Cloudflare Turnstile is an alternative to Google reCAPTCHA that loads faster, and It provides a user-friendly experience without the need for user interaction.
In this post, I will guide you on how to add Cloudflare Turnstile to your WordPress website without using plugins, using just code.

How to Add Cloudflare Turnstile CAPTCHA in WordPress without Plugins

Get a site key and a secret key from Cloudflare

To access this information, simply open a new browser tab and navigate to the Cloudflare login page. If you don’t have an account yet, you’ll need to create one using your email address.

Once you’ve successfully logged into your Cloudflare dashboard, locate the ‘Turnstile’ option in the left-hand menu and click on it.

Then click on the ‘Add site’ button.

On this screen, start by typing in a ‘Site Name.’
Next, type your website’s domain name into the ‘Domain’ field.

In the next step, we need to add some codes to WordPress.
To do this, you can either add these codes to the functions.php file of your child theme or use the Code Snippets plugin for this purpose.

Place the code below in your child theme’s functions file and replace the site_key and secret_key with the ones you obtained in the previous step.

function cloudflare_key(){
	$sitekey= "00000000000000000000000";
	$secretkey= "000000000000000000000";
	return [$sitekey,$secretkey]; 	
}

add_action("wp_head", function(){	
	wp_enqueue_script('cloudflare-turnstile', 'https://challenges.cloudflare.com/turnstile/v0/api.js');
	} );
	

Adding Cloudflare Turnstile to WordPress Login Form

Show code

To add Cloudflare Turnstile to your login form, simply copy the code below into the functions.php file of your child theme or into the code snippets plugin.

/*
 * Adding Cloudflare Turnstile to Login Form by wpcookie
 * https://redpishi.com/wordpress-tutorials/cloudflare-turnstile-captcha-wordpress/
 */	
function login_style() {
    wp_register_script('login-recaptcha', 'https://challenges.cloudflare.com/turnstile/v0/api.js', false, NULL);
    wp_enqueue_script('login-recaptcha');
	echo "<style>p.submit, p.forgetmenot {margin-top: 10px!important;}.login form{width: 303px;} div#login_error {width: 322px;}</style>";
}
add_action('login_enqueue_scripts', 'login_style');

add_action('login_form', function(){
	echo '<div class="cf-turnstile" data-sitekey="'.cloudflare_key()[0].'"></div>';
	} );

add_action('wp_authenticate_user', function($user, $password) {
	$captcha=$_POST['cf-turnstile-response'];
	if (!$captcha) {
     return new WP_Error('Captcha Invalid', __('<center>Captcha Invalid! Please check the captcha!</center>'));
	   die();
           exit;
   }
   $secretKey = cloudflare_key()[1];
   $ip = $_SERVER['REMOTE_ADDR'];

   $url_path = 'https://challenges.cloudflare.com/turnstile/v0/siteverify';
   $data = array('secret' => $secretKey, 'response' => $captcha, 'remoteip' => $ip);
	
	$options = array(
		'http' => array(
		'method' => 'POST',
		'content' => http_build_query($data))
	);
	
	$stream = stream_context_create($options);
	
	$result = file_get_contents(
			$url_path, false, $stream);
	
	$response =  $result;
   
   $responseKeys = json_decode($response,true);
	  if(intval($responseKeys["success"]) !== 1) {
		   return new WP_Error('Captcha Invalid', __('<center>Captcha Invalid! Please check the captcha!</center>'));
		  die();
		   exit;
	  } else { 
		  return $user;
}
	
	} , 10, 2);

Adding Cloudflare Turnstile to WordPress Comment

Show code

To add Cloudflare Turnstile to your comment form, simply copy the code below into the functions.php file of your child theme or into the code snippets plugin.

/*
 * Adding Cloudflare Turnstile to WordPress Comment
 * https://redpishi.com/wordpress-tutorials/cloudflare-turnstile-captcha-wordpress/
 */	
function is_valid_captcha($captcha) {
	
   if (!$captcha) {
     return false;
   }
   $secretKey = cloudflare_key()[1];
   $ip = $_SERVER['REMOTE_ADDR'];

   $url_path = 'https://challenges.cloudflare.com/turnstile/v0/siteverify';
   $data = array('secret' => $secretKey, 'response' => $captcha, 'remoteip' => $ip);
	
	$options = array(
		'http' => array(
		'method' => 'POST',
		'content' => http_build_query($data))
	);
	
	$stream = stream_context_create($options);
	
	$result = file_get_contents(
			$url_path, false, $stream);
	
	$response =  $result;
   
   $responseKeys = json_decode($response,true);
	  if(intval($responseKeys["success"]) !== 1) {
		   return false;
	  } else { 
		  return true;
}
}
	
add_action('init', function(){
	if (!is_user_logged_in() ) {
    add_action('pre_comment_on_post', function(){
		$recaptcha = $_POST['cf-turnstile-response'];
	if (empty($recaptcha))
    wp_die( __("<b>ERROR:</b> please select <b>I'm not a robot!</b><p><a href='javascript:history.back()'>« Back</a></p>"));
	else if (!is_valid_captcha($recaptcha))
    wp_die( __("<b>please select I'm not a robot!</b>"));
		
		} );

    add_filter('comment_form_defaults',function ($submit_field) {

		    $submit_field['submit_field'] = '<div class="cf-turnstile" data-sitekey="'.cloudflare_key()[0].'"></div><br>'.$submit_field['submit_field'];
    return $submit_field;	
	});
}	
	});

Adding Cloudflare Turnstile to WordPress Lost Password page

Show code

To add Cloudflare Turnstile to to WordPress Lost Password form, simply copy the code below into the functions.php file of your child theme or into the code snippets plugin.

This code is for WPCookie PLUS+ members only.

Join WPCookie PLUS+
Already a member? Log in here

Adding Cloudflare Turnstile to WordPress Registration Form

To activate registration in WordPress, first go to the admin dashboard and select the “General” option from the “Setting” menu. Then, check the box next to the “Anyone can register” option.

Show code

To add Cloudflare Turnstile to to WordPress Registration Form, simply copy the code below into the functions.php file of your child theme or into the code snippets plugin.

This code is for WPCookie PLUS+ members only.

Join WPCookie PLUS+
Already a member? Log in here

Adding Cloudflare Turnstile to contact form 7 forms

First, edit the form you want to add Cloudflare Turnstile to and add the following snippet before the submit button:

[c][text c]

Next copy the code below into the functions.php file of your child theme or into the code snippets plugin.

Show code
add_action("wp_footer", function(){ ?>
    <style>
        .wpcf7 input[name="c"] {
            display: none;
        }
        .wpcf7 .cf-turnstile {
            margin-top: -10px;
        }
    </style>
<?php });

function add_custom_element_before_submit($form) {
    $cf = '<div class="cf-turnstile" data-sitekey="' . cloudflare_key()[0] . '"></div>';
    $form = preg_replace('/\[\s*c\s*\]/', $cf, $form);    
    return $form;
}
add_filter('wpcf7_form_elements', 'add_custom_element_before_submit');

function validate_turnstile($result, $tag) {
    $turnstile_token = isset($_POST['cf-turnstile-response']) ? sanitize_text_field($_POST['cf-turnstile-response']) : '';
    $ct = "c";
    if (empty($turnstile_token)) {
        $result->invalidate($ct, 'You must complete the Turnstile challenge to submit this form.');
        // Mark as spam since no token was provided
        add_filter('wpcf7_spam', function() { return true; }, 100, 2);
        return $result;
    }

    $secret_key = cloudflare_key()[1];
    $remote_ip = $_SERVER['REMOTE_ADDR'];

    $response = wp_remote_post('https://challenges.cloudflare.com/turnstile/v0/siteverify', array(
        'body' => array(
            'secret' => $secret_key,
            'response' => $turnstile_token,
            'remoteip' => $remote_ip,
        ),
    ));

    if (is_wp_error($response)) {
        $result->invalidate($ct, 'The Turnstile token is invalid. Please try again.');
        // Mark as spam for invalid token
        add_filter('wpcf7_spam', function() { return true; }, 100, 2);
        return $result;
    }

    $body = wp_remote_retrieve_body($response);
    $data = json_decode($body, true);

    if (!$data['success']) {
        $result->invalidate($ct, 'There was an error validating the Turnstile challenge. Please try again.');
        // Mark as spam if validation failed
        add_filter('wpcf7_spam', function() { return true; }, 100, 2);
        return $result;
    }

    return $result;
}
add_filter('wpcf7_validate', 'validate_turnstile', 10, 2);

If this article is difficult for you to read in text, you can watch the video version below.

Share this post
Maya
Maya

Hi, my name is Maya and I’m a WordPress plugin developer. I created this website to share some of the helpful codes that I’ve used in my own projects.
If you’re looking for a custom plugin for your website, you can contact me by clicking on Hire a developer in the menu. I’d love to hear from you.

Articles: 58

10 Comments

  1. Hi

    I like your code for adding Turnstile to Contact Form 7 forms, it works but it is not effective. It\’s missing a tie in with the wpcf7_spam filter. Just using $result->invalidate will not prevent a spam/bot submission to go through, spammy emails will still go through. Please clarify. Thanks.

    • You\’re correct that simply invalidating with $result->invalidate doesn\’t prevent the form from being submitted as spam. I\’ve updated the code to include wpcf7_spam filter integration within the validation process.
      Expect an email from me with a one-year free subscription to WPCookie Plus+!

    • To add Cloudflare Turnstile to HTML forms in WordPress:
      1.Insert the Turnstile script snippet into the head of your Page, You can use wp_head hook.

      2. Add Turnstile field to Your Form:

      Replace your_site_key with your actual site key.

      3.Verify Turnstile Response in PHP
      In your form handling PHP script, you need to verify the Turnstile response. You can use wp_ajax and wp_ajax_nopriv hooks.

      Here’s a basic example:

      if ($_SERVER[\’REQUEST_METHOD\’] === \’POST\’) {
      $turnstile_response = $_POST[\’cf-turnstile-response\’];
      $secret_key = \’your_secret_key\’;

      $response = file_get_contents(\”https://challenges.cloudflare.com/turnstile/v0/siteverify\”, false, stream_context_create([
      \’http\’ => [
      \’method\’ => \’POST\’,
      \’header\’ => \’Content-type: application/x-www-form-urlencoded\’,
      \’content\’ => http_build_query([
      \’secret\’ => $secret_key,
      \’response\’ => $turnstile_response,
      \’remoteip\’ => $_SERVER[\’REMOTE_ADDR\’]
      ])
      ]
      ]));

      $result = json_decode($response, true);

      if ($result[\’success\’]) {
      // Turnstile verification passed
      // Process the form data
      } else {
      // Turnstile verification failed
      echo \’Verification failed. Please try again.\’;
      }
      }

      Replace your_secret_key with your actual secret key.

  2. Hi Maya, thank you so much for the codes. I tried it to add Cloudflare Turnstile to my WordPress Comment, and it really work! All the best to you.

    • I am not familiar with bbPress, but if I get the chance to work with it, I may create a tutorial about it. Thank you for the suggestion!

Leave a Reply

Your email address will not be published. Required fields are marked *