dubbo RpcContextInterceptor 源码

  • 2022-10-20
  • 浏览 (355)

dubbo RpcContextInterceptor 代码

文件路径:/dubbo-rpc/dubbo-rpc-grpc/src/main/java/org/apache/dubbo/rpc/protocol/grpc/interceptors/RpcContextInterceptor.java

/*
 * Licensed to the Apache Software Foundation (ASF) under one or more
 * contributor license agreements.  See the NOTICE file distributed with
 * this work for additional information regarding copyright ownership.
 * The ASF licenses this file to You under the Apache License, Version 2.0
 * (the "License"); you may not use this file except in compliance with
 * the License.  You may obtain a copy of the License at
 *
 *     http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */
package org.apache.dubbo.rpc.protocol.grpc.interceptors;

import org.apache.dubbo.common.extension.Activate;
import org.apache.dubbo.rpc.RpcContext;

import io.grpc.CallOptions;
import io.grpc.Channel;
import io.grpc.ClientCall;
import io.grpc.ForwardingClientCall;
import io.grpc.ForwardingServerCallListener;
import io.grpc.Metadata;
import io.grpc.MethodDescriptor;
import io.grpc.ServerCall;
import io.grpc.ServerCallHandler;

import java.util.HashMap;
import java.util.Map;
import java.util.Set;

import static io.grpc.Metadata.ASCII_STRING_MARSHALLER;
import static org.apache.dubbo.common.constants.CommonConstants.CONSUMER;
import static org.apache.dubbo.common.constants.CommonConstants.PROVIDER;

/**
 * Hand over context information from Dubbo to gRPC.
 */
@Activate(group = {PROVIDER, CONSUMER})
public class RpcContextInterceptor implements ClientInterceptor, ServerInterceptor {

    private static final String DUBBO = "D-";

    @Override
    public <ReqT, RespT> ClientCall<ReqT, RespT> interceptCall(MethodDescriptor<ReqT, RespT> method, CallOptions callOptions, Channel next) {
        RpcContext rpcContext = RpcContext.getClientAttachment();
        Map<String, Object> attachments = new HashMap<>(rpcContext.getObjectAttachments());

        return new ForwardingClientCall.SimpleForwardingClientCall<ReqT, RespT>(next.newCall(method, callOptions)) {
            @Override
            public void start(Listener<RespT> responseListener, Metadata headers) {
                if (!attachments.isEmpty()) {
                    for (Map.Entry<String, Object> entry : attachments.entrySet()) {
                        // only used for string
                        if (entry.getValue() instanceof String) {
                            headers.put(Metadata.Key.of(DUBBO + entry.getKey(), ASCII_STRING_MARSHALLER), ((String) entry.getValue()));
                        }
                    }
                }
                super.start(responseListener, headers);
            }
        };
    }

    @Override
    public <ReqT, RespT> ServerCall.Listener<ReqT> interceptCall(ServerCall<ReqT, RespT> call, Metadata headers, ServerCallHandler<ReqT, RespT> next) {
        Set<String> keys = headers.keys();
        Map<String, Object> attachments = new HashMap<>(keys.size());
        // filter out all dubbo attachments and save in map
        if (keys != null) {
            keys.stream().filter(k -> k.toUpperCase().startsWith(DUBBO)).forEach(k ->
                    attachments.put(k.substring(DUBBO.length()), headers.get(Metadata.Key.of(k, Metadata.ASCII_STRING_MARSHALLER)))
            );
        }

        return new ForwardingServerCallListener.SimpleForwardingServerCallListener<ReqT>(next.startCall(call, headers)) {
            @Override
            public void onHalfClose() {
                // the client completed all message sending and server will call the biz method if client is not the streaming
                if (call.getMethodDescriptor().getType().clientSendsOneMessage()) {
                    RpcContext.getServerAttachment().setObjectAttachments(attachments);
                }
                super.onHalfClose();
            }

            @Override
            public void onMessage(ReqT message) {
                //server receive the request from client and call the biz method if client is streaming
                if (!call.getMethodDescriptor().getType().clientSendsOneMessage()) {
                    RpcContext.getServerAttachment().setObjectAttachments(attachments);
                }
                super.onMessage(message);
            }
        };
    }

}

相关信息

dubbo 源码目录

相关文章

dubbo ClientInterceptor 源码

dubbo GrpcConfigurator 源码

dubbo ServerInterceptor 源码

dubbo ServerTransportFilter 源码

0  赞