Skip to main content
aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorJoakim Erdfelt2015-08-07 16:40:11 +0000
committerJoakim Erdfelt2015-08-07 16:40:11 +0000
commit25b692046f85fa2276c30e4ae2e7a1e09f5ac4d3 (patch)
tree775a81168b640b6ce3a353294a3f2cb85088f6b0 /jetty-websocket
parent1b33ded7848a849453f71cbcebaa88c84ce65053 (diff)
parent55862e229e530cbde495e073fff37671f3baedd0 (diff)
downloadorg.eclipse.jetty.project-25b692046f85fa2276c30e4ae2e7a1e09f5ac4d3.tar.gz
org.eclipse.jetty.project-25b692046f85fa2276c30e4ae2e7a1e09f5ac4d3.tar.xz
org.eclipse.jetty.project-25b692046f85fa2276c30e4ae2e7a1e09f5ac4d3.zip
Merge branch 'jetty-9.2.x'
Diffstat (limited to 'jetty-websocket')
-rw-r--r--jetty-websocket/javax-websocket-server-impl/src/test/java/org/eclipse/jetty/websocket/jsr356/server/ExtensionStackProcessingTest.java55
-rw-r--r--jetty-websocket/websocket-common/src/main/java/org/eclipse/jetty/websocket/common/extensions/compress/ByteAccumulator.java2
-rw-r--r--jetty-websocket/websocket-common/src/main/java/org/eclipse/jetty/websocket/common/extensions/compress/CompressExtension.java13
-rw-r--r--jetty-websocket/websocket-common/src/main/java/org/eclipse/jetty/websocket/common/extensions/compress/DeflateFrameExtension.java76
-rw-r--r--jetty-websocket/websocket-common/src/main/java/org/eclipse/jetty/websocket/common/extensions/compress/XWebkitDeflateFrameExtension.java32
-rw-r--r--jetty-websocket/websocket-common/src/main/resources/META-INF/services/org.eclipse.jetty.websocket.api.extensions.Extension4
-rw-r--r--jetty-websocket/websocket-common/src/test/java/org/eclipse/jetty/websocket/common/extensions/compress/DeflateFrameExtensionTest.java451
7 files changed, 628 insertions, 5 deletions
diff --git a/jetty-websocket/javax-websocket-server-impl/src/test/java/org/eclipse/jetty/websocket/jsr356/server/ExtensionStackProcessingTest.java b/jetty-websocket/javax-websocket-server-impl/src/test/java/org/eclipse/jetty/websocket/jsr356/server/ExtensionStackProcessingTest.java
index 30c2e9fd1a..3096f47a43 100644
--- a/jetty-websocket/javax-websocket-server-impl/src/test/java/org/eclipse/jetty/websocket/jsr356/server/ExtensionStackProcessingTest.java
+++ b/jetty-websocket/javax-websocket-server-impl/src/test/java/org/eclipse/jetty/websocket/jsr356/server/ExtensionStackProcessingTest.java
@@ -39,7 +39,12 @@ import org.eclipse.jetty.server.Server;
import org.eclipse.jetty.server.ServerConnector;
import org.eclipse.jetty.servlet.ServletContextHandler;
import org.eclipse.jetty.websocket.api.extensions.ExtensionFactory;
+import org.eclipse.jetty.websocket.api.extensions.OutgoingFrames;
+import org.eclipse.jetty.websocket.client.io.WebSocketClientConnection;
+import org.eclipse.jetty.websocket.common.extensions.ExtensionStack;
+import org.eclipse.jetty.websocket.common.extensions.compress.DeflateFrameExtension;
import org.eclipse.jetty.websocket.jsr356.JsrExtension;
+import org.eclipse.jetty.websocket.jsr356.JsrSession;
import org.eclipse.jetty.websocket.jsr356.server.deploy.WebSocketServerContainerInitializer;
import org.eclipse.jetty.websocket.jsr356.server.samples.echo.BasicEchoEndpoint;
import org.eclipse.jetty.websocket.server.WebSocketUpgradeFilter;
@@ -83,7 +88,55 @@ public class ExtensionStackProcessingTest
{
server.stop();
}
-
+
+ @Test
+ public void testDeflateFrameExtension() throws Exception
+ {
+ Assume.assumeTrue("Server has deflate-frame extension registered",serverExtensionFactory.isAvailable("deflate-frame"));
+
+ ClientEndpointConfig config = ClientEndpointConfig.Builder.create()
+ .extensions(Arrays.<Extension>asList(new JsrExtension("deflate-frame")))
+ .build();
+
+ final String content = "deflate_me";
+ final CountDownLatch messageLatch = new CountDownLatch(1);
+ URI uri = URI.create("ws://localhost:" + connector.getLocalPort());
+ Session session = client.connectToServer(new EndpointAdapter()
+ {
+ @Override
+ public void onMessage(String message)
+ {
+ Assert.assertEquals(content, message);
+ messageLatch.countDown();
+ }
+ }, config, uri);
+
+ // Make sure everything is wired properly.
+ OutgoingFrames firstOut = ((JsrSession)session).getOutgoingHandler();
+ Assert.assertTrue(firstOut instanceof ExtensionStack);
+ ExtensionStack extensionStack = (ExtensionStack)firstOut;
+ Assert.assertTrue(extensionStack.isRunning());
+ OutgoingFrames secondOut = extensionStack.getNextOutgoing();
+ Assert.assertTrue(secondOut instanceof DeflateFrameExtension);
+ DeflateFrameExtension deflateExtension = (DeflateFrameExtension)secondOut;
+ Assert.assertTrue(deflateExtension.isRunning());
+ OutgoingFrames thirdOut = deflateExtension.getNextOutgoing();
+ Assert.assertTrue(thirdOut instanceof WebSocketClientConnection);
+
+ final CountDownLatch latch = new CountDownLatch(1);
+ session.getAsyncRemote().sendText(content, new SendHandler()
+ {
+ @Override
+ public void onResult(SendResult result)
+ {
+ latch.countDown();
+ }
+ });
+
+ Assert.assertTrue(latch.await(5, TimeUnit.SECONDS));
+ Assert.assertTrue(messageLatch.await(5, TimeUnit.SECONDS));
+ }
+
@Test
public void testPerMessageDeflateExtension() throws Exception
{
diff --git a/jetty-websocket/websocket-common/src/main/java/org/eclipse/jetty/websocket/common/extensions/compress/ByteAccumulator.java b/jetty-websocket/websocket-common/src/main/java/org/eclipse/jetty/websocket/common/extensions/compress/ByteAccumulator.java
index 279cdbfc90..57a6fa823e 100644
--- a/jetty-websocket/websocket-common/src/main/java/org/eclipse/jetty/websocket/common/extensions/compress/ByteAccumulator.java
+++ b/jetty-websocket/websocket-common/src/main/java/org/eclipse/jetty/websocket/common/extensions/compress/ByteAccumulator.java
@@ -49,7 +49,7 @@ public class ByteAccumulator
chunks.add(copy);
this.length += length;
}
-
+
public int getLength()
{
return length;
diff --git a/jetty-websocket/websocket-common/src/main/java/org/eclipse/jetty/websocket/common/extensions/compress/CompressExtension.java b/jetty-websocket/websocket-common/src/main/java/org/eclipse/jetty/websocket/common/extensions/compress/CompressExtension.java
index dbb2e9d126..5c27104c59 100644
--- a/jetty-websocket/websocket-common/src/main/java/org/eclipse/jetty/websocket/common/extensions/compress/CompressExtension.java
+++ b/jetty-websocket/websocket-common/src/main/java/org/eclipse/jetty/websocket/common/extensions/compress/CompressExtension.java
@@ -67,6 +67,9 @@ public abstract class CompressExtension extends AbstractExtension
/** Deflater / Inflater: Maximum Input Buffer Size */
protected static final int INPUT_MAX_BUFFER_SIZE = 8 * 1024;
+ /** Inflater : Output Buffer Size */
+ private static final int DECOMPRESS_BUF_SIZE = 8 * 1024;
+
private final static boolean NOWRAP = true;
private final Queue<FrameEntry> entries = new ConcurrentArrayQueue<>();
@@ -150,7 +153,7 @@ public abstract class CompressExtension extends AbstractExtension
{
return;
}
- byte[] output = new byte[1024]; // TODO: make configurable size
+ byte[] output = new byte[DECOMPRESS_BUF_SIZE];
if (inflater.needsInput() && !supplyInput(inflater,buf))
{
@@ -369,9 +372,15 @@ public abstract class CompressExtension extends AbstractExtension
private class Flusher extends IteratingCallback implements WriteCallback
{
- private static final int INPUT_BUFSIZE = 32 * 1024;
private FrameEntry current;
private boolean finished = true;
+
+ @Override
+ public void failed(Throwable x)
+ {
+ LOG.warn(x);
+ super.failed(x);
+ }
@Override
protected Action process() throws Exception
diff --git a/jetty-websocket/websocket-common/src/main/java/org/eclipse/jetty/websocket/common/extensions/compress/DeflateFrameExtension.java b/jetty-websocket/websocket-common/src/main/java/org/eclipse/jetty/websocket/common/extensions/compress/DeflateFrameExtension.java
new file mode 100644
index 0000000000..8114a269d6
--- /dev/null
+++ b/jetty-websocket/websocket-common/src/main/java/org/eclipse/jetty/websocket/common/extensions/compress/DeflateFrameExtension.java
@@ -0,0 +1,76 @@
+//
+// ========================================================================
+// Copyright (c) 1995-2015 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.zip.DataFormatException;
+
+import org.eclipse.jetty.websocket.api.BadPayloadException;
+import org.eclipse.jetty.websocket.api.extensions.Frame;
+
+/**
+ * Implementation of the
+ * <a href="https://tools.ietf.org/id/draft-tyoshino-hybi-websocket-perframe-deflate.txt">deflate-frame</a>
+ * extension seen out in the wild.
+ */
+public class DeflateFrameExtension extends CompressExtension
+{
+ @Override
+ public String getName()
+ {
+ return "deflate-frame";
+ }
+
+ @Override
+ int getRsvUseMode()
+ {
+ return RSV_USE_ALWAYS;
+ }
+
+ @Override
+ int getTailDropMode()
+ {
+ return TAIL_DROP_ALWAYS;
+ }
+
+ @Override
+ public void incomingFrame(Frame frame)
+ {
+ // Incoming frames are always non concurrent because
+ // they are read and parsed with a single thread, and
+ // therefore there is no need for synchronization.
+
+ if ( frame.getType().isControl() || !frame.isRsv1() || !frame.hasPayload() )
+ {
+ nextIncomingFrame(frame);
+ return;
+ }
+
+ try
+ {
+ ByteAccumulator accumulator = newByteAccumulator();
+ decompress(accumulator, frame.getPayload());
+ decompress(accumulator, TAIL_BYTES_BUF.slice());
+ forwardIncoming(frame, accumulator);
+ }
+ catch (DataFormatException e)
+ {
+ throw new BadPayloadException(e);
+ }
+ }
+}
diff --git a/jetty-websocket/websocket-common/src/main/java/org/eclipse/jetty/websocket/common/extensions/compress/XWebkitDeflateFrameExtension.java b/jetty-websocket/websocket-common/src/main/java/org/eclipse/jetty/websocket/common/extensions/compress/XWebkitDeflateFrameExtension.java
new file mode 100644
index 0000000000..bcee2423b4
--- /dev/null
+++ b/jetty-websocket/websocket-common/src/main/java/org/eclipse/jetty/websocket/common/extensions/compress/XWebkitDeflateFrameExtension.java
@@ -0,0 +1,32 @@
+//
+// ========================================================================
+// Copyright (c) 1995-2015 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";
+ }
+}
diff --git a/jetty-websocket/websocket-common/src/main/resources/META-INF/services/org.eclipse.jetty.websocket.api.extensions.Extension b/jetty-websocket/websocket-common/src/main/resources/META-INF/services/org.eclipse.jetty.websocket.api.extensions.Extension
index 221481e72a..a1ad7d5a0f 100644
--- a/jetty-websocket/websocket-common/src/main/resources/META-INF/services/org.eclipse.jetty.websocket.api.extensions.Extension
+++ b/jetty-websocket/websocket-common/src/main/resources/META-INF/services/org.eclipse.jetty.websocket.api.extensions.Extension
@@ -1,3 +1,5 @@
org.eclipse.jetty.websocket.common.extensions.identity.IdentityExtension
+org.eclipse.jetty.websocket.common.extensions.fragment.FragmentExtension
org.eclipse.jetty.websocket.common.extensions.compress.PerMessageDeflateExtension
-org.eclipse.jetty.websocket.common.extensions.fragment.FragmentExtension \ No newline at end of file
+org.eclipse.jetty.websocket.common.extensions.compress.DeflateFrameExtension
+org.eclipse.jetty.websocket.common.extensions.compress.XWebkitDeflateFrameExtension
diff --git a/jetty-websocket/websocket-common/src/test/java/org/eclipse/jetty/websocket/common/extensions/compress/DeflateFrameExtensionTest.java b/jetty-websocket/websocket-common/src/test/java/org/eclipse/jetty/websocket/common/extensions/compress/DeflateFrameExtensionTest.java
new file mode 100644
index 0000000000..df1a5287d7
--- /dev/null
+++ b/jetty-websocket/websocket-common/src/test/java/org/eclipse/jetty/websocket/common/extensions/compress/DeflateFrameExtensionTest.java
@@ -0,0 +1,451 @@
+//
+// ========================================================================
+// Copyright (c) 1995-2015 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 static org.hamcrest.Matchers.contains;
+import static org.hamcrest.Matchers.greaterThan;
+import static org.hamcrest.Matchers.is;
+
+import java.io.ByteArrayOutputStream;
+import java.io.IOException;
+import java.nio.ByteBuffer;
+import java.nio.charset.StandardCharsets;
+import java.util.Collections;
+import java.util.List;
+import java.util.Random;
+import java.util.zip.Deflater;
+import java.util.zip.Inflater;
+
+import org.eclipse.jetty.io.RuntimeIOException;
+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.BatchMode;
+import org.eclipse.jetty.websocket.api.WebSocketPolicy;
+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.api.extensions.IncomingFrames;
+import org.eclipse.jetty.websocket.api.extensions.OutgoingFrames;
+import org.eclipse.jetty.websocket.common.Generator;
+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.extensions.AbstractExtensionTest;
+import org.eclipse.jetty.websocket.common.extensions.ExtensionTool.Tester;
+import org.eclipse.jetty.websocket.common.frames.BinaryFrame;
+import org.eclipse.jetty.websocket.common.frames.TextFrame;
+import org.eclipse.jetty.websocket.common.test.ByteBufferAssert;
+import org.eclipse.jetty.websocket.common.test.IncomingFramesCapture;
+import org.eclipse.jetty.websocket.common.test.LeakTrackingBufferPoolRule;
+import org.eclipse.jetty.websocket.common.test.OutgoingNetworkBytesCapture;
+import org.eclipse.jetty.websocket.common.test.UnitParser;
+import org.junit.Assert;
+import org.junit.Rule;
+import org.junit.Test;
+
+public class DeflateFrameExtensionTest extends AbstractExtensionTest
+{
+ private static final Logger LOG = Log.getLogger(DeflateFrameExtensionTest.class);
+
+ @Rule
+ public LeakTrackingBufferPoolRule bufferPool = new LeakTrackingBufferPoolRule("Test");
+
+ private void assertIncoming(byte[] raw, String... expectedTextDatas)
+ {
+ WebSocketPolicy policy = WebSocketPolicy.newClientPolicy();
+
+ DeflateFrameExtension ext = new DeflateFrameExtension();
+ ext.setBufferPool(bufferPool);
+ ext.setPolicy(policy);
+
+ ExtensionConfig config = ExtensionConfig.parse("deflate-frame");
+ ext.setConfig(config);
+
+ // Setup capture of incoming frames
+ IncomingFramesCapture capture = new IncomingFramesCapture();
+
+ // Wire up stack
+ ext.setNextIncomingFrames(capture);
+
+ Parser parser = new UnitParser(policy);
+ parser.configureFromExtensions(Collections.singletonList(ext));
+ parser.setIncomingFramesHandler(ext);
+
+ parser.parse(ByteBuffer.wrap(raw));
+
+ int len = expectedTextDatas.length;
+ capture.assertFrameCount(len);
+ capture.assertHasFrame(OpCode.TEXT, len);
+
+ int i = 0;
+ for (WebSocketFrame actual : capture.getFrames())
+ {
+ String prefix = "Frame[" + i + "]";
+ Assert.assertThat(prefix + ".opcode", actual.getOpCode(), is(OpCode.TEXT));
+ Assert.assertThat(prefix + ".fin", actual.isFin(), is(true));
+ Assert.assertThat(prefix + ".rsv1", actual.isRsv1(), is(false)); // RSV1 should be unset at this point
+ Assert.assertThat(prefix + ".rsv2", actual.isRsv2(), is(false));
+ Assert.assertThat(prefix + ".rsv3", actual.isRsv3(), is(false));
+
+ ByteBuffer expected = BufferUtil.toBuffer(expectedTextDatas[i], StandardCharsets.UTF_8);
+ Assert.assertThat(prefix + ".payloadLength", actual.getPayloadLength(), is(expected.remaining()));
+ ByteBufferAssert.assertEquals(prefix + ".payload", expected, actual.getPayload().slice());
+ i++;
+ }
+ }
+
+ private void assertOutgoing(String text, String expectedHex) throws IOException
+ {
+ WebSocketPolicy policy = WebSocketPolicy.newClientPolicy();
+
+ DeflateFrameExtension ext = new DeflateFrameExtension();
+ ext.setBufferPool(bufferPool);
+ ext.setPolicy(policy);
+
+ ExtensionConfig config = ExtensionConfig.parse("deflate-frame");
+ ext.setConfig(config);
+
+ Generator generator = new Generator(policy, bufferPool, true);
+ generator.configureFromExtensions(Collections.singletonList(ext));
+
+ OutgoingNetworkBytesCapture capture = new OutgoingNetworkBytesCapture(generator);
+ ext.setNextOutgoingFrames(capture);
+
+ Frame frame = new TextFrame().setPayload(text);
+ ext.outgoingFrame(frame, null, BatchMode.OFF);
+
+ capture.assertBytes(0, expectedHex);
+ }
+
+ @Test
+ public void testBlockheadClient_HelloThere()
+ {
+ Tester tester = serverExtensions.newTester("deflate-frame");
+
+ tester.assertNegotiated("deflate-frame");
+
+ tester.parseIncomingHex(// Captured from Blockhead Client - "Hello" then "There" via unit test
+ "c18700000000f248cdc9c90700", // "Hello"
+ "c187000000000ac9482d4a0500" // "There"
+ );
+
+ tester.assertHasFrames("Hello", "There");
+ }
+
+ @Test
+ public void testChrome20_Hello()
+ {
+ Tester tester = serverExtensions.newTester("deflate-frame");
+
+ tester.assertNegotiated("deflate-frame");
+
+ tester.parseIncomingHex(// Captured from Chrome 20.x - "Hello" (sent from browser)
+ "c187832b5c11716391d84a2c5c" // "Hello"
+ );
+
+ tester.assertHasFrames("Hello");
+ }
+
+ @Test
+ public void testChrome20_HelloThere()
+ {
+ Tester tester = serverExtensions.newTester("deflate-frame");
+
+ tester.assertNegotiated("deflate-frame");
+
+ tester.parseIncomingHex(// Captured from Chrome 20.x - "Hello" then "There" (sent from browser)
+ "c1877b1971db8951bc12b21e71", // "Hello"
+ "c18759edc8f4532480d913e8c8" // There
+ );
+
+ tester.assertHasFrames("Hello", "There");
+ }
+
+ @Test
+ public void testChrome20_Info()
+ {
+ Tester tester = serverExtensions.newTester("deflate-frame");
+
+ tester.assertNegotiated("deflate-frame");
+
+ tester.parseIncomingHex(// Captured from Chrome 20.x - "info:" (sent from browser)
+ "c187ca4def7f0081a4b47d4fef" // example payload
+ );
+
+ tester.assertHasFrames("info:");
+ }
+
+ @Test
+ public void testChrome20_TimeTime()
+ {
+ Tester tester = serverExtensions.newTester("deflate-frame");
+
+ tester.assertNegotiated("deflate-frame");
+
+ tester.parseIncomingHex(// Captured from Chrome 20.x - "time:" then "time:" once more (sent from browser)
+ "c18782467424a88fb869374474", // "time:"
+ "c1853cfda17f16fcb07f3c" // "time:"
+ );
+
+ tester.assertHasFrames("time:", "time:");
+ }
+
+ @Test
+ public void testPyWebSocket_TimeTimeTime()
+ {
+ Tester tester = serverExtensions.newTester("deflate-frame");
+
+ tester.assertNegotiated("deflate-frame");
+
+ tester.parseIncomingHex(// Captured from Pywebsocket (r781) - "time:" sent 3 times.
+ "c1876b100104" + "41d9cd49de1201", // "time:"
+ "c1852ae3ff01" + "00e2ee012a", // "time:"
+ "c18435558caa" + "37468caa" // "time:"
+ );
+
+ tester.assertHasFrames("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, BatchMode.OFF);
+ ext.outgoingFrame(new TextFrame().setPayload("time:"), null, BatchMode.OFF);
+ ext.outgoingFrame(new TextFrame().setPayload("time:"), null, BatchMode.OFF);
+
+ List<String> actual = capture.getCaptured();
+
+ Assert.assertThat("Compressed Payloads", actual, contains(expected));
+ }
+
+ private void init(DeflateFrameExtension ext)
+ {
+ ext.setConfig(new ExtensionConfig(ext.getName()));
+ ext.setBufferPool(bufferPool);
+ }
+
+ @Test
+ public void testDeflateBasics() throws Exception
+ {
+ // Setup deflater basics
+ Deflater compressor = new Deflater(Deflater.BEST_COMPRESSION, true);
+ compressor.setStrategy(Deflater.DEFAULT_STRATEGY);
+
+ // Text to compress
+ String text = "info:";
+ byte uncompressed[] = StringUtil.getUtf8Bytes(text);
+
+ // Prime the compressor
+ compressor.reset();
+ compressor.setInput(uncompressed, 0, uncompressed.length);
+ compressor.finish();
+
+ // Perform compression
+ ByteBuffer outbuf = ByteBuffer.allocate(64);
+ BufferUtil.clearToFill(outbuf);
+
+ while (!compressor.finished())
+ {
+ byte out[] = new byte[64];
+ int len = compressor.deflate(out, 0, out.length, Deflater.SYNC_FLUSH);
+ if (len > 0)
+ {
+ outbuf.put(out, 0, len);
+ }
+ }
+ compressor.end();
+
+ BufferUtil.flipToFlush(outbuf, 0);
+ byte compressed[] = BufferUtil.toArray(outbuf);
+ // Clear the BFINAL bit that has been set by the compressor.end() call.
+ // In the real implementation we never end() the compressor.
+ compressed[0] &= 0xFE;
+
+ String actual = TypeUtil.toHexString(compressed);
+ String expected = "CaCc4bCbB70200"; // what pywebsocket produces
+
+ Assert.assertThat("Compressed data", actual, is(expected));
+ }
+
+ @Test
+ public void testGeneratedTwoFrames() throws IOException
+ {
+ WebSocketPolicy policy = WebSocketPolicy.newClientPolicy();
+
+ DeflateFrameExtension ext = new DeflateFrameExtension();
+ ext.setBufferPool(bufferPool);
+ ext.setPolicy(policy);
+ ext.setConfig(new ExtensionConfig(ext.getName()));
+
+ Generator generator = new Generator(policy, bufferPool, true);
+ generator.configureFromExtensions(Collections.singletonList(ext));
+
+ OutgoingNetworkBytesCapture capture = new OutgoingNetworkBytesCapture(generator);
+ ext.setNextOutgoingFrames(capture);
+
+ ext.outgoingFrame(new TextFrame().setPayload("Hello"), null, BatchMode.OFF);
+ ext.outgoingFrame(new TextFrame().setPayload("There"), null, BatchMode.OFF);
+
+ capture.assertBytes(0, "c107f248cdc9c90700");
+ }
+
+ @Test
+ public void testInflateBasics() throws Exception
+ {
+ // should result in "info:" text if properly inflated
+ byte rawbuf[] = TypeUtil.fromHexString("CaCc4bCbB70200"); // what pywebsocket produces
+ // byte rawbuf[] = TypeUtil.fromHexString("CbCc4bCbB70200"); // what java produces
+
+ Inflater inflater = new Inflater(true);
+ inflater.reset();
+ inflater.setInput(rawbuf, 0, rawbuf.length);
+
+ byte outbuf[] = new byte[64];
+ int len = inflater.inflate(outbuf);
+ inflater.end();
+ Assert.assertThat("Inflated length", len, greaterThan(4));
+
+ String actual = StringUtil.toUTF8String(outbuf, 0, len);
+ Assert.assertThat("Inflated text", actual, is("info:"));
+ }
+
+ @Test
+ public void testPyWebSocketServer_Hello()
+ {
+ // Captured from PyWebSocket - "Hello" (echo from server)
+ byte rawbuf[] = TypeUtil.fromHexString("c107f248cdc9c90700");
+ assertIncoming(rawbuf, "Hello");
+ }
+
+ @Test
+ public void testPyWebSocketServer_Long()
+ {
+ // Captured from PyWebSocket - Long Text (echo from server)
+ byte rawbuf[] = TypeUtil.fromHexString("c1421cca410a80300c44d1abccce9df7" + "f018298634d05631138ab7b7b8fdef1f" + "dc0282e2061d575a45f6f2686bab25e1"
+ + "3fb7296fa02b5885eb3b0379c394f461" + "98cafd03");
+ assertIncoming(rawbuf, "It's a big enough umbrella but it's always me that ends up getting wet.");
+ }
+
+ @Test
+ public void testPyWebSocketServer_Medium()
+ {
+ // Captured from PyWebSocket - "stackoverflow" (echo from server)
+ byte rawbuf[] = TypeUtil.fromHexString("c10f2a2e494ccece2f4b2d4acbc92f0700");
+ assertIncoming(rawbuf, "stackoverflow");
+ }
+
+ /**
+ * Make sure that the server generated compressed form for "Hello" is consistent with what PyWebSocket creates.
+ * @throws IOException on test failure
+ */
+ @Test
+ public void testServerGeneratedHello() throws IOException
+ {
+ assertOutgoing("Hello", "c107f248cdc9c90700");
+ }
+
+ /**
+ * Make sure that the server generated compressed form for "There" is consistent with what PyWebSocket creates.
+ * @throws IOException on test failure
+ */
+ @Test
+ public void testServerGeneratedThere() throws IOException
+ {
+ assertOutgoing("There", "c1070ac9482d4a0500");
+ }
+
+ @Test
+ public void testCompressAndDecompressBigPayload() throws Exception
+ {
+ byte[] input = new byte[1024 * 1024];
+ // Make them not compressible.
+ new Random().nextBytes(input);
+
+ int maxMessageSize = (1024 * 1024) + 8192;
+
+ DeflateFrameExtension clientExtension = new DeflateFrameExtension();
+ clientExtension.setBufferPool(bufferPool);
+ clientExtension.setPolicy(WebSocketPolicy.newClientPolicy());
+ clientExtension.getPolicy().setMaxBinaryMessageSize(maxMessageSize);
+ clientExtension.getPolicy().setMaxBinaryMessageBufferSize(maxMessageSize);
+ clientExtension.setConfig(ExtensionConfig.parse("deflate-frame"));
+
+ final DeflateFrameExtension serverExtension = new DeflateFrameExtension();
+ serverExtension.setBufferPool(bufferPool);
+ serverExtension.setPolicy(WebSocketPolicy.newServerPolicy());
+ serverExtension.getPolicy().setMaxBinaryMessageSize(maxMessageSize);
+ serverExtension.getPolicy().setMaxBinaryMessageBufferSize(maxMessageSize);
+ serverExtension.setConfig(ExtensionConfig.parse("deflate-frame"));
+
+ // Chain the next element to decompress.
+ clientExtension.setNextOutgoingFrames(new OutgoingFrames()
+ {
+ @Override
+ public void outgoingFrame(Frame frame, WriteCallback callback, BatchMode batchMode)
+ {
+ LOG.debug("outgoingFrame({})", frame);
+ serverExtension.incomingFrame(frame);
+ callback.writeSuccess();
+ }
+ });
+
+ final ByteArrayOutputStream result = new ByteArrayOutputStream(input.length);
+ serverExtension.setNextIncomingFrames(new IncomingFrames()
+ {
+ @Override
+ public void incomingFrame(Frame frame)
+ {
+ LOG.debug("incomingFrame({})", frame);
+ try
+ {
+ result.write(BufferUtil.toArray(frame.getPayload()));
+ }
+ catch (IOException x)
+ {
+ throw new RuntimeIOException(x);
+ }
+ }
+
+ @Override
+ public void incomingError(Throwable t)
+ {
+ }
+ });
+
+ BinaryFrame frame = new BinaryFrame();
+ frame.setPayload(input);
+ frame.setFin(true);
+ clientExtension.outgoingFrame(frame, null, BatchMode.OFF);
+
+ Assert.assertArrayEquals(input, result.toByteArray());
+ }
+}

Back to the top