2013-04-17 17:24:50 +02:00
|
|
|
<?php
|
|
|
|
|
|
|
|
# Copyright (c) 2009-2013 Yubico AB
|
|
|
|
# All rights reserved.
|
|
|
|
#
|
|
|
|
# Redistribution and use in source and binary forms, with or without
|
|
|
|
# modification, are permitted provided that the following conditions are
|
|
|
|
# met:
|
|
|
|
#
|
|
|
|
# * Redistributions of source code must retain the above copyright
|
|
|
|
# notice, this list of conditions and the following disclaimer.
|
|
|
|
#
|
|
|
|
# * Redistributions in binary form must reproduce the above
|
|
|
|
# copyright notice, this list of conditions and the following
|
|
|
|
# disclaimer in the documentation and/or other materials provided
|
|
|
|
# with the distribution.
|
|
|
|
#
|
|
|
|
# THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
|
|
|
# "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
|
|
|
|
# LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
|
|
|
|
# A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
|
|
|
|
# OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
|
|
|
|
# SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
|
|
|
|
# LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
|
|
|
|
# DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
|
|
|
|
# THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
|
|
|
# (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
|
|
|
# OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
|
|
|
|
|
|
define('S_OK', 'OK');
|
|
|
|
define('S_BAD_OTP', 'BAD_OTP');
|
|
|
|
define('S_REPLAYED_OTP', 'REPLAYED_OTP');
|
|
|
|
define('S_DELAYED_OTP', 'DELAYED_OTP');
|
|
|
|
define('S_BAD_SIGNATURE', 'BAD_SIGNATURE');
|
|
|
|
define('S_MISSING_PARAMETER', 'MISSING_PARAMETER');
|
|
|
|
define('S_NO_SUCH_CLIENT', 'NO_SUCH_CLIENT');
|
|
|
|
define('S_OPERATION_NOT_ALLOWED', 'OPERATION_NOT_ALLOWED');
|
|
|
|
define('S_BACKEND_ERROR', 'BACKEND_ERROR');
|
|
|
|
define('S_NOT_ENOUGH_ANSWERS', 'NOT_ENOUGH_ANSWERS');
|
|
|
|
define('S_REPLAYED_REQUEST', 'REPLAYED_REQUEST');
|
|
|
|
|
|
|
|
|
|
|
|
define('TS_SEC', 1/8);
|
|
|
|
define('TS_REL_TOLERANCE', 0.3);
|
|
|
|
define('TS_ABS_TOLERANCE', 20);
|
|
|
|
|
|
|
|
define('TOKEN_LEN', 32);
|
|
|
|
define('OTP_MAX_LEN', 48); // TOKEN_LEN plus public identity of 0..16
|
|
|
|
|
|
|
|
function logdie ($logger, $str)
|
|
|
|
{
|
|
|
|
$logger->log(LOG_INFO, $str);
|
|
|
|
die($str . "\n");
|
|
|
|
}
|
|
|
|
|
|
|
|
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];
|
|
|
|
}
|
2015-07-16 22:14:36 +02:00
|
|
|
$v = trim($val);
|
2015-07-17 10:11:43 +02:00
|
|
|
$v = str_replace('\\', "", $v);
|
2013-04-17 17:24:50 +02:00
|
|
|
return $v;
|
|
|
|
}
|
|
|
|
|
|
|
|
function log_format() {
|
|
|
|
$str = "";
|
|
|
|
foreach (func_get_args() as $msg)
|
|
|
|
{
|
|
|
|
if (is_array($msg)) {
|
|
|
|
foreach($msg as $key => $value){
|
|
|
|
$str .= "$key=$value ";
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
$str .= $msg . " ";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return $str;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Sign a http query string in the array of key-value pairs
|
|
|
|
// return b64 encoded hmac hash
|
|
|
|
function sign($a, $apiKey, $logger) {
|
|
|
|
ksort($a);
|
|
|
|
$qs = urldecode(http_build_query($a));
|
|
|
|
|
|
|
|
// 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);
|
|
|
|
|
|
|
|
$logger->log(LOG_DEBUG, 'SIGN: ' . $qs . ' H=' . $hmac);
|
|
|
|
|
|
|
|
return $hmac;
|
|
|
|
|
|
|
|
} // sign an array of query string
|
|
|
|
|
2015-07-16 22:26:10 +02:00
|
|
|
function curl_settings($logger, $ident, $handle, $url, $timeout, $curlopts)
|
|
|
|
{
|
|
|
|
$logger->log(LOG_DEBUG, $ident . ' adding URL : ' . $url);
|
|
|
|
|
|
|
|
curl_setopt($handle, CURLOPT_URL, $url);
|
|
|
|
curl_setopt($handle, CURLOPT_TIMEOUT, $timeout);
|
|
|
|
curl_setopt($handle, CURLOPT_USERAGENT, 'YK-VAL');
|
|
|
|
curl_setopt($handle, CURLOPT_RETURNTRANSFER, TRUE);
|
|
|
|
curl_setopt($handle, CURLOPT_FAILONERROR, TRUE);
|
|
|
|
|
|
|
|
if (is_array($curlopts) === FALSE)
|
|
|
|
{
|
|
|
|
$logger->log(LOG_WARN, $ident . 'curl options must be an array');
|
|
|
|
return;
|
|
|
|
}
|
2015-04-13 17:35:02 +02:00
|
|
|
|
2015-07-16 22:26:10 +02:00
|
|
|
foreach ($curlopts as $key => $val)
|
|
|
|
{
|
|
|
|
if (curl_setopt($handle, $key, $val) === FALSE)
|
|
|
|
{
|
|
|
|
$logger->log(LOG_WARN, $ident . ' failed to set ' . curl_opt_name($key));
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
}
|
2015-04-13 17:35:02 +02:00
|
|
|
}
|
|
|
|
|
2015-07-16 22:26:10 +02:00
|
|
|
// returns the string name of a curl constant,
|
2015-04-13 17:35:02 +02:00
|
|
|
// or "curl option" if constant not found.
|
|
|
|
// e.g.
|
|
|
|
// curl_opt_name(CURLOPT_URL) returns "CURLOPT_URL"
|
|
|
|
// curl_opt_name(CURLOPT_BLABLA) returns "curl option"
|
2015-07-16 22:26:10 +02:00
|
|
|
function curl_opt_name($opt)
|
|
|
|
{
|
|
|
|
$consts = get_defined_constants(true);
|
|
|
|
$consts = $consts['curl'];
|
2015-04-13 17:35:02 +02:00
|
|
|
|
2015-07-16 22:26:10 +02:00
|
|
|
$name = array_search($opt, $consts, TRUE);
|
2015-04-13 17:35:02 +02:00
|
|
|
|
2015-07-16 22:26:10 +02:00
|
|
|
// array_search may return either on failure...
|
|
|
|
if ($name === FALSE || $name === NULL)
|
|
|
|
return 'curl option';
|
2015-04-13 17:35:02 +02:00
|
|
|
|
2015-07-16 22:26:10 +02:00
|
|
|
return $name;
|
2015-04-13 17:35:02 +02:00
|
|
|
}
|
|
|
|
|
2013-04-17 17:24:50 +02:00
|
|
|
// This function takes a list of URLs. It will return the content of
|
|
|
|
// the first successfully retrieved URL, whose content matches ^OK.
|
|
|
|
// The request are sent asynchronously. Some of the URLs can fail
|
|
|
|
// with unknown host, connection errors, or network timeout, but as
|
|
|
|
// long as one of the URLs given work, data will be returned. If all
|
|
|
|
// URLs fail, data from some URL that did not match parameter $match
|
|
|
|
// (defaults to ^OK) is returned, or if all URLs failed, false.
|
2015-07-17 21:54:51 +02:00
|
|
|
function retrieveURLasync($ident, $urls, $logger, $ans_req=1, $match="^OK", $returl=False, $timeout=10, $curlopts)
|
|
|
|
{
|
|
|
|
$mh = curl_multi_init();
|
|
|
|
$ch = array();
|
2013-04-17 17:24:50 +02:00
|
|
|
|
2015-07-17 22:03:49 +02:00
|
|
|
foreach ($urls as $url)
|
2015-07-17 21:54:51 +02:00
|
|
|
{
|
|
|
|
$handle = curl_init();
|
|
|
|
curl_settings($logger, $ident, $handle, $url, $timeout, $curlopts);
|
|
|
|
curl_multi_add_handle($mh, $handle);
|
|
|
|
$ch[$handle] = $handle;
|
2013-04-17 17:24:50 +02:00
|
|
|
}
|
|
|
|
|
2015-07-17 21:54:51 +02:00
|
|
|
$str = false;
|
|
|
|
$ans_count = 0;
|
|
|
|
$ans_arr = array();
|
2013-04-17 17:24:50 +02:00
|
|
|
|
2015-07-17 21:54:51 +02:00
|
|
|
do
|
|
|
|
{
|
2015-07-17 21:55:46 +02:00
|
|
|
while (curl_multi_exec($mh, $active) == CURLM_CALL_MULTI_PERFORM);
|
2013-04-17 17:24:50 +02:00
|
|
|
|
2015-07-17 21:54:51 +02:00
|
|
|
while ($info = curl_multi_info_read($mh))
|
|
|
|
{
|
|
|
|
$logger->log(LOG_DEBUG, $ident . " curl multi info : ", $info);
|
|
|
|
|
|
|
|
if ($info['result'] == CURLE_OK)
|
|
|
|
{
|
|
|
|
$str = curl_multi_getcontent($info['handle']);
|
2015-07-17 22:00:11 +02:00
|
|
|
$logger->log(LOG_DEBUG, "$ident curl multi content : $str");
|
2015-07-17 21:57:16 +02:00
|
|
|
|
2015-07-17 22:00:11 +02:00
|
|
|
if (preg_match("/$match/", $str))
|
2015-07-17 21:57:16 +02:00
|
|
|
{
|
2015-07-17 22:00:11 +02:00
|
|
|
$logger->log(LOG_DEBUG, "$ident response matches $match");
|
2015-07-17 21:57:16 +02:00
|
|
|
$error = curl_error($info['handle']);
|
|
|
|
$errno = curl_errno($info['handle']);
|
|
|
|
$cinfo = curl_getinfo($info['handle']);
|
2015-07-17 22:00:11 +02:00
|
|
|
$logger->log(LOG_INFO, "$ident errno/error: $errno/$error", $cinfo);
|
2015-07-17 21:54:51 +02:00
|
|
|
$ans_count++;
|
2015-07-17 22:04:09 +02:00
|
|
|
|
|
|
|
if ($returl)
|
|
|
|
$ans_arr[] = "url=" . $cinfo['url'] . "\n" . $str;
|
|
|
|
else
|
|
|
|
$ans_arr[] = $str;
|
2015-07-17 21:54:51 +02:00
|
|
|
}
|
|
|
|
|
2015-07-17 21:57:16 +02:00
|
|
|
if ($ans_count >= $ans_req)
|
|
|
|
{
|
|
|
|
foreach ($ch as $h)
|
|
|
|
{
|
|
|
|
curl_multi_remove_handle($mh, $h);
|
|
|
|
curl_close($h);
|
2015-07-17 21:54:51 +02:00
|
|
|
}
|
2015-07-17 21:57:16 +02:00
|
|
|
curl_multi_close($mh);
|
2015-07-17 21:54:51 +02:00
|
|
|
|
|
|
|
return $ans_arr;
|
|
|
|
}
|
|
|
|
|
2015-07-17 21:57:16 +02:00
|
|
|
curl_multi_remove_handle($mh, $info['handle']);
|
|
|
|
curl_close($info['handle']);
|
|
|
|
unset($ch[$info['handle']]);
|
2015-07-17 21:54:51 +02:00
|
|
|
}
|
|
|
|
|
2015-07-17 21:57:16 +02:00
|
|
|
curl_multi_select($mh);
|
2015-07-17 21:54:51 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
while($active);
|
2013-04-17 17:24:50 +02:00
|
|
|
|
2015-07-17 21:54:51 +02:00
|
|
|
foreach ($ch as $h)
|
|
|
|
{
|
2015-07-17 21:57:16 +02:00
|
|
|
curl_multi_remove_handle($mh, $h);
|
|
|
|
curl_close($h);
|
2015-07-17 21:54:51 +02:00
|
|
|
}
|
2015-07-17 21:57:16 +02:00
|
|
|
curl_multi_close($mh);
|
2015-07-17 21:54:51 +02:00
|
|
|
|
2015-07-17 22:04:09 +02:00
|
|
|
if ($ans_count > 0)
|
|
|
|
return $ans_arr;
|
|
|
|
|
2015-07-17 21:54:51 +02:00
|
|
|
return $str;
|
2013-04-17 17:24:50 +02:00
|
|
|
}
|
|
|
|
|
2015-07-17 21:38:00 +02:00
|
|
|
function KSMdecryptOTP($urls, $logger, $curlopts)
|
|
|
|
{
|
|
|
|
$ret = array();
|
2013-10-16 13:56:34 +02:00
|
|
|
|
2015-07-17 21:38:00 +02:00
|
|
|
if (!is_array($urls))
|
|
|
|
{
|
|
|
|
$urls = array($urls);
|
|
|
|
}
|
|
|
|
|
2015-07-17 21:50:28 +02:00
|
|
|
$response = retrieveURLasync('YK-KSM', $urls, $logger, $ans_req=1, $match='^OK', $returl=False, $timeout=10, $curlopts);
|
2015-07-17 21:38:00 +02:00
|
|
|
|
|
|
|
if (is_array($response))
|
|
|
|
{
|
|
|
|
$response = $response[0];
|
|
|
|
}
|
|
|
|
|
|
|
|
if ($response)
|
|
|
|
{
|
2015-07-17 21:50:28 +02:00
|
|
|
$logger->log(LOG_DEBUG, log_format('YK-KSM response: ', $response));
|
2015-07-17 21:38:00 +02:00
|
|
|
}
|
|
|
|
|
2015-07-17 21:50:28 +02:00
|
|
|
if (sscanf($response,
|
|
|
|
'OK counter=%04x low=%04x high=%02x use=%02x',
|
|
|
|
$ret['session_counter'],
|
|
|
|
$ret['low'],
|
|
|
|
$ret['high'],
|
|
|
|
$ret['session_use'] !== 4)
|
2015-07-17 21:38:00 +02:00
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return $ret;
|
2015-07-16 22:26:10 +02:00
|
|
|
}
|
2013-04-17 17:24:50 +02:00
|
|
|
|
|
|
|
function sendResp($status, $logger, $apiKey = '', $extra = null) {
|
|
|
|
$a['status'] = $status;
|
2015-07-17 00:42:42 +02:00
|
|
|
|
|
|
|
// 2008-11-21T06:11:55Z0711
|
|
|
|
$t = substr(microtime(false), 2, 3);
|
2015-07-17 00:43:11 +02:00
|
|
|
$t = gmdate('Y-m-d\TH:i:s\Z0') . $t;
|
2015-07-17 00:42:42 +02:00
|
|
|
|
|
|
|
$a['t'] = $t;
|
2015-07-17 00:29:22 +02:00
|
|
|
|
|
|
|
if ($extra)
|
|
|
|
foreach ($extra as $param => $value)
|
|
|
|
$a[$param] = $value;
|
|
|
|
|
2013-04-17 17:24:50 +02:00
|
|
|
$h = sign($a, $apiKey, $logger);
|
|
|
|
|
2015-07-17 00:33:53 +02:00
|
|
|
$str = "";
|
|
|
|
$str .= "h=" . $h . "\r\n";
|
2015-07-17 00:29:22 +02:00
|
|
|
$str .= "t=" . $a['t'] . "\r\n";
|
2015-07-17 00:33:53 +02:00
|
|
|
|
|
|
|
if ($extra)
|
|
|
|
foreach ($extra as $param => $value)
|
2013-04-17 17:24:50 +02:00
|
|
|
$str .= $param . "=" . $value . "\r\n";
|
2015-07-17 00:33:53 +02:00
|
|
|
|
2015-07-17 00:29:22 +02:00
|
|
|
$str .= "status=" . $a['status'] . "\r\n";
|
2013-04-17 17:24:50 +02:00
|
|
|
$str .= "\r\n";
|
|
|
|
|
2015-07-17 00:42:42 +02:00
|
|
|
$logger->log(LOG_INFO, "Response: " . $str . " (at " . gmdate("c") . " " . microtime() . ")");
|
2013-04-17 17:24:50 +02:00
|
|
|
|
|
|
|
echo $str;
|
2015-07-16 22:47:16 +02:00
|
|
|
exit;
|
2013-04-17 17:24:50 +02:00
|
|
|
}
|
2014-09-27 15:40:53 +02:00
|
|
|
|
2015-07-16 22:19:47 +02:00
|
|
|
// backport from PHP 5.6
|
|
|
|
if (function_exists('hash_equals') === FALSE)
|
|
|
|
{
|
|
|
|
function hash_equals($a, $b)
|
|
|
|
{
|
|
|
|
// hashes are a (known) fixed length,
|
|
|
|
// so this doesn't leak anything.
|
|
|
|
if (strlen($a) != strlen($b))
|
|
|
|
return false;
|
2014-09-27 15:40:53 +02:00
|
|
|
|
2015-07-16 22:19:47 +02:00
|
|
|
$result = 0;
|
|
|
|
|
|
|
|
for ($i = 0; $i < strlen($a); $i++)
|
|
|
|
$result |= ord($a[$i]) ^ ord($b[$i]);
|
|
|
|
|
|
|
|
return (0 === $result);
|
|
|
|
}
|
2014-09-27 15:40:53 +02:00
|
|
|
}
|