395444 - Websockets not working with Chrome (deflate problem)
+ Hopefully final fix to deflate-frame + Splitting out extension named 'deflate-frame' (last spec'd standard) from 'x-webkit-deflate-frame' (standard in use by chrome + safari)
This commit is contained in:
parent
ac0e30bbb7
commit
20fbd95bf1
|
@ -0,0 +1,228 @@
|
|||
//
|
||||
// ========================================================================
|
||||
// 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.extensions.compress;
|
||||
|
||||
import java.nio.ByteBuffer;
|
||||
import java.util.zip.DataFormatException;
|
||||
import java.util.zip.Deflater;
|
||||
import java.util.zip.Inflater;
|
||||
|
||||
import org.eclipse.jetty.util.BufferUtil;
|
||||
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.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.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">deflate-frame</a> extension seen out in the
|
||||
* wild.
|
||||
*/
|
||||
public class DeflateFrameExtension extends AbstractExtension
|
||||
{
|
||||
private static final boolean BFINAL_HACK = Boolean.parseBoolean(System.getProperty("jetty.websocket.bfinal.hack","true"));
|
||||
private static final Logger LOG = Log.getLogger(DeflateFrameExtension.class);
|
||||
|
||||
private static final int OVERHEAD = 64;
|
||||
/** Tail Bytes per Spec */
|
||||
private static final byte[] TAIL = new byte[]
|
||||
{ 0x00, 0x00, (byte)0xFF, (byte)0xFF };
|
||||
private int bufferSize = 64 * 1024;
|
||||
private Deflater compressor;
|
||||
private Inflater decompressor;
|
||||
|
||||
@Override
|
||||
public String getName()
|
||||
{
|
||||
return "deflate-frame";
|
||||
}
|
||||
|
||||
@Override
|
||||
public synchronized void incomingFrame(Frame frame)
|
||||
{
|
||||
if (OpCode.isControlFrame(frame.getOpCode()) || !frame.isRsv1())
|
||||
{
|
||||
// Cannot modify incoming control frames or ones with RSV1 set.
|
||||
nextIncomingFrame(frame);
|
||||
return;
|
||||
}
|
||||
|
||||
if (!frame.hasPayload())
|
||||
{
|
||||
// no payload? nothing to do.
|
||||
nextIncomingFrame(frame);
|
||||
return;
|
||||
}
|
||||
|
||||
// Prime the decompressor
|
||||
ByteBuffer payload = frame.getPayload();
|
||||
int inlen = payload.remaining();
|
||||
byte compressed[] = new byte[inlen + TAIL.length];
|
||||
payload.get(compressed,0,inlen);
|
||||
System.arraycopy(TAIL,0,compressed,inlen,TAIL.length);
|
||||
decompressor.setInput(compressed,0,compressed.length);
|
||||
|
||||
// Perform decompression
|
||||
while (decompressor.getRemaining() > 0 && !decompressor.finished())
|
||||
{
|
||||
DataFrame out = new DataFrame(frame);
|
||||
out.setRsv1(false); // Unset RSV1
|
||||
byte outbuf[] = new byte[Math.min(inlen * 2,bufferSize)];
|
||||
try
|
||||
{
|
||||
int len = decompressor.inflate(outbuf);
|
||||
if (len == 0)
|
||||
{
|
||||
if (decompressor.needsInput())
|
||||
{
|
||||
throw new BadPayloadException("Unable to inflate frame, not enough input on frame");
|
||||
}
|
||||
if (decompressor.needsDictionary())
|
||||
{
|
||||
throw new BadPayloadException("Unable to inflate frame, frame erroneously says it needs a dictionary");
|
||||
}
|
||||
}
|
||||
if (len > 0)
|
||||
{
|
||||
out.setPayload(ByteBuffer.wrap(outbuf,0,len));
|
||||
}
|
||||
nextIncomingFrame(out);
|
||||
}
|
||||
catch (DataFormatException e)
|
||||
{
|
||||
LOG.warn(e);
|
||||
throw new BadPayloadException(e);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Indicates use of RSV1 flag for indicating deflation is in use.
|
||||
* <p>
|
||||
* Also known as the "COMP" framing header bit
|
||||
*/
|
||||
@Override
|
||||
public boolean isRsv1User()
|
||||
{
|
||||
return true;
|
||||
}
|
||||
|
||||
@Override
|
||||
public synchronized void outgoingFrame(Frame frame, WriteCallback callback)
|
||||
{
|
||||
if (OpCode.isControlFrame(frame.getOpCode()))
|
||||
{
|
||||
// skip, cannot compress control frames.
|
||||
nextOutgoingFrame(frame,callback);
|
||||
return;
|
||||
}
|
||||
|
||||
if (!frame.hasPayload())
|
||||
{
|
||||
// pass through, nothing to do
|
||||
nextOutgoingFrame(frame,callback);
|
||||
return;
|
||||
}
|
||||
|
||||
if (LOG.isDebugEnabled())
|
||||
{
|
||||
LOG.debug("outgoingFrame({}, {}) - {}",OpCode.name(frame.getOpCode()),callback != null?callback.getClass().getSimpleName():"<null>",
|
||||
BufferUtil.toDetailString(frame.getPayload()));
|
||||
}
|
||||
|
||||
// Prime the compressor
|
||||
byte uncompressed[] = BufferUtil.toArray(frame.getPayload());
|
||||
|
||||
// Perform the compression
|
||||
if (!compressor.finished())
|
||||
{
|
||||
compressor.setInput(uncompressed,0,uncompressed.length);
|
||||
byte compressed[] = new byte[uncompressed.length + OVERHEAD];
|
||||
|
||||
while (!compressor.needsInput())
|
||||
{
|
||||
int len = compressor.deflate(compressed,0,compressed.length,Deflater.SYNC_FLUSH);
|
||||
ByteBuffer outbuf = getBufferPool().acquire(len,true);
|
||||
BufferUtil.clearToFill(outbuf);
|
||||
|
||||
if (len > 0)
|
||||
{
|
||||
outbuf.put(compressed,0,len - 4);
|
||||
}
|
||||
|
||||
BufferUtil.flipToFlush(outbuf,0);
|
||||
|
||||
if (len > 0 && BFINAL_HACK)
|
||||
{
|
||||
/*
|
||||
* Per the spec, it says that BFINAL 1 or 0 are allowed.
|
||||
*
|
||||
* However, Java always uses BFINAL 1, whereas the browsers Chromium and Safari fail to decompress when it encounters BFINAL 1.
|
||||
*
|
||||
* This hack will always set BFINAL 0
|
||||
*/
|
||||
byte b0 = outbuf.get(0);
|
||||
if ((b0 & 1) != 0) // if BFINAL 1
|
||||
{
|
||||
outbuf.put(0,(b0 ^= 1)); // flip bit to BFINAL 0
|
||||
}
|
||||
}
|
||||
|
||||
DataFrame out = new DataFrame(frame);
|
||||
out.setRsv1(true);
|
||||
out.setPooledBuffer(true);
|
||||
out.setPayload(outbuf);
|
||||
|
||||
if (!compressor.needsInput())
|
||||
{
|
||||
// this is fragmented
|
||||
out.setFin(false);
|
||||
nextOutgoingFrame(out,null); // non final frames have no callback
|
||||
}
|
||||
else
|
||||
{
|
||||
// pass through the callback
|
||||
nextOutgoingFrame(out,callback);
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
@Override
|
||||
public void setConfig(ExtensionConfig config)
|
||||
{
|
||||
super.setConfig(config);
|
||||
|
||||
boolean nowrap = true;
|
||||
compressor = new Deflater(Deflater.BEST_COMPRESSION,nowrap);
|
||||
compressor.setStrategy(Deflater.DEFAULT_STRATEGY);
|
||||
|
||||
decompressor = new Inflater(nowrap);
|
||||
}
|
||||
|
||||
@Override
|
||||
public String toString()
|
||||
{
|
||||
return this.getClass().getSimpleName() + "[]";
|
||||
}
|
||||
}
|
|
@ -1,131 +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.extensions.compress;
|
||||
|
||||
|
||||
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.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
|
||||
* in the wild.
|
||||
*/
|
||||
public class FrameCompressionExtension extends AbstractExtension
|
||||
{
|
||||
private CompressionMethod method = new DeflateCompressionMethod();
|
||||
|
||||
@Override
|
||||
public String getName()
|
||||
{
|
||||
return "x-webkit-deflate-frame";
|
||||
}
|
||||
|
||||
@Override
|
||||
public synchronized void incomingFrame(Frame frame)
|
||||
{
|
||||
if (OpCode.isControlFrame(frame.getOpCode()) || !frame.isRsv1())
|
||||
{
|
||||
// Cannot modify incoming control frames or ones with RSV1 set.
|
||||
nextIncomingFrame(frame);
|
||||
return;
|
||||
}
|
||||
|
||||
ByteBuffer data = frame.getPayload();
|
||||
method.decompress().input(data);
|
||||
while (!method.decompress().isDone())
|
||||
{
|
||||
ByteBuffer uncompressed = method.decompress().process();
|
||||
DataFrame out = new DataFrame(frame);
|
||||
out.setPayload(uncompressed);
|
||||
if (!method.decompress().isDone())
|
||||
{
|
||||
out.setFin(false);
|
||||
}
|
||||
out.setRsv1(false); // Unset RSV1 on decompressed frame
|
||||
nextIncomingFrame(out);
|
||||
}
|
||||
|
||||
// reset on every frame.
|
||||
// method.decompress().end();
|
||||
}
|
||||
|
||||
/**
|
||||
* Indicates use of RSV1 flag for indicating deflation is in use.
|
||||
* <p>
|
||||
* Also known as the "COMP" framing header bit
|
||||
*/
|
||||
@Override
|
||||
public boolean isRsv1User()
|
||||
{
|
||||
return true;
|
||||
}
|
||||
|
||||
@Override
|
||||
public synchronized void outgoingFrame(Frame frame, WriteCallback callback)
|
||||
{
|
||||
if (OpCode.isControlFrame(frame.getOpCode()))
|
||||
{
|
||||
// skip, cannot compress control frames.
|
||||
nextOutgoingFrame(frame,callback);
|
||||
return;
|
||||
}
|
||||
|
||||
ByteBuffer data = frame.getPayload();
|
||||
|
||||
// deflate data
|
||||
method.compress().input(data);
|
||||
while (!method.compress().isDone())
|
||||
{
|
||||
ByteBuffer buf = method.compress().process();
|
||||
DataFrame out = new DataFrame(frame);
|
||||
out.setPayload(buf);
|
||||
out.setRsv1(true);
|
||||
if (!method.compress().isDone())
|
||||
{
|
||||
out.setFin(false);
|
||||
nextOutgoingFrame(frame,null); // no callback for start/end frames
|
||||
}
|
||||
else
|
||||
{
|
||||
nextOutgoingFrame(out,callback); // pass thru callback
|
||||
}
|
||||
}
|
||||
|
||||
// reset on every frame.
|
||||
method.compress().end();
|
||||
}
|
||||
|
||||
@Override
|
||||
public void setConfig(ExtensionConfig config)
|
||||
{
|
||||
super.setConfig(config);
|
||||
}
|
||||
|
||||
@Override
|
||||
public String toString()
|
||||
{
|
||||
return this.getClass().getSimpleName() + "[]";
|
||||
}
|
||||
}
|
|
@ -0,0 +1,38 @@
|
|||
//
|
||||
// ========================================================================
|
||||
// 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.extensions.compress;
|
||||
|
||||
/**
|
||||
* 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
|
||||
* in the wild. Using the alternate extension identification
|
||||
*/
|
||||
public class XWebkitDeflateFrameExtension extends DeflateFrameExtension
|
||||
{
|
||||
@Override
|
||||
public String getName()
|
||||
{
|
||||
return "x-webkit-deflate-frame";
|
||||
}
|
||||
|
||||
@Override
|
||||
public String toString()
|
||||
{
|
||||
return this.getClass().getSimpleName() + "[]";
|
||||
}
|
||||
}
|
|
@ -18,6 +18,7 @@
|
|||
|
||||
package org.eclipse.jetty.websocket.common.frames;
|
||||
|
||||
import org.eclipse.jetty.io.ByteBufferPool;
|
||||
import org.eclipse.jetty.websocket.api.extensions.Frame;
|
||||
import org.eclipse.jetty.websocket.common.OpCode;
|
||||
import org.eclipse.jetty.websocket.common.WebSocketFrame;
|
||||
|
@ -27,6 +28,8 @@ import org.eclipse.jetty.websocket.common.WebSocketFrame;
|
|||
*/
|
||||
public class DataFrame extends WebSocketFrame
|
||||
{
|
||||
private boolean isPooledBuffer = false;
|
||||
|
||||
protected DataFrame(byte opcode)
|
||||
{
|
||||
super(opcode);
|
||||
|
@ -63,14 +66,6 @@ public class DataFrame extends WebSocketFrame
|
|||
/* 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()
|
||||
{
|
||||
|
@ -82,4 +77,28 @@ public class DataFrame extends WebSocketFrame
|
|||
{
|
||||
return true;
|
||||
}
|
||||
|
||||
/**
|
||||
* @return true if payload buffer is from a {@link ByteBufferPool} and can be released when appropriate to do so
|
||||
*/
|
||||
public boolean isPooledBuffer()
|
||||
{
|
||||
return isPooledBuffer;
|
||||
}
|
||||
|
||||
/**
|
||||
* Set the data frame to continuation mode
|
||||
*/
|
||||
public void setIsContinuation()
|
||||
{
|
||||
setOpCode(OpCode.CONTINUATION);
|
||||
}
|
||||
|
||||
/**
|
||||
* Sets a flag indicating that the underlying payload is from a {@link ByteBufferPool} and can be released when appropriate to do so
|
||||
*/
|
||||
public void setPooledBuffer(boolean isPooledBuffer)
|
||||
{
|
||||
this.isPooledBuffer = isPooledBuffer;
|
||||
}
|
||||
}
|
||||
|
|
|
@ -35,6 +35,7 @@ import org.eclipse.jetty.util.log.Logger;
|
|||
import org.eclipse.jetty.websocket.api.extensions.Frame;
|
||||
import org.eclipse.jetty.websocket.common.Generator;
|
||||
import org.eclipse.jetty.websocket.common.OpCode;
|
||||
import org.eclipse.jetty.websocket.common.frames.DataFrame;
|
||||
|
||||
/**
|
||||
* Interface for working with bytes destined for {@link EndPoint#write(Callback, ByteBuffer...)}
|
||||
|
@ -101,6 +102,7 @@ public class WriteBytesProvider implements Callback
|
|||
generator.getBufferPool().release(headerBuffer);
|
||||
headerBuffer = null;
|
||||
}
|
||||
releasePayloadBuffer(frame);
|
||||
}
|
||||
|
||||
/**
|
||||
|
@ -342,6 +344,24 @@ public class WriteBytesProvider implements Callback
|
|||
}
|
||||
}
|
||||
|
||||
public void releasePayloadBuffer(Frame frame)
|
||||
{
|
||||
if (!frame.hasPayload())
|
||||
{
|
||||
return;
|
||||
}
|
||||
|
||||
if (frame instanceof DataFrame)
|
||||
{
|
||||
DataFrame data = (DataFrame)frame;
|
||||
if (data.isPooledBuffer())
|
||||
{
|
||||
ByteBuffer payload = frame.getPayload();
|
||||
generator.getBufferPool().release(payload);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Set the buffer size used for generating ByteBuffers from the frames.
|
||||
* <p>
|
||||
|
|
|
@ -1,4 +1,5 @@
|
|||
org.eclipse.jetty.websocket.common.extensions.identity.IdentityExtension
|
||||
org.eclipse.jetty.websocket.common.extensions.compress.FrameCompressionExtension
|
||||
org.eclipse.jetty.websocket.common.extensions.compress.DeflateFrameExtension
|
||||
org.eclipse.jetty.websocket.common.extensions.compress.XWebkitDeflateFrameExtension
|
||||
org.eclipse.jetty.websocket.common.extensions.compress.MessageDeflateCompressionExtension
|
||||
org.eclipse.jetty.websocket.common.extensions.fragment.FragmentExtension
|
|
@ -44,6 +44,19 @@ public class IncomingFramesCapture implements IncomingFrames
|
|||
|
||||
public void assertFrameCount(int expectedCount)
|
||||
{
|
||||
if (frames.size() != expectedCount)
|
||||
{
|
||||
// dump details
|
||||
System.err.printf("Expected %d frame(s)%n",expectedCount);
|
||||
System.err.printf("But actually captured %d frame(s)%n",frames.size());
|
||||
for (int i = 0; i < frames.size(); i++)
|
||||
{
|
||||
Frame frame = frames.get(i);
|
||||
System.err.printf(" [%d] Frame[%s] - %s%n", i,
|
||||
OpCode.name(frame.getOpCode()),
|
||||
BufferUtil.toDetailString(frame.getPayload()));
|
||||
}
|
||||
}
|
||||
Assert.assertThat("Captured frame count",frames.size(),is(expectedCount));
|
||||
}
|
||||
|
||||
|
|
|
@ -20,14 +20,14 @@ package org.eclipse.jetty.websocket.common.extensions;
|
|||
|
||||
import org.eclipse.jetty.websocket.common.extensions.compress.DeflateCompressionMethodTest;
|
||||
import org.eclipse.jetty.websocket.common.extensions.compress.MessageCompressionExtensionTest;
|
||||
import org.eclipse.jetty.websocket.common.extensions.compress.FrameCompressionExtensionTest;
|
||||
import org.eclipse.jetty.websocket.common.extensions.compress.DeflateFrameExtensionTest;
|
||||
import org.junit.runner.RunWith;
|
||||
import org.junit.runners.Suite;
|
||||
|
||||
@RunWith(Suite.class)
|
||||
@Suite.SuiteClasses(
|
||||
{ ExtensionStackTest.class, DeflateCompressionMethodTest.class, MessageCompressionExtensionTest.class, FragmentExtensionTest.class,
|
||||
IdentityExtensionTest.class, FrameCompressionExtensionTest.class })
|
||||
IdentityExtensionTest.class, DeflateFrameExtensionTest.class })
|
||||
public class AllTests
|
||||
{
|
||||
/* nothing to do here, its all done in the annotations */
|
||||
|
|
|
@ -23,6 +23,7 @@ import org.eclipse.jetty.util.log.Logger;
|
|||
import org.eclipse.jetty.websocket.api.WriteCallback;
|
||||
import org.eclipse.jetty.websocket.api.extensions.Frame;
|
||||
import org.eclipse.jetty.websocket.api.extensions.OutgoingFrames;
|
||||
import org.junit.rules.TestName;
|
||||
|
||||
/**
|
||||
* Dummy implementation of {@link OutgoingFrames} used for testing
|
||||
|
@ -37,6 +38,11 @@ public class DummyOutgoingFrames implements OutgoingFrames
|
|||
this.id = id;
|
||||
}
|
||||
|
||||
public DummyOutgoingFrames(TestName testname)
|
||||
{
|
||||
this(testname.getMethodName());
|
||||
}
|
||||
|
||||
@Override
|
||||
public void outgoingFrame(Frame frame, WriteCallback callback)
|
||||
{
|
||||
|
|
|
@ -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.extensions.compress;
|
||||
|
||||
import java.util.ArrayList;
|
||||
import java.util.List;
|
||||
|
||||
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.Frame;
|
||||
import org.eclipse.jetty.websocket.api.extensions.OutgoingFrames;
|
||||
import org.eclipse.jetty.websocket.common.Hex;
|
||||
import org.eclipse.jetty.websocket.common.OpCode;
|
||||
|
||||
public class CapturedHexPayloads implements OutgoingFrames
|
||||
{
|
||||
private static final Logger LOG = Log.getLogger(CapturedHexPayloads.class);
|
||||
private List<String> captured = new ArrayList<>();
|
||||
|
||||
@Override
|
||||
public void outgoingFrame(Frame frame, WriteCallback callback)
|
||||
{
|
||||
String hexPayload = Hex.asHex(frame.getPayload());
|
||||
LOG.debug("outgoingFrame({}: \"{}\", {})",
|
||||
OpCode.name(frame.getOpCode()),
|
||||
hexPayload, callback!=null?callback.getClass().getSimpleName():"<null>");
|
||||
captured.add(hexPayload);
|
||||
if (callback != null)
|
||||
{
|
||||
callback.writeSuccess();
|
||||
}
|
||||
}
|
||||
|
||||
public List<String> getCaptured()
|
||||
{
|
||||
return captured;
|
||||
}
|
||||
}
|
|
@ -23,6 +23,7 @@ import static org.hamcrest.Matchers.*;
|
|||
import java.io.IOException;
|
||||
import java.nio.ByteBuffer;
|
||||
import java.util.Collections;
|
||||
import java.util.List;
|
||||
import java.util.zip.Deflater;
|
||||
import java.util.zip.Inflater;
|
||||
|
||||
|
@ -36,6 +37,7 @@ import org.eclipse.jetty.websocket.api.extensions.ExtensionConfig;
|
|||
import org.eclipse.jetty.websocket.api.extensions.Frame;
|
||||
import org.eclipse.jetty.websocket.common.ByteBufferAssert;
|
||||
import org.eclipse.jetty.websocket.common.Generator;
|
||||
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.OutgoingNetworkBytesCapture;
|
||||
|
@ -44,19 +46,24 @@ import org.eclipse.jetty.websocket.common.UnitParser;
|
|||
import org.eclipse.jetty.websocket.common.WebSocketFrame;
|
||||
import org.eclipse.jetty.websocket.common.frames.TextFrame;
|
||||
import org.junit.Assert;
|
||||
import org.junit.Rule;
|
||||
import org.junit.Test;
|
||||
import org.junit.rules.TestName;
|
||||
|
||||
public class FrameCompressionExtensionTest
|
||||
public class DeflateFrameExtensionTest
|
||||
{
|
||||
@Rule
|
||||
public TestName testname = new TestName();
|
||||
|
||||
private void assertIncoming(byte[] raw, String... expectedTextDatas)
|
||||
{
|
||||
WebSocketPolicy policy = WebSocketPolicy.newClientPolicy();
|
||||
|
||||
FrameCompressionExtension ext = new FrameCompressionExtension();
|
||||
DeflateFrameExtension ext = new DeflateFrameExtension();
|
||||
ext.setBufferPool(new MappedByteBufferPool());
|
||||
ext.setPolicy(policy);
|
||||
|
||||
ExtensionConfig config = ExtensionConfig.parse("x-webkit-deflate-frame");
|
||||
ExtensionConfig config = ExtensionConfig.parse("deflate-frame");
|
||||
ext.setConfig(config);
|
||||
|
||||
// Setup capture of incoming frames
|
||||
|
@ -95,11 +102,11 @@ public class FrameCompressionExtensionTest
|
|||
{
|
||||
WebSocketPolicy policy = WebSocketPolicy.newClientPolicy();
|
||||
|
||||
FrameCompressionExtension ext = new FrameCompressionExtension();
|
||||
DeflateFrameExtension ext = new DeflateFrameExtension();
|
||||
ext.setBufferPool(new MappedByteBufferPool());
|
||||
ext.setPolicy(policy);
|
||||
|
||||
ExtensionConfig config = ExtensionConfig.parse("x-webkit-deflate-frame");
|
||||
ExtensionConfig config = ExtensionConfig.parse("deflate-frame");
|
||||
ext.setConfig(config);
|
||||
|
||||
ByteBufferPool bufferPool = new MappedByteBufferPool();
|
||||
|
@ -120,48 +127,92 @@ public class FrameCompressionExtensionTest
|
|||
public void testBlockheadClient_HelloThere()
|
||||
{
|
||||
// Captured from Blockhead Client - "Hello" then "There" via unit test
|
||||
String hello = "c1 87 00 00 00 00 f2 48 cd c9 c9 07 00".replaceAll("\\s*","");
|
||||
String there = "c1 87 00 00 00 00 0a c9 48 2d 4a 05 00".replaceAll("\\s*","");
|
||||
byte rawbuf[] = TypeUtil.fromHexString(hello + there);
|
||||
String hello = "c18700000000f248cdc9c90700";
|
||||
String there = "c187000000000ac9482d4a0500";
|
||||
byte rawbuf[] = Hex.asByteArray(hello + there);
|
||||
assertIncoming(rawbuf,"Hello","There");
|
||||
}
|
||||
|
||||
@Test
|
||||
public void testChrome20_Hello()
|
||||
{
|
||||
// Captured from Chrome 20.x - "Hello" (sent from browser/client)
|
||||
byte rawbuf[] = TypeUtil.fromHexString("c187832b5c11716391d84a2c5c");
|
||||
// Captured from Chrome 20.x - "Hello" (sent from browser)
|
||||
byte rawbuf[] = Hex.asByteArray("c187832b5c11716391d84a2c5c");
|
||||
assertIncoming(rawbuf,"Hello");
|
||||
}
|
||||
|
||||
@Test
|
||||
public void testChrome20_HelloThere()
|
||||
{
|
||||
// Captured from Chrome 20.x - "Hello" then "There" (sent from browser/client)
|
||||
String hello = "c1 87 7b 19 71 db 89 51 bc 12 b2 1e 71".replaceAll("\\s*","");
|
||||
String there = "c1 87 59 ed c8 f4 53 24 80 d9 13 e8 c8".replaceAll("\\s*","");
|
||||
byte rawbuf[] = TypeUtil.fromHexString(hello + there);
|
||||
// Captured from Chrome 20.x - "Hello" then "There" (sent from browser)
|
||||
String hello = "c1877b1971db8951bc12b21e71";
|
||||
String there = "c18759edc8f4532480d913e8c8";
|
||||
byte rawbuf[] = Hex.asByteArray(hello + there);
|
||||
assertIncoming(rawbuf,"Hello","There");
|
||||
}
|
||||
|
||||
@Test
|
||||
public void testChrome20_Info()
|
||||
{
|
||||
// Captured from Chrome 20.x - "info:" (sent from browser/client)
|
||||
byte rawbuf[] = TypeUtil.fromHexString("c187ca4def7f0081a4b47d4fef");
|
||||
// Captured from Chrome 20.x - "info:" (sent from browser)
|
||||
byte rawbuf[] = Hex.asByteArray("c187ca4def7f0081a4b47d4fef");
|
||||
assertIncoming(rawbuf,"info:");
|
||||
}
|
||||
|
||||
@Test
|
||||
public void testChrome20_TimeTime()
|
||||
{
|
||||
// Captured from Chrome 20.x - "time:" then "time:" once more (sent from browser/client)
|
||||
String time1 = "c1 87 82 46 74 24 a8 8f b8 69 37 44 74".replaceAll("\\s*","");
|
||||
String time2 = "c1 85 3c fd a1 7f 16 fc b0 7f 3c".replaceAll("\\s*","");
|
||||
byte rawbuf[] = TypeUtil.fromHexString(time1 + time2);
|
||||
// Captured from Chrome 20.x - "time:" then "time:" once more (sent from browser)
|
||||
String time1 = "c18782467424a88fb869374474";
|
||||
String time2 = "c1853cfda17f16fcb07f3c";
|
||||
byte rawbuf[] = Hex.asByteArray(time1 + time2);
|
||||
assertIncoming(rawbuf,"time:","time:");
|
||||
}
|
||||
|
||||
@Test
|
||||
public void testPyWebSocket_TimeTimeTime()
|
||||
{
|
||||
// Captured from Pywebsocket (r781) - "time:" sent 3 times.
|
||||
String time1 = "c1876b100104" + "41d9cd49de1201";
|
||||
String time2 = "c1852ae3ff01" + "00e2ee012a";
|
||||
String time3 = "c18435558caa" + "37468caa";
|
||||
byte rawbuf[] = Hex.asByteArray(time1 + time2 + time3);
|
||||
assertIncoming(rawbuf,"time:","time:","time:");
|
||||
}
|
||||
|
||||
@Test
|
||||
public void testCompress_TimeTimeTime()
|
||||
{
|
||||
// What pywebsocket produces for "time:", "time:", "time:"
|
||||
String expected[] = new String[]
|
||||
{ "2AC9CC4DB50200", "2A01110000", "02130000" };
|
||||
|
||||
// Lets see what we produce
|
||||
CapturedHexPayloads capture = new CapturedHexPayloads();
|
||||
DeflateFrameExtension ext = new DeflateFrameExtension();
|
||||
init(ext);
|
||||
ext.setNextOutgoingFrames(capture);
|
||||
|
||||
ext.outgoingFrame(new TextFrame().setPayload("time:"),null);
|
||||
ext.outgoingFrame(new TextFrame().setPayload("time:"),null);
|
||||
ext.outgoingFrame(new TextFrame().setPayload("time:"),null);
|
||||
|
||||
List<String> actual = capture.getCaptured();
|
||||
|
||||
for (String entry : actual)
|
||||
{
|
||||
System.err.printf("actual: \"%s\"%n",entry);
|
||||
}
|
||||
|
||||
Assert.assertThat("Compressed Payloads",actual,contains(expected));
|
||||
}
|
||||
|
||||
private void init(DeflateFrameExtension ext)
|
||||
{
|
||||
ext.setConfig(new ExtensionConfig(ext.getName()));
|
||||
ext.setBufferPool(new MappedByteBufferPool());
|
||||
}
|
||||
|
||||
@Test
|
||||
public void testDeflateBasics() throws Exception
|
||||
{
|
||||
|
@ -204,7 +255,6 @@ public class FrameCompressionExtensionTest
|
|||
|
||||
String actual = TypeUtil.toHexString(compressed);
|
||||
String expected = "CaCc4bCbB70200"; // what pywebsocket produces
|
||||
// String expected = "CbCc4bCbB70200"; // what java produces
|
||||
|
||||
Assert.assertThat("Compressed data",actual,is(expected));
|
||||
}
|
||||
|
@ -214,12 +264,10 @@ public class FrameCompressionExtensionTest
|
|||
{
|
||||
WebSocketPolicy policy = WebSocketPolicy.newClientPolicy();
|
||||
|
||||
FrameCompressionExtension ext = new FrameCompressionExtension();
|
||||
DeflateFrameExtension ext = new DeflateFrameExtension();
|
||||
ext.setBufferPool(new MappedByteBufferPool());
|
||||
ext.setPolicy(policy);
|
||||
|
||||
ExtensionConfig config = ExtensionConfig.parse("x-webkit-deflate-frame");
|
||||
ext.setConfig(config);
|
||||
ext.setConfig(new ExtensionConfig(ext.getName()));
|
||||
|
||||
ByteBufferPool bufferPool = new MappedByteBufferPool();
|
||||
boolean validating = true;
|
||||
|
@ -233,7 +281,6 @@ public class FrameCompressionExtensionTest
|
|||
ext.outgoingFrame(new TextFrame().setPayload("There"),null);
|
||||
|
||||
capture.assertBytes(0,"c107f248cdc9c90700");
|
||||
capture.assertBytes(1,"c1070ac9482d4a0500");
|
||||
}
|
||||
|
||||
@Test
|
|
@ -0,0 +1,82 @@
|
|||
//
|
||||
// ========================================================================
|
||||
// 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.extensions.compress;
|
||||
|
||||
import java.nio.ByteBuffer;
|
||||
import java.util.zip.Deflater;
|
||||
|
||||
import org.eclipse.jetty.util.StringUtil;
|
||||
import org.eclipse.jetty.websocket.common.Hex;
|
||||
import org.junit.Ignore;
|
||||
import org.junit.Test;
|
||||
|
||||
public class DeflateTest
|
||||
{
|
||||
private int bufSize = 8 * 1024;
|
||||
|
||||
public String deflate(String inputHex, Deflater deflater, int flushMode)
|
||||
{
|
||||
byte uncompressed[] = Hex.asByteArray(inputHex);
|
||||
deflater.setInput(uncompressed,0,uncompressed.length);
|
||||
deflater.finish();
|
||||
|
||||
ByteBuffer out = ByteBuffer.allocate(bufSize);
|
||||
byte buf[] = new byte[64];
|
||||
while (!deflater.finished())
|
||||
{
|
||||
int len = deflater.deflate(buf,0,buf.length,flushMode);
|
||||
out.put(buf,0,len);
|
||||
}
|
||||
|
||||
out.flip();
|
||||
return Hex.asHex(out);
|
||||
}
|
||||
|
||||
@Test
|
||||
@Ignore("just noisy")
|
||||
public void deflateAllTypes()
|
||||
{
|
||||
int levels[] = new int[]
|
||||
{ 0, 1, 2, 3, 4, 5, 6, 7, 8, 9 };
|
||||
boolean nowraps[] = new boolean[]
|
||||
{ true, false };
|
||||
int strategies[] = new int[]
|
||||
{ Deflater.DEFAULT_STRATEGY, Deflater.FILTERED, Deflater.HUFFMAN_ONLY };
|
||||
int flushmodes[] = new int[]
|
||||
{ Deflater.NO_FLUSH, Deflater.SYNC_FLUSH, Deflater.FULL_FLUSH };
|
||||
|
||||
String inputHex = Hex.asHex(StringUtil.getUtf8Bytes("time:"));
|
||||
for (int level : levels)
|
||||
{
|
||||
for (boolean nowrap : nowraps)
|
||||
{
|
||||
Deflater deflater = new Deflater(level,nowrap);
|
||||
for (int strategy : strategies)
|
||||
{
|
||||
deflater.setStrategy(strategy);
|
||||
for (int flushmode : flushmodes)
|
||||
{
|
||||
String result = deflate(inputHex,deflater,flushmode);
|
||||
System.out.printf("%d | %b | %d | %d | \"%s\"%n",level,nowrap,strategy,flushmode,result);
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
|
@ -4,4 +4,4 @@ org.eclipse.jetty.LEVEL=WARN
|
|||
# org.eclipse.jetty.websocket.protocol.Parser.LEVEL=DEBUG
|
||||
# org.eclipse.jetty.websocket.protocol.LEVEL=DEBUG
|
||||
# org.eclipse.jetty.websocket.io.payload.LEVEL=DEBUG
|
||||
# org.eclipse.jetty.websocket.core.extensions.compress.LEVEL=DEBUG
|
||||
# org.eclipse.jetty.websocket.common.extensions.LEVEL=DEBUG
|
||||
|
|
|
@ -23,8 +23,6 @@ import org.eclipse.jetty.server.ServerConnector;
|
|||
import org.eclipse.jetty.server.handler.ResourceHandler;
|
||||
import org.eclipse.jetty.util.log.Log;
|
||||
import org.eclipse.jetty.util.log.Logger;
|
||||
import org.eclipse.jetty.websocket.common.extensions.compress.FrameCompressionExtension;
|
||||
import org.eclipse.jetty.websocket.common.extensions.compress.MessageDeflateCompressionExtension;
|
||||
import org.eclipse.jetty.websocket.server.WebSocketHandler;
|
||||
import org.eclipse.jetty.websocket.servlet.ServletUpgradeRequest;
|
||||
import org.eclipse.jetty.websocket.servlet.ServletUpgradeResponse;
|
||||
|
@ -84,6 +82,8 @@ public class BrowserDebugTool implements WebSocketCreator
|
|||
|
||||
String ua = req.getHeader("User-Agent");
|
||||
String rexts = req.getHeader("Sec-WebSocket-Extensions");
|
||||
LOG.debug("User-Agent: {}", ua);
|
||||
LOG.debug("Sec-WebSocket-Extensions: {}", rexts);
|
||||
BrowserSocket socket = new BrowserSocket(ua,rexts);
|
||||
return socket;
|
||||
}
|
||||
|
@ -110,15 +110,11 @@ public class BrowserDebugTool implements WebSocketCreator
|
|||
{
|
||||
LOG.debug("Configuring WebSocketServerFactory ...");
|
||||
|
||||
// Setup some extensions we want to test against
|
||||
factory.getExtensionFactory().register("x-webkit-deflate-frame",FrameCompressionExtension.class);
|
||||
factory.getExtensionFactory().register("permessage-compress",MessageDeflateCompressionExtension.class);
|
||||
|
||||
// Setup the desired Socket to use for all incoming upgrade requests
|
||||
factory.setCreator(BrowserDebugTool.this);
|
||||
|
||||
// Set the timeout
|
||||
factory.getPolicy().setIdleTimeout(2000);
|
||||
factory.getPolicy().setIdleTimeout(20000);
|
||||
}
|
||||
};
|
||||
|
||||
|
|
|
@ -18,7 +18,7 @@
|
|||
|
||||
package org.eclipse.jetty.websocket.server.helper;
|
||||
|
||||
import org.eclipse.jetty.websocket.common.extensions.compress.FrameCompressionExtension;
|
||||
import org.eclipse.jetty.websocket.common.extensions.compress.DeflateFrameExtension;
|
||||
import org.eclipse.jetty.websocket.common.extensions.compress.MessageDeflateCompressionExtension;
|
||||
import org.eclipse.jetty.websocket.servlet.WebSocketServlet;
|
||||
import org.eclipse.jetty.websocket.servlet.WebSocketServletFactory;
|
||||
|
@ -33,7 +33,7 @@ public class EchoServlet extends WebSocketServlet
|
|||
public void configure(WebSocketServletFactory factory)
|
||||
{
|
||||
// Setup some extensions we want to test against
|
||||
factory.getExtensionFactory().register("x-webkit-deflate-frame",FrameCompressionExtension.class);
|
||||
factory.getExtensionFactory().register("x-webkit-deflate-frame",DeflateFrameExtension.class);
|
||||
factory.getExtensionFactory().register("permessage-compress",MessageDeflateCompressionExtension.class);
|
||||
|
||||
// Setup the desired Socket to use for all incoming upgrade requests
|
||||
|
|
Loading…
Reference in New Issue