Netty In Action中文版 – 第十一章:WebSocket

释放双眼,带上耳机,听听看~!

Netty In Action中文版 – 第十一章:WebSocket

本章介绍

  • WebSocket
  • ChannelHandler,Decoder and Encoder
  • 引导一个Netty基础程序
  • 测试WebSocket

“real-time-web”实时web现在随处可见,很多的用户希望能从web站点实时获取信息。Netty支持WebSocket实现,并包含了不同的版本,我们可以非常容易的实现WebSocket应用。使用Netty附带的WebSocket,我们不需要关注协议内部实现,只需要使用Netty提供的一些简单的方法就可以实现。本章将通过的例子应用帮助你来使用WebSocket并了解它是如何工作。

11.1 WebSockets some background

        关于WebSocket的一些概念和背景,可以查询网上相关介绍。这里不赘述。

11.2 面临的挑战

        要显示“real-time”支持的WebSocket,应用程序将显示如何使用Netty中的WebSocket实现一个在浏览器中进行聊天的IRC应用程序。你可能知道从Facebook可以发送文本消息到另一个人,在这里,我们将进一步了解其实现。在这个应用程序中,不同的用户可以同时交谈,非常像IRC(Internet Relay Chat,互联网中继聊天)。

上图显示的逻辑很简单:

  1. 一个客户端发送一条消息
  2. 消息被广播到其他已连接的客户端

它的工作原理就像聊天室一样,在这里例子中,我们将编写服务器,然后使用浏览器作为客户端。带着这样的思路,我们将会很简单的实现它。

11.3 实现

        WebSocket使用HTTP升级机制从一个普通的HTTP连接WebSocket,因为这个应用程序使用WebSocket总是开始于HTTP(s),然后再升级。什么时候升级取决于应用程序本身。直接执行升级作为第一个操作一般是使用特定的url请求。

        在这里,如果url的结尾以/ws结束,我们将只会升级到WebSocket,否则服务器将发送一个网页给客户端。升级后的连接将通过WebSocket传输所有数据。逻辑图如下:

11.3.1 处理http请求

        服务器将作为一种混合式以允许同时处理http和websocket,所以服务器还需要html页面,html用来充当客户端角色,连接服务器并交互消息。因此,如果客户端不发送/ws的uri,我们需要写一个ChannelInboundHandler用来处理FullHttpRequest。看下面代码:

[java] view plain
copy

package netty.in.action;
1.
1.
import io.netty.channel.ChannelFuture;
1.
import io.netty.channel.ChannelFutureListener;
1.
import io.netty.channel.ChannelHandlerContext;
1.
import io.netty.channel.DefaultFileRegion;
1.
import io.netty.channel.SimpleChannelInboundHandler;
1.
import io.netty.handler.codec.http.DefaultFullHttpResponse;
1.
import io.netty.handler.codec.http.DefaultHttpResponse;
1.
import io.netty.handler.codec.http.FullHttpRequest;
1.
import io.netty.handler.codec.http.FullHttpResponse;
1.
import io.netty.handler.codec.http.HttpHeaders;
1.
import io.netty.handler.codec.http.HttpResponse;
1.
import io.netty.handler.codec.http.HttpResponseStatus;
1.
import io.netty.handler.codec.http.HttpVersion;
1.
import io.netty.handler.codec.http.LastHttpContent;
1.
import io.netty.handler.ssl.SslHandler;
1.
import io.netty.handler.stream.ChunkedNioFile;
1.
1.
import java.io.RandomAccessFile;
1.

  1. /**
  2.  * WebSocket,处理http请求
  3.  * 
  4.  * @author c.k
  5.  * 
  6.  */

public 
class HttpRequestHandler 
extends

  1.         SimpleChannelInboundHandler<FullHttpRequest> {
  2.     

//websocket标识

  1.     

private 
final String wsUri;
1.

  1.     

public HttpRequestHandler(String wsUri) {

  1.         

this.wsUri = wsUri;

  1.     }
  2.     

@Override

  1.     

protected 
void channelRead0(ChannelHandlerContext ctx, FullHttpRequest msg)

  1.             

throws Exception {

  1.         

//如果是websocket请求,请求地址uri等于wsuri

  1.         

if (wsUri.equalsIgnoreCase(msg.getUri())) {

  1.             

//将消息转发到下一个ChannelHandler

  1.             ctx.fireChannelRead(msg.retain());
  2.         } 

else {
//如果不是websocket请求

  1.             

if (HttpHeaders.is100ContinueExpected(msg)) {

  1.                 

//如果HTTP请求头部包含Expect: 100-continue,

  1.                 

//则响应请求

  1.                 FullHttpResponse response = 

new DefaultFullHttpResponse(

  1.                         HttpVersion.HTTP_1_1, HttpResponseStatus.CONTINUE);
  2.                 ctx.writeAndFlush(response);
  3.             }
  4.             

//获取index.html的内容响应给客户端

  1.             RandomAccessFile file = 

new RandomAccessFile(

  1.                     System.getProperty(

"user.dir") + 
"/index.html", 
"r");

  1.             HttpResponse response = 

new DefaultHttpResponse(

  1.                     msg.getProtocolVersion(), HttpResponseStatus.OK);
  2.             response.headers().set(HttpHeaders.Names.CONTENT_TYPE,
  3.                     

"text/html; charset=UTF-8");

  1.             

boolean keepAlive = HttpHeaders.isKeepAlive(msg);

  1.             

//如果http请求保持活跃,设置http请求头部信息

  1.             

//并响应请求

  1.             

if (keepAlive) {

  1.                 response.headers().set(HttpHeaders.Names.CONTENT_LENGTH,
  2.                         file.length());
  3.                 response.headers().set(HttpHeaders.Names.CONNECTION,
  4.                         HttpHeaders.Values.KEEP_ALIVE);
  5.             }
  6.             ctx.write(response);
  7.             

//如果不是https请求,将index.html内容写入通道

  1.             

if (ctx.pipeline().get(SslHandler.
class) == 
null) {

  1.                 ctx.write(

new DefaultFileRegion(file.getChannel(), 
0, file

  1.                         .length()));
  2.             } 

else {

  1.                 ctx.write(

new ChunkedNioFile(file.getChannel()));

  1.             }
  2.             

//标识响应内容结束并刷新通道

  1.             ChannelFuture future = ctx
  2.                     .writeAndFlush(LastHttpContent.EMPTY_LAST_CONTENT);
  3.             

if (!keepAlive) {

  1.                 

//如果http请求不活跃,关闭http连接

  1.                 future.addListener(ChannelFutureListener.CLOSE);
  2.             }
  3.             file.close();
  4.         }
  5.     }
  6.     

@Override

  1.     

public 
void exceptionCaught(ChannelHandlerContext ctx, Throwable cause)

  1.             

throws Exception {

  1.         cause.printStackTrace();
  2.         ctx.close();
  3.     }
  4. }

11.3.2 处理WebSocket框架

        WebSocket支持6种不同框架,如下图:

我们的程序只需要使用下面4个框架:

  • CloseWebSocketFrame
  • PingWebSocketFrame
  • PongWebSocketFrame
  • TextWebSocketFrame

我们只需要显示处理TextWebSocketFrame,其他的会自动由WebSocketServerProtocolHandler处理,看下面代码:

[java] view plain
copy

package netty.in.action;
1.
1.
import io.netty.channel.ChannelHandlerContext;
1.
import io.netty.channel.SimpleChannelInboundHandler;
1.
import io.netty.channel.group.ChannelGroup;
1.
import io.netty.handler.codec.http.websocketx.TextWebSocketFrame;
1.
import io.netty.handler.codec.http.websocketx.WebSocketServerProtocolHandler;
1.

  1. /**
  2.  * WebSocket,处理消息
  3.  * @author c.k
  4.  *
  5.  */

public 
class TextWebSocketFrameHandler 
extends

  1.         SimpleChannelInboundHandler<TextWebSocketFrame> {
  2.     

private 
final ChannelGroup group;
1.

  1.     

public TextWebSocketFrameHandler(ChannelGroup group) {

  1.         

this.group = group;

  1.     }
  2.     

@Override

  1.     

public 
void userEventTriggered(ChannelHandlerContext ctx, Object evt)

  1.             

throws Exception {

  1.         

//如果WebSocket握手完成

  1.         

if (evt == WebSocketServerProtocolHandler.ServerHandshakeStateEvent.HANDSHAKE_COMPLETE) {

  1.             

//删除ChannelPipeline中的HttpRequestHandler

  1.             ctx.pipeline().remove(HttpRequestHandler.

class);

  1.             

//写一个消息到ChannelGroup

  1.             group.writeAndFlush(

new TextWebSocketFrame(
"Client " + ctx.channel()

  1.                     + 

" joined"));

  1.             

//将Channel添加到ChannelGroup

  1.             group.add(ctx.channel());
  2.         }

else {

  1.             

super.userEventTriggered(ctx, evt);

  1.         }
  2.     }
  3.     

@Override

  1.     

protected 
void channelRead0(ChannelHandlerContext ctx, TextWebSocketFrame msg)

  1.             

throws Exception {

  1.         

//将接收的消息通过ChannelGroup转发到所以已连接的客户端

  1.         group.writeAndFlush(msg.retain());
  2.     }
  3. }

11.3.3 初始化ChannelPipeline

        看下面代码:

[java] view plain
copy

package netty.in.action;
1.
1.
import io.netty.channel.Channel;
1.
import io.netty.channel.ChannelInitializer;
1.
import io.netty.channel.ChannelPipeline;
1.
import io.netty.channel.group.ChannelGroup;
1.
import io.netty.handler.codec.http.HttpObjectAggregator;
1.
import io.netty.handler.codec.http.HttpServerCodec;
1.
import io.netty.handler.codec.http.websocketx.WebSocketServerProtocolHandler;
1.
import io.netty.handler.stream.ChunkedWriteHandler;
1.

  1. /**
  2.  * WebSocket,初始化ChannelHandler
  3.  * @author c.k
  4.  *
  5.  */

public 
class ChatServerInitializer 
extends ChannelInitializer<Channel> {

  1.     

private 
final ChannelGroup group;
1.

  1.     

public ChatServerInitializer(ChannelGroup group){

  1.         

this.group = group;

  1.     }
  2.     

@Override

  1.     

protected 
void initChannel(Channel ch) 
throws Exception {

  1.         ChannelPipeline pipeline = ch.pipeline();
  2.         

//编解码http请求

  1.         pipeline.addLast(

new HttpServerCodec());

  1.         

//写文件内容

  1.         pipeline.addLast(

new ChunkedWriteHandler());

  1.         

//聚合解码HttpRequest/HttpContent/LastHttpContent到FullHttpRequest

  1.         

//保证接收的Http请求的完整性

  1.         pipeline.addLast(

new HttpObjectAggregator(
64 * 
1024));

  1.         

//处理FullHttpRequest

  1.         pipeline.addLast(

new HttpRequestHandler(
"/ws"));

  1.         

//处理其他的WebSocketFrame

  1.         pipeline.addLast(

new WebSocketServerProtocolHandler(
"/ws"));

  1.         

//处理TextWebSocketFrame

  1.         pipeline.addLast(

new TextWebSocketFrameHandler(group));

  1.     }
  2. }

WebSocketServerProtcolHandler不仅处理Ping/Pong/CloseWebSocketFrame,还和它自己握手并帮助升级WebSocket。这是执行完成握手和成功修改ChannelPipeline,并且添加需要的编码器/解码器和删除不需要的ChannelHandler。

        看下图:

ChannelPipeline通过ChannelInitializer的initChannel(…)方法完成初始化,完成握手后就会更改事情。一旦这样做了,WebSocketServerProtocolHandler将取代HttpRequestDecoder、WebSocketFrameDecoder13和HttpResponseEncoder、WebSocketFrameEncoder13。另外也要删除所有不需要的ChannelHandler已获得最佳性能。这些都是HttpObjectAggregator和HttpRequestHandler。下图显示ChannelPipeline握手完成:

我们甚至没注意到它,因为它是在底层执行的。以非常灵活的方式动态更新ChannelPipeline让单独的任务在不同的ChannelHandler中实现。

11.4 结合在一起使用

        一如既往,我们要将它们结合在一起使用。使用Bootstrap引导服务器和设置正确的ChannelInitializer。看下面代码:

[java] view plain
copy

package netty.in.action;
1.
1.
import io.netty.bootstrap.ServerBootstrap;
1.
import io.netty.channel.Channel;
1.
import io.netty.channel.ChannelFuture;
1.
import io.netty.channel.ChannelInitializer;
1.
import io.netty.channel.EventLoopGroup;
1.
import io.netty.channel.group.ChannelGroup;
1.
import io.netty.channel.group.DefaultChannelGroup;
1.
import io.netty.channel.nio.NioEventLoopGroup;
1.
import io.netty.channel.socket.nio.NioServerSocketChannel;
1.
import io.netty.util.concurrent.ImmediateEventExecutor;
1.
1.
import java.net.InetSocketAddress;
1.

  1. /**
  2.  * 访问地址:http://localhost:2048
  3.  * 
  4.  * @author c.k
  5.  * 
  6.  */

public 
class ChatServer {
1.

  1.     

private 
final ChannelGroup group = 
new DefaultChannelGroup(

  1.             ImmediateEventExecutor.INSTANCE);
  2.     

private 
final EventLoopGroup workerGroup = 
new NioEventLoopGroup();

  1.     

private Channel channel;
1.

  1.     

public ChannelFuture start(InetSocketAddress address) {

  1.         ServerBootstrap b = 

new ServerBootstrap();

  1.         b.group(workerGroup).channel(NioServerSocketChannel.

class)

  1.                 .childHandler(createInitializer(group));
  2.         ChannelFuture f = b.bind(address).syncUninterruptibly();
  3.         channel = f.channel();
  4.         

return f;

  1.     }
  2.     

public 
void destroy() {

  1.         

if (channel != 
null)

  1.             channel.close();
  2.         group.close();
  3.         workerGroup.shutdownGracefully();
  4.     }
  5.     

protected ChannelInitializer<Channel> createInitializer(ChannelGroup group) {

  1.         

return 
new ChatServerInitializer(group);

  1.     }
  2.     

public 
static 
void main(String[] args) {

  1.         

final ChatServer server = 
new ChatServer();

  1.         ChannelFuture f = server.start(

new InetSocketAddress(
2048));

  1.         Runtime.getRuntime().addShutdownHook(

new Thread() {

  1.             

@Override

  1.             

public 
void run() {

  1.                 server.destroy();
  2.             }
  3.         });
  4.         f.channel().closeFuture().syncUninterruptibly();
  5.     }
  6. }

另外,需要将index.html文件放在项目根目录,index.html内容如下:

[html] view plain
copy

<
html

<
head

<
title

Web Socket Test

</
title

</
head

<
body

<
script 
type=
"text/javascript"

  1. var socket;
  2. if (!window.WebSocket) {
  3.   

window.WebSocket = 
window.MozWebSocket;

  1. }
  2. if (window.WebSocket) {
  3.   

socket = 
new WebSocket("ws://localhost:2048/ws");

  1.   

socket.onmessage = 
function(event) {

  1.     var 

ta = 
document.getElementById('responseText');

  1.     

ta.value = 
ta.value + '\n' + event.data

  1.   };
  2.   

socket.onopen = 
function(event) {

  1.     var 

ta = 
document.getElementById('responseText');

  1.     

ta.value = 
"Web Socket opened!";

  1.   };
  2.   

socket.onclose = 
function(event) {

  1.     var 

ta = 
document.getElementById('responseText');

  1.     

ta.value = 
ta.value + "Web Socket closed";

  1.   };
  2. } else {
  3.   alert("Your browser does not support Web Socket.");
  4. }
  5. function send(message) {
  6.   if (!window.WebSocket) { return; }
  7.   if (

socket.readyState == WebSocket.OPEN) {

  1.     socket.send(message);
  2.   } else {
  3.     alert("The socket is not open.");
  4.   }
  5. }

</
script

  1.     

<
form 
onsubmit=
"return false;"

  1.         

<
input 
type=
"text" 
name=
"message" 
value=
"Hello, World!"

<
input

  1.             

type=
"button" 
value=
"Send Web Socket Data"

  1.             

onclick=
"send(this.form.message.value)"

  1.         

<
h3

Output

</
h3

  1.         

<
textarea 
id=
"responseText" 
style=
"width: 500px; height: 300px;"

</
textarea

  1.     

</
form

</
body

</
html

最后在浏览器中输入:http://localhost:2048,多开几个窗口就可以聊天了。

11.5 给WebSocket加密

        上面的应用程序虽然工作的很好,但是在网络上收发消息存在很大的安全隐患,所以有必要对消息进行加密。添加这样一个加密的功能一般比较复杂,需要对代码有较大的改动。但是使用Netty就可以很容易的添加这样的功能,只需要将SslHandler加入到ChannelPipeline中就可以了。实际上还需要添加SslContext,但这不在本例子范围内。

        首先我们创建一个用于添加加密Handler的handler初始化类,看下面代码:

[java] view plain
copy

package netty.in.action;
1.
1.
import io.netty.channel.Channel;
1.
import io.netty.channel.group.ChannelGroup;
1.
import io.netty.handler.ssl.SslHandler;
1.
1.
import javax.net.ssl.SSLContext;
1.
import javax.net.ssl.SSLEngine;
1.
1.
public 
class SecureChatServerIntializer 
extends ChatServerInitializer {

  1.     

private 
final SSLContext context;
1.

  1.     

public SecureChatServerIntializer(ChannelGroup group,SSLContext context) {

  1.         

super(group);

  1.         

this.context = context;

  1.     }
  2.     

@Override

  1.     

protected 
void initChannel(Channel ch) 
throws Exception {

  1.         

super.initChannel(ch);

  1.         SSLEngine engine = context.createSSLEngine();
  2.         engine.setUseClientMode(

false);

  1.         ch.pipeline().addFirst(

new SslHandler(engine));

  1.     }
  2. }

最后我们创建一个用于引导配置的类,看下面代码:

[java] view plain
copy

package netty.in.action;
1.
1.
import io.netty.channel.Channel;
1.
import io.netty.channel.ChannelFuture;
1.
import io.netty.channel.ChannelInitializer;
1.
import io.netty.channel.group.ChannelGroup;
1.
import java.net.InetSocketAddress;
1.
import javax.net.ssl.SSLContext;
1.

  1. /**
  2.  * 访问地址:https://localhost:4096
  3.  * 
  4.  * @author c.k
  5.  * 
  6.  */

public 
class SecureChatServer 
extends ChatServer {

  1.     

private 
final SSLContext context;
1.

  1.     

public SecureChatServer(SSLContext context) {

  1.         

this.context = context;

  1.     }
  2.     

@Override

  1.     

protected ChannelInitializer<Channel> createInitializer(ChannelGroup group) {

  1.         

return 
new SecureChatServerIntializer(group, context);

  1.     }
  2.     

/**

  1.      * 获取SSLContext需要相关的keystore文件,这里没有 关于HTTPS可以查阅相关资料,这里只介绍在Netty中如何使用
  2.      * 
  3.      * @return
  4.      */
  5.     

private 
static SSLContext getSslContext() {

  1.         

return 
null;

  1.     }
  2.     

public 
static 
void main(String[] args) {

  1.         SSLContext context = getSslContext();
  2.         

final SecureChatServer server = 
new SecureChatServer(context);

  1.         ChannelFuture future = server.start(

new InetSocketAddress(
4096));

  1.         Runtime.getRuntime().addShutdownHook(

new Thread() {

  1.             

@Override

  1.             

public 
void run() {

  1.                 server.destroy();
  2.             }
  3.         });
  4.         future.channel().closeFuture().syncUninterruptibly();
  5.     }
  6. }

11.6 Summary

原文地址http://www.bieryun.com/2182.html

给TA打赏
共{{data.count}}人
人已打赏
安全技术

Bootstrap框架之排版

2021-12-21 16:36:11

安全技术

从零搭建自己的SpringBoot后台框架(二十三)

2022-1-12 12:36:11

个人中心
购物车
优惠劵
今日签到
有新私信 私信列表
搜索