From 987ebb90ec43b928823f2591a558072b6de3e798 Mon Sep 17 00:00:00 2001 From: Norman Maurer Date: Tue, 28 Jun 2016 06:42:02 +0200 Subject: [PATCH] Share code between ReadTimeoutHandler and IdleStateHandler Motivation: ReadTimeoutHandler and IdleStateHandler have duplicated code, we should share whatever possible. Modifications: Let ReadTimeoutHandler extend IdleStateHandler. Result: Remove code duplication. --- .../handler/timeout/ReadTimeoutHandler.java | 141 +----------------- 1 file changed, 5 insertions(+), 136 deletions(-) diff --git a/handler/src/main/java/io/netty/handler/timeout/ReadTimeoutHandler.java b/handler/src/main/java/io/netty/handler/timeout/ReadTimeoutHandler.java index f06b254295..1fe47889d8 100644 --- a/handler/src/main/java/io/netty/handler/timeout/ReadTimeoutHandler.java +++ b/handler/src/main/java/io/netty/handler/timeout/ReadTimeoutHandler.java @@ -19,10 +19,8 @@ import io.netty.bootstrap.ServerBootstrap; import io.netty.channel.Channel; import io.netty.channel.ChannelDuplexHandler; import io.netty.channel.ChannelHandlerContext; -import io.netty.channel.ChannelInboundHandlerAdapter; import io.netty.channel.ChannelInitializer; -import java.util.concurrent.ScheduledFuture; import java.util.concurrent.TimeUnit; /** @@ -61,18 +59,7 @@ import java.util.concurrent.TimeUnit; * @see WriteTimeoutHandler * @see IdleStateHandler */ -public class ReadTimeoutHandler extends ChannelInboundHandlerAdapter { - private static final long MIN_TIMEOUT_NANOS = TimeUnit.MILLISECONDS.toNanos(1); - - private final long timeoutNanos; - - private long lastReadTime; - - private volatile ScheduledFuture timeout; - - private volatile int state; // 0 - none, 1 - Initialized, 2 - Destroyed; - - private volatile boolean reading; +public class ReadTimeoutHandler extends IdleStateHandler { private boolean closed; /** @@ -94,97 +81,13 @@ public class ReadTimeoutHandler extends ChannelInboundHandlerAdapter { * the {@link TimeUnit} of {@code timeout} */ public ReadTimeoutHandler(long timeout, TimeUnit unit) { - if (unit == null) { - throw new NullPointerException("unit"); - } - - if (timeout <= 0) { - timeoutNanos = 0; - } else { - timeoutNanos = Math.max(unit.toNanos(timeout), MIN_TIMEOUT_NANOS); - } + super(timeout, 0, 0, unit); } @Override - public void handlerAdded(ChannelHandlerContext ctx) throws Exception { - if (ctx.channel().isActive() && ctx.channel().isRegistered()) { - // channelActvie() event has been fired already, which means this.channelActive() will - // not be invoked. We have to initialize here instead. - initialize(ctx); - } else { - // channelActive() event has not been fired yet. this.channelActive() will be invoked - // and initialization will occur there. - } - } - - @Override - public void handlerRemoved(ChannelHandlerContext ctx) throws Exception { - destroy(); - } - - @Override - public void channelRegistered(ChannelHandlerContext ctx) throws Exception { - // Initialize early if channel is active already. - if (ctx.channel().isActive()) { - initialize(ctx); - } - super.channelRegistered(ctx); - } - - @Override - public void channelActive(ChannelHandlerContext ctx) throws Exception { - // This method will be invoked only if this handler was added - // before channelActive() event is fired. If a user adds this handler - // after the channelActive() event, initialize() will be called by beforeAdd(). - initialize(ctx); - super.channelActive(ctx); - } - - @Override - public void channelInactive(ChannelHandlerContext ctx) throws Exception { - destroy(); - super.channelInactive(ctx); - } - - @Override - public void channelRead(ChannelHandlerContext ctx, Object msg) throws Exception { - reading = true; - ctx.fireChannelRead(msg); - } - - @Override - public void channelReadComplete(ChannelHandlerContext ctx) throws Exception { - lastReadTime = System.nanoTime(); - reading = false; - ctx.fireChannelReadComplete(); - } - - private void initialize(ChannelHandlerContext ctx) { - // Avoid the case where destroy() is called before scheduling timeouts. - // See: https://github.com/netty/netty/issues/143 - switch (state) { - case 1: - case 2: - return; - } - - state = 1; - - lastReadTime = System.nanoTime(); - if (timeoutNanos > 0) { - timeout = ctx.executor().schedule( - new ReadTimeoutTask(ctx), - timeoutNanos, TimeUnit.NANOSECONDS); - } - } - - private void destroy() { - state = 2; - - if (timeout != null) { - timeout.cancel(false); - timeout = null; - } + protected final void channelIdle(ChannelHandlerContext ctx, IdleStateEvent evt) throws Exception { + assert evt.state() == IdleState.READER_IDLE; + readTimedOut(ctx); } /** @@ -197,38 +100,4 @@ public class ReadTimeoutHandler extends ChannelInboundHandlerAdapter { closed = true; } } - - private final class ReadTimeoutTask implements Runnable { - - private final ChannelHandlerContext ctx; - - ReadTimeoutTask(ChannelHandlerContext ctx) { - this.ctx = ctx; - } - - @Override - public void run() { - if (!ctx.channel().isOpen()) { - return; - } - - long nextDelay = timeoutNanos; - if (!reading) { - nextDelay -= System.nanoTime() - lastReadTime; - } - - if (nextDelay <= 0) { - // Read timed out - set a new timeout and notify the callback. - timeout = ctx.executor().schedule(this, timeoutNanos, TimeUnit.NANOSECONDS); - try { - readTimedOut(ctx); - } catch (Throwable t) { - ctx.fireExceptionCaught(t); - } - } else { - // Read occurred before the timeout - set a new timeout with shorter delay. - timeout = ctx.executor().schedule(this, nextDelay, TimeUnit.NANOSECONDS); - } - } - } }