tree: 4412f4197a868611e8c064e75067d39a7924189f [path history] [tgz]
  1. client/
  2. server/
  3. README.md
examples/features/metadata_interceptor/README.md

Metadata interceptor example

This example shows how to update metadata from unary and streaming interceptors on the server. Please see grpc-metadata.md for more information.

Try it

go run server/main.go
go run client/main.go

Explanation

Unary interceptor

The interceptor can read existing metadata from the RPC context passed to it. Since Go contexts are immutable, the interceptor will have to create a new context with updated metadata and pass it to the provided handler.

func SomeInterceptor(ctx context.Context, req interface{}, info *grpc.UnaryServerInfo, handler grpc.UnaryHandler) (interface{}, error) {
    // Get the incoming metadata from the RPC context, and add a new
    // key-value pair to it.
    md, ok := metadata.FromIncomingContext(ctx)
    md.Append("key1", "value1")

    // Create a context with the new metadata and pass it to handler.
    ctx = metadata.NewIncomingContext(ctx, md)
    return handler(ctx, req)
}

Streaming interceptor

grpc.ServerStream does not provide a way to modify its RPC context. The streaming interceptor therefore needs to implement the grpc.ServerStream interface and return a context with updated metadata.

The easiest way to do this would be to create a type which embeds the grpc.ServerStream interface and overrides only the Context() method to return a context with updated metadata. The streaming interceptor would then pass this wrapped stream to the provided handler.

type wrappedStream struct {
    grpc.ServerStream
    ctx context.Context
}

func (s *wrappedStream) Context() context.Context {
    return s.ctx
}

func SomeStreamInterceptor(srv interface{}, ss grpc.ServerStream, info *grpc.StreamServerInfo, handler grpc.StreamHandler) error {
    // Get the incoming metadata from the RPC context, and add a new 
    // key-value pair to it.
    md, ok := metadata.FromIncomingContext(ctx)
    md.Append("key1", "value1")

    // Create a context with the new metadata and pass it to handler.
    ctx = metadata.NewIncomingContext(ctx, md)

    return handler(srv, &wrappedStream{ss, ctx})
}