文章详情

短信预约-IT技能 免费直播动态提醒

请输入下面的图形验证码

提交验证

短信预约提醒成功

Java gRPC拦截器如何实现分布式日志链路追踪器

2023-07-05 07:47

关注

这篇文章主要介绍“Java gRPC拦截器如何实现分布式日志链路追踪器”的相关知识,小编通过实际案例向大家展示操作过程,操作方法简单快捷,实用性强,希望这篇“Java gRPC拦截器如何实现分布式日志链路追踪器”文章能帮助大家解决问题。

跨进程链路追踪原理

想要实现跨进程间的分布式链路追踪,就要在发起远程调用的时候通过请求头或者公共的自定义域将链路参数放进去,然后服务端收到请求后将链路参数从请求头或者自定义域中或取出来,就这样一层一层的将链路参数传递下去直至调用结束。

JAVA的gRPC库io.grpc提供了在RPC调用中客户端和服务端的拦截器(Interceptor),通过客户端拦截器我们可以将链路追踪的参数放到gRPC调用的Metadata中,通过服务端拦截器能够从Metadata中获取到链路追踪所传递的参数;io.grpc提供的客户端拦截器和服务端拦截器分别是io.grpc.ClientInterceptorio.grpc.ServerInterceptor

代码实现

maven依赖

    <dependencies>        <dependency>            <groupId>io.grpc</groupId>            <artifactId>grpc-all</artifactId>            <version>${grpc.version}</version>            <scope>provided</scope>        </dependency>        <dependency>            <groupId>net.devh</groupId>            <artifactId>grpc-server-spring-boot-starter</artifactId>            <version>${grpc.starter.version}</version>            <scope>provided</scope>        </dependency>        <dependency>            <groupId>net.devh</groupId>            <artifactId>grpc-client-spring-boot-starter</artifactId>            <version>${grpc.starter.version}</version>            <scope>provided</scope>        </dependency>        <dependency>            <groupId>io.github.redick01</groupId>            <artifactId>log-helper-spring-boot-starter-common</artifactId>            <version>1.0.3-RELEASE</version>        </dependency>    </dependencies>

拦截器实现

@Slf4j@GrpcGlobalClientInterceptor@GrpcGlobalServerInterceptorpublic class GrpcInterceptor extends AbstractInterceptor implements ServerInterceptor, ClientInterceptor {    // 链路追踪参数traceId    private static final Metadata.Key<String> TRACE = Metadata.Key.of("traceId", Metadata.ASCII_STRING_MARSHALLER)    // 链路追踪参数spanId    private static final Metadata.Key<String> SPAN = Metadata.Key.of("spanId", Metadata.ASCII_STRING_MARSHALLER);    // 链路追踪参数parentId    private static final Metadata.Key<String> PARENT = Metadata.Key.of("parentId", Metadata.ASCII_STRING_MARSHALLER);    @Override    public <ReqT, RespT> ClientCall<ReqT, RespT> interceptCall(            MethodDescriptor<ReqT, RespT> methodDescriptor, CallOptions callOptions,            Channel channel) {        StopWatch stopWatch = new StopWatch();        stopWatch.start();        try {            return new ForwardingClientCall.SimpleForwardingClientCall<ReqT, RespT>(channel.newCall(methodDescriptor, callOptions)) {                @Override                public void start(Listener<RespT> responseListener, Metadata headers) {                    // 客户端传递链路追中数据,将数据放到headers中                    String traceId = traceId();                    if (StringUtils.isNotBlank(traceId)) {                        headers.put(TRACE, traceId);                        headers.put(SPAN, spanId());                        headers.put(PARENT, parentId());                    }                    // 继续下一步                    super.start(new ForwardingClientCallListener.SimpleForwardingClientCallListener<RespT>(responseListener) {                        @Override                        public void onHeaders(Metadata headers) {                            // 服务端传递回来的header                            super.onHeaders(headers);                        }                    }, headers);                }            };        } finally {            stopWatch.stop();            log.info(LogUtil.marker(stopWatch.getTime()), "GRPC调用耗时");        }    }    @Override    public <ReqT, RespT> Listener<ReqT> interceptCall(ServerCall<ReqT, RespT> serverCall,            Metadata headers, ServerCallHandler<ReqT, RespT> serverCallHandler) {        // 服务端从headers中获取到链路追踪参数        String traceId = headers.get(TRACE);        String spanId = headers.get(SPAN);        String parentId = headers.get(PARENT);        // 构建当前进程的链路追踪数据并体现在日志中        Tracer.trace(traceId, spanId, parentId);        log.info(LogUtil.marker(), "开始处理");        return serverCallHandler.startCall(new ForwardingServerCall.SimpleForwardingServerCall<ReqT, RespT>(serverCall) {            @Override            public void sendHeaders(Metadata responseHeaders) {                super.sendHeaders(responseHeaders);            }            @Override            public void close(Status status, Metadata trailers) {                super.close(status, trailers);            }        }, headers);    }}

客户端使用

客户端使用代码如下,该使用示例是在我开源的日志工具中的例子,我这里通过springboot自动装配将GrpcInterceptor交由spring容器管理。所以可以直接通过自动注入的方式使用。

@RestControllerpublic class TestController {    @GrpcClient("userClient")    private UserServiceGrpc.UserServiceBlockingStub userService;    @Autowired    private GrpcInterceptor grpcInterceptor;    //@LogMarker(businessDescription = "获取用户名")    @GetMapping("/getUser")    public String getUser()     {        User user = User.newBuilder()                .setUserId(100)                .putHobbys("pingpong", "play pingpong")                .setCode(200)                .build();        Channel channel = ClientInterceptors.intercept(userService.getChannel(), grpcInterceptor);        userService = UserServiceGrpc.newBlockingStub(channel);        User u = userService.getUser(user);        return u.getName();    }}

关于“Java gRPC拦截器如何实现分布式日志链路追踪器”的内容就介绍到这里了,感谢大家的阅读。如果想了解更多行业相关的知识,可以关注编程网行业资讯频道,小编每天都会为大家更新不同的知识点。

阅读原文内容投诉

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

软考中级精品资料免费领

  • 历年真题答案解析
  • 备考技巧名师总结
  • 高频考点精准押题
  • 2024年上半年信息系统项目管理师第二批次真题及答案解析(完整版)

    难度     813人已做
    查看
  • 【考后总结】2024年5月26日信息系统项目管理师第2批次考情分析

    难度     354人已做
    查看
  • 【考后总结】2024年5月25日信息系统项目管理师第1批次考情分析

    难度     318人已做
    查看
  • 2024年上半年软考高项第一、二批次真题考点汇总(完整版)

    难度     435人已做
    查看
  • 2024年上半年系统架构设计师考试综合知识真题

    难度     224人已做
    查看

相关文章

发现更多好内容

猜你喜欢

AI推送时光机
位置:首页-资讯-后端开发
咦!没有更多了?去看看其它编程学习网 内容吧
首页课程
资料下载
问答资讯