338 lines
8.6 KiB
PHP
Executable File
338 lines
8.6 KiB
PHP
Executable File
<?php
|
|
|
|
namespace IXR;
|
|
|
|
use Typecho\Widget\Exception as WidgetException;
|
|
|
|
/**
|
|
* IXR服务器
|
|
*
|
|
* @package IXR
|
|
*/
|
|
class Server
|
|
{
|
|
/**
|
|
* 回调函数
|
|
*
|
|
* @var array
|
|
*/
|
|
private array $callbacks;
|
|
|
|
/**
|
|
* 默认参数
|
|
*
|
|
* @var array
|
|
*/
|
|
private array $capabilities;
|
|
|
|
/**
|
|
* @var Hook
|
|
*/
|
|
private Hook $hook;
|
|
|
|
/**
|
|
* 构造函数
|
|
*
|
|
* @param array $callbacks 回调函数
|
|
*/
|
|
public function __construct(array $callbacks = [])
|
|
{
|
|
$this->setCapabilities();
|
|
$this->callbacks = $callbacks;
|
|
$this->setCallbacks();
|
|
}
|
|
|
|
/**
|
|
* 获取默认参数
|
|
*
|
|
* @access public
|
|
* @return array
|
|
*/
|
|
public function getCapabilities(): array
|
|
{
|
|
return $this->capabilities;
|
|
}
|
|
|
|
/**
|
|
* 列出所有方法
|
|
*
|
|
* @access public
|
|
* @return array
|
|
*/
|
|
public function listMethods(): array
|
|
{
|
|
// Returns a list of methods - uses array_reverse to ensure user defined
|
|
// methods are listed before server defined methods
|
|
return array_reverse(array_keys($this->callbacks));
|
|
}
|
|
|
|
/**
|
|
* 一次处理多个请求
|
|
*
|
|
* @param array $methodcalls
|
|
* @return array
|
|
*/
|
|
public function multiCall(array $methodcalls): array
|
|
{
|
|
// See http://www.xmlrpc.com/discuss/msgReader$1208
|
|
$return = [];
|
|
foreach ($methodcalls as $call) {
|
|
$method = $call['methodName'];
|
|
$params = $call['params'];
|
|
if ($method == 'system.multicall') {
|
|
$result = new Error(-32600, 'Recursive calls to system.multicall are forbidden');
|
|
} else {
|
|
$result = $this->call($method, $params);
|
|
}
|
|
if (is_a($result, 'Error')) {
|
|
$return[] = [
|
|
'faultCode' => $result->code,
|
|
'faultString' => $result->message
|
|
];
|
|
} else {
|
|
$return[] = [$result];
|
|
}
|
|
}
|
|
return $return;
|
|
}
|
|
|
|
/**
|
|
* @param string $methodName
|
|
* @return string|Error
|
|
*/
|
|
public function methodHelp(string $methodName)
|
|
{
|
|
if (!$this->hasMethod($methodName)) {
|
|
return new Error(-32601, 'server error. requested method ' . $methodName . ' does not exist.');
|
|
}
|
|
|
|
[$object, $method] = $this->callbacks[$methodName];
|
|
|
|
try {
|
|
$ref = new \ReflectionMethod($object, $method);
|
|
$doc = $ref->getDocComment();
|
|
|
|
return $doc ?: '';
|
|
} catch (\ReflectionException $e) {
|
|
return '';
|
|
}
|
|
}
|
|
|
|
/**
|
|
* @param Hook $hook
|
|
*/
|
|
public function setHook(Hook $hook)
|
|
{
|
|
$this->hook = $hook;
|
|
}
|
|
|
|
/**
|
|
* 呼叫内部方法
|
|
*
|
|
* @param string $methodName 方法名
|
|
* @param array $args 参数
|
|
* @return mixed
|
|
*/
|
|
private function call(string $methodName, array $args)
|
|
{
|
|
if (!$this->hasMethod($methodName)) {
|
|
return new Error(-32601, 'server error. requested method ' . $methodName . ' does not exist.');
|
|
}
|
|
$method = $this->callbacks[$methodName];
|
|
|
|
if (!is_callable($method)) {
|
|
return new Error(
|
|
-32601,
|
|
'server error. requested class method "' . $methodName . '" does not exist.'
|
|
);
|
|
}
|
|
|
|
[$object, $objectMethod] = $method;
|
|
|
|
try {
|
|
$ref = new \ReflectionMethod($object, $objectMethod);
|
|
$requiredArgs = $ref->getNumberOfRequiredParameters();
|
|
if (count($args) < $requiredArgs) {
|
|
return new Error(
|
|
-32602,
|
|
'server error. requested class method "' . $methodName . '" require ' . $requiredArgs . ' params.'
|
|
);
|
|
}
|
|
|
|
foreach ($ref->getParameters() as $key => $parameter) {
|
|
if ($parameter->hasType() && !settype($args[$key], $parameter->getType()->getName())) {
|
|
return new Error(
|
|
-32602,
|
|
'server error. requested class method "'
|
|
. $methodName . '" ' . $key . ' param has wrong type.'
|
|
);
|
|
}
|
|
}
|
|
|
|
if (isset($this->hook)) {
|
|
$result = $this->hook->beforeRpcCall($methodName, $ref, $args);
|
|
|
|
if (isset($result)) {
|
|
return $result;
|
|
}
|
|
}
|
|
|
|
$result = call_user_func_array($method, $args);
|
|
|
|
if (isset($this->hook)) {
|
|
$this->hook->afterRpcCall($methodName, $result);
|
|
}
|
|
|
|
return $result;
|
|
} catch (\ReflectionException $e) {
|
|
return new Error(
|
|
-32601,
|
|
'server error. requested class method "' . $methodName . '" does not exist.'
|
|
);
|
|
} catch (Exception $e) {
|
|
return new Error(
|
|
$e->getCode(),
|
|
$e->getMessage()
|
|
);
|
|
} catch (WidgetException $e) {
|
|
return new Error(
|
|
-32001,
|
|
$e->getMessage()
|
|
);
|
|
} catch (\Exception $e) {
|
|
return new Error(
|
|
-32001,
|
|
'server error. requested class method "' . $methodName . '" failed.'
|
|
);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* 抛出错误
|
|
*
|
|
* @access private
|
|
* @param integer|Error $error 错误代码
|
|
* @param string|null $message 错误消息
|
|
* @return void
|
|
*/
|
|
private function error($error, ?string $message = null)
|
|
{
|
|
// Accepts either an error object or an error code and message
|
|
if (!$error instanceof Error) {
|
|
$error = new Error($error, $message);
|
|
}
|
|
|
|
$this->output($error->getXml());
|
|
}
|
|
|
|
/**
|
|
* 输出xml
|
|
*
|
|
* @access private
|
|
* @param string $xml 输出xml
|
|
*/
|
|
private function output(string $xml)
|
|
{
|
|
$xml = '<?xml version="1.0"?>' . "\n" . $xml;
|
|
$length = strlen($xml);
|
|
header('Connection: close');
|
|
header('Content-Length: ' . $length);
|
|
header('Content-Type: text/xml');
|
|
header('Date: ' . date('r'));
|
|
echo $xml;
|
|
exit;
|
|
}
|
|
|
|
/**
|
|
* 是否存在方法
|
|
*
|
|
* @access private
|
|
* @param string $method 方法名
|
|
* @return bool
|
|
*/
|
|
private function hasMethod(string $method): bool
|
|
{
|
|
return in_array($method, array_keys($this->callbacks));
|
|
}
|
|
|
|
/**
|
|
* 设置默认参数
|
|
*
|
|
* @access public
|
|
* @return void
|
|
*/
|
|
private function setCapabilities()
|
|
{
|
|
// Initialises capabilities array
|
|
$this->capabilities = [
|
|
'xmlrpc' => [
|
|
'specUrl' => 'http://www.xmlrpc.com/spec',
|
|
'specVersion' => 1
|
|
],
|
|
'faults_interop' => [
|
|
'specUrl' => 'http://xmlrpc-epi.sourceforge.net/specs/rfc.fault_codes.php',
|
|
'specVersion' => 20010516
|
|
],
|
|
'system.multicall' => [
|
|
'specUrl' => 'http://www.xmlrpc.com/discuss/msgReader$1208',
|
|
'specVersion' => 1
|
|
],
|
|
];
|
|
}
|
|
|
|
/**
|
|
* 设置默认方法
|
|
*
|
|
* @access private
|
|
* @return void
|
|
*/
|
|
private function setCallbacks()
|
|
{
|
|
$this->callbacks['system.getCapabilities'] = [$this, 'getCapabilities'];
|
|
$this->callbacks['system.listMethods'] = [$this, 'listMethods'];
|
|
$this->callbacks['system.multicall'] = [$this, 'multiCall'];
|
|
$this->callbacks['system.methodHelp'] = [$this, 'methodHelp'];
|
|
}
|
|
|
|
/**
|
|
* 服务入口
|
|
*/
|
|
public function serve()
|
|
{
|
|
$message = new Message(file_get_contents('php://input') ?: '');
|
|
|
|
if (!$message->parse()) {
|
|
$this->error(-32700, 'parse error. not well formed');
|
|
} elseif ($message->messageType != 'methodCall') {
|
|
$this->error(-32600, 'server error. invalid xml-rpc. not conforming to spec. Request must be a methodCall');
|
|
}
|
|
|
|
$result = $this->call($message->methodName, $message->params);
|
|
// Is the result an error?
|
|
if ($result instanceof Error) {
|
|
$this->error($result);
|
|
}
|
|
|
|
// Encode the result
|
|
$r = new Value($result);
|
|
$resultXml = $r->getXml();
|
|
|
|
// Create the XML
|
|
$xml = <<<EOD
|
|
<methodResponse>
|
|
<params>
|
|
<param>
|
|
<value>
|
|
$resultXml
|
|
</value>
|
|
</param>
|
|
</params>
|
|
</methodResponse>
|
|
|
|
EOD;
|
|
|
|
// Send it
|
|
$this->output($xml);
|
|
}
|
|
}
|