commit a7683eaf4084da049f6295545e29d62affc6c8a0 Author: mkm <727897186@qq.com> Date: Tue Sep 12 20:37:00 2023 +0800 init diff --git a/.gitignore b/.gitignore new file mode 100644 index 0000000..0307853 --- /dev/null +++ b/.gitignore @@ -0,0 +1,7 @@ +/.idea +/.vscode +/vendor +*.log +.env +/tests/tmp +/tests/.phpunit.result.cache diff --git a/LICENSE b/LICENSE new file mode 100644 index 0000000..2c66292 --- /dev/null +++ b/LICENSE @@ -0,0 +1,21 @@ +MIT License + +Copyright (c) 2021 walkor and contributors (see https://github.com/walkor/webman/contributors) + +Permission is hereby granted, free of charge, to any person obtaining a copy +of this software and associated documentation files (the "Software"), to deal +in the Software without restriction, including without limitation the rights +to use, copy, modify, merge, publish, distribute, sublicense, and/or sell +copies of the Software, and to permit persons to whom the Software is +furnished to do so, subject to the following conditions: + +The above copyright notice and this permission notice shall be included in all +copies or substantial portions of the Software. + +THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR +IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, +FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE +AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER +LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, +OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE +SOFTWARE. diff --git a/README.md b/README.md new file mode 100644 index 0000000..9538cd9 --- /dev/null +++ b/README.md @@ -0,0 +1,59 @@ +
+

webman

+ +基于workerman开发的超高性能PHP框架 + + +

学习

+ + + +
+ +

赞助商

+ +

特别赞助

+ + + + +

铂金赞助

+ + + +

金牌赞助

+ + +
+ + +
+

LICENSE

+The webman is open-sourced software licensed under the MIT. +
+ +
+ + diff --git a/app/Pusher.php b/app/Pusher.php new file mode 100644 index 0000000..32a6b6d --- /dev/null +++ b/app/Pusher.php @@ -0,0 +1,123 @@ +assembleAuthUrl('ws://spark-api.xf-yun.com/v2.1/chat'); + $con = new AsyncTcpConnection($uri); + $con->onWebSocketConnect = function (AsyncTcpConnection $con,$data) { + $res = $this->getBody($this->app_id,$data); + $con->send($res); + }; + // $con->onMessage = function (TcpConnection $connection, $res) { + // // var_dump($res); + // // $con->send($res); + // $connection->send($res); + + // }; + $con->connect(); + } + } + + public function onClose(TcpConnection $connection) + { + echo "onClose\n"; + } + + function getBody($appid, $question) + { + $header = array( + "app_id" => $appid, + "uid" => "1" + ); + + $parameter = array( + "chat" => array( + "domain" => "generalv2", + "temperature" => 0.5, + "max_tokens" => 1024 + ) + ); + + $payload = array( + "message" => array( + "text" => array( + array("role" => "user", "content" => $question) + ) + ) + ); + + $json_string = json_encode(array( + "header" => $header, + "parameter" => $parameter, + "payload" => $payload + )); + + return $json_string; + } + + function assembleAuthUrl($addr, $method = 'GET') + { + $apiKey = $this->api_key; + $apiSecret = $this->api_secret; + if ($apiKey == "" && $apiSecret == "") { // 不鉴权 + return $addr; + } + + $ul = parse_url($addr); // 解析地址 + if ($ul === false) { // 地址不对,也不鉴权 + return $addr; + } + // // $date = date(DATE_RFC1123); // 获取当前时间并格式化为RFC1123格式的字符串 + $timestamp = time(); + $rfc1123_format = gmdate("D, d M Y H:i:s \G\M\T", $timestamp); + // $rfc1123_format = "Mon, 31 Jul 2023 08:24:03 GMT"; + + // 参与签名的字段 host, date, request-line + $signString = array("host: " . $ul["host"], "date: " . $rfc1123_format, $method . " " . $ul["path"] . " HTTP/1.1"); + // 对签名字符串进行排序,确保顺序一致 + // ksort($signString); + // 将签名字符串拼接成一个字符串 + $sgin = implode("\n", $signString); + + // 对签名字符串进行HMAC-SHA256加密,得到签名结果 + $sha = hash_hmac('sha256', $sgin, $apiSecret, true); + + $signature_sha_base64 = base64_encode($sha); + + // 将API密钥、算法、头部信息和签名结果拼接成一个授权URL + $authUrl = "api_key=\"$apiKey\",algorithm=\"hmac-sha256\",headers=\"host date request-line\",signature=\"$signature_sha_base64\""; + // 对授权URL进行Base64编码,并添加到原始地址后面作为查询参数 + $authAddr = $addr . '?' . http_build_query(array( + 'host' => $ul['host'], + 'date' => $rfc1123_format, + 'authorization' => base64_encode($authUrl), + )); + return $authAddr; + } +} diff --git a/app/controller/IndexController.php b/app/controller/IndexController.php new file mode 100644 index 0000000..ee818f4 --- /dev/null +++ b/app/controller/IndexController.php @@ -0,0 +1,29 @@ + 'webman']); + } + + public function json(Request $request) + { + return json(['code' => 0, 'msg' => 'ok']); + } + +} diff --git a/app/extend/IFlytek/Xfyun/Core/Config/ConfigInterface.php b/app/extend/IFlytek/Xfyun/Core/Config/ConfigInterface.php new file mode 100644 index 0000000..46fb651 --- /dev/null +++ b/app/extend/IFlytek/Xfyun/Core/Config/ConfigInterface.php @@ -0,0 +1,41 @@ +client = $client; + } + + /** + * 接收Psr-7的request对象和请求参数,返回Psr-7的response对象 + * + * @param RequestInterface $request + * @param array $options + * @return ResponseInterface + */ + public function __invoke(RequestInterface $request, array $options = []) + { + return $this->client->send($request, $options); + } +} \ No newline at end of file diff --git a/app/extend/IFlytek/Xfyun/Core/Handler/HttpHandlerFactory.php b/app/extend/IFlytek/Xfyun/Core/Handler/HttpHandlerFactory.php new file mode 100644 index 0000000..d7eb339 --- /dev/null +++ b/app/extend/IFlytek/Xfyun/Core/Handler/HttpHandlerFactory.php @@ -0,0 +1,45 @@ +client = new Client($uri); + $this->client->setTimeout($timeout); + $this->input = $input; + $this->logger = $logger ?: new NullLogger(); + } + + /** + * 发送并等待获取返回 + * + * 这是一个同步阻塞的过程,调用将持续到结果返回或者出现超时 + */ + public function sendAndReceive() + { + $result = ''; + try { + $this->logger->info("Start to send data, input: {$this->input}"); + $this->client->send($this->input); + $printSid = true; + while (true) { + $message = $this->jsonDecode($this->client->receive()); + if ($message->code !== 0) { + throw new \Exception(json_encode($message)); + } + if ($printSid) { + $this->logger->info("Receiving data, sid-[{$message->sid}]"); + $printSid = false; + } + switch ($message->data->status) { + case 1: + $result .= base64_decode($message->data->audio); + break; + case 2: + $result .= base64_decode($message->data->audio); + break 2; + } + } + $this->logger->info("Receive data successfully, total length: " . strlen($result)); + return new Response(200, [], $result); + } catch (Exception $ex) { + throw $ex; + } + } + + public function send($message = null) + { + try { + if (empty($message)) { + if (!empty($this->input)) { + $message = $this->input; + } else { + throw new Exception(); + } + } + return $this->client->send($message); + } catch (Exception $ex) { + throw $ex; + } + } + + public function receive() + { + try { + return $this->client->receive(); + } catch (Exception $ex) { + throw $ex; + } + } + + public function setLogger(LoggerInterface $logger = null) + { + $this->logger = $logger ?: new NullLogger(); + } + +} diff --git a/app/extend/IFlytek/Xfyun/Core/HttpClient.php b/app/extend/IFlytek/Xfyun/Core/HttpClient.php new file mode 100644 index 0000000..8f69c17 --- /dev/null +++ b/app/extend/IFlytek/Xfyun/Core/HttpClient.php @@ -0,0 +1,163 @@ + null, + 'httpHeaders' => [], + 'requestTimeout' => 3000, + 'retries' => 3, + 'decideRetryFunction' => null, + 'delayFunction' => null, + 'calcDelayFunction' => null + ]; + $this->httpHandler = $config['httpHandler'] ?: HttpHandlerFactory::build(); + $this->httpHeaders = $config['httpHeaders']; + $this->retries = $config['retries']; + $this->decideRetryFunction = $config['decideRetryFunction'] ?: $this->getDecideRetryFunction(); + $this->calcDelayFunction = $config['calcDelayFunction'] ?: [$this, 'calculateDelay']; + $this->delayFunction = $config['delayFunction'] ?: static function ($delay) { + usleep($delay); + }; + } + + /** + * 发送请求,并接受返回 + * + * @param RequestInterface $request + * @param array $options 请求的配置参数 + * @return Response + * @throws Exception + */ + public function sendAndReceive(RequestInterface $request, array $options = []) + { + try { + $delayFunction = $this->delayFunction; + $calcDelayFunction = $this->calcDelayFunction; + $retryAttempt = 0; + + while (true) { + try { + return call_user_func_array($this->httpHandler, [$this->applyHeaders($request), $options]); + } catch (Exception $exception) { + if ($this->decideRetryFunction) { + if (!call_user_func($this->decideRetryFunction, $exception)) { + throw $exception; + } + } + + if ($retryAttempt >= $this->retries) { + break; + } + + $delayFunction($calcDelayFunction($retryAttempt)); + $retryAttempt++; + } + } + + throw $exception; + } catch (Exception $ex) { + throw $ex; + } + } + + /** + * 将头部信息添加到请求对象中 + * + * @param $request 请求对象 + * @return RequestInterface + */ + private function applyHeaders($request) + { + return Utils::modifyRequest($request, ['set_headers' => $this->httpHeaders]); + } + + /** + * 根据重试次数计算下次重试延迟时间 + * + * @param int $attempt 重试次数 + * @return int + */ + public static function calculateDelay($attempt) + { + return min( + mt_rand(0, 1000000) + (pow(2, $attempt) * 1000000), + self::MAX_DELAY_MICROSECONDS + ); + } +} diff --git a/app/extend/IFlytek/Xfyun/Core/Traits/ArrayTrait.php b/app/extend/IFlytek/Xfyun/Core/Traits/ArrayTrait.php new file mode 100644 index 0000000..72f53f4 --- /dev/null +++ b/app/extend/IFlytek/Xfyun/Core/Traits/ArrayTrait.php @@ -0,0 +1,45 @@ + $item) { + if (is_array($item)) { + $array[$key] = self::removeNull($item); + } else if (is_null($item)) { + unset($array[$key]); + } + } + return $array; + } +} diff --git a/app/extend/IFlytek/Xfyun/Core/Traits/DecideRetryTrait.php b/app/extend/IFlytek/Xfyun/Core/Traits/DecideRetryTrait.php new file mode 100644 index 0000000..d7801d3 --- /dev/null +++ b/app/extend/IFlytek/Xfyun/Core/Traits/DecideRetryTrait.php @@ -0,0 +1,96 @@ +httpRetryCodes; + $httpRetryMessages = $this->httpRetryMessages; + + return function (\Exception $ex) use ($httpRetryCodes, $httpRetryMessages, $shouldRetryMessages) { + $statusCode = $ex->getCode(); + + if (in_array($statusCode, $httpRetryCodes)) { + return true; + } + + if (!$shouldRetryMessages) { + return false; + } + + $message = ($ex instanceof RequestException && $ex->hasResponse()) + ? (string) $ex->getResponse()->getBody() + : $ex->getMessage(); + + try { + $message = $this->jsonDecode( + $message, + true + ); + } catch (\InvalidArgumentException $ex) { + return false; + } + + if (!isset($message['errors'])) { + return false; + } + + foreach ($message['errors'] as $error) { + if (in_array($error['reason'], $httpRetryMessages)) { + return true; + } + } + + return false; + }; + } +} diff --git a/app/extend/IFlytek/Xfyun/Core/Traits/JsonTrait.php b/app/extend/IFlytek/Xfyun/Core/Traits/JsonTrait.php new file mode 100644 index 0000000..cb8383b --- /dev/null +++ b/app/extend/IFlytek/Xfyun/Core/Traits/JsonTrait.php @@ -0,0 +1,68 @@ + $host, + 'date' => $date, + 'authorization' => $authrization + ]); + return $uri; + } + + /** + * 根据所提供信息返回签名 + * + * @param string $appId appid + * @param string $secretKey secretKey + * @param string $timestamp 时间戳,不传的话使用系统时间 + * @return string + */ + public static function signV1($appId, $secretKey, $timestamp = null) + { + $timestamp = $timestamp ?: time(); + $baseString = $appId . $timestamp; + $signa_origin = hash_hmac('sha1', md5($baseString), $secretKey, true); + return base64_encode($signa_origin); + } + + /** + * https调用的鉴权参数构造 + * + * @param string $appId appId + * @param string $apiKey apiKey + * @param string $curTime curTime + * @param string $param param + * @return array + */ + public static function signV2($appId, $apiKey, $param, $curTime = null) + { + if (empty($curTime)) { + $curTime = time(); + } + return [ + 'X-Appid' => $appId, + 'X-CurTime' => $curTime, + 'X-Param' => base64_encode($param), + 'X-CheckSum' => md5($apiKey . $curTime . base64_encode($param)) + ]; + } +} diff --git a/app/extend/IFlytek/Xfyun/Core/WsClient.php b/app/extend/IFlytek/Xfyun/Core/WsClient.php new file mode 100644 index 0000000..0d6035e --- /dev/null +++ b/app/extend/IFlytek/Xfyun/Core/WsClient.php @@ -0,0 +1,70 @@ + null, + ]; + $this->handler = $config['handler']; + } + + /** + * 发送请求,并接受返回 + * + * @return Response + */ + public function sendAndReceive() + { + try { + return call_user_func_array([$this->handler, 'sendAndReceive'], []); + } catch (\Exception $ex) { + throw $ex; + } + } + + public function send($message = null) + { + try { + return call_user_func_array([$this->handler, 'send'], [$message]); + } catch (\Exception $ex) { + throw $ex; + } + } + + public function receive() + { + try { + return call_user_func_array([$this->handler, 'receive'], []); + } catch (\Exception $ex) { + throw $ex; + } + } +} diff --git a/app/extend/IFlytek/Xfyun/Speech/ChatClient.php b/app/extend/IFlytek/Xfyun/Speech/ChatClient.php new file mode 100644 index 0000000..3215662 --- /dev/null +++ b/app/extend/IFlytek/Xfyun/Speech/ChatClient.php @@ -0,0 +1,84 @@ +appId = $appId; + $this->apiKey = $apiKey; + $this->apiSecret = $apiSecret; + $this->uid = $uid; + $this->resId = $resId; + } + + function assembleAuthUrl($addr,$method='GET') { + $apiKey=$this->apiKey; + $apiSecret=$this->apiSecret; + if ($apiKey == "" && $apiSecret == "") { // 不鉴权 + return $addr; + } + + $ul = parse_url($addr); // 解析地址 + if ($ul === false) { // 地址不对,也不鉴权 + return $addr; + } + // // $date = date(DATE_RFC1123); // 获取当前时间并格式化为RFC1123格式的字符串 + $timestamp = time(); + $rfc1123_format = gmdate("D, d M Y H:i:s \G\M\T", $timestamp); + // $rfc1123_format = "Mon, 31 Jul 2023 08:24:03 GMT"; + + // 参与签名的字段 host, date, request-line + $signString = array("host: " . $ul["host"], "date: " . $rfc1123_format, $method . " " . $ul["path"] . " HTTP/1.1"); + // 对签名字符串进行排序,确保顺序一致 + // ksort($signString); + // 将签名字符串拼接成一个字符串 + $sgin = implode("\n", $signString); + + // 对签名字符串进行HMAC-SHA256加密,得到签名结果 + $sha = hash_hmac('sha256', $sgin, $apiSecret,true); + + $signature_sha_base64 = base64_encode($sha); + + // 将API密钥、算法、头部信息和签名结果拼接成一个授权URL + $authUrl = "api_key=\"$apiKey\",algorithm=\"hmac-sha256\",headers=\"host date request-line\",signature=\"$signature_sha_base64\""; + // 对授权URL进行Base64编码,并添加到原始地址后面作为查询参数 + $authAddr = $addr . '?' . http_build_query(array( + 'host' => $ul['host'], + 'date' => $rfc1123_format, + 'authorization' => base64_encode($authUrl), + )); + return $authAddr; + } +} diff --git a/app/extend/IFlytek/Xfyun/Speech/Config/IgrConfig.php b/app/extend/IFlytek/Xfyun/Speech/Config/IgrConfig.php new file mode 100644 index 0000000..b7f83fe --- /dev/null +++ b/app/extend/IFlytek/Xfyun/Speech/Config/IgrConfig.php @@ -0,0 +1,110 @@ + 'raw', + 'rate' => 16000 + ]; + + $this->ent = 'igr'; + $this->aue = $config['aue']; + $this->rate = $config['rate']; + } + + /** + * 去除null项后返回数组形式 + * + * @return array + */ + public function toArray() + { + return $this->removeNull([ + 'ent' => $this->ent, + 'aue' => $this->aue, + 'rate' => $this->rate + ]); + } + + /** + * 返回toArray的Json格式 + * + * @return string + */ + public function toJson() + { + return $this->jsonEncode($this->toArray()); + } + + /** + * @return string + */ + public function getAue() + { + return $this->aue; + } + + /** + * @return int + */ + public function getRate() + { + return $this->rate; + } +} diff --git a/app/extend/IFlytek/Xfyun/Speech/Config/IseConfig.php b/app/extend/IFlytek/Xfyun/Speech/Config/IseConfig.php new file mode 100644 index 0000000..8c22a94 --- /dev/null +++ b/app/extend/IFlytek/Xfyun/Speech/Config/IseConfig.php @@ -0,0 +1,265 @@ + 'cn_vip', + 'category' => 'read_sentence', + 'aus' => 1, + 'cmd' => 'ssb', + 'text' => '', + 'tte' => 'utf-8', + 'ttp_skip' => true, + 'extra_ability' => null, + 'aue' => 'raw', + 'rstcd' => 'utf8', + 'group' => 'adult', + 'check_type' => 'common', + 'grade' => 'middle', + 'rst' => 'entirety', + 'ise_unite' => '0', + 'plev' => '0' + ]; + + $this->sub = 'ise'; + $this->ent = $config['ent']; + $this->category = $config['category']; + $this->aus = $config['aus']; + $this->cmd = $config['cmd']; + $this->text = chr(239) . chr(187) . chr(191) . $config['text']; + $this->tte = $config['tte']; + $this->ttpSkip = $config['ttp_skip']; + $this->extraAbility = $config['extra_ability']; + $this->aue = $config['aue']; + $this->rstcd = $config['rstcd']; + $this->group = $config['group']; + $this->checkType = $config['check_type']; + $this->grade = $config['grade']; + $this->rst = $config['rst']; + $this->iseUnite = $config['ise_unite']; + $this->plev = $config['plev']; + } + + /** + * 去除null项后返回数组形式 + * + * @return array + */ + public function toArray() + { + return $this->removeNull([ + 'sub' => $this->sub, + 'ent' => $this->ent, + 'category' => $this->category, + 'aus' => $this->aus, + 'cmd' => $this->cmd, + 'text' => $this->text, + 'tte' => $this->tte, + 'ttp_skip' => $this->ttpSkip, + 'extra_ability' => $this->extraAbility, + 'aue' => $this->aue, + 'rstcd' => $this->rstcd, + 'group' => $this->group, + 'check_type' => $this->checkType, + 'grade' => $this->grade, + 'rst' => $this->rst, + 'ise_unite' => $this->iseUnite, + 'plev' => $this->plev + ]); + } + + /** + * 返回toArray的Json格式 + * + * @return string + */ + public function toJson() + { + return $this->jsonEncode($this->toArray()); + } + + public function setText($text) + { + $this->text = chr(239) . chr(187) . chr(191) . $text; + } +} diff --git a/app/extend/IFlytek/Xfyun/Speech/Config/LfasrConfig.php b/app/extend/IFlytek/Xfyun/Speech/Config/LfasrConfig.php new file mode 100644 index 0000000..cdc2b65 --- /dev/null +++ b/app/extend/IFlytek/Xfyun/Speech/Config/LfasrConfig.php @@ -0,0 +1,142 @@ + '0', + 'has_participle' => 'false', + 'max_alternatives' => '0', + 'speaker_number' => '2', + 'has_seperate' => 'false', + 'role_type' => '1', + 'language' => 'cn', + 'pd' => null + ]; + + $this->lfasrType = $config['lfasr_type']; + $this->hasParticiple = $config['has_participle']; + $this->maxAlternatives = $config['max_alternatives']; + $this->speakerNumber = $config['speaker_number']; + $this->hasSeperate = $config['has_seperate']; + $this->roleType = $config['role_type']; + $this->language = $config['language']; + $this->pd = $config['pd']; + } + + /** + * 去除null项后返回数组形式 + * + * @return array + */ + public function toArray() + { + return $this->removeNull([ + 'lfasr_type' => $this->lfasrType, + 'has_participle' => $this->hasParticiple, + 'max_alternatives' => $this->maxAlternatives, + 'speaker_number' => $this->speakerNumber, + 'has_seperate' => $this->hasSeperate, + 'role_type' => $this->roleType, + 'language' => $this->language, + 'pd' => $this->pd + ]); + } + + /** + * 返回toArray的Json格式 + * + * @return string + */ + public function toJson() + { + return $this->jsonEncode($this->toArray()); + } +} diff --git a/app/extend/IFlytek/Xfyun/Speech/Config/TcConfig.php b/app/extend/IFlytek/Xfyun/Speech/Config/TcConfig.php new file mode 100644 index 0000000..980417b --- /dev/null +++ b/app/extend/IFlytek/Xfyun/Speech/Config/TcConfig.php @@ -0,0 +1,76 @@ + 'utf8', + 'resultCompress' => 'raw', + 'resultFormat' => 'json', + 'inputEncoding' => 'utf8', + 'inputCompress' => 'raw', + 'inputFormat' => 'json' + ]; + + $this->resultEncoding = $config['resultEncoding']; + $this->resultCompress = $config['resultCompress']; + $this->resultFormat = $config['resultFormat']; + $this->inputEncoding = $config['inputEncoding']; + $this->inputCompress = $config['inputCompress']; + $this->inputFormat = $config['inputFormat']; + } + + public function toJson() + { + // TODO: Implement toJson() method. + } + + public function toArray() + { + return self::removeNull([ + 'resultEncoding' => $this->resultEncoding, + 'resultCompress' => $this->resultCompress, + 'resultFormat' => $this->resultFormat, + 'inputEncoding' => $this->inputEncoding, + 'inputCompress' => $this->inputCompress, + 'inputFormat' => $this->inputFormat + ]); + } +} diff --git a/app/extend/IFlytek/Xfyun/Speech/Config/TtsConfig.php b/app/extend/IFlytek/Xfyun/Speech/Config/TtsConfig.php new file mode 100644 index 0000000..c9f6b31 --- /dev/null +++ b/app/extend/IFlytek/Xfyun/Speech/Config/TtsConfig.php @@ -0,0 +1,191 @@ + 'lame', + 'sfl' => 1, + 'auf' => null, + 'vcn' => 'xiaoyan', + 'speed' => 50, + 'volume' => 50, + 'pitch' => 50, + 'bgs' => 0, + 'tte' => 'UTF8', + 'reg' => '2', + 'rdn' => '0', + 'ent' => '' + ]; + + $this->aue = $config['aue']; + $this->sfl = $config['sfl']; + $this->auf = $config['auf']; + $this->vcn = $config['vcn']; + $this->speed = $config['speed']; + $this->volume = $config['volume']; + $this->pitch = $config['pitch']; + $this->bgs = $config['bgs']; + $this->tte = $config['tte']; + $this->reg = $config['reg']; + $this->rdn = $config['rdn']; + $this->ent = $config['ent']; + } + + /** + * 去除null项后返回数组形式 + * + * @return array + */ + public function toArray() + { + return $this->removeNull([ + 'aue' => $this->aue, + 'sfl' => $this->sfl, + 'auf' => $this->auf, + 'vcn' => $this->vcn, + 'speed' => $this->speed, + 'volume' => $this->volume, + 'pitch' => $this->pitch, + 'bgs' => $this->bgs, + 'tte' => $this->tte, + 'reg' => $this->reg, + 'rdn' => $this->rdn, + 'ent' => $this->ent + ]); + } + + /** + * 返回toArray的Json格式 + * + * @return string + */ + public function toJson() + { + return $this->jsonEncode($this->toArray()); + } +} diff --git a/app/extend/IFlytek/Xfyun/Speech/Constants/ChatConstants.php b/app/extend/IFlytek/Xfyun/Speech/Constants/ChatConstants.php new file mode 100644 index 0000000..fa97099 --- /dev/null +++ b/app/extend/IFlytek/Xfyun/Speech/Constants/ChatConstants.php @@ -0,0 +1,30 @@ +id = LfasrConstants::ORIGIN_SLICE_ID; + } + + /** + * 返回当前切片ID,并生成下一个切片ID,赋值给对象的当前ID + * + * @return string + */ + public function getId() + { + $currentId = $this->id; + $nextId = $currentId; + $pos = strlen($currentId) - 1; + while ($pos >= 0) { + $charAtPos = $nextId[$pos]; + if ($charAtPos != 'z') { + $nextId = substr($nextId, 0, $pos) . chr((ord($charAtPos) + 1)) . substr($nextId, $pos + 1); + break; + } else { + $nextId = substr($nextId, 0, $pos) . 'a'; + $pos = $pos - 1; + } + } + $this->id = $nextId; + return $currentId; + } +} diff --git a/app/extend/IFlytek/Xfyun/Speech/IgrClient.php b/app/extend/IFlytek/Xfyun/Speech/IgrClient.php new file mode 100644 index 0000000..423a004 --- /dev/null +++ b/app/extend/IFlytek/Xfyun/Speech/IgrClient.php @@ -0,0 +1,111 @@ +appId = $appId; + $this->apiKey = $apiKey; + $this->apiSecret = $apiSecret; + $this->requestConfig = new IgrConfig($requestConfig); + } + + /** + * 请求并返回结果 + * + * @param string $audioPath 待识别音频路径 + * @return string + * @throws Exception + */ + public function request($audioPath) + { + $ttsHandler = new WsHandler( + $this->signUriV1(IgrConstants::URI, [ + 'appId' => $this->appId, + 'apiKey' => $this->apiKey, + 'apiSecret' => $this->apiSecret, + 'host' => IgrConstants::HOST, + 'requestLine' => IgrConstants::REQUEST_LINE, + ]), + null + ); + $client = new WsClient([ + 'handler' => $ttsHandler + ]); + + // 音频上传 + $frameNum = ceil(fileSize($audioPath) / IgrConstants::FRAME_SIZE); + $fileStream = new Stream(fopen($audioPath, 'r')); + // 发送第一帧 + $client->send($this->generateAudioInput($fileStream->read(IgrConstants::FRAME_SIZE), true, false)); + + // 发送中间帧 + for ($i = 1; $i < $frameNum; $i++) { + $client->send($this->generateAudioInput($fileStream->read(IgrConstants::FRAME_SIZE), false, false)); + usleep(4000); + } + // 发送最后一帧 + $client->send($this->generateAudioInput('', false, true)); + + // 接受数据 + $message = $this->jsonDecode($client->receive(), true); + if ($message['code'] !== 0) { + throw new Exception(json_encode($message)); + } + return $message['data']; + } +} diff --git a/app/extend/IFlytek/Xfyun/Speech/IseClient.php b/app/extend/IFlytek/Xfyun/Speech/IseClient.php new file mode 100644 index 0000000..8e9cf68 --- /dev/null +++ b/app/extend/IFlytek/Xfyun/Speech/IseClient.php @@ -0,0 +1,126 @@ +appId = $appId; + $this->apiKey = $apiKey; + $this->apiSecret = $apiSecret; + $this->requestConfig = new IseConfig($requestConfig); + } + + /** + * 请求评测,并返回结果(xml格式) + * + * @param string $audioPath 待评测音频路径 + * @param string $text 待评测文本 + * @return string + */ + public function request($audioPath, $text = null) + { + $ttsHandler = new WsHandler( + $this->signUriV1(IseConstants::URI, [ + 'appId' => $this->appId, + 'apiKey' => $this->apiKey, + 'apiSecret' => $this->apiSecret, + 'host' => IseConstants::HOST, + 'requestLine' => IseConstants::REQUEST_LINE, + ]), + null + ); + $client = new WsClient([ + 'handler' => $ttsHandler + ]); + + // 参数上传 + if (!empty($text)) { + $this->requestConfig->setText($text); + } + $client->send($this->generateParamsInput($this->appId, $this->requestConfig->toArray())); + + // 音频上传 + $frameSize = ceil(fileSize($audioPath) / IseConstants::FRAME_SIZE); + $fileStream = new Stream(fopen($audioPath, 'r')); + // 发送第一帧 + $result = $client->send($this->generateAudioInput($fileStream->read(IseConstants::FRAME_SIZE), true, false)); + // 发送中间帧 + for ($i = 1; $i < $frameSize - 1; $i++) { + $client->send($this->generateAudioInput($fileStream->read(IseConstants::FRAME_SIZE), false, false)); + usleep(4000); + } + // 发送最后一帧 + $client->send($this->generateAudioInput($fileStream->read(IseConstants::FRAME_SIZE), false, true)); + + // 接受数据 + $result = ''; + while (true) { + $message = $this->jsonDecode($client->receive()); + if ($message->code !== 0) { + throw new \Exception(json_encode($message)); + } + switch ($message->data->status) { + case 1: + $result .= base64_decode($message->data->data); + break; + case 2: + $result .= base64_decode($message->data->data); + break 2; + } + } + return $result; + } +} diff --git a/app/extend/IFlytek/Xfyun/Speech/LfasrClient.php b/app/extend/IFlytek/Xfyun/Speech/LfasrClient.php new file mode 100644 index 0000000..ddc2962 --- /dev/null +++ b/app/extend/IFlytek/Xfyun/Speech/LfasrClient.php @@ -0,0 +1,231 @@ +appId = $appId; + $this->secretKey = $secretKey; + $this->requestConfig = new LfasrConfig($requestConfig); + $this->client = new HttpClient([]); + $timestamp = time(); + $this->requestBody = [ + 'app_id' => $this->appId, + 'signa' => $this->signV1($this->appId, $this->secretKey, $timestamp), + 'ts' => $timestamp, + ]; + } + + /** + * 打包上传接口,封装了准备、分片上传和合并三个接口,并返回task_id + * + * @param string $filePath 文件路径 + * @return string + * @throws Exception + */ + public function combineUpload($filePath) + { + $prepareResponse = $this->prepare($filePath); + $prepareContent = $this->jsonDecode($prepareResponse->getBody()->getContents(), true); + $taskId = $prepareContent['data']; + $this->upload($taskId, $filePath); + $this->merge($taskId); + return $taskId; + } + + /** + * 准备接口 + * + * @param $file_path + * @return Response + * @throws Exception + */ + public function prepare($file_path) + { + $this->requestBody += $this->fileInfo($file_path); + $this->requestBody += $this->sliceInfo($file_path); + $this->requestBody += $this->requestConfig->toArray(); + + return $this->client->sendAndReceive( + new Request( + 'POST', + LfasrConstants::URI_PREPARE, + ['Content-Type' => 'application/x-www-form-urlencoded;charset=UTF-8'], + Query::build($this->requestBody) + ) + ); + } + + /** + * 分片上传接口 + * + * @param string $taskId task_id + * @param string $filePath 文件路径 + * @return Response + * @throws Exception + */ + public function upload($taskId, $filePath) + { + $sliceIdGenerator = new SliceIdGenerator(); + $sliceInfo = $this->sliceInfo($filePath); + $fileStream = new Stream(fopen($filePath, 'r')); + + $this->requestBody += [ + 'task_id' => $taskId + ]; + + $request = new Request( + 'POST', + LfasrConstants::URI_UPLOAD + ); + + for ($i = 0; $i < $sliceInfo['slice_num']; $i++) { + $multipartStream = new MultipartStream([ + [ + 'name' => 'app_id', + 'contents' => $this->requestBody['app_id'] + ], + [ + 'name' => 'signa', + 'contents' => $this->requestBody['signa'] + ], + [ + 'name' => 'ts', + 'contents' => $this->requestBody['ts'] + ], + [ + 'name' => 'task_id', + 'contents' => $taskId, + ], + [ + 'name' => 'slice_id', + 'contents' => $sliceIdGenerator->getId() + ], + [ + 'name' => 'content', + 'contents' => $fileStream->read(LfasrConstants::SLICE_PIECE_SIZE), + 'filename' => '1.pcm' + ] + ]); + + $this->client->sendAndReceive( + $request->withBody($multipartStream) + ); + } + return new Response(200); + } + + /** + * 合并接口 + * + * @param string $taskId task_id + * @return Response + */ + public function merge($taskId) + { + return $this->process(LfasrConstants::URI_MERGE, $taskId); + } + + /** + * 查询进度接口 + * + * @param string $taskId task_id + * @return Response + */ + public function getProgress($taskId) + { + return $this->process(LfasrConstants::URI_GET_PROGRESS, $taskId); + } + + /** + * 获取结果接口 + * + * @param string $taskId task_id + * @return Response + */ + public function getResult($taskId) + { + return $this->process(LfasrConstants::URI_GET_RESULT, $taskId); + } + + /** + * 封装操作 + * + * @param string $task 操作 + * @param string $taskId task_id + * @return Response + */ + private function process($task, $taskId) + { + $this->requestBody += ['task_id' => $taskId]; + return $this->client->sendAndReceive( + new Request( + 'POST', + $task, + ['Content-Type' => 'application/x-www-form-urlencoded;charset=UTF-8'], + Query::build($this->requestBody) + ) + ); + } +} diff --git a/app/extend/IFlytek/Xfyun/Speech/TcClient.php b/app/extend/IFlytek/Xfyun/Speech/TcClient.php new file mode 100644 index 0000000..d328d2b --- /dev/null +++ b/app/extend/IFlytek/Xfyun/Speech/TcClient.php @@ -0,0 +1,104 @@ +appId = $appId; + $this->apiKey = $apiKey; + $this->apiSecret = $apiSecret; + $this->uid = $uid; + $this->resId = $resId; + $this->requestConfig = new TcConfig($requestConfig); + $this->client = new HttpClient([]); + } + + public function request($text) + { + $uri = self::signUriV1(TcConstants::URI, [ + 'apiKey' => $this->apiKey, + 'apiSecret' => $this->apiSecret, + 'host' => TcConstants::HOST, + 'requestLine' => TcConstants::REQUEST_LINE + ]); + $body = self::generateInput($text, $this->appId, $this->uid, $this->resId, $this->requestConfig->toArray()); + return $this->client->sendAndReceive( + new Request( + 'POST', + $uri, + ['Content-Type' => 'application/json;charset=UTF-8'], + $body + ) + ); + } + + public function listUpload($whiteList, $blackList) + { + if (empty($this->uid) || empty($this->resId)) { + return false; + } + $response = $this->client->sendAndReceive( + new Request( + 'POST', + TcConstants::LIST_UPLOAD_URI, + ['Content-Type' => 'application/x-www-form-urlencoded;charset=UTF-8'], + self::jsonEncode([ + 'common' => [ + 'app_id' => $this->appId, + 'uid' => $this->uid, + + ], + 'business' => [ + 'res_id' => $this->resId + ], + 'data' => base64_encode(self::jsonEncode([ + 'white_list' => $whiteList, + 'black_list' => $blackList + ])) + ]) + ) + ); + $content = json_decode($response->getBody()->getContents(), true); + return $content['message'] == 'success'; + } +} diff --git a/app/extend/IFlytek/Xfyun/Speech/Traits/IgrTrait.php b/app/extend/IFlytek/Xfyun/Speech/Traits/IgrTrait.php new file mode 100644 index 0000000..216184a --- /dev/null +++ b/app/extend/IFlytek/Xfyun/Speech/Traits/IgrTrait.php @@ -0,0 +1,59 @@ + !$isFirstFrame ? null : [ + "app_id" => $this->appId + ], + 'business' => !$isFirstFrame ? null : [ + "aue" => $this->requestConfig->getAue(), + "rate" => $this->requestConfig->getRate() + ], + 'data' => [ + 'status' => $isFirstFrame ? 0 : ($isLastFrame ? 2 : 1), + 'audio' => base64_encode($frameData) + ] + ]) + ); + } +} diff --git a/app/extend/IFlytek/Xfyun/Speech/Traits/IseTrait.php b/app/extend/IFlytek/Xfyun/Speech/Traits/IseTrait.php new file mode 100644 index 0000000..c8fe85b --- /dev/null +++ b/app/extend/IFlytek/Xfyun/Speech/Traits/IseTrait.php @@ -0,0 +1,78 @@ + [ + 'app_id' => $appId + ], + 'business' => $iseConfigArray, + 'data' => [ + 'status' => 0 + ] + ]) + ); + } + + /** + * 根据音频数据、是否是第一帧、最后一帧,生成音频上传请求体 + * + * @param string $frameData 音频数据 + * @param boolean $isFirstFrame 是否是第一帧 + * @param boolean $isLastFrame 是否是最后一帧 + * @return string + */ + public static function generateAudioInput($frameData, $isFirstFrame = false, $isLastFrame = false) + { + return self::jsonEncode( + self::removeNull([ + 'business' => [ + "cmd" => "auw", + "aus" => $isFirstFrame ? 1 : ($isLastFrame ? 4 : 2) + ], + 'data' => [ + 'status' => $isLastFrame ? 2 : 1, + 'data' => base64_encode($frameData) + ] + ]) + ); + } +} diff --git a/app/extend/IFlytek/Xfyun/Speech/Traits/LfasrTrait.php b/app/extend/IFlytek/Xfyun/Speech/Traits/LfasrTrait.php new file mode 100644 index 0000000..fb1bf1c --- /dev/null +++ b/app/extend/IFlytek/Xfyun/Speech/Traits/LfasrTrait.php @@ -0,0 +1,56 @@ + basename($filePath), + 'file_len' => filesize($filePath) + ]; + } + + /** + * 根据文件大小和SDK默认分片大小,获取文件分片数目信息 + * + * @param string $filePath 文件路径 + * @return array + */ + public static function sliceInfo($filePath) + { + $fileSize = filesize($filePath); + return [ + 'slice_num' => ceil($fileSize / LfasrConstants::SLICE_PIECE_SIZE) + ]; + } +} diff --git a/app/extend/IFlytek/Xfyun/Speech/Traits/TcTrait.php b/app/extend/IFlytek/Xfyun/Speech/Traits/TcTrait.php new file mode 100644 index 0000000..a6cbb94 --- /dev/null +++ b/app/extend/IFlytek/Xfyun/Speech/Traits/TcTrait.php @@ -0,0 +1,64 @@ + [ + 'app_id' => $appId, + 'uid' => $uid, + 'status' => 3 + ], + 'parameter' => [ + 's9a87e3ec' => [ + 'res_id' => $resId, + 'result' => [ + 'encoding' => $tcConfigArray['resultEncoding'], + 'compress' => $tcConfigArray['resultCompress'], + 'format' => $tcConfigArray['resultFormat'], + ] + ] + ], + 'payload' => [ + 'input' => [ + 'encoding' => $tcConfigArray['inputEncoding'], + 'compress' => $tcConfigArray['inputCompress'], + 'format' => $tcConfigArray['inputFormat'], + 'status' => 3, + 'text' => base64_encode($text) + ] + ] + ]) + ); + } +} diff --git a/app/extend/IFlytek/Xfyun/Speech/Traits/TtsTrait.php b/app/extend/IFlytek/Xfyun/Speech/Traits/TtsTrait.php new file mode 100644 index 0000000..d2de507 --- /dev/null +++ b/app/extend/IFlytek/Xfyun/Speech/Traits/TtsTrait.php @@ -0,0 +1,56 @@ + [ + 'app_id' => $appId + ], + 'business' => $ttsConfigArray, + 'data' => [ + 'text' => base64_encode($text), + 'status' => 2 + ] + ]) + ); + } +} diff --git a/app/extend/IFlytek/Xfyun/Speech/TtsClient.php b/app/extend/IFlytek/Xfyun/Speech/TtsClient.php new file mode 100644 index 0000000..f4978f8 --- /dev/null +++ b/app/extend/IFlytek/Xfyun/Speech/TtsClient.php @@ -0,0 +1,100 @@ +appId = $appId; + $this->apiKey = $apiKey; + $this->apiSecret = $apiSecret; + $this->requestConfig = new TtsConfig($requestConfig); + $this->logger = $logger; + } + + /** + * 合成文本,并返回结果(字节数组)在Response->getBody()->getContents() + * + * @param string $text 待合成的文本 + * @return \GuzzleHttp\Psr7\Response + * @throws Exception + */ + public function request($text) + { + $ttsHandler = new WsHandler( + $this->signUriV1(TtsConstants::URI, [ + 'appId' => $this->appId, + 'apiKey' => $this->apiKey, + 'apiSecret' => $this->apiSecret, + 'host' => TtsConstants::HOST, + 'requestLine' => TtsConstants::REQUEST_LINE, + ]), + $this->generateInput($text, $this->appId, $this->requestConfig->toArray()) + ); + if ($this->logger) { + $ttsHandler->setLogger($this->logger); + } + $client = new WsClient([ + 'handler' => $ttsHandler + ]); + return $client->sendAndReceive(); + } +} diff --git a/app/functions.php b/app/functions.php new file mode 100644 index 0000000..5c9c58d --- /dev/null +++ b/app/functions.php @@ -0,0 +1,4 @@ + + * @copyright walkor + * @link http://www.workerman.net/ + * @license http://www.opensource.org/licenses/mit-license.php MIT License + */ + +namespace app\middleware; + +use Webman\MiddlewareInterface; +use Webman\Http\Response; +use Webman\Http\Request; + +/** + * Class StaticFile + * @package app\middleware + */ +class StaticFile implements MiddlewareInterface +{ + public function process(Request $request, callable $next): Response + { + // Access to files beginning with. Is prohibited + if (strpos($request->path(), '/.') !== false) { + return response('

403 forbidden

', 403); + } + /** @var Response $response */ + $response = $next($request); + // Add cross domain HTTP header + /*$response->withHeaders([ + 'Access-Control-Allow-Origin' => '*', + 'Access-Control-Allow-Credentials' => 'true', + ]);*/ + return $response; + } +} diff --git a/app/model/Test.php b/app/model/Test.php new file mode 100644 index 0000000..92d70e3 --- /dev/null +++ b/app/model/Test.php @@ -0,0 +1,29 @@ + + + + + + + + webman + + + +hello + + diff --git a/composer.json b/composer.json new file mode 100644 index 0000000..0bab6ae --- /dev/null +++ b/composer.json @@ -0,0 +1,57 @@ +{ + "name": "workerman/webman", + "type": "project", + "keywords": [ + "high performance", + "http service" + ], + "homepage": "https://www.workerman.net", + "license": "MIT", + "description": "High performance HTTP Service Framework.", + "authors": [ + { + "name": "walkor", + "email": "walkor@workerman.net", + "homepage": "https://www.workerman.net", + "role": "Developer" + } + ], + "support": { + "email": "walkor@workerman.net", + "issues": "https://github.com/walkor/webman/issues", + "forum": "https://wenda.workerman.net/", + "wiki": "https://workerman.net/doc/webman", + "source": "https://github.com/walkor/webman" + }, + "require": { + "php": ">=7.2", + "workerman/webman-framework": "^1.5.0", + "monolog/monolog": "^2.0", + "textalk/websocket": "^1.6" + }, + "suggest": { + "ext-event": "For better performance. " + }, + "autoload": { + "psr-4": { + "": "./", + "app\\": "./app", + "App\\": "./app", + "app\\View\\Components\\": "./app/view/components" + }, + "files": [ + "./support/helpers.php" + ] + }, + "scripts": { + "post-package-install": [ + "support\\Plugin::install" + ], + "post-package-update": [ + "support\\Plugin::install" + ], + "pre-package-uninstall": [ + "support\\Plugin::uninstall" + ] + } +} diff --git a/composer.lock b/composer.lock new file mode 100644 index 0000000..c68e8a6 --- /dev/null +++ b/composer.lock @@ -0,0 +1,533 @@ +{ + "_readme": [ + "This file locks the dependencies of your project to a known state", + "Read more about it at https://getcomposer.org/doc/01-basic-usage.md#installing-dependencies", + "This file is @generated automatically" + ], + "content-hash": "742c251f2e4bc7f8e8ebe86c60fdac51", + "packages": [ + { + "name": "monolog/monolog", + "version": "2.9.1", + "dist": { + "type": "zip", + "url": "https://mirrors.cloud.tencent.com/repository/composer/monolog/monolog/2.9.1/monolog-monolog-2.9.1.zip", + "reference": "f259e2b15fb95494c83f52d3caad003bbf5ffaa1", + "shasum": "" + }, + "require": { + "php": ">=7.2", + "psr/log": "^1.0.1 || ^2.0 || ^3.0" + }, + "provide": { + "psr/log-implementation": "1.0.0 || 2.0.0 || 3.0.0" + }, + "require-dev": { + "aws/aws-sdk-php": "^2.4.9 || ^3.0", + "doctrine/couchdb": "~1.0@dev", + "elasticsearch/elasticsearch": "^7 || ^8", + "ext-json": "*", + "graylog2/gelf-php": "^1.4.2 || ^2@dev", + "guzzlehttp/guzzle": "^7.4", + "guzzlehttp/psr7": "^2.2", + "mongodb/mongodb": "^1.8", + "php-amqplib/php-amqplib": "~2.4 || ^3", + "phpspec/prophecy": "^1.15", + "phpstan/phpstan": "^0.12.91", + "phpunit/phpunit": "^8.5.14", + "predis/predis": "^1.1 || ^2.0", + "rollbar/rollbar": "^1.3 || ^2 || ^3", + "ruflin/elastica": "^7", + "swiftmailer/swiftmailer": "^5.3|^6.0", + "symfony/mailer": "^5.4 || ^6", + "symfony/mime": "^5.4 || ^6" + }, + "suggest": { + "aws/aws-sdk-php": "Allow sending log messages to AWS services like DynamoDB", + "doctrine/couchdb": "Allow sending log messages to a CouchDB server", + "elasticsearch/elasticsearch": "Allow sending log messages to an Elasticsearch server via official client", + "ext-amqp": "Allow sending log messages to an AMQP server (1.0+ required)", + "ext-curl": "Required to send log messages using the IFTTTHandler, the LogglyHandler, the SendGridHandler, the SlackWebhookHandler or the TelegramBotHandler", + "ext-mbstring": "Allow to work properly with unicode symbols", + "ext-mongodb": "Allow sending log messages to a MongoDB server (via driver)", + "ext-openssl": "Required to send log messages using SSL", + "ext-sockets": "Allow sending log messages to a Syslog server (via UDP driver)", + "graylog2/gelf-php": "Allow sending log messages to a GrayLog2 server", + "mongodb/mongodb": "Allow sending log messages to a MongoDB server (via library)", + "php-amqplib/php-amqplib": "Allow sending log messages to an AMQP server using php-amqplib", + "rollbar/rollbar": "Allow sending log messages to Rollbar", + "ruflin/elastica": "Allow sending log messages to an Elastic Search server" + }, + "type": "library", + "extra": { + "branch-alias": { + "dev-main": "2.x-dev" + } + }, + "autoload": { + "psr-4": { + "Monolog\\": "src/Monolog" + } + }, + "license": [ + "MIT" + ], + "authors": [ + { + "name": "Jordi Boggiano", + "email": "j.boggiano@seld.be", + "homepage": "https://seld.be" + } + ], + "description": "Sends your logs to files, sockets, inboxes, databases and various web services", + "homepage": "https://github.com/Seldaek/monolog", + "keywords": [ + "log", + "logging", + "psr-3" + ], + "time": "2023-02-06T13:44:46+00:00" + }, + { + "name": "nikic/fast-route", + "version": "v1.3.0", + "dist": { + "type": "zip", + "url": "https://mirrors.tencent.com/repository/composer/nikic/fast-route/v1.3.0/nikic-fast-route-v1.3.0.zip", + "reference": "181d480e08d9476e61381e04a71b34dc0432e812", + "shasum": "" + }, + "require": { + "php": ">=5.4.0" + }, + "require-dev": { + "phpunit/phpunit": "^4.8.35|~5.7" + }, + "type": "library", + "autoload": { + "files": [ + "src/functions.php" + ], + "psr-4": { + "FastRoute\\": "src/" + } + }, + "license": [ + "BSD-3-Clause" + ], + "authors": [ + { + "name": "Nikita Popov", + "email": "nikic@php.net" + } + ], + "description": "Fast request router for PHP", + "keywords": [ + "router", + "routing" + ], + "time": "2018-02-13T20:26:39+00:00" + }, + { + "name": "phrity/net-uri", + "version": "1.3.0", + "dist": { + "type": "zip", + "url": "https://mirrors.cloud.tencent.com/repository/composer/phrity/net-uri/1.3.0/phrity-net-uri-1.3.0.zip", + "reference": "3f458e0c4d1ddc0e218d7a5b9420127c63925f43", + "shasum": "" + }, + "require": { + "php": "^7.4 | ^8.0", + "psr/http-factory": "^1.0", + "psr/http-message": "^1.0 | ^2.0" + }, + "require-dev": { + "php-coveralls/php-coveralls": "^2.0", + "phpunit/phpunit": "^9.0 | ^10.0", + "squizlabs/php_codesniffer": "^3.0" + }, + "type": "library", + "autoload": { + "psr-4": { + "Phrity\\Net\\": "src/" + } + }, + "license": [ + "MIT" + ], + "authors": [ + { + "name": "Sören Jensen", + "email": "sirn@sirn.se", + "homepage": "https://phrity.sirn.se" + } + ], + "description": "PSR-7 Uri and PSR-17 UriFactory implementation", + "homepage": "https://phrity.sirn.se/net-uri", + "keywords": [ + "psr-17", + "psr-7", + "uri", + "uri factory" + ], + "time": "2023-08-21T10:33:06+00:00" + }, + { + "name": "phrity/util-errorhandler", + "version": "1.0.1", + "dist": { + "type": "zip", + "url": "https://mirrors.cloud.tencent.com/repository/composer/phrity/util-errorhandler/1.0.1/phrity-util-errorhandler-1.0.1.zip", + "reference": "dc9ac8fb70d733c48a9d9d1eb50f7022172da6bc", + "shasum": "" + }, + "require": { + "php": "^7.2|^8.0" + }, + "require-dev": { + "php-coveralls/php-coveralls": "^2.0", + "phpunit/phpunit": "^8.0|^9.0", + "squizlabs/php_codesniffer": "^3.5" + }, + "type": "library", + "autoload": { + "psr-4": { + "": "src/" + } + }, + "license": [ + "MIT" + ], + "authors": [ + { + "name": "Sören Jensen", + "email": "sirn@sirn.se", + "homepage": "https://phrity.sirn.se" + } + ], + "description": "Inline error handler; catch and resolve errors for code block.", + "homepage": "https://phrity.sirn.se/util-errorhandler", + "keywords": [ + "error", + "warning" + ], + "time": "2022-10-27T12:14:42+00:00" + }, + { + "name": "psr/container", + "version": "2.0.2", + "dist": { + "type": "zip", + "url": "https://mirrors.cloud.tencent.com/repository/composer/psr/container/2.0.2/psr-container-2.0.2.zip", + "reference": "c71ecc56dfe541dbd90c5360474fbc405f8d5963", + "shasum": "" + }, + "require": { + "php": ">=7.4.0" + }, + "type": "library", + "extra": { + "branch-alias": { + "dev-master": "2.0.x-dev" + } + }, + "autoload": { + "psr-4": { + "Psr\\Container\\": "src/" + } + }, + "license": [ + "MIT" + ], + "authors": [ + { + "name": "PHP-FIG", + "homepage": "https://www.php-fig.org/" + } + ], + "description": "Common Container Interface (PHP FIG PSR-11)", + "homepage": "https://github.com/php-fig/container", + "keywords": [ + "PSR-11", + "container", + "container-interface", + "container-interop", + "psr" + ], + "time": "2021-11-05T16:47:00+00:00" + }, + { + "name": "psr/http-factory", + "version": "1.0.2", + "dist": { + "type": "zip", + "url": "https://mirrors.cloud.tencent.com/repository/composer/psr/http-factory/1.0.2/psr-http-factory-1.0.2.zip", + "reference": "e616d01114759c4c489f93b099585439f795fe35", + "shasum": "" + }, + "require": { + "php": ">=7.0.0", + "psr/http-message": "^1.0 || ^2.0" + }, + "type": "library", + "extra": { + "branch-alias": { + "dev-master": "1.0.x-dev" + } + }, + "autoload": { + "psr-4": { + "Psr\\Http\\Message\\": "src/" + } + }, + "license": [ + "MIT" + ], + "authors": [ + { + "name": "PHP-FIG", + "homepage": "https://www.php-fig.org/" + } + ], + "description": "Common interfaces for PSR-7 HTTP message factories", + "keywords": [ + "factory", + "http", + "message", + "psr", + "psr-17", + "psr-7", + "request", + "response" + ], + "time": "2023-04-10T20:10:41+00:00" + }, + { + "name": "psr/http-message", + "version": "1.1", + "dist": { + "type": "zip", + "url": "https://mirrors.cloud.tencent.com/repository/composer/psr/http-message/1.1/psr-http-message-1.1.zip", + "reference": "cb6ce4845ce34a8ad9e68117c10ee90a29919eba", + "shasum": "" + }, + "require": { + "php": "^7.2 || ^8.0" + }, + "type": "library", + "extra": { + "branch-alias": { + "dev-master": "1.1.x-dev" + } + }, + "autoload": { + "psr-4": { + "Psr\\Http\\Message\\": "src/" + } + }, + "license": [ + "MIT" + ], + "authors": [ + { + "name": "PHP-FIG", + "homepage": "http://www.php-fig.org/" + } + ], + "description": "Common interface for HTTP messages", + "homepage": "https://github.com/php-fig/http-message", + "keywords": [ + "http", + "http-message", + "psr", + "psr-7", + "request", + "response" + ], + "time": "2023-04-04T09:50:52+00:00" + }, + { + "name": "psr/log", + "version": "3.0.0", + "dist": { + "type": "zip", + "url": "https://mirrors.cloud.tencent.com/repository/composer/psr/log/3.0.0/psr-log-3.0.0.zip", + "reference": "fe5ea303b0887d5caefd3d431c3e61ad47037001", + "shasum": "" + }, + "require": { + "php": ">=8.0.0" + }, + "type": "library", + "extra": { + "branch-alias": { + "dev-master": "3.x-dev" + } + }, + "autoload": { + "psr-4": { + "Psr\\Log\\": "src" + } + }, + "license": [ + "MIT" + ], + "authors": [ + { + "name": "PHP-FIG", + "homepage": "https://www.php-fig.org/" + } + ], + "description": "Common interface for logging libraries", + "homepage": "https://github.com/php-fig/log", + "keywords": [ + "log", + "psr", + "psr-3" + ], + "time": "2021-07-14T16:46:02+00:00" + }, + { + "name": "textalk/websocket", + "version": "1.6.3", + "dist": { + "type": "zip", + "url": "https://mirrors.cloud.tencent.com/repository/composer/textalk/websocket/1.6.3/textalk-websocket-1.6.3.zip", + "reference": "67de79745b1a357caf812bfc44e0abf481cee012", + "shasum": "" + }, + "require": { + "php": "^7.4 | ^8.0", + "phrity/net-uri": "^1.0", + "phrity/util-errorhandler": "^1.0", + "psr/http-message": "^1.0", + "psr/log": "^1.0 | ^2.0 | ^3.0" + }, + "require-dev": { + "php-coveralls/php-coveralls": "^2.0", + "phpunit/phpunit": "^9.0", + "squizlabs/php_codesniffer": "^3.5" + }, + "type": "library", + "autoload": { + "psr-4": { + "WebSocket\\": "lib" + } + }, + "license": [ + "ISC" + ], + "authors": [ + { + "name": "Fredrik Liljegren" + }, + { + "name": "Sören Jensen" + } + ], + "description": "WebSocket client and server", + "time": "2022-11-07T18:59:33+00:00" + }, + { + "name": "workerman/webman-framework", + "version": "v1.5.9", + "dist": { + "type": "zip", + "url": "https://mirrors.cloud.tencent.com/repository/composer/workerman/webman-framework/v1.5.9/workerman-webman-framework-v1.5.9.zip", + "reference": "6c43faa4f8e3ec61b8d97ba9e8ff53965b9a8b69", + "shasum": "" + }, + "require": { + "ext-json": "*", + "nikic/fast-route": "^1.3", + "php": ">=7.2", + "psr/container": ">=1.0", + "workerman/workerman": "^4.0.4 || ^5.0.0" + }, + "suggest": { + "ext-event": "For better performance. " + }, + "type": "library", + "autoload": { + "psr-4": { + "Webman\\": "./src", + "Support\\": "./src/support", + "support\\": "./src/support", + "Support\\View\\": "./src/support/view", + "Support\\Bootstrap\\": "./src/support/bootstrap", + "Support\\Exception\\": "./src/support/exception" + } + }, + "license": [ + "MIT" + ], + "authors": [ + { + "name": "walkor", + "email": "walkor@workerman.net", + "homepage": "https://www.workerman.net", + "role": "Developer" + } + ], + "description": "High performance HTTP Service Framework.", + "homepage": "https://www.workerman.net", + "keywords": [ + "High Performance", + "http service" + ], + "time": "2023-09-11T07:55:50+00:00" + }, + { + "name": "workerman/workerman", + "version": "v4.1.13", + "dist": { + "type": "zip", + "url": "https://mirrors.tencent.com/repository/composer/workerman/workerman/v4.1.13/workerman-workerman-v4.1.13.zip", + "reference": "807780ff672775fcd08f89e573a2824e939021ce", + "shasum": "" + }, + "require": { + "php": ">=7.0" + }, + "suggest": { + "ext-event": "For better performance. " + }, + "type": "library", + "autoload": { + "psr-4": { + "Workerman\\": "./" + } + }, + "license": [ + "MIT" + ], + "authors": [ + { + "name": "walkor", + "email": "walkor@workerman.net", + "homepage": "http://www.workerman.net", + "role": "Developer" + } + ], + "description": "An asynchronous event driven PHP framework for easily building fast, scalable network applications.", + "homepage": "http://www.workerman.net", + "keywords": [ + "asynchronous", + "event-loop" + ], + "time": "2023-07-31T05:57:25+00:00" + } + ], + "packages-dev": [], + "aliases": [], + "minimum-stability": "stable", + "stability-flags": [], + "prefer-stable": false, + "prefer-lowest": false, + "platform": { + "php": ">=7.2" + }, + "platform-dev": [], + "plugin-api-version": "2.3.0" +} diff --git a/composer.phar b/composer.phar new file mode 100755 index 0000000..d39c3e6 Binary files /dev/null and b/composer.phar differ diff --git a/config/app.php b/config/app.php new file mode 100644 index 0000000..f26e358 --- /dev/null +++ b/config/app.php @@ -0,0 +1,26 @@ + + * @copyright walkor + * @link http://www.workerman.net/ + * @license http://www.opensource.org/licenses/mit-license.php MIT License + */ + +use support\Request; + +return [ + 'debug' => true, + 'error_reporting' => E_ALL, + 'default_timezone' => 'Asia/Shanghai', + 'request_class' => Request::class, + 'public_path' => base_path() . DIRECTORY_SEPARATOR . 'public', + 'runtime_path' => base_path(false) . DIRECTORY_SEPARATOR . 'runtime', + 'controller_suffix' => 'Controller', + 'controller_reuse' => false, +]; diff --git a/config/autoload.php b/config/autoload.php new file mode 100644 index 0000000..69a8135 --- /dev/null +++ b/config/autoload.php @@ -0,0 +1,21 @@ + + * @copyright walkor + * @link http://www.workerman.net/ + * @license http://www.opensource.org/licenses/mit-license.php MIT License + */ + +return [ + 'files' => [ + base_path() . '/app/functions.php', + base_path() . '/support/Request.php', + base_path() . '/support/Response.php', + ] +]; diff --git a/config/bootstrap.php b/config/bootstrap.php new file mode 100644 index 0000000..44054e0 --- /dev/null +++ b/config/bootstrap.php @@ -0,0 +1,18 @@ + + * @copyright walkor + * @link http://www.workerman.net/ + * @license http://www.opensource.org/licenses/mit-license.php MIT License + */ + +return [ + support\bootstrap\Session::class, + support\bootstrap\LaravelDb::class, +]; diff --git a/config/container.php b/config/container.php new file mode 100644 index 0000000..106b7b4 --- /dev/null +++ b/config/container.php @@ -0,0 +1,15 @@ + + * @copyright walkor + * @link http://www.workerman.net/ + * @license http://www.opensource.org/licenses/mit-license.php MIT License + */ + +return new Webman\Container; \ No newline at end of file diff --git a/config/database.php b/config/database.php new file mode 100644 index 0000000..7dc463a --- /dev/null +++ b/config/database.php @@ -0,0 +1,15 @@ + + * @copyright walkor + * @link http://www.workerman.net/ + * @license http://www.opensource.org/licenses/mit-license.php MIT License + */ + +return []; diff --git a/config/dependence.php b/config/dependence.php new file mode 100644 index 0000000..8e964ed --- /dev/null +++ b/config/dependence.php @@ -0,0 +1,15 @@ + + * @copyright walkor + * @link http://www.workerman.net/ + * @license http://www.opensource.org/licenses/mit-license.php MIT License + */ + +return []; \ No newline at end of file diff --git a/config/exception.php b/config/exception.php new file mode 100644 index 0000000..f2aede3 --- /dev/null +++ b/config/exception.php @@ -0,0 +1,17 @@ + + * @copyright walkor + * @link http://www.workerman.net/ + * @license http://www.opensource.org/licenses/mit-license.php MIT License + */ + +return [ + '' => support\exception\Handler::class, +]; \ No newline at end of file diff --git a/config/log.php b/config/log.php new file mode 100644 index 0000000..7f05de5 --- /dev/null +++ b/config/log.php @@ -0,0 +1,32 @@ + + * @copyright walkor + * @link http://www.workerman.net/ + * @license http://www.opensource.org/licenses/mit-license.php MIT License + */ + +return [ + 'default' => [ + 'handlers' => [ + [ + 'class' => Monolog\Handler\RotatingFileHandler::class, + 'constructor' => [ + runtime_path() . '/logs/webman.log', + 7, //$maxFiles + Monolog\Logger::DEBUG, + ], + 'formatter' => [ + 'class' => Monolog\Formatter\LineFormatter::class, + 'constructor' => [null, 'Y-m-d H:i:s', true], + ], + ] + ], + ], +]; diff --git a/config/middleware.php b/config/middleware.php new file mode 100644 index 0000000..8e964ed --- /dev/null +++ b/config/middleware.php @@ -0,0 +1,15 @@ + + * @copyright walkor + * @link http://www.workerman.net/ + * @license http://www.opensource.org/licenses/mit-license.php MIT License + */ + +return []; \ No newline at end of file diff --git a/config/process.php b/config/process.php new file mode 100644 index 0000000..781dd0a --- /dev/null +++ b/config/process.php @@ -0,0 +1,49 @@ + + * @copyright walkor + * @link http://www.workerman.net/ + * @license http://www.opensource.org/licenses/mit-license.php MIT License + */ + +global $argv; + +return [ + // File update detection and automatic reload + 'monitor' => [ + 'handler' => process\Monitor::class, + 'reloadable' => false, + 'constructor' => [ + // Monitor these directories + 'monitorDir' => array_merge([ + app_path(), + config_path(), + base_path() . '/process', + base_path() . '/support', + base_path() . '/resource', + base_path() . '/.env', + ], glob(base_path() . '/plugin/*/app'), glob(base_path() . '/plugin/*/config'), glob(base_path() . '/plugin/*/api')), + // Files with these suffixes will be monitored + 'monitorExtensions' => [ + 'php', 'html', 'htm', 'env' + ], + 'options' => [ + 'enable_file_monitor' => !in_array('-d', $argv) && DIRECTORY_SEPARATOR === '/', + 'enable_memory_monitor' => DIRECTORY_SEPARATOR === '/', + ] + ] + ], + 'websocket_test' => [ + // 这里指定进程类,就是上面定义的Pusher类 + 'handler' => app\Pusher::class, + 'listen' => 'websocket://0.0.0.0:8888', + 'count' => 1, + ], +]; diff --git a/config/redis.php b/config/redis.php new file mode 100644 index 0000000..2f9757a --- /dev/null +++ b/config/redis.php @@ -0,0 +1,22 @@ + + * @copyright walkor + * @link http://www.workerman.net/ + * @license http://www.opensource.org/licenses/mit-license.php MIT License + */ + +return [ + 'default' => [ + 'host' => '127.0.0.1', + 'password' => null, + 'port' => 6379, + 'database' => 0, + ], +]; diff --git a/config/route.php b/config/route.php new file mode 100644 index 0000000..a5064fc --- /dev/null +++ b/config/route.php @@ -0,0 +1,21 @@ + + * @copyright walkor + * @link http://www.workerman.net/ + * @license http://www.opensource.org/licenses/mit-license.php MIT License + */ + +use Webman\Route; + + + + + + diff --git a/config/server.php b/config/server.php new file mode 100644 index 0000000..f55ce3c --- /dev/null +++ b/config/server.php @@ -0,0 +1,31 @@ + + * @copyright walkor + * @link http://www.workerman.net/ + * @license http://www.opensource.org/licenses/mit-license.php MIT License + */ + +return [ + 'listen' => 'http://0.0.0.0:8787', + 'transport' => 'tcp', + 'context' => [], + 'name' => 'webman', + 'count' => cpu_count() * 4, + 'user' => '', + 'group' => '', + 'reusePort' => false, + 'event_loop' => '', + 'stop_timeout' => 2, + 'pid_file' => runtime_path() . '/webman.pid', + 'status_file' => runtime_path() . '/webman.status', + 'stdout_file' => runtime_path() . '/logs/stdout.log', + 'log_file' => runtime_path() . '/logs/workerman.log', + 'max_package_size' => 10 * 1024 * 1024 +]; diff --git a/config/session.php b/config/session.php new file mode 100644 index 0000000..043f8c4 --- /dev/null +++ b/config/session.php @@ -0,0 +1,65 @@ + + * @copyright walkor + * @link http://www.workerman.net/ + * @license http://www.opensource.org/licenses/mit-license.php MIT License + */ + +use Webman\Session\FileSessionHandler; +use Webman\Session\RedisSessionHandler; +use Webman\Session\RedisClusterSessionHandler; + +return [ + + 'type' => 'file', // or redis or redis_cluster + + 'handler' => FileSessionHandler::class, + + 'config' => [ + 'file' => [ + 'save_path' => runtime_path() . '/sessions', + ], + 'redis' => [ + 'host' => '127.0.0.1', + 'port' => 6379, + 'auth' => '', + 'timeout' => 2, + 'database' => '', + 'prefix' => 'redis_session_', + ], + 'redis_cluster' => [ + 'host' => ['127.0.0.1:7000', '127.0.0.1:7001', '127.0.0.1:7001'], + 'timeout' => 2, + 'auth' => '', + 'prefix' => 'redis_session_', + ] + ], + + 'session_name' => 'PHPSID', + + 'auto_update_timestamp' => false, + + 'lifetime' => 7*24*60*60, + + 'cookie_lifetime' => 365*24*60*60, + + 'cookie_path' => '/', + + 'domain' => '', + + 'http_only' => true, + + 'secure' => false, + + 'same_site' => '', + + 'gc_probability' => [1, 1000], + +]; diff --git a/config/static.php b/config/static.php new file mode 100644 index 0000000..2f76cf3 --- /dev/null +++ b/config/static.php @@ -0,0 +1,23 @@ + + * @copyright walkor + * @link http://www.workerman.net/ + * @license http://www.opensource.org/licenses/mit-license.php MIT License + */ + +/** + * Static file settings + */ +return [ + 'enable' => true, + 'middleware' => [ // Static file Middleware + //app\middleware\StaticFile::class, + ], +]; \ No newline at end of file diff --git a/config/translation.php b/config/translation.php new file mode 100644 index 0000000..96589b2 --- /dev/null +++ b/config/translation.php @@ -0,0 +1,25 @@ + + * @copyright walkor + * @link http://www.workerman.net/ + * @license http://www.opensource.org/licenses/mit-license.php MIT License + */ + +/** + * Multilingual configuration + */ +return [ + // Default language + 'locale' => 'zh_CN', + // Fallback language + 'fallback_locale' => ['zh_CN', 'en'], + // Folder where language files are stored + 'path' => base_path() . '/resource/translations', +]; \ No newline at end of file diff --git a/config/view.php b/config/view.php new file mode 100644 index 0000000..e3a7b85 --- /dev/null +++ b/config/view.php @@ -0,0 +1,22 @@ + + * @copyright walkor + * @link http://www.workerman.net/ + * @license http://www.opensource.org/licenses/mit-license.php MIT License + */ + +use support\view\Raw; +use support\view\Twig; +use support\view\Blade; +use support\view\ThinkPHP; + +return [ + 'handler' => Raw::class +]; diff --git a/process/Monitor.php b/process/Monitor.php new file mode 100644 index 0000000..92b3716 --- /dev/null +++ b/process/Monitor.php @@ -0,0 +1,243 @@ + + * @copyright walkor + * @link http://www.workerman.net/ + * @license http://www.opensource.org/licenses/mit-license.php MIT License + */ + +namespace process; + +use FilesystemIterator; +use RecursiveDirectoryIterator; +use RecursiveIteratorIterator; +use SplFileInfo; +use Workerman\Timer; +use Workerman\Worker; + +/** + * Class FileMonitor + * @package process + */ +class Monitor +{ + /** + * @var array + */ + protected $paths = []; + + /** + * @var array + */ + protected $extensions = []; + + /** + * @var string + */ + public static $lockFile = __DIR__ . '/../runtime/monitor.lock'; + + /** + * Pause monitor + * @return void + */ + public static function pause() + { + file_put_contents(static::$lockFile, time()); + } + + /** + * Resume monitor + * @return void + */ + public static function resume(): void + { + clearstatcache(); + if (is_file(static::$lockFile)) { + unlink(static::$lockFile); + } + } + + /** + * Whether monitor is paused + * @return bool + */ + public static function isPaused(): bool + { + clearstatcache(); + return file_exists(static::$lockFile); + } + + /** + * FileMonitor constructor. + * @param $monitorDir + * @param $monitorExtensions + * @param array $options + */ + public function __construct($monitorDir, $monitorExtensions, array $options = []) + { + static::resume(); + $this->paths = (array)$monitorDir; + $this->extensions = $monitorExtensions; + if (!Worker::getAllWorkers()) { + return; + } + $disableFunctions = explode(',', ini_get('disable_functions')); + if (in_array('exec', $disableFunctions, true)) { + echo "\nMonitor file change turned off because exec() has been disabled by disable_functions setting in " . PHP_CONFIG_FILE_PATH . "/php.ini\n"; + } else { + if ($options['enable_file_monitor'] ?? true) { + Timer::add(1, function () { + $this->checkAllFilesChange(); + }); + } + } + + $memoryLimit = $this->getMemoryLimit($options['memory_limit'] ?? null); + if ($memoryLimit && ($options['enable_memory_monitor'] ?? true)) { + Timer::add(60, [$this, 'checkMemory'], [$memoryLimit]); + } + } + + /** + * @param $monitorDir + * @return bool + */ + public function checkFilesChange($monitorDir): bool + { + static $lastMtime, $tooManyFilesCheck; + if (!$lastMtime) { + $lastMtime = time(); + } + clearstatcache(); + if (!is_dir($monitorDir)) { + if (!is_file($monitorDir)) { + return false; + } + $iterator = [new SplFileInfo($monitorDir)]; + } else { + // recursive traversal directory + $dirIterator = new RecursiveDirectoryIterator($monitorDir, FilesystemIterator::SKIP_DOTS | FilesystemIterator::FOLLOW_SYMLINKS); + $iterator = new RecursiveIteratorIterator($dirIterator); + } + $count = 0; + foreach ($iterator as $file) { + $count ++; + /** var SplFileInfo $file */ + if (is_dir($file->getRealPath())) { + continue; + } + // check mtime + if (in_array($file->getExtension(), $this->extensions, true) && $lastMtime < $file->getMTime()) { + $var = 0; + exec('"'.PHP_BINARY . '" -l ' . $file, $out, $var); + $lastMtime = $file->getMTime(); + if ($var) { + continue; + } + echo $file . " update and reload\n"; + // send SIGUSR1 signal to master process for reload + if (DIRECTORY_SEPARATOR === '/') { + posix_kill(posix_getppid(), SIGUSR1); + } else { + return true; + } + break; + } + } + if (!$tooManyFilesCheck && $count > 1000) { + echo "Monitor: There are too many files ($count files) in $monitorDir which makes file monitoring very slow\n"; + $tooManyFilesCheck = 1; + } + return false; + } + + /** + * @return bool + */ + public function checkAllFilesChange(): bool + { + if (static::isPaused()) { + return false; + } + foreach ($this->paths as $path) { + if ($this->checkFilesChange($path)) { + return true; + } + } + return false; + } + + /** + * @param $memoryLimit + * @return void + */ + public function checkMemory($memoryLimit) + { + if (static::isPaused() || $memoryLimit <= 0) { + return; + } + $ppid = posix_getppid(); + $childrenFile = "/proc/$ppid/task/$ppid/children"; + if (!is_file($childrenFile) || !($children = file_get_contents($childrenFile))) { + return; + } + foreach (explode(' ', $children) as $pid) { + $pid = (int)$pid; + $statusFile = "/proc/$pid/status"; + if (!is_file($statusFile) || !($status = file_get_contents($statusFile))) { + continue; + } + $mem = 0; + if (preg_match('/VmRSS\s*?:\s*?(\d+?)\s*?kB/', $status, $match)) { + $mem = $match[1]; + } + $mem = (int)($mem / 1024); + if ($mem >= $memoryLimit) { + posix_kill($pid, SIGINT); + } + } + } + + /** + * Get memory limit + * @return float + */ + protected function getMemoryLimit($memoryLimit) + { + if ($memoryLimit === 0) { + return 0; + } + $usePhpIni = false; + if (!$memoryLimit) { + $memoryLimit = ini_get('memory_limit'); + $usePhpIni = true; + } + + if ($memoryLimit == -1) { + return 0; + } + $unit = strtolower($memoryLimit[strlen($memoryLimit) - 1]); + if ($unit === 'g') { + $memoryLimit = 1024 * (int)$memoryLimit; + } else if ($unit === 'm') { + $memoryLimit = (int)$memoryLimit; + } else if ($unit === 'k') { + $memoryLimit = ((int)$memoryLimit / 1024); + } else { + $memoryLimit = ((int)$memoryLimit / (1024 * 1024)); + } + if ($memoryLimit < 30) { + $memoryLimit = 30; + } + if ($usePhpIni) { + $memoryLimit = (int)(0.8 * $memoryLimit); + } + return $memoryLimit; + } +} diff --git a/public/404.html b/public/404.html new file mode 100644 index 0000000..2bde119 --- /dev/null +++ b/public/404.html @@ -0,0 +1,12 @@ + + + 404 Not Found - webman + + +
+

404 Not Found

+
+
+
webman
+ + diff --git a/public/favicon.ico b/public/favicon.ico new file mode 100644 index 0000000..b9f722e Binary files /dev/null and b/public/favicon.ico differ diff --git a/runtime/.gitignore b/runtime/.gitignore new file mode 100644 index 0000000..1283e48 --- /dev/null +++ b/runtime/.gitignore @@ -0,0 +1,4 @@ +* +!logs +!views +!.gitignore diff --git a/runtime/logs/.gitignore b/runtime/logs/.gitignore new file mode 100644 index 0000000..c96a04f --- /dev/null +++ b/runtime/logs/.gitignore @@ -0,0 +1,2 @@ +* +!.gitignore \ No newline at end of file diff --git a/runtime/views/.gitignore b/runtime/views/.gitignore new file mode 100644 index 0000000..d6b7ef3 --- /dev/null +++ b/runtime/views/.gitignore @@ -0,0 +1,2 @@ +* +!.gitignore diff --git a/start.php b/start.php new file mode 100755 index 0000000..489e447 --- /dev/null +++ b/start.php @@ -0,0 +1,4 @@ +#!/usr/bin/env php + + * @copyright walkor + * @link http://www.workerman.net/ + * @license http://www.opensource.org/licenses/mit-license.php MIT License + */ + +namespace support; + +/** + * Class Request + * @package support + */ +class Request extends \Webman\Http\Request +{ + +} \ No newline at end of file diff --git a/support/Response.php b/support/Response.php new file mode 100644 index 0000000..9bc4e1e --- /dev/null +++ b/support/Response.php @@ -0,0 +1,24 @@ + + * @copyright walkor + * @link http://www.workerman.net/ + * @license http://www.opensource.org/licenses/mit-license.php MIT License + */ + +namespace support; + +/** + * Class Response + * @package support + */ +class Response extends \Webman\Http\Response +{ + +} \ No newline at end of file diff --git a/support/bootstrap.php b/support/bootstrap.php new file mode 100644 index 0000000..d9471e6 --- /dev/null +++ b/support/bootstrap.php @@ -0,0 +1,133 @@ + + * @copyright walkor + * @link http://www.workerman.net/ + * @license http://www.opensource.org/licenses/mit-license.php MIT License + */ + +use Dotenv\Dotenv; +use support\Log; +use Webman\Bootstrap; +use Webman\Config; +use Webman\Middleware; +use Webman\Route; +use Webman\Util; + +$worker = $worker ?? null; + +set_error_handler(function ($level, $message, $file = '', $line = 0) { + if (error_reporting() & $level) { + throw new ErrorException($message, 0, $level, $file, $line); + } +}); + +if ($worker) { + register_shutdown_function(function ($startTime) { + if (time() - $startTime <= 0.1) { + sleep(1); + } + }, time()); +} + +if (class_exists('Dotenv\Dotenv') && file_exists(base_path(false) . '/.env')) { + if (method_exists('Dotenv\Dotenv', 'createUnsafeMutable')) { + Dotenv::createUnsafeMutable(base_path(false))->load(); + } else { + Dotenv::createMutable(base_path(false))->load(); + } +} + +Config::clear(); +support\App::loadAllConfig(['route']); +if ($timezone = config('app.default_timezone')) { + date_default_timezone_set($timezone); +} + +foreach (config('autoload.files', []) as $file) { + include_once $file; +} +foreach (config('plugin', []) as $firm => $projects) { + foreach ($projects as $name => $project) { + if (!is_array($project)) { + continue; + } + foreach ($project['autoload']['files'] ?? [] as $file) { + include_once $file; + } + } + foreach ($projects['autoload']['files'] ?? [] as $file) { + include_once $file; + } +} + +Middleware::load(config('middleware', [])); +foreach (config('plugin', []) as $firm => $projects) { + foreach ($projects as $name => $project) { + if (!is_array($project) || $name === 'static') { + continue; + } + Middleware::load($project['middleware'] ?? []); + } + Middleware::load($projects['middleware'] ?? [], $firm); + if ($staticMiddlewares = config("plugin.$firm.static.middleware")) { + Middleware::load(['__static__' => $staticMiddlewares], $firm); + } +} +Middleware::load(['__static__' => config('static.middleware', [])]); + +foreach (config('bootstrap', []) as $className) { + if (!class_exists($className)) { + $log = "Warning: Class $className setting in config/bootstrap.php not found\r\n"; + echo $log; + Log::error($log); + continue; + } + /** @var Bootstrap $className */ + $className::start($worker); +} + +foreach (config('plugin', []) as $firm => $projects) { + foreach ($projects as $name => $project) { + if (!is_array($project)) { + continue; + } + foreach ($project['bootstrap'] ?? [] as $className) { + if (!class_exists($className)) { + $log = "Warning: Class $className setting in config/plugin/$firm/$name/bootstrap.php not found\r\n"; + echo $log; + Log::error($log); + continue; + } + /** @var Bootstrap $className */ + $className::start($worker); + } + } + foreach ($projects['bootstrap'] ?? [] as $className) { + /** @var string $className */ + if (!class_exists($className)) { + $log = "Warning: Class $className setting in plugin/$firm/config/bootstrap.php not found\r\n"; + echo $log; + Log::error($log); + continue; + } + /** @var Bootstrap $className */ + $className::start($worker); + } +} + +$directory = base_path() . '/plugin'; +$paths = [config_path()]; +foreach (Util::scanDir($directory) as $path) { + if (is_dir($path = "$path/config")) { + $paths[] = $path; + } +} +Route::load($paths); + diff --git a/support/helpers.php b/support/helpers.php new file mode 100644 index 0000000..e200040 --- /dev/null +++ b/support/helpers.php @@ -0,0 +1,517 @@ + + * @copyright walkor + * @link http://www.workerman.net/ + * @license http://www.opensource.org/licenses/mit-license.php MIT License + */ + +use support\Container; +use support\Request; +use support\Response; +use support\Translation; +use support\view\Blade; +use support\view\Raw; +use support\view\ThinkPHP; +use support\view\Twig; +use Twig\Error\LoaderError; +use Twig\Error\RuntimeError; +use Twig\Error\SyntaxError; +use Webman\App; +use Webman\Config; +use Webman\Route; +use Workerman\Protocols\Http\Session; +use Workerman\Worker; + +// Project base path +define('BASE_PATH', dirname(__DIR__)); + +/** + * return the program execute directory + * @param string $path + * @return string + */ +function run_path(string $path = ''): string +{ + static $runPath = ''; + if (!$runPath) { + $runPath = is_phar() ? dirname(Phar::running(false)) : BASE_PATH; + } + return path_combine($runPath, $path); +} + +/** + * if the param $path equal false,will return this program current execute directory + * @param string|false $path + * @return string + */ +function base_path($path = ''): string +{ + if (false === $path) { + return run_path(); + } + return path_combine(BASE_PATH, $path); +} + +/** + * App path + * @param string $path + * @return string + */ +function app_path(string $path = ''): string +{ + return path_combine(BASE_PATH . DIRECTORY_SEPARATOR . 'app', $path); +} + +/** + * Public path + * @param string $path + * @return string + */ +function public_path(string $path = ''): string +{ + static $publicPath = ''; + if (!$publicPath) { + $publicPath = \config('app.public_path') ?: run_path('public'); + } + return path_combine($publicPath, $path); +} + +/** + * Config path + * @param string $path + * @return string + */ +function config_path(string $path = ''): string +{ + return path_combine(BASE_PATH . DIRECTORY_SEPARATOR . 'config', $path); +} + +/** + * Runtime path + * @param string $path + * @return string + */ +function runtime_path(string $path = ''): string +{ + static $runtimePath = ''; + if (!$runtimePath) { + $runtimePath = \config('app.runtime_path') ?: run_path('runtime'); + } + return path_combine($runtimePath, $path); +} + +/** + * Generate paths based on given information + * @param string $front + * @param string $back + * @return string + */ +function path_combine(string $front, string $back): string +{ + return $front . ($back ? (DIRECTORY_SEPARATOR . ltrim($back, DIRECTORY_SEPARATOR)) : $back); +} + +/** + * Response + * @param int $status + * @param array $headers + * @param string $body + * @return Response + */ +function response(string $body = '', int $status = 200, array $headers = []): Response +{ + return new Response($status, $headers, $body); +} + +/** + * Json response + * @param $data + * @param int $options + * @return Response + */ +function json($data, int $options = JSON_UNESCAPED_UNICODE): Response +{ + return new Response(200, ['Content-Type' => 'application/json'], json_encode($data, $options)); +} + +/** + * Xml response + * @param $xml + * @return Response + */ +function xml($xml): Response +{ + if ($xml instanceof SimpleXMLElement) { + $xml = $xml->asXML(); + } + return new Response(200, ['Content-Type' => 'text/xml'], $xml); +} + +/** + * Jsonp response + * @param $data + * @param string $callbackName + * @return Response + */ +function jsonp($data, string $callbackName = 'callback'): Response +{ + if (!is_scalar($data) && null !== $data) { + $data = json_encode($data); + } + return new Response(200, [], "$callbackName($data)"); +} + +/** + * Redirect response + * @param string $location + * @param int $status + * @param array $headers + * @return Response + */ +function redirect(string $location, int $status = 302, array $headers = []): Response +{ + $response = new Response($status, ['Location' => $location]); + if (!empty($headers)) { + $response->withHeaders($headers); + } + return $response; +} + +/** + * View response + * @param string $template + * @param array $vars + * @param string|null $app + * @param string|null $plugin + * @return Response + */ +function view(string $template, array $vars = [], string $app = null, string $plugin = null): Response +{ + $request = \request(); + $plugin = $plugin === null ? ($request->plugin ?? '') : $plugin; + $handler = \config($plugin ? "plugin.$plugin.view.handler" : 'view.handler'); + return new Response(200, [], $handler::render($template, $vars, $app, $plugin)); +} + +/** + * Raw view response + * @param string $template + * @param array $vars + * @param string|null $app + * @return Response + * @throws Throwable + */ +function raw_view(string $template, array $vars = [], string $app = null): Response +{ + return new Response(200, [], Raw::render($template, $vars, $app)); +} + +/** + * Blade view response + * @param string $template + * @param array $vars + * @param string|null $app + * @return Response + */ +function blade_view(string $template, array $vars = [], string $app = null): Response +{ + return new Response(200, [], Blade::render($template, $vars, $app)); +} + +/** + * Think view response + * @param string $template + * @param array $vars + * @param string|null $app + * @return Response + */ +function think_view(string $template, array $vars = [], string $app = null): Response +{ + return new Response(200, [], ThinkPHP::render($template, $vars, $app)); +} + +/** + * Twig view response + * @param string $template + * @param array $vars + * @param string|null $app + * @return Response + * @throws LoaderError + * @throws RuntimeError + * @throws SyntaxError + */ +function twig_view(string $template, array $vars = [], string $app = null): Response +{ + return new Response(200, [], Twig::render($template, $vars, $app)); +} + +/** + * Get request + * @return \Webman\Http\Request|Request|null + */ +function request() +{ + return App::request(); +} + +/** + * Get config + * @param string|null $key + * @param $default + * @return array|mixed|null + */ +function config(string $key = null, $default = null) +{ + return Config::get($key, $default); +} + +/** + * Create url + * @param string $name + * @param ...$parameters + * @return string + */ +function route(string $name, ...$parameters): string +{ + $route = Route::getByName($name); + if (!$route) { + return ''; + } + + if (!$parameters) { + return $route->url(); + } + + if (is_array(current($parameters))) { + $parameters = current($parameters); + } + + return $route->url($parameters); +} + +/** + * Session + * @param mixed $key + * @param mixed $default + * @return mixed|bool|Session + */ +function session($key = null, $default = null) +{ + $session = \request()->session(); + if (null === $key) { + return $session; + } + if (is_array($key)) { + $session->put($key); + return null; + } + if (strpos($key, '.')) { + $keyArray = explode('.', $key); + $value = $session->all(); + foreach ($keyArray as $index) { + if (!isset($value[$index])) { + return $default; + } + $value = $value[$index]; + } + return $value; + } + return $session->get($key, $default); +} + +/** + * Translation + * @param string $id + * @param array $parameters + * @param string|null $domain + * @param string|null $locale + * @return string + */ +function trans(string $id, array $parameters = [], string $domain = null, string $locale = null): string +{ + $res = Translation::trans($id, $parameters, $domain, $locale); + return $res === '' ? $id : $res; +} + +/** + * Locale + * @param string|null $locale + * @return string + */ +function locale(string $locale = null): string +{ + if (!$locale) { + return Translation::getLocale(); + } + Translation::setLocale($locale); + return $locale; +} + +/** + * 404 not found + * @return Response + */ +function not_found(): Response +{ + return new Response(404, [], file_get_contents(public_path() . '/404.html')); +} + +/** + * Copy dir + * @param string $source + * @param string $dest + * @param bool $overwrite + * @return void + */ +function copy_dir(string $source, string $dest, bool $overwrite = false) +{ + if (is_dir($source)) { + if (!is_dir($dest)) { + mkdir($dest); + } + $files = scandir($source); + foreach ($files as $file) { + if ($file !== "." && $file !== "..") { + copy_dir("$source/$file", "$dest/$file"); + } + } + } else if (file_exists($source) && ($overwrite || !file_exists($dest))) { + copy($source, $dest); + } +} + +/** + * Remove dir + * @param string $dir + * @return bool + */ +function remove_dir(string $dir): bool +{ + if (is_link($dir) || is_file($dir)) { + return unlink($dir); + } + $files = array_diff(scandir($dir), array('.', '..')); + foreach ($files as $file) { + (is_dir("$dir/$file") && !is_link($dir)) ? remove_dir("$dir/$file") : unlink("$dir/$file"); + } + return rmdir($dir); +} + +/** + * Bind worker + * @param $worker + * @param $class + */ +function worker_bind($worker, $class) +{ + $callbackMap = [ + 'onConnect', + 'onMessage', + 'onClose', + 'onError', + 'onBufferFull', + 'onBufferDrain', + 'onWorkerStop', + 'onWebSocketConnect', + 'onWorkerReload' + ]; + foreach ($callbackMap as $name) { + if (method_exists($class, $name)) { + $worker->$name = [$class, $name]; + } + } + if (method_exists($class, 'onWorkerStart')) { + call_user_func([$class, 'onWorkerStart'], $worker); + } +} + +/** + * Start worker + * @param $processName + * @param $config + * @return void + */ +function worker_start($processName, $config) +{ + $worker = new Worker($config['listen'] ?? null, $config['context'] ?? []); + $propertyMap = [ + 'count', + 'user', + 'group', + 'reloadable', + 'reusePort', + 'transport', + 'protocol', + ]; + $worker->name = $processName; + foreach ($propertyMap as $property) { + if (isset($config[$property])) { + $worker->$property = $config[$property]; + } + } + + $worker->onWorkerStart = function ($worker) use ($config) { + require_once base_path('/support/bootstrap.php'); + if (isset($config['handler'])) { + if (!class_exists($config['handler'])) { + echo "process error: class {$config['handler']} not exists\r\n"; + return; + } + + $instance = Container::make($config['handler'], $config['constructor'] ?? []); + worker_bind($worker, $instance); + } + }; +} + +/** + * Get realpath + * @param string $filePath + * @return string + */ +function get_realpath(string $filePath): string +{ + if (strpos($filePath, 'phar://') === 0) { + return $filePath; + } else { + return realpath($filePath); + } +} + +/** + * Is phar + * @return bool + */ +function is_phar(): bool +{ + return class_exists(Phar::class, false) && Phar::running(); +} + +/** + * Get cpu count + * @return int + */ +function cpu_count(): int +{ + // Windows does not support the number of processes setting. + if (DIRECTORY_SEPARATOR === '\\') { + return 1; + } + $count = 4; + if (is_callable('shell_exec')) { + if (strtolower(PHP_OS) === 'darwin') { + $count = (int)shell_exec('sysctl -n machdep.cpu.core_count'); + } else { + $count = (int)shell_exec('nproc'); + } + } + return $count > 0 ? $count : 4; +} diff --git a/windows.bat b/windows.bat new file mode 100644 index 0000000..f07ce53 --- /dev/null +++ b/windows.bat @@ -0,0 +1,3 @@ +CHCP 65001 +php windows.php +pause \ No newline at end of file diff --git a/windows.php b/windows.php new file mode 100644 index 0000000..d1975d6 --- /dev/null +++ b/windows.php @@ -0,0 +1,116 @@ +load(); + } else { + Dotenv::createMutable(base_path())->load(); + } +} + +App::loadAllConfig(['route']); + +$errorReporting = config('app.error_reporting'); +if (isset($errorReporting)) { + error_reporting($errorReporting); +} + +$runtimeProcessPath = runtime_path() . DIRECTORY_SEPARATOR . '/windows'; +if (!is_dir($runtimeProcessPath)) { + mkdir($runtimeProcessPath); +} +$processFiles = [ + __DIR__ . DIRECTORY_SEPARATOR . 'start.php' +]; +foreach (config('process', []) as $processName => $config) { + $processFiles[] = write_process_file($runtimeProcessPath, $processName, ''); +} + +foreach (config('plugin', []) as $firm => $projects) { + foreach ($projects as $name => $project) { + if (!is_array($project)) { + continue; + } + foreach ($project['process'] ?? [] as $processName => $config) { + $processFiles[] = write_process_file($runtimeProcessPath, $processName, "$firm.$name"); + } + } + foreach ($projects['process'] ?? [] as $processName => $config) { + $processFiles[] = write_process_file($runtimeProcessPath, $processName, $firm); + } +} + +function write_process_file($runtimeProcessPath, $processName, $firm): string +{ + $processParam = $firm ? "plugin.$firm.$processName" : $processName; + $configParam = $firm ? "config('plugin.$firm.process')['$processName']" : "config('process')['$processName']"; + $fileContent = << true]); + if (!$resource) { + exit("Can not execute $cmd\r\n"); + } + return $resource; +} + +$resource = popen_processes($processFiles); +echo "\r\n"; +while (1) { + sleep(1); + if (!empty($monitor) && $monitor->checkAllFilesChange()) { + $status = proc_get_status($resource); + $pid = $status['pid']; + shell_exec("taskkill /F /T /PID $pid"); + proc_close($resource); + $resource = popen_processes($processFiles); + } +}