Files
kiri-rpc/RpcJsonp.php
T

313 lines
8.0 KiB
PHP
Raw Normal View History

2022-01-09 14:00:32 +08:00
<?php
namespace Kiri\Rpc;
2022-03-03 16:39:46 +08:00
use Exception;
2022-05-04 03:11:10 +08:00
use JetBrains\PhpStorm\ArrayShape;
2022-01-14 11:29:16 +08:00
use Kiri;
2022-01-09 14:00:32 +08:00
use Kiri\Abstracts\Component;
use Kiri\Abstracts\Config;
2022-01-14 11:29:16 +08:00
use Kiri\Annotation\Annotation;
2022-01-09 14:00:32 +08:00
use Kiri\Consul\Agent;
use Kiri\Context;
2022-03-03 18:30:59 +08:00
use Kiri\Events\EventProvider;
2022-01-09 14:00:32 +08:00
use Kiri\Exception\ConfigException;
2022-01-14 11:29:16 +08:00
use Kiri\Message\Constrict\RequestInterface;
2022-03-02 16:27:15 +08:00
use Kiri\Message\Handler\DataGrip;
2022-01-14 11:29:16 +08:00
use Kiri\Message\Handler\Router;
2022-03-02 16:27:15 +08:00
use Kiri\Message\Handler\RouterCollector;
2022-01-14 11:29:16 +08:00
use Kiri\Message\ServerRequest;
2022-01-10 11:39:56 +08:00
use Kiri\Server\Contract\OnCloseInterface;
use Kiri\Server\Contract\OnConnectInterface;
use Kiri\Server\Contract\OnReceiveInterface;
use Kiri\Server\Events\OnBeforeShutdown;
use Kiri\Server\Events\OnServerBeforeStart;
use Kiri\Server\Events\OnTaskerStart;
2022-01-14 11:29:16 +08:00
use Kiri\Server\Events\OnWorkerExit;
2022-01-10 11:39:56 +08:00
use Kiri\Server\Events\OnWorkerStart;
2022-01-14 11:29:16 +08:00
use Psr\Container\ContainerExceptionInterface;
2022-03-03 16:39:46 +08:00
use Psr\Container\ContainerInterface;
2022-01-14 11:29:16 +08:00
use Psr\Container\NotFoundExceptionInterface;
use Psr\Http\Message\ServerRequestInterface;
2022-02-23 16:32:08 +08:00
use ReflectionException;
2022-01-09 14:00:32 +08:00
use Swoole\Coroutine;
use Swoole\Coroutine\Channel;
use Swoole\Server;
use Swoole\Timer;
/**
*
*/
class RpcJsonp extends Component implements OnConnectInterface, OnReceiveInterface, OnCloseInterface
{
2022-01-14 11:29:16 +08:00
private int $timerId;
2022-02-23 16:32:08 +08:00
2022-03-03 16:39:46 +08:00
/**
* @param ContainerInterface $container
2022-03-03 18:30:59 +08:00
* @param Router $router
* @param Annotation $annotation
* @param DataGrip $dataGrip
* @param RpcManager $manager
* @param RouterCollector $collector
* @param EventProvider $eventProvider
2022-03-03 16:39:46 +08:00
* @param array $config
* @throws Exception
*/
2022-03-03 18:30:59 +08:00
public function __construct(public ContainerInterface $container,
public Router $router,
public Annotation $annotation,
public DataGrip $dataGrip,
public RpcManager $manager,
public RouterCollector $collector,
public EventProvider $eventProvider,
array $config = [])
2022-03-03 16:39:46 +08:00
{
parent::__construct($config);
}
2022-01-09 14:00:32 +08:00
/**
2022-02-23 16:32:08 +08:00
* @return void
* @throws ReflectionException
2022-01-09 14:00:32 +08:00
*/
public function init(): void
{
2022-03-03 18:30:59 +08:00
$this->eventProvider->on(OnBeforeShutdown::class, [$this, 'onBeforeShutdown']);
2022-01-09 14:00:32 +08:00
2022-01-13 18:41:16 +08:00
scan_directory(APP_PATH . 'rpc', 'app\Rpc');
2022-01-09 14:00:32 +08:00
2022-03-03 18:30:59 +08:00
$this->eventProvider->on(OnWorkerStart::class, [$this, 'consulWatches']);
$this->eventProvider->on(OnWorkerExit::class, [$this, 'onWorkerExit']);
$this->eventProvider->on(OnServerBeforeStart::class, [$this, 'register']);
2022-03-02 16:27:15 +08:00
2022-03-03 18:30:59 +08:00
$this->collector = $this->dataGrip->get('rpc');
2022-01-09 14:00:32 +08:00
}
/**
* @param OnBeforeShutdown $beforeShutdown
* @return void
* @throws ContainerExceptionInterface
* @throws NotFoundExceptionInterface
*/
2022-05-04 03:11:10 +08:00
public function onBeforeShutdown(OnBeforeShutdown $beforeShutdown): void
2022-01-09 14:00:32 +08:00
{
$doneList = $this->manager->doneList();
2022-03-03 18:30:59 +08:00
$agent = $this->container->get(Agent::class);
2022-01-09 14:00:32 +08:00
foreach ($doneList as $value) {
$agent->service->deregister($value['config']['ID']);
$agent->checks->deregister($value['config']['Check']['CheckId']);
}
}
/**
* @param OnWorkerStart|OnTaskerStart $server
* @throws ConfigException
*/
public function consulWatches(OnWorkerStart|OnTaskerStart $server)
{
if ($server->workerId != 0) {
return;
}
$async_time = (int)Config::get('consul.async_time', 1000);
2022-01-14 11:29:16 +08:00
$this->timerId = Timer::tick($async_time, static function () {
2022-01-09 14:00:32 +08:00
Kiri::getDi()->get(RpcManager::class)->tick();
});
}
2022-01-14 11:29:16 +08:00
/**
* @param OnWorkerExit $exit
* @return void
*/
2022-05-04 03:11:10 +08:00
public function onWorkerExit(OnWorkerExit $exit): void
2022-01-14 11:29:16 +08:00
{
Timer::clear($this->timerId);
}
2022-01-09 14:00:32 +08:00
/**
* @param OnServerBeforeStart $server
*/
public function register(OnServerBeforeStart $server)
{
$this->manager->register();
}
/**
* @param Server $server
* @param int $fd
*/
public function onConnect(Server $server, int $fd): void
{
// TODO: Implement onConnect() method.
}
/**
* @param Server $server
* @param int $fd
* @param int $reactor_id
* @param string $data
*/
public function onReceive(Server $server, int $fd, int $reactor_id, string $data): void
{
$data = json_decode($data, true);
if (is_null($data)) {
$this->failure(-32700, 'Parse error语法解析错误');
} else if (!isset($data['jsonrpc']) || !isset($data['method']) || $data['jsonrpc'] != '2.0') {
$this->failure(-32600, 'Invalid Request无效请求');
} else {
$this->batchDispatch($server, $fd, $data);
}
}
/**
* @param Server $server
* @param int $fd
* @param array $data
* @return void
*/
private function batchDispatch(Server $server, int $fd, array $data): void
{
if (isset($data['jsonrpc'])) {
$dispatch = $this->dispatch($data);
if (!isset($data['id'])) {
$dispatch = [1];
}
$result = json_encode($dispatch, JSON_UNESCAPED_UNICODE);
} else {
$channel = new Channel($total = count($data));
foreach ($data as $datum) {
$this->_execute($channel, $datum);
}
$result = [];
for ($i = 0; $i < $total; $i++) {
$result[] = $channel->pop();
}
}
$server->send($fd, json_encode($result, JSON_UNESCAPED_UNICODE));
}
/**
* @param $channel
* @param $datum
*/
private function _execute($channel, $datum)
{
Coroutine::create(function () use ($channel, $datum) {
if (empty($datum) || !isset($datum['jsonrpc'])) {
$channel->push($this->failure(-32700, 'Parse error语法解析错误'));
} else if (!isset($datum['method'])) {
$channel->push($this->failure(-32700, 'Parse error语法解析错误'));
} else {
$dispatch = $this->dispatch($datum);
if (!isset($dispatch['id'])) {
$dispatch = [1];
}
$channel->push($dispatch);
}
});
}
/**
* @param $data
* @return array
*/
private function dispatch($data): array
{
try {
2022-03-02 16:27:15 +08:00
$handler = $this->collector->find($data['service'], 'GET');
if (is_integer($handler) || is_null($handler)) {
2022-05-04 03:11:10 +08:00
throw new Exception('Handler not found', -32601);
}
2022-03-02 16:27:15 +08:00
2022-05-04 03:11:10 +08:00
$controller = $handler->callback[0];
if (!method_exists($controller, $data['method'])) {
throw new Exception('Method not found', -32601);
}
$params = $this->container->getMethodParameters($controller::class, $data['method']);
2022-01-09 14:00:32 +08:00
2022-05-04 03:11:10 +08:00
Context::setContext(RequestInterface::class, $this->createServerRequest($params));
return $this->handler($controller, $data['method'], $params);
} catch (\Throwable $throwable) {
2022-01-09 14:00:32 +08:00
$code = $throwable->getCode() == 0 ? -32603 : $throwable->getCode();
return $this->failure($code, jTraceEx($throwable), [], $data['id'] ?? null);
}
}
/**
* @param $params
* @return ServerRequestInterface
2022-03-03 16:39:46 +08:00
* @throws Exception
2022-01-09 14:00:32 +08:00
*/
private function createServerRequest($params): ServerRequestInterface
{
return (new ServerRequest())->withParsedBody($params);
}
/**
2022-03-02 16:47:02 +08:00
* @param $controller
2022-03-02 16:27:15 +08:00
* @param string $method
* @param $params
2022-01-09 14:00:32 +08:00
* @return array
*/
2022-05-04 03:11:10 +08:00
#[ArrayShape([])]
private function handler($controller, string $method, $params): array
2022-01-09 14:00:32 +08:00
{
2022-03-02 16:47:02 +08:00
$result = call_user_func([$controller, $method], ...$params);
2022-01-09 14:00:32 +08:00
return [
'jsonrpc' => '2.0',
2022-03-02 16:27:15 +08:00
'result' => $result,
2022-01-09 14:00:32 +08:00
'id' => $data['id'] ?? null
];
}
/**
* @param $code
* @param $message
* @param array $data
* @param null $id
* @return array
*/
2022-05-04 03:11:10 +08:00
#[ArrayShape([])]
2022-01-09 14:00:32 +08:00
protected function failure($code, $message, array $data = [], $id = null): array
{
$error = [
'jsonrpc' => '2.0',
'error' => [
'code' => $code,
'message' => $message,
'data' => $data
]
];
if (!is_null($id)) {
$error['id'] = $id;
}
return $error;
}
/**
2022-02-28 14:59:34 +08:00
* @param \Swoole\WebSocket\Server $server
2022-01-09 14:00:32 +08:00
* @param int $fd
2022-02-28 14:59:34 +08:00
* @return void
2022-01-09 14:00:32 +08:00
*/
2022-02-28 14:59:34 +08:00
public function onClose(\Swoole\WebSocket\Server $server, int $fd): void
2022-01-09 14:00:32 +08:00
{
// TODO: Implement onClose() method.
}
}