386 lines
13 KiB
PHP
386 lines
13 KiB
PHP
|
<?php
|
||
|
/* vim: set noexpandtab tabstop=2 softtabstop=2 shiftwidth=2: */
|
||
|
|
||
|
////////////////////////////////////////////////////////////////
|
||
|
//
|
||
|
// File: XMLRPC DB ACCESS 2.1
|
||
|
// Date: 28.05.2009
|
||
|
// Author: Gilles Masson
|
||
|
// Updated: Xymph
|
||
|
//
|
||
|
////////////////////////////////////////////////////////////////
|
||
|
|
||
|
class XmlrpcDB {
|
||
|
|
||
|
//-----------------------------
|
||
|
// Fields
|
||
|
//-----------------------------
|
||
|
|
||
|
var $_webaccess;
|
||
|
var $_url;
|
||
|
var $_server;
|
||
|
var $_callbacks;
|
||
|
var $_requests;
|
||
|
var $_auth_cb;
|
||
|
var $_bad;
|
||
|
var $_bad_time;
|
||
|
var $_debug;
|
||
|
|
||
|
//-----------------------------
|
||
|
// Methods
|
||
|
//-----------------------------
|
||
|
|
||
|
function XmlrpcDB($webaccess, $url, $game, $login, $password, $tool, $version, $nation, $packmask = '') {
|
||
|
|
||
|
$this->_debug = 0; // max debug level = 3
|
||
|
$this->_webaccess = $webaccess;
|
||
|
$this->_url = $url;
|
||
|
$this->_server = array('Game' => $game,
|
||
|
'Login' => $login,
|
||
|
'Password' => $password,
|
||
|
'Tool' => $tool,
|
||
|
'Version' => $version,
|
||
|
'Nation' => $nation,
|
||
|
'Packmask' => $packmask,
|
||
|
'PlayersGame' => true
|
||
|
);
|
||
|
$this->_auth_cb = array('xmlrpc_auth_cb');
|
||
|
|
||
|
$this->_bad = false;
|
||
|
$this->_bad_time = -1;
|
||
|
// in case webaccess URL connection was previously in error, ask to retry
|
||
|
$this->_webaccess->retry($this->_url);
|
||
|
|
||
|
// prepare to add requests
|
||
|
$this->_initRequest();
|
||
|
} // XmlrpcDB
|
||
|
|
||
|
// change the packmask value
|
||
|
function setPackmask($packmask = '') {
|
||
|
|
||
|
$this->_server['Packmask'] = $packmask;
|
||
|
} // setPackmask
|
||
|
|
||
|
// is the connection in recurrent error?
|
||
|
function isBad() {
|
||
|
|
||
|
return $this->_bad;
|
||
|
} // isBad
|
||
|
|
||
|
// get time since the error state was set
|
||
|
function badTime() {
|
||
|
|
||
|
return (time() - $this->_bad_time);
|
||
|
} // badTime
|
||
|
|
||
|
// stop the bad state: will try again at next RequestWait(),
|
||
|
// sendRequestsWait() or sendRequests()
|
||
|
function retry() {
|
||
|
|
||
|
$this->_bad = false;
|
||
|
$this->_bad_time = -1;
|
||
|
// set webaccess object to retry on that URL too
|
||
|
$this->_webaccess->retry($this->_url);
|
||
|
} // retry
|
||
|
|
||
|
// clear all requests, and get them if asked
|
||
|
function clearRequests($get_requests = false) {
|
||
|
|
||
|
if ($get_requests) {
|
||
|
$return = array($_requests, $_callbacks);
|
||
|
$this->_initRequest();
|
||
|
return $return;
|
||
|
}
|
||
|
$this->_initRequest();
|
||
|
} // clearRequests
|
||
|
|
||
|
// add a request
|
||
|
function addRequest($callback, $method) {
|
||
|
|
||
|
$args = func_get_args();
|
||
|
$callback = array_shift($args);
|
||
|
$method = array_shift($args);
|
||
|
return $this->addRequestArray($callback, $method, $args);
|
||
|
} // addRequest
|
||
|
|
||
|
// add a request
|
||
|
function addRequestArray($callback, $method, $args) {
|
||
|
|
||
|
$this->_callbacks[] = $callback;
|
||
|
$this->_requests[] = array('methodName' => $method, 'params' => $args);
|
||
|
return count($this->_requests) - 1;
|
||
|
} // addRequestArray
|
||
|
|
||
|
// send added requests, callbacks will be called when response come
|
||
|
function sendRequests() {
|
||
|
global $aseco;
|
||
|
|
||
|
if (count($this->_callbacks) > 1) {
|
||
|
$this->addRequest(null, 'dedimania.WarningsAndTTR');
|
||
|
$webdatas = $this->_makeXMLdatas();
|
||
|
$response = $this->_webaccess->request($this->_url,
|
||
|
array( array($this, '_callCB'), $this->_callbacks, $this->_requests),
|
||
|
$webdatas, true);
|
||
|
$this->_initRequest();
|
||
|
if ($response === false) {
|
||
|
if (!$this->_bad) {
|
||
|
$this->_bad = true;
|
||
|
$this->_bad_time = time();
|
||
|
}
|
||
|
if ($this->_debug > 2)
|
||
|
$aseco->console_text('XmlrpcDB->sendRequests - this' . CRLF . print_r($this, true));
|
||
|
return false;
|
||
|
}
|
||
|
}
|
||
|
return true;
|
||
|
} // sendRequests
|
||
|
|
||
|
// send added requests, wait response, then call callbacks
|
||
|
function sendRequestsWait() {
|
||
|
global $aseco;
|
||
|
|
||
|
if (count($this->_callbacks) > 1) {
|
||
|
$this->addRequest(null, 'dedimania.WarningsAndTTR');
|
||
|
$webdatas = $this->_makeXMLdatas();
|
||
|
$response = $this->_webaccess->request($this->_url,
|
||
|
null, $webdatas, true);
|
||
|
if ($response === false) {
|
||
|
if (!$this->_bad) {
|
||
|
$this->_bad = true;
|
||
|
$this->_bad_time = time();
|
||
|
}
|
||
|
if ($this->_debug > 0)
|
||
|
$aseco->console_text('XmlrpcDB->sendRequestsWait - this' . CRLF . print_r($this, true));
|
||
|
$this->_initRequest();
|
||
|
return false;
|
||
|
} else {
|
||
|
$this->_callCB($response, $this->_callbacks, $this->_requests);
|
||
|
$this->_initRequest();
|
||
|
}
|
||
|
}
|
||
|
return true;
|
||
|
} // sendRequestsWait
|
||
|
|
||
|
// send a request, wait response, and return the response
|
||
|
function RequestWait($method) {
|
||
|
|
||
|
$args = func_get_args();
|
||
|
$method = array_shift($args);
|
||
|
return $this->RequestWaitArray($method, $args);
|
||
|
} // RequestWait
|
||
|
|
||
|
// send a request, wait response, and return the response
|
||
|
function RequestWaitArray($method, $args) {
|
||
|
global $aseco;
|
||
|
|
||
|
if ($this->sendRequestsWait() === false) {
|
||
|
if (!$this->_bad) {
|
||
|
$this->_bad = true;
|
||
|
$this->_bad_time = time();
|
||
|
}
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
$reqnum = $this->addRequestArray(null, $method, $args);
|
||
|
|
||
|
$this->addRequest(null, 'dedimania.WarningsAndTTR');
|
||
|
$webdatas = $this->_makeXMLdatas();
|
||
|
$response = $this->_webaccess->request($this->_url, null, $webdatas, true);
|
||
|
if (isset($response['Message']) && is_string($response['Message'])) {
|
||
|
if ($this->_debug > 1)
|
||
|
$aseco->console_text('XmlrpcDB->RequestWaitArray() - response[Message]' . CRLF . print_r($response['Message'], true));
|
||
|
|
||
|
$xmlrpc_message = new IXR_Message($response['Message']);
|
||
|
if ($xmlrpc_message->parse() && $xmlrpc_message->messageType != 'fault') {
|
||
|
if ($this->_debug > 1) {
|
||
|
$aseco->console_text('XmlrpcDB->RequestWaitArray() - message' . CRLF . print_r($xmlrpc_message->message, true));
|
||
|
$aseco->console_text('XmlrpcDB->RequestWaitArray() - params' . CRLF . print_r($xmlrpc_message->params, true));
|
||
|
}
|
||
|
|
||
|
//$datas = array('methodName' => $xmlrpc_message->methodName, 'params' => $xmlrpc_message->params);
|
||
|
$datas = $this->_makeResponseDatas($xmlrpc_message->methodName, $xmlrpc_message->params, $this->_requests);
|
||
|
} else {
|
||
|
if ($this->_debug > 0)
|
||
|
$aseco->console_text('XmlrpcDB->RequestWaitArray() - message fault' . CRLF . print_r($xmlrpc_message->message, true));
|
||
|
$datas = array();
|
||
|
}
|
||
|
} else {
|
||
|
$datas = array();
|
||
|
}
|
||
|
|
||
|
if ($this->_debug > 0)
|
||
|
$aseco->console_text('XmlrpcDB->RequestWaitArray() - datas' . CRLF . print_r($datas, true));
|
||
|
if (isset($datas['params']) && isset($datas['params'][$reqnum])) {
|
||
|
$response['Data'] = $datas['params'][$reqnum];
|
||
|
$param_end = end($datas['params']);
|
||
|
if (isset($param_end['globalTTR']) && !isset($response['Data']['globalTTR']))
|
||
|
$response['Data']['globalTTR'] = $param_end['globalTTR'];
|
||
|
} else {
|
||
|
$response['Data'] = $datas;
|
||
|
}
|
||
|
if ($this->_debug > 0)
|
||
|
$aseco->console_text('XmlrpcDB->RequestWaitArray() - response[Data]' . CRLF . print_r($response['Data'], true));
|
||
|
|
||
|
$this->_initRequest();
|
||
|
return $response;
|
||
|
} // RequestWaitArray
|
||
|
|
||
|
// init the request and callback array
|
||
|
function _initRequest() {
|
||
|
|
||
|
$this->_callbacks = array();
|
||
|
$this->_requests = array();
|
||
|
$this->addRequest($this->_auth_cb, 'dedimania.Authenticate', $this->_server);
|
||
|
} // _initRequest
|
||
|
|
||
|
// make the xmlrpc string, encode it in base64, and pass it as value
|
||
|
// to xmlrpc URL post parameter
|
||
|
function _makeXMLdatas() {
|
||
|
global $aseco;
|
||
|
|
||
|
$xmlrpc_request = new IXR_RequestStd('system.multicall', $this->_requests);
|
||
|
if ($this->_debug > 1)
|
||
|
$aseco->console_text('XmlrpcDB->_makeXMLdatas() - getXml()' . CRLF . print_r($xmlrpc_request->getXml(), true));
|
||
|
return $xmlrpc_request->getXml();
|
||
|
} // _makeXMLdatas
|
||
|
|
||
|
function _callCB($response, $callbacks, $requests) {
|
||
|
global $aseco;
|
||
|
|
||
|
$globalTTR = 0;
|
||
|
if (isset($response['Message']) && is_string($response['Message'])) {
|
||
|
$xmlrpc_message = new IXR_Message($response['Message']);
|
||
|
if ($xmlrpc_message->parse() && $xmlrpc_message->messageType != 'fault') {
|
||
|
if ($this->_debug > 1)
|
||
|
$aseco->console_text('XmlrpcDB->_callCB() - message' . CRLF . print_r($xmlrpc_message->message, true));
|
||
|
|
||
|
//$datas = array('methodName' => $xmlrpc_message->methodName, 'params' => $xmlrpc_message->params);
|
||
|
$datas = $this->_makeResponseDatas($xmlrpc_message->methodName, $xmlrpc_message->params, $requests);
|
||
|
|
||
|
if (isset($datas['params']) && is_array($datas['params'])) {
|
||
|
$param_end = end($datas['params']);
|
||
|
if (isset($param_end['globalTTR']))
|
||
|
$globalTTR = $param_end['globalTTR'];
|
||
|
} else {
|
||
|
if ($this->_debug > 0)
|
||
|
$aseco->console_text('XmlrpcDB->_callCB() - message fault' . CRLF . print_r($xmlrpc_message->message, true));
|
||
|
$datas = array();
|
||
|
}
|
||
|
} else {
|
||
|
if ($this->_debug > 0)
|
||
|
$aseco->console_text('XmlrpcDB->_callCB() - message fault' . CRLF . print_r($xmlrpc_message->message, true));
|
||
|
$datas = array();
|
||
|
}
|
||
|
} else {
|
||
|
if (!isset($response['Error']) ||
|
||
|
(strpos($response['Error'], 'connection failed') === false && strpos($response['Error'], 'Request timeout') === false)) {
|
||
|
$infos = array('Url'=>$this->_url, 'Requests'=>$requests, 'Callbacks'=>$callbacks, 'Response'=>$response);
|
||
|
$serinfos = serialize($infos);
|
||
|
$aseco->console_text('XmlrpcDB->_callCB() - no response message:' . CRLF . $serinfos);
|
||
|
}
|
||
|
$datas = array();
|
||
|
}
|
||
|
|
||
|
for ($i = 0; $i < count($callbacks); $i++) {
|
||
|
if ($callbacks[$i] != null) {
|
||
|
$callback = $callbacks[$i][0];
|
||
|
if (isset($datas['params']) && isset($datas['params'][$i])) {
|
||
|
$response['Data'] = $datas['params'][$i];
|
||
|
if (!isset($response['Data']['globalTTR']))
|
||
|
$response['Data']['globalTTR'] = $globalTTR;
|
||
|
} else {
|
||
|
$response['Data'] = $datas;
|
||
|
}
|
||
|
$callbacks[$i][0] = $response;
|
||
|
call_user_func_array($callback, $callbacks[$i]);
|
||
|
}
|
||
|
}
|
||
|
} // _callCB
|
||
|
|
||
|
// build the datas array from XAseco or Dedimania server
|
||
|
// remove the first array level into params if needed
|
||
|
// add methodResponse name if needed
|
||
|
// rename sub responses params array from [0] to ['params'] if needed
|
||
|
function _makeResponseDatas($methodname, $params, $requests) {
|
||
|
global $aseco;
|
||
|
|
||
|
if (is_array($params) && count($params) == 1 && is_array($params[0]))
|
||
|
$params = $params[0];
|
||
|
if ($this->_debug > 2)
|
||
|
$aseco->console_text('XmlrpcDB->_makeResponseDatas() - requests' . CRLF . print_r($requests, true));
|
||
|
if ($this->_debug > 1)
|
||
|
$aseco->console_text('XmlrpcDB->_makeResponseDatas() - params' . CRLF . print_r($params, true));
|
||
|
|
||
|
if (is_array($params) && is_array($params[0]) && !isset($params[0]['methodResponse'])) {
|
||
|
$params2 = array();
|
||
|
foreach ($params as $key => $param) {
|
||
|
$errors = null;
|
||
|
if (isset($param['faultCode'])) {
|
||
|
$errors[] = array('Code' => $param['faultCode'], 'Message' => $param['faultString']);
|
||
|
}
|
||
|
|
||
|
if (isset($requests[$key]['methodName']))
|
||
|
$methodresponse = $requests[$key]['methodName'];
|
||
|
else
|
||
|
$methodresponse = 'Unknown';
|
||
|
|
||
|
$ttr = 0.000001;
|
||
|
|
||
|
if (isset($param[0]))
|
||
|
$param = $param[0];
|
||
|
else
|
||
|
$param = array();
|
||
|
|
||
|
$params2[$key] = array('methodResponse' => $methodresponse,
|
||
|
'params' => $param,
|
||
|
'errors' => $errors,
|
||
|
'TTR' => $ttr,
|
||
|
'globalTTR' => $ttr
|
||
|
);
|
||
|
|
||
|
if ($methodresponse == 'dedimania.WarningsAndTTR') {
|
||
|
if ($this->_debug > 1) {
|
||
|
$aseco->console_text('XmlrpcDB->_makeResponseDatas() - param' . CRLF . print_r($param, true));
|
||
|
$aseco->console_text('XmlrpcDB->_makeResponseDatas() - params2' . CRLF . print_r($params2, true));
|
||
|
}
|
||
|
$globalTTR = $param['globalTTR'];
|
||
|
foreach ($param['methods'] as $key3 => $param3) {
|
||
|
$key2 = 0;
|
||
|
while ($key2 < count($params2) && $params2[$key2]['methodResponse'] != $param3['methodName']) {
|
||
|
$params2[$key2]['globalTTR'] = $globalTTR;
|
||
|
$key2++;
|
||
|
}
|
||
|
if ($this->_debug > 1)
|
||
|
$aseco->console_text("XmlrpcDB->_makeResponseDatas() - key2=$key2 - key3=$key3 - param3" . CRLF . print_r($param3, true));
|
||
|
if ($key2 < count($params2)) {
|
||
|
$params2[$key2]['errors'] = $param3['errors'];
|
||
|
$params2[$key2]['TTR'] = $param3['TTR'];
|
||
|
$params2[$key2]['globalTTR'] = $globalTTR;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
if ($this->_debug > 1)
|
||
|
$aseco->console_text('XmlrpcDB->_makeResponseDatas() - params2' . CRLF . print_r($params2, true));
|
||
|
return array('methodName' => $methodname, 'params' => $params2);
|
||
|
|
||
|
} else {
|
||
|
return array('methodName' => $methodname, 'params' => $params);
|
||
|
}
|
||
|
} // _makeResponseDatas
|
||
|
} // class XmlrpcDB
|
||
|
|
||
|
|
||
|
// Dedimania.Authenticate callback used to catch errors
|
||
|
function xmlrpc_auth_cb($response) {
|
||
|
global $aseco;
|
||
|
|
||
|
if (isset($response['Data']['errors']) && $response['Data']['errors'] != '') {
|
||
|
if (is_array($response['Data']['errors']))
|
||
|
$aseco->console('xmlrpc_auth_cb() - response[Data][errors]' . CRLF . print_r($response['Data']['errors'], true));
|
||
|
else
|
||
|
$aseco->console('xmlrpc_auth_cb() - response[Data][errors]' . CRLF . print_r($response, true));
|
||
|
}
|
||
|
} // xmlrpc_auth_cb
|
||
|
?>
|