Captchalot
A dictionary/symbol based captcha generator and validator RESTful service
Copyright (C) 2014 Gael Abadinbr/
License: MIT Expat
, (*1)
, (*2)
Motivation
I wanted to implement a simple, easy to use, scalable and independent RESTful captcha service on my web app., (*3)
Requirements
- PHP >= 5.3 with PDO support
- MySQL / MariaDB, Postgres; or a PDO supported DB
Deployment
- You can install and deploy captchalot using composer:
php composer.phar create-project -s "beta" elcodedocle/captchalot
-
You need to edit config.php.dist
in order to provide basic database connection parameters; then save it as config.php
., (*4)
-
Here is a basic client written in javascript to AJAX request/refresh/validate a captcha:, (*5)
var captchalot = {
'validate' : function(opts){
let XHR = new XMLHttpRequest(),
responseJSON,
parameters,
options = (typeof (opts) === 'object')?opts:{
uuid: '',
magicword: '',
width: 350,
height: 50,
callbackSuccess: function(responseJSON){ alert('success!'); console.log(responseJSON); },
callbackError: function(responseJSON){ alert('error!'); console.log(responseJSON); }
};
XHR.addEventListener("load", function(event) {
try {
responseJSON = JSON.parse(event.target.responseText);
if (
!('data' in responseJSON) ||
!('validationResult' in responseJSON.data) ||
responseJSON.data['validationResult'] !== 'ERROR'
&& responseJSON.data['validationResult'] !== 'PENDING'
&& responseJSON.data['validationResult'] !== 'OK'
){
console.log(
"Cannot understand response from server:\n"
+ responseJSON
);
options.callbackError(responseJSON);
} else {
if ( responseJSON.data['validationResult'] === 'OK' ){
options.callbackSuccess(responseJSON);
} else {
options.callbackError(responseJSON);
}
}
} catch (e) {
console.error("Parsing error:", e);
console.log(event.target.responseText);
}
});
XHR.addEventListener("error", function(event) {
alert('Something went wrong ¯\\(º_o)/¯');
console.log(event.target.responseText);
});
XHR.open("POST", 'validate.php', true);
XHR.setRequestHeader(
'Content-type',
'application/x-www-form-urlencoded'
);
parameters = 'uuid='
+ options.uuid
+ '&magicword='
+ encodeURIComponent(options.magicword)
+ '&width='+options.width
+ '&height='+options.height;
XHR.send(parameters);
}
};
- And here is the server-side PHP controller for this client:
<?php
require_once __DIR__ . '/vendor/autoload.php';
use info\synapp\tools\captcha\Session;
use info\synapp\tools\uuid\UUID;
use info\synapp\tools\captcha\CaptchaWord;
use info\synapp\tools\captcha\CaptchaImage;
use info\synapp\tools\captcha\Captcha;
session_start();
ob_start();
$VALIDATION_RESULT_PENDING = 'PENDING';
$VALIDATION_RESULT_OK = 'OK';
$VALIDATION_RESULT_ERROR = 'ERROR';
$session = new Session(session_id());
$session->removeExpiredCaptchas();
$uuidGenerator = new UUID();
$captchaWord = new CaptchaWord();
$options=array('options'=>array('default'=>420, 'min_range'=>70, 'max_range'=>4200));
$width=filter_input(INPUT_POST, 'width', FILTER_VALIDATE_INT, $options);
$options=array('options'=>array('default'=>60, 'min_range'=>10, 'max_range'=>600));
$height=filter_input(INPUT_POST, 'height', FILTER_VALIDATE_INT, $options);
$captchaImage = new CaptchaImage($width,$height);
$captcha = new Captcha($session,null,$uuidGenerator,$captchaWord,$captchaImage);
$output = array(
'data' => array(
'validationResult' => $VALIDATION_RESULT_PENDING,
),
);
if (
isset($_REQUEST['uuid'])
&& $_REQUEST['uuid']!==''
&& isset($_REQUEST['magicword'])
&& $_REQUEST['magicword']!==''
){
if($captcha->validate($_REQUEST['uuid'],$_REQUEST['magicword'])){
$output['data']['validationResult'] = $VALIDATION_RESULT_OK;
} else {
$output['data']['validationResult'] = $VALIDATION_RESULT_ERROR;
}
}
if($output['data']['validationResult']!==$VALIDATION_RESULT_OK){
$output['data']=array_merge(
$output['data'],
$captcha->getCaptchaUuidAndImgBase64SrcAttrVal()
);
}
header('Content-type: application/json');
header('Expires: Sun, 1 Jan 2000 12:00:00 GMT');
header('Last-Modified: '.gmdate("D, d M Y H:i:s").'GMT');
header('Cache-Control: no-store, no-cache, must-revalidate');
header('Cache-Control: post-check=0, pre-check=0', false);
header('Pragma: no-cache');
$jsonOutput = json_encode($output);
echo $jsonOutput;
ob_end_flush();
- Check the code (or generate the docs using phpdocumentor) if you want more info on tweaks and supported config/input parameters and values.
Web app
You can test the code above on the provided web app (validate.php, captchalot.js and index.php) you can test
by giving those files public execution/access permissions on your web server, then pointing your browser to index.php
, (*6)
Here is the demo: https://synapp.info/tools/captchalot, (*7)
Service pitfalls
The web app was designed in the simplest possible way for embedding on a couple of quite low load services that very seldom require captcha actions. That means it wasn't designed with efficiency or performance in mind, although scalability was considered (if your server starts to choke just move the captcha service to AWS or something like that and start throwing on-demand instances at the problem ;-))., (*8)
Also, it doesn't implement accessibility features such as an audio reader for blind or short sighted people (sorry :-(), (*9)
Acks
Some anonymous internet citizen for posting a blog entry years ago (which I wasn't able to find back) showing how to use imagepng for creating an image and drawing lines and adding text to it
Peter Norvig, publisher of the compilation of the 1/3 million most frequent English words on the natural language corpus data from where the word list used by the default dictionary source for this project has been derived., (*10)
And that's all for now, folks. If you like this project, feel free to buy me a beer ;-), (*11)
bitcoin: 1G4d1Ak4aRXqN8SFqiiiGMFSaffxFbu5EX, (*12)
dogecoin: D8axrRWBZA686kEey1rCXXXamjGg9f6A6s, (*13)
paypal: http://goo.gl/Q2kRFG, (*14)
Have fun.-, (*15)