com.bloom.zerofs.rest.NettyResponseChannel.java Source code

Java tutorial

Introduction

Here is the source code for com.bloom.zerofs.rest.NettyResponseChannel.java

Source

/**
 * Copyright 2016 Bloom Corp. All rights reserved.
 *
 * 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.
 */
package com.bloom.zerofs.rest;

import io.netty.buffer.ByteBuf;
import io.netty.buffer.Unpooled;
import io.netty.channel.ChannelFuture;
import io.netty.channel.ChannelFutureListener;
import io.netty.channel.ChannelHandlerContext;
import io.netty.channel.ChannelProgressiveFuture;
import io.netty.channel.ChannelProgressivePromise;
import io.netty.channel.ChannelPromise;
import io.netty.handler.codec.http.DefaultFullHttpResponse;
import io.netty.handler.codec.http.DefaultHttpContent;
import io.netty.handler.codec.http.DefaultHttpResponse;
import io.netty.handler.codec.http.DefaultLastHttpContent;
import io.netty.handler.codec.http.FullHttpResponse;
import io.netty.handler.codec.http.HttpContent;
import io.netty.handler.codec.http.HttpHeaders;
import io.netty.handler.codec.http.HttpResponse;
import io.netty.handler.codec.http.HttpResponseStatus;
import io.netty.handler.codec.http.HttpVersion;
import io.netty.handler.codec.http.LastHttpContent;
import io.netty.handler.stream.ChunkedInput;
import io.netty.handler.stream.ChunkedWriteHandler;
import io.netty.util.concurrent.GenericFutureListener;
import io.netty.util.concurrent.GenericProgressiveFutureListener;

import java.nio.ByteBuffer;
import java.nio.channels.ClosedChannelException;
import java.util.ArrayList;
import java.util.Date;
import java.util.GregorianCalendar;
import java.util.List;
import java.util.Queue;
import java.util.concurrent.ConcurrentLinkedQueue;
import java.util.concurrent.Future;
import java.util.concurrent.atomic.AtomicBoolean;
import java.util.concurrent.atomic.AtomicLong;

import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import com.bloom.zerofs.api.rest.ResponseStatus;
import com.bloom.zerofs.api.rest.RestMethod;
import com.bloom.zerofs.api.rest.RestResponseChannel;
import com.bloom.zerofs.api.rest.RestServiceErrorCode;
import com.bloom.zerofs.api.rest.RestServiceException;
import com.bloom.zerofs.api.router.Callback;
import com.bloom.zerofs.api.router.FutureResult;
import com.bloom.zerofs.tools.Utils;

/**
 * Netty specific implementation of {@link RestResponseChannel} used to return responses via Netty. It is supported by
 * an underlying Netty channel whose handle this class has in the form of a {@link ChannelHandlerContext}.
 * <p/>
 * Data is sent in the order that threads call {@link #write(ByteBuffer, Callback)}.
 * <p/>
 * If a write through this class fails at any time, the underlying channel will be closed immediately and no more writes
 * will be accepted and all scheduled writes will be notified of the failure.
 */
class NettyResponseChannel implements RestResponseChannel {
    // add to this list if the connection needs to be closed on certain errors on GET, DELETE and HEAD.
    // for a POST, we always close the connection on error because we expect the channel to be in a bad state.
    static final List<HttpResponseStatus> CLOSE_CONNECTION_ERROR_STATUSES = new ArrayList<>();

    private final ChannelHandlerContext ctx;
    private final NettyMetrics nettyMetrics;
    private final ChannelProgressivePromise writeFuture;
    private final ChunkedWriteHandler chunkedWriteHandler;

    private final Logger logger = LoggerFactory.getLogger(getClass());
    private final HttpResponse responseMetadata = new DefaultHttpResponse(HttpVersion.HTTP_1_1,
            HttpResponseStatus.OK);
    // tracks whether onResponseComplete() has been called. Helps make it idempotent and also treats response channel as
    // closed if this is true.
    private final AtomicBoolean responseCompleteCalled = new AtomicBoolean(false);
    // tracks whether response metadata write has been initiated. Rejects any more attempts at writing metadata after this
    // has been set to true.
    private final AtomicBoolean responseMetadataWriteInitiated = new AtomicBoolean(false);
    private final AtomicLong totalBytesReceived = new AtomicLong(0);
    private final Queue<Chunk> chunksToWrite = new ConcurrentLinkedQueue<Chunk>();
    private final Queue<Chunk> chunksAwaitingCallback = new ConcurrentLinkedQueue<Chunk>();
    private final AtomicLong chunksToWriteCount = new AtomicLong(0);

    private NettyRequest request = null;
    // marked as true if force close is required because close() was called.
    private volatile boolean forceClose = false;
    // the ResponseStatus that will be sent (or has been sent) as a part of the response.
    private volatile ResponseStatus responseStatus = ResponseStatus.Ok;
    // the response metadata that was actually sent.
    private volatile HttpResponse finalResponseMetadata = null;
    // temp variable to hold the error response status which will be overwritten on responseStatus if the error response
    // was successfully sent
    private ResponseStatus errorResponseStatus = null;

    /**
     * Create an instance of NettyResponseChannel that will use {@code ctx} to return responses.
     * @param ctx the {@link ChannelHandlerContext} to use.
     * @param nettyMetrics the {@link NettyMetrics} instance to use.
     */
    public NettyResponseChannel(ChannelHandlerContext ctx, NettyMetrics nettyMetrics) {
        this.ctx = ctx;
        this.nettyMetrics = nettyMetrics;
        chunkedWriteHandler = ctx.pipeline().get(ChunkedWriteHandler.class);
        writeFuture = ctx.newProgressivePromise();
        logger.trace("Instantiated NettyResponseChannel");
    }

    @Override
    public Future<Long> write(ByteBuffer src, Callback<Long> callback) {
        long writeProcessingStartTime = System.currentTimeMillis();
        if (!responseMetadataWriteInitiated.get()) {
            maybeWriteResponseMetadata(responseMetadata, new ResponseMetadataWriteListener());
        }
        Chunk chunk = new Chunk(src, callback);
        chunksToWrite.add(chunk);
        if (!isOpen()) {
            // the isOpen() check is not before addition to the queue because chunks need to be acknowledged in the order
            // they were received. If we don't add it to the queue and clean up, chunks may be acknowledged out of order.
            logger.debug("Scheduling a chunk cleanup on channel {} because response channel is closed",
                    ctx.channel());
            writeFuture.addListener(new CleanupCallback(new ClosedChannelException()));
        } else if (finalResponseMetadata instanceof FullHttpResponse) {
            logger.debug("Scheduling a chunk cleanup on channel {} because Content-Length is 0", ctx.channel());
            Exception exception = null;
            // this is only allowed to be a 0 sized buffer.
            if (src.remaining() > 0) {
                exception = new IllegalStateException(
                        "Provided non zero size content after setting Content-Length to 0");
                if (!writeFuture.isDone()) {
                    writeFuture.setFailure(exception);
                }
            }
            writeFuture.addListener(new CleanupCallback(exception));
        } else if (HttpHeaders.isContentLengthSet(finalResponseMetadata)
                && totalBytesReceived.get() > HttpHeaders.getContentLength(finalResponseMetadata)) {
            Exception exception = new IllegalStateException("Size of provided content [" + totalBytesReceived.get()
                    + "] is greater than Content-Length set [" + HttpHeaders.getContentLength(finalResponseMetadata)
                    + "]");
            if (!writeFuture.isDone()) {
                writeFuture.setFailure(exception);
            }
            writeFuture.addListener(new CleanupCallback(exception));
        } else {
            chunkedWriteHandler.resumeTransfer();
        }

        long writeProcessingTime = System.currentTimeMillis() - writeProcessingStartTime;
        nettyMetrics.writeProcessingTimeInMs.update(writeProcessingTime);
        if (request != null) {
            request.getMetricsTracker().nioMetricsTracker.addToResponseProcessingTime(writeProcessingTime);
        }
        return chunk.future;
    }

    @Override
    public boolean isOpen() {
        return !responseCompleteCalled.get() && ctx.channel().isActive();
    }

    /**
     * {@inheritDoc}
     * <p/>
     * Marks the channel as closed. No further communication will be possible. Any pending writes (that are not already
     * flushed) might be discarded. The process of closing the network channel is also initiated.
     * <p/>
     * The underlying network channel might not be closed immediately but no more writes will be accepted and any calls to
     * {@link #isOpen()} after a call to this function will return {@code false}.
     */
    @Override
    public void close() {
        close(new ClosedChannelException());
    }

    @Override
    public void onResponseComplete(Exception exception) {
        long responseCompleteStartTime = System.currentTimeMillis();
        try {
            if (responseCompleteCalled.compareAndSet(false, true)) {
                logger.trace("Finished responding to current request on channel {}", ctx.channel());
                nettyMetrics.requestCompletionRate.mark();
                if (exception == null) {
                    if (!maybeWriteResponseMetadata(responseMetadata, new ResponseMetadataWriteListener())) {
                        // There were other writes. Let ChunkedWriteHandler finish if it has been kicked off.
                        chunkedWriteHandler.resumeTransfer();
                    }
                } else {
                    log(exception);
                    if (request != null) {
                        request.getMetricsTracker().markFailure();
                    }
                    // need to set writeFuture as failed in case writes have started or chunks have been queued.
                    if (!writeFuture.isDone()) {
                        writeFuture.setFailure(exception);
                    }
                    if (!maybeSendErrorResponse(exception)) {
                        completeRequest(true);
                    }
                }
            } else if (exception != null) {
                // this is probably an attempt to force close the channel *after* the response is already complete.
                log(exception);
                if (!writeFuture.isDone()) {
                    writeFuture.setFailure(exception);
                }
                completeRequest(true);
            }
            long responseFinishProcessingTime = System.currentTimeMillis() - responseCompleteStartTime;
            nettyMetrics.responseFinishProcessingTimeInMs.update(responseFinishProcessingTime);
            if (request != null) {
                request.getMetricsTracker().nioMetricsTracker
                        .addToResponseProcessingTime(responseFinishProcessingTime);
            }
        } catch (Exception e) {
            logger.error("Swallowing exception encountered during onResponseComplete tasks", e);
            nettyMetrics.responseCompleteTasksError.inc();
            if (!writeFuture.isDone()) {
                writeFuture.setFailure(exception);
            }
            completeRequest(true);
        }
    }

    @Override
    public void setStatus(ResponseStatus status) throws RestServiceException {
        responseMetadata.setStatus(getHttpResponseStatus(status));
        responseStatus = status;
        logger.trace("Set status to {} for response on channel {}", responseMetadata.getStatus(), ctx.channel());
    }

    @Override
    public ResponseStatus getStatus() {
        return responseStatus;
    }

    @Override
    public void setHeader(String headerName, Object headerValue) throws RestServiceException {
        setResponseHeader(headerName, headerValue);
    }

    @Override
    public Object getHeader(String headerName) {
        HttpResponse response = finalResponseMetadata;
        if (response == null) {
            response = responseMetadata;
        }
        return HttpHeaders.getHeader(response, headerName);
    }

    /**
     * Sets the request whose response is being served through this instance of NettyResponseChannel.
     * @param request the {@link NettyRequest} whose response is being served through this instance of
     *                NettyResponseChannel.
     */
    void setRequest(NettyRequest request) {
        if (request != null) {
            if (this.request == null) {
                this.request = request;
                HttpHeaders.setKeepAlive(responseMetadata, request.isKeepAlive());
            } else {
                throw new IllegalStateException(
                        "Request has already been set inside NettyResponseChannel for channel {} " + ctx.channel());
            }
        } else {
            throw new IllegalArgumentException("RestRequest provided is null");
        }
    }

    /**
     * Sends a response to the client based on the {@code exception} and marks the channel as closed. The process of
     * closing the network channel is also initiated.
     * <p/>
     * Functionality is the same as {@link #onResponseComplete(Exception)} except that the channel is always closed after
     * the response sending is complete.
     * @param exception the {@link Exception} to use while constructing an error message for the client.
     */
    void close(Exception exception) {
        forceClose = true;
        onResponseComplete(exception);
    }

    /**
     * Closes the request associated with this NettyResponseChannel.
     */
    private void closeRequest() {
        if (request != null && request.isOpen()) {
            request.close();
        }
    }

    /**
     * Writes response metadata to the channel if not already written previously and channel is active.
     * @param responseMetadata the {@link HttpResponse} that needs to be written.
     * @param listener the {@link GenericFutureListener} that needs to be attached to the write.
     * @return {@code true} if response metadata was written to the channel in this call. {@code false} otherwise.
     */
    private boolean maybeWriteResponseMetadata(HttpResponse responseMetadata,
            GenericFutureListener<ChannelFuture> listener) {
        long writeProcessingStartTime = System.currentTimeMillis();
        boolean writtenThisTime = false;
        if (responseMetadataWriteInitiated.compareAndSet(false, true) && ctx.channel().isActive()) {
            // we do some manipulation here for chunking. According to the HTTP spec, we can have either a Content-Length
            // or Transfer-Encoding:chunked, never both. So we check for Content-Length - if it is not there, we add
            // Transfer-Encoding:chunked. Note that sending HttpContent chunks data anyway - we are just explicitly specifying
            // this in the header.
            if (!HttpHeaders.isContentLengthSet(responseMetadata)) {
                // This makes sure that we don't stomp on any existing transfer-encoding.
                HttpHeaders.setTransferEncodingChunked(responseMetadata);
            } else if (HttpHeaders.getContentLength(responseMetadata) == 0
                    && !(responseMetadata instanceof FullHttpResponse)) {
                // if the Content-Length is 0, we can send a FullHttpResponse since there is no content expected.
                FullHttpResponse fullHttpResponse = new DefaultFullHttpResponse(HttpVersion.HTTP_1_1,
                        responseMetadata.getStatus());
                fullHttpResponse.headers().set(responseMetadata.headers());
                responseMetadata = fullHttpResponse;
            }
            logger.trace("Sending response with status {} on channel {}", responseMetadata.getStatus(),
                    ctx.channel());
            finalResponseMetadata = responseMetadata;
            ChannelPromise writePromise = ctx.newPromise().addListener(listener);
            ctx.writeAndFlush(responseMetadata, writePromise);
            writtenThisTime = true;
            long writeProcessingTime = System.currentTimeMillis() - writeProcessingStartTime;
            nettyMetrics.responseMetadataProcessingTimeInMs.update(writeProcessingTime);
        }
        return writtenThisTime;
    }

    /**
     * Sets the value of response headers after making sure that the response metadata is not already sent.
     * @param headerName The name of the header.
     * @param headerValue The intended value of the header.
     * @throws IllegalArgumentException if any of {@code headerName} or {@code headerValue} is null.
     * @throws IllegalStateException if response metadata has already been written to the channel.
     */
    private void setResponseHeader(String headerName, Object headerValue) {
        if (headerName != null && headerValue != null) {
            long startTime = System.currentTimeMillis();
            if (headerValue instanceof Date) {
                HttpHeaders.setDateHeader(responseMetadata, headerName, (Date) headerValue);
            } else {
                HttpHeaders.setHeader(responseMetadata, headerName, headerValue);
            }
            if (responseMetadataWriteInitiated.get()) {
                nettyMetrics.deadResponseAccessError.inc();
                throw new IllegalStateException(
                        "Response metadata changed after it has already been written to the channel");
            } else {
                logger.trace("Header {} set to {} for channel {}", headerName,
                        responseMetadata.headers().get(headerName), ctx.channel());
                nettyMetrics.headerSetTimeInMs.update(System.currentTimeMillis() - startTime);
            }
        } else {
            throw new IllegalArgumentException(
                    "Header name [" + headerName + "] or header value [" + headerValue + "] null");
        }
    }

    /**
     * Builds and sends an error response to the client based on {@code cause}.
     * @param exception the cause of the request handling failure.
     * @return {@code true} if error response was scheduled to be sent. {@code false} otherwise.
     */
    private boolean maybeSendErrorResponse(Exception exception) {
        long processingStartTime = System.currentTimeMillis();
        boolean responseSent = false;
        logger.trace("Sending error response to client on channel {}", ctx.channel());
        FullHttpResponse errorResponse = getErrorResponse(exception);
        if (maybeWriteResponseMetadata(errorResponse, new ErrorResponseWriteListener())) {
            logger.trace("Scheduled error response sending on channel {}", ctx.channel());
            responseStatus = errorResponseStatus;
            responseSent = true;
            long processingTime = System.currentTimeMillis() - processingStartTime;
            nettyMetrics.errorResponseProcessingTimeInMs.update(processingTime);
        } else {
            logger.debug("Could not send error response on channel {}", ctx.channel());
        }
        return responseSent;
    }

    /**
     * Provided a cause, returns an error response with the right status and error message.
     * @param cause the cause of the error.
     * @return a {@link FullHttpResponse} with the error message that can be sent to the client.
     */
    private FullHttpResponse getErrorResponse(Throwable cause) {
        HttpResponseStatus status;
        StringBuilder errReason = new StringBuilder();
        if (cause instanceof RestServiceException) {
            RestServiceErrorCode restServiceErrorCode = ((RestServiceException) cause).getErrorCode();
            errorResponseStatus = ResponseStatus.getResponseStatus(restServiceErrorCode);
            status = getHttpResponseStatus(errorResponseStatus);
            if (status == HttpResponseStatus.BAD_REQUEST) {
                errReason.append(" [").append(Utils.getRootCause(cause).getMessage()).append("]");
            }
        } else {
            nettyMetrics.internalServerErrorCount.inc();
            status = HttpResponseStatus.INTERNAL_SERVER_ERROR;
            errorResponseStatus = ResponseStatus.InternalServerError;
        }
        String fullMsg = "Failure: " + status + errReason;
        logger.trace("Constructed error response for the client - [{}]", fullMsg);
        FullHttpResponse response;
        if (request != null && !request.getRestMethod().equals(RestMethod.HEAD)) {
            response = new DefaultFullHttpResponse(HttpVersion.HTTP_1_1, status,
                    Unpooled.wrappedBuffer(fullMsg.getBytes()));
        } else {
            // for HEAD, we cannot send the actual body but we need to return what the length would have been if this was GET.
            // https://www.w3.org/Protocols/rfc2616/rfc2616-sec9.html (Section 9.4)
            response = new DefaultFullHttpResponse(HttpVersion.HTTP_1_1, status);
        }
        HttpHeaders.setDate(response, new GregorianCalendar().getTime());
        HttpHeaders.setContentLength(response, fullMsg.length());
        HttpHeaders.setHeader(response, HttpHeaders.Names.CONTENT_TYPE, "text/plain; charset=UTF-8");
        boolean keepAlive = !forceClose && HttpHeaders.isKeepAlive(responseMetadata) && request != null
                && !request.getRestMethod().equals(RestMethod.POST)
                && !CLOSE_CONNECTION_ERROR_STATUSES.contains(status);
        HttpHeaders.setKeepAlive(response, keepAlive);
        return response;
    }

    /**
     * Converts a {@link ResponseStatus} into a {@link HttpResponseStatus}.
     * @param responseStatus {@link ResponseStatus} that needs to be mapped to a {@link HttpResponseStatus}.
     * @return the {@link HttpResponseStatus} that maps to the {@link ResponseStatus}.
     */
    private HttpResponseStatus getHttpResponseStatus(ResponseStatus responseStatus) {
        HttpResponseStatus status;
        switch (responseStatus) {
        case Ok:
            status = HttpResponseStatus.OK;
            break;
        case Created:
            status = HttpResponseStatus.CREATED;
            break;
        case Accepted:
            status = HttpResponseStatus.ACCEPTED;
            break;
        case NotModified:
            status = HttpResponseStatus.NOT_MODIFIED;
            break;
        case BadRequest:
            nettyMetrics.badRequestCount.inc();
            status = HttpResponseStatus.BAD_REQUEST;
            break;
        case Unauthorized:
            nettyMetrics.unauthorizedCount.inc();
            status = HttpResponseStatus.UNAUTHORIZED;
            break;
        case NotFound:
            nettyMetrics.notFoundCount.inc();
            status = HttpResponseStatus.NOT_FOUND;
            break;
        case Gone:
            nettyMetrics.goneCount.inc();
            status = HttpResponseStatus.GONE;
            break;
        case Forbidden:
            nettyMetrics.forbiddenCount.inc();
            status = HttpResponseStatus.FORBIDDEN;
            break;
        case ProxyAuthenticationRequired:
            nettyMetrics.proxyAuthRequiredCount.inc();
            status = HttpResponseStatus.PROXY_AUTHENTICATION_REQUIRED;
            break;
        case InternalServerError:
            nettyMetrics.internalServerErrorCount.inc();
            status = HttpResponseStatus.INTERNAL_SERVER_ERROR;
            break;
        default:
            nettyMetrics.unknownResponseStatusCount.inc();
            status = HttpResponseStatus.INTERNAL_SERVER_ERROR;
            break;
        }
        return status;
    }

    /**
     * Completes the request by closing the request and network channel (if {@code closeNetworkChannel} is {@code true}).
     * </p>
     * May also close the channel if the class internally is forcing a close (i.e. if {@link #close()} is called.
     * @param closeNetworkChannel network channel is closed if {@code true}.
     */
    private void completeRequest(boolean closeNetworkChannel) {
        if ((closeNetworkChannel || forceClose) && ctx.channel().isOpen()) {
            writeFuture.addListener(ChannelFutureListener.CLOSE);
            logger.trace("Requested closing of channel {}", ctx.channel());
        }
        closeRequest();
    }

    /**
     * Handles post-mortem of writes that have failed.
     * @param cause the cause of the failure.
     * @param propagateErrorIfRequired if {@code true} and {@code cause} is not an instance of {@link Exception}, the
     *                                 error is propagated through the netty pipeline.
     */
    private void handleChannelWriteFailure(Throwable cause, boolean propagateErrorIfRequired) {
        long writeFailureProcessingStartTime = System.currentTimeMillis();
        try {
            nettyMetrics.channelWriteError.inc();
            Exception exception;
            if (!(cause instanceof Exception)) {
                logger.warn("Encountered a throwable on channel write failure", cause);
                exception = new IllegalStateException("Encountered a Throwable - " + cause.getMessage());
                if (propagateErrorIfRequired) {
                    // we can't ignore throwables - so we let Netty deal with it.
                    ctx.fireExceptionCaught(cause);
                    nettyMetrics.throwableCount.inc();
                }
            } else {
                exception = (Exception) cause;
            }
            onResponseComplete(exception);
            cleanupChunks(exception);
        } finally {
            nettyMetrics.channelWriteFailureProcessingTimeInMs
                    .update(System.currentTimeMillis() - writeFailureProcessingStartTime);
        }
    }

    /**
     * Logs the exception at the appropriate level.
     * @param exception the {@link Exception} that has to be logged.
     */
    private void log(Exception exception) {
        if (ctx.channel().isActive()) {
            String uri = "unknown";
            RestMethod restMethod = RestMethod.UNKNOWN;
            if (request != null) {
                uri = request.getUri();
                restMethod = request.getRestMethod();
            }
            if (exception instanceof RestServiceException) {
                RestServiceErrorCode errorCode = ((RestServiceException) exception).getErrorCode();
                ResponseStatus responseStatus = ResponseStatus.getResponseStatus(errorCode);
                if (responseStatus == ResponseStatus.InternalServerError) {
                    logger.error("Internal error handling request {} with method {}.", uri, restMethod, exception);
                } else {
                    logger.trace("Error handling request {} with method {}.", uri, restMethod, exception);
                }
            } else {
                logger.error("Unexpected error handling request {} with method {}.", uri, restMethod, exception);
            }
        } else {
            logger.debug("Exception encountered after channel {} became inactive", ctx.channel(), exception);
        }
    }

    /**
     * Cleans up all the chunks remaining by invoking their callbacks.
     * @param exception the {@link Exception} to provide in the callback. Can be {@code null}.
     */
    private void cleanupChunks(Exception exception) {
        logger.trace("Cleaning up remaining chunks");
        Chunk chunk = chunksAwaitingCallback.poll();
        while (chunk != null) {
            chunk.resolveChunk(exception);
            chunk = chunksAwaitingCallback.poll();
        }
        chunk = chunksToWrite.poll();
        while (chunk != null) {
            chunk.onDequeue();
            chunk.resolveChunk(exception);
            chunk = chunksToWrite.poll();
        }
    }

    // helper classes

    /**
     * Class that represents a chunk of data.
     */
    private class Chunk {
        /**
         * The future that will be set on chunk resolution.
         */
        final FutureResult<Long> future = new FutureResult<Long>();
        /**
         * The bytes associated with this chunk.
         */
        final ByteBuffer buffer;
        /**
         * The number of bytes that will need to be written.
         */
        final long bytesToBeWritten;
        /**
         * If progress in {@link #writeFuture} becomes greater than this number, the future/callback will be triggered.
         */
        final long writeCompleteThreshold;
        /**
         * Information on whether this is the last chunk.
         * <p/>
         * This value will be {@code true} for at most one chunk. If {@link HttpHeaders.Names#CONTENT_LENGTH} is not set,
         * there will be no chunks with this value as {@code true}.
         */
        final boolean isLast;
        private final Callback<Long> callback;
        private final long chunkQueueStartTime = System.currentTimeMillis();

        private long chunkWriteStartTime;

        /**
         * Creates a chunk.
         * @param buffer the {@link ByteBuffer} that forms the data of this chunk.
         * @param callback the {@link Callback} to invoke when {@link #writeCompleteThreshold} is reached.
         */
        public Chunk(ByteBuffer buffer, Callback<Long> callback) {
            this.buffer = buffer;
            bytesToBeWritten = buffer.remaining();
            this.callback = callback;
            writeCompleteThreshold = totalBytesReceived.addAndGet(bytesToBeWritten);
            chunksToWriteCount.incrementAndGet();

            // if we are here, it means that finalResponseMetadata has been set and there is no danger of it being null
            long contentLength = HttpHeaders.getContentLength(finalResponseMetadata, -1);
            isLast = contentLength != -1 && writeCompleteThreshold >= contentLength;
        }

        /**
         * Does tasks (like tracking) that need to be done when a chunk is dequeued for processing.
         */
        public void onDequeue() {
            chunksToWriteCount.decrementAndGet();
            chunkWriteStartTime = System.currentTimeMillis();
            long chunkQueueTime = chunkWriteStartTime - chunkQueueStartTime;
            nettyMetrics.chunkQueueTimeInMs.update(chunkQueueTime);
            if (request != null) {
                request.getMetricsTracker().nioMetricsTracker.addToResponseProcessingTime(chunkQueueTime);
            }
        }

        /**
         * Marks a chunk as handled and invokes the callback and future that accompanied this chunk of data. Once a chunk is
         * resolved, the data inside it is considered void.
         * @param exception the reason for chunk handling failure.
         */
        public void resolveChunk(Exception exception) {
            long chunkWriteFinishTime = System.currentTimeMillis();
            long bytesWritten = 0;
            if (exception == null) {
                bytesWritten = bytesToBeWritten;
                buffer.position(buffer.limit());
            }
            nettyMetrics.bytesWriteRate.mark(bytesWritten);
            future.done(bytesWritten, exception);
            if (callback != null) {
                callback.onCompletion(bytesWritten, exception);
            }
            long chunkResolutionProcessingTime = System.currentTimeMillis() - chunkWriteFinishTime;
            long chunkWriteTime = chunkWriteFinishTime - chunkWriteStartTime;
            nettyMetrics.channelWriteTimeInMs.update(chunkWriteTime);
            nettyMetrics.chunkResolutionProcessingTimeInMs.update(chunkResolutionProcessingTime);
            if (request != null) {
                request.getMetricsTracker().nioMetricsTracker
                        .addToResponseProcessingTime(chunkWriteTime + chunkResolutionProcessingTime);
            }
        }
    }

    /**
     * Dispenses chunks when asked to by the {@link ChunkedWriteHandler}.
     */
    private class ChunkDispenser implements ChunkedInput<HttpContent> {
        // NOTE: Due to a bug in HttpChunkedInput, some code there has been reproduced here instead of using it directly.
        private boolean sentLastChunk = false;

        /**
         * Determines if input has ended by examining response state and number of chunks pending for write.
         * @return {@code true} if there are no more chunks to write and the end marker has been sent. {@code false}
         *         otherwise.
         */
        @Override
        public boolean isEndOfInput() {
            return sentLastChunk;
        }

        @Override
        public void close() {
            // nothing to do.
        }

        /**
         * Dispenses the next chunk from {@link #chunksToWrite} if one is available and adds the chunk dispensed to
         * {@link #chunksAwaitingCallback}.
         * @param ctx the {@link ChannelHandlerContext} for the channel being written to.
         * @return a chunk of data if one is available. {@code null} otherwise.
         */
        @Override
        public HttpContent readChunk(ChannelHandlerContext ctx) {
            long chunkDispenseStartTime = System.currentTimeMillis();
            logger.trace("Servicing request for next chunk on channel {}", ctx.channel());
            HttpContent content = null;
            Chunk chunk = chunksToWrite.poll();
            if (chunk != null) {
                chunk.onDequeue();
                ByteBuf buf = Unpooled.wrappedBuffer(chunk.buffer);
                chunksAwaitingCallback.add(chunk);
                if (chunk.isLast) {
                    content = new DefaultLastHttpContent(buf);
                    sentLastChunk = true;
                } else {
                    content = new DefaultHttpContent(buf);
                }
            } else if (allChunksWritten() && !sentLastChunk) {
                // Send last chunk for this input
                sentLastChunk = true;
                content = LastHttpContent.EMPTY_LAST_CONTENT;
                logger.trace("Last chunk was sent on channel {}", ctx.channel());
            }
            nettyMetrics.chunkDispenseTimeInMs.update(System.currentTimeMillis() - chunkDispenseStartTime);
            return content;
        }

        /**
         * Determines if input has ended by examining response state and number of chunks pending for write.
         * @return {@code true} if there are no more chunks to write. {@code false} otherwise.
         */
        private boolean allChunksWritten() {
            return responseCompleteCalled.get() && !writeFuture.isDone() && chunksToWriteCount.get() == 0;
        }
    }

    // callback classes

    /**
     * Invokes callback on any chunks that are eligible for callback.
     */
    private class CallbackInvoker implements GenericProgressiveFutureListener<ChannelProgressiveFuture> {

        /**
         * Uses {@code progress} to determine chunks whose callbacks need to be invoked.
         * @param future the {@link ChannelProgressiveFuture} that is being listened on.
         * @param progress the total number of bytes that have been written starting from the time writes were invoked via
         *                 {@link ChunkedWriteHandler}.
         * @param total the total number of bytes that need to be written i.e. the target number. This is not relevant to
         *              {@link ChunkedWriteHandler} because there is no target number. All calls to this function except
         *              the very last one will have a negative {@code total}.
         */
        @Override
        public void operationProgressed(ChannelProgressiveFuture future, long progress, long total) {
            logger.trace("{} bytes of response written on channel {}", progress, ctx.channel());
            while (chunksAwaitingCallback.peek() != null
                    && progress >= chunksAwaitingCallback.peek().writeCompleteThreshold) {
                chunksAwaitingCallback.poll().resolveChunk(null);
            }
        }

        /**
         * Called once the write is complete i.e. either all chunks that were needed to be written were written or there
         * was an error  writing the chunks.
         * @param future the {@link ChannelProgressiveFuture} that is being listened on.
         */
        @Override
        public void operationComplete(ChannelProgressiveFuture future) {
            if (future.isSuccess()) {
                logger.trace("Response sending complete on channel {}", ctx.channel());
                completeRequest(request == null || !request.isKeepAlive());
            } else {
                handleChannelWriteFailure(future.cause(), true);
            }
        }
    }

    /**
     * Callback for writes of response metadata.
     */
    private class ResponseMetadataWriteListener implements GenericFutureListener<ChannelFuture> {
        private final long responseWriteStartTime = System.currentTimeMillis();

        /**
         * If the operation completed successfully, a write via the {@link ChunkedWriteHandler} is initiated. Otherwise,
         * failure is handled.
         * @param future the {@link ChannelFuture} that is being listened on.
         */
        @Override
        public void operationComplete(ChannelFuture future) {
            long writeFinishTime = System.currentTimeMillis();
            if (future.isSuccess()) {
                if (finalResponseMetadata instanceof LastHttpContent) {
                    // this is the case if finalResponseMetadata is a FullHttpResponse.
                    // in this case there is nothing more to write.
                    if (!writeFuture.isDone()) {
                        writeFuture.setSuccess();
                        completeRequest(!HttpHeaders.isKeepAlive(finalResponseMetadata));
                    }
                } else {
                    // otherwise there is some content to write.
                    logger.trace("Starting ChunkedWriteHandler on channel {}", ctx.channel());
                    writeFuture.addListener(new CallbackInvoker());
                    ctx.writeAndFlush(new ChunkDispenser(), writeFuture);
                }
            } else {
                handleChannelWriteFailure(future.cause(), true);
            }
            long responseAfterWriteProcessingTime = System.currentTimeMillis() - writeFinishTime;
            long channelWriteTime = writeFinishTime - responseWriteStartTime;
            nettyMetrics.channelWriteTimeInMs.update(channelWriteTime);
            nettyMetrics.responseMetadataAfterWriteProcessingTimeInMs.update(responseAfterWriteProcessingTime);
            if (request != null) {
                request.getMetricsTracker().nioMetricsTracker
                        .addToResponseProcessingTime(channelWriteTime + responseAfterWriteProcessingTime);
            }
        }
    }

    /**
     * Listener that will be attached to the write of an error response.
     */
    private class ErrorResponseWriteListener implements GenericFutureListener<ChannelFuture> {
        private final long responseWriteStartTime = System.currentTimeMillis();

        @Override
        public void operationComplete(ChannelFuture future) throws Exception {
            long writeFinishTime = System.currentTimeMillis();
            long channelWriteTime = writeFinishTime - responseWriteStartTime;
            if (future.isSuccess()) {
                completeRequest(!HttpHeaders.isKeepAlive(finalResponseMetadata));
            } else {
                handleChannelWriteFailure(future.cause(), true);
            }
            long responseAfterWriteProcessingTime = System.currentTimeMillis() - writeFinishTime;
            nettyMetrics.channelWriteTimeInMs.update(channelWriteTime);
            nettyMetrics.responseMetadataAfterWriteProcessingTimeInMs.update(responseAfterWriteProcessingTime);
            if (request != null) {
                request.getMetricsTracker().nioMetricsTracker
                        .addToResponseProcessingTime(channelWriteTime + responseAfterWriteProcessingTime);
            }
        }
    }

    /**
     * Cleans up chunks that are remaining. This serves two purposes:
     * 1. Guarding against the case where {@link CallbackInvoker#operationComplete(ChannelProgressiveFuture)} might have
     * finished already.
     * 2. Cleaning up any zero sized chunks when {@link HttpHeaders.Names#CONTENT_LENGTH} is 0 and a
     * {@link FullHttpResponse} has been sent.
     */
    private class CleanupCallback implements GenericFutureListener<ChannelFuture> {
        private final Exception exception;

        /**
         * Instantiate a CleanupCallback with an exception to return once cleanup is activated.
         * @param exception the {@link Exception} to return as a part of the callback. Can be {@code null}. This can be
         *                  overriden if the channel write ended in failure.
         */
        CleanupCallback(Exception exception) {
            this.exception = exception;
        }

        @Override
        public void operationComplete(ChannelFuture future) throws Exception {
            Throwable cause = future.cause() == null ? exception : future.cause();
            if (cause != null) {
                handleChannelWriteFailure(cause, false);
            } else {
                cleanupChunks(null);
            }
            logger.debug("Chunk cleanup complete on channel {}", ctx.channel());
        }
    }
}