Fix infinite loop while handling a client socket / Retrofit EchoClient
This commit is contained in:
parent
b4610acda1
commit
4b673c4ebb
@ -15,15 +15,22 @@
|
|||||||
*/
|
*/
|
||||||
package io.netty.example.echo;
|
package io.netty.example.echo;
|
||||||
|
|
||||||
import java.net.InetSocketAddress;
|
import io.netty.buffer.ChannelBuffer;
|
||||||
import java.util.concurrent.Executors;
|
import io.netty.buffer.ChannelBuffers;
|
||||||
|
import io.netty.channel.ChannelBufferHolder;
|
||||||
|
import io.netty.channel.ChannelBufferHolders;
|
||||||
|
import io.netty.channel.ChannelInboundHandlerAdapter;
|
||||||
|
import io.netty.channel.ChannelInboundHandlerContext;
|
||||||
|
import io.netty.channel.EventLoop;
|
||||||
|
import io.netty.channel.MultithreadEventLoop;
|
||||||
|
import io.netty.channel.socket.SocketChannel;
|
||||||
|
import io.netty.channel.socket.nio.NioSocketChannel;
|
||||||
|
import io.netty.channel.socket.nio.SelectorEventLoop;
|
||||||
|
import io.netty.handler.logging.LoggingHandler;
|
||||||
|
import io.netty.logging.InternalLogLevel;
|
||||||
|
|
||||||
import io.netty.bootstrap.ClientBootstrap;
|
import java.net.InetSocketAddress;
|
||||||
import io.netty.channel.ChannelFuture;
|
import java.util.concurrent.atomic.AtomicLong;
|
||||||
import io.netty.channel.ChannelPipeline;
|
|
||||||
import io.netty.channel.ChannelPipelineFactory;
|
|
||||||
import io.netty.channel.Channels;
|
|
||||||
import io.netty.channel.socket.nio.NioClientSocketChannelFactory;
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Sends one message when a connection is open and echoes back any received
|
* Sends one message when a connection is open and echoes back any received
|
||||||
@ -36,6 +43,7 @@ public class EchoClient {
|
|||||||
private final String host;
|
private final String host;
|
||||||
private final int port;
|
private final int port;
|
||||||
private final int firstMessageSize;
|
private final int firstMessageSize;
|
||||||
|
private final AtomicLong transferredBytes = new AtomicLong();
|
||||||
|
|
||||||
public EchoClient(String host, int port, int firstMessageSize) {
|
public EchoClient(String host, int port, int firstMessageSize) {
|
||||||
this.host = host;
|
this.host = host;
|
||||||
@ -43,28 +51,54 @@ public class EchoClient {
|
|||||||
this.firstMessageSize = firstMessageSize;
|
this.firstMessageSize = firstMessageSize;
|
||||||
}
|
}
|
||||||
|
|
||||||
public void run() {
|
public void run() throws Exception {
|
||||||
// Configure the client.
|
EventLoop loop = new MultithreadEventLoop(SelectorEventLoop.class);
|
||||||
ClientBootstrap bootstrap = new ClientBootstrap(
|
SocketChannel s = new NioSocketChannel();
|
||||||
new NioClientSocketChannelFactory(
|
s.config().setTcpNoDelay(true);
|
||||||
Executors.newCachedThreadPool()));
|
s.pipeline().addLast("logger", new LoggingHandler(InternalLogLevel.INFO));
|
||||||
|
s.pipeline().addLast("echoer", new ChannelInboundHandlerAdapter<Byte>() {
|
||||||
|
|
||||||
// Set up the pipeline factory.
|
private final ChannelBuffer firstMessage;
|
||||||
bootstrap.setPipelineFactory(new ChannelPipelineFactory() {
|
{
|
||||||
public ChannelPipeline getPipeline() throws Exception {
|
if (firstMessageSize <= 0) {
|
||||||
return Channels.pipeline(
|
throw new IllegalArgumentException(
|
||||||
new EchoClientHandler(firstMessageSize));
|
"firstMessageSize: " + firstMessageSize);
|
||||||
|
}
|
||||||
|
firstMessage = ChannelBuffers.buffer(firstMessageSize);
|
||||||
|
for (int i = 0; i < firstMessage.capacity(); i ++) {
|
||||||
|
firstMessage.writeByte((byte) i);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public ChannelBufferHolder<Byte> newInboundBuffer(ChannelInboundHandlerContext<Byte> ctx) {
|
||||||
|
return ChannelBufferHolders.byteBuffer(ChannelBuffers.dynamicBuffer());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void channelActive(ChannelInboundHandlerContext<Byte> ctx)
|
||||||
|
throws Exception {
|
||||||
|
ctx.write(firstMessage);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void inboundBufferUpdated(
|
||||||
|
ChannelInboundHandlerContext<Byte> ctx) throws Exception {
|
||||||
|
ChannelBuffer in = ctx.in().byteBuffer();
|
||||||
|
ChannelBuffer out = ctx.out().byteBuffer();
|
||||||
|
transferredBytes.addAndGet(in.readableBytes());
|
||||||
|
|
||||||
|
out.discardReadBytes();
|
||||||
|
out.writeBytes(in);
|
||||||
|
in.clear();
|
||||||
|
ctx.flush();
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
loop.register(s).awaitUninterruptibly().rethrowIfFailed();
|
||||||
|
s.connect(new InetSocketAddress(host, port)).awaitUninterruptibly().rethrowIfFailed();
|
||||||
|
|
||||||
// Start the connection attempt.
|
// FIXME: Wait until the connection is closed or the connection attempt fails.
|
||||||
ChannelFuture future = bootstrap.connect(new InetSocketAddress(host, port));
|
// FIXME: Show how to shut down.
|
||||||
|
|
||||||
// Wait until the connection is closed or the connection attempt fails.
|
|
||||||
future.channel().getCloseFuture().awaitUninterruptibly();
|
|
||||||
|
|
||||||
// Shut down thread pools to exit.
|
|
||||||
bootstrap.releaseExternalResources();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public static void main(String[] args) throws Exception {
|
public static void main(String[] args) throws Exception {
|
||||||
|
@ -33,6 +33,7 @@ import io.netty.logging.InternalLogLevel;
|
|||||||
import java.net.InetSocketAddress;
|
import java.net.InetSocketAddress;
|
||||||
import java.util.ArrayDeque;
|
import java.util.ArrayDeque;
|
||||||
import java.util.Queue;
|
import java.util.Queue;
|
||||||
|
import java.util.concurrent.atomic.AtomicLong;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Echoes back any received data from a client.
|
* Echoes back any received data from a client.
|
||||||
@ -40,6 +41,7 @@ import java.util.Queue;
|
|||||||
public class EchoServer {
|
public class EchoServer {
|
||||||
|
|
||||||
private final int port;
|
private final int port;
|
||||||
|
private final AtomicLong transferredBytes = new AtomicLong();
|
||||||
|
|
||||||
public EchoServer(int port) {
|
public EchoServer(int port) {
|
||||||
this.port = port;
|
this.port = port;
|
||||||
@ -69,6 +71,7 @@ public class EchoServer {
|
|||||||
if (s == null) {
|
if (s == null) {
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
s.config().setTcpNoDelay(true);
|
||||||
s.pipeline().addLast("logger", new LoggingHandler(InternalLogLevel.INFO));
|
s.pipeline().addLast("logger", new LoggingHandler(InternalLogLevel.INFO));
|
||||||
s.pipeline().addLast("echoer", new ChannelInboundHandlerAdapter<Byte>() {
|
s.pipeline().addLast("echoer", new ChannelInboundHandlerAdapter<Byte>() {
|
||||||
@Override
|
@Override
|
||||||
@ -80,6 +83,8 @@ public class EchoServer {
|
|||||||
public void inboundBufferUpdated(ChannelInboundHandlerContext<Byte> ctx) {
|
public void inboundBufferUpdated(ChannelInboundHandlerContext<Byte> ctx) {
|
||||||
ChannelBuffer in = ctx.in().byteBuffer();
|
ChannelBuffer in = ctx.in().byteBuffer();
|
||||||
ChannelBuffer out = ctx.out().byteBuffer();
|
ChannelBuffer out = ctx.out().byteBuffer();
|
||||||
|
transferredBytes.addAndGet(in.readableBytes());
|
||||||
|
|
||||||
out.discardReadBytes();
|
out.discardReadBytes();
|
||||||
out.writeBytes(in);
|
out.writeBytes(in);
|
||||||
in.clear();
|
in.clear();
|
||||||
|
@ -546,8 +546,12 @@ public abstract class AbstractChannel extends DefaultAttributeMap implements Cha
|
|||||||
assert eventLoop().inEventLoop();
|
assert eventLoop().inEventLoop();
|
||||||
assert connectFuture != null;
|
assert connectFuture != null;
|
||||||
try {
|
try {
|
||||||
|
boolean wasActive = isActive();
|
||||||
doFinishConnect();
|
doFinishConnect();
|
||||||
connectFuture.setSuccess();
|
connectFuture.setSuccess();
|
||||||
|
if (!wasActive && isActive()) {
|
||||||
|
pipeline().fireChannelActive();
|
||||||
|
}
|
||||||
} catch (Throwable t) {
|
} catch (Throwable t) {
|
||||||
connectFuture.setFailure(t);
|
connectFuture.setFailure(t);
|
||||||
pipeline().fireExceptionCaught(t);
|
pipeline().fireExceptionCaught(t);
|
||||||
|
@ -87,6 +87,6 @@ public abstract class AbstractNioChannel extends AbstractChannel {
|
|||||||
}
|
}
|
||||||
|
|
||||||
SelectorEventLoop loop = (SelectorEventLoop) eventLoop();
|
SelectorEventLoop loop = (SelectorEventLoop) eventLoop();
|
||||||
selectionKey = javaChannel().register(loop.selector, SelectionKey.OP_READ, this);
|
selectionKey = javaChannel().register(loop.selector, isActive()? SelectionKey.OP_READ : 0, this);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -116,8 +116,10 @@ public class NioSocketChannel extends AbstractNioChannel implements io.netty.cha
|
|||||||
boolean success = false;
|
boolean success = false;
|
||||||
try {
|
try {
|
||||||
boolean connected = javaChannel().connect(remoteAddress);
|
boolean connected = javaChannel().connect(remoteAddress);
|
||||||
if (!connected) {
|
if (connected) {
|
||||||
selectionKey().interestOps(selectionKey().interestOps() | SelectionKey.OP_CONNECT);
|
selectionKey().interestOps(SelectionKey.OP_READ);
|
||||||
|
} else {
|
||||||
|
selectionKey().interestOps(SelectionKey.OP_CONNECT);
|
||||||
}
|
}
|
||||||
success = true;
|
success = true;
|
||||||
return connected;
|
return connected;
|
||||||
@ -133,6 +135,7 @@ public class NioSocketChannel extends AbstractNioChannel implements io.netty.cha
|
|||||||
if (!javaChannel().finishConnect()) {
|
if (!javaChannel().finishConnect()) {
|
||||||
throw new Error();
|
throw new Error();
|
||||||
}
|
}
|
||||||
|
selectionKey().interestOps(SelectionKey.OP_READ);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
Loading…
Reference in New Issue
Block a user