io.grpc.netty.NettyClientHandler.java Source code

Java tutorial

Introduction

Here is the source code for io.grpc.netty.NettyClientHandler.java

Source

/*
 * 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.netty;

import static io.netty.handler.codec.http2.DefaultHttp2LocalFlowController.DEFAULT_WINDOW_UPDATE_RATIO;
import static io.netty.util.CharsetUtil.UTF_8;
import static io.netty.util.internal.ObjectUtil.checkNotNull;

import com.google.common.annotations.VisibleForTesting;
import com.google.common.base.Preconditions;
import com.google.common.base.Stopwatch;
import com.google.common.base.Supplier;
import io.grpc.Attributes;
import io.grpc.InternalChannelz;
import io.grpc.Metadata;
import io.grpc.Status;
import io.grpc.StatusException;
import io.grpc.internal.ClientStreamListener.RpcProgress;
import io.grpc.internal.ClientTransport.PingCallback;
import io.grpc.internal.GrpcAttributes;
import io.grpc.internal.GrpcUtil;
import io.grpc.internal.Http2Ping;
import io.grpc.internal.InUseStateAggregator;
import io.grpc.internal.KeepAliveManager;
import io.grpc.internal.TransportTracer;
import io.grpc.netty.GrpcHttp2HeadersUtils.GrpcHttp2ClientHeadersDecoder;
import io.netty.buffer.ByteBuf;
import io.netty.buffer.ByteBufUtil;
import io.netty.buffer.Unpooled;
import io.netty.channel.Channel;
import io.netty.channel.ChannelFuture;
import io.netty.channel.ChannelFutureListener;
import io.netty.channel.ChannelHandlerContext;
import io.netty.channel.ChannelPromise;
import io.netty.handler.codec.http2.DefaultHttp2Connection;
import io.netty.handler.codec.http2.DefaultHttp2ConnectionDecoder;
import io.netty.handler.codec.http2.DefaultHttp2ConnectionEncoder;
import io.netty.handler.codec.http2.DefaultHttp2FrameReader;
import io.netty.handler.codec.http2.DefaultHttp2FrameWriter;
import io.netty.handler.codec.http2.DefaultHttp2LocalFlowController;
import io.netty.handler.codec.http2.DefaultHttp2RemoteFlowController;
import io.netty.handler.codec.http2.Http2CodecUtil;
import io.netty.handler.codec.http2.Http2Connection;
import io.netty.handler.codec.http2.Http2ConnectionAdapter;
import io.netty.handler.codec.http2.Http2ConnectionDecoder;
import io.netty.handler.codec.http2.Http2Error;
import io.netty.handler.codec.http2.Http2Exception;
import io.netty.handler.codec.http2.Http2FlowController;
import io.netty.handler.codec.http2.Http2FrameAdapter;
import io.netty.handler.codec.http2.Http2FrameLogger;
import io.netty.handler.codec.http2.Http2FrameReader;
import io.netty.handler.codec.http2.Http2FrameWriter;
import io.netty.handler.codec.http2.Http2Headers;
import io.netty.handler.codec.http2.Http2HeadersDecoder;
import io.netty.handler.codec.http2.Http2InboundFrameLogger;
import io.netty.handler.codec.http2.Http2OutboundFrameLogger;
import io.netty.handler.codec.http2.Http2Settings;
import io.netty.handler.codec.http2.Http2Stream;
import io.netty.handler.codec.http2.Http2StreamVisitor;
import io.netty.handler.codec.http2.StreamBufferingEncoder;
import io.netty.handler.codec.http2.WeightedFairQueueByteDistributor;
import io.netty.handler.logging.LogLevel;
import io.perfmark.PerfMark;
import io.perfmark.Tag;
import java.nio.channels.ClosedChannelException;
import java.util.concurrent.Executor;
import java.util.logging.Level;
import java.util.logging.Logger;
import javax.annotation.Nullable;

/**
 * Client-side Netty handler for GRPC processing. All event handlers are executed entirely within
 * the context of the Netty Channel thread.
 */
class NettyClientHandler extends AbstractNettyHandler {
    private static final Logger logger = Logger.getLogger(NettyClientHandler.class.getName());

    /**
     * A message that simply passes through the channel without any real processing. It is useful to
     * check if buffers have been drained and test the health of the channel in a single operation.
     */
    static final Object NOOP_MESSAGE = new Object();

    /**
     * Status used when the transport has exhausted the number of streams.
     */
    private static final Status EXHAUSTED_STREAMS_STATUS = Status.UNAVAILABLE
            .withDescription("Stream IDs have been exhausted");
    private static final long USER_PING_PAYLOAD = 1111;

    private final Http2Connection.PropertyKey streamKey;
    private final ClientTransportLifecycleManager lifecycleManager;
    private final KeepAliveManager keepAliveManager;
    // Returns new unstarted stopwatches
    private final Supplier<Stopwatch> stopwatchFactory;
    private final TransportTracer transportTracer;
    private final Attributes eagAttributes;
    private final String authority;
    private final InUseStateAggregator<Http2Stream> inUseState = new InUseStateAggregator<Http2Stream>() {
        @Override
        protected void handleInUse() {
            lifecycleManager.notifyInUse(true);
        }

        @Override
        protected void handleNotInUse() {
            lifecycleManager.notifyInUse(false);
        }
    };

    private WriteQueue clientWriteQueue;
    private Http2Ping ping;
    private Attributes attributes;
    private InternalChannelz.Security securityInfo;

    static NettyClientHandler newHandler(ClientTransportLifecycleManager lifecycleManager,
            @Nullable KeepAliveManager keepAliveManager, int flowControlWindow, int maxHeaderListSize,
            Supplier<Stopwatch> stopwatchFactory, Runnable tooManyPingsRunnable, TransportTracer transportTracer,
            Attributes eagAttributes, String authority) {
        Preconditions.checkArgument(maxHeaderListSize > 0, "maxHeaderListSize must be positive");
        Http2HeadersDecoder headersDecoder = new GrpcHttp2ClientHeadersDecoder(maxHeaderListSize);
        Http2FrameReader frameReader = new DefaultHttp2FrameReader(headersDecoder);
        Http2FrameWriter frameWriter = new DefaultHttp2FrameWriter();
        Http2Connection connection = new DefaultHttp2Connection(false);
        WeightedFairQueueByteDistributor dist = new WeightedFairQueueByteDistributor(connection);
        dist.allocationQuantum(16 * 1024); // Make benchmarks fast again.
        DefaultHttp2RemoteFlowController controller = new DefaultHttp2RemoteFlowController(connection, dist);
        connection.remote().flowController(controller);

        return newHandler(connection, frameReader, frameWriter, lifecycleManager, keepAliveManager,
                flowControlWindow, maxHeaderListSize, stopwatchFactory, tooManyPingsRunnable, transportTracer,
                eagAttributes, authority);
    }

    @VisibleForTesting
    static NettyClientHandler newHandler(final Http2Connection connection, Http2FrameReader frameReader,
            Http2FrameWriter frameWriter, ClientTransportLifecycleManager lifecycleManager,
            KeepAliveManager keepAliveManager, int flowControlWindow, int maxHeaderListSize,
            Supplier<Stopwatch> stopwatchFactory, Runnable tooManyPingsRunnable, TransportTracer transportTracer,
            Attributes eagAttributes, String authority) {
        Preconditions.checkNotNull(connection, "connection");
        Preconditions.checkNotNull(frameReader, "frameReader");
        Preconditions.checkNotNull(lifecycleManager, "lifecycleManager");
        Preconditions.checkArgument(flowControlWindow > 0, "flowControlWindow must be positive");
        Preconditions.checkArgument(maxHeaderListSize > 0, "maxHeaderListSize must be positive");
        Preconditions.checkNotNull(stopwatchFactory, "stopwatchFactory");
        Preconditions.checkNotNull(tooManyPingsRunnable, "tooManyPingsRunnable");
        Preconditions.checkNotNull(eagAttributes, "eagAttributes");
        Preconditions.checkNotNull(authority, "authority");

        Http2FrameLogger frameLogger = new Http2FrameLogger(LogLevel.DEBUG, NettyClientHandler.class);
        frameReader = new Http2InboundFrameLogger(frameReader, frameLogger);
        frameWriter = new Http2OutboundFrameLogger(frameWriter, frameLogger);

        StreamBufferingEncoder encoder = new StreamBufferingEncoder(
                new DefaultHttp2ConnectionEncoder(connection, frameWriter));

        // Create the local flow controller configured to auto-refill the connection window.
        connection.local()
                .flowController(new DefaultHttp2LocalFlowController(connection, DEFAULT_WINDOW_UPDATE_RATIO, true));

        Http2ConnectionDecoder decoder = new DefaultHttp2ConnectionDecoder(connection, encoder, frameReader);

        transportTracer.setFlowControlWindowReader(new TransportTracer.FlowControlReader() {
            final Http2FlowController local = connection.local().flowController();
            final Http2FlowController remote = connection.remote().flowController();

            @Override
            public TransportTracer.FlowControlWindows read() {
                return new TransportTracer.FlowControlWindows(local.windowSize(connection.connectionStream()),
                        remote.windowSize(connection.connectionStream()));
            }
        });

        Http2Settings settings = new Http2Settings();
        settings.pushEnabled(false);
        settings.initialWindowSize(flowControlWindow);
        settings.maxConcurrentStreams(0);
        settings.maxHeaderListSize(maxHeaderListSize);

        return new NettyClientHandler(decoder, encoder, settings, lifecycleManager, keepAliveManager,
                stopwatchFactory, tooManyPingsRunnable, transportTracer, eagAttributes, authority);
    }

    private NettyClientHandler(Http2ConnectionDecoder decoder, StreamBufferingEncoder encoder,
            Http2Settings settings, ClientTransportLifecycleManager lifecycleManager,
            KeepAliveManager keepAliveManager, Supplier<Stopwatch> stopwatchFactory,
            final Runnable tooManyPingsRunnable, TransportTracer transportTracer, Attributes eagAttributes,
            String authority) {
        super(/* channelUnused= */ null, decoder, encoder, settings);
        this.lifecycleManager = lifecycleManager;
        this.keepAliveManager = keepAliveManager;
        this.stopwatchFactory = stopwatchFactory;
        this.transportTracer = Preconditions.checkNotNull(transportTracer);
        this.eagAttributes = eagAttributes;
        this.authority = authority;
        this.attributes = Attributes.newBuilder().set(GrpcAttributes.ATTR_CLIENT_EAG_ATTRS, eagAttributes).build();

        // Set the frame listener on the decoder.
        decoder().frameListener(new FrameListener());

        Http2Connection connection = encoder.connection();
        streamKey = connection.newKey();

        connection.addListener(new Http2ConnectionAdapter() {
            @Override
            public void onGoAwayReceived(int lastStreamId, long errorCode, ByteBuf debugData) {
                byte[] debugDataBytes = ByteBufUtil.getBytes(debugData);
                goingAway(statusFromGoAway(errorCode, debugDataBytes));
                if (errorCode == Http2Error.ENHANCE_YOUR_CALM.code()) {
                    String data = new String(debugDataBytes, UTF_8);
                    logger.log(Level.WARNING, "Received GOAWAY with ENHANCE_YOUR_CALM. Debug data: {1}", data);
                    if ("too_many_pings".equals(data)) {
                        tooManyPingsRunnable.run();
                    }
                }
            }

            @Override
            public void onStreamActive(Http2Stream stream) {
                if (connection().numActiveStreams() == 1 && NettyClientHandler.this.keepAliveManager != null) {
                    NettyClientHandler.this.keepAliveManager.onTransportActive();
                }
            }

            @Override
            public void onStreamClosed(Http2Stream stream) {
                // Although streams with CALL_OPTIONS_RPC_OWNED_BY_BALANCER are not marked as "in-use" in
                // the first place, we don't propagate that option here, and it's safe to reset the in-use
                // state for them, which will be a cheap no-op.
                inUseState.updateObjectInUse(stream, false);
                if (connection().numActiveStreams() == 0 && NettyClientHandler.this.keepAliveManager != null) {
                    NettyClientHandler.this.keepAliveManager.onTransportIdle();
                }
            }
        });
    }

    /**
     * The protocol negotiation attributes, available once the protocol negotiation completes;
     * otherwise returns {@code Attributes.EMPTY}.
     */
    Attributes getAttributes() {
        return attributes;
    }

    /**
     * Handler for commands sent from the stream.
     */
    @Override
    public void write(ChannelHandlerContext ctx, Object msg, ChannelPromise promise) throws Exception {
        if (msg instanceof CreateStreamCommand) {
            createStream((CreateStreamCommand) msg, promise);
        } else if (msg instanceof SendGrpcFrameCommand) {
            sendGrpcFrame(ctx, (SendGrpcFrameCommand) msg, promise);
        } else if (msg instanceof CancelClientStreamCommand) {
            cancelStream(ctx, (CancelClientStreamCommand) msg, promise);
        } else if (msg instanceof SendPingCommand) {
            sendPingFrame(ctx, (SendPingCommand) msg, promise);
        } else if (msg instanceof GracefulCloseCommand) {
            gracefulClose(ctx, (GracefulCloseCommand) msg, promise);
        } else if (msg instanceof ForcefulCloseCommand) {
            forcefulClose(ctx, (ForcefulCloseCommand) msg, promise);
        } else if (msg == NOOP_MESSAGE) {
            ctx.write(Unpooled.EMPTY_BUFFER, promise);
        } else {
            throw new AssertionError("Write called for unexpected type: " + msg.getClass().getName());
        }
    }

    void startWriteQueue(Channel channel) {
        clientWriteQueue = new WriteQueue(channel);
    }

    WriteQueue getWriteQueue() {
        return clientWriteQueue;
    }

    ClientTransportLifecycleManager getLifecycleManager() {
        return lifecycleManager;
    }

    /**
     * Returns the given processed bytes back to inbound flow control.
     */
    void returnProcessedBytes(Http2Stream stream, int bytes) {
        try {
            decoder().flowController().consumeBytes(stream, bytes);
        } catch (Http2Exception e) {
            throw new RuntimeException(e);
        }
    }

    private void onHeadersRead(int streamId, Http2Headers headers, boolean endStream) {
        // Stream 1 is reserved for the Upgrade response, so we should ignore its headers here:
        if (streamId != Http2CodecUtil.HTTP_UPGRADE_STREAM_ID) {
            NettyClientStream.TransportState stream = clientStream(requireHttp2Stream(streamId));
            PerfMark.event("NettyClientHandler.onHeadersRead", stream.tag());
            stream.transportHeadersReceived(headers, endStream);
        }

        if (keepAliveManager != null) {
            keepAliveManager.onDataReceived();
        }
    }

    /**
     * Handler for an inbound HTTP/2 DATA frame.
     */
    private void onDataRead(int streamId, ByteBuf data, int padding, boolean endOfStream) {
        flowControlPing().onDataRead(data.readableBytes(), padding);
        NettyClientStream.TransportState stream = clientStream(requireHttp2Stream(streamId));
        PerfMark.event("NettyClientHandler.onDataRead", stream.tag());
        stream.transportDataReceived(data, endOfStream);
        if (keepAliveManager != null) {
            keepAliveManager.onDataReceived();
        }
    }

    /**
     * Handler for an inbound HTTP/2 RST_STREAM frame, terminating a stream.
     */
    private void onRstStreamRead(int streamId, long errorCode) {
        NettyClientStream.TransportState stream = clientStream(connection().stream(streamId));
        if (stream != null) {
            PerfMark.event("NettyClientHandler.onRstStreamRead", stream.tag());
            Status status = GrpcUtil.Http2Error.statusForCode((int) errorCode)
                    .augmentDescription("Received Rst Stream");
            stream.transportReportStatus(status,
                    errorCode == Http2Error.REFUSED_STREAM.code() ? RpcProgress.REFUSED : RpcProgress.PROCESSED,
                    false /*stop delivery*/, new Metadata());
            if (keepAliveManager != null) {
                keepAliveManager.onDataReceived();
            }
        }
    }

    @Override
    public void close(ChannelHandlerContext ctx, ChannelPromise promise) throws Exception {
        logger.fine("Network channel being closed by the application.");
        if (ctx.channel().isActive()) { // Ignore notification that the socket was closed
            lifecycleManager
                    .notifyShutdown(Status.UNAVAILABLE.withDescription("Transport closed for unknown reason"));
        }
        super.close(ctx, promise);
    }

    /**
     * Handler for the Channel shutting down.
     */
    @Override
    public void channelInactive(ChannelHandlerContext ctx) throws Exception {
        try {
            logger.fine("Network channel is closed");
            Status status = Status.UNAVAILABLE.withDescription("Network closed for unknown reason");
            lifecycleManager.notifyShutdown(status);
            try {
                cancelPing(lifecycleManager.getShutdownThrowable());
                // Report status to the application layer for any open streams
                connection().forEachActiveStream(new Http2StreamVisitor() {
                    @Override
                    public boolean visit(Http2Stream stream) throws Http2Exception {
                        NettyClientStream.TransportState clientStream = clientStream(stream);
                        if (clientStream != null) {
                            clientStream.transportReportStatus(lifecycleManager.getShutdownStatus(), false,
                                    new Metadata());
                        }
                        return true;
                    }
                });
            } finally {
                lifecycleManager.notifyTerminated(status);
            }
        } finally {
            // Close any open streams
            super.channelInactive(ctx);
            if (keepAliveManager != null) {
                keepAliveManager.onTransportTermination();
            }
        }
    }

    @Override
    public void handleProtocolNegotiationCompleted(Attributes attributes, InternalChannelz.Security securityInfo) {
        this.attributes = this.attributes.toBuilder().setAll(attributes).build();
        this.securityInfo = securityInfo;
        super.handleProtocolNegotiationCompleted(attributes, securityInfo);
        writeBufferingAndRemove(ctx().channel());
    }

    static void writeBufferingAndRemove(Channel channel) {
        checkNotNull(channel, "channel");
        ChannelHandlerContext handlerCtx = channel.pipeline().context(WriteBufferingAndExceptionHandler.class);
        if (handlerCtx == null) {
            return;
        }
        ((WriteBufferingAndExceptionHandler) handlerCtx.handler()).writeBufferedAndRemove(handlerCtx);
    }

    @Override
    public Attributes getEagAttributes() {
        return eagAttributes;
    }

    @Override
    public String getAuthority() {
        return authority;
    }

    InternalChannelz.Security getSecurityInfo() {
        return securityInfo;
    }

    @Override
    protected void onConnectionError(ChannelHandlerContext ctx, boolean outbound, Throwable cause,
            Http2Exception http2Ex) {
        logger.log(Level.FINE, "Caught a connection error", cause);
        lifecycleManager.notifyShutdown(Utils.statusFromThrowable(cause));
        // Parent class will shut down the Channel
        super.onConnectionError(ctx, outbound, cause, http2Ex);
    }

    @Override
    protected void onStreamError(ChannelHandlerContext ctx, boolean outbound, Throwable cause,
            Http2Exception.StreamException http2Ex) {
        // Close the stream with a status that contains the cause.
        NettyClientStream.TransportState stream = clientStream(connection().stream(http2Ex.streamId()));
        if (stream != null) {
            stream.transportReportStatus(Utils.statusFromThrowable(cause), false, new Metadata());
        } else {
            logger.log(Level.FINE, "Stream error for unknown stream " + http2Ex.streamId(), cause);
        }

        // Delegate to the base class to send a RST_STREAM.
        super.onStreamError(ctx, outbound, cause, http2Ex);
    }

    @Override
    protected boolean isGracefulShutdownComplete() {
        // Only allow graceful shutdown to complete after all pending streams have completed.
        return super.isGracefulShutdownComplete() && ((StreamBufferingEncoder) encoder()).numBufferedStreams() == 0;
    }

    /**
     * Attempts to create a new stream from the given command. If there are too many active streams,
     * the creation request is queued.
     */
    private void createStream(CreateStreamCommand command, ChannelPromise promise) throws Exception {
        if (lifecycleManager.getShutdownThrowable() != null) {
            command.stream().setNonExistent();
            // The connection is going away (it is really the GOAWAY case),
            // just terminate the stream now.
            command.stream().transportReportStatus(lifecycleManager.getShutdownStatus(), RpcProgress.REFUSED, true,
                    new Metadata());
            promise.setFailure(lifecycleManager.getShutdownThrowable());
            return;
        }

        // Get the stream ID for the new stream.
        int streamId;
        try {
            streamId = incrementAndGetNextStreamId();
        } catch (StatusException e) {
            command.stream().setNonExistent();
            // Stream IDs have been exhausted for this connection. Fail the promise immediately.
            promise.setFailure(e);

            // Initiate a graceful shutdown if we haven't already.
            if (!connection().goAwaySent()) {
                logger.fine("Stream IDs have been exhausted for this connection. "
                        + "Initiating graceful shutdown of the connection.");
                lifecycleManager.notifyShutdown(e.getStatus());
                close(ctx(), ctx().newPromise());
            }
            return;
        }

        NettyClientStream.TransportState stream = command.stream();
        Http2Headers headers = command.headers();
        stream.setId(streamId);

        PerfMark.startTask("NettyClientHandler.createStream", stream.tag());
        PerfMark.linkIn(command.getLink());
        try {
            createStreamTraced(streamId, stream, headers, command.isGet(), command.shouldBeCountedForInUse(),
                    promise);
        } finally {
            PerfMark.stopTask("NettyClientHandler.createStream", stream.tag());
        }
    }

    private void createStreamTraced(final int streamId, final NettyClientStream.TransportState stream,
            final Http2Headers headers, boolean isGet, final boolean shouldBeCountedForInUse,
            final ChannelPromise promise) {
        // Create an intermediate promise so that we can intercept the failure reported back to the
        // application.
        ChannelPromise tempPromise = ctx().newPromise();
        encoder().writeHeaders(ctx(), streamId, headers, 0, isGet, tempPromise)
                .addListener(new ChannelFutureListener() {
                    @Override
                    public void operationComplete(ChannelFuture future) throws Exception {
                        if (future.isSuccess()) {
                            // The http2Stream will be null in case a stream buffered in the encoder
                            // was canceled via RST_STREAM.
                            Http2Stream http2Stream = connection().stream(streamId);
                            if (http2Stream != null) {
                                stream.getStatsTraceContext().clientOutboundHeaders();
                                http2Stream.setProperty(streamKey, stream);

                                // This delays the in-use state until the I/O completes, which technically may
                                // be later than we would like.
                                if (shouldBeCountedForInUse) {
                                    inUseState.updateObjectInUse(http2Stream, true);
                                }

                                // Attach the client stream to the HTTP/2 stream object as user data.
                                stream.setHttp2Stream(http2Stream);
                            }
                            // Otherwise, the stream has been cancelled and Netty is sending a
                            // RST_STREAM frame which causes it to purge pending writes from the
                            // flow-controller and delete the http2Stream. The stream listener has already
                            // been notified of cancellation so there is nothing to do.

                            // Just forward on the success status to the original promise.
                            promise.setSuccess();
                        } else {
                            final Throwable cause = future.cause();
                            if (cause instanceof StreamBufferingEncoder.Http2GoAwayException) {
                                StreamBufferingEncoder.Http2GoAwayException e = (StreamBufferingEncoder.Http2GoAwayException) cause;
                                lifecycleManager.notifyShutdown(statusFromGoAway(e.errorCode(), e.debugData()));
                                promise.setFailure(lifecycleManager.getShutdownThrowable());
                            } else {
                                promise.setFailure(cause);
                            }
                        }
                    }
                });
    }

    /**
     * Cancels this stream.
     */
    private void cancelStream(ChannelHandlerContext ctx, CancelClientStreamCommand cmd, ChannelPromise promise) {
        NettyClientStream.TransportState stream = cmd.stream();
        PerfMark.startTask("NettyClientHandler.cancelStream", stream.tag());
        PerfMark.linkIn(cmd.getLink());
        try {
            Status reason = cmd.reason();
            if (reason != null) {
                stream.transportReportStatus(reason, true, new Metadata());
            }
            if (!cmd.stream().isNonExistent()) {
                encoder().writeRstStream(ctx, stream.id(), Http2Error.CANCEL.code(), promise);
            } else {
                promise.setSuccess();
            }
        } finally {
            PerfMark.stopTask("NettyClientHandler.cancelStream", stream.tag());
        }
    }

    /**
     * Sends the given GRPC frame for the stream.
     */
    private void sendGrpcFrame(ChannelHandlerContext ctx, SendGrpcFrameCommand cmd, ChannelPromise promise) {
        PerfMark.startTask("NettyClientHandler.sendGrpcFrame", cmd.stream().tag());
        PerfMark.linkIn(cmd.getLink());
        try {
            // Call the base class to write the HTTP/2 DATA frame.
            // Note: no need to flush since this is handled by the outbound flow controller.
            encoder().writeData(ctx, cmd.stream().id(), cmd.content(), 0, cmd.endStream(), promise);
        } finally {
            PerfMark.stopTask("NettyClientHandler.sendGrpcFrame", cmd.stream().tag());
        }
    }

    private void sendPingFrame(ChannelHandlerContext ctx, SendPingCommand msg, ChannelPromise promise) {
        PerfMark.startTask("NettyClientHandler.sendPingFrame");
        PerfMark.linkIn(msg.getLink());
        try {
            sendPingFrameTraced(ctx, msg, promise);
        } finally {
            PerfMark.stopTask("NettyClientHandler.sendPingFrame");
        }
    }

    /**
     * Sends a PING frame. If a ping operation is already outstanding, the callback in the message is
     * registered to be called when the existing operation completes, and no new frame is sent.
     */
    private void sendPingFrameTraced(ChannelHandlerContext ctx, SendPingCommand msg, ChannelPromise promise) {
        // Don't check lifecycleManager.getShutdownStatus() since we want to allow pings after shutdown
        // but before termination. After termination, messages will no longer arrive because the
        // pipeline clears all handlers on channel close.

        PingCallback callback = msg.callback();
        Executor executor = msg.executor();
        // we only allow one outstanding ping at a time, so just add the callback to
        // any outstanding operation
        if (ping != null) {
            promise.setSuccess();
            ping.addCallback(callback, executor);
            return;
        }

        // Use a new promise to prevent calling the callback twice on write failure: here and in
        // NettyClientTransport.ping(). It may appear strange, but it will behave the same as if
        // ping != null above.
        promise.setSuccess();
        promise = ctx().newPromise();
        // set outstanding operation
        long data = USER_PING_PAYLOAD;
        Stopwatch stopwatch = stopwatchFactory.get();
        stopwatch.start();
        ping = new Http2Ping(data, stopwatch);
        ping.addCallback(callback, executor);
        // and then write the ping
        encoder().writePing(ctx, false, USER_PING_PAYLOAD, promise);
        ctx.flush();
        final Http2Ping finalPing = ping;
        promise.addListener(new ChannelFutureListener() {
            @Override
            public void operationComplete(ChannelFuture future) throws Exception {
                if (future.isSuccess()) {
                    transportTracer.reportKeepAliveSent();
                } else {
                    Throwable cause = future.cause();
                    if (cause instanceof ClosedChannelException) {
                        cause = lifecycleManager.getShutdownThrowable();
                        if (cause == null) {
                            cause = Status.UNKNOWN.withDescription("Ping failed but for unknown reason.")
                                    .withCause(future.cause()).asException();
                        }
                    }
                    finalPing.failed(cause);
                    if (ping == finalPing) {
                        ping = null;
                    }
                }
            }
        });
    }

    private void gracefulClose(ChannelHandlerContext ctx, GracefulCloseCommand msg, ChannelPromise promise)
            throws Exception {
        lifecycleManager.notifyShutdown(msg.getStatus());
        // Explicitly flush to create any buffered streams before sending GOAWAY.
        // TODO(ejona): determine if the need to flush is a bug in Netty
        flush(ctx);
        close(ctx, promise);
    }

    private void forcefulClose(final ChannelHandlerContext ctx, final ForcefulCloseCommand msg,
            ChannelPromise promise) throws Exception {
        // close() already called by NettyClientTransport, so just need to clean up streams
        connection().forEachActiveStream(new Http2StreamVisitor() {
            @Override
            public boolean visit(Http2Stream stream) throws Http2Exception {
                NettyClientStream.TransportState clientStream = clientStream(stream);
                Tag tag = clientStream != null ? clientStream.tag() : PerfMark.createTag();
                PerfMark.startTask("NettyClientHandler.forcefulClose", tag);
                PerfMark.linkIn(msg.getLink());
                try {
                    if (clientStream != null) {
                        clientStream.transportReportStatus(msg.getStatus(), true, new Metadata());
                        resetStream(ctx, stream.id(), Http2Error.CANCEL.code(), ctx.newPromise());
                    }
                    stream.close();
                    return true;
                } finally {
                    PerfMark.stopTask("NettyClientHandler.forcefulClose", tag);
                }
            }
        });
        promise.setSuccess();
    }

    /**
     * Handler for a GOAWAY being received. Fails any streams created after the
     * last known stream.
     */
    private void goingAway(Status status) {
        lifecycleManager.notifyShutdown(status);
        final Status goAwayStatus = lifecycleManager.getShutdownStatus();
        final int lastKnownStream = connection().local().lastStreamKnownByPeer();
        try {
            connection().forEachActiveStream(new Http2StreamVisitor() {
                @Override
                public boolean visit(Http2Stream stream) throws Http2Exception {
                    if (stream.id() > lastKnownStream) {
                        NettyClientStream.TransportState clientStream = clientStream(stream);
                        if (clientStream != null) {
                            clientStream.transportReportStatus(goAwayStatus, RpcProgress.REFUSED, false,
                                    new Metadata());
                        }
                        stream.close();
                    }
                    return true;
                }
            });
        } catch (Http2Exception e) {
            throw new RuntimeException(e);
        }
    }

    private void cancelPing(Throwable t) {
        if (ping != null) {
            ping.failed(t);
            ping = null;
        }
    }

    private Status statusFromGoAway(long errorCode, byte[] debugData) {
        Status status = GrpcUtil.Http2Error.statusForCode((int) errorCode).augmentDescription("Received Goaway");
        if (debugData != null && debugData.length > 0) {
            // If a debug message was provided, use it.
            String msg = new String(debugData, UTF_8);
            status = status.augmentDescription(msg);
        }
        return status;
    }

    /**
     * Gets the client stream associated to the given HTTP/2 stream object.
     */
    private NettyClientStream.TransportState clientStream(Http2Stream stream) {
        return stream == null ? null : (NettyClientStream.TransportState) stream.getProperty(streamKey);
    }

    private int incrementAndGetNextStreamId() throws StatusException {
        int nextStreamId = connection().local().incrementAndGetNextStreamId();
        if (nextStreamId < 0) {
            logger.fine("Stream IDs have been exhausted for this connection. "
                    + "Initiating graceful shutdown of the connection.");
            throw EXHAUSTED_STREAMS_STATUS.asException();
        }
        return nextStreamId;
    }

    private Http2Stream requireHttp2Stream(int streamId) {
        Http2Stream stream = connection().stream(streamId);
        if (stream == null) {
            // This should never happen.
            throw new AssertionError("Stream does not exist: " + streamId);
        }
        return stream;
    }

    private class FrameListener extends Http2FrameAdapter {
        private boolean firstSettings = true;

        @Override
        public void onSettingsRead(ChannelHandlerContext ctx, Http2Settings settings) {
            if (firstSettings) {
                firstSettings = false;
                lifecycleManager.notifyReady();
            }
        }

        @Override
        public int onDataRead(ChannelHandlerContext ctx, int streamId, ByteBuf data, int padding,
                boolean endOfStream) throws Http2Exception {
            NettyClientHandler.this.onDataRead(streamId, data, padding, endOfStream);
            return padding;
        }

        @Override
        public void onHeadersRead(ChannelHandlerContext ctx, int streamId, Http2Headers headers,
                int streamDependency, short weight, boolean exclusive, int padding, boolean endStream)
                throws Http2Exception {
            NettyClientHandler.this.onHeadersRead(streamId, headers, endStream);
        }

        @Override
        public void onRstStreamRead(ChannelHandlerContext ctx, int streamId, long errorCode) throws Http2Exception {
            NettyClientHandler.this.onRstStreamRead(streamId, errorCode);
        }

        @Override
        public void onPingAckRead(ChannelHandlerContext ctx, long ackPayload) throws Http2Exception {
            Http2Ping p = ping;
            if (ackPayload == flowControlPing().payload()) {
                flowControlPing().updateWindow();
                if (logger.isLoggable(Level.FINE)) {
                    logger.log(Level.FINE, String.format("Window: %d",
                            decoder().flowController().initialWindowSize(connection().connectionStream())));
                }
            } else if (p != null) {
                if (p.payload() == ackPayload) {
                    p.complete();
                    ping = null;
                } else {
                    logger.log(Level.WARNING, String.format("Received unexpected ping ack. Expecting %d, got %d",
                            p.payload(), ackPayload));
                }
            } else {
                logger.warning("Received unexpected ping ack. No ping outstanding");
            }
            if (keepAliveManager != null) {
                keepAliveManager.onDataReceived();
            }
        }

        @Override
        public void onPingRead(ChannelHandlerContext ctx, long data) throws Http2Exception {
            if (keepAliveManager != null) {
                keepAliveManager.onDataReceived();
            }
        }
    }
}