recaptcha1Сегодня мы поговорим об установке капчи (captcha) в ваши проекты на CakePHP. Капча бывает очень полезной для защиты от ботов, к примеру, в публичных формах обратной связи, при регистрации и во многих других местах, которые могут отказаться целью автоматической обработки с корыстной целью. К примеру, для спама. Под катом я расскажу сразу о двух разных компонентах для капчи реализованных для CakePHP.

В поисках капчи я натолкнулся на две реализации — простую, и сложную. Забавно то, что простая реализация в результате даёт капчу значительно сложнее, а сложная — значительно проще.

Первым делом, я расскажу о Recaptcha.

Recaptcha

Теория

Рекапча была придуманна в университете Карнеги-Меллона, что находится в Питтсбурге. Идея проекта заключается в том, что время, которое вы тратите на распознание капчи, идёт на распознание… текстов книг и статей.

Процесс работы рекапчи достаточно прост. Сканы книг и статей сканируются двумя независимыми системами распознания изображений. Когда результаты их работы отличаются, из нераспознанного слова делается капча, к которой добавляют контрольное слово, значение которого уже известно системе.

Кроме того, все процессы отображения и проверки капчи происходят на удалённом сервере, что затрудняет процесс её взлома.

Recaptcha  так же позволяет использовать так называемый механизм mailhide. Он защищает адреса электронной почты от ботов, которые собираю их в свои спам-списки. При использовании mailhide на странице выводится адрес в виде co..@meelk.com.ua. После щелчка по “…” необходимо ввести капчу, после чего вы увидите адрес полностью.

Использование

Для реализации reCaptcha в ваших проектах, предлогаю использовать хелпер, созданный Говардом Линсом (Howard Lince). Для начала надо получить приватный и публичный ключ для рекапчи. Сделать это можно на сайте. Публичный ключ используется для получения изображения, а приватный – для проверки введённого текста.

Теперь создадим компонент app/controllers/components/recaptcha.php

<?php 
 class RecaptchaComponent extends Object {
	var $publickey = "";
	var $privatekey= "";
 
	var $is_valid = false;
	var $error = "";
 
	function startup(&$controller){
		Configure::write("Recaptcha.apiServer","http://api.recaptcha.net");
		Configure::write("Recaptcha.apiSecureServer","https://api-secure.recaptcha.net");
		Configure::write("Recaptcha.verifyServer","api-verify.recaptcha.net");
		Configure::write("Recaptcha.pubKey", $this->publickey);
		Configure::write("Recaptcha.privateKey", $this->privatekey);
 
		$this->controller =& $controller;
		$this->controller->helpers[] = "Recaptcha";
	}
 
	function valid($form){
        if (isset($form['recaptcha_challenge_field']) && isset($form['recaptcha_response_field'])){
        	if($this->recaptcha_check_answer(
                $this->privatekey, 
                $_SERVER["REMOTE_ADDR"],
                $form['recaptcha_challenge_field'], 
                $form['recaptcha_response_field']
            ) == 0)
            	return false;
 
            if ($this->is_valid)
                return true;
        }
        return false;
    }
 
	/**
	  * Calls an HTTP POST function to verify if the user's guess was correct
	  * @param string $privkey
	  * @param string $remoteip
	  * @param string $challenge
	  * @param string $response
	  * @param array $extra_params an array of extra variables to post to the server
	  * @return ReCaptchaResponse
	  */
	function recaptcha_check_answer ($privkey, $remoteip, $challenge, $response, $extra_params = array()){
		if ($privkey == null || $privkey == ''){
			die ("To use reCAPTCHA you must get an API key from <a href='http://recaptcha.net/api/getkey'>http://recaptcha.net/api/getkey</a>");
		}
 
		if ($remoteip == null || $remoteip == ''){
			die ("For security reasons, you must pass the remote ip to reCAPTCHA");
		}		
 
	        //discard spam submissions
	        if ($challenge == null || strlen($challenge) == 0 || $response == null || strlen($response) == 0) {
	                $this->is_valid = false;
	                $this->error = 'incorrect-captcha-sol';
	                return 0;
	        }
 
	        $response = $this->_recaptcha_http_post(Configure::read('Recaptcha.verifyServer'), "/verify",
	                                          array (
	                                                 'privatekey' => $privkey,
	                                                 'remoteip' => $remoteip,
	                                                 'challenge' => $challenge,
	                                                 'response' => $response
	                                                 ) + $extra_params
	                                          );
 
	        $answers = explode ("\n", $response [1]);
 
	        if (trim ($answers [0]) == 'true') {
	                $this->is_valid = true;
	                return 1;
	        }else{
	                $this->is_valid = false;
	                $this->error = $answers [1];
	                return 0;
	        }
	}
 
 
	/**
	 * Submits an HTTP POST to a reCAPTCHA server
	 * @param string $host
	 * @param string $path
	 * @param array $data
	 * @param int port
	 * @return array response
	 */
	function _recaptcha_http_post($host, $path, $data, $port = 80) {
 
        $req = $this->_recaptcha_qsencode ($data);
 
        $http_request  = "POST $path HTTP/1.0\r\n";
        $http_request .= "Host: $host\r\n";
        $http_request .= "Content-Type: application/x-www-form-urlencoded;\r\n";
        $http_request .= "Content-Length: " . strlen($req) . "\r\n";
        $http_request .= "User-Agent: reCAPTCHA/PHP\r\n";
        $http_request .= "\r\n";
        $http_request .= $req;
 
        $response = '';
        if( false == ( $fs = @fsockopen($host, $port, $errno, $errstr, 10) ) ) {
                die ('Could not open socket');
        }
 
        fwrite($fs, $http_request);
 
        while ( !feof($fs) )
                $response .= fgets($fs, 1160); // One TCP-IP packet
        fclose($fs);
        $response = explode("\r\n\r\n", $response, 2);
 
        return $response;
	}
 
 
	/**
	 * Encodes the given data into a query string format
	 * @param $data - array of string elements to be encoded
	 * @return string - encoded request
	 */
	function _recaptcha_qsencode ($data) {
        $req = "";
        foreach ( $data as $key => $value )
                $req .= $key . '=' . urlencode( stripslashes($value) ) . '&';
 
        // Cut the last '&'
        $req=substr($req,0,strlen($req)-1);
        return $req;
	}
}
?>

Теперь создадим хелпер app/views/helpers/recaptcha.php

<?php 
class RecaptchaHelper extends AppHelper {
	var $helpers = array('form'); 
 
	function display_form($output_method = 'return', $error = null, $use_ssl = false){
		$data = $this->__form(Configure::read("Recaptcha.pubKey"),$error,$use_ssl);
		if($output_method == "echo")
			echo $data;
		else
			return $data;
	}
 
	function hide_mail($email = '',$output_method = 'return'){
		$data = $this->recaptcha_mailhide_html(Configure::read('Recaptcha.pubKey'), Configure::read('Recaptcha.privateKey'), $email);
		if($output_method == "echo")
			echo $data;
		else
			return $data;
	}
 
	/**
	 * Gets the challenge HTML (javascript and non-javascript version).
	 * This is called from the browser, and the resulting reCAPTCHA HTML widget
	 * is embedded within the HTML form it was called from.
	 * @param string $pubkey A public key for reCAPTCHA
	 * @param string $error The error given by reCAPTCHA (optional, default is null)
	 * @param boolean $use_ssl Should the request be made over ssl? (optional, default is false)
 
	 * @return string - The HTML to be embedded in the user's form.
	 */
	function __form($pubkey, $error = null, $use_ssl = false){
		if ($pubkey == null || $pubkey == '') {
			die ("To use reCAPTCHA you must get an API key from <a href='http://recaptcha.net/api/getkey'>http://recaptcha.net/api/getkey</a>");
		}
 
		if ($use_ssl) {
	                $server = Configure::read('Recaptcha.apiSecureServer');
	        } else {
	                $server = Configure::read('Recaptcha.apiServer');
	        }
 
	        $errorpart = "";
	        if ($error) {
	           $errorpart = "&amp;error=" . $error;
	        }
	        return '<script type="text/javascript" src="'. $server . '/challenge?k=' . $pubkey . $errorpart . '"></script>
 
		<noscript>
	  		<iframe src="'. $server . '/noscript?k=' . $pubkey . $errorpart . '" height="300" width="500" frameborder="0"></iframe><br/>
	  			<textarea name="recaptcha_challenge_field" rows="3" cols="40"></textarea>
				<input type="hidden" name="recaptcha_response_field" value="manual_challenge"/>
	  		<input type="hidden" name="recaptcha_response_field" value="manual_challenge"/>
		</noscript>';
	}
 
	/* Mailhide related code */
	function _recaptcha_aes_encrypt($val,$ky) {
		if (! function_exists ("mcrypt_encrypt")) {
			die ("To use reCAPTCHA Mailhide, you need to have the mcrypt php module installed.");
		}
		$mode=MCRYPT_MODE_CBC;   
		$enc=MCRYPT_RIJNDAEL_128;
		$val=$this->_recaptcha_aes_pad($val);
		return mcrypt_encrypt($enc, $ky, $val, $mode, "\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0");
	}
 
	function _recaptcha_mailhide_urlbase64 ($x) {
		return strtr(base64_encode ($x), '+/', '-_');
	}
 
	/* gets the reCAPTCHA Mailhide url for a given email, public key and private key */
	function recaptcha_mailhide_url($pubkey, $privkey, $email) {
		if ($pubkey == '' || $pubkey == null || $privkey == "" || $privkey == null) {
			die ("To use reCAPTCHA Mailhide, you have to sign up for a public and private key, " .
			     "you can do so at <a href='http://mailhide.recaptcha.net/apikey'>http://mailhide.recaptcha.net/apikey</a>");
		}
 
 
		$ky = pack('H*', $privkey);
		$cryptmail = $this->_recaptcha_aes_encrypt ($email, $ky);
 
		return "http://mailhide.recaptcha.net/d?k=" . $pubkey . "&c=" . $this->_recaptcha_mailhide_urlbase64 ($cryptmail);
	}
 
	/**
	 * gets the parts of the email to expose to the user.
	 * eg, given johndoe@example,com return ["john", "example.com"].
	 * the email is then displayed as john...@example.com
	 */
	function _recaptcha_mailhide_email_parts ($email) {
		$arr = preg_split("/@/", $email );
 
		if (strlen ($arr[0]) <= 4) {
			$arr[0] = substr ($arr[0], 0, 1);
		} else if (strlen ($arr[0]) <= 6) {
			$arr[0] = substr ($arr[0], 0, 3);
		} else {
			$arr[0] = substr ($arr[0], 0, 4);
		}
		return $arr;
	}
 
	/**
	 * Gets html to display an email address given a public an private key.
	 * to get a key, go to:
	 *
	 * http://mailhide.recaptcha.net/apikey
	 */
	function recaptcha_mailhide_html($pubkey, $privkey, $email) {
		$emailparts = $this->_recaptcha_mailhide_email_parts ($email);
		$url = $this->recaptcha_mailhide_url ($pubkey, $privkey, $email);
 
		return htmlentities($emailparts[0]) . "<a href='" . htmlentities ($url) .
			"' onclick=\"window.open('" . htmlentities ($url) . "', '', 'toolbar=0,scrollbars=0,location=0,statusbar=0,menubar=0,resizable=0,width=500,height=300'); return false;\" title=\"Reveal this e-mail address\">...</a>@" . htmlentities ($emailparts [1]);
 
	}
 
 
}
?>

Это всё. Теперь добавим в контроллер что-то вроде этого, не забыв ввести полученные уже публичный и приватный ключ.

$components = array("recaptcha");
$helpers = array("recaptcha");
$this->recaptcha->publickey = "";
$this->recaptcha->privatekey = "";
}
?>

Теперь в вид можно добавить следующий код

<?
display_form('echo')
//Спрятать адрес.
 $recaptcha->hide_mail("someuser@somdomain.tld",'echo');
?>

Метод отображения капчи принимает параметром тип вывода. В примере в указанном месте хелпер сам выведет содержимое (‘echo’), однако можно и просто его вернуть в виде строки (‘return’). Таким же образом метод hide_mail принимает в качестве параметра собственно адрес, и тип вывода.

В контроллере можно сделать следущую проверку:

<?
Recaptcha->valid($this->params['form']))
  //Введёный текст верен
else
  //Пользователь ошибся при вводе
?>

Капча будет выглядеть следующим образом:

recaptcha, вот она какая

imageAuth

imageAuth – это уже компонент, который генерирует капчу уже у вас на машине. Такая капча зависит только от вас, да и выглядит не такой громоздкой как предыдущая. Этот компонент я нашёл в блоге Rossoft где вы наверняка сможете найти для себя много интересного.

Вот и все, удачного вам использования капчи в ваших проектах, и не забывайте, не стоит вставлять капчу  для подтверждения каждого действия пользователя — это очень быстро надоедает и отталкивает пользователя от проекта.

This entry was posted on Saturday, February 28th, 2009 and is filed under cakephp, PHP-разработка, Пізнавальне. You can follow any responses to this entry through the RSS 2.0 feed. You can leave a response, or trackback from your own site.

12 Responses to “Использование reCaptcha в CakePHP”

  1. А ящитаю, капча – это тупиковая ветвь. Будущее за байезианскими фильтрами. Капча – репрессивная мера для честных пользователей, они тратят свое время для расшифровки непонятной фигни. С другой стороны, время ботов намного менее ценно. Лучше наказывать ботов, путем составления черных списков.

  2. Так вот кто это всё читает! :)
    Ну, капча это как бы "временное решение", понятно, что было бы круто, если были бы системы которые бы сами отделяли ботов от людей. Есть разные интересные решения, но пока что единственное действенное решение – капча. Плюс, фильтры использовать не везде можно. Разве что для постов. Да и то, под бдительным контролем модератора.

  3. для комментариев есть akismet. очень хорошо чистит и карает анально даже биороботов сеошников.

  4. а вот паранойи по поводу выкладывания своего email не понимаю совершенно. я свой gmail везде пишу и ниче.

  5. Вот я и говорю, что фильтры подходят для постов. А, скажем, регистрации или специализированные формы, к примеру, форма отправки заказа или предложения. Ведь это рай для спама. Но как отфильтровать спам (предложение о продаже чайников), от реального предложения по покупке холодильников?

  6. С этим согласен :) Описал этот метод, поскольку он есть и может оказаться всё-таки полезным.

  7. Ну, этот маркетинговый ход был в моде пару лет назад. Сейчас задрало то, что все сайты обязательно должны иметь закрытую регистрацию через инвайт, который у всех полюбасу уже есть )

  8. Супер, долго искал, большое спасибо автору!

  9. а сами почему не поставили этот плагин?

  10. очень интересная статья, автор, еще статьи по cakephp будут?

  11. Спасибо! Сейчас к сожалению времени совсем мало, да и работа в основном ведётся с ипользованием WP. Но думаю всё-таки продолжить :)

  12. High class services; I ll undoubtedly come back to your site for additional help and advice.
    Plumbing Lake Worth FL

Leave a Reply

free counters

Designed by Gabfire
Rambler's Top100