Files
blog/var/IXR/Server.php
2026-03-21 17:04:18 +08:00

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);
}
}