91超碰碰碰碰久久久久久综合_超碰av人澡人澡人澡人澡人掠_国产黄大片在线观看画质优化_txt小说免费全本

溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

Java?gRPC攔截器如何實現分布式日志鏈路追蹤器

發布時間:2023-03-01 15:25:41 來源:億速云 閱讀:117 作者:iii 欄目:開發技術

這篇文章主要介紹“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
@GrpcGlobalServerInterceptor
public 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容器管理。所以可以直接通過自動注入的方式使用。

@RestController
public 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攔截器如何實現分布式日志鏈路追蹤器”的內容就介紹到這里了,感謝大家的閱讀。如果想了解更多行業相關的知識,可以關注億速云行業資訊頻道,小編每天都會為大家更新不同的知識點。

向AI問一下細節

免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

AI

南宫市| 咸宁市| 韶关市| 新竹市| 澜沧| 德惠市| 灵石县| 白沙| 汶川县| 福州市| 吉木萨尔县| 金乡县| 吕梁市| 河南省| 吉首市| 英超| 锡林浩特市| 利川市| 南溪县| 镇远县| 棋牌| 新和县| 托里县| 于田县| 通海县| 合山市| 孝感市| 阜宁县| 梨树县| 西和县| 田林县| 广德县| 伊吾县| 延安市| 义马市| 海淀区| 天等县| 额敏县| 嘉善县| 名山县| 视频|