View Javadoc
1   /*
2    * Copyright 2014 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.example.spdy.server;
17  
18  import io.netty.bootstrap.ServerBootstrap;
19  import io.netty.channel.Channel;
20  import io.netty.channel.ChannelOption;
21  import io.netty.channel.EventLoopGroup;
22  import io.netty.channel.nio.NioEventLoopGroup;
23  import io.netty.channel.socket.nio.NioServerSocketChannel;
24  import io.netty.handler.logging.LogLevel;
25  import io.netty.handler.logging.LoggingHandler;
26  import io.netty.handler.ssl.ApplicationProtocolConfig;
27  import io.netty.handler.ssl.ApplicationProtocolConfig.Protocol;
28  import io.netty.handler.ssl.ApplicationProtocolConfig.SelectedListenerFailureBehavior;
29  import io.netty.handler.ssl.ApplicationProtocolConfig.SelectorFailureBehavior;
30  import io.netty.handler.ssl.ApplicationProtocolNames;
31  import io.netty.handler.ssl.SslContext;
32  import io.netty.handler.ssl.SslContextBuilder;
33  import io.netty.handler.ssl.util.SelfSignedCertificate;
34  
35  /**
36   * A SPDY Server that responds to a GET request with a Hello World.
37   * <p>
38   * This class must be run with the JVM parameter: {@code java -Xbootclasspath/p:<path_to_npn_boot_jar> ...}.
39   * The "path_to_npn_boot_jar" is the path on the file system for the NPN Boot Jar file which can be downloaded from
40   * Maven at coordinates org.mortbay.jetty.npn:npn-boot. Different versions applies to different OpenJDK versions.
41   * See <a href="https://www.eclipse.org/jetty/documentation/current/npn-chapter.html">Jetty docs</a> for more
42   * information.
43   * <p>
44   * You may also use the {@code run-example.sh} script to start the server from the command line:
45   * <pre>
46   *     ./run-example.sh spdy-server
47   * </pre>
48   * <p>
49   * Once started, you can test the server with your
50   * <a href="https://en.wikipedia.org/wiki/SPDY#Browser_support_and_usage">SPDY enabled web browser</a> by navigating
51   * to <a href="https://localhost:8443/">https://localhost:8443/</a>
52   */
53  public final class SpdyServer {
54  
55      static final int PORT = Integer.parseInt(System.getProperty("port", "8443"));
56  
57      public static void main(String[] args) throws Exception {
58          // Configure SSL.
59          SelfSignedCertificate ssc = new SelfSignedCertificate();
60          SslContext sslCtx = SslContextBuilder.forServer(ssc.certificate(), ssc.privateKey())
61              .applicationProtocolConfig(new ApplicationProtocolConfig(
62                          Protocol.NPN,
63                          // NO_ADVERTISE is currently the only mode supported by both OpenSsl and JDK providers.
64                          SelectorFailureBehavior.NO_ADVERTISE,
65                          // ACCEPT is currently the only mode supported by both OpenSsl and JDK providers.
66                          SelectedListenerFailureBehavior.ACCEPT,
67                          ApplicationProtocolNames.SPDY_3_1,
68                          ApplicationProtocolNames.HTTP_1_1))
69              .build();
70  
71          // Configure the server.
72          EventLoopGroup bossGroup = new NioEventLoopGroup(1);
73          EventLoopGroup workerGroup = new NioEventLoopGroup();
74          try {
75              ServerBootstrap b = new ServerBootstrap();
76              b.option(ChannelOption.SO_BACKLOG, 1024);
77              b.group(bossGroup, workerGroup)
78               .channel(NioServerSocketChannel.class)
79               .handler(new LoggingHandler(LogLevel.INFO))
80               .childHandler(new SpdyServerInitializer(sslCtx));
81  
82              Channel ch = b.bind(PORT).sync().channel();
83  
84              System.err.println("Open your SPDY-enabled web browser and navigate to https://127.0.0.1:" + PORT + '/');
85              System.err.println("If using Chrome browser, check your SPDY sessions at chrome://net-internals/#spdy");
86  
87              ch.closeFuture().sync();
88          } finally {
89              bossGroup.shutdownGracefully();
90              workerGroup.shutdownGracefully();
91          }
92      }
93  }