Merge branch 'jetty-9.1-wssplitframe' into jetty-9.1

This commit is contained in:
Joakim Erdfelt 2013-08-19 15:27:09 -07:00
commit 8e66da396e
74 changed files with 1321 additions and 1220 deletions

View File

@ -186,12 +186,14 @@ public class EchoTest
.expect("Hello World|OhMy");
// Partial message based
/*
EchoCase.add(TESTCASES,PartialTextSocket.class)
.addSplitMessage("Saved"," by ","zero")
.expect("('Saved',false)(' by ',false)('zero',true)");
EchoCase.add(TESTCASES,PartialTextSessionSocket.class)
.addSplitMessage("Built"," for"," the"," future")
.expect("('Built',false)(' for',false)(' the',false)(' future',true)");
*/
}
@BeforeClass

View File

@ -1,7 +1,7 @@
org.eclipse.jetty.util.log.class=org.eclipse.jetty.util.log.StdErrLog
org.eclipse.jetty.LEVEL=WARN
# org.eclipse.jetty.websocket.LEVEL=DEBUG
org.eclipse.jetty.websocket.LEVEL=DEBUG
# org.eclipse.jetty.websocket.LEVEL=INFO
# org.eclipse.jetty.websocket.LEVEL=WARN
# org.eclipse.jetty.websocket.common.io.LEVEL=DEBUG

View File

@ -66,19 +66,6 @@ public interface Extension extends IncomingFrames, OutgoingFrames
*/
public abstract boolean isRsv3User();
/**
* Used to indicate that the extension works as a decoder of TEXT Data Frames.
* <p>
* This is used to adjust validation during parsing/generating, as per spec TEXT Data Frames can only contain UTF8 encoded String data.
* <p>
* Example: a compression extension will process a compressed set of text data, the parser/generator should no longer be concerned about the validity of the
* TEXT Data Frames as this is now the responsibility of the extension.
*
* @return true if extension will process TEXT Data Frames, false if extension makes no modifications of TEXT Data Frames. If false, the parser/generator is
* now free to validate the conformance to spec of TEXT Data Frames.
*/
public abstract boolean isTextDataDecoder();
/**
* Set the next {@link IncomingFrames} to call in the chain.
*

View File

@ -92,8 +92,6 @@ public interface Frame
public boolean hasPayload();
public boolean isContinuation();
public boolean isFin();
/**

View File

@ -64,6 +64,7 @@ import org.eclipse.jetty.websocket.common.Parser;
import org.eclipse.jetty.websocket.common.WebSocketFrame;
import org.eclipse.jetty.websocket.common.extensions.ExtensionStack;
import org.eclipse.jetty.websocket.common.extensions.WebSocketExtensionFactory;
import org.eclipse.jetty.websocket.common.frames.CloseFrame;
import org.junit.Assert;
/**
@ -117,7 +118,7 @@ public class BlockheadServer
public void close() throws IOException
{
write(new WebSocketFrame(OpCode.CLOSE));
write(new CloseFrame());
flush();
disconnect();
}
@ -217,8 +218,7 @@ public class BlockheadServer
{
LOG.info("Server parsed {} frames",count);
}
WebSocketFrame copy = new WebSocketFrame(frame);
incomingFrames.incomingFrame(copy);
incomingFrames.incomingFrame(WebSocketFrame.copy(frame));
if (frame.getType() == Type.CLOSE)
{

View File

@ -130,7 +130,7 @@ public class IncomingFramesCapture implements IncomingFrames
@Override
public void incomingFrame(Frame frame)
{
WebSocketFrame copy = new WebSocketFrame(frame);
WebSocketFrame copy = WebSocketFrame.copy(frame);
Assert.assertThat("frame.masking must be set",frame.isMasked(),is(true));
frames.add(copy);
}

View File

@ -29,6 +29,7 @@ import org.eclipse.jetty.websocket.api.BadPayloadException;
import org.eclipse.jetty.websocket.api.ProtocolException;
import org.eclipse.jetty.websocket.api.StatusCode;
import org.eclipse.jetty.websocket.api.extensions.Frame;
import org.eclipse.jetty.websocket.common.frames.CloseFrame;
public class CloseInfo
{
@ -128,7 +129,7 @@ public class CloseInfo
this.reason = reason;
}
private byte[] asByteBuffer()
private ByteBuffer asByteBuffer()
{
if ((statusCode == StatusCode.NO_CLOSE) || (statusCode == StatusCode.NO_CODE) || (statusCode == (-1)))
{
@ -144,21 +145,22 @@ public class CloseInfo
len += utf.length;
}
byte buf[] = new byte[len];
buf[0] = (byte)((statusCode >>> 8) & 0xFF);
buf[1] = (byte)((statusCode >>> 0) & 0xFF);
ByteBuffer buf = ByteBuffer.allocate(len);
buf.put((byte)((statusCode >>> 8) & 0xFF));
buf.put((byte)((statusCode >>> 0) & 0xFF));
if (utf != null)
{
System.arraycopy(utf,0,buf,2,utf.length);
buf.put(utf,0,utf.length);
}
buf.flip();
return buf;
}
public WebSocketFrame asFrame()
public CloseFrame asFrame()
{
WebSocketFrame frame = new WebSocketFrame(OpCode.CLOSE);
CloseFrame frame = new CloseFrame();
frame.setFin(true);
frame.setPayload(asByteBuffer());
return frame;

View File

@ -224,7 +224,7 @@ public class Generator
// NOTE: using .getOpCode() here, not .getType().getOpCode() for testing reasons
byte opcode = frame.getOpCode();
if (frame.isContinuation())
if (frame.getOpCode() == OpCode.CONTINUATION)
{
// Continuations are not the same OPCODE
opcode = OpCode.CONTINUATION;

View File

@ -33,11 +33,15 @@ import org.eclipse.jetty.websocket.api.WebSocketPolicy;
import org.eclipse.jetty.websocket.api.extensions.Extension;
import org.eclipse.jetty.websocket.api.extensions.Frame;
import org.eclipse.jetty.websocket.api.extensions.IncomingFrames;
import org.eclipse.jetty.websocket.common.io.payload.CloseReasonValidator;
import org.eclipse.jetty.websocket.common.frames.BinaryFrame;
import org.eclipse.jetty.websocket.common.frames.CloseFrame;
import org.eclipse.jetty.websocket.common.frames.ContinuationFrame;
import org.eclipse.jetty.websocket.common.frames.ControlFrame;
import org.eclipse.jetty.websocket.common.frames.PingFrame;
import org.eclipse.jetty.websocket.common.frames.PongFrame;
import org.eclipse.jetty.websocket.common.frames.TextFrame;
import org.eclipse.jetty.websocket.common.io.payload.DeMaskProcessor;
import org.eclipse.jetty.websocket.common.io.payload.NoOpValidator;
import org.eclipse.jetty.websocket.common.io.payload.PayloadProcessor;
import org.eclipse.jetty.websocket.common.io.payload.UTF8Validator;
/**
* Parsing of a frames in WebSocket land.
@ -70,7 +74,7 @@ public class Parser
private ByteBuffer payload;
private int payloadLength;
private PayloadProcessor maskProcessor = new DeMaskProcessor();
private PayloadProcessor strictnessProcessor;
// private PayloadProcessor strictnessProcessor;
/** Is there an extension using RSV1 */
private boolean rsv1InUse = false;
@ -78,8 +82,6 @@ public class Parser
private boolean rsv2InUse = false;
/** Is there an extension using RSV3 */
private boolean rsv3InUse = false;
/** Is there an extension that processes invalid UTF8 text messages (such as compressed content) */
private boolean isTextFrameValidated = true;
private IncomingFrames incomingFramesHandler;
@ -113,10 +115,10 @@ public class Parser
// fall thru
case OpCode.PING:
case OpCode.PONG:
if (len > WebSocketFrame.MAX_CONTROL_PAYLOAD)
if (len > ControlFrame.MAX_CONTROL_PAYLOAD)
{
throw new ProtocolException("Invalid control frame payload length, [" + payloadLength + "] cannot exceed ["
+ WebSocketFrame.MAX_CONTROL_PAYLOAD + "]");
+ ControlFrame.MAX_CONTROL_PAYLOAD + "]");
}
break;
case OpCode.TEXT:
@ -134,7 +136,6 @@ public class Parser
this.rsv1InUse = false;
this.rsv2InUse = false;
this.rsv3InUse = false;
this.isTextFrameValidated = true;
// configure from list of extensions in use
for (Extension ext : exts)
@ -151,10 +152,6 @@ public class Parser
{
this.rsv3InUse = true;
}
if (ext.isTextDataDecoder())
{
this.isTextFrameValidated = false;
}
}
}
@ -343,39 +340,26 @@ public class Parser
throw new ProtocolException("RSV3 not allowed to be set");
}
boolean isContinuation = false;
switch (opcode)
{
// base framing flags
switch(opcode) {
case OpCode.TEXT:
if (isTextFrameValidated)
frame = new TextFrame();
// data validation
if ((priorDataFrame != null) && (!priorDataFrame.isFin()))
{
strictnessProcessor = new UTF8Validator();
}
else
{
strictnessProcessor = NoOpValidator.INSTANCE;
throw new ProtocolException("Unexpected " + OpCode.name(opcode) + " frame, was expecting CONTINUATION");
}
break;
case OpCode.CLOSE:
strictnessProcessor = new CloseReasonValidator();
case OpCode.BINARY:
frame = new BinaryFrame();
// data validation
if ((priorDataFrame != null) && (!priorDataFrame.isFin()))
{
throw new ProtocolException("Unexpected " + OpCode.name(opcode) + " frame, was expecting CONTINUATION");
}
break;
default:
strictnessProcessor = NoOpValidator.INSTANCE;
break;
}
if (OpCode.isControlFrame(opcode))
{
// control frame validation
if (!fin)
{
throw new ProtocolException("Fragmented Control Frame [" + OpCode.name(opcode) + "]");
}
}
else if (opcode == OpCode.CONTINUATION)
{
isContinuation = true;
case OpCode.CONTINUATION:
frame = new ContinuationFrame();
// continuation validation
if (priorDataFrame == null)
{
@ -383,23 +367,37 @@ public class Parser
}
// Be careful to use the original opcode
opcode = lastDataOpcode;
}
else if (OpCode.isDataFrame(opcode))
break;
case OpCode.CLOSE:
frame = new CloseFrame();
// control frame validation
if (!fin)
{
// data validation
if ((priorDataFrame != null) && (!priorDataFrame.isFin()))
{
throw new ProtocolException("Unexpected " + OpCode.name(opcode) + " frame, was expecting CONTINUATION");
throw new ProtocolException("Fragmented Close Frame [" + OpCode.name(opcode) + "]");
}
break;
case OpCode.PING:
frame = new PingFrame();
// control frame validation
if (!fin)
{
throw new ProtocolException("Fragmented Ping Frame [" + OpCode.name(opcode) + "]");
}
break;
case OpCode.PONG:
frame = new PongFrame();
// control frame validation
if (!fin)
{
throw new ProtocolException("Fragmented Pong Frame [" + OpCode.name(opcode) + "]");
}
break;
}
// base framing flags
frame = new WebSocketFrame(opcode);
frame.setFin(fin);
frame.setRsv1(rsv1);
frame.setRsv2(rsv2);
frame.setRsv3(rsv3);
frame.setContinuation(isContinuation);
if (frame.isDataFrame())
{
@ -581,7 +579,6 @@ public class Parser
}
maskProcessor.process(window);
strictnessProcessor.process(window);
int len = BufferUtil.put(window,payload);
buffer.position(buffer.position() + len); // update incoming buffer position

View File

@ -22,9 +22,13 @@ import java.nio.ByteBuffer;
import java.util.Arrays;
import org.eclipse.jetty.util.BufferUtil;
import org.eclipse.jetty.util.StringUtil;
import org.eclipse.jetty.websocket.api.ProtocolException;
import org.eclipse.jetty.websocket.api.extensions.Frame;
import org.eclipse.jetty.websocket.common.frames.BinaryFrame;
import org.eclipse.jetty.websocket.common.frames.CloseFrame;
import org.eclipse.jetty.websocket.common.frames.ContinuationFrame;
import org.eclipse.jetty.websocket.common.frames.PingFrame;
import org.eclipse.jetty.websocket.common.frames.PongFrame;
import org.eclipse.jetty.websocket.common.frames.TextFrame;
/**
* A Base Frame as seen in <a href="https://tools.ietf.org/html/rfc6455#section-5.2">RFC 6455. Sec 5.2</a>
@ -50,44 +54,80 @@ import org.eclipse.jetty.websocket.api.extensions.Frame;
* +---------------------------------------------------------------+
* </pre>
*/
public class WebSocketFrame implements Frame
public abstract class WebSocketFrame implements Frame
{
/** Maximum size of Control frame, per RFC 6455 */
public static final int MAX_CONTROL_PAYLOAD = 125;
public static WebSocketFrame binary()
public static BinaryFrame binary()
{
return new WebSocketFrame(OpCode.BINARY);
return new BinaryFrame();
}
public static WebSocketFrame binary(byte buf[])
public static BinaryFrame binary(byte buf[])
{
return new WebSocketFrame(OpCode.BINARY).setPayload(buf);
return new BinaryFrame(buf);
}
public static WebSocketFrame ping()
public static BinaryFrame binary(ByteBuffer buf)
{
return new WebSocketFrame(OpCode.PING);
return new BinaryFrame(buf);
}
public static WebSocketFrame pong()
public static WebSocketFrame copy(Frame copy)
{
return new WebSocketFrame(OpCode.PONG);
WebSocketFrame frame = null;
switch (copy.getOpCode())
{
case OpCode.BINARY:
frame = new BinaryFrame();
break;
case OpCode.TEXT:
frame = new TextFrame();
break;
case OpCode.CLOSE:
frame = new CloseFrame();
break;
case OpCode.CONTINUATION:
frame = new ContinuationFrame();
break;
case OpCode.PING:
frame = new PingFrame();
break;
case OpCode.PONG:
frame = new PongFrame();
break;
default:
throw new IllegalArgumentException("Cannot copy frame with opcode " + copy.getOpCode() + " - " + copy);
}
public static WebSocketFrame text()
{
return new WebSocketFrame(OpCode.TEXT);
frame.copyHeaders(copy);
frame.setPayload(copy.getPayload());
return frame;
}
public static WebSocketFrame text(String msg)
public static PingFrame ping()
{
return new WebSocketFrame(OpCode.TEXT).setPayload(msg);
return new PingFrame();
}
public static PongFrame pong()
{
return new PongFrame();
}
public static TextFrame text()
{
return new TextFrame();
}
public static TextFrame text(String msg)
{
return new TextFrame(msg);
}
/**
* Combined FIN + RSV1 + RSV2 + RSV3 + OpCode byte.
* <p>
*
* <pre>
* 1000_0000 (0x80) = fin
* 0100_0000 (0x40) = rsv1
@ -97,54 +137,40 @@ public class WebSocketFrame implements Frame
* </pre>
*/
protected byte finRsvOp;
protected boolean masked = false;
private boolean masked = false;
private byte mask[];
protected byte mask[];
/**
* The payload data.
* <p>
* It is assumed to always be in FLUSH mode (ready to read) in this object.
*/
private ByteBuffer data;
private int payloadLength = 0;
protected ByteBuffer data;
protected int payloadLength = 0;
// FIXME: Remove
private Type type;
private boolean continuation = false;
private int continuationIndex = 0;
/**
* Default constructor
*/
public WebSocketFrame()
{
this(OpCode.UNDEFINED);
}
/**
* Construct form opcode
*/
public WebSocketFrame(byte opcode)
protected WebSocketFrame(byte opcode)
{
reset();
setOpCode(opcode);
}
/**
* Copy constructor for the websocket frame.
*
* @param frame
* the websocket frame to copy.
*/
public WebSocketFrame(Frame frame)
public abstract void assertValid();
protected void copy(WebSocketFrame copy, ByteBuffer payload)
{
if (frame instanceof WebSocketFrame)
{
WebSocketFrame wsf = (WebSocketFrame)frame;
copy(wsf,wsf.data);
copyHeaders(copy);
setPayload(payload);
}
else
protected void copyHeaders(Frame frame)
{
// Copy manually
finRsvOp = 0x00;
finRsvOp |= frame.isFin()?0x80:0x00;
finRsvOp |= frame.isRsv1()?0x40:0x00;
@ -154,84 +180,17 @@ public class WebSocketFrame implements Frame
type = frame.getType();
masked = frame.isMasked();
if (masked)
{
mask = frame.getMask();
}
else
{
mask = null;
byte maskCopy[] = frame.getMask();
if (maskCopy != null)
{
mask = new byte[maskCopy.length];
System.arraycopy(maskCopy,0,mask,0,mask.length);
}
setPayload(frame.getPayload());
}
}
/**
* Copy constructor for the websocket frame.
* <p>
* Note: the underlying payload is merely a {@link ByteBuffer#slice()} of the input frame.
*
* @param copy
* the websocket to copy.
*/
public WebSocketFrame(WebSocketFrame copy)
{
copy(copy,copy.data);
}
/**
* Copy constructor for the websocket frame, with an alternate payload.
* <p>
* This is especially useful for Extensions to utilize when mutating the payload.
*
* @param copy
* the websocket to copy.
* @param altPayload
* the alternate payload to use for this frame.
*/
public WebSocketFrame(WebSocketFrame copy, ByteBuffer altPayload)
{
copy(copy,altPayload);
}
public void assertValid()
{
if (isControlFrame())
{
if (getPayloadLength() > WebSocketFrame.MAX_CONTROL_PAYLOAD)
{
throw new ProtocolException("Desired payload length [" + getPayloadLength() + "] exceeds maximum control payload length ["
+ MAX_CONTROL_PAYLOAD + "]");
}
if ((finRsvOp & 0x80) == 0)
{
throw new ProtocolException("Cannot have FIN==false on Control frames");
}
if ((finRsvOp & 0x40) != 0)
{
throw new ProtocolException("Cannot have RSV1==true on Control frames");
}
if ((finRsvOp & 0x20) != 0)
{
throw new ProtocolException("Cannot have RSV2==true on Control frames");
}
if ((finRsvOp & 0x10) != 0)
{
throw new ProtocolException("Cannot have RSV3==true on Control frames");
}
if (isContinuation())
{
throw new ProtocolException("Control frames cannot be Continuations");
}
}
}
private final void copy(WebSocketFrame copy, ByteBuffer payload)
protected void copyHeaders(WebSocketFrame copy)
{
finRsvOp = copy.finRsvOp;
type = copy.type;
@ -242,10 +201,6 @@ public class WebSocketFrame implements Frame
mask = new byte[copy.mask.length];
System.arraycopy(copy.mask,0,mask,0,mask.length);
}
continuationIndex = copy.continuationIndex;
continuation = copy.continuation;
setPayload(payload);
}
@Override
@ -264,14 +219,6 @@ public class WebSocketFrame implements Frame
return false;
}
WebSocketFrame other = (WebSocketFrame)obj;
if (continuation != other.continuation)
{
return false;
}
if (continuationIndex != other.continuationIndex)
{
return false;
}
if (data == null)
{
if (other.data != null)
@ -298,20 +245,6 @@ public class WebSocketFrame implements Frame
return true;
}
/**
* The number of fragments this frame consists of.
* <p>
* For every {@link OpCode#CONTINUATION} opcode encountered, this increments by one.
* <p>
* Note: Not part of the Base Framing Protocol / header information.
*
* @return the number of continuation fragments encountered.
*/
public int getContinuationIndex()
{
return continuationIndex;
}
@Override
public byte[] getMask()
{
@ -344,11 +277,7 @@ public class WebSocketFrame implements Frame
public String getPayloadAsUTF8()
{
if (data == null)
{
return null;
}
return BufferUtil.toUTF8String(data);
return BufferUtil.toUTF8String(getPayload());
}
@Override
@ -372,8 +301,6 @@ public class WebSocketFrame implements Frame
{
final int prime = 31;
int result = 1;
result = (prime * result) + (continuation?1231:1237);
result = (prime * result) + continuationIndex;
result = (prime * result) + ((data == null)?0:data.hashCode());
result = (prime * result) + finRsvOp;
result = (prime * result) + Arrays.hashCode(mask);
@ -386,21 +313,9 @@ public class WebSocketFrame implements Frame
return ((data != null) && (payloadLength > 0));
}
@Override
public boolean isContinuation()
{
return continuation;
}
public abstract boolean isControlFrame();
public boolean isControlFrame()
{
return OpCode.isControlFrame(getOpCode());
}
public boolean isDataFrame()
{
return OpCode.isDataFrame(getOpCode());
}
public abstract boolean isDataFrame();
@Override
public boolean isFin()
@ -414,12 +329,6 @@ public class WebSocketFrame implements Frame
return isFin();
}
// FIXME: remove
public boolean isLastFrame()
{
return isFin();
}
@Override
public boolean isMasked()
{
@ -484,20 +393,6 @@ public class WebSocketFrame implements Frame
data = null;
payloadLength = 0;
mask = null;
continuationIndex = 0;
continuation = false;
}
public Frame setContinuation(boolean continuation)
{
this.continuation = continuation;
return this;
}
public Frame setContinuationIndex(int continuationIndex)
{
this.continuationIndex = continuationIndex;
return this;
}
public WebSocketFrame setFin(boolean fin)
@ -520,7 +415,7 @@ public class WebSocketFrame implements Frame
return this;
}
public WebSocketFrame setOpCode(byte op)
protected WebSocketFrame setOpCode(byte op)
{
this.finRsvOp = (byte)((finRsvOp & 0xF0) | (op & 0x0F));
@ -535,60 +430,6 @@ public class WebSocketFrame implements Frame
return this;
}
/**
* Set the data and payload length.
*
* @param buf
* the bytebuffer to set
*/
public WebSocketFrame setPayload(byte buf[])
{
if (buf == null)
{
data = null;
return this;
}
if (isControlFrame())
{
if (buf.length > WebSocketFrame.MAX_CONTROL_PAYLOAD)
{
throw new ProtocolException("Control Payloads can not exceed 125 bytes in length.");
}
}
data = ByteBuffer.wrap(buf);
payloadLength = data.remaining();
return this;
}
/**
* Set the data and payload length.
*
* @param buf
* the bytebuffer to set
*/
public WebSocketFrame setPayload(byte buf[], int offset, int len)
{
if (buf == null)
{
data = null;
return this;
}
if (isControlFrame())
{
if (len > WebSocketFrame.MAX_CONTROL_PAYLOAD)
{
throw new ProtocolException("Control Payloads can not exceed 125 bytes in length.");
}
}
data = BufferUtil.toBuffer(buf,offset,len);
payloadLength = data.limit();
return this;
}
/**
* Set the data payload.
* <p>
@ -607,25 +448,11 @@ public class WebSocketFrame implements Frame
return this;
}
if (isControlFrame())
{
if (buf.remaining() > WebSocketFrame.MAX_CONTROL_PAYLOAD)
{
throw new ProtocolException("Control Payloads can not exceed 125 bytes in length. (was " + buf.remaining() + " bytes)");
}
}
data = buf.slice();
payloadLength = data.limit();
return this;
}
public WebSocketFrame setPayload(String str)
{
setPayload(BufferUtil.toBuffer(str,StringUtil.__UTF8_CHARSET));
return this;
}
public WebSocketFrame setRsv1(boolean rsv1)
{
// set bit 2
@ -660,7 +487,6 @@ public class WebSocketFrame implements Frame
b.append(((finRsvOp & 0x20) != 0)?'1':'.');
b.append(((finRsvOp & 0x10) != 0)?'1':'.');
b.append(",masked=").append(masked);
b.append(",continuation=").append(continuation);
b.append(",remaining=").append(remaining());
b.append(",position=").append(position());
b.append(']');

View File

@ -33,6 +33,12 @@ import org.eclipse.jetty.util.log.Logger;
import org.eclipse.jetty.websocket.api.RemoteEndpoint;
import org.eclipse.jetty.websocket.api.WriteCallback;
import org.eclipse.jetty.websocket.api.extensions.OutgoingFrames;
import org.eclipse.jetty.websocket.common.frames.BinaryFrame;
import org.eclipse.jetty.websocket.common.frames.ContinuationFrame;
import org.eclipse.jetty.websocket.common.frames.DataFrame;
import org.eclipse.jetty.websocket.common.frames.PingFrame;
import org.eclipse.jetty.websocket.common.frames.PongFrame;
import org.eclipse.jetty.websocket.common.frames.TextFrame;
import org.eclipse.jetty.websocket.common.io.FutureWriteCallback;
/**
@ -186,10 +192,13 @@ public class WebSocketRemoteEndpoint implements RemoteEndpoint
{
LOG.debug("sendPartialBytes({}, {})",BufferUtil.toDetailString(fragment),isLast);
}
WebSocketFrame frame = WebSocketFrame.binary().setPayload(fragment).setFin(isLast);
DataFrame frame = null;
if(partialStarted) {
frame.setContinuation(true);
frame = new ContinuationFrame(fragment);
} else {
frame = new BinaryFrame(fragment);
}
frame.setFin(isLast);
blockingWrite(frame);
partialStarted = !isLast;
}
@ -225,10 +234,13 @@ public class WebSocketRemoteEndpoint implements RemoteEndpoint
{
LOG.debug("sendPartialString({}, {})",fragment,isLast);
}
WebSocketFrame frame = WebSocketFrame.text(fragment).setFin(isLast);
DataFrame frame = null;
if(partialStarted) {
frame.setContinuation(true);
frame = new ContinuationFrame(fragment);
} else {
frame = new TextFrame(fragment);
}
frame.setFin(isLast);
blockingWrite(frame);
partialStarted = !isLast;
}
@ -259,7 +271,7 @@ public class WebSocketRemoteEndpoint implements RemoteEndpoint
{
LOG.debug("sendPing with {}",BufferUtil.toDetailString(applicationData));
}
WebSocketFrame frame = WebSocketFrame.ping().setPayload(applicationData);
WebSocketFrame frame = new PingFrame(applicationData);
blockingWrite(frame);
}
finally
@ -286,7 +298,7 @@ public class WebSocketRemoteEndpoint implements RemoteEndpoint
{
LOG.debug("sendPong with {}",BufferUtil.toDetailString(applicationData));
}
WebSocketFrame frame = WebSocketFrame.pong().setPayload(applicationData);
WebSocketFrame frame = new PongFrame(applicationData);
blockingWrite(frame);
}
finally

View File

@ -22,7 +22,6 @@ import java.io.IOException;
import java.nio.ByteBuffer;
import org.eclipse.jetty.util.BufferUtil;
import org.eclipse.jetty.util.StringUtil;
import org.eclipse.jetty.util.Utf8Appendable.NotUtf8Exception;
import org.eclipse.jetty.util.log.Log;
import org.eclipse.jetty.util.log.Logger;
@ -33,8 +32,8 @@ import org.eclipse.jetty.websocket.api.extensions.Frame;
import org.eclipse.jetty.websocket.api.extensions.IncomingFrames;
import org.eclipse.jetty.websocket.common.CloseInfo;
import org.eclipse.jetty.websocket.common.OpCode;
import org.eclipse.jetty.websocket.common.WebSocketFrame;
import org.eclipse.jetty.websocket.common.WebSocketSession;
import org.eclipse.jetty.websocket.common.frames.CloseFrame;
import org.eclipse.jetty.websocket.common.message.MessageAppender;
/**
@ -117,7 +116,8 @@ public abstract class AbstractEventDriver implements IncomingFrames, EventDriver
case OpCode.CLOSE:
{
boolean validate = true;
CloseInfo close = new CloseInfo(frame,validate);
CloseFrame closeframe = (CloseFrame)frame;
CloseInfo close = new CloseInfo(closeframe,validate);
// notify user websocket pojo
onClose(close);
@ -224,10 +224,8 @@ public abstract class AbstractEventDriver implements IncomingFrames, EventDriver
protected void terminateConnection(int statusCode, String rawreason)
{
String reason = rawreason;
reason = StringUtil.truncate(reason,(WebSocketFrame.MAX_CONTROL_PAYLOAD - 2));
LOG.debug("terminateConnection({},{})",statusCode,rawreason);
session.close(statusCode,reason);
session.close(statusCode,CloseFrame.truncate(rawreason));
}
private void unhandled(Throwable t)

View File

@ -151,23 +151,6 @@ public abstract class AbstractExtension extends ContainerLifeCycle implements Ex
return false;
}
/**
* Used to indicate that the extension works as a decoder of TEXT Data Frames.
* <p>
* This is used to adjust validation during parsing/generating, as per spec TEXT Data Frames can only contain UTF8 encoded String data.
* <p>
* Example: a compression extension will process a compressed set of text data, the parser/generator should no longer be concerned about the validity of the
* TEXT Data Frames as this is now the responsibility of the extension.
*
* @return true if extension will process TEXT Data Frames, false if extension makes no modifications of TEXT Data Frames. If false, the parser/generator is
* now free to validate the conformance to spec of TEXT Data Frames.
*/
@Override
public boolean isTextDataDecoder()
{
return false;
}
protected void nextIncomingError(Throwable e)
{
this.nextIncoming.incomingError(e);

View File

@ -24,8 +24,8 @@ import java.nio.ByteBuffer;
import org.eclipse.jetty.websocket.api.WriteCallback;
import org.eclipse.jetty.websocket.api.extensions.ExtensionConfig;
import org.eclipse.jetty.websocket.api.extensions.Frame;
import org.eclipse.jetty.websocket.common.WebSocketFrame;
import org.eclipse.jetty.websocket.common.extensions.AbstractExtension;
import org.eclipse.jetty.websocket.common.frames.DataFrame;
/**
* Implementation of the <a href="https://tools.ietf.org/id/draft-tyoshino-hybi-websocket-perframe-deflate-05.txt">x-webkit-deflate-frame</a> extension seen out
@ -56,7 +56,8 @@ public class FrameCompressionExtension extends AbstractExtension
while (!method.decompress().isDone())
{
ByteBuffer uncompressed = method.decompress().process();
WebSocketFrame out = new WebSocketFrame(frame).setPayload(uncompressed);
DataFrame out = new DataFrame(frame);
out.setPayload(uncompressed);
if (!method.decompress().isDone())
{
out.setFin(false);
@ -80,15 +81,6 @@ public class FrameCompressionExtension extends AbstractExtension
return true;
}
/**
* Indicate that this extensions is now responsible for TEXT Data Frame compliance to the WebSocket spec.
*/
@Override
public boolean isTextDataDecoder()
{
return true;
}
@Override
public synchronized void outgoingFrame(Frame frame, WriteCallback callback)
{
@ -106,7 +98,8 @@ public class FrameCompressionExtension extends AbstractExtension
while (!method.compress().isDone())
{
ByteBuffer buf = method.compress().process();
WebSocketFrame out = new WebSocketFrame(frame).setPayload(buf);
DataFrame out = new DataFrame(frame);
out.setPayload(buf);
out.setRsv1(true);
if (!method.compress().isDone())
{

View File

@ -23,8 +23,8 @@ import java.nio.ByteBuffer;
import org.eclipse.jetty.websocket.api.WriteCallback;
import org.eclipse.jetty.websocket.api.extensions.ExtensionConfig;
import org.eclipse.jetty.websocket.api.extensions.Frame;
import org.eclipse.jetty.websocket.common.WebSocketFrame;
import org.eclipse.jetty.websocket.common.extensions.AbstractExtension;
import org.eclipse.jetty.websocket.common.frames.DataFrame;
/**
* Per Message Deflate Compression extension for WebSocket.
@ -60,7 +60,9 @@ public class MessageDeflateCompressionExtension extends AbstractExtension
{
continue;
}
WebSocketFrame out = new WebSocketFrame(frame).setPayload(uncompressed);
DataFrame out = new DataFrame(frame);
out.setPayload(uncompressed);
if (!method.decompress().isDone())
{
out.setFin(false);
@ -85,13 +87,6 @@ public class MessageDeflateCompressionExtension extends AbstractExtension
return true;
}
@Override
public boolean isTextDataDecoder()
{
// this extension is responsible for text data frames
return true;
}
@Override
public void outgoingFrame(Frame frame, WriteCallback callback)
{
@ -108,7 +103,8 @@ public class MessageDeflateCompressionExtension extends AbstractExtension
while (!method.compress().isDone())
{
ByteBuffer buf = method.compress().process();
WebSocketFrame out = new WebSocketFrame(frame).setPayload(buf);
DataFrame out = new DataFrame(frame);
out.setPayload(buf);
out.setRsv1(true);
if (!method.compress().isDone())
{

View File

@ -24,9 +24,8 @@ import java.nio.ByteBuffer;
import org.eclipse.jetty.websocket.api.WriteCallback;
import org.eclipse.jetty.websocket.api.extensions.ExtensionConfig;
import org.eclipse.jetty.websocket.api.extensions.Frame;
import org.eclipse.jetty.websocket.common.OpCode;
import org.eclipse.jetty.websocket.common.WebSocketFrame;
import org.eclipse.jetty.websocket.common.extensions.AbstractExtension;
import org.eclipse.jetty.websocket.common.frames.DataFrame;
/**
* Fragment Extension
@ -67,7 +66,6 @@ public class FragmentExtension extends AbstractExtension
int length = frame.getPayloadLength();
byte opcode = frame.getType().getOpCode(); // original opcode
ByteBuffer payload = frame.getPayload().slice();
int originalLimit = payload.limit();
int currentPosition = payload.position();
@ -84,10 +82,8 @@ public class FragmentExtension extends AbstractExtension
// break apart payload based on maxLength rules
while (length > maxLength)
{
WebSocketFrame frag = new WebSocketFrame(frame);
frag.setOpCode(opcode);
DataFrame frag = new DataFrame(frame,continuation);
frag.setFin(false); // always false here
frag.setContinuation(continuation);
payload.position(currentPosition);
payload.limit(Math.min(payload.position() + maxLength,originalLimit));
frag.setPayload(payload);
@ -96,16 +92,13 @@ public class FragmentExtension extends AbstractExtension
nextOutgoingFrame(frag,null);
length -= maxLength;
opcode = OpCode.CONTINUATION;
continuation = true;
currentPosition = payload.limit();
}
// write remaining
WebSocketFrame frag = new WebSocketFrame(frame);
frag.setOpCode(opcode);
DataFrame frag = new DataFrame(frame,continuation);
frag.setFin(frame.isFin()); // use original fin
frag.setContinuation(continuation);
payload.position(currentPosition);
payload.limit(originalLimit);
frag.setPayload(payload);

View File

@ -16,35 +16,40 @@
// ========================================================================
//
package org.eclipse.jetty.websocket.common.io.payload;
package org.eclipse.jetty.websocket.common.frames;
import java.nio.ByteBuffer;
import org.eclipse.jetty.util.StringUtil;
import org.eclipse.jetty.websocket.common.OpCode;
/**
* Validate UTF8 correctness for {@link OpCode#CLOSE} Reason message.
*/
public class CloseReasonValidator extends UTF8Validator implements PayloadProcessor
public class BinaryFrame extends DataFrame
{
private int statusCodeBytes = 2;
public BinaryFrame()
{
super(OpCode.BINARY);
}
public BinaryFrame(byte[] buf)
{
this();
setPayload(ByteBuffer.wrap(buf));
}
public BinaryFrame(ByteBuffer buf)
{
this();
setPayload(buf);
}
public BinaryFrame(String payload)
{
this(StringUtil.getBytes(payload));
}
@Override
public void process(ByteBuffer payload)
public Type getType()
{
if ((payload == null) || (payload.remaining() <= 2))
{
// no validation needed
return;
}
ByteBuffer copy = payload.slice();
while (statusCodeBytes > 0)
{
copy.get();
statusCodeBytes--;
}
super.process(copy);
return Type.BINARY;
}
}

View File

@ -0,0 +1,42 @@
//
// ========================================================================
// Copyright (c) 1995-2013 Mort Bay Consulting Pty. Ltd.
// ------------------------------------------------------------------------
// All rights reserved. This program and the accompanying materials
// are made available under the terms of the Eclipse Public License v1.0
// and Apache License v2.0 which accompanies this distribution.
//
// The Eclipse Public License is available at
// http://www.eclipse.org/legal/epl-v10.html
//
// The Apache License v2.0 is available at
// http://www.opensource.org/licenses/apache2.0.php
//
// You may elect to redistribute this code under either of these licenses.
// ========================================================================
//
package org.eclipse.jetty.websocket.common.frames;
import org.eclipse.jetty.util.StringUtil;
import org.eclipse.jetty.websocket.common.OpCode;
public class CloseFrame extends ControlFrame
{
public CloseFrame()
{
super(OpCode.CLOSE);
}
/**
* Truncate arbitrary reason into something that will fit into the CloseFrame limits.
*
* @param reason
* the arbitrary reason to possibly truncate.
* @return the possibly truncated reason string.
*/
public static String truncate(String reason)
{
return StringUtil.truncate(reason,(ControlFrame.MAX_CONTROL_PAYLOAD - 2));
}
}

View File

@ -0,0 +1,55 @@
//
// ========================================================================
// Copyright (c) 1995-2013 Mort Bay Consulting Pty. Ltd.
// ------------------------------------------------------------------------
// All rights reserved. This program and the accompanying materials
// are made available under the terms of the Eclipse Public License v1.0
// and Apache License v2.0 which accompanies this distribution.
//
// The Eclipse Public License is available at
// http://www.eclipse.org/legal/epl-v10.html
//
// The Apache License v2.0 is available at
// http://www.opensource.org/licenses/apache2.0.php
//
// You may elect to redistribute this code under either of these licenses.
// ========================================================================
//
package org.eclipse.jetty.websocket.common.frames;
import java.nio.ByteBuffer;
import org.eclipse.jetty.util.StringUtil;
import org.eclipse.jetty.websocket.common.OpCode;
public class ContinuationFrame extends DataFrame
{
public ContinuationFrame()
{
super(OpCode.CONTINUATION);
}
public ContinuationFrame(byte[] buf)
{
this();
setPayload(ByteBuffer.wrap(buf));
}
public ContinuationFrame(ByteBuffer buf)
{
this();
setPayload(buf);
}
public ContinuationFrame(String payload)
{
this(StringUtil.getBytes(payload));
}
@Override
public Type getType()
{
return Type.CONTINUATION;
}
}

View File

@ -0,0 +1,137 @@
//
// ========================================================================
// Copyright (c) 1995-2013 Mort Bay Consulting Pty. Ltd.
// ------------------------------------------------------------------------
// All rights reserved. This program and the accompanying materials
// are made available under the terms of the Eclipse Public License v1.0
// and Apache License v2.0 which accompanies this distribution.
//
// The Eclipse Public License is available at
// http://www.eclipse.org/legal/epl-v10.html
//
// The Apache License v2.0 is available at
// http://www.opensource.org/licenses/apache2.0.php
//
// You may elect to redistribute this code under either of these licenses.
// ========================================================================
//
package org.eclipse.jetty.websocket.common.frames;
import java.nio.ByteBuffer;
import java.util.Arrays;
import org.eclipse.jetty.websocket.api.ProtocolException;
import org.eclipse.jetty.websocket.common.WebSocketFrame;
public abstract class ControlFrame extends WebSocketFrame
{
/** Maximum size of Control frame, per RFC 6455 */
public static final int MAX_CONTROL_PAYLOAD = 125;
public ControlFrame(byte opcode)
{
super(opcode);
}
public void assertValid()
{
if (isControlFrame())
{
if (getPayloadLength() > ControlFrame.MAX_CONTROL_PAYLOAD)
{
throw new ProtocolException("Desired payload length [" + getPayloadLength() + "] exceeds maximum control payload length ["
+ MAX_CONTROL_PAYLOAD + "]");
}
if ((finRsvOp & 0x80) == 0)
{
throw new ProtocolException("Cannot have FIN==false on Control frames");
}
if ((finRsvOp & 0x40) != 0)
{
throw new ProtocolException("Cannot have RSV1==true on Control frames");
}
if ((finRsvOp & 0x20) != 0)
{
throw new ProtocolException("Cannot have RSV2==true on Control frames");
}
if ((finRsvOp & 0x10) != 0)
{
throw new ProtocolException("Cannot have RSV3==true on Control frames");
}
}
}
@Override
public boolean equals(Object obj)
{
if (this == obj)
{
return true;
}
if (obj == null)
{
return false;
}
if (getClass() != obj.getClass())
{
return false;
}
ControlFrame other = (ControlFrame)obj;
if (data == null)
{
if (other.data != null)
{
return false;
}
}
else if (!data.equals(other.data))
{
return false;
}
if (finRsvOp != other.finRsvOp)
{
return false;
}
if (!Arrays.equals(mask,other.mask))
{
return false;
}
if (masked != other.masked)
{
return false;
}
return true;
}
public boolean isControlFrame()
{
return true;
}
@Override
public boolean isDataFrame()
{
return false;
}
@Override
public WebSocketFrame setPayload(ByteBuffer buf)
{
if (buf == null)
{
data = null;
return this;
}
if (buf.remaining() > ControlFrame.MAX_CONTROL_PAYLOAD)
{
throw new ProtocolException("Control Payloads can not exceed 125 bytes in length.");
}
return super.setPayload(buf);
}
}

View File

@ -0,0 +1,85 @@
//
// ========================================================================
// Copyright (c) 1995-2013 Mort Bay Consulting Pty. Ltd.
// ------------------------------------------------------------------------
// All rights reserved. This program and the accompanying materials
// are made available under the terms of the Eclipse Public License v1.0
// and Apache License v2.0 which accompanies this distribution.
//
// The Eclipse Public License is available at
// http://www.eclipse.org/legal/epl-v10.html
//
// The Apache License v2.0 is available at
// http://www.opensource.org/licenses/apache2.0.php
//
// You may elect to redistribute this code under either of these licenses.
// ========================================================================
//
package org.eclipse.jetty.websocket.common.frames;
import org.eclipse.jetty.websocket.api.extensions.Frame;
import org.eclipse.jetty.websocket.common.OpCode;
import org.eclipse.jetty.websocket.common.WebSocketFrame;
/**
* A Data Frame
*/
public class DataFrame extends WebSocketFrame
{
protected DataFrame(byte opcode)
{
super(opcode);
}
/**
* Construct new DataFrame based on headers of provided frame.
* <p>
* Useful for when working in extensions and a new frame needs to be created.
*/
public DataFrame(Frame basedOn)
{
this(basedOn,false);
}
/**
* Construct new DataFrame based on headers of provided frame, overriding for continuations if needed.
* <p>
* Useful for when working in extensions and a new frame needs to be created.
*/
public DataFrame(Frame basedOn, boolean continuation)
{
super(basedOn.getOpCode());
copyHeaders(basedOn);
if (continuation)
{
setOpCode(OpCode.CONTINUATION);
}
}
@Override
public void assertValid()
{
/* no extra validation for data frames (yet) here */
}
/**
* Set the data frame to continuation mode
*/
public void setIsContinuation()
{
setOpCode(OpCode.CONTINUATION);
}
@Override
public boolean isControlFrame()
{
return false;
}
@Override
public boolean isDataFrame()
{
return true;
}
}

View File

@ -16,28 +16,40 @@
// ========================================================================
//
package org.eclipse.jetty.websocket.common.io.payload;
package org.eclipse.jetty.websocket.common.frames;
import java.nio.ByteBuffer;
import org.eclipse.jetty.websocket.api.extensions.Frame;
import org.eclipse.jetty.util.StringUtil;
import org.eclipse.jetty.websocket.common.OpCode;
/**
* payload validator does no validation.
*/
public class NoOpValidator implements PayloadProcessor
public class PingFrame extends ControlFrame
{
public static final NoOpValidator INSTANCE = new NoOpValidator();
public PingFrame()
{
super(OpCode.PING);
}
@Override
public void process(ByteBuffer payload)
public PingFrame(ByteBuffer buf)
{
/* all payloads are valid in this case */
this();
setPayload(buf);
}
public PingFrame(byte[] bytes)
{
this();
setPayload(ByteBuffer.wrap(bytes));
}
public PingFrame(String payload)
{
this(StringUtil.getUtf8Bytes(payload));
}
@Override
public void reset(Frame frame)
public Type getType()
{
/* do nothing */
return Type.PING;
}
}

View File

@ -0,0 +1,55 @@
//
// ========================================================================
// Copyright (c) 1995-2013 Mort Bay Consulting Pty. Ltd.
// ------------------------------------------------------------------------
// All rights reserved. This program and the accompanying materials
// are made available under the terms of the Eclipse Public License v1.0
// and Apache License v2.0 which accompanies this distribution.
//
// The Eclipse Public License is available at
// http://www.eclipse.org/legal/epl-v10.html
//
// The Apache License v2.0 is available at
// http://www.opensource.org/licenses/apache2.0.php
//
// You may elect to redistribute this code under either of these licenses.
// ========================================================================
//
package org.eclipse.jetty.websocket.common.frames;
import java.nio.ByteBuffer;
import org.eclipse.jetty.util.StringUtil;
import org.eclipse.jetty.websocket.common.OpCode;
public class PongFrame extends ControlFrame
{
public PongFrame()
{
super(OpCode.PONG);
}
public PongFrame(ByteBuffer buf)
{
this();
setPayload(buf);
}
public PongFrame(byte[] bytes)
{
this();
setPayload(ByteBuffer.wrap(bytes));
}
public PongFrame(String payload)
{
this(StringUtil.getUtf8Bytes(payload));
}
@Override
public Type getType()
{
return Type.PONG;
}
}

View File

@ -0,0 +1,61 @@
//
// ========================================================================
// Copyright (c) 1995-2013 Mort Bay Consulting Pty. Ltd.
// ------------------------------------------------------------------------
// All rights reserved. This program and the accompanying materials
// are made available under the terms of the Eclipse Public License v1.0
// and Apache License v2.0 which accompanies this distribution.
//
// The Eclipse Public License is available at
// http://www.eclipse.org/legal/epl-v10.html
//
// The Apache License v2.0 is available at
// http://www.opensource.org/licenses/apache2.0.php
//
// You may elect to redistribute this code under either of these licenses.
// ========================================================================
//
package org.eclipse.jetty.websocket.common.frames;
import java.nio.ByteBuffer;
import org.eclipse.jetty.util.BufferUtil;
import org.eclipse.jetty.util.StringUtil;
import org.eclipse.jetty.websocket.common.OpCode;
import org.eclipse.jetty.websocket.common.WebSocketFrame;
public class TextFrame extends DataFrame
{
public TextFrame()
{
super(OpCode.TEXT);
}
public TextFrame(String msg)
{
this();
setPayload(msg);
}
@Override
public Type getType()
{
return Type.TEXT;
}
public WebSocketFrame setPayload(String str)
{
setPayload(ByteBuffer.wrap(StringUtil.getUtf8Bytes(str)));
return this;
}
public String getPayloadAsUTF8()
{
if (data == null)
{
return null;
}
return BufferUtil.toUTF8String(data);
}
}

View File

@ -1,111 +0,0 @@
//
// ========================================================================
// Copyright (c) 1995-2013 Mort Bay Consulting Pty. Ltd.
// ------------------------------------------------------------------------
// All rights reserved. This program and the accompanying materials
// are made available under the terms of the Eclipse Public License v1.0
// and Apache License v2.0 which accompanies this distribution.
//
// The Eclipse Public License is available at
// http://www.eclipse.org/legal/epl-v10.html
//
// The Apache License v2.0 is available at
// http://www.opensource.org/licenses/apache2.0.php
//
// You may elect to redistribute this code under either of these licenses.
// ========================================================================
//
package org.eclipse.jetty.websocket.common.io.payload;
import java.io.IOException;
import java.nio.ByteBuffer;
import org.eclipse.jetty.util.BufferUtil;
import org.eclipse.jetty.util.Utf8Appendable;
import org.eclipse.jetty.util.log.Log;
import org.eclipse.jetty.util.log.Logger;
import org.eclipse.jetty.websocket.api.BadPayloadException;
import org.eclipse.jetty.websocket.api.extensions.Frame;
/**
* Used to perform validation of UTF8 payload contents (for fast-fail reasons)
*/
public class UTF8Validator extends Utf8Appendable implements PayloadProcessor
{
private static class EmptyAppender implements Appendable
{
private int length = 0;
@Override
public Appendable append(char c) throws IOException
{
length++;
return this;
}
@Override
public Appendable append(CharSequence csq) throws IOException
{
length += csq.length();
return this;
}
@Override
public Appendable append(CharSequence csq, int start, int end) throws IOException
{
length += (end - start);
return this;
}
public int getLength()
{
return length;
}
}
private static final Logger LOG = Log.getLogger(UTF8Validator.class);
private EmptyAppender buffer;
public UTF8Validator()
{
super(new EmptyAppender());
this.buffer = (EmptyAppender)_appendable;
}
@Override
public int length()
{
return this.buffer.getLength();
}
@Override
public void process(ByteBuffer payload)
{
if (LOG.isDebugEnabled())
{
LOG.debug("Payload: {}",BufferUtil.toDetailString(payload));
}
if ((payload == null) || (payload.remaining() <= 0))
{
return;
}
try
{
append(payload.slice());
}
catch (NotUtf8Exception e)
{
throw new BadPayloadException(e);
}
}
@Override
public void reset(Frame frame)
{
/* do nothing */
}
}

View File

@ -29,9 +29,8 @@ import org.eclipse.jetty.util.log.Log;
import org.eclipse.jetty.util.log.Logger;
import org.eclipse.jetty.websocket.api.WriteCallback;
import org.eclipse.jetty.websocket.api.extensions.OutgoingFrames;
import org.eclipse.jetty.websocket.common.OpCode;
import org.eclipse.jetty.websocket.common.WebSocketFrame;
import org.eclipse.jetty.websocket.common.WebSocketSession;
import org.eclipse.jetty.websocket.common.frames.BinaryFrame;
import org.eclipse.jetty.websocket.common.io.FutureWriteCallback;
/**
@ -43,7 +42,7 @@ public class MessageOutputStream extends OutputStream
private final OutgoingFrames outgoing;
private final ByteBufferPool bufferPool;
private long frameCount = 0;
private WebSocketFrame frame;
private BinaryFrame frame;
private ByteBuffer buffer;
private FutureWriteCallback blocker;
private WriteCallback callback;
@ -55,7 +54,7 @@ public class MessageOutputStream extends OutputStream
this.bufferPool = bufferPool;
this.buffer = bufferPool.acquire(bufferSize,true);
BufferUtil.flipToFill(buffer);
this.frame = new WebSocketFrame(OpCode.BINARY);
this.frame = new BinaryFrame();
}
public MessageOutputStream(WebSocketSession session)
@ -146,7 +145,7 @@ public class MessageOutputStream extends OutputStream
blocker.get();
// block success
frameCount++;
frame.setOpCode(OpCode.CONTINUATION);
frame.setIsContinuation();
}
catch (ExecutionException e)
{

View File

@ -29,9 +29,8 @@ import org.eclipse.jetty.util.log.Log;
import org.eclipse.jetty.util.log.Logger;
import org.eclipse.jetty.websocket.api.WriteCallback;
import org.eclipse.jetty.websocket.api.extensions.OutgoingFrames;
import org.eclipse.jetty.websocket.common.OpCode;
import org.eclipse.jetty.websocket.common.WebSocketFrame;
import org.eclipse.jetty.websocket.common.WebSocketSession;
import org.eclipse.jetty.websocket.common.frames.TextFrame;
import org.eclipse.jetty.websocket.common.io.FutureWriteCallback;
/**
@ -45,7 +44,7 @@ public class MessageWriter extends Writer
private final OutgoingFrames outgoing;
private final ByteBufferPool bufferPool;
private long frameCount = 0;
private WebSocketFrame frame;
private TextFrame frame;
private ByteBuffer buffer;
private Utf8CharBuffer utf;
private FutureWriteCallback blocker;
@ -59,7 +58,7 @@ public class MessageWriter extends Writer
this.buffer = bufferPool.acquire(bufferSize,true);
BufferUtil.flipToFill(buffer);
this.utf = Utf8CharBuffer.wrap(buffer);
this.frame = new WebSocketFrame(OpCode.TEXT);
this.frame = new TextFrame();
}
public MessageWriter(WebSocketSession session)
@ -129,7 +128,7 @@ public class MessageWriter extends Writer
// clear utf buffer
utf.clear();
frameCount++;
frame.setOpCode(OpCode.CONTINUATION);
frame.setIsContinuation();
}
catch (ExecutionException e)
{

View File

@ -27,6 +27,7 @@ import org.eclipse.jetty.io.ByteBufferPool;
import org.eclipse.jetty.io.MappedByteBufferPool;
import org.eclipse.jetty.util.BufferUtil;
import org.eclipse.jetty.websocket.api.WebSocketPolicy;
import org.eclipse.jetty.websocket.common.frames.TextFrame;
import org.junit.Assert;
import org.junit.Test;
@ -68,7 +69,7 @@ public class GeneratorParserRoundtripTest
capture.assertNoErrors();
capture.assertHasFrame(OpCode.TEXT,1);
WebSocketFrame txt = capture.getFrames().get(0);
TextFrame txt = (TextFrame)capture.getFrames().get(0);
Assert.assertThat("Text parsed",txt.getPayloadAsUTF8(),is(message));
}
@ -115,7 +116,7 @@ public class GeneratorParserRoundtripTest
capture.assertNoErrors();
capture.assertHasFrame(OpCode.TEXT,1);
WebSocketFrame txt = capture.getFrames().get(0);
TextFrame txt = (TextFrame)capture.getFrames().get(0);
Assert.assertTrue("Text.isMasked",txt.isMasked());
Assert.assertThat("Text parsed",txt.getPayloadAsUTF8(),is(message));
}

View File

@ -30,6 +30,8 @@ import org.eclipse.jetty.util.log.Logger;
import org.eclipse.jetty.websocket.api.StatusCode;
import org.eclipse.jetty.websocket.api.WebSocketPolicy;
import org.eclipse.jetty.websocket.api.extensions.Frame;
import org.eclipse.jetty.websocket.common.frames.CloseFrame;
import org.eclipse.jetty.websocket.common.frames.PingFrame;
import org.junit.Assert;
import org.junit.Test;
@ -131,7 +133,7 @@ public class GeneratorTest
public void testClose_Empty()
{
// 0 byte payload (no status code)
assertGeneratedBytes("8800",new WebSocketFrame(OpCode.CLOSE));
assertGeneratedBytes("8800",new CloseFrame());
}
@Test
@ -212,8 +214,7 @@ public class GeneratorTest
WebSocketFrame[] frames = new WebSocketFrame[pingCount + 1];
for (int i = 0; i < pingCount; i++)
{
String payload = String.format("ping-%d",i);
frames[i] = WebSocketFrame.ping().setPayload(payload);
frames[i] = new PingFrame(String.format("ping-%d",i));
}
frames[pingCount] = new CloseInfo(StatusCode.NORMAL).asFrame();

View File

@ -56,6 +56,11 @@ public final class Hex
return buf;
}
public static ByteBuffer asByteBuffer(String hstr)
{
return ByteBuffer.wrap(asByteArray(hstr));
}
public static String asHex(byte buf[])
{
int len = buf.length;

View File

@ -59,17 +59,18 @@ public class IncomingFramesCapture implements IncomingFrames
public void assertHasFrame(byte op, int expectedCount)
{
Assert.assertThat(OpCode.name(op),getFrameCount(op),is(expectedCount));
String msg = String.format("%s frame count",OpCode.name(op));
Assert.assertThat(msg,getFrameCount(op),is(expectedCount));
}
public void assertHasNoFrames()
{
Assert.assertThat("Has no frames",frames.size(),is(0));
Assert.assertThat("Frame count",frames.size(),is(0));
}
public void assertNoErrors()
{
Assert.assertThat("Has no errors",errors.size(),is(0));
Assert.assertThat("Error count",errors.size(),is(0));
}
public void dump()
@ -129,8 +130,7 @@ public class IncomingFramesCapture implements IncomingFrames
@Override
public void incomingFrame(Frame frame)
{
WebSocketFrame copy = new WebSocketFrame(frame);
frames.add(copy);
frames.add(WebSocketFrame.copy(frame));
}
public int size()

View File

@ -84,8 +84,7 @@ public class OutgoingFramesCapture implements OutgoingFrames
@Override
public void outgoingFrame(Frame frame, WriteCallback callback)
{
WebSocketFrame copy = new WebSocketFrame(frame);
frames.add(copy);
frames.add(WebSocketFrame.copy(frame));
if (callback != null)
{
callback.writeSuccess();

View File

@ -24,33 +24,20 @@ import java.nio.ByteBuffer;
import java.util.ArrayList;
import java.util.List;
import org.eclipse.jetty.util.BufferUtil;
import org.eclipse.jetty.util.StringUtil;
import org.eclipse.jetty.util.TypeUtil;
import org.eclipse.jetty.util.log.Log;
import org.eclipse.jetty.util.log.Logger;
import org.eclipse.jetty.websocket.api.BadPayloadException;
import org.eclipse.jetty.websocket.api.StatusCode;
import org.eclipse.jetty.websocket.api.WebSocketBehavior;
import org.eclipse.jetty.websocket.api.WebSocketPolicy;
import org.eclipse.jetty.websocket.common.frames.ContinuationFrame;
import org.eclipse.jetty.websocket.common.frames.DataFrame;
import org.eclipse.jetty.websocket.common.frames.PingFrame;
import org.eclipse.jetty.websocket.common.frames.PongFrame;
import org.eclipse.jetty.websocket.common.frames.TextFrame;
import org.junit.Assert;
import org.junit.Test;
public class ParserTest
{
private static final Logger LOG = Log.getLogger(ParserTest.class);
/** Parse, but be quiet about stack traces */
private void parseQuietly(UnitParser parser, ByteBuffer buf)
{
LogShush.disableStacks(Parser.class);
try {
parser.parse(buf);
} finally {
LogShush.enableStacks(Parser.class);
}
}
/**
* Similar to the server side 5.15 testcase. A normal 2 fragment text text message, followed by another continuation.
*/
@ -58,10 +45,10 @@ public class ParserTest
public void testParseCase5_15()
{
List<WebSocketFrame> send = new ArrayList<>();
send.add(new WebSocketFrame(OpCode.TEXT).setPayload("fragment1").setFin(false));
send.add(new WebSocketFrame(OpCode.CONTINUATION).setPayload("fragment2").setFin(true));
send.add(new WebSocketFrame(OpCode.CONTINUATION).setPayload("fragment3").setFin(false)); // bad frame
send.add(new WebSocketFrame(OpCode.TEXT).setPayload("fragment4").setFin(true));
send.add(new TextFrame("fragment1").setFin(false));
send.add(new ContinuationFrame("fragment2").setFin(true));
send.add(new ContinuationFrame("fragment3").setFin(false)); // bad frame
send.add(new TextFrame("fragment4").setFin(true));
send.add(new CloseInfo(StatusCode.NORMAL).asFrame());
ByteBuffer completeBuf = UnitGenerator.generate(send);
@ -69,10 +56,11 @@ public class ParserTest
IncomingFramesCapture capture = new IncomingFramesCapture();
parser.setIncomingFramesHandler(capture);
parseQuietly(parser,completeBuf);
parser.parseQuietly(completeBuf);
capture.assertErrorCount(1);
capture.assertHasFrame(OpCode.TEXT,2);
capture.assertHasFrame(OpCode.TEXT,1);
capture.assertHasFrame(OpCode.CONTINUATION,1);
}
/**
@ -82,45 +70,45 @@ public class ParserTest
public void testParseCase5_18()
{
List<WebSocketFrame> send = new ArrayList<>();
send.add(new WebSocketFrame(OpCode.TEXT).setPayload("fragment1").setFin(false));
send.add(new WebSocketFrame(OpCode.TEXT).setPayload("fragment2").setFin(true)); // bad frame, must be continuation
send.add(new TextFrame("fragment1").setFin(false));
send.add(new TextFrame("fragment2").setFin(true)); // bad frame, must be continuation
send.add(new CloseInfo(StatusCode.NORMAL).asFrame());
ByteBuffer completeBuf = UnitGenerator.generate(send);
UnitParser parser = new UnitParser();
IncomingFramesCapture capture = new IncomingFramesCapture();
parser.setIncomingFramesHandler(capture);
parseQuietly(parser,completeBuf);
parser.parseQuietly(completeBuf);
capture.assertErrorCount(1);
capture.assertHasFrame(OpCode.TEXT,1); // fragment 1
}
/**
* Similar to the server side 5.19 testcase.
* text message, send in 5 frames/fragments, with 2 pings in the mix.
* Similar to the server side 5.19 testcase. text message, send in 5 frames/fragments, with 2 pings in the mix.
*/
@Test
public void testParseCase5_19()
{
List<WebSocketFrame> send = new ArrayList<>();
send.add(new WebSocketFrame(OpCode.TEXT).setPayload("f1").setFin(false));
send.add(new WebSocketFrame(OpCode.CONTINUATION).setPayload(",f2").setFin(false));
send.add(new WebSocketFrame(OpCode.PING).setPayload("pong-1"));
send.add(new WebSocketFrame(OpCode.CONTINUATION).setPayload(",f3").setFin(false));
send.add(new WebSocketFrame(OpCode.CONTINUATION).setPayload(",f4").setFin(false));
send.add(new WebSocketFrame(OpCode.PING).setPayload("pong-2"));
send.add(new WebSocketFrame(OpCode.CONTINUATION).setPayload(",f5").setFin(true));
send.add(new TextFrame("f1").setFin(false));
send.add(new ContinuationFrame(",f2").setFin(false));
send.add(new PingFrame("pong-1"));
send.add(new ContinuationFrame(",f3").setFin(false));
send.add(new ContinuationFrame(",f4").setFin(false));
send.add(new PingFrame("pong-2"));
send.add(new ContinuationFrame(",f5").setFin(true));
send.add(new CloseInfo(StatusCode.NORMAL).asFrame());
ByteBuffer completeBuf = UnitGenerator.generate(send);
UnitParser parser = new UnitParser();
IncomingFramesCapture capture = new IncomingFramesCapture();
parser.setIncomingFramesHandler(capture);
parseQuietly(parser,completeBuf);
parser.parseQuietly(completeBuf);
capture.assertErrorCount(0);
capture.assertHasFrame(OpCode.TEXT,5);
capture.assertHasFrame(OpCode.TEXT,1);
capture.assertHasFrame(OpCode.CONTINUATION,4);
capture.assertHasFrame(OpCode.CLOSE,1);
capture.assertHasFrame(OpCode.PING,2);
}
@ -132,8 +120,8 @@ public class ParserTest
public void testParseCase5_6()
{
List<WebSocketFrame> send = new ArrayList<>();
send.add(WebSocketFrame.pong().setPayload("ping"));
send.add(WebSocketFrame.text("hello, world"));
send.add(new PongFrame("ping"));
send.add(new TextFrame("hello, world"));
send.add(new CloseInfo(StatusCode.NORMAL).asFrame());
ByteBuffer completeBuf = UnitGenerator.generate(send);
@ -158,19 +146,31 @@ public class ParserTest
byte msg[] = StringUtil.getUtf8Bytes(utf8);
List<WebSocketFrame> send = new ArrayList<>();
int textCount = 0;
int continuationCount = 0;
int len = msg.length;
byte opcode = OpCode.TEXT;
boolean continuation = false;
byte mini[];
for (int i = 0; i < len; i++)
{
WebSocketFrame frame = new WebSocketFrame(opcode);
DataFrame frame = null;
if (continuation)
{
frame = new ContinuationFrame();
continuationCount++;
}
else
{
frame = new TextFrame();
textCount++;
}
mini = new byte[1];
mini[0] = msg[i];
frame.setPayload(mini);
frame.setPayload(ByteBuffer.wrap(mini));
boolean isLast = (i >= (len - 1));
frame.setFin(isLast);
send.add(frame);
opcode = OpCode.CONTINUATION;
continuation = true;
}
send.add(new CloseInfo(StatusCode.NORMAL).asFrame());
@ -181,57 +181,11 @@ public class ParserTest
parser.parse(completeBuf);
capture.assertErrorCount(0);
capture.assertHasFrame(OpCode.TEXT,len);
capture.assertHasFrame(OpCode.TEXT,textCount);
capture.assertHasFrame(OpCode.CONTINUATION,continuationCount);
capture.assertHasFrame(OpCode.CLOSE,1);
}
/**
* Similar to the server side 6.4.3 testcase.
*/
@Test
public void testParseCase6_4_3()
{
ByteBuffer payload = ByteBuffer.allocate(64);
BufferUtil.clearToFill(payload);
payload.put(TypeUtil.fromHexString("cebae1bdb9cf83cebcceb5")); // good
payload.put(TypeUtil.fromHexString("f4908080")); // INVALID
payload.put(TypeUtil.fromHexString("656469746564")); // good
BufferUtil.flipToFlush(payload,0);
WebSocketFrame text = new WebSocketFrame();
text.setMask(TypeUtil.fromHexString("11223344"));
text.setPayload(payload);
text.setOpCode(OpCode.TEXT);
ByteBuffer buf = UnitGenerator.generate(text);
ByteBuffer part1 = ByteBuffer.allocate(17); // header + good
ByteBuffer part2 = ByteBuffer.allocate(4); // invalid
ByteBuffer part3 = ByteBuffer.allocate(10); // the rest (all good utf)
BufferUtil.put(buf,part1);
BufferUtil.put(buf,part2);
BufferUtil.put(buf,part3);
BufferUtil.flipToFlush(part1,0);
BufferUtil.flipToFlush(part2,0);
BufferUtil.flipToFlush(part3,0);
LOG.debug("Part1: {}",BufferUtil.toDetailString(part1));
LOG.debug("Part2: {}",BufferUtil.toDetailString(part2));
LOG.debug("Part3: {}",BufferUtil.toDetailString(part3));
UnitParser parser = new UnitParser();
IncomingFramesCapture capture = new IncomingFramesCapture();
parser.setIncomingFramesHandler(capture);
parseQuietly(parser,part1);
capture.assertErrorCount(0);
parseQuietly(parser,part2);
capture.assertErrorCount(1);
capture.assertHasErrors(BadPayloadException.class,1);
}
@Test
public void testParseNothing()
{

View File

@ -25,6 +25,7 @@ import java.nio.ByteBuffer;
import org.eclipse.jetty.util.BufferUtil;
import org.eclipse.jetty.websocket.api.WebSocketBehavior;
import org.eclipse.jetty.websocket.api.WebSocketPolicy;
import org.eclipse.jetty.websocket.common.frames.PingFrame;
import org.junit.Assert;
import org.junit.Test;
@ -47,8 +48,9 @@ public class PingPayloadParserTest
capture.assertNoErrors();
capture.assertHasFrame(OpCode.PING,1);
WebSocketFrame ping = capture.getFrames().get(0);
PingFrame ping = (PingFrame)capture.getFrames().get(0);
Assert.assertThat("PingFrame.payload",ping.getPayloadAsUTF8(),is("Hello"));
String actual = BufferUtil.toUTF8String(ping.getPayload());
Assert.assertThat("PingFrame.payload",actual,is("Hello"));
}
}

View File

@ -21,6 +21,11 @@ package org.eclipse.jetty.websocket.common;
import java.nio.ByteBuffer;
import java.util.Arrays;
import org.eclipse.jetty.websocket.common.frames.BinaryFrame;
import org.eclipse.jetty.websocket.common.frames.ContinuationFrame;
import org.eclipse.jetty.websocket.common.frames.PingFrame;
import org.eclipse.jetty.websocket.common.frames.PongFrame;
import org.eclipse.jetty.websocket.common.frames.TextFrame;
import org.junit.Test;
public class RFC6455ExamplesGeneratorTest
@ -30,8 +35,8 @@ public class RFC6455ExamplesGeneratorTest
@Test
public void testFragmentedUnmaskedTextMessage()
{
WebSocketFrame text1 = WebSocketFrame.text("Hel").setFin(false);
WebSocketFrame text2 = new WebSocketFrame(OpCode.CONTINUATION).setPayload("lo");
WebSocketFrame text1 = new TextFrame("Hel").setFin(false);
WebSocketFrame text2 = new ContinuationFrame("lo");
ByteBuffer actual1 = UnitGenerator.generate(text1);
ByteBuffer actual2 = UnitGenerator.generate(text2);
@ -56,8 +61,7 @@ public class RFC6455ExamplesGeneratorTest
@Test
public void testSingleMaskedPongRequest()
{
WebSocketFrame pong = new WebSocketFrame(OpCode.PONG);
pong.setPayload("Hello");
PongFrame pong = new PongFrame("Hello");
pong.setMask(new byte[]
{ 0x37, (byte)0xfa, 0x21, 0x3d });
@ -97,10 +101,10 @@ public class RFC6455ExamplesGeneratorTest
{
int dataSize = 256;
WebSocketFrame binary = WebSocketFrame.binary();
BinaryFrame binary = new BinaryFrame();
byte payload[] = new byte[dataSize];
Arrays.fill(payload,(byte)0x44);
binary.setPayload(payload);
binary.setPayload(ByteBuffer.wrap(payload));
ByteBuffer actual = UnitGenerator.generate(binary);
@ -126,10 +130,10 @@ public class RFC6455ExamplesGeneratorTest
{
int dataSize = 1024 * 64;
WebSocketFrame binary = WebSocketFrame.binary();
BinaryFrame binary = new BinaryFrame();
byte payload[] = new byte[dataSize];
Arrays.fill(payload,(byte)0x44);
binary.setPayload(payload);
binary.setPayload(ByteBuffer.wrap(payload));
ByteBuffer actual = UnitGenerator.generate(binary);
@ -154,7 +158,7 @@ public class RFC6455ExamplesGeneratorTest
@Test
public void testSingleUnmaskedPingRequest() throws Exception
{
WebSocketFrame ping = new WebSocketFrame(OpCode.PING).setPayload("Hello");
PingFrame ping = new PingFrame("Hello");
ByteBuffer actual = UnitGenerator.generate(ping);

View File

@ -64,12 +64,15 @@ public class RFC6455ExamplesParserTest
parser.parse(buf);
capture.assertNoErrors();
capture.assertHasFrame(OpCode.TEXT,2);
capture.assertHasFrame(OpCode.TEXT,1);
capture.assertHasFrame(OpCode.CONTINUATION,1);
WebSocketFrame txt = capture.getFrames().get(0);
Assert.assertThat("TextFrame[0].data",txt.getPayloadAsUTF8(),is("Hel"));
String actual = BufferUtil.toUTF8String(txt.getPayload());
Assert.assertThat("TextFrame[0].data",actual,is("Hel"));
txt = capture.getFrames().get(1);
Assert.assertThat("TextFrame[1].data",txt.getPayloadAsUTF8(),is("lo"));
actual = BufferUtil.toUTF8String(txt.getPayload());
Assert.assertThat("TextFrame[1].data",actual,is("lo"));
}
@Test
@ -92,7 +95,8 @@ public class RFC6455ExamplesParserTest
capture.assertHasFrame(OpCode.PONG,1);
WebSocketFrame pong = capture.getFrames().get(0);
Assert.assertThat("PongFrame.payload",pong.getPayloadAsUTF8(),is("Hello"));
String actual = BufferUtil.toUTF8String(pong.getPayload());
Assert.assertThat("PongFrame.payload",actual,is("Hello"));
}
@Test
@ -115,7 +119,8 @@ public class RFC6455ExamplesParserTest
capture.assertHasFrame(OpCode.TEXT,1);
WebSocketFrame txt = capture.getFrames().get(0);
Assert.assertThat("TextFrame.payload",txt.getPayloadAsUTF8(),is("Hello"));
String actual = BufferUtil.toUTF8String(txt.getPayload());
Assert.assertThat("TextFrame.payload",actual,is("Hello"));
}
@Test
@ -215,7 +220,8 @@ public class RFC6455ExamplesParserTest
capture.assertHasFrame(OpCode.PING,1);
WebSocketFrame ping = capture.getFrames().get(0);
Assert.assertThat("PingFrame.payload",ping.getPayloadAsUTF8(),is("Hello"));
String actual = BufferUtil.toUTF8String(ping.getPayload());
Assert.assertThat("PingFrame.payload",actual,is("Hello"));
}
@Test
@ -238,6 +244,7 @@ public class RFC6455ExamplesParserTest
capture.assertHasFrame(OpCode.TEXT,1);
WebSocketFrame txt = capture.getFrames().get(0);
Assert.assertThat("TextFrame.payload",txt.getPayloadAsUTF8(),is("Hello"));
String actual = BufferUtil.toUTF8String(txt.getPayload());
Assert.assertThat("TextFrame.payload",actual,is("Hello"));
}
}

View File

@ -169,7 +169,8 @@ public class TextPayloadParserTest
parser.parse(buf);
capture.assertNoErrors();
capture.assertHasFrame(OpCode.TEXT,2);
capture.assertHasFrame(OpCode.TEXT,1);
capture.assertHasFrame(OpCode.CONTINUATION,1);
WebSocketFrame txt = capture.getFrames().get(0);
Assert.assertThat("TextFrame[0].data",txt.getPayloadAsUTF8(),is(part1));
txt = capture.getFrames().get(1);

View File

@ -22,6 +22,7 @@ import java.nio.ByteBuffer;
import org.eclipse.jetty.io.ByteBufferPool;
import org.eclipse.jetty.io.MappedByteBufferPool;
import org.eclipse.jetty.util.log.StacklessLogging;
import org.eclipse.jetty.websocket.api.WebSocketPolicy;
public class UnitParser extends Parser
@ -56,14 +57,13 @@ public class UnitParser extends Parser
*/
public void parseQuietly(ByteBuffer buf)
{
try
try (StacklessLogging suppress = new StacklessLogging(Parser.class))
{
LogShush.disableStacks(Parser.class);
parse(buf);
}
finally
catch (Exception ignore)
{
LogShush.enableStacks(Parser.class);
/* ignore */
}
}

View File

@ -28,6 +28,9 @@ import org.eclipse.jetty.util.BufferUtil;
import org.eclipse.jetty.websocket.api.StatusCode;
import org.eclipse.jetty.websocket.api.WebSocketPolicy;
import org.eclipse.jetty.websocket.api.extensions.Frame;
import org.eclipse.jetty.websocket.common.frames.CloseFrame;
import org.eclipse.jetty.websocket.common.frames.PingFrame;
import org.eclipse.jetty.websocket.common.frames.TextFrame;
import org.junit.Assert;
import org.junit.BeforeClass;
import org.junit.Test;
@ -54,13 +57,6 @@ public class WebSocketFrameTest
laxGenerator = new Generator(policy,bufferPool,false);
}
private void assertEqual(String message, ByteBuffer expected, ByteBuffer actual)
{
BufferUtil.flipToFlush(expected,0);
ByteBufferAssert.assertEquals(message,expected,actual);
}
private void assertFrameHex(String message, String expectedHex, ByteBuffer actual)
{
String actualHex = Hex.asHex(actual);
@ -70,7 +66,7 @@ public class WebSocketFrameTest
@Test
public void testLaxInvalidClose()
{
WebSocketFrame frame = new WebSocketFrame(OpCode.CLOSE).setFin(false);
WebSocketFrame frame = new CloseFrame().setFin(false);
ByteBuffer actual = generateWholeFrame(laxGenerator,frame);
String expected = "0800";
assertFrameHex("Lax Invalid Close Frame",expected,actual);
@ -79,7 +75,7 @@ public class WebSocketFrameTest
@Test
public void testLaxInvalidPing()
{
WebSocketFrame frame = new WebSocketFrame(OpCode.PING).setFin(false);
WebSocketFrame frame = new PingFrame().setFin(false);
ByteBuffer actual = generateWholeFrame(laxGenerator,frame);
String expected = "0900";
assertFrameHex("Lax Invalid Ping Frame",expected,actual);
@ -97,7 +93,7 @@ public class WebSocketFrameTest
@Test
public void testStrictValidPing()
{
WebSocketFrame frame = new WebSocketFrame(OpCode.PING);
WebSocketFrame frame = new PingFrame();
ByteBuffer actual = generateWholeFrame(strictGenerator,frame);
String expected = "8900";
assertFrameHex("Strict Valid Ping Frame",expected,actual);
@ -106,7 +102,7 @@ public class WebSocketFrameTest
@Test
public void testRsv1()
{
WebSocketFrame frame = new WebSocketFrame(OpCode.TEXT);
TextFrame frame = new TextFrame();
frame.setPayload("Hi");
frame.setRsv1(true);
ByteBuffer actual = generateWholeFrame(laxGenerator,frame);
@ -117,7 +113,7 @@ public class WebSocketFrameTest
@Test
public void testRsv2()
{
WebSocketFrame frame = new WebSocketFrame(OpCode.TEXT);
TextFrame frame = new TextFrame();
frame.setPayload("Hi");
frame.setRsv2(true);
ByteBuffer actual = generateWholeFrame(laxGenerator,frame);
@ -128,7 +124,7 @@ public class WebSocketFrameTest
@Test
public void testRsv3()
{
WebSocketFrame frame = new WebSocketFrame(OpCode.TEXT);
TextFrame frame = new TextFrame();
frame.setPayload("Hi");
frame.setRsv3(true);
ByteBuffer actual = generateWholeFrame(laxGenerator,frame);

View File

@ -23,6 +23,7 @@ import static org.hamcrest.Matchers.*;
import java.nio.ByteBuffer;
import java.util.Arrays;
import org.eclipse.jetty.util.StringUtil;
import org.eclipse.jetty.websocket.api.ProtocolException;
import org.eclipse.jetty.websocket.api.WebSocketBehavior;
import org.eclipse.jetty.websocket.api.WebSocketException;
@ -36,6 +37,7 @@ import org.eclipse.jetty.websocket.common.Parser;
import org.eclipse.jetty.websocket.common.UnitGenerator;
import org.eclipse.jetty.websocket.common.UnitParser;
import org.eclipse.jetty.websocket.common.WebSocketFrame;
import org.eclipse.jetty.websocket.common.frames.PingFrame;
import org.junit.Assert;
import org.junit.Test;
@ -53,7 +55,7 @@ public class TestABCase2
bytes[i] = Integer.valueOf(Integer.toOctalString(i)).byteValue();
}
WebSocketFrame pingFrame = WebSocketFrame.ping().setPayload(bytes);
PingFrame pingFrame = new PingFrame(bytes);
ByteBuffer actual = UnitGenerator.generate(pingFrame);
@ -77,7 +79,7 @@ public class TestABCase2
{
byte[] bytes = new byte[] { 0x00, 0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07, 0x08 };
WebSocketFrame pingFrame = WebSocketFrame.ping().setPayload(bytes);
PingFrame pingFrame = new PingFrame(bytes);
ByteBuffer actual = UnitGenerator.generate(pingFrame);
@ -118,9 +120,9 @@ public class TestABCase2
public void testGenerateHelloPingCase2_2()
{
String message = "Hello, world!";
byte[] messageBytes = message.getBytes();
byte[] messageBytes = StringUtil.getUtf8Bytes(message);
WebSocketFrame pingFrame = WebSocketFrame.ping().setPayload(messageBytes);
PingFrame pingFrame = new PingFrame(messageBytes);
ByteBuffer actual = UnitGenerator.generate(pingFrame);
@ -143,28 +145,22 @@ public class TestABCase2
public void testGenerateOversizedBinaryPingCase2_5_A()
{
byte[] bytes = new byte[126];
Arrays.fill(bytes,(byte)0x00);
for ( int i = 0 ; i < bytes.length ; ++i )
{
bytes[i] = 0x00;
}
WebSocketFrame.ping().setPayload(bytes);
PingFrame pingFrame = new PingFrame();
pingFrame.setPayload(ByteBuffer.wrap(bytes)); // should throw exception
}
@Test( expected=WebSocketException.class )
public void testGenerateOversizedBinaryPingCase2_5_B()
{
byte[] bytes = new byte[126];
Arrays.fill(bytes, (byte)0x00);
for ( int i = 0 ; i < bytes.length ; ++i )
{
bytes[i] = 0x00;
}
PingFrame pingFrame = new PingFrame();
pingFrame.setPayload(ByteBuffer.wrap(bytes)); // should throw exception
WebSocketFrame pingFrame = WebSocketFrame.ping().setPayload(bytes);
UnitGenerator.generate(pingFrame);
// FIXME: Remove? UnitGenerator.generate(pingFrame);
}
@Test

View File

@ -31,12 +31,13 @@ import org.eclipse.jetty.websocket.api.WebSocketPolicy;
import org.eclipse.jetty.websocket.api.extensions.Frame;
import org.eclipse.jetty.websocket.common.ByteBufferAssert;
import org.eclipse.jetty.websocket.common.CloseInfo;
import org.eclipse.jetty.websocket.common.Hex;
import org.eclipse.jetty.websocket.common.IncomingFramesCapture;
import org.eclipse.jetty.websocket.common.OpCode;
import org.eclipse.jetty.websocket.common.Parser;
import org.eclipse.jetty.websocket.common.UnitGenerator;
import org.eclipse.jetty.websocket.common.UnitParser;
import org.eclipse.jetty.websocket.common.WebSocketFrame;
import org.eclipse.jetty.websocket.common.frames.CloseFrame;
import org.junit.Assert;
import org.junit.Test;
@ -88,8 +89,8 @@ public class TestABCase7_3
@Test(expected = ProtocolException.class)
public void testCase7_3_2Generate1BytePayloadClose()
{
WebSocketFrame closeFrame = new WebSocketFrame(OpCode.CLOSE).setPayload(new byte[]
{ 0x00 });
CloseFrame closeFrame = new CloseFrame();
closeFrame.setPayload(Hex.asByteBuffer("00"));
UnitGenerator.generate(closeFrame);
}
@ -97,12 +98,7 @@ public class TestABCase7_3
@Test
public void testCase7_3_2Parse1BytePayloadClose()
{
ByteBuffer expected = ByteBuffer.allocate(32);
expected.put(new byte[]
{ (byte)0x88, 0x01, 0x00 });
expected.flip();
ByteBuffer expected = Hex.asByteBuffer("880100");
UnitParser parser = new UnitParser(policy);
IncomingFramesCapture capture = new IncomingFramesCapture();
@ -293,16 +289,16 @@ public class TestABCase7_3
byte[] messageBytes = message.toString().getBytes();
WebSocketFrame closeFrame = new WebSocketFrame(OpCode.CLOSE);
CloseFrame closeFrame = new CloseFrame();
ByteBuffer bb = ByteBuffer.allocate(WebSocketFrame.MAX_CONTROL_PAYLOAD + 1); // 126 which is too big for control
ByteBuffer bb = ByteBuffer.allocate(CloseFrame.MAX_CONTROL_PAYLOAD + 1); // 126 which is too big for control
bb.putChar((char)1000);
bb.put(messageBytes);
BufferUtil.flipToFlush(bb,0);
closeFrame.setPayload(BufferUtil.toArray(bb));
closeFrame.setPayload(bb);
UnitGenerator.generate(closeFrame);
}

View File

@ -26,8 +26,10 @@ import org.eclipse.jetty.websocket.api.WebSocketException;
import org.eclipse.jetty.websocket.api.WebSocketPolicy;
import org.eclipse.jetty.websocket.api.extensions.Frame;
import org.eclipse.jetty.websocket.common.CloseInfo;
import org.eclipse.jetty.websocket.common.OpCode;
import org.eclipse.jetty.websocket.common.WebSocketFrame;
import org.eclipse.jetty.websocket.common.frames.BinaryFrame;
import org.eclipse.jetty.websocket.common.frames.PingFrame;
import org.eclipse.jetty.websocket.common.frames.TextFrame;
import org.eclipse.jetty.websocket.common.io.LocalWebSocketSession;
import org.junit.Rule;
import org.junit.Test;
@ -47,7 +49,7 @@ public class EventDriverTest
private Frame makeBinaryFrame(String content, boolean fin)
{
return WebSocketFrame.binary().setFin(fin).setPayload(content);
return new BinaryFrame(content).setFin(fin);
}
@Test
@ -114,9 +116,9 @@ public class EventDriverTest
try (LocalWebSocketSession conn = new LocalWebSocketSession(testname,driver))
{
conn.open();
driver.incomingFrame(new WebSocketFrame(OpCode.PING).setPayload("PING"));
driver.incomingFrame(WebSocketFrame.text("Text Me"));
driver.incomingFrame(WebSocketFrame.binary().setPayload("Hello Bin"));
driver.incomingFrame(new PingFrame("PING"));
driver.incomingFrame(new TextFrame("Text Me"));
driver.incomingFrame(new BinaryFrame("Hello Bin"));
driver.incomingFrame(new CloseInfo(StatusCode.SHUTDOWN,"testcase").asFrame());
socket.capture.assertEventCount(6);

View File

@ -38,6 +38,9 @@ import org.eclipse.jetty.websocket.common.OpCode;
import org.eclipse.jetty.websocket.common.OutgoingFramesCapture;
import org.eclipse.jetty.websocket.common.WebSocketFrame;
import org.eclipse.jetty.websocket.common.extensions.fragment.FragmentExtension;
import org.eclipse.jetty.websocket.common.frames.ContinuationFrame;
import org.eclipse.jetty.websocket.common.frames.PingFrame;
import org.eclipse.jetty.websocket.common.frames.TextFrame;
import org.junit.Assert;
import org.junit.Test;
@ -112,7 +115,7 @@ public class FragmentExtensionTest
ext.setNextIncomingFrames(capture);
String payload = "Are you there?";
Frame ping = WebSocketFrame.ping().setPayload(payload);
Frame ping = new PingFrame(payload);
ext.incomingFrame(ping);
capture.assertFrameCount(1);
@ -161,16 +164,16 @@ public class FragmentExtensionTest
// Expected Frames
List<WebSocketFrame> expectedFrames = new ArrayList<>();
expectedFrames.add(new WebSocketFrame(OpCode.TEXT).setPayload("No amount of experim").setFin(false));
expectedFrames.add(new WebSocketFrame(OpCode.CONTINUATION).setPayload("entation can ever pr").setFin(false));
expectedFrames.add(new WebSocketFrame(OpCode.CONTINUATION).setPayload("ove me right;").setFin(true));
expectedFrames.add(new TextFrame("No amount of experim").setFin(false));
expectedFrames.add(new ContinuationFrame("entation can ever pr").setFin(false));
expectedFrames.add(new ContinuationFrame("ove me right;").setFin(true));
expectedFrames.add(new WebSocketFrame(OpCode.TEXT).setPayload("a single experiment ").setFin(false));
expectedFrames.add(new WebSocketFrame(OpCode.CONTINUATION).setPayload("can prove me wrong.").setFin(true));
expectedFrames.add(new TextFrame("a single experiment ").setFin(false));
expectedFrames.add(new ContinuationFrame("can prove me wrong.").setFin(true));
expectedFrames.add(new WebSocketFrame(OpCode.TEXT).setPayload("-- Albert Einstein").setFin(true));
expectedFrames.add(new TextFrame("-- Albert Einstein").setFin(true));
// capture.dump();
capture.dump();
int len = expectedFrames.size();
capture.assertFrameCount(len);
@ -183,6 +186,9 @@ public class FragmentExtensionTest
WebSocketFrame actualFrame = frames.get(i);
WebSocketFrame expectedFrame = expectedFrames.get(i);
System.out.printf("actual: %s%n",actualFrame);
System.out.printf("expect: %s%n",expectedFrame);
// Validate Frame
Assert.assertThat(prefix + ".opcode",actualFrame.getOpCode(),is(expectedFrame.getOpCode()));
Assert.assertThat(prefix + ".fin",actualFrame.isFin(),is(expectedFrame.isFin()));
@ -230,9 +236,9 @@ public class FragmentExtensionTest
// Expected Frames
List<WebSocketFrame> expectedFrames = new ArrayList<>();
expectedFrames.add(new WebSocketFrame(OpCode.TEXT).setPayload("No amount of experimentation can ever prove me right;"));
expectedFrames.add(new WebSocketFrame(OpCode.TEXT).setPayload("a single experiment can prove me wrong."));
expectedFrames.add(new WebSocketFrame(OpCode.TEXT).setPayload("-- Albert Einstein"));
expectedFrames.add(new TextFrame("No amount of experimentation can ever prove me right;"));
expectedFrames.add(new TextFrame("a single experiment can prove me wrong."));
expectedFrames.add(new TextFrame("-- Albert Einstein"));
// capture.dump();
@ -280,7 +286,7 @@ public class FragmentExtensionTest
ext.setNextOutgoingFrames(capture);
String payload = "Are you there?";
Frame ping = WebSocketFrame.ping().setPayload(payload);
Frame ping = new PingFrame(payload);
ext.outgoingFrame(ping,null);

View File

@ -39,6 +39,8 @@ import org.eclipse.jetty.websocket.common.OpCode;
import org.eclipse.jetty.websocket.common.OutgoingFramesCapture;
import org.eclipse.jetty.websocket.common.WebSocketFrame;
import org.eclipse.jetty.websocket.common.extensions.compress.CompressionMethod.Process;
import org.eclipse.jetty.websocket.common.frames.PingFrame;
import org.eclipse.jetty.websocket.common.frames.TextFrame;
import org.junit.Assert;
import org.junit.Test;
@ -64,9 +66,9 @@ public class MessageCompressionExtensionTest
// Receive frame
String hex = hexStr.replaceAll("\\s*0x","");
byte net[] = TypeUtil.fromHexString(hex);
WebSocketFrame frame = WebSocketFrame.text();
TextFrame frame = new TextFrame();
frame.setRsv1(true);
frame.setPayload(net);
frame.setPayload(ByteBuffer.wrap(net));
// Send frame into stack
ext.incomingFrame(frame);
@ -175,7 +177,7 @@ public class MessageCompressionExtensionTest
ext.setNextIncomingFrames(capture);
String payload = "Are you there?";
Frame ping = WebSocketFrame.ping().setPayload(payload);
Frame ping = new PingFrame(payload);
ext.incomingFrame(ping);
capture.assertFrameCount(1);
@ -220,8 +222,7 @@ public class MessageCompressionExtensionTest
// leave frames as-is, no compression, and pass into extension
for (String q : quote)
{
WebSocketFrame frame = new WebSocketFrame(OpCode.TEXT);
frame.setPayload(q);
TextFrame frame = new TextFrame(q);
frame.setRsv1(false); // indication to extension that frame is not compressed (ie: a normal frame)
ext.incomingFrame(frame);
}
@ -338,7 +339,7 @@ public class MessageCompressionExtensionTest
ext.setNextOutgoingFrames(capture);
String payload = "Are you there?";
Frame ping = WebSocketFrame.ping().setPayload(payload);
Frame ping = new PingFrame(payload);
ext.outgoingFrame(ping,null);

View File

@ -1,57 +0,0 @@
//
// ========================================================================
// Copyright (c) 1995-2013 Mort Bay Consulting Pty. Ltd.
// ------------------------------------------------------------------------
// All rights reserved. This program and the accompanying materials
// are made available under the terms of the Eclipse Public License v1.0
// and Apache License v2.0 which accompanies this distribution.
//
// The Eclipse Public License is available at
// http://www.eclipse.org/legal/epl-v10.html
//
// The Apache License v2.0 is available at
// http://www.opensource.org/licenses/apache2.0.php
//
// You may elect to redistribute this code under either of these licenses.
// ========================================================================
//
package org.eclipse.jetty.websocket.common.io.payload;
import java.nio.ByteBuffer;
import org.eclipse.jetty.util.TypeUtil;
import org.eclipse.jetty.websocket.api.BadPayloadException;
import org.eclipse.jetty.websocket.common.io.payload.UTF8Validator;
import org.junit.Assert;
import org.junit.Test;
public class UTF8ValidatorTest
{
private ByteBuffer asByteBuffer(String hexStr)
{
byte buf[] = TypeUtil.fromHexString(hexStr);
return ByteBuffer.wrap(buf);
}
@Test
public void testCase6_4_3()
{
ByteBuffer part1 = asByteBuffer("cebae1bdb9cf83cebcceb5"); // good
ByteBuffer part2 = asByteBuffer("f4908080"); // INVALID
ByteBuffer part3 = asByteBuffer("656469746564"); // good
UTF8Validator validator = new UTF8Validator();
validator.process(part1); // good
try
{
validator.process(part2); // bad
Assert.fail("Expected a " + BadPayloadException.class);
}
catch (BadPayloadException e)
{
// expected path
}
validator.process(part3); // good
}
}

View File

@ -98,14 +98,14 @@ public class MuxParser
return;
}
if (frame.isContinuation())
if (frame.getOpCode() == OpCode.CONTINUATION)
{
muxframe.reset();
muxframe.setFin(frame.isFin());
muxframe.setFin(frame.isRsv1());
muxframe.setFin(frame.isRsv2());
muxframe.setFin(frame.isRsv3());
muxframe.setContinuation(true);
muxframe.setIsContinuation();
parseDataFramePayload(buffer);
}
else
@ -223,12 +223,12 @@ public class MuxParser
if (opcode == OpCode.CONTINUATION)
{
muxframe.setContinuation(true);
muxframe.setIsContinuation();
}
else
{
muxframe.reset();
muxframe.setOpCode(opcode);
muxframe.setOp(opcode);
}
muxframe.setChannelId(channelId);

View File

@ -20,14 +20,15 @@ package org.eclipse.jetty.websocket.mux;
import org.eclipse.jetty.websocket.common.OpCode;
import org.eclipse.jetty.websocket.common.WebSocketFrame;
import org.eclipse.jetty.websocket.common.frames.DataFrame;
public class MuxedFrame extends WebSocketFrame
public class MuxedFrame extends DataFrame
{
private long channelId = -1;
public MuxedFrame()
{
super();
super(OpCode.BINARY);
}
public MuxedFrame(MuxedFrame frame)
@ -66,8 +67,13 @@ public class MuxedFrame extends WebSocketFrame
b.append(isRsv1()?'1':'.');
b.append(isRsv2()?'1':'.');
b.append(isRsv3()?'1':'.');
b.append(",continuation=").append(isContinuation());
b.append(']');
return b.toString();
}
public void setOp(byte opcode)
{
// TODO Auto-generated method stub
}
}

View File

@ -38,6 +38,7 @@ import org.eclipse.jetty.websocket.api.extensions.IncomingFrames;
import org.eclipse.jetty.websocket.api.extensions.OutgoingFrames;
import org.eclipse.jetty.websocket.common.LogicalConnection;
import org.eclipse.jetty.websocket.common.WebSocketFrame;
import org.eclipse.jetty.websocket.common.frames.ControlFrame;
import org.eclipse.jetty.websocket.mux.add.MuxAddClient;
import org.eclipse.jetty.websocket.mux.add.MuxAddServer;
import org.eclipse.jetty.websocket.mux.op.MuxAddChannelRequest;
@ -196,7 +197,7 @@ public class Muxer implements IncomingFrames, MuxParser.Listener
}
String reason = "Mux[MUST FAIL]" + drop.getPhrase();
reason = StringUtil.truncate(reason,WebSocketFrame.MAX_CONTROL_PAYLOAD);
reason = StringUtil.truncate(reason,ControlFrame.MAX_CONTROL_PAYLOAD);
this.physicalConnection.close(StatusCode.SERVER_ERROR,reason);
// TODO: trigger abnormal close for all sub-channels.
@ -308,7 +309,7 @@ public class Muxer implements IncomingFrames, MuxParser.Listener
MuxChannel channel = getChannel(channelId,false);
String reason = "Mux " + drop.toString();
reason = StringUtil.truncate(reason,(WebSocketFrame.MAX_CONTROL_PAYLOAD - 2));
reason = StringUtil.truncate(reason,(ControlFrame.MAX_CONTROL_PAYLOAD - 2));
channel.close(StatusCode.PROTOCOL,reason);
// TODO: set channel to inactive?
}

View File

@ -43,7 +43,7 @@ public interface MuxAddServer
* the channel to attach the {@link WebSocketSession} to.
* @param requestHandshake
* the request handshake (request headers)
* @throws AbstractMuxException
* @throws MuxException
* if unable to handshake
* @throws IOException
* if unable to parse request headers

View File

@ -71,11 +71,11 @@ public class MuxAddHandler implements MuxAddServer
/**
* An incoming MuxAddChannel request.
*
* @param the
* @param muxer the muxer handling this
* @param channel the
* channel this request should be bound to
* @param request
* the incoming request headers (complete and merged if delta encoded)
* @return the outgoing response headers
*/
@Override
public void handshake(Muxer muxer, MuxChannel channel, UpgradeRequest request) throws MuxException, IOException

View File

@ -40,6 +40,7 @@ import org.eclipse.jetty.websocket.common.extensions.AbstractExtension;
import org.eclipse.jetty.websocket.mux.helper.IncomingFramesCapture;
import org.eclipse.jetty.websocket.mux.helper.UnitParser;
import org.junit.Assert;
import org.junit.Ignore;
import org.junit.Test;
public class MuxParserRFCTest
@ -94,6 +95,7 @@ public class MuxParserRFCTest
}
@Test
@Ignore
public void testRFCExample1() throws IOException
{
// Create RFC detailed frames
@ -122,6 +124,7 @@ public class MuxParserRFCTest
}
@Test
@Ignore
public void testRFCExample2() throws IOException
{
// Create RFC detailed frames
@ -158,7 +161,6 @@ public class MuxParserRFCTest
Assert.assertThat(prefix + ".rsv2",mux.isRsv2(),is(false));
Assert.assertThat(prefix + ".rsv3",mux.isRsv3(),is(false));
Assert.assertThat(prefix + ".masked",mux.isMasked(),is(false));
Assert.assertThat(prefix + ".continuation",mux.isContinuation(),is(true));
// (BUG IN DRAFT) Assert.assertThat(prefix + ".opcode",mux.getOpCode(),is(OpCode.BINARY));
payload = mux.getPayloadAsUTF8();
@ -166,6 +168,7 @@ public class MuxParserRFCTest
}
@Test
@Ignore
public void testRFCExample3() throws IOException
{
// Create RFC detailed frames
@ -188,7 +191,6 @@ public class MuxParserRFCTest
Assert.assertThat(prefix + ".rsv2",mux.isRsv2(),is(false));
Assert.assertThat(prefix + ".rsv3",mux.isRsv3(),is(false));
Assert.assertThat(prefix + ".masked",mux.isMasked(),is(false));
Assert.assertThat(prefix + ".continuation",mux.isContinuation(),is(false));
Assert.assertThat(prefix + ".opcode",mux.getOpCode(),is(OpCode.TEXT));
String payload = mux.getPayloadAsUTF8();
@ -203,7 +205,6 @@ public class MuxParserRFCTest
Assert.assertThat(prefix + ".rsv2",mux.isRsv2(),is(false));
Assert.assertThat(prefix + ".rsv3",mux.isRsv3(),is(false));
Assert.assertThat(prefix + ".masked",mux.isMasked(),is(false));
Assert.assertThat(prefix + ".continuation",mux.isContinuation(),is(false));
Assert.assertThat(prefix + ".opcode",mux.getOpCode(),is(OpCode.TEXT));
payload = mux.getPayloadAsUTF8();
@ -218,7 +219,6 @@ public class MuxParserRFCTest
Assert.assertThat(prefix + ".rsv2",mux.isRsv2(),is(false));
Assert.assertThat(prefix + ".rsv3",mux.isRsv3(),is(false));
Assert.assertThat(prefix + ".masked",mux.isMasked(),is(false));
Assert.assertThat(prefix + ".continuation",mux.isContinuation(),is(true));
Assert.assertThat(prefix + ".opcode",mux.getOpCode(),is(OpCode.TEXT));
payload = mux.getPayloadAsUTF8();

View File

@ -131,7 +131,7 @@ public class IncomingFramesCapture implements IncomingFrames
@Override
public void incomingFrame(Frame frame)
{
WebSocketFrame copy = new WebSocketFrame(frame);
WebSocketFrame copy = WebSocketFrame.copy(frame);
frames.add(copy);
}

View File

@ -86,7 +86,7 @@ public class OutgoingFramesCapture implements OutgoingFrames
@Override
public void outgoingFrame(Frame frame, WriteCallback callback)
{
WebSocketFrame copy = new WebSocketFrame(frame);
WebSocketFrame copy = WebSocketFrame.copy(frame);
frames.add(copy);
if (callback != null)
{

View File

@ -18,23 +18,14 @@
package org.eclipse.jetty.websocket.server;
import static org.hamcrest.Matchers.*;
import java.nio.ByteBuffer;
import java.util.concurrent.TimeUnit;
import org.eclipse.jetty.util.BufferUtil;
import org.eclipse.jetty.websocket.api.StatusCode;
import org.eclipse.jetty.websocket.common.CloseInfo;
import org.eclipse.jetty.websocket.common.OpCode;
import org.eclipse.jetty.websocket.common.WebSocketFrame;
import org.eclipse.jetty.websocket.server.blockhead.BlockheadClient;
import org.eclipse.jetty.websocket.server.helper.IncomingFramesCapture;
import org.eclipse.jetty.websocket.server.helper.RFCSocket;
import org.eclipse.jetty.websocket.servlet.WebSocketServlet;
import org.eclipse.jetty.websocket.servlet.WebSocketServletFactory;
import org.junit.AfterClass;
import org.junit.Assert;
import org.junit.BeforeClass;
import org.junit.Test;

View File

@ -37,7 +37,11 @@ import org.eclipse.jetty.websocket.common.OpCode;
import org.eclipse.jetty.websocket.common.Parser;
import org.eclipse.jetty.websocket.common.WebSocketFrame;
import org.eclipse.jetty.websocket.common.events.EventDriver;
import org.eclipse.jetty.websocket.common.frames.BinaryFrame;
import org.eclipse.jetty.websocket.common.frames.ContinuationFrame;
import org.eclipse.jetty.websocket.common.frames.TextFrame;
import org.eclipse.jetty.websocket.server.blockhead.BlockheadClient;
import org.eclipse.jetty.websocket.server.helper.Hex;
import org.eclipse.jetty.websocket.server.helper.IncomingFramesCapture;
import org.eclipse.jetty.websocket.server.helper.RFCServlet;
import org.eclipse.jetty.websocket.servlet.WebSocketServlet;
@ -99,15 +103,15 @@ public class WebSocketServletRFCTest
WebSocketFrame bin;
bin = WebSocketFrame.binary(buf1).setFin(false);
bin = new BinaryFrame(buf1).setFin(false);
client.write(bin); // write buf1 (fin=false)
bin = new WebSocketFrame(OpCode.CONTINUATION).setPayload(buf2).setFin(false);
bin = new ContinuationFrame(buf2).setFin(false);
client.write(bin); // write buf2 (fin=false)
bin = new WebSocketFrame(OpCode.CONTINUATION).setPayload(buf3).setFin(true);
bin = new ContinuationFrame(buf3).setFin(true);
client.write(bin); // write buf3 (fin=true)
@ -284,7 +288,8 @@ public class WebSocketServletRFCTest
byte buf[] = new byte[]
{ (byte)0xC2, (byte)0xC3 };
WebSocketFrame txt = WebSocketFrame.text().setPayload(buf);
WebSocketFrame txt = new TextFrame().setPayload(ByteBuffer.wrap(buf));
txt.setMask(Hex.asByteArray("11223344"));
ByteBuffer bbHeader = generator.generateHeaderBytes(txt);
client.writeRaw(bbHeader);
client.writeRaw(txt.getPayload());

View File

@ -24,9 +24,12 @@ import org.eclipse.jetty.io.ByteBufferPool;
import org.eclipse.jetty.io.MappedByteBufferPool;
import org.eclipse.jetty.util.BufferUtil;
import org.eclipse.jetty.util.StringUtil;
import org.eclipse.jetty.util.log.StacklessLogging;
import org.eclipse.jetty.util.log.StdErrLog;
import org.eclipse.jetty.websocket.api.WebSocketPolicy;
import org.eclipse.jetty.websocket.common.Generator;
import org.eclipse.jetty.websocket.common.OpCode;
import org.eclipse.jetty.websocket.common.WebSocketFrame;
import org.eclipse.jetty.websocket.server.SimpleServletServer;
import org.junit.AfterClass;
import org.junit.BeforeClass;
@ -35,6 +38,34 @@ import org.junit.rules.TestName;
public abstract class AbstractABCase
{
// Allow Fuzzer / Generator to create bad frames for testing frame validation
protected static class BadFrame extends WebSocketFrame
{
public BadFrame(byte opcode)
{
super(OpCode.CONTINUATION);
super.finRsvOp = (byte)((finRsvOp & 0xF0) | (opcode & 0x0F));
// NOTE: Not setting Frame.Type intentionally
}
@Override
public void assertValid()
{
}
@Override
public boolean isControlFrame()
{
return false;
}
@Override
public boolean isDataFrame()
{
return false;
}
}
protected static final byte FIN = (byte)0x80;
protected static final byte NOFIN = 0x00;
private static final byte MASKED_BIT = (byte)0x80;
@ -76,10 +107,27 @@ public abstract class AbstractABCase
* @param payload the payload to copy
* @return a new byte array of the payload contents
*/
protected byte[] copyOf(byte[] payload)
protected ByteBuffer copyOf(byte[] payload)
{
byte copy[] = new byte[payload.length];
System.arraycopy(payload,0,copy,0,payload.length);
return ByteBuffer.wrap(copy);
}
/**
* Make a copy of a byte buffer.
* <p>
* This is important in some tests, as the underlying byte buffer contained in a Frame can be modified through
* masking and make it difficult to compare the results in the fuzzer.
*
* @param payload the payload to copy
* @return a new byte array of the payload contents
*/
protected ByteBuffer clone(ByteBuffer payload)
{
ByteBuffer copy = ByteBuffer.allocate(payload.remaining());
copy.put(payload.slice());
copy.flip();
return copy;
}
@ -124,6 +172,10 @@ public abstract class AbstractABCase
@Rule
public TestName testname = new TestName();
/**
* @deprecated use {@link StacklessLogging} in a try-with-resources block instead
*/
@Deprecated
protected void enableStacks(Class<?> clazz, boolean enabled)
{
StdErrLog log = StdErrLog.getLogger(clazz);

View File

@ -109,7 +109,6 @@ public class Fuzzer
buflen += f.getPayloadLength() + Generator.OVERHEAD;
}
ByteBuffer buf = ByteBuffer.allocate(buflen);
BufferUtil.clearToFill(buf);
// Generate frames
for (WebSocketFrame f : send)
@ -117,7 +116,7 @@ public class Fuzzer
setClientMask(f);
generator.generateWholeFrame(f,buf);
}
BufferUtil.flipToFlush(buf,0);
buf.flip();
return buf;
}

View File

@ -18,6 +18,7 @@
package org.eclipse.jetty.websocket.server.ab;
import java.nio.ByteBuffer;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.List;
@ -67,13 +68,14 @@ public class TestABCase1 extends AbstractABCase
{
byte payload[] = new byte[125];
Arrays.fill(payload,(byte)'*');
ByteBuffer buf = ByteBuffer.wrap(payload);
List<WebSocketFrame> send = new ArrayList<>();
send.add(WebSocketFrame.text().setPayload(payload));
send.add(WebSocketFrame.text().setPayload(buf));
send.add(new CloseInfo(StatusCode.NORMAL).asFrame());
List<WebSocketFrame> expect = new ArrayList<>();
expect.add(WebSocketFrame.text().setPayload(copyOf(payload)));
expect.add(WebSocketFrame.text().setPayload(clone(buf)));
expect.add(new CloseInfo(StatusCode.NORMAL).asFrame());
Fuzzer fuzzer = new Fuzzer(this);
@ -99,13 +101,14 @@ public class TestABCase1 extends AbstractABCase
{
byte payload[] = new byte[126];
Arrays.fill(payload,(byte)'*');
ByteBuffer buf = ByteBuffer.wrap(payload);
List<WebSocketFrame> send = new ArrayList<>();
send.add(WebSocketFrame.text().setPayload(payload));
send.add(WebSocketFrame.text().setPayload(buf));
send.add(new CloseInfo(StatusCode.NORMAL).asFrame());
List<WebSocketFrame> expect = new ArrayList<>();
expect.add(WebSocketFrame.text().setPayload(copyOf(payload)));
expect.add(WebSocketFrame.text().setPayload(clone(buf)));
expect.add(new CloseInfo(StatusCode.NORMAL).asFrame());
Fuzzer fuzzer = new Fuzzer(this);
@ -131,13 +134,14 @@ public class TestABCase1 extends AbstractABCase
{
byte payload[] = new byte[127];
Arrays.fill(payload,(byte)'*');
ByteBuffer buf = ByteBuffer.wrap(payload);
List<WebSocketFrame> send = new ArrayList<>();
send.add(WebSocketFrame.text().setPayload(payload));
send.add(WebSocketFrame.text().setPayload(buf));
send.add(new CloseInfo(StatusCode.NORMAL).asFrame());
List<WebSocketFrame> expect = new ArrayList<>();
expect.add(WebSocketFrame.text().setPayload(copyOf(payload)));
expect.add(WebSocketFrame.text().setPayload(clone(buf)));
expect.add(new CloseInfo(StatusCode.NORMAL).asFrame());
Fuzzer fuzzer = new Fuzzer(this);
@ -163,13 +167,14 @@ public class TestABCase1 extends AbstractABCase
{
byte payload[] = new byte[128];
Arrays.fill(payload,(byte)'*');
ByteBuffer buf = ByteBuffer.wrap(payload);
List<WebSocketFrame> send = new ArrayList<>();
send.add(WebSocketFrame.text().setPayload(payload));
send.add(WebSocketFrame.text().setPayload(buf));
send.add(new CloseInfo(StatusCode.NORMAL).asFrame());
List<WebSocketFrame> expect = new ArrayList<>();
expect.add(WebSocketFrame.text().setPayload(copyOf(payload)));
expect.add(WebSocketFrame.text().setPayload(clone(buf)));
expect.add(new CloseInfo(StatusCode.NORMAL).asFrame());
Fuzzer fuzzer = new Fuzzer(this);
@ -195,13 +200,14 @@ public class TestABCase1 extends AbstractABCase
{
byte payload[] = new byte[65535];
Arrays.fill(payload,(byte)'*');
ByteBuffer buf = ByteBuffer.wrap(payload);
List<WebSocketFrame> send = new ArrayList<>();
send.add(WebSocketFrame.text().setPayload(payload));
send.add(WebSocketFrame.text().setPayload(buf));
send.add(new CloseInfo(StatusCode.NORMAL).asFrame());
List<WebSocketFrame> expect = new ArrayList<>();
expect.add(WebSocketFrame.text().setPayload(copyOf(payload)));
expect.add(WebSocketFrame.text().setPayload(clone(buf)));
expect.add(new CloseInfo(StatusCode.NORMAL).asFrame());
Fuzzer fuzzer = new Fuzzer(this);
@ -227,13 +233,14 @@ public class TestABCase1 extends AbstractABCase
{
byte payload[] = new byte[65536];
Arrays.fill(payload,(byte)'*');
ByteBuffer buf = ByteBuffer.wrap(payload);
List<WebSocketFrame> send = new ArrayList<>();
send.add(WebSocketFrame.text().setPayload(payload));
send.add(WebSocketFrame.text().setPayload(buf));
send.add(new CloseInfo(StatusCode.NORMAL).asFrame());
List<WebSocketFrame> expect = new ArrayList<>();
expect.add(WebSocketFrame.text().setPayload(copyOf(payload)));
expect.add(WebSocketFrame.text().setPayload(clone(buf)));
expect.add(new CloseInfo(StatusCode.NORMAL).asFrame());
Fuzzer fuzzer = new Fuzzer(this);
@ -263,14 +270,15 @@ public class TestABCase1 extends AbstractABCase
{
byte payload[] = new byte[65536];
Arrays.fill(payload,(byte)'*');
ByteBuffer buf = ByteBuffer.wrap(payload);
int segmentSize = 997;
List<WebSocketFrame> send = new ArrayList<>();
send.add(WebSocketFrame.text().setPayload(payload));
send.add(WebSocketFrame.text().setPayload(buf));
send.add(new CloseInfo(StatusCode.NORMAL).asFrame());
List<WebSocketFrame> expect = new ArrayList<>();
expect.add(WebSocketFrame.text().setPayload(copyOf(payload)));
expect.add(WebSocketFrame.text().setPayload(clone(buf)));
expect.add(new CloseInfo(StatusCode.NORMAL).asFrame());
Fuzzer fuzzer = new Fuzzer(this);
@ -326,13 +334,14 @@ public class TestABCase1 extends AbstractABCase
{
byte payload[] = new byte[125];
Arrays.fill(payload,(byte)0xFE);
ByteBuffer buf = ByteBuffer.wrap(payload);
List<WebSocketFrame> send = new ArrayList<>();
send.add(WebSocketFrame.binary().setPayload(payload));
send.add(WebSocketFrame.binary().setPayload(buf));
send.add(new CloseInfo(StatusCode.NORMAL).asFrame());
List<WebSocketFrame> expect = new ArrayList<>();
expect.add(WebSocketFrame.binary().setPayload(copyOf(payload)));
expect.add(WebSocketFrame.binary().setPayload(clone(buf)));
expect.add(new CloseInfo(StatusCode.NORMAL).asFrame());
Fuzzer fuzzer = new Fuzzer(this);
@ -358,13 +367,14 @@ public class TestABCase1 extends AbstractABCase
{
byte payload[] = new byte[126];
Arrays.fill(payload,(byte)0xFE);
ByteBuffer buf = ByteBuffer.wrap(payload);
List<WebSocketFrame> send = new ArrayList<>();
send.add(WebSocketFrame.binary().setPayload(payload));
send.add(WebSocketFrame.binary().setPayload(buf));
send.add(new CloseInfo(StatusCode.NORMAL).asFrame());
List<WebSocketFrame> expect = new ArrayList<>();
expect.add(WebSocketFrame.binary().setPayload(copyOf(payload)));
expect.add(WebSocketFrame.binary().setPayload(clone(buf)));
expect.add(new CloseInfo(StatusCode.NORMAL).asFrame());
Fuzzer fuzzer = new Fuzzer(this);
@ -390,13 +400,14 @@ public class TestABCase1 extends AbstractABCase
{
byte payload[] = new byte[127];
Arrays.fill(payload,(byte)0xFE);
ByteBuffer buf = ByteBuffer.wrap(payload);
List<WebSocketFrame> send = new ArrayList<>();
send.add(WebSocketFrame.binary().setPayload(payload));
send.add(WebSocketFrame.binary().setPayload(buf));
send.add(new CloseInfo(StatusCode.NORMAL).asFrame());
List<WebSocketFrame> expect = new ArrayList<>();
expect.add(WebSocketFrame.binary().setPayload(copyOf(payload)));
expect.add(WebSocketFrame.binary().setPayload(clone(buf)));
expect.add(new CloseInfo(StatusCode.NORMAL).asFrame());
Fuzzer fuzzer = new Fuzzer(this);
@ -422,13 +433,14 @@ public class TestABCase1 extends AbstractABCase
{
byte payload[] = new byte[128];
Arrays.fill(payload,(byte)0xFE);
ByteBuffer buf = ByteBuffer.wrap(payload);
List<WebSocketFrame> send = new ArrayList<>();
send.add(WebSocketFrame.binary().setPayload(payload));
send.add(WebSocketFrame.binary().setPayload(buf));
send.add(new CloseInfo(StatusCode.NORMAL).asFrame());
List<WebSocketFrame> expect = new ArrayList<>();
expect.add(WebSocketFrame.binary().setPayload(copyOf(payload)));
expect.add(WebSocketFrame.binary().setPayload(clone(buf)));
expect.add(new CloseInfo(StatusCode.NORMAL).asFrame());
Fuzzer fuzzer = new Fuzzer(this);
@ -454,13 +466,14 @@ public class TestABCase1 extends AbstractABCase
{
byte payload[] = new byte[65535];
Arrays.fill(payload,(byte)0xFE);
ByteBuffer buf = ByteBuffer.wrap(payload);
List<WebSocketFrame> send = new ArrayList<>();
send.add(WebSocketFrame.binary().setPayload(payload));
send.add(WebSocketFrame.binary().setPayload(buf));
send.add(new CloseInfo(StatusCode.NORMAL).asFrame());
List<WebSocketFrame> expect = new ArrayList<>();
expect.add(WebSocketFrame.binary().setPayload(copyOf(payload)));
expect.add(WebSocketFrame.binary().setPayload(clone(buf)));
expect.add(new CloseInfo(StatusCode.NORMAL).asFrame());
Fuzzer fuzzer = new Fuzzer(this);
@ -486,13 +499,14 @@ public class TestABCase1 extends AbstractABCase
{
byte payload[] = new byte[65536];
Arrays.fill(payload,(byte)0xFE);
ByteBuffer buf = ByteBuffer.wrap(payload);
List<WebSocketFrame> send = new ArrayList<>();
send.add(WebSocketFrame.binary().setPayload(payload));
send.add(WebSocketFrame.binary().setPayload(buf));
send.add(new CloseInfo(StatusCode.NORMAL).asFrame());
List<WebSocketFrame> expect = new ArrayList<>();
expect.add(WebSocketFrame.binary().setPayload(copyOf(payload)));
expect.add(WebSocketFrame.binary().setPayload(clone(buf)));
expect.add(new CloseInfo(StatusCode.NORMAL).asFrame());
Fuzzer fuzzer = new Fuzzer(this);
@ -522,14 +536,15 @@ public class TestABCase1 extends AbstractABCase
{
byte payload[] = new byte[65536];
Arrays.fill(payload,(byte)0xFE);
ByteBuffer buf = ByteBuffer.wrap(payload);
int segmentSize = 997;
List<WebSocketFrame> send = new ArrayList<>();
send.add(WebSocketFrame.binary().setPayload(payload));
send.add(WebSocketFrame.binary().setPayload(buf));
send.add(new CloseInfo(StatusCode.NORMAL).asFrame());
List<WebSocketFrame> expect = new ArrayList<>();
expect.add(WebSocketFrame.binary().setPayload(copyOf(payload)));
expect.add(WebSocketFrame.binary().setPayload(clone(buf)));
expect.add(new CloseInfo(StatusCode.NORMAL).asFrame());
Fuzzer fuzzer = new Fuzzer(this);

View File

@ -18,6 +18,7 @@
package org.eclipse.jetty.websocket.server.ab;
import java.nio.ByteBuffer;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.List;
@ -30,6 +31,8 @@ import org.eclipse.jetty.websocket.common.CloseInfo;
import org.eclipse.jetty.websocket.common.OpCode;
import org.eclipse.jetty.websocket.common.Parser;
import org.eclipse.jetty.websocket.common.WebSocketFrame;
import org.eclipse.jetty.websocket.common.frames.PingFrame;
import org.eclipse.jetty.websocket.common.frames.PongFrame;
import org.junit.Test;
import org.junit.runner.RunWith;
@ -79,8 +82,8 @@ public class TestABCase2 extends AbstractABCase
for (int i = 0; i < pingCount; i++)
{
String payload = String.format("ping-%d[%X]",i,i);
send.add(WebSocketFrame.ping().setPayload(payload));
expect.add(WebSocketFrame.pong().setPayload(payload));
send.add(new PingFrame(payload));
expect.add(new PongFrame(payload));
}
send.add(new CloseInfo(StatusCode.NORMAL).asFrame());
expect.add(new CloseInfo(StatusCode.NORMAL).asFrame());
@ -118,8 +121,8 @@ public class TestABCase2 extends AbstractABCase
for (int i = 0; i < pingCount; i++)
{
String payload = String.format("ping-%d[%X]",i,i);
send.add(WebSocketFrame.ping().setPayload(payload));
expect.add(WebSocketFrame.pong().setPayload(payload));
send.add(new PingFrame(payload));
expect.add(new PongFrame(payload));
}
send.add(new CloseInfo(StatusCode.NORMAL).asFrame());
expect.add(new CloseInfo(StatusCode.NORMAL).asFrame());
@ -148,11 +151,11 @@ public class TestABCase2 extends AbstractABCase
byte payload[] = StringUtil.getUtf8Bytes("Hello world");
List<WebSocketFrame> send = new ArrayList<>();
send.add(WebSocketFrame.ping().setPayload(payload));
send.add(new PingFrame(payload));
send.add(new CloseInfo(StatusCode.NORMAL).asFrame());
List<WebSocketFrame> expect = new ArrayList<>();
expect.add(WebSocketFrame.pong().setPayload(copyOf(payload)));
expect.add(new PongFrame(copyOf(payload)));
expect.add(new CloseInfo(StatusCode.NORMAL).asFrame());
Fuzzer fuzzer = new Fuzzer(this);
@ -179,11 +182,11 @@ public class TestABCase2 extends AbstractABCase
{ 0x00, (byte)0xFF, (byte)0xFE, (byte)0xFD, (byte)0xFC, (byte)0xFB, 0x00, (byte)0xFF };
List<WebSocketFrame> send = new ArrayList<>();
send.add(WebSocketFrame.ping().setPayload(payload));
send.add(new PingFrame(payload));
send.add(new CloseInfo(StatusCode.NORMAL).asFrame());
List<WebSocketFrame> expect = new ArrayList<>();
expect.add(WebSocketFrame.pong().setPayload(copyOf(payload)));
expect.add(new PongFrame(copyOf(payload)));
expect.add(new CloseInfo(StatusCode.NORMAL).asFrame());
Fuzzer fuzzer = new Fuzzer(this);
@ -210,11 +213,11 @@ public class TestABCase2 extends AbstractABCase
Arrays.fill(payload,(byte)0xFE);
List<WebSocketFrame> send = new ArrayList<>();
send.add(WebSocketFrame.ping().setPayload(payload));
send.add(new PingFrame(payload));
send.add(new CloseInfo(StatusCode.NORMAL).asFrame());
List<WebSocketFrame> expect = new ArrayList<>();
expect.add(WebSocketFrame.pong().setPayload(copyOf(payload)));
expect.add(new PongFrame(copyOf(payload)));
expect.add(new CloseInfo(StatusCode.NORMAL).asFrame());
Fuzzer fuzzer = new Fuzzer(this);
@ -241,10 +244,11 @@ public class TestABCase2 extends AbstractABCase
{
byte payload[] = new byte[126]; // intentionally too big
Arrays.fill(payload,(byte)'5');
ByteBuffer buf = ByteBuffer.wrap(payload);
List<WebSocketFrame> send = new ArrayList<>();
// trick websocket frame into making extra large payload for ping
send.add(WebSocketFrame.binary(payload).setOpCode(OpCode.PING));
send.add(new BadFrame(OpCode.PING).setPayload(buf));
send.add(new CloseInfo(StatusCode.NORMAL,"Test 2.5").asFrame());
List<WebSocketFrame> expect = new ArrayList<>();
@ -276,11 +280,11 @@ public class TestABCase2 extends AbstractABCase
Arrays.fill(payload,(byte)'6');
List<WebSocketFrame> send = new ArrayList<>();
send.add(WebSocketFrame.ping().setPayload(payload));
send.add(new PingFrame(payload));
send.add(new CloseInfo(StatusCode.NORMAL,"Test 2.6").asFrame());
List<WebSocketFrame> expect = new ArrayList<>();
expect.add(WebSocketFrame.pong().setPayload(copyOf(payload)));
expect.add(new PongFrame(copyOf(payload)));
expect.add(new CloseInfo(StatusCode.NORMAL,"Test 2.6").asFrame());
Fuzzer fuzzer = new Fuzzer(this);
@ -332,7 +336,7 @@ public class TestABCase2 extends AbstractABCase
public void testCase2_8() throws Exception
{
List<WebSocketFrame> send = new ArrayList<>();
send.add(WebSocketFrame.pong().setPayload("unsolicited")); // unsolicited pong
send.add(new PongFrame("unsolicited")); // unsolicited pong
send.add(new CloseInfo(StatusCode.NORMAL).asFrame());
List<WebSocketFrame> expect = new ArrayList<>();
@ -359,12 +363,12 @@ public class TestABCase2 extends AbstractABCase
public void testCase2_9() throws Exception
{
List<WebSocketFrame> send = new ArrayList<>();
send.add(WebSocketFrame.pong().setPayload("unsolicited")); // unsolicited pong
send.add(WebSocketFrame.ping().setPayload("our ping")); // our ping
send.add(new PongFrame("unsolicited")); // unsolicited pong
send.add(new PingFrame("our ping")); // our ping
send.add(new CloseInfo(StatusCode.NORMAL).asFrame());
List<WebSocketFrame> expect = new ArrayList<>();
expect.add(WebSocketFrame.pong().setPayload("our ping")); // our pong
expect.add(new PongFrame("our ping")); // our pong
expect.add(new CloseInfo(StatusCode.NORMAL).asFrame());
Fuzzer fuzzer = new Fuzzer(this);

View File

@ -25,10 +25,8 @@ import java.util.List;
import org.eclipse.jetty.toolchain.test.AdvancedRunner;
import org.eclipse.jetty.websocket.api.StatusCode;
import org.eclipse.jetty.websocket.common.CloseInfo;
import org.eclipse.jetty.websocket.common.Parser;
import org.eclipse.jetty.websocket.common.WebSocketFrame;
import org.junit.After;
import org.junit.Before;
import org.eclipse.jetty.websocket.common.frames.PingFrame;
import org.junit.Test;
import org.junit.runner.RunWith;
@ -38,18 +36,6 @@ import org.junit.runner.RunWith;
@RunWith(AdvancedRunner.class)
public class TestABCase3 extends AbstractABCase
{
@After
public void enableParserStacks()
{
enableStacks(Parser.class,true);
}
@Before
public void quietParserStacks()
{
enableStacks(Parser.class,false);
}
/**
* Send small text frame, with RSV1 == true, with no extensions defined.
*/
@ -83,7 +69,7 @@ public class TestABCase3 extends AbstractABCase
List<WebSocketFrame> send = new ArrayList<>();
send.add(WebSocketFrame.text("small"));
send.add(WebSocketFrame.text("small").setRsv2(true)); // intentionally bad
send.add(WebSocketFrame.ping().setPayload("ping"));
send.add(new PingFrame("ping"));
List<WebSocketFrame> expect = new ArrayList<>();
expect.add(WebSocketFrame.text("small")); // echo on good frame
@ -112,7 +98,7 @@ public class TestABCase3 extends AbstractABCase
List<WebSocketFrame> send = new ArrayList<>();
send.add(WebSocketFrame.text("small"));
send.add(WebSocketFrame.text("small").setRsv1(true).setRsv2(true)); // intentionally bad
send.add(WebSocketFrame.ping().setPayload("ping"));
send.add(new PingFrame("ping"));
List<WebSocketFrame> expect = new ArrayList<>();
expect.add(WebSocketFrame.text("small")); // echo on good frame
@ -141,7 +127,7 @@ public class TestABCase3 extends AbstractABCase
List<WebSocketFrame> send = new ArrayList<>();
send.add(WebSocketFrame.text("small"));
send.add(WebSocketFrame.text("small").setRsv3(true)); // intentionally bad
send.add(WebSocketFrame.ping().setPayload("ping"));
send.add(new PingFrame("ping"));
List<WebSocketFrame> expect = new ArrayList<>();
expect.add(WebSocketFrame.text("small")); // echo on good frame
@ -201,7 +187,7 @@ public class TestABCase3 extends AbstractABCase
Arrays.fill(payload,(byte)0xFF);
List<WebSocketFrame> send = new ArrayList<>();
send.add(WebSocketFrame.ping().setPayload(payload).setRsv3(true).setRsv2(true)); // intentionally bad
send.add(new PingFrame(payload).setRsv3(true).setRsv2(true)); // intentionally bad
List<WebSocketFrame> expect = new ArrayList<>();
expect.add(new CloseInfo(StatusCode.PROTOCOL).asFrame());

View File

@ -18,6 +18,7 @@
package org.eclipse.jetty.websocket.server.ab;
import java.nio.ByteBuffer;
import java.util.ArrayList;
import java.util.List;
@ -25,10 +26,7 @@ import org.eclipse.jetty.toolchain.test.AdvancedRunner;
import org.eclipse.jetty.util.StringUtil;
import org.eclipse.jetty.websocket.api.StatusCode;
import org.eclipse.jetty.websocket.common.CloseInfo;
import org.eclipse.jetty.websocket.common.Parser;
import org.eclipse.jetty.websocket.common.WebSocketFrame;
import org.junit.After;
import org.junit.Before;
import org.junit.Test;
import org.junit.runner.RunWith;
@ -38,29 +36,6 @@ import org.junit.runner.RunWith;
@RunWith(AdvancedRunner.class)
public class TestABCase4 extends AbstractABCase
{
// Allow Fuzzer / Generator to create bad frames for testing frame validation
private static class BadFrame extends WebSocketFrame
{
public BadFrame(byte opcode)
{
super();
super.finRsvOp = (byte)((finRsvOp & 0xF0) | (opcode & 0x0F));
// NOTE: Not setting Frame.Type intentionally
}
}
@After
public void enableParserStacks()
{
enableStacks(Parser.class,true);
}
@Before
public void quietParserStacks()
{
enableStacks(Parser.class,false);
}
/**
* Send opcode 3 (reserved)
*/
@ -94,9 +69,10 @@ public class TestABCase4 extends AbstractABCase
public void testCase4_1_2() throws Exception
{
byte payload[] = StringUtil.getUtf8Bytes("reserved payload");
ByteBuffer buf = ByteBuffer.wrap(payload);
List<WebSocketFrame> send = new ArrayList<>();
send.add(new BadFrame((byte)4).setPayload(payload)); // intentionally bad
send.add(new BadFrame((byte)4).setPayload(buf)); // intentionally bad
List<WebSocketFrame> expect = new ArrayList<>();
expect.add(new CloseInfo(StatusCode.PROTOCOL).asFrame());
@ -150,9 +126,11 @@ public class TestABCase4 extends AbstractABCase
@Test
public void testCase4_1_4() throws Exception
{
ByteBuffer buf = ByteBuffer.wrap(StringUtil.getUtf8Bytes("bad"));
List<WebSocketFrame> send = new ArrayList<>();
send.add(WebSocketFrame.text("hello"));
send.add(new BadFrame((byte)6).setPayload("bad")); // intentionally bad
send.add(new BadFrame((byte)6).setPayload(buf)); // intentionally bad
send.add(WebSocketFrame.ping());
List<WebSocketFrame> expect = new ArrayList<>();
@ -179,9 +157,11 @@ public class TestABCase4 extends AbstractABCase
@Test
public void testCase4_1_5() throws Exception
{
ByteBuffer buf = ByteBuffer.wrap(StringUtil.getUtf8Bytes("bad"));
List<WebSocketFrame> send = new ArrayList<>();
send.add(WebSocketFrame.text("hello"));
send.add(new BadFrame((byte)7).setPayload("bad")); // intentionally bad
send.add(new BadFrame((byte)7).setPayload(buf)); // intentionally bad
send.add(WebSocketFrame.ping());
List<WebSocketFrame> expect = new ArrayList<>();
@ -234,8 +214,10 @@ public class TestABCase4 extends AbstractABCase
@Test
public void testCase4_2_2() throws Exception
{
ByteBuffer buf = ByteBuffer.wrap(StringUtil.getUtf8Bytes("bad"));
List<WebSocketFrame> send = new ArrayList<>();
send.add(new BadFrame((byte)12).setPayload("bad")); // intentionally bad
send.add(new BadFrame((byte)12).setPayload(buf)); // intentionally bad
List<WebSocketFrame> expect = new ArrayList<>();
expect.add(new CloseInfo(StatusCode.PROTOCOL).asFrame());
@ -289,9 +271,11 @@ public class TestABCase4 extends AbstractABCase
@Test
public void testCase4_2_4() throws Exception
{
ByteBuffer buf = ByteBuffer.wrap(StringUtil.getUtf8Bytes("bad"));
List<WebSocketFrame> send = new ArrayList<>();
send.add(WebSocketFrame.text("hello"));
send.add(new BadFrame((byte)14).setPayload("bad")); // intentionally bad
send.add(new BadFrame((byte)14).setPayload(buf)); // intentionally bad
send.add(WebSocketFrame.ping());
List<WebSocketFrame> expect = new ArrayList<>();
@ -318,9 +302,11 @@ public class TestABCase4 extends AbstractABCase
@Test
public void testCase4_2_5() throws Exception
{
ByteBuffer buf = ByteBuffer.wrap(StringUtil.getUtf8Bytes("bad"));
List<WebSocketFrame> send = new ArrayList<>();
send.add(WebSocketFrame.text("hello"));
send.add(new BadFrame((byte)15).setPayload("bad")); // intentionally bad
send.add(new BadFrame((byte)15).setPayload(buf)); // intentionally bad
send.add(WebSocketFrame.ping());
List<WebSocketFrame> expect = new ArrayList<>();

View File

@ -24,11 +24,15 @@ import java.util.concurrent.TimeUnit;
import org.eclipse.jetty.toolchain.test.AdvancedRunner;
import org.eclipse.jetty.toolchain.test.annotation.Slow;
import org.eclipse.jetty.util.log.StacklessLogging;
import org.eclipse.jetty.websocket.api.StatusCode;
import org.eclipse.jetty.websocket.common.CloseInfo;
import org.eclipse.jetty.websocket.common.OpCode;
import org.eclipse.jetty.websocket.common.Parser;
import org.eclipse.jetty.websocket.common.WebSocketFrame;
import org.eclipse.jetty.websocket.common.frames.ContinuationFrame;
import org.eclipse.jetty.websocket.common.frames.PingFrame;
import org.eclipse.jetty.websocket.common.frames.PongFrame;
import org.eclipse.jetty.websocket.common.frames.TextFrame;
import org.junit.Test;
import org.junit.runner.RunWith;
@ -44,19 +48,16 @@ public class TestABCase5 extends AbstractABCase
@Test
public void testCase5_1() throws Exception
{
// Disable Long Stacks from Parser (we know this test will throw an exception)
enableStacks(Parser.class,false);
List<WebSocketFrame> send = new ArrayList<>();
send.add(new WebSocketFrame(OpCode.PING).setPayload("hello, ").setFin(false));
send.add(new WebSocketFrame(OpCode.CONTINUATION).setPayload("world"));
send.add(new PingFrame("hello, ").setFin(false));
send.add(new ContinuationFrame("world"));
send.add(new CloseInfo(StatusCode.NORMAL).asFrame());
List<WebSocketFrame> expect = new ArrayList<>();
expect.add(new CloseInfo(StatusCode.PROTOCOL).asFrame());
Fuzzer fuzzer = new Fuzzer(this);
try
try(StacklessLogging supress = new StacklessLogging(Parser.class))
{
fuzzer.connect();
fuzzer.setSendMode(Fuzzer.SendMode.BULK);
@ -66,7 +67,6 @@ public class TestABCase5 extends AbstractABCase
finally
{
fuzzer.close();
enableStacks(Parser.class,true);
}
}
@ -76,19 +76,16 @@ public class TestABCase5 extends AbstractABCase
@Test
public void testCase5_10() throws Exception
{
// Disable Long Stacks from Parser (we know this test will throw an exception)
enableStacks(Parser.class,false);
List<WebSocketFrame> send = new ArrayList<>();
send.add(new WebSocketFrame(OpCode.CONTINUATION).setPayload("sorry").setFin(true));
send.add(new WebSocketFrame(OpCode.TEXT).setPayload("hello, world"));
send.add(new ContinuationFrame("sorry").setFin(true));
send.add(new TextFrame("hello, world"));
send.add(new CloseInfo(StatusCode.NORMAL).asFrame());
List<WebSocketFrame> expect = new ArrayList<>();
expect.add(new CloseInfo(StatusCode.PROTOCOL).asFrame());
Fuzzer fuzzer = new Fuzzer(this);
try
try(StacklessLogging supress = new StacklessLogging(Parser.class))
{
fuzzer.connect();
fuzzer.setSendMode(Fuzzer.SendMode.PER_FRAME);
@ -97,7 +94,6 @@ public class TestABCase5 extends AbstractABCase
}
finally
{
enableStacks(Parser.class,true);
fuzzer.close();
}
}
@ -108,19 +104,16 @@ public class TestABCase5 extends AbstractABCase
@Test
public void testCase5_11() throws Exception
{
// Disable Long Stacks from Parser (we know this test will throw an exception)
enableStacks(Parser.class,false);
List<WebSocketFrame> send = new ArrayList<>();
send.add(new WebSocketFrame(OpCode.CONTINUATION).setPayload("sorry").setFin(true));
send.add(new WebSocketFrame(OpCode.TEXT).setPayload("hello, world"));
send.add(new ContinuationFrame("sorry").setFin(true));
send.add(new TextFrame("hello, world"));
send.add(new CloseInfo(StatusCode.NORMAL).asFrame());
List<WebSocketFrame> expect = new ArrayList<>();
expect.add(new CloseInfo(StatusCode.PROTOCOL).asFrame());
Fuzzer fuzzer = new Fuzzer(this);
try
try(StacklessLogging supress = new StacklessLogging(Parser.class))
{
fuzzer.connect();
fuzzer.setSendMode(Fuzzer.SendMode.SLOW);
@ -130,7 +123,6 @@ public class TestABCase5 extends AbstractABCase
}
finally
{
enableStacks(Parser.class,true);
fuzzer.close();
}
}
@ -141,19 +133,16 @@ public class TestABCase5 extends AbstractABCase
@Test
public void testCase5_12() throws Exception
{
// Disable Long Stacks from Parser (we know this test will throw an exception)
enableStacks(Parser.class,false);
List<WebSocketFrame> send = new ArrayList<>();
send.add(new WebSocketFrame(OpCode.CONTINUATION).setPayload("sorry").setFin(false));
send.add(new WebSocketFrame(OpCode.TEXT).setPayload("hello, world"));
send.add(new ContinuationFrame("sorry").setFin(false));
send.add(new TextFrame("hello, world"));
send.add(new CloseInfo(StatusCode.NORMAL).asFrame());
List<WebSocketFrame> expect = new ArrayList<>();
expect.add(new CloseInfo(StatusCode.PROTOCOL).asFrame());
Fuzzer fuzzer = new Fuzzer(this);
try
try(StacklessLogging supress = new StacklessLogging(Parser.class))
{
fuzzer.connect();
fuzzer.setSendMode(Fuzzer.SendMode.BULK);
@ -162,7 +151,6 @@ public class TestABCase5 extends AbstractABCase
}
finally
{
enableStacks(Parser.class,true);
fuzzer.close();
}
}
@ -173,18 +161,16 @@ public class TestABCase5 extends AbstractABCase
@Test
public void testCase5_13() throws Exception
{
// Disable Long Stacks from Parser (we know this test will throw an exception)
enableStacks(Parser.class,false);
List<WebSocketFrame> send = new ArrayList<>();
send.add(new WebSocketFrame(OpCode.CONTINUATION).setPayload("sorry").setFin(false));
send.add(new WebSocketFrame(OpCode.TEXT).setPayload("hello, world"));
send.add(new ContinuationFrame("sorry").setFin(false));
send.add(new TextFrame("hello, world"));
send.add(new CloseInfo(StatusCode.NORMAL).asFrame());
List<WebSocketFrame> expect = new ArrayList<>();
expect.add(new CloseInfo(StatusCode.PROTOCOL).asFrame());
Fuzzer fuzzer = new Fuzzer(this);
try
try(StacklessLogging supress = new StacklessLogging(Parser.class))
{
fuzzer.connect();
fuzzer.setSendMode(Fuzzer.SendMode.PER_FRAME);
@ -193,7 +179,6 @@ public class TestABCase5 extends AbstractABCase
}
finally
{
enableStacks(Parser.class,true);
fuzzer.close();
}
}
@ -204,19 +189,16 @@ public class TestABCase5 extends AbstractABCase
@Test
public void testCase5_14() throws Exception
{
// Disable Long Stacks from Parser (we know this test will throw an exception)
enableStacks(Parser.class,false);
List<WebSocketFrame> send = new ArrayList<>();
send.add(new WebSocketFrame(OpCode.CONTINUATION).setPayload("sorry").setFin(false));
send.add(new WebSocketFrame(OpCode.TEXT).setPayload("hello, world"));
send.add(new ContinuationFrame("sorry").setFin(false));
send.add(new TextFrame("hello, world"));
send.add(new CloseInfo(StatusCode.NORMAL).asFrame());
List<WebSocketFrame> expect = new ArrayList<>();
expect.add(new CloseInfo(StatusCode.PROTOCOL).asFrame());
Fuzzer fuzzer = new Fuzzer(this);
try
try(StacklessLogging supress = new StacklessLogging(Parser.class))
{
fuzzer.connect();
fuzzer.setSendMode(Fuzzer.SendMode.SLOW);
@ -226,7 +208,6 @@ public class TestABCase5 extends AbstractABCase
}
finally
{
enableStacks(Parser.class,false);
fuzzer.close();
}
}
@ -237,14 +218,11 @@ public class TestABCase5 extends AbstractABCase
@Test
public void testCase5_15() throws Exception
{
// Disable Long Stacks from Parser (we know this test will throw an exception)
enableStacks(Parser.class,false);
List<WebSocketFrame> send = new ArrayList<>();
send.add(new WebSocketFrame(OpCode.TEXT).setPayload("fragment1").setFin(false));
send.add(new WebSocketFrame(OpCode.CONTINUATION).setPayload("fragment2").setFin(true));
send.add(new WebSocketFrame(OpCode.CONTINUATION).setPayload("fragment3").setFin(false)); // bad frame
send.add(new WebSocketFrame(OpCode.TEXT).setPayload("fragment4").setFin(true));
send.add(new TextFrame("fragment1").setFin(false));
send.add(new ContinuationFrame("fragment2").setFin(true));
send.add(new ContinuationFrame("fragment3").setFin(false)); // bad frame
send.add(new TextFrame("fragment4").setFin(true));
send.add(new CloseInfo(StatusCode.NORMAL).asFrame());
List<WebSocketFrame> expect = new ArrayList<>();
@ -252,7 +230,7 @@ public class TestABCase5 extends AbstractABCase
expect.add(new CloseInfo(StatusCode.PROTOCOL).asFrame());
Fuzzer fuzzer = new Fuzzer(this);
try
try(StacklessLogging supress = new StacklessLogging(Parser.class))
{
fuzzer.connect();
fuzzer.setSendMode(Fuzzer.SendMode.BULK);
@ -261,7 +239,6 @@ public class TestABCase5 extends AbstractABCase
}
finally
{
enableStacks(Parser.class,true);
fuzzer.close();
}
}
@ -272,23 +249,20 @@ public class TestABCase5 extends AbstractABCase
@Test
public void testCase5_16() throws Exception
{
// Disable Long Stacks from Parser (we know this test will throw an exception)
enableStacks(Parser.class,false);
List<WebSocketFrame> send = new ArrayList<>();
send.add(new WebSocketFrame(OpCode.CONTINUATION).setPayload("fragment1").setFin(false)); // bad frame
send.add(new WebSocketFrame(OpCode.TEXT).setPayload("fragment2").setFin(false));
send.add(new WebSocketFrame(OpCode.CONTINUATION).setPayload("fragment3").setFin(true));
send.add(new WebSocketFrame(OpCode.CONTINUATION).setPayload("fragment4").setFin(false)); // bad frame
send.add(new WebSocketFrame(OpCode.TEXT).setPayload("fragment5").setFin(false));
send.add(new WebSocketFrame(OpCode.CONTINUATION).setPayload("fragment6").setFin(true));
send.add(new ContinuationFrame("fragment1").setFin(false)); // bad frame
send.add(new TextFrame("fragment2").setFin(false));
send.add(new ContinuationFrame("fragment3").setFin(true));
send.add(new ContinuationFrame("fragment4").setFin(false)); // bad frame
send.add(new TextFrame("fragment5").setFin(false));
send.add(new ContinuationFrame("fragment6").setFin(true));
send.add(new CloseInfo(StatusCode.NORMAL).asFrame());
List<WebSocketFrame> expect = new ArrayList<>();
expect.add(new CloseInfo(StatusCode.PROTOCOL).asFrame());
Fuzzer fuzzer = new Fuzzer(this);
try
try(StacklessLogging supress = new StacklessLogging(Parser.class))
{
fuzzer.connect();
fuzzer.setSendMode(Fuzzer.SendMode.BULK);
@ -297,7 +271,6 @@ public class TestABCase5 extends AbstractABCase
}
finally
{
enableStacks(Parser.class,true);
fuzzer.close();
}
}
@ -308,23 +281,20 @@ public class TestABCase5 extends AbstractABCase
@Test
public void testCase5_17() throws Exception
{
// Disable Long Stacks from Parser (we know this test will throw an exception)
enableStacks(Parser.class,false);
List<WebSocketFrame> send = new ArrayList<>();
send.add(new WebSocketFrame(OpCode.CONTINUATION).setPayload("fragment1").setFin(true)); // nothing to continue
send.add(new WebSocketFrame(OpCode.TEXT).setPayload("fragment2").setFin(false));
send.add(new WebSocketFrame(OpCode.CONTINUATION).setPayload("fragment3").setFin(true));
send.add(new WebSocketFrame(OpCode.CONTINUATION).setPayload("fragment4").setFin(true)); // nothing to continue
send.add(new WebSocketFrame(OpCode.TEXT).setPayload("fragment5").setFin(false));
send.add(new WebSocketFrame(OpCode.CONTINUATION).setPayload("fragment6").setFin(true));
send.add(new ContinuationFrame("fragment1").setFin(true)); // nothing to continue
send.add(new TextFrame("fragment2").setFin(false));
send.add(new ContinuationFrame("fragment3").setFin(true));
send.add(new ContinuationFrame("fragment4").setFin(true)); // nothing to continue
send.add(new TextFrame("fragment5").setFin(false));
send.add(new ContinuationFrame("fragment6").setFin(true));
send.add(new CloseInfo(StatusCode.NORMAL).asFrame());
List<WebSocketFrame> expect = new ArrayList<>();
expect.add(new CloseInfo(StatusCode.PROTOCOL).asFrame());
Fuzzer fuzzer = new Fuzzer(this);
try
try(StacklessLogging supress = new StacklessLogging(Parser.class))
{
fuzzer.connect();
fuzzer.setSendMode(Fuzzer.SendMode.BULK);
@ -333,7 +303,6 @@ public class TestABCase5 extends AbstractABCase
}
finally
{
enableStacks(Parser.class,true);
fuzzer.close();
}
}
@ -344,19 +313,16 @@ public class TestABCase5 extends AbstractABCase
@Test
public void testCase5_18() throws Exception
{
// Disable Long Stacks from Parser (we know this test will throw an exception)
enableStacks(Parser.class,false);
List<WebSocketFrame> send = new ArrayList<>();
send.add(new WebSocketFrame(OpCode.TEXT).setPayload("fragment1").setFin(false));
send.add(new WebSocketFrame(OpCode.TEXT).setPayload("fragment2").setFin(true)); // bad frame, must be continuation
send.add(new TextFrame("fragment1").setFin(false));
send.add(new TextFrame("fragment2").setFin(true)); // bad frame, must be continuation
send.add(new CloseInfo(StatusCode.NORMAL).asFrame());
List<WebSocketFrame> expect = new ArrayList<>();
expect.add(new CloseInfo(StatusCode.PROTOCOL).asFrame());
Fuzzer fuzzer = new Fuzzer(this);
try
try(StacklessLogging supress = new StacklessLogging(Parser.class))
{
fuzzer.connect();
fuzzer.setSendMode(Fuzzer.SendMode.BULK);
@ -365,7 +331,6 @@ public class TestABCase5 extends AbstractABCase
}
finally
{
enableStacks(Parser.class,true);
fuzzer.close();
}
}
@ -379,28 +344,28 @@ public class TestABCase5 extends AbstractABCase
{
// phase 1
List<WebSocketFrame> send1 = new ArrayList<>();
send1.add(new WebSocketFrame(OpCode.TEXT).setPayload("f1").setFin(false));
send1.add(new WebSocketFrame(OpCode.CONTINUATION).setPayload(",f2").setFin(false));
send1.add(new WebSocketFrame(OpCode.PING).setPayload("pong-1"));
send1.add(new TextFrame("f1").setFin(false));
send1.add(new ContinuationFrame(",f2").setFin(false));
send1.add(new PingFrame("pong-1"));
List<WebSocketFrame> expect1 = new ArrayList<>();
expect1.add(WebSocketFrame.pong().setPayload("pong-1"));
expect1.add(new PongFrame("pong-1"));
// phase 2
List<WebSocketFrame> send2 = new ArrayList<>();
send2.add(new WebSocketFrame(OpCode.CONTINUATION).setPayload(",f3").setFin(false));
send2.add(new WebSocketFrame(OpCode.CONTINUATION).setPayload(",f4").setFin(false));
send2.add(new WebSocketFrame(OpCode.PING).setPayload("pong-2"));
send2.add(new WebSocketFrame(OpCode.CONTINUATION).setPayload(",f5").setFin(true));
send2.add(new ContinuationFrame(",f3").setFin(false));
send2.add(new ContinuationFrame(",f4").setFin(false));
send2.add(new PingFrame("pong-2"));
send2.add(new ContinuationFrame(",f5").setFin(true));
send2.add(new CloseInfo(StatusCode.NORMAL).asFrame());
List<WebSocketFrame> expect2 = new ArrayList<>();
expect2.add(WebSocketFrame.pong().setPayload("pong-2"));
expect2.add(new PongFrame("pong-2"));
expect2.add(WebSocketFrame.text("f1,f2,f3,f4,f5"));
expect2.add(new CloseInfo(StatusCode.NORMAL).asFrame());
Fuzzer fuzzer = new Fuzzer(this);
try
try(StacklessLogging supress = new StacklessLogging(Parser.class))
{
fuzzer.connect();
fuzzer.setSendMode(Fuzzer.SendMode.BULK);
@ -428,19 +393,16 @@ public class TestABCase5 extends AbstractABCase
@Test
public void testCase5_2() throws Exception
{
// Disable Long Stacks from Parser (we know this test will throw an exception)
enableStacks(Parser.class,false);
List<WebSocketFrame> send = new ArrayList<>();
send.add(new WebSocketFrame(OpCode.PONG).setPayload("hello, ").setFin(false));
send.add(new WebSocketFrame(OpCode.CONTINUATION).setPayload("world"));
send.add(new PongFrame("hello, ").setFin(false));
send.add(new ContinuationFrame("world"));
send.add(new CloseInfo(StatusCode.NORMAL).asFrame());
List<WebSocketFrame> expect = new ArrayList<>();
expect.add(new CloseInfo(StatusCode.PROTOCOL).asFrame());
Fuzzer fuzzer = new Fuzzer(this);
try
try(StacklessLogging supress = new StacklessLogging(Parser.class))
{
fuzzer.connect();
fuzzer.setSendMode(Fuzzer.SendMode.BULK);
@ -449,7 +411,6 @@ public class TestABCase5 extends AbstractABCase
}
finally
{
enableStacks(Parser.class,true);
fuzzer.close();
}
}
@ -461,27 +422,27 @@ public class TestABCase5 extends AbstractABCase
public void testCase5_20() throws Exception
{
List<WebSocketFrame> send1 = new ArrayList<>();
send1.add(new WebSocketFrame(OpCode.TEXT).setPayload("f1").setFin(false));
send1.add(new WebSocketFrame(OpCode.CONTINUATION).setPayload(",f2").setFin(false));
send1.add(new WebSocketFrame(OpCode.PING).setPayload("pong-1"));
send1.add(new TextFrame("f1").setFin(false));
send1.add(new ContinuationFrame(",f2").setFin(false));
send1.add(new PingFrame("pong-1"));
List<WebSocketFrame> send2 = new ArrayList<>();
send2.add(new WebSocketFrame(OpCode.CONTINUATION).setPayload(",f3").setFin(false));
send2.add(new WebSocketFrame(OpCode.CONTINUATION).setPayload(",f4").setFin(false));
send2.add(new WebSocketFrame(OpCode.PING).setPayload("pong-2"));
send2.add(new WebSocketFrame(OpCode.CONTINUATION).setPayload(",f5").setFin(true));
send2.add(new ContinuationFrame(",f3").setFin(false));
send2.add(new ContinuationFrame(",f4").setFin(false));
send2.add(new PingFrame("pong-2"));
send2.add(new ContinuationFrame(",f5").setFin(true));
send2.add(new CloseInfo(StatusCode.NORMAL).asFrame());
List<WebSocketFrame> expect1 = new ArrayList<>();
expect1.add(WebSocketFrame.pong().setPayload("pong-1"));
expect1.add(new PongFrame("pong-1"));
List<WebSocketFrame> expect2 = new ArrayList<>();
expect2.add(WebSocketFrame.pong().setPayload("pong-2"));
expect2.add(new PongFrame("pong-2"));
expect2.add(WebSocketFrame.text("f1,f2,f3,f4,f5"));
expect2.add(new CloseInfo(StatusCode.NORMAL).asFrame());
Fuzzer fuzzer = new Fuzzer(this);
try
try(StacklessLogging supress = new StacklessLogging(Parser.class))
{
fuzzer.connect();
fuzzer.setSendMode(Fuzzer.SendMode.PER_FRAME);
@ -507,27 +468,27 @@ public class TestABCase5 extends AbstractABCase
public void testCase5_20_slow() throws Exception
{
List<WebSocketFrame> send1 = new ArrayList<>();
send1.add(new WebSocketFrame(OpCode.TEXT).setPayload("f1").setFin(false));
send1.add(new WebSocketFrame(OpCode.CONTINUATION).setPayload(",f2").setFin(false));
send1.add(new WebSocketFrame(OpCode.PING).setPayload("pong-1"));
send1.add(new TextFrame("f1").setFin(false));
send1.add(new ContinuationFrame(",f2").setFin(false));
send1.add(new PingFrame("pong-1"));
List<WebSocketFrame> send2 = new ArrayList<>();
send2.add(new WebSocketFrame(OpCode.CONTINUATION).setPayload(",f3").setFin(false));
send2.add(new WebSocketFrame(OpCode.CONTINUATION).setPayload(",f4").setFin(false));
send2.add(new WebSocketFrame(OpCode.PING).setPayload("pong-2"));
send2.add(new WebSocketFrame(OpCode.CONTINUATION).setPayload(",f5").setFin(true));
send2.add(new ContinuationFrame(",f3").setFin(false));
send2.add(new ContinuationFrame(",f4").setFin(false));
send2.add(new PingFrame("pong-2"));
send2.add(new ContinuationFrame(",f5").setFin(true));
send2.add(new CloseInfo(StatusCode.NORMAL).asFrame());
List<WebSocketFrame> expect1 = new ArrayList<>();
expect1.add(WebSocketFrame.pong().setPayload("pong-1"));
expect1.add(new PongFrame("pong-1"));
List<WebSocketFrame> expect2 = new ArrayList<>();
expect2.add(WebSocketFrame.pong().setPayload("pong-2"));
expect2.add(new PongFrame("pong-2"));
expect2.add(WebSocketFrame.text("f1,f2,f3,f4,f5"));
expect2.add(new CloseInfo(StatusCode.NORMAL).asFrame());
Fuzzer fuzzer = new Fuzzer(this);
try
try(StacklessLogging supress = new StacklessLogging(Parser.class))
{
fuzzer.connect();
fuzzer.setSendMode(Fuzzer.SendMode.SLOW);
@ -554,8 +515,8 @@ public class TestABCase5 extends AbstractABCase
public void testCase5_3() throws Exception
{
List<WebSocketFrame> send = new ArrayList<>();
send.add(new WebSocketFrame(OpCode.TEXT).setPayload("hello, ").setFin(false));
send.add(new WebSocketFrame(OpCode.CONTINUATION).setPayload("world").setFin(true));
send.add(new TextFrame("hello, ").setFin(false));
send.add(new ContinuationFrame("world").setFin(true));
send.add(new CloseInfo(StatusCode.NORMAL).asFrame());
List<WebSocketFrame> expect = new ArrayList<>();
@ -563,7 +524,7 @@ public class TestABCase5 extends AbstractABCase
expect.add(new CloseInfo(StatusCode.NORMAL).asFrame());
Fuzzer fuzzer = new Fuzzer(this);
try
try(StacklessLogging supress = new StacklessLogging(Parser.class))
{
fuzzer.connect();
fuzzer.setSendMode(Fuzzer.SendMode.BULK);
@ -583,8 +544,8 @@ public class TestABCase5 extends AbstractABCase
public void testCase5_4() throws Exception
{
List<WebSocketFrame> send = new ArrayList<>();
send.add(new WebSocketFrame(OpCode.TEXT).setPayload("hello, ").setFin(false));
send.add(new WebSocketFrame(OpCode.CONTINUATION).setPayload("world").setFin(true));
send.add(new TextFrame("hello, ").setFin(false));
send.add(new ContinuationFrame("world").setFin(true));
send.add(new CloseInfo(StatusCode.NORMAL).asFrame());
List<WebSocketFrame> expect = new ArrayList<>();
@ -592,7 +553,7 @@ public class TestABCase5 extends AbstractABCase
expect.add(new CloseInfo(StatusCode.NORMAL).asFrame());
Fuzzer fuzzer = new Fuzzer(this);
try
try(StacklessLogging supress = new StacklessLogging(Parser.class))
{
fuzzer.connect();
fuzzer.setSendMode(Fuzzer.SendMode.PER_FRAME);
@ -612,8 +573,8 @@ public class TestABCase5 extends AbstractABCase
public void testCase5_5() throws Exception
{
List<WebSocketFrame> send = new ArrayList<>();
send.add(new WebSocketFrame(OpCode.TEXT).setPayload("hello, ").setFin(false));
send.add(new WebSocketFrame(OpCode.CONTINUATION).setPayload("world").setFin(true));
send.add(new TextFrame("hello, ").setFin(false));
send.add(new ContinuationFrame("world").setFin(true));
send.add(new CloseInfo(StatusCode.NORMAL).asFrame());
List<WebSocketFrame> expect = new ArrayList<>();
@ -621,7 +582,7 @@ public class TestABCase5 extends AbstractABCase
expect.add(new CloseInfo(StatusCode.NORMAL).asFrame());
Fuzzer fuzzer = new Fuzzer(this);
try
try(StacklessLogging supress = new StacklessLogging(Parser.class))
{
fuzzer.connect();
fuzzer.setSendMode(Fuzzer.SendMode.SLOW);
@ -642,18 +603,18 @@ public class TestABCase5 extends AbstractABCase
public void testCase5_6() throws Exception
{
List<WebSocketFrame> send = new ArrayList<>();
send.add(new WebSocketFrame(OpCode.TEXT).setPayload("hello, ").setFin(false));
send.add(new WebSocketFrame(OpCode.PING).setPayload("ping"));
send.add(new WebSocketFrame(OpCode.CONTINUATION).setPayload("world").setFin(true));
send.add(new TextFrame("hello, ").setFin(false));
send.add(new PingFrame("ping"));
send.add(new ContinuationFrame("world").setFin(true));
send.add(new CloseInfo(StatusCode.NORMAL).asFrame());
List<WebSocketFrame> expect = new ArrayList<>();
expect.add(WebSocketFrame.pong().setPayload("ping"));
expect.add(new PongFrame("ping"));
expect.add(WebSocketFrame.text("hello, world"));
expect.add(new CloseInfo(StatusCode.NORMAL).asFrame());
Fuzzer fuzzer = new Fuzzer(this);
try
try(StacklessLogging supress = new StacklessLogging(Parser.class))
{
fuzzer.connect();
fuzzer.setSendMode(Fuzzer.SendMode.BULK);
@ -673,18 +634,18 @@ public class TestABCase5 extends AbstractABCase
public void testCase5_7() throws Exception
{
List<WebSocketFrame> send = new ArrayList<>();
send.add(new WebSocketFrame(OpCode.TEXT).setPayload("hello, ").setFin(false));
send.add(new WebSocketFrame(OpCode.PING).setPayload("ping"));
send.add(new WebSocketFrame(OpCode.CONTINUATION).setPayload("world").setFin(true));
send.add(new TextFrame("hello, ").setFin(false));
send.add(new PingFrame("ping"));
send.add(new ContinuationFrame("world").setFin(true));
send.add(new CloseInfo(StatusCode.NORMAL).asFrame());
List<WebSocketFrame> expect = new ArrayList<>();
expect.add(WebSocketFrame.pong().setPayload("ping"));
expect.add(new PongFrame("ping"));
expect.add(WebSocketFrame.text("hello, world"));
expect.add(new CloseInfo(StatusCode.NORMAL).asFrame());
Fuzzer fuzzer = new Fuzzer(this);
try
try(StacklessLogging supress = new StacklessLogging(Parser.class))
{
fuzzer.connect();
fuzzer.setSendMode(Fuzzer.SendMode.PER_FRAME);
@ -704,18 +665,18 @@ public class TestABCase5 extends AbstractABCase
public void testCase5_8() throws Exception
{
List<WebSocketFrame> send = new ArrayList<>();
send.add(new WebSocketFrame(OpCode.TEXT).setPayload("hello, ").setFin(false));
send.add(new WebSocketFrame(OpCode.PING).setPayload("ping"));
send.add(new WebSocketFrame(OpCode.CONTINUATION).setPayload("world").setFin(true));
send.add(new TextFrame("hello, ").setFin(false));
send.add(new PingFrame("ping"));
send.add(new ContinuationFrame("world").setFin(true));
send.add(new CloseInfo(StatusCode.NORMAL).asFrame());
List<WebSocketFrame> expect = new ArrayList<>();
expect.add(WebSocketFrame.pong().setPayload("ping"));
expect.add(new PongFrame("ping"));
expect.add(WebSocketFrame.text("hello, world"));
expect.add(new CloseInfo(StatusCode.NORMAL).asFrame());
Fuzzer fuzzer = new Fuzzer(this);
try
try(StacklessLogging supress = new StacklessLogging(Parser.class))
{
fuzzer.connect();
fuzzer.setSendMode(Fuzzer.SendMode.SLOW);
@ -735,19 +696,17 @@ public class TestABCase5 extends AbstractABCase
@Test
public void testCase5_9() throws Exception
{
// Disable Long Stacks from Parser (we know this test will throw an exception)
enableStacks(Parser.class,false);
List<WebSocketFrame> send = new ArrayList<>();
send.add(new WebSocketFrame(OpCode.CONTINUATION).setPayload("sorry").setFin(true));
send.add(new WebSocketFrame(OpCode.TEXT).setPayload("hello, world"));
send.add(new ContinuationFrame("sorry").setFin(true));
send.add(new TextFrame("hello, world"));
send.add(new CloseInfo(StatusCode.NORMAL).asFrame());
List<WebSocketFrame> expect = new ArrayList<>();
expect.add(new CloseInfo(StatusCode.PROTOCOL).asFrame());
Fuzzer fuzzer = new Fuzzer(this);
try
try(StacklessLogging supress = new StacklessLogging(Parser.class))
{
fuzzer.connect();
fuzzer.setSendMode(Fuzzer.SendMode.BULK);
@ -756,7 +715,6 @@ public class TestABCase5 extends AbstractABCase
}
finally
{
enableStacks(Parser.class,true);
fuzzer.close();
}
}

View File

@ -31,9 +31,11 @@ import org.eclipse.jetty.util.TypeUtil;
import org.eclipse.jetty.util.log.StacklessLogging;
import org.eclipse.jetty.websocket.api.StatusCode;
import org.eclipse.jetty.websocket.common.CloseInfo;
import org.eclipse.jetty.websocket.common.OpCode;
import org.eclipse.jetty.websocket.common.Parser;
import org.eclipse.jetty.websocket.common.WebSocketFrame;
import org.eclipse.jetty.websocket.common.frames.ContinuationFrame;
import org.eclipse.jetty.websocket.common.frames.DataFrame;
import org.eclipse.jetty.websocket.common.frames.TextFrame;
import org.eclipse.jetty.websocket.server.helper.Hex;
import org.junit.Test;
import org.junit.runner.RunWith;
@ -50,18 +52,26 @@ public class TestABCase6 extends AbstractABCase
protected void fragmentText(List<WebSocketFrame> frames, byte msg[])
{
int len = msg.length;
byte opcode = OpCode.TEXT;
boolean continuation = false;
byte mini[];
for (int i = 0; i < len; i++)
{
WebSocketFrame frame = new WebSocketFrame(opcode);
DataFrame frame = null;
if (continuation)
{
frame = new ContinuationFrame();
}
else
{
frame = new TextFrame();
}
mini = new byte[1];
mini[0] = msg[i];
frame.setPayload(mini);
frame.setPayload(ByteBuffer.wrap(mini));
boolean isLast = (i >= (len - 1));
frame.setFin(isLast);
frames.add(frame);
opcode = OpCode.CONTINUATION;
continuation = true;
}
}
@ -100,9 +110,9 @@ public class TestABCase6 extends AbstractABCase
public void testCase6_1_2() throws Exception
{
List<WebSocketFrame> send = new ArrayList<>();
send.add(new WebSocketFrame(OpCode.TEXT).setFin(false));
send.add(new WebSocketFrame(OpCode.CONTINUATION).setFin(false));
send.add(new WebSocketFrame(OpCode.CONTINUATION).setFin(true));
send.add(new TextFrame().setFin(false));
send.add(new ContinuationFrame().setFin(false));
send.add(new ContinuationFrame().setFin(true));
send.add(new CloseInfo(StatusCode.NORMAL).asFrame());
List<WebSocketFrame> expect = new ArrayList<>();
@ -130,9 +140,9 @@ public class TestABCase6 extends AbstractABCase
public void testCase6_1_3() throws Exception
{
List<WebSocketFrame> send = new ArrayList<>();
send.add(new WebSocketFrame(OpCode.TEXT).setFin(false));
send.add(new WebSocketFrame(OpCode.CONTINUATION).setFin(false).setPayload("middle"));
send.add(new WebSocketFrame(OpCode.CONTINUATION).setFin(true));
send.add(new TextFrame().setFin(false));
send.add(new ContinuationFrame("middle").setFin(false));
send.add(new ContinuationFrame().setFin(true));
send.add(new CloseInfo(StatusCode.NORMAL).asFrame());
List<WebSocketFrame> expect = new ArrayList<>();
@ -159,16 +169,23 @@ public class TestABCase6 extends AbstractABCase
@Test
public void testCase6_2_2() throws Exception
{
String utf8[] =
{ "Hello-\uC2B5@\uC39F\uC3A4", "\uC3BC\uC3A0\uC3A1-UTF-8!!" };
String utf1 = "Hello-\uC2B5@\uC39F\uC3A4";
String utf2 = "\uC3BC\uC3A0\uC3A1-UTF-8!!";
ByteBuffer b1 = ByteBuffer.wrap(StringUtil.getUtf8Bytes(utf1));
ByteBuffer b2 = ByteBuffer.wrap(StringUtil.getUtf8Bytes(utf2));
List<WebSocketFrame> send = new ArrayList<>();
send.add(new WebSocketFrame(OpCode.TEXT).setPayload(utf8[0]).setFin(false));
send.add(new WebSocketFrame(OpCode.CONTINUATION).setPayload(utf8[1]).setFin(true));
send.add(new TextFrame().setPayload(b1).setFin(false));
send.add(new ContinuationFrame(b2).setFin(true));
send.add(new CloseInfo(StatusCode.NORMAL).asFrame());
List<WebSocketFrame> expect = new ArrayList<>();
expect.add(new WebSocketFrame(OpCode.TEXT).setPayload(utf8[0] + utf8[1]));
ByteBuffer e1 = ByteBuffer.allocate(100);
e1.put(StringUtil.getUtf8Bytes(utf1));
e1.put(StringUtil.getUtf8Bytes(utf2));
e1.flip();
expect.add(new TextFrame().setPayload(e1));
expect.add(new CloseInfo(StatusCode.NORMAL).asFrame());
Fuzzer fuzzer = new Fuzzer(this);
@ -199,7 +216,7 @@ public class TestABCase6 extends AbstractABCase
send.add(new CloseInfo(StatusCode.NORMAL).asFrame());
List<WebSocketFrame> expect = new ArrayList<>();
expect.add(new WebSocketFrame(OpCode.TEXT).setPayload(msg));
expect.add(new TextFrame().setPayload(ByteBuffer.wrap(msg)));
expect.add(new CloseInfo(StatusCode.NORMAL).asFrame());
Fuzzer fuzzer = new Fuzzer(this);
@ -229,7 +246,7 @@ public class TestABCase6 extends AbstractABCase
send.add(new CloseInfo(StatusCode.NORMAL).asFrame());
List<WebSocketFrame> expect = new ArrayList<>();
expect.add(new WebSocketFrame(OpCode.TEXT).setPayload(msg));
expect.add(new TextFrame().setPayload(ByteBuffer.wrap(msg)));
expect.add(new CloseInfo(StatusCode.NORMAL).asFrame());
Fuzzer fuzzer = new Fuzzer(this);
@ -299,11 +316,11 @@ public class TestABCase6 extends AbstractABCase
fuzzer.connect();
fuzzer.setSendMode(Fuzzer.SendMode.BULK);
fuzzer.send(new WebSocketFrame(OpCode.TEXT).setPayload(part1).setFin(false));
fuzzer.send(new TextFrame().setPayload(ByteBuffer.wrap(part1)).setFin(false));
TimeUnit.SECONDS.sleep(1);
fuzzer.send(new WebSocketFrame(OpCode.CONTINUATION).setPayload(part2).setFin(false));
fuzzer.send(new ContinuationFrame().setPayload(ByteBuffer.wrap(part2)).setFin(false));
TimeUnit.SECONDS.sleep(1);
fuzzer.send(new WebSocketFrame(OpCode.CONTINUATION).setPayload(part3).setFin(true));
fuzzer.send(new ContinuationFrame().setPayload(ByteBuffer.wrap(part3)).setFin(true));
fuzzer.expect(expect);
}
@ -338,11 +355,11 @@ public class TestABCase6 extends AbstractABCase
{
fuzzer.connect();
fuzzer.setSendMode(Fuzzer.SendMode.BULK);
fuzzer.send(new WebSocketFrame(OpCode.TEXT).setPayload(part1).setFin(false));
fuzzer.send(new TextFrame().setPayload(ByteBuffer.wrap(part1)).setFin(false));
TimeUnit.SECONDS.sleep(1);
fuzzer.send(new WebSocketFrame(OpCode.CONTINUATION).setPayload(part2).setFin(false));
fuzzer.send(new ContinuationFrame().setPayload(ByteBuffer.wrap(part2)).setFin(false));
TimeUnit.SECONDS.sleep(1);
fuzzer.send(new WebSocketFrame(OpCode.CONTINUATION).setPayload(part3).setFin(true));
fuzzer.send(new ContinuationFrame().setPayload(ByteBuffer.wrap(part3)).setFin(true));
fuzzer.expect(expect);
}
finally
@ -369,7 +386,7 @@ public class TestABCase6 extends AbstractABCase
BufferUtil.flipToFlush(payload,0);
List<WebSocketFrame> send = new ArrayList<>();
send.add(new WebSocketFrame(OpCode.TEXT).setPayload(payload));
send.add(new TextFrame().setPayload(payload));
send.add(new CloseInfo(StatusCode.NORMAL).asFrame());
List<WebSocketFrame> expect = new ArrayList<>();
@ -395,13 +412,13 @@ public class TestABCase6 extends AbstractABCase
part3.limit(splits[2]);
fuzzer.send(part1); // the header + good utf
TimeUnit.SECONDS.sleep(1);
TimeUnit.MILLISECONDS.sleep(500);
fuzzer.send(part2); // the bad UTF
TimeUnit.MILLISECONDS.sleep(500);
fuzzer.send(part3); // the rest (shouldn't work)
fuzzer.expect(expect);
TimeUnit.SECONDS.sleep(1);
fuzzer.send(part3); // the rest (shouldn't work)
fuzzer.expectServerDisconnect(Fuzzer.DisconnectMode.UNCLEAN);
}
finally
@ -418,20 +435,17 @@ public class TestABCase6 extends AbstractABCase
@Slow
public void testCase6_4_4() throws Exception
{
// Disable Long Stacks from Parser (we know this test will throw an exception)
enableStacks(Parser.class,false);
byte invalid[] = Hex.asByteArray("CEBAE1BDB9CF83CEBCCEB5F49080808080656469746564");
List<WebSocketFrame> send = new ArrayList<>();
send.add(new WebSocketFrame(OpCode.TEXT).setPayload(invalid));
send.add(new TextFrame().setPayload(ByteBuffer.wrap(invalid)));
send.add(new CloseInfo(StatusCode.NORMAL).asFrame());
List<WebSocketFrame> expect = new ArrayList<>();
expect.add(new CloseInfo(StatusCode.BAD_PAYLOAD).asFrame());
Fuzzer fuzzer = new Fuzzer(this);
try
try(StacklessLogging scope = new StacklessLogging(Parser.class))
{
fuzzer.connect();
@ -447,7 +461,6 @@ public class TestABCase6 extends AbstractABCase
}
finally
{
enableStacks(Parser.class,true);
fuzzer.close();
}
}

View File

@ -18,17 +18,19 @@
package org.eclipse.jetty.websocket.server.ab;
import java.nio.ByteBuffer;
import java.util.ArrayList;
import java.util.Collection;
import java.util.List;
import org.eclipse.jetty.util.log.Log;
import org.eclipse.jetty.util.log.Logger;
import org.eclipse.jetty.util.log.StacklessLogging;
import org.eclipse.jetty.websocket.api.StatusCode;
import org.eclipse.jetty.websocket.common.CloseInfo;
import org.eclipse.jetty.websocket.common.OpCode;
import org.eclipse.jetty.websocket.common.Parser;
import org.eclipse.jetty.websocket.common.WebSocketFrame;
import org.eclipse.jetty.websocket.common.frames.TextFrame;
import org.eclipse.jetty.websocket.server.helper.Hex;
import org.junit.Test;
import org.junit.runner.RunWith;
@ -154,16 +156,15 @@ public class TestABCase6_BadUTF extends AbstractABCase
public void assertBadTextPayload() throws Exception
{
List<WebSocketFrame> send = new ArrayList<>();
send.add(new WebSocketFrame(OpCode.TEXT).setPayload(invalid));
send.add(new TextFrame().setPayload(ByteBuffer.wrap(invalid)));
send.add(new CloseInfo(StatusCode.NORMAL).asFrame());
List<WebSocketFrame> expect = new ArrayList<>();
expect.add(new CloseInfo(StatusCode.BAD_PAYLOAD).asFrame());
Fuzzer fuzzer = new Fuzzer(this);
try
try(StacklessLogging supress = new StacklessLogging(Parser.class))
{
enableStacks(Parser.class,false);
fuzzer.connect();
fuzzer.setSendMode(Fuzzer.SendMode.BULK);
fuzzer.send(send);
@ -173,7 +174,6 @@ public class TestABCase6_BadUTF extends AbstractABCase
finally
{
fuzzer.close();
enableStacks(Parser.class,true);
}
}
}

View File

@ -18,6 +18,7 @@
package org.eclipse.jetty.websocket.server.ab;
import java.nio.ByteBuffer;
import java.util.ArrayList;
import java.util.Collection;
import java.util.List;
@ -26,8 +27,8 @@ import org.eclipse.jetty.util.log.Log;
import org.eclipse.jetty.util.log.Logger;
import org.eclipse.jetty.websocket.api.StatusCode;
import org.eclipse.jetty.websocket.common.CloseInfo;
import org.eclipse.jetty.websocket.common.OpCode;
import org.eclipse.jetty.websocket.common.WebSocketFrame;
import org.eclipse.jetty.websocket.common.frames.TextFrame;
import org.eclipse.jetty.websocket.server.helper.Hex;
import org.junit.Test;
import org.junit.runner.RunWith;
@ -118,23 +119,23 @@ public class TestABCase6_GoodUTF extends AbstractABCase
return data;
}
private final byte[] msg;
private final ByteBuffer msg;
public TestABCase6_GoodUTF(String testId, String hexMsg)
{
LOG.debug("Test ID: {}",testId);
this.msg = Hex.asByteArray(hexMsg);
this.msg = Hex.asByteBuffer(hexMsg);
}
@Test
public void assertEchoTextMessage() throws Exception
{
List<WebSocketFrame> send = new ArrayList<>();
send.add(new WebSocketFrame(OpCode.TEXT).setPayload(msg));
send.add(new TextFrame().setPayload(msg));
send.add(new CloseInfo(StatusCode.NORMAL).asFrame());
List<WebSocketFrame> expect = new ArrayList<>();
expect.add(new WebSocketFrame(OpCode.TEXT).setPayload(copyOf(msg)));
expect.add(new TextFrame().setPayload(clone(msg)));
expect.add(new CloseInfo(StatusCode.NORMAL).asFrame());
Fuzzer fuzzer = new Fuzzer(this);

View File

@ -26,11 +26,16 @@ import java.util.List;
import org.eclipse.jetty.toolchain.test.TestTracker;
import org.eclipse.jetty.util.BufferUtil;
import org.eclipse.jetty.util.StringUtil;
import org.eclipse.jetty.util.log.StacklessLogging;
import org.eclipse.jetty.websocket.api.StatusCode;
import org.eclipse.jetty.websocket.common.CloseInfo;
import org.eclipse.jetty.websocket.common.OpCode;
import org.eclipse.jetty.websocket.common.Parser;
import org.eclipse.jetty.websocket.common.WebSocketFrame;
import org.eclipse.jetty.websocket.common.frames.CloseFrame;
import org.eclipse.jetty.websocket.common.frames.ContinuationFrame;
import org.eclipse.jetty.websocket.common.frames.PingFrame;
import org.eclipse.jetty.websocket.common.frames.TextFrame;
import org.eclipse.jetty.websocket.server.helper.Hex;
import org.junit.Rule;
import org.junit.Test;
@ -107,7 +112,7 @@ public class TestABCase7 extends AbstractABCase
{
List<WebSocketFrame> send = new ArrayList<>();
send.add(new CloseInfo(StatusCode.NORMAL).asFrame());
send.add(WebSocketFrame.ping().setPayload("out of band ping"));
send.add(new PingFrame("out of band ping"));
List<WebSocketFrame> expect = new ArrayList<>();
expect.add(new CloseInfo(StatusCode.NORMAL).asFrame());
@ -162,9 +167,9 @@ public class TestABCase7 extends AbstractABCase
public void testCase7_1_5() throws Exception
{
List<WebSocketFrame> send = new ArrayList<>();
send.add(new WebSocketFrame(OpCode.TEXT).setPayload("an").setFin(false));
send.add(new TextFrame().setPayload("an").setFin(false));
send.add(new CloseInfo(StatusCode.NORMAL).asFrame());
send.add(new WebSocketFrame(OpCode.CONTINUATION).setPayload("ticipation").setFin(true));
send.add(new ContinuationFrame("ticipation").setFin(true));
List<WebSocketFrame> expect = new ArrayList<>();
expect.add(new CloseInfo(StatusCode.NORMAL).asFrame());
@ -192,14 +197,15 @@ public class TestABCase7 extends AbstractABCase
{
byte msg[] = new byte[256 * 1024];
Arrays.fill(msg,(byte)'*');
ByteBuffer buf = ByteBuffer.wrap(msg);
List<WebSocketFrame> send = new ArrayList<>();
send.add(new WebSocketFrame(OpCode.TEXT).setPayload(msg));
send.add(new TextFrame().setPayload(buf));
send.add(new CloseInfo(StatusCode.NORMAL).asFrame());
send.add(new WebSocketFrame(OpCode.PING).setPayload("out of band"));
send.add(new PingFrame("out of band"));
List<WebSocketFrame> expect = new ArrayList<>();
expect.add(new WebSocketFrame(OpCode.TEXT).setPayload(copyOf(msg)));
expect.add(new TextFrame().setPayload(clone(buf)));
expect.add(new CloseInfo(StatusCode.NORMAL).asFrame());
Fuzzer fuzzer = new Fuzzer(this);
@ -224,10 +230,10 @@ public class TestABCase7 extends AbstractABCase
public void testCase7_3_1() throws Exception
{
List<WebSocketFrame> send = new ArrayList<>();
send.add(new WebSocketFrame(OpCode.CLOSE));
send.add(new CloseFrame());
List<WebSocketFrame> expect = new ArrayList<>();
expect.add(new WebSocketFrame(OpCode.CLOSE));
expect.add(new CloseFrame());
Fuzzer fuzzer = new Fuzzer(this);
try
@ -252,17 +258,17 @@ public class TestABCase7 extends AbstractABCase
{
byte payload[] = new byte[]
{ 0x00 };
ByteBuffer buf = ByteBuffer.wrap(payload);
List<WebSocketFrame> send = new ArrayList<>();
send.add(new WebSocketFrame(OpCode.CLOSE).setPayload(payload));
send.add(new CloseFrame().setPayload(buf));
List<WebSocketFrame> expect = new ArrayList<>();
expect.add(new CloseInfo(StatusCode.PROTOCOL).asFrame());
Fuzzer fuzzer = new Fuzzer(this);
try
try(StacklessLogging scope = new StacklessLogging(Parser.class))
{
enableStacks(Parser.class,false);
fuzzer.connect();
fuzzer.setSendMode(Fuzzer.SendMode.BULK);
fuzzer.send(send);
@ -271,7 +277,6 @@ public class TestABCase7 extends AbstractABCase
}
finally
{
enableStacks(Parser.class,true);
fuzzer.close();
}
}
@ -376,18 +381,16 @@ public class TestABCase7 extends AbstractABCase
BufferUtil.flipToFlush(payload,0);
List<WebSocketFrame> send = new ArrayList<>();
WebSocketFrame close = new WebSocketFrame(); // anonymous (no opcode) intentionally
WebSocketFrame close = new BadFrame(OpCode.CLOSE);
close.setPayload(payload); // intentionally bad payload
close.setOpCode(OpCode.CLOSE); // set opcode after payload (to prevent early bad payload detection)
send.add(close);
List<WebSocketFrame> expect = new ArrayList<>();
expect.add(new CloseInfo(StatusCode.BAD_PAYLOAD).asFrame());
Fuzzer fuzzer = new Fuzzer(this);
try
try(StacklessLogging scope = new StacklessLogging(Parser.class))
{
enableStacks(Parser.class,false);
fuzzer.connect();
fuzzer.setSendMode(Fuzzer.SendMode.BULK);
fuzzer.send(send);
@ -396,7 +399,6 @@ public class TestABCase7 extends AbstractABCase
}
finally
{
enableStacks(Parser.class,true);
fuzzer.close();
}
}

View File

@ -29,8 +29,8 @@ import org.eclipse.jetty.util.log.Log;
import org.eclipse.jetty.util.log.Logger;
import org.eclipse.jetty.websocket.api.StatusCode;
import org.eclipse.jetty.websocket.common.CloseInfo;
import org.eclipse.jetty.websocket.common.OpCode;
import org.eclipse.jetty.websocket.common.WebSocketFrame;
import org.eclipse.jetty.websocket.common.frames.CloseFrame;
import org.junit.Test;
import org.junit.runner.RunWith;
import org.junit.runners.Parameterized;
@ -92,7 +92,7 @@ public class TestABCase7_BadStatusCodes extends AbstractABCase
BufferUtil.flipToFlush(payload,0);
List<WebSocketFrame> send = new ArrayList<>();
send.add(new WebSocketFrame(OpCode.CLOSE).setPayload(payload.slice()));
send.add(new CloseFrame().setPayload(payload.slice()));
List<WebSocketFrame> expect = new ArrayList<>();
expect.add(new CloseInfo(StatusCode.PROTOCOL).asFrame());
@ -125,7 +125,7 @@ public class TestABCase7_BadStatusCodes extends AbstractABCase
BufferUtil.flipToFlush(payload,0);
List<WebSocketFrame> send = new ArrayList<>();
send.add(new WebSocketFrame(OpCode.CLOSE).setPayload(payload.slice()));
send.add(new CloseFrame().setPayload(payload.slice()));
List<WebSocketFrame> expect = new ArrayList<>();
expect.add(new CloseInfo(StatusCode.PROTOCOL).asFrame());

View File

@ -27,8 +27,8 @@ import org.eclipse.jetty.util.BufferUtil;
import org.eclipse.jetty.util.StringUtil;
import org.eclipse.jetty.util.log.Log;
import org.eclipse.jetty.util.log.Logger;
import org.eclipse.jetty.websocket.common.OpCode;
import org.eclipse.jetty.websocket.common.WebSocketFrame;
import org.eclipse.jetty.websocket.common.frames.CloseFrame;
import org.junit.Test;
import org.junit.runner.RunWith;
import org.junit.runners.Parameterized;
@ -87,10 +87,10 @@ public class TestABCase7_GoodStatusCodes extends AbstractABCase
BufferUtil.flipToFlush(payload,0);
List<WebSocketFrame> send = new ArrayList<>();
send.add(new WebSocketFrame(OpCode.CLOSE).setPayload(payload.slice()));
send.add(new CloseFrame().setPayload(payload.slice()));
List<WebSocketFrame> expect = new ArrayList<>();
expect.add(new WebSocketFrame(OpCode.CLOSE).setPayload(copyOf(payload)));
expect.add(new CloseFrame().setPayload(clone(payload)));
Fuzzer fuzzer = new Fuzzer(this);
try
@ -114,16 +114,15 @@ public class TestABCase7_GoodStatusCodes extends AbstractABCase
public void testStatusCodeWithReason() throws Exception
{
ByteBuffer payload = ByteBuffer.allocate(256);
BufferUtil.clearToFill(payload);
payload.putChar((char)statusCode);
payload.put(StringUtil.getBytes("Reason"));
BufferUtil.flipToFlush(payload,0);
payload.flip();
List<WebSocketFrame> send = new ArrayList<>();
send.add(new WebSocketFrame(OpCode.CLOSE).setPayload(payload.slice()));
send.add(new CloseFrame().setPayload(payload.slice()));
List<WebSocketFrame> expect = new ArrayList<>();
expect.add(new WebSocketFrame(OpCode.CLOSE).setPayload(copyOf(payload)));
expect.add(new CloseFrame().setPayload(clone(payload)));
Fuzzer fuzzer = new Fuzzer(this);
try

View File

@ -18,6 +18,7 @@
package org.eclipse.jetty.websocket.server.ab;
import java.nio.ByteBuffer;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.List;
@ -30,6 +31,10 @@ import org.eclipse.jetty.websocket.api.StatusCode;
import org.eclipse.jetty.websocket.common.CloseInfo;
import org.eclipse.jetty.websocket.common.OpCode;
import org.eclipse.jetty.websocket.common.WebSocketFrame;
import org.eclipse.jetty.websocket.common.frames.BinaryFrame;
import org.eclipse.jetty.websocket.common.frames.ContinuationFrame;
import org.eclipse.jetty.websocket.common.frames.DataFrame;
import org.eclipse.jetty.websocket.common.frames.TextFrame;
import org.junit.Test;
import org.junit.runner.RunWith;
@ -42,6 +47,21 @@ public class TestABCase9 extends AbstractABCase
private static final int KBYTE = 1024;
private static final int MBYTE = KBYTE * KBYTE;
private DataFrame toDataFrame(byte op)
{
switch (op)
{
case OpCode.BINARY:
return new BinaryFrame();
case OpCode.TEXT:
return new TextFrame();
case OpCode.CONTINUATION:
return new ContinuationFrame();
default:
throw new IllegalArgumentException("Not a data frame: " + op);
}
}
private void assertMultiFrameEcho(byte opcode, int overallMsgSize, int fragmentSize) throws Exception
{
byte msg[] = new byte[overallMsgSize];
@ -52,6 +72,8 @@ public class TestABCase9 extends AbstractABCase
int remaining = msg.length;
int offset = 0;
boolean fin;
ByteBuffer buf;
;
byte op = opcode;
while (remaining > 0)
{
@ -60,14 +82,17 @@ public class TestABCase9 extends AbstractABCase
System.arraycopy(msg,offset,frag,0,len);
remaining -= len;
fin = (remaining <= 0);
send.add(new WebSocketFrame(op).setPayload(frag).setFin(fin));
buf = ByteBuffer.wrap(frag);
send.add(toDataFrame(op).setPayload(buf).setFin(fin));
offset += len;
op = OpCode.CONTINUATION;
}
send.add(new CloseInfo(StatusCode.NORMAL).asFrame());
List<WebSocketFrame> expect = new ArrayList<>();
expect.add(new WebSocketFrame(opcode).setPayload(copyOf(msg)));
expect.add(toDataFrame(opcode).setPayload(copyOf(msg)));
expect.add(new CloseInfo(StatusCode.NORMAL).asFrame());
Fuzzer fuzzer = new Fuzzer(this);
@ -88,13 +113,14 @@ public class TestABCase9 extends AbstractABCase
{
byte msg[] = new byte[overallMsgSize];
Arrays.fill(msg,(byte)'M');
ByteBuffer buf = ByteBuffer.wrap(msg);
List<WebSocketFrame> send = new ArrayList<>();
send.add(new WebSocketFrame(opcode).setPayload(msg));
send.add(toDataFrame(opcode).setPayload(buf));
send.add(new CloseInfo(StatusCode.NORMAL).asFrame());
List<WebSocketFrame> expect = new ArrayList<>();
expect.add(new WebSocketFrame(opcode).setPayload(copyOf(msg)));
expect.add(toDataFrame(opcode).setPayload(clone(buf)));
expect.add(new CloseInfo(StatusCode.NORMAL).asFrame());
Fuzzer fuzzer = new Fuzzer(this);
@ -152,13 +178,14 @@ public class TestABCase9 extends AbstractABCase
{
byte utf[] = new byte[256 * KBYTE];
Arrays.fill(utf,(byte)'y');
ByteBuffer buf = ByteBuffer.wrap(utf);
List<WebSocketFrame> send = new ArrayList<>();
send.add(WebSocketFrame.text().setPayload(utf));
send.add(new TextFrame().setPayload(buf));
send.add(new CloseInfo(StatusCode.NORMAL).asFrame());
List<WebSocketFrame> expect = new ArrayList<>();
expect.add(WebSocketFrame.text().setPayload(copyOf(utf)));
expect.add(WebSocketFrame.text().setPayload(clone(buf)));
expect.add(new CloseInfo(StatusCode.NORMAL).asFrame());
Fuzzer fuzzer = new Fuzzer(this);
@ -184,13 +211,14 @@ public class TestABCase9 extends AbstractABCase
{
byte utf[] = new byte[1 * MBYTE];
Arrays.fill(utf,(byte)'y');
ByteBuffer buf = ByteBuffer.wrap(utf);
List<WebSocketFrame> send = new ArrayList<>();
send.add(WebSocketFrame.text().setPayload(utf));
send.add(WebSocketFrame.text().setPayload(buf));
send.add(new CloseInfo(StatusCode.NORMAL).asFrame());
List<WebSocketFrame> expect = new ArrayList<>();
expect.add(WebSocketFrame.text().setPayload(utf));
expect.add(WebSocketFrame.text().setPayload(clone(buf)));
expect.add(new CloseInfo(StatusCode.NORMAL).asFrame());
Fuzzer fuzzer = new Fuzzer(this);
@ -216,13 +244,14 @@ public class TestABCase9 extends AbstractABCase
{
byte utf[] = new byte[4 * MBYTE];
Arrays.fill(utf,(byte)'y');
ByteBuffer buf = ByteBuffer.wrap(utf);
List<WebSocketFrame> send = new ArrayList<>();
send.add(WebSocketFrame.text().setPayload(utf));
send.add(WebSocketFrame.text().setPayload(buf));
send.add(new CloseInfo(StatusCode.NORMAL).asFrame());
List<WebSocketFrame> expect = new ArrayList<>();
expect.add(WebSocketFrame.text().setPayload(utf));
expect.add(WebSocketFrame.text().setPayload(clone(buf)));
expect.add(new CloseInfo(StatusCode.NORMAL).asFrame());
Fuzzer fuzzer = new Fuzzer(this);
@ -248,13 +277,14 @@ public class TestABCase9 extends AbstractABCase
{
byte utf[] = new byte[8 * MBYTE];
Arrays.fill(utf,(byte)'y');
ByteBuffer buf = ByteBuffer.wrap(utf);
List<WebSocketFrame> send = new ArrayList<>();
send.add(WebSocketFrame.text().setPayload(utf));
send.add(WebSocketFrame.text().setPayload(buf));
send.add(new CloseInfo(StatusCode.NORMAL).asFrame());
List<WebSocketFrame> expect = new ArrayList<>();
expect.add(WebSocketFrame.text().setPayload(utf));
expect.add(WebSocketFrame.text().setPayload(clone(buf)));
expect.add(new CloseInfo(StatusCode.NORMAL).asFrame());
Fuzzer fuzzer = new Fuzzer(this);
@ -280,13 +310,14 @@ public class TestABCase9 extends AbstractABCase
{
byte utf[] = new byte[16 * MBYTE];
Arrays.fill(utf,(byte)'y');
ByteBuffer buf = ByteBuffer.wrap(utf);
List<WebSocketFrame> send = new ArrayList<>();
send.add(WebSocketFrame.text().setPayload(utf));
send.add(WebSocketFrame.text().setPayload(buf));
send.add(new CloseInfo(StatusCode.NORMAL).asFrame());
List<WebSocketFrame> expect = new ArrayList<>();
expect.add(WebSocketFrame.text().setPayload(utf));
expect.add(WebSocketFrame.text().setPayload(clone(buf)));
expect.add(new CloseInfo(StatusCode.NORMAL).asFrame());
Fuzzer fuzzer = new Fuzzer(this);
@ -342,13 +373,14 @@ public class TestABCase9 extends AbstractABCase
{
byte data[] = new byte[256 * KBYTE];
Arrays.fill(data,(byte)0x22);
ByteBuffer buf = ByteBuffer.wrap(data);
List<WebSocketFrame> send = new ArrayList<>();
send.add(WebSocketFrame.binary().setPayload(data));
send.add(WebSocketFrame.binary().setPayload(buf));
send.add(new CloseInfo(StatusCode.NORMAL).asFrame());
List<WebSocketFrame> expect = new ArrayList<>();
expect.add(WebSocketFrame.binary().setPayload(copyOf(data)));
expect.add(WebSocketFrame.binary().setPayload(clone(buf)));
expect.add(new CloseInfo(StatusCode.NORMAL).asFrame());
Fuzzer fuzzer = new Fuzzer(this);
@ -374,13 +406,14 @@ public class TestABCase9 extends AbstractABCase
{
byte data[] = new byte[1 * MBYTE];
Arrays.fill(data,(byte)0x23);
ByteBuffer buf = ByteBuffer.wrap(data);
List<WebSocketFrame> send = new ArrayList<>();
send.add(WebSocketFrame.binary().setPayload(data));
send.add(WebSocketFrame.binary().setPayload(buf));
send.add(new CloseInfo(StatusCode.NORMAL).asFrame());
List<WebSocketFrame> expect = new ArrayList<>();
expect.add(WebSocketFrame.binary().setPayload(data));
expect.add(WebSocketFrame.binary().setPayload(clone(buf)));
expect.add(new CloseInfo(StatusCode.NORMAL).asFrame());
Fuzzer fuzzer = new Fuzzer(this);
@ -406,13 +439,14 @@ public class TestABCase9 extends AbstractABCase
{
byte data[] = new byte[4 * MBYTE];
Arrays.fill(data,(byte)0x24);
ByteBuffer buf = ByteBuffer.wrap(data);
List<WebSocketFrame> send = new ArrayList<>();
send.add(WebSocketFrame.binary().setPayload(data));
send.add(WebSocketFrame.binary().setPayload(buf));
send.add(new CloseInfo(StatusCode.NORMAL).asFrame());
List<WebSocketFrame> expect = new ArrayList<>();
expect.add(WebSocketFrame.binary().setPayload(data));
expect.add(WebSocketFrame.binary().setPayload(clone(buf)));
expect.add(new CloseInfo(StatusCode.NORMAL).asFrame());
Fuzzer fuzzer = new Fuzzer(this);
@ -438,13 +472,14 @@ public class TestABCase9 extends AbstractABCase
{
byte data[] = new byte[8 * MBYTE];
Arrays.fill(data,(byte)0x25);
ByteBuffer buf = ByteBuffer.wrap(data);
List<WebSocketFrame> send = new ArrayList<>();
send.add(WebSocketFrame.binary().setPayload(data));
send.add(WebSocketFrame.binary().setPayload(buf));
send.add(new CloseInfo(StatusCode.NORMAL).asFrame());
List<WebSocketFrame> expect = new ArrayList<>();
expect.add(WebSocketFrame.binary().setPayload(data));
expect.add(WebSocketFrame.binary().setPayload(clone(buf)));
expect.add(new CloseInfo(StatusCode.NORMAL).asFrame());
Fuzzer fuzzer = new Fuzzer(this);
@ -470,13 +505,14 @@ public class TestABCase9 extends AbstractABCase
{
byte data[] = new byte[16 * MBYTE];
Arrays.fill(data,(byte)0x26);
ByteBuffer buf = ByteBuffer.wrap(data);
List<WebSocketFrame> send = new ArrayList<>();
send.add(WebSocketFrame.binary().setPayload(data));
send.add(WebSocketFrame.binary().setPayload(buf));
send.add(new CloseInfo(StatusCode.NORMAL).asFrame());
List<WebSocketFrame> expect = new ArrayList<>();
expect.add(WebSocketFrame.binary().setPayload(data));
expect.add(WebSocketFrame.binary().setPayload(clone(buf)));
expect.add(new CloseInfo(StatusCode.NORMAL).asFrame());
Fuzzer fuzzer = new Fuzzer(this);

View File

@ -434,7 +434,7 @@ public class BlockheadClient implements IncomingFrames, OutgoingFrames, Connecti
ioState.onCloseRemote(close);
}
WebSocketFrame copy = new WebSocketFrame(frame);
WebSocketFrame copy = WebSocketFrame.copy(frame);
incomingFrames.incomingFrame(copy);
}

View File

@ -18,6 +18,8 @@
package org.eclipse.jetty.websocket.server.helper;
import java.nio.ByteBuffer;
public final class Hex
{
private static final char[] hexcodes = "0123456789ABCDEF".toCharArray();
@ -52,6 +54,11 @@ public final class Hex
return buf;
}
public static ByteBuffer asByteBuffer(String hstr)
{
return ByteBuffer.wrap(asByteArray(hstr));
}
public static String asHex(byte buf[])
{
int len = buf.length;

View File

@ -136,7 +136,7 @@ public class IncomingFramesCapture implements IncomingFrames
@Override
public void incomingFrame(Frame frame)
{
WebSocketFrame copy = new WebSocketFrame(frame);
WebSocketFrame copy = WebSocketFrame.copy(frame);
frames.add(copy);
}

View File

@ -93,7 +93,7 @@ public class OutgoingFramesCapture implements OutgoingFrames
@Override
public void outgoingFrame(Frame frame, WriteCallback callback)
{
WebSocketFrame copy = new WebSocketFrame(frame);
WebSocketFrame copy = WebSocketFrame.copy(frame);
frames.add(copy);
if (callback != null)
{

View File

@ -2,7 +2,7 @@ org.eclipse.jetty.util.log.class=org.eclipse.jetty.util.log.StdErrLog
org.eclipse.jetty.LEVEL=WARN
# org.eclipse.jetty.io.WriteFlusher.LEVEL=DEBUG
# org.eclipse.jetty.websocket.LEVEL=DEBUG
org.eclipse.jetty.websocket.LEVEL=DEBUG
# org.eclipse.jetty.websocket.LEVEL=WARN
# org.eclipse.jetty.websocket.common.io.LEVEL=DEBUG
# org.eclipse.jetty.websocket.server.ab.LEVEL=DEBUG