2008-09-26 03:21:11 +00:00
|
|
|
<?php
|
2009-03-11 01:42:23 +00:00
|
|
|
|
2008-09-26 03:21:11 +00:00
|
|
|
define('S_OK', 'OK');
|
|
|
|
define('S_BAD_OTP', 'BAD_OTP');
|
|
|
|
define('S_REPLAYED_OTP', 'REPLAYED_OTP');
|
2009-03-11 00:19:59 +00:00
|
|
|
define('S_DELAYED_OTP', 'DELAYED_OTP');
|
2008-09-26 03:21:11 +00:00
|
|
|
define('S_BAD_SIGNATURE', 'BAD_SIGNATURE');
|
|
|
|
define('S_MISSING_PARAMETER', 'MISSING_PARAMETER');
|
2009-03-10 22:01:46 +00:00
|
|
|
define('S_NO_SUCH_CLIENT', 'NO_SUCH_CLIENT');
|
2008-09-26 03:21:11 +00:00
|
|
|
define('S_OPERATION_NOT_ALLOWED', 'OPERATION_NOT_ALLOWED');
|
|
|
|
define('S_BACKEND_ERROR', 'BACKEND_ERROR');
|
2009-03-11 01:42:23 +00:00
|
|
|
|
|
|
|
define('TS_SEC', 1/8);
|
2009-03-11 00:26:57 +00:00
|
|
|
define('TS_REL_TOLERANCE', 0.3);
|
|
|
|
define('TS_ABS_TOLERANCE', 20);
|
2008-09-26 03:21:11 +00:00
|
|
|
|
2009-03-18 11:00:48 +00:00
|
|
|
define('TOKEN_LEN', 32);
|
2009-03-10 22:50:35 +00:00
|
|
|
|
|
|
|
function unescape($s) {
|
|
|
|
return str_replace('\\', "", $s);
|
|
|
|
}
|
|
|
|
|
|
|
|
function getHttpVal($key, $defaultVal) {
|
|
|
|
$val = $defaultVal;
|
|
|
|
if (array_key_exists($key, $_GET)) {
|
|
|
|
$val = $_GET[$key];
|
|
|
|
} else if (array_key_exists($key, $_POST)) {
|
|
|
|
$val = $_POST[$key];
|
|
|
|
}
|
|
|
|
$v = unescape(trim($val));
|
|
|
|
return $v;
|
|
|
|
}
|
|
|
|
|
2009-03-11 02:06:02 +00:00
|
|
|
function query($conn, $q) {
|
|
|
|
debug('Query: '.$q);
|
|
|
|
$result = mysql_query($q, $conn);
|
|
|
|
if (!$result) {
|
|
|
|
die("Query error: " . mysql_error());
|
|
|
|
}
|
|
|
|
return $result;
|
2009-03-10 22:50:35 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
function mysql_quote($value) {
|
|
|
|
return "'" . mysql_real_escape_string($value) . "'";
|
|
|
|
}
|
|
|
|
|
2009-03-11 00:54:19 +00:00
|
|
|
function debug($msg) {
|
|
|
|
if (is_array($msg)) {
|
|
|
|
$str = "";
|
|
|
|
foreach($msg as $key => $value){
|
|
|
|
$str .= "$key=$value ";
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
$str = $msg;
|
|
|
|
}
|
|
|
|
error_log($str);
|
2008-09-26 03:21:11 +00:00
|
|
|
}
|
|
|
|
|
2009-03-10 19:01:07 +00:00
|
|
|
// Return eg. 2008-11-21T06:11:55Z0711
|
2008-11-21 06:41:13 +00:00
|
|
|
//
|
2008-09-27 09:04:49 +00:00
|
|
|
function getUTCTimeStamp() {
|
|
|
|
date_default_timezone_set('UTC');
|
2009-03-10 19:01:07 +00:00
|
|
|
$tiny = substr(microtime(false), 2, 3);
|
|
|
|
return date('Y-m-d\TH:i:s\Z0', time()) . $tiny;
|
2008-09-27 09:04:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Sign a http query string in the array of key-value pairs
|
|
|
|
// return b64 encoded hmac hash
|
2009-03-11 02:06:02 +00:00
|
|
|
function sign($a, $apiKey) {
|
2008-09-27 09:04:49 +00:00
|
|
|
ksort($a);
|
|
|
|
$qs = '';
|
|
|
|
$n = count($a);
|
|
|
|
$i = 0;
|
|
|
|
foreach (array_keys($a) as $key) {
|
2008-11-21 21:41:26 +00:00
|
|
|
$qs .= trim($key).'='.trim($a[$key]);
|
2008-09-27 09:04:49 +00:00
|
|
|
if (++$i < $n) {
|
|
|
|
$qs .= '&';
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// the TRUE at the end states we want the raw value, not hexadecimal form
|
|
|
|
$hmac = hash_hmac('sha1', utf8_encode($qs), $apiKey, true);
|
|
|
|
$hmac = base64_encode($hmac);
|
2009-03-11 01:53:38 +00:00
|
|
|
|
|
|
|
debug('SIGN: ' . $qs . ' H=' . $hmac);
|
|
|
|
|
2008-09-27 09:04:49 +00:00
|
|
|
return $hmac;
|
|
|
|
|
2008-10-08 06:29:19 +00:00
|
|
|
} // sign an array of query string
|
2008-09-27 09:04:49 +00:00
|
|
|
|
2009-03-11 01:37:07 +00:00
|
|
|
function hex2b64 ($hex_str) {
|
|
|
|
$bin = pack("H*", $hex_str);
|
|
|
|
return base64_encode($bin);
|
2009-03-10 22:50:35 +00:00
|
|
|
}
|
|
|
|
|
2009-03-11 01:37:07 +00:00
|
|
|
function modhex2b64 ($modhex_str) {
|
|
|
|
$hex_str = strtr ($modhex_str, "cbdefghijklnrtuv", "0123456789abcdef");
|
|
|
|
return hex2b64($hex_str);
|
2009-03-10 22:50:35 +00:00
|
|
|
}
|
|
|
|
|
2009-03-11 01:42:23 +00:00
|
|
|
// $otp: A yubikey OTP
|
|
|
|
function decryptOTP($otp, $base_url) {
|
|
|
|
$url = $base_url . $otp;
|
2009-03-11 01:37:07 +00:00
|
|
|
$ch = curl_init($url);
|
|
|
|
curl_setopt($ch, CURLOPT_USERAGENT, "YK-VAL");
|
|
|
|
curl_setopt($ch, CURLOPT_RETURNTRANSFER, 1);
|
2009-03-11 11:14:50 +00:00
|
|
|
curl_setopt($ch, CURLOPT_FAILONERROR, true);
|
2009-03-11 02:12:36 +00:00
|
|
|
curl_setopt($ch, CURLOPT_TIMEOUT, 5);
|
2009-03-11 01:37:07 +00:00
|
|
|
$response = curl_exec($ch);
|
2009-03-11 11:14:50 +00:00
|
|
|
$error = curl_error ($ch);
|
2009-03-11 11:18:56 +00:00
|
|
|
$errno = curl_errno ($ch);
|
|
|
|
debug("YK-KSM response: $response errno: " . $errno . " error: " . $error);
|
2009-03-11 11:14:50 +00:00
|
|
|
$info = curl_getinfo ($ch);
|
|
|
|
debug($info);
|
2009-03-11 01:37:07 +00:00
|
|
|
curl_close($ch);
|
|
|
|
|
|
|
|
if (sscanf ($response,
|
2009-03-18 15:16:18 +00:00
|
|
|
"OK counter=%04x low=%04x high=%02x use=%02x",
|
2009-03-18 15:21:50 +00:00
|
|
|
$ret["session_counter"], $ret["low"], $ret["high"],
|
|
|
|
$ret["session_use"]) != 4) {
|
2009-03-11 01:37:07 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return $ret;
|
|
|
|
} // End decryptOTP
|
2009-03-10 22:50:35 +00:00
|
|
|
|
|
|
|
// $devId: The first 12 chars from the OTP
|
2009-03-11 02:06:02 +00:00
|
|
|
function getAuthData($conn, $devId) {
|
2009-03-11 01:37:07 +00:00
|
|
|
$tokenId = modhex2b64($devId);
|
2009-03-18 20:44:14 +00:00
|
|
|
$stmt =
|
2009-04-01 16:09:29 +00:00
|
|
|
'SELECT id, active, client_id, counter, sessionUse, low, high, accessed '.
|
2009-03-18 20:44:14 +00:00
|
|
|
'FROM yubikeys '.
|
2009-04-01 16:09:29 +00:00
|
|
|
'WHERE tokenId='.mysql_quote($tokenId);
|
2009-03-11 02:06:02 +00:00
|
|
|
$r = query($conn, $stmt);
|
2009-03-10 22:50:35 +00:00
|
|
|
if (mysql_num_rows($r) > 0) {
|
|
|
|
$row = mysql_fetch_assoc($r);
|
|
|
|
mysql_free_result($r);
|
|
|
|
return $row;
|
|
|
|
}
|
|
|
|
return null;
|
|
|
|
} // End getAuthData
|
|
|
|
|
2009-04-01 16:09:29 +00:00
|
|
|
function addNewKey($conn, $devId) {
|
|
|
|
$tokenId = modhex2b64($devId);
|
|
|
|
$stmt = 'INSERT INTO yubikeys (client_id, active, created, tokenId, counter) '.
|
|
|
|
'VALUES (1, true, NOW(), ' . mysql_quote($tokenId) . ', 0)';
|
|
|
|
$r = query($conn, $stmt);
|
|
|
|
}
|
|
|
|
|
2009-03-10 22:50:35 +00:00
|
|
|
// $clientId: The decimal client identity
|
2009-03-11 02:06:02 +00:00
|
|
|
function getClientData($conn, $clientId) {
|
2009-03-18 20:44:14 +00:00
|
|
|
$stmt =
|
|
|
|
'SELECT id, secret, chk_time '.
|
|
|
|
'FROM clients '.
|
|
|
|
'WHERE active AND id='.mysql_quote($clientId);
|
2009-03-11 02:06:02 +00:00
|
|
|
$r = query($conn, $stmt);
|
2009-03-10 22:50:35 +00:00
|
|
|
if (mysql_num_rows($r) > 0) {
|
|
|
|
$row = mysql_fetch_assoc($r);
|
|
|
|
mysql_free_result($r);
|
|
|
|
return $row;
|
|
|
|
}
|
|
|
|
return null;
|
|
|
|
} // End getClientData
|
2008-09-26 03:21:11 +00:00
|
|
|
?>
|