2011-01-31 09:50:27 +00:00
< ? php
2011-10-14 21:30:16 +00:00
/**
2011-01-31 09:50:27 +00:00
* This is a PHP library that handles calling reCAPTCHA .
* - Documentation and latest version
* http :// recaptcha . net / plugins / php /
* - Get a reCAPTCHA API Key
2011-09-29 21:31:11 +00:00
* https :// www . google . com / recaptcha / admin / create
2011-01-31 09:50:27 +00:00
* - Discussion group
* http :// groups . google . com / group / recaptcha
*
* Copyright ( c ) 2007 reCAPTCHA -- http :// recaptcha . net
* AUTHORS :
* Mike Crawford
* Ben Maurer
*
* Permission is hereby granted , free of charge , to any person obtaining a copy
* of this software and associated documentation files ( the " Software " ), to deal
* in the Software without restriction , including without limitation the rights
* to use , copy , modify , merge , publish , distribute , sublicense , and / or sell
* copies of the Software , and to permit persons to whom the Software is
* furnished to do so , subject to the following conditions :
*
* The above copyright notice and this permission notice shall be included in
* all copies or substantial portions of the Software .
*
* THE SOFTWARE IS PROVIDED " AS IS " , WITHOUT WARRANTY OF ANY KIND , EXPRESS OR
* IMPLIED , INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY ,
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT . IN NO EVENT SHALL THE
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM , DAMAGES OR OTHER
* LIABILITY , WHETHER IN AN ACTION OF CONTRACT , TORT OR OTHERWISE , ARISING FROM ,
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
* THE SOFTWARE .
*/
/**
* The reCAPTCHA server URL ' s
*/
2012-01-12 08:58:40 +00:00
define ( " RECAPTCHA_API_SERVER " , " http://www.google.com/recaptcha/api " );
define ( " RECAPTCHA_API_SECURE_SERVER " , " https://www.google.com/recaptcha/api " );
define ( " RECAPTCHA_VERIFY_SERVER " , " www.google.com " );
2011-01-31 09:50:27 +00:00
/**
* Encodes the given data into a query string format
* @ param $data - array of string elements to be encoded
* @ return string - encoded request
*/
2012-01-12 08:58:40 +00:00
function _recaptcha_qsencode ( $data ) {
2011-01-31 09:50:27 +00:00
$req = " " ;
foreach ( $data as $key => $value )
2012-01-12 08:58:40 +00:00
$req .= $key . '=' . urlencode ( stripslashes ( $value ) ) . '&' ;
2011-01-31 09:50:27 +00:00
// Cut the last '&'
2012-01-12 08:58:40 +00:00
$req = substr ( $req , 0 , strlen ( $req ) - 1 );
2011-01-31 09:50:27 +00:00
return $req ;
}
/**
* Submits an HTTP POST to a reCAPTCHA server
* @ param string $host
* @ param string $path
* @ param array $data
* @ param int port
* @ return array response
*/
2012-01-12 08:58:40 +00:00
function _recaptcha_http_post ( $host , $path , $data , $port = 80 ) {
2011-01-31 09:50:27 +00:00
2012-01-12 08:58:40 +00:00
$req = _recaptcha_qsencode ( $data );
2011-01-31 09:50:27 +00:00
$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 " ;
2012-01-12 08:58:40 +00:00
$http_request .= " Content-Length: " . strlen ( $req ) . " \r \n " ;
2011-01-31 09:50:27 +00:00
$http_request .= " User-Agent: reCAPTCHA/PHP \r \n " ;
$http_request .= " \r \n " ;
$http_request .= $req ;
$response = '' ;
2012-01-12 08:58:40 +00:00
if ( false == ( $fs = @ fsockopen ( $host , $port , $errno , $errstr , 10 ) ) ) {
die ( 'Could not open socket' );
2011-01-31 09:50:27 +00:00
}
2012-01-12 08:58:40 +00:00
fwrite ( $fs , $http_request );
2011-01-31 09:50:27 +00:00
2012-01-12 08:58:40 +00:00
while ( ! feof ( $fs ) )
$response .= fgets ( $fs , 1160 ); // One TCP-IP packet
fclose ( $fs );
$response = explode ( " \r \n \r \n " , $response , 2 );
2011-01-31 09:50:27 +00:00
return $response ;
}
2011-09-29 21:31:11 +00:00
2011-01-31 09:50:27 +00:00
/**
* 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 .
*/
2012-01-12 08:58:40 +00:00
function recaptcha_get_html ( $pubkey , $error = null , $use_ssl = false )
2011-01-31 09:50:27 +00:00
{
2012-01-12 08:58:40 +00:00
if ( $pubkey == null || $pubkey == '' ) {
die ( " To use reCAPTCHA you must get an API key from <a href='https://www.google.com/recaptcha/admin/create'>https://www.google.com/recaptcha/admin/create</a> " );
2011-01-31 09:50:27 +00:00
}
2011-10-14 21:30:16 +00:00
2012-01-12 08:58:40 +00:00
if ( $use_ssl ) {
2011-09-29 21:31:11 +00:00
$server = RECAPTCHA_API_SECURE_SERVER ;
2011-01-31 09:50:27 +00:00
} else {
2011-09-29 21:31:11 +00:00
$server = RECAPTCHA_API_SERVER ;
2011-01-31 09:50:27 +00:00
}
2011-09-29 21:31:11 +00:00
2011-01-31 09:50:27 +00:00
$errorpart = " " ;
2012-01-12 08:58:40 +00:00
if ( $error ) {
2011-01-31 09:50:27 +00:00
$errorpart = " &error= " . $error ;
}
2012-01-12 08:58:40 +00:00
return '<script type="text/javascript" src="' . $server . '/challenge?k=' . $pubkey . $errorpart . ' " ></script>
2011-01-31 09:50:27 +00:00
< noscript >
2012-01-12 08:58:40 +00:00
< iframe src = " ' . $server . '/noscript?k=' . $pubkey . $errorpart . ' " height = " 300 " width = " 500 " frameborder = " 0 " ></ iframe >< br />
2011-01-31 09:50:27 +00:00
< textarea name = " recaptcha_challenge_field " rows = " 3 " cols = " 40 " ></ textarea >
2011-09-29 21:31:11 +00:00
< input type = " hidden " name = " recaptcha_response_field " value = " manual_challenge " />
2011-01-31 09:50:27 +00:00
</ noscript > ' ;
}
2011-09-29 21:31:11 +00:00
2011-01-31 09:50:27 +00:00
/**
* A ReCaptchaResponse is returned from recaptcha_check_answer ()
*/
class ReCaptchaResponse {
var $is_valid ;
var $error ;
}
2011-09-29 21:31:11 +00:00
2011-01-31 09:50:27 +00:00
/**
* 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
2011-09-29 21:31:11 +00:00
* @ param array $extra_params an array of extra variables to post to the server
2011-01-31 09:50:27 +00:00
* @ return ReCaptchaResponse
*/
2012-01-12 08:58:40 +00:00
function recaptcha_check_answer ( $privkey , $remoteip , $challenge , $response , $extra_params = array () )
2011-01-31 09:50:27 +00:00
{
2012-01-12 08:58:40 +00:00
if ( $privkey == null || $privkey == '' ) {
die ( " To use reCAPTCHA you must get an API key from <a href='https://www.google.com/recaptcha/admin/create'>https://www.google.com/recaptcha/admin/create</a> " );
2011-01-31 09:50:27 +00:00
}
2012-01-12 08:58:40 +00:00
if ( $remoteip == null || $remoteip == '' ) {
die ( " For security reasons, you must pass the remote ip to reCAPTCHA " );
2011-01-31 09:50:27 +00:00
}
2011-10-14 21:30:16 +00:00
2012-01-12 08:58:40 +00:00
// discard spam submissions
if ( $challenge == null || strlen ( $challenge ) == 0 || $response == null || strlen ( $response ) == 0 ) {
2011-01-31 09:50:27 +00:00
$recaptcha_response = new ReCaptchaResponse ();
$recaptcha_response -> is_valid = false ;
$recaptcha_response -> error = 'incorrect-captcha-sol' ;
return $recaptcha_response ;
}
2012-01-12 08:58:40 +00:00
$response = _recaptcha_http_post ( RECAPTCHA_VERIFY_SERVER , " /recaptcha/api/verify " ,
2011-01-31 09:50:27 +00:00
array (
'privatekey' => $privkey ,
'remoteip' => $remoteip ,
'challenge' => $challenge ,
'response' => $response
2011-09-29 21:31:11 +00:00
) + $extra_params
2011-01-31 09:50:27 +00:00
);
2012-01-12 08:58:40 +00:00
$answers = explode ( " \n " , $response [ 1 ] );
2011-01-31 09:50:27 +00:00
$recaptcha_response = new ReCaptchaResponse ();
2012-01-12 08:58:40 +00:00
if ( trim ( $answers [ 0 ] ) == 'true' ) {
2011-01-31 09:50:27 +00:00
$recaptcha_response -> is_valid = true ;
}
else {
$recaptcha_response -> is_valid = false ;
$recaptcha_response -> error = $answers [ 1 ];
}
return $recaptcha_response ;
}
/**
* gets a URL where the user can sign up for reCAPTCHA . If your application
* has a configuration page where you enter a key , you should provide a link
* using this function .
* @ param string $domain The domain where the page is hosted
* @ param string $appname The name of your application
*/
2012-01-12 08:58:40 +00:00
function recaptcha_get_signup_url ( $domain = null , $appname = null ) {
return " https://www.google.com/recaptcha/admin/create? " . _recaptcha_qsencode ( array ( 'domains' => $domain , 'app' => $appname ) );
2011-09-29 21:31:11 +00:00
}
2012-01-12 08:58:40 +00:00
function _recaptcha_aes_pad ( $val ) {
2011-09-29 21:31:11 +00:00
$block_size = 16 ;
2012-01-12 08:58:40 +00:00
$numpad = $block_size - ( strlen ( $val ) % $block_size );
return str_pad ( $val , strlen ( $val ) + $numpad , chr ( $numpad ) );
2011-01-31 09:50:27 +00:00
}
/* Mailhide related code */
2012-01-12 08:58:40 +00:00
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. " );
2011-01-31 09:50:27 +00:00
}
2012-01-12 08:58:40 +00:00
$mode = MCRYPT_MODE_CBC ;
$enc = MCRYPT_RIJNDAEL_128 ;
$val = _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 " );
2011-01-31 09:50:27 +00:00
}
2012-01-12 08:58:40 +00:00
function _recaptcha_mailhide_urlbase64 ( $x ) {
return strtr ( base64_encode ( $x ), '+/' , '-_' );
2011-01-31 09:50:27 +00:00
}
/* gets the reCAPTCHA Mailhide url for a given email, public key and private key */
2012-01-12 08:58:40 +00:00
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://www.google.com/recaptcha/mailhide/apikey'>http://www.google.com/recaptcha/mailhide/apikey</a> " );
2011-01-31 09:50:27 +00:00
}
2011-10-14 21:30:16 +00:00
2011-01-31 09:50:27 +00:00
2012-01-12 08:58:40 +00:00
$ky = pack ( 'H*' , $privkey );
$cryptmail = _recaptcha_aes_encrypt ( $email , $ky );
2011-10-14 21:30:16 +00:00
2012-01-12 08:58:40 +00:00
return " http://www.google.com/recaptcha/mailhide/d?k= " . $pubkey . " &c= " . _recaptcha_mailhide_urlbase64 ( $cryptmail );
2011-01-31 09:50:27 +00:00
}
/**
* 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
*/
2012-01-12 08:58:40 +00:00
function _recaptcha_mailhide_email_parts ( $email ) {
$arr = preg_split ( " /@/ " , $email );
2011-01-31 09:50:27 +00:00
2012-01-12 08:58:40 +00:00
if ( strlen ( $arr [ 0 ] ) <= 4 ) {
$arr [ 0 ] = substr ( $arr [ 0 ], 0 , 1 );
} elseif ( strlen ( $arr [ 0 ] ) <= 6 ) {
$arr [ 0 ] = substr ( $arr [ 0 ], 0 , 3 );
2011-01-31 09:50:27 +00:00
} else {
2012-01-12 08:58:40 +00:00
$arr [ 0 ] = substr ( $arr [ 0 ], 0 , 4 );
2011-01-31 09:50:27 +00:00
}
return $arr ;
}
/**
* Gets html to display an email address given a public an private key .
* to get a key , go to :
*
2011-09-29 21:31:11 +00:00
* http :// www . google . com / recaptcha / mailhide / apikey
2011-01-31 09:50:27 +00:00
*/
2012-01-12 08:58:40 +00:00
function recaptcha_mailhide_html ( $pubkey , $privkey , $email ) {
$emailparts = _recaptcha_mailhide_email_parts ( $email );
$url = recaptcha_mailhide_url ( $pubkey , $privkey , $email );
2011-10-14 21:30:16 +00:00
2012-01-12 08:58:40 +00:00
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 ] );
2011-01-31 09:50:27 +00:00
}