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();
        }
    }
}
相关推荐
等一场春雨19 分钟前
springboot 3 websocket react 系统提示,选手实时数据更新监控
spring boot·websocket·react.js
却道天凉_好个秋37 分钟前
音视频学习(二十八):websocket-flv
websocket·音视频·flv
龙哥·三年风水1 天前
workman服务端开发模式-应用开发-vue-element-admin封装websocket
分布式·websocket·vue
ZoeLandia2 天前
WebSocket | 背景 概念 原理 使用 优缺点及适用场景
网络·websocket·网络协议
zquwei2 天前
SpringCloudGateway+Nacos注册与转发Netty+WebSocket
java·网络·分布式·后端·websocket·网络协议·spring
carterslam2 天前
解决:websocket 1002 connection rejected 426upgrade required
网络·websocket·网络协议
抓住鼹鼠不撒手2 天前
xterm.js结合websocket实现web ssh
前端·javascript·websocket
学前端的小朱2 天前
Echarts实现大屏可视化
websocket·echarts·nodejs·vue3·vite·koa·cors
龙少95434 天前
【Http,Netty,Socket,WebSocket的应用场景和区别】
java·后端·websocket·网络协议·http
m0_748232924 天前
前端在WebSocket中加入Token
前端·websocket·网络协议