Compare commits
97 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| b96bd159d3 | |||
| c31211a6fa | |||
| 45d8931267 | |||
| 7bcb789ed2 | |||
| 1d9d761798 | |||
| 22f425ceb7 | |||
| a52610cf2a | |||
| 5697102a85 | |||
| 0c0a2fadd5 | |||
| 5061a17342 | |||
| 9c1bc63f78 | |||
| 345af9e727 | |||
| 196db497c1 | |||
| 6da3575619 | |||
| 9814e06e8d | |||
| df32502723 | |||
| 7f702e5784 | |||
| fd61b6f1d9 | |||
| 9e062a7a33 | |||
| 43c8f9b8e6 | |||
| 7b14fa075e | |||
| 165f62113c | |||
| 3c3b6f2e65 | |||
| 2ca20d3a96 | |||
| b2800b806c | |||
| 092daa022a | |||
| 57337cbe16 | |||
| ad8b20723b | |||
| f2421b71e7 | |||
| 02685a9e61 | |||
| 1b5232f5e7 | |||
| c76fb40689 | |||
| 18b44b32f2 | |||
| 14574e7a6c | |||
| 13fc812ae4 | |||
| edd1751f7a | |||
| 7dbce8b1e8 | |||
| eb7ffecdb9 | |||
| dd85596907 | |||
| e6b10bbd70 | |||
| c149afa3f8 | |||
| f195d001c6 | |||
| b45666d9db | |||
| 650aaee974 | |||
| d589e096af | |||
| f17a55f8d0 | |||
| 3cc98214c7 | |||
| aaeb538849 | |||
| 0c7b8eb57f | |||
| 48808ded98 | |||
| eb25fe06ed | |||
| 4d51f6e5d6 | |||
| 21fbbe52e5 | |||
| a0a753d8b4 | |||
| dcc5e67f0a | |||
| b8e7a5f086 | |||
| 88a414f0bb | |||
| cba27222fa | |||
| 593f05754d | |||
| 044c5c4a28 | |||
| 121642e825 | |||
| 5eecf66b43 | |||
| a88787a530 | |||
| aed0e61531 | |||
| c94883375a | |||
| 6565f1d7a3 | |||
| cefe7ae010 | |||
| 56f42ac97b | |||
| 9c7355d7f6 | |||
| 2ff2e3cdb8 | |||
| 18c6f9594e | |||
| a5c2467cdf | |||
| 698868b6ee | |||
| cbee38ae54 | |||
| e19536ce99 | |||
| b92ab0bf1d | |||
| bc049a3c77 | |||
| a026a610a1 | |||
| e71a17cbce | |||
| 86c09f3b25 | |||
| 788ecee65d | |||
| 98cbb07845 | |||
| 93bab6d5eb | |||
| 99f79b7616 | |||
| 47dc988e49 | |||
| 072de12cc1 | |||
| 77f755d07c | |||
| bc5f9b1085 | |||
| 4660025775 | |||
| 336cc875a1 | |||
| d943b2ebc2 | |||
| f982b51ff6 | |||
| 5b7a33f48b | |||
| 753e521a41 | |||
| d479f36662 | |||
| 6984f78746 | |||
| 030f337e74 |
+37
@@ -0,0 +1,37 @@
|
||||
# Created by .ignore support plugin (hsz.mobi)
|
||||
### Yii template
|
||||
assets/*
|
||||
!assets/.gitignore
|
||||
protected/runtime/*
|
||||
!protected/runtime/.gitignore
|
||||
protected/data/*.db
|
||||
themes/classic/views/
|
||||
|
||||
### Example user template template
|
||||
### Example user template
|
||||
|
||||
# IntelliJ project files
|
||||
.idea
|
||||
*.iml
|
||||
out
|
||||
gen
|
||||
|
||||
db/
|
||||
async-queue/
|
||||
|
||||
composer.lock
|
||||
|
||||
*.log
|
||||
commands/result
|
||||
config/setting.php
|
||||
tests/
|
||||
vendor/*
|
||||
runtime/
|
||||
|
||||
*.xml
|
||||
*.lock
|
||||
|
||||
oot
|
||||
d
|
||||
|
||||
composer.lock
|
||||
-177
@@ -1,177 +0,0 @@
|
||||
<?php
|
||||
/**
|
||||
* Created by PhpStorm.
|
||||
* User: whwyy
|
||||
* Date: 2018/5/24 0024
|
||||
* Time: 11:34
|
||||
*/
|
||||
declare(strict_types=1);
|
||||
|
||||
namespace Kiri;
|
||||
|
||||
use Exception;
|
||||
use Kiri\Message\Stream;
|
||||
use Kiri\Abstracts\Logger;
|
||||
use Swoole\Client as SwowClient;
|
||||
|
||||
/**
|
||||
* Class Client
|
||||
* @package Kiri\Kiri\Http
|
||||
*/
|
||||
class AsyncClient extends ClientAbstracts
|
||||
{
|
||||
|
||||
|
||||
use TSwooleClient;
|
||||
|
||||
/**
|
||||
* @param string $method
|
||||
* @param $path
|
||||
* @param array $params
|
||||
* @return void
|
||||
* @throws Exception
|
||||
*/
|
||||
public function request(string $method, $path, array $params = []): void
|
||||
{
|
||||
$this->withMethod($method)
|
||||
->coroutine(
|
||||
$this->matchHost($path),
|
||||
$this->paramEncode($params)
|
||||
);
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* @param $path
|
||||
* @return $this
|
||||
*/
|
||||
public function withCAInfo($path): static
|
||||
{
|
||||
return $this;
|
||||
}
|
||||
|
||||
/**
|
||||
* @param $url
|
||||
* @param array|string $data
|
||||
* @throws Exception 使用swoole协程方式请求
|
||||
*/
|
||||
private function coroutine($url, array|string $data = []): void
|
||||
{
|
||||
try {
|
||||
$this->generate_client($data, ...$url);
|
||||
} catch (\Throwable $exception) {
|
||||
Kiri::getDi()->get(Logger::class)->error('rpc', [$exception]);
|
||||
$this->setStatusCode(-1);
|
||||
$this->setBody(jTraceEx($exception));
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* @param $data
|
||||
* @param $host
|
||||
* @param $isHttps
|
||||
* @param $path
|
||||
* @throws Exception
|
||||
*/
|
||||
private function generate_client($data, $host, $isHttps, $path): void
|
||||
{
|
||||
$this->client = new SwowClient(SWOOLE_TCP, FALSE);
|
||||
if (!$this->client->connect($host, $this->getPort())) {
|
||||
throw new Exception('链接失败');
|
||||
}
|
||||
if ($isHttps || $this->isSSL()) {
|
||||
$this->client->enableSSL();
|
||||
}
|
||||
$this->client->set(array_merge($this->settings(), ['open_http_protocol' => true]));
|
||||
if (!empty($this->getAgent())) {
|
||||
$this->withAddedHeader('User-Agent', $this->getAgent());
|
||||
}
|
||||
|
||||
$path = $this->setParams($path, $data);
|
||||
|
||||
$this->withAddedHeader('Accept', ' text/html,application/xhtml+xml,application/json,application/xml;q=0.9,image/avif,image/webp,image/apng,*/*;q=0.8,application/signed-exchange;v=b3;q=0.9');
|
||||
// $this->withAddedHeader('Accept-Encoding', 'gzip');
|
||||
$this->withAddedHeader('Content-Length', $this->getData()->getSize());
|
||||
|
||||
$this->execute($path, $this->getData()->getContents());
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* @param string $path
|
||||
* @param string $content
|
||||
* @return void
|
||||
*/
|
||||
private function execute(string $path, string $content)
|
||||
{
|
||||
$array = [];
|
||||
$array[] = strtoupper($this->getMethod()) . ' ' . $path . ' HTTP/1.1';
|
||||
if (!empty($this->getHeader())) {
|
||||
foreach ($this->getHeader() as $key => $value) {
|
||||
$array[] = sprintf('%s: %s', $key, $value);
|
||||
}
|
||||
}
|
||||
$this->client->send(implode("\r\n", $array) . "\r\n\r\n" . $content);
|
||||
$receive = $this->client->recv();
|
||||
|
||||
[$header, $body] = explode("\r\n\r\n", $receive);
|
||||
|
||||
$header = explode("\r\n", $header);
|
||||
$status = array_shift($header);
|
||||
|
||||
$this->setStatusCode(intval(explode(' ', $status)[1]));
|
||||
$this->parseResponseHeaders($header);
|
||||
$this->setBody($body);
|
||||
}
|
||||
|
||||
|
||||
private function chunked()
|
||||
{
|
||||
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* @param array $headers
|
||||
* @return void
|
||||
*/
|
||||
private function parseResponseHeaders(array $headers)
|
||||
{
|
||||
$array = [];
|
||||
foreach ($headers as $header) {
|
||||
[$key, $value] = explode(': ', $header);
|
||||
|
||||
$array[$key] = trim($value);
|
||||
}
|
||||
$this->setResponseHeader($array);
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* @param $path
|
||||
* @param $data
|
||||
* @return string
|
||||
*/
|
||||
private function setParams($path, $data): string
|
||||
{
|
||||
if ($this->isGet()) {
|
||||
if (!empty($data)) $path .= '?' . $data;
|
||||
} else {
|
||||
$data = $this->mergeParams($data);
|
||||
if (!empty($data)) {
|
||||
$this->withBody(new Stream($data));
|
||||
}
|
||||
}
|
||||
return $path;
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
*
|
||||
*/
|
||||
public function close(): void
|
||||
{
|
||||
$this->client->close();
|
||||
}
|
||||
}
|
||||
+8
-8
@@ -2,7 +2,7 @@
|
||||
|
||||
namespace Kiri;
|
||||
|
||||
use Kiri\Context;
|
||||
use Swoole\Coroutine;
|
||||
|
||||
|
||||
/**
|
||||
@@ -12,17 +12,17 @@ class Client
|
||||
{
|
||||
|
||||
|
||||
private CoroutineClient|CurlClient|AsyncClient $abstracts;
|
||||
private CoroutineClient|CurlClient $abstracts;
|
||||
|
||||
|
||||
/**
|
||||
* @param string $host
|
||||
* @param int $port
|
||||
* @param bool $isSsl
|
||||
*/
|
||||
/**
|
||||
* @param string $host
|
||||
* @param int $port
|
||||
* @param bool $isSsl
|
||||
*/
|
||||
public function __construct(string $host, int $port, bool $isSsl = false)
|
||||
{
|
||||
if (Context::inCoroutine()) {
|
||||
if (class_exists(Coroutine::class) && Coroutine::getCid() > -1) {
|
||||
$this->abstracts = new CoroutineClient($host, $port, $isSsl);
|
||||
} else {
|
||||
$this->abstracts = new CurlClient($host, $port, $isSsl);
|
||||
|
||||
+569
-759
File diff suppressed because it is too large
Load Diff
+121
-91
@@ -10,120 +10,150 @@ declare(strict_types=1);
|
||||
namespace Kiri;
|
||||
|
||||
use Exception;
|
||||
use JetBrains\PhpStorm\Pure;
|
||||
use Kiri\Abstracts\Logger;
|
||||
use Kiri\Kiri;
|
||||
use Swoole\Coroutine\Http\Client as SwowClient;
|
||||
|
||||
/**
|
||||
* Class Client
|
||||
* @package Kiri\Kiri\Http
|
||||
* @package Kiri\Http
|
||||
*/
|
||||
class CoroutineClient extends ClientAbstracts
|
||||
{
|
||||
|
||||
use TSwooleClient;
|
||||
|
||||
/**
|
||||
* @param string $method
|
||||
* @param $path
|
||||
* @param array $params
|
||||
* @return void
|
||||
* @throws Exception
|
||||
*/
|
||||
public function request(string $method, $path, array $params = []): void
|
||||
{
|
||||
$this->withMethod($method)
|
||||
->coroutine(
|
||||
$this->matchHost($path),
|
||||
$this->paramEncode($params)
|
||||
);
|
||||
}
|
||||
/**
|
||||
* @param string $method
|
||||
* @param $path
|
||||
* @param array|string $params
|
||||
* @return void
|
||||
* @throws
|
||||
*/
|
||||
public function request(string $method, $path, array|string $params = []): void
|
||||
{
|
||||
if (!str_starts_with($path, '/')) {
|
||||
$path = '/' . $path;
|
||||
}
|
||||
|
||||
$host = $this->host;
|
||||
if (!preg_match('/(\d{1,3}\.){3}\d{1,3}/', $host)) {
|
||||
$this->withAddedHeader('Host', $host);
|
||||
}
|
||||
$this->withMethod($method)
|
||||
->coroutine(
|
||||
$path,
|
||||
$this->paramEncode($params)
|
||||
);
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* @param $path
|
||||
* @param string $path
|
||||
* @return $this
|
||||
*/
|
||||
public function withCAInfo($path): static
|
||||
public function withCAInfo(string $path): static
|
||||
{
|
||||
return $this;
|
||||
}
|
||||
|
||||
/**
|
||||
* @param $url
|
||||
* @param array|string $data
|
||||
* @throws Exception 使用swoole协程方式请求
|
||||
*/
|
||||
private function coroutine($url, array|string $data = []): void
|
||||
{
|
||||
try {
|
||||
$this->generate_client($data, ...$url);
|
||||
if ($this->client->statusCode < 0) {
|
||||
throw new Exception($this->client->errMsg);
|
||||
}
|
||||
$this->setStatusCode($this->client->getStatusCode());
|
||||
$this->setBody($this->client->getBody());
|
||||
$this->setResponseHeader($this->client->headers);
|
||||
} catch (\Throwable $exception) {
|
||||
Kiri::getDi()->get(Logger::class)->error('rpc', [$exception]);
|
||||
$this->setStatusCode(-1);
|
||||
$this->setBody(jTraceEx($exception));
|
||||
}
|
||||
}
|
||||
/**
|
||||
* @param string $url
|
||||
* @param array|string $data
|
||||
*/
|
||||
private function coroutine(string $url, array|string $data = []): void
|
||||
{
|
||||
try {
|
||||
$this->generate_client($this->host, $this->isSSL);
|
||||
if ($this->client->statusCode < 0) {
|
||||
throw new Exception($this->client->errMsg);
|
||||
}
|
||||
|
||||
$this->execute($url, $data);
|
||||
} catch (\Throwable $exception) {
|
||||
$this->statusCode = -1;
|
||||
$this->body = json_encode(['code' => 500, 'message' => $exception->getMessage()]);
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* @param $data
|
||||
* @param $host
|
||||
* @param $isHttps
|
||||
* @param $path
|
||||
*/
|
||||
private function generate_client($data, $host, $isHttps, $path): void
|
||||
{
|
||||
if ($isHttps || $this->isSSL()) {
|
||||
$this->client = new SwowClient($host, 443, true);
|
||||
} else {
|
||||
$this->client = new SwowClient($host, $this->getPort(), false);
|
||||
}
|
||||
$this->client->set($this->settings());
|
||||
if (!empty($this->getAgent())) {
|
||||
$this->withAddedHeader('User-Agent', $this->getAgent());
|
||||
}
|
||||
$this->client->setHeaders($this->getHeader());
|
||||
$this->client->setMethod(strtoupper($this->getMethod()));
|
||||
$this->client->execute($this->setParams($path, $data));
|
||||
}
|
||||
/**
|
||||
* @param string $path
|
||||
* @param array|string $data
|
||||
* @return void
|
||||
*/
|
||||
private function execute(string $path, array|string $data): void
|
||||
{
|
||||
$this->client->execute($this->setParams($path, $data));
|
||||
if (in_array($this->client->getStatusCode(), [502, 404])) {
|
||||
$this->retry($path, $data);
|
||||
} else {
|
||||
$this->statusCode = $this->client->getStatusCode();
|
||||
$this->body = $this->client->getBody();
|
||||
$this->setResponseHeader($this->client->headers);
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* @param $path
|
||||
* @param $data
|
||||
* @return string
|
||||
*/
|
||||
private function setParams($path, $data): string
|
||||
{
|
||||
$content = $this->getData()->getContents();
|
||||
if (!empty($content)) {
|
||||
$this->client->setData($content);
|
||||
}
|
||||
if ($this->isGet()) {
|
||||
if (!empty($data)) $path .= '?' . $data;
|
||||
} else {
|
||||
$data = $this->mergeParams($data);
|
||||
if (!empty($data)) {
|
||||
$this->client->setData($data);
|
||||
}
|
||||
}
|
||||
return $path;
|
||||
}
|
||||
/**
|
||||
* @param string $path
|
||||
* @param array|string $data
|
||||
* @return void
|
||||
*/
|
||||
private function retry(string $path, array|string $data): void
|
||||
{
|
||||
if (($this->num += 1) <= $this->retryNum) {
|
||||
sleep($this->retryTimeout);
|
||||
|
||||
/**
|
||||
*
|
||||
*/
|
||||
public function close(): void
|
||||
{
|
||||
$this->client->close();
|
||||
}
|
||||
$this->execute($path, $data);
|
||||
} else {
|
||||
$this->statusCode = $this->client->statusCode;
|
||||
$this->body = $this->client->errMsg;
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* @param string $host
|
||||
* @param bool $isHttps
|
||||
*/
|
||||
private function generate_client(string $host, bool $isHttps): void
|
||||
{
|
||||
if ($isHttps || $this->isSSL) {
|
||||
$this->client = new SwowClient($host, 443, true);
|
||||
} else {
|
||||
$this->client = new SwowClient($host, $this->getPort(), false);
|
||||
}
|
||||
$this->client->set($this->settings());
|
||||
if (!empty($this->agent)) {
|
||||
$this->withAddedHeader('User-Agent', $this->agent);
|
||||
}
|
||||
$this->client->setHeaders($this->header);
|
||||
$this->client->setMethod(strtoupper($this->method));
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* @param string $path
|
||||
* @param mixed $data
|
||||
* @return string
|
||||
*/
|
||||
private function setParams(string $path, mixed $data): string
|
||||
{
|
||||
$content = $this->_data;
|
||||
if (!empty($content)) {
|
||||
$this->client->setData($content);
|
||||
}
|
||||
if ($this->isGet()) {
|
||||
if (!empty($data)) $path .= '?' . $data;
|
||||
} else if (!empty($data)) {
|
||||
$this->client->setData($data);
|
||||
}
|
||||
return $path;
|
||||
}
|
||||
|
||||
/**
|
||||
*
|
||||
*/
|
||||
public function close(): void
|
||||
{
|
||||
$this->client->close();
|
||||
}
|
||||
}
|
||||
|
||||
+196
-174
@@ -4,205 +4,227 @@ declare(strict_types=1);
|
||||
namespace Kiri;
|
||||
|
||||
|
||||
use Exception;
|
||||
use Kiri\Message\Response;
|
||||
use Kiri\Message\Stream;
|
||||
use JetBrains\PhpStorm\Pure;
|
||||
use Psr\Http\Message\ResponseInterface;
|
||||
|
||||
|
||||
/**
|
||||
* Class CurlClient
|
||||
*
|
||||
* @package Http\Handler\Client
|
||||
*/
|
||||
class CurlClient extends ClientAbstracts
|
||||
{
|
||||
|
||||
/**
|
||||
* @param $method
|
||||
* @param $path
|
||||
* @param array $params
|
||||
* @throws Exception
|
||||
*/
|
||||
public function request($method, $path, array $params = []): void
|
||||
{
|
||||
if ($method == self::GET) {
|
||||
$path = $this->joinGetParams($path, $params);
|
||||
}
|
||||
|
||||
$this->getCurlHandler($path, $method, $params);
|
||||
/**
|
||||
* @param string $method
|
||||
* @param string $path
|
||||
* @param array|string $params
|
||||
*
|
||||
* @return void
|
||||
*/
|
||||
public function request(string $method, string $path, array|string $params = []): void
|
||||
{
|
||||
if (!str_starts_with($path, '/')) {
|
||||
$path = '/' . $path;
|
||||
}
|
||||
if ($method == self::GET) {
|
||||
$path = $this->joinGetParams($path, $params);
|
||||
}
|
||||
|
||||
$this->execute();
|
||||
}
|
||||
$this->getCurlHandler($path, $method, $params);
|
||||
|
||||
$this->execute();
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* @param $path
|
||||
* @param $method
|
||||
* @param $params
|
||||
* @throws Exception
|
||||
*/
|
||||
private function getCurlHandler($path, $method, $params): void
|
||||
{
|
||||
[$host, $isHttps, $path] = $this->matchHost($path);
|
||||
|
||||
$host = $isHttps ? 'https://' . $host : 'http://' . $host;
|
||||
if ($this->getPort() != 443 && $this->getPort() != 80) {
|
||||
$host .= ':' . $this->getPort();
|
||||
}
|
||||
$this->do(curl_init($host . $path), $host . $path, $method);
|
||||
if ($isHttps !== FALSE) {
|
||||
$this->curlHandlerSslSet();
|
||||
}
|
||||
$contents = $this->getData()->getContents();
|
||||
if (empty($params) && empty($contents)) {
|
||||
return;
|
||||
}
|
||||
if (!empty($contents)) {
|
||||
curl_setopt($this->client, CURLOPT_POSTFIELDS, $contents);
|
||||
} else if ($method === self::POST) {
|
||||
curl_setopt($this->client, CURLOPT_POSTFIELDS, $this->mergeParams($params));
|
||||
} else if ($method === self::UPLOAD) {
|
||||
curl_setopt($this->client, CURLOPT_POSTFIELDS, $params);
|
||||
}
|
||||
}
|
||||
/**
|
||||
* @param string $path
|
||||
* @param string $method
|
||||
* @param $params
|
||||
*
|
||||
* @return void
|
||||
*/
|
||||
private function getCurlHandler(string $path, string $method, $params): void
|
||||
{
|
||||
$host = $this->isSSL ? 'https://' . $this->host : 'http://' . $this->host;
|
||||
if ($this->getPort() != 443 && $this->getPort() != 80) {
|
||||
$host .= ':' . $this->getPort();
|
||||
}
|
||||
$this->do(curl_init($host . $path), $host . $path, $method);
|
||||
if ($this->isSSL) {
|
||||
$this->curlHandlerSslSet();
|
||||
}
|
||||
$contents = $this->_data;
|
||||
if (empty($params) && empty($contents)) {
|
||||
return;
|
||||
}
|
||||
if (!empty($contents)) {
|
||||
curl_setopt($this->client, CURLOPT_POSTFIELDS, $contents);
|
||||
} else if ($method === self::UPLOAD) {
|
||||
curl_setopt($this->client, CURLOPT_POSTFIELDS, $params);
|
||||
} else if ($method === self::POST) {
|
||||
if (is_array($params)) {
|
||||
$params = http_build_query($params);
|
||||
}
|
||||
curl_setopt($this->client, CURLOPT_POSTFIELDS, $params);
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* @return void
|
||||
* @throws Exception
|
||||
*/
|
||||
private function curlHandlerSslSet(): void
|
||||
{
|
||||
if (!empty($this->getSslKeyFile()) && file_exists($this->getSslKeyFile())) {
|
||||
curl_setopt($this->client, CURLOPT_SSLKEY, $this->getSslKeyFile());
|
||||
}
|
||||
if (!empty($this->getSslCertFile()) && file_exists($this->getSslCertFile())) {
|
||||
curl_setopt($this->client, CURLOPT_SSLCERT, $this->getSslCertFile());
|
||||
}
|
||||
}
|
||||
/**
|
||||
* @return void
|
||||
* @throws
|
||||
*/
|
||||
private function curlHandlerSslSet(): void
|
||||
{
|
||||
if (!empty($this->ssl_key_file) && file_exists($this->ssl_key_file)) {
|
||||
curl_setopt($this->client, CURLOPT_SSLKEY, $this->ssl_key_file);
|
||||
}
|
||||
if (!empty($this->ssl_cert_file) && file_exists($this->ssl_cert_file)) {
|
||||
curl_setopt($this->client, CURLOPT_SSLCERT, $this->ssl_cert_file);
|
||||
}
|
||||
if (!empty($this->ca) && file_exists($this->ca)) {
|
||||
curl_setopt($this->client, CURLOPT_CAINFO, $this->ca);
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* @param $resource
|
||||
* @param $path
|
||||
* @param $method
|
||||
* @throws Exception
|
||||
*/
|
||||
private function do($resource, $path, $method): void
|
||||
{
|
||||
curl_setopt($resource, CURLOPT_URL, $path);
|
||||
curl_setopt($resource, CURLOPT_TIMEOUT, $this->getTimeout()); // 超时设置
|
||||
curl_setopt($resource, CURLOPT_CONNECTTIMEOUT, $this->getConnectTimeout()); // 超时设置
|
||||
curl_setopt($resource, CURLOPT_HEADER, TRUE);
|
||||
curl_setopt($resource, CURLOPT_FAILONERROR, TRUE);
|
||||
curl_setopt($resource, CURLOPT_HTTPHEADER, $this->parseHeaderMat());
|
||||
if (defined('CURLOPT_SSL_FALSESTART')) {
|
||||
curl_setopt($resource, CURLOPT_SSL_FALSESTART, TRUE);
|
||||
}
|
||||
curl_setopt($resource, CURLOPT_FORBID_REUSE, FALSE);
|
||||
curl_setopt($resource, CURLOPT_FRESH_CONNECT, FALSE);
|
||||
if (!empty($this->getAgent())) {
|
||||
curl_setopt($resource, CURLOPT_USERAGENT, $this->getAgent());
|
||||
}
|
||||
curl_setopt($resource, CURLOPT_NOBODY, FALSE);
|
||||
curl_setopt($resource, CURLOPT_RETURNTRANSFER, TRUE);//返回内容
|
||||
curl_setopt($resource, CURLOPT_FOLLOWLOCATION, TRUE);// 跟踪重定向
|
||||
curl_setopt($resource, CURLOPT_ENCODING, 'gzip,deflate');
|
||||
if ($method === self::POST || $method == self::UPLOAD) {
|
||||
curl_setopt($resource, CURLOPT_POST, 1);
|
||||
}
|
||||
curl_setopt($resource, CURLOPT_CUSTOMREQUEST, strtoupper($method));
|
||||
$this->client = $resource;
|
||||
if (!empty($this->caPath)) {
|
||||
curl_setopt($this->client,CURLOPT_CAINFO, $this->caPath);
|
||||
}
|
||||
}
|
||||
/**
|
||||
* @param mixed $resource
|
||||
* @param string $path
|
||||
* @param string $method
|
||||
*
|
||||
* @return void
|
||||
*/
|
||||
private function do(mixed $resource, string $path, string $method): void
|
||||
{
|
||||
curl_setopt($resource, CURLOPT_URL, $path);
|
||||
curl_setopt($resource, CURLOPT_TIMEOUT, $this->timeout); // 超时设置
|
||||
curl_setopt($resource, CURLOPT_CONNECTTIMEOUT, $this->connect_timeout); // 超时设置
|
||||
curl_setopt($resource, CURLOPT_HEADER, TRUE);
|
||||
curl_setopt($resource, CURLOPT_FAILONERROR, TRUE);
|
||||
curl_setopt($resource, CURLOPT_HTTPHEADER, $this->parseHeaderMat());
|
||||
if (defined('CURLOPT_SSL_FALSESTART')) {
|
||||
curl_setopt($resource, CURLOPT_SSL_FALSESTART, TRUE);
|
||||
}
|
||||
curl_setopt($resource, CURLOPT_FORBID_REUSE, FALSE);
|
||||
curl_setopt($resource, CURLOPT_FRESH_CONNECT, FALSE);
|
||||
if (!empty($this->agent)) {
|
||||
curl_setopt($resource, CURLOPT_USERAGENT, $this->agent);
|
||||
}
|
||||
curl_setopt($resource, CURLOPT_NOBODY, FALSE);
|
||||
curl_setopt($resource, CURLOPT_RETURNTRANSFER, TRUE);//返回内容
|
||||
curl_setopt($resource, CURLOPT_FOLLOWLOCATION, TRUE);// 跟踪重定向
|
||||
curl_setopt($resource, CURLOPT_ENCODING, 'gzip,deflate');
|
||||
if ($method === self::POST || $method == self::UPLOAD) {
|
||||
curl_setopt($resource, CURLOPT_POST, 1);
|
||||
}
|
||||
[$proxy, $port] = [$this->proxyHost, $this->proxyPort];
|
||||
if (!empty($proxy) && $port > 0) {
|
||||
curl_setopt($resource, CURLOPT_PROXYPORT, $port);
|
||||
curl_setopt($resource, CURLOPT_PROXY, $proxy);
|
||||
}
|
||||
curl_setopt($resource, CURLOPT_CUSTOMREQUEST, strtoupper($method));
|
||||
$this->client = $resource;
|
||||
if (!empty($this->caPath)) {
|
||||
curl_setopt($this->client, CURLOPT_CAINFO, $this->caPath);
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
private string $caPath = '';
|
||||
/**
|
||||
* @var string
|
||||
*/
|
||||
private string $caPath = '';
|
||||
|
||||
|
||||
/**
|
||||
* @param $path
|
||||
* @return $this
|
||||
*/
|
||||
public function withCAInfo($path): static
|
||||
{
|
||||
$this->caPath = $path;
|
||||
return $this;
|
||||
}
|
||||
/**
|
||||
* @param string $path
|
||||
*
|
||||
* @return $this
|
||||
*/
|
||||
public function withCAInfo(string $path): static
|
||||
{
|
||||
$this->caPath = $path;
|
||||
return $this;
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* @throws Exception
|
||||
*/
|
||||
private function execute(): void
|
||||
{
|
||||
$output = curl_exec($this->client);
|
||||
if ($output === FALSE) {
|
||||
$this->setStatusCode(curl_errno($this->client));
|
||||
$this->setBody(curl_error($this->client));
|
||||
} else {
|
||||
$this->explode($output);
|
||||
}
|
||||
}
|
||||
/**
|
||||
* @throws
|
||||
*/
|
||||
private function execute(): void
|
||||
{
|
||||
$output = curl_exec($this->client);
|
||||
if ($output !== FALSE) {
|
||||
$this->explode($output);
|
||||
} else {
|
||||
$this->statusCode = curl_errno($this->client);
|
||||
$this->body = curl_error($this->client);
|
||||
}
|
||||
$this->close();
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
*
|
||||
*/
|
||||
public function close(): void
|
||||
{
|
||||
curl_close($this->client);
|
||||
}
|
||||
/**
|
||||
* @return void
|
||||
* @throws
|
||||
*/
|
||||
private function retry(): void
|
||||
{
|
||||
if (($this->num += 1) <= $this->retryNum) {
|
||||
sleep($this->retryTimeout);
|
||||
|
||||
$this->execute();
|
||||
} else {
|
||||
$this->statusCode = curl_errno($this->client);
|
||||
$this->body = curl_error($this->client);
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* @param $output
|
||||
* @return void
|
||||
* @throws Exception
|
||||
*/
|
||||
private function explode($output): void
|
||||
{
|
||||
[$header, $body] = explode("\r\n\r\n", $output, 2);
|
||||
if ($header == 'HTTP/1.1 100 Continue') {
|
||||
[$header, $body] = explode("\r\n\r\n", $body, 2);
|
||||
}
|
||||
|
||||
$header = explode("\r\n", $header);
|
||||
$status = explode(' ', array_shift($header));
|
||||
|
||||
$this->setStatusCode(intval($status[1]));
|
||||
$this->setBody($body);
|
||||
$this->setResponseHeader($header);
|
||||
}
|
||||
|
||||
/**
|
||||
* @param $headers
|
||||
* @return array
|
||||
*/
|
||||
private function headerFormat($headers): array
|
||||
{
|
||||
$_tmp = [];
|
||||
foreach ($headers as $val) {
|
||||
$trim = explode(': ', trim($val));
|
||||
|
||||
$_tmp[strtolower($trim[0])] = [$trim[1] ?? ''];
|
||||
}
|
||||
return $_tmp;
|
||||
}
|
||||
/**
|
||||
*
|
||||
*/
|
||||
public function close(): void
|
||||
{
|
||||
if (PHP_VERSION < '8.0') {
|
||||
curl_close($this->client);
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* @return array
|
||||
*/
|
||||
#[Pure] private function parseHeaderMat(): array
|
||||
{
|
||||
$headers = [];
|
||||
foreach ($this->getHeader() as $key => $val) {
|
||||
$headers[$key] = $key . ': ' . $val;
|
||||
}
|
||||
return array_values($headers);
|
||||
}
|
||||
/**
|
||||
* @param string $output
|
||||
*
|
||||
* @return void
|
||||
* @throws
|
||||
*/
|
||||
private function explode(string $output): void
|
||||
{
|
||||
// 获取 HTTP 状态码
|
||||
$statusCode = +curl_getinfo($this->client, CURLINFO_HTTP_CODE);
|
||||
|
||||
// 获取 header 的大小(不包括最后的 \r\n\r\n 分隔符)
|
||||
$headerSize = curl_getinfo($this->client, CURLINFO_HEADER_SIZE);
|
||||
$header = substr($output, 0, $headerSize);
|
||||
if (in_array($statusCode, [502, 404])) {
|
||||
$this->retry();
|
||||
} else {
|
||||
$this->statusCode = $statusCode;
|
||||
$this->body = substr($output, $headerSize);
|
||||
$this->setResponseHeader(explode("\r\n", $header));
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* @return array
|
||||
*/
|
||||
private function parseHeaderMat(): array
|
||||
{
|
||||
$headers = [];
|
||||
foreach ($this->header as $key => $val) {
|
||||
$headers[$key] = $key . ': ' . $val;
|
||||
}
|
||||
return array_values($headers);
|
||||
}
|
||||
}
|
||||
|
||||
+88
-89
@@ -11,99 +11,98 @@ use Exception;
|
||||
*/
|
||||
class HttpParse
|
||||
{
|
||||
/**
|
||||
* @param mixed ...$object
|
||||
* @return string
|
||||
*/
|
||||
private static function getKey(...$object): string
|
||||
{
|
||||
$first = '';
|
||||
$tp = [];
|
||||
foreach ($object as $key => $value) {
|
||||
if ($value === null) {
|
||||
continue;
|
||||
}
|
||||
if (is_array($value)) {
|
||||
$value = key($value);
|
||||
}
|
||||
if ($first === '') {
|
||||
$first = $value;
|
||||
} else {
|
||||
$tp[] = $value;
|
||||
}
|
||||
}
|
||||
$key = $first . '[' . implode('][', $tp) . ']';
|
||||
if (count($tp) < 1) {
|
||||
$key = $first;
|
||||
}
|
||||
return $key;
|
||||
}
|
||||
/**
|
||||
* @param mixed ...$object
|
||||
* @return string
|
||||
*/
|
||||
private static function getKey(...$object): string
|
||||
{
|
||||
$first = '';
|
||||
$tp = [];
|
||||
foreach ($object as $value) {
|
||||
if ($value === null) {
|
||||
continue;
|
||||
}
|
||||
if (is_array($value)) {
|
||||
$value = key($value);
|
||||
}
|
||||
if ($first === '') {
|
||||
$first = $value;
|
||||
} else {
|
||||
$tp[] = $value;
|
||||
}
|
||||
}
|
||||
$key = $first . '[' . implode('][', $tp) . ']';
|
||||
if (count($tp) < 1) {
|
||||
$key = $first;
|
||||
}
|
||||
return $key;
|
||||
}
|
||||
|
||||
/**
|
||||
* @param $data
|
||||
* @return string
|
||||
* @throws Exception
|
||||
*/
|
||||
public static function parse($data): string
|
||||
{
|
||||
$tmp = [];
|
||||
if (is_string($data)) {
|
||||
return $data;
|
||||
}
|
||||
foreach ($data as $key => $datum) {
|
||||
if ($datum === null) {
|
||||
continue;
|
||||
}
|
||||
$tmp[] = static::ifElse($key, $datum);
|
||||
}
|
||||
return implode('&', $tmp);
|
||||
}
|
||||
/**
|
||||
* @param array|string $data
|
||||
* @return string
|
||||
*/
|
||||
public static function parse(array|string $data): string
|
||||
{
|
||||
$tmp = [];
|
||||
if (is_string($data)) {
|
||||
return $data;
|
||||
}
|
||||
foreach ($data as $key => $datum) {
|
||||
if ($datum === null) {
|
||||
continue;
|
||||
}
|
||||
$tmp[] = static::ifElse($key, $datum);
|
||||
}
|
||||
return implode('&', $tmp);
|
||||
}
|
||||
|
||||
/**
|
||||
* @param $t
|
||||
* @param $qt
|
||||
* @return string
|
||||
* @throws Exception
|
||||
*/
|
||||
private static function ifElse($t, $qt): string
|
||||
{
|
||||
if (is_numeric($qt)) {
|
||||
return $t . '=' . $qt;
|
||||
}
|
||||
if (is_string($qt)) {
|
||||
$string = $t . '=' . urlencode($qt);
|
||||
} else {
|
||||
$string = static::encode($t, $qt);
|
||||
}
|
||||
return $string;
|
||||
}
|
||||
/**
|
||||
* @param $t
|
||||
* @param $qt
|
||||
* @return string
|
||||
* @throws
|
||||
*/
|
||||
private static function ifElse(string $t, mixed $qt): string
|
||||
{
|
||||
if (is_numeric($qt)) {
|
||||
return $t . '=' . $qt;
|
||||
}
|
||||
if (is_string($qt)) {
|
||||
$string = $t . '=' . urlencode($qt);
|
||||
} else {
|
||||
$string = static::encode($t, $qt);
|
||||
}
|
||||
return $string;
|
||||
}
|
||||
|
||||
/**
|
||||
* @param mixed ...$object
|
||||
* @return string
|
||||
* @throws Exception
|
||||
*/
|
||||
private static function encode(...$object): string
|
||||
{
|
||||
$ret = [];
|
||||
/**
|
||||
* @param mixed ...$object
|
||||
* @return string
|
||||
* @throws
|
||||
*/
|
||||
private static function encode(...$object): string
|
||||
{
|
||||
$ret = [];
|
||||
|
||||
$data = $object[count($object) - 1];
|
||||
$key = static::getKey(...$object);
|
||||
foreach ($data as $s => $datum) {
|
||||
if (is_array($datum)) {
|
||||
$object[count($object) - 1] = $s;
|
||||
$object[] = $datum;
|
||||
$string = static::encode(...$object);
|
||||
} else {
|
||||
if (is_object($datum)) {
|
||||
throw new Exception('Http body con\'t object.');
|
||||
}
|
||||
$string = $key . '=' . urlencode($datum);
|
||||
}
|
||||
$ret[] = $string;
|
||||
}
|
||||
return implode('&', $ret);
|
||||
}
|
||||
$data = $object[count($object) - 1];
|
||||
$key = static::getKey(...$object);
|
||||
foreach ($data as $s => $datum) {
|
||||
if (is_array($datum)) {
|
||||
$object[count($object) - 1] = $s;
|
||||
$object[] = $datum;
|
||||
$string = static::encode(...$object);
|
||||
} else {
|
||||
if (is_object($datum)) {
|
||||
throw new Exception('Http body con\'t object.');
|
||||
}
|
||||
$string = $key . '=' . urlencode($datum);
|
||||
}
|
||||
$ret[] = $string;
|
||||
}
|
||||
return implode('&', $ret);
|
||||
}
|
||||
|
||||
|
||||
}
|
||||
|
||||
+42
-45
@@ -5,25 +5,23 @@ namespace Kiri;
|
||||
|
||||
|
||||
use Closure;
|
||||
use Psr\Http\Message\ResponseInterface;
|
||||
use Psr\Http\Message\StreamInterface;
|
||||
|
||||
interface IClient
|
||||
{
|
||||
|
||||
|
||||
/**
|
||||
* @param string $path
|
||||
* @param array $params
|
||||
*/
|
||||
public function get(string $path, array $params = []): void;
|
||||
/**
|
||||
* @param string $path
|
||||
* @param array|string $params
|
||||
*/
|
||||
public function get(string $path, array|string $params = []): void;
|
||||
|
||||
|
||||
/**
|
||||
* @param string $path
|
||||
* @param array $params
|
||||
*/
|
||||
public function post(string $path, array $params = []): void;
|
||||
/**
|
||||
* @param string $path
|
||||
* @param array|string $params
|
||||
*/
|
||||
public function post(string $path, array|string $params = []): void;
|
||||
|
||||
|
||||
/**
|
||||
@@ -32,47 +30,47 @@ interface IClient
|
||||
public function close(): void;
|
||||
|
||||
|
||||
/**
|
||||
* @param string $path
|
||||
* @param array $params
|
||||
*/
|
||||
public function delete(string $path, array $params = []): void;
|
||||
/**
|
||||
* @param string $path
|
||||
* @param array|string $params
|
||||
*/
|
||||
public function delete(string $path, array|string $params = []): void;
|
||||
|
||||
|
||||
/**
|
||||
* @param string $path
|
||||
* @param array $params
|
||||
*/
|
||||
public function options(string $path, array $params = []): void;
|
||||
/**
|
||||
* @param string $path
|
||||
* @param array|string $params
|
||||
*/
|
||||
public function options(string $path, array|string $params = []): void;
|
||||
|
||||
|
||||
/**
|
||||
* @param string $path
|
||||
* @param array $params
|
||||
*/
|
||||
public function upload(string $path, array $params = []): void;
|
||||
/**
|
||||
* @param string $path
|
||||
* @param array|string $params
|
||||
*/
|
||||
public function upload(string $path, array|string $params = []): void;
|
||||
|
||||
|
||||
/**
|
||||
* @param string $path
|
||||
* @param array $params
|
||||
*/
|
||||
public function put(string $path, array $params = []): void;
|
||||
/**
|
||||
* @param string $path
|
||||
* @param array|string $params
|
||||
*/
|
||||
public function put(string $path, array|string $params = []): void;
|
||||
|
||||
|
||||
/**
|
||||
* @param string $path
|
||||
* @param array $params
|
||||
*/
|
||||
public function head(string $path, array $params = []): void;
|
||||
/**
|
||||
* @param string $path
|
||||
* @param array|string $params
|
||||
*/
|
||||
public function head(string $path, array|string $params = []): void;
|
||||
|
||||
|
||||
/**
|
||||
* @param string $method
|
||||
* @param string $path
|
||||
* @param array $params
|
||||
* @param array|string $params
|
||||
*/
|
||||
public function request(string $method, string $path, array $params = []): void;
|
||||
public function request(string $method, string $path, array|string $params = []): void;
|
||||
|
||||
|
||||
/**
|
||||
@@ -154,10 +152,10 @@ interface IClient
|
||||
|
||||
|
||||
/**
|
||||
* @param string|StreamInterface $data
|
||||
* @param string $data
|
||||
* @return static
|
||||
*/
|
||||
public function withBody(string|StreamInterface $data): static;
|
||||
public function withBody(string $data): static;
|
||||
|
||||
|
||||
/**
|
||||
@@ -174,8 +172,7 @@ interface IClient
|
||||
public function withContentType(string $contentType): static;
|
||||
|
||||
|
||||
/**
|
||||
* @return mixed
|
||||
*/
|
||||
public function getBody(): mixed;
|
||||
public null|string $body {
|
||||
get;
|
||||
}
|
||||
}
|
||||
|
||||
+26
-23
@@ -2,37 +2,40 @@
|
||||
|
||||
namespace Kiri;
|
||||
|
||||
use JetBrains\PhpStorm\Pure;
|
||||
|
||||
trait TSwooleClient
|
||||
{
|
||||
|
||||
|
||||
/**
|
||||
* @return array
|
||||
*/
|
||||
private function settings(): array
|
||||
{
|
||||
$sslCert = $this->ssl_cert_file;
|
||||
$sslKey = $this->ssl_key_file;
|
||||
$sslCa = $this->ca;
|
||||
|
||||
/**
|
||||
* @return array
|
||||
*/
|
||||
#[Pure] private function settings(): array
|
||||
{
|
||||
$sslCert = $this->getSslCertFile();
|
||||
$sslKey = $this->getSslKeyFile();
|
||||
$sslCa = $this->getCa();
|
||||
$params = [];
|
||||
if ($this->connect_timeout > 0) {
|
||||
$params['timeout'] = $this->connect_timeout;
|
||||
}
|
||||
|
||||
$params = [];
|
||||
if ($this->getConnectTimeout() > 0) {
|
||||
$params['timeout'] = $this->getConnectTimeout();
|
||||
}
|
||||
if (empty($sslCert) || empty($sslKey) || empty($sslCa)) {
|
||||
return $params;
|
||||
[$proxy, $port] = [$this->proxyHost, $this->proxyPort];
|
||||
if (!empty($proxy) && $port > 0) {
|
||||
$params['http_proxy_host'] = $proxy;
|
||||
$params['http_proxy_port'] = $port;
|
||||
}
|
||||
if (empty($sslCert) || empty($sslKey) || empty($sslCa)) {
|
||||
return $params;
|
||||
}
|
||||
|
||||
$params['ssl_host_name'] = $this->getHost();
|
||||
$params['ssl_cert_file'] = $this->getSslCertFile();
|
||||
$params['ssl_key_file'] = $this->getSslKeyFile();
|
||||
$params['ssl_verify_peer'] = TRUE;
|
||||
$params['ssl_cafile'] = $sslCa;
|
||||
$params['ssl_host_name'] = $this->host;
|
||||
$params['ssl_cert_file'] = $this->ssl_cert_file;
|
||||
$params['ssl_key_file'] = $this->ssl_key_file;
|
||||
$params['ssl_verify_peer'] = TRUE;
|
||||
$params['ssl_cafile'] = $sslCa;
|
||||
|
||||
return $params;
|
||||
}
|
||||
return $params;
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
@@ -0,0 +1,87 @@
|
||||
<?php
|
||||
|
||||
namespace Kiri;
|
||||
|
||||
use Exception;
|
||||
use Kiri\Di\Context;
|
||||
use Swoole\Coroutine\Client as CoroutineClient;
|
||||
use Swoole\Client as AsyncClient;
|
||||
|
||||
|
||||
class TcpClient
|
||||
{
|
||||
|
||||
|
||||
/**
|
||||
* @var AsyncClient|CoroutineClient
|
||||
*/
|
||||
protected AsyncClient|CoroutineClient $client;
|
||||
|
||||
|
||||
/**
|
||||
* @param string $host
|
||||
* @param int $port
|
||||
* @param int $socket
|
||||
* @throws
|
||||
*/
|
||||
public function __construct(readonly public string $host, readonly public int $port, readonly public int $socket = SWOOLE_SOCK_TCP)
|
||||
{
|
||||
$this->reconnect();
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* @return void
|
||||
* @throws
|
||||
*/
|
||||
public function reconnect(): void
|
||||
{
|
||||
$this->client?->close();
|
||||
if (Context::inCoroutine()) {
|
||||
$this->client = new CoroutineClient($this->socket);
|
||||
} else {
|
||||
$this->client = new AsyncClient($this->socket);
|
||||
}
|
||||
if (!$this->client->connect($this->host, $this->port, 1)) {
|
||||
throw new Exception('Connect ' . $this->host . '::' . $this->port . ' fail');
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* @param string $data
|
||||
* @return int|bool
|
||||
*/
|
||||
public function send(string $data): int|bool
|
||||
{
|
||||
return $this->client->send($data);
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* @return bool|string
|
||||
*/
|
||||
public function read(): bool|string
|
||||
{
|
||||
return $this->client->recv();
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* @return bool
|
||||
*/
|
||||
public function isConnected(): bool
|
||||
{
|
||||
return $this->client->isConnected();
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* @return bool
|
||||
*/
|
||||
public function close(): bool
|
||||
{
|
||||
return $this->client->close();
|
||||
}
|
||||
|
||||
}
|
||||
+2
-4
@@ -9,11 +9,9 @@
|
||||
],
|
||||
"license": "MIT",
|
||||
"require": {
|
||||
"php": ">=8.0",
|
||||
"php": ">=8.5",
|
||||
"ext-json": "*",
|
||||
"ext-swoole": "*",
|
||||
"psr/http-client": "^1.0",
|
||||
"psr/http-message": "^1.0"
|
||||
"ext-swoole": "*"
|
||||
},
|
||||
"autoload": {
|
||||
"psr-4": {
|
||||
|
||||
Reference in New Issue
Block a user