Finish porting the codec package to the new API

- Removed deprecated classes
- Changed type parameter of StreamToMessageDecoder and
  MessageToMessageDecoder for more flexibility
- Made all tests in the codec package pass
This commit is contained in:
Trustin Lee 2012-05-17 16:38:25 +09:00
parent 1bf0dfe64a
commit 5344dc242c
17 changed files with 122 additions and 301 deletions

View File

@ -55,7 +55,7 @@ import io.netty.channel.ChannelInboundHandlerContext;
* </pre> * </pre>
* @apiviz.uses io.netty.handler.codec.frame.Delimiters - - useful * @apiviz.uses io.netty.handler.codec.frame.Delimiters - - useful
*/ */
public class DelimiterBasedFrameDecoder extends StreamToMessageDecoder<ChannelBuffer> { public class DelimiterBasedFrameDecoder extends StreamToMessageDecoder<Object> {
private final ChannelBuffer[] delimiters; private final ChannelBuffer[] delimiters;
private final int maxFrameLength; private final int maxFrameLength;
@ -187,7 +187,7 @@ public class DelimiterBasedFrameDecoder extends StreamToMessageDecoder<ChannelBu
} }
@Override @Override
public ChannelBuffer decode(ChannelInboundHandlerContext<Byte> ctx, ChannelBuffer buffer) throws Exception { public Object decode(ChannelInboundHandlerContext<Byte> ctx, ChannelBuffer buffer) throws Exception {
// Try all delimiters and choose the delimiter which yields the shortest frame. // Try all delimiters and choose the delimiter which yields the shortest frame.
int minFrameLength = Integer.MAX_VALUE; int minFrameLength = Integer.MAX_VALUE;
ChannelBuffer minDelim = null; ChannelBuffer minDelim = null;

View File

@ -37,7 +37,7 @@ import io.netty.channel.ChannelInboundHandlerContext;
* +-----+-----+-----+ * +-----+-----+-----+
* </pre> * </pre>
*/ */
public class FixedLengthFrameDecoder extends StreamToMessageDecoder<ChannelBuffer> { public class FixedLengthFrameDecoder extends StreamToMessageDecoder<Object> {
private final int frameLength; private final int frameLength;
private final boolean allocateFullBuffer; private final boolean allocateFullBuffer;
@ -75,7 +75,7 @@ public class FixedLengthFrameDecoder extends StreamToMessageDecoder<ChannelBuffe
} }
@Override @Override
public ChannelBuffer decode(ChannelInboundHandlerContext<Byte> ctx, ChannelBuffer in) throws Exception { public Object decode(ChannelInboundHandlerContext<Byte> ctx, ChannelBuffer in) throws Exception {
if (in.readableBytes() < frameLength) { if (in.readableBytes() < frameLength) {
return null; return null;
} else { } else {

View File

@ -180,7 +180,7 @@ import io.netty.handler.codec.serialization.ObjectDecoder;
* </pre> * </pre>
* @see LengthFieldPrepender * @see LengthFieldPrepender
*/ */
public class LengthFieldBasedFrameDecoder extends StreamToMessageDecoder<ChannelBuffer> { public class LengthFieldBasedFrameDecoder extends StreamToMessageDecoder<Object> {
private final int maxFrameLength; private final int maxFrameLength;
private final int lengthFieldOffset; private final int lengthFieldOffset;
@ -308,7 +308,7 @@ public class LengthFieldBasedFrameDecoder extends StreamToMessageDecoder<Channel
} }
@Override @Override
public ChannelBuffer decode(ChannelInboundHandlerContext<Byte> ctx, ChannelBuffer in) throws Exception { public Object decode(ChannelInboundHandlerContext<Byte> ctx, ChannelBuffer in) throws Exception {
if (discardingTooLongFrame) { if (discardingTooLongFrame) {
long bytesToDiscard = this.bytesToDiscard; long bytesToDiscard = this.bytesToDiscard;
int localBytesToDiscard = (int) Math.min(bytesToDiscard, in.readableBytes()); int localBytesToDiscard = (int) Math.min(bytesToDiscard, in.readableBytes());

View File

@ -45,7 +45,22 @@ public abstract class StreamToMessageDecoder<O> extends ChannelInboundHandlerAda
boolean decoded = false; boolean decoded = false;
for (;;) { for (;;) {
try { try {
if (unfoldAndAdd(ctx, ctx.nextIn(), decode(ctx, in))) { int oldInputLength = in.readableBytes();
O o = decode(ctx, in);
if (o == null) {
if (oldInputLength == in.readableBytes()) {
break;
} else {
continue;
}
} else {
if (oldInputLength == in.readableBytes()) {
throw new IllegalStateException(
"decode() did not read anything but decoded a message.");
}
}
if (unfoldAndAdd(ctx, ctx.nextIn(), o)) {
decoded = true; decoded = true;
} else { } else {
break; break;

View File

@ -18,7 +18,6 @@ package io.netty.handler.codec.embedder;
import io.netty.channel.Channel; import io.netty.channel.Channel;
import io.netty.channel.ChannelBufferHolder; import io.netty.channel.ChannelBufferHolder;
import io.netty.channel.ChannelBufferHolders; import io.netty.channel.ChannelBufferHolders;
import io.netty.channel.ChannelException;
import io.netty.channel.ChannelHandler; import io.netty.channel.ChannelHandler;
import io.netty.channel.ChannelInboundHandlerAdapter; import io.netty.channel.ChannelInboundHandlerAdapter;
import io.netty.channel.ChannelInboundHandlerContext; import io.netty.channel.ChannelInboundHandlerContext;
@ -84,14 +83,11 @@ abstract class AbstractCodecEmbedder<E> implements CodecEmbedder<E> {
@SuppressWarnings("unchecked") @SuppressWarnings("unchecked")
private E product(Object p) { private E product(Object p) {
if (p instanceof CodecEmbedderException) {
throw (CodecEmbedderException) p;
}
if (p instanceof Throwable) { if (p instanceof Throwable) {
if (p instanceof RuntimeException) { throw new CodecEmbedderException((Throwable) p);
throw (RuntimeException) p;
}
if (p instanceof Error) {
throw (Error) p;
}
throw new ChannelException((Throwable) p);
} }
return (E) p; return (E) p;
} }
@ -157,6 +153,11 @@ abstract class AbstractCodecEmbedder<E> implements CodecEmbedder<E> {
return ChannelBufferHolders.messageBuffer(productQueue); return ChannelBufferHolders.messageBuffer(productQueue);
} }
@Override
public void inboundBufferUpdated(ChannelInboundHandlerContext<Object> ctx) throws Exception {
// NOOP
}
@Override @Override
public void exceptionCaught(ChannelInboundHandlerContext<Object> ctx, Throwable cause) throws Exception { public void exceptionCaught(ChannelInboundHandlerContext<Object> ctx, Throwable cause) throws Exception {
productQueue.add(cause); productQueue.add(cause);

View File

@ -37,7 +37,7 @@ import com.google.protobuf.CodedInputStream;
* *
* @see com.google.protobuf.CodedInputStream * @see com.google.protobuf.CodedInputStream
*/ */
public class ProtobufVarint32FrameDecoder extends StreamToMessageDecoder<ChannelBuffer> { public class ProtobufVarint32FrameDecoder extends StreamToMessageDecoder<Object> {
// TODO maxFrameLength + safe skip + fail-fast option // TODO maxFrameLength + safe skip + fail-fast option
// (just like LengthFieldBasedFrameDecoder) // (just like LengthFieldBasedFrameDecoder)
@ -49,7 +49,7 @@ public class ProtobufVarint32FrameDecoder extends StreamToMessageDecoder<Channel
} }
@Override @Override
public ChannelBuffer decode(ChannelInboundHandlerContext<Byte> ctx, ChannelBuffer in) throws Exception { public Object decode(ChannelInboundHandlerContext<Byte> ctx, ChannelBuffer in) throws Exception {
in.markReaderIndex(); in.markReaderIndex();
final byte[] buf = new byte[5]; final byte[] buf = new byte[5];
for (int i = 0; i < buf.length; i ++) { for (int i = 0; i < buf.length; i ++) {

View File

@ -1,110 +0,0 @@
/*
* Copyright 2011 The Netty Project
*
* The Netty Project licenses this file to you under the Apache License,
* version 2.0 (the "License"); you may not use this file except in compliance
* with the License. You may obtain a copy of the License at:
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
* WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
* License for the specific language governing permissions and limitations
* under the License.
*/
package io.netty.handler.codec.serialization;
import java.io.InputStream;
import java.io.ObjectInputStream;
import java.io.ObjectOutputStream;
import java.io.ObjectStreamConstants;
import io.netty.buffer.ChannelBuffer;
import io.netty.buffer.ChannelBufferInputStream;
import io.netty.channel.Channel;
import io.netty.channel.ChannelHandlerContext;
import io.netty.handler.codec.ReplayingDecoder;
/**
* A decoder which deserializes the received {@link ChannelBuffer}s into Java
* objects (interoperability version).
* <p>
* This decoder is interoperable with the standard Java object
* streams such as {@link ObjectInputStream} and {@link ObjectOutputStream}.
* <p>
* However, this decoder might perform worse than {@link ObjectDecoder} if
* the serialized object is big and complex. Also, it does not limit the
* maximum size of the object, and consequently your application might face
* the risk of <a href="http://en.wikipedia.org/wiki/DoS">DoS attack</a>.
* Please use {@link ObjectEncoder} and {@link ObjectDecoder} if you are not
* required to keep the interoperability with the standard object streams.
*
* @deprecated This decoder has a known critical bug which fails to decode and
* raises a random exception in some circumstances. Avoid to use
* it whenever you can. The only workaround is to replace
* {@link CompatibleObjectEncoder}, {@link CompatibleObjectDecoder},
* {@link ObjectInputStream}, and {@link ObjectOutputStream} with
* {@link ObjectEncoder}, {@link ObjectDecoder},
* {@link ObjectEncoderOutputStream}, and
* {@link ObjectDecoderInputStream} respectively. This workaround
* requires both a client and a server to be modified.
*/
@Deprecated
public class CompatibleObjectDecoder extends ReplayingDecoder<CompatibleObjectDecoderState> {
private final SwitchableInputStream bin = new SwitchableInputStream();
private ObjectInputStream oin;
/**
* Creates a new decoder.
*/
public CompatibleObjectDecoder() {
super(CompatibleObjectDecoderState.READ_HEADER);
}
/**
* Creates a new {@link ObjectInputStream} which wraps the specified
* {@link InputStream}. Override this method to use a subclass of the
* {@link ObjectInputStream}.
*/
protected ObjectInputStream newObjectInputStream(InputStream in) throws Exception {
return new ObjectInputStream(in);
}
@Override
protected Object decode(
ChannelHandlerContext ctx, Channel channel, ChannelBuffer buffer, CompatibleObjectDecoderState state) throws Exception {
bin.switchStream(new ChannelBufferInputStream(buffer));
switch (state) {
case READ_HEADER:
oin = newObjectInputStream(bin);
checkpoint(CompatibleObjectDecoderState.READ_OBJECT);
case READ_OBJECT:
return oin.readObject();
default:
throw new IllegalStateException("Unknown state: " + state);
}
}
@Override
protected Object decodeLast(ChannelHandlerContext ctx, Channel channel,
ChannelBuffer buffer, CompatibleObjectDecoderState state)
throws Exception {
switch (buffer.readableBytes()) {
case 0:
return null;
case 1:
// Ignore the last TC_RESET
if (buffer.getByte(buffer.readerIndex()) == ObjectStreamConstants.TC_RESET) {
buffer.skipBytes(1);
oin.close();
return null;
}
}
Object decoded = decode(ctx, channel, buffer, state);
oin.close();
return decoded;
}
}

View File

@ -1,21 +0,0 @@
/*
* Copyright 2011 The Netty Project
*
* The Netty Project licenses this file to you under the Apache License,
* version 2.0 (the "License"); you may not use this file except in compliance
* with the License. You may obtain a copy of the License at:
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
* WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
* License for the specific language governing permissions and limitations
* under the License.
*/
package io.netty.handler.codec.serialization;
enum CompatibleObjectDecoderState {
READ_HEADER,
READ_OBJECT,
}

View File

@ -15,18 +15,16 @@
*/ */
package io.netty.handler.codec.serialization; package io.netty.handler.codec.serialization;
import io.netty.buffer.ChannelBuffer;
import io.netty.buffer.ChannelBufferOutputStream;
import io.netty.channel.ChannelOutboundHandlerContext;
import io.netty.handler.codec.MessageToStreamEncoder;
import io.netty.util.Attribute;
import io.netty.util.AttributeKey;
import java.io.ObjectInputStream; import java.io.ObjectInputStream;
import java.io.ObjectOutputStream; import java.io.ObjectOutputStream;
import java.io.OutputStream; import java.io.OutputStream;
import java.util.concurrent.atomic.AtomicReference;
import io.netty.buffer.ChannelBuffer;
import io.netty.buffer.ChannelBufferFactory;
import io.netty.buffer.ChannelBufferOutputStream;
import io.netty.buffer.ChannelBuffers;
import io.netty.channel.Channel;
import io.netty.channel.ChannelHandlerContext;
import io.netty.handler.codec.oneone.OneToOneEncoder;
/** /**
* An encoder which serializes a Java object into a {@link ChannelBuffer} * An encoder which serializes a Java object into a {@link ChannelBuffer}
@ -35,12 +33,13 @@ import io.netty.handler.codec.oneone.OneToOneEncoder;
* This encoder is interoperable with the standard Java object streams such as * This encoder is interoperable with the standard Java object streams such as
* {@link ObjectInputStream} and {@link ObjectOutputStream}. * {@link ObjectInputStream} and {@link ObjectOutputStream}.
*/ */
public class CompatibleObjectEncoder extends OneToOneEncoder { public class CompatibleObjectEncoder extends MessageToStreamEncoder<Object> {
private static final AttributeKey<ObjectOutputStream> OOS =
new AttributeKey<ObjectOutputStream>(
CompatibleObjectEncoder.class.getName() + ".oos", ObjectOutputStream.class);
private final AtomicReference<ChannelBuffer> buffer =
new AtomicReference<ChannelBuffer>();
private final int resetInterval; private final int resetInterval;
private volatile ObjectOutputStream oout;
private int writtenObjects; private int writtenObjects;
/** /**
@ -77,44 +76,31 @@ public class CompatibleObjectEncoder extends OneToOneEncoder {
} }
@Override @Override
protected Object encode(ChannelHandlerContext context, Channel channel, Object msg) throws Exception { public void encode(ChannelOutboundHandlerContext<Object> ctx, Object msg, ChannelBuffer out) throws Exception {
ChannelBuffer buffer = buffer(context); Attribute<ObjectOutputStream> oosAttr = ctx.attr(OOS);
ObjectOutputStream oout = this.oout; ObjectOutputStream oos = oosAttr.get();
if (oos == null) {
oos = newObjectOutputStream(new ChannelBufferOutputStream(out));
ObjectOutputStream newOos = oosAttr.setIfAbsent(oos);
if (newOos != null) {
oos = newOos;
}
}
synchronized (oos) {
if (resetInterval != 0) { if (resetInterval != 0) {
// Resetting will prevent OOM on the receiving side. // Resetting will prevent OOM on the receiving side.
writtenObjects ++; writtenObjects ++;
if (writtenObjects % resetInterval == 0) { if (writtenObjects % resetInterval == 0) {
oout.reset(); oos.reset();
// Also discard the byproduct to avoid OOM on the sending side. // Also discard the byproduct to avoid OOM on the sending side.
buffer.discardReadBytes(); out.discardReadBytes();
} }
} }
oout.writeObject(msg);
oout.flush();
return buffer.readBytes(buffer.readableBytes());
}
private ChannelBuffer buffer(ChannelHandlerContext ctx) throws Exception { oos.writeObject(msg);
ChannelBuffer buf = buffer.get(); oos.flush();
if (buf == null) {
ChannelBufferFactory factory = ctx.channel().getConfig().getBufferFactory();
buf = ChannelBuffers.dynamicBuffer(factory);
if (buffer.compareAndSet(null, buf)) {
boolean success = false;
try {
oout = newObjectOutputStream(new ChannelBufferOutputStream(buf));
success = true;
} finally {
if (!success) {
oout = null;
} }
} }
} else {
buf = buffer.get();
}
}
return buf;
}
} }

View File

@ -15,15 +15,14 @@
*/ */
package io.netty.handler.codec.serialization; package io.netty.handler.codec.serialization;
import java.io.ObjectOutputStream;
import java.io.StreamCorruptedException;
import io.netty.buffer.ChannelBuffer; import io.netty.buffer.ChannelBuffer;
import io.netty.buffer.ChannelBufferInputStream; import io.netty.buffer.ChannelBufferInputStream;
import io.netty.channel.Channel; import io.netty.channel.ChannelInboundHandlerContext;
import io.netty.channel.ChannelHandlerContext;
import io.netty.handler.codec.LengthFieldBasedFrameDecoder; import io.netty.handler.codec.LengthFieldBasedFrameDecoder;
import java.io.ObjectOutputStream;
import java.io.StreamCorruptedException;
/** /**
* A decoder which deserializes the received {@link ChannelBuffer}s into Java * A decoder which deserializes the received {@link ChannelBuffer}s into Java
* objects. * objects.
@ -39,20 +38,6 @@ public class ObjectDecoder extends LengthFieldBasedFrameDecoder {
private final ClassResolver classResolver; private final ClassResolver classResolver;
/**
* Creates a new decoder whose maximum object size is {@code 1048576}
* bytes. If the size of the received object is greater than
* {@code 1048576} bytes, a {@link StreamCorruptedException} will be
* raised.
*
* @deprecated use {@link #ObjectDecoder(ClassResolver)}
*/
@Deprecated
public ObjectDecoder() {
this(1048576);
}
/** /**
* Creates a new decoder whose maximum object size is {@code 1048576} * Creates a new decoder whose maximum object size is {@code 1048576}
* bytes. If the size of the received object is greater than * bytes. If the size of the received object is greater than
@ -65,20 +50,6 @@ public class ObjectDecoder extends LengthFieldBasedFrameDecoder {
this(1048576, classResolver); this(1048576, classResolver);
} }
/**
* Creates a new decoder with the specified maximum object size.
*
* @param maxObjectSize the maximum byte length of the serialized object.
* if the length of the received object is greater
* than this value, {@link StreamCorruptedException}
* will be raised.
* @deprecated use {@link #ObjectDecoder(int, ClassResolver)}
*/
@Deprecated
public ObjectDecoder(int maxObjectSize) {
this(maxObjectSize, ClassResolvers.weakCachingResolver(null));
}
/** /**
* Creates a new decoder with the specified maximum object size. * Creates a new decoder with the specified maximum object size.
* *
@ -94,27 +65,9 @@ public class ObjectDecoder extends LengthFieldBasedFrameDecoder {
this.classResolver = classResolver; this.classResolver = classResolver;
} }
/**
* Create a new decoder with the specified maximum object size and the {@link ClassLoader} wrapped in {@link ClassResolvers#weakCachingResolver(ClassLoader)}
*
* @param maxObjectSize the maximum byte length of the serialized object.
* if the length of the received object is greater
* than this value, {@link StreamCorruptedException}
* will be raised.
* @param classLoader the the classloader to use
* @deprecated use {@link #ObjectDecoder(int, ClassResolver)}
*/
@Deprecated
public ObjectDecoder(int maxObjectSize, ClassLoader classLoader) {
this(maxObjectSize, ClassResolvers.weakCachingResolver(classLoader));
}
@Override @Override
protected Object decode( public Object decode(ChannelInboundHandlerContext<Byte> ctx, ChannelBuffer in) throws Exception {
ChannelHandlerContext ctx, Channel channel, ChannelBuffer buffer) throws Exception { ChannelBuffer frame = (ChannelBuffer) super.decode(ctx, in);
ChannelBuffer frame = (ChannelBuffer) super.decode(ctx, channel, buffer);
if (frame == null) { if (frame == null) {
return null; return null;
} }

View File

@ -15,18 +15,15 @@
*/ */
package io.netty.handler.codec.serialization; package io.netty.handler.codec.serialization;
import static io.netty.buffer.ChannelBuffers.*; import io.netty.buffer.ChannelBuffer;
import io.netty.buffer.ChannelBufferOutputStream;
import io.netty.channel.ChannelHandler.Sharable;
import io.netty.channel.ChannelOutboundHandlerContext;
import io.netty.handler.codec.MessageToStreamEncoder;
import java.io.ObjectInputStream; import java.io.ObjectInputStream;
import java.io.ObjectOutputStream; import java.io.ObjectOutputStream;
import io.netty.buffer.ChannelBuffer;
import io.netty.buffer.ChannelBufferOutputStream;
import io.netty.channel.Channel;
import io.netty.channel.ChannelHandlerContext;
import io.netty.channel.ChannelHandler.Sharable;
import io.netty.handler.codec.oneone.OneToOneEncoder;
/** /**
* An encoder which serializes a Java object into a {@link ChannelBuffer}. * An encoder which serializes a Java object into a {@link ChannelBuffer}.
* <p> * <p>
@ -38,7 +35,7 @@ import io.netty.handler.codec.oneone.OneToOneEncoder;
* @apiviz.has io.netty.handler.codec.serialization.ObjectEncoderOutputStream - - - compatible with * @apiviz.has io.netty.handler.codec.serialization.ObjectEncoderOutputStream - - - compatible with
*/ */
@Sharable @Sharable
public class ObjectEncoder extends OneToOneEncoder { public class ObjectEncoder extends MessageToStreamEncoder<Object> {
private static final byte[] LENGTH_PLACEHOLDER = new byte[4]; private static final byte[] LENGTH_PLACEHOLDER = new byte[4];
private final int estimatedLength; private final int estimatedLength;
@ -70,18 +67,18 @@ public class ObjectEncoder extends OneToOneEncoder {
} }
@Override @Override
protected Object encode(ChannelHandlerContext ctx, Channel channel, Object msg) throws Exception { public void encode(ChannelOutboundHandlerContext<Object> ctx, Object msg, ChannelBuffer out) throws Exception {
ChannelBufferOutputStream bout = int startIdx = out.writerIndex();
new ChannelBufferOutputStream(dynamicBuffer(
estimatedLength, ctx.channel().getConfig().getBufferFactory())); ChannelBufferOutputStream bout = new ChannelBufferOutputStream(out);
bout.write(LENGTH_PLACEHOLDER); bout.write(LENGTH_PLACEHOLDER);
ObjectOutputStream oout = new CompactObjectOutputStream(bout); ObjectOutputStream oout = new CompactObjectOutputStream(bout);
oout.writeObject(msg); oout.writeObject(msg);
oout.flush(); oout.flush();
oout.close(); oout.close();
ChannelBuffer encoded = bout.buffer(); int endIdx = out.writerIndex();
encoded.setInt(0, encoded.writerIndex() - 4);
return encoded; out.setInt(startIdx, endIdx - startIdx - 4);
} }
} }

View File

@ -15,18 +15,16 @@
*/ */
package io.netty.handler.codec.string; package io.netty.handler.codec.string;
import java.nio.charset.Charset;
import io.netty.buffer.ChannelBuffer; import io.netty.buffer.ChannelBuffer;
import io.netty.channel.Channel;
import io.netty.channel.ChannelHandlerContext;
import io.netty.channel.ChannelPipeline;
import io.netty.channel.MessageEvent;
import io.netty.channel.ChannelHandler.Sharable; import io.netty.channel.ChannelHandler.Sharable;
import io.netty.channel.ChannelHandlerContext;
import io.netty.channel.ChannelInboundHandlerContext;
import io.netty.channel.ChannelPipeline;
import io.netty.handler.codec.DelimiterBasedFrameDecoder; import io.netty.handler.codec.DelimiterBasedFrameDecoder;
import io.netty.handler.codec.Delimiters; import io.netty.handler.codec.Delimiters;
import io.netty.handler.codec.FrameDecoder; import io.netty.handler.codec.MessageToMessageDecoder;
import io.netty.handler.codec.oneone.OneToOneDecoder;
import java.nio.charset.Charset;
/** /**
* Decodes a received {@link ChannelBuffer} into a {@link String}. Please * Decodes a received {@link ChannelBuffer} into a {@link String}. Please
@ -55,7 +53,7 @@ import io.netty.handler.codec.oneone.OneToOneDecoder;
* @apiviz.landmark * @apiviz.landmark
*/ */
@Sharable @Sharable
public class StringDecoder extends OneToOneDecoder { public class StringDecoder extends MessageToMessageDecoder<ChannelBuffer, String> {
// TODO Use CharsetDecoder instead. // TODO Use CharsetDecoder instead.
private final Charset charset; private final Charset charset;
@ -78,11 +76,7 @@ public class StringDecoder extends OneToOneDecoder {
} }
@Override @Override
protected Object decode( public String decode(ChannelInboundHandlerContext<ChannelBuffer> ctx, ChannelBuffer msg) throws Exception {
ChannelHandlerContext ctx, Channel channel, Object msg) throws Exception { return msg.toString(charset);
if (!(msg instanceof ChannelBuffer)) {
return msg;
}
return ((ChannelBuffer) msg).toString(charset);
} }
} }

View File

@ -15,19 +15,17 @@
*/ */
package io.netty.handler.codec.string; package io.netty.handler.codec.string;
import static io.netty.buffer.ChannelBuffers.*;
import java.nio.charset.Charset;
import io.netty.buffer.ChannelBuffer; import io.netty.buffer.ChannelBuffer;
import io.netty.channel.Channel; import io.netty.buffer.ChannelBuffers;
import io.netty.channel.ChannelHandlerContext;
import io.netty.channel.ChannelPipeline;
import io.netty.channel.MessageEvent;
import io.netty.channel.ChannelHandler.Sharable; import io.netty.channel.ChannelHandler.Sharable;
import io.netty.channel.ChannelHandlerContext;
import io.netty.channel.ChannelOutboundHandlerContext;
import io.netty.channel.ChannelPipeline;
import io.netty.handler.codec.DelimiterBasedFrameDecoder; import io.netty.handler.codec.DelimiterBasedFrameDecoder;
import io.netty.handler.codec.Delimiters; import io.netty.handler.codec.Delimiters;
import io.netty.handler.codec.oneone.OneToOneEncoder; import io.netty.handler.codec.MessageToMessageEncoder;
import java.nio.charset.Charset;
/** /**
* Encodes the requested {@link String} into a {@link ChannelBuffer}. * Encodes the requested {@link String} into a {@link ChannelBuffer}.
@ -53,7 +51,7 @@ import io.netty.handler.codec.oneone.OneToOneEncoder;
* @apiviz.landmark * @apiviz.landmark
*/ */
@Sharable @Sharable
public class StringEncoder extends OneToOneEncoder { public class StringEncoder extends MessageToMessageEncoder<String, ChannelBuffer> {
// TODO Use CharsetEncoder instead. // TODO Use CharsetEncoder instead.
private final Charset charset; private final Charset charset;
@ -76,11 +74,7 @@ public class StringEncoder extends OneToOneEncoder {
} }
@Override @Override
protected Object encode( public ChannelBuffer encode(ChannelOutboundHandlerContext<String> ctx, String msg) throws Exception {
ChannelHandlerContext ctx, Channel channel, Object msg) throws Exception { return ChannelBuffers.copiedBuffer(msg, charset);
if (!(msg instanceof String)) {
return msg;
}
return copiedBuffer((String) msg, charset);
} }
} }

View File

@ -18,6 +18,7 @@ package io.netty.handler.codec.bytes;
import static io.netty.buffer.ChannelBuffers.*; import static io.netty.buffer.ChannelBuffers.*;
import static org.hamcrest.core.Is.*; import static org.hamcrest.core.Is.*;
import static org.junit.Assert.*; import static org.junit.Assert.*;
import io.netty.handler.codec.embedder.CodecEmbedderException;
import io.netty.handler.codec.embedder.DecoderEmbedder; import io.netty.handler.codec.embedder.DecoderEmbedder;
import java.util.Random; import java.util.Random;
@ -58,8 +59,9 @@ public class ByteArrayDecoderTest {
try { try {
embedder.poll(); embedder.poll();
fail(); fail();
} catch (ClassCastException e) { } catch (CodecEmbedderException e) {
// Expected // Expected
assertTrue(e.getCause() instanceof ClassCastException);
} }
} }
} }

View File

@ -20,6 +20,7 @@ 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.ChannelBuffer; import io.netty.buffer.ChannelBuffer;
import io.netty.handler.codec.embedder.CodecEmbedderException;
import io.netty.handler.codec.embedder.EncoderEmbedder; import io.netty.handler.codec.embedder.EncoderEmbedder;
import java.util.Random; import java.util.Random;
@ -60,8 +61,9 @@ public class ByteArrayEncoderTest {
try { try {
embedder.poll(); embedder.poll();
fail(); fail();
} catch (ClassCastException e) { } catch (CodecEmbedderException e) {
// Expected // Expected
assertTrue(e.getCause() instanceof ClassCastException);
} }
} }

View File

@ -15,6 +15,7 @@
*/ */
package io.netty.handler.codec.frame; package io.netty.handler.codec.frame;
import static org.junit.Assert.*;
import io.netty.buffer.ChannelBuffer; import io.netty.buffer.ChannelBuffer;
import io.netty.buffer.ChannelBuffers; import io.netty.buffer.ChannelBuffers;
import io.netty.handler.codec.DelimiterBasedFrameDecoder; import io.netty.handler.codec.DelimiterBasedFrameDecoder;
@ -23,6 +24,7 @@ import io.netty.handler.codec.TooLongFrameException;
import io.netty.handler.codec.embedder.CodecEmbedderException; import io.netty.handler.codec.embedder.CodecEmbedderException;
import io.netty.handler.codec.embedder.DecoderEmbedder; import io.netty.handler.codec.embedder.DecoderEmbedder;
import io.netty.util.CharsetUtil; import io.netty.util.CharsetUtil;
import org.junit.Assert; import org.junit.Assert;
import org.junit.Test; import org.junit.Test;
@ -35,7 +37,8 @@ public class DelimiterBasedFrameDecoderTest {
for (int i = 0; i < 2; i ++) { for (int i = 0; i < 2; i ++) {
embedder.offer(ChannelBuffers.wrappedBuffer(new byte[] { 1, 2 })); embedder.offer(ChannelBuffers.wrappedBuffer(new byte[] { 1, 2 }));
try { try {
embedder.offer(ChannelBuffers.wrappedBuffer(new byte[] { 0 })); assertTrue(embedder.offer(ChannelBuffers.wrappedBuffer(new byte[] { 0 })));
embedder.poll();
Assert.fail(CodecEmbedderException.class.getSimpleName() + " must be raised."); Assert.fail(CodecEmbedderException.class.getSimpleName() + " must be raised.");
} catch (CodecEmbedderException e) { } catch (CodecEmbedderException e) {
Assert.assertTrue(e.getCause() instanceof TooLongFrameException); Assert.assertTrue(e.getCause() instanceof TooLongFrameException);
@ -55,7 +58,8 @@ public class DelimiterBasedFrameDecoderTest {
for (int i = 0; i < 2; i ++) { for (int i = 0; i < 2; i ++) {
try { try {
embedder.offer(ChannelBuffers.wrappedBuffer(new byte[] { 1, 2 })); assertTrue(embedder.offer(ChannelBuffers.wrappedBuffer(new byte[] { 1, 2 })));
embedder.poll();
Assert.fail(CodecEmbedderException.class.getSimpleName() + " must be raised."); Assert.fail(CodecEmbedderException.class.getSimpleName() + " must be raised.");
} catch (CodecEmbedderException e) { } catch (CodecEmbedderException e) {
Assert.assertTrue(e.getCause() instanceof TooLongFrameException); Assert.assertTrue(e.getCause() instanceof TooLongFrameException);

View File

@ -15,6 +15,7 @@
*/ */
package io.netty.handler.codec.frame; package io.netty.handler.codec.frame;
import static org.junit.Assert.*;
import io.netty.buffer.ChannelBuffer; import io.netty.buffer.ChannelBuffer;
import io.netty.buffer.ChannelBuffers; import io.netty.buffer.ChannelBuffers;
import io.netty.handler.codec.LengthFieldBasedFrameDecoder; import io.netty.handler.codec.LengthFieldBasedFrameDecoder;
@ -22,6 +23,7 @@ import io.netty.handler.codec.TooLongFrameException;
import io.netty.handler.codec.embedder.CodecEmbedderException; import io.netty.handler.codec.embedder.CodecEmbedderException;
import io.netty.handler.codec.embedder.DecoderEmbedder; import io.netty.handler.codec.embedder.DecoderEmbedder;
import io.netty.util.CharsetUtil; import io.netty.util.CharsetUtil;
import org.junit.Assert; import org.junit.Assert;
import org.junit.Test; import org.junit.Test;
@ -32,9 +34,10 @@ public class LengthFieldBasedFrameDecoderTest {
new LengthFieldBasedFrameDecoder(5, 0, 4, 0, 4, false)); new LengthFieldBasedFrameDecoder(5, 0, 4, 0, 4, false));
for (int i = 0; i < 2; i ++) { for (int i = 0; i < 2; i ++) {
embedder.offer(ChannelBuffers.wrappedBuffer(new byte[] { 0, 0, 0, 2 })); assertFalse(embedder.offer(ChannelBuffers.wrappedBuffer(new byte[] { 0, 0, 0, 2 })));
try { try {
embedder.offer(ChannelBuffers.wrappedBuffer(new byte[] { 0, 0 })); assertTrue(embedder.offer(ChannelBuffers.wrappedBuffer(new byte[] { 0, 0 })));
embedder.poll();
Assert.fail(CodecEmbedderException.class.getSimpleName() + " must be raised."); Assert.fail(CodecEmbedderException.class.getSimpleName() + " must be raised.");
} catch (CodecEmbedderException e) { } catch (CodecEmbedderException e) {
Assert.assertTrue(e.getCause() instanceof TooLongFrameException); Assert.assertTrue(e.getCause() instanceof TooLongFrameException);
@ -54,7 +57,8 @@ public class LengthFieldBasedFrameDecoderTest {
for (int i = 0; i < 2; i ++) { for (int i = 0; i < 2; i ++) {
try { try {
embedder.offer(ChannelBuffers.wrappedBuffer(new byte[] { 0, 0, 0, 2 })); assertTrue(embedder.offer(ChannelBuffers.wrappedBuffer(new byte[] { 0, 0, 0, 2 })));
embedder.poll();
Assert.fail(CodecEmbedderException.class.getSimpleName() + " must be raised."); Assert.fail(CodecEmbedderException.class.getSimpleName() + " must be raised.");
} catch (CodecEmbedderException e) { } catch (CodecEmbedderException e) {
Assert.assertTrue(e.getCause() instanceof TooLongFrameException); Assert.assertTrue(e.getCause() instanceof TooLongFrameException);