This commit is contained in:
Aether
2025-09-28 14:51:38 +08:00
parent afd384a52a
commit 10b58a8c6d
8 changed files with 326 additions and 4 deletions

View File

@@ -0,0 +1,96 @@
<?php
declare(strict_types=1);
namespace Aether\Middleware;
use Aether\RpcException\BusinessException;
use Aether\RpcException\ErrorCode;
use Hyperf\Context\ApplicationContext;
use Hyperf\Context\Context;
use Hyperf\Contract\StdoutLoggerInterface;
use Psr\Http\Message\ResponseInterface;
use Psr\Http\Message\ServerRequestInterface;
use Psr\Http\Server\MiddlewareInterface;
use Psr\Http\Server\RequestHandlerInterface;
use Throwable;
use function Hyperf\support\env;
class GatewayExceptionHandler implements MiddlewareInterface
{
public function process(ServerRequestInterface $request, RequestHandlerInterface $handler): ResponseInterface
{
Context::set(ServerRequestInterface::class, $request);
try {
return $handler->handle($request);
} catch (Throwable $e) {
// 尝试解析异常信息判断是否为RPC服务返回的业务异常
$parsed = $this->parseRpcException($e);
if ($parsed) {
// 转换为业务异常
throw new BusinessException(
$parsed['code'],
$parsed['message'],
$parsed['data'] ?? []
);
}
// 记录非业务异常日志
$logger = ApplicationContext::getContainer()->get(StdoutLoggerInterface::class);
$logger->error(sprintf('服务调用异常: %s', $e->getMessage()), [
'trace' => $e->getTraceAsString(),
'file' => $e->getFile(),
'line' => $e->getLine(),
]);
// 非业务异常,使用系统错误码
if ($e instanceof BusinessException) {
throw $e;
}
// 统一转换为系统错误
throw new BusinessException(
env('APP_ENV') === 'dev' ? $e->getMessage() : ErrorCode::getMessage(ErrorCode::RPC_CALL_ERROR),
ErrorCode::RPC_CALL_ERROR,
);
}
}
/**
* 解析RPC异常信息.
*/
private function parseRpcException(Throwable $e): ?array
{
try {
// 从异常消息中解析JSON数据
$message = $e->getMessage();
$data = json_decode($message, true);
// 检查是否为有效的JSON-RPC错误响应
if (json_last_error() === JSON_ERROR_NONE && isset($data['jsonrpc']) && $data['jsonrpc'] === '2.0') {
$error = $data['error'] ?? [];
// 检查是否包含业务异常标识
return [
'code' => $error['code'] ?? ErrorCode::RPC_CALL_ERROR,
'message' => $error['message'] ?? '服务调用异常',
'data' => $error['data'] ?? [],
];
// 普通RPC错误
}
// 检查是否为直接返回的错误数组
if (is_array($data) && isset($data['code'], $data['message'])) {
return $data;
}
} catch (Throwable $parseError) {
// 解析失败不影响主流程
}
return null;
}
}