WindChat/windchat-connector/src/main/java/com/windchat/im/connector/ws/WsServer.java

115 lines
3.9 KiB
Java
Raw Normal View History

2019-11-27 23:33:33 +08:00
package com.windchat.im.connector.ws;
2019-07-29 23:42:16 +08:00
2019-11-27 23:33:33 +08:00
import com.windchat.im.connector.constant.AkxProject;
import com.windchat.im.connector.ws.handler.WsServerHandler;
2019-07-29 23:42:16 +08:00
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
2019-11-27 23:36:54 +08:00
import com.windchat.common.command.Command;
import com.windchat.common.command.CommandResponse;
import com.windchat.common.executor.AbstracteExecutor;
import com.windchat.common.executor.SimpleExecutor;
2019-07-29 23:42:16 +08:00
import io.netty.bootstrap.ServerBootstrap;
import io.netty.channel.ChannelFuture;
import io.netty.channel.ChannelInitializer;
import io.netty.channel.EventLoopGroup;
import io.netty.channel.nio.NioEventLoopGroup;
import io.netty.channel.socket.SocketChannel;
import io.netty.channel.socket.nio.NioServerSocketChannel;
import io.netty.handler.codec.http.HttpObjectAggregator;
import io.netty.handler.codec.http.HttpServerCodec;
import io.netty.handler.stream.ChunkedWriteHandler;
import io.netty.handler.timeout.IdleStateHandler;
import io.netty.util.concurrent.Future;
import io.netty.util.concurrent.GenericFutureListener;
/**
* WebSocket服务启动静态类
*
* @author Sam{@link an.guoyue254@gmail.com}
* @since 2018-05-02 15:15:21
*/
public abstract class WsServer {
private static Logger logger = LoggerFactory.getLogger(WsServer.class);
private AbstracteExecutor<Command, CommandResponse> executor;
private ServerBootstrap bootstrap;
private EventLoopGroup parentGroup;
private EventLoopGroup childGroup;
public WsServer() {
executor = new SimpleExecutor<Command, CommandResponse>();
loadExecutor(executor);
// 负责对外连接线程
parentGroup = new NioEventLoopGroup();
// 负责对内分发业务的线程
childGroup = new NioEventLoopGroup();
bootstrap = new ServerBootstrap();
bootstrap.group(parentGroup, childGroup);
bootstrap.channel(NioServerSocketChannel.class);
bootstrap.childHandler(new ChannelInitializer<SocketChannel>() {
@Override
protected void initChannel(SocketChannel ch) throws Exception {
// 30秒空闲时间设置
ch.pipeline().addLast(new IdleStateHandler(30, 0, 60));
// HttpServerCodec将请求和应答消息解码为HTTP消息
ch.pipeline().addLast(new HttpServerCodec());
// 针对大文件上传时,把 HttpMessage 和 HttpContent 聚合成一个
// FullHttpRequest,并定义可以接受的数据大小64M(可以支持params+multipart)
ch.pipeline().addLast(new HttpObjectAggregator(64 * 1024));
// 针对大文件下发,分块写数据
ch.pipeline().addLast(new ChunkedWriteHandler());
// WebSocket 访问地址
// ch.pipeline().addLast(new WebSocketServerProtocolHandler("/akaxin/ws"));
// 自定义handler
ch.pipeline().addLast(new WsServerHandler(executor));
}
});
}
// 启动websocket服务
public void start(String address, int port) throws Exception {
try {
if (bootstrap != null) {
ChannelFuture channelFuture = bootstrap.bind(address, port).sync();
channelFuture.channel().closeFuture().addListener(new GenericFutureListener<Future<? super Void>>() {
@Override
public void operationComplete(Future<? super Void> future) throws Exception {
closeGracefully();
}
});
} else {
throw new Exception();
}
} catch (Exception e) {
closeGracefully();
throw new Exception("start websocket server error", e);
}
}
private void closeGracefully() {
try {
if (parentGroup != null) {
// terminate all threads
parentGroup.shutdownGracefully();
// wait for all threads terminated
parentGroup.terminationFuture().sync();
}
if (childGroup != null) {
// terminate all threads
childGroup.shutdownGracefully();
// wait for all threads terminated
childGroup.terminationFuture().sync();
}
} catch (Exception es) {
logger.error(AkxProject.PLN + " shutdown netty gracefully error.", es);
}
}
public abstract void loadExecutor(AbstracteExecutor<Command, CommandResponse> executor);
}