2008-03-10 17:32:32 +00:00
< ? php
2009-05-26 03:23:32 +00:00
/**
2008-03-10 17:32:32 +00:00
* This is a PHP library that handles calling reCAPTCHA .
* - Documentation and latest version
2010-10-06 07:53:25 +00:00
* { @ link http :// code . google . com / apis / recaptcha / docs / php . html }
2008-03-10 17:32:32 +00:00
* - Get a reCAPTCHA API Key
2010-10-06 07:53:25 +00:00
* { @ link https :// www . google . com / recaptcha / admin / create }
2008-03-10 17:32:32 +00:00
* - Discussion group
2009-05-26 03:23:32 +00:00
* { @ link http :// groups . google . com / group / recaptcha }
2008-03-10 17:32:32 +00:00
*
2010-10-06 07:53:25 +00:00
* Copyright ( c ) 2007 reCAPTCHA -- { @ link http :// www . google . com / recaptcha }
2008-03-10 17:32:32 +00:00
* 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 .
2009-05-26 03:23:32 +00:00
*
* @ package moodlecore
2010-10-06 07:53:25 +00:00
* @ copyright ( c ) 2007 reCAPTCHA -- { @ link http :// www . google . com / recaptcha }
2008-03-10 17:32:32 +00:00
*/
/**
* The reCAPTCHA server URL ' s
*/
2010-10-06 07:53:25 +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 " );
2008-03-10 17:32:32 +00:00
/**
* 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 )
2008-06-09 16:53:30 +00:00
$req .= $key . '=' . urlencode ( $value ) . '&' ;
2008-03-10 17:32:32 +00:00
// Cut the last '&'
$req = substr ( $req , 0 , strlen ( $req ) - 1 );
return $req ;
}
/**
* Submits an HTTP POST to a reCAPTCHA server
2009-05-26 03:23:32 +00:00
*
* @ global object
2008-03-10 17:32:32 +00:00
* @ param string $host
* @ param string $path
* @ param array $data
* @ param int port
* @ return array response
*/
2008-03-31 12:13:46 +00:00
function _recaptcha_http_post ( $host , $path , $data , $port = 80 , $https = false ) {
2008-03-11 10:34:50 +00:00
global $CFG ;
2008-03-31 12:13:46 +00:00
$protocol = 'http' ;
if ( $https ) {
$protocol = 'https' ;
}
2008-03-13 15:45:16 +00:00
require_once $CFG -> libdir . '/filelib.php' ;
2008-03-10 17:32:32 +00:00
2008-03-13 15:45:16 +00:00
$req = _recaptcha_qsencode ( $data );
2009-06-12 09:09:28 +00:00
2008-03-13 15:45:16 +00:00
$headers = array ();
$headers [ 'Host' ] = $host ;
$headers [ 'Content-Type' ] = 'application/x-www-form-urlencoded' ;
$headers [ 'Content-Length' ] = strlen ( $req );
$headers [ 'User-Agent' ] = 'reCAPTCHA/PHP' ;
2009-06-12 09:09:28 +00:00
2008-03-31 12:13:46 +00:00
$results = download_file_content ( " $protocol :// " . $host . $path , $headers , $data , false , 300 , 20 , true );
2009-06-12 09:09:28 +00:00
2008-03-13 15:45:16 +00:00
if ( $results ) {
return array ( 1 => $results );
2008-03-11 10:34:50 +00:00
} else {
return false ;
2008-03-10 17:32:32 +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 .
2009-05-26 03:23:32 +00:00
*
* @ global object
2008-03-10 17:32:32 +00:00
* @ 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 .
*/
2008-03-13 15:45:16 +00:00
function recaptcha_get_html ( $pubkey , $error = null , $use_ssl = false ) {
2009-06-24 07:39:57 +00:00
global $CFG , $PAGE ;
2008-03-13 15:45:16 +00:00
$recaptchatype = optional_param ( 'recaptcha' , 'image' , PARAM_TEXT );
2009-06-12 09:09:28 +00:00
if ( $pubkey == null || $pubkey == '' ) {
2010-10-06 07:53:25 +00:00
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> " );
2009-06-12 09:09:28 +00:00
}
2008-03-10 17:32:32 +00:00
2009-06-12 09:09:28 +00:00
if ( $use_ssl ) {
$server = RECAPTCHA_API_SECURE_SERVER ;
} else {
$server = RECAPTCHA_API_SERVER ;
}
$errorpart = " " ;
if ( $error ) {
$errorpart = " &error= " . $error ;
}
2008-03-10 17:32:32 +00:00
2008-03-13 15:45:16 +00:00
require_once $CFG -> libdir . '/filelib.php' ;
2008-03-31 12:13:46 +00:00
$html = download_file_content ( $server . '/noscript?k=' . $pubkey . $errorpart , null , null , false , 300 , 20 , true );
2008-03-13 15:45:16 +00:00
preg_match ( '/image\?c\=([A-Za-z0-9\-\_]*)\"/' , $html , $matches );
$challenge_hash = $matches [ 1 ];
$image_url = $server . '/image?c=' . $challenge_hash ;
2009-06-12 09:09:28 +00:00
2008-03-13 15:45:16 +00:00
$strincorrectpleasetryagain = get_string ( 'incorrectpleasetryagain' , 'auth' );
$strenterthewordsabove = get_string ( 'enterthewordsabove' , 'auth' );
$strenterthenumbersyouhear = get_string ( 'enterthenumbersyouhear' , 'auth' );
$strgetanothercaptcha = get_string ( 'getanothercaptcha' , 'auth' );
$strgetanaudiocaptcha = get_string ( 'getanaudiocaptcha' , 'auth' );
$strgetanimagecaptcha = get_string ( 'getanimagecaptcha' , 'auth' );
2008-09-26 11:24:52 +00:00
2010-01-26 10:22:58 +00:00
$return = html_writer :: script ( '' , $server . '/challenge?k=' . $pubkey . $errorpart );
2008-09-26 11:24:52 +00:00
$return .= ' < noscript >
2008-03-13 15:45:16 +00:00
< div id = " recaptcha_widget_noscript " >
< div id = " recaptcha_image_noscript " >< img src = " ' . $image_url . ' " alt = " reCAPTCHA " /></ div > ' ;
2008-09-26 11:24:52 +00:00
2008-03-13 15:45:16 +00:00
if ( $error == 'incorrect-captcha-sol' ) {
$return .= '<div class="recaptcha_only_if_incorrect_sol" style="color:red">' . $strincorrectpleasetryagain . '</div>' ;
}
if ( $recaptchatype == 'image' ) {
$return .= '<span class="recaptcha_only_if_image">' . $strenterthewordsabove . '</span>' ;
} elseif ( $recaptchatype == 'audio' ) {
$return .= '<span class="recaptcha_only_if_audio">' . $strenterthenumbersyouhear . '</span>' ;
}
$return .= '<input type="text" id="recaptcha_response_field_noscript" name="recaptcha_response_field" />' ;
$return .= '<input type="hidden" id="recaptcha_challenge_field_noscript" name="recaptcha_challenge_field" value="' . $challenge_hash . '" />' ;
$return .= '<div><a href="signup.php">' . $strgetanothercaptcha . '</a></div>' ;
// Disabling audio recaptchas for now: not language-independent
/*
if ( $recaptchatype == 'image' ) {
$return .= '<div class="recaptcha_only_if_image"><a href="signup.php?recaptcha=audio">' . $strgetanaudiocaptcha . '</a></div>' ;
} elseif ( $recaptchatype == 'audio' ) {
$return .= '<div class="recaptcha_only_if_audio"><a href="signup.php?recaptcha=image">' . $strgetanimagecaptcha . '</a></div>' ;
}
*/
$return .= '
</ div >
2009-06-12 09:09:28 +00:00
</ noscript > ' ;
2008-03-13 15:45:16 +00:00
return $return ;
2008-03-10 17:32:32 +00:00
}
/**
* A ReCaptchaResponse is returned from recaptcha_check_answer ()
2009-05-26 03:23:32 +00:00
*
* @ package moodlecore
2010-10-06 07:53:25 +00:00
* @ copyright ( c ) 2007 reCAPTCHA -- { @ link http :// www . google . com / recaptcha }
2008-03-10 17:32:32 +00:00
*/
class ReCaptchaResponse {
var $is_valid ;
var $error ;
}
/**
* 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
* @ return ReCaptchaResponse
*/
2008-03-31 12:13:46 +00:00
function recaptcha_check_answer ( $privkey , $remoteip , $challenge , $response , $https = false )
2008-03-10 17:32:32 +00:00
{
2009-06-12 09:09:28 +00:00
if ( $privkey == null || $privkey == '' ) {
2010-10-06 07:53:25 +00:00
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> " );
2009-06-12 09:09:28 +00:00
}
2008-03-10 17:32:32 +00:00
2009-06-12 09:09:28 +00:00
if ( $remoteip == null || $remoteip == '' ) {
die ( " For security reasons, you must pass the remote ip to reCAPTCHA " );
}
2008-03-10 17:32:32 +00:00
//discard spam submissions
if ( $challenge == null || strlen ( $challenge ) == 0 || $response == null || strlen ( $response ) == 0 ) {
$recaptcha_response = new ReCaptchaResponse ();
$recaptcha_response -> is_valid = false ;
$recaptcha_response -> error = 'incorrect-captcha-sol' ;
return $recaptcha_response ;
}
2010-10-06 07:53:25 +00:00
$response = _recaptcha_http_post ( RECAPTCHA_VERIFY_SERVER , " /recaptcha/api/verify " ,
2008-03-31 12:13:46 +00:00
array (
'privatekey' => $privkey ,
'remoteip' => $remoteip ,
'challenge' => $challenge ,
'response' => $response
),
$https
);
2008-03-10 17:32:32 +00:00
$answers = explode ( " \n " , $response [ 1 ]);
$recaptcha_response = new ReCaptchaResponse ();
if ( trim ( $answers [ 0 ]) == 'true' ) {
$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
*/
function recaptcha_get_signup_url ( $domain = null , $appname = null ) {
2010-10-06 07:53:25 +00:00
return " https://www.google.com/recaptcha/admin/create? " . _recaptcha_qsencode ( array ( 'domains' => $domain , 'app' => $appname ));
2008-03-10 17:32:32 +00:00
}
function _recaptcha_aes_pad ( $val ) {
2009-06-12 09:09:28 +00:00
$block_size = 16 ;
$numpad = $block_size - ( strlen ( $val ) % $block_size );
return str_pad ( $val , strlen ( $val ) + $numpad , chr ( $numpad ));
2008-03-10 17:32:32 +00:00
}
/* Mailhide related code */
function _recaptcha_aes_encrypt ( $val , $ky ) {
2009-06-12 09:09:28 +00:00
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 = _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 " );
2008-03-10 17:32:32 +00:00
}
function _recaptcha_mailhide_urlbase64 ( $x ) {
2009-06-12 09:09:28 +00:00
return strtr ( base64_encode ( $x ), '+/' , '-_' );
2008-03-10 17:32:32 +00:00
}
/* gets the reCAPTCHA Mailhide url for a given email, public key and private key */
function recaptcha_mailhide_url ( $pubkey , $privkey , $email ) {
2009-06-12 09:09:28 +00:00
if ( $pubkey == '' || $pubkey == null || $privkey == " " || $privkey == null ) {
die ( " To use reCAPTCHA Mailhide, you have to sign up for a public and private key, " .
2010-10-06 07:53:25 +00:00
" you can do so at <a href='http://www.google.com/recaptcha/mailhide/apikey'>http://www.google.com/recaptcha/mailhide/apikey</a> " );
2009-06-12 09:09:28 +00:00
}
$ky = pack ( 'H*' , $privkey );
$cryptmail = _recaptcha_aes_encrypt ( $email , $ky );
2010-10-06 07:53:25 +00:00
return " http://www.google.com/recaptcha/mailhide/d?k= " . $pubkey . " &c= " . _recaptcha_mailhide_urlbase64 ( $cryptmail );
2008-03-10 17:32:32 +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
*/
function _recaptcha_mailhide_email_parts ( $email ) {
2009-06-12 09:09:28 +00:00
$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 ;
2008-03-10 17:32:32 +00:00
}
/**
* Gets html to display an email address given a public an private key .
* to get a key , go to :
*
2010-10-06 07:53:25 +00:00
* http :// www . google . com / recaptcha / mailhide / apikey
2008-03-10 17:32:32 +00:00
*/
function recaptcha_mailhide_html ( $pubkey , $privkey , $email ) {
2009-06-12 09:09:28 +00:00
$emailparts = _recaptcha_mailhide_email_parts ( $email );
$url = recaptcha_mailhide_url ( $pubkey , $privkey , $email );
2008-03-10 17:32:32 +00:00
2009-06-12 09:09:28 +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 ]);
2008-03-10 17:32:32 +00:00
2009-06-12 09:09:28 +00:00
}