Protect your forms
  Last reviewed:  3 months ago  
      This tutorial will guide you through integrating Cloudflare Turnstile to protect your web forms, such as login, signup, or contact forms. Learn how to implement the Turnstile widget on the client side and verify the Turnstile token via the Siteverify API on the server side.
- You must have a Cloudflare account.
- You must have a web application with a form you want to protect.
- You must have basic knowledge of HTML and your server-side language of choice, such as Node.js or Python.
- Log in to the Cloudflare dashboard ↗ and select your account.
- Go to Turnstile and create a new Turnstile widget.
- Copy the sitekey and the secret key to use in the next step.
- Add the Turnstile widget to your form.
- Replace <YOUR-SITE-KEY>with the sitekey from Cloudflare.
- Add a data-callbackattribute to the Turnstile widget div. This JavaScript function will be called when the challenge is successful.
- Ensure your submit button is initially disabled.
<!DOCTYPE html><html lang="en"><head>    <meta charset="UTF-8">    <title>Contact Form</title>    <script src="https://challenges.cloudflare.com/turnstile/v0/api.js" async defer></script>    <script>        function enableSubmit() {            document.getElementById("submit-button").disabled = false;        }    </script></head><body>    <form id="contact-form" action="/submit" method="POST">        <input type="text" name="name" placeholder="Name" required>        <input type="email" name="email" placeholder="Email" required>        <textarea name="message" placeholder="Message" required></textarea>
        <!-- Turnstile widget -->        <div class="cf-turnstile" data-sitekey="<YOUR-SITE-KEY>" data-callback="enableSubmit"></div>
        <button type="submit" id="submit-button" disabled>Submit</button>    </form></body></html>You will need to verify the Turnstile token sent from the client side. Below is an example in Node.js.
const express = require('express');const axios = require('axios');const bodyParser = require('body-parser');const app = express();
app.use(bodyParser.urlencoded({ extended: true }));
app.post('/submit', async (req, res) => {    const turnstileToken = req.body['cf-turnstile-response'];    const secretKey = 'your-secret-key';
    try {        const response = await axios.post('https://challenges.cloudflare.com/turnstile/v0/siteverify', null, {            params: {                secret: secretKey,                response: turnstileToken            }        });
        if (response.data.success) {            // Token is valid, proceed with form submission            const name = req.body.name;            const email = req.body.email;            const message = req.body.message;            // Your form processing logic here            res.send('Form submission successful');        } else {            res.status(400).send('Turnstile verification failed');        }    } catch (error) {        res.status(500).send('Error verifying Turnstile token');    }});
app.listen(3000, () => {    console.log('Server is running on port 3000');});It is crucial to handle the verification of the Turnstile token correctly. This section covers some key points to keep in mind.
- Ensure that you verify the Turnstile token after the user has filled out the form and selected submit.
- If you verify the token before the user inputs their data, a malicious actor could potentially bypass the protection by manipulating the form submission after obtaining a valid token.
- When the user submits the form, send both the form data and the Turnstile token to your server.
- On the server side, verify the Turnstile token first.
- Based on the verification response, decide whether to proceed with processing the form data.
Was this helpful?
- Resources
- API
- New to Cloudflare?
- Directory
- Sponsorships
- Open Source
- Support
- Help Center
- System Status
- Compliance
- GDPR
- Company
- cloudflare.com
- Our team
- Careers
- © 2025 Cloudflare, Inc.
- Privacy Policy
- Terms of Use
- Report Security Issues
- Trademark