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

io.grpc.stub.MetadataUtils Maven / Gradle / Ivy

There is a newer version: 1.68.1
Show newest version
/*
 * Copyright 2014 The gRPC Authors
 *
 * Licensed 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 io.grpc.stub;

import static com.google.common.base.Preconditions.checkNotNull;

import io.grpc.CallOptions;
import io.grpc.Channel;
import io.grpc.ClientCall;
import io.grpc.ClientInterceptor;
import io.grpc.ExperimentalApi;
import io.grpc.ForwardingClientCall.SimpleForwardingClientCall;
import io.grpc.ForwardingClientCallListener.SimpleForwardingClientCallListener;
import io.grpc.ForwardingServerCall.SimpleForwardingServerCall;
import io.grpc.Metadata;
import io.grpc.MethodDescriptor;
import io.grpc.ServerCall;
import io.grpc.ServerCallHandler;
import io.grpc.ServerInterceptor;
import io.grpc.Status;
import java.util.concurrent.atomic.AtomicReference;

/**
 * Utility functions for binding and receiving headers.
 */
public final class MetadataUtils {
  // Prevent instantiation
  private MetadataUtils() {}

  /**
   * Returns a client interceptor that attaches a set of headers to requests.
   *
   * @param extraHeaders the headers to be passed by each call that is processed by the returned
   *                     interceptor
   */
  public static ClientInterceptor newAttachHeadersInterceptor(Metadata extraHeaders) {
    return new HeaderAttachingClientInterceptor(extraHeaders);
  }

  private static final class HeaderAttachingClientInterceptor implements ClientInterceptor {

    private final Metadata extraHeaders;

    // Non private to avoid synthetic class
    HeaderAttachingClientInterceptor(Metadata extraHeaders) {
      this.extraHeaders = checkNotNull(extraHeaders, "extraHeaders");
    }

    @Override
    public  ClientCall interceptCall(
        MethodDescriptor method, CallOptions callOptions, Channel next) {
      return new HeaderAttachingClientCall<>(next.newCall(method, callOptions));
    }

    private final class HeaderAttachingClientCall
        extends SimpleForwardingClientCall {

      // Non private to avoid synthetic class
      HeaderAttachingClientCall(ClientCall call) {
        super(call);
      }

      @Override
      public void start(Listener responseListener, Metadata headers) {
        headers.merge(extraHeaders);
        super.start(responseListener, headers);
      }
    }
  }

  /**
   * Captures the last received metadata on a channel. Useful for testing.
   *
   * @param headersCapture to record the last received headers
   * @param trailersCapture to record the last received trailers
   * @return an implementation of the channel with captures installed.
   */
  public static ClientInterceptor newCaptureMetadataInterceptor(
      AtomicReference headersCapture, AtomicReference trailersCapture) {
    return new MetadataCapturingClientInterceptor(headersCapture, trailersCapture);
  }

  private static final class MetadataCapturingClientInterceptor implements ClientInterceptor {

    final AtomicReference headersCapture;
    final AtomicReference trailersCapture;

    // Non private to avoid synthetic class
    MetadataCapturingClientInterceptor(
        AtomicReference headersCapture, AtomicReference trailersCapture) {
      this.headersCapture = checkNotNull(headersCapture, "headersCapture");
      this.trailersCapture = checkNotNull(trailersCapture, "trailersCapture");
    }

    @Override
    public  ClientCall interceptCall(
        MethodDescriptor method, CallOptions callOptions, Channel next) {
      return new MetadataCapturingClientCall<>(next.newCall(method, callOptions));
    }

    private final class MetadataCapturingClientCall
        extends SimpleForwardingClientCall {

      // Non private to avoid synthetic class
      MetadataCapturingClientCall(ClientCall call) {
        super(call);
      }

      @Override
      public void start(ClientCall.Listener responseListener, Metadata headers) {
        headersCapture.set(null);
        trailersCapture.set(null);
        super.start(new MetadataCapturingClientCallListener(responseListener), headers);
      }

      private final class MetadataCapturingClientCallListener
          extends SimpleForwardingClientCallListener {

        MetadataCapturingClientCallListener(ClientCall.Listener responseListener) {
          super(responseListener);
        }

        @Override
        public void onHeaders(Metadata headers) {
          headersCapture.set(headers);
          super.onHeaders(headers);
        }

        @Override
        public void onClose(Status status, Metadata trailers) {
          trailersCapture.set(trailers);
          super.onClose(status, trailers);
        }
      }
    }
  }

  /**
   * Returns a ServerInterceptor that adds the specified Metadata to every response stream, one way
   * or another.
   *
   * 

If, absent this interceptor, a stream would have headers, 'extras' will be added to those * headers. Otherwise, 'extras' will be sent as trailers. This pattern is useful when you have * some fixed information, server identity say, that should be included no matter how the call * turns out. The fallback to trailers avoids artificially committing clients to error responses * that could otherwise be retried (see https://grpc.io/docs/guides/retry/ for more). * *

For correct operation, be sure to arrange for this interceptor to run *before* any others * that might add headers. * * @param extras the Metadata to be added to each stream. Caller gives up ownership. */ @ExperimentalApi("https://github.com/grpc/grpc-java/issues/11462") public static ServerInterceptor newAttachMetadataServerInterceptor(Metadata extras) { return new MetadataAttachingServerInterceptor(extras); } private static final class MetadataAttachingServerInterceptor implements ServerInterceptor { private final Metadata extras; MetadataAttachingServerInterceptor(Metadata extras) { this.extras = extras; } @Override public ServerCall.Listener interceptCall( ServerCall call, Metadata headers, ServerCallHandler next) { return next.startCall(new MetadataAttachingServerCall<>(call), headers); } final class MetadataAttachingServerCall extends SimpleForwardingServerCall { boolean headersSent; MetadataAttachingServerCall(ServerCall delegate) { super(delegate); } @Override public void sendHeaders(Metadata headers) { headers.merge(extras); headersSent = true; super.sendHeaders(headers); } @Override public void close(Status status, Metadata trailers) { if (!headersSent) { trailers.merge(extras); } super.close(status, trailers); } } } }





© 2015 - 2024 Weber Informatics LLC | Privacy Policy