com.linecorp.armeria.server.http.Http2RequestDecoder.java Source code

Java tutorial

Introduction

Here is the source code for com.linecorp.armeria.server.http.Http2RequestDecoder.java

Source

/*
 * Copyright 2016 LINE Corporation
 *
 * LINE Corporation licenses this file to you 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 com.linecorp.armeria.server.http;

import static io.netty.handler.codec.http2.Http2Error.INTERNAL_ERROR;
import static io.netty.handler.codec.http2.Http2Error.PROTOCOL_ERROR;
import static io.netty.handler.codec.http2.Http2Exception.connectionError;
import static io.netty.handler.codec.http2.Http2Exception.streamError;

import java.nio.charset.StandardCharsets;

import com.google.common.net.MediaType;

import com.linecorp.armeria.common.ContentTooLargeException;
import com.linecorp.armeria.common.http.DefaultHttpRequest;
import com.linecorp.armeria.common.http.HttpData;
import com.linecorp.armeria.common.http.HttpHeaderNames;
import com.linecorp.armeria.common.http.HttpHeaders;
import com.linecorp.armeria.internal.InboundTrafficController;
import com.linecorp.armeria.internal.http.ArmeriaHttpUtil;

import io.netty.buffer.ByteBuf;
import io.netty.buffer.Unpooled;
import io.netty.channel.Channel;
import io.netty.channel.ChannelHandlerContext;
import io.netty.handler.codec.http.HttpResponseStatus;
import io.netty.handler.codec.http2.DefaultHttp2Headers;
import io.netty.handler.codec.http2.Http2ConnectionEncoder;
import io.netty.handler.codec.http2.Http2Error;
import io.netty.handler.codec.http2.Http2EventAdapter;
import io.netty.handler.codec.http2.Http2Exception;
import io.netty.handler.codec.http2.Http2Headers;
import io.netty.handler.codec.http2.Http2Settings;
import io.netty.handler.codec.http2.Http2Stream;
import io.netty.util.collection.IntObjectHashMap;
import io.netty.util.collection.IntObjectMap;

final class Http2RequestDecoder extends Http2EventAdapter {

    private final Http2ConnectionEncoder writer;
    private final InboundTrafficController inboundTrafficController;
    private final IntObjectMap<DecodedHttpRequest> requests = new IntObjectHashMap<>();
    private int nextId;

    Http2RequestDecoder(Channel channel, Http2ConnectionEncoder writer) {
        this.writer = writer;
        inboundTrafficController = new InboundTrafficController(channel);
    }

    @Override
    public void onSettingsRead(ChannelHandlerContext ctx, Http2Settings settings) {
        ctx.fireChannelRead(settings);
    }

    @Override
    public void onHeadersRead(ChannelHandlerContext ctx, int streamId, Http2Headers headers, int padding,
            boolean endOfStream) throws Http2Exception {
        final HttpHeaders convertedHeaders = ArmeriaHttpUtil.toArmeria(headers);
        DecodedHttpRequest req = requests.get(streamId);
        if (req == null) {
            // Validate the 'content-length' header if exists.
            if (headers.contains(HttpHeaderNames.CONTENT_LENGTH)) {
                final long contentLength = headers.getLong(HttpHeaderNames.CONTENT_LENGTH, -1L);
                if (contentLength < 0) {
                    writeErrorResponse(ctx, streamId, HttpResponseStatus.BAD_REQUEST);
                    return;
                }
            }

            req = new DecodedHttpRequest(ctx.channel().eventLoop(), ++nextId, streamId, convertedHeaders, true,
                    inboundTrafficController);

            requests.put(streamId, req);
            ctx.fireChannelRead(req);
        } else {
            try {
                req.write(convertedHeaders);
            } catch (Throwable t) {
                req.close(t);
                throw connectionError(INTERNAL_ERROR, t, "failed to consume a HEADERS frame");
            }
        }

        if (endOfStream) {
            req.close();
        }
    }

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

    @Override
    public void onStreamRemoved(Http2Stream stream) {
    }

    @Override
    public int onDataRead(ChannelHandlerContext ctx, int streamId, ByteBuf data, int padding, boolean endOfStream)
            throws Http2Exception {

        final DecodedHttpRequest req = requests.get(streamId);
        if (req == null) {
            throw connectionError(PROTOCOL_ERROR, "received a DATA Frame for an unknown stream: %d", streamId);
        }

        final int dataLength = data.readableBytes();

        if (req.isOpen()) {
            final long maxContentLength = req.maxRequestLength();
            if (maxContentLength > 0 && req.writtenBytes() > maxContentLength - dataLength) {
                if (isWritable(streamId)) {
                    writeErrorResponse(ctx, streamId, HttpResponseStatus.REQUEST_ENTITY_TOO_LARGE);
                    req.close(ContentTooLargeException.get());
                } else {
                    req.close(ContentTooLargeException.get());
                    throw connectionError(INTERNAL_ERROR, "content length too large: %d + %d > %d (stream: %d)",
                            req.writtenBytes(), dataLength, maxContentLength, streamId);
                }
            } else {
                try {
                    req.write(HttpData.of(data));
                } catch (Throwable t) {
                    req.close(t);
                    throw connectionError(INTERNAL_ERROR, t, "failed to consume a DATA frame");
                }

                if (endOfStream) {
                    req.close();
                }
            }
        }

        // All bytes have been processed.
        return dataLength + padding;
    }

    private boolean isWritable(int streamId) {
        switch (writer.connection().stream(streamId).state()) {
        case OPEN:
        case HALF_CLOSED_REMOTE:
            return true;
        default:
            return false;
        }
    }

    private void writeErrorResponse(ChannelHandlerContext ctx, int streamId, HttpResponseStatus status) {
        final byte[] content = status.toString().getBytes(StandardCharsets.UTF_8);

        writer.writeHeaders(ctx, streamId,
                new DefaultHttp2Headers(false).status(status.codeAsText())
                        .set(HttpHeaderNames.CONTENT_TYPE, MediaType.PLAIN_TEXT_UTF_8.toString())
                        .setInt(HttpHeaderNames.CONTENT_LENGTH, content.length),
                0, false, ctx.voidPromise());

        writer.writeData(ctx, streamId, Unpooled.wrappedBuffer(content), 0, true, ctx.voidPromise());
    }

    @Override
    public void onRstStreamRead(ChannelHandlerContext ctx, int streamId, long errorCode) throws Http2Exception {
        final DefaultHttpRequest req = requests.get(streamId);
        if (req == null) {
            throw connectionError(PROTOCOL_ERROR, "received a RST_STREAM frame for an unknown stream: %d",
                    streamId);
        }

        req.close(streamError(streamId, Http2Error.valueOf(errorCode), "received a RST_STREAM frame"));
    }

    @Override
    public void onPushPromiseRead(ChannelHandlerContext ctx, int streamId, int promisedStreamId,
            Http2Headers headers, int padding) throws Http2Exception {
        throw connectionError(PROTOCOL_ERROR, "received a PUSH_PROMISE frame which only a server can send");
    }
}