Compare commits
19 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| ca274a3910 | |||
| 4d6f7d1d13 | |||
| 8296cda1f2 | |||
| 3bb14c5acb | |||
| b5bfff66dd | |||
| adb2269df0 | |||
| b557f15a98 | |||
| 862acf1db1 | |||
| a2272edc76 | |||
| 555d653288 | |||
| 45cf88e52c | |||
| a8f840bfb2 | |||
| d8222366b1 | |||
| fbe13eaa7e | |||
| 466df3387f | |||
| fa76b5170a | |||
| b0c66c9c6a | |||
| 2ffdf83645 | |||
| c3a3551ba3 |
@@ -14,7 +14,6 @@ use Kiri\Application;
|
|||||||
use Kiri\Core\Json;
|
use Kiri\Core\Json;
|
||||||
use Kiri\Di\Container;
|
use Kiri\Di\Container;
|
||||||
use Kiri\Environmental;
|
use Kiri\Environmental;
|
||||||
use Kiri\Server\Tasker\AsyncTaskExecute;
|
|
||||||
use Psr\Container\ContainerInterface;
|
use Psr\Container\ContainerInterface;
|
||||||
use Swoole\Coroutine;
|
use Swoole\Coroutine;
|
||||||
use Swoole\Process;
|
use Swoole\Process;
|
||||||
@@ -139,25 +138,6 @@ class Kiri
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @param $port
|
|
||||||
* @return bool
|
|
||||||
* @throws Exception
|
|
||||||
*/
|
|
||||||
public static function port_already($port): bool
|
|
||||||
{
|
|
||||||
if (empty($port)) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
if (Kiri::getPlatform()->isLinux()) {
|
|
||||||
exec('netstat -tunlp | grep ' . $port, $output);
|
|
||||||
} else {
|
|
||||||
exec('lsof -i :' . $port . ' | grep -i "LISTEN"', $output);
|
|
||||||
}
|
|
||||||
return !empty($output);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @return Annotation
|
* @return Annotation
|
||||||
* @throws Exception
|
* @throws Exception
|
||||||
@@ -168,15 +148,6 @@ class Kiri
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @param $service
|
|
||||||
* @return string
|
|
||||||
*/
|
|
||||||
#[Pure] public static function listen($service): string
|
|
||||||
{
|
|
||||||
return sprintf('Check listen %s::%d -> ok', $service['host'], $service['port']);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @param $className
|
* @param $className
|
||||||
@@ -215,15 +186,6 @@ class Kiri
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @return bool
|
|
||||||
*/
|
|
||||||
public static function inCoroutine(): bool
|
|
||||||
{
|
|
||||||
return Coroutine::getCid() > 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @return Container
|
* @return Container
|
||||||
*/
|
*/
|
||||||
@@ -242,40 +204,6 @@ class Kiri
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @param $workerId
|
|
||||||
* @return mixed
|
|
||||||
* @throws Exception
|
|
||||||
*/
|
|
||||||
public static function setManagerId($workerId): mixed
|
|
||||||
{
|
|
||||||
if (empty($workerId) || static::isDocker()) {
|
|
||||||
return $workerId;
|
|
||||||
}
|
|
||||||
|
|
||||||
$tmpFile = storage($workerId . '.sock', 'pid/manager');
|
|
||||||
|
|
||||||
return self::writeFile($tmpFile, $workerId);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @param $workerId
|
|
||||||
* @return mixed
|
|
||||||
* @throws Exception
|
|
||||||
*/
|
|
||||||
public static function setProcessId($workerId): mixed
|
|
||||||
{
|
|
||||||
if (empty($workerId) || static::isDocker()) {
|
|
||||||
return $workerId;
|
|
||||||
}
|
|
||||||
|
|
||||||
$tmpFile = storage($workerId . '.sock', 'pid/process');
|
|
||||||
|
|
||||||
return self::writeFile($tmpFile, $workerId);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @return bool
|
* @return bool
|
||||||
*/
|
*/
|
||||||
@@ -289,39 +217,6 @@ class Kiri
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @param $workerId
|
|
||||||
* @return mixed
|
|
||||||
* @throws Exception
|
|
||||||
*/
|
|
||||||
public static function setWorkerId($workerId): mixed
|
|
||||||
{
|
|
||||||
if (empty($workerId) || static::isDocker()) {
|
|
||||||
return $workerId;
|
|
||||||
}
|
|
||||||
|
|
||||||
$tmpFile = storage($workerId . '.sock', 'pid/worker');
|
|
||||||
|
|
||||||
return self::writeFile($tmpFile, $workerId);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @param $workerId
|
|
||||||
* @return mixed
|
|
||||||
* @throws Exception
|
|
||||||
*/
|
|
||||||
public static function setTaskId($workerId): mixed
|
|
||||||
{
|
|
||||||
if (empty($workerId) || static::isDocker()) {
|
|
||||||
return $workerId;
|
|
||||||
}
|
|
||||||
|
|
||||||
$tmpFile = storage($workerId . '.sock', 'pid/task');
|
|
||||||
|
|
||||||
return self::writeFile($tmpFile, $workerId);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @param $fileName
|
* @param $fileName
|
||||||
* @param $content
|
* @param $content
|
||||||
@@ -334,7 +229,7 @@ class Kiri
|
|||||||
if ($is_append !== null) {
|
if ($is_append !== null) {
|
||||||
$params[] = $is_append;
|
$params[] = $is_append;
|
||||||
}
|
}
|
||||||
return !self::inCoroutine() ? file_put_contents(...$params) : Coroutine::writeFile(...$params);
|
return !(Coroutine::getCid() > 0) ? file_put_contents(...$params) : Coroutine::writeFile(...$params);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@@ -355,95 +250,6 @@ class Kiri
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @param $workerId
|
|
||||||
* @param bool $isWorker
|
|
||||||
* @throws Exception
|
|
||||||
*/
|
|
||||||
public static function clearProcessId($workerId, bool $isWorker = false)
|
|
||||||
{
|
|
||||||
clearstatcache();
|
|
||||||
$directory = $isWorker === true ? 'pid/worker' : 'pid/task';
|
|
||||||
if (!file_exists($file = storage($workerId, $directory))) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
shell_exec('rm -rf ' . $file);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @param string|null $taskPid
|
|
||||||
* @throws Exception
|
|
||||||
*/
|
|
||||||
public static function clearTaskPid(string $taskPid = null)
|
|
||||||
{
|
|
||||||
if (empty($taskPid)) {
|
|
||||||
exec('rm -rf ' . storage(null, 'pid/task'));
|
|
||||||
} else {
|
|
||||||
static::clearProcessId($taskPid);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @param $taskPid
|
|
||||||
* @throws Exception
|
|
||||||
*/
|
|
||||||
public static function clearWorkerPid($taskPid = null)
|
|
||||||
{
|
|
||||||
if (empty($taskPid)) {
|
|
||||||
exec('rm -rf ' . storage(null, 'pid/worker'));
|
|
||||||
} else {
|
|
||||||
static::clearProcessId($taskPid, true);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @return Server|null
|
|
||||||
* @throws
|
|
||||||
*/
|
|
||||||
public static function getWebSocket(): ?\Swoole\Server
|
|
||||||
{
|
|
||||||
$server = static::app()->getSwoole();
|
|
||||||
if (!($server instanceof \Swoole\Server)) {
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
return $server;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @return false|string
|
|
||||||
* @throws Exception
|
|
||||||
*/
|
|
||||||
public static function getMasterPid(): bool|string
|
|
||||||
{
|
|
||||||
$pid = Kiri::app()->getSwoole()->setting['pid_file'];
|
|
||||||
|
|
||||||
return file_get_contents($pid);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @param int $fd
|
|
||||||
* @param $data
|
|
||||||
* @return mixed
|
|
||||||
* @throws Exception
|
|
||||||
*/
|
|
||||||
public static function push(int $fd, $data): mixed
|
|
||||||
{
|
|
||||||
$server = static::getWebSocket();
|
|
||||||
if (empty($server) || !$server->isEstablished($fd)) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
if (!is_string($data)) {
|
|
||||||
$data = Json::encode($data);
|
|
||||||
}
|
|
||||||
return $server->push($fd, $data);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @return mixed
|
* @return mixed
|
||||||
*/
|
*/
|
||||||
@@ -453,19 +259,6 @@ class Kiri
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @param string $class
|
|
||||||
* @param array $params
|
|
||||||
* @throws ReflectionException
|
|
||||||
* @throws Exception
|
|
||||||
*/
|
|
||||||
public static function async(string $class, array $params = [])
|
|
||||||
{
|
|
||||||
$manager = di(AsyncTaskExecute::class);
|
|
||||||
$manager->execute(new $class(...$params));
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @param array $v1
|
* @param array $v1
|
||||||
* @param array $v2
|
* @param array $v2
|
||||||
@@ -490,19 +283,6 @@ class Kiri
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @param $process
|
|
||||||
* @throws Exception
|
|
||||||
*/
|
|
||||||
public static function shutdown($process): void
|
|
||||||
{
|
|
||||||
static::app()->getSwoole()->shutdown();
|
|
||||||
if ($process instanceof Process) {
|
|
||||||
$process->exit(0);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @param $tmp_name
|
* @param $tmp_name
|
||||||
* @return string
|
* @return string
|
||||||
@@ -540,34 +320,10 @@ class Kiri
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
private static array $_autoload = [];
|
|
||||||
|
|
||||||
|
|
||||||
const PROCESS = 'process';
|
const PROCESS = 'process';
|
||||||
const TASK = 'task';
|
const TASK = 'task';
|
||||||
const WORKER = 'worker';
|
const WORKER = 'worker';
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @param string $event
|
|
||||||
* @param null $data
|
|
||||||
* @return false|string
|
|
||||||
* @throws Exception
|
|
||||||
*/
|
|
||||||
public static function param(string $event, $data = NULL): bool|string
|
|
||||||
{
|
|
||||||
if (is_object($data)) {
|
|
||||||
if ($data instanceof ModelInterface || $data instanceof Collection) {
|
|
||||||
$data = $data->getAttributes();
|
|
||||||
} else {
|
|
||||||
$data = get_object_vars($data);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if (!is_array($data)) $data = ['data' => $data];
|
|
||||||
return json_encode(array_merge(['callback' => $event], $data));
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @return string|null
|
* @return string|null
|
||||||
*/
|
*/
|
||||||
@@ -603,35 +359,6 @@ class Kiri
|
|||||||
return static::getEnvironmental() == static::PROCESS;
|
return static::getEnvironmental() == static::PROCESS;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @param $class
|
|
||||||
* @param $file
|
|
||||||
*/
|
|
||||||
public static function setAutoload($class, $file)
|
|
||||||
{
|
|
||||||
if (isset(static::$_autoload[$class])) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
static::$_autoload[$class] = $file;
|
|
||||||
include_once "Kiri.php";
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @param $className
|
|
||||||
*/
|
|
||||||
public static function autoload($className)
|
|
||||||
{
|
|
||||||
if (!isset(static::$_autoload[$className])) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
$file = static::$_autoload[$className];
|
|
||||||
require_once "Kiri.php";
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
//spl_autoload_register([Kiri::class, 'autoload'], true, true);
|
|
||||||
Kiri::setContainer(new Container());
|
Kiri::setContainer(new Container());
|
||||||
|
|||||||
@@ -27,6 +27,7 @@
|
|||||||
"psr/container": "^2.0",
|
"psr/container": "^2.0",
|
||||||
"psr/http-server-middleware": "1.0.1",
|
"psr/http-server-middleware": "1.0.1",
|
||||||
"game-worker/kiri-event": "~v2.0",
|
"game-worker/kiri-event": "~v2.0",
|
||||||
|
"game-worker/kiri-di": "~v1.0",
|
||||||
"ext-inotify": "*"
|
"ext-inotify": "*"
|
||||||
},
|
},
|
||||||
"autoload": {
|
"autoload": {
|
||||||
@@ -36,6 +37,7 @@
|
|||||||
"Kiri\\Websocket\\": "kiri-websocket-server/",
|
"Kiri\\Websocket\\": "kiri-websocket-server/",
|
||||||
"Gii\\": "kiri-gii/",
|
"Gii\\": "kiri-gii/",
|
||||||
"Kiri\\Annotation\\": "kiri-annotation/",
|
"Kiri\\Annotation\\": "kiri-annotation/",
|
||||||
|
"Kiri\\Server\\": "kiri-server/",
|
||||||
"Kiri\\Task\\": "kiri-task/"
|
"Kiri\\Task\\": "kiri-task/"
|
||||||
},
|
},
|
||||||
"files": [
|
"files": [
|
||||||
|
|||||||
@@ -297,22 +297,6 @@ if (!function_exists('injectRuntime')) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
if (!function_exists('swoole')) {
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @return Server|null
|
|
||||||
* @throws Exception
|
|
||||||
*/
|
|
||||||
function swoole(): ?Server
|
|
||||||
{
|
|
||||||
return Kiri::getWebSocket();
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
if (!function_exists('directory')) {
|
if (!function_exists('directory')) {
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|||||||
@@ -1,44 +0,0 @@
|
|||||||
<?php
|
|
||||||
|
|
||||||
|
|
||||||
namespace Kiri\Annotation;
|
|
||||||
|
|
||||||
|
|
||||||
use Exception;
|
|
||||||
use Kiri;
|
|
||||||
use Kiri\Server\Tasker\AsyncTaskExecute;
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Class Task
|
|
||||||
* @package Annotation
|
|
||||||
* Task任务
|
|
||||||
*/
|
|
||||||
#[\Attribute(\Attribute::TARGET_CLASS)] class Task extends Attribute
|
|
||||||
{
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Task constructor.
|
|
||||||
* @param string $name
|
|
||||||
*/
|
|
||||||
public function __construct(public string $name)
|
|
||||||
{
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @param mixed $class
|
|
||||||
* @param mixed|null $method
|
|
||||||
* @return bool
|
|
||||||
* @throws Exception
|
|
||||||
*/
|
|
||||||
public function execute(mixed $class, mixed $method = null): bool
|
|
||||||
{
|
|
||||||
$task = Kiri::getDi()->get(AsyncTaskExecute::class);
|
|
||||||
$task->reg($this->name, $class);
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
@@ -12,17 +12,19 @@ namespace Kiri\Abstracts;
|
|||||||
|
|
||||||
use Database\Connection;
|
use Database\Connection;
|
||||||
use Exception;
|
use Exception;
|
||||||
use Kiri\Message\Handler\Router;
|
|
||||||
use Kafka\KafkaProvider;
|
use Kafka\KafkaProvider;
|
||||||
use Kiri\Async;
|
|
||||||
use Kiri;
|
use Kiri;
|
||||||
use Kiri\Annotation\Annotation as SAnnotation;
|
use Kiri\Annotation\Annotation as SAnnotation;
|
||||||
|
use Kiri\Async;
|
||||||
use Kiri\Cache\Redis;
|
use Kiri\Cache\Redis;
|
||||||
use Kiri\Di\LocalService;
|
use Kiri\Di\LocalService;
|
||||||
use Kiri\Error\{ErrorHandler, Logger};
|
use Kiri\Error\{ErrorHandler, Logger};
|
||||||
use Kiri\Exception\{InitException, NotFindClassException};
|
use Kiri\Exception\{InitException, NotFindClassException};
|
||||||
|
use Kiri\Message\Handler\Router;
|
||||||
|
use Kiri\Server\{Server, ServerManager};
|
||||||
|
use Kiri\Task\AsyncTaskExecute;
|
||||||
|
use Kiri\Task\OnTaskInterface;
|
||||||
use ReflectionException;
|
use ReflectionException;
|
||||||
use Kiri\Server\{Contract\OnTaskInterface, Server, ServerManager, Tasker\AsyncTaskExecute};
|
|
||||||
use Swoole\Table;
|
use Swoole\Table;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -200,7 +202,7 @@ abstract class BaseApplication extends Component
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* @param OnTaskInterface $execute
|
* @param OnTaskInterface $execute
|
||||||
* @throws ReflectionException
|
* @throws ReflectionException|Exception
|
||||||
*/
|
*/
|
||||||
public function task(OnTaskInterface $execute): void
|
public function task(OnTaskInterface $execute): void
|
||||||
{
|
{
|
||||||
@@ -217,20 +219,20 @@ abstract class BaseApplication extends Component
|
|||||||
private function addEvent($key, $value): void
|
private function addEvent($key, $value): void
|
||||||
{
|
{
|
||||||
if ($value instanceof \Closure || is_object($value)) {
|
if ($value instanceof \Closure || is_object($value)) {
|
||||||
$this->eventProvider->on($key, $value, 0);
|
$this->getEventProvider()->on($key, $value, 0);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
if (is_array($value)) {
|
if (is_array($value)) {
|
||||||
if (is_object($value[0]) && !($value[0] instanceof \Closure)) {
|
if (is_object($value[0]) && !($value[0] instanceof \Closure)) {
|
||||||
$this->eventProvider->on($key, $value, 0);
|
$this->getEventProvider()->on($key, $value, 0);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (is_string($value[0])) {
|
if (is_string($value[0])) {
|
||||||
$value[0] = Kiri::createObject($value[0]);
|
$value[0] = Kiri::createObject($value[0]);
|
||||||
$this->eventProvider->on($key, $value, 0);
|
$this->getEventProvider()->on($key, $value, 0);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -239,7 +241,7 @@ abstract class BaseApplication extends Component
|
|||||||
if (!is_callable($item, true)) {
|
if (!is_callable($item, true)) {
|
||||||
throw new InitException("Class does not hav callback.");
|
throw new InitException("Class does not hav callback.");
|
||||||
}
|
}
|
||||||
$this->eventProvider->on($key, $item, 0);
|
$this->getEventProvider()->on($key, $item, 0);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -23,9 +23,6 @@ use Psr\Container\NotFoundExceptionInterface;
|
|||||||
/**
|
/**
|
||||||
* Class Component
|
* Class Component
|
||||||
* @package Kiri\Base
|
* @package Kiri\Base
|
||||||
* @property ContainerInterface|Container $container
|
|
||||||
* @property EventProvider $eventProvider
|
|
||||||
* @property EventDispatch $eventDispatch
|
|
||||||
*/
|
*/
|
||||||
class Component implements Configure
|
class Component implements Configure
|
||||||
{
|
{
|
||||||
@@ -67,6 +64,8 @@ class Component implements Configure
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* @return EventDispatch
|
* @return EventDispatch
|
||||||
|
* @throws ContainerExceptionInterface
|
||||||
|
* @throws NotFoundExceptionInterface
|
||||||
*/
|
*/
|
||||||
protected function getEventDispatch(): EventDispatch
|
protected function getEventDispatch(): EventDispatch
|
||||||
{
|
{
|
||||||
@@ -89,38 +88,6 @@ class Component implements Configure
|
|||||||
return static::class;
|
return static::class;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* @param $name
|
|
||||||
* @param $value
|
|
||||||
*
|
|
||||||
* @throws Exception
|
|
||||||
*/
|
|
||||||
public function __set($name, $value)
|
|
||||||
{
|
|
||||||
$method = 'set' . ucfirst($name);
|
|
||||||
if (method_exists($this, $method)) {
|
|
||||||
$this->{$method}($value);
|
|
||||||
} else {
|
|
||||||
throw new Exception('The set name ' . $name . ' not find in class ' . static::class);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @param $name
|
|
||||||
*
|
|
||||||
* @return mixed
|
|
||||||
* @throws Exception
|
|
||||||
*/
|
|
||||||
public function __get($name): mixed
|
|
||||||
{
|
|
||||||
$method = 'get' . ucfirst($name);
|
|
||||||
if (method_exists($this, $method)) {
|
|
||||||
return $this->$method();
|
|
||||||
} else {
|
|
||||||
throw new Exception('The get name ' . $name . ' not find in class ' . static::class);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @param $message
|
* @param $message
|
||||||
|
|||||||
@@ -239,7 +239,7 @@ class Application extends BaseApplication
|
|||||||
scan_directory(MODEL_PATH, 'app\Model');
|
scan_directory(MODEL_PATH, 'app\Model');
|
||||||
}
|
}
|
||||||
|
|
||||||
$this->container->setBindings(OutputInterface::class, $output);
|
$this->getContainer()->setBindings(OutputInterface::class, $output);
|
||||||
|
|
||||||
$class->run($input, $output);
|
$class->run($input, $output);
|
||||||
fire(new OnAfterCommandExecute());
|
fire(new OnAfterCommandExecute());
|
||||||
|
|||||||
@@ -7,7 +7,7 @@ namespace Kiri;
|
|||||||
use Exception;
|
use Exception;
|
||||||
use Kiri\Abstracts\Component;
|
use Kiri\Abstracts\Component;
|
||||||
use Kiri\Server\ServerManager;
|
use Kiri\Server\ServerManager;
|
||||||
use Kiri\Server\Tasker\AsyncTaskExecute;
|
use Kiri\Task\AsyncTaskExecute;
|
||||||
use Kiri;
|
use Kiri;
|
||||||
/**
|
/**
|
||||||
* Class Async
|
* Class Async
|
||||||
|
|||||||
@@ -45,8 +45,6 @@ class Redis implements StopHeartbeatCheck
|
|||||||
private int $_last = 0;
|
private int $_last = 0;
|
||||||
|
|
||||||
|
|
||||||
private EventProvider $eventProvider;
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @param array $config
|
* @param array $config
|
||||||
@@ -67,9 +65,6 @@ class Redis implements StopHeartbeatCheck
|
|||||||
public function init()
|
public function init()
|
||||||
{
|
{
|
||||||
$this->heartbeat_check();
|
$this->heartbeat_check();
|
||||||
|
|
||||||
$this->eventProvider = Kiri::getDi()->get(EventProvider::class);
|
|
||||||
$this->eventProvider->on(OnWorkerExit::class, [$this, 'onWorkerExit']);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@@ -110,8 +105,6 @@ class Redis implements StopHeartbeatCheck
|
|||||||
if (time() - $this->_last > intval($this->pool['tick'] ?? 60)) {
|
if (time() - $this->_last > intval($this->pool['tick'] ?? 60)) {
|
||||||
$this->stopHeartbeatCheck();
|
$this->stopHeartbeatCheck();
|
||||||
|
|
||||||
$this->eventProvider->off(OnWorkerExit::class, [$this, 'stopHeartbeatCheck']);
|
|
||||||
|
|
||||||
$this->pdo = null;
|
$this->pdo = null;
|
||||||
}
|
}
|
||||||
} catch (\Throwable $throwable) {
|
} catch (\Throwable $throwable) {
|
||||||
|
|||||||
@@ -19,6 +19,8 @@ use Kiri;
|
|||||||
use Kiri\Pool\Redis as PoolRedis;
|
use Kiri\Pool\Redis as PoolRedis;
|
||||||
use Kiri\Annotation\Inject;
|
use Kiri\Annotation\Inject;
|
||||||
use Kiri\Server\Events\OnWorkerExit;
|
use Kiri\Server\Events\OnWorkerExit;
|
||||||
|
use Psr\Container\ContainerExceptionInterface;
|
||||||
|
use Psr\Container\NotFoundExceptionInterface;
|
||||||
use Swoole\Timer;
|
use Swoole\Timer;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -43,7 +45,10 @@ class Redis extends Component
|
|||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
* @return void
|
||||||
* @throws ConfigException
|
* @throws ConfigException
|
||||||
|
* @throws ContainerExceptionInterface
|
||||||
|
* @throws NotFoundExceptionInterface
|
||||||
* @throws Exception
|
* @throws Exception
|
||||||
*/
|
*/
|
||||||
public function init()
|
public function init()
|
||||||
@@ -54,7 +59,7 @@ class Redis extends Component
|
|||||||
|
|
||||||
$length = Config::get('cache.redis.pool.max', 10);
|
$length = Config::get('cache.redis.pool.max', 10);
|
||||||
|
|
||||||
$this->eventProvider->on(OnWorkerExit::class, [$this, 'destroy'], 0);
|
$this->getEventProvider()->on(OnWorkerExit::class, [$this, 'destroy'], 0);
|
||||||
|
|
||||||
$connections->initConnections('Redis:' . $config['host'], true, $length);
|
$connections->initConnections('Redis:' . $config['host'], true, $length);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -4,8 +4,9 @@ namespace Kiri\Core;
|
|||||||
|
|
||||||
use JetBrains\PhpStorm\Pure;
|
use JetBrains\PhpStorm\Pure;
|
||||||
use ReturnTypeWillChange;
|
use ReturnTypeWillChange;
|
||||||
|
use Traversable;
|
||||||
|
|
||||||
class HashMap implements \ArrayAccess
|
class HashMap implements \ArrayAccess, \IteratorAggregate
|
||||||
{
|
{
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -14,7 +15,16 @@ class HashMap implements \ArrayAccess
|
|||||||
private array $lists = [];
|
private array $lists = [];
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
* @return Traversable
|
||||||
|
*/
|
||||||
|
public function getIterator(): Traversable
|
||||||
|
{
|
||||||
|
return new \ArrayIterator($this->lists);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
* @param string $key
|
* @param string $key
|
||||||
* @param $value
|
* @param $value
|
||||||
*/
|
*/
|
||||||
|
|||||||
@@ -1,454 +0,0 @@
|
|||||||
<?php
|
|
||||||
/**
|
|
||||||
* Created by PhpStorm.
|
|
||||||
* User: whwyy
|
|
||||||
* Date: 2018/4/24 0024
|
|
||||||
* Time: 17:27
|
|
||||||
*/
|
|
||||||
declare(strict_types=1);
|
|
||||||
|
|
||||||
namespace Kiri\Di;
|
|
||||||
|
|
||||||
use Closure;
|
|
||||||
use Exception;
|
|
||||||
use Kiri;
|
|
||||||
use Kiri\Abstracts\Logger;
|
|
||||||
use Kiri\Annotation\Inject;
|
|
||||||
use Psr\Container\ContainerInterface;
|
|
||||||
use Psr\Log\LoggerInterface;
|
|
||||||
use ReflectionClass;
|
|
||||||
use ReflectionException;
|
|
||||||
use ReflectionFunction;
|
|
||||||
use ReflectionMethod;
|
|
||||||
use ReflectionProperty;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Class Container
|
|
||||||
* @package Kiri\Di
|
|
||||||
*/
|
|
||||||
class Container implements ContainerInterface
|
|
||||||
{
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @var array
|
|
||||||
*
|
|
||||||
* instance class by className
|
|
||||||
*/
|
|
||||||
private array $_singletons = [];
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @var ReflectionMethod[]
|
|
||||||
*
|
|
||||||
* class new instance construct parameter
|
|
||||||
*/
|
|
||||||
private array $_constructs = [];
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @var array
|
|
||||||
*
|
|
||||||
* implements \ReflectClass
|
|
||||||
*/
|
|
||||||
private array $_reflection = [];
|
|
||||||
|
|
||||||
|
|
||||||
/** @var array */
|
|
||||||
private array $_parameters = [];
|
|
||||||
|
|
||||||
|
|
||||||
/** @var array|string[] */
|
|
||||||
private array $_interfaces = [
|
|
||||||
LoggerInterface::class => Logger::class
|
|
||||||
];
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @param string $id
|
|
||||||
* @return mixed
|
|
||||||
* @throws
|
|
||||||
*/
|
|
||||||
public function get(string $id): mixed
|
|
||||||
{
|
|
||||||
if ($id == ContainerInterface::class) {
|
|
||||||
return $this;
|
|
||||||
}
|
|
||||||
return $this->make($id, [], []);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @param $class
|
|
||||||
* @param array $constrict
|
|
||||||
* @param array $config
|
|
||||||
* @return mixed
|
|
||||||
* @throws
|
|
||||||
*/
|
|
||||||
public function make($class, array $constrict = [], array $config = []): mixed
|
|
||||||
{
|
|
||||||
if ($class == ContainerInterface::class) {
|
|
||||||
return $this;
|
|
||||||
}
|
|
||||||
if ($this->isInterface($class)) {
|
|
||||||
$class = $this->_interfaces[$class];
|
|
||||||
}
|
|
||||||
if (!isset($this->_singletons[$class])) {
|
|
||||||
$this->_singletons[$class] = $this->resolve($class, $constrict, $config);
|
|
||||||
}
|
|
||||||
return $this->_singletons[$class];
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @param string $interface
|
|
||||||
* @param string $class
|
|
||||||
*/
|
|
||||||
public function mapping(string $interface, string $class)
|
|
||||||
{
|
|
||||||
$this->_interfaces[$interface] = $class;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @param $class
|
|
||||||
* @return bool
|
|
||||||
*/
|
|
||||||
public function isInterface($class): bool
|
|
||||||
{
|
|
||||||
$reflect = $this->getReflect($class);
|
|
||||||
if ($reflect->isInterface()) {
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @param string $interface
|
|
||||||
* @param $object
|
|
||||||
*/
|
|
||||||
public function setBindings(string $interface, $object)
|
|
||||||
{
|
|
||||||
if (is_string($object)) {
|
|
||||||
$this->_interfaces[$interface] = $object;
|
|
||||||
} else {
|
|
||||||
$className = get_class($object);
|
|
||||||
$this->_interfaces[$interface] = $className;
|
|
||||||
$this->_singletons[$className] = $object;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @param $class
|
|
||||||
* @param array $constrict
|
|
||||||
* @param array $config
|
|
||||||
* @return object
|
|
||||||
* @throws
|
|
||||||
*/
|
|
||||||
public function create($class, array $constrict = [], array $config = []): object
|
|
||||||
{
|
|
||||||
return $this->resolve($class, $constrict, $config);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @param $class
|
|
||||||
* @param $constrict
|
|
||||||
* @param $config
|
|
||||||
*
|
|
||||||
* @return object
|
|
||||||
* @throws Exception
|
|
||||||
*/
|
|
||||||
private function resolve($class, $constrict, $config): object
|
|
||||||
{
|
|
||||||
$reflect = $this->resolveDependencies($class);
|
|
||||||
if (!$reflect->isInstantiable()) {
|
|
||||||
throw new ReflectionException('Class ' . $class . ' cannot be instantiated');
|
|
||||||
}
|
|
||||||
|
|
||||||
$object = $this->newInstance($reflect, $constrict);
|
|
||||||
|
|
||||||
$this->propertyInject($reflect, $object);
|
|
||||||
|
|
||||||
return $this->onAfterInit($object, $config);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @param ReflectionClass $reflect
|
|
||||||
* @param $dependencies
|
|
||||||
* @return object
|
|
||||||
* @throws ReflectionException
|
|
||||||
*/
|
|
||||||
private function newInstance(ReflectionClass $reflect, $dependencies): object
|
|
||||||
{
|
|
||||||
if (!isset($this->_constructs[$reflect->getName()])) {
|
|
||||||
return $reflect->newInstance();
|
|
||||||
}
|
|
||||||
$construct = $this->_constructs[$reflect->getName()];
|
|
||||||
if ($construct->getNumberOfParameters() < 1) {
|
|
||||||
return $reflect->newInstance();
|
|
||||||
}
|
|
||||||
$parameters = $this->mergeParam($this->resolveMethodParameters($construct), $dependencies);
|
|
||||||
return $reflect->newInstanceArgs($parameters);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @param ReflectionClass $reflect
|
|
||||||
* @param $object
|
|
||||||
* @return mixed
|
|
||||||
* @throws Exception
|
|
||||||
*/
|
|
||||||
public function propertyInject(ReflectionClass $reflect, $object): mixed
|
|
||||||
{
|
|
||||||
foreach (NoteManager::getPropertyAnnotation($reflect) as $property => $inject) {
|
|
||||||
/** @var Inject $inject */
|
|
||||||
$inject->execute($object, $property);
|
|
||||||
}
|
|
||||||
return $object;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @param $className
|
|
||||||
* @param $method
|
|
||||||
* @return array
|
|
||||||
*/
|
|
||||||
public function getMethodAttribute($className, $method = null): array
|
|
||||||
{
|
|
||||||
$methods = NoteManager::getMethodAnnotation($this->getReflect($className));
|
|
||||||
if (!empty($method)) {
|
|
||||||
return $methods[$method] ?? [];
|
|
||||||
}
|
|
||||||
return $methods;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @param string $class
|
|
||||||
* @param string|null $property
|
|
||||||
* @return ReflectionProperty|ReflectionProperty[]|null
|
|
||||||
*/
|
|
||||||
public function getClassReflectionProperty(string $class, string $property = null): ReflectionProperty|null|array
|
|
||||||
{
|
|
||||||
$lists = NoteManager::getProperty($this->getReflect($class));
|
|
||||||
if (empty($lists)) {
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
if (!empty($property)) {
|
|
||||||
return $lists[$property] ?? null;
|
|
||||||
}
|
|
||||||
return $lists;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @param $object
|
|
||||||
* @param $config
|
|
||||||
* @return mixed
|
|
||||||
*/
|
|
||||||
private function onAfterInit($object, $config): mixed
|
|
||||||
{
|
|
||||||
Kiri::configure($object, $config);
|
|
||||||
if (method_exists($object, 'init') && is_callable([$object, 'init'])) {
|
|
||||||
call_user_func([$object, 'init']);
|
|
||||||
}
|
|
||||||
return $object;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @param $class
|
|
||||||
* @return ReflectionClass
|
|
||||||
*/
|
|
||||||
private function resolveDependencies($class): ReflectionClass
|
|
||||||
{
|
|
||||||
if (isset($this->_reflection[$class])) {
|
|
||||||
return $this->_reflection[$class];
|
|
||||||
}
|
|
||||||
$reflect = new ReflectionClass($class);
|
|
||||||
if ($reflect->isAbstract() || $reflect->isTrait() || $reflect->isInterface()) {
|
|
||||||
return $this->_reflection[$class] = $reflect;
|
|
||||||
}
|
|
||||||
$construct = NoteManager::resolveTarget($reflect);
|
|
||||||
if (!empty($construct) && $construct->getNumberOfParameters() > 0) {
|
|
||||||
$this->_constructs[$class] = $construct;
|
|
||||||
}
|
|
||||||
return $this->_reflection[$class] = $reflect;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @param ReflectionClass|string $class
|
|
||||||
* @return ReflectionMethod[]
|
|
||||||
* @throws ReflectionException
|
|
||||||
*/
|
|
||||||
public function getReflectMethods(ReflectionClass|string $class): array
|
|
||||||
{
|
|
||||||
if (is_string($class)) {
|
|
||||||
$class = $this->getReflect($class);
|
|
||||||
}
|
|
||||||
return NoteManager::getMethods($class);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @param ReflectionClass|string $class
|
|
||||||
* @param string $method
|
|
||||||
* @return ReflectionMethod|null
|
|
||||||
* @throws ReflectionException
|
|
||||||
*/
|
|
||||||
public function getReflectMethod(ReflectionClass|string $class, string $method): ?ReflectionMethod
|
|
||||||
{
|
|
||||||
return $this->getReflectMethods($class)[$method] ?? null;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @param string $className
|
|
||||||
* @param string $method
|
|
||||||
* @return array|null
|
|
||||||
* @throws ReflectionException
|
|
||||||
*/
|
|
||||||
public function getMethodParameters(string $className, string $method): ?array
|
|
||||||
{
|
|
||||||
if (isset($this->_parameters[$className]) && isset($this->_parameters[$className][$method])) {
|
|
||||||
return $this->_parameters[$className][$method];
|
|
||||||
}
|
|
||||||
$reflectMethod = $this->getReflectMethod($this->getReflect($className), $method);
|
|
||||||
if (!($reflectMethod instanceof ReflectionMethod)) {
|
|
||||||
throw new ReflectionException("Class does not have a function $className::$method");
|
|
||||||
}
|
|
||||||
$className = $reflectMethod->getDeclaringClass()->getName();
|
|
||||||
if (isset($this->_parameters[$className]) && isset($this->_parameters[$className][$reflectMethod->getName()])) {
|
|
||||||
return $this->_parameters[$className][$reflectMethod->getName()];
|
|
||||||
}
|
|
||||||
return $this->setParameters($className, $reflectMethod->getName(), $this->resolveMethodParameters($reflectMethod));
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @param $class
|
|
||||||
* @param $method
|
|
||||||
* @param $parameters
|
|
||||||
* @return mixed
|
|
||||||
*/
|
|
||||||
private function setParameters($class, $method, $parameters): mixed
|
|
||||||
{
|
|
||||||
if (!isset($this->_parameters[$class])) {
|
|
||||||
$this->_parameters[$class] = [];
|
|
||||||
}
|
|
||||||
return $this->_parameters[$class][$method] = $parameters;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @param Closure $reflectionMethod
|
|
||||||
* @return array
|
|
||||||
* @throws ReflectionException
|
|
||||||
*/
|
|
||||||
public function getFunctionParameters(Closure $reflectionMethod): array
|
|
||||||
{
|
|
||||||
return $this->resolveMethodParameters(new ReflectionFunction($reflectionMethod));
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @param ReflectionMethod|ReflectionFunction $reflectionMethod
|
|
||||||
* @return array
|
|
||||||
*/
|
|
||||||
private function resolveMethodParameters(ReflectionMethod|ReflectionFunction $reflectionMethod): array
|
|
||||||
{
|
|
||||||
if ($reflectionMethod->getNumberOfParameters() < 1) {
|
|
||||||
return [];
|
|
||||||
}
|
|
||||||
$params = [];
|
|
||||||
foreach ($reflectionMethod->getParameters() as $key => $parameter) {
|
|
||||||
if ($parameter->isDefaultValueAvailable()) {
|
|
||||||
$params[$key] = $parameter->getDefaultValue();
|
|
||||||
} else if ($parameter->getType() === null) {
|
|
||||||
$params[$key] = $parameter->getType();
|
|
||||||
} else {
|
|
||||||
$type = $parameter->getType()->getName();
|
|
||||||
if (is_string($type) && class_exists($type) || isset($this->_interfaces[$type])) {
|
|
||||||
$type = Kiri::getDi()->get($type);
|
|
||||||
}
|
|
||||||
$params[$key] = match ($parameter->getType()) {
|
|
||||||
'string' => '',
|
|
||||||
'int', 'float' => 0,
|
|
||||||
'', null, 'object', 'mixed' => NULL,
|
|
||||||
'bool' => false,
|
|
||||||
default => $type
|
|
||||||
};
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return $params;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @param $class
|
|
||||||
* @return ReflectionClass|null
|
|
||||||
*/
|
|
||||||
public function getReflect($class): ?ReflectionClass
|
|
||||||
{
|
|
||||||
if (!isset($this->_reflection[$class])) {
|
|
||||||
return $this->resolveDependencies($class);
|
|
||||||
}
|
|
||||||
return $this->_reflection[$class];
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @param $class
|
|
||||||
*/
|
|
||||||
public function unset($class)
|
|
||||||
{
|
|
||||||
if (is_array($class) && isset($class['class'])) {
|
|
||||||
$class = $class['class'];
|
|
||||||
} else if (is_object($class)) {
|
|
||||||
$class = $class::class;
|
|
||||||
}
|
|
||||||
unset(
|
|
||||||
$this->_reflection[$class], $this->_singletons[$class], $this->_constructs[$class]
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @return $this
|
|
||||||
*/
|
|
||||||
public function flush(): static
|
|
||||||
{
|
|
||||||
$this->_reflection = [];
|
|
||||||
$this->_singletons = [];
|
|
||||||
$this->_constructs = [];
|
|
||||||
return $this;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @param $old
|
|
||||||
* @param $newParam
|
|
||||||
*
|
|
||||||
* @return mixed
|
|
||||||
*/
|
|
||||||
private function mergeParam($old, $newParam): array
|
|
||||||
{
|
|
||||||
if (empty($old)) {
|
|
||||||
return $newParam;
|
|
||||||
} else if (empty($newParam)) {
|
|
||||||
return $old;
|
|
||||||
}
|
|
||||||
foreach ($newParam as $key => $val) {
|
|
||||||
$old[$key] = $val;
|
|
||||||
}
|
|
||||||
return $old;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @param string $id
|
|
||||||
* @return bool
|
|
||||||
*/
|
|
||||||
public function has(string $id): bool
|
|
||||||
{
|
|
||||||
return isset($this->_singletons[$id]) || isset($this->_interfaces[$id]);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,88 +0,0 @@
|
|||||||
<?php
|
|
||||||
|
|
||||||
namespace Kiri\Di;
|
|
||||||
|
|
||||||
use Kiri\Abstracts\Component;
|
|
||||||
use Kiri;
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* 服务定位器
|
|
||||||
*/
|
|
||||||
class LocalService extends Component
|
|
||||||
{
|
|
||||||
|
|
||||||
|
|
||||||
private array $_components = [];
|
|
||||||
|
|
||||||
|
|
||||||
private array $_definition = [];
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @param $name
|
|
||||||
* @param $define
|
|
||||||
*/
|
|
||||||
public function set($name, $define)
|
|
||||||
{
|
|
||||||
unset($this->_components[$name]);
|
|
||||||
|
|
||||||
$this->_definition[$name] = $define;
|
|
||||||
if (is_object($define) || $define instanceof \Closure) {
|
|
||||||
$this->_components[$name] = $define;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @throws \Exception
|
|
||||||
*/
|
|
||||||
public function get(string $name, $throwException = true)
|
|
||||||
{
|
|
||||||
if (isset($this->_components[$name])) {
|
|
||||||
return $this->_components[$name];
|
|
||||||
}
|
|
||||||
if (isset($this->_definition[$name])) {
|
|
||||||
$definition = $this->_definition[$name];
|
|
||||||
if (is_object($definition) && !$definition instanceof \Closure) {
|
|
||||||
return $this->_components[$name] = $definition;
|
|
||||||
}
|
|
||||||
return $this->_components[$name] = Kiri::createObject($definition);
|
|
||||||
} else if ($throwException) {
|
|
||||||
throw new \Exception("Unknown component ID: $name");
|
|
||||||
}
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @param array $components
|
|
||||||
*/
|
|
||||||
public function setComponents(array $components)
|
|
||||||
{
|
|
||||||
foreach ($components as $name => $component) {
|
|
||||||
$this->set($name, $component);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @param $id
|
|
||||||
* @return bool
|
|
||||||
*/
|
|
||||||
public function has($id): bool
|
|
||||||
{
|
|
||||||
return isset($this->_components[$id]) || isset($this->_definition[$id]);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @param $id
|
|
||||||
*/
|
|
||||||
public function remove($id): void
|
|
||||||
{
|
|
||||||
unset($this->_components[$id], $this->_definition[$id]);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
}
|
|
||||||
@@ -1,303 +0,0 @@
|
|||||||
<?php
|
|
||||||
|
|
||||||
namespace Kiri\Di;
|
|
||||||
|
|
||||||
use JetBrains\PhpStorm\Pure;
|
|
||||||
use ReflectionAttribute;
|
|
||||||
use ReflectionClass;
|
|
||||||
use ReflectionProperty;
|
|
||||||
|
|
||||||
class NoteManager
|
|
||||||
{
|
|
||||||
|
|
||||||
|
|
||||||
private static array $_classTarget = [];
|
|
||||||
private static array $_classMethodAnnotation = [];
|
|
||||||
private static array $_classMethod = [];
|
|
||||||
private static array $_classPropertyAnnotation = [];
|
|
||||||
private static array $_classProperty = [];
|
|
||||||
private static array $_mapping = [];
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @return void
|
|
||||||
*/
|
|
||||||
public static function clear()
|
|
||||||
{
|
|
||||||
static::$_classTarget = [];
|
|
||||||
static::$_classMethodAnnotation = [];
|
|
||||||
static::$_classMethod = [];
|
|
||||||
static::$_classPropertyAnnotation = [];
|
|
||||||
static::$_classProperty = [];
|
|
||||||
static::$_mapping = [];
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @param ReflectionClass $class
|
|
||||||
*/
|
|
||||||
public static function setTargetAnnotation(ReflectionClass $class)
|
|
||||||
{
|
|
||||||
$className = $class->getName();
|
|
||||||
if (!isset(static::$_classTarget[$className])) {
|
|
||||||
static::$_classTarget[$className] = [];
|
|
||||||
}
|
|
||||||
foreach ($class->getAttributes() as $attribute) {
|
|
||||||
if (!class_exists($attribute->getName())) {
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
$instance = $attribute->newInstance();
|
|
||||||
|
|
||||||
static::$_classTarget[$className][] = $instance;
|
|
||||||
|
|
||||||
self::setMappingClass($attribute, $className);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @param ReflectionAttribute $attribute
|
|
||||||
* @param string $class
|
|
||||||
*/
|
|
||||||
public static function setMappingClass(ReflectionAttribute $attribute, string $class)
|
|
||||||
{
|
|
||||||
if (!isset(static::$_mapping[$attribute->getName()])) {
|
|
||||||
static::$_mapping[$attribute->getName()] = [];
|
|
||||||
}
|
|
||||||
if (!isset(static::$_mapping[$attribute->getName()][$class])) {
|
|
||||||
static::$_mapping[$attribute->getName()][$class] = [];
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @param ReflectionAttribute $attribute
|
|
||||||
* @param string $class
|
|
||||||
* @param string $method
|
|
||||||
* @param mixed $instance
|
|
||||||
*/
|
|
||||||
public static function setMappingMethod(ReflectionAttribute $attribute, string $class, string $method, mixed $instance)
|
|
||||||
{
|
|
||||||
self::setMappingClass($attribute, $class);
|
|
||||||
|
|
||||||
if (!isset(static::$_mapping[$attribute->getName()][$class]['method'])) {
|
|
||||||
static::$_mapping[$attribute->getName()][$class]['method'] = [];
|
|
||||||
}
|
|
||||||
static::$_mapping[$attribute->getName()][$class]['method'][] = [$method => $instance];
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @param ReflectionAttribute $attribute
|
|
||||||
* @param string $class
|
|
||||||
* @param string $property
|
|
||||||
* @param $instance
|
|
||||||
*/
|
|
||||||
public static function setMappingProperty(ReflectionAttribute $attribute, string $class, string $property, $instance)
|
|
||||||
{
|
|
||||||
self::setMappingClass($attribute, $class);
|
|
||||||
|
|
||||||
$mapping = static::$_mapping[$attribute->getName()][$class];
|
|
||||||
if (!isset($mapping['property'])) {
|
|
||||||
$mapping['property'] = [];
|
|
||||||
}
|
|
||||||
$mapping['property'][] = [$property => $instance];
|
|
||||||
static::$_mapping[$attribute->getName()][$class] = $mapping;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @param mixed $class
|
|
||||||
* @return array
|
|
||||||
*/
|
|
||||||
public static function getTargetAnnotation(mixed $class): array
|
|
||||||
{
|
|
||||||
if (!is_string($class)) {
|
|
||||||
$class = $class::class;
|
|
||||||
}
|
|
||||||
return static::$_classTarget[$class] ?? [];
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @param ReflectionClass $class
|
|
||||||
*/
|
|
||||||
public static function setMethodAnnotation(ReflectionClass $class)
|
|
||||||
{
|
|
||||||
$className = $class->getName();
|
|
||||||
static::$_classMethodAnnotation[$className] = static::$_classMethod[$className] = [];
|
|
||||||
foreach ($class->getMethods() as $ReflectionMethod) {
|
|
||||||
static::$_classMethod[$className][$ReflectionMethod->getName()] = $ReflectionMethod;
|
|
||||||
static::$_classMethodAnnotation[$className][$ReflectionMethod->getName()] = [];
|
|
||||||
foreach ($ReflectionMethod->getAttributes() as $attribute) {
|
|
||||||
if (!class_exists($attribute->getName())) {
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
$instance = $attribute->newInstance();
|
|
||||||
|
|
||||||
static::$_classMethodAnnotation[$className][$ReflectionMethod->getName()][] = $instance;
|
|
||||||
|
|
||||||
self::setMappingMethod($attribute, $className, $ReflectionMethod->getName(), $instance);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @param string $class
|
|
||||||
* @param string $method
|
|
||||||
* @return bool
|
|
||||||
*/
|
|
||||||
public static function hasMethod(string $class, string $method): bool
|
|
||||||
{
|
|
||||||
return isset(static::$_classMethod[$class]) && isset(static::$_classMethod[$class][$method]);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @param ReflectionClass $class
|
|
||||||
* @return array
|
|
||||||
*/
|
|
||||||
#[Pure] public static function getMethodAnnotation(ReflectionClass $class): array
|
|
||||||
{
|
|
||||||
return static::$_classMethodAnnotation[$class->getName()] ?? [];
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @param \ReflectionClass $reflect
|
|
||||||
* @return \ReflectionMethod|null
|
|
||||||
*/
|
|
||||||
public static function resolveTarget(ReflectionClass $reflect): ?\ReflectionMethod
|
|
||||||
{
|
|
||||||
NoteManager::setPropertyAnnotation($reflect);
|
|
||||||
NoteManager::setTargetAnnotation($reflect);
|
|
||||||
NoteManager::setMethodAnnotation($reflect);
|
|
||||||
|
|
||||||
return $reflect->getConstructor();
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @param ReflectionClass $class
|
|
||||||
*/
|
|
||||||
public static function setPropertyAnnotation(ReflectionClass $class)
|
|
||||||
{
|
|
||||||
$className = $class->getName();
|
|
||||||
static::$_classProperty[$className] = static::$_classPropertyAnnotation[$className] = [];
|
|
||||||
foreach ($class->getProperties(ReflectionProperty::IS_PRIVATE | ReflectionProperty::IS_PUBLIC |
|
|
||||||
ReflectionProperty::IS_PROTECTED) as $ReflectionMethod) {
|
|
||||||
static::$_classProperty[$className][$ReflectionMethod->getName()] = $ReflectionMethod;
|
|
||||||
foreach ($ReflectionMethod->getAttributes() as $attribute) {
|
|
||||||
if (!class_exists($attribute->getName())) {
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
$instance = $attribute->newInstance();
|
|
||||||
|
|
||||||
static::$_classPropertyAnnotation[$className][$ReflectionMethod->getName()] = $instance;
|
|
||||||
|
|
||||||
self::setMappingProperty($attribute, $className, $ReflectionMethod->getName(), $instance);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @param string $attribute
|
|
||||||
* @param string|null $class
|
|
||||||
* @return array[]
|
|
||||||
*/
|
|
||||||
public static function getAttributeTrees(string $attribute, string $class = null): array
|
|
||||||
{
|
|
||||||
$mapping = static::$_mapping[$attribute] ?? [];
|
|
||||||
if (empty($mapping) || empty($class)) {
|
|
||||||
return $mapping;
|
|
||||||
}
|
|
||||||
return $mapping[$class] ?? [];
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @param string $attribute
|
|
||||||
* @param string $class
|
|
||||||
* @param string|null $method
|
|
||||||
* @return array
|
|
||||||
*/
|
|
||||||
public static function getSpecify_annotation(string $attribute, string $class, string $method = null): mixed
|
|
||||||
{
|
|
||||||
$class = self::getAttributeTrees($attribute, $class);
|
|
||||||
if (empty($class) || !isset($class['method'])) {
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
if (empty($method)) {
|
|
||||||
return $class['method'];
|
|
||||||
}
|
|
||||||
foreach ($class['method'] as $value) {
|
|
||||||
$key = key($value);
|
|
||||||
if ($method == $key) {
|
|
||||||
return $value[$key];
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @param string $attribute
|
|
||||||
* @param string $class
|
|
||||||
* @param string $method
|
|
||||||
* @return mixed
|
|
||||||
*/
|
|
||||||
public static function getPropertyByAnnotation(string $attribute, string $class, string $method): mixed
|
|
||||||
{
|
|
||||||
$class = self::getAttributeTrees($attribute, $class);
|
|
||||||
if (empty($class) || !isset($class['property'])) {
|
|
||||||
return [];
|
|
||||||
}
|
|
||||||
foreach ($class['property'] as $value) {
|
|
||||||
$key = key($value);
|
|
||||||
if ($method == $key) {
|
|
||||||
return $value[$key];
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @param ReflectionClass|string $class
|
|
||||||
* @return array
|
|
||||||
* @throws \ReflectionException
|
|
||||||
*/
|
|
||||||
public static function getMethods(ReflectionClass|string $class): array
|
|
||||||
{
|
|
||||||
if (is_string($class)) {
|
|
||||||
$class = self::getReflect($class);
|
|
||||||
}
|
|
||||||
return static::$_classMethod[$class->getName()] ?? [];
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @param ReflectionClass $class
|
|
||||||
* @return ReflectionProperty[]
|
|
||||||
*/
|
|
||||||
#[Pure] public static function getProperty(ReflectionClass $class): array
|
|
||||||
{
|
|
||||||
return static::$_classProperty[$class->getName()] ?? [];
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @param ReflectionClass $class
|
|
||||||
* @return array
|
|
||||||
*/
|
|
||||||
#[Pure] public static function getPropertyAnnotation(ReflectionClass $class): array
|
|
||||||
{
|
|
||||||
return static::$_classPropertyAnnotation[$class->getName()] ?? [];
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
}
|
|
||||||
@@ -36,13 +36,8 @@ class ErrorHandler extends Component implements ErrorInterface
|
|||||||
*/
|
*/
|
||||||
public function register()
|
public function register()
|
||||||
{
|
{
|
||||||
// ini_set('display_errors', '1');
|
|
||||||
set_exception_handler([$this, 'exceptionHandler']);
|
set_exception_handler([$this, 'exceptionHandler']);
|
||||||
if (defined('HHVM_VERSION')) {
|
set_error_handler([$this, 'errorHandler']);
|
||||||
set_error_handler([$this, 'errorHandler']);
|
|
||||||
} else {
|
|
||||||
set_error_handler([$this, 'errorHandler']);
|
|
||||||
}
|
|
||||||
register_shutdown_function([$this, 'shutdown']);
|
register_shutdown_function([$this, 'shutdown']);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -118,8 +113,6 @@ class ErrorHandler extends Component implements ErrorInterface
|
|||||||
{
|
{
|
||||||
$path = ['file' => $file, 'line' => $line];
|
$path = ['file' => $file, 'line' => $line];
|
||||||
|
|
||||||
var_dump(func_get_args());
|
|
||||||
|
|
||||||
$data = Json::to($code, $this->category . ': ' . $message, $path);
|
$data = Json::to($code, $this->category . ': ' . $message, $path);
|
||||||
|
|
||||||
write($data, $this->category);
|
write($data, $this->category);
|
||||||
|
|||||||
@@ -204,7 +204,6 @@ class HotReload extends Command
|
|||||||
public function trigger_reload(string $path = '')
|
public function trigger_reload(string $path = '')
|
||||||
{
|
{
|
||||||
$this->logger->warning('change reload');
|
$this->logger->warning('change reload');
|
||||||
var_dump($path);
|
|
||||||
if (!empty($path) && str_starts_with($path, CONTROLLER_PATH)) {
|
if (!empty($path) && str_starts_with($path, CONTROLLER_PATH)) {
|
||||||
$pid = file_get_contents(storage('.swoole.pid'));
|
$pid = file_get_contents(storage('.swoole.pid'));
|
||||||
if (!empty($pid) && Process::kill($pid, 0)) {
|
if (!empty($pid) && Process::kill($pid, 0)) {
|
||||||
|
|||||||
@@ -86,13 +86,12 @@ class Inotify
|
|||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
$search = array_search($ev['wd'], $this->watchFiles);
|
||||||
var_dump($ev);
|
|
||||||
|
|
||||||
//非重启类型
|
//非重启类型
|
||||||
if (str_ends_with($ev['name'], '.php')) {
|
if (str_ends_with($ev['name'], '.php')) {
|
||||||
|
|
||||||
Timer::after(3000, fn() => $this->reload($ev['name']));
|
Timer::after(3000, fn() => $this->reload($search));
|
||||||
$this->isReloading = TRUE;
|
$this->isReloading = TRUE;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -6,10 +6,10 @@ namespace Kiri\Pool;
|
|||||||
use Closure;
|
use Closure;
|
||||||
use Database\Mysql\PDO;
|
use Database\Mysql\PDO;
|
||||||
use Exception;
|
use Exception;
|
||||||
|
use Kiri;
|
||||||
use Kiri\Abstracts\Component;
|
use Kiri\Abstracts\Component;
|
||||||
use Kiri\Abstracts\Config;
|
use Kiri\Abstracts\Config;
|
||||||
use Kiri\Context;
|
use Kiri\Context;
|
||||||
use Kiri;
|
|
||||||
use Swoole\Error;
|
use Swoole\Error;
|
||||||
use Throwable;
|
use Throwable;
|
||||||
|
|
||||||
@@ -118,6 +118,19 @@ class Connection extends Component
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param string $name
|
||||||
|
* @param PDO $PDO
|
||||||
|
* @return void
|
||||||
|
* @throws Kiri\Exception\ConfigException
|
||||||
|
* @throws Exception
|
||||||
|
*/
|
||||||
|
public function addItem(string $name, PDO $PDO)
|
||||||
|
{
|
||||||
|
$this->getPool()->push($name, $PDO);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @param $name
|
* @param $name
|
||||||
* @param $isMaster
|
* @param $isMaster
|
||||||
@@ -126,21 +139,24 @@ class Connection extends Component
|
|||||||
*/
|
*/
|
||||||
public function initConnections($name, $isMaster, $max)
|
public function initConnections($name, $isMaster, $max)
|
||||||
{
|
{
|
||||||
$this->getPool()->initConnections($name, $isMaster, $max);
|
$pool = $this->getPool();
|
||||||
|
$pool->initConnections($name, $isMaster, $max);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @param $coroutineName
|
* @param $coroutineName
|
||||||
* @param $isMaster
|
* @param $isMaster
|
||||||
|
* @param array $config
|
||||||
|
* @throws Kiri\Exception\ConfigException
|
||||||
* @throws Exception
|
* @throws Exception
|
||||||
*/
|
*/
|
||||||
public function release($coroutineName, $isMaster)
|
public function release($coroutineName, $isMaster, array $config)
|
||||||
{
|
{
|
||||||
$coroutineName = $this->name('Mysql:' . $coroutineName, $isMaster);
|
$coroutineName = $this->name('Mysql:' . $coroutineName, $isMaster);
|
||||||
/** @var PDO $client */
|
/** @var PDO $client */
|
||||||
if (!($client = Context::getContext($coroutineName)) instanceof PDO) {
|
if (!($client = Context::getContext($coroutineName)) instanceof PDO) {
|
||||||
return;
|
$client = call_user_func($this->create($coroutineName, $config));
|
||||||
}
|
}
|
||||||
if ($client->inTransaction()) {
|
if ($client->inTransaction()) {
|
||||||
return;
|
return;
|
||||||
@@ -184,7 +200,7 @@ class Connection extends Component
|
|||||||
} else {
|
} else {
|
||||||
$result = true;
|
$result = true;
|
||||||
}
|
}
|
||||||
} catch (Error | Throwable $exception) {
|
} catch (Error|Throwable $exception) {
|
||||||
$result = $this->addError($exception, 'mysql');
|
$result = $this->addError($exception, 'mysql');
|
||||||
} finally {
|
} finally {
|
||||||
return $result;
|
return $result;
|
||||||
|
|||||||
@@ -41,11 +41,10 @@ class Pool extends Component
|
|||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @param Channel $channel
|
* @param Channel|SplQueue $channel
|
||||||
* @param $retain_number
|
* @param $retain_number
|
||||||
* @throws Exception
|
|
||||||
*/
|
*/
|
||||||
protected function pop(Channel $channel, $retain_number): void
|
protected function pop(Channel|SplQueue $channel, $retain_number): void
|
||||||
{
|
{
|
||||||
while ($channel->length() > $retain_number) {
|
while ($channel->length() > $retain_number) {
|
||||||
if (Context::inCoroutine()) {
|
if (Context::inCoroutine()) {
|
||||||
|
|||||||
@@ -31,7 +31,7 @@ class GiiTask extends GiiBase
|
|||||||
|
|
||||||
namespace App\Async;
|
namespace App\Async;
|
||||||
|
|
||||||
use Kiri\Server\Contract\OnTaskInterface;
|
use Kiri\Task\OnTaskInterface;
|
||||||
|
|
||||||
';
|
';
|
||||||
|
|
||||||
|
|||||||
@@ -0,0 +1,115 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
namespace Kiri\Server;
|
||||||
|
|
||||||
|
use Kiri;
|
||||||
|
use Kiri\Context;
|
||||||
|
use Kiri\Events\EventDispatch;
|
||||||
|
use Kiri\Server\Abstracts\BaseProcess;
|
||||||
|
use Kiri\Server\Contract\OnProcessInterface;
|
||||||
|
use Kiri\Server\Events\OnProcessStart;
|
||||||
|
use Swoole\Coroutine;
|
||||||
|
use Swoole\Coroutine\Http\Server;
|
||||||
|
use Swoole\Http\Request;
|
||||||
|
use Swoole\Http\Response;
|
||||||
|
use Swoole\Process\Manager;
|
||||||
|
use Swoole\Process\Pool;
|
||||||
|
|
||||||
|
|
||||||
|
class CoroutineServer
|
||||||
|
{
|
||||||
|
|
||||||
|
|
||||||
|
private Manager $manager;
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param string|OnProcessInterface|BaseProcess $customProcess
|
||||||
|
* @param string $system
|
||||||
|
* @return void
|
||||||
|
*/
|
||||||
|
public function addProcess(string|OnProcessInterface|BaseProcess $customProcess, string $system)
|
||||||
|
{
|
||||||
|
if (is_string($customProcess)) {
|
||||||
|
$customProcess = Kiri::getDi()->get($customProcess);
|
||||||
|
}
|
||||||
|
$this->manager->add(function (Pool $pool, int $workerId) use ($customProcess, $system) {
|
||||||
|
$process = $pool->getProcess($workerId);
|
||||||
|
|
||||||
|
if (Kiri::getPlatform()->isLinux()) {
|
||||||
|
$process->name($system . '(' . $customProcess->getName() . ')');
|
||||||
|
}
|
||||||
|
|
||||||
|
Kiri::getDi()->get(EventDispatch::class)->dispatch(new OnProcessStart());
|
||||||
|
|
||||||
|
set_env('environmental', Kiri::PROCESS);
|
||||||
|
$channel = Coroutine::create(function () use ($process, $customProcess) {
|
||||||
|
while (!$customProcess->isStop()) {
|
||||||
|
$message = $process->read();
|
||||||
|
if (!empty($message)) {
|
||||||
|
$message = unserialize($message);
|
||||||
|
}
|
||||||
|
if (is_null($message)) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
$customProcess->onBroadcast($message);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
Context::setContext('waite:process:message', $channel);
|
||||||
|
|
||||||
|
$customProcess->onSigterm()->process($process);
|
||||||
|
|
||||||
|
}, $customProcess->isEnableCoroutine());
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param array $settings
|
||||||
|
* @return void
|
||||||
|
*/
|
||||||
|
public function httpServer(array $settings = []): void
|
||||||
|
{
|
||||||
|
$this->manager->add(function (Pool $pool, int $workerId) use ($settings) {
|
||||||
|
$host = $settings['host'];
|
||||||
|
$port = $settings['port'];
|
||||||
|
|
||||||
|
$server = new Server($host, $port, false, true);
|
||||||
|
$server->set($settings);
|
||||||
|
|
||||||
|
$callback = $settings['events'][Constant::REQUEST] ?? null;
|
||||||
|
if (is_null($callback)) {
|
||||||
|
$pool->shutdown();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if (is_string($callback[0])) {
|
||||||
|
$callback[0] = Kiri::getDi()->get($callback[0]);
|
||||||
|
}
|
||||||
|
$server->handle('/', $callback);
|
||||||
|
$server->start();
|
||||||
|
}, true);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param array $settings
|
||||||
|
* @return void
|
||||||
|
*/
|
||||||
|
public function websocketServer(array $settings)
|
||||||
|
{
|
||||||
|
$this->manager->add(function () use ($settings) {
|
||||||
|
$host = $settings['host'];
|
||||||
|
$port = $settings['port'];
|
||||||
|
|
||||||
|
$server = new Server($host, $port, false, true);
|
||||||
|
$server->set($settings);
|
||||||
|
$hServer = \Kiri::getDi()->get(\Kiri\Message\Server::class);
|
||||||
|
$server->handle('/', function (Request $request, Response $response) use ($hServer) {
|
||||||
|
call_user_func([$hServer, 'onRequest'], $request, $response);
|
||||||
|
});
|
||||||
|
$server->start();
|
||||||
|
}, true);
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
}
|
||||||
@@ -0,0 +1,54 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
namespace Kiri\Server;
|
||||||
|
|
||||||
|
use Kiri\Abstracts\Component;
|
||||||
|
use Kiri\Abstracts\Config;
|
||||||
|
use Kiri\Exception\ConfigException;
|
||||||
|
use Kiri\Task\OnServerTask;
|
||||||
|
use Swoole\Coroutine;
|
||||||
|
use Swoole\Coroutine\Channel;
|
||||||
|
use Swoole\Server\Task;
|
||||||
|
|
||||||
|
class CoroutineTasker extends Component
|
||||||
|
{
|
||||||
|
|
||||||
|
|
||||||
|
public Channel $channel;
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return void
|
||||||
|
* @throws ConfigException
|
||||||
|
*/
|
||||||
|
public function initCore()
|
||||||
|
{
|
||||||
|
$taskWorker = Config::get('server.settings.task_worker_num', 0);
|
||||||
|
|
||||||
|
if ($taskWorker > 1) {
|
||||||
|
$this->channel = new Channel($taskWorker);
|
||||||
|
for ($i = 0; $i < $taskWorker; $i++) {
|
||||||
|
|
||||||
|
Coroutine::create(function () {
|
||||||
|
while ($this->channel->capacity) {
|
||||||
|
$data = $this->channel->pop(-1);
|
||||||
|
|
||||||
|
$execute = $this->getContainer()->get(OnServerTask::class);
|
||||||
|
|
||||||
|
$task = new Task();
|
||||||
|
$task->data = $data;
|
||||||
|
$execute->onCoroutineTask(null, $task);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
public function dispatch($data)
|
||||||
|
{
|
||||||
|
$this->channel->push($data);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
}
|
||||||
@@ -0,0 +1,33 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
namespace Kiri\Task\Annotation;
|
||||||
|
|
||||||
|
|
||||||
|
use Kiri\Annotation\Attribute;
|
||||||
|
use Kiri\Task\TaskManager;
|
||||||
|
|
||||||
|
#[\Attribute(\Attribute::TARGET_CLASS)] class AsynchronousTask extends Attribute
|
||||||
|
{
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param string $name
|
||||||
|
*/
|
||||||
|
public function __construct(public string $name)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param mixed $class
|
||||||
|
* @param mixed $method
|
||||||
|
* @return mixed
|
||||||
|
*/
|
||||||
|
public function execute(mixed $class, mixed $method = ''): mixed
|
||||||
|
{
|
||||||
|
$AsyncTaskExecute = \Kiri::getDi()->get(TaskManager::class);
|
||||||
|
$AsyncTaskExecute->add($this->name, $class::class);
|
||||||
|
return parent::execute($class, $method); // TODO: Change the autogenerated stub
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
@@ -0,0 +1,42 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
namespace Kiri\Task;
|
||||||
|
|
||||||
|
use Exception;
|
||||||
|
use Kiri;
|
||||||
|
use Kiri\Abstracts\Component;
|
||||||
|
use Kiri\Server\SwooleServerInterface;
|
||||||
|
use Swoole\Coroutine;
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
class AsyncTaskExecute extends Component
|
||||||
|
{
|
||||||
|
|
||||||
|
|
||||||
|
use TaskResolve;
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param OnTaskInterface|string $handler
|
||||||
|
* @param array $params
|
||||||
|
* @param int $workerId
|
||||||
|
* @throws Exception
|
||||||
|
*/
|
||||||
|
public function execute(OnTaskInterface|string $handler, array $params = [], int $workerId = -1)
|
||||||
|
{
|
||||||
|
if (is_string($handler)) {
|
||||||
|
$handler = $this->handle($handler, $params);
|
||||||
|
}
|
||||||
|
|
||||||
|
$server = Kiri::getDi()->get(SwooleServerInterface::class);
|
||||||
|
if ($workerId < 0 || $workerId > $server->setting['task_worker_num']) {
|
||||||
|
$workerId = random_int(0, $server->setting['task_worker_num'] - 1);
|
||||||
|
}
|
||||||
|
$server->task(serialize($handler), $workerId);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
}
|
||||||
@@ -0,0 +1,86 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
namespace Kiri\Task;
|
||||||
|
|
||||||
|
use Kiri\Abstracts\Component;
|
||||||
|
use Kiri\Core\HashMap;
|
||||||
|
use Kiri\Exception\ConfigException;
|
||||||
|
use ReflectionException;
|
||||||
|
use Swoole\Coroutine;
|
||||||
|
use Swoole\Server\Task;
|
||||||
|
|
||||||
|
class CoroutineTaskExecute extends Component
|
||||||
|
{
|
||||||
|
|
||||||
|
use TaskResolve;
|
||||||
|
|
||||||
|
|
||||||
|
private HashMap $hashMap;
|
||||||
|
|
||||||
|
|
||||||
|
private Coroutine\Channel $channel;
|
||||||
|
|
||||||
|
|
||||||
|
private OnServerTask $taskServer;
|
||||||
|
|
||||||
|
|
||||||
|
private int $total = 50;
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
public function init()
|
||||||
|
{
|
||||||
|
$this->hashMap = new HashMap();
|
||||||
|
|
||||||
|
$this->channel = new Coroutine\Channel($this->total);
|
||||||
|
|
||||||
|
$this->taskServer = \Kiri::getDi()->get(OnServerTask::class);
|
||||||
|
|
||||||
|
Coroutine::create(function () {
|
||||||
|
$barrier = Coroutine\Barrier::make();
|
||||||
|
for ($i = 0; $i < 50; $i++) {
|
||||||
|
Coroutine::create(function () {
|
||||||
|
$this->handler();
|
||||||
|
});
|
||||||
|
}
|
||||||
|
Coroutine\Barrier::wait($barrier);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return void
|
||||||
|
* @throws ConfigException
|
||||||
|
*/
|
||||||
|
protected function handler()
|
||||||
|
{
|
||||||
|
Coroutine\defer(function () {
|
||||||
|
$this->handler();
|
||||||
|
});
|
||||||
|
$data = $this->channel->pop(-1);
|
||||||
|
|
||||||
|
$task = new Task();
|
||||||
|
$task->data = $data;
|
||||||
|
|
||||||
|
$this->taskServer->onCoroutineTask(null, $task);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param OnTaskInterface|string $handler
|
||||||
|
* @param array $params
|
||||||
|
* @param int $workerId
|
||||||
|
* @return void
|
||||||
|
* @throws ReflectionException
|
||||||
|
*/
|
||||||
|
public function execute(OnTaskInterface|string $handler, array $params = [], int $workerId = -1)
|
||||||
|
{
|
||||||
|
if (is_string($handler)) {
|
||||||
|
$handler = $this->handle($handler, $params);
|
||||||
|
}
|
||||||
|
$this->channel->push(serialize($handler));
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
@@ -0,0 +1,94 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
|
||||||
|
namespace Kiri\Task;
|
||||||
|
|
||||||
|
|
||||||
|
use Kiri\Annotation\Inject;
|
||||||
|
use Kiri\Abstracts\Logger;
|
||||||
|
use Kiri\Exception\ConfigException;
|
||||||
|
use Kiri\Task\OnTaskInterface;
|
||||||
|
use Swoole\Server;
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Class OnServerTask
|
||||||
|
* @package Server\Task
|
||||||
|
*/
|
||||||
|
class OnServerTask
|
||||||
|
{
|
||||||
|
|
||||||
|
|
||||||
|
#[Inject(Logger::class)]
|
||||||
|
public Logger $logger;
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param Server $server
|
||||||
|
* @param int $task_id
|
||||||
|
* @param int $src_worker_id
|
||||||
|
* @param mixed $data
|
||||||
|
* @throws ConfigException
|
||||||
|
*/
|
||||||
|
public function onTask(Server $server, int $task_id, int $src_worker_id, mixed $data)
|
||||||
|
{
|
||||||
|
try {
|
||||||
|
$data = $this->resolve($data);
|
||||||
|
} catch (\Throwable $exception) {
|
||||||
|
$data = jTraceEx($exception);
|
||||||
|
|
||||||
|
$this->logger->error('task', [error_trigger_format($exception)]);
|
||||||
|
} finally {
|
||||||
|
$server->finish($data);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param Server|null $server
|
||||||
|
* @param Server\Task $task
|
||||||
|
* @throws ConfigException
|
||||||
|
*/
|
||||||
|
public function onCoroutineTask(?Server $server, Server\Task $task)
|
||||||
|
{
|
||||||
|
try {
|
||||||
|
$data = $this->resolve($task->data);
|
||||||
|
} catch (\Throwable $exception) {
|
||||||
|
$data = jTraceEx($exception);
|
||||||
|
|
||||||
|
$this->logger->error('task', [error_trigger_format($exception)]);
|
||||||
|
} finally {
|
||||||
|
$task->finish($data);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param $data
|
||||||
|
* @return null
|
||||||
|
*/
|
||||||
|
private function resolve($data)
|
||||||
|
{
|
||||||
|
$execute = unserialize($data);
|
||||||
|
if ($execute instanceof OnTaskInterface) {
|
||||||
|
return $execute->execute();
|
||||||
|
}
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param Server $server
|
||||||
|
* @param int $task_id
|
||||||
|
* @param mixed $data
|
||||||
|
*/
|
||||||
|
public function onFinish(Server $server, int $task_id, mixed $data)
|
||||||
|
{
|
||||||
|
if (!($data instanceof OnTaskInterface)) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
$data->finish($server, $task_id);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
}
|
||||||
@@ -0,0 +1,17 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
|
||||||
|
namespace Kiri\Task;
|
||||||
|
|
||||||
|
|
||||||
|
use Swoole\Server;
|
||||||
|
|
||||||
|
interface OnTaskInterface
|
||||||
|
{
|
||||||
|
|
||||||
|
public function execute();
|
||||||
|
|
||||||
|
|
||||||
|
public function finish(Server $server, int $task_id);
|
||||||
|
|
||||||
|
}
|
||||||
@@ -0,0 +1,77 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
namespace Kiri\Task;
|
||||||
|
|
||||||
|
use Kiri\Abstracts\Component;
|
||||||
|
use Kiri\Core\HashMap;
|
||||||
|
use Psr\Container\ContainerExceptionInterface;
|
||||||
|
use Psr\Container\NotFoundExceptionInterface;
|
||||||
|
use Swoole\Server;
|
||||||
|
|
||||||
|
class TaskManager extends Component
|
||||||
|
{
|
||||||
|
|
||||||
|
|
||||||
|
private HashMap $hashMap;
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
public function init()
|
||||||
|
{
|
||||||
|
$this->hashMap = new HashMap();
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param Server $swollen
|
||||||
|
* @return void
|
||||||
|
* @throws ContainerExceptionInterface
|
||||||
|
* @throws NotFoundExceptionInterface
|
||||||
|
*/
|
||||||
|
public function taskListener(Server $swollen)
|
||||||
|
{
|
||||||
|
if (!isset($swollen->setting['task_worker_num']) || $swollen->setting['task_worker_num'] < 1) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
$task_use_object = $swollen->setting['task_object'] ?? $swollen->setting['task_use_object'] ?? false;
|
||||||
|
$reflect = $this->getContainer()->get(OnServerTask::class);
|
||||||
|
|
||||||
|
$swollen->on('finish', [$reflect, 'onFinish']);
|
||||||
|
if ($task_use_object || $swollen->setting['task_enable_coroutine']) {
|
||||||
|
$swollen->on('task', [$reflect, 'onCoroutineTask']);
|
||||||
|
} else {
|
||||||
|
$swollen->on('task', [$reflect, 'onTask']);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param string $key
|
||||||
|
* @param $handler
|
||||||
|
*/
|
||||||
|
public function add(string $key, $handler)
|
||||||
|
{
|
||||||
|
$this->hashMap->put($key, $handler);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param string $key
|
||||||
|
* @return OnTaskInterface
|
||||||
|
* @throws ContainerExceptionInterface
|
||||||
|
* @throws NotFoundExceptionInterface
|
||||||
|
*/
|
||||||
|
public function get(string $key): OnTaskInterface
|
||||||
|
{
|
||||||
|
$task = $this->hashMap->get($key);
|
||||||
|
if (is_string($task)) {
|
||||||
|
$task = $this->getContainer()->get($task);
|
||||||
|
}
|
||||||
|
return $task;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
}
|
||||||
@@ -0,0 +1,31 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
namespace Kiri\Task;
|
||||||
|
|
||||||
|
use Exception;
|
||||||
|
use ReflectionException;
|
||||||
|
|
||||||
|
trait TaskResolve
|
||||||
|
{
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param $handler
|
||||||
|
* @param $params
|
||||||
|
* @return object
|
||||||
|
* @throws ReflectionException
|
||||||
|
* @throws Exception
|
||||||
|
*/
|
||||||
|
private function handle($handler, $params): object
|
||||||
|
{
|
||||||
|
if (!class_exists($handler) && $this->hashMap->has($handler)) {
|
||||||
|
$handler = $this->hashMap->get($handler);
|
||||||
|
}
|
||||||
|
$implements = $this->getContainer()->getReflect($handler);
|
||||||
|
if (!in_array(OnTaskInterface::class, $implements->getInterfaceNames())) {
|
||||||
|
throw new Exception('Task must instance ' . OnTaskInterface::class);
|
||||||
|
}
|
||||||
|
return $implements->newInstanceArgs($params);
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
@@ -28,6 +28,16 @@ class Sender implements WebSocketInterface
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param AliasServer|Server $server
|
||||||
|
*/
|
||||||
|
public function setServer(mixed $server): void
|
||||||
|
{
|
||||||
|
$this->server = $server;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @param int $fd
|
* @param int $fd
|
||||||
* @param mixed $data
|
* @param mixed $data
|
||||||
|
|||||||
@@ -3,15 +3,15 @@
|
|||||||
namespace Kiri\Websocket;
|
namespace Kiri\Websocket;
|
||||||
|
|
||||||
use Exception;
|
use Exception;
|
||||||
use Kiri\Message\Handler\DataGrip;
|
|
||||||
use Kiri\Abstracts\AbstractServer;
|
use Kiri\Abstracts\AbstractServer;
|
||||||
|
use Kiri\Message\Handler\DataGrip;
|
||||||
use Kiri\Message\Handler\RouterCollector;
|
use Kiri\Message\Handler\RouterCollector;
|
||||||
use Psr\Container\ContainerExceptionInterface;
|
|
||||||
use Psr\Container\NotFoundExceptionInterface;
|
|
||||||
use Kiri\Server\Contract\OnCloseInterface;
|
use Kiri\Server\Contract\OnCloseInterface;
|
||||||
use Kiri\Server\Contract\OnHandshakeInterface;
|
use Kiri\Server\Contract\OnHandshakeInterface;
|
||||||
use Kiri\Server\Contract\OnMessageInterface;
|
use Kiri\Server\Contract\OnMessageInterface;
|
||||||
use Kiri\Server\Contract\OnOpenInterface;
|
use Kiri\Server\Contract\OnOpenInterface;
|
||||||
|
use Psr\Container\ContainerExceptionInterface;
|
||||||
|
use Psr\Container\NotFoundExceptionInterface;
|
||||||
use Swoole\Http\Request;
|
use Swoole\Http\Request;
|
||||||
use Swoole\Http\Response;
|
use Swoole\Http\Response;
|
||||||
use Swoole\WebSocket\Frame;
|
use Swoole\WebSocket\Frame;
|
||||||
@@ -32,18 +32,24 @@ class Server extends AbstractServer
|
|||||||
public mixed $callback = null;
|
public mixed $callback = null;
|
||||||
|
|
||||||
|
|
||||||
|
public Sender $sender;
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @throws ContainerExceptionInterface
|
* @throws ContainerExceptionInterface
|
||||||
* @throws NotFoundExceptionInterface
|
* @throws NotFoundExceptionInterface
|
||||||
*/
|
*/
|
||||||
public function init()
|
public function init()
|
||||||
{
|
{
|
||||||
$this->router = $this->container->get(DataGrip::class)->get('ws');
|
$this->router = $this->getContainer()->get(DataGrip::class)->get('ws');
|
||||||
$handler = $this->router->find('/', 'GET');
|
$handler = $this->router->find('/', 'GET');
|
||||||
if (is_int($handler) || is_null($handler)) {
|
if (is_int($handler) || is_null($handler)) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
$this->callback = $handler->callback[0];
|
$this->callback = $handler->callback[0];
|
||||||
|
|
||||||
|
$this->sender = $this->getContainer()->get(Sender::class);
|
||||||
|
$this->sender->setServer($this->server);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
Reference in New Issue
Block a user