Compare commits
203 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| ca274a3910 | |||
| 4d6f7d1d13 | |||
| 8296cda1f2 | |||
| 3bb14c5acb | |||
| b5bfff66dd | |||
| adb2269df0 | |||
| b557f15a98 | |||
| 862acf1db1 | |||
| a2272edc76 | |||
| 555d653288 | |||
| 45cf88e52c | |||
| a8f840bfb2 | |||
| d8222366b1 | |||
| fbe13eaa7e | |||
| 466df3387f | |||
| fa76b5170a | |||
| b0c66c9c6a | |||
| 2ffdf83645 | |||
| c3a3551ba3 | |||
| 829c063700 | |||
| eaf251ac53 | |||
| 776cc80495 | |||
| 103cc96103 | |||
| f2da99efce | |||
| 49d0ba7b3c | |||
| a6ed92206e | |||
| bb9b9dbcd2 | |||
| 9782147a47 | |||
| 7598bdbd63 | |||
| a569bd6897 | |||
| ffeef3ff4b | |||
| 7f6a0c01e0 | |||
| 32f9ebc2a2 | |||
| f35ffe6ff1 | |||
| 9a76ee0184 | |||
| 249f9b1c6f | |||
| 1df2d43b8b | |||
| 2daac841a3 | |||
| 28cd946219 | |||
| 70015f7ab8 | |||
| a467056c86 | |||
| a2a3f0fc40 | |||
| cfbebeb951 | |||
| 94851c3f51 | |||
| 1896dc90b4 | |||
| 6168adb401 | |||
| 5e742e7196 | |||
| 4605fc9162 | |||
| 54f19fb058 | |||
| a9165b601a | |||
| 5b35c4de4a | |||
| 8b6aff5c7b | |||
| 1d3b45e2c3 | |||
| f2ad97c7f0 | |||
| 19b3f0f1e9 | |||
| ebd643772d | |||
| 08d9551245 | |||
| 59eec80939 | |||
| 241385b575 | |||
| 73ce2b8a50 | |||
| 6f80b2fe6a | |||
| 8955891c2f | |||
| 035138a779 | |||
| c6e6c8d68d | |||
| 6a30fdfa8d | |||
| 367e1cd122 | |||
| 667d311d73 | |||
| 375f396467 | |||
| 7b1cc1bd7b | |||
| 1ae1d78ddf | |||
| 00212d133d | |||
| 7fbe9fbf44 | |||
| de1aff9efd | |||
| cef09a11ef | |||
| 7122018a2a | |||
| f8763953c5 | |||
| 22560d77d6 | |||
| 483c898f51 | |||
| 1fa651c587 | |||
| f7ca56a9b0 | |||
| 4fa5c23c10 | |||
| f46af653f2 | |||
| 5e02a79bf0 | |||
| 08dc3e262b | |||
| 8c5e52940f | |||
| 4dbcacdd1f | |||
| 97cd1a0ebf | |||
| 7165a67294 | |||
| 4bc7451424 | |||
| d7d0f685dc | |||
| 8ee7d30d6a | |||
| 224009e7e6 | |||
| b4551ae2fd | |||
| 5ca94925e3 | |||
| 0afa8fc400 | |||
| 842220e4de | |||
| c9726e5778 | |||
| f867f4be9e | |||
| 45319c3733 | |||
| 06c9459f14 | |||
| 5fd2abcb69 | |||
| 58a3d91df3 | |||
| c61842402a | |||
| 623bae9e97 | |||
| 12610c218c | |||
| c3ca24884e | |||
| 02c14874a2 | |||
| 6eff48e22f | |||
| 6c46a54d4b | |||
| 03287cfd65 | |||
| 93ce4c16b6 | |||
| 4b92edd40f | |||
| a76c81df8e | |||
| aa59caad07 | |||
| 6fd8a5dd34 | |||
| 9ca53a73ce | |||
| d904e78864 | |||
| 0d64ef7ac4 | |||
| 154d9d74d6 | |||
| 2c61abff01 | |||
| b4ce762cf3 | |||
| 4b3c2234af | |||
| 7ee78a9642 | |||
| f9838f781d | |||
| 848416af4f | |||
| 0216e761be | |||
| acf6631c5c | |||
| 64e4307a57 | |||
| ab672127e6 | |||
| 6e5b545a1e | |||
| 8229395e6d | |||
| d5d2b04321 | |||
| ce9c184c83 | |||
| 620de34559 | |||
| b01c71ea5e | |||
| 90e1f7eb29 | |||
| f3ad09ef66 | |||
| dfccb8816c | |||
| 1670ff3fef | |||
| 8870a7ca27 | |||
| ebb7ac9673 | |||
| 5aad8d2001 | |||
| 80713788c9 | |||
| 9f36acbbca | |||
| 044d213a69 | |||
| e5fe525f82 | |||
| 39e4e52908 | |||
| f62014ff34 | |||
| 7935e6a6a3 | |||
| d2acd50352 | |||
| d02337ec22 | |||
| 57f12b6701 | |||
| d500fd21ce | |||
| 30d7b8684e | |||
| d8eb4d4e45 | |||
| 7004c5c0f8 | |||
| 81e55ecdf1 | |||
| 20adc186d4 | |||
| 8c16d9f4b3 | |||
| b3e06a680a | |||
| 3176443e5c | |||
| 83962fa3ba | |||
| 5475f2cd51 | |||
| 30212c0b86 | |||
| 4731463897 | |||
| 87e901f5b1 | |||
| 66f87b6da4 | |||
| 0007242b70 | |||
| 8653e6914b | |||
| 816fec8ef4 | |||
| b0ef09fd35 | |||
| f7a2d6f30e | |||
| 2950ba8fd5 | |||
| 2de5c82a73 | |||
| 758c4e7d5b | |||
| 33f045aec7 | |||
| 35189de442 | |||
| 34e7fc0392 | |||
| a3ce9f52ba | |||
| 6e7da1b0ed | |||
| a6e0b5c1a2 | |||
| 13ab14f965 | |||
| 6729094f63 | |||
| e1aeb17a5a | |||
| b9e20051ef | |||
| 4ab6332176 | |||
| abe2dad521 | |||
| b0f70a13da | |||
| 1e8aca91dd | |||
| b4ac5c4758 | |||
| d161477957 | |||
| a26b99dd1e | |||
| 86e4a92ab0 | |||
| ea425bb82d | |||
| ad0154d319 | |||
| edb70d2b9b | |||
| 9f3355cab4 | |||
| 4107b5bb07 | |||
| 048960c572 | |||
| b826e1f594 | |||
| e5b57cbcdb | |||
| 2870a64792 | |||
| 4d1587bc8d |
+18
-287
@@ -2,36 +2,29 @@
|
||||
declare(strict_types=1);
|
||||
|
||||
|
||||
namespace Kiri;
|
||||
error_reporting(0);
|
||||
|
||||
|
||||
use Annotation\Annotation;
|
||||
use Database\Collection;
|
||||
use Database\ModelInterface;
|
||||
use Exception;
|
||||
use JetBrains\PhpStorm\Pure;
|
||||
use Kiri\Abstracts\Config;
|
||||
use Kiri\Annotation\Annotation;
|
||||
use Kiri\Application;
|
||||
use Kiri\Core\Json;
|
||||
use Kiri\Di\Container;
|
||||
use Kiri\Environmental;
|
||||
use Psr\Container\ContainerInterface;
|
||||
use ReflectionException;
|
||||
use Server\ServerManager;
|
||||
use Swoole\Coroutine;
|
||||
use Swoole\Process;
|
||||
use Swoole\WebSocket\Server;
|
||||
|
||||
|
||||
defined('DB_ERROR_BUSY') or define('DB_ERROR_BUSY', 'The database is busy. Please try again later.');
|
||||
defined('SELECT_IS_NULL') or define('SELECT_IS_NULL', 'Query data does not exist, please check the relevant conditions.');
|
||||
defined('PARAMS_IS_NULL') or define('PARAMS_IS_NULL', 'Required items cannot be empty, please add.');
|
||||
defined('CONTROLLER_PATH') or define('CONTROLLER_PATH', APP_PATH . 'app/Controllers/');
|
||||
defined('CRONTAB_PATH') or define('CRONTAB_PATH', APP_PATH . 'app/Crontab/');
|
||||
defined('CLIENT_PATH') or define('CLIENT_PATH', APP_PATH . 'app/Client/');
|
||||
defined('TASK_PATH') or define('TASK_PATH', APP_PATH . 'app/Async/');
|
||||
defined('LISTENER_PATH') or define('LISTENER_PATH', APP_PATH . 'app/Listener/');
|
||||
defined('KAFKA_PATH') or define('KAFKA_PATH', APP_PATH . 'app/Kafka/');
|
||||
defined('RPC_CLIENT_PATH') or define('RPC_CLIENT_PATH', APP_PATH . 'app/Client/Rpc/');
|
||||
defined('MODEL_PATH') or define('MODEL_PATH', APP_PATH . 'app/Model/');
|
||||
defined('CONTROLLER_PATH') or define('CONTROLLER_PATH', realpath(APP_PATH . 'controllers/'));
|
||||
defined('MODEL_PATH') or define('MODEL_PATH', realpath(APP_PATH . 'models/'));
|
||||
defined('COMPONENT_PATH') or define('COMPONENT_PATH', realpath(APP_PATH . 'components/'));
|
||||
|
||||
|
||||
/**
|
||||
@@ -70,6 +63,15 @@ class Kiri
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* @return Container
|
||||
*/
|
||||
public static function getContainer(): Container
|
||||
{
|
||||
return static::$container;
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* @param $alias
|
||||
* @param array $array
|
||||
@@ -136,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
|
||||
* @throws Exception
|
||||
@@ -165,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
|
||||
@@ -196,6 +170,7 @@ class Kiri
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* @return string
|
||||
* @throws Exception
|
||||
@@ -211,15 +186,6 @@ class Kiri
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* @return bool
|
||||
*/
|
||||
public static function inCoroutine(): bool
|
||||
{
|
||||
return Coroutine::getCid() > 0;
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* @return Container
|
||||
*/
|
||||
@@ -238,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
|
||||
*/
|
||||
@@ -285,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 $content
|
||||
@@ -330,7 +229,7 @@ class Kiri
|
||||
if ($is_append !== null) {
|
||||
$params[] = $is_append;
|
||||
}
|
||||
return !self::inCoroutine() ? file_put_contents(...$params) : Coroutine::writeFile(...$params);
|
||||
return !(Coroutine::getCid() > 0) ? file_put_contents(...$params) : Coroutine::writeFile(...$params);
|
||||
}
|
||||
|
||||
|
||||
@@ -351,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
|
||||
*/
|
||||
@@ -449,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(ServerManager::class);
|
||||
$manager->task(new $class(...$params));
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* @param array $v1
|
||||
* @param array $v2
|
||||
@@ -486,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
|
||||
* @return string
|
||||
@@ -536,34 +320,10 @@ class Kiri
|
||||
}
|
||||
|
||||
|
||||
private static array $_autoload = [];
|
||||
|
||||
|
||||
const PROCESS = 'process';
|
||||
const TASK = 'task';
|
||||
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
|
||||
*/
|
||||
@@ -599,35 +359,6 @@ class Kiri
|
||||
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());
|
||||
@@ -3,21 +3,21 @@
|
||||
|
||||
function version($oldVersion, $newVersion): bool
|
||||
{
|
||||
$first = explode('.', $oldVersion);
|
||||
$end = explode('.', $newVersion);
|
||||
while (count($first) > 0) {
|
||||
$shift = (int)array_shift($first);
|
||||
$endShift = (int)array_shift($end);
|
||||
if ($endShift == $shift) {
|
||||
continue;
|
||||
}
|
||||
if ($endShift < $shift) {
|
||||
return true;
|
||||
} else {
|
||||
return false;
|
||||
}
|
||||
}
|
||||
return false;
|
||||
$first = explode('.', $oldVersion);
|
||||
$end = explode('.', $newVersion);
|
||||
while (count($first) > 0) {
|
||||
$shift = (int)array_shift($first);
|
||||
$endShift = (int)array_shift($end);
|
||||
if ($endShift == $shift) {
|
||||
continue;
|
||||
}
|
||||
if ($endShift < $shift) {
|
||||
return TRUE;
|
||||
} else {
|
||||
return FALSE;
|
||||
}
|
||||
}
|
||||
return FALSE;
|
||||
}
|
||||
|
||||
var_dump(version('1.4.4','1.4.3'));
|
||||
var_dump(version('1.4.4', '1.4.3'));
|
||||
|
||||
+10
-8
@@ -9,7 +9,7 @@
|
||||
],
|
||||
"license": "MIT",
|
||||
"require": {
|
||||
"php": ">=8.0",
|
||||
"php": ">=8.1",
|
||||
"ext-json": "*",
|
||||
"ext-fileinfo": "*",
|
||||
"ext-pdo": "*",
|
||||
@@ -21,25 +21,27 @@
|
||||
"ext-xml": "*",
|
||||
"ext-curl": "*",
|
||||
"ext-openssl": "*",
|
||||
"symfony/console": "v5.3.10",
|
||||
"symfony/console": "~v5.3.10",
|
||||
"psr/log": "1.*",
|
||||
"ext-sockets": "*",
|
||||
"ext-pcntl": "*",
|
||||
"ext-posix": "*",
|
||||
"composer-runtime-api": "^2.0",
|
||||
"swiftmailer/swiftmailer": "v6.3.*",
|
||||
"psr/container": "^2.0",
|
||||
"psr/http-server-middleware": "1.0.1",
|
||||
"game-worker/kiri-event": "v1.0",
|
||||
"game-worker/kiri-event": "~v2.0",
|
||||
"game-worker/kiri-di": "~v1.0",
|
||||
"ext-inotify": "*"
|
||||
},
|
||||
"autoload": {
|
||||
"psr-4": {
|
||||
"Kiri\\": "kiri-engine/",
|
||||
"Kiri\\Gateway\\": "kiri-gateway/",
|
||||
"Kiri\\Websocket\\": "kiri-websocket-server/",
|
||||
"Gii\\": "kiri-gii/",
|
||||
"Annotation\\": "kiri-note/"
|
||||
"Kiri\\Annotation\\": "kiri-annotation/",
|
||||
"Kiri\\Server\\": "kiri-server/",
|
||||
"Kiri\\Task\\": "kiri-task/"
|
||||
},
|
||||
"files": [
|
||||
"Kiri.php",
|
||||
"error.php",
|
||||
"function.php"
|
||||
]
|
||||
|
||||
+157
-71
@@ -3,10 +3,10 @@
|
||||
defined('APP_PATH') or define('APP_PATH', realpath(__DIR__ . '/../../'));
|
||||
|
||||
|
||||
use Annotation\Annotation;
|
||||
use Http\Handler\Router;
|
||||
use JetBrains\PhpStorm\Pure;
|
||||
use Kiri\Abstracts\Config;
|
||||
use Kiri\Annotation\Annotation;
|
||||
use Kiri\Annotation\Route\Route;
|
||||
use Kiri\Application;
|
||||
use Kiri\Core\ArrayAccess;
|
||||
use Kiri\Di\NoteManager;
|
||||
@@ -14,10 +14,8 @@ use Kiri\Error\Logger;
|
||||
use Kiri\Events\EventDispatch;
|
||||
use Kiri\Events\EventProvider;
|
||||
use Kiri\Exception\ConfigException;
|
||||
use Kiri\Exception\NotFindClassException;
|
||||
use Kiri\Kiri;
|
||||
use Kiri\Message\Handler\Router;
|
||||
use Psr\Log\LoggerInterface;
|
||||
use Server\ServerManager;
|
||||
use Swoole\Process;
|
||||
use Swoole\WebSocket\Server;
|
||||
|
||||
@@ -30,7 +28,7 @@ if (!function_exists('make')) {
|
||||
* @return mixed
|
||||
* @throws
|
||||
*/
|
||||
function make($name, $default = null): mixed
|
||||
function make($name, $default = NULL): mixed
|
||||
{
|
||||
if (class_exists($name)) {
|
||||
return Kiri::createObject($name);
|
||||
@@ -45,7 +43,7 @@ if (!function_exists('make')) {
|
||||
return Kiri::app()->get($default);
|
||||
}
|
||||
$class = Kiri::createObject($default);
|
||||
class_alias($name, $default, true);
|
||||
class_alias($name, $default, TRUE);
|
||||
return $class;
|
||||
}
|
||||
|
||||
@@ -53,6 +51,22 @@ if (!function_exists('make')) {
|
||||
}
|
||||
|
||||
|
||||
if (!function_exists('map')) {
|
||||
|
||||
|
||||
/**
|
||||
* @param array $params
|
||||
* @param Closure $closure
|
||||
* @return mixed
|
||||
*/
|
||||
function map(array $params, Closure $closure): mixed
|
||||
{
|
||||
return array_map($closure, $params);
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
|
||||
if (!function_exists('checkPortIsAlready')) {
|
||||
|
||||
|
||||
@@ -65,7 +79,7 @@ if (!function_exists('checkPortIsAlready')) {
|
||||
{
|
||||
if (!Kiri::getPlatform()->isLinux()) {
|
||||
exec("lsof -i :" . $port . " | grep -i 'LISTEN' | awk '{print $2}'", $output);
|
||||
if (empty($output)) return false;
|
||||
if (empty($output)) return FALSE;
|
||||
$output = explode(PHP_EOL, $output[0]);
|
||||
return $output[0];
|
||||
}
|
||||
@@ -77,7 +91,7 @@ if (!function_exists('checkPortIsAlready')) {
|
||||
|
||||
exec('netstat -lnp | grep ' . $port . ' | grep "LISTEN" | awk \'{print $7}\'', $output);
|
||||
if (empty($output)) {
|
||||
return false;
|
||||
return FALSE;
|
||||
}
|
||||
return explode('/', $output[0])[0];
|
||||
}
|
||||
@@ -157,7 +171,7 @@ if (!function_exists('now')) {
|
||||
*/
|
||||
function now(): string
|
||||
{
|
||||
return date('Y-m-d H:i:s') . '.' . str_replace(time() . '.', '', (string)microtime(true));
|
||||
return date('Y-m-d H:i:s') . '.' . str_replace(time() . '.', '', (string)microtime(TRUE));
|
||||
}
|
||||
}
|
||||
|
||||
@@ -177,7 +191,7 @@ if (!function_exists('workerName')) {
|
||||
}
|
||||
|
||||
|
||||
if (!function_exists('annotation')) {
|
||||
if (!function_exists('Annotation')) {
|
||||
|
||||
|
||||
/**
|
||||
@@ -213,6 +227,22 @@ if (!function_exists('scan_directory')) {
|
||||
|
||||
}
|
||||
|
||||
if (!class_exists('ReturnTypeWillChange')) {
|
||||
|
||||
/**
|
||||
* @since 8.1
|
||||
*/
|
||||
#[Attribute(Attribute::TARGET_METHOD)]
|
||||
final class ReturnTypeWillChange
|
||||
{
|
||||
public function __construct()
|
||||
{
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
}
|
||||
|
||||
|
||||
if (!function_exists('injectRuntime')) {
|
||||
|
||||
@@ -227,8 +257,13 @@ if (!function_exists('injectRuntime')) {
|
||||
{
|
||||
$fileLists = Kiri::getAnnotation()->runtime($path, $exclude);
|
||||
$di = Kiri::getDi();
|
||||
|
||||
$router = [];
|
||||
foreach ($fileLists as $class) {
|
||||
foreach (NoteManager::getTargetNote($class) as $value) {
|
||||
foreach (NoteManager::getTargetAnnotation($class) as $value) {
|
||||
if (!method_exists($value, 'execute')) {
|
||||
continue;
|
||||
}
|
||||
$value->execute($class);
|
||||
}
|
||||
$methods = $di->getMethodAttribute($class);
|
||||
@@ -237,31 +272,31 @@ if (!function_exists('injectRuntime')) {
|
||||
continue;
|
||||
}
|
||||
foreach ($attribute as $item) {
|
||||
$item->execute($class, $method);
|
||||
if ($item instanceof Route) {
|
||||
$router[] = [$item, $class, $method];
|
||||
} else {
|
||||
if (!method_exists($item, 'execute')) {
|
||||
continue;
|
||||
}
|
||||
$item->execute($class, $method);
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
if (!empty($router)) {
|
||||
foreach ($router as $class) {
|
||||
[$item, $class, $method] = $class;
|
||||
if (!method_exists($item, 'execute')) {
|
||||
continue;
|
||||
}
|
||||
$item->execute($class, $method);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
|
||||
if (!function_exists('swoole')) {
|
||||
|
||||
|
||||
/**
|
||||
* @return Server|null
|
||||
* @throws Exception
|
||||
*/
|
||||
function swoole(): ?Server
|
||||
{
|
||||
return Kiri::getWebSocket();
|
||||
}
|
||||
|
||||
|
||||
}
|
||||
|
||||
|
||||
if (!function_exists('directory')) {
|
||||
|
||||
/**
|
||||
@@ -285,25 +320,28 @@ if (!function_exists('isUrl')) {
|
||||
* @param bool $get_info
|
||||
* @return false|array
|
||||
*/
|
||||
function isUrl($url, bool $get_info = true): bool|array
|
||||
function isUrl($url, bool $get_info = TRUE): bool|array
|
||||
{
|
||||
if (str_starts_with($url, '/')) {
|
||||
return false;
|
||||
return FALSE;
|
||||
}
|
||||
$queryMatch = '/((http[s]?):\/\/)?(([\w\-\_]+\.)+\w+(:\d+)?)(\/.*)?/';
|
||||
if (!preg_match($queryMatch, $url, $outPut)) {
|
||||
return false;
|
||||
return FALSE;
|
||||
}
|
||||
$port = str_replace(':', '', $outPut[5]);
|
||||
|
||||
[$isHttps, $domain, $port, $path] = [$outPut[2] == 'https', $outPut[3], $port, $outPut[6] ?? ''];
|
||||
if ($isHttps && empty($port)) {
|
||||
[$scheme, $host, $port, $user, $pass, $query, $path, $fragment] = parse_url($url);
|
||||
if ($scheme == 'https' && empty($port)) {
|
||||
$port = 443;
|
||||
}
|
||||
|
||||
if (!empty($query)) $path .= '?' . $query;
|
||||
if (!empty($fragment)) $path .= '#' . $fragment;
|
||||
|
||||
|
||||
unset($outPut);
|
||||
|
||||
return [$isHttps == 'https', $domain, $port, $path];
|
||||
return [$scheme == 'https', $host, $port, $path];
|
||||
}
|
||||
|
||||
}
|
||||
@@ -318,8 +356,8 @@ if (!function_exists('split_request_uri')) {
|
||||
*/
|
||||
function split_request_uri($url): bool|array
|
||||
{
|
||||
if (($parse = isUrl($url, null)) === false) {
|
||||
return false;
|
||||
if (($parse = isUrl($url, NULL)) === FALSE) {
|
||||
return FALSE;
|
||||
}
|
||||
|
||||
[$isHttps, $domain, $port, $path] = $parse;
|
||||
@@ -343,7 +381,7 @@ if (!function_exists('hadDomain')) {
|
||||
function hadDomain($url): bool|array
|
||||
{
|
||||
$param = split_request_uri($url);
|
||||
return !is_array($param) ? false : $param[0];
|
||||
return !is_array($param) ? FALSE : $param[0];
|
||||
}
|
||||
|
||||
}
|
||||
@@ -392,7 +430,7 @@ if (!function_exists('loadByDir')) {
|
||||
classAutoload($namespace, $value);
|
||||
} else {
|
||||
$pos = strpos($value, '.php');
|
||||
if ($pos === false || strlen($value) - 4 != $pos) {
|
||||
if ($pos === FALSE || strlen($value) - 4 != $pos) {
|
||||
continue;
|
||||
}
|
||||
|
||||
@@ -445,7 +483,7 @@ if (!function_exists('fire')) {
|
||||
|
||||
/**
|
||||
* @param object $event
|
||||
*/
|
||||
*/
|
||||
function fire(object $event)
|
||||
{
|
||||
di(EventDispatch::class)->dispatch($event);
|
||||
@@ -470,7 +508,7 @@ if (!function_exists('instance_load')) {
|
||||
|
||||
function instance_load()
|
||||
{
|
||||
$content = json_decode(file_get_contents(__DIR__ . '/composer.json'), true);
|
||||
$content = json_decode(file_get_contents(__DIR__ . '/composer.json'), TRUE);
|
||||
if (isset($content['autoload']) && isset($content['autoload']['psr-4'])) {
|
||||
$psr4 = $content['autoload']['psr-4'];
|
||||
foreach ($psr4 as $namespace => $dirname) {
|
||||
@@ -520,7 +558,7 @@ if (!function_exists('trim_blank')) {
|
||||
* @param bool $htmlTags
|
||||
* @return array|string|null
|
||||
*/
|
||||
function trim_blank(string $content, int $len = 0, string $encode = 'utf-8', bool $htmlTags = true): array|string|null
|
||||
function trim_blank(string $content, int $len = 0, string $encode = 'utf-8', bool $htmlTags = TRUE): array|string|null
|
||||
{
|
||||
$str = trim($content);
|
||||
if ($htmlTags) {
|
||||
@@ -540,7 +578,7 @@ if (!function_exists('trim_blank')) {
|
||||
if (!function_exists('get_file_extension')) {
|
||||
|
||||
function get_file_extension($filename): bool|int|string
|
||||
{
|
||||
{
|
||||
$mime_types = [
|
||||
'txt' => 'text/plain',
|
||||
'htm' => 'text/html',
|
||||
@@ -760,12 +798,12 @@ if (!function_exists('get_file_extension')) {
|
||||
$ext = strtolower(array_pop($explode));
|
||||
if (array_key_exists($ext, $mime_types)) {
|
||||
return $ext;
|
||||
} elseif (function_exists('finfo_open')) {
|
||||
} else if (function_exists('finfo_open')) {
|
||||
$fInfo = finfo_open(FILEINFO_MIME);
|
||||
$mimeType = finfo_file($fInfo, $filename);
|
||||
finfo_close($fInfo);
|
||||
$mimeType = current(explode('; ', $mimeType));
|
||||
if (($search = array_search($mimeType, $mime_types)) == false) {
|
||||
if (($search = array_search($mimeType, $mime_types)) == FALSE) {
|
||||
return $mimeType;
|
||||
}
|
||||
return $search;
|
||||
@@ -790,13 +828,13 @@ if (!function_exists('storage')) {
|
||||
if (!empty($path)) {
|
||||
$path = ltrim($path, '/');
|
||||
if (!is_dir($basePath . '/' . $path)) {
|
||||
mkdir($basePath . '/' . $path, 0777, true);
|
||||
mkdir($basePath . '/' . $path, 0777, TRUE);
|
||||
}
|
||||
}
|
||||
if (empty($fileName)) {
|
||||
return $basePath . '/' . $path . '/';
|
||||
return $basePath . '/' . $path;
|
||||
}
|
||||
$fileName = $basePath . '/' . $path . '/' . $fileName;
|
||||
$fileName = $basePath . '/' . $path . $fileName;
|
||||
if (!file_exists($fileName)) {
|
||||
touch($fileName);
|
||||
}
|
||||
@@ -814,7 +852,7 @@ if (!function_exists('event')) {
|
||||
* @param bool $isAppend
|
||||
* @throws Exception
|
||||
*/
|
||||
function event($name, $callback, bool $isAppend = true)
|
||||
function event($name, $callback, bool $isAppend = TRUE)
|
||||
{
|
||||
$pro = di(EventProvider::class);
|
||||
$pro->on($name, $callback, 0);
|
||||
@@ -831,7 +869,7 @@ if (!function_exists('name')) {
|
||||
* @throws ConfigException
|
||||
* @throws Exception
|
||||
*/
|
||||
function name(int $pid, string $prefix = null)
|
||||
function name(int $pid, string $prefix = NULL)
|
||||
{
|
||||
if (Kiri::getPlatform()->isMac()) {
|
||||
return;
|
||||
@@ -862,10 +900,10 @@ if (!function_exists('env')) {
|
||||
* @param null $default
|
||||
* @return array|string|null
|
||||
*/
|
||||
#[Pure] function env($key, $default = null): null|array|string
|
||||
#[Pure] function env($key, $default = NULL): null|array|string
|
||||
{
|
||||
$env = getenv($key);
|
||||
if ($env === false) {
|
||||
if ($env === FALSE) {
|
||||
return $default;
|
||||
}
|
||||
return $env;
|
||||
@@ -877,11 +915,10 @@ if (!function_exists('env')) {
|
||||
if (!function_exists('di')) {
|
||||
|
||||
|
||||
/**
|
||||
* @param string $className
|
||||
* @return mixed
|
||||
* @throws ReflectionException
|
||||
*/
|
||||
/**
|
||||
* @param string $className
|
||||
* @return mixed
|
||||
*/
|
||||
function di(string $className): mixed
|
||||
{
|
||||
return Kiri::getDi()->get($className);
|
||||
@@ -889,18 +926,43 @@ if (!function_exists('di')) {
|
||||
|
||||
}
|
||||
|
||||
|
||||
if (!function_exists('interval')) {
|
||||
|
||||
|
||||
/**
|
||||
* @param callable $callback
|
||||
* @param int $interval
|
||||
* @param bool $is
|
||||
*/
|
||||
function interval(callable $callback, int $interval = 1000, bool $is = FALSE)
|
||||
{
|
||||
usleep($interval * 1000);
|
||||
|
||||
$callback();
|
||||
|
||||
interval($callback, $interval, $is);
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
|
||||
if (!function_exists('duplicate')) {
|
||||
|
||||
|
||||
/**
|
||||
* @param string $className
|
||||
* @return mixed
|
||||
* @throws ReflectionException
|
||||
*/
|
||||
/**
|
||||
* @param string $className
|
||||
* @return mixed
|
||||
* @throws ReflectionException
|
||||
*/
|
||||
function duplicate(string $className): mixed
|
||||
{
|
||||
$class = di($className);
|
||||
return clone $class;
|
||||
$clone = clone $class;
|
||||
if (method_exists($clone, 'clear')) {
|
||||
$clone->clear();
|
||||
}
|
||||
return $clone;
|
||||
}
|
||||
|
||||
}
|
||||
@@ -952,7 +1014,7 @@ if (!function_exists('swoole_unserialize')) {
|
||||
function swoole_unserialize($data): mixed
|
||||
{
|
||||
if (empty($data)) {
|
||||
return null;
|
||||
return NULL;
|
||||
}
|
||||
// if (class_exists('swoole_serialize')) {
|
||||
// return \swoole_serialize::unpack($data);
|
||||
@@ -1033,7 +1095,7 @@ if (!function_exists('jTraceEx')) {
|
||||
* @param bool $toHtml
|
||||
* @return string
|
||||
*/
|
||||
function jTraceEx($e, $seen = null, bool $toHtml = false): string
|
||||
function jTraceEx($e, $seen = NULL, bool $toHtml = FALSE): string
|
||||
{
|
||||
$starter = $seen ? 'Caused by: ' : '';
|
||||
$result = [];
|
||||
@@ -1049,12 +1111,12 @@ if (!function_exists('jTraceEx')) {
|
||||
count($value) && array_key_exists('class', $value) ? str_replace('\\', '.', $value['class']) : '',
|
||||
count($value) && array_key_exists('class', $value) && array_key_exists('function', $value) ? '.' : '',
|
||||
count($value) && array_key_exists('function', $value) ? str_replace('\\', '.', $value['function']) : '(main)',
|
||||
$line === null ? $file : basename($file),
|
||||
$line === null ? '' : ':',
|
||||
$line === null ? '' : $line);
|
||||
$line === NULL ? $file : basename($file),
|
||||
$line === NULL ? '' : ':',
|
||||
$line === NULL ? '' : $line);
|
||||
|
||||
$file = array_key_exists('file', $value) ? $value['file'] : 'Unknown Source';
|
||||
$line = array_key_exists('file', $value) && array_key_exists('line', $value) && $value['line'] ? $value['line'] : null;
|
||||
$line = array_key_exists('file', $value) && array_key_exists('line', $value) && $value['line'] ? $value['line'] : NULL;
|
||||
}
|
||||
$result = join($toHtml ? "<br>" : "\n", $result);
|
||||
if ($prev) {
|
||||
@@ -1078,7 +1140,7 @@ if (!function_exists('swoole_substr_json_decode')) {
|
||||
*/
|
||||
function swoole_substr_json_decode($packet, int $length = 0): mixed
|
||||
{
|
||||
return json_decode($packet, true);
|
||||
return json_decode($packet, TRUE);
|
||||
}
|
||||
|
||||
}
|
||||
@@ -1153,3 +1215,27 @@ if (!function_exists('success')) {
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
if (!function_exists('error_trigger_format')) {
|
||||
|
||||
|
||||
/**
|
||||
* @param Throwable|Error $throwable
|
||||
* @return string
|
||||
*/
|
||||
function error_trigger_format(\Throwable|\Error $throwable): string
|
||||
{
|
||||
$message = "Throwable: " . $throwable->getMessage() . "\n" . ' ' . $throwable->getFile() . " at line " . $throwable->getLine() . "\n";
|
||||
|
||||
$message .= "trance\n";
|
||||
foreach ($throwable->getTrace() as $value) {
|
||||
if (!isset($value['file'])) {
|
||||
continue;
|
||||
}
|
||||
$message .= $value['file'] . " -> " . $value['line'] . "(" . (isset($value['class']) ? $value['class'] . '::' : '') . ($value['function'] ?? 'Closure') . ")\n";
|
||||
}
|
||||
return "\033[41;37m" . $message . "\033[0m";
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
|
||||
@@ -1,7 +1,7 @@
|
||||
<?php
|
||||
|
||||
|
||||
namespace Annotation;
|
||||
namespace Kiri\Annotation;
|
||||
|
||||
|
||||
use DirectoryIterator;
|
||||
@@ -1,7 +1,7 @@
|
||||
<?php
|
||||
|
||||
|
||||
namespace Annotation;
|
||||
namespace Kiri\Annotation;
|
||||
|
||||
|
||||
use Exception;
|
||||
@@ -1,7 +1,7 @@
|
||||
<?php
|
||||
|
||||
|
||||
namespace Annotation;
|
||||
namespace Kiri\Annotation;
|
||||
|
||||
|
||||
/**
|
||||
@@ -17,6 +17,7 @@ abstract class Attribute implements IAnnotation
|
||||
* @param mixed|string $method
|
||||
* @return mixed
|
||||
*/
|
||||
#[\ReturnTypeWillChange]
|
||||
public function execute(mixed $class, mixed $method = ''): mixed
|
||||
{
|
||||
// TODO: Implement execute() method.
|
||||
@@ -1,12 +1,12 @@
|
||||
<?php
|
||||
|
||||
|
||||
namespace Annotation;
|
||||
namespace Kiri\Annotation;
|
||||
|
||||
|
||||
use Exception;
|
||||
use Kiri\Events\EventProvider;
|
||||
use Kiri\Kiri;
|
||||
use Kiri;
|
||||
|
||||
|
||||
/**
|
||||
@@ -1,7 +1,7 @@
|
||||
<?php
|
||||
|
||||
|
||||
namespace Annotation;
|
||||
namespace Kiri\Annotation;
|
||||
|
||||
|
||||
|
||||
@@ -1,12 +1,12 @@
|
||||
<?php
|
||||
|
||||
|
||||
namespace Annotation;
|
||||
namespace Kiri\Annotation;
|
||||
|
||||
|
||||
use Exception;
|
||||
use Kiri\Core\Str;
|
||||
use Kiri\Kiri;
|
||||
use Kiri;
|
||||
use ReflectionException;
|
||||
use ReflectionProperty;
|
||||
|
||||
@@ -70,7 +70,6 @@ use ReflectionProperty;
|
||||
* @param $class
|
||||
* @param $method
|
||||
* @return ReflectionProperty|bool
|
||||
* @throws ReflectionException
|
||||
*/
|
||||
private function getProperty($class, $method): ReflectionProperty|bool
|
||||
{
|
||||
@@ -1,13 +1,13 @@
|
||||
<?php
|
||||
|
||||
|
||||
namespace Annotation;
|
||||
namespace Kiri\Annotation;
|
||||
|
||||
|
||||
use DirectoryIterator;
|
||||
use Exception;
|
||||
use Kiri\Abstracts\BaseObject;
|
||||
use Kiri\Kiri;
|
||||
use Kiri;
|
||||
use Kiri\Abstracts\Component;
|
||||
use ReflectionClass;
|
||||
use ReflectionException;
|
||||
use Throwable;
|
||||
@@ -17,19 +17,13 @@ use Throwable;
|
||||
* Class Loader
|
||||
* @package Annotation
|
||||
*/
|
||||
class Loader extends BaseObject
|
||||
class Loader extends Component
|
||||
{
|
||||
|
||||
|
||||
private array $_classes = [];
|
||||
|
||||
|
||||
private array $_directory = [];
|
||||
|
||||
|
||||
private array $_property = [];
|
||||
|
||||
|
||||
private array $_methods = [];
|
||||
|
||||
|
||||
@@ -43,17 +37,6 @@ class Loader extends BaseObject
|
||||
$this->_scanDir(new DirectoryIterator($path), $namespace);
|
||||
}
|
||||
|
||||
/**
|
||||
* @param string $class
|
||||
* @param string $property
|
||||
* @return \ReflectionProperty|array|null
|
||||
* @throws ReflectionException
|
||||
*/
|
||||
public function getProperty(string $class, string $property = ''): \ReflectionProperty|array|null
|
||||
{
|
||||
return Kiri::getDi()->getClassReflectionProperty($class, $property);
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* @param string $class
|
||||
@@ -101,6 +84,9 @@ class Loader extends BaseObject
|
||||
public function _scanDir(DirectoryIterator $paths, $namespace, array $exclude = [])
|
||||
{
|
||||
foreach ($paths as $path) {
|
||||
if (function_exists('opcache_invalidate')) {
|
||||
opcache_invalidate($path->getRealPath(), true);
|
||||
}
|
||||
if ($path->isDot() || str_starts_with($path->getFilename(), '.')) {
|
||||
continue;
|
||||
}
|
||||
@@ -147,7 +133,6 @@ class Loader extends BaseObject
|
||||
* @param DirectoryIterator $path
|
||||
* @param string $namespace
|
||||
* @return ReflectionClass|null
|
||||
* @throws ReflectionException
|
||||
*/
|
||||
private function getReflect(DirectoryIterator $path, string $namespace): ?ReflectionClass
|
||||
{
|
||||
@@ -1,8 +1,8 @@
|
||||
<?php
|
||||
|
||||
namespace Annotation;
|
||||
namespace Kiri\Annotation;
|
||||
|
||||
use Kiri\Kiri;
|
||||
use Kiri;
|
||||
|
||||
#[\Attribute(\Attribute::TARGET_CLASS)] class Mapping extends Attribute
|
||||
{
|
||||
@@ -1,10 +1,10 @@
|
||||
<?php
|
||||
|
||||
|
||||
namespace Annotation\Route;
|
||||
namespace Kiri\Annotation\Route;
|
||||
|
||||
|
||||
use Annotation\Attribute;
|
||||
use Kiri\Annotation\Attribute;
|
||||
|
||||
/**
|
||||
* Class Document
|
||||
@@ -1,11 +1,11 @@
|
||||
<?php
|
||||
|
||||
|
||||
namespace Annotation\Route;
|
||||
namespace Kiri\Annotation\Route;
|
||||
|
||||
|
||||
use Annotation\Attribute;
|
||||
use Http\Handler\Abstracts\MiddlewareManager;
|
||||
use Kiri\Annotation\Attribute;
|
||||
use Kiri\Message\Handler\Abstracts\MiddlewareManager;
|
||||
use Psr\Http\Server\MiddlewareInterface;
|
||||
|
||||
/**
|
||||
@@ -41,6 +41,7 @@ use Psr\Http\Server\MiddlewareInterface;
|
||||
* @param mixed $class
|
||||
* @param mixed|null $method
|
||||
* @return $this
|
||||
* @throws \ReflectionException
|
||||
*/
|
||||
public function execute(mixed $class, mixed $method = null): mixed
|
||||
{
|
||||
@@ -1,12 +1,12 @@
|
||||
<?php
|
||||
|
||||
|
||||
namespace Annotation\Route;
|
||||
namespace Kiri\Annotation\Route;
|
||||
|
||||
|
||||
use Annotation\Attribute;
|
||||
use Http\Handler\Router;
|
||||
use Kiri\Kiri;
|
||||
use Kiri\Annotation\Attribute;
|
||||
use Kiri\Message\Handler\Router;
|
||||
use Kiri;
|
||||
|
||||
#[\Attribute(\Attribute::TARGET_METHOD | \Attribute::IS_REPEATABLE)] class Route extends Attribute
|
||||
{
|
||||
@@ -1,10 +1,10 @@
|
||||
<?php
|
||||
|
||||
|
||||
namespace Annotation\Route;
|
||||
namespace Kiri\Annotation\Route;
|
||||
|
||||
|
||||
use Annotation\Attribute;
|
||||
use Kiri\Annotation\Attribute;
|
||||
|
||||
/**
|
||||
* Class Socket
|
||||
@@ -1,7 +1,7 @@
|
||||
<?php
|
||||
|
||||
|
||||
namespace Annotation;
|
||||
namespace Kiri\Annotation;
|
||||
|
||||
|
||||
/**
|
||||
@@ -0,0 +1,12 @@
|
||||
<?php
|
||||
|
||||
namespace Kiri\Abstracts;
|
||||
|
||||
abstract class AbstractServer extends Component
|
||||
{
|
||||
|
||||
|
||||
public string $name = 'http';
|
||||
|
||||
|
||||
}
|
||||
@@ -10,30 +10,26 @@ declare(strict_types=1);
|
||||
namespace Kiri\Abstracts;
|
||||
|
||||
|
||||
use Annotation\Annotation as SAnnotation;
|
||||
use Database\Connection;
|
||||
use Exception;
|
||||
use Http\Handler\Router;
|
||||
use Kiri\Events\OnBeforeCommandExecute;
|
||||
use Server\Server;
|
||||
use Kafka\KafkaProvider;
|
||||
use Kiri;
|
||||
use Kiri\Annotation\Annotation as SAnnotation;
|
||||
use Kiri\Async;
|
||||
use Kiri\Cache\Redis;
|
||||
use Kiri\Di\LocalService;
|
||||
use Kiri\Error\ErrorHandler;
|
||||
use Kiri\Error\Logger;
|
||||
use Kiri\Events\EventProvider;
|
||||
use Kiri\Exception\InitException;
|
||||
use Kiri\Exception\NotFindClassException;
|
||||
use Kiri\Kiri;
|
||||
use Kiri\Error\{ErrorHandler, Logger};
|
||||
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 Server\ServerManager;
|
||||
use Server\Contract\OnTaskInterface;
|
||||
use Swoole\Table;
|
||||
|
||||
/**
|
||||
* Class BaseApplication
|
||||
* @package Kiri\Kiri\Base
|
||||
* @package Kiri\Base
|
||||
*/
|
||||
abstract class BaseApplication extends Component
|
||||
{
|
||||
@@ -206,11 +202,11 @@ abstract class BaseApplication extends Component
|
||||
|
||||
/**
|
||||
* @param OnTaskInterface $execute
|
||||
* @throws ReflectionException
|
||||
* @throws ReflectionException|Exception
|
||||
*/
|
||||
public function task(OnTaskInterface $execute): void
|
||||
{
|
||||
di(ServerManager::class)->task($execute);
|
||||
di(AsyncTaskExecute::class)->execute($execute);
|
||||
}
|
||||
|
||||
|
||||
@@ -222,22 +218,21 @@ abstract class BaseApplication extends Component
|
||||
*/
|
||||
private function addEvent($key, $value): void
|
||||
{
|
||||
$eventProvider = di(EventProvider::class);
|
||||
if ($value instanceof \Closure || is_object($value)) {
|
||||
$eventProvider->on($key, $value, 0);
|
||||
$this->getEventProvider()->on($key, $value, 0);
|
||||
return;
|
||||
}
|
||||
|
||||
|
||||
if (is_array($value)) {
|
||||
if (is_object($value[0]) && !($value[0] instanceof \Closure)) {
|
||||
$eventProvider->on($key, $value, 0);
|
||||
$this->getEventProvider()->on($key, $value, 0);
|
||||
return;
|
||||
}
|
||||
|
||||
if (is_string($value[0])) {
|
||||
$value[0] = Kiri::createObject($value[0]);
|
||||
$eventProvider->on($key, $value, 0);
|
||||
$this->getEventProvider()->on($key, $value, 0);
|
||||
return;
|
||||
}
|
||||
|
||||
@@ -246,7 +241,7 @@ abstract class BaseApplication extends Component
|
||||
if (!is_callable($item, true)) {
|
||||
throw new InitException("Class does not hav callback.");
|
||||
}
|
||||
$eventProvider->on($key, $item, 0);
|
||||
$this->getEventProvider()->on($key, $item, 0);
|
||||
}
|
||||
}
|
||||
|
||||
@@ -377,7 +372,7 @@ abstract class BaseApplication extends Component
|
||||
*/
|
||||
public function getServer(): Server
|
||||
{
|
||||
return $this->get('server');
|
||||
return Kiri::getDi()->get(Server::class);
|
||||
}
|
||||
|
||||
|
||||
@@ -397,7 +392,7 @@ abstract class BaseApplication extends Component
|
||||
*/
|
||||
public function getAnnotation(): SAnnotation
|
||||
{
|
||||
return $this->get('annotation');
|
||||
return $this->get('Annotation');
|
||||
}
|
||||
|
||||
|
||||
@@ -411,7 +406,6 @@ abstract class BaseApplication extends Component
|
||||
}
|
||||
|
||||
|
||||
|
||||
/**
|
||||
* @param $array
|
||||
*/
|
||||
@@ -450,9 +444,8 @@ abstract class BaseApplication extends Component
|
||||
'error' => ['class' => ErrorHandler::class],
|
||||
'config' => ['class' => Config::class],
|
||||
'logger' => ['class' => Logger::class],
|
||||
'annotation' => ['class' => SAnnotation::class],
|
||||
'Annotation' => ['class' => SAnnotation::class],
|
||||
'databases' => ['class' => Connection::class],
|
||||
'jwt' => ['class' => Jwt::class],
|
||||
'async' => ['class' => Async::class],
|
||||
'kafka-container' => ['class' => KafkaProvider::class],
|
||||
]);
|
||||
|
||||
@@ -1,243 +0,0 @@
|
||||
<?php
|
||||
/**
|
||||
* Created by PhpStorm.
|
||||
* User: whwyy
|
||||
* Date: 2018/3/30 0030
|
||||
* Time: 14:10
|
||||
*/
|
||||
declare(strict_types=1);
|
||||
|
||||
namespace Kiri\Abstracts;
|
||||
|
||||
use Exception;
|
||||
use JetBrains\PhpStorm\Pure;
|
||||
use Kiri\Kiri;
|
||||
use Swoole\Coroutine;
|
||||
|
||||
/**
|
||||
* Class BaseObject
|
||||
* @package Kiri\Kiri\Base
|
||||
*/
|
||||
class BaseObject implements Configure
|
||||
{
|
||||
|
||||
/**
|
||||
* BaseAbstract constructor.
|
||||
*
|
||||
* @param array $config
|
||||
* @throws Exception
|
||||
*/
|
||||
public function __construct(array $config = [])
|
||||
{
|
||||
if (!empty($config) && is_array($config)) {
|
||||
Kiri::configure($this, $config);
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* @throws Exception
|
||||
*/
|
||||
public function init()
|
||||
{
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* @param array|callable $callback
|
||||
* @param object $scope
|
||||
*/
|
||||
public function async_create(array|callable $callback, object $scope)
|
||||
{
|
||||
Coroutine::create($callback, $scope);
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* @return string
|
||||
*/
|
||||
#[Pure] public static function className(): string
|
||||
{
|
||||
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 string $model
|
||||
* @return bool
|
||||
* @throws Exception
|
||||
*/
|
||||
public function addError($message, string $model = 'app'): bool
|
||||
{
|
||||
if ($message instanceof \Throwable) {
|
||||
$this->error(jTraceEx($message));
|
||||
} else {
|
||||
if (!is_string($message)) {
|
||||
$message = json_encode($message, JSON_UNESCAPED_UNICODE);
|
||||
}
|
||||
$this->error($message);
|
||||
}
|
||||
return FALSE;
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* @return Logger
|
||||
* @throws Exception
|
||||
*/
|
||||
private function logger(): Logger
|
||||
{
|
||||
return Kiri::getDi()->get(Logger::class);
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* @param mixed $message
|
||||
* @param string $method
|
||||
* @param string $file
|
||||
* @throws Exception
|
||||
*/
|
||||
public function debug(mixed $message, string $method = '', string $file = '')
|
||||
{
|
||||
if (!is_string($message)) {
|
||||
$message = print_r($message, true);
|
||||
}
|
||||
$message = "\033[35m" . $message . "\033[0m";
|
||||
|
||||
|
||||
$context = [];
|
||||
if (!empty($method)) $context['method'] = $method;
|
||||
if (!empty($file)) $context['file'] = $file;
|
||||
|
||||
$this->logger()->debug($message, $context);
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* @param mixed $message
|
||||
* @param string $method
|
||||
* @param string $file
|
||||
* @throws Exception
|
||||
*/
|
||||
public function info(mixed $message, string $method = '', string $file = '')
|
||||
{
|
||||
if (!is_string($message)) {
|
||||
$message = print_r($message, true);
|
||||
}
|
||||
$message = "\033[34m" . $message . "\033[0m";
|
||||
|
||||
|
||||
$context = [];
|
||||
if (!empty($method)) $context['method'] = $method;
|
||||
if (!empty($file)) $context['file'] = $file;
|
||||
|
||||
$this->logger()->info($message, $context);
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* @param mixed $message
|
||||
* @param string $method
|
||||
* @param string $file
|
||||
* @throws Exception
|
||||
*/
|
||||
public function success(mixed $message, string $method = '', string $file = '')
|
||||
{
|
||||
if (!is_string($message)) {
|
||||
$message = print_r($message, true);
|
||||
}
|
||||
|
||||
$message = "\033[36m" . $message . "\033[0m";
|
||||
|
||||
|
||||
$context = [];
|
||||
if (!empty($method)) $context['method'] = $method;
|
||||
if (!empty($file)) $context['file'] = $file;
|
||||
|
||||
$this->logger()->notice($message, $context);
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* @param mixed $message
|
||||
* @param string $method
|
||||
* @param string $file
|
||||
* @throws Exception
|
||||
*/
|
||||
public function warning(mixed $message, string $method = '', string $file = '')
|
||||
{
|
||||
if (!is_string($message)) {
|
||||
$message = print_r($message, true);
|
||||
}
|
||||
|
||||
$message = "\033[33m" . $message . "\033[0m";
|
||||
|
||||
|
||||
$context = [];
|
||||
if (!empty($method)) $context['method'] = $method;
|
||||
if (!empty($file)) $context['file'] = $file;
|
||||
|
||||
$this->logger()->critical($message, $context);
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* @param mixed $message
|
||||
* @param null $method
|
||||
* @param null $file
|
||||
* @throws Exception
|
||||
*/
|
||||
public function error(mixed $message, $method = null, $file = null)
|
||||
{
|
||||
if ($message instanceof \Throwable) {
|
||||
$message = $message->getMessage() . " on line " . $message->getLine() . " at file " . $message->getFile();
|
||||
}
|
||||
$content = (empty($method) ? '' : $method . ': ') . $message;
|
||||
|
||||
$message = "\033[41;37m" . $content . "\033[0m";
|
||||
|
||||
if (!empty($file)) {
|
||||
$message .= PHP_EOL . "\03341;37m[" . $file . "\033[0m";
|
||||
}
|
||||
|
||||
$context = [];
|
||||
if (!empty($method)) $context['method'] = $method;
|
||||
if (!empty($file)) $context['file'] = $file;
|
||||
|
||||
$this->logger()->error($message, $context);
|
||||
}
|
||||
|
||||
}
|
||||
@@ -11,45 +11,217 @@ namespace Kiri\Abstracts;
|
||||
|
||||
|
||||
use Exception;
|
||||
|
||||
use JetBrains\PhpStorm\Pure;
|
||||
use Kiri\Exception\ComponentException;
|
||||
use Kiri\Kiri;
|
||||
use Kiri\Di\Container;
|
||||
use Kiri\Events\EventDispatch;
|
||||
use Kiri\Events\EventProvider;
|
||||
use Kiri;
|
||||
use Psr\Container\ContainerExceptionInterface;
|
||||
use Psr\Container\ContainerInterface;
|
||||
use Psr\Container\NotFoundExceptionInterface;
|
||||
|
||||
/**
|
||||
* Class Component
|
||||
* @package Kiri\Kiri\Base
|
||||
* @package Kiri\Base
|
||||
*/
|
||||
class Component extends BaseObject
|
||||
class Component implements Configure
|
||||
{
|
||||
|
||||
|
||||
/**
|
||||
* @param $name
|
||||
* @param $value
|
||||
* BaseAbstract constructor.
|
||||
*
|
||||
* @param array $config
|
||||
* @throws Exception
|
||||
*/
|
||||
public function __set($name, $value)
|
||||
public function __construct(array $config = [])
|
||||
{
|
||||
if (property_exists($this, $name)) {
|
||||
$this->$name = $value;
|
||||
} else {
|
||||
parent::__set($name, $value);
|
||||
if (!empty($config) && is_array($config)) {
|
||||
Kiri::configure($this, $config);
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* @param $name
|
||||
* @return mixed
|
||||
* @return Container|ContainerInterface
|
||||
*/
|
||||
#[Pure] public function getContainer(): ContainerInterface|Container
|
||||
{
|
||||
return Kiri::getDi();
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* @return EventProvider
|
||||
* @throws ContainerExceptionInterface
|
||||
* @throws NotFoundExceptionInterface
|
||||
*/
|
||||
public function getEventProvider(): EventProvider
|
||||
{
|
||||
return $this->getContainer()->get(EventProvider::class);
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* @return EventDispatch
|
||||
* @throws ContainerExceptionInterface
|
||||
* @throws NotFoundExceptionInterface
|
||||
*/
|
||||
protected function getEventDispatch(): EventDispatch
|
||||
{
|
||||
return $this->getContainer()->get(EventDispatch::class);
|
||||
}
|
||||
|
||||
/**
|
||||
* @throws Exception
|
||||
*/
|
||||
public function __get($name): mixed
|
||||
public function init()
|
||||
{
|
||||
if (property_exists($this, $name)) {
|
||||
return $this->$name ?? null;
|
||||
} else {
|
||||
return parent::__get($name);
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* @return string
|
||||
*/
|
||||
#[Pure] public static function className(): string
|
||||
{
|
||||
return static::class;
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* @param $message
|
||||
* @param string $model
|
||||
* @return bool
|
||||
* @throws Exception
|
||||
*/
|
||||
public function addError($message, string $model = 'app'): bool
|
||||
{
|
||||
if ($message instanceof \Throwable) {
|
||||
$this->error($message = jTraceEx($message));
|
||||
} else {
|
||||
if (!is_string($message)) {
|
||||
$message = json_encode($message, JSON_UNESCAPED_UNICODE);
|
||||
}
|
||||
$this->error($message);
|
||||
}
|
||||
Kiri::app()->getLogger()->fail($message, $model);
|
||||
return FALSE;
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* @return Logger
|
||||
* @throws Exception
|
||||
*/
|
||||
protected function logger(): Logger
|
||||
{
|
||||
return Kiri::getDi()->get(Logger::class);
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* @param mixed $message
|
||||
* @param string $method
|
||||
* @param string $file
|
||||
* @throws Exception
|
||||
*/
|
||||
public function debug(mixed $message, string $method = '', string $file = '')
|
||||
{
|
||||
if (!is_string($message)) {
|
||||
$message = print_r($message, true);
|
||||
}
|
||||
$context = [];
|
||||
if (!empty($method)) $context['method'] = $method;
|
||||
if (!empty($file)) $context['file'] = $file;
|
||||
|
||||
$this->logger()->debug($message, $context);
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* @param mixed $message
|
||||
* @param string $method
|
||||
* @param string $file
|
||||
* @throws Exception
|
||||
*/
|
||||
public function info(mixed $message, string $method = '', string $file = '')
|
||||
{
|
||||
if (!is_string($message)) {
|
||||
$message = print_r($message, true);
|
||||
}
|
||||
$context = [];
|
||||
if (!empty($method)) $context['method'] = $method;
|
||||
if (!empty($file)) $context['file'] = $file;
|
||||
|
||||
$this->logger()->info($message, $context);
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* @param mixed $message
|
||||
* @param string $method
|
||||
* @param string $file
|
||||
* @throws Exception
|
||||
*/
|
||||
public function success(mixed $message, string $method = '', string $file = '')
|
||||
{
|
||||
if (!is_string($message)) {
|
||||
$message = print_r($message, true);
|
||||
}
|
||||
$context = [];
|
||||
if (!empty($method)) $context['method'] = $method;
|
||||
if (!empty($file)) $context['file'] = $file;
|
||||
|
||||
$this->logger()->notice($message, $context);
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* @param mixed $message
|
||||
* @param string $method
|
||||
* @param string $file
|
||||
* @throws Exception
|
||||
*/
|
||||
public function warning(mixed $message, string $method = '', string $file = '')
|
||||
{
|
||||
if (!is_string($message)) {
|
||||
$message = print_r($message, true);
|
||||
}
|
||||
|
||||
$context = [];
|
||||
if (!empty($method)) $context['method'] = $method;
|
||||
if (!empty($file)) $context['file'] = $file;
|
||||
|
||||
$this->logger()->critical($message, $context);
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* @param mixed $message
|
||||
* @param null $method
|
||||
* @param null $file
|
||||
* @throws Exception
|
||||
*/
|
||||
public function error(mixed $message, $method = null, $file = null)
|
||||
{
|
||||
if ($message instanceof \Throwable) {
|
||||
$message = $message->getMessage() . " on line " . $message->getLine() . " at file " . $message->getFile();
|
||||
}
|
||||
|
||||
$context = [];
|
||||
if (is_string($method)) {
|
||||
$message = (empty($method) ? '' : $method . ': ') . $message;
|
||||
} else {
|
||||
if (is_null($method)) {
|
||||
$method = [];
|
||||
}
|
||||
$context = $method;
|
||||
}
|
||||
if (!empty($method)) $context['method'] = $method;
|
||||
if (!empty($file)) $context['file'] = $file;
|
||||
|
||||
$this->logger()->error($message, $context);
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
@@ -9,29 +9,27 @@ declare(strict_types=1);
|
||||
|
||||
namespace Kiri\Abstracts;
|
||||
|
||||
use Exception;
|
||||
use Kiri\Exception\ConfigException;
|
||||
use Kiri\Kiri;
|
||||
|
||||
|
||||
/**
|
||||
* Class Config
|
||||
* @package Kiri\Kiri\Base
|
||||
* @package Kiri\Base
|
||||
*/
|
||||
class Config extends Component
|
||||
{
|
||||
|
||||
const ERROR_MESSAGE = 'The not find %s in app configs.';
|
||||
|
||||
protected mixed $data = [];
|
||||
protected static mixed $data = [];
|
||||
|
||||
|
||||
/**
|
||||
* @return mixed
|
||||
*/
|
||||
public function getData(): mixed
|
||||
public static function getData(): mixed
|
||||
{
|
||||
return $this->data;
|
||||
return static::$data;
|
||||
}
|
||||
|
||||
|
||||
@@ -40,23 +38,21 @@ class Config extends Component
|
||||
* @param $value
|
||||
* @return mixed
|
||||
*/
|
||||
public function setData($key, $value): mixed
|
||||
public static function setData($key, $value): mixed
|
||||
{
|
||||
return $this->data[$key] = $value;
|
||||
return static::$data[$key] = $value;
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* @param array $configs
|
||||
* @throws Exception
|
||||
*/
|
||||
public static function sets(array $configs)
|
||||
{
|
||||
$config = Kiri::app()->getConfig();
|
||||
if (empty($configs)) {
|
||||
return;
|
||||
}
|
||||
$config->data = $configs;
|
||||
static::$data = $configs;
|
||||
}
|
||||
|
||||
/**
|
||||
@@ -64,11 +60,11 @@ class Config extends Component
|
||||
* @param bool $try
|
||||
* @param mixed|null $default
|
||||
* @return mixed
|
||||
* @throws
|
||||
* @throws ConfigException
|
||||
*/
|
||||
public static function get($key, mixed $default = null, bool $try = FALSE): mixed
|
||||
{
|
||||
$instance = Kiri::app()->getConfig()->getData();
|
||||
$instance = static::$data;
|
||||
if (!str_contains($key, '.')) {
|
||||
return $instance[$key] ?? $default;
|
||||
}
|
||||
@@ -94,27 +90,35 @@ class Config extends Component
|
||||
* @param $key
|
||||
* @param $value
|
||||
* @return mixed
|
||||
* @throws Exception
|
||||
*/
|
||||
public static function set($key, $value): mixed
|
||||
{
|
||||
$config = Kiri::app()->getConfig();
|
||||
return $config->setData($key, $value);
|
||||
$explode = explode('.', $key);
|
||||
$parent = &static::$data;
|
||||
foreach ($explode as $item) {
|
||||
if (!isset($parent[$item])) {
|
||||
$parent[$item] = [];
|
||||
}
|
||||
$parent = &$parent[$item];
|
||||
}
|
||||
$parent = $value;
|
||||
|
||||
unset($parent);
|
||||
|
||||
return static::$data;
|
||||
}
|
||||
|
||||
/**
|
||||
* @param $key
|
||||
* @param bool $must_not_null
|
||||
* @return bool
|
||||
* @throws Exception
|
||||
*/
|
||||
public static function has($key, bool $must_not_null = false): bool
|
||||
{
|
||||
$config = Kiri::app()->getConfig();
|
||||
if (!isset($config->data[$key])) {
|
||||
if (!isset(static::$data[$key])) {
|
||||
return false;
|
||||
}
|
||||
$config = $config->data[$key];
|
||||
$config = static::$data[$key];
|
||||
if ($must_not_null === false) {
|
||||
return true;
|
||||
}
|
||||
|
||||
@@ -10,7 +10,7 @@ namespace Kiri\Abstracts;
|
||||
|
||||
/**
|
||||
* Interface Configure
|
||||
* @package Kiri\Kiri\Base
|
||||
* @package Kiri\Base
|
||||
*/
|
||||
interface Configure
|
||||
{
|
||||
|
||||
@@ -2,12 +2,14 @@
|
||||
|
||||
namespace Kiri\Abstracts;
|
||||
|
||||
use Annotation\Inject;
|
||||
use DirectoryIterator;
|
||||
use Exception;
|
||||
use Kiri\Events\EventProvider;
|
||||
use Kiri\Exception\ConfigException;
|
||||
use Kiri;
|
||||
use Kiri\Server\Events\OnWorkerStop;
|
||||
use Psr\Log\LoggerInterface;
|
||||
use Server\Events\OnWorkerStop;
|
||||
use ReflectionException;
|
||||
|
||||
|
||||
/**
|
||||
@@ -26,12 +28,6 @@ class Logger implements LoggerInterface
|
||||
const DEBUG = 'debug';
|
||||
|
||||
|
||||
/**
|
||||
* @var EventProvider
|
||||
*/
|
||||
#[Inject(EventProvider::class)]
|
||||
public EventProvider $eventProvider;
|
||||
|
||||
private array $_loggers = [];
|
||||
|
||||
|
||||
@@ -39,11 +35,12 @@ class Logger implements LoggerInterface
|
||||
|
||||
|
||||
/**
|
||||
* 监听事件
|
||||
* @return void
|
||||
* @throws ReflectionException
|
||||
*/
|
||||
public function init()
|
||||
{
|
||||
$this->eventProvider->on(OnWorkerStop::class, [$this, 'onAfterRequest']);
|
||||
Kiri::getDi()->get(EventProvider::class)->on(OnWorkerStop::class, [$this, 'onAfterRequest']);
|
||||
}
|
||||
|
||||
|
||||
@@ -158,7 +155,7 @@ class Logger implements LoggerInterface
|
||||
{
|
||||
// TODO: Implement log() method.
|
||||
$levels = Config::get('log.level', Logger::LOGGER_LEVELS);
|
||||
if (!in_array($level, $levels)) {
|
||||
if (!in_array($level, $levels) || str_contains($message, 'Event::rshutdown')) {
|
||||
return;
|
||||
}
|
||||
|
||||
@@ -179,13 +176,44 @@ class Logger implements LoggerInterface
|
||||
$loggers = implode(PHP_EOL, $this->_loggers);
|
||||
$this->_loggers = [];
|
||||
if (!empty($loggers)) {
|
||||
$filename = storage('log-' . date('Y-m-d') . '.log', 'logs/');
|
||||
$filename = storage('log-' . date('Y-m-d') . '.log', 'log/');
|
||||
|
||||
file_put_contents($filename, $loggers);
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* @return void
|
||||
* @throws Exception
|
||||
*/
|
||||
public function flush()
|
||||
{
|
||||
$this->removeFile(storage());
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* @param string $dirname
|
||||
* @return void
|
||||
*/
|
||||
private function removeFile(string $dirname)
|
||||
{
|
||||
$paths = new DirectoryIterator($dirname);
|
||||
/** @var DirectoryIterator $path */
|
||||
foreach ($paths as $path) {
|
||||
if ($path->isDot() || str_starts_with($path->getFilename(), '.')) {
|
||||
continue;
|
||||
}
|
||||
if ($path->isDir()) {
|
||||
$directory = rtrim($path->getRealPath(), '/');
|
||||
$this->removeFile($directory);
|
||||
}
|
||||
@unlink($path->getRealPath());
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* @param $message
|
||||
* @param $context
|
||||
@@ -194,7 +222,7 @@ class Logger implements LoggerInterface
|
||||
private function _string($message, $context): string
|
||||
{
|
||||
if (!empty($context)) {
|
||||
return $message . ' ' . print_r($context, true) . PHP_EOL;
|
||||
return $message . ' ' . PHP_EOL . print_r($context, true) . PHP_EOL;
|
||||
}
|
||||
return $message . PHP_EOL;
|
||||
}
|
||||
|
||||
@@ -4,14 +4,11 @@
|
||||
namespace Kiri\Abstracts;
|
||||
|
||||
|
||||
use Annotation\Annotation as SAnnotation;
|
||||
use Kiri\Annotation\Annotation as SAnnotation;
|
||||
use Database\Connection;
|
||||
use Database\DatabasesProviders;
|
||||
use Http\Handler\Client\Client;
|
||||
use Http\Handler\Client\Curl;
|
||||
use Http\Handler\Router;
|
||||
use Server\Server;
|
||||
use Kiri\Crontab\Producer;
|
||||
use Kiri\Message\Handler\Router;
|
||||
use Kiri\Server\Server;
|
||||
use Kiri\Async;
|
||||
use Kiri\Error\Logger;
|
||||
use Kiri\Jwt\JWTAuth;
|
||||
@@ -27,9 +24,7 @@ use Kiri\Jwt\JWTAuth;
|
||||
* @property JWTAuth $jwt
|
||||
* @property SAnnotation $annotation
|
||||
* @property BaseGoto $goto
|
||||
* @property Client $client
|
||||
* @property Connection $databases
|
||||
* @property Curl $curl
|
||||
*/
|
||||
trait TraitApplication
|
||||
{
|
||||
|
||||
+23
-19
@@ -13,24 +13,23 @@ namespace Kiri;
|
||||
use Closure;
|
||||
use Database\DatabasesProviders;
|
||||
use Exception;
|
||||
use Kiri\Abstracts\BaseApplication;
|
||||
use Kiri\Abstracts\Config;
|
||||
use Kiri\Abstracts\Kernel;
|
||||
use Kiri;
|
||||
use Kiri\Abstracts\{BaseApplication, Config, Kernel};
|
||||
use Kiri\Crontab\CrontabProviders;
|
||||
use Kiri\Events\OnAfterCommandExecute;
|
||||
use Kiri\Events\OnBeforeCommandExecute;
|
||||
use Kiri\Exception\NotFindClassException;
|
||||
use Kiri\Events\{OnAfterCommandExecute, OnBeforeCommandExecute};
|
||||
use Kiri\FileListen\HotReload;
|
||||
use Kiri\Server\ServerProviders;
|
||||
use ReflectionException;
|
||||
use Server\ServerProviders;
|
||||
use stdClass;
|
||||
use Swoole\Process;
|
||||
use Swoole\Timer;
|
||||
use Symfony\Component\Console\Application as ConsoleApplication;
|
||||
use Symfony\Component\Console\Command\Command;
|
||||
use Symfony\Component\Console\Input\ArgvInput;
|
||||
use Symfony\Component\Console\Input\InputInterface;
|
||||
use Symfony\Component\Console\Output\ConsoleOutput;
|
||||
use Symfony\Component\Console\{Application as ConsoleApplication,
|
||||
Command\Command,
|
||||
Input\ArgvInput,
|
||||
Output\ConsoleOutput,
|
||||
Output\OutputInterface
|
||||
};
|
||||
|
||||
|
||||
/**
|
||||
* Class Init
|
||||
@@ -192,7 +191,8 @@ class Application extends BaseApplication
|
||||
*/
|
||||
public function execute(array $argv): void
|
||||
{
|
||||
/** @var InputInterface $input */
|
||||
ini_set('swoole.enable_preemptive_scheduler', 'On');
|
||||
ini_set('swoole.enable_library', 'On');
|
||||
[$input, $output] = $this->argument($argv);
|
||||
try {
|
||||
$console = di(ConsoleApplication::class);
|
||||
@@ -212,7 +212,6 @@ class Application extends BaseApplication
|
||||
}
|
||||
|
||||
|
||||
|
||||
/**
|
||||
* @param $argv
|
||||
* @return array
|
||||
@@ -223,9 +222,7 @@ class Application extends BaseApplication
|
||||
}
|
||||
|
||||
|
||||
|
||||
/**
|
||||
* @throws NotFindClassException
|
||||
* @throws ReflectionException
|
||||
* @throws Exception
|
||||
*/
|
||||
@@ -233,16 +230,23 @@ class Application extends BaseApplication
|
||||
{
|
||||
fire(new OnBeforeCommandExecute());
|
||||
if (!($class instanceof HotReload)) {
|
||||
scan_directory(directory('app'), 'App');
|
||||
$config = Config::get('scanner', []);
|
||||
if (!empty($config)) {
|
||||
foreach ($config as $key => $value) {
|
||||
scan_directory($value, $key);
|
||||
}
|
||||
}
|
||||
scan_directory(MODEL_PATH, 'app\Model');
|
||||
}
|
||||
|
||||
$this->getContainer()->setBindings(OutputInterface::class, $output);
|
||||
|
||||
$class->run($input, $output);
|
||||
fire(new OnAfterCommandExecute());
|
||||
$output->writeln('ok' . PHP_EOL);
|
||||
}
|
||||
|
||||
|
||||
|
||||
|
||||
/**
|
||||
* @param $className
|
||||
* @param null $abstracts
|
||||
|
||||
@@ -6,8 +6,9 @@ namespace Kiri;
|
||||
|
||||
use Exception;
|
||||
use Kiri\Abstracts\Component;
|
||||
use Server\ServerManager;
|
||||
|
||||
use Kiri\Server\ServerManager;
|
||||
use Kiri\Task\AsyncTaskExecute;
|
||||
use Kiri;
|
||||
/**
|
||||
* Class Async
|
||||
* @package Kiri
|
||||
@@ -36,8 +37,8 @@ class Async extends Component
|
||||
*/
|
||||
public function dispatch(string $name, array $params = [])
|
||||
{
|
||||
$context = di(ServerManager::class);
|
||||
$context->task(static::$_absences[$name], $params);
|
||||
$context = di(AsyncTaskExecute::class);
|
||||
$context->execute(static::$_absences[$name], $params);
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
@@ -2,12 +2,14 @@
|
||||
|
||||
namespace Kiri\Cache\Base;
|
||||
|
||||
use Exception;
|
||||
use Kiri\Abstracts\Logger;
|
||||
use Kiri\Events\EventProvider;
|
||||
use Kiri\Exception\RedisConnectException;
|
||||
use Kiri\Kiri;
|
||||
use Kiri;
|
||||
use Kiri\Pool\StopHeartbeatCheck;
|
||||
use Kiri\Server\Events\OnWorkerExit;
|
||||
use RedisException;
|
||||
use Kiri\Context;
|
||||
use Swoole\Timer;
|
||||
|
||||
|
||||
@@ -22,28 +24,41 @@ class Redis implements StopHeartbeatCheck
|
||||
|
||||
private ?\Redis $pdo = null;
|
||||
|
||||
public string $host;
|
||||
|
||||
private int $_transaction = 0;
|
||||
public int $port;
|
||||
|
||||
public int $database = 0;
|
||||
|
||||
public string $auth = '';
|
||||
|
||||
public string $prefix = '';
|
||||
|
||||
public int $timeout = 30;
|
||||
|
||||
public int $read_timeout = 30;
|
||||
|
||||
public array $pool = [];
|
||||
|
||||
private int $_timer = -1;
|
||||
|
||||
private int $_last = 0;
|
||||
|
||||
|
||||
|
||||
/**
|
||||
* @param string $host
|
||||
* @param int $port
|
||||
* @param int $database
|
||||
* @param string $auth
|
||||
* @param string $prefix
|
||||
* @param int $timeout
|
||||
* @param int $read_timeout
|
||||
* @param array $config
|
||||
*/
|
||||
public function __construct(public string $host, public int $port, public int $database = 0,
|
||||
public string $auth = '', public string $prefix = '', public int $timeout = 30,
|
||||
public int $read_timeout = 30)
|
||||
public function __construct(array $config)
|
||||
{
|
||||
$this->host = $config['host'];
|
||||
$this->port = $config['port'];
|
||||
$this->database = $config['databases'];
|
||||
$this->auth = $config['auth'];
|
||||
$this->prefix = $config['prefix'];
|
||||
$this->timeout = $config['timeout'];
|
||||
$this->read_timeout = $config['read_timeout'];
|
||||
$this->pool = $config['pool'];
|
||||
}
|
||||
|
||||
|
||||
@@ -53,29 +68,47 @@ class Redis implements StopHeartbeatCheck
|
||||
}
|
||||
|
||||
|
||||
|
||||
|
||||
/**
|
||||
* @param Kiri\Server\Events\OnWorkerExit $exit
|
||||
* @return void
|
||||
*/
|
||||
public function onWorkerExit(OnWorkerExit $exit)
|
||||
{
|
||||
$this->stopHeartbeatCheck();
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
*
|
||||
*/
|
||||
public function heartbeat_check(): void
|
||||
{
|
||||
if (env('state', 'start') == 'exit') {
|
||||
return;
|
||||
|
||||
if ($this->_timer === -1) {
|
||||
$this->_timer = Timer::tick(1000, fn() => $this->waite());
|
||||
}
|
||||
if ($this->_timer === -1 && Context::inCoroutine()) {
|
||||
$this->_timer = Timer::tick(1000, function () {
|
||||
try {
|
||||
if (env('state', 'start') == 'exit') {
|
||||
Kiri::getDi()->get(Logger::class)->critical('timer end');
|
||||
$this->stopHeartbeatCheck();
|
||||
}
|
||||
if (time() - $this->_last > 10 * 60) {
|
||||
$this->stopHeartbeatCheck();
|
||||
$this->pdo = null;
|
||||
}
|
||||
} catch (\Throwable $throwable) {
|
||||
error($throwable);
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* @throws Exception
|
||||
*/
|
||||
private function waite(): void
|
||||
{
|
||||
try {
|
||||
if ($this->_timer === -1) {
|
||||
Kiri::getDi()->get(Logger::class)->critical('timer end');
|
||||
$this->stopHeartbeatCheck();
|
||||
}
|
||||
if (time() - $this->_last > intval($this->pool['tick'] ?? 60)) {
|
||||
$this->stopHeartbeatCheck();
|
||||
|
||||
$this->pdo = null;
|
||||
}
|
||||
} catch (\Throwable $throwable) {
|
||||
error($throwable);
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
@@ -16,7 +16,7 @@ use Swoole\Coroutine\System;
|
||||
|
||||
/**
|
||||
* Class File
|
||||
* @package Kiri\Kiri\Cache
|
||||
* @package Kiri\Cache
|
||||
*/
|
||||
class File extends Component implements ICache
|
||||
{
|
||||
|
||||
@@ -11,7 +11,7 @@ namespace Kiri\Cache;
|
||||
|
||||
/**
|
||||
* Interface ICache
|
||||
* @package Kiri\Kiri\Cache
|
||||
* @package Kiri\Cache
|
||||
*/
|
||||
interface ICache
|
||||
{
|
||||
|
||||
+45
-14
@@ -9,45 +9,57 @@ declare(strict_types=1);
|
||||
|
||||
namespace Kiri\Cache;
|
||||
|
||||
use Annotation\Inject;
|
||||
use Exception;
|
||||
use Server\Events\OnWorkerExit;
|
||||
use Kiri\Abstracts\Component;
|
||||
use Kiri\Abstracts\Config;
|
||||
use Kiri\Core\Json;
|
||||
use Kiri\Events\EventProvider;
|
||||
use Kiri\Exception\ConfigException;
|
||||
use Kiri\Kiri;
|
||||
use Kiri;
|
||||
use Kiri\Pool\Redis as PoolRedis;
|
||||
use Kiri\Annotation\Inject;
|
||||
use Kiri\Server\Events\OnWorkerExit;
|
||||
use Psr\Container\ContainerExceptionInterface;
|
||||
use Psr\Container\NotFoundExceptionInterface;
|
||||
use Swoole\Timer;
|
||||
|
||||
/**
|
||||
* Class Redis
|
||||
* @package Kiri\Kiri\Cache
|
||||
* @package Kiri\Cache
|
||||
* @mixin \Redis
|
||||
*/
|
||||
class Redis extends Component
|
||||
{
|
||||
|
||||
/**
|
||||
* @var EventProvider
|
||||
*/
|
||||
#[Inject(EventProvider::class)]
|
||||
public EventProvider $eventProvider;
|
||||
|
||||
const REDIS_OPTION_HOST = 'host';
|
||||
const REDIS_OPTION_PORT = 'port';
|
||||
const REDIS_OPTION_PREFIX = 'prefix';
|
||||
const REDIS_OPTION_AUTH = 'auth';
|
||||
const REDIS_OPTION_DATABASES = 'databases';
|
||||
const REDIS_OPTION_TIMEOUT = 'timeout';
|
||||
const REDIS_OPTION_POOL = 'pool';
|
||||
const REDIS_OPTION_POOL_TICK = 'tick';
|
||||
const REDIS_OPTION_POOL_MIN = 'min';
|
||||
const REDIS_OPTION_POOL_MAX = 'max';
|
||||
|
||||
|
||||
/**
|
||||
* @return void
|
||||
* @throws ConfigException
|
||||
* @throws ContainerExceptionInterface
|
||||
* @throws NotFoundExceptionInterface
|
||||
* @throws Exception
|
||||
*/
|
||||
public function init()
|
||||
{
|
||||
$connections = Kiri::getDi()->get(PoolRedis::class);
|
||||
$connections = Kiri::getDi()->get(PoolRedis::class);
|
||||
|
||||
$config = $this->get_config();
|
||||
|
||||
$length = Config::get('connections.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);
|
||||
}
|
||||
@@ -74,6 +86,25 @@ class Redis extends Component
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* @param $key
|
||||
* @param int $timeout
|
||||
* @return bool
|
||||
*/
|
||||
public function waite($key, int $timeout = 5): bool
|
||||
{
|
||||
$time = time();
|
||||
while (!$this->setNx($key, 1)) {
|
||||
if (time()- $time >= $timeout) {
|
||||
return FALSE;
|
||||
}
|
||||
usleep(1000);
|
||||
}
|
||||
$this->expire($key, $timeout);
|
||||
return TRUE;
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* @param $key
|
||||
* @param int $timeout
|
||||
@@ -111,7 +142,7 @@ SCRIPT;
|
||||
*/
|
||||
public function release()
|
||||
{
|
||||
$connections = Kiri::getDi()->get(PoolRedis::class);
|
||||
$connections = Kiri::getDi()->get(PoolRedis::class);
|
||||
$connections->release($this->get_config(), true);
|
||||
}
|
||||
|
||||
@@ -135,7 +166,7 @@ SCRIPT;
|
||||
*/
|
||||
public function proxy($name, $arguments): mixed
|
||||
{
|
||||
$connections = Kiri::getDi()->get(PoolRedis::class);
|
||||
$connections = Kiri::getDi()->get(PoolRedis::class);
|
||||
|
||||
$config = $this->get_config();
|
||||
|
||||
|
||||
+29
-9
@@ -5,7 +5,7 @@ namespace Kiri;
|
||||
|
||||
use Kiri\Abstracts\BaseContext;
|
||||
use Swoole\Coroutine;
|
||||
|
||||
use Kiri;
|
||||
/**
|
||||
* Class Context
|
||||
* @package Yoc\http
|
||||
@@ -24,10 +24,13 @@ class Context extends BaseContext
|
||||
*/
|
||||
public static function setContext($id, $context, $coroutineId = null): mixed
|
||||
{
|
||||
if (Coroutine::getCid() === -1) {
|
||||
return static::$_contents[$id] = $context;
|
||||
}
|
||||
return Coroutine::getContext($coroutineId)[$id] = $context;
|
||||
if (is_null($coroutineId)) {
|
||||
$coroutineId = Coroutine::getCid();
|
||||
}
|
||||
if (Coroutine::getCid() !== -1) {
|
||||
return Coroutine::getContext($coroutineId)[$id] = $context;
|
||||
}
|
||||
return static::$_contents[$id] = $context;
|
||||
}
|
||||
|
||||
/**
|
||||
@@ -38,6 +41,9 @@ class Context extends BaseContext
|
||||
*/
|
||||
public static function increment($id, int $value = 1, $coroutineId = null): bool|int
|
||||
{
|
||||
if (is_null($coroutineId)) {
|
||||
$coroutineId = Coroutine::getCid();
|
||||
}
|
||||
if (!isset(Coroutine::getContext($coroutineId)[$id])) {
|
||||
Coroutine::getContext($coroutineId)[$id] = 0;
|
||||
}
|
||||
@@ -52,6 +58,9 @@ class Context extends BaseContext
|
||||
*/
|
||||
public static function decrement($id, int $value = 1, $coroutineId = null): bool|int
|
||||
{
|
||||
if (is_null($coroutineId)) {
|
||||
$coroutineId = Coroutine::getCid();
|
||||
}
|
||||
if (!isset(Coroutine::getContext($coroutineId)[$id])) {
|
||||
Coroutine::getContext($coroutineId)[$id] = 0;
|
||||
}
|
||||
@@ -81,6 +90,9 @@ class Context extends BaseContext
|
||||
*/
|
||||
private static function loadByContext($id, $default = null, $coroutineId = null): mixed
|
||||
{
|
||||
if (is_null($coroutineId)) {
|
||||
$coroutineId = Coroutine::getCid();
|
||||
}
|
||||
return Coroutine::getContext($coroutineId)[$id] ?? $default;
|
||||
}
|
||||
|
||||
@@ -115,6 +127,9 @@ class Context extends BaseContext
|
||||
*/
|
||||
public static function remove(string $id, $coroutineId = null)
|
||||
{
|
||||
if (is_null($coroutineId)) {
|
||||
$coroutineId = Coroutine::getCid();
|
||||
}
|
||||
if (!static::hasContext($id, $coroutineId)) {
|
||||
return;
|
||||
}
|
||||
@@ -150,8 +165,9 @@ class Context extends BaseContext
|
||||
if (!isset(static::$_contents[$id])) {
|
||||
return false;
|
||||
}
|
||||
if (!empty($key) && !isset(static::$_contents[$id][$key])) {
|
||||
return false;
|
||||
$value = static::$_contents[$id];
|
||||
if (!empty($key) && is_array($value)) {
|
||||
return isset($value[$key]);
|
||||
}
|
||||
return true;
|
||||
}
|
||||
@@ -165,11 +181,15 @@ class Context extends BaseContext
|
||||
*/
|
||||
private static function searchByCoroutine($id, $key = null, $coroutineId = null): bool
|
||||
{
|
||||
if (is_null($coroutineId)) {
|
||||
$coroutineId = Coroutine::getCid();
|
||||
}
|
||||
if (!isset(Coroutine::getContext($coroutineId)[$id])) {
|
||||
return false;
|
||||
}
|
||||
if ($key !== null) {
|
||||
return isset((Coroutine::getContext($coroutineId)[$id] ?? [])[$key]);
|
||||
$value = Coroutine::getContext($coroutineId)[$id];
|
||||
if ($key !== null && is_array($value)) {
|
||||
return isset($value[$key]);
|
||||
}
|
||||
return true;
|
||||
}
|
||||
|
||||
@@ -13,7 +13,7 @@ namespace Kiri\Core;
|
||||
|
||||
/**
|
||||
* Class DateFormat
|
||||
* @package Kiri\Kiri\Core
|
||||
* @package Kiri\Core
|
||||
*/
|
||||
class DateFormat
|
||||
{
|
||||
@@ -27,7 +27,7 @@ class DateFormat
|
||||
if ($time === null) {
|
||||
$time = time();
|
||||
} else if (is_numeric($time)) {
|
||||
$length = strlen(floatval($time));
|
||||
$length = strlen((string)$time);
|
||||
if ($length != 10 && $length != 13) {
|
||||
return false;
|
||||
}
|
||||
@@ -86,9 +86,7 @@ class DateFormat
|
||||
return false;
|
||||
}
|
||||
|
||||
$time = date('t', $time);
|
||||
|
||||
return $time;
|
||||
return date('t', $time);
|
||||
}
|
||||
|
||||
/**
|
||||
|
||||
@@ -21,6 +21,7 @@ class Dtl extends Component
|
||||
/**
|
||||
* Dtl constructor.
|
||||
* @param $params
|
||||
* @throws Exception
|
||||
*/
|
||||
public function __construct($params)
|
||||
{
|
||||
@@ -35,9 +36,6 @@ class Dtl extends Component
|
||||
*/
|
||||
public function toArray(): array
|
||||
{
|
||||
if (!is_array($this->params)) {
|
||||
return ArrayAccess::toArray($this->params);
|
||||
}
|
||||
return $this->params;
|
||||
}
|
||||
|
||||
|
||||
@@ -0,0 +1,111 @@
|
||||
<?php
|
||||
|
||||
namespace Kiri\Core;
|
||||
|
||||
use JetBrains\PhpStorm\Pure;
|
||||
use ReturnTypeWillChange;
|
||||
use Traversable;
|
||||
|
||||
class HashMap implements \ArrayAccess, \IteratorAggregate
|
||||
{
|
||||
|
||||
/**
|
||||
* @var array
|
||||
*/
|
||||
private array $lists = [];
|
||||
|
||||
|
||||
/**
|
||||
* @return Traversable
|
||||
*/
|
||||
public function getIterator(): Traversable
|
||||
{
|
||||
return new \ArrayIterator($this->lists);
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* @param string $key
|
||||
* @param $value
|
||||
*/
|
||||
public function put(string $key, $value)
|
||||
{
|
||||
$this->lists[$key] = $value;
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* @param string $key
|
||||
* @return mixed
|
||||
*/
|
||||
#[Pure] public function get(string $key): mixed
|
||||
{
|
||||
if (!$this->has($key)) {
|
||||
return null;
|
||||
}
|
||||
return $this->lists[$key];
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* @param string $key
|
||||
*/
|
||||
public function del(string $key)
|
||||
{
|
||||
if (!$this->has($key)) {
|
||||
return;
|
||||
}
|
||||
unset($this->lists[$key]);
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* @param string $key
|
||||
* @return bool
|
||||
*/
|
||||
public function has(string $key): bool
|
||||
{
|
||||
return array_key_exists($key, $this->lists);
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* @param mixed $offset
|
||||
* @return bool
|
||||
*/
|
||||
public function offsetExists(mixed $offset): bool
|
||||
{
|
||||
return isset($this->lists[$offset]);
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* @param mixed $offset
|
||||
* @return mixed
|
||||
*/
|
||||
#[Pure] public function offsetGet(mixed $offset): mixed
|
||||
{
|
||||
return $this->get($offset);
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* @param mixed $offset
|
||||
* @param mixed $value
|
||||
*/
|
||||
#[ReturnTypeWillChange]
|
||||
public function offsetSet(mixed $offset, mixed $value)
|
||||
{
|
||||
$this->put($offset, $value);
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* @param mixed $offset
|
||||
*/
|
||||
#[ReturnTypeWillChange]
|
||||
public function offsetUnset(mixed $offset)
|
||||
{
|
||||
unset($this->lists[$offset]);
|
||||
}
|
||||
}
|
||||
@@ -10,7 +10,7 @@ use Exception;
|
||||
|
||||
/**
|
||||
* Class Help
|
||||
* @package Kiri\Kiri\Core
|
||||
* @package Kiri\Core
|
||||
*/
|
||||
class Help
|
||||
{
|
||||
@@ -59,6 +59,7 @@ class Help
|
||||
/**
|
||||
* @param $xml
|
||||
* @return mixed
|
||||
* @throws Exception
|
||||
*/
|
||||
public static function toArray($xml): mixed
|
||||
{
|
||||
@@ -200,6 +201,9 @@ class Help
|
||||
*/
|
||||
public static function sendEmail($email, string $Subject, $messageContent)
|
||||
{
|
||||
if (!class_exists('\Swift_Mailer')) {
|
||||
return;
|
||||
}
|
||||
$mailer = new \Swift_Mailer((new \Swift_SmtpTransport($email['host'], $email['port']))
|
||||
->setUsername($email['username'])->setPassword($email['password']));
|
||||
$message = (new \Swift_Message($Subject))
|
||||
|
||||
+11
-12
@@ -16,7 +16,7 @@ use Throwable;
|
||||
|
||||
/**
|
||||
* Class JSON
|
||||
* @package Kiri\Kiri\Core
|
||||
* @package Kiri\Core
|
||||
*/
|
||||
class Json
|
||||
{
|
||||
@@ -43,7 +43,7 @@ class Json
|
||||
* @param bool $asArray
|
||||
* @return mixed
|
||||
*/
|
||||
public static function decode($data, $asArray = true): mixed
|
||||
public static function decode($data, bool $asArray = true): mixed
|
||||
{
|
||||
if (is_array($data) || is_numeric($data)) {
|
||||
return $data;
|
||||
@@ -55,22 +55,21 @@ class Json
|
||||
|
||||
/**
|
||||
* @param $code
|
||||
* @param string $message
|
||||
* @param array $data
|
||||
* @param string|array $message
|
||||
* @param array|int $data
|
||||
* @param int $count
|
||||
* @param array $exPageInfo
|
||||
* @return mixed
|
||||
* @throws
|
||||
* @return string|bool
|
||||
*/
|
||||
public static function to($code, $message = '', $data = [], $count = 0, $exPageInfo = []): mixed
|
||||
public static function to($code, string|array $message = '', array|int $data = [], int $count = 0, array $exPageInfo = []): string|bool
|
||||
{
|
||||
$params['code'] = $code;
|
||||
if (!is_string($message)) {
|
||||
$params['param'] = $message;
|
||||
if (!empty($data)) {
|
||||
$params['exPageInfo'] = $data;
|
||||
}
|
||||
$params['message'] = 'System success.';
|
||||
$params['message'] = 'System success.';
|
||||
$params['param'] = $message;
|
||||
if (!empty($data)) {
|
||||
$params['exPageInfo'] = $data;
|
||||
}
|
||||
} else {
|
||||
$params['message'] = $message;
|
||||
$params['param'] = $data;
|
||||
|
||||
@@ -0,0 +1,18 @@
|
||||
<?php
|
||||
|
||||
namespace Kiri\Core;
|
||||
|
||||
class Network
|
||||
{
|
||||
|
||||
|
||||
/**
|
||||
* @return string
|
||||
*/
|
||||
public static function local(): string
|
||||
{
|
||||
return current(swoole_get_local_ip());
|
||||
}
|
||||
|
||||
|
||||
}
|
||||
@@ -6,7 +6,7 @@ namespace Kiri\Core;
|
||||
|
||||
/**
|
||||
* Class Reader
|
||||
* @package Kiri\Kiri\Core
|
||||
* @package Kiri\Core
|
||||
*/
|
||||
class Reader
|
||||
{
|
||||
|
||||
@@ -9,7 +9,7 @@ use Exception;
|
||||
|
||||
/**
|
||||
* Class Str
|
||||
* @package Kiri\Kiri\Core
|
||||
* @package Kiri\Core
|
||||
*/
|
||||
class Str
|
||||
{
|
||||
|
||||
@@ -13,7 +13,7 @@ use Exception;
|
||||
|
||||
/**
|
||||
* Class Xml
|
||||
* @package Kiri\Kiri\Core
|
||||
* @package Kiri\Core
|
||||
*/
|
||||
class Xml
|
||||
{
|
||||
|
||||
@@ -1,461 +0,0 @@
|
||||
<?php
|
||||
/**
|
||||
* Created by PhpStorm.
|
||||
* User: whwyy
|
||||
* Date: 2018/4/24 0024
|
||||
* Time: 17:27
|
||||
*/
|
||||
declare(strict_types=1);
|
||||
|
||||
namespace Kiri\Di;
|
||||
|
||||
use Annotation\Inject;
|
||||
use Closure;
|
||||
use Exception;
|
||||
use Kiri\Abstracts\BaseObject;
|
||||
use Kiri\Abstracts\Logger;
|
||||
use Kiri\Kiri;
|
||||
use Psr\Log\LoggerInterface;
|
||||
use ReflectionClass;
|
||||
use ReflectionException;
|
||||
use ReflectionFunction;
|
||||
use ReflectionMethod;
|
||||
use ReflectionProperty;
|
||||
use Psr\Container\ContainerInterface;
|
||||
|
||||
/**
|
||||
* Class Container
|
||||
* @package Kiri\Di
|
||||
*/
|
||||
class Container extends BaseObject 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 ReflectionException
|
||||
*/
|
||||
public function get(string $id): mixed
|
||||
{
|
||||
return $this->make($id, [], []);
|
||||
}
|
||||
|
||||
|
||||
|
||||
/**
|
||||
* @param $class
|
||||
* @param array $constrict
|
||||
* @param array $config
|
||||
* @return mixed
|
||||
* @throws ReflectionException
|
||||
* @throws Exception
|
||||
*/
|
||||
public function make($class, array $constrict = [], array $config = []): mixed
|
||||
{
|
||||
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
|
||||
* @throws ReflectionException
|
||||
*/
|
||||
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::getPropertyNote($reflect) as $property => $inject) {
|
||||
/** @var Inject $inject */
|
||||
$inject->execute($object, $property);
|
||||
}
|
||||
return $object;
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* @param $className
|
||||
* @param $method
|
||||
* @return array
|
||||
* @throws ReflectionException
|
||||
*/
|
||||
public function getMethodAttribute($className, $method = null): array
|
||||
{
|
||||
$methods = NoteManager::getMethodNote($this->getReflect($className));
|
||||
if (!empty($method)) {
|
||||
return $methods[$method] ?? [];
|
||||
}
|
||||
return $methods;
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* @param string $class
|
||||
* @param string|null $property
|
||||
* @return ReflectionProperty|ReflectionProperty[]|null
|
||||
* @throws ReflectionException
|
||||
*/
|
||||
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
|
||||
* @throws ReflectionException
|
||||
*/
|
||||
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] = [];
|
||||
}
|
||||
if (!isset($this->_parameters[$class][$method])) {
|
||||
$this->_parameters[$class][$method] = [];
|
||||
}
|
||||
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
|
||||
* @throws ReflectionException
|
||||
*/
|
||||
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
|
||||
* @throws ReflectionException
|
||||
*/
|
||||
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\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,289 +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 $_classMethodNote = [];
|
||||
private static array $_classMethod = [];
|
||||
private static array $_classPropertyNote = [];
|
||||
private static array $_classProperty = [];
|
||||
private static array $_mapping = [];
|
||||
|
||||
|
||||
/**
|
||||
* @param ReflectionClass $class
|
||||
*/
|
||||
public static function setTargetNote(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 getTargetNote(mixed $class): array
|
||||
{
|
||||
if (!is_string($class)) {
|
||||
$class = $class::class;
|
||||
}
|
||||
return static::$_classTarget[$class] ?? [];
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* @param ReflectionClass $class
|
||||
*/
|
||||
public static function setMethodNote(ReflectionClass $class)
|
||||
{
|
||||
$className = $class->getName();
|
||||
static::$_classMethodNote[$className] = static::$_classMethod[$className] = [];
|
||||
foreach ($class->getMethods() as $ReflectionMethod) {
|
||||
static::$_classMethod[$className][$ReflectionMethod->getName()] = $ReflectionMethod;
|
||||
static::$_classMethodNote[$className][$ReflectionMethod->getName()] = [];
|
||||
foreach ($ReflectionMethod->getAttributes() as $attribute) {
|
||||
if (!class_exists($attribute->getName())) {
|
||||
continue;
|
||||
}
|
||||
$instance = $attribute->newInstance();
|
||||
|
||||
static::$_classMethodNote[$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 getMethodNote(ReflectionClass $class): array
|
||||
{
|
||||
return static::$_classMethodNote[$class->getName()] ?? [];
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* @param \ReflectionClass $reflect
|
||||
* @return \ReflectionMethod|null
|
||||
*/
|
||||
public static function resolveTarget(ReflectionClass $reflect): ?\ReflectionMethod
|
||||
{
|
||||
NoteManager::setPropertyNote($reflect);
|
||||
NoteManager::setTargetNote($reflect);
|
||||
NoteManager::setMethodNote($reflect);
|
||||
|
||||
return $reflect->getConstructor();
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* @param ReflectionClass $class
|
||||
*/
|
||||
public static function setPropertyNote(ReflectionClass $class)
|
||||
{
|
||||
$className = $class->getName();
|
||||
static::$_classProperty[$className] = static::$_classPropertyNote[$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::$_classPropertyNote[$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 getPropertyNote(ReflectionClass $class): array
|
||||
{
|
||||
return static::$_classPropertyNote[$class->getName()] ?? [];
|
||||
}
|
||||
|
||||
|
||||
}
|
||||
@@ -5,7 +5,7 @@ namespace Kiri;
|
||||
|
||||
|
||||
use JetBrains\PhpStorm\Pure;
|
||||
|
||||
use Kiri;
|
||||
|
||||
/**
|
||||
* Class Environmental
|
||||
|
||||
@@ -10,17 +10,17 @@ declare(strict_types=1);
|
||||
namespace Kiri\Error;
|
||||
|
||||
use Exception;
|
||||
use Http\Handler\Formatter\IFormatter;
|
||||
use Kiri\Message\Handler\Formatter\IFormatter;
|
||||
use Kiri\Abstracts\Component;
|
||||
use Kiri\Core\Json;
|
||||
use Kiri\Events\EventDispatch;
|
||||
use Kiri\Kiri;
|
||||
use Http\Events\OnAfterRequest;
|
||||
use Kiri;
|
||||
use Kiri\Message\Events\OnAfterRequest;
|
||||
|
||||
/**
|
||||
* Class ErrorHandler
|
||||
*
|
||||
* @package Kiri\Kiri\Base
|
||||
* @package Kiri\Base
|
||||
* @property-read $asError
|
||||
*/
|
||||
class ErrorHandler extends Component implements ErrorInterface
|
||||
@@ -36,13 +36,8 @@ class ErrorHandler extends Component implements ErrorInterface
|
||||
*/
|
||||
public function register()
|
||||
{
|
||||
// ini_set('display_errors', '1');
|
||||
set_exception_handler([$this, 'exceptionHandler']);
|
||||
if (defined('HHVM_VERSION')) {
|
||||
set_error_handler([$this, 'errorHandler']);
|
||||
} else {
|
||||
set_error_handler([$this, 'errorHandler']);
|
||||
}
|
||||
set_error_handler([$this, 'errorHandler']);
|
||||
register_shutdown_function([$this, 'shutdown']);
|
||||
}
|
||||
|
||||
@@ -118,8 +113,6 @@ class ErrorHandler extends Component implements ErrorInterface
|
||||
{
|
||||
$path = ['file' => $file, 'line' => $line];
|
||||
|
||||
var_dump(func_get_args());
|
||||
|
||||
$data = Json::to($code, $this->category . ': ' . $message, $path);
|
||||
|
||||
write($data, $this->category);
|
||||
|
||||
@@ -11,7 +11,7 @@ namespace Kiri\Error;
|
||||
|
||||
/**
|
||||
* Interface ErrorInterface
|
||||
* @package Kiri\Kiri\Error
|
||||
* @package Kiri\Error
|
||||
*/
|
||||
interface ErrorInterface
|
||||
{
|
||||
|
||||
@@ -9,18 +9,17 @@ declare(strict_types=1);
|
||||
|
||||
namespace Kiri\Error;
|
||||
|
||||
use Annotation\Inject;
|
||||
use Exception;
|
||||
use Kiri\Abstracts\Component;
|
||||
use Kiri\Core\Json;
|
||||
use Kiri\Events\EventProvider;
|
||||
use Kiri\Kiri;
|
||||
use Kiri;
|
||||
use Kiri\Annotation\Inject;
|
||||
use Psr\Log\LoggerInterface;
|
||||
use Throwable;
|
||||
|
||||
/**
|
||||
* Class Logger
|
||||
* @package Kiri\Kiri\Error
|
||||
* @package Kiri\Error
|
||||
* @mixin \Kiri\Abstracts\Logger
|
||||
*/
|
||||
class Logger extends Component
|
||||
@@ -29,11 +28,6 @@ class Logger extends Component
|
||||
private array $logs = [];
|
||||
|
||||
|
||||
/** @var EventProvider */
|
||||
#[Inject(EventProvider::class)]
|
||||
public EventProvider $eventProvider;
|
||||
|
||||
|
||||
/**
|
||||
* inject logger
|
||||
*
|
||||
@@ -52,9 +46,21 @@ class Logger extends Component
|
||||
*/
|
||||
public function getLastError(string $application = 'app'): string
|
||||
{
|
||||
return 'Unknown error.';
|
||||
return $this->logs[$application] ?? 'Unknown error.';
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* @param $message
|
||||
* @param $method
|
||||
* @return void
|
||||
*/
|
||||
public function fail($message, $method)
|
||||
{
|
||||
$this->logs[$method] = $message;
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* @param string $messages
|
||||
* @param string $method
|
||||
|
||||
@@ -5,10 +5,11 @@ namespace Kiri\Error;
|
||||
|
||||
|
||||
use Exception;
|
||||
use Http\Aspect\OnAspectInterface;
|
||||
use Http\Aspect\OnJoinPointInterface;
|
||||
use Http\Constrict\RequestInterface;
|
||||
use Kiri\Kiri;
|
||||
use Kiri\Message\Aspect\OnAspectInterface;
|
||||
use Kiri\Message\Aspect\OnJoinPointInterface;
|
||||
use Kiri\Message\Constrict\RequestInterface;
|
||||
use Kiri;
|
||||
use Psr\Log\LoggerInterface;
|
||||
|
||||
|
||||
/**
|
||||
@@ -43,10 +44,10 @@ class LoggerAspect implements OnAspectInterface
|
||||
private function print_runtime($startTime)
|
||||
{
|
||||
$request = Kiri::getDi()->get(RequestInterface::class);
|
||||
|
||||
$runTime = round(microtime(true) - $startTime, 6);
|
||||
echo sprintf('run %s use time %6f', $request->getUri()->__toString(), $runTime);
|
||||
echo PHP_EOL;
|
||||
|
||||
$logger = Kiri::getDi()->get(LoggerInterface::class);
|
||||
$logger->debug(sprintf('run %s use time %6f', $request->getUri()->__toString(), $runTime));
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
@@ -5,13 +5,14 @@ namespace Kiri\Error;
|
||||
|
||||
|
||||
use Exception;
|
||||
use JetBrains\PhpStorm\Pure;
|
||||
use Kiri\Core\Json;
|
||||
use Kiri\Exception\ComponentException;
|
||||
use Kiri\Kiri;
|
||||
use Kiri;
|
||||
use Kiri\Server\Abstracts\BaseProcess;
|
||||
use Kiri\Server\Broadcast\OnBroadcastInterface;
|
||||
use Psr\Log\LoggerInterface;
|
||||
use Swoole\Coroutine;
|
||||
use Swoole\Process;
|
||||
use Server\Abstracts\BaseProcess;
|
||||
|
||||
/**
|
||||
* Class LoggerProcess
|
||||
@@ -24,7 +25,6 @@ class LoggerProcess extends BaseProcess
|
||||
public string $name = 'logger process';
|
||||
|
||||
|
||||
|
||||
/**
|
||||
* @param Process $process
|
||||
* @throws ComponentException
|
||||
@@ -36,6 +36,17 @@ class LoggerProcess extends BaseProcess
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* @param OnBroadcastInterface $message
|
||||
* @return void
|
||||
*/
|
||||
public function onBroadcast(OnBroadcastInterface $message): void
|
||||
{
|
||||
$logger = Kiri::getDi()->get(LoggerInterface::class);
|
||||
$logger->debug($message->data . '::' . static::class);
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* @param Process $process
|
||||
* @throws ComponentException
|
||||
@@ -43,6 +54,9 @@ class LoggerProcess extends BaseProcess
|
||||
*/
|
||||
public function message(Process $process)
|
||||
{
|
||||
if ($this->isStop()) {
|
||||
return;
|
||||
}
|
||||
$message = Json::decode($process->read());
|
||||
if (!empty($message)) {
|
||||
Kiri::writeFile($this->getDirName($message), $message[0], FILE_APPEND);
|
||||
|
||||
@@ -6,14 +6,13 @@ namespace Kiri;
|
||||
|
||||
|
||||
use Exception;
|
||||
use Kiri\Abstracts\BaseObject;
|
||||
use Swoole\Coroutine;
|
||||
|
||||
use Kiri\Abstracts\Component;
|
||||
use Kiri;
|
||||
/**
|
||||
* Class Event
|
||||
* @package Kiri
|
||||
*/
|
||||
class Event extends BaseObject
|
||||
class Event extends Component
|
||||
{
|
||||
|
||||
public bool $isVide = true;
|
||||
|
||||
@@ -15,7 +15,7 @@ use Throwable;
|
||||
|
||||
/**
|
||||
* Class ComponentException
|
||||
* @package Kiri\Kiri\Exception
|
||||
* @package Kiri\Exception
|
||||
*/
|
||||
class ComponentException extends \Exception
|
||||
{
|
||||
|
||||
@@ -16,7 +16,7 @@ use Throwable;
|
||||
|
||||
/**
|
||||
* Class NotFindClassException
|
||||
* @package Kiri\Kiri\Exception
|
||||
* @package Kiri\Exception
|
||||
*/
|
||||
class NotFindClassException extends \Exception
|
||||
{
|
||||
|
||||
@@ -15,7 +15,7 @@ use Throwable;
|
||||
|
||||
/**
|
||||
* Class NotFindClassException
|
||||
* @package Kiri\Kiri\Exception
|
||||
* @package Kiri\Exception
|
||||
*/
|
||||
class NotFindPropertyException extends \Exception
|
||||
{
|
||||
|
||||
@@ -2,14 +2,16 @@
|
||||
|
||||
namespace Kiri\FileListen;
|
||||
|
||||
use Annotation\Inject;
|
||||
use Exception;
|
||||
use Kiri;
|
||||
use Kiri\Abstracts\Config;
|
||||
use Kiri\Annotation\Inject;
|
||||
use Kiri\Core\Json;
|
||||
use Kiri\Error\Logger;
|
||||
use Kiri\Exception\ConfigException;
|
||||
use Kiri\Kiri;
|
||||
use Swoole\Coroutine;
|
||||
use Swoole\Process;
|
||||
use Swoole\Timer;
|
||||
use Symfony\Component\Console\Command\Command;
|
||||
use Symfony\Component\Console\Input\InputInterface;
|
||||
use Symfony\Component\Console\Output\OutputInterface;
|
||||
@@ -22,8 +24,8 @@ class HotReload extends Command
|
||||
{
|
||||
|
||||
|
||||
public bool $isReloading = false;
|
||||
public bool $isReloadingOut = false;
|
||||
public bool $isReloading = FALSE;
|
||||
public bool $isReloadingOut = FALSE;
|
||||
public ?array $dirs = [];
|
||||
|
||||
public int $events;
|
||||
@@ -31,7 +33,7 @@ class HotReload extends Command
|
||||
public int $int = -1;
|
||||
|
||||
|
||||
private ?Process $process = null;
|
||||
private ?Process $process = NULL;
|
||||
|
||||
|
||||
public Inotify|Scaner $driver;
|
||||
@@ -41,20 +43,81 @@ class HotReload extends Command
|
||||
public Logger $logger;
|
||||
|
||||
|
||||
protected mixed $source = null;
|
||||
protected mixed $source = NULL;
|
||||
|
||||
protected mixed $pipes = [];
|
||||
|
||||
protected ?Coroutine\Channel $channel = null;
|
||||
protected ?Coroutine\Channel $channel = NULL;
|
||||
|
||||
|
||||
/**
|
||||
*
|
||||
*/
|
||||
protected function configure()
|
||||
{
|
||||
$this->setName('sw:wather')
|
||||
->setDescription('server start');
|
||||
$this->setName('sw:wather')->setDescription('server start');
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* @throws ConfigException
|
||||
* @throws Exception
|
||||
*/
|
||||
protected function initCore()
|
||||
{
|
||||
set_error_handler([$this, 'errorHandler']);
|
||||
$this->dirs = Config::get('inotify', [APP_PATH . 'app']);
|
||||
if (!extension_loaded('inotify')) {
|
||||
$this->driver = Kiri::getDi()->make(Scaner::class, [$this->dirs, $this]);
|
||||
} else {
|
||||
$this->driver = Kiri::getDi()->make(Inotify::class, [$this->dirs, $this]);
|
||||
}
|
||||
$this->clearOtherService();
|
||||
$this->setProcessName();
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* @throws ConfigException
|
||||
*/
|
||||
public function setProcessName()
|
||||
{
|
||||
swoole_async_set(['enable_coroutine' => FALSE]);
|
||||
if (Kiri::getPlatform()->isLinux()) {
|
||||
swoole_set_process_name('[' . Config::get('id', 'sw service.') . '].sw:wather');
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* @throws Exception
|
||||
*/
|
||||
public function clearOtherService()
|
||||
{
|
||||
if (file_exists(storage('.manager.pid'))) {
|
||||
$pid = (int)file_get_contents(storage('.manager.pid'));
|
||||
if ($pid > 0 && Process::kill($pid, 0)) {
|
||||
Process::kill($pid, 15) && Process::wait(TRUE);
|
||||
}
|
||||
}
|
||||
file_put_contents(storage('.manager.pid'), getmypid());
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* @throws Exception
|
||||
*/
|
||||
public function errorHandler()
|
||||
{
|
||||
$error = func_get_args();
|
||||
|
||||
$path = ['file' => $error[2], 'line' => $error[3]];
|
||||
|
||||
if ($error[0] === 0) {
|
||||
$error[0] = 500;
|
||||
}
|
||||
$data = Json::to(500, $error[1], $path);
|
||||
|
||||
$this->logger->error($data, 'error');
|
||||
}
|
||||
|
||||
|
||||
@@ -67,21 +130,11 @@ class HotReload extends Command
|
||||
*/
|
||||
public function execute(InputInterface $input, OutputInterface $output): int
|
||||
{
|
||||
// TODO: Implement onHandler() method.
|
||||
set_error_handler([$this, 'onErrorHandler']);
|
||||
$this->dirs = Config::get('inotify', [APP_PATH . 'app']);
|
||||
swoole_async_set(['enable_coroutine' => false]);
|
||||
if (!extension_loaded('inotify')) {
|
||||
$this->driver = Kiri::getDi()->make(Scaner::class, [$this->dirs, $this]);
|
||||
} else {
|
||||
$this->driver = Kiri::getDi()->make(Inotify::class, [$this->dirs, $this]);
|
||||
}
|
||||
if (Kiri::getPlatform()->isLinux()) {
|
||||
swoole_set_process_name('[' . Config::get('id', 'sw service.') . '].sw:wather');
|
||||
}
|
||||
$this->trigger_reload();
|
||||
$this->initCore();
|
||||
|
||||
$this->trigger_reload();
|
||||
Timer::tick(1000, fn() => $this->healthCheck());
|
||||
|
||||
var_dump(getmypid());
|
||||
Process::signal(SIGTERM, [$this, 'onSignal']);
|
||||
Process::signal(SIGKILL, [$this, 'onSignal']);
|
||||
|
||||
@@ -90,6 +143,25 @@ class HotReload extends Command
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* @throws Exception
|
||||
*/
|
||||
public function healthCheck()
|
||||
{
|
||||
$pid = (int)file_get_contents(storage('.swoole.pid'));
|
||||
if ($this->int == 1) {
|
||||
return;
|
||||
}
|
||||
if (empty($pid)) {
|
||||
$this->logger->warning('service is shutdown you need reload.');
|
||||
$this->trigger_reload();
|
||||
} else if (!Process::kill($pid, 0)) {
|
||||
$this->logger->warning('service is shutdown you need reload.');
|
||||
$this->trigger_reload();
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* @param $data
|
||||
* @throws Exception
|
||||
@@ -99,15 +171,10 @@ class HotReload extends Command
|
||||
if (!$data) {
|
||||
return;
|
||||
}
|
||||
Timer::clearAll();
|
||||
$this->driver->clear();
|
||||
$pid = file_get_contents(storage('.swoole.pid'));
|
||||
if (!empty($pid) && Process::kill($pid, 0)) {
|
||||
Process::kill($pid, SIGTERM);
|
||||
}
|
||||
if ($this->process && Process::kill($this->process->pid, 0)) {
|
||||
Process::kill($this->process->pid) && Process::wait(true);
|
||||
}
|
||||
while ($ret = Process::wait(true)) {
|
||||
$this->stopServer();
|
||||
while ($ret = Process::wait(TRUE)) {
|
||||
echo "PID={$ret['pid']}\n";
|
||||
sleep(1);
|
||||
}
|
||||
@@ -115,18 +182,17 @@ class HotReload extends Command
|
||||
|
||||
|
||||
/**
|
||||
* @param $code
|
||||
* @param $message
|
||||
* @param $file
|
||||
* @param $line
|
||||
* @throws Exception
|
||||
*/
|
||||
public function onErrorHandler($code, $message, $file, $line)
|
||||
protected function stopServer()
|
||||
{
|
||||
if (str_contains($message, 'The file descriptor is not an inotify instance')) {
|
||||
return;
|
||||
$pid = file_get_contents(storage('.swoole.pid'));
|
||||
if (!empty($pid) && Process::kill($pid, 0)) {
|
||||
Process::kill($pid, SIGTERM);
|
||||
}
|
||||
if ($this->process && Process::kill($this->process->pid, 0)) {
|
||||
Process::kill($this->process->pid) && Process::wait(TRUE);
|
||||
}
|
||||
debug('Error:' . $message . ' at ' . $file . ':' . $line);
|
||||
}
|
||||
|
||||
|
||||
@@ -135,19 +201,23 @@ class HotReload extends Command
|
||||
*
|
||||
* @throws Exception
|
||||
*/
|
||||
public function trigger_reload()
|
||||
public function trigger_reload(string $path = '')
|
||||
{
|
||||
$this->logger->warning('change reload');
|
||||
$pid = $this->process?->pid;
|
||||
$process = new Process(function (Process $process) {
|
||||
$process->exec(PHP_BINARY, [APP_PATH . "kiri.php", "sw:server", "restart"]);
|
||||
});
|
||||
$process->start();
|
||||
if ($pid && Process::kill($pid, 0)) {
|
||||
Process::kill($pid) && Process::wait(true);
|
||||
if (!empty($path) && str_starts_with($path, CONTROLLER_PATH)) {
|
||||
$pid = file_get_contents(storage('.swoole.pid'));
|
||||
if (!empty($pid) && Process::kill($pid, 0)) {
|
||||
Process::kill($pid, SIGUSR1);
|
||||
}
|
||||
} else {
|
||||
$this->int = 1;
|
||||
$this->stopServer();
|
||||
$this->process = new Process(function (Process $process) {
|
||||
$process->exec(PHP_BINARY, [APP_PATH . "kiri.php", "sw:server", "start"]);
|
||||
});
|
||||
$this->process->start();
|
||||
$this->int = -1;
|
||||
}
|
||||
$this->process = null;
|
||||
$this->process = $process;
|
||||
}
|
||||
|
||||
|
||||
|
||||
@@ -15,7 +15,7 @@ class Inotify
|
||||
private array $watchFiles = [];
|
||||
|
||||
|
||||
protected bool $isReloading = FALSE;
|
||||
public bool $isReloading = FALSE;
|
||||
|
||||
|
||||
protected int $cid;
|
||||
@@ -29,6 +29,17 @@ class Inotify
|
||||
*/
|
||||
public function __construct(protected array $dirs, public HotReload $process)
|
||||
{
|
||||
set_error_handler([$this, 'error']);
|
||||
set_exception_handler([$this, 'error']);
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* @return void
|
||||
*/
|
||||
public function error(): void
|
||||
{
|
||||
|
||||
}
|
||||
|
||||
|
||||
@@ -43,6 +54,7 @@ class Inotify
|
||||
if (!is_dir($dir)) continue;
|
||||
$this->watch($dir);
|
||||
}
|
||||
$this->process->int = -1;
|
||||
Event::add($this->inotify, [$this, 'check']);
|
||||
Event::wait();
|
||||
}
|
||||
@@ -73,10 +85,14 @@ class Inotify
|
||||
if (!in_array($ev['mask'], $LISTEN_TYPE)) {
|
||||
continue;
|
||||
}
|
||||
|
||||
$search = array_search($ev['wd'], $this->watchFiles);
|
||||
|
||||
//非重启类型
|
||||
if (str_ends_with($ev['name'], '.php')) {
|
||||
Timer::after(3000, fn()=>$this->reload());
|
||||
$this->isReloading = TRUE;
|
||||
|
||||
Timer::after(3000, fn() => $this->reload($search));
|
||||
$this->isReloading = TRUE;
|
||||
}
|
||||
}
|
||||
}
|
||||
@@ -84,15 +100,16 @@ class Inotify
|
||||
/**
|
||||
* @throws Exception
|
||||
*/
|
||||
public function reload()
|
||||
public function reload($path)
|
||||
{
|
||||
$this->process->trigger_reload();
|
||||
$this->process->trigger_reload($path);
|
||||
$this->process->int = -1;
|
||||
|
||||
$this->clearWatch();
|
||||
foreach ($this->dirs as $root) {
|
||||
$this->watch($root);
|
||||
}
|
||||
$this->process->int = -1;
|
||||
$this->isReloading = FALSE;
|
||||
$this->isReloading = FALSE;
|
||||
}
|
||||
|
||||
|
||||
@@ -102,11 +119,7 @@ class Inotify
|
||||
public function clearWatch()
|
||||
{
|
||||
foreach ($this->watchFiles as $wd) {
|
||||
try {
|
||||
inotify_rm_watch($this->inotify, $wd);
|
||||
} catch (\Throwable $exception) {
|
||||
logger()->addError($exception, 'throwable');
|
||||
}
|
||||
@inotify_rm_watch($this->inotify, $wd);
|
||||
}
|
||||
$this->watchFiles = [];
|
||||
}
|
||||
|
||||
@@ -9,6 +9,9 @@ class Scaner
|
||||
|
||||
private array $md5Map = [];
|
||||
|
||||
public bool $isReloading = FALSE;
|
||||
|
||||
|
||||
/**
|
||||
* @param array $dirs
|
||||
* @param HotReload $process
|
||||
@@ -32,7 +35,7 @@ class Scaner
|
||||
* @param bool $isReload
|
||||
* @throws Exception
|
||||
*/
|
||||
private function loadDirs(bool $isReload = false)
|
||||
private function loadDirs(bool $isReload = FALSE)
|
||||
{
|
||||
foreach ($this->dirs as $value) {
|
||||
if (is_bool($path = realpath($value))) {
|
||||
@@ -52,7 +55,7 @@ class Scaner
|
||||
* @return void
|
||||
* @throws Exception
|
||||
*/
|
||||
private function loadByDir($path, bool $isReload = false): void
|
||||
private function loadByDir($path, bool $isReload = FALSE): void
|
||||
{
|
||||
if (!is_string($path)) {
|
||||
return;
|
||||
@@ -64,7 +67,11 @@ class Scaner
|
||||
}
|
||||
if (is_file($value)) {
|
||||
if ($this->checkFile($value, $isReload)) {
|
||||
$this->timerReload();
|
||||
$this->isReloading = TRUE;
|
||||
|
||||
sleep(2);
|
||||
|
||||
$this->timerReload($value);
|
||||
break;
|
||||
}
|
||||
}
|
||||
@@ -83,48 +90,46 @@ class Scaner
|
||||
$mTime = filectime($value);
|
||||
if (!isset($this->md5Map[$md5])) {
|
||||
if ($isReload) {
|
||||
return true;
|
||||
return TRUE;
|
||||
}
|
||||
$this->md5Map[$md5] = $mTime;
|
||||
} else {
|
||||
if ($this->md5Map[$md5] != $mTime) {
|
||||
if ($isReload) {
|
||||
return true;
|
||||
return TRUE;
|
||||
}
|
||||
$this->md5Map[$md5] = $mTime;
|
||||
}
|
||||
}
|
||||
return false;
|
||||
return FALSE;
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* @throws Exception
|
||||
*/
|
||||
public function timerReload()
|
||||
public function timerReload($path)
|
||||
{
|
||||
if ($this->process->isReloading) {
|
||||
return;
|
||||
}
|
||||
$this->process->isReloading = true;
|
||||
$this->process->trigger_reload();
|
||||
$this->isReloading = TRUE;
|
||||
|
||||
$this->process->int = -1;
|
||||
$this->process->trigger_reload($path);
|
||||
|
||||
$this->loadDirs();
|
||||
|
||||
$this->process->isReloading = FALSE;
|
||||
$this->process->int = -1;
|
||||
|
||||
$this->isReloading = FALSE;
|
||||
$this->process->isReloadingOut = FALSE;
|
||||
|
||||
$this->tick();
|
||||
}
|
||||
|
||||
|
||||
private bool $isStop = false;
|
||||
private bool $isStop = FALSE;
|
||||
|
||||
public function clear()
|
||||
{
|
||||
$this->isStop = true;
|
||||
$this->isStop = TRUE;
|
||||
}
|
||||
|
||||
|
||||
@@ -133,11 +138,11 @@ class Scaner
|
||||
*/
|
||||
public function tick()
|
||||
{
|
||||
if ($this->process->isReloading || $this->isStop) {
|
||||
if ($this->isReloading || $this->isStop) {
|
||||
return;
|
||||
}
|
||||
|
||||
$this->loadDirs(true);
|
||||
$this->loadDirs(TRUE);
|
||||
|
||||
sleep(2);
|
||||
|
||||
|
||||
@@ -6,10 +6,10 @@ namespace Kiri\Pool;
|
||||
use Closure;
|
||||
use Database\Mysql\PDO;
|
||||
use Exception;
|
||||
use Kiri;
|
||||
use Kiri\Abstracts\Component;
|
||||
use Kiri\Abstracts\Config;
|
||||
use Kiri\Context;
|
||||
use Kiri\Kiri;
|
||||
use Swoole\Error;
|
||||
use Throwable;
|
||||
|
||||
@@ -113,13 +113,24 @@ class Connection extends Component
|
||||
public function create($coroutineName, $config): Closure
|
||||
{
|
||||
return static function () use ($coroutineName, $config) {
|
||||
return Kiri::getDi()->create(PDO::class, [
|
||||
$config['database'], $config['cds'], $config['username'], $config['password'], $config['charset'] ?? 'utf8mb4'
|
||||
]);
|
||||
return Kiri::getDi()->create(PDO::class, [$config]);
|
||||
};
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* @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 $isMaster
|
||||
@@ -128,21 +139,24 @@ class Connection extends Component
|
||||
*/
|
||||
public function initConnections($name, $isMaster, $max)
|
||||
{
|
||||
$this->getPool()->initConnections($name, $isMaster, $max);
|
||||
$pool = $this->getPool();
|
||||
$pool->initConnections($name, $isMaster, $max);
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* @param $coroutineName
|
||||
* @param $isMaster
|
||||
* @param array $config
|
||||
* @throws Kiri\Exception\ConfigException
|
||||
* @throws Exception
|
||||
*/
|
||||
public function release($coroutineName, $isMaster)
|
||||
public function release($coroutineName, $isMaster, array $config)
|
||||
{
|
||||
$coroutineName = $this->name('Mysql:' . $coroutineName, $isMaster);
|
||||
/** @var PDO $client */
|
||||
if (!($client = Context::getContext($coroutineName)) instanceof PDO) {
|
||||
return;
|
||||
$client = call_user_func($this->create($coroutineName, $config));
|
||||
}
|
||||
if ($client->inTransaction()) {
|
||||
return;
|
||||
@@ -186,7 +200,7 @@ class Connection extends Component
|
||||
} else {
|
||||
$result = true;
|
||||
}
|
||||
} catch (Error | Throwable $exception) {
|
||||
} catch (Error|Throwable $exception) {
|
||||
$result = $this->addError($exception, 'mysql');
|
||||
} finally {
|
||||
return $result;
|
||||
|
||||
@@ -41,11 +41,10 @@ class Pool extends Component
|
||||
|
||||
|
||||
/**
|
||||
* @param Channel $channel
|
||||
* @param Channel|SplQueue $channel
|
||||
* @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) {
|
||||
if (Context::inCoroutine()) {
|
||||
|
||||
@@ -10,11 +10,11 @@ use Exception;
|
||||
use Kiri\Abstracts\Component;
|
||||
use Kiri\Context;
|
||||
use Kiri\Exception\ConfigException;
|
||||
use Kiri\Kiri;
|
||||
use Kiri;
|
||||
|
||||
/**
|
||||
* Class RedisClient
|
||||
* @package Kiri\Kiri\Pool
|
||||
* @package Kiri\Pool
|
||||
*/
|
||||
class Redis extends Component
|
||||
{
|
||||
@@ -50,11 +50,7 @@ class Redis extends Component
|
||||
public function create(string $name, mixed $config): Closure
|
||||
{
|
||||
return static function () use ($name, $config) {
|
||||
return Kiri::getDi()->create(\Kiri\Cache\Base\Redis::class, [
|
||||
$config['host'], (int)$config['port'], $config['databases'] ?? 0,
|
||||
$config['auth'], $config['prefix'] ?? '', $config['timeout'] ?? 30,
|
||||
$config['read_timeout'] ?? 30
|
||||
]);
|
||||
return Kiri::getDi()->create(\Kiri\Cache\Base\Redis::class, [$config]);
|
||||
};
|
||||
}
|
||||
|
||||
|
||||
@@ -5,11 +5,12 @@ namespace Kiri;
|
||||
|
||||
|
||||
use Exception;
|
||||
use Kiri\Abstracts\Config;
|
||||
use Kiri\Abstracts\Input;
|
||||
use Symfony\Component\Console\Command\Command;
|
||||
use Symfony\Component\Console\Input\InputInterface;
|
||||
use Symfony\Component\Console\Output\OutputInterface;
|
||||
|
||||
use Kiri;
|
||||
|
||||
/**
|
||||
* Class Runtime
|
||||
@@ -62,8 +63,7 @@ class Runtime extends Command
|
||||
public function configEach(): string
|
||||
{
|
||||
$array = [];
|
||||
$configs = Kiri::app()->getConfig();
|
||||
foreach ($configs->getData() as $key => $datum) {
|
||||
foreach (Config::getData() as $key => $datum) {
|
||||
if ($datum instanceof \Closure) {
|
||||
continue;
|
||||
}
|
||||
|
||||
@@ -0,0 +1,14 @@
|
||||
<?php
|
||||
|
||||
namespace Kiri\Gateway;
|
||||
|
||||
class Collector
|
||||
{
|
||||
|
||||
|
||||
public function get()
|
||||
{
|
||||
}
|
||||
|
||||
|
||||
}
|
||||
@@ -0,0 +1,23 @@
|
||||
<?php
|
||||
|
||||
namespace Kiri\Gateway;
|
||||
|
||||
use Swoole\Http\Request;
|
||||
use Swoole\Http\Response;
|
||||
|
||||
class GatewayServer
|
||||
{
|
||||
|
||||
|
||||
/**
|
||||
* @param Request $request
|
||||
* @param Response $response
|
||||
*/
|
||||
public function onRequest(Request $request, Response $response)
|
||||
{
|
||||
|
||||
|
||||
}
|
||||
|
||||
|
||||
}
|
||||
@@ -0,0 +1,35 @@
|
||||
<?php
|
||||
|
||||
namespace Kiri\Gateway;
|
||||
|
||||
class HashMap
|
||||
{
|
||||
|
||||
|
||||
const HTTP = 1;
|
||||
const TCP = 2;
|
||||
const UDP = 2;
|
||||
|
||||
|
||||
public string $domain;
|
||||
|
||||
|
||||
public string $path;
|
||||
|
||||
|
||||
public string $scheme;
|
||||
|
||||
|
||||
public string $method;
|
||||
|
||||
|
||||
public string $proxy_host;
|
||||
|
||||
|
||||
public string $proxy_port;
|
||||
|
||||
|
||||
public int $type = self::HTTP;
|
||||
|
||||
|
||||
}
|
||||
+5
-6
@@ -13,7 +13,7 @@ use Database\Connection;
|
||||
use Database\Db;
|
||||
use Exception;
|
||||
use Kiri\Cache\Redis;
|
||||
use Kiri\Kiri;
|
||||
use Kiri;
|
||||
use Symfony\Component\Console\Input\InputInterface;
|
||||
|
||||
/**
|
||||
@@ -68,7 +68,7 @@ class Gii
|
||||
$this->input = $input;
|
||||
$this->db = $db;
|
||||
|
||||
$make = $this->input->getArgument('action');
|
||||
$make = $this->input->getOption('make');
|
||||
if (empty($make)) {
|
||||
throw new Exception('构建类型不能为空~');
|
||||
}
|
||||
@@ -119,10 +119,8 @@ class Gii
|
||||
*/
|
||||
private function makeByDatabases($make, InputInterface $input): array
|
||||
{
|
||||
$redis = Kiri::getDi()->get(Redis::class);
|
||||
if ($input->hasArgument('name')) {
|
||||
$this->tableName = $input->getArgument('name');
|
||||
$redis->del('column:' . $this->tableName);
|
||||
if ($input->hasOption('name')) {
|
||||
$this->tableName = $input->getOption('name');
|
||||
}
|
||||
return match ($make) {
|
||||
'controller' => $this->getTable(1, 0),
|
||||
@@ -334,6 +332,7 @@ class Gii
|
||||
private function getClassName($tableName): string
|
||||
{
|
||||
$res = [];
|
||||
$tableName = str_replace($this->db->tablePrefix,'', $tableName);
|
||||
foreach (explode('_', $tableName) as $n => $val) {
|
||||
$res[] = ucfirst($val);
|
||||
}
|
||||
|
||||
@@ -7,7 +7,7 @@ namespace Gii;
|
||||
use Exception;
|
||||
use Kiri\Abstracts\Config;
|
||||
use Kiri\Exception\ConfigException;
|
||||
use Kiri\Kiri;
|
||||
use Kiri;
|
||||
use Symfony\Component\Console\Command\Command;
|
||||
use Symfony\Component\Console\Input\InputArgument;
|
||||
use Symfony\Component\Console\Input\InputInterface;
|
||||
@@ -32,9 +32,9 @@ class GiiCommand extends Command
|
||||
protected function configure()
|
||||
{
|
||||
$this->setName('sw:gii')
|
||||
->addArgument('action', InputArgument::REQUIRED)
|
||||
->addArgument('name', InputArgument::OPTIONAL)
|
||||
->addArgument('databases', InputArgument::OPTIONAL)
|
||||
->addOption('make','m', InputArgument::OPTIONAL)
|
||||
->addOption('name','t', InputArgument::OPTIONAL)
|
||||
->addOption('databases','d', InputArgument::OPTIONAL)
|
||||
->setDescription('./snowflake sw:gii make=model|controller|task|interceptor|limits|middleware name=xxxx');
|
||||
}
|
||||
|
||||
@@ -52,12 +52,12 @@ class GiiCommand extends Command
|
||||
$gii = Kiri::app()->get('gii');
|
||||
|
||||
$connections = Kiri::app()->get('db');
|
||||
if (($db = $input->getArgument('databases')) != null) {
|
||||
if (($db = $input->getOption('databases')) != null) {
|
||||
$gii->run($connections->get($db), $input);
|
||||
return 1;
|
||||
}
|
||||
|
||||
$action = $input->getArgument('action');
|
||||
$action = $input->getOption('make');
|
||||
if (!in_array($action, ['model', 'controller'])) {
|
||||
$gii->run(null, $input);
|
||||
return 1;
|
||||
|
||||
@@ -6,7 +6,7 @@ namespace Gii;
|
||||
|
||||
use Exception;
|
||||
use ReflectionException;
|
||||
use Kiri\Kiri;
|
||||
use Kiri;
|
||||
|
||||
/**
|
||||
* Class GiiController
|
||||
@@ -47,9 +47,6 @@ class GiiController extends GiiBase
|
||||
$namespace = rtrim($path['namespace'], '\\');
|
||||
$model_namespace = rtrim($modelPath['namespace'], '\\');
|
||||
|
||||
$prefix = str_replace('_', '', $this->db->tablePrefix);
|
||||
$managerName = str_replace(ucfirst($prefix), '', $managerName);
|
||||
|
||||
$class = '';
|
||||
$controller = str_replace('\\\\', '\\', "$namespace\\{$managerName}Controller");
|
||||
|
||||
@@ -68,14 +65,14 @@ namespace {$namespace};
|
||||
} else {
|
||||
$import = "use Kiri;
|
||||
use Exception;
|
||||
use Annotation\Target;
|
||||
use Annotation\Route\Middleware;
|
||||
use Annotation\Route\Route;
|
||||
use Kiri\Annotation\Target;
|
||||
use Kiri\Annotation\Route\Middleware;
|
||||
use Kiri\Annotation\Route\Route;
|
||||
use Kiri\Core\Str;
|
||||
use Kiri\Core\Json;
|
||||
use Http\Context\Request;
|
||||
use Http\Context\Response;
|
||||
use Http\Controller;
|
||||
use Kiri\Message\Context\Request;
|
||||
use Kiri\Message\Context\Response;
|
||||
use Kiri\Message\Controller;
|
||||
use JetBrains\PhpStorm\ArrayShape;
|
||||
use {$model_namespace}\\{$managerName};
|
||||
";
|
||||
|
||||
@@ -51,8 +51,8 @@ interface ' . ucfirst($name) . 'RpcInterface
|
||||
namespace Rpc\Producers;
|
||||
|
||||
|
||||
use Annotation\Target;
|
||||
use Annotation\Mapping;
|
||||
use Kiri\Annotation\Target;
|
||||
use Kiri\Annotation\Mapping;
|
||||
use Rpc\\' . ucfirst($name) . 'RpcInterface;
|
||||
use Exception;
|
||||
use Kiri\Rpc\JsonRpcConsumers;
|
||||
@@ -86,9 +86,9 @@ class ' . ucfirst($name) . 'RpcService extends JsonRpcConsumers implements ' . u
|
||||
namespace Rpc\Consumers;
|
||||
|
||||
|
||||
use Annotation\Target;
|
||||
use Kiri\Annotation\Target;
|
||||
use Kiri\Rpc\Annotation\JsonRpc;
|
||||
use Http\Handler\Controller;
|
||||
use Kiri\Message\Handler\Controller;
|
||||
use Rpc\\' . ucfirst($name) . 'RpcInterface;
|
||||
|
||||
|
||||
|
||||
@@ -6,7 +6,7 @@ namespace Gii;
|
||||
|
||||
|
||||
use Exception;
|
||||
use Kiri\Kiri;
|
||||
use Kiri;
|
||||
|
||||
/**
|
||||
* Class GiiMiddleware
|
||||
@@ -34,7 +34,7 @@ namespace App\Middleware;
|
||||
|
||||
use Closure;
|
||||
use Psr\Http\Server\MiddlewareInterface;
|
||||
use Server\Constrict\RequestInterface;
|
||||
use Kiri\Server\Constrict\RequestInterface;
|
||||
|
||||
';
|
||||
|
||||
|
||||
+12
-12
@@ -8,7 +8,7 @@ use Database\Db;
|
||||
use Database\Model;
|
||||
use Exception;
|
||||
use ReflectionException;
|
||||
use Kiri\Kiri;
|
||||
use Kiri;
|
||||
|
||||
/**
|
||||
* Class GiiModel
|
||||
@@ -51,9 +51,6 @@ class GiiModel extends GiiBase
|
||||
|
||||
$namespace = rtrim($modelPath['namespace'], '\\');
|
||||
|
||||
$prefix = str_replace('_', '', $this->db->tablePrefix);
|
||||
$managerName = str_replace(ucfirst($prefix), '', $managerName);
|
||||
|
||||
if (file_exists($modelPath['path'] . '/' . $managerName . '.php')) {
|
||||
try {
|
||||
$className = str_replace('\\\\', '\\', "{$modelPath['namespace']}\\{$managerName}");
|
||||
@@ -87,7 +84,7 @@ namespace ' . $namespace . ';
|
||||
|
||||
|
||||
use Exception;
|
||||
use Annotation\Target;
|
||||
use Kiri\Annotation\Target;
|
||||
use Kiri\Core\Json;
|
||||
use Database\Connection;
|
||||
use Database\Annotation\Get;
|
||||
@@ -263,7 +260,7 @@ use Database\Model;
|
||||
$field = '\'' . current($val)['Field'] . '\'';
|
||||
}
|
||||
$_field_one .= '
|
||||
[' . $field . ', \'' . $key . '\'],';
|
||||
[' . $field . ', \'' . $key . '\'],';
|
||||
}
|
||||
foreach ($data as $key => $val) {
|
||||
$length = $this->getLength($val);
|
||||
@@ -279,8 +276,11 @@ use Database\Model;
|
||||
/**
|
||||
* @return array
|
||||
*/
|
||||
protected array $rules = [' . $_field_one . '
|
||||
];
|
||||
public function rules(): array
|
||||
{
|
||||
return [' . $_field_one . '
|
||||
];
|
||||
}
|
||||
';
|
||||
}
|
||||
|
||||
@@ -311,10 +311,10 @@ use Database\Model;
|
||||
}
|
||||
if (count($_val) == 1) {
|
||||
$_tmp = '
|
||||
[\'' . $_val[0][3] . '\', ' . ($_val[0][1] == 'enum' ? '\'enum\' => [' . $key .']' : $key) . ']';
|
||||
[\'' . $_val[0][3] . '\', ' . ($_val[0][1] == 'enum' ? '\'enum\' => [' . $key .']' : $key) . ']';
|
||||
} else {
|
||||
$_tmp = '
|
||||
[[\'' . implode('\', \'', array_column($_val, 3)) . '\'], ' . $key . ']';
|
||||
[[\'' . implode('\', \'', array_column($_val, 3)) . '\'], ' . $key . ']';
|
||||
}
|
||||
$string[] = $_tmp;
|
||||
}
|
||||
@@ -338,7 +338,7 @@ use Database\Model;
|
||||
return '';
|
||||
}
|
||||
return '
|
||||
[[\'' . implode('\', \'', $data) . '\'], \'unique\'],';
|
||||
[[\'' . implode('\', \'', $data) . '\'], \'unique\'],';
|
||||
}
|
||||
|
||||
/**
|
||||
@@ -358,7 +358,7 @@ use Database\Model;
|
||||
return '';
|
||||
}
|
||||
return '
|
||||
[[\'' . implode('\', \'', $data) . '\'], \'required\'],';
|
||||
[[\'' . implode('\', \'', $data) . '\'], \'required\'],';
|
||||
}
|
||||
|
||||
/**
|
||||
|
||||
@@ -8,7 +8,7 @@ namespace Gii;
|
||||
use Exception;
|
||||
use Kiri\Abstracts\Providers;
|
||||
use Kiri\Application;
|
||||
use Kiri\Kiri;
|
||||
use Kiri;
|
||||
|
||||
/**
|
||||
* Class DatabasesProviders
|
||||
|
||||
@@ -5,7 +5,7 @@ namespace Gii;
|
||||
|
||||
|
||||
use Exception;
|
||||
use Kiri\Kiri;
|
||||
use Kiri;
|
||||
|
||||
/**
|
||||
* Class GiiRpcClient
|
||||
@@ -36,13 +36,13 @@ class GiiRpcClient extends GiiBase
|
||||
|
||||
namespace App\Client\Rpc;
|
||||
|
||||
use Annotation\Rpc\Consumer;
|
||||
use Annotation\Rpc\RpcClient;
|
||||
use Annotation\Target;
|
||||
use Kiri\Annotation\Rpc\Consumer;
|
||||
use Kiri\Annotation\Rpc\RpcClient;
|
||||
use Kiri\Annotation\Target;
|
||||
use Exception;
|
||||
use Rpc\Client;
|
||||
use Kiri\Core\Json;
|
||||
use Kiri\Kiri;
|
||||
use Kiri;
|
||||
|
||||
';
|
||||
|
||||
|
||||
@@ -5,7 +5,7 @@ namespace Gii;
|
||||
|
||||
|
||||
use Exception;
|
||||
use Kiri\Kiri;
|
||||
use Kiri;
|
||||
|
||||
/**
|
||||
* Class GiiRpcClient
|
||||
@@ -33,10 +33,10 @@ class GiiRpcService extends GiiBase
|
||||
|
||||
namespace App\Rpc;
|
||||
|
||||
use Annotation\Route\RpcProducer;
|
||||
use Annotation\Target;
|
||||
use Kiri\Annotation\Route\RpcProducer;
|
||||
use Kiri\Annotation\Target;
|
||||
use Exception;
|
||||
use Http\Controller;
|
||||
use Kiri\Message\Controller;
|
||||
use Kiri\Core\Json;
|
||||
|
||||
';
|
||||
|
||||
@@ -5,7 +5,7 @@ declare(strict_types=1);
|
||||
namespace Gii;
|
||||
|
||||
use Exception;
|
||||
use Kiri\Kiri;
|
||||
use Kiri;
|
||||
|
||||
/**
|
||||
* Class GiiModel
|
||||
@@ -31,7 +31,7 @@ class GiiTask extends GiiBase
|
||||
|
||||
namespace App\Async;
|
||||
|
||||
use Server\Contract\OnTaskInterface;
|
||||
use Kiri\Task\OnTaskInterface;
|
||||
|
||||
';
|
||||
|
||||
|
||||
@@ -1,43 +0,0 @@
|
||||
<?php
|
||||
|
||||
|
||||
namespace Annotation;
|
||||
|
||||
|
||||
use Exception;
|
||||
use Kiri\Kiri;
|
||||
|
||||
|
||||
/**
|
||||
* Class Asynchronous
|
||||
* @package Annotation
|
||||
* Task任务
|
||||
*/
|
||||
#[\Attribute(\Attribute::TARGET_CLASS)] class Asynchronous extends Attribute
|
||||
{
|
||||
|
||||
|
||||
/**
|
||||
* Asynchronous 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
|
||||
{
|
||||
$async = Kiri::app()->getAsync();
|
||||
$async->addAsync($this->name, $class);
|
||||
return true;
|
||||
}
|
||||
|
||||
}
|
||||
@@ -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);
|
||||
}
|
||||
|
||||
}
|
||||
@@ -0,0 +1,14 @@
|
||||
<?php
|
||||
|
||||
namespace Kiri\Websocket;
|
||||
|
||||
class FdCollector
|
||||
{
|
||||
|
||||
|
||||
public function set($fd)
|
||||
{
|
||||
|
||||
}
|
||||
|
||||
}
|
||||
@@ -0,0 +1,102 @@
|
||||
<?php
|
||||
|
||||
namespace Kiri\Websocket;
|
||||
|
||||
use Kiri;
|
||||
use Swoole\{Coroutine\Http\Server as AliasServer, WebSocket\Server};
|
||||
|
||||
|
||||
/**
|
||||
*
|
||||
*/
|
||||
class Sender implements WebSocketInterface
|
||||
{
|
||||
|
||||
|
||||
/**
|
||||
* @var AliasServer|Server
|
||||
*/
|
||||
private AliasServer|Server $server;
|
||||
|
||||
|
||||
/**
|
||||
*
|
||||
*/
|
||||
public function __construct()
|
||||
{
|
||||
$this->server = Kiri::getDi()->get(WebSocketInterface::class);
|
||||
}
|
||||
|
||||
|
||||
|
||||
/**
|
||||
* @param AliasServer|Server $server
|
||||
*/
|
||||
public function setServer(mixed $server): void
|
||||
{
|
||||
$this->server = $server;
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* @param int $fd
|
||||
* @param mixed $data
|
||||
* @param int $opcode
|
||||
* @param int $flags
|
||||
* @return bool
|
||||
*/
|
||||
public function push(int $fd, string $data, int $opcode = WEBSOCKET_OPCODE_TEXT, int $flags = SWOOLE_WEBSOCKET_FLAG_FIN): bool
|
||||
{
|
||||
if ($this->isEstablished($fd)) {
|
||||
return $this->server->push($fd, $data, $opcode, $flags);
|
||||
}
|
||||
return false;
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* @param $fd
|
||||
* @param $reactor_id
|
||||
* @return array|null
|
||||
*/
|
||||
public function connection_info($fd, $reactor_id = null): ?array
|
||||
{
|
||||
return $this->server->getClientInfo($fd, $reactor_id);
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* @param int $fd
|
||||
* @param int $code
|
||||
* @param string $reason
|
||||
* @return bool
|
||||
*/
|
||||
public function disconnect(int $fd, int $code = SWOOLE_WEBSOCKET_CLOSE_NORMAL, string $reason = ''): bool
|
||||
{
|
||||
if ($this->isEstablished($fd)) {
|
||||
return $this->server->disconnect($fd, $code, $reason);
|
||||
}
|
||||
return false;
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* @param int $fd
|
||||
* @return bool
|
||||
*/
|
||||
public function isEstablished(int $fd): bool
|
||||
{
|
||||
return $this->exist($fd) && $this->server->isEstablished($fd);
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* @param int $fd
|
||||
* @return bool
|
||||
*/
|
||||
public function exist(int $fd): bool
|
||||
{
|
||||
return $this->server->exist($fd);
|
||||
}
|
||||
|
||||
}
|
||||
@@ -0,0 +1,160 @@
|
||||
<?php
|
||||
|
||||
namespace Kiri\Websocket;
|
||||
|
||||
use Exception;
|
||||
use Kiri\Abstracts\AbstractServer;
|
||||
use Kiri\Message\Handler\DataGrip;
|
||||
use Kiri\Message\Handler\RouterCollector;
|
||||
use Kiri\Server\Contract\OnCloseInterface;
|
||||
use Kiri\Server\Contract\OnHandshakeInterface;
|
||||
use Kiri\Server\Contract\OnMessageInterface;
|
||||
use Kiri\Server\Contract\OnOpenInterface;
|
||||
use Psr\Container\ContainerExceptionInterface;
|
||||
use Psr\Container\NotFoundExceptionInterface;
|
||||
use Swoole\Http\Request;
|
||||
use Swoole\Http\Response;
|
||||
use Swoole\WebSocket\Frame;
|
||||
|
||||
|
||||
/**
|
||||
* websocket server
|
||||
*/
|
||||
class Server extends AbstractServer
|
||||
{
|
||||
|
||||
public RouterCollector $router;
|
||||
|
||||
|
||||
const SHA1_KEY = '258EAFA5-E914-47DA-95CA-C5AB0DC85B11';
|
||||
|
||||
|
||||
public mixed $callback = null;
|
||||
|
||||
|
||||
public Sender $sender;
|
||||
|
||||
|
||||
/**
|
||||
* @throws ContainerExceptionInterface
|
||||
* @throws NotFoundExceptionInterface
|
||||
*/
|
||||
public function init()
|
||||
{
|
||||
$this->router = $this->getContainer()->get(DataGrip::class)->get('ws');
|
||||
$handler = $this->router->find('/', 'GET');
|
||||
if (is_int($handler) || is_null($handler)) {
|
||||
return;
|
||||
}
|
||||
$this->callback = $handler->callback[0];
|
||||
|
||||
$this->sender = $this->getContainer()->get(Sender::class);
|
||||
$this->sender->setServer($this->server);
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* @param \Swoole\Server $server
|
||||
* @param int $fd
|
||||
*/
|
||||
public function onClose(\Swoole\Server $server, int $fd): void
|
||||
{
|
||||
$clientInfo = $server->getClientInfo($fd);
|
||||
if (!isset($clientInfo['websocket_status'])) {
|
||||
return;
|
||||
}
|
||||
if ($this->callback instanceof OnCloseInterface) {
|
||||
$this->callback->onClose($fd);
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* @param Request $request
|
||||
* @param Response $response
|
||||
* @return void
|
||||
* @throws Exception
|
||||
*/
|
||||
protected function setWebSocketProtocol(Request $request, Response $response)
|
||||
{
|
||||
$secWebSocketKey = $request->header['sec-websocket-key'];
|
||||
$patten = '#^[+/0-9A-Za-z]{21}[AQgw]==$#';
|
||||
if (preg_match($patten, $secWebSocketKey) === 0 || strlen(base64_decode($secWebSocketKey)) !== 16) {
|
||||
throw new Exception('protocol error.', 500);
|
||||
}
|
||||
$key = base64_encode(sha1($request->header['sec-websocket-key'] . self::SHA1_KEY, true));
|
||||
$headers = [
|
||||
'Upgrade' => 'websocket',
|
||||
'Connection' => 'Upgrade',
|
||||
'Sec-Websocket-Accept' => $key,
|
||||
'Sec-Websocket-Version' => '13',
|
||||
];
|
||||
if (isset($request->header['sec-websocket-protocol'])) {
|
||||
$headers['Sec-Websocket-Protocol'] = $request->header['sec-websocket-protocol'];
|
||||
}
|
||||
foreach ($headers as $key => $val) {
|
||||
$response->header($key, $val);
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* @param Request $request
|
||||
* @param Response $response
|
||||
*/
|
||||
public function onHandshake(Request $request, Response $response): void
|
||||
{
|
||||
try {
|
||||
$this->setWebSocketProtocol($request, $response);
|
||||
if ($this->callback instanceof OnHandshakeInterface) {
|
||||
$this->callback->onHandshake($request, $response);
|
||||
} else {
|
||||
$response->setStatusCode(101, 'connection success.');
|
||||
$response->end();
|
||||
}
|
||||
// if ($this->server instanceof \Swoole\Coroutine\Http\Server) {
|
||||
// $response->upgrade();
|
||||
// $this->deferOpen($request);
|
||||
// while (true) {
|
||||
// $receive = $response->recv();
|
||||
// if ($receive === '' || $receive instanceof CloseFrame) {
|
||||
// $response->close();
|
||||
// if ($this->callback instanceof OnCloseInterface) {
|
||||
// $this->callback->onClose($this->server, $response->fd);
|
||||
// }
|
||||
// break;
|
||||
// }
|
||||
// $this->callback->onMessage($this->server, $receive);
|
||||
// }
|
||||
// } else {
|
||||
// $this->deferOpen($request);
|
||||
// }
|
||||
if ($response->isWritable()) {
|
||||
$this->deferOpen($request);
|
||||
}
|
||||
} catch (\Throwable $throwable) {
|
||||
$response->status(4000 + $throwable->getCode(), $throwable->getMessage());
|
||||
$response->end();
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
private function deferOpen($request)
|
||||
{
|
||||
if ($this->callback instanceof OnOpenInterface) {
|
||||
$this->callback->onOpen($request);
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* @param $server
|
||||
* @param Frame $frame
|
||||
*/
|
||||
public function onMessage($server, Frame $frame): void
|
||||
{
|
||||
if ($this->callback instanceof OnMessageInterface) {
|
||||
$this->callback->onMessage($frame);
|
||||
}
|
||||
}
|
||||
}
|
||||
@@ -0,0 +1,47 @@
|
||||
<?php
|
||||
|
||||
namespace Kiri\Websocket;
|
||||
|
||||
|
||||
/**
|
||||
* @mixin \Swoole\WebSocket\Server
|
||||
* @mixin \Swoole\Coroutine\Http\Server
|
||||
*/
|
||||
interface WebSocketInterface
|
||||
{
|
||||
|
||||
|
||||
/**
|
||||
* @param int $fd
|
||||
* @param mixed $data
|
||||
* @param int $opcode
|
||||
* @param int $flags
|
||||
* @return bool
|
||||
*/
|
||||
public function push(int $fd, string $data, int $opcode = WEBSOCKET_OPCODE_TEXT, int $flags = SWOOLE_WEBSOCKET_FLAG_FIN): bool;
|
||||
|
||||
|
||||
/**
|
||||
* @param int $fd
|
||||
* @param int $code
|
||||
* @param string $reason
|
||||
* @return mixed
|
||||
*/
|
||||
public function disconnect(int $fd, int $code = SWOOLE_WEBSOCKET_CLOSE_NORMAL, string $reason = ''): bool;
|
||||
|
||||
|
||||
/**
|
||||
* @param int $fd
|
||||
* @return bool
|
||||
*/
|
||||
public function isEstablished(int $fd): bool;
|
||||
|
||||
|
||||
/**
|
||||
* @param int $fd
|
||||
* @return bool
|
||||
*/
|
||||
public function exist(int $fd): bool;
|
||||
|
||||
|
||||
}
|
||||
@@ -49,3 +49,42 @@
|
||||
// after($process);
|
||||
//});
|
||||
|
||||
|
||||
var_dump(json_encode([
|
||||
"Datacenter" => "dc1",
|
||||
"Node" => "iz8vbi3edjyskl7kpuwudqz",
|
||||
"SkipNodeUpdate" => FALSE,
|
||||
"Service" => [
|
||||
"ID" => "redis1",
|
||||
"Service" => "FriendRpcService",
|
||||
"Address" => "172.26.221.211",
|
||||
"TaggedAddresses" => [
|
||||
"lan" => [
|
||||
"address" => "127.0.0.1",
|
||||
"port" => 9627,
|
||||
],
|
||||
"wan" => [
|
||||
"address" => "172.26.221.211",
|
||||
"port" => 9627,
|
||||
],
|
||||
],
|
||||
"Meta" => [
|
||||
"redis_version" => "4.0",
|
||||
],
|
||||
"Port" => 9627,
|
||||
],
|
||||
"Check" => [
|
||||
"Node" => "iz8vbi3edjyskl7kpuwudqz",
|
||||
"CheckId" => "service:redis1",
|
||||
"Name" => "Redis health check",
|
||||
"Notes" => "Script based health check",
|
||||
"Status" => "passing",
|
||||
"ServiceID" => "redis1",
|
||||
"Definition" => [
|
||||
"Http" => "http://172.26.221.211:9627",
|
||||
"Interval" => "5s",
|
||||
"Timeout" => "1s",
|
||||
"DeregisterCriticalServiceAfter" => "30s",
|
||||
],
|
||||
],
|
||||
]));
|
||||
|
||||
@@ -27,11 +27,21 @@ namespace Ar;
|
||||
use Swoole\Coroutine\Http\Client;
|
||||
use function Swoole\Coroutine\run;
|
||||
|
||||
run(function () {
|
||||
//run(function () {
|
||||
//
|
||||
// $client = new Client('47.92.194.207',8500);
|
||||
// $client->get('/v1/agent/services?filter=Service == FriendRpcService');
|
||||
// $client->close();
|
||||
// var_dump($client->getBody());
|
||||
//
|
||||
//});
|
||||
|
||||
$client = new Client('47.92.194.207',8500);
|
||||
$client->get('/v1/agent/services?filter=Service == FriendRpcService');
|
||||
$client->close();
|
||||
var_dump($client->getBody());
|
||||
function order(int $userId): string
|
||||
{
|
||||
$explode = current(explode(' ', str_replace('0.', '', round((float)microtime(),6))));
|
||||
|
||||
});
|
||||
return 'N'.sprintf('%09d', $userId) . '.' . date('YmdHis') . '.' . str_pad($explode,6,0);
|
||||
}
|
||||
var_dump(
|
||||
order(1)
|
||||
);
|
||||
Reference in New Issue
Block a user