Issue #207 - moved some JSR356 tests to websocket-tests
# Conflicts: # jetty-websocket/javax-websocket-client-impl/src/test/java/org/eclipse/jetty/websocket/jsr356/DecoderReaderManySmallTest.java # jetty-websocket/javax-websocket-client-impl/src/test/java/org/eclipse/jetty/websocket/jsr356/DecoderReaderTest.java # jetty-websocket/javax-websocket-client-impl/src/test/java/org/eclipse/jetty/websocket/jsr356/EncoderTest.java
This commit is contained in:
parent
06f204f794
commit
88b2c47904
|
@ -1,228 +0,0 @@
|
||||||
//
|
|
||||||
// ========================================================================
|
|
||||||
// Copyright (c) 1995-2017 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.jsr356;
|
|
||||||
|
|
||||||
import java.io.BufferedReader;
|
|
||||||
import java.io.IOException;
|
|
||||||
import java.io.Reader;
|
|
||||||
import java.util.ArrayList;
|
|
||||||
import java.util.List;
|
|
||||||
import java.util.concurrent.CountDownLatch;
|
|
||||||
import java.util.concurrent.TimeUnit;
|
|
||||||
|
|
||||||
import javax.websocket.ClientEndpoint;
|
|
||||||
import javax.websocket.CloseReason;
|
|
||||||
import javax.websocket.ContainerProvider;
|
|
||||||
import javax.websocket.DecodeException;
|
|
||||||
import javax.websocket.Decoder;
|
|
||||||
import javax.websocket.EndpointConfig;
|
|
||||||
import javax.websocket.OnClose;
|
|
||||||
import javax.websocket.OnMessage;
|
|
||||||
import javax.websocket.WebSocketContainer;
|
|
||||||
|
|
||||||
import org.eclipse.jetty.toolchain.test.EventQueue;
|
|
||||||
import org.eclipse.jetty.toolchain.test.TestTracker;
|
|
||||||
import org.eclipse.jetty.util.component.LifeCycle;
|
|
||||||
import org.eclipse.jetty.util.log.Log;
|
|
||||||
import org.eclipse.jetty.util.log.Logger;
|
|
||||||
import org.eclipse.jetty.websocket.common.frames.TextFrame;
|
|
||||||
import org.eclipse.jetty.websocket.common.test.XBlockheadServer;
|
|
||||||
import org.eclipse.jetty.websocket.common.test.IBlockheadServerConnection;
|
|
||||||
import org.junit.After;
|
|
||||||
import org.junit.Assert;
|
|
||||||
import org.junit.Before;
|
|
||||||
import org.junit.Ignore;
|
|
||||||
import org.junit.Rule;
|
|
||||||
import org.junit.Test;
|
|
||||||
|
|
||||||
@Ignore("Not working atm")
|
|
||||||
public class DecoderReaderManySmallTest
|
|
||||||
{
|
|
||||||
public static class EventId
|
|
||||||
{
|
|
||||||
public int eventId;
|
|
||||||
}
|
|
||||||
|
|
||||||
public static class EventIdDecoder implements Decoder.TextStream<EventId>
|
|
||||||
{
|
|
||||||
@Override
|
|
||||||
public void init(EndpointConfig config)
|
|
||||||
{
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void destroy()
|
|
||||||
{
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public EventId decode(Reader reader) throws DecodeException, IOException
|
|
||||||
{
|
|
||||||
EventId id = new EventId();
|
|
||||||
try (BufferedReader buf = new BufferedReader(reader))
|
|
||||||
{
|
|
||||||
String line;
|
|
||||||
while ((line = buf.readLine()) != null)
|
|
||||||
{
|
|
||||||
id.eventId = Integer.parseInt(line);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return id;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@ClientEndpoint(decoders = { EventIdDecoder.class })
|
|
||||||
public static class EventIdSocket
|
|
||||||
{
|
|
||||||
public EventQueue<EventId> messageQueue = new EventQueue<>();
|
|
||||||
private CountDownLatch closeLatch = new CountDownLatch(1);
|
|
||||||
|
|
||||||
@OnClose
|
|
||||||
public void onClose(CloseReason close)
|
|
||||||
{
|
|
||||||
closeLatch.countDown();
|
|
||||||
}
|
|
||||||
|
|
||||||
@OnMessage
|
|
||||||
public void onMessage(EventId msg)
|
|
||||||
{
|
|
||||||
messageQueue.add(msg);
|
|
||||||
}
|
|
||||||
|
|
||||||
public void awaitClose() throws InterruptedException
|
|
||||||
{
|
|
||||||
closeLatch.await(4,TimeUnit.SECONDS);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private static class EventIdServer implements Runnable
|
|
||||||
{
|
|
||||||
private XBlockheadServer server;
|
|
||||||
private IBlockheadServerConnection sconnection;
|
|
||||||
private CountDownLatch connectLatch = new CountDownLatch(1);
|
|
||||||
|
|
||||||
public EventIdServer(XBlockheadServer server)
|
|
||||||
{
|
|
||||||
this.server = server;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void run()
|
|
||||||
{
|
|
||||||
try
|
|
||||||
{
|
|
||||||
sconnection = server.accept();
|
|
||||||
sconnection.setSoTimeout(60000);
|
|
||||||
sconnection.upgrade();
|
|
||||||
}
|
|
||||||
catch (Exception e)
|
|
||||||
{
|
|
||||||
LOG.warn(e);
|
|
||||||
}
|
|
||||||
finally
|
|
||||||
{
|
|
||||||
connectLatch.countDown();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public void writeSequentialIds(int from, int to) throws IOException
|
|
||||||
{
|
|
||||||
for (int id = from; id < to; id++)
|
|
||||||
{
|
|
||||||
TextFrame frame = new TextFrame();
|
|
||||||
frame.setPayload(Integer.toString(id));
|
|
||||||
sconnection.write(frame);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public void close() throws IOException
|
|
||||||
{
|
|
||||||
sconnection.close();
|
|
||||||
}
|
|
||||||
|
|
||||||
public void awaitConnect() throws InterruptedException
|
|
||||||
{
|
|
||||||
connectLatch.await(1,TimeUnit.SECONDS);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private static final Logger LOG = Log.getLogger(DecoderReaderManySmallTest.class);
|
|
||||||
|
|
||||||
@Rule
|
|
||||||
public TestTracker tt = new TestTracker();
|
|
||||||
|
|
||||||
private XBlockheadServer server;
|
|
||||||
private WebSocketContainer client;
|
|
||||||
|
|
||||||
@Before
|
|
||||||
public void initClient()
|
|
||||||
{
|
|
||||||
client = ContainerProvider.getWebSocketContainer();
|
|
||||||
}
|
|
||||||
|
|
||||||
@After
|
|
||||||
public void stopClient() throws Exception
|
|
||||||
{
|
|
||||||
((LifeCycle)client).stop();
|
|
||||||
}
|
|
||||||
|
|
||||||
@Before
|
|
||||||
public void startServer() throws Exception
|
|
||||||
{
|
|
||||||
server = new XBlockheadServer();
|
|
||||||
server.start();
|
|
||||||
}
|
|
||||||
|
|
||||||
@After
|
|
||||||
public void stopServer() throws Exception
|
|
||||||
{
|
|
||||||
server.stop();
|
|
||||||
}
|
|
||||||
|
|
||||||
@Test
|
|
||||||
public void testManyIds() throws Exception
|
|
||||||
{
|
|
||||||
EventIdSocket ids = new EventIdSocket();
|
|
||||||
EventIdServer idserver = new EventIdServer(server);
|
|
||||||
new Thread(idserver).start();
|
|
||||||
client.connectToServer(ids,server.getWsUri());
|
|
||||||
idserver.awaitConnect();
|
|
||||||
int from = 1000;
|
|
||||||
int to = 2000;
|
|
||||||
idserver.writeSequentialIds(from,to);
|
|
||||||
idserver.close();
|
|
||||||
int count = from - to;
|
|
||||||
ids.messageQueue.awaitEventCount(count,4,TimeUnit.SECONDS);
|
|
||||||
ids.awaitClose();
|
|
||||||
// collect seen ids
|
|
||||||
List<Integer> seen = new ArrayList<>();
|
|
||||||
for(EventId id: ids.messageQueue)
|
|
||||||
{
|
|
||||||
// validate that ids don't repeat.
|
|
||||||
Assert.assertFalse("Already saw ID: " + id.eventId, seen.contains(id.eventId));
|
|
||||||
seen.add(id.eventId);
|
|
||||||
}
|
|
||||||
|
|
||||||
// validate that all expected ids have been seen (order is irrelevant here)
|
|
||||||
for(int expected=from; expected<to; expected++)
|
|
||||||
{
|
|
||||||
Assert.assertTrue("Has expected id:"+expected,seen.contains(expected));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -1,300 +0,0 @@
|
||||||
//
|
|
||||||
// ========================================================================
|
|
||||||
// Copyright (c) 1995-2017 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.jsr356;
|
|
||||||
|
|
||||||
import static org.hamcrest.Matchers.*;
|
|
||||||
|
|
||||||
import java.io.BufferedReader;
|
|
||||||
import java.io.File;
|
|
||||||
import java.io.FileReader;
|
|
||||||
import java.io.IOException;
|
|
||||||
import java.io.Reader;
|
|
||||||
import java.util.ArrayList;
|
|
||||||
import java.util.List;
|
|
||||||
import java.util.concurrent.CountDownLatch;
|
|
||||||
import java.util.concurrent.TimeUnit;
|
|
||||||
|
|
||||||
import javax.websocket.ClientEndpoint;
|
|
||||||
import javax.websocket.CloseReason;
|
|
||||||
import javax.websocket.ContainerProvider;
|
|
||||||
import javax.websocket.DecodeException;
|
|
||||||
import javax.websocket.Decoder;
|
|
||||||
import javax.websocket.EndpointConfig;
|
|
||||||
import javax.websocket.OnClose;
|
|
||||||
import javax.websocket.OnMessage;
|
|
||||||
import javax.websocket.WebSocketContainer;
|
|
||||||
|
|
||||||
import org.eclipse.jetty.toolchain.test.EventQueue;
|
|
||||||
import org.eclipse.jetty.toolchain.test.MavenTestingUtils;
|
|
||||||
import org.eclipse.jetty.toolchain.test.TestTracker;
|
|
||||||
import org.eclipse.jetty.util.BufferUtil;
|
|
||||||
import org.eclipse.jetty.util.component.LifeCycle;
|
|
||||||
import org.eclipse.jetty.util.log.Log;
|
|
||||||
import org.eclipse.jetty.util.log.Logger;
|
|
||||||
import org.eclipse.jetty.websocket.common.WebSocketFrame;
|
|
||||||
import org.eclipse.jetty.websocket.common.frames.ContinuationFrame;
|
|
||||||
import org.eclipse.jetty.websocket.common.frames.TextFrame;
|
|
||||||
import org.eclipse.jetty.websocket.common.test.XBlockheadServer;
|
|
||||||
import org.eclipse.jetty.websocket.common.test.IBlockheadServerConnection;
|
|
||||||
import org.junit.After;
|
|
||||||
import org.junit.Assert;
|
|
||||||
import org.junit.Before;
|
|
||||||
import org.junit.Ignore;
|
|
||||||
import org.junit.Rule;
|
|
||||||
import org.junit.Test;
|
|
||||||
|
|
||||||
public class DecoderReaderTest
|
|
||||||
{
|
|
||||||
public static class Quotes
|
|
||||||
{
|
|
||||||
private String author;
|
|
||||||
private List<String> quotes = new ArrayList<>();
|
|
||||||
|
|
||||||
public String getAuthor()
|
|
||||||
{
|
|
||||||
return author;
|
|
||||||
}
|
|
||||||
|
|
||||||
public void setAuthor(String author)
|
|
||||||
{
|
|
||||||
this.author = author;
|
|
||||||
}
|
|
||||||
|
|
||||||
public List<String> getQuotes()
|
|
||||||
{
|
|
||||||
return quotes;
|
|
||||||
}
|
|
||||||
|
|
||||||
public void addQuote(String quote)
|
|
||||||
{
|
|
||||||
quotes.add(quote);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public static class QuotesDecoder implements Decoder.TextStream<Quotes>
|
|
||||||
{
|
|
||||||
@Override
|
|
||||||
public void init(EndpointConfig config)
|
|
||||||
{
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void destroy()
|
|
||||||
{
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public Quotes decode(Reader reader) throws DecodeException, IOException
|
|
||||||
{
|
|
||||||
Quotes quotes = new Quotes();
|
|
||||||
try (BufferedReader buf = new BufferedReader(reader))
|
|
||||||
{
|
|
||||||
String line;
|
|
||||||
while ((line = buf.readLine()) != null)
|
|
||||||
{
|
|
||||||
switch (line.charAt(0))
|
|
||||||
{
|
|
||||||
case 'a':
|
|
||||||
quotes.setAuthor(line.substring(2));
|
|
||||||
break;
|
|
||||||
case 'q':
|
|
||||||
quotes.addQuote(line.substring(2));
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return quotes;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@ClientEndpoint(decoders = { QuotesDecoder.class })
|
|
||||||
public static class QuotesSocket
|
|
||||||
{
|
|
||||||
public EventQueue<Quotes> messageQueue = new EventQueue<>();
|
|
||||||
private CountDownLatch closeLatch = new CountDownLatch(1);
|
|
||||||
|
|
||||||
@OnClose
|
|
||||||
public void onClose(CloseReason close)
|
|
||||||
{
|
|
||||||
closeLatch.countDown();
|
|
||||||
}
|
|
||||||
|
|
||||||
@OnMessage
|
|
||||||
public synchronized void onMessage(Quotes msg)
|
|
||||||
{
|
|
||||||
Integer h=hashCode();
|
|
||||||
messageQueue.add(msg);
|
|
||||||
System.out.printf("%x: Quotes from: %s%n",h,msg.author);
|
|
||||||
for (String quote : msg.quotes)
|
|
||||||
{
|
|
||||||
System.out.printf("%x: - %s%n",h,quote);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public void awaitClose() throws InterruptedException
|
|
||||||
{
|
|
||||||
closeLatch.await(4,TimeUnit.SECONDS);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private static class QuoteServer implements Runnable
|
|
||||||
{
|
|
||||||
private XBlockheadServer server;
|
|
||||||
private IBlockheadServerConnection sconnection;
|
|
||||||
private CountDownLatch connectLatch = new CountDownLatch(1);
|
|
||||||
|
|
||||||
public QuoteServer(XBlockheadServer server)
|
|
||||||
{
|
|
||||||
this.server = server;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void run()
|
|
||||||
{
|
|
||||||
try
|
|
||||||
{
|
|
||||||
sconnection = server.accept();
|
|
||||||
sconnection.setSoTimeout(60000);
|
|
||||||
sconnection.upgrade();
|
|
||||||
}
|
|
||||||
catch (Exception e)
|
|
||||||
{
|
|
||||||
LOG.warn(e);
|
|
||||||
}
|
|
||||||
finally
|
|
||||||
{
|
|
||||||
connectLatch.countDown();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public void writeQuotes(String filename) throws IOException
|
|
||||||
{
|
|
||||||
// read file
|
|
||||||
File qfile = MavenTestingUtils.getTestResourceFile(filename);
|
|
||||||
List<String> lines = new ArrayList<>();
|
|
||||||
try (FileReader reader = new FileReader(qfile); BufferedReader buf = new BufferedReader(reader))
|
|
||||||
{
|
|
||||||
String line;
|
|
||||||
while ((line = buf.readLine()) != null)
|
|
||||||
{
|
|
||||||
lines.add(line);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
// write file out, each line on a separate frame, but as
|
|
||||||
// 1 whole message
|
|
||||||
for (int i = 0; i < lines.size(); i++)
|
|
||||||
{
|
|
||||||
WebSocketFrame frame;
|
|
||||||
if (i == 0)
|
|
||||||
{
|
|
||||||
frame = new TextFrame();
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
frame = new ContinuationFrame();
|
|
||||||
}
|
|
||||||
frame.setFin((i >= (lines.size() - 1)));
|
|
||||||
frame.setPayload(BufferUtil.toBuffer(lines.get(i) + "\n"));
|
|
||||||
sconnection.write(frame);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public void close() throws IOException
|
|
||||||
{
|
|
||||||
sconnection.close();
|
|
||||||
}
|
|
||||||
|
|
||||||
public void awaitConnect() throws InterruptedException
|
|
||||||
{
|
|
||||||
connectLatch.await(1,TimeUnit.SECONDS);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private static final Logger LOG = Log.getLogger(DecoderReaderTest.class);
|
|
||||||
|
|
||||||
@Rule
|
|
||||||
public TestTracker tt = new TestTracker();
|
|
||||||
|
|
||||||
private XBlockheadServer server;
|
|
||||||
private WebSocketContainer client;
|
|
||||||
|
|
||||||
@Before
|
|
||||||
public void initClient()
|
|
||||||
{
|
|
||||||
client = ContainerProvider.getWebSocketContainer();
|
|
||||||
}
|
|
||||||
|
|
||||||
@After
|
|
||||||
public void stopClient() throws Exception
|
|
||||||
{
|
|
||||||
((LifeCycle)client).stop();
|
|
||||||
}
|
|
||||||
|
|
||||||
@Before
|
|
||||||
public void startServer() throws Exception
|
|
||||||
{
|
|
||||||
server = new XBlockheadServer();
|
|
||||||
server.start();
|
|
||||||
}
|
|
||||||
|
|
||||||
@After
|
|
||||||
public void stopServer() throws Exception
|
|
||||||
{
|
|
||||||
server.stop();
|
|
||||||
}
|
|
||||||
|
|
||||||
// TODO analyse and fix
|
|
||||||
@Ignore
|
|
||||||
@Test
|
|
||||||
public void testSingleQuotes() throws Exception
|
|
||||||
{
|
|
||||||
QuotesSocket quoter = new QuotesSocket();
|
|
||||||
QuoteServer qserver = new QuoteServer(server);
|
|
||||||
new Thread(qserver).start();
|
|
||||||
client.connectToServer(quoter,server.getWsUri());
|
|
||||||
qserver.awaitConnect();
|
|
||||||
qserver.writeQuotes("quotes-ben.txt");
|
|
||||||
quoter.messageQueue.awaitEventCount(1,1000,TimeUnit.MILLISECONDS);
|
|
||||||
qserver.close();
|
|
||||||
quoter.awaitClose();
|
|
||||||
Quotes quotes = quoter.messageQueue.poll();
|
|
||||||
Assert.assertThat("Quotes Author",quotes.author,is("Benjamin Franklin"));
|
|
||||||
Assert.assertThat("Quotes Count",quotes.quotes.size(),is(3));
|
|
||||||
}
|
|
||||||
|
|
||||||
// TODO analyse and fix
|
|
||||||
@Test
|
|
||||||
@Ignore ("Quotes appear to be able to arrive in any order?")
|
|
||||||
public void testTwoQuotes() throws Exception
|
|
||||||
{
|
|
||||||
QuotesSocket quoter = new QuotesSocket();
|
|
||||||
QuoteServer qserver = new QuoteServer(server);
|
|
||||||
new Thread(qserver).start();
|
|
||||||
client.connectToServer(quoter,server.getWsUri());
|
|
||||||
qserver.awaitConnect();
|
|
||||||
qserver.writeQuotes("quotes-ben.txt");
|
|
||||||
qserver.writeQuotes("quotes-twain.txt");
|
|
||||||
quoter.messageQueue.awaitEventCount(2,1000,TimeUnit.MILLISECONDS);
|
|
||||||
qserver.close();
|
|
||||||
quoter.awaitClose();
|
|
||||||
Quotes quotes = quoter.messageQueue.poll();
|
|
||||||
Assert.assertThat("Quotes Author",quotes.author,is("Benjamin Franklin"));
|
|
||||||
Assert.assertThat("Quotes Count",quotes.quotes.size(),is(3));
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -1,325 +0,0 @@
|
||||||
//
|
|
||||||
// ========================================================================
|
|
||||||
// Copyright (c) 1995-2017 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.jsr356;
|
|
||||||
|
|
||||||
import static org.hamcrest.Matchers.containsString;
|
|
||||||
|
|
||||||
import java.io.BufferedReader;
|
|
||||||
import java.io.File;
|
|
||||||
import java.io.FileReader;
|
|
||||||
import java.io.IOException;
|
|
||||||
import java.util.ArrayList;
|
|
||||||
import java.util.List;
|
|
||||||
import java.util.concurrent.CountDownLatch;
|
|
||||||
import java.util.concurrent.TimeUnit;
|
|
||||||
|
|
||||||
import javax.websocket.ClientEndpointConfig;
|
|
||||||
import javax.websocket.ContainerProvider;
|
|
||||||
import javax.websocket.EncodeException;
|
|
||||||
import javax.websocket.Encoder;
|
|
||||||
import javax.websocket.Endpoint;
|
|
||||||
import javax.websocket.EndpointConfig;
|
|
||||||
import javax.websocket.MessageHandler;
|
|
||||||
import javax.websocket.Session;
|
|
||||||
import javax.websocket.WebSocketContainer;
|
|
||||||
|
|
||||||
import org.eclipse.jetty.toolchain.test.EventQueue;
|
|
||||||
import org.eclipse.jetty.toolchain.test.MavenTestingUtils;
|
|
||||||
import org.eclipse.jetty.toolchain.test.TestTracker;
|
|
||||||
import org.eclipse.jetty.util.component.LifeCycle;
|
|
||||||
import org.eclipse.jetty.util.log.Log;
|
|
||||||
import org.eclipse.jetty.util.log.Logger;
|
|
||||||
import org.eclipse.jetty.websocket.common.test.XBlockheadServer;
|
|
||||||
import org.eclipse.jetty.websocket.common.test.IBlockheadServerConnection;
|
|
||||||
import org.junit.After;
|
|
||||||
import org.junit.Assert;
|
|
||||||
import org.junit.Before;
|
|
||||||
import org.junit.Rule;
|
|
||||||
import org.junit.Test;
|
|
||||||
|
|
||||||
public class EncoderTest
|
|
||||||
{
|
|
||||||
private static class EchoServer implements Runnable
|
|
||||||
{
|
|
||||||
private Thread thread;
|
|
||||||
private XBlockheadServer server;
|
|
||||||
private IBlockheadServerConnection sconnection;
|
|
||||||
private CountDownLatch connectLatch = new CountDownLatch(1);
|
|
||||||
|
|
||||||
public EchoServer(XBlockheadServer server)
|
|
||||||
{
|
|
||||||
this.server = server;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void run()
|
|
||||||
{
|
|
||||||
try
|
|
||||||
{
|
|
||||||
sconnection = server.accept();
|
|
||||||
sconnection.setSoTimeout(60000);
|
|
||||||
sconnection.upgrade();
|
|
||||||
sconnection.startEcho();
|
|
||||||
}
|
|
||||||
catch (Exception e)
|
|
||||||
{
|
|
||||||
LOG.warn(e);
|
|
||||||
}
|
|
||||||
finally
|
|
||||||
{
|
|
||||||
connectLatch.countDown();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public void start()
|
|
||||||
{
|
|
||||||
this.thread = new Thread(this,"EchoServer");
|
|
||||||
this.thread.start();
|
|
||||||
}
|
|
||||||
|
|
||||||
public void stop()
|
|
||||||
{
|
|
||||||
if (this.sconnection != null)
|
|
||||||
{
|
|
||||||
this.sconnection.stopEcho();
|
|
||||||
try
|
|
||||||
{
|
|
||||||
this.sconnection.close();
|
|
||||||
}
|
|
||||||
catch (IOException ignore)
|
|
||||||
{
|
|
||||||
/* ignore */
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public static class Quotes
|
|
||||||
{
|
|
||||||
private String author;
|
|
||||||
private List<String> quotes = new ArrayList<>();
|
|
||||||
|
|
||||||
public void addQuote(String quote)
|
|
||||||
{
|
|
||||||
quotes.add(quote);
|
|
||||||
}
|
|
||||||
|
|
||||||
public String getAuthor()
|
|
||||||
{
|
|
||||||
return author;
|
|
||||||
}
|
|
||||||
|
|
||||||
public List<String> getQuotes()
|
|
||||||
{
|
|
||||||
return quotes;
|
|
||||||
}
|
|
||||||
|
|
||||||
public void setAuthor(String author)
|
|
||||||
{
|
|
||||||
this.author = author;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public static class QuotesEncoder implements Encoder.Text<Quotes>
|
|
||||||
{
|
|
||||||
@Override
|
|
||||||
public void destroy()
|
|
||||||
{
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public String encode(Quotes q) throws EncodeException
|
|
||||||
{
|
|
||||||
StringBuilder buf = new StringBuilder();
|
|
||||||
buf.append("Author: ").append(q.getAuthor());
|
|
||||||
buf.append(System.lineSeparator());
|
|
||||||
for (String quote : q.quotes)
|
|
||||||
{
|
|
||||||
buf.append("Quote: ").append(quote);
|
|
||||||
buf.append(System.lineSeparator());
|
|
||||||
}
|
|
||||||
return buf.toString();
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void init(EndpointConfig config)
|
|
||||||
{
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public static class QuotesSocket extends Endpoint implements MessageHandler.Whole<String>
|
|
||||||
{
|
|
||||||
private Session session;
|
|
||||||
private EventQueue<String> messageQueue = new EventQueue<>();
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void onMessage(String message)
|
|
||||||
{
|
|
||||||
messageQueue.add(message);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void onOpen(Session session, EndpointConfig config)
|
|
||||||
{
|
|
||||||
this.session = session;
|
|
||||||
this.session.addMessageHandler(this);
|
|
||||||
}
|
|
||||||
|
|
||||||
public void write(Quotes quotes) throws IOException, EncodeException
|
|
||||||
{
|
|
||||||
if (LOG.isDebugEnabled())
|
|
||||||
LOG.debug("Writing Quotes: {}",quotes);
|
|
||||||
this.session.getBasicRemote().sendObject(quotes);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private static final Logger LOG = Log.getLogger(EncoderTest.class);
|
|
||||||
|
|
||||||
@Rule
|
|
||||||
public TestTracker tt = new TestTracker();
|
|
||||||
private XBlockheadServer server;
|
|
||||||
|
|
||||||
private WebSocketContainer client;
|
|
||||||
|
|
||||||
private void assertReceivedQuotes(String result, Quotes quotes)
|
|
||||||
{
|
|
||||||
Assert.assertThat("Quote Author",result,containsString("Author: " + quotes.getAuthor()));
|
|
||||||
for (String quote : quotes.quotes)
|
|
||||||
{
|
|
||||||
Assert.assertThat("Quote",result,containsString("Quote: " + quote));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@SuppressWarnings("Duplicates")
|
|
||||||
private Quotes getQuotes(String filename) throws IOException
|
|
||||||
{
|
|
||||||
Quotes quotes = new Quotes();
|
|
||||||
|
|
||||||
// read file
|
|
||||||
File qfile = MavenTestingUtils.getTestResourceFile(filename);
|
|
||||||
try (FileReader reader = new FileReader(qfile); BufferedReader buf = new BufferedReader(reader))
|
|
||||||
{
|
|
||||||
String line;
|
|
||||||
while ((line = buf.readLine()) != null)
|
|
||||||
{
|
|
||||||
switch (line.charAt(0))
|
|
||||||
{
|
|
||||||
case 'a':
|
|
||||||
quotes.setAuthor(line.substring(2));
|
|
||||||
break;
|
|
||||||
case 'q':
|
|
||||||
quotes.addQuote(line.substring(2));
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return quotes;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Before
|
|
||||||
public void initClient()
|
|
||||||
{
|
|
||||||
client = ContainerProvider.getWebSocketContainer();
|
|
||||||
}
|
|
||||||
|
|
||||||
@After
|
|
||||||
public void stopClient() throws Exception
|
|
||||||
{
|
|
||||||
((LifeCycle)client).stop();
|
|
||||||
}
|
|
||||||
|
|
||||||
@Before
|
|
||||||
public void startServer() throws Exception
|
|
||||||
{
|
|
||||||
server = new XBlockheadServer();
|
|
||||||
server.start();
|
|
||||||
}
|
|
||||||
|
|
||||||
@After
|
|
||||||
public void stopServer() throws Exception
|
|
||||||
{
|
|
||||||
server.stop();
|
|
||||||
}
|
|
||||||
|
|
||||||
@Test
|
|
||||||
public void testSingleQuotes() throws Exception
|
|
||||||
{
|
|
||||||
EchoServer echoServer = new EchoServer(server);
|
|
||||||
try
|
|
||||||
{
|
|
||||||
echoServer.start();
|
|
||||||
|
|
||||||
QuotesSocket quoter = new QuotesSocket();
|
|
||||||
|
|
||||||
ClientEndpointConfig.Builder builder = ClientEndpointConfig.Builder.create();
|
|
||||||
List<Class<? extends Encoder>> encoders = new ArrayList<>();
|
|
||||||
encoders.add(QuotesEncoder.class);
|
|
||||||
builder.encoders(encoders);
|
|
||||||
ClientEndpointConfig cec = builder.build();
|
|
||||||
client.connectToServer(quoter,cec,server.getWsUri());
|
|
||||||
|
|
||||||
Quotes ben = getQuotes("quotes-ben.txt");
|
|
||||||
quoter.write(ben);
|
|
||||||
|
|
||||||
quoter.messageQueue.awaitEventCount(1,1000,TimeUnit.MILLISECONDS);
|
|
||||||
|
|
||||||
String result = quoter.messageQueue.poll();
|
|
||||||
assertReceivedQuotes(result,ben);
|
|
||||||
}
|
|
||||||
finally
|
|
||||||
{
|
|
||||||
echoServer.stop();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@Test
|
|
||||||
public void testTwoQuotes() throws Exception
|
|
||||||
{
|
|
||||||
EchoServer echoServer = new EchoServer(server);
|
|
||||||
try
|
|
||||||
{
|
|
||||||
echoServer.start();
|
|
||||||
|
|
||||||
QuotesSocket quoter = new QuotesSocket();
|
|
||||||
ClientEndpointConfig.Builder builder = ClientEndpointConfig.Builder.create();
|
|
||||||
List<Class<? extends Encoder>> encoders = new ArrayList<>();
|
|
||||||
encoders.add(QuotesEncoder.class);
|
|
||||||
builder.encoders(encoders);
|
|
||||||
ClientEndpointConfig cec = builder.build();
|
|
||||||
client.connectToServer(quoter,cec,server.getWsUri());
|
|
||||||
|
|
||||||
Quotes ben = getQuotes("quotes-ben.txt");
|
|
||||||
Quotes twain = getQuotes("quotes-twain.txt");
|
|
||||||
quoter.write(ben);
|
|
||||||
quoter.write(twain);
|
|
||||||
|
|
||||||
quoter.messageQueue.awaitEventCount(2,1000,TimeUnit.MILLISECONDS);
|
|
||||||
|
|
||||||
String result = quoter.messageQueue.poll();
|
|
||||||
assertReceivedQuotes(result,ben);
|
|
||||||
result = quoter.messageQueue.poll();
|
|
||||||
assertReceivedQuotes(result,twain);
|
|
||||||
}
|
|
||||||
finally
|
|
||||||
{
|
|
||||||
echoServer.stop();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -1 +0,0 @@
|
||||||
org.eclipse.jetty.websocket.jsr356.server.PathParamArgIdentifier
|
|
|
@ -20,6 +20,11 @@
|
||||||
<artifactId>websocket-api</artifactId>
|
<artifactId>websocket-api</artifactId>
|
||||||
<version>${project.version}</version>
|
<version>${project.version}</version>
|
||||||
</dependency>
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>org.eclipse.jetty.websocket</groupId>
|
||||||
|
<artifactId>javax-websocket-client-impl</artifactId>
|
||||||
|
<version>${project.version}</version>
|
||||||
|
</dependency>
|
||||||
<dependency>
|
<dependency>
|
||||||
<groupId>org.eclipse.jetty.websocket</groupId>
|
<groupId>org.eclipse.jetty.websocket</groupId>
|
||||||
<artifactId>websocket-client</artifactId>
|
<artifactId>websocket-client</artifactId>
|
||||||
|
@ -30,6 +35,11 @@
|
||||||
<artifactId>websocket-server</artifactId>
|
<artifactId>websocket-server</artifactId>
|
||||||
<version>${project.version}</version>
|
<version>${project.version}</version>
|
||||||
</dependency>
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>org.eclipse.jetty.websocket</groupId>
|
||||||
|
<artifactId>javax-websocket-server-impl</artifactId>
|
||||||
|
<version>${project.version}</version>
|
||||||
|
</dependency>
|
||||||
<dependency>
|
<dependency>
|
||||||
<groupId>org.eclipse.jetty</groupId>
|
<groupId>org.eclipse.jetty</groupId>
|
||||||
<artifactId>jetty-util</artifactId>
|
<artifactId>jetty-util</artifactId>
|
||||||
|
|
|
@ -0,0 +1,46 @@
|
||||||
|
//
|
||||||
|
// ========================================================================
|
||||||
|
// Copyright (c) 1995-2017 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.tests;
|
||||||
|
|
||||||
|
import javax.websocket.CloseReason;
|
||||||
|
import javax.websocket.EndpointConfig;
|
||||||
|
import javax.websocket.OnClose;
|
||||||
|
import javax.websocket.OnOpen;
|
||||||
|
import javax.websocket.Session;
|
||||||
|
|
||||||
|
@SuppressWarnings("unused")
|
||||||
|
public abstract class AbstractJsrTrackingEndpoint extends AbstractTrackingEndpoint<Session>
|
||||||
|
{
|
||||||
|
public AbstractJsrTrackingEndpoint(String id)
|
||||||
|
{
|
||||||
|
super(id);
|
||||||
|
}
|
||||||
|
|
||||||
|
@OnOpen
|
||||||
|
public void onOpen(Session session, EndpointConfig config)
|
||||||
|
{
|
||||||
|
super.onWSOpen(session);
|
||||||
|
}
|
||||||
|
|
||||||
|
@OnClose
|
||||||
|
public void onClose(CloseReason closeReason)
|
||||||
|
{
|
||||||
|
super.onWSClose(closeReason.getCloseCode().getCode(), closeReason.getReasonPhrase());
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,119 @@
|
||||||
|
//
|
||||||
|
// ========================================================================
|
||||||
|
// Copyright (c) 1995-2017 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.tests;
|
||||||
|
|
||||||
|
import static org.hamcrest.CoreMatchers.notNullValue;
|
||||||
|
import static org.junit.Assert.assertThat;
|
||||||
|
import static org.junit.Assert.assertTrue;
|
||||||
|
import static org.junit.Assert.fail;
|
||||||
|
|
||||||
|
import java.util.concurrent.CountDownLatch;
|
||||||
|
import java.util.concurrent.TimeUnit;
|
||||||
|
import java.util.concurrent.atomic.AtomicReference;
|
||||||
|
|
||||||
|
import org.eclipse.jetty.util.log.Log;
|
||||||
|
import org.eclipse.jetty.util.log.Logger;
|
||||||
|
import org.eclipse.jetty.websocket.common.CloseInfo;
|
||||||
|
import org.hamcrest.Matcher;
|
||||||
|
import org.hamcrest.Matchers;
|
||||||
|
|
||||||
|
public abstract class AbstractTrackingEndpoint<T>
|
||||||
|
{
|
||||||
|
public final Logger LOG;
|
||||||
|
|
||||||
|
public T session;
|
||||||
|
|
||||||
|
public CountDownLatch openLatch = new CountDownLatch(1);
|
||||||
|
public CountDownLatch closeLatch = new CountDownLatch(1);
|
||||||
|
public AtomicReference<CloseInfo> closeInfo = new AtomicReference<>();
|
||||||
|
public AtomicReference<Throwable> error = new AtomicReference<>();
|
||||||
|
|
||||||
|
public AbstractTrackingEndpoint(String id)
|
||||||
|
{
|
||||||
|
LOG = Log.getLogger(this.getClass().getName() + "." + id);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void assertCloseInfo(String prefix, int expectedCloseStatusCode, Matcher<String> reasonMatcher) throws InterruptedException
|
||||||
|
{
|
||||||
|
CloseInfo close = closeInfo.get();
|
||||||
|
assertThat(prefix + " close info", close, Matchers.notNullValue());
|
||||||
|
assertThat(prefix + " received close code", close.getStatusCode(), Matchers.is(expectedCloseStatusCode));
|
||||||
|
assertThat(prefix + " received close reason", close.getReason(), reasonMatcher);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void assertErrorEvent(String prefix, Matcher<Throwable> throwableMatcher, Matcher<? super String> messageMatcher)
|
||||||
|
{
|
||||||
|
assertThat(prefix + " error event type", error.get(), throwableMatcher);
|
||||||
|
assertThat(prefix + " error event message", error.get().getMessage(), messageMatcher);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void assertNoErrorEvents(String prefix)
|
||||||
|
{
|
||||||
|
assertTrue(prefix + " error event should not have occurred", error.get() == null);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void assertNotClosed(String prefix)
|
||||||
|
{
|
||||||
|
assertTrue(prefix + " close event should not have occurred", closeLatch.getCount() > 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void assertNotOpened(String prefix)
|
||||||
|
{
|
||||||
|
assertTrue(prefix + " open event should not have occurred", openLatch.getCount() > 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void awaitCloseEvent(String prefix) throws InterruptedException
|
||||||
|
{
|
||||||
|
assertTrue(prefix + " onClose event", closeLatch.await(Defaults.CLOSE_EVENT_TIMEOUT_MS, TimeUnit.MILLISECONDS));
|
||||||
|
}
|
||||||
|
|
||||||
|
public void awaitOpenEvent(String prefix) throws InterruptedException
|
||||||
|
{
|
||||||
|
assertTrue(prefix + " onOpen event", openLatch.await(Defaults.OPEN_EVENT_TIMEOUT_MS, TimeUnit.MILLISECONDS));
|
||||||
|
}
|
||||||
|
|
||||||
|
protected void onWSOpen(T session)
|
||||||
|
{
|
||||||
|
this.session = session;
|
||||||
|
if (LOG.isDebugEnabled())
|
||||||
|
{
|
||||||
|
LOG.debug("onWSOpen()");
|
||||||
|
}
|
||||||
|
this.openLatch.countDown();
|
||||||
|
}
|
||||||
|
|
||||||
|
protected void onWSClose(int statusCode, String reason)
|
||||||
|
{
|
||||||
|
CloseInfo close = new CloseInfo(statusCode, reason);
|
||||||
|
boolean closeTracked = closeInfo.compareAndSet(null, close);
|
||||||
|
this.closeLatch.countDown();
|
||||||
|
assertTrue("Close only happened once", closeTracked);
|
||||||
|
}
|
||||||
|
|
||||||
|
protected void onWSError(Throwable cause)
|
||||||
|
{
|
||||||
|
assertThat("Error must have value", cause, notNullValue());
|
||||||
|
if (error.compareAndSet(null, cause) == false)
|
||||||
|
{
|
||||||
|
LOG.warn("onError should only happen once - Original Cause", error.get());
|
||||||
|
LOG.warn("onError should only happen once - Extra/Excess Cause", cause);
|
||||||
|
fail("onError should only happen once!");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -19,23 +19,14 @@
|
||||||
package org.eclipse.jetty.websocket.tests;
|
package org.eclipse.jetty.websocket.tests;
|
||||||
|
|
||||||
import static org.hamcrest.CoreMatchers.instanceOf;
|
import static org.hamcrest.CoreMatchers.instanceOf;
|
||||||
import static org.hamcrest.CoreMatchers.is;
|
|
||||||
import static org.hamcrest.CoreMatchers.notNullValue;
|
|
||||||
import static org.junit.Assert.assertThat;
|
import static org.junit.Assert.assertThat;
|
||||||
import static org.junit.Assert.assertTrue;
|
|
||||||
import static org.junit.Assert.fail;
|
|
||||||
|
|
||||||
import java.nio.ByteBuffer;
|
import java.nio.ByteBuffer;
|
||||||
import java.util.concurrent.BlockingQueue;
|
import java.util.concurrent.BlockingQueue;
|
||||||
import java.util.concurrent.CountDownLatch;
|
|
||||||
import java.util.concurrent.LinkedBlockingDeque;
|
import java.util.concurrent.LinkedBlockingDeque;
|
||||||
import java.util.concurrent.TimeUnit;
|
|
||||||
import java.util.concurrent.atomic.AtomicReference;
|
|
||||||
|
|
||||||
import org.eclipse.jetty.io.EndPoint;
|
import org.eclipse.jetty.io.EndPoint;
|
||||||
import org.eclipse.jetty.util.BufferUtil;
|
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.RemoteEndpoint;
|
import org.eclipse.jetty.websocket.api.RemoteEndpoint;
|
||||||
import org.eclipse.jetty.websocket.api.Session;
|
import org.eclipse.jetty.websocket.api.Session;
|
||||||
import org.eclipse.jetty.websocket.api.UpgradeRequest;
|
import org.eclipse.jetty.websocket.api.UpgradeRequest;
|
||||||
|
@ -43,75 +34,23 @@ import org.eclipse.jetty.websocket.api.UpgradeResponse;
|
||||||
import org.eclipse.jetty.websocket.api.WebSocketFrameListener;
|
import org.eclipse.jetty.websocket.api.WebSocketFrameListener;
|
||||||
import org.eclipse.jetty.websocket.api.WebSocketListener;
|
import org.eclipse.jetty.websocket.api.WebSocketListener;
|
||||||
import org.eclipse.jetty.websocket.api.extensions.Frame;
|
import org.eclipse.jetty.websocket.api.extensions.Frame;
|
||||||
import org.eclipse.jetty.websocket.common.CloseInfo;
|
|
||||||
import org.eclipse.jetty.websocket.common.LogicalConnection;
|
import org.eclipse.jetty.websocket.common.LogicalConnection;
|
||||||
import org.eclipse.jetty.websocket.common.WebSocketFrame;
|
import org.eclipse.jetty.websocket.common.WebSocketFrame;
|
||||||
import org.eclipse.jetty.websocket.common.WebSocketSession;
|
import org.eclipse.jetty.websocket.common.WebSocketSession;
|
||||||
import org.eclipse.jetty.websocket.common.io.AbstractWebSocketConnection;
|
import org.eclipse.jetty.websocket.common.io.AbstractWebSocketConnection;
|
||||||
import org.hamcrest.Matcher;
|
|
||||||
import org.hamcrest.Matchers;
|
|
||||||
|
|
||||||
public class TrackingEndpoint implements WebSocketListener, WebSocketFrameListener
|
public class TrackingEndpoint extends AbstractTrackingEndpoint<WebSocketSession> implements WebSocketListener, WebSocketFrameListener
|
||||||
{
|
{
|
||||||
public final Logger LOG;
|
|
||||||
|
|
||||||
public CountDownLatch openLatch = new CountDownLatch(1);
|
|
||||||
|
|
||||||
public UpgradeRequest openUpgradeRequest;
|
public UpgradeRequest openUpgradeRequest;
|
||||||
public UpgradeResponse openUpgradeResponse;
|
public UpgradeResponse openUpgradeResponse;
|
||||||
|
|
||||||
public CountDownLatch closeLatch = new CountDownLatch(1);
|
public BlockingQueue<WebSocketFrame> framesQueue = new LinkedBlockingDeque<>();
|
||||||
public AtomicReference<CloseInfo> closeInfo = new AtomicReference<>();
|
|
||||||
public AtomicReference<Throwable> error = new AtomicReference<>();
|
|
||||||
|
|
||||||
public BlockingQueue<String> messageQueue = new LinkedBlockingDeque<>();
|
public BlockingQueue<String> messageQueue = new LinkedBlockingDeque<>();
|
||||||
public BlockingQueue<ByteBuffer> bufferQueue = new LinkedBlockingDeque<>();
|
public BlockingQueue<ByteBuffer> bufferQueue = new LinkedBlockingDeque<>();
|
||||||
public BlockingQueue<WebSocketFrame> framesQueue = new LinkedBlockingDeque<>();
|
|
||||||
|
|
||||||
public WebSocketSession session;
|
|
||||||
|
|
||||||
public TrackingEndpoint(String id)
|
public TrackingEndpoint(String id)
|
||||||
{
|
{
|
||||||
LOG = Log.getLogger(this.getClass().getName() + "." + id);
|
super(id);
|
||||||
}
|
|
||||||
|
|
||||||
public void assertCloseInfo(String prefix, int expectedCloseStatusCode, Matcher<String> reasonMatcher) throws InterruptedException
|
|
||||||
{
|
|
||||||
CloseInfo close = closeInfo.get();
|
|
||||||
assertThat(prefix + " close info", close, Matchers.notNullValue());
|
|
||||||
assertThat(prefix + " received close code", close.getStatusCode(), Matchers.is(expectedCloseStatusCode));
|
|
||||||
assertThat(prefix + " received close reason", close.getReason(), reasonMatcher);
|
|
||||||
}
|
|
||||||
|
|
||||||
public void assertNotOpened(String prefix)
|
|
||||||
{
|
|
||||||
assertTrue(prefix + " open event should not have occurred", openLatch.getCount() > 0);
|
|
||||||
}
|
|
||||||
|
|
||||||
public void assertNotClosed(String prefix)
|
|
||||||
{
|
|
||||||
assertTrue(prefix + " close event should not have occurred", closeLatch.getCount() > 0);
|
|
||||||
}
|
|
||||||
|
|
||||||
public void assertNoErrorEvents(String prefix)
|
|
||||||
{
|
|
||||||
assertTrue(prefix + " error event should not have occurred", error.get() == null);
|
|
||||||
}
|
|
||||||
|
|
||||||
public void assertErrorEvent(String prefix, Matcher<Throwable> throwableMatcher, Matcher<? super String> messageMatcher)
|
|
||||||
{
|
|
||||||
assertThat(prefix + " error event type", error.get(), throwableMatcher);
|
|
||||||
assertThat(prefix + " error event message", error.get().getMessage(), messageMatcher);
|
|
||||||
}
|
|
||||||
|
|
||||||
public void awaitOpenEvent(String prefix) throws InterruptedException
|
|
||||||
{
|
|
||||||
assertTrue(prefix + " onOpen event", openLatch.await(Defaults.OPEN_EVENT_TIMEOUT_MS, TimeUnit.MILLISECONDS));
|
|
||||||
}
|
|
||||||
|
|
||||||
public void awaitCloseEvent(String prefix) throws InterruptedException
|
|
||||||
{
|
|
||||||
assertTrue(prefix + " onClose event", closeLatch.await(Defaults.CLOSE_EVENT_TIMEOUT_MS, TimeUnit.MILLISECONDS));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public void close(int statusCode, String reason)
|
public void close(int statusCode, String reason)
|
||||||
|
@ -138,35 +77,22 @@ public class TrackingEndpoint implements WebSocketListener, WebSocketFrameListen
|
||||||
@Override
|
@Override
|
||||||
public void onWebSocketClose(int statusCode, String reason)
|
public void onWebSocketClose(int statusCode, String reason)
|
||||||
{
|
{
|
||||||
this.closeLatch.countDown();
|
super.onWSClose(statusCode, reason);
|
||||||
CloseInfo close = new CloseInfo(statusCode, reason);
|
|
||||||
assertThat("Close only happened once", closeInfo.compareAndSet(null, close), is(true));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void onWebSocketConnect(Session session)
|
public void onWebSocketConnect(Session session)
|
||||||
{
|
{
|
||||||
assertThat("Session type", session, instanceOf(WebSocketSession.class));
|
assertThat("Session type", session, instanceOf(WebSocketSession.class));
|
||||||
this.session = (WebSocketSession) session;
|
super.onWSOpen((WebSocketSession) session);
|
||||||
this.openUpgradeRequest = session.getUpgradeRequest();
|
this.openUpgradeRequest = session.getUpgradeRequest();
|
||||||
this.openUpgradeResponse = session.getUpgradeResponse();
|
this.openUpgradeResponse = session.getUpgradeResponse();
|
||||||
if (LOG.isDebugEnabled())
|
|
||||||
{
|
|
||||||
LOG.debug("onWebSocketConnect()");
|
|
||||||
}
|
|
||||||
this.openLatch.countDown();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void onWebSocketError(Throwable cause)
|
public void onWebSocketError(Throwable cause)
|
||||||
{
|
{
|
||||||
assertThat("Error must have value", cause, notNullValue());
|
super.onWSError(cause);
|
||||||
if (error.compareAndSet(null, cause) == false)
|
|
||||||
{
|
|
||||||
LOG.warn("onError should only happen once - Original Cause", error.get());
|
|
||||||
LOG.warn("onError should only happen once - Extra/Excess Cause", cause);
|
|
||||||
fail("onError should only happen once!");
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
|
|
@ -0,0 +1,202 @@
|
||||||
|
//
|
||||||
|
// ========================================================================
|
||||||
|
// Copyright (c) 1995-2017 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.tests.client.jsr356;
|
||||||
|
|
||||||
|
import java.io.BufferedReader;
|
||||||
|
import java.io.IOException;
|
||||||
|
import java.io.Reader;
|
||||||
|
import java.net.URI;
|
||||||
|
import java.util.ArrayList;
|
||||||
|
import java.util.List;
|
||||||
|
import java.util.concurrent.BlockingQueue;
|
||||||
|
import java.util.concurrent.LinkedBlockingDeque;
|
||||||
|
import java.util.concurrent.TimeUnit;
|
||||||
|
|
||||||
|
import javax.websocket.ClientEndpoint;
|
||||||
|
import javax.websocket.ContainerProvider;
|
||||||
|
import javax.websocket.DecodeException;
|
||||||
|
import javax.websocket.Decoder;
|
||||||
|
import javax.websocket.EndpointConfig;
|
||||||
|
import javax.websocket.OnMessage;
|
||||||
|
import javax.websocket.Session;
|
||||||
|
import javax.websocket.WebSocketContainer;
|
||||||
|
|
||||||
|
import org.eclipse.jetty.websocket.api.BatchMode;
|
||||||
|
import org.eclipse.jetty.websocket.api.WebSocketBehavior;
|
||||||
|
import org.eclipse.jetty.websocket.servlet.ServletUpgradeRequest;
|
||||||
|
import org.eclipse.jetty.websocket.servlet.ServletUpgradeResponse;
|
||||||
|
import org.eclipse.jetty.websocket.servlet.WebSocketCreator;
|
||||||
|
import org.eclipse.jetty.websocket.tests.AbstractJsrTrackingEndpoint;
|
||||||
|
import org.eclipse.jetty.websocket.tests.UntrustedWSEndpoint;
|
||||||
|
import org.eclipse.jetty.websocket.tests.UntrustedWSServer;
|
||||||
|
import org.junit.After;
|
||||||
|
import org.junit.Assert;
|
||||||
|
import org.junit.Before;
|
||||||
|
import org.junit.Rule;
|
||||||
|
import org.junit.Test;
|
||||||
|
import org.junit.rules.TestName;
|
||||||
|
|
||||||
|
public class DecoderReaderManySmallTest
|
||||||
|
{
|
||||||
|
public static class EventId
|
||||||
|
{
|
||||||
|
public int eventId;
|
||||||
|
}
|
||||||
|
|
||||||
|
public static class EventIdDecoder implements Decoder.TextStream<EventId>
|
||||||
|
{
|
||||||
|
@Override
|
||||||
|
public void init(EndpointConfig config)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void destroy()
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public EventId decode(Reader reader) throws DecodeException, IOException
|
||||||
|
{
|
||||||
|
EventId id = new EventId();
|
||||||
|
try (BufferedReader buf = new BufferedReader(reader))
|
||||||
|
{
|
||||||
|
String line;
|
||||||
|
while ((line = buf.readLine()) != null)
|
||||||
|
{
|
||||||
|
id.eventId = Integer.parseInt(line);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return id;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@ClientEndpoint(decoders = EventIdDecoder.class, subprotocols = "eventids")
|
||||||
|
public static class EventIdSocket extends AbstractJsrTrackingEndpoint
|
||||||
|
{
|
||||||
|
public BlockingQueue<EventId> messageQueue = new LinkedBlockingDeque<>();
|
||||||
|
|
||||||
|
public EventIdSocket(String id)
|
||||||
|
{
|
||||||
|
super(id);
|
||||||
|
}
|
||||||
|
|
||||||
|
@SuppressWarnings("unused")
|
||||||
|
@OnMessage
|
||||||
|
public void onMessage(EventId msg)
|
||||||
|
{
|
||||||
|
messageQueue.offer(msg);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public static class EventIdServerCreator implements WebSocketCreator
|
||||||
|
{
|
||||||
|
@Override
|
||||||
|
public Object createWebSocket(ServletUpgradeRequest req, ServletUpgradeResponse resp)
|
||||||
|
{
|
||||||
|
EventIdServerEndpoint endpoint = new EventIdServerEndpoint(WebSocketBehavior.SERVER.name());
|
||||||
|
resp.setAcceptedSubProtocol("eventids");
|
||||||
|
return endpoint;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private static class EventIdServerEndpoint extends UntrustedWSEndpoint
|
||||||
|
{
|
||||||
|
public EventIdServerEndpoint(String id)
|
||||||
|
{
|
||||||
|
super(id);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onWebSocketText(String text)
|
||||||
|
{
|
||||||
|
super.onWebSocketText(text);
|
||||||
|
|
||||||
|
if (text.startsWith("seq|"))
|
||||||
|
{
|
||||||
|
String parts[] = text.split("\\|");
|
||||||
|
int from = Integer.parseInt(parts[1]);
|
||||||
|
int to = Integer.parseInt(parts[2]);
|
||||||
|
|
||||||
|
session.getRemote().setBatchMode(BatchMode.OFF);
|
||||||
|
|
||||||
|
for (int id = from; id < to; id++)
|
||||||
|
{
|
||||||
|
session.getRemote().sendStringByFuture(Integer.toString(id));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Rule
|
||||||
|
public TestName testname = new TestName();
|
||||||
|
|
||||||
|
private UntrustedWSServer server;
|
||||||
|
private WebSocketContainer client;
|
||||||
|
|
||||||
|
@Before
|
||||||
|
public void initClient()
|
||||||
|
{
|
||||||
|
client = ContainerProvider.getWebSocketContainer();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Before
|
||||||
|
public void startServer() throws Exception
|
||||||
|
{
|
||||||
|
server = new UntrustedWSServer();
|
||||||
|
server.start();
|
||||||
|
}
|
||||||
|
|
||||||
|
@After
|
||||||
|
public void stopServer() throws Exception
|
||||||
|
{
|
||||||
|
server.stop();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testManyIds() throws Exception
|
||||||
|
{
|
||||||
|
server.registerWebSocket("/eventids", new EventIdServerCreator());
|
||||||
|
|
||||||
|
URI wsUri = server.getWsUri().resolve("/eventids");
|
||||||
|
EventIdSocket clientSocket = new EventIdSocket(testname.getMethodName());
|
||||||
|
Session clientSession = client.connectToServer(clientSocket, wsUri);
|
||||||
|
|
||||||
|
final int from = 1000;
|
||||||
|
final int to = 2000;
|
||||||
|
|
||||||
|
clientSession.getAsyncRemote().sendText("seq|" + from + "|" + to);
|
||||||
|
|
||||||
|
// collect seen ids
|
||||||
|
List<Integer> seen = new ArrayList<>();
|
||||||
|
for (int i = from; i < to; i++)
|
||||||
|
{
|
||||||
|
// validate that ids don't repeat.
|
||||||
|
EventId receivedId = clientSocket.messageQueue.poll(5, TimeUnit.SECONDS);
|
||||||
|
Assert.assertFalse("Already saw ID: " + receivedId.eventId, seen.contains(receivedId.eventId));
|
||||||
|
seen.add(receivedId.eventId);
|
||||||
|
}
|
||||||
|
|
||||||
|
// validate that all expected ids have been seen (order is irrelevant here)
|
||||||
|
for (int expected = from; expected < to; expected++)
|
||||||
|
{
|
||||||
|
Assert.assertTrue("Has expected id:" + expected, seen.contains(expected));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,48 @@
|
||||||
|
//
|
||||||
|
// ========================================================================
|
||||||
|
// Copyright (c) 1995-2017 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.tests.client.jsr356;
|
||||||
|
|
||||||
|
import java.util.ArrayList;
|
||||||
|
import java.util.List;
|
||||||
|
|
||||||
|
public class Quotes
|
||||||
|
{
|
||||||
|
private String author;
|
||||||
|
private List<String> quotes = new ArrayList<>();
|
||||||
|
|
||||||
|
public void addQuote(String quote)
|
||||||
|
{
|
||||||
|
quotes.add(quote);
|
||||||
|
}
|
||||||
|
|
||||||
|
public String getAuthor()
|
||||||
|
{
|
||||||
|
return author;
|
||||||
|
}
|
||||||
|
|
||||||
|
public List<String> getQuotes()
|
||||||
|
{
|
||||||
|
return quotes;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setAuthor(String author)
|
||||||
|
{
|
||||||
|
this.author = author;
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,73 @@
|
||||||
|
//
|
||||||
|
// ========================================================================
|
||||||
|
// Copyright (c) 1995-2017 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.tests.client.jsr356;
|
||||||
|
|
||||||
|
import java.io.BufferedReader;
|
||||||
|
import java.io.IOException;
|
||||||
|
import java.io.Reader;
|
||||||
|
|
||||||
|
import javax.websocket.DecodeException;
|
||||||
|
import javax.websocket.Decoder;
|
||||||
|
import javax.websocket.EndpointConfig;
|
||||||
|
|
||||||
|
import org.eclipse.jetty.util.log.Log;
|
||||||
|
import org.eclipse.jetty.util.log.Logger;
|
||||||
|
|
||||||
|
public class QuotesDecoder implements Decoder.TextStream<Quotes>
|
||||||
|
{
|
||||||
|
private static final Logger LOG = Log.getLogger(QuotesDecoder.class);
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void init(EndpointConfig config)
|
||||||
|
{
|
||||||
|
// TODO: verify init called
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void destroy()
|
||||||
|
{
|
||||||
|
// TODO: verify destroy called
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Quotes decode(Reader reader) throws DecodeException, IOException
|
||||||
|
{
|
||||||
|
Quotes quotes = new Quotes();
|
||||||
|
try (BufferedReader buf = new BufferedReader(reader))
|
||||||
|
{
|
||||||
|
LOG.debug("decode() begin");
|
||||||
|
String line;
|
||||||
|
while ((line = buf.readLine()) != null)
|
||||||
|
{
|
||||||
|
LOG.debug("decode() line = {}", line);
|
||||||
|
switch (line.charAt(0))
|
||||||
|
{
|
||||||
|
case 'a':
|
||||||
|
quotes.setAuthor(line.substring(2));
|
||||||
|
break;
|
||||||
|
case 'q':
|
||||||
|
quotes.addQuote(line.substring(2));
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
LOG.debug("decode() complete");
|
||||||
|
}
|
||||||
|
return quotes;
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,215 @@
|
||||||
|
//
|
||||||
|
// ========================================================================
|
||||||
|
// Copyright (c) 1995-2017 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.tests.client.jsr356;
|
||||||
|
|
||||||
|
import static org.hamcrest.Matchers.is;
|
||||||
|
import static org.junit.Assert.assertThat;
|
||||||
|
|
||||||
|
import java.io.BufferedReader;
|
||||||
|
import java.io.File;
|
||||||
|
import java.io.FileReader;
|
||||||
|
import java.net.URI;
|
||||||
|
import java.util.ArrayList;
|
||||||
|
import java.util.List;
|
||||||
|
import java.util.concurrent.BlockingQueue;
|
||||||
|
import java.util.concurrent.LinkedBlockingDeque;
|
||||||
|
import java.util.concurrent.TimeUnit;
|
||||||
|
|
||||||
|
import javax.websocket.ClientEndpoint;
|
||||||
|
import javax.websocket.ContainerProvider;
|
||||||
|
import javax.websocket.OnMessage;
|
||||||
|
import javax.websocket.Session;
|
||||||
|
import javax.websocket.WebSocketContainer;
|
||||||
|
|
||||||
|
import org.eclipse.jetty.toolchain.test.MavenTestingUtils;
|
||||||
|
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.WebSocketBehavior;
|
||||||
|
import org.eclipse.jetty.websocket.common.WebSocketFrame;
|
||||||
|
import org.eclipse.jetty.websocket.common.frames.ContinuationFrame;
|
||||||
|
import org.eclipse.jetty.websocket.common.frames.TextFrame;
|
||||||
|
import org.eclipse.jetty.websocket.servlet.ServletUpgradeRequest;
|
||||||
|
import org.eclipse.jetty.websocket.servlet.ServletUpgradeResponse;
|
||||||
|
import org.eclipse.jetty.websocket.servlet.WebSocketCreator;
|
||||||
|
import org.eclipse.jetty.websocket.tests.AbstractJsrTrackingEndpoint;
|
||||||
|
import org.eclipse.jetty.websocket.tests.UntrustedWSConnection;
|
||||||
|
import org.eclipse.jetty.websocket.tests.UntrustedWSEndpoint;
|
||||||
|
import org.eclipse.jetty.websocket.tests.UntrustedWSServer;
|
||||||
|
import org.eclipse.jetty.websocket.tests.UntrustedWSSession;
|
||||||
|
import org.junit.After;
|
||||||
|
import org.junit.Before;
|
||||||
|
import org.junit.Rule;
|
||||||
|
import org.junit.Test;
|
||||||
|
import org.junit.rules.TestName;
|
||||||
|
|
||||||
|
public class QuotesDecoderTest
|
||||||
|
{
|
||||||
|
@ClientEndpoint(decoders = QuotesDecoder.class, subprotocols = "quotes")
|
||||||
|
public static class QuotesSocket extends AbstractJsrTrackingEndpoint
|
||||||
|
{
|
||||||
|
public BlockingQueue<Quotes> messageQueue = new LinkedBlockingDeque<>();
|
||||||
|
|
||||||
|
public QuotesSocket(String id)
|
||||||
|
{
|
||||||
|
super(id);
|
||||||
|
}
|
||||||
|
|
||||||
|
@SuppressWarnings("unused")
|
||||||
|
@OnMessage
|
||||||
|
public void onMessage(Quotes quote)
|
||||||
|
{
|
||||||
|
System.err.printf("QuotesSocket.onMessage(%s)%n",quote);
|
||||||
|
messageQueue.offer(quote);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public static class QuoteServingCreator implements WebSocketCreator
|
||||||
|
{
|
||||||
|
@Override
|
||||||
|
public Object createWebSocket(ServletUpgradeRequest req, ServletUpgradeResponse resp)
|
||||||
|
{
|
||||||
|
QuoterServerEndpoint endpoint = new QuoterServerEndpoint(WebSocketBehavior.SERVER.name());
|
||||||
|
resp.setAcceptedSubProtocol("quotes");
|
||||||
|
return endpoint;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public static class QuoterServerEndpoint extends UntrustedWSEndpoint
|
||||||
|
{
|
||||||
|
public QuoterServerEndpoint(String id)
|
||||||
|
{
|
||||||
|
super(id);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onWebSocketText(String filename)
|
||||||
|
{
|
||||||
|
super.onWebSocketText(filename);
|
||||||
|
try
|
||||||
|
{
|
||||||
|
UntrustedWSSession untrustedWSSession = (UntrustedWSSession) session;
|
||||||
|
UntrustedWSConnection untrustedWSConnection = untrustedWSSession.getUntrustedConnection();
|
||||||
|
writeQuotes(filename, untrustedWSConnection);
|
||||||
|
}
|
||||||
|
catch (Exception e)
|
||||||
|
{
|
||||||
|
e.printStackTrace();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public void writeQuotes(String filename, UntrustedWSConnection connection) throws Exception
|
||||||
|
{
|
||||||
|
// read file
|
||||||
|
File qfile = MavenTestingUtils.getTestResourceFile(filename);
|
||||||
|
List<String> lines = new ArrayList<>();
|
||||||
|
try (FileReader reader = new FileReader(qfile); BufferedReader buf = new BufferedReader(reader))
|
||||||
|
{
|
||||||
|
String line;
|
||||||
|
while ((line = buf.readLine()) != null)
|
||||||
|
{
|
||||||
|
lines.add(line);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// write file out, each line on a separate frame, but as
|
||||||
|
// 1 whole message
|
||||||
|
for (int i = 0; i < lines.size(); i++)
|
||||||
|
{
|
||||||
|
WebSocketFrame frame;
|
||||||
|
if (i == 0)
|
||||||
|
{
|
||||||
|
frame = new TextFrame();
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
frame = new ContinuationFrame();
|
||||||
|
}
|
||||||
|
frame.setFin((i >= (lines.size() - 1)));
|
||||||
|
frame.setPayload(BufferUtil.toBuffer(lines.get(i) + "\n"));
|
||||||
|
connection.write(frame);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private static final Logger LOG = Log.getLogger(QuotesDecoderTest.class);
|
||||||
|
|
||||||
|
@Rule
|
||||||
|
public TestName testname = new TestName();
|
||||||
|
|
||||||
|
private UntrustedWSServer server;
|
||||||
|
private WebSocketContainer client;
|
||||||
|
|
||||||
|
@Before
|
||||||
|
public void initClient()
|
||||||
|
{
|
||||||
|
client = ContainerProvider.getWebSocketContainer();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Before
|
||||||
|
public void startServer() throws Exception
|
||||||
|
{
|
||||||
|
server = new UntrustedWSServer();
|
||||||
|
server.start();
|
||||||
|
}
|
||||||
|
|
||||||
|
@After
|
||||||
|
public void stopServer() throws Exception
|
||||||
|
{
|
||||||
|
server.stop();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testSingleQuotes() throws Exception
|
||||||
|
{
|
||||||
|
server.registerWebSocket("/quoter", new QuoteServingCreator());
|
||||||
|
|
||||||
|
URI wsUri = server.getWsUri().resolve("/quoter");
|
||||||
|
QuotesSocket clientSocket = new QuotesSocket(testname.getMethodName());
|
||||||
|
Session clientSession = client.connectToServer(clientSocket, wsUri);
|
||||||
|
|
||||||
|
clientSession.getAsyncRemote().sendText("quotes-ben.txt");
|
||||||
|
|
||||||
|
Quotes quotes = clientSocket.messageQueue.poll(5, TimeUnit.SECONDS);
|
||||||
|
|
||||||
|
assertThat("Quotes Author", quotes.getAuthor(), is("Benjamin Franklin"));
|
||||||
|
assertThat("Quotes Count", quotes.getQuotes().size(), is(3));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testTwoQuotes() throws Exception
|
||||||
|
{
|
||||||
|
server.registerWebSocket("/quoter", new QuoteServingCreator());
|
||||||
|
|
||||||
|
URI wsUri = server.getWsUri().resolve("/quoter");
|
||||||
|
QuotesSocket clientSocket = new QuotesSocket(testname.getMethodName());
|
||||||
|
Session clientSession = client.connectToServer(clientSocket, wsUri);
|
||||||
|
|
||||||
|
clientSession.getAsyncRemote().sendText("quotes-ben.txt");
|
||||||
|
clientSession.getAsyncRemote().sendText("quotes-twain.txt");
|
||||||
|
|
||||||
|
Quotes quotes;
|
||||||
|
quotes = clientSocket.messageQueue.poll(5, TimeUnit.SECONDS);
|
||||||
|
assertThat("Quotes Author", quotes.getAuthor(), is("Benjamin Franklin"));
|
||||||
|
assertThat("Quotes Count", quotes.getQuotes().size(), is(3));
|
||||||
|
|
||||||
|
quotes = clientSocket.messageQueue.poll(5, TimeUnit.SECONDS);
|
||||||
|
assertThat("Quotes Author", quotes.getAuthor(), is("Mark Twain"));
|
||||||
|
assertThat("Quotes Count", quotes.getQuotes().size(), is(4));
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,52 @@
|
||||||
|
//
|
||||||
|
// ========================================================================
|
||||||
|
// Copyright (c) 1995-2017 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.tests.client.jsr356;
|
||||||
|
|
||||||
|
import javax.websocket.EncodeException;
|
||||||
|
import javax.websocket.Encoder;
|
||||||
|
import javax.websocket.EndpointConfig;
|
||||||
|
|
||||||
|
public class QuotesEncoder implements Encoder.Text<Quotes>
|
||||||
|
{
|
||||||
|
@Override
|
||||||
|
public void destroy()
|
||||||
|
{
|
||||||
|
// TODO: verify destroy called
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public String encode(Quotes q) throws EncodeException
|
||||||
|
{
|
||||||
|
StringBuilder buf = new StringBuilder();
|
||||||
|
buf.append("Author: ").append(q.getAuthor());
|
||||||
|
buf.append(System.lineSeparator());
|
||||||
|
for (String quote : q.getQuotes())
|
||||||
|
{
|
||||||
|
buf.append("Quote: ").append(quote);
|
||||||
|
buf.append(System.lineSeparator());
|
||||||
|
}
|
||||||
|
return buf.toString();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void init(EndpointConfig config)
|
||||||
|
{
|
||||||
|
// TODO: verify init called
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,197 @@
|
||||||
|
//
|
||||||
|
// ========================================================================
|
||||||
|
// Copyright (c) 1995-2017 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.tests.client.jsr356;
|
||||||
|
|
||||||
|
import static org.hamcrest.Matchers.containsString;
|
||||||
|
|
||||||
|
import java.io.BufferedReader;
|
||||||
|
import java.io.File;
|
||||||
|
import java.io.FileReader;
|
||||||
|
import java.io.IOException;
|
||||||
|
import java.net.URI;
|
||||||
|
import java.util.concurrent.BlockingQueue;
|
||||||
|
import java.util.concurrent.LinkedBlockingDeque;
|
||||||
|
import java.util.concurrent.TimeUnit;
|
||||||
|
|
||||||
|
import javax.websocket.ClientEndpoint;
|
||||||
|
import javax.websocket.ContainerProvider;
|
||||||
|
import javax.websocket.EncodeException;
|
||||||
|
import javax.websocket.OnMessage;
|
||||||
|
import javax.websocket.Session;
|
||||||
|
import javax.websocket.WebSocketContainer;
|
||||||
|
|
||||||
|
import org.eclipse.jetty.toolchain.test.MavenTestingUtils;
|
||||||
|
import org.eclipse.jetty.websocket.tests.AbstractJsrTrackingEndpoint;
|
||||||
|
import org.eclipse.jetty.websocket.tests.UntrustedWSServer;
|
||||||
|
import org.junit.After;
|
||||||
|
import org.junit.Assert;
|
||||||
|
import org.junit.Before;
|
||||||
|
import org.junit.Rule;
|
||||||
|
import org.junit.Test;
|
||||||
|
import org.junit.rules.TestName;
|
||||||
|
|
||||||
|
public class QuotesEncoderTest
|
||||||
|
{
|
||||||
|
@ClientEndpoint(encoders = QuotesEncoder.class, subprotocols = "echo")
|
||||||
|
public static class QuotesSocket extends AbstractJsrTrackingEndpoint
|
||||||
|
{
|
||||||
|
public BlockingQueue<String> messageQueue = new LinkedBlockingDeque<>();
|
||||||
|
|
||||||
|
public QuotesSocket(String id)
|
||||||
|
{
|
||||||
|
super(id);
|
||||||
|
}
|
||||||
|
|
||||||
|
@SuppressWarnings("unused")
|
||||||
|
@OnMessage
|
||||||
|
public void onMessage(String message)
|
||||||
|
{
|
||||||
|
messageQueue.offer(message);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void write(Quotes quotes) throws IOException, EncodeException
|
||||||
|
{
|
||||||
|
if (LOG.isDebugEnabled())
|
||||||
|
LOG.debug("Writing Quotes: {}", quotes);
|
||||||
|
this.session.getBasicRemote().sendObject(quotes);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Rule
|
||||||
|
public TestName testname = new TestName();
|
||||||
|
|
||||||
|
private UntrustedWSServer server;
|
||||||
|
private WebSocketContainer client;
|
||||||
|
|
||||||
|
@Before
|
||||||
|
public void initClient()
|
||||||
|
{
|
||||||
|
client = ContainerProvider.getWebSocketContainer();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Before
|
||||||
|
public void startServer() throws Exception
|
||||||
|
{
|
||||||
|
server = new UntrustedWSServer();
|
||||||
|
server.start();
|
||||||
|
}
|
||||||
|
|
||||||
|
@After
|
||||||
|
public void stopServer() throws Exception
|
||||||
|
{
|
||||||
|
server.stop();
|
||||||
|
}
|
||||||
|
|
||||||
|
private void assertReceivedQuotes(String result, Quotes quotes)
|
||||||
|
{
|
||||||
|
Assert.assertThat("Quote Author", result, containsString("Author: " + quotes.getAuthor()));
|
||||||
|
for (String quote : quotes.getQuotes())
|
||||||
|
{
|
||||||
|
Assert.assertThat("Quote", result, containsString("Quote: " + quote));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@SuppressWarnings("Duplicates")
|
||||||
|
private Quotes getQuotes(String filename) throws IOException
|
||||||
|
{
|
||||||
|
Quotes quotes = new Quotes();
|
||||||
|
|
||||||
|
// read file
|
||||||
|
File qfile = MavenTestingUtils.getTestResourceFile(filename);
|
||||||
|
try (FileReader reader = new FileReader(qfile); BufferedReader buf = new BufferedReader(reader))
|
||||||
|
{
|
||||||
|
String line;
|
||||||
|
while ((line = buf.readLine()) != null)
|
||||||
|
{
|
||||||
|
switch (line.charAt(0))
|
||||||
|
{
|
||||||
|
case 'a':
|
||||||
|
quotes.setAuthor(line.substring(2));
|
||||||
|
break;
|
||||||
|
case 'q':
|
||||||
|
quotes.addQuote(line.substring(2));
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return quotes;
|
||||||
|
}
|
||||||
|
|
||||||
|
private void close(Session session) throws IOException
|
||||||
|
{
|
||||||
|
if (session != null)
|
||||||
|
{
|
||||||
|
session.close();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testSingleQuotes() throws Exception
|
||||||
|
{
|
||||||
|
URI wsUri = server.getUntrustedWsUri(this.getClass(), testname);
|
||||||
|
QuotesSocket quoter = new QuotesSocket(testname.getMethodName());
|
||||||
|
|
||||||
|
Session session = null;
|
||||||
|
try
|
||||||
|
{
|
||||||
|
session = client.connectToServer(quoter, wsUri);
|
||||||
|
|
||||||
|
Quotes ben = getQuotes("quotes-ben.txt");
|
||||||
|
quoter.write(ben);
|
||||||
|
|
||||||
|
String incomingMessage = quoter.messageQueue.poll(5, TimeUnit.SECONDS);
|
||||||
|
assertReceivedQuotes(incomingMessage, ben);
|
||||||
|
}
|
||||||
|
finally
|
||||||
|
{
|
||||||
|
close(session);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testTwoQuotes() throws Exception
|
||||||
|
{
|
||||||
|
URI wsUri = server.getUntrustedWsUri(this.getClass(), testname);
|
||||||
|
QuotesSocket quoter = new QuotesSocket(testname.getMethodName());
|
||||||
|
|
||||||
|
Session session = null;
|
||||||
|
try
|
||||||
|
{
|
||||||
|
session = client.connectToServer(quoter, wsUri);
|
||||||
|
|
||||||
|
Quotes ben = getQuotes("quotes-ben.txt");
|
||||||
|
Quotes twain = getQuotes("quotes-twain.txt");
|
||||||
|
quoter.write(ben);
|
||||||
|
quoter.write(twain);
|
||||||
|
|
||||||
|
String incomingQuote;
|
||||||
|
|
||||||
|
incomingQuote = quoter.messageQueue.poll(5, TimeUnit.SECONDS);
|
||||||
|
assertReceivedQuotes(incomingQuote, ben);
|
||||||
|
|
||||||
|
incomingQuote = quoter.messageQueue.poll(5, TimeUnit.SECONDS);
|
||||||
|
assertReceivedQuotes(incomingQuote, twain);
|
||||||
|
}
|
||||||
|
finally
|
||||||
|
{
|
||||||
|
close(session);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -16,10 +16,10 @@
|
||||||
// ========================================================================
|
// ========================================================================
|
||||||
//
|
//
|
||||||
|
|
||||||
package org.eclipse.jetty.websocket.jsr356.server;
|
package org.eclipse.jetty.websocket.tests.server.jsr356;
|
||||||
|
|
||||||
import static org.hamcrest.Matchers.is;
|
import static org.hamcrest.Matchers.is;
|
||||||
import static org.hamcrest.Matchers.nullValue;
|
import static org.hamcrest.Matchers.notNullValue;
|
||||||
import static org.junit.Assert.assertThat;
|
import static org.junit.Assert.assertThat;
|
||||||
|
|
||||||
import java.io.PrintWriter;
|
import java.io.PrintWriter;
|
||||||
|
@ -35,6 +35,7 @@ import java.util.List;
|
||||||
import java.util.Locale;
|
import java.util.Locale;
|
||||||
import java.util.Map;
|
import java.util.Map;
|
||||||
import java.util.TimeZone;
|
import java.util.TimeZone;
|
||||||
|
import java.util.concurrent.Future;
|
||||||
import java.util.concurrent.TimeUnit;
|
import java.util.concurrent.TimeUnit;
|
||||||
import java.util.concurrent.atomic.AtomicInteger;
|
import java.util.concurrent.atomic.AtomicInteger;
|
||||||
import java.util.concurrent.atomic.AtomicReference;
|
import java.util.concurrent.atomic.AtomicReference;
|
||||||
|
@ -54,20 +55,25 @@ import javax.websocket.server.ServerEndpointConfig;
|
||||||
import org.eclipse.jetty.server.Server;
|
import org.eclipse.jetty.server.Server;
|
||||||
import org.eclipse.jetty.server.ServerConnector;
|
import org.eclipse.jetty.server.ServerConnector;
|
||||||
import org.eclipse.jetty.servlet.ServletContextHandler;
|
import org.eclipse.jetty.servlet.ServletContextHandler;
|
||||||
import org.eclipse.jetty.toolchain.test.EventQueue;
|
|
||||||
import org.eclipse.jetty.util.log.Log;
|
import org.eclipse.jetty.util.log.Log;
|
||||||
import org.eclipse.jetty.util.log.Logger;
|
import org.eclipse.jetty.util.log.Logger;
|
||||||
|
import org.eclipse.jetty.websocket.api.extensions.ExtensionConfig;
|
||||||
import org.eclipse.jetty.websocket.api.util.QuoteUtil;
|
import org.eclipse.jetty.websocket.api.util.QuoteUtil;
|
||||||
import org.eclipse.jetty.websocket.common.WebSocketFrame;
|
import org.eclipse.jetty.websocket.api.util.WSURI;
|
||||||
import org.eclipse.jetty.websocket.common.frames.TextFrame;
|
import org.eclipse.jetty.websocket.client.ClientUpgradeRequest;
|
||||||
import org.eclipse.jetty.websocket.common.test.XBlockheadClient;
|
import org.eclipse.jetty.websocket.client.WebSocketClient;
|
||||||
import org.eclipse.jetty.websocket.common.test.HttpResponse;
|
import org.eclipse.jetty.websocket.jsr356.server.JsrCreator;
|
||||||
import org.eclipse.jetty.websocket.common.test.IBlockheadClient;
|
import org.eclipse.jetty.websocket.jsr356.server.ServerContainer;
|
||||||
import org.eclipse.jetty.websocket.jsr356.server.deploy.WebSocketServerContainerInitializer;
|
import org.eclipse.jetty.websocket.jsr356.server.deploy.WebSocketServerContainerInitializer;
|
||||||
|
import org.eclipse.jetty.websocket.tests.Defaults;
|
||||||
|
import org.eclipse.jetty.websocket.tests.TrackingEndpoint;
|
||||||
|
import org.junit.After;
|
||||||
import org.junit.AfterClass;
|
import org.junit.AfterClass;
|
||||||
import org.junit.Assert;
|
import org.junit.Before;
|
||||||
import org.junit.BeforeClass;
|
import org.junit.BeforeClass;
|
||||||
|
import org.junit.Rule;
|
||||||
import org.junit.Test;
|
import org.junit.Test;
|
||||||
|
import org.junit.rules.TestName;
|
||||||
|
|
||||||
public class ConfiguratorTest
|
public class ConfiguratorTest
|
||||||
{
|
{
|
||||||
|
@ -382,13 +388,7 @@ public class ConfiguratorTest
|
||||||
container.addEndpoint(TimeDecoderSocket.class);
|
container.addEndpoint(TimeDecoderSocket.class);
|
||||||
|
|
||||||
server.start();
|
server.start();
|
||||||
String host = connector.getHost();
|
baseServerUri = WSURI.toWebsocket(server.getURI()).resolve("/");
|
||||||
if (host == null)
|
|
||||||
{
|
|
||||||
host = "localhost";
|
|
||||||
}
|
|
||||||
int port = connector.getLocalPort();
|
|
||||||
baseServerUri = new URI(String.format("ws://%s:%d/", host, port));
|
|
||||||
if (LOG.isDebugEnabled())
|
if (LOG.isDebugEnabled())
|
||||||
LOG.debug("Server started on {}", baseServerUri);
|
LOG.debug("Server started on {}", baseServerUri);
|
||||||
}
|
}
|
||||||
|
@ -408,115 +408,132 @@ public class ConfiguratorTest
|
||||||
server.stop();
|
server.stop();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Rule
|
||||||
|
public TestName testname = new TestName();
|
||||||
|
|
||||||
|
private WebSocketClient client;
|
||||||
|
|
||||||
|
@Before
|
||||||
|
public void startClient() throws Exception
|
||||||
|
{
|
||||||
|
client = new WebSocketClient();
|
||||||
|
client.start();
|
||||||
|
}
|
||||||
|
|
||||||
|
@After
|
||||||
|
public void stopClient() throws Exception
|
||||||
|
{
|
||||||
|
client.stop();
|
||||||
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
public void testEmptyConfigurator() throws Exception
|
public void testEmptyConfigurator() throws Exception
|
||||||
{
|
{
|
||||||
URI uri = baseServerUri.resolve("/empty");
|
URI wsUri = baseServerUri.resolve("/empty");
|
||||||
|
|
||||||
try (IBlockheadClient client = new XBlockheadClient(uri))
|
TrackingEndpoint clientSocket = new TrackingEndpoint(testname.getMethodName());
|
||||||
{
|
ClientUpgradeRequest upgradeRequest = new ClientUpgradeRequest();
|
||||||
client.addExtensions("identity");
|
upgradeRequest.addExtensions("identity");
|
||||||
client.connect();
|
Future<org.eclipse.jetty.websocket.api.Session> clientConnectFuture = client.connect(clientSocket, wsUri, upgradeRequest);
|
||||||
client.sendStandardRequest();
|
|
||||||
HttpResponse response = client.readResponseHeader();
|
org.eclipse.jetty.websocket.api.Session clientSession = clientConnectFuture.get(Defaults.CONNECT_TIMEOUT_MS, TimeUnit.MILLISECONDS);
|
||||||
Assert.assertThat("response.extensions", response.getExtensionsHeader(), is("identity"));
|
List<ExtensionConfig> negotiatedExtensions = clientSession.getUpgradeResponse().getExtensions();
|
||||||
}
|
assertThat("UpgradeResponse.extensions", negotiatedExtensions, notNullValue());
|
||||||
|
assertThat("UpgradeResponse.extensions.size", negotiatedExtensions.size(), is(1));
|
||||||
|
assertThat("UpgradeResponse.extensions[0]", negotiatedExtensions.get(0).toString(), is("identity"));
|
||||||
|
|
||||||
|
clientSession.close();
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
public void testNoExtensionsConfigurator() throws Exception
|
public void testNoExtensionsConfigurator() throws Exception
|
||||||
{
|
{
|
||||||
URI uri = baseServerUri.resolve("/no-extensions");
|
URI wsUri = baseServerUri.resolve("/no-extensions");
|
||||||
|
|
||||||
try (IBlockheadClient client = new XBlockheadClient(uri))
|
TrackingEndpoint clientSocket = new TrackingEndpoint(testname.getMethodName());
|
||||||
{
|
ClientUpgradeRequest upgradeRequest = new ClientUpgradeRequest();
|
||||||
client.addExtensions("identity");
|
upgradeRequest.addExtensions("identity");
|
||||||
client.connect();
|
Future<org.eclipse.jetty.websocket.api.Session> clientConnectFuture = client.connect(clientSocket, wsUri, upgradeRequest);
|
||||||
client.sendStandardRequest();
|
|
||||||
HttpResponse response = client.expectUpgradeResponse();
|
|
||||||
assertThat("response.extensions", response.getExtensionsHeader(), nullValue());
|
|
||||||
|
|
||||||
client.write(new TextFrame().setPayload("NegoExts"));
|
org.eclipse.jetty.websocket.api.Session clientSession = clientConnectFuture.get(Defaults.CONNECT_TIMEOUT_MS, TimeUnit.MILLISECONDS);
|
||||||
EventQueue<WebSocketFrame> frames = client.readFrames(1, 1, TimeUnit.SECONDS);
|
clientSession.getRemote().sendString("NegoExts");
|
||||||
WebSocketFrame frame = frames.poll();
|
|
||||||
assertThat("Frame Response", frame.getPayloadAsUTF8(), is("negotiatedExtensions=[]"));
|
String incomingMessage = clientSocket.messageQueue.poll(5, TimeUnit.SECONDS);
|
||||||
}
|
assertThat("Incoming Message", incomingMessage, is("negotiatedExtensions=[]"));
|
||||||
|
|
||||||
|
clientSession.close();
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
public void testCaptureRequestHeadersConfigurator() throws Exception
|
public void testCaptureRequestHeadersConfigurator() throws Exception
|
||||||
{
|
{
|
||||||
URI uri = baseServerUri.resolve("/capture-request-headers");
|
URI wsUri = baseServerUri.resolve("/capture-request-headers");
|
||||||
|
|
||||||
try (IBlockheadClient client = new XBlockheadClient(uri))
|
TrackingEndpoint clientSocket = new TrackingEndpoint(testname.getMethodName());
|
||||||
{
|
ClientUpgradeRequest upgradeRequest = new ClientUpgradeRequest();
|
||||||
client.addHeader("X-Dummy: Bogus\r\n");
|
upgradeRequest.setHeader("X-Dummy", "Bogus");
|
||||||
client.connect();
|
Future<org.eclipse.jetty.websocket.api.Session> clientConnectFuture = client.connect(clientSocket, wsUri, upgradeRequest);
|
||||||
client.sendStandardRequest();
|
|
||||||
client.expectUpgradeResponse();
|
|
||||||
|
|
||||||
client.write(new TextFrame().setPayload("X-Dummy"));
|
org.eclipse.jetty.websocket.api.Session clientSession = clientConnectFuture.get(Defaults.CONNECT_TIMEOUT_MS, TimeUnit.MILLISECONDS);
|
||||||
EventQueue<WebSocketFrame> frames = client.readFrames(1, 1, TimeUnit.SECONDS);
|
clientSession.getRemote().sendString("X-Dummy");
|
||||||
WebSocketFrame frame = frames.poll();
|
|
||||||
Assert.assertThat("Frame Response", frame.getPayloadAsUTF8(), is("Request Header [X-Dummy]: \"Bogus\""));
|
String incomingMessage = clientSocket.messageQueue.poll(5, TimeUnit.SECONDS);
|
||||||
}
|
assertThat("Incoming Message", incomingMessage, is("Request Header [X-Dummy]: \"Bogus\""));
|
||||||
|
|
||||||
|
clientSession.close();
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
public void testUniqueUserPropsConfigurator() throws Exception
|
public void testUniqueUserPropsConfigurator() throws Exception
|
||||||
{
|
{
|
||||||
URI uri = baseServerUri.resolve("/unique-user-props");
|
URI wsUri = baseServerUri.resolve("/unique-user-props");
|
||||||
|
|
||||||
// First request
|
// First Request
|
||||||
try (IBlockheadClient client = new XBlockheadClient(uri))
|
TrackingEndpoint clientSocket = new TrackingEndpoint(testname.getMethodName());
|
||||||
{
|
ClientUpgradeRequest upgradeRequest = new ClientUpgradeRequest();
|
||||||
client.connect();
|
Future<org.eclipse.jetty.websocket.api.Session> clientConnectFuture = client.connect(clientSocket, wsUri, upgradeRequest);
|
||||||
client.sendStandardRequest();
|
|
||||||
client.expectUpgradeResponse();
|
|
||||||
|
|
||||||
client.write(new TextFrame().setPayload("apple"));
|
org.eclipse.jetty.websocket.api.Session clientSession = clientConnectFuture.get(Defaults.CONNECT_TIMEOUT_MS, TimeUnit.MILLISECONDS);
|
||||||
EventQueue<WebSocketFrame> frames = client.readFrames(1, 1, TimeUnit.SECONDS);
|
clientSession.getRemote().sendString("apple"); // first request has this UserProperty
|
||||||
WebSocketFrame frame = frames.poll();
|
|
||||||
Assert.assertThat("Frame Response", frame.getPayloadAsUTF8(), is("Requested User Property: [apple] = \"fruit from tree\""));
|
String incomingMessage = clientSocket.messageQueue.poll(5, TimeUnit.SECONDS);
|
||||||
}
|
assertThat("Incoming Message", incomingMessage, is("Requested User Property: [apple] = \"fruit from tree\""));
|
||||||
|
|
||||||
|
clientSession.close();
|
||||||
|
|
||||||
// Second request
|
// Second request
|
||||||
try (IBlockheadClient client = new XBlockheadClient(uri))
|
clientSocket = new TrackingEndpoint(testname.getMethodName());
|
||||||
{
|
upgradeRequest = new ClientUpgradeRequest();
|
||||||
client.connect();
|
clientConnectFuture = client.connect(clientSocket, wsUri, upgradeRequest);
|
||||||
client.sendStandardRequest();
|
|
||||||
client.expectUpgradeResponse();
|
|
||||||
|
|
||||||
client.write(new TextFrame().setPayload("apple"));
|
clientSession = clientConnectFuture.get(Defaults.CONNECT_TIMEOUT_MS, TimeUnit.MILLISECONDS);
|
||||||
client.write(new TextFrame().setPayload("blueberry"));
|
clientSession.getRemote().sendString("apple"); // as this is second request, this should be null
|
||||||
EventQueue<WebSocketFrame> frames = client.readFrames(2, 1, TimeUnit.SECONDS);
|
clientSession.getRemote().sendString("blueberry"); // second request has this UserProperty
|
||||||
WebSocketFrame frame = frames.poll();
|
|
||||||
// should have no value
|
|
||||||
Assert.assertThat("Frame Response", frame.getPayloadAsUTF8(), is("Requested User Property: [apple] = <null>"));
|
|
||||||
|
|
||||||
frame = frames.poll();
|
incomingMessage = clientSocket.messageQueue.poll(5, TimeUnit.SECONDS);
|
||||||
Assert.assertThat("Frame Response", frame.getPayloadAsUTF8(), is("Requested User Property: [blueberry] = \"fruit from bush\""));
|
assertThat("Incoming Message", incomingMessage, is("Requested User Property: [apple] = <null>"));
|
||||||
}
|
incomingMessage = clientSocket.messageQueue.poll(5, TimeUnit.SECONDS);
|
||||||
|
assertThat("Incoming Message", incomingMessage, is("Requested User Property: [blueberry] = \"fruit from bush\""));
|
||||||
|
|
||||||
|
clientSession.close();
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
public void testUserPropsAddress() throws Exception
|
public void testUserPropsAddress() throws Exception
|
||||||
{
|
{
|
||||||
URI uri = baseServerUri.resolve("/addr");
|
URI wsUri = baseServerUri.resolve("/addr");
|
||||||
|
|
||||||
// First request
|
TrackingEndpoint clientSocket = new TrackingEndpoint(testname.getMethodName());
|
||||||
try (IBlockheadClient client = new XBlockheadClient(uri))
|
ClientUpgradeRequest upgradeRequest = new ClientUpgradeRequest();
|
||||||
{
|
Future<org.eclipse.jetty.websocket.api.Session> clientConnectFuture = client.connect(clientSocket, wsUri, upgradeRequest);
|
||||||
client.connect();
|
|
||||||
client.sendStandardRequest();
|
|
||||||
client.expectUpgradeResponse();
|
|
||||||
|
|
||||||
InetSocketAddress expectedLocal = client.getLocalSocketAddress();
|
org.eclipse.jetty.websocket.api.Session clientSession = clientConnectFuture.get(Defaults.CONNECT_TIMEOUT_MS, TimeUnit.MILLISECONDS);
|
||||||
InetSocketAddress expectedRemote = client.getRemoteSocketAddress();
|
InetSocketAddress expectedLocal = clientSession.getLocalAddress();
|
||||||
|
InetSocketAddress expectedRemote = clientSession.getRemoteAddress();
|
||||||
|
|
||||||
client.write(new TextFrame().setPayload("addr"));
|
clientSession.getRemote().sendString("addr");
|
||||||
EventQueue<WebSocketFrame> frames = client.readFrames(1, 1, TimeUnit.SECONDS);
|
|
||||||
WebSocketFrame frame = frames.poll();
|
String incomingMessage = clientSocket.messageQueue.poll(5, TimeUnit.SECONDS);
|
||||||
|
|
||||||
StringWriter expected = new StringWriter();
|
StringWriter expected = new StringWriter();
|
||||||
PrintWriter out = new PrintWriter(expected);
|
PrintWriter out = new PrintWriter(expected);
|
||||||
|
@ -526,104 +543,110 @@ public class ConfiguratorTest
|
||||||
out.printf("[found.local] = %s%n", toSafeAddr(expectedRemote));
|
out.printf("[found.local] = %s%n", toSafeAddr(expectedRemote));
|
||||||
out.printf("[found.remote] = %s%n", toSafeAddr(expectedLocal));
|
out.printf("[found.remote] = %s%n", toSafeAddr(expectedLocal));
|
||||||
|
|
||||||
Assert.assertThat("Frame Response", frame.getPayloadAsUTF8(), is(expected.toString()));
|
assertThat("Frame Response", incomingMessage, is(expected.toString()));
|
||||||
}
|
|
||||||
|
clientSession.close();
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Test of Sec-WebSocket-Protocol, as seen in RFC-6455, 1 protocol
|
* Test of Sec-WebSocket-Protocol, as seen in RFC-6455, 1 protocol
|
||||||
|
*
|
||||||
* @throws Exception on test failure
|
* @throws Exception on test failure
|
||||||
*/
|
*/
|
||||||
@Test
|
@Test
|
||||||
public void testProtocol_Single() throws Exception
|
public void testProtocol_Single() throws Exception
|
||||||
{
|
{
|
||||||
URI uri = baseServerUri.resolve("/protocols");
|
URI wsUri = baseServerUri.resolve("/protocols");
|
||||||
ProtocolsConfigurator.seenProtocols.set(null);
|
ProtocolsConfigurator.seenProtocols.set(null);
|
||||||
|
|
||||||
try (IBlockheadClient client = new XBlockheadClient(uri))
|
TrackingEndpoint clientSocket = new TrackingEndpoint(testname.getMethodName());
|
||||||
{
|
ClientUpgradeRequest upgradeRequest = new ClientUpgradeRequest();
|
||||||
client.addHeader("Sec-WebSocket-Protocol: echo\r\n");
|
upgradeRequest.setSubProtocols("echo");
|
||||||
client.connect();
|
Future<org.eclipse.jetty.websocket.api.Session> clientConnectFuture = client.connect(clientSocket, wsUri, upgradeRequest);
|
||||||
client.sendStandardRequest();
|
|
||||||
client.expectUpgradeResponse();
|
|
||||||
|
|
||||||
client.write(new TextFrame().setPayload("getProtocols"));
|
org.eclipse.jetty.websocket.api.Session clientSession = clientConnectFuture.get(Defaults.CONNECT_TIMEOUT_MS, TimeUnit.MILLISECONDS);
|
||||||
EventQueue<WebSocketFrame> frames = client.readFrames(1, 1, TimeUnit.SECONDS);
|
clientSession.getRemote().sendString("getProtocols");
|
||||||
WebSocketFrame frame = frames.poll();
|
|
||||||
Assert.assertThat("Frame Response", frame.getPayloadAsUTF8(), is("Requested Protocols: [\"echo\"]"));
|
String incomingMessage = clientSocket.messageQueue.poll(5, TimeUnit.SECONDS);
|
||||||
}
|
assertThat("Incoming message", incomingMessage, is("Requested Protocols: [\"echo\"]"));
|
||||||
|
|
||||||
|
clientSession.close();
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Test of Sec-WebSocket-Protocol, as seen in RFC-6455, 3 protocols
|
* Test of Sec-WebSocket-Protocol, as seen in RFC-6455, 3 protocols
|
||||||
|
*
|
||||||
* @throws Exception on test failure
|
* @throws Exception on test failure
|
||||||
*/
|
*/
|
||||||
@Test
|
@Test
|
||||||
public void testProtocol_Triple() throws Exception
|
public void testProtocol_Triple() throws Exception
|
||||||
{
|
{
|
||||||
URI uri = baseServerUri.resolve("/protocols");
|
URI wsUri = baseServerUri.resolve("/protocols");
|
||||||
ProtocolsConfigurator.seenProtocols.set(null);
|
ProtocolsConfigurator.seenProtocols.set(null);
|
||||||
|
|
||||||
try (IBlockheadClient client = new XBlockheadClient(uri))
|
TrackingEndpoint clientSocket = new TrackingEndpoint(testname.getMethodName());
|
||||||
{
|
ClientUpgradeRequest upgradeRequest = new ClientUpgradeRequest();
|
||||||
client.addHeader("Sec-WebSocket-Protocol: echo, chat, status\r\n");
|
upgradeRequest.setSubProtocols("echo", "chat", "status");
|
||||||
client.connect();
|
Future<org.eclipse.jetty.websocket.api.Session> clientConnectFuture = client.connect(clientSocket, wsUri, upgradeRequest);
|
||||||
client.sendStandardRequest();
|
|
||||||
client.expectUpgradeResponse();
|
|
||||||
|
|
||||||
client.write(new TextFrame().setPayload("getProtocols"));
|
org.eclipse.jetty.websocket.api.Session clientSession = clientConnectFuture.get(Defaults.CONNECT_TIMEOUT_MS, TimeUnit.MILLISECONDS);
|
||||||
EventQueue<WebSocketFrame> frames = client.readFrames(1, 1, TimeUnit.SECONDS);
|
clientSession.getRemote().sendString("getProtocols");
|
||||||
WebSocketFrame frame = frames.poll();
|
|
||||||
Assert.assertThat("Frame Response", frame.getPayloadAsUTF8(), is("Requested Protocols: [\"echo\",\"chat\",\"status\"]"));
|
String incomingMessage = clientSocket.messageQueue.poll(5, TimeUnit.SECONDS);
|
||||||
}
|
assertThat("Incoming message", incomingMessage, is("Requested Protocols: [\"echo\",\"chat\",\"status\"]"));
|
||||||
|
|
||||||
|
clientSession.close();
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Test of Sec-WebSocket-Protocol, using all lowercase header
|
* Test of Sec-WebSocket-Protocol, using all lowercase header
|
||||||
|
*
|
||||||
* @throws Exception on test failure
|
* @throws Exception on test failure
|
||||||
*/
|
*/
|
||||||
@Test
|
@Test
|
||||||
public void testProtocol_LowercaseHeader() throws Exception
|
public void testProtocol_LowercaseHeader() throws Exception
|
||||||
{
|
{
|
||||||
URI uri = baseServerUri.resolve("/protocols");
|
URI wsUri = baseServerUri.resolve("/protocols");
|
||||||
ProtocolsConfigurator.seenProtocols.set(null);
|
ProtocolsConfigurator.seenProtocols.set(null);
|
||||||
|
|
||||||
try (IBlockheadClient client = new XBlockheadClient(uri))
|
TrackingEndpoint clientSocket = new TrackingEndpoint(testname.getMethodName());
|
||||||
{
|
ClientUpgradeRequest upgradeRequest = new ClientUpgradeRequest();
|
||||||
client.addHeader("sec-websocket-protocol: echo, chat, status\r\n");
|
upgradeRequest.setHeader("sec-websocket-protocol", "echo, chat, status");
|
||||||
client.connect();
|
Future<org.eclipse.jetty.websocket.api.Session> clientConnectFuture = client.connect(clientSocket, wsUri, upgradeRequest);
|
||||||
client.sendStandardRequest();
|
|
||||||
client.expectUpgradeResponse();
|
|
||||||
|
|
||||||
client.write(new TextFrame().setPayload("getProtocols"));
|
org.eclipse.jetty.websocket.api.Session clientSession = clientConnectFuture.get(Defaults.CONNECT_TIMEOUT_MS, TimeUnit.MILLISECONDS);
|
||||||
EventQueue<WebSocketFrame> frames = client.readFrames(1, 1, TimeUnit.SECONDS);
|
clientSession.getRemote().sendString("getProtocols");
|
||||||
WebSocketFrame frame = frames.poll();
|
|
||||||
Assert.assertThat("Frame Response", frame.getPayloadAsUTF8(), is("Requested Protocols: [\"echo\",\"chat\",\"status\"]"));
|
String incomingMessage = clientSocket.messageQueue.poll(5, TimeUnit.SECONDS);
|
||||||
}
|
assertThat("Incoming message", incomingMessage, is("Requested Protocols: [\"echo\",\"chat\",\"status\"]"));
|
||||||
|
|
||||||
|
clientSession.close();
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Test of Sec-WebSocket-Protocol, using non-spec case header
|
* Test of Sec-WebSocket-Protocol, using non-spec case header
|
||||||
|
*
|
||||||
* @throws Exception on test failure
|
* @throws Exception on test failure
|
||||||
*/
|
*/
|
||||||
@Test
|
@Test
|
||||||
public void testProtocol_AltHeaderCase() throws Exception
|
public void testProtocol_AltHeaderCase() throws Exception
|
||||||
{
|
{
|
||||||
URI uri = baseServerUri.resolve("/protocols");
|
URI wsUri = baseServerUri.resolve("/protocols");
|
||||||
ProtocolsConfigurator.seenProtocols.set(null);
|
ProtocolsConfigurator.seenProtocols.set(null);
|
||||||
|
|
||||||
try (IBlockheadClient client = new XBlockheadClient(uri))
|
TrackingEndpoint clientSocket = new TrackingEndpoint(testname.getMethodName());
|
||||||
{
|
ClientUpgradeRequest upgradeRequest = new ClientUpgradeRequest();
|
||||||
client.addHeader("Sec-Websocket-Protocol: echo, chat, status\r\n");
|
// header name is not to spec (case wise)
|
||||||
client.connect();
|
upgradeRequest.setHeader("Sec-Websocket-Protocol", "echo, chat, status");
|
||||||
client.sendStandardRequest();
|
Future<org.eclipse.jetty.websocket.api.Session> clientConnectFuture = client.connect(clientSocket, wsUri, upgradeRequest);
|
||||||
client.expectUpgradeResponse();
|
|
||||||
|
|
||||||
client.write(new TextFrame().setPayload("getProtocols"));
|
org.eclipse.jetty.websocket.api.Session clientSession = clientConnectFuture.get(Defaults.CONNECT_TIMEOUT_MS, TimeUnit.MILLISECONDS);
|
||||||
EventQueue<WebSocketFrame> frames = client.readFrames(1, 1, TimeUnit.SECONDS);
|
clientSession.getRemote().sendString("getProtocols");
|
||||||
WebSocketFrame frame = frames.poll();
|
|
||||||
Assert.assertThat("Frame Response", frame.getPayloadAsUTF8(), is("Requested Protocols: [\"echo\",\"chat\",\"status\"]"));
|
String incomingMessage = clientSocket.messageQueue.poll(5, TimeUnit.SECONDS);
|
||||||
}
|
assertThat("Incoming message", incomingMessage, is("Requested Protocols: [\"echo\",\"chat\",\"status\"]"));
|
||||||
|
|
||||||
|
clientSession.close();
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -632,19 +655,19 @@ public class ConfiguratorTest
|
||||||
@Test
|
@Test
|
||||||
public void testDecoderWithProtocol() throws Exception
|
public void testDecoderWithProtocol() throws Exception
|
||||||
{
|
{
|
||||||
URI uri = baseServerUri.resolve("/timedecoder");
|
URI wsUri = baseServerUri.resolve("/timedecoder");
|
||||||
|
|
||||||
try (XBlockheadClient client = new XBlockheadClient(uri))
|
TrackingEndpoint clientSocket = new TrackingEndpoint(testname.getMethodName());
|
||||||
{
|
ClientUpgradeRequest upgradeRequest = new ClientUpgradeRequest();
|
||||||
client.addHeader("Sec-Websocket-Protocol: gmt\r\n");
|
upgradeRequest.setSubProtocols("gmt");
|
||||||
client.connect();
|
Future<org.eclipse.jetty.websocket.api.Session> clientConnectFuture = client.connect(clientSocket, wsUri, upgradeRequest);
|
||||||
client.sendStandardRequest();
|
|
||||||
client.expectUpgradeResponse();
|
|
||||||
|
|
||||||
client.write(new TextFrame().setPayload("2016-06-20T14:27:44"));
|
org.eclipse.jetty.websocket.api.Session clientSession = clientConnectFuture.get(Defaults.CONNECT_TIMEOUT_MS, TimeUnit.MILLISECONDS);
|
||||||
EventQueue<WebSocketFrame> frames = client.readFrames(1, 1, TimeUnit.SECONDS);
|
clientSession.getRemote().sendString("2016-06-20T14:27:44");
|
||||||
WebSocketFrame frame = frames.poll();
|
|
||||||
Assert.assertThat("Frame Response", frame.getPayloadAsUTF8(), is("cal=2016.06.20 AD at 14:27:44 +0000"));
|
String incomingMessage = clientSocket.messageQueue.poll(5, TimeUnit.SECONDS);
|
||||||
}
|
assertThat("Incoming message", incomingMessage, is("cal=2016.06.20 AD at 14:27:44 +0000"));
|
||||||
|
|
||||||
|
clientSession.close();
|
||||||
}
|
}
|
||||||
}
|
}
|
|
@ -26,6 +26,10 @@ 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=INFO
|
||||||
# org.eclipse.jetty.websocket.tests.LEVEL=DEBUG
|
# org.eclipse.jetty.websocket.tests.LEVEL=DEBUG
|
||||||
|
# org.eclipse.jetty.websocket.tests.client.LEVEL=DEBUG
|
||||||
|
# org.eclipse.jetty.websocket.tests.client.jsr356.LEVEL=DEBUG
|
||||||
|
# org.eclipse.jetty.websocket.tests.server.LEVEL=DEBUG
|
||||||
|
# org.eclipse.jetty.websocket.tests.server.jsr356.LEVEL=DEBUG
|
||||||
# org.eclipse.jetty.websocket.common.io.LEVEL=DEBUG
|
# org.eclipse.jetty.websocket.common.io.LEVEL=DEBUG
|
||||||
# org.eclipse.jetty.websocket.server.ab.LEVEL=DEBUG
|
# org.eclipse.jetty.websocket.server.ab.LEVEL=DEBUG
|
||||||
# org.eclipse.jetty.websocket.common.WebSocketSession.LEVEL=DEBUG
|
# org.eclipse.jetty.websocket.common.WebSocketSession.LEVEL=DEBUG
|
||||||
|
@ -36,17 +40,5 @@ org.eclipse.jetty.LEVEL=WARN
|
||||||
# org.eclipse.jetty.websocket.server.helper.LEVEL=DEBUG
|
# org.eclipse.jetty.websocket.server.helper.LEVEL=DEBUG
|
||||||
org.eclipse.jetty.websocket.common.CompletionCallback.LEVEL=ALL
|
org.eclipse.jetty.websocket.common.CompletionCallback.LEVEL=ALL
|
||||||
|
|
||||||
# org.eclipse.jetty.websocket.client.io.ConnectPromise.LEVEL=DEBUG
|
|
||||||
# org.eclipse.jetty.websocket.common.WebSocketSession_OPEN.LEVEL=DEBUG
|
|
||||||
# org.eclipse.jetty.websocket.common.io.AbstractWebSocketConnection_OPEN.LEVEL=DEBUG
|
|
||||||
|
|
||||||
### Show state changes on BrowserDebugTool
|
|
||||||
# -- LEAVE THIS AT DEBUG LEVEL --
|
|
||||||
org.eclipse.jetty.websocket.server.browser.LEVEL=DEBUG
|
|
||||||
|
|
||||||
### Disabling intentional error out of RFCSocket
|
### Disabling intentional error out of RFCSocket
|
||||||
org.eclipse.jetty.websocket.server.helper.RFCSocket.LEVEL=OFF
|
org.eclipse.jetty.websocket.server.helper.RFCSocket.LEVEL=OFF
|
||||||
|
|
||||||
### Hiding Stack Traces from various test cases
|
|
||||||
org.eclipse.jetty.websocket.tests.server.ABSocket.STACKS=OFF
|
|
||||||
org.eclipse.jetty.websocket.server.WebSocketCloseTest$FastFailSocket.STACKS=OFF
|
|
|
@ -0,0 +1,4 @@
|
||||||
|
a|Benjamin Franklin
|
||||||
|
q|There never was a good war or a bad peace.
|
||||||
|
q|We must, indeed, all hang together, or assuredly we shall all hang separately.
|
||||||
|
q|Our new Constitution is now established, and has an appearance that promises permanency; but in this world nothing can be said to be certain, except death and taxes.
|
|
@ -0,0 +1,5 @@
|
||||||
|
a|Mark Twain
|
||||||
|
q|He is now fast rising from affluence to poverty.
|
||||||
|
q|A baby is an inestimable blessing and bother.
|
||||||
|
q|As I slowly grow wise I briskly grow cautious.
|
||||||
|
q|A circle is a round straight line with a hole in the middle.
|
Loading…
Reference in New Issue