604 lines
12 KiB
PHP
604 lines
12 KiB
PHP
<?php
|
|
declare(strict_types=1);
|
|
|
|
namespace Http\Route;
|
|
|
|
use Closure;
|
|
use Exception;
|
|
use Http\Abstracts\HttpService;
|
|
use Http\Controller;
|
|
use Http\Exception\RequestException;
|
|
use Http\Http\Request;
|
|
use Http\Http\Response;
|
|
use Http\IInterface\Middleware;
|
|
use Http\IInterface\RouterInterface;
|
|
use JetBrains\PhpStorm\Pure;
|
|
use ReflectionException;
|
|
use Server\RequestInterface;
|
|
use Kiri\Abstracts\Config;
|
|
use Kiri\Exception\ConfigException;
|
|
use Kiri\Exception\NotFindClassException;
|
|
use Kiri\Kiri;
|
|
|
|
defined('ROUTER_TREE') or define('ROUTER_TREE', 1);
|
|
defined('ROUTER_HASH') or define('ROUTER_HASH', 2);
|
|
|
|
/**
|
|
* Class Router
|
|
* @package Kiri\Kiri\Route
|
|
*/
|
|
class Router extends HttpService implements RouterInterface
|
|
{
|
|
/** @var Node[] $nodes */
|
|
public array $nodes = [];
|
|
public array $groupTacks = [];
|
|
public ?string $dir = 'App\\Http\\Controllers';
|
|
|
|
const NOT_FOUND = 'Page not found or method not allowed.';
|
|
|
|
/** @var string[] */
|
|
public array $methods = ['GET', 'POST', 'OPTIONS', 'PUT', 'DELETE', 'RECEIVE', 'HEAD'];
|
|
|
|
|
|
public ?Closure $middleware = null;
|
|
|
|
public int $useTree = ROUTER_TREE;
|
|
|
|
public ?Response $response = null;
|
|
|
|
|
|
/**
|
|
* @param Closure $middleware
|
|
*/
|
|
public function setMiddleware(\Closure $middleware): void
|
|
{
|
|
$this->middleware = $middleware;
|
|
}
|
|
|
|
|
|
/**
|
|
* @throws ConfigException
|
|
* @throws Exception
|
|
* 初始化函数路径
|
|
*/
|
|
public function init()
|
|
{
|
|
$this->dir = Config::get('http.namespace', $this->dir);
|
|
|
|
$this->response = Kiri::app()->get('response');
|
|
}
|
|
|
|
|
|
/**
|
|
* @param bool $useTree
|
|
*/
|
|
public function setUseTree(bool $useTree): void
|
|
{
|
|
$this->useTree = $useTree ? ROUTER_TREE : ROUTER_HASH;
|
|
}
|
|
|
|
|
|
/**
|
|
* @param $path
|
|
* @param $handler
|
|
* @param string $method
|
|
* @return ?Node
|
|
* @throws Exception
|
|
*/
|
|
public function addRoute($path, $handler, string $method = 'any'): ?Node
|
|
{
|
|
if (!isset($this->nodes[$method])) {
|
|
$this->nodes[$method] = [];
|
|
}
|
|
if ($handler instanceof Closure) {
|
|
$handler = Closure::bind($handler, di(Controller::class));
|
|
}
|
|
return $this->tree($path, $handler, $method);
|
|
}
|
|
|
|
|
|
/**
|
|
* @param $path
|
|
* @param $handler
|
|
* @param string $method
|
|
* @return ?Node
|
|
*/
|
|
private function hash($path, $handler, string $method = 'any'): ?Node
|
|
{
|
|
$path = $this->resolve($path);
|
|
$this->nodes[$method][$path] = $this->NodeInstance($path, 0, $method);
|
|
|
|
return $this->nodes[$method][$path]->setHandler($handler);
|
|
}
|
|
|
|
|
|
/**
|
|
* @param $path
|
|
* @return string
|
|
*/
|
|
#[Pure] private function resolve($path): string
|
|
{
|
|
$paths = array_column($this->groupTacks, 'prefix');
|
|
if (empty($paths)) {
|
|
return '/' . ltrim($path, '/');
|
|
}
|
|
$paths = '/' . implode('/', $paths);
|
|
if ($path != '/') {
|
|
return $paths . '/' . ltrim($path, '/');
|
|
}
|
|
return $paths . '/';
|
|
}
|
|
|
|
|
|
/**
|
|
* @param $path
|
|
* @param $handler
|
|
* @param string $method
|
|
* @return Node
|
|
* @throws Exception
|
|
*/
|
|
private function tree($path, $handler, string $method = 'any'): Node
|
|
{
|
|
$explode = $this->split($path);
|
|
if (!isset($this->nodes[$method]['/'])) {
|
|
$this->nodes[$method]['/'] = $this->NodeInstance('/', 0, $method);
|
|
}
|
|
$parent = $this->nodes[$method]['/'];
|
|
if (!empty($explode)) {
|
|
$parent = $this->bindNode($parent, $explode, $method);
|
|
}
|
|
return $parent->setHandler($handler, $path);
|
|
}
|
|
|
|
|
|
/**
|
|
* @param Node $parent
|
|
* @param array $explode
|
|
* @param $method
|
|
* @return Node
|
|
* @throws Exception
|
|
*/
|
|
private function bindNode(Node $parent, array $explode, $method): Node
|
|
{
|
|
$a = 0;
|
|
foreach ($explode as $value) {
|
|
++$a;
|
|
$search = $parent->findNode($value);
|
|
if ($search === null) {
|
|
$parent = $parent->addChild($this->NodeInstance($value, $a, $method));
|
|
} else {
|
|
$parent = $search;
|
|
}
|
|
}
|
|
return $parent;
|
|
}
|
|
|
|
/**
|
|
* @param $route
|
|
* @param $handler
|
|
* @return Node|null
|
|
* @throws
|
|
*/
|
|
public function socket($route, $handler): ?Node
|
|
{
|
|
return $this->addRoute($route, $handler, 'socket');
|
|
}
|
|
|
|
|
|
/**
|
|
* @param $route
|
|
* @param $handler
|
|
* @return Node|null
|
|
* @throws
|
|
*/
|
|
public function post($route, $handler): ?Node
|
|
{
|
|
return $this->addRoute($route, $handler, 'POST');
|
|
}
|
|
|
|
/**
|
|
* @param $route
|
|
* @param $handler
|
|
* @return Node|null
|
|
* @throws
|
|
*/
|
|
public function get($route, $handler): ?Node
|
|
{
|
|
return $this->addRoute($route, $handler, 'GET');
|
|
}
|
|
|
|
|
|
/**
|
|
* @param $port
|
|
* @param callable $callback
|
|
* @return mixed
|
|
* @throws
|
|
*/
|
|
public function addRpcService($port, callable $callback): mixed
|
|
{
|
|
return call_user_func($callback, new Actuator($port));
|
|
}
|
|
|
|
|
|
/**
|
|
* @param $route
|
|
* @param $handler
|
|
* @return Node|null
|
|
* @throws
|
|
*/
|
|
public function options($route, $handler): ?Node
|
|
{
|
|
return $this->addRoute($route, $handler, 'OPTIONS');
|
|
}
|
|
|
|
|
|
/**
|
|
* @param $route
|
|
* @param $handler
|
|
* @return Any
|
|
* @throws
|
|
*/
|
|
public function any($route, $handler): Any
|
|
{
|
|
$nodes = [];
|
|
foreach ($this->methods as $method) {
|
|
$nodes[] = $this->addRoute($route, $handler, $method);
|
|
}
|
|
return new Any($nodes);
|
|
}
|
|
|
|
/**
|
|
* @param $route
|
|
* @param $handler
|
|
* @return Node|null
|
|
* @throws
|
|
*/
|
|
public function delete($route, $handler): ?Node
|
|
{
|
|
return $this->addRoute($route, $handler, 'DELETE');
|
|
}
|
|
|
|
|
|
/**
|
|
* @param $route
|
|
* @param $handler
|
|
* @return Node|null
|
|
* @throws Exception
|
|
*/
|
|
public function head($route, $handler): ?Node
|
|
{
|
|
return $this->addRoute($route, $handler, 'HEAD');
|
|
}
|
|
|
|
|
|
/**
|
|
* @param $route
|
|
* @param $handler
|
|
* @return Node|null
|
|
* @throws
|
|
*/
|
|
public function put($route, $handler): ?Node
|
|
{
|
|
return $this->addRoute($route, $handler, 'PUT');
|
|
}
|
|
|
|
/**
|
|
* @param $value
|
|
* @param int $index
|
|
* @param string $method
|
|
* @return Node
|
|
* @throws
|
|
*/
|
|
public function NodeInstance($value, int $index = 0, string $method = 'GET'): Node
|
|
{
|
|
$node = new Node();
|
|
$node->childes = [];
|
|
$node->path = $value;
|
|
$node->index = $index;
|
|
$node->method = $method;
|
|
$node->namespace = $this->loadNamespace($method);
|
|
|
|
$name = array_column($this->groupTacks, 'middleware');
|
|
if ($this->middleware instanceof \Closure) {
|
|
$node->addMiddleware([$this->middleware]);
|
|
}
|
|
if (is_array($name)) {
|
|
$node->addMiddleware($this->resolve_middleware($name));
|
|
}
|
|
return $node;
|
|
}
|
|
|
|
|
|
/**
|
|
* @param string|array $middleware
|
|
* @return array
|
|
* @throws NotFindClassException
|
|
* @throws ReflectionException
|
|
*/
|
|
private function resolve_middleware(string|array $middleware): array
|
|
{
|
|
if (is_string($middleware)) {
|
|
$middleware = [$middleware];
|
|
}
|
|
|
|
$array = [];
|
|
foreach ($middleware as $value) {
|
|
if (is_array($value)) {
|
|
foreach ($value as $item) {
|
|
$array[] = $this->getMiddlewareInstance($item);
|
|
}
|
|
} else {
|
|
$array[] = $this->getMiddlewareInstance($value);
|
|
}
|
|
}
|
|
return array_filter($array);
|
|
}
|
|
|
|
|
|
/**
|
|
* @param $value
|
|
* @return Closure|array|null
|
|
* @throws NotFindClassException
|
|
* @throws ReflectionException
|
|
*/
|
|
private function getMiddlewareInstance($value): null|Closure|array
|
|
{
|
|
if (!is_string($value)) {
|
|
return $value;
|
|
}
|
|
$value = Kiri::createObject($value);
|
|
if (!($value instanceof Middleware)) {
|
|
return null;
|
|
}
|
|
return [$value, 'onHandler'];
|
|
}
|
|
|
|
|
|
/**
|
|
* @param $method
|
|
* @return array
|
|
*/
|
|
private function loadNamespace($method): array
|
|
{
|
|
$name = array_column($this->groupTacks, 'namespace');
|
|
array_unshift($name, $this->dir);
|
|
return array_filter($name);
|
|
}
|
|
|
|
/**
|
|
* @param array $config
|
|
* @param callable $callback
|
|
* 路由分组
|
|
* @param null $stdClass
|
|
*/
|
|
public function group(array $config, callable $callback, $stdClass = null)
|
|
{
|
|
$this->groupTacks[] = $config;
|
|
if ($stdClass) {
|
|
$callback($stdClass);
|
|
} else {
|
|
$callback($this);
|
|
}
|
|
array_pop($this->groupTacks);
|
|
}
|
|
|
|
/**
|
|
* @return string
|
|
*/
|
|
public function addPrefix(): string
|
|
{
|
|
$prefix = array_column($this->groupTacks, 'prefix');
|
|
|
|
$prefix = array_filter($prefix);
|
|
|
|
if (empty($prefix)) {
|
|
return '';
|
|
}
|
|
|
|
return '/' . implode('/', $prefix);
|
|
}
|
|
|
|
/**
|
|
* @param array|null $explode
|
|
* @param $method
|
|
* @return Node|null
|
|
* 查找指定路由
|
|
*/
|
|
public function tree_search(?array $explode, $method): ?Node
|
|
{
|
|
if (!isset($this->nodes[$method])) {
|
|
return null;
|
|
}
|
|
$parent = $this->nodes[$method]['/'];
|
|
while ($value = array_shift($explode)) {
|
|
$node = $parent->findNode($value);
|
|
if (!$node) {
|
|
return null;
|
|
}
|
|
$parent = $node;
|
|
}
|
|
return $parent;
|
|
}
|
|
|
|
/**
|
|
* @param $path
|
|
* @return array|null
|
|
*/
|
|
public function split($path): ?array
|
|
{
|
|
$path = $this->addPrefix() . '/' . ltrim($path, '/');
|
|
if ($path === '/') {
|
|
return [];
|
|
}
|
|
$filter = array_filter(explode('/', $path));
|
|
if (!empty($filter)) {
|
|
return $filter;
|
|
}
|
|
return [];
|
|
}
|
|
|
|
/**
|
|
* @return array
|
|
*/
|
|
public function each(): array
|
|
{
|
|
$paths = [];
|
|
foreach ($this->nodes as $node) {
|
|
/** @var Node[] $node */
|
|
foreach ($node as $path => $_node) {
|
|
$paths[] = strtoupper($_node->method) . ' : ' . $path;
|
|
}
|
|
}
|
|
return $paths;
|
|
}
|
|
|
|
|
|
/**
|
|
* @param $exception
|
|
* @return mixed
|
|
* @throws Exception
|
|
*/
|
|
public function exception($exception): mixed
|
|
{
|
|
return Kiri::app()->getLogger()->exception($exception);
|
|
}
|
|
|
|
|
|
/**
|
|
* @param Request $request
|
|
* @return Node|null 树干搜索
|
|
* 树干搜索
|
|
* @throws Exception
|
|
*/
|
|
public function find_path(Request $request): ?Node
|
|
{
|
|
$method = $request->getMethod();
|
|
|
|
$methods = $this->nodes[$method][\request()->getUri()] ?? null;
|
|
if (!is_null($methods)) {
|
|
return $methods;
|
|
}
|
|
if ($request->isOption) {
|
|
return $this->nodes[$method]['*'] ?? null;
|
|
}
|
|
return null;
|
|
}
|
|
|
|
|
|
/**
|
|
* @param $uri
|
|
* @param $method
|
|
* @return Node|null
|
|
*/
|
|
public function search($uri, $method): Node|null
|
|
{
|
|
if (!isset($this->nodes[$method])) {
|
|
return null;
|
|
}
|
|
$methods = $this->nodes[$method];
|
|
if (isset($methods[$uri])) {
|
|
return $methods[$uri];
|
|
}
|
|
return $methods['/'] ?? null;
|
|
}
|
|
|
|
|
|
/**
|
|
* @param $request
|
|
* @return Node|null
|
|
*/
|
|
private function search_options($request): ?Node
|
|
{
|
|
$method = $request->getMethod();
|
|
if (!isset($this->nodes[$method])) {
|
|
return null;
|
|
}
|
|
if (!isset($this->nodes[$method]['*'])) {
|
|
return null;
|
|
}
|
|
return $this->nodes[$method]['*'];
|
|
}
|
|
|
|
|
|
/**
|
|
* @param RequestInterface $request
|
|
* @return Node|null
|
|
* 树杈搜索
|
|
*/
|
|
public function Branch_search(RequestInterface $request): ?Node
|
|
{
|
|
$node = $this->tree_search($request->getExplode(), $request->getMethod());
|
|
if ($node instanceof Node) {
|
|
return $node;
|
|
}
|
|
if (!$request->isOption) {
|
|
return null;
|
|
}
|
|
$node = $this->tree_search(['*'], $request->getMethod());
|
|
if (!($node instanceof Node)) {
|
|
return null;
|
|
}
|
|
return $node;
|
|
}
|
|
|
|
|
|
/**
|
|
* @throws
|
|
*/
|
|
public function _loader()
|
|
{
|
|
$this->loadRouteDir(APP_PATH . 'routes');
|
|
// $classes = Kiri::getAnnotation()->runtime(CONTROLLER_PATH);
|
|
//
|
|
// $di = Kiri::getDi();
|
|
// foreach ($classes as $class) {
|
|
// $methods = $di->getMethodAttribute($class);
|
|
// foreach ($methods as $method => $attribute) {
|
|
// if (empty($attribute)) {
|
|
// continue;
|
|
// }
|
|
// foreach ($attribute as $item) {
|
|
// $item->execute($class, $method);
|
|
// }
|
|
// }
|
|
// }
|
|
}
|
|
|
|
/**
|
|
* @param $path
|
|
* @throws Exception
|
|
* 加载目录下的路由文件
|
|
*/
|
|
private function loadRouteDir($path)
|
|
{
|
|
$files = glob($path . '/*');
|
|
for ($i = 0; $i < count($files); $i++) {
|
|
if (is_dir($files[$i])) {
|
|
$this->loadRouteDir($files[$i]);
|
|
} else {
|
|
$this->loadRouterFile($files[$i]);
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
/**
|
|
* @param $files
|
|
* @throws Exception
|
|
*/
|
|
private function loadRouterFile($files)
|
|
{
|
|
try {
|
|
$router = $this;
|
|
include_once "{$files}";
|
|
} catch (\Throwable $exception) {
|
|
$this->addError($exception, 'throwable');
|
|
} finally {
|
|
if (isset($exception)) {
|
|
unset($exception);
|
|
}
|
|
}
|
|
}
|
|
|
|
}
|