Compare commits

...

95 Commits

Author SHA1 Message Date
as2252258 45d8931267 eee 2026-07-03 15:06:12 +08:00
as2252258 7bcb789ed2 eee 2026-06-12 23:57:21 +08:00
as2252258 1d9d761798 eee 2025-12-31 00:19:30 +08:00
as2252258 22f425ceb7 ea 2025-11-28 11:43:24 +08:00
as2252258 a52610cf2a ea 2025-11-28 11:39:14 +08:00
as2252258 5697102a85 ea 2025-11-28 11:38:29 +08:00
as2252258 0c0a2fadd5 ea 2025-11-28 11:37:09 +08:00
as2252258 5061a17342 ea 2025-11-28 11:30:07 +08:00
as2252258 9c1bc63f78 ea 2025-11-26 07:52:46 +08:00
as2252258 345af9e727 eee 2025-07-14 15:35:37 +08:00
as2252258 196db497c1 eee 2023-12-18 18:21:09 +08:00
as2252258 6da3575619 eee 2023-12-12 15:35:36 +08:00
as2252258 9814e06e8d eee 2023-12-12 14:08:01 +08:00
as2252258 df32502723 eee 2023-11-09 14:46:15 +08:00
as2252258 7f702e5784 eee 2023-11-09 14:45:59 +08:00
as2252258 fd61b6f1d9 eee 2023-11-07 16:16:34 +08:00
as2252258 9e062a7a33 eee 2023-11-07 16:15:48 +08:00
as2252258 43c8f9b8e6 eee 2023-10-27 22:36:05 +08:00
as2252258 7b14fa075e eee 2023-10-27 22:23:11 +08:00
as2252258 165f62113c eee 2023-10-24 17:22:30 +08:00
as2252258 3c3b6f2e65 qqq 2023-08-24 11:25:42 +08:00
as2252258 2ca20d3a96 qqq 2023-08-18 19:53:00 +08:00
as2252258 b2800b806c qqq 2023-08-18 19:50:43 +08:00
as2252258 092daa022a qqq 2023-08-18 17:33:06 +08:00
as2252258 57337cbe16 qqq 2023-08-18 16:37:47 +08:00
as2252258 ad8b20723b qqq 2023-08-18 16:37:18 +08:00
as2252258 f2421b71e7 qqq 2023-08-18 16:36:05 +08:00
as2252258 02685a9e61 qqq 2023-08-18 15:45:30 +08:00
as2252258 1b5232f5e7 qqq 2023-08-18 15:43:52 +08:00
as2252258 c76fb40689 qqq 2023-08-17 16:32:02 +08:00
as2252258 18b44b32f2 qqq 2023-07-12 21:38:27 +08:00
as2252258 14574e7a6c qqq 2023-06-12 17:51:10 +08:00
as2252258 13fc812ae4 变更 2022-12-12 17:31:12 +08:00
as2252258 edd1751f7a 变更 2022-09-08 14:07:22 +08:00
as2252258 7dbce8b1e8 modify plugin name 2022-07-08 17:40:25 +08:00
as2252258 eb7ffecdb9 modify plugin name 2022-06-22 16:29:42 +08:00
as2252258 dd85596907 变更 2022-05-31 11:48:20 +08:00
as2252258 e6b10bbd70 变更 2022-05-31 08:24:44 +08:00
as2252258 c149afa3f8 变更 2022-05-31 08:23:26 +08:00
as2252258 f195d001c6 变更 2022-05-31 07:46:45 +08:00
as2252258 b45666d9db 变更 2022-05-31 07:46:01 +08:00
as2252258 650aaee974 modify plugin name 2022-03-01 18:47:45 +08:00
as2252258 d589e096af modify plugin name 2022-03-01 18:46:08 +08:00
as2252258 f17a55f8d0 modify plugin name 2022-02-21 15:39:55 +08:00
as2252258 3cc98214c7 modify plugin name 2022-02-21 15:23:41 +08:00
as2252258 aaeb538849 modify plugin name 2022-02-21 15:21:34 +08:00
as2252258 0c7b8eb57f modify plugin name 2022-02-21 15:17:42 +08:00
as2252258 48808ded98 modify plugin name 2022-02-18 13:53:51 +08:00
as2252258 eb25fe06ed modify plugin name 2022-02-18 13:45:30 +08:00
as2252258 4d51f6e5d6 Revert "改名"
This reverts commit fdf58326
2022-01-12 14:10:33 +08:00
as2252258 21fbbe52e5 Revert "改名"
This reverts commit fdf58326
2022-01-11 17:56:16 +08:00
as2252258 a0a753d8b4 Revert "改名"
This reverts commit fdf58326
2022-01-11 16:25:17 +08:00
as2252258 dcc5e67f0a Revert "改名"
This reverts commit fdf58326
2022-01-11 16:21:29 +08:00
as2252258 b8e7a5f086 Revert "改名"
This reverts commit fdf58326
2022-01-11 16:20:49 +08:00
as2252258 88a414f0bb Revert "改名"
This reverts commit fdf58326
2022-01-11 16:19:05 +08:00
as2252258 cba27222fa Revert "改名"
This reverts commit fdf58326
2022-01-11 16:16:04 +08:00
as2252258 593f05754d Revert "改名"
This reverts commit fdf58326
2022-01-11 15:44:56 +08:00
as2252258 044c5c4a28 Revert "改名"
This reverts commit fdf58326
2022-01-11 15:44:07 +08:00
as2252258 121642e825 Revert "改名"
This reverts commit fdf58326
2022-01-11 15:42:55 +08:00
as2252258 5eecf66b43 Revert "改名"
This reverts commit fdf58326
2022-01-11 15:41:53 +08:00
as2252258 a88787a530 Revert "改名"
This reverts commit fdf58326
2022-01-11 15:40:55 +08:00
as2252258 aed0e61531 Revert "改名"
This reverts commit fdf58326
2022-01-11 15:38:56 +08:00
as2252258 c94883375a Revert "改名"
This reverts commit fdf58326
2022-01-11 15:38:06 +08:00
as2252258 6565f1d7a3 Revert "改名"
This reverts commit fdf58326
2022-01-11 15:35:59 +08:00
as2252258 cefe7ae010 Revert "改名"
This reverts commit fdf58326
2022-01-11 15:35:14 +08:00
as2252258 56f42ac97b Revert "改名"
This reverts commit fdf58326
2022-01-11 15:34:27 +08:00
as2252258 9c7355d7f6 Revert "改名"
This reverts commit fdf58326
2022-01-11 15:33:27 +08:00
as2252258 2ff2e3cdb8 Revert "改名"
This reverts commit fdf58326
2022-01-11 15:32:19 +08:00
as2252258 18c6f9594e Revert "改名"
This reverts commit fdf58326
2022-01-11 15:31:37 +08:00
as2252258 a5c2467cdf Revert "改名"
This reverts commit fdf58326
2022-01-11 15:30:10 +08:00
as2252258 698868b6ee Revert "改名"
This reverts commit fdf58326
2022-01-11 15:28:30 +08:00
as2252258 cbee38ae54 Revert "改名"
This reverts commit fdf58326
2022-01-11 15:27:10 +08:00
as2252258 e19536ce99 Revert "改名"
This reverts commit fdf58326
2022-01-11 15:25:22 +08:00
as2252258 b92ab0bf1d Revert "改名"
This reverts commit fdf58326
2022-01-11 15:24:22 +08:00
as2252258 bc049a3c77 Revert "改名"
This reverts commit fdf58326
2022-01-11 15:20:00 +08:00
as2252258 a026a610a1 Revert "改名"
This reverts commit fdf58326
2022-01-11 15:12:59 +08:00
as2252258 e71a17cbce Revert "改名"
This reverts commit fdf58326
2022-01-11 15:11:18 +08:00
as2252258 86c09f3b25 Revert "改名"
This reverts commit fdf58326
2022-01-11 15:09:14 +08:00
as2252258 788ecee65d Revert "改名"
This reverts commit fdf58326
2022-01-11 15:08:18 +08:00
as2252258 98cbb07845 Revert "改名"
This reverts commit fdf58326
2022-01-11 15:04:16 +08:00
as2252258 93bab6d5eb Revert "改名"
This reverts commit fdf58326
2022-01-11 15:01:42 +08:00
as2252258 99f79b7616 Revert "改名"
This reverts commit fdf58326
2022-01-11 14:59:59 +08:00
as2252258 47dc988e49 Revert "改名"
This reverts commit fdf58326
2022-01-11 14:59:24 +08:00
as2252258 072de12cc1 Revert "改名"
This reverts commit fdf58326
2022-01-11 14:58:21 +08:00
as2252258 77f755d07c Revert "改名"
This reverts commit fdf58326
2022-01-11 14:54:21 +08:00
as2252258 bc5f9b1085 Revert "改名"
This reverts commit fdf58326
2022-01-11 14:51:50 +08:00
as2252258 4660025775 Revert "改名"
This reverts commit fdf58326
2022-01-11 14:46:42 +08:00
as2252258 336cc875a1 Revert "改名"
This reverts commit fdf58326
2022-01-11 14:46:08 +08:00
as2252258 d943b2ebc2 Revert "改名"
This reverts commit fdf58326
2022-01-11 14:44:37 +08:00
as2252258 f982b51ff6 Revert "改名"
This reverts commit fdf58326
2022-01-10 18:51:56 +08:00
as2252258 5b7a33f48b Revert "改名"
This reverts commit fdf58326
2022-01-10 18:48:17 +08:00
as2252258 753e521a41 Revert "改名"
This reverts commit fdf58326
2022-01-10 18:47:31 +08:00
as2252258 d479f36662 Revert "改名"
This reverts commit fdf58326
2022-01-10 18:45:21 +08:00
as2252258 6984f78746 Revert "改名"
This reverts commit fdf58326
2022-01-10 18:44:20 +08:00
as2252258 030f337e74 Revert "改名"
This reverts commit fdf58326
2022-01-10 18:36:35 +08:00
11 changed files with 1258 additions and 1370 deletions
+37
View File
@@ -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
View File
@@ -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
View File
@@ -2,7 +2,7 @@
namespace Kiri; 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 string $host
* @param int $port * @param int $port
* @param bool $isSsl * @param bool $isSsl
*/ */
public function __construct(string $host, int $port, bool $isSsl = false) 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); $this->abstracts = new CoroutineClient($host, $port, $isSsl);
} else { } else {
$this->abstracts = new CurlClient($host, $port, $isSsl); $this->abstracts = new CurlClient($host, $port, $isSsl);
+651 -759
View File
File diff suppressed because it is too large Load Diff
+121 -91
View File
@@ -10,120 +10,150 @@ declare(strict_types=1);
namespace Kiri; namespace Kiri;
use Exception; use Exception;
use JetBrains\PhpStorm\Pure;
use Kiri\Abstracts\Logger;
use Kiri\Kiri;
use Swoole\Coroutine\Http\Client as SwowClient; use Swoole\Coroutine\Http\Client as SwowClient;
/** /**
* Class Client * Class Client
* @package Kiri\Kiri\Http * @package Kiri\Http
*/ */
class CoroutineClient extends ClientAbstracts class CoroutineClient extends ClientAbstracts
{ {
use TSwooleClient; use TSwooleClient;
/** /**
* @param string $method * @param string $method
* @param $path * @param $path
* @param array $params * @param array|string $params
* @return void * @return void
* @throws Exception * @throws
*/ */
public function request(string $method, $path, array $params = []): void public function request(string $method, $path, array|string $params = []): void
{ {
$this->withMethod($method) if (!str_starts_with($path, '/')) {
->coroutine( $path = '/' . $path;
$this->matchHost($path), }
$this->paramEncode($params)
);
}
$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 * @return $this
*/ */
public function withCAInfo($path): static public function withCAInfo(string $path): static
{ {
return $this; return $this;
} }
/** /**
* @param $url * @param string $url
* @param array|string $data * @param array|string $data
* @throws Exception 使用swoole协程方式请求 */
*/ private function coroutine(string $url, array|string $data = []): void
private function coroutine($url, array|string $data = []): void {
{ try {
try { $this->generate_client($this->host, $this->isSSL);
$this->generate_client($data, ...$url); if ($this->client->statusCode < 0) {
if ($this->client->statusCode < 0) { throw new Exception($this->client->errMsg);
throw new Exception($this->client->errMsg); }
}
$this->setStatusCode($this->client->getStatusCode()); $this->execute($url, $data);
$this->setBody($this->client->getBody()); } catch (\Throwable $exception) {
$this->setResponseHeader($this->client->headers); $this->statusCode = -1;
} catch (\Throwable $exception) { $this->body = json_encode(['code' => 500, 'message' => $exception->getMessage()]);
Kiri::getDi()->get(Logger::class)->error('rpc', [$exception]); }
$this->setStatusCode(-1); }
$this->setBody(jTraceEx($exception));
}
}
/** /**
* @param $data * @param string $path
* @param $host * @param array|string $data
* @param $isHttps * @return void
* @param $path */
*/ private function execute(string $path, array|string $data): void
private function generate_client($data, $host, $isHttps, $path): void {
{ $this->client->execute($this->setParams($path, $data));
if ($isHttps || $this->isSSL()) { if (in_array($this->client->getStatusCode(), [502, 404])) {
$this->client = new SwowClient($host, 443, true); $this->retry($path, $data);
} else { } else {
$this->client = new SwowClient($host, $this->getPort(), false); $this->statusCode = $this->client->getStatusCode();
} $this->body = $this->client->getBody();
$this->client->set($this->settings()); $this->setResponseHeader($this->client->headers);
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 $path * @param string $path
* @param $data * @param array|string $data
* @return string * @return void
*/ */
private function setParams($path, $data): string private function retry(string $path, array|string $data): void
{ {
$content = $this->getData()->getContents(); if (($this->num += 1) <= $this->retryNum) {
if (!empty($content)) { sleep($this->retryTimeout);
$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;
}
/** $this->execute($path, $data);
* } else {
*/ $this->statusCode = $this->client->statusCode;
public function close(): void $this->body = $this->client->errMsg;
{ }
$this->client->close(); }
}
/**
* @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
View File
@@ -4,205 +4,227 @@ declare(strict_types=1);
namespace Kiri; namespace Kiri;
use Exception;
use Kiri\Message\Response;
use Kiri\Message\Stream;
use JetBrains\PhpStorm\Pure;
use Psr\Http\Message\ResponseInterface;
/** /**
* Class CurlClient * Class CurlClient
*
* @package Http\Handler\Client * @package Http\Handler\Client
*/ */
class CurlClient extends ClientAbstracts 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 string $path
* @param $method * @param string $method
* @param $params * @param $params
* @throws Exception *
*/ * @return void
private function getCurlHandler($path, $method, $params): void */
{ private function getCurlHandler(string $path, string $method, $params): void
[$host, $isHttps, $path] = $this->matchHost($path); {
$host = $this->isSSL ? 'https://' . $this->host : 'http://' . $this->host;
$host = $isHttps ? 'https://' . $host : 'http://' . $host; if ($this->getPort() != 443 && $this->getPort() != 80) {
if ($this->getPort() != 443 && $this->getPort() != 80) { $host .= ':' . $this->getPort();
$host .= ':' . $this->getPort(); }
} $this->do(curl_init($host . $path), $host . $path, $method);
$this->do(curl_init($host . $path), $host . $path, $method); if ($this->isSSL) {
if ($isHttps !== FALSE) { $this->curlHandlerSslSet();
$this->curlHandlerSslSet(); }
} $contents = $this->_data;
$contents = $this->getData()->getContents(); if (empty($params) && empty($contents)) {
if (empty($params) && empty($contents)) { return;
return; }
} if (!empty($contents)) {
if (!empty($contents)) { curl_setopt($this->client, CURLOPT_POSTFIELDS, $contents);
curl_setopt($this->client, CURLOPT_POSTFIELDS, $contents); } else if ($method === self::UPLOAD) {
} else if ($method === self::POST) { curl_setopt($this->client, CURLOPT_POSTFIELDS, $params);
curl_setopt($this->client, CURLOPT_POSTFIELDS, $this->mergeParams($params)); } else if ($method === self::POST) {
} else if ($method === self::UPLOAD) { if (is_array($params)) {
curl_setopt($this->client, CURLOPT_POSTFIELDS, $params); $params = http_build_query($params);
} }
} curl_setopt($this->client, CURLOPT_POSTFIELDS, $params);
}
}
/** /**
* @return void * @return void
* @throws Exception * @throws
*/ */
private function curlHandlerSslSet(): void private function curlHandlerSslSet(): void
{ {
if (!empty($this->getSslKeyFile()) && file_exists($this->getSslKeyFile())) { if (!empty($this->ssl_key_file) && file_exists($this->ssl_key_file)) {
curl_setopt($this->client, CURLOPT_SSLKEY, $this->getSslKeyFile()); curl_setopt($this->client, CURLOPT_SSLKEY, $this->ssl_key_file);
} }
if (!empty($this->getSslCertFile()) && file_exists($this->getSslCertFile())) { if (!empty($this->ssl_cert_file) && file_exists($this->ssl_cert_file)) {
curl_setopt($this->client, CURLOPT_SSLCERT, $this->getSslCertFile()); 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 mixed $resource
* @param $path * @param string $path
* @param $method * @param string $method
* @throws Exception *
*/ * @return void
private function do($resource, $path, $method): void */
{ private function do(mixed $resource, string $path, string $method): void
curl_setopt($resource, CURLOPT_URL, $path); {
curl_setopt($resource, CURLOPT_TIMEOUT, $this->getTimeout()); // 超时设置 curl_setopt($resource, CURLOPT_URL, $path);
curl_setopt($resource, CURLOPT_CONNECTTIMEOUT, $this->getConnectTimeout()); // 超时设置 curl_setopt($resource, CURLOPT_TIMEOUT, $this->timeout); // 超时设置
curl_setopt($resource, CURLOPT_HEADER, TRUE); curl_setopt($resource, CURLOPT_CONNECTTIMEOUT, $this->connect_timeout); // 超时设置
curl_setopt($resource, CURLOPT_FAILONERROR, TRUE); curl_setopt($resource, CURLOPT_HEADER, TRUE);
curl_setopt($resource, CURLOPT_HTTPHEADER, $this->parseHeaderMat()); curl_setopt($resource, CURLOPT_FAILONERROR, TRUE);
if (defined('CURLOPT_SSL_FALSESTART')) { curl_setopt($resource, CURLOPT_HTTPHEADER, $this->parseHeaderMat());
curl_setopt($resource, CURLOPT_SSL_FALSESTART, TRUE); 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); curl_setopt($resource, CURLOPT_FORBID_REUSE, FALSE);
if (!empty($this->getAgent())) { curl_setopt($resource, CURLOPT_FRESH_CONNECT, FALSE);
curl_setopt($resource, CURLOPT_USERAGENT, $this->getAgent()); 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_NOBODY, FALSE);
curl_setopt($resource, CURLOPT_FOLLOWLOCATION, TRUE);// 跟踪重定向 curl_setopt($resource, CURLOPT_RETURNTRANSFER, TRUE);//返回内容
curl_setopt($resource, CURLOPT_ENCODING, 'gzip,deflate'); curl_setopt($resource, CURLOPT_FOLLOWLOCATION, TRUE);// 跟踪重定向
if ($method === self::POST || $method == self::UPLOAD) { curl_setopt($resource, CURLOPT_ENCODING, 'gzip,deflate');
curl_setopt($resource, CURLOPT_POST, 1); if ($method === self::POST || $method == self::UPLOAD) {
} curl_setopt($resource, CURLOPT_POST, 1);
curl_setopt($resource, CURLOPT_CUSTOMREQUEST, strtoupper($method)); }
$this->client = $resource; [$proxy, $port] = [$this->proxyHost, $this->proxyPort];
if (!empty($this->caPath)) { if (!empty($proxy) && $port > 0) {
curl_setopt($this->client,CURLOPT_CAINFO, $this->caPath); 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 * @param string $path
* @return $this *
*/ * @return $this
public function withCAInfo($path): static */
{ public function withCAInfo(string $path): static
$this->caPath = $path; {
return $this; $this->caPath = $path;
} return $this;
}
/** /**
* @throws Exception * @throws
*/ */
private function execute(): void private function execute(): void
{ {
$output = curl_exec($this->client); $output = curl_exec($this->client);
if ($output === FALSE) { if ($output !== FALSE) {
$this->setStatusCode(curl_errno($this->client)); $this->explode($output);
$this->setBody(curl_error($this->client)); } else {
} else { $this->statusCode = curl_errno($this->client);
$this->explode($output); $this->body = curl_error($this->client);
} }
} $this->close();
}
/** /**
* * @return void
*/ * @throws
public function close(): void */
{ private function retry(): void
curl_close($this->client); {
} 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 public function close(): void
*/ {
private function explode($output): void if (PHP_VERSION < 80000) {
{ curl_close($this->client);
[$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;
}
/** /**
* @return array * @param string $output
*/ *
#[Pure] private function parseHeaderMat(): array * @return void
{ * @throws
$headers = []; */
foreach ($this->getHeader() as $key => $val) { private function explode(string $output): void
$headers[$key] = $key . ': ' . $val; {
} // 获取 HTTP 状态码
return array_values($headers); $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
View File
@@ -11,99 +11,98 @@ use Exception;
*/ */
class HttpParse class HttpParse
{ {
/** /**
* @param mixed ...$object * @param mixed ...$object
* @return string * @return string
*/ */
private static function getKey(...$object): string private static function getKey(...$object): string
{ {
$first = ''; $first = '';
$tp = []; $tp = [];
foreach ($object as $key => $value) { foreach ($object as $value) {
if ($value === null) { if ($value === null) {
continue; continue;
} }
if (is_array($value)) { if (is_array($value)) {
$value = key($value); $value = key($value);
} }
if ($first === '') { if ($first === '') {
$first = $value; $first = $value;
} else { } else {
$tp[] = $value; $tp[] = $value;
} }
} }
$key = $first . '[' . implode('][', $tp) . ']'; $key = $first . '[' . implode('][', $tp) . ']';
if (count($tp) < 1) { if (count($tp) < 1) {
$key = $first; $key = $first;
} }
return $key; return $key;
} }
/** /**
* @param $data * @param array|string $data
* @return string * @return string
* @throws Exception */
*/ public static function parse(array|string $data): string
public static function parse($data): string {
{ $tmp = [];
$tmp = []; if (is_string($data)) {
if (is_string($data)) { return $data;
return $data; }
} foreach ($data as $key => $datum) {
foreach ($data as $key => $datum) { if ($datum === null) {
if ($datum === null) { continue;
continue; }
} $tmp[] = static::ifElse($key, $datum);
$tmp[] = static::ifElse($key, $datum); }
} return implode('&', $tmp);
return implode('&', $tmp); }
}
/** /**
* @param $t * @param $t
* @param $qt * @param $qt
* @return string * @return string
* @throws Exception * @throws
*/ */
private static function ifElse($t, $qt): string private static function ifElse(string $t, mixed $qt): string
{ {
if (is_numeric($qt)) { if (is_numeric($qt)) {
return $t . '=' . $qt; return $t . '=' . $qt;
} }
if (is_string($qt)) { if (is_string($qt)) {
$string = $t . '=' . urlencode($qt); $string = $t . '=' . urlencode($qt);
} else { } else {
$string = static::encode($t, $qt); $string = static::encode($t, $qt);
} }
return $string; return $string;
} }
/** /**
* @param mixed ...$object * @param mixed ...$object
* @return string * @return string
* @throws Exception * @throws
*/ */
private static function encode(...$object): string private static function encode(...$object): string
{ {
$ret = []; $ret = [];
$data = $object[count($object) - 1]; $data = $object[count($object) - 1];
$key = static::getKey(...$object); $key = static::getKey(...$object);
foreach ($data as $s => $datum) { foreach ($data as $s => $datum) {
if (is_array($datum)) { if (is_array($datum)) {
$object[count($object) - 1] = $s; $object[count($object) - 1] = $s;
$object[] = $datum; $object[] = $datum;
$string = static::encode(...$object); $string = static::encode(...$object);
} else { } else {
if (is_object($datum)) { if (is_object($datum)) {
throw new Exception('Http body con\'t object.'); throw new Exception('Http body con\'t object.');
} }
$string = $key . '=' . urlencode($datum); $string = $key . '=' . urlencode($datum);
} }
$ret[] = $string; $ret[] = $string;
} }
return implode('&', $ret); return implode('&', $ret);
} }
} }
+42 -45
View File
@@ -5,25 +5,23 @@ namespace Kiri;
use Closure; use Closure;
use Psr\Http\Message\ResponseInterface;
use Psr\Http\Message\StreamInterface;
interface IClient interface IClient
{ {
/** /**
* @param string $path * @param string $path
* @param array $params * @param array|string $params
*/ */
public function get(string $path, array $params = []): void; public function get(string $path, array|string $params = []): void;
/** /**
* @param string $path * @param string $path
* @param array $params * @param array|string $params
*/ */
public function post(string $path, array $params = []): void; public function post(string $path, array|string $params = []): void;
/** /**
@@ -32,47 +30,47 @@ interface IClient
public function close(): void; public function close(): void;
/** /**
* @param string $path * @param string $path
* @param array $params * @param array|string $params
*/ */
public function delete(string $path, array $params = []): void; public function delete(string $path, array|string $params = []): void;
/** /**
* @param string $path * @param string $path
* @param array $params * @param array|string $params
*/ */
public function options(string $path, array $params = []): void; public function options(string $path, array|string $params = []): void;
/** /**
* @param string $path * @param string $path
* @param array $params * @param array|string $params
*/ */
public function upload(string $path, array $params = []): void; public function upload(string $path, array|string $params = []): void;
/** /**
* @param string $path * @param string $path
* @param array $params * @param array|string $params
*/ */
public function put(string $path, array $params = []): void; public function put(string $path, array|string $params = []): void;
/** /**
* @param string $path * @param string $path
* @param array $params * @param array|string $params
*/ */
public function head(string $path, array $params = []): void; public function head(string $path, array|string $params = []): void;
/** /**
* @param string $method * @param string $method
* @param string $path * @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 * @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; public function withContentType(string $contentType): static;
/** public null|string $body {
* @return mixed get;
*/ }
public function getBody(): mixed;
} }
+26 -23
View File
@@ -2,37 +2,40 @@
namespace Kiri; namespace Kiri;
use JetBrains\PhpStorm\Pure;
trait TSwooleClient trait TSwooleClient
{ {
/**
* @return array
*/
private function settings(): array
{
$sslCert = $this->ssl_cert_file;
$sslKey = $this->ssl_key_file;
$sslCa = $this->ca;
/** $params = [];
* @return array if ($this->connect_timeout > 0) {
*/ $params['timeout'] = $this->connect_timeout;
#[Pure] private function settings(): array }
{
$sslCert = $this->getSslCertFile();
$sslKey = $this->getSslKeyFile();
$sslCa = $this->getCa();
$params = []; [$proxy, $port] = [$this->proxyHost, $this->proxyPort];
if ($this->getConnectTimeout() > 0) { if (!empty($proxy) && $port > 0) {
$params['timeout'] = $this->getConnectTimeout(); $params['http_proxy_host'] = $proxy;
} $params['http_proxy_port'] = $port;
if (empty($sslCert) || empty($sslKey) || empty($sslCa)) {
return $params;
} }
if (empty($sslCert) || empty($sslKey) || empty($sslCa)) {
return $params;
}
$params['ssl_host_name'] = $this->getHost(); $params['ssl_host_name'] = $this->host;
$params['ssl_cert_file'] = $this->getSslCertFile(); $params['ssl_cert_file'] = $this->ssl_cert_file;
$params['ssl_key_file'] = $this->getSslKeyFile(); $params['ssl_key_file'] = $this->ssl_key_file;
$params['ssl_verify_peer'] = TRUE; $params['ssl_verify_peer'] = TRUE;
$params['ssl_cafile'] = $sslCa; $params['ssl_cafile'] = $sslCa;
return $params; return $params;
} }
} }
+87
View File
@@ -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
View File
@@ -9,11 +9,9 @@
], ],
"license": "MIT", "license": "MIT",
"require": { "require": {
"php": ">=8.0", "php": ">=8.5",
"ext-json": "*", "ext-json": "*",
"ext-swoole": "*", "ext-swoole": "*"
"psr/http-client": "^1.0",
"psr/http-message": "^1.0"
}, },
"autoload": { "autoload": {
"psr-4": { "psr-4": {