分享一个hyperf的grpc 日志切面

分享一个hyperf的grpc 日志切面, 对于 gRPC 接口,由于返回的是 Protocol Buffers 格式的消息(message),所有需要特殊处理一下。

<?php

declare(strict_types=1);


namespace App\Aspect;

use Hyperf\Codec\Json;
use Hyperf\Context\Context;
use Hyperf\Contract\StdoutLoggerInterface;
use Hyperf\Di\Annotation\Aspect;
use Hyperf\Di\Annotation\Inject;
use Hyperf\Di\Aop\AbstractAspect;
use Hyperf\Di\Aop\ProceedingJoinPoint;
use Hyperf\Grpc\Parser;
use Hyperf\GrpcClient\GrpcClient;
use Hyperf\GrpcClient\Request;
use Swoole\Http2\Response;
use Throwable;

#[Aspect]
class GrpcLogAspect extends AbstractAspect
{
    public array $classes = [
        GrpcClient::class . '::send',
        GrpcClient::class . '::recv',
    ];

    #[Inject]
    protected StdoutLoggerInterface $logger;

    public function process(ProceedingJoinPoint $proceedingJoinPoint)
    {
        return match ($proceedingJoinPoint->methodName) {
            'send' => $this->send($proceedingJoinPoint),
            'recv' => $this->recv($proceedingJoinPoint),
            default => $proceedingJoinPoint->process(),
        };
    }

    private function send(ProceedingJoinPoint $proceedingJoinPoint)
    {
        $request = $proceedingJoinPoint->getArguments()[0];
        $logInfo = [];
        if ($request instanceof Request) {
            // 截取path中的方法名 为类名
            $path = explode('/', $request->path);
            // 截取$path[1].前面的字符串
            $serviceName = substr($path[1], 0, strpos($path[1], '.'));
            // $path[2]首字母大写
            $methodName = ucfirst($path[2]);
            $className = $serviceName . '\\' . $methodName;
            Context::set('grpcSendClassName', $className);
            $data = Parser::deserializeMessage([$className . 'Request', 'decode'], $request->data);
            $logInfo = [
                'path' => $request->path,
                'method' => $request->method,
                'header' => $request->headers,
                'cookies' => $request->cookies,
                'pipeline' => $request->pipeline,
                'usePipelineRead' => $request->usePipelineRead,
            ];
            $this->logger->info(sprintf('grpc send: %s, data: %s', Json::encode($logInfo), $data->serializeToJsonString()));
        }

        try {
            $result = $proceedingJoinPoint->process();
        } catch (Throwable $th) {
            $this->logger->error(sprintf('grpc send error: %s, %s, %s', $th->getMessage(), $th->getCode(), $th->getTraceAsString()));
            throw $th;
        }
        return $result;
    }

    private function recv(ProceedingJoinPoint $proceedingJoinPoint)
    {
        try {
            $response = $proceedingJoinPoint->process();
            if ($response instanceof Response) {
                $className = Context::get('grpcSendClassName');
                $data = Parser::deserializeMessage([$className . 'Response', 'decode'], $response->data);
                $logInfo = [
                    'streamId' => $response->streamId,
                    'errCode' => $response->errCode,
                    'statusCode' => $response->statusCode,
                    'pipeline' => $response->pipeline,
                    'cookies' => $response->cookies,
                    'headers' => $response->headers,
                    'set_cookie_headers' => $response->set_cookie_headers,
                ];
                $this->logger->info(sprintf('grpc recv: %s, data: %s', Json::encode($logInfo), $data->serializeToJsonString()));
            }
        } catch (Throwable $th) {
            $this->logger->error(sprintf('grpc recv error: %s, %s, %s', $th->getMessage(), $th->getCode(), $th->getTraceAsString()));
            throw $th;
        }

        return $response;
    }
}

All posts

Other pages


Deprecated: 文件 没有 comments.php 的主题 自版本 3.0.0 起已弃用,且没有可用的替代。 请在您的主题中包含一个 comments.php 模板。 in /www/wwwroot/liguoqi.site/wp-includes/functions.php on line 6078

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注