空闲5分钟后,Netty websocket客户端无法从服务器读取新帧

时间:2016-01-31 23:16:12

标签: java websocket netty

我正在使用Netty服务器端和客户端来建立和控制websocket连接。我在服务器端有一个IdleStateHandler,它会在频道读取器,写入器或两者都闲置一段时间时发送用户事件。我有它,以便写入空闲事件将在空闲5分钟后触发,读取器空闲事件将在空闲6分钟后触发。在写入器空闲事件期间,服务器将向客户端发送ping帧,一旦从客户端接收到乒乓帧,该ping帧将重置写入器空闲时间以及读取器空闲时间。

问题是netty客户端在闲置5分钟后似乎没有读取任何新帧。我在客户端的通道上进行了一些状态检查,看看在5分钟的空闲时间之后它是否可写,已注册,打开和活动,并且所有状态都是真的,但是没有读取新帧。为了解决这个问题,我只是简单地将服务器端的IdleStateHandler时间更改为3分钟而不是5分钟,以便客户端接收ping帧并在空闲5分钟之前使用pong帧进行响应。

但这并不能解决根本问题。我想了解并能够控制客户端读卡器何时空闲并能够防止将来丢失或未读数据的问题。查看下面的代码,如果没有从客户端收到pong或heartbeat帧,则idle事件处理程序将关闭通道连接,但由于客户端不读取新帧,因此它永远不会获得关闭帧,因此服务器认为客户端没有连接,客户端认为它已连接,这显然会导致问题。有没有办法使用Netty在客户端获得对这个神奇的5分钟超时的更多控制?我在文档或来源中找不到任何相关内容。

以下是服务器中相关的空闲事件处理代码:

private class ConnectServerInitializer extends ChannelInitializer<SocketChannel> {

    private final IdleEventHandler idleEventHandler = new IdleEventHandler();
    private final SslContext sslCtx;

    private ConnectServerInitializer(SslContext sslCtx) {
        this.sslCtx = sslCtx;
    }

    @Override
    public void initChannel(SocketChannel ch) throws Exception {
        ChannelPipeline pipeline = ch.pipeline();
        if (sslCtx != null) {
            pipeline.addLast(sslCtx.newHandler(ch.alloc()));
        }
        pipeline.addLast(new HttpServerCodec());
        pipeline.addLast(new HttpObjectAggregator(65536));
        pipeline.addLast(idleEventHandler.newStateHandler());
        pipeline.addLast(idleEventHandler);
        pipeline.addLast(getHandler());
    }

}

@Sharable
private class IdleEventHandler extends ChannelDuplexHandler {

    private static final String HEARTBEAT_CONTENT = "--heartbeat--";
    private static final int READER_IDLE_TIMEOUT = 200; // 20 seconds more that writer to allow for pong response
    private static final int WRITER_IDLE_TIMEOUT = 180; // NOTE: netty clients will not read frames after 5 minutes of being idle
    // This is a fallback for when clients do not support ping/pong frames
    private final AttributeKey<Boolean> USE_HEARTBEAT = AttributeKey.valueOf("use-heartbeat");

    @Override
    public void userEventTriggered(ChannelHandlerContext ctx, Object event) throws Exception {
        if (event instanceof IdleStateEvent) {
            IdleStateEvent e = (IdleStateEvent) event;
            Boolean useHeartbeat = ctx.attr(USE_HEARTBEAT).get();
            if (e.state() == IdleState.READER_IDLE) {
                if (useHeartbeat == null) {
                    logger.info("Client " + ctx.channel() + " has not responded to ping frame. Sending heartbeat message...");
                    ctx.attr(USE_HEARTBEAT).set(true);
                    sendHeartbeat(ctx);
                } else {
                    logger.warn("Client " + ctx.channel() + " has been idle for too long. Closing websocket connection...");
                    ctx.close();
                }
            } else if (e.state() == IdleState.WRITER_IDLE || e.state() == IdleState.ALL_IDLE) {
                if (useHeartbeat == null || !useHeartbeat) {
                    ByteBuf ping = Unpooled.wrappedBuffer(HEARTBEAT_CONTENT.getBytes());
                    ctx.writeAndFlush(new PingWebSocketFrame(ping));
                } else {
                    sendHeartbeat(ctx);
                }
            }
        }
    }

    private void sendHeartbeat(ChannelHandlerContext ctx) {
        String json = getHandler().getMessenger().serialize(new HeartbeatMessage(HEARTBEAT_CONTENT));
        ctx.writeAndFlush(new TextWebSocketFrame(json));
    }

    private IdleStateHandler newStateHandler() {
        return new IdleStateHandler(READER_IDLE_TIMEOUT, WRITER_IDLE_TIMEOUT, WRITER_IDLE_TIMEOUT);
    }
}

1 个答案:

答案 0 :(得分:1)

您的问题与防火墙的超时有关。某些防火墙在5分钟后超时,如果超过此超时,则会以静默方式丢弃连接。因此,客户端和服务器需要有一些读取超时来检查这个事实,服务器,客户端或两者都有某种ping消息。当您通过IPv6运行协议时,防火墙问题将会减少,因为大多数IPv6防火墙主要是无状态的,并且通常不会更改连接的端口,因此来自客户端的数据包会再次重新激活防火墙中的条目。

当您有5分钟超时的许多时刻时,您应该考虑是否可以将每个1分钟的来自websockets的额外负载与简单轮询http循环的负载进行比较,因为这会减少服务器上的内存压力。 / p>