Captcha Protection for your API Routes
Learn how to set up captcha protection for your API routes.
For captcha protection, we use Cloudflare Turnstile. To enable it, you need to set the following environment variables:
CAPTCHA_SECRET_TOKEN=NEXT_PUBLIC_CAPTCHA_SITE_KEY=
You can find the CAPTCHA_SECRET_TOKEN
in the Turnstile configuration. The NEXT_PUBLIC_CAPTCHA_SITE_KEY
is public and safe to share. Instead, the CAPTCHA_SECRET_TOKEN
should be kept secret.
This guide assumes you have correctly set up your Turnstile configuration. If you haven't, please refer to the Turnstile documentation.
Authentication
When you set the token in the environment variables, the kit will automatically protect your API routes with captcha.
NB: you also need to set the token in the Supabase Dashboard!
Retrieving the Token
To retrieve the captcha token, you can use the useCaptchaToken
function from @kit/auth/captcha/client
:
import { useCaptchaToken } from '@kit/auth/captcha/client';function MyComponent() { const { captchaToken } = useCaptchaToken(); const handleSubmit = async (e) => { e.preventDefault(); const response = await fetch('/my-endpoint', { method: 'POST', headers: { 'Content-Type': 'application/json', 'x-captcha-token': captchaToken, }, body: JSON.stringify({ message: 'Hello, world!' }), }); }; // your component code}
When using Server Actions, use enhanceAction
from @kit/next/actions
:
const MySchema = z.object({ message: z.string(), captchaToken: z.string().min(1),});export const myServerAction = enhanceAction(async (data) => { // your action code}, { captcha: true, schema: MySchema});
When calling the server action, you need to pass the captcha:
function MyForm() { const { captchaToken, resetCaptchaToken } = useCaptchaToken(); const handleSubmit = async (e) => { e.preventDefault(); try { const response = await myServerAction({ message: 'Hello, world!', captchaToken, }); } finally { // always reset the token! resetCaptchaToken(); } }; // your component code}
A token is valid for one request only. You need to reset the token after each request.
The library will take care of renewing the token automatically when needed - but you need to reset it manually when consuming the token.
Resetting the Token
The underlying library React Turnstile resets the token automatically - but we need to reset it manually when actually consuming the token since the token is only valid for one request.
To reset the token, please call resetCaptchaToken
from the useCaptchaToken
hook - as shown in the example above.
You should be doing so in all situations, whether the request was successful or not.
Verifying the Token
To verify the captcha manually server-side, please use the following code:
import { verifyCaptchaToken } from '@kit/auth/captcha/server';function assertCaptchaValidity(request: Request) { const token = request.headers.get('x-captcha-token'); await verifyCaptchaToken(token);}
If you use the utilities provided by the kit enhanceAction
and enhanceRouteHandler
, you don't need to worry about this, as the kit will automatically verify the captcha token for you (as long as it is passed).