All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.apache.dubbo.rpc.protocol.grpc.interceptors.RpcContextInterceptor Maven / Gradle / Ivy

There is a newer version: 3.3.0-beta.2
Show newest version
/*
 * 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 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 org.apache.dubbo.common.extension.Activate;
import org.apache.dubbo.rpc.RpcContext;

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  ClientCall interceptCall(MethodDescriptor method, CallOptions callOptions, Channel next) {
        RpcContext rpcContext = RpcContext.getContext();
        Map attachments = new HashMap<>(rpcContext.getObjectAttachments());

        return new ForwardingClientCall.SimpleForwardingClientCall(next.newCall(method, callOptions)) {
            @Override
            public void start(Listener responseListener, Metadata headers) {
                if (!attachments.isEmpty()) {
                    for (Map.Entry 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  ServerCall.Listener interceptCall(ServerCall call, Metadata headers, ServerCallHandler next) {
        Set keys = headers.keys();
        Map attachments = new HashMap<>();
        // 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(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.getContext().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.getContext().setObjectAttachments(attachments);
                }
                super.onMessage(message);
            }
        };
    }

}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy