io.jsync.http.impl.DefaultHttpServerResponse.java Source code

Java tutorial

Introduction

Here is the source code for io.jsync.http.impl.DefaultHttpServerResponse.java

Source

/*
 * Copyright (c) 2011-2013 The original author or authors
 * ------------------------------------------------------
 * All rights reserved. This program and the accompanying materials
 * are made available under the terms of the Eclipse Public License v1.0
 * and Apache License v2.0 which accompanies this distribution.
 *
 *     The Eclipse Public License is available at
 *     http://www.eclipse.org/legal/epl-v10.html
 *
 *     The Apache License v2.0 is available at
 *     http://www.opensource.org/licenses/apache2.0.php
 *
 * You may elect to redistribute this code under either of these licenses.
 */

package io.jsync.http.impl;

import io.jsync.AsyncResult;
import io.jsync.Handler;
import io.jsync.MultiMap;
import io.jsync.buffer.Buffer;
import io.jsync.file.impl.PathAdjuster;
import io.jsync.http.HttpServerResponse;
import io.jsync.impl.AsyncInternal;
import io.jsync.impl.DefaultFutureResult;
import io.jsync.utils.MimeUtils;
import io.netty.buffer.ByteBuf;
import io.netty.buffer.Unpooled;
import io.netty.channel.ChannelFuture;
import io.netty.channel.ChannelFutureListener;
import io.netty.handler.codec.http.*;

import java.io.File;

/**
 * @author <a href="http://tfox.org">Tim Fox</a>
 */
public class DefaultHttpServerResponse implements HttpServerResponse {

    private static final Buffer NOT_FOUND = new Buffer("<html><body>Resource not found</body><html>");
    private static final Buffer FORBIDDEN = new Buffer("<html><body>Forbidden</body><html>");

    private final AsyncInternal async;
    private final ServerConnection conn;
    private final HttpResponse response;
    private final HttpVersion version;
    private final boolean keepAlive;
    private boolean headWritten;
    private boolean written;
    private Handler<Void> drainHandler;
    private Handler<Throwable> exceptionHandler;
    private Handler<Void> closeHandler;
    private boolean chunked;
    private boolean closed;
    private ChannelFuture channelFuture;
    private MultiMap headers;
    private LastHttpContent trailing;
    private MultiMap trailers;
    private String statusMessage;

    DefaultHttpServerResponse(final AsyncInternal async, ServerConnection conn, HttpRequest request) {
        this.async = async;
        this.conn = conn;
        this.version = request.protocolVersion();
        this.response = new DefaultHttpResponse(version, HttpResponseStatus.OK, false);
        this.keepAlive = version == HttpVersion.HTTP_1_1 || (version == HttpVersion.HTTP_1_0 && request.headers()
                .contains(io.jsync.http.HttpHeaders.CONNECTION, io.jsync.http.HttpHeaders.KEEP_ALIVE, true));
    }

    @Override
    public MultiMap headers() {
        if (headers == null) {
            headers = new HttpHeadersAdapter(response.headers());
        }
        return headers;
    }

    @Override
    public MultiMap trailers() {
        if (trailers == null) {
            if (trailing == null) {
                trailing = new DefaultLastHttpContent(Unpooled.EMPTY_BUFFER, false);
            }
            trailers = new HttpHeadersAdapter(trailing.trailingHeaders());
        }
        return trailers;
    }

    @Override
    public int getStatusCode() {
        return response.status().code();
    }

    @Override
    public HttpServerResponse setStatusCode(int statusCode) {
        HttpResponseStatus status = statusMessage != null ? new HttpResponseStatus(statusCode, statusMessage)
                : HttpResponseStatus.valueOf(statusCode);
        this.response.setStatus(status);
        return this;
    }

    @Override
    public HttpServerResponse setContentLength(int contentLength) {
        checkWritten();
        headers().set(HttpHeaderNames.CONTENT_LENGTH, Integer.toString(contentLength));

        return this;
    }

    @Override
    public HttpServerResponse setContentType(String contentType) {
        checkWritten();
        headers().set(HttpHeaderNames.CONTENT_TYPE, contentType);

        return this;
    }

    @Override
    public String getStatusMessage() {
        return response.status().reasonPhrase();
    }

    @Override
    public HttpServerResponse setStatusMessage(String statusMessage) {
        this.statusMessage = statusMessage;
        this.response.setStatus(new HttpResponseStatus(response.status().code(), statusMessage));
        return this;
    }

    @Override
    public boolean isChunked() {
        return chunked;
    }

    @Override
    public DefaultHttpServerResponse setChunked(boolean chunked) {
        checkWritten();
        // HTTP 1.0 does not support chunking so we ignore this if HTTP 1.0
        if (version != HttpVersion.HTTP_1_0) {
            this.chunked = chunked;
        }
        return this;
    }

    @Override
    public DefaultHttpServerResponse putHeader(String key, String value) {
        checkWritten();
        headers().set(key, value);
        return this;
    }

    @Override
    public DefaultHttpServerResponse putHeader(String key, Iterable<String> values) {
        checkWritten();
        headers().set(key, values);
        return this;
    }

    @Override
    public DefaultHttpServerResponse putTrailer(String key, String value) {
        checkWritten();
        trailers().set(key, value);
        return this;
    }

    @Override
    public DefaultHttpServerResponse putTrailer(String key, Iterable<String> values) {
        checkWritten();
        trailers().set(key, values);
        return this;
    }

    @Override
    public HttpServerResponse putHeader(CharSequence name, CharSequence value) {
        checkWritten();
        headers().set(name, value);
        return this;
    }

    @Override
    public HttpServerResponse putHeader(CharSequence name, Iterable<CharSequence> values) {
        checkWritten();
        headers().set(name, values);
        return this;
    }

    @Override
    public HttpServerResponse putTrailer(CharSequence name, CharSequence value) {
        checkWritten();
        trailers().set(name, value);
        return this;
    }

    @Override
    public HttpServerResponse putTrailer(CharSequence name, Iterable<CharSequence> value) {
        checkWritten();
        trailers().set(name, value);
        return this;
    }

    @Override
    public HttpServerResponse setWriteQueueMaxSize(int size) {
        checkWritten();
        conn.doSetWriteQueueMaxSize(size);
        return this;
    }

    @Override
    public boolean writeQueueFull() {
        checkWritten();
        return conn.doWriteQueueFull();
    }

    @Override
    public HttpServerResponse drainHandler(Handler<Void> handler) {
        checkWritten();
        this.drainHandler = handler;
        conn.handleInterestedOpsChanged(); //If the channel is already drained, we want to call it immediately
        return this;
    }

    @Override
    public HttpServerResponse exceptionHandler(Handler<Throwable> handler) {
        checkWritten();
        this.exceptionHandler = handler;
        return this;
    }

    @Override
    public HttpServerResponse closeHandler(Handler<Void> handler) {
        checkWritten();
        this.closeHandler = handler;
        return this;
    }

    @Override
    public DefaultHttpServerResponse write(Buffer chunk) {
        ByteBuf buf = chunk.getByteBuf();
        return write(buf, null);
    }

    @Override
    public DefaultHttpServerResponse write(String chunk, String enc) {
        return write(new Buffer(chunk, enc).getByteBuf(), null);
    }

    @Override
    public DefaultHttpServerResponse write(String chunk) {
        return write(new Buffer(chunk).getByteBuf(), null);
    }

    @Override
    public void end(String chunk) {
        end(new Buffer(chunk));
    }

    @Override
    public void end(String chunk, String enc) {
        end(new Buffer(chunk, enc));
    }

    @Override
    public void end(Buffer chunk) {
        if (!chunked && !contentLengthSet()) {
            headers().set(io.jsync.http.HttpHeaders.CONTENT_LENGTH, String.valueOf(chunk.length()));
        }
        ByteBuf buf = chunk.getByteBuf();
        end0(buf);
    }

    @Override
    public void close() {
        if (!closed) {
            if (headWritten) {
                closeConnAfterWrite();
            } else {
                conn.close();
            }
            closed = true;
        }
    }

    @Override
    public void end() {
        end0(Unpooled.EMPTY_BUFFER);
    }

    private void end0(ByteBuf data) {
        checkWritten();

        if (!headWritten) {
            // if the head was not written yet we can write out everything in on go
            // which is more cheap.
            prepareHeaders();
            FullHttpResponse resp;
            if (trailing != null) {
                resp = new AssembledFullHttpResponse(response, data, trailing.trailingHeaders(),
                        trailing.decoderResult());
            } else {
                resp = new AssembledFullHttpResponse(response, data);
            }
            channelFuture = conn.write(resp);
            headWritten = true;
        } else {
            if (!data.isReadable()) {
                if (trailing == null) {
                    channelFuture = conn.write(LastHttpContent.EMPTY_LAST_CONTENT);
                } else {
                    channelFuture = conn.write(trailing);
                }
            } else {
                LastHttpContent content;
                if (trailing != null) {
                    content = new AssembledLastHttpContent(data, trailing.trailingHeaders(),
                            trailing.decoderResult());
                } else {
                    content = new DefaultLastHttpContent(data, false);
                }
                channelFuture = conn.write(content);
            }
        }

        if (!keepAlive) {
            closeConnAfterWrite();
        }
        written = true;
        conn.responseComplete();
    }

    @Override
    public DefaultHttpServerResponse sendFile(String filename) {
        return sendFile(filename, (String) null);
    }

    @Override
    public DefaultHttpServerResponse sendFile(String filename, String notFoundResource) {
        doSendFile(filename, notFoundResource, null);
        return this;
    }

    @Override
    public HttpServerResponse sendFile(String filename, Handler<AsyncResult<Void>> resultHandler) {
        return sendFile(filename, null, resultHandler);
    }

    @Override
    public HttpServerResponse sendFile(String filename, String notFoundFile,
            Handler<AsyncResult<Void>> resultHandler) {
        doSendFile(filename, notFoundFile, resultHandler);
        return this;
    }

    private void doSendFile(String filename, String notFoundResource,
            final Handler<AsyncResult<Void>> resultHandler) {
        if (headWritten) {
            throw new IllegalStateException("Head already written");
        }
        checkWritten();
        File file = new File(PathAdjuster.adjust(async, filename));
        if (!file.exists()) {
            if (notFoundResource != null) {
                setStatusCode(HttpResponseStatus.NOT_FOUND.code());
                sendFile(notFoundResource, (String) null, resultHandler);
            } else {
                sendNotFound();
            }
        } else if (file.isDirectory()) {
            // send over a 403 Forbidden
            sendForbidden();
        } else {
            if (!contentLengthSet()) {
                putHeader(io.jsync.http.HttpHeaders.CONTENT_LENGTH, String.valueOf(file.length()));
            }
            if (!contentTypeSet()) {
                int li = filename.lastIndexOf('.');
                if (li != -1 && li != filename.length() - 1) {
                    String ext = filename.substring(li + 1, filename.length());
                    String contentType = MimeUtils.getMimeTypeForExtension(ext);
                    if (contentType != null) {
                        putHeader(io.jsync.http.HttpHeaders.CONTENT_TYPE, contentType);
                    }
                }
            }
            prepareHeaders();
            conn.queueForWrite(response);
            conn.sendFile(file);

            // write an empty last content to let the http encoder know the response is complete
            channelFuture = conn.write(LastHttpContent.EMPTY_LAST_CONTENT);
            headWritten = written = true;

            if (resultHandler != null) {
                channelFuture.addListener(new ChannelFutureListener() {
                    public void operationComplete(ChannelFuture future) throws Exception {
                        final AsyncResult<Void> res;
                        if (future.isSuccess()) {
                            res = new DefaultFutureResult<>((Void) null);
                        } else {
                            res = new DefaultFutureResult<>(future.cause());
                        }
                        async.runOnContext(new Handler<Void>() {
                            @Override
                            public void handle(Void v) {
                                resultHandler.handle(res);
                            }
                        });
                    }
                });
            }

            if (!keepAlive) {
                closeConnAfterWrite();
            }
            conn.responseComplete();
        }
    }

    private boolean contentLengthSet() {
        if (headers == null) {
            return false;
        }
        return response.headers().contains(io.jsync.http.HttpHeaders.CONTENT_LENGTH);
    }

    private boolean contentTypeSet() {
        if (headers == null) {
            return false;
        }
        return response.headers().contains(io.jsync.http.HttpHeaders.CONTENT_TYPE);
    }

    private void closeConnAfterWrite() {
        if (channelFuture != null) {
            channelFuture.addListener(new ChannelFutureListener() {
                public void operationComplete(ChannelFuture future) throws Exception {
                    conn.close();
                }
            });
        }
    }

    private void sendForbidden() {
        setStatusCode(HttpResponseStatus.FORBIDDEN.code());
        putHeader(io.jsync.http.HttpHeaders.CONTENT_TYPE, io.jsync.http.HttpHeaders.TEXT_HTML);
        end(FORBIDDEN);
    }

    private void sendNotFound() {
        setStatusCode(HttpResponseStatus.NOT_FOUND.code());
        putHeader(io.jsync.http.HttpHeaders.CONTENT_TYPE, io.jsync.http.HttpHeaders.TEXT_HTML);
        end(NOT_FOUND);
    }

    void handleDrained() {
        if (drainHandler != null) {
            drainHandler.handle(null);
        }
    }

    void handleException(Throwable t) {
        if (exceptionHandler != null) {
            exceptionHandler.handle(t);
        }
    }

    void handleClosed() {
        if (closeHandler != null) {
            closeHandler.handle(null);
        }
    }

    private void checkWritten() {
        if (written) {
            throw new IllegalStateException("Response has already been written");
        }
    }

    private void prepareHeaders() {
        if (version == HttpVersion.HTTP_1_0 && keepAlive) {
            response.headers().set(io.jsync.http.HttpHeaders.CONNECTION, io.jsync.http.HttpHeaders.KEEP_ALIVE);
        }
        if (chunked) {
            response.headers().set(io.jsync.http.HttpHeaders.TRANSFER_ENCODING, io.jsync.http.HttpHeaders.CHUNKED);
        } else if (version != HttpVersion.HTTP_1_0 && !contentLengthSet()) {
            response.headers().set(io.jsync.http.HttpHeaders.CONTENT_LENGTH, "0");
        }
    }

    private DefaultHttpServerResponse write(ByteBuf chunk, final Handler<AsyncResult<Void>> doneHandler) {
        checkWritten();
        if (!headWritten && version != HttpVersion.HTTP_1_0 && !chunked && !contentLengthSet()) {
            throw new IllegalStateException(
                    "You must set the Content-Length header to be the total size of the message "
                            + "body BEFORE sending any data if you are not using HTTP chunked encoding.");
        }

        if (!headWritten) {
            prepareHeaders();
            channelFuture = conn.write(new AssembledHttpResponse(response, chunk));
            headWritten = true;
        } else {
            channelFuture = conn.write(new DefaultHttpContent(chunk));
        }

        conn.addFuture(doneHandler, channelFuture);
        return this;
    }
}