Improve error handling in ByteToMessageDecoder when expand fails (#9822)
Motivation: The buffer which the decoder allocates for the expansion can be leaked if there is a subsequent issue writing to it. Modifications: The error handling has been improved so that the new buffer always is released on failure in the expand. Result: The decoder will not leak in this scenario any more. Fixes: https://github.com/netty/netty/issues/9812
This commit is contained in:
parent
0a252b48b1
commit
585ed4d08f
@ -25,14 +25,14 @@ import io.netty.buffer.Unpooled;
|
|||||||
import io.netty.channel.ChannelHandlerAdapter;
|
import io.netty.channel.ChannelHandlerAdapter;
|
||||||
import io.netty.channel.ChannelConfig;
|
import io.netty.channel.ChannelConfig;
|
||||||
import io.netty.channel.ChannelHandlerContext;
|
import io.netty.channel.ChannelHandlerContext;
|
||||||
import io.netty.channel.ChannelInboundHandler;
|
import io.netty.channel.ChannelHandler;
|
||||||
import io.netty.channel.socket.ChannelInputShutdownEvent;
|
import io.netty.channel.socket.ChannelInputShutdownEvent;
|
||||||
import io.netty.util.internal.StringUtil;
|
import io.netty.util.internal.StringUtil;
|
||||||
|
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* {@link ChannelInboundHandler} which decodes bytes in a stream-like fashion from one {@link ByteBuf} to an
|
* {@link ChannelHandler} which decodes bytes in a stream-like fashion from one {@link ByteBuf} to an
|
||||||
* other Message type.
|
* other Message type.
|
||||||
*
|
*
|
||||||
* For example here is an implementation which reads all readable bytes from
|
* For example here is an implementation which reads all readable bytes from
|
||||||
@ -72,16 +72,15 @@ import java.util.List;
|
|||||||
* is not released or added to the <tt>out</tt> {@link List}. Use derived buffers like {@link ByteBuf#readSlice(int)}
|
* is not released or added to the <tt>out</tt> {@link List}. Use derived buffers like {@link ByteBuf#readSlice(int)}
|
||||||
* to avoid leaking memory.
|
* to avoid leaking memory.
|
||||||
*/
|
*/
|
||||||
public abstract class ByteToMessageDecoder extends ChannelHandlerAdapter implements ChannelInboundHandler {
|
public abstract class ByteToMessageDecoder extends ChannelHandlerAdapter {
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Cumulate {@link ByteBuf}s by merge them into one {@link ByteBuf}'s, using memory copies.
|
* Cumulate {@link ByteBuf}s by merge them into one {@link ByteBuf}'s, using memory copies.
|
||||||
*/
|
*/
|
||||||
public static final Cumulator MERGE_CUMULATOR = (alloc, cumulation, in) -> {
|
public static final Cumulator MERGE_CUMULATOR = (alloc, cumulation, in) -> {
|
||||||
try {
|
try {
|
||||||
final ByteBuf buffer;
|
|
||||||
if (cumulation.writerIndex() > cumulation.maxCapacity() - in.readableBytes()
|
if (cumulation.writerIndex() > cumulation.maxCapacity() - in.readableBytes()
|
||||||
|| cumulation.refCnt() > 1 || cumulation.isReadOnly()) {
|
|| cumulation.refCnt() > 1 || cumulation.isReadOnly()) {
|
||||||
// Expand cumulation (by replace it) when either there is not more room in the buffer
|
// Expand cumulation (by replace it) when either there is not more room in the buffer
|
||||||
// or if the refCnt is greater then 1 which may happen when the user use slice().retain() or
|
// or if the refCnt is greater then 1 which may happen when the user use slice().retain() or
|
||||||
// duplicate().retain() or if its read-only.
|
// duplicate().retain() or if its read-only.
|
||||||
@ -89,12 +88,11 @@ public abstract class ByteToMessageDecoder extends ChannelHandlerAdapter impleme
|
|||||||
// See:
|
// See:
|
||||||
// - https://github.com/netty/netty/issues/2327
|
// - https://github.com/netty/netty/issues/2327
|
||||||
// - https://github.com/netty/netty/issues/1764
|
// - https://github.com/netty/netty/issues/1764
|
||||||
buffer = expandCumulation(alloc, cumulation, in.readableBytes());
|
cumulation = expandCumulation(alloc, cumulation, in);
|
||||||
} else {
|
} else {
|
||||||
buffer = cumulation;
|
cumulation.writeBytes(in);
|
||||||
}
|
}
|
||||||
buffer.writeBytes(in);
|
return cumulation;
|
||||||
return buffer;
|
|
||||||
} finally {
|
} finally {
|
||||||
// We must release in in all cases as otherwise it may produce a leak if writeBytes(...) throw
|
// We must release in in all cases as otherwise it may produce a leak if writeBytes(...) throw
|
||||||
// for whatever release (for example because of OutOfMemoryError)
|
// for whatever release (for example because of OutOfMemoryError)
|
||||||
@ -117,8 +115,7 @@ public abstract class ByteToMessageDecoder extends ChannelHandlerAdapter impleme
|
|||||||
// See:
|
// See:
|
||||||
// - https://github.com/netty/netty/issues/2327
|
// - https://github.com/netty/netty/issues/2327
|
||||||
// - https://github.com/netty/netty/issues/1764
|
// - https://github.com/netty/netty/issues/1764
|
||||||
buffer = expandCumulation(alloc, cumulation, in.readableBytes());
|
buffer = expandCumulation(alloc, cumulation, in);
|
||||||
buffer.writeBytes(in);
|
|
||||||
} else {
|
} else {
|
||||||
CompositeByteBuf composite;
|
CompositeByteBuf composite;
|
||||||
if (cumulation instanceof CompositeByteBuf) {
|
if (cumulation instanceof CompositeByteBuf) {
|
||||||
@ -521,12 +518,17 @@ public abstract class ByteToMessageDecoder extends ChannelHandlerAdapter impleme
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
static ByteBuf expandCumulation(ByteBufAllocator alloc, ByteBuf cumulation, int readable) {
|
static ByteBuf expandCumulation(ByteBufAllocator alloc, ByteBuf oldCumulation, ByteBuf in) {
|
||||||
ByteBuf oldCumulation = cumulation;
|
ByteBuf newCumulation = alloc.buffer(oldCumulation.readableBytes() + in.readableBytes());
|
||||||
cumulation = alloc.buffer(oldCumulation.readableBytes() + readable);
|
ByteBuf toRelease = newCumulation;
|
||||||
cumulation.writeBytes(oldCumulation);
|
try {
|
||||||
oldCumulation.release();
|
newCumulation.writeBytes(oldCumulation);
|
||||||
return cumulation;
|
newCumulation.writeBytes(in);
|
||||||
|
toRelease = oldCumulation;
|
||||||
|
return newCumulation;
|
||||||
|
} finally {
|
||||||
|
toRelease.release();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -15,7 +15,9 @@
|
|||||||
*/
|
*/
|
||||||
package io.netty.handler.codec;
|
package io.netty.handler.codec;
|
||||||
|
|
||||||
|
import io.netty.buffer.AbstractByteBufAllocator;
|
||||||
import io.netty.buffer.ByteBuf;
|
import io.netty.buffer.ByteBuf;
|
||||||
|
import io.netty.buffer.ByteBufAllocator;
|
||||||
import io.netty.buffer.CompositeByteBuf;
|
import io.netty.buffer.CompositeByteBuf;
|
||||||
import io.netty.buffer.Unpooled;
|
import io.netty.buffer.Unpooled;
|
||||||
import io.netty.buffer.UnpooledByteBufAllocator;
|
import io.netty.buffer.UnpooledByteBufAllocator;
|
||||||
@ -32,7 +34,12 @@ import java.util.concurrent.LinkedBlockingDeque;
|
|||||||
import java.util.concurrent.ThreadLocalRandom;
|
import java.util.concurrent.ThreadLocalRandom;
|
||||||
|
|
||||||
import static io.netty.buffer.Unpooled.wrappedBuffer;
|
import static io.netty.buffer.Unpooled.wrappedBuffer;
|
||||||
import static org.junit.Assert.*;
|
import static org.junit.Assert.assertEquals;
|
||||||
|
import static org.junit.Assert.assertFalse;
|
||||||
|
import static org.junit.Assert.assertNull;
|
||||||
|
import static org.junit.Assert.assertSame;
|
||||||
|
import static org.junit.Assert.assertTrue;
|
||||||
|
import static org.junit.Assert.fail;
|
||||||
|
|
||||||
public class ByteToMessageDecoderTest {
|
public class ByteToMessageDecoderTest {
|
||||||
|
|
||||||
@ -308,23 +315,94 @@ public class ByteToMessageDecoderTest {
|
|||||||
assertFalse(channel.finish());
|
assertFalse(channel.finish());
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
static class WriteFailingByteBuf extends UnpooledHeapByteBuf {
|
||||||
public void releaseWhenMergeCumulateThrows() {
|
private final Error error = new Error();
|
||||||
final Error error = new Error();
|
private int untilFailure;
|
||||||
|
|
||||||
ByteBuf cumulation = new UnpooledHeapByteBuf(UnpooledByteBufAllocator.DEFAULT, 0, 64) {
|
WriteFailingByteBuf(int untilFailure, int capacity) {
|
||||||
@Override
|
super(UnpooledByteBufAllocator.DEFAULT, capacity, capacity);
|
||||||
public ByteBuf writeBytes(ByteBuf src) {
|
this.untilFailure = untilFailure;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public ByteBuf writeBytes(ByteBuf src) {
|
||||||
|
if (--untilFailure <= 0) {
|
||||||
throw error;
|
throw error;
|
||||||
}
|
}
|
||||||
};
|
return super.writeBytes(src);
|
||||||
|
}
|
||||||
|
|
||||||
|
Error writeError() {
|
||||||
|
return error;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void releaseWhenMergeCumulateThrows() {
|
||||||
|
WriteFailingByteBuf oldCumulation = new WriteFailingByteBuf(1, 64);
|
||||||
ByteBuf in = Unpooled.buffer().writeZero(12);
|
ByteBuf in = Unpooled.buffer().writeZero(12);
|
||||||
|
|
||||||
|
Throwable thrown = null;
|
||||||
try {
|
try {
|
||||||
ByteToMessageDecoder.MERGE_CUMULATOR.cumulate(UnpooledByteBufAllocator.DEFAULT, cumulation, in);
|
ByteToMessageDecoder.MERGE_CUMULATOR.cumulate(UnpooledByteBufAllocator.DEFAULT, oldCumulation, in);
|
||||||
fail();
|
} catch (Throwable t) {
|
||||||
} catch (Error expected) {
|
thrown = t;
|
||||||
assertSame(error, expected);
|
}
|
||||||
assertEquals(0, in.refCnt());
|
|
||||||
|
assertSame(oldCumulation.writeError(), thrown);
|
||||||
|
assertEquals(0, in.refCnt());
|
||||||
|
assertEquals(1, oldCumulation.refCnt());
|
||||||
|
oldCumulation.release();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void releaseWhenMergeCumulateThrowsInExpand() {
|
||||||
|
releaseWhenMergeCumulateThrowsInExpand(1, true);
|
||||||
|
releaseWhenMergeCumulateThrowsInExpand(2, true);
|
||||||
|
releaseWhenMergeCumulateThrowsInExpand(3, false); // sentinel test case
|
||||||
|
}
|
||||||
|
|
||||||
|
private void releaseWhenMergeCumulateThrowsInExpand(int untilFailure, boolean shouldFail) {
|
||||||
|
ByteBuf oldCumulation = UnpooledByteBufAllocator.DEFAULT.heapBuffer(8, 8);
|
||||||
|
final WriteFailingByteBuf newCumulation = new WriteFailingByteBuf(untilFailure, 16);
|
||||||
|
|
||||||
|
ByteBufAllocator allocator = new AbstractByteBufAllocator(false) {
|
||||||
|
@Override
|
||||||
|
public boolean isDirectBufferPooled() {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected ByteBuf newHeapBuffer(int initialCapacity, int maxCapacity) {
|
||||||
|
return newCumulation;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected ByteBuf newDirectBuffer(int initialCapacity, int maxCapacity) {
|
||||||
|
throw new UnsupportedOperationException();
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
ByteBuf in = Unpooled.buffer().writeZero(12);
|
||||||
|
Throwable thrown = null;
|
||||||
|
try {
|
||||||
|
ByteToMessageDecoder.MERGE_CUMULATOR.cumulate(allocator, oldCumulation, in);
|
||||||
|
} catch (Throwable t) {
|
||||||
|
thrown = t;
|
||||||
|
}
|
||||||
|
|
||||||
|
assertEquals(0, in.refCnt());
|
||||||
|
|
||||||
|
if (shouldFail) {
|
||||||
|
assertSame(newCumulation.writeError(), thrown);
|
||||||
|
assertEquals(1, oldCumulation.refCnt());
|
||||||
|
oldCumulation.release();
|
||||||
|
assertEquals(0, newCumulation.refCnt());
|
||||||
|
} else {
|
||||||
|
assertNull(thrown);
|
||||||
|
assertEquals(0, oldCumulation.refCnt());
|
||||||
|
assertEquals(1, newCumulation.refCnt());
|
||||||
|
newCumulation.release();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -345,6 +423,7 @@ public class ByteToMessageDecoderTest {
|
|||||||
} catch (Error expected) {
|
} catch (Error expected) {
|
||||||
assertSame(error, expected);
|
assertSame(error, expected);
|
||||||
assertEquals(0, in.refCnt());
|
assertEquals(0, in.refCnt());
|
||||||
|
cumulation.release();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Loading…
x
Reference in New Issue
Block a user