[#1876] Make use of proper state machine in WebSocket08FrameDecoder for performance reasons
This commit is contained in:
parent
449ce0dd7a
commit
60acd54c7e
@ -22,6 +22,8 @@ import io.netty.handler.codec.TooLongFrameException;
|
|||||||
|
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
|
|
||||||
|
import static io.netty.buffer.ByteBufUtil.readBytes;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Decodes {@link ByteBuf}s into {@link WebSocketFrame}s.
|
* Decodes {@link ByteBuf}s into {@link WebSocketFrame}s.
|
||||||
* <p>
|
* <p>
|
||||||
@ -91,8 +93,7 @@ public class WebSocket00FrameDecoder extends ReplayingDecoder<Void> implements W
|
|||||||
receivedClosingHandshake = true;
|
receivedClosingHandshake = true;
|
||||||
return new CloseWebSocketFrame();
|
return new CloseWebSocketFrame();
|
||||||
}
|
}
|
||||||
ByteBuf payload = ctx.alloc().buffer((int) frameSize);
|
ByteBuf payload = readBytes(ctx.alloc(), buffer, (int) frameSize);
|
||||||
buffer.readBytes(payload);
|
|
||||||
return new BinaryWebSocketFrame(payload);
|
return new BinaryWebSocketFrame(payload);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -116,12 +117,12 @@ public class WebSocket00FrameDecoder extends ReplayingDecoder<Void> implements W
|
|||||||
throw new TooLongFrameException();
|
throw new TooLongFrameException();
|
||||||
}
|
}
|
||||||
|
|
||||||
ByteBuf binaryData = ctx.alloc().buffer(frameSize);
|
ByteBuf binaryData = readBytes(ctx.alloc(), buffer, frameSize);
|
||||||
buffer.readBytes(binaryData);
|
|
||||||
buffer.skipBytes(1);
|
buffer.skipBytes(1);
|
||||||
|
|
||||||
int ffDelimPos = binaryData.indexOf(binaryData.readerIndex(), binaryData.writerIndex(), (byte) 0xFF);
|
int ffDelimPos = binaryData.indexOf(binaryData.readerIndex(), binaryData.writerIndex(), (byte) 0xFF);
|
||||||
if (ffDelimPos >= 0) {
|
if (ffDelimPos >= 0) {
|
||||||
|
binaryData.release();
|
||||||
throw new IllegalArgumentException("a text frame should not contain 0xFF.");
|
throw new IllegalArgumentException("a text frame should not contain 0xFF.");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -57,21 +57,32 @@ import io.netty.buffer.ByteBuf;
|
|||||||
import io.netty.buffer.Unpooled;
|
import io.netty.buffer.Unpooled;
|
||||||
import io.netty.channel.ChannelFutureListener;
|
import io.netty.channel.ChannelFutureListener;
|
||||||
import io.netty.channel.ChannelHandlerContext;
|
import io.netty.channel.ChannelHandlerContext;
|
||||||
|
import io.netty.handler.codec.ByteToMessageDecoder;
|
||||||
import io.netty.handler.codec.CorruptedFrameException;
|
import io.netty.handler.codec.CorruptedFrameException;
|
||||||
import io.netty.handler.codec.ReplayingDecoder;
|
|
||||||
import io.netty.handler.codec.TooLongFrameException;
|
import io.netty.handler.codec.TooLongFrameException;
|
||||||
import io.netty.util.internal.logging.InternalLogger;
|
import io.netty.util.internal.logging.InternalLogger;
|
||||||
import io.netty.util.internal.logging.InternalLoggerFactory;
|
import io.netty.util.internal.logging.InternalLoggerFactory;
|
||||||
|
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
|
|
||||||
|
import static io.netty.buffer.ByteBufUtil.readBytes;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Decodes a web socket frame from wire protocol version 8 format. This code was forked from <a
|
* Decodes a web socket frame from wire protocol version 8 format. This code was forked from <a
|
||||||
* href="https://github.com/joewalnes/webbit">webbit</a> and modified.
|
* href="https://github.com/joewalnes/webbit">webbit</a> and modified.
|
||||||
*/
|
*/
|
||||||
public class WebSocket08FrameDecoder extends ReplayingDecoder<WebSocket08FrameDecoder.State>
|
public class WebSocket08FrameDecoder extends ByteToMessageDecoder
|
||||||
implements WebSocketFrameDecoder {
|
implements WebSocketFrameDecoder {
|
||||||
|
|
||||||
|
enum State {
|
||||||
|
READING_FIRST,
|
||||||
|
READING_SECOND,
|
||||||
|
READING_SIZE,
|
||||||
|
MASKING_KEY,
|
||||||
|
PAYLOAD,
|
||||||
|
CORRUPT
|
||||||
|
}
|
||||||
|
|
||||||
private static final InternalLogger logger = InternalLoggerFactory.getInstance(WebSocket08FrameDecoder.class);
|
private static final InternalLogger logger = InternalLoggerFactory.getInstance(WebSocket08FrameDecoder.class);
|
||||||
|
|
||||||
private static final byte OPCODE_CONT = 0x0;
|
private static final byte OPCODE_CONT = 0x0;
|
||||||
@ -83,24 +94,19 @@ public class WebSocket08FrameDecoder extends ReplayingDecoder<WebSocket08FrameDe
|
|||||||
|
|
||||||
private UTF8Output fragmentedFramesText;
|
private UTF8Output fragmentedFramesText;
|
||||||
private int fragmentedFramesCount;
|
private int fragmentedFramesCount;
|
||||||
|
|
||||||
private final long maxFramePayloadLength;
|
|
||||||
private boolean frameFinalFlag;
|
private boolean frameFinalFlag;
|
||||||
private int frameRsv;
|
private int frameRsv;
|
||||||
private int frameOpcode;
|
private int frameOpcode;
|
||||||
private long framePayloadLength;
|
private long framePayloadLength;
|
||||||
private ByteBuf framePayload;
|
|
||||||
private int framePayloadBytesRead;
|
|
||||||
private byte[] maskingKey;
|
private byte[] maskingKey;
|
||||||
private ByteBuf payloadBuffer;
|
private int framePayloadLen1;
|
||||||
|
|
||||||
private final boolean allowExtensions;
|
|
||||||
private final boolean maskedPayload;
|
|
||||||
private boolean receivedClosingHandshake;
|
private boolean receivedClosingHandshake;
|
||||||
|
|
||||||
enum State {
|
private final long maxFramePayloadLength;
|
||||||
FRAME_START, MASKING_KEY, PAYLOAD, CORRUPT
|
private final boolean allowExtensions;
|
||||||
}
|
private final boolean maskedPayload;
|
||||||
|
|
||||||
|
private State state = State.READING_FIRST;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Constructor
|
* Constructor
|
||||||
@ -115,7 +121,6 @@ public class WebSocket08FrameDecoder extends ReplayingDecoder<WebSocket08FrameDe
|
|||||||
* helps check for denial of services attacks.
|
* helps check for denial of services attacks.
|
||||||
*/
|
*/
|
||||||
public WebSocket08FrameDecoder(boolean maskedPayload, boolean allowExtensions, int maxFramePayloadLength) {
|
public WebSocket08FrameDecoder(boolean maskedPayload, boolean allowExtensions, int maxFramePayloadLength) {
|
||||||
super(State.FRAME_START);
|
|
||||||
this.maskedPayload = maskedPayload;
|
this.maskedPayload = maskedPayload;
|
||||||
this.allowExtensions = allowExtensions;
|
this.allowExtensions = allowExtensions;
|
||||||
this.maxFramePayloadLength = maxFramePayloadLength;
|
this.maxFramePayloadLength = maxFramePayloadLength;
|
||||||
@ -129,14 +134,9 @@ public class WebSocket08FrameDecoder extends ReplayingDecoder<WebSocket08FrameDe
|
|||||||
in.skipBytes(actualReadableBytes());
|
in.skipBytes(actualReadableBytes());
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
switch (state) {
|
||||||
try {
|
case READING_FIRST:
|
||||||
switch (state()) {
|
framePayloadLength = 0;
|
||||||
case FRAME_START:
|
|
||||||
framePayloadBytesRead = 0;
|
|
||||||
framePayloadLength = -1;
|
|
||||||
framePayload = null;
|
|
||||||
payloadBuffer = null;
|
|
||||||
|
|
||||||
// FIN, RSV, OPCODE
|
// FIN, RSV, OPCODE
|
||||||
byte b = in.readByte();
|
byte b = in.readByte();
|
||||||
@ -148,10 +148,15 @@ public class WebSocket08FrameDecoder extends ReplayingDecoder<WebSocket08FrameDe
|
|||||||
logger.debug("Decoding WebSocket Frame opCode={}", frameOpcode);
|
logger.debug("Decoding WebSocket Frame opCode={}", frameOpcode);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
state = State.READING_SECOND;
|
||||||
|
case READING_SECOND:
|
||||||
|
if (!in.isReadable()) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
// MASK, PAYLOAD LEN 1
|
// MASK, PAYLOAD LEN 1
|
||||||
b = in.readByte();
|
b = in.readByte();
|
||||||
boolean frameMasked = (b & 0x80) != 0;
|
boolean frameMasked = (b & 0x80) != 0;
|
||||||
int framePayloadLen1 = b & 0x7F;
|
framePayloadLen1 = b & 0x7F;
|
||||||
|
|
||||||
if (frameRsv != 0 && !allowExtensions) {
|
if (frameRsv != 0 && !allowExtensions) {
|
||||||
protocolViolation(ctx, "RSV != 0 and no extension negotiated, RSV:" + frameRsv);
|
protocolViolation(ctx, "RSV != 0 and no extension negotiated, RSV:" + frameRsv);
|
||||||
@ -212,14 +217,23 @@ public class WebSocket08FrameDecoder extends ReplayingDecoder<WebSocket08FrameDe
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
state = State.READING_SIZE;
|
||||||
|
case READING_SIZE:
|
||||||
|
|
||||||
// Read frame payload length
|
// Read frame payload length
|
||||||
if (framePayloadLen1 == 126) {
|
if (framePayloadLen1 == 126) {
|
||||||
|
if (in.readableBytes() < 2) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
framePayloadLength = in.readUnsignedShort();
|
framePayloadLength = in.readUnsignedShort();
|
||||||
if (framePayloadLength < 126) {
|
if (framePayloadLength < 126) {
|
||||||
protocolViolation(ctx, "invalid data frame length (not using minimal length encoding)");
|
protocolViolation(ctx, "invalid data frame length (not using minimal length encoding)");
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
} else if (framePayloadLen1 == 127) {
|
} else if (framePayloadLen1 == 127) {
|
||||||
|
if (in.readableBytes() < 8) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
framePayloadLength = in.readLong();
|
framePayloadLength = in.readLong();
|
||||||
// TODO: check if it's bigger than 0x7FFFFFFFFFFFFFFF, Maybe
|
// TODO: check if it's bigger than 0x7FFFFFFFFFFFFFFF, Maybe
|
||||||
// just check if it's negative?
|
// just check if it's negative?
|
||||||
@ -241,169 +255,130 @@ public class WebSocket08FrameDecoder extends ReplayingDecoder<WebSocket08FrameDe
|
|||||||
logger.debug("Decoding WebSocket Frame length={}", framePayloadLength);
|
logger.debug("Decoding WebSocket Frame length={}", framePayloadLength);
|
||||||
}
|
}
|
||||||
|
|
||||||
checkpoint(State.MASKING_KEY);
|
state = State.MASKING_KEY;
|
||||||
case MASKING_KEY:
|
case MASKING_KEY:
|
||||||
if (maskedPayload) {
|
if (maskedPayload) {
|
||||||
|
if (in.readableBytes() < 4) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
if (maskingKey == null) {
|
if (maskingKey == null) {
|
||||||
maskingKey = new byte[4];
|
maskingKey = new byte[4];
|
||||||
}
|
}
|
||||||
in.readBytes(maskingKey);
|
in.readBytes(maskingKey);
|
||||||
}
|
}
|
||||||
checkpoint(State.PAYLOAD);
|
state = State.PAYLOAD;
|
||||||
case PAYLOAD:
|
case PAYLOAD:
|
||||||
// Sometimes, the payload may not be delivered in 1 nice packet
|
if (in.readableBytes() < framePayloadLength) {
|
||||||
// We need to accumulate the data until we have it all
|
return;
|
||||||
int rbytes = actualReadableBytes();
|
}
|
||||||
|
|
||||||
long willHaveReadByteCount = framePayloadBytesRead + rbytes;
|
ByteBuf payloadBuffer = null;
|
||||||
// logger.debug("Frame rbytes=" + rbytes + " willHaveReadByteCount="
|
try {
|
||||||
// + willHaveReadByteCount + " framePayloadLength=" +
|
payloadBuffer = readBytes(ctx.alloc(), in, toFrameLength(framePayloadLength));
|
||||||
// framePayloadLength);
|
|
||||||
if (willHaveReadByteCount == framePayloadLength) {
|
|
||||||
// We have all our content so proceed to process
|
|
||||||
payloadBuffer = ctx.alloc().buffer(rbytes);
|
|
||||||
payloadBuffer.writeBytes(in, rbytes);
|
|
||||||
} else if (willHaveReadByteCount < framePayloadLength) {
|
|
||||||
|
|
||||||
// We don't have all our content so accumulate payload.
|
// Now we have all the data, the next checkpoint must be the next
|
||||||
// Returning null means we will get called back
|
// frame
|
||||||
if (framePayload == null) {
|
state = State.READING_FIRST;
|
||||||
framePayload = ctx.alloc().buffer(toFrameLength(framePayloadLength));
|
|
||||||
|
// Unmask data if needed
|
||||||
|
if (maskedPayload) {
|
||||||
|
unmask(payloadBuffer);
|
||||||
}
|
}
|
||||||
framePayload.writeBytes(in, rbytes);
|
|
||||||
framePayloadBytesRead += rbytes;
|
|
||||||
|
|
||||||
// Return null to wait for more bytes to arrive
|
// Processing ping/pong/close frames because they cannot be
|
||||||
return;
|
// fragmented
|
||||||
} else if (willHaveReadByteCount > framePayloadLength) {
|
if (frameOpcode == OPCODE_PING) {
|
||||||
// We have more than what we need so read up to the end of frame
|
out.add(new PingWebSocketFrame(frameFinalFlag, frameRsv, payloadBuffer));
|
||||||
// Leave the remainder in the buffer for next frame
|
payloadBuffer = null;
|
||||||
if (framePayload == null) {
|
return;
|
||||||
framePayload = ctx.alloc().buffer(toFrameLength(framePayloadLength));
|
|
||||||
}
|
}
|
||||||
framePayload.writeBytes(in, toFrameLength(framePayloadLength - framePayloadBytesRead));
|
if (frameOpcode == OPCODE_PONG) {
|
||||||
}
|
out.add(new PongWebSocketFrame(frameFinalFlag, frameRsv, payloadBuffer));
|
||||||
|
payloadBuffer = null;
|
||||||
// Now we have all the data, the next checkpoint must be the next
|
return;
|
||||||
// frame
|
|
||||||
checkpoint(State.FRAME_START);
|
|
||||||
|
|
||||||
// Take the data that we have in this packet
|
|
||||||
if (framePayload == null) {
|
|
||||||
framePayload = payloadBuffer;
|
|
||||||
payloadBuffer = null;
|
|
||||||
} else if (payloadBuffer != null) {
|
|
||||||
framePayload.writeBytes(payloadBuffer);
|
|
||||||
payloadBuffer.release();
|
|
||||||
payloadBuffer = null;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Unmask data if needed
|
|
||||||
if (maskedPayload) {
|
|
||||||
unmask(framePayload);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Processing ping/pong/close frames because they cannot be
|
|
||||||
// fragmented
|
|
||||||
if (frameOpcode == OPCODE_PING) {
|
|
||||||
out.add(new PingWebSocketFrame(frameFinalFlag, frameRsv, framePayload));
|
|
||||||
framePayload = null;
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
if (frameOpcode == OPCODE_PONG) {
|
|
||||||
out.add(new PongWebSocketFrame(frameFinalFlag, frameRsv, framePayload));
|
|
||||||
framePayload = null;
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
if (frameOpcode == OPCODE_CLOSE) {
|
|
||||||
checkCloseFrameBody(ctx, framePayload);
|
|
||||||
receivedClosingHandshake = true;
|
|
||||||
out.add(new CloseWebSocketFrame(frameFinalFlag, frameRsv, framePayload));
|
|
||||||
framePayload = null;
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Processing for possible fragmented messages for text and binary
|
|
||||||
// frames
|
|
||||||
String aggregatedText = null;
|
|
||||||
if (frameFinalFlag) {
|
|
||||||
// Final frame of the sequence. Apparently ping frames are
|
|
||||||
// allowed in the middle of a fragmented message
|
|
||||||
if (frameOpcode != OPCODE_PING) {
|
|
||||||
fragmentedFramesCount = 0;
|
|
||||||
|
|
||||||
// Check text for UTF8 correctness
|
|
||||||
if (frameOpcode == OPCODE_TEXT || fragmentedFramesText != null) {
|
|
||||||
// Check UTF-8 correctness for this payload
|
|
||||||
checkUTF8String(ctx, framePayload);
|
|
||||||
|
|
||||||
// This does a second check to make sure UTF-8
|
|
||||||
// correctness for entire text message
|
|
||||||
aggregatedText = fragmentedFramesText.toString();
|
|
||||||
|
|
||||||
fragmentedFramesText = null;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
} else {
|
if (frameOpcode == OPCODE_CLOSE) {
|
||||||
// Not final frame so we can expect more frames in the
|
checkCloseFrameBody(ctx, payloadBuffer);
|
||||||
// fragmented sequence
|
receivedClosingHandshake = true;
|
||||||
if (fragmentedFramesCount == 0) {
|
out.add(new CloseWebSocketFrame(frameFinalFlag, frameRsv, payloadBuffer));
|
||||||
// First text or binary frame for a fragmented set
|
payloadBuffer = null;
|
||||||
fragmentedFramesText = null;
|
return;
|
||||||
if (frameOpcode == OPCODE_TEXT) {
|
}
|
||||||
checkUTF8String(ctx, framePayload);
|
|
||||||
|
// Processing for possible fragmented messages for text and binary
|
||||||
|
// frames
|
||||||
|
String aggregatedText = null;
|
||||||
|
if (frameFinalFlag) {
|
||||||
|
// Final frame of the sequence. Apparently ping frames are
|
||||||
|
// allowed in the middle of a fragmented message
|
||||||
|
if (frameOpcode != OPCODE_PING) {
|
||||||
|
fragmentedFramesCount = 0;
|
||||||
|
|
||||||
|
// Check text for UTF8 correctness
|
||||||
|
if (frameOpcode == OPCODE_TEXT || fragmentedFramesText != null) {
|
||||||
|
// Check UTF-8 correctness for this payload
|
||||||
|
checkUTF8String(ctx, payloadBuffer);
|
||||||
|
|
||||||
|
// This does a second check to make sure UTF-8
|
||||||
|
// correctness for entire text message
|
||||||
|
aggregatedText = fragmentedFramesText.toString();
|
||||||
|
|
||||||
|
fragmentedFramesText = null;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
// Subsequent frames - only check if init frame is text
|
// Not final frame so we can expect more frames in the
|
||||||
if (fragmentedFramesText != null) {
|
// fragmented sequence
|
||||||
checkUTF8String(ctx, framePayload);
|
if (fragmentedFramesCount == 0) {
|
||||||
|
// First text or binary frame for a fragmented set
|
||||||
|
fragmentedFramesText = null;
|
||||||
|
if (frameOpcode == OPCODE_TEXT) {
|
||||||
|
checkUTF8String(ctx, payloadBuffer);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
// Subsequent frames - only check if init frame is text
|
||||||
|
if (fragmentedFramesText != null) {
|
||||||
|
checkUTF8String(ctx, payloadBuffer);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Increment counter
|
||||||
|
fragmentedFramesCount++;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Increment counter
|
// Return the frame
|
||||||
fragmentedFramesCount++;
|
if (frameOpcode == OPCODE_TEXT) {
|
||||||
}
|
out.add(new TextWebSocketFrame(frameFinalFlag, frameRsv, payloadBuffer));
|
||||||
|
payloadBuffer = null;
|
||||||
// Return the frame
|
return;
|
||||||
if (frameOpcode == OPCODE_TEXT) {
|
} else if (frameOpcode == OPCODE_BINARY) {
|
||||||
out.add(new TextWebSocketFrame(frameFinalFlag, frameRsv, framePayload));
|
out.add(new BinaryWebSocketFrame(frameFinalFlag, frameRsv, payloadBuffer));
|
||||||
framePayload = null;
|
payloadBuffer = null;
|
||||||
return;
|
return;
|
||||||
} else if (frameOpcode == OPCODE_BINARY) {
|
} else if (frameOpcode == OPCODE_CONT) {
|
||||||
out.add(new BinaryWebSocketFrame(frameFinalFlag, frameRsv, framePayload));
|
out.add(new ContinuationWebSocketFrame(frameFinalFlag, frameRsv,
|
||||||
framePayload = null;
|
payloadBuffer, aggregatedText));
|
||||||
return;
|
payloadBuffer = null;
|
||||||
} else if (frameOpcode == OPCODE_CONT) {
|
return;
|
||||||
out.add(new ContinuationWebSocketFrame(frameFinalFlag, frameRsv, framePayload, aggregatedText));
|
} else {
|
||||||
framePayload = null;
|
throw new UnsupportedOperationException("Cannot decode web socket frame with opcode: "
|
||||||
return;
|
+ frameOpcode);
|
||||||
} else {
|
}
|
||||||
throw new UnsupportedOperationException("Cannot decode web socket frame with opcode: "
|
} finally {
|
||||||
+ frameOpcode);
|
if (payloadBuffer != null) {
|
||||||
|
payloadBuffer.release();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
case CORRUPT:
|
case CORRUPT:
|
||||||
// If we don't keep reading Netty will throw an exception saying
|
if (in.isReadable()) {
|
||||||
// we can't return null if no bytes read and state not changed.
|
// If we don't keep reading Netty will throw an exception saying
|
||||||
in.readByte();
|
// we can't return null if no bytes read and state not changed.
|
||||||
|
in.readByte();
|
||||||
|
}
|
||||||
return;
|
return;
|
||||||
default:
|
default:
|
||||||
throw new Error("Shouldn't reach here.");
|
throw new Error("Shouldn't reach here.");
|
||||||
}
|
}
|
||||||
} catch (Exception e) {
|
|
||||||
if (payloadBuffer != null) {
|
|
||||||
if (payloadBuffer.refCnt() > 0) {
|
|
||||||
payloadBuffer.release();
|
|
||||||
}
|
|
||||||
payloadBuffer = null;
|
|
||||||
}
|
|
||||||
if (framePayload != null) {
|
|
||||||
if (framePayload.refCnt() > 0) {
|
|
||||||
framePayload.release();
|
|
||||||
}
|
|
||||||
framePayload = null;
|
|
||||||
}
|
|
||||||
throw e;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private void unmask(ByteBuf frame) {
|
private void unmask(ByteBuf frame) {
|
||||||
@ -413,7 +388,7 @@ public class WebSocket08FrameDecoder extends ReplayingDecoder<WebSocket08FrameDe
|
|||||||
}
|
}
|
||||||
|
|
||||||
private void protocolViolation(ChannelHandlerContext ctx, String reason) {
|
private void protocolViolation(ChannelHandlerContext ctx, String reason) {
|
||||||
checkpoint(State.CORRUPT);
|
state = State.CORRUPT;
|
||||||
if (ctx.channel().isActive()) {
|
if (ctx.channel().isActive()) {
|
||||||
ctx.writeAndFlush(Unpooled.EMPTY_BUFFER).addListener(ChannelFutureListener.CLOSE);
|
ctx.writeAndFlush(Unpooled.EMPTY_BUFFER).addListener(ChannelFutureListener.CLOSE);
|
||||||
}
|
}
|
||||||
@ -473,18 +448,4 @@ public class WebSocket08FrameDecoder extends ReplayingDecoder<WebSocket08FrameDe
|
|||||||
// Restore reader index
|
// Restore reader index
|
||||||
buffer.readerIndex(idx);
|
buffer.readerIndex(idx);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
|
||||||
public void channelInactive(ChannelHandlerContext ctx) throws Exception {
|
|
||||||
super.channelInactive(ctx);
|
|
||||||
|
|
||||||
// release all not complete frames data to prevent leaks.
|
|
||||||
// https://github.com/netty/netty/issues/1874
|
|
||||||
if (framePayload != null) {
|
|
||||||
framePayload.release();
|
|
||||||
}
|
|
||||||
if (payloadBuffer != null) {
|
|
||||||
payloadBuffer.release();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
Loading…
x
Reference in New Issue
Block a user