123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525 |
- <?php
- /* Copyright (C) 2013 Laurent Destailleur <eldy@users.sourceforge.net>
- *
- * This program is free software; you can redistribute it and/or modify
- * it under the terms of the GNU General Public License as published by
- * the Free Software Foundation; either version 3 of the License, or
- * (at your option) any later version.
- *
- * This program is distributed in the hope that it will be useful,
- * but WITHOUT ANY WARRANTY; without even the implied warranty of
- * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
- * GNU General Public License for more details.
- *
- * You should have received a copy of the GNU General Public License
- * along with this program. If not, see <http://www.gnu.org/licenses/>.
- */
- /**
- * \file htdocs/core/class/openid.class.php
- * \ingroup core
- * \brief Class to manage authentication with OpenId
- */
- /**
- * Class to manage OpenID
- */
- class SimpleOpenID
- {
- var $openid_url_identity;
- var $URLs = array();
- var $error = array();
- var $fields = array(
- 'required' => array(),
- 'optional' => array(),
- );
- /**
- * Constructor
- */
- function __construct()
- {
- if (!function_exists('curl_exec'))
- {
- die('Error: Class SimpleOpenID requires curl extension to work');
- }
- }
- /**
- * SetOpenIDServer
- *
- * @param string $a Server
- * @return void
- */
- function SetOpenIDServer($a)
- {
- $this->URLs['openid_server'] = $a;
- }
- /**
- * SetOpenIDServer
- *
- * @param string $a Server
- * @return void
- */
- function SetTrustRoot($a)
- {
- $this->URLs['trust_root'] = $a;
- }
- /**
- * SetOpenIDServer
- *
- * @param string $a Server
- * @return void
- */
- function SetCancelURL($a)
- {
- $this->URLs['cancel'] = $a;
- }
- /**
- * SetApprovedURL
- *
- * @param string $a Server
- * @return void
- */
- function SetApprovedURL($a)
- {
- $this->URLs['approved'] = $a;
- }
- /**
- * SetRequiredFields
- *
- * @param string $a Server
- * @return void
- */
- function SetRequiredFields($a)
- {
- if (is_array($a)){
- $this->fields['required'] = $a;
- }else{
- $this->fields['required'][] = $a;
- }
- }
- /**
- * SetOptionalFields
- *
- * @param string $a Server
- * @return void
- */
- function SetOptionalFields($a)
- {
- if (is_array($a)){
- $this->fields['optional'] = $a;
- }else{
- $this->fields['optional'][] = $a;
- }
- }
- /**
- * SetIdentity
- *
- * @param string $a Server
- * @return void
- */
- function SetIdentity($a)
- { // Set Identity URL
- if ((stripos($a, 'http://') === false)
- && (stripos($a, 'https://') === false)){
- $a = 'http://'.$a;
- }
- /*
- $u = parse_url(trim($a));
- if (!isset($u['path'])){
- $u['path'] = '/';
- }else if(substr($u['path'],-1,1) == '/'){
- $u['path'] = substr($u['path'], 0, strlen($u['path'])-1);
- }
- if (isset($u['query'])){ // If there is a query string, then use identity as is
- $identity = $a;
- }else{
- $identity = $u['scheme'] . '://' . $u['host'] . $u['path'];
- }
- //*/
- $this->openid_url_identity = $a;
- }
- /**
- * GetIdentity
- *
- * @return string
- */
- function GetIdentity()
- { // Get Identity
- return $this->openid_url_identity;
- }
- /**
- * SetOpenIDServer
- *
- * @return void
- */
- function GetError()
- {
- $e = $this->error;
- return array('code'=>$e[0],'description'=>$e[1]);
- }
- /**
- * ErrorStore
- *
- * @param string $code Code
- * @param string $desc Description
- * @return void
- */
- function ErrorStore($code, $desc = null)
- {
- $errs['OPENID_NOSERVERSFOUND'] = 'Cannot find OpenID Server TAG on Identity page.';
- if ($desc == null){
- $desc = $errs[$code];
- }
- $this->error = array($code,$desc);
- }
- /**
- * IsError
- *
- * @return boolean
- */
- function IsError()
- {
- if (count($this->error) > 0)
- {
- return true;
- }
- else
- {
- return false;
- }
- }
- /**
- * splitResponse
- *
- * @param string $response Server
- * @return void
- */
- function splitResponse($response)
- {
- $r = array();
- $response = explode("\n", $response);
- foreach($response as $line) {
- $line = trim($line);
- if ($line != "") {
- list($key, $value) = explode(":", $line, 2);
- $r[trim($key)] = trim($value);
- }
- }
- return $r;
- }
- /**
- * OpenID_Standarize
- *
- * @param string $openid_identity Server
- * @return string
- */
- function OpenID_Standarize($openid_identity = null)
- {
- if ($openid_identity === null)
- $openid_identity = $this->openid_url_identity;
- $u = parse_url(strtolower(trim($openid_identity)));
- if (!isset($u['path']) || ($u['path'] == '/')) {
- $u['path'] = '';
- }
- if(substr($u['path'],-1,1) == '/'){
- $u['path'] = substr($u['path'], 0, strlen($u['path'])-1);
- }
- if (isset($u['query'])){ // If there is a query string, then use identity as is
- return $u['host'] . $u['path'] . '?' . $u['query'];
- }else{
- return $u['host'] . $u['path'];
- }
- }
- /**
- * array2url
- *
- * @param array $arr An array
- * @return false|string false if KO, string of url if OK
- */
- function array2url($arr)
- { // converts associated array to URL Query String
- if (!is_array($arr)){
- return false;
- }
- $query = '';
- foreach($arr as $key => $value){
- $query .= $key . "=" . $value . "&";
- }
- return $query;
- }
- /**
- * FSOCK_Request
- *
- * @param string $url URL
- * @param string $method Method
- * @param string $params Params
- * @return boolean|unknown
- */
- function FSOCK_Request($url, $method="GET", $params = "")
- {
- $fp = fsockopen("ssl://www.myopenid.com", 443, $errno, $errstr, 3); // Connection timeout is 3 seconds
- if (!$fp) {
- $this->ErrorStore('OPENID_SOCKETERROR', $errstr);
- return false;
- } else {
- $request = $method . " /server HTTP/1.0\r\n";
- $request .= "User-Agent: Dolibarr\r\n";
- $request .= "Connection: close\r\n\r\n";
- fwrite($fp, $request);
- stream_set_timeout($fp, 4); // Connection response timeout is 4 seconds
- $res = fread($fp, 2000);
- $info = stream_get_meta_data($fp);
- fclose($fp);
- if ($info['timed_out']) {
- $this->ErrorStore('OPENID_SOCKETTIMEOUT');
- } else {
- return $res;
- }
- }
- }
- /**
- * CURL_Request
- *
- * @param string $url URL
- * @param string $method Method
- * @param string $params Params
- * @return string
- */
- function CURL_Request($url, $method="GET", $params = "")
- { // Remember, SSL MUST BE SUPPORTED
- if (is_array($params)) $params = $this->array2url($params);
- $curl = curl_init($url . ($method == "GET" && $params != "" ? "?" . $params : ""));
- @curl_setopt($curl, CURLOPT_FOLLOWLOCATION, true);
- curl_setopt($curl, CURLOPT_HEADER, false);
- curl_setopt($curl, CURLOPT_SSL_VERIFYPEER, false);
- curl_setopt($curl, CURLOPT_HTTPGET, ($method == "GET"));
- curl_setopt($curl, CURLOPT_POST, ($method == "POST"));
- if ($method == "POST") curl_setopt($curl, CURLOPT_POSTFIELDS, $params);
- curl_setopt($curl, CURLOPT_RETURNTRANSFER, true);
- $response = curl_exec($curl);
- if (curl_errno($curl) == 0){
- $response;
- }else{
- $this->ErrorStore('OPENID_CURL', curl_error($curl));
- }
- return $response;
- }
- /**
- * HTML2OpenIDServer
- *
- * @param string $content Content
- * @return array Array of servers
- */
- function HTML2OpenIDServer($content)
- {
- $get = array();
- // Get details of their OpenID server and (optional) delegate
- preg_match_all('/<link[^>]*rel=[\'"]openid.server[\'"][^>]*href=[\'"]([^\'"]+)[\'"][^>]*\/?>/i', $content, $matches1);
- preg_match_all('/<link[^>]*href=\'"([^\'"]+)[\'"][^>]*rel=[\'"]openid.server[\'"][^>]*\/?>/i', $content, $matches2);
- $servers = array_merge($matches1[1], $matches2[1]);
- preg_match_all('/<link[^>]*rel=[\'"]openid.delegate[\'"][^>]*href=[\'"]([^\'"]+)[\'"][^>]*\/?>/i', $content, $matches1);
- preg_match_all('/<link[^>]*href=[\'"]([^\'"]+)[\'"][^>]*rel=[\'"]openid.delegate[\'"][^>]*\/?>/i', $content, $matches2);
- $delegates = array_merge($matches1[1], $matches2[1]);
- $ret = array($servers, $delegates);
- return $ret;
- }
- /**
- * Get openid server
- *
- * @param string $url Url to found endpoint
- * @return string Endpoint
- */
- function GetOpenIDServer($url='')
- {
- global $conf;
- include_once DOL_DOCUMENT_ROOT.'/core/lib/geturl.lib.php';
- if (empty($url)) $url=$conf->global->MAIN_AUTHENTICATION_OPENID_URL;
- $response = getURLContent($url);
- list($servers, $delegates) = $this->HTML2OpenIDServer($response);
- if (count($servers) == 0){
- $this->ErrorStore('OPENID_NOSERVERSFOUND');
- return false;
- }
- if (isset($delegates[0])
- && ($delegates[0] != "")){
- $this->SetIdentity($delegates[0]);
- }
- $this->SetOpenIDServer($servers[0]);
- return $servers[0];
- }
- /**
- * GetRedirectURL
- *
- * @return string
- */
- function GetRedirectURL()
- {
- $params = array();
- $params['openid.return_to'] = urlencode($this->URLs['approved']);
- $params['openid.mode'] = 'checkid_setup';
- $params['openid.identity'] = urlencode($this->openid_url_identity);
- $params['openid.trust_root'] = urlencode($this->URLs['trust_root']);
- if (isset($this->fields['required'])
- && (count($this->fields['required']) > 0)) {
- $params['openid.sreg.required'] = implode(',',$this->fields['required']);
- }
- if (isset($this->fields['optional'])
- && (count($this->fields['optional']) > 0)) {
- $params['openid.sreg.optional'] = implode(',',$this->fields['optional']);
- }
- return $this->URLs['openid_server'] . "?". $this->array2url($params);
- }
- /**
- * Redirect
- *
- * @return void
- */
- function Redirect()
- {
- $redirect_to = $this->GetRedirectURL();
- if (headers_sent())
- { // Use JavaScript to redirect if content has been previously sent (not recommended, but safe)
- echo '<script language="JavaScript" type="text/javascript">window.location=\'';
- echo $redirect_to;
- echo '\';</script>';
- }
- else
- { // Default Header Redirect
- header('Location: ' . $redirect_to);
- }
- }
- /**
- * ValidateWithServer
- *
- * @return boolean
- */
- function ValidateWithServer()
- {
- $params = array(
- 'openid.assoc_handle' => urlencode($_GET['openid_assoc_handle']),
- 'openid.signed' => urlencode($_GET['openid_signed']),
- 'openid.sig' => urlencode($_GET['openid_sig'])
- );
- // Send only required parameters to confirm validity
- $arr_signed = explode(",",str_replace('sreg.','sreg_',$_GET['openid_signed']));
- $num = count($arr_signed);
- for ($i = 0; $i < $num; $i++)
- {
- $s = str_replace('sreg_','sreg.', $arr_signed[$i]);
- $c = $_GET['openid_' . $arr_signed[$i]];
- // if ($c != ""){
- $params['openid.' . $s] = urlencode($c);
- // }
- }
- $params['openid.mode'] = "check_authentication";
- $openid_server = $this->GetOpenIDServer();
- if ($openid_server == false)
- {
- return false;
- }
- $response = $this->CURL_Request($openid_server,'POST',$params);
- $data = $this->splitResponse($response);
- if ($data['is_valid'] == "true")
- {
- return true;
- }
- else
- {
- return false;
- }
- }
- /**
- * Get XRDS response and set possible servers.
- *
- * @param string $url Url of endpoint to request
- * @return string First endpoint OpenID server found. False if it failed to found.
- */
- function sendDiscoveryRequestToGetXRDS($url='')
- {
- global $conf;
- include_once DOL_DOCUMENT_ROOT.'/core/lib/geturl.lib.php';
- if (empty($url)) $url=$conf->global->MAIN_AUTHENTICATION_OPENID_URL;
- dol_syslog(get_class($this).'::sendDiscoveryRequestToGetXRDS get XRDS');
- $addheaders=array('Accept: application/xrds+xml');
- $response = getURLContent($url, 'GET', '', 1, $addheaders);
- /* response should like this:
- <?xml version="1.0" encoding="UTF-8"?>
- <xrds:XRDS xmlns:xrds="xri://$xrds" xmlns="xri://$xrd*($v*2.0)">
- <XRD>
- <Service priority="0">
- <Type>http://specs.openid.net/auth/2.0/server</Type>
- <Type>http://openid.net/srv/ax/1.0</Type>
- ...
- <URI>https://www.google.com/accounts/o8/ud</URI>
- </Service>
- </XRD>
- </xrds:XRDS>
- */
- $content=$response['content'];
- $server='';
- if (preg_match('/'.preg_quote('<URI>','/').'(.*)'.preg_quote('</URI>','/').'/is', $content, $reg))
- {
- $server=$reg[1];
- }
- if (empty($server))
- {
- $this->ErrorStore('OPENID_NOSERVERSFOUND');
- return false;
- }
- else
- {
- dol_syslog(get_class($this).'::sendDiscoveryRequestToGetXRDS found endpoint = '.$server);
- $this->SetOpenIDServer($server);
- return $server;
- }
- }
- }
|