netty websockt之断连重试

断连重试有以下两点考虑:

1、连接异常,比如网络抖动导致连接失败;

2、连接过程中断开连接重试;

主要用到两个工具类:

ChannelFutureListener监听ChannelFuture..isSuccess();

ChannelInboundHandlerAdapter重写channelInactive,当连接变为不活跃,则回调该方法。

完整代码如下:

java 复制代码
@Component
public class WebSocketClient {

    private Channel channel;

    private Bootstrap bootstrap;

    private URI uri;

    private MessageHandler messageHandler;

    private WebSocketClientHandler handler;

    private volatile AtomicInteger atomicCount = new AtomicInteger(0);

    public WebSocketClient initClient(String host, MessageHandler messageHandler) throws Exception {
        this.messageHandler = messageHandler;
        if (StringUtils.isEmpty(host)) {
            throw new RuntimeException("未配置host.");
        }
        uri = new URI(host);
        String scheme = uri.getScheme() == null? WssSchemeEnum.WS.getValue() : uri.getScheme();

        //判断是否ssl连接,如果是则设置为可信
        final boolean ssl = WssSchemeEnum.WSS.getValue().equalsIgnoreCase(scheme);
        final SslContext sslCtx;
        if (ssl) {
            sslCtx = SslContextBuilder.forClient()
                    .trustManager(InsecureTrustManagerFactory.INSTANCE).build();
        } else {
            sslCtx = null;
        }

        EventLoopGroup group = new NioEventLoopGroup();
        try {
            bootstrap = new Bootstrap();
            bootstrap.group(group)
                    .channel(NioSocketChannel.class)
                    .handler(new ChannelInitializer<SocketChannel>() {
                        @Override
                        protected void initChannel(SocketChannel ch) {
                            ChannelPipeline p = ch.pipeline();
                            p.addFirst(new ChannelInboundHandlerAdapter() {
                                @Override
                                public void channelInactive(ChannelHandlerContext ctx) throws Exception {
                                    log.error("【{}】检测到wss断连, 第 {} 次发起重连.", exchange, atomicCount.incrementAndGet());
                                    super.channelInactive(ctx);
                                    ctx.channel().eventLoop().schedule(WebSocketClient.this::doConnect, 3000, TimeUnit.MILLISECONDS);
                                }
                            });
                            if (sslCtx != null) {
                                p.addLast(sslCtx.newHandler(ch.alloc(), uri.getHost(), getUriPort(uri)));
                            }
                            p.addLast(new HttpClientCodec());
                            p.addLast(new HttpObjectAggregator(8192));
                            p.addLast(WebSocketClientCompressionHandler.INSTANCE);
                            handler = new WebSocketClientHandler(
                                    WebSocketClientHandshakerFactory.newHandshaker(
                                            uri, WebSocketVersion.V13, null, true, new DefaultHttpHeaders()), exchange, messageHandler);
                            p.addLast(handler);
                        }
                    });
        } catch (Exception e) {
            log.error("wss创建client异常. e:", e);
            if (bootstrap != null) {
                bootstrap.config().group().shutdownGracefully();
            }
            throw new RuntimeException("初始化wss连接异常. e: " + e);
        }
        doConnect();
        return this;
    }

    public void doConnect() {
        try {
            ChannelFuture future = bootstrap.connect(uri.getHost(), getUriPort(uri)).sync();
            handler.handshakeFuture().sync();
            future.addListener((ChannelFutureListener) cf -> {
                        if (future.isSuccess()) {
                            channel = future.channel();
                            WssManger.addChannel(exchange, channel);
                            log.info("连接成功.");
                            messageHandler.connectSuccessAction(future.channel());
                            atomicCount.set(0);
                        } else {
                            log.error("监听断连, wss第 {} 次发起重连. ", atomicCount.incrementAndGet());
                            future.channel().eventLoop().schedule(WebSocketClient.this::doConnect, 3000, TimeUnit.MILLISECONDS);
                        }
                    });
        }catch (Exception e) {
            log.error("连接异常. e:" + e);
            if (bootstrap != null) {
                log.info("wss连接异常,第 {} 次发起重连.", atomicCount.incrementAndGet());
                bootstrap.config().group().schedule(WebSocketClient.this::doConnect, 3000, TimeUnit.MILLISECONDS);
            }
        }
    }

    /**
     * 根据URI获取对应的port
     *
     * @param uri uri
     * @return port
     */
    private int getUriPort(URI uri) {
        String scheme = uri.getScheme() == null? WssSchemeEnum.WS.getValue() : uri.getScheme();
        if (!WssSchemeEnum.allScheme().contains(scheme)) {
            throw new RuntimeException("Only WS(S) is supported.");
        }
        if (uri.getPort() == -1) {
            if (WssSchemeEnum.WS.getValue().equalsIgnoreCase(scheme)) {
                return WssSchemeEnum.WS.getPort();
            } else if (WssSchemeEnum.WSS.getValue().equalsIgnoreCase(scheme)) {
                return WssSchemeEnum.WSS.getPort();
            } else {
                return -1;
            }
        } else {
            return uri.getPort();
        }
    }
}
相关推荐
咖啡星人k1 天前
MonkeyCode 网络架构:WebSocket、SSE与实时协作的技术选型
网络·websocket·架构·monkeycode
tobias.b1 天前
JumpServer4\.10\.16离线部署\+外部Nginx反向代理 解决30分钟空闲断开WebSocket超时(延长10天)
运维·websocket·nginx
Jacob程序员1 天前
WebSSH技术实现全解析
linux·运维·服务器·websocket
brycegao3211 天前
金融交易App客户端架构实战 | 模块化、WebSocket治理、多线路容灾全解
websocket·金融·组件化·android架构·客户端模块化·移动端稳定性·多线路网络
ttwuai2 天前
XYGo Admin 扩展开发:WebSocket 事件注册与实时推送实战
python·websocket·网络协议·golang·后台框架
喵个咪2 天前
实时游戏网络协议深度对比:KCP vs WebRTC vs WebSocket
后端·websocket·webrtc
AIFQuant3 天前
量化私募回测系统:高质量股票/外汇历史数据 API 选型与接入
python·websocket·金融·ai量化
下北沢美食家4 天前
WebSocket入门
网络·websocket·网络协议
zh路西法4 天前
【rosbridge-websocket】跨网络的ROS1与ROS2通讯法(上)
linux·网络·c++·python·websocket·网络协议
必胜刻4 天前
一个异步生成游戏功能的落地复盘:Redis Stream + WebSocket + 状态补偿
redis·websocket·golang·gin·状态补偿