Add the 'nextBufferType' parameter to ByteArrayEncoder like did to StringEncoder / Consistent parameter order

This commit is contained in:
Trustin Lee 2013-02-08 01:36:41 +09:00
parent b8c0751023
commit d8c0bf3be2
12 changed files with 147 additions and 59 deletions

View File

@ -15,14 +15,16 @@
*/ */
package io.netty.handler.codec.bytes; package io.netty.handler.codec.bytes;
import io.netty.buffer.BufType;
import io.netty.buffer.ByteBuf; import io.netty.buffer.ByteBuf;
import io.netty.buffer.MessageBuf; import io.netty.buffer.MessageBuf;
import io.netty.buffer.Unpooled; import io.netty.buffer.Unpooled;
import io.netty.channel.ChannelHandlerContext; import io.netty.channel.ChannelHandlerContext;
import io.netty.channel.ChannelOutboundMessageHandlerAdapter;
import io.netty.channel.ChannelPipeline; import io.netty.channel.ChannelPipeline;
import io.netty.channel.ChannelPromise;
import io.netty.handler.codec.LengthFieldBasedFrameDecoder; import io.netty.handler.codec.LengthFieldBasedFrameDecoder;
import io.netty.handler.codec.LengthFieldPrepender; import io.netty.handler.codec.LengthFieldPrepender;
import io.netty.handler.codec.MessageToMessageEncoder;
/** /**
* Encodes the requested array of bytes into a {@link ByteBuf}. * Encodes the requested array of bytes into a {@link ByteBuf}.
@ -48,22 +50,53 @@ import io.netty.handler.codec.MessageToMessageEncoder;
* } * }
* </pre> * </pre>
*/ */
public class ByteArrayEncoder extends MessageToMessageEncoder<byte[]> { public class ByteArrayEncoder extends ChannelOutboundMessageHandlerAdapter<byte[]> {
public ByteArrayEncoder() { private final BufType nextBufferType;
super(byte[].class);
}
@Override public ByteArrayEncoder(BufType nextBufferType) {
public MessageBuf<byte[]> newOutboundBuffer(ChannelHandlerContext ctx) throws Exception { if (nextBufferType == null) {
return Unpooled.messageBuffer(); throw new NullPointerException("nextBufferType");
} }
this.nextBufferType = nextBufferType;
@Override }
protected ByteBuf encode(ChannelHandlerContext ctx, byte[] msg) throws Exception {
if (msg.length == 0) { @Override
return null; public void flush(ChannelHandlerContext ctx, ChannelPromise promise) throws Exception {
MessageBuf<Object> in = ctx.outboundMessageBuffer();
MessageBuf<Object> msgOut = ctx.nextOutboundMessageBuffer();
ByteBuf byteOut = ctx.nextOutboundByteBuffer();
try {
for (;;) {
Object m = in.poll();
if (m == null) {
break;
}
if (!(m instanceof byte[])) {
msgOut.add(m);
continue;
}
byte[] a = (byte[]) m;
if (a.length == 0) {
continue;
}
switch (nextBufferType) {
case BYTE:
byteOut.writeBytes(a);
break;
case MESSAGE:
msgOut.add(Unpooled.wrappedBuffer(a));
break;
default:
throw new Error();
}
}
} finally {
ctx.flush(promise);
} }
return Unpooled.wrappedBuffer(msg);
} }
} }

View File

@ -15,13 +15,16 @@
*/ */
package io.netty.handler.codec.string; package io.netty.handler.codec.string;
import io.netty.buffer.BufType;
import io.netty.buffer.ByteBuf; import io.netty.buffer.ByteBuf;
import io.netty.buffer.MessageBuf;
import io.netty.buffer.Unpooled; import io.netty.buffer.Unpooled;
import io.netty.channel.ChannelHandler.Sharable; import io.netty.channel.ChannelHandler.Sharable;
import io.netty.channel.ChannelHandlerContext; import io.netty.channel.ChannelHandlerContext;
import io.netty.channel.ChannelOutboundMessageHandlerAdapter;
import io.netty.channel.ChannelPipeline; import io.netty.channel.ChannelPipeline;
import io.netty.channel.ChannelPromise;
import io.netty.handler.codec.LineBasedFrameDecoder; import io.netty.handler.codec.LineBasedFrameDecoder;
import io.netty.handler.codec.MessageToMessageEncoder;
import java.nio.charset.Charset; import java.nio.charset.Charset;
@ -48,32 +51,67 @@ import java.nio.charset.Charset;
* @apiviz.landmark * @apiviz.landmark
*/ */
@Sharable @Sharable
public class StringEncoder extends MessageToMessageEncoder<CharSequence> { public class StringEncoder extends ChannelOutboundMessageHandlerAdapter<CharSequence> {
private final BufType nextBufferType;
// TODO Use CharsetEncoder instead. // TODO Use CharsetEncoder instead.
private final Charset charset; private final Charset charset;
/** /**
* Creates a new instance with the current system character set. * Creates a new instance with the current system character set.
*/ */
public StringEncoder() { public StringEncoder(BufType nextBufferType) {
this(Charset.defaultCharset()); this(nextBufferType, Charset.defaultCharset());
} }
/** /**
* Creates a new instance with the specified character set. * Creates a new instance with the specified character set.
*/ */
public StringEncoder(Charset charset) { public StringEncoder(BufType nextBufferType, Charset charset) {
super(CharSequence.class); if (nextBufferType == null) {
throw new NullPointerException("nextBufferType");
}
if (charset == null) { if (charset == null) {
throw new NullPointerException("charset"); throw new NullPointerException("charset");
} }
this.nextBufferType = nextBufferType;
this.charset = charset; this.charset = charset;
} }
@Override @Override
protected Object encode(ChannelHandlerContext ctx, CharSequence msg) throws Exception { public void flush(ChannelHandlerContext ctx, ChannelPromise promise) throws Exception {
return Unpooled.copiedBuffer(msg, charset); MessageBuf<Object> in = ctx.outboundMessageBuffer();
MessageBuf<Object> msgOut = ctx.nextOutboundMessageBuffer();
ByteBuf byteOut = ctx.nextOutboundByteBuffer();
try {
for (;;) {
Object m = in.poll();
if (m == null) {
break;
}
if (!(m instanceof CharSequence)) {
msgOut.add(m);
continue;
}
CharSequence s = (CharSequence) m;
ByteBuf encoded = Unpooled.copiedBuffer(s, charset);
switch (nextBufferType) {
case BYTE:
byteOut.writeBytes(encoded);
break;
case MESSAGE:
msgOut.add(encoded);
break;
default:
throw new Error();
}
}
} finally {
ctx.flush(promise);
}
} }
} }

View File

@ -15,27 +15,27 @@
*/ */
package io.netty.handler.codec.bytes; package io.netty.handler.codec.bytes;
import io.netty.buffer.BufType;
import io.netty.buffer.ByteBuf;
import io.netty.channel.embedded.EmbeddedMessageChannel;
import org.junit.Before;
import org.junit.Test;
import java.util.Random;
import static io.netty.buffer.Unpooled.*; import static io.netty.buffer.Unpooled.*;
import static org.hamcrest.core.Is.*; import static org.hamcrest.core.Is.*;
import static org.hamcrest.core.IsNull.*; import static org.hamcrest.core.IsNull.*;
import static org.junit.Assert.*; import static org.junit.Assert.*;
import io.netty.buffer.ByteBuf;
import io.netty.channel.embedded.EmbeddedMessageChannel;
import java.util.Random; @SuppressWarnings("ZeroLengthArrayAllocation")
import org.junit.Before;
import org.junit.Test;
/**
*/
public class ByteArrayEncoderTest { public class ByteArrayEncoderTest {
private EmbeddedMessageChannel ch; private EmbeddedMessageChannel ch;
@Before @Before
public void setUp() { public void setUp() {
ch = new EmbeddedMessageChannel(new ByteArrayEncoder()); ch = new EmbeddedMessageChannel(new ByteArrayEncoder(BufType.MESSAGE));
} }
@Test @Test

View File

@ -16,14 +16,15 @@
package io.netty.example.filetransfer; package io.netty.example.filetransfer;
import io.netty.bootstrap.ServerBootstrap; import io.netty.bootstrap.ServerBootstrap;
import io.netty.buffer.BufType;
import io.netty.channel.ChannelFuture; import io.netty.channel.ChannelFuture;
import io.netty.channel.ChannelHandlerContext; import io.netty.channel.ChannelHandlerContext;
import io.netty.channel.ChannelInboundMessageHandlerAdapter; import io.netty.channel.ChannelInboundMessageHandlerAdapter;
import io.netty.channel.ChannelInitializer; import io.netty.channel.ChannelInitializer;
import io.netty.channel.ChannelOption; import io.netty.channel.ChannelOption;
import io.netty.channel.DefaultFileRegion; import io.netty.channel.DefaultFileRegion;
import io.netty.channel.socket.SocketChannel;
import io.netty.channel.nio.NioEventLoopGroup; import io.netty.channel.nio.NioEventLoopGroup;
import io.netty.channel.socket.SocketChannel;
import io.netty.channel.socket.nio.NioServerSocketChannel; import io.netty.channel.socket.nio.NioServerSocketChannel;
import io.netty.handler.codec.LineBasedFrameDecoder; import io.netty.handler.codec.LineBasedFrameDecoder;
import io.netty.handler.codec.string.StringDecoder; import io.netty.handler.codec.string.StringDecoder;
@ -58,7 +59,7 @@ public class FileServer {
@Override @Override
public void initChannel(SocketChannel ch) throws Exception { public void initChannel(SocketChannel ch) throws Exception {
ch.pipeline().addLast( ch.pipeline().addLast(
new StringEncoder(CharsetUtil.UTF_8), new StringEncoder(BufType.BYTE, CharsetUtil.UTF_8),
new LineBasedFrameDecoder(8192), new LineBasedFrameDecoder(8192),
new StringDecoder(CharsetUtil.UTF_8), new StringDecoder(CharsetUtil.UTF_8),
new FileHandler()); new FileHandler());

View File

@ -16,11 +16,12 @@
package io.netty.example.rxtx; package io.netty.example.rxtx;
import io.netty.bootstrap.Bootstrap; import io.netty.bootstrap.Bootstrap;
import io.netty.buffer.BufType;
import io.netty.channel.ChannelFuture; import io.netty.channel.ChannelFuture;
import io.netty.channel.ChannelInitializer; import io.netty.channel.ChannelInitializer;
import io.netty.channel.oio.OioEventLoopGroup;
import io.netty.channel.rxtx.RxtxChannel; import io.netty.channel.rxtx.RxtxChannel;
import io.netty.channel.rxtx.RxtxDeviceAddress; import io.netty.channel.rxtx.RxtxDeviceAddress;
import io.netty.channel.oio.OioEventLoopGroup;
import io.netty.handler.codec.LineBasedFrameDecoder; import io.netty.handler.codec.LineBasedFrameDecoder;
import io.netty.handler.codec.string.StringDecoder; import io.netty.handler.codec.string.StringDecoder;
import io.netty.handler.codec.string.StringEncoder; import io.netty.handler.codec.string.StringEncoder;
@ -40,7 +41,7 @@ public final class RxtxClient {
public void initChannel(RxtxChannel ch) throws Exception { public void initChannel(RxtxChannel ch) throws Exception {
ch.pipeline().addLast( ch.pipeline().addLast(
new LineBasedFrameDecoder(32768), new LineBasedFrameDecoder(32768),
new StringEncoder(), new StringEncoder(BufType.BYTE),
new StringDecoder(), new StringDecoder(),
new RxtxClientHandler() new RxtxClientHandler()
); );

View File

@ -15,6 +15,7 @@
*/ */
package io.netty.example.securechat; package io.netty.example.securechat;
import io.netty.buffer.BufType;
import io.netty.channel.ChannelInitializer; import io.netty.channel.ChannelInitializer;
import io.netty.channel.ChannelPipeline; import io.netty.channel.ChannelPipeline;
import io.netty.channel.socket.SocketChannel; import io.netty.channel.socket.SocketChannel;
@ -51,7 +52,7 @@ public class SecureChatClientInitializer extends ChannelInitializer<SocketChanne
pipeline.addLast("framer", new DelimiterBasedFrameDecoder( pipeline.addLast("framer", new DelimiterBasedFrameDecoder(
8192, Delimiters.lineDelimiter())); 8192, Delimiters.lineDelimiter()));
pipeline.addLast("decoder", new StringDecoder()); pipeline.addLast("decoder", new StringDecoder());
pipeline.addLast("encoder", new StringEncoder()); pipeline.addLast("encoder", new StringEncoder(BufType.BYTE));
// and then business logic. // and then business logic.
pipeline.addLast("handler", new SecureChatClientHandler()); pipeline.addLast("handler", new SecureChatClientHandler());

View File

@ -15,6 +15,7 @@
*/ */
package io.netty.example.securechat; package io.netty.example.securechat;
import io.netty.buffer.BufType;
import io.netty.channel.ChannelInitializer; import io.netty.channel.ChannelInitializer;
import io.netty.channel.ChannelPipeline; import io.netty.channel.ChannelPipeline;
import io.netty.channel.socket.SocketChannel; import io.netty.channel.socket.SocketChannel;
@ -54,7 +55,7 @@ public class SecureChatServerInitializer extends ChannelInitializer<SocketChanne
pipeline.addLast("framer", new DelimiterBasedFrameDecoder( pipeline.addLast("framer", new DelimiterBasedFrameDecoder(
8192, Delimiters.lineDelimiter())); 8192, Delimiters.lineDelimiter()));
pipeline.addLast("decoder", new StringDecoder()); pipeline.addLast("decoder", new StringDecoder());
pipeline.addLast("encoder", new StringEncoder()); pipeline.addLast("encoder", new StringEncoder(BufType.BYTE));
// and then business logic. // and then business logic.
pipeline.addLast("handler", new SecureChatServerHandler()); pipeline.addLast("handler", new SecureChatServerHandler());

View File

@ -15,6 +15,7 @@
*/ */
package io.netty.example.telnet; package io.netty.example.telnet;
import io.netty.buffer.BufType;
import io.netty.channel.ChannelInitializer; import io.netty.channel.ChannelInitializer;
import io.netty.channel.ChannelPipeline; import io.netty.channel.ChannelPipeline;
import io.netty.channel.socket.SocketChannel; import io.netty.channel.socket.SocketChannel;
@ -28,7 +29,7 @@ import io.netty.handler.codec.string.StringEncoder;
*/ */
public class TelnetClientInitializer extends ChannelInitializer<SocketChannel> { public class TelnetClientInitializer extends ChannelInitializer<SocketChannel> {
private static final StringDecoder DECODER = new StringDecoder(); private static final StringDecoder DECODER = new StringDecoder();
private static final StringEncoder ENCODER = new StringEncoder(); private static final StringEncoder ENCODER = new StringEncoder(BufType.BYTE);
private static final TelnetClientHandler CLIENTHANDLER = new TelnetClientHandler(); private static final TelnetClientHandler CLIENTHANDLER = new TelnetClientHandler();
@Override @Override
public void initChannel(SocketChannel ch) throws Exception { public void initChannel(SocketChannel ch) throws Exception {

View File

@ -15,6 +15,7 @@
*/ */
package io.netty.example.telnet; package io.netty.example.telnet;
import io.netty.buffer.BufType;
import io.netty.channel.ChannelInitializer; import io.netty.channel.ChannelInitializer;
import io.netty.channel.ChannelPipeline; import io.netty.channel.ChannelPipeline;
import io.netty.channel.socket.SocketChannel; import io.netty.channel.socket.SocketChannel;
@ -28,7 +29,7 @@ import io.netty.handler.codec.string.StringEncoder;
*/ */
public class TelnetServerPipelineFactory extends ChannelInitializer<SocketChannel> { public class TelnetServerPipelineFactory extends ChannelInitializer<SocketChannel> {
private static final StringDecoder DECODER = new StringDecoder(); private static final StringDecoder DECODER = new StringDecoder();
private static final StringEncoder ENCODER = new StringEncoder(); private static final StringEncoder ENCODER = new StringEncoder(BufType.BYTE);
private static final TelnetServerHandler SERVERHANDLER = new TelnetServerHandler(); private static final TelnetServerHandler SERVERHANDLER = new TelnetServerHandler();
@Override @Override

View File

@ -17,6 +17,7 @@ package io.netty.testsuite.transport.socket;
import io.netty.bootstrap.Bootstrap; import io.netty.bootstrap.Bootstrap;
import io.netty.bootstrap.ServerBootstrap; import io.netty.bootstrap.ServerBootstrap;
import io.netty.buffer.BufType;
import io.netty.channel.Channel; import io.netty.channel.Channel;
import io.netty.channel.ChannelFuture; import io.netty.channel.ChannelFuture;
import io.netty.channel.ChannelHandlerContext; import io.netty.channel.ChannelHandlerContext;
@ -76,7 +77,7 @@ public class SocketStartTlsTest extends AbstractSocketTest {
public void initChannel(SocketChannel sch) throws Exception { public void initChannel(SocketChannel sch) throws Exception {
ChannelPipeline p = sch.pipeline(); ChannelPipeline p = sch.pipeline();
p.addLast("logger", new ByteLoggingHandler(LOG_LEVEL)); p.addLast("logger", new ByteLoggingHandler(LOG_LEVEL));
p.addLast(new LineBasedFrameDecoder(64), new StringDecoder(), new StringEncoder()); p.addLast(new LineBasedFrameDecoder(64), new StringDecoder(), new StringEncoder(BufType.BYTE));
p.addLast(executor, sh); p.addLast(executor, sh);
} }
}); });
@ -86,7 +87,7 @@ public class SocketStartTlsTest extends AbstractSocketTest {
public void initChannel(SocketChannel sch) throws Exception { public void initChannel(SocketChannel sch) throws Exception {
ChannelPipeline p = sch.pipeline(); ChannelPipeline p = sch.pipeline();
p.addLast("logger", new ByteLoggingHandler(LOG_LEVEL)); p.addLast("logger", new ByteLoggingHandler(LOG_LEVEL));
p.addLast(new LineBasedFrameDecoder(64), new StringDecoder(), new StringEncoder()); p.addLast(new LineBasedFrameDecoder(64), new StringDecoder(), new StringEncoder(BufType.BYTE));
p.addLast(executor, ch); p.addLast(executor, ch);
} }
}); });

View File

@ -17,6 +17,7 @@ package io.netty.testsuite.transport.socket;
import io.netty.bootstrap.Bootstrap; import io.netty.bootstrap.Bootstrap;
import io.netty.bootstrap.ServerBootstrap; import io.netty.bootstrap.ServerBootstrap;
import io.netty.buffer.BufType;
import io.netty.channel.Channel; import io.netty.channel.Channel;
import io.netty.channel.ChannelHandlerContext; import io.netty.channel.ChannelHandlerContext;
import io.netty.channel.ChannelInboundMessageHandlerAdapter; import io.netty.channel.ChannelInboundMessageHandlerAdapter;
@ -66,7 +67,7 @@ public class SocketStringEchoTest extends AbstractSocketTest {
public void initChannel(SocketChannel sch) throws Exception { public void initChannel(SocketChannel sch) throws Exception {
sch.pipeline().addLast("framer", new DelimiterBasedFrameDecoder(512, Delimiters.lineDelimiter())); sch.pipeline().addLast("framer", new DelimiterBasedFrameDecoder(512, Delimiters.lineDelimiter()));
sch.pipeline().addLast("decoder", new StringDecoder(CharsetUtil.ISO_8859_1)); sch.pipeline().addLast("decoder", new StringDecoder(CharsetUtil.ISO_8859_1));
sch.pipeline().addBefore("decoder", "encoder", new StringEncoder(CharsetUtil.ISO_8859_1)); sch.pipeline().addBefore("decoder", "encoder", new StringEncoder(BufType.BYTE, CharsetUtil.ISO_8859_1));
sch.pipeline().addAfter("decoder", "handler", sh); sch.pipeline().addAfter("decoder", "handler", sh);
} }
}); });
@ -76,7 +77,7 @@ public class SocketStringEchoTest extends AbstractSocketTest {
public void initChannel(SocketChannel sch) throws Exception { public void initChannel(SocketChannel sch) throws Exception {
sch.pipeline().addLast("framer", new DelimiterBasedFrameDecoder(512, Delimiters.lineDelimiter())); sch.pipeline().addLast("framer", new DelimiterBasedFrameDecoder(512, Delimiters.lineDelimiter()));
sch.pipeline().addLast("decoder", new StringDecoder(CharsetUtil.ISO_8859_1)); sch.pipeline().addLast("decoder", new StringDecoder(CharsetUtil.ISO_8859_1));
sch.pipeline().addBefore("decoder", "encoder", new StringEncoder(CharsetUtil.ISO_8859_1)); sch.pipeline().addBefore("decoder", "encoder", new StringEncoder(BufType.BYTE, CharsetUtil.ISO_8859_1));
sch.pipeline().addAfter("decoder", "handler", ch); sch.pipeline().addAfter("decoder", "handler", ch);
} }
}); });

View File

@ -1274,23 +1274,32 @@ final class DefaultChannelPipeline implements ChannelPipeline {
@Override @Override
public void flush(ChannelHandlerContext ctx, ChannelPromise promise) throws Exception { public void flush(ChannelHandlerContext ctx, ChannelPromise promise) throws Exception {
int msgSinkSize = msgSink.size(); int discardedMessages = 0;
if (msgSinkSize != 0) { MessageBuf<Object> in = msgSink;
MessageBuf<Object> in = msgSink; for (;;) {
for (;;) { Object m = in.poll();
Object m = in.poll(); if (m == null) {
if (m == null) { break;
break;
}
if (m instanceof Freeable) {
((Freeable) m).free();
}
} }
if (m instanceof ByteBuf) {
ByteBuf src = (ByteBuf) m;
byteSink.writeBytes(src, src.readerIndex(), src.readableBytes());
} else {
discardedMessages ++;
}
if (m instanceof Freeable) {
((Freeable) m).free();
}
}
if (discardedMessages != 0) {
logger.warn( logger.warn(
"Discarded {} outbound message(s) that reached at the end of the pipeline. " + "Discarded {} outbound message(s) that reached at the end of the pipeline. " +
"Please check your pipeline configuration.", msgSinkSize); "Please check your pipeline configuration.", discardedMessages);
} }
unsafe.flush(promise); unsafe.flush(promise);
} }
} }