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();
        }
    }
}
相关推荐
weixin79893765432...43 分钟前
深入浅出 WebSocket 协议
websocket·http·socket·sse
callJJ1 小时前
WebSocket 两种实现方式对比与入门
java·python·websocket·网络协议·stomp
今晚务必早点睡14 小时前
系统通信方式实战详解:HTTP、RPC、MQ、WebSocket 各用在什么场景?(附 SDK 示例)
websocket·http·rpc
*才华有限公司*19 小时前
RTSP视频流播放系统
java·git·websocket·网络协议·信息与通信
栗子叶20 小时前
网页接收服务端消息的几种方式
前端·websocket·http·通信
栗子叶1 天前
SSE、长轮询与 WebSocket 连接资源对比及 Spring Boot 配置指南
spring boot·websocket·网络协议
softshow10261 天前
Vue3 :封装 WebRTC 低延迟视频流与 WebSocket 实时状态驱动的大屏可视化
websocket·网络协议·webrtc
克里斯蒂亚诺更新2 天前
理解即时通信Socket以及用NodeJs实现WebSocket
网络·websocket·网络协议
码农很忙2 天前
从0到1搭建实时日志监控系统:基于WebSocket + Elasticsearch的实战方案
websocket·网络协议·elasticsearch
寂寞恋上夜3 天前
异步任务怎么设计:轮询/WebSocket/回调(附PRD写法)
网络·人工智能·websocket·网络协议·markdown转xmind·deepseek思维导图