View Javadoc
1   /*
2    * Copyright 2016 The Netty Project
3    *
4    * The Netty Project licenses this file to you under the Apache License,
5    * version 2.0 (the "License"); you may not use this file except in compliance
6    * with the License. You may obtain a copy of the License at:
7    *
8    *   https://www.apache.org/licenses/LICENSE-2.0
9    *
10   * Unless required by applicable law or agreed to in writing, software
11   * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
12   * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
13   * License for the specific language governing permissions and limitations
14   * under the License.
15   */
16  package io.netty.handler.codec.http2;
17  
18  import io.netty.channel.ChannelHandlerContext;
19  import io.netty.channel.ChannelInboundHandlerAdapter;
20  import io.netty.handler.codec.http.FullHttpMessage;
21  import io.netty.handler.codec.http.HttpHeaders;
22  import io.netty.handler.codec.http.HttpScheme;
23  
24  /**
25   * Translates HTTP/1.x object reads into HTTP/2 frames.
26   */
27  public class InboundHttpToHttp2Adapter extends ChannelInboundHandlerAdapter {
28      private final Http2Connection connection;
29      private final Http2FrameListener listener;
30  
31      public InboundHttpToHttp2Adapter(Http2Connection connection, Http2FrameListener listener) {
32          this.connection = connection;
33          this.listener = listener;
34      }
35  
36      private static int getStreamId(Http2Connection connection, HttpHeaders httpHeaders) {
37          return httpHeaders.getInt(HttpConversionUtil.ExtensionHeaderNames.STREAM_ID.text(),
38                                    connection.remote().incrementAndGetNextStreamId());
39      }
40  
41      @Override
42      public void channelRead(ChannelHandlerContext ctx, Object msg) throws Exception {
43          if (msg instanceof FullHttpMessage) {
44              handle(ctx, connection, listener, (FullHttpMessage) msg);
45          } else {
46              super.channelRead(ctx, msg);
47          }
48      }
49  
50      // note that this may behave strangely when used for the initial upgrade
51      // message when using h2c, since that message is ineligible for flow
52      // control, but there is not yet an API for signaling that.
53      static void handle(ChannelHandlerContext ctx, Http2Connection connection,
54                                Http2FrameListener listener, FullHttpMessage message) throws Http2Exception {
55          try {
56              int streamId = getStreamId(connection, message.headers());
57              Http2Stream stream = connection.stream(streamId);
58              if (stream == null) {
59                  stream = connection.remote().createStream(streamId, false);
60              }
61              message.headers().set(HttpConversionUtil.ExtensionHeaderNames.SCHEME.text(), HttpScheme.HTTP.name());
62              Http2Headers messageHeaders = HttpConversionUtil.toHttp2Headers(message, true);
63              boolean hasContent = message.content().isReadable();
64              boolean hasTrailers = !message.trailingHeaders().isEmpty();
65              listener.onHeadersRead(
66                      ctx, streamId, messageHeaders, 0, !(hasContent || hasTrailers));
67              if (hasContent) {
68                  listener.onDataRead(ctx, streamId, message.content(), 0, !hasTrailers);
69              }
70              if (hasTrailers) {
71                  Http2Headers headers = HttpConversionUtil.toHttp2Headers(message.trailingHeaders(), true);
72                  listener.onHeadersRead(ctx, streamId, headers, 0, true);
73              }
74              stream.closeRemoteSide();
75          } finally {
76              message.release();
77          }
78      }
79  }