Merge branch 'jetty-9' of ssh://git.eclipse.org/gitroot/jetty/org.eclipse.jetty.project into jetty-9

This commit is contained in:
Jesse McConnell 2012-06-26 10:58:30 -05:00
commit 6e2d9549c9
27 changed files with 1045 additions and 322 deletions

View File

@ -634,6 +634,7 @@ public class HttpGenerator
{ {
// special case for websocket connection ordering // special case for websocket connection ordering
header.put(HttpHeader.CONNECTION.getBytesColonSpace()).put(HttpHeader.UPGRADE.getBytes()); header.put(HttpHeader.CONNECTION.getBytesColonSpace()).put(HttpHeader.UPGRADE.getBytes());
header.put(CRLF);
break; break;
} }

View File

@ -13,6 +13,7 @@
package org.eclipse.jetty.http; package org.eclipse.jetty.http;
import static org.hamcrest.Matchers.*;
import static org.hamcrest.Matchers.containsString; import static org.hamcrest.Matchers.containsString;
import static org.hamcrest.Matchers.either; import static org.hamcrest.Matchers.either;
import static org.hamcrest.Matchers.equalTo; import static org.hamcrest.Matchers.equalTo;
@ -30,6 +31,7 @@ import org.eclipse.jetty.http.HttpGenerator.Action;
import org.eclipse.jetty.http.HttpGenerator.ResponseInfo; import org.eclipse.jetty.http.HttpGenerator.ResponseInfo;
import org.eclipse.jetty.util.BufferUtil; import org.eclipse.jetty.util.BufferUtil;
import org.hamcrest.Matchers; import org.hamcrest.Matchers;
import org.junit.Assert;
import org.junit.Test; import org.junit.Test;
/** /**
@ -392,6 +394,40 @@ public class HttpGeneratorServerTest
assertThat(head,containsString("Content-Length: 0")); assertThat(head,containsString("Content-Length: 0"));
} }
@Test
public void testResponseUpgrade() throws Exception
{
ByteBuffer header=BufferUtil.allocate(8096);
HttpGenerator gen = new HttpGenerator();
HttpGenerator.Result
result=gen.generate(null,null,null,null,null,Action.COMPLETE);
assertEquals(HttpGenerator.State.COMMITTING_COMPLETING,gen.getState());
assertEquals(HttpGenerator.Result.NEED_INFO,result);
ResponseInfo info = new ResponseInfo(HttpVersion.HTTP_1_1,new HttpFields(),-1,101,null,false);
info.getHttpFields().add("Upgrade","WebSocket");
info.getHttpFields().add("Connection","Upgrade");
info.getHttpFields().add("Sec-WebSocket-Accept","123456789==");
result=gen.generate(info,header,null,null,null,null);
assertEquals(HttpGenerator.Result.FLUSH,result);
String head = BufferUtil.toString(header);
System.out.println(head);
BufferUtil.clear(header);
result=gen.generate(info,null,null,null,null,null);
assertEquals(HttpGenerator.Result.OK,result);
assertEquals(HttpGenerator.State.END,gen.getState());
assertEquals(0,gen.getContentPrepared());
assertThat(head,startsWith("HTTP/1.1 101 Switching Protocols"));
assertThat(head,containsString("Upgrade: WebSocket\r\n"));
assertThat(head,containsString("Connection: Upgrade\r\n"));
}
@Test @Test
public void testResponseWithChunkedContent() throws Exception public void testResponseWithChunkedContent() throws Exception
{ {
@ -497,6 +533,7 @@ public class HttpGeneratorServerTest
assertThat(head,containsString("Transfer-Encoding: chunked")); assertThat(head,containsString("Transfer-Encoding: chunked"));
assertTrue(head.endsWith("\r\n\r\n10\r\n")); assertTrue(head.endsWith("\r\n\r\n10\r\n"));
} }
@Test @Test
public void testResponseWithKnownContent() throws Exception public void testResponseWithKnownContent() throws Exception
{ {

View File

@ -62,7 +62,7 @@ public class Server extends HandlerWrapper implements Attributes
Server.class.getPackage().getImplementationVersion()!=null) Server.class.getPackage().getImplementationVersion()!=null)
__version=Server.class.getPackage().getImplementationVersion(); __version=Server.class.getPackage().getImplementationVersion();
else else
__version=System.getProperty("jetty.version","8.0.y.z-SNAPSHOT"); __version=System.getProperty("jetty.version","9.0.y.z-SNAPSHOT");
} }
private final Container _container=new Container(); private final Container _container=new Container();

View File

@ -0,0 +1,151 @@
package org.eclipse.jetty.servlet;
import static org.hamcrest.Matchers.*;
import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStream;
import java.io.InputStreamReader;
import java.io.OutputStream;
import java.net.InetSocketAddress;
import java.net.Socket;
import java.net.SocketAddress;
import java.net.URI;
import javax.servlet.ServletException;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import org.eclipse.jetty.server.SelectChannelConnector;
import org.eclipse.jetty.server.Server;
import org.eclipse.jetty.util.IO;
import org.junit.AfterClass;
import org.junit.Assert;
import org.junit.BeforeClass;
import org.junit.Test;
public class ResponseHeadersTest
{
/** Pretend to be a WebSocket Upgrade (not real) */
@SuppressWarnings("serial")
private static class SimulateUpgradeServlet extends HttpServlet
{
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse response) throws ServletException, IOException
{
response.setHeader("Upgrade","WebSocket");
response.addHeader("Connection","Upgrade");
response.addHeader("Sec-WebSocket-Accept","123456789==");
response.setStatus(HttpServletResponse.SC_SWITCHING_PROTOCOLS);
}
}
private static Server server;
private static SelectChannelConnector connector;
private static URI serverUri;
@BeforeClass
public static void startServer() throws Exception
{
// Configure Server
server = new Server();
connector = new SelectChannelConnector();
server.addConnector(connector);
ServletContextHandler context = new ServletContextHandler();
context.setContextPath("/");
server.setHandler(context);
// Serve capture servlet
context.addServlet(new ServletHolder(new SimulateUpgradeServlet()),"/*");
// Start Server
server.start();
String host = connector.getHost();
if (host == null)
{
host = "localhost";
}
int port = connector.getLocalPort();
serverUri = new URI(String.format("http://%s:%d/",host,port));
System.out.printf("Server URI: %s%n",serverUri);
}
@AfterClass
public static void stopServer()
{
try
{
server.stop();
}
catch (Exception e)
{
e.printStackTrace(System.err);
}
}
@Test
public void testResponseHeaderFormat() throws IOException
{
Socket socket = new Socket();
SocketAddress endpoint = new InetSocketAddress(serverUri.getHost(),serverUri.getPort());
socket.connect(endpoint);
StringBuilder req = new StringBuilder();
req.append("GET / HTTP/1.1\r\n");
req.append(String.format("Host: %s:%d\r\n",serverUri.getHost(),serverUri.getPort()));
req.append("\r\n");
OutputStream out = null;
InputStream in = null;
try
{
out = socket.getOutputStream();
in = socket.getInputStream();
// Write request
out.write(req.toString().getBytes());
out.flush();
// Read response
String respHeader = readResponseHeader(in);
System.out.println("RESPONSE: " + respHeader);
// Now test for properly formatted HTTP Response Headers.
Assert.assertThat("Response Code",respHeader,startsWith("HTTP/1.1 101 Switching Protocols"));
Assert.assertThat("Response Header Upgrade",respHeader,containsString("Upgrade: WebSocket\r\n"));
Assert.assertThat("Response Header Connection",respHeader,containsString("Connection: Upgrade\r\n"));
}
finally
{
IO.close(in);
IO.close(out);
socket.close();
}
}
private String readResponseHeader(InputStream in) throws IOException
{
InputStreamReader isr = new InputStreamReader(in);
BufferedReader reader = new BufferedReader(isr);
StringBuilder header = new StringBuilder();
// Read the response header
String line = reader.readLine();
Assert.assertNotNull(line);
Assert.assertThat(line,startsWith("HTTP/1.1 "));
header.append(line).append("\r\n");
while ((line = reader.readLine()) != null)
{
if (line.trim().length() == 0)
{
break;
}
header.append(line).append("\r\n");
}
return header.toString();
}
}

View File

@ -0,0 +1,98 @@
package org.eclipse.jetty.websocket.api;
import java.util.HashMap;
import java.util.Map;
import org.eclipse.jetty.util.QuotedStringTokenizer;
/**
* Proposed interface for API (not yet settled)
*/
public class ExtensionConfig
{
public static ExtensionConfig parse(String parameterizedName)
{
QuotedStringTokenizer tok = new QuotedStringTokenizer(parameterizedName,";");
ExtensionConfig ext = new ExtensionConfig(tok.nextToken().trim());
while (tok.hasMoreTokens())
{
QuotedStringTokenizer nv = new QuotedStringTokenizer(tok.nextToken().trim(),"=");
String key = nv.nextToken().trim();
String value = nv.hasMoreTokens()?nv.nextToken().trim():null;
ext.setParameter(key,value);
}
return ext;
}
private final String name;
private Map<String, String> parameters;
public ExtensionConfig(String name)
{
this.name = name;
this.parameters = new HashMap<>();
}
public String getName()
{
return name;
}
public int getParameter(String key, int defValue)
{
String val = parameters.get(key);
if(val == null) {
return defValue;
}
return Integer.valueOf(val);
}
public String getParameter(String key, String defValue)
{
String val = parameters.get(key);
if(val == null) {
return defValue;
}
return val;
}
public String getParameterizedName()
{
StringBuilder str = new StringBuilder();
str.append(name);
for (String param : parameters.keySet())
{
str.append(';').append(param).append('=').append(QuotedStringTokenizer.quoteIfNeeded(parameters.get(param),";="));
}
return str.toString();
}
/**
* Initialize the parameters on this config from the other configuration.
* @param other the other configuration.
*/
public void init(ExtensionConfig other)
{
this.parameters.clear();
this.parameters.putAll(other.parameters);
}
public void setParameter(String key, int value)
{
parameters.put(key,Integer.toString(value));
}
public void setParameter(String key, String value)
{
parameters.put(key,value);
}
@Override
public String toString()
{
return getParameterizedName();
}
}

View File

@ -1,13 +0,0 @@
package org.eclipse.jetty.websocket.api;
import java.util.Map;
/**
* Proposed interface for API (not yet settled)
*/
public interface ExtensionRef
{
String getName();
Map<String, String> getParameters();
}

View File

@ -15,14 +15,6 @@ import org.eclipse.jetty.websocket.frames.TextFrame;
*/ */
public interface WebSocket public interface WebSocket
{ {
/**
* Server side interface (to be moved once API settles down)
*/
public static interface Accept
{
WebSocket acceptWebSocket(WebSocketHandshakeRequest request, WebSocketHandshakeResponse response);
}
/** /**
* Advanced usage, for those interested in flags * Advanced usage, for those interested in flags
*/ */

View File

@ -1,25 +0,0 @@
package org.eclipse.jetty.websocket.api;
import java.util.List;
/**
* Proposed interface for API (not yet settled)
*/
public interface WebSocketHandshakeRequest
{
// get/set arbitrary Http Header Fields? (if so, then this should not use servlet-api)
String getEndpoint();
List<ExtensionRef> getExtensions();
String getHost();
String getOrigin();
String getWebSocketKey();
String[] getWebSocketProtocols();
int getWebSocketVersion();
}

View File

@ -1,17 +0,0 @@
package org.eclipse.jetty.websocket.api;
import java.util.List;
/**
* Proposed interface for API (not yet settled)
*/
public interface WebSocketHandshakeResponse
{
String getWebSocketAccept();
boolean isUpgradeAccepted();
void setExtensions(List<ExtensionRef> refs);
void setUpgradeAccepted(boolean accept);
}

View File

@ -15,64 +15,40 @@
*******************************************************************************/ *******************************************************************************/
package org.eclipse.jetty.websocket.extensions; package org.eclipse.jetty.websocket.extensions;
import java.nio.ByteBuffer; import org.eclipse.jetty.websocket.api.ExtensionConfig;
import java.util.HashMap;
import java.util.Map;
import org.eclipse.jetty.util.QuotedStringTokenizer;
public class AbstractExtension implements Extension public class AbstractExtension implements Extension
{ {
private final String name; private final ExtensionConfig config;
private final Map<String, String> parameters = new HashMap<String, String>();
public AbstractExtension(String name) public AbstractExtension(String name)
{ {
this.name = name; this.config = new ExtensionConfig(name);
} }
public int getInitParameter(String name, int dft) @Override
public ExtensionConfig getConfig()
{ {
String v = parameters.get(name); // TODO Auto-generated method stub
if (v==null) return null;
{
return dft;
}
return Integer.valueOf(v);
}
public String getInitParameter(String name,String dft)
{
if (!parameters.containsKey(name))
{
return dft;
}
return parameters.get(name);
} }
@Override @Override
public String getName() public String getName()
{ {
return name; return config.getName();
} }
@Override @Override
public String getParameterizedName() public String getParameterizedName()
{ {
StringBuilder name = new StringBuilder(); return config.getParameterizedName();
name.append(name);
for (String param : parameters.keySet())
{
name.append(';').append(param).append('=').append(QuotedStringTokenizer.quoteIfNeeded(parameters.get(param),";="));
}
return name.toString();
} }
@Override @Override
public boolean init(Map<String, String> parameters) public void setConfig(ExtensionConfig config)
{ {
parameters.putAll(parameters); this.config.init(config);
return true;
} }
@Override @Override

View File

@ -15,12 +15,12 @@
*******************************************************************************/ *******************************************************************************/
package org.eclipse.jetty.websocket.extensions; package org.eclipse.jetty.websocket.extensions;
import java.util.Map; import org.eclipse.jetty.websocket.api.ExtensionConfig;
public interface Extension public interface Extension
{ {
public ExtensionConfig getConfig();
public String getName(); public String getName();
public String getParameterizedName(); public String getParameterizedName();
public void setConfig(ExtensionConfig config);
public boolean init(Map<String,String> parameters);
} }

View File

@ -15,12 +15,12 @@
*******************************************************************************/ *******************************************************************************/
package org.eclipse.jetty.websocket.extensions.deflate; package org.eclipse.jetty.websocket.extensions.deflate;
import java.util.Map;
import java.util.zip.Deflater; import java.util.zip.Deflater;
import java.util.zip.Inflater; import java.util.zip.Inflater;
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.ExtensionConfig;
import org.eclipse.jetty.websocket.extensions.AbstractExtension; import org.eclipse.jetty.websocket.extensions.AbstractExtension;
/** /**
@ -40,22 +40,14 @@ public class DeflateFrameExtension extends AbstractExtension
} }
@Override @Override
public boolean init(Map<String, String> parameters) public void setConfig(ExtensionConfig config)
{ {
if (!parameters.containsKey("minLength")) super.setConfig(config);
{
parameters.put("minLength",Integer.toString(_minLength)); _minLength = config.getParameter("minLength",_minLength);
}
if(super.init(parameters))
{
_minLength=getInitParameter("minLength",_minLength);
_deflater = new Deflater(); _deflater = new Deflater();
_inflater = new Inflater(); _inflater = new Inflater();
return true;
}
return false;
} }
/* (non-Javadoc) /* (non-Javadoc)

View File

@ -15,9 +15,7 @@
*******************************************************************************/ *******************************************************************************/
package org.eclipse.jetty.websocket.extensions.fragment; package org.eclipse.jetty.websocket.extensions.fragment;
import java.io.IOException; import org.eclipse.jetty.websocket.api.ExtensionConfig;
import java.util.Map;
import org.eclipse.jetty.websocket.extensions.AbstractExtension; import org.eclipse.jetty.websocket.extensions.AbstractExtension;
public class FragmentExtension extends AbstractExtension public class FragmentExtension extends AbstractExtension
@ -31,15 +29,13 @@ public class FragmentExtension extends AbstractExtension
} }
@Override @Override
public boolean init(Map<String, String> parameters) public void setConfig(ExtensionConfig config)
{ {
if(super.init(parameters)) super.setConfig(config);
{
_maxLength=getInitParameter("maxLength",_maxLength); _maxLength = config.getParameter("maxLength",_maxLength);
_minFragments=getInitParameter("minFragments",_minFragments); _minFragments = config.getParameter("minFragments",_minFragments);
return true;
}
return false;
} }
/* TODO: Migrate to new Jetty9 IO /* TODO: Migrate to new Jetty9 IO

View File

@ -1,29 +1,151 @@
package org.eclipse.jetty.websocket.server; package org.eclipse.jetty.websocket.server;
import java.io.IOException;
import java.nio.ByteBuffer;
import java.util.ArrayList;
import java.util.List;
import java.util.concurrent.Executor; import java.util.concurrent.Executor;
import org.eclipse.jetty.io.AbstractAsyncConnection; import org.eclipse.jetty.io.AbstractAsyncConnection;
import org.eclipse.jetty.io.AsyncEndPoint; import org.eclipse.jetty.io.AsyncEndPoint;
import org.eclipse.jetty.io.ByteBufferPool;
import org.eclipse.jetty.io.RuntimeIOException;
import org.eclipse.jetty.io.StandardByteBufferPool;
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.ExtensionConfig;
import org.eclipse.jetty.websocket.api.StatusCode;
import org.eclipse.jetty.websocket.api.WebSocketPolicy;
import org.eclipse.jetty.websocket.frames.CloseFrame;
import org.eclipse.jetty.websocket.generator.Generator;
import org.eclipse.jetty.websocket.parser.Parser;
// TODO: implement WebSocket.Connection (for API access)?
public class AsyncWebSocketConnection extends AbstractAsyncConnection public class AsyncWebSocketConnection extends AbstractAsyncConnection
{ {
// TODO: track extensions? (only those that need to operate at this level?) private static final Logger LOG = Log.getLogger(AsyncWebSocketConnection.class);
// TODO: track generic WebSocket.Connection (for API access)? private static final ThreadLocal<AsyncWebSocketConnection> CURRENT_CONNECTION = new ThreadLocal<AsyncWebSocketConnection>();
public AsyncWebSocketConnection(AsyncEndPoint endp, Executor executor) public static AsyncWebSocketConnection getCurrentConnection()
{ {
super(endp,executor); return CURRENT_CONNECTION.get();
} }
public AsyncWebSocketConnection(AsyncEndPoint endp, Executor executor, boolean executeOnlyFailure) protected static void setCurrentConnection(AsyncWebSocketConnection connection)
{ {
super(endp,executor,executeOnlyFailure); CURRENT_CONNECTION.set(connection);
}
private final ByteBufferPool bufferPool;
private Generator generator;
private Parser parser;
private WebSocketPolicy policy;
// TODO: track extensions? (only those that need to operate at this level?)
// TODO: are extensions going to layer the endpoint?
// TODO: are extensions going to layer the connection?
private List<ExtensionConfig> extensions;
public AsyncWebSocketConnection(AsyncEndPoint endp, Executor executor, WebSocketPolicy policy)
{
super(endp,executor);
this.policy = policy;
this.bufferPool = new StandardByteBufferPool(policy.getBufferSize());
this.generator = new Generator(bufferPool,policy);
this.parser = new Parser(policy);
this.extensions = new ArrayList<>();
}
private int fill(AsyncEndPoint endPoint, ByteBuffer buffer)
{
try
{
return endPoint.fill(buffer);
}
catch (IOException e)
{
terminateConnection(StatusCode.PROTOCOL,e.getMessage());
throw new RuntimeIOException(e);
}
}
/**
* Get the list of extensions in use.
* <p>
* This list is negotiated during the WebSocket Upgrade Request/Response handshake.
*
* @return the list of negotiated extensions in use.
*/
public List<ExtensionConfig> getExtensions()
{
return extensions;
} }
@Override @Override
public void onFillable() public void onFillable()
{ {
// TODO Auto-generated method stub LOG.debug("onFillable");
setCurrentConnection(this);
ByteBuffer buffer = bufferPool.acquire(policy.getBufferSize(),false);
BufferUtil.clear(buffer);
try
{
read(buffer);
}
finally
{
setCurrentConnection(null);
bufferPool.release(buffer);
}
}
private void read(ByteBuffer buffer)
{
while (true)
{
int filled = fill(getEndPoint(),buffer);
if (filled == 0)
{
break;
}
if (filled < 0)
{
// IO error
terminateConnection(StatusCode.PROTOCOL,null);
break;
}
parser.parse(buffer);
}
}
/**
* Get the list of extensions in use.
* <p>
* This list is negotiated during the WebSocket Upgrade Request/Response handshake.
*
* @param extensions
* the list of negotiated extensions in use.
*/
public void setExtensions(List<ExtensionConfig> extensions)
{
this.extensions = extensions;
}
/**
* For terminating connections forcefully.
*
* @param statusCode
* the WebSocket status code.
* @param reason
* the (optiona) reason string. (null is allowed)
* @see StatusCode
*/
private void terminateConnection(short statusCode, String reason)
{
CloseFrame close = new CloseFrame(statusCode);
close.setReason(reason);
// fire and forget -> close frame
getEndPoint().write(null,new WebSocketCloseCallback(this),generator.generate(close));
} }
} }

View File

@ -0,0 +1,25 @@
package org.eclipse.jetty.websocket.server;
import org.eclipse.jetty.util.Callback;
public class WebSocketCloseCallback implements Callback<Void>
{
private AsyncWebSocketConnection conn;
public WebSocketCloseCallback(AsyncWebSocketConnection conn)
{
this.conn = conn;
}
@Override
public void completed(Void context)
{
this.conn.getEndPoint().close();
}
@Override
public void failed(Void context, Throwable cause)
{
this.conn.getEndPoint().close();
}
}

View File

@ -39,7 +39,7 @@ import org.eclipse.jetty.server.handler.HandlerWrapper;
import org.eclipse.jetty.websocket.api.WebSocketBehavior; import org.eclipse.jetty.websocket.api.WebSocketBehavior;
import org.eclipse.jetty.websocket.api.WebSocketPolicy; import org.eclipse.jetty.websocket.api.WebSocketPolicy;
public abstract class WebSocketHandler extends HandlerWrapper implements WebSocketServerFactory.Acceptor public abstract class WebSocketHandler extends HandlerWrapper implements WebSocketServer.Acceptor
{ {
private final WebSocketServerFactory webSocketFactory; private final WebSocketServerFactory webSocketFactory;

View File

@ -1,9 +0,0 @@
package org.eclipse.jetty.websocket.server;
/**
* <a href="https://tools.ietf.org/html/draft-hixie-thewebsocketprotocol-76">Hixie-76 Draft for WebSocket protocol</a>
* Seen in use by Safari/OSX
*/
public class WebSocketHixie76 {
/* Put Hixie-76 specifics in here */
}

View File

@ -0,0 +1,18 @@
package org.eclipse.jetty.websocket.server;
import org.eclipse.jetty.util.FutureCallback;
public class WebSocketOpenCallback extends FutureCallback<String>
{
@Override
public void completed(String context)
{
// TODO notify API on connection open
}
@Override
public void failed(String context, Throwable x)
{
// TODO notify API on open failure
}
}

View File

@ -0,0 +1,60 @@
package org.eclipse.jetty.websocket.server;
import java.io.IOException;
import java.util.List;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import org.eclipse.jetty.websocket.WebSocket;
import org.eclipse.jetty.websocket.extensions.Extension;
/**
* Main API class for WebSocket servers
*/
public interface WebSocketServer
{
public static interface Acceptor
{
/**
* <p>
* Checks the origin of an incoming WebSocket handshake request.
* </p>
*
* @param request
* the incoming HTTP upgrade request
* @param origin
* the origin URI
* @return boolean to indicate that the origin is acceptable.
*/
boolean checkOrigin(HttpServletRequest request, String origin);
/* ------------------------------------------------------------ */
/**
* <p>
* Factory method that applications needs to implement to return a {@link WebSocket} object.
* </p>
*
* @param request
* the incoming HTTP upgrade request
* @param protocol
* the websocket sub protocol
* @return a new {@link WebSocket} object that will handle websocket events.
*/
WebSocket doWebSocketConnect(HttpServletRequest request, String protocol);
}
public static interface Handshake
{
/**
* Formulate a WebSocket upgrade handshake response.
*
* @param request
* @param response
* @param extensions
* @param acceptedSubProtocol
*/
public void doHandshakeResponse(HttpServletRequest request, HttpServletResponse response, List<Extension> extensions, String acceptedSubProtocol)
throws IOException;
}
}

View File

@ -15,78 +15,92 @@ package org.eclipse.jetty.websocket.server;
import java.io.IOException; import java.io.IOException;
import java.util.ArrayList; import java.util.ArrayList;
import java.util.Collections;
import java.util.Enumeration; import java.util.Enumeration;
import java.util.HashMap; import java.util.HashMap;
import java.util.List; import java.util.List;
import java.util.Map; import java.util.Map;
import java.util.Queue; import java.util.Queue;
import java.util.concurrent.ConcurrentLinkedQueue; import java.util.concurrent.ConcurrentLinkedQueue;
import java.util.concurrent.Executor;
import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse; import javax.servlet.http.HttpServletResponse;
import org.eclipse.jetty.http.HttpStatus; import org.eclipse.jetty.http.HttpStatus;
import org.eclipse.jetty.io.AsyncEndPoint; import org.eclipse.jetty.io.AsyncEndPoint;
import org.eclipse.jetty.io.ByteBufferPool;
import org.eclipse.jetty.io.StandardByteBufferPool;
import org.eclipse.jetty.server.HttpConnection; import org.eclipse.jetty.server.HttpConnection;
import org.eclipse.jetty.util.QuotedStringTokenizer; import org.eclipse.jetty.util.QuotedStringTokenizer;
import org.eclipse.jetty.util.component.AbstractLifeCycle; import org.eclipse.jetty.util.component.AbstractLifeCycle;
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.WebSocket; import org.eclipse.jetty.websocket.WebSocket;
import org.eclipse.jetty.websocket.api.ExtensionConfig;
import org.eclipse.jetty.websocket.api.WebSocketPolicy; import org.eclipse.jetty.websocket.api.WebSocketPolicy;
import org.eclipse.jetty.websocket.extensions.Extension; import org.eclipse.jetty.websocket.extensions.Extension;
import org.eclipse.jetty.websocket.extensions.deflate.DeflateFrameExtension; import org.eclipse.jetty.websocket.extensions.deflate.DeflateFrameExtension;
import org.eclipse.jetty.websocket.extensions.fragment.FragmentExtension; import org.eclipse.jetty.websocket.extensions.fragment.FragmentExtension;
import org.eclipse.jetty.websocket.extensions.identity.IdentityExtension; import org.eclipse.jetty.websocket.extensions.identity.IdentityExtension;
import org.eclipse.jetty.websocket.server.handshake.HandshakeHixie76;
import org.eclipse.jetty.websocket.server.handshake.HandshakeRFC6455;
/** /**
* Factory to create WebSocket connections * Factory to create WebSocket connections
*/ */
public class WebSocketServerFactory extends AbstractLifeCycle public class WebSocketServerFactory extends AbstractLifeCycle
{ {
public interface Acceptor
{
/* ------------------------------------------------------------ */
/**
* <p>Checks the origin of an incoming WebSocket handshake request.</p>
* @param request the incoming HTTP upgrade request
* @param origin the origin URI
* @return boolean to indicate that the origin is acceptable.
*/
boolean checkOrigin(HttpServletRequest request, String origin);
/* ------------------------------------------------------------ */
/**
* <p>Factory method that applications needs to implement to return a
* {@link WebSocket} object.</p>
* @param request the incoming HTTP upgrade request
* @param protocol the websocket sub protocol
* @return a new {@link WebSocket} object that will handle websocket events.
*/
WebSocket doWebSocketConnect(HttpServletRequest request, String protocol);
}
private static final Logger LOG = Log.getLogger(WebSocketServerFactory.class); private static final Logger LOG = Log.getLogger(WebSocketServerFactory.class);
private static final int RESPONSE_BUFFER_SIZE = 8192;
private final Queue<WebSocketServletConnection> connections = new ConcurrentLinkedQueue<WebSocketServletConnection>(); private final Queue<AsyncWebSocketConnection> connections = new ConcurrentLinkedQueue<AsyncWebSocketConnection>();
private final Map<String,Class<? extends Extension>> _extensionClasses = new HashMap<String, Class<? extends Extension>>(); // TODO: replace with ExtensionRegistry in websocket-core
private final Map<String, Class<? extends Extension>> extensionClasses = new HashMap<>();
{ {
_extensionClasses.put("identity",IdentityExtension.class); extensionClasses.put("identity",IdentityExtension.class);
_extensionClasses.put("fragment",FragmentExtension.class); extensionClasses.put("fragment",FragmentExtension.class);
_extensionClasses.put("x-deflate-frame",DeflateFrameExtension.class); extensionClasses.put("x-deflate-frame",DeflateFrameExtension.class);
} }
private final Acceptor _acceptor; private final Map<Integer, WebSocketServer.Handshake> handshakes = new HashMap<>();
{
handshakes.put(HandshakeRFC6455.VERSION,new HandshakeRFC6455());
handshakes.put(HandshakeHixie76.VERSION,new HandshakeHixie76());
}
private final WebSocketServer.Acceptor acceptor;
private final ByteBufferPool bufferPool;
private final String supportedVersions;
private WebSocketPolicy policy; private WebSocketPolicy policy;
public WebSocketServerFactory(Acceptor acceptor, WebSocketPolicy policy) public WebSocketServerFactory(WebSocketServer.Acceptor acceptor, WebSocketPolicy policy)
{ {
this._acceptor = acceptor; this.acceptor = acceptor;
this.policy = policy; this.policy = policy;
this.bufferPool = new StandardByteBufferPool(RESPONSE_BUFFER_SIZE);
// Create supportedVersions
List<Integer> versions = new ArrayList<>();
for (int v : handshakes.keySet())
{
versions.add(v);
}
Collections.sort(versions,Collections.reverseOrder()); // newest first
StringBuilder rv = new StringBuilder();
for (int v : versions)
{
if (rv.length() > 0)
{
rv.append(", ");
}
rv.append(v);
}
supportedVersions = rv.toString();
} }
public boolean acceptWebSocket(HttpServletRequest request, HttpServletResponse response) public boolean acceptWebSocket(HttpServletRequest request, HttpServletResponse response) throws IOException
throws IOException
{ {
if ("websocket".equalsIgnoreCase(request.getHeader("Upgrade"))) if ("websocket".equalsIgnoreCase(request.getHeader("Upgrade")))
{ {
@ -95,7 +109,7 @@ public class WebSocketServerFactory extends AbstractLifeCycle
{ {
origin = request.getHeader("Sec-WebSocket-Origin"); origin = request.getHeader("Sec-WebSocket-Origin");
} }
if (!_acceptor.checkOrigin(request,origin)) if (!acceptor.checkOrigin(request,origin))
{ {
response.sendError(HttpServletResponse.SC_FORBIDDEN); response.sendError(HttpServletResponse.SC_FORBIDDEN);
return false; return false;
@ -104,7 +118,6 @@ public class WebSocketServerFactory extends AbstractLifeCycle
// Try each requested protocol // Try each requested protocol
WebSocket websocket = null; WebSocket websocket = null;
@SuppressWarnings("unchecked")
Enumeration<String> protocols = request.getHeaders("Sec-WebSocket-Protocol"); Enumeration<String> protocols = request.getHeaders("Sec-WebSocket-Protocol");
String protocol = null; String protocol = null;
while ((protocol == null) && (protocols != null) && protocols.hasMoreElements()) while ((protocol == null) && (protocols != null) && protocols.hasMoreElements())
@ -112,7 +125,7 @@ public class WebSocketServerFactory extends AbstractLifeCycle
String candidate = protocols.nextElement(); String candidate = protocols.nextElement();
for (String p : parseProtocols(candidate)) for (String p : parseProtocols(candidate))
{ {
websocket = _acceptor.doWebSocketConnect(request, p); websocket = acceptor.doWebSocketConnect(request,p);
if (websocket != null) if (websocket != null)
{ {
protocol = p; protocol = p;
@ -125,7 +138,7 @@ public class WebSocketServerFactory extends AbstractLifeCycle
if (websocket == null) if (websocket == null)
{ {
// Try with no protocol // Try with no protocol
websocket = _acceptor.doWebSocketConnect(request, null); websocket = acceptor.doWebSocketConnect(request,null);
if (websocket == null) if (websocket == null)
{ {
@ -142,16 +155,16 @@ public class WebSocketServerFactory extends AbstractLifeCycle
return false; return false;
} }
protected boolean addConnection(WebSocketServletConnection connection) protected boolean addConnection(AsyncWebSocketConnection connection)
{ {
return isRunning() && connections.add(connection); return isRunning() && connections.add(connection);
} }
protected void closeConnections() protected void closeConnections()
{ {
for (WebSocketServletConnection connection : connections) for (AsyncWebSocketConnection connection : connections)
{ {
// TODO connection.shutdown(); connection.getEndPoint().close();
} }
} }
@ -166,7 +179,7 @@ public class WebSocketServerFactory extends AbstractLifeCycle
*/ */
public Map<String, Class<? extends Extension>> getExtensionClassesMap() public Map<String, Class<? extends Extension>> getExtensionClassesMap()
{ {
return _extensionClasses; return extensionClasses;
} }
/** /**
@ -179,35 +192,23 @@ public class WebSocketServerFactory extends AbstractLifeCycle
return policy; return policy;
} }
public List<Extension> initExtensions(List<String> requested,int maxDataOpcodes,int maxControlOpcodes,int maxReservedBits) public List<Extension> initExtensions(List<ExtensionConfig> requested)
{ {
List<Extension> extensions = new ArrayList<Extension>(); List<Extension> extensions = new ArrayList<Extension>();
for (String rExt : requested)
{
QuotedStringTokenizer tok = new QuotedStringTokenizer(rExt,";");
String extName=tok.nextToken().trim();
Map<String,String> parameters = new HashMap<String,String>();
while (tok.hasMoreTokens())
{
QuotedStringTokenizer nv = new QuotedStringTokenizer(tok.nextToken().trim(),"=");
String name=nv.nextToken().trim();
String value=nv.hasMoreTokens()?nv.nextToken().trim():null;
parameters.put(name,value);
}
Extension extension = newExtension(extName); for (ExtensionConfig cfg : requested)
{
Extension extension = newExtension(cfg.getName());
if (extension == null) if (extension == null)
{ {
continue; continue;
} }
if (extension.init(parameters)) extension.setConfig(cfg);
{ LOG.debug("added {}",extension);
LOG.debug("add {} {}",extName,parameters);
extensions.add(extension); extensions.add(extension);
} }
}
LOG.debug("extensions={}",extensions); LOG.debug("extensions={}",extensions);
return extensions; return extensions;
} }
@ -216,7 +217,7 @@ public class WebSocketServerFactory extends AbstractLifeCycle
{ {
try try
{ {
Class<? extends Extension> extClass = _extensionClasses.get(name); Class<? extends Extension> extClass = extensionClasses.get(name);
if (extClass != null) if (extClass != null)
{ {
return extClass.newInstance(); return extClass.newInstance();
@ -234,12 +235,14 @@ public class WebSocketServerFactory extends AbstractLifeCycle
{ {
if (protocol == null) if (protocol == null)
{ {
return new String[]{null}; return new String[]
{ null };
} }
protocol = protocol.trim(); protocol = protocol.trim();
if ((protocol == null) || (protocol.length() == 0)) if ((protocol == null) || (protocol.length() == 0))
{ {
return new String[]{null}; return new String[]
{ null };
} }
String[] passed = protocol.split("\\s*,\\s*"); String[] passed = protocol.split("\\s*,\\s*");
String[] protocols = new String[passed.length + 1]; String[] protocols = new String[passed.length + 1];
@ -254,87 +257,79 @@ public class WebSocketServerFactory extends AbstractLifeCycle
/** /**
* Upgrade the request/response to a WebSocket Connection. * Upgrade the request/response to a WebSocket Connection.
* <p>This method will not normally return, but will instead throw a * <p>
* UpgradeConnectionException, to exit HTTP handling and initiate * This method will not normally return, but will instead throw a UpgradeConnectionException, to exit HTTP handling and initiate WebSocket handling of the
* WebSocket handling of the connection. * connection.
* *
* @param request The request to upgrade * @param request
* @param response The response to upgrade * The request to upgrade
* @param websocket The websocket handler implementation to use * @param response
* @param protocol The websocket protocol * The response to upgrade
* @throws IOException in case of I/O errors * @param websocket
* The websocket handler implementation to use
* @param acceptedSubProtocol
* The accepted websocket sub protocol
* @throws IOException
* in case of I/O errors
*/ */
public void upgrade(HttpServletRequest request, HttpServletResponse response, WebSocket websocket, String protocol) public void upgrade(HttpServletRequest request, HttpServletResponse response, WebSocket websocket, String acceptedSubProtocol) throws IOException
throws IOException
{ {
if (!"websocket".equalsIgnoreCase(request.getHeader("Upgrade"))) if (!"websocket".equalsIgnoreCase(request.getHeader("Upgrade")))
{ {
throw new IllegalStateException("!Upgrade:websocket"); throw new IllegalStateException("Not a 'WebSocket: Ugprade' request");
} }
if (!"HTTP/1.1".equals(request.getProtocol())) if (!"HTTP/1.1".equals(request.getProtocol()))
{ {
throw new IllegalStateException("!HTTP/1.1"); throw new IllegalStateException("Not a 'HTTP/1.1' request");
} }
int draft = request.getIntHeader("Sec-WebSocket-Version"); int version = request.getIntHeader("Sec-WebSocket-Version");
if (draft < 0) { if (version < 0)
{
// Old pre-RFC version specifications (header not present in RFC-6455) // Old pre-RFC version specifications (header not present in RFC-6455)
draft = request.getIntHeader("Sec-WebSocket-Draft"); version = request.getIntHeader("Sec-WebSocket-Draft");
} }
HttpConnection http = HttpConnection.getCurrentConnection();
AsyncEndPoint endp = http.getEndPoint();
List<String> extensions_requested = new ArrayList<String>(); List<ExtensionConfig> extensionsRequested = new ArrayList<>();
@SuppressWarnings("unchecked")
Enumeration<String> e = request.getHeaders("Sec-WebSocket-Extensions"); Enumeration<String> e = request.getHeaders("Sec-WebSocket-Extensions");
while (e.hasMoreElements()) while (e.hasMoreElements())
{ {
QuotedStringTokenizer tok = new QuotedStringTokenizer(e.nextElement(),","); QuotedStringTokenizer tok = new QuotedStringTokenizer(e.nextElement(),",");
while (tok.hasMoreTokens()) while (tok.hasMoreTokens())
{ {
extensions_requested.add(tok.nextToken()); extensionsRequested.add(ExtensionConfig.parse(tok.nextToken()));
} }
} }
final WebSocketServletConnection connection; WebSocketServer.Handshake handshaker = handshakes.get(version);
switch (draft) if (handshaker == null)
{ {
case org.eclipse.jetty.websocket.api.WebSocket.VERSION: // RFC 6455 Version LOG.warn("Unsupported Websocket version: " + version);
{
// List<Extension> extensions = initExtensions(extensions_requested,
// 8 - WebSocketConnectionRFC6455.OP_EXT_DATA,
// 16 - WebSocketConnectionRFC6455.OP_EXT_CTRL,
// 3);
// connection = new WebSocketServletConnectionRFC6455(this, websocket, endp, _buffers, http.getTimeStamp(), _maxIdleTime, protocol, extensions, draft);
break;
}
default:
{
LOG.warn("Unsupported Websocket version: " + draft);
// Per RFC 6455 - 4.4 - Supporting Multiple Versions of WebSocket Protocol // Per RFC 6455 - 4.4 - Supporting Multiple Versions of WebSocket Protocol
// Using the examples as outlined // Using the examples as outlined
response.setHeader("Sec-WebSocket-Version", "" + org.eclipse.jetty.websocket.api.WebSocket.VERSION /*+ ", 0"*/); response.setHeader("Sec-WebSocket-Version",supportedVersions);
response.setStatus(HttpStatus.BAD_REQUEST_400); response.sendError(HttpStatus.BAD_REQUEST_400,"Unsupported websocket version specification");
return; return;
} }
}
// addConnection(connection); // Create connection
HttpConnection http = HttpConnection.getCurrentConnection();
AsyncEndPoint endp = http.getEndPoint();
Executor executor = http.getConnector().findExecutor();
final AsyncWebSocketConnection connection = new AsyncWebSocketConnection(endp,executor,policy);
endp.setAsyncConnection(connection);
// Set the defaults // Initialize / Negotiate Extensions
// connection.getConnection().setMaxBinaryMessageSize(_maxBinaryMessageSize); List<Extension> extensions = initExtensions(extensionsRequested);
// connection.getConnection().setMaxTextMessageSize(_maxTextMessageSize);
// Let the connection finish processing the handshake // Process (version specific) handshake response
// connection.handshake(request, response, protocol); handshaker.doHandshakeResponse(request,response,extensions,acceptedSubProtocol);
response.flushBuffer();
// Give the connection any unused data from the HTTP connection. // Add connection
// connection.fillBuffersFrom(((HttpParser)http.getParser()).getHeaderBuffer()); addConnection(connection);
// connection.fillBuffersFrom(((HttpParser)http.getParser()).getBodyBuffer());
// Tell jetty about the new connection // Tell jetty about the new connection
// LOG.debug("Websocket upgrade {} {} {} {}",request.getRequestURI(),draft,protocol,connection); LOG.debug("Websocket upgrade {} {} {} {}",request.getRequestURI(),version,acceptedSubProtocol,connection);
// request.setAttribute("org.eclipse.jetty.io.Connection", connection); request.setAttribute("org.eclipse.jetty.io.Connection",connection); // TODO: this still needed?
} }
} }

View File

@ -29,25 +29,58 @@ import org.eclipse.jetty.websocket.api.WebSocketBehavior;
import org.eclipse.jetty.websocket.api.WebSocketPolicy; import org.eclipse.jetty.websocket.api.WebSocketPolicy;
/** /**
* Servlet to upgrade connections to WebSocket * Abstract Servlet used to bridge the Servlet API to the WebSocket API.
* <p/> * <p>
* The request must have the correct upgrade headers, else it is * This servlet implements the {@link WebSocketServer.Acceptor}, with a default implementation of
* handled as a normal servlet request. * {@link WebSocketServer.Acceptor#checkOrigin(HttpServletRequest, String)} leaving you to implement the
* <p/> * {@link WebSocketServer.Acceptor#doWebSocketConnect(HttpServletRequest, String)}.
* The initParameter "bufferSize" can be used to set the buffer size, * <p>
* which is also the max frame byte size (default 8192). * The most basic implementation would be as follows.
* <p/> *
* The initParameter "maxIdleTime" can be used to set the time in ms * <pre>
* that a websocket may be idle before closing. * package my.example;
* <p/> *
* The initParameter "maxTextMessagesSize" can be used to set the size in characters * import javax.servlet.http.HttpServletRequest;
* that a websocket may be accept before closing. * import org.eclipse.jetty.websocket.WebSocket;
* <p/> * import org.eclipse.jetty.websocket.server.WebSocketServlet;
* The initParameter "maxBinaryMessagesSize" can be used to set the size in bytes *
* that a websocket may be accept before closing. * public class MyEchoServlet extends WebSocketServlet
* {
* &#064;Override
* public WebSocket doWebSocketConnect(HttpServletRequest request, String protocol)
* {
* return new MyEchoSocket();
* }
* }
* </pre>
*
* Note: this servlet will only forward on a incoming request that hits this servlet to the
* {@link WebSocketServer.Acceptor#doWebSocketConnect(HttpServletRequest, String)} if it conforms to a "WebSocket: Upgrade" handshake request. <br>
* All other requests are treated as normal servlet requets.
*
* <p>
* <b>Configuration / Init-Parameters:</b>
*
* <dl>
* <dt>bufferSize</dt>
* <dd>can be used to set the buffer size, which is also the max frame byte size<br>
* <i>Default: 8192</i></dd>
*
* <dt>maxIdleTime</dt>
* <dd>set the time in ms that a websocket may be idle before closing<br>
* <i>Default:</i></dd>
*
* <dt>maxTextMessagesSize</dt>
* <dd>set the size in characters that a websocket may be accept before closing<br>
* <i>Default:</i></dd>
*
* <dt>maxBinaryMessagesSize</dt>
* <dd>set the size in bytes that a websocket may be accept before closing<br>
* <i>Default:</i></dd>
* </dl>
*/ */
@SuppressWarnings("serial") @SuppressWarnings("serial")
public abstract class WebSocketServlet extends HttpServlet implements WebSocketServerFactory.Acceptor public abstract class WebSocketServlet extends HttpServlet implements WebSocketServer.Acceptor
{ {
private final Logger LOG = Log.getLogger(getClass()); private final Logger LOG = Log.getLogger(getClass());
private WebSocketServerFactory webSocketFactory; private WebSocketServerFactory webSocketFactory;
@ -81,22 +114,26 @@ public abstract class WebSocketServlet extends HttpServlet implements WebSocketS
{ {
String bs = getInitParameter("bufferSize"); String bs = getInitParameter("bufferSize");
WebSocketPolicy policy = new WebSocketPolicy(WebSocketBehavior.SERVER); WebSocketPolicy policy = new WebSocketPolicy(WebSocketBehavior.SERVER);
if(bs != null) { if (bs != null)
{
policy.setBufferSize(Integer.parseInt(bs)); policy.setBufferSize(Integer.parseInt(bs));
} }
String max = getInitParameter("maxIdleTime"); String max = getInitParameter("maxIdleTime");
if (max != null) { if (max != null)
{
policy.setMaxIdleTime(Integer.parseInt(max)); policy.setMaxIdleTime(Integer.parseInt(max));
} }
max = getInitParameter("maxTextMessageSize"); max = getInitParameter("maxTextMessageSize");
if (max != null) { if (max != null)
{
policy.setMaxTextMessageSize(Integer.parseInt(max)); policy.setMaxTextMessageSize(Integer.parseInt(max));
} }
max = getInitParameter("maxBinaryMessageSize"); max = getInitParameter("maxBinaryMessageSize");
if (max != null) { if (max != null)
{
policy.setMaxBinaryMessageSize(Integer.parseInt(max)); policy.setMaxBinaryMessageSize(Integer.parseInt(max));
} }

View File

@ -0,0 +1,29 @@
package org.eclipse.jetty.websocket.server.handshake;
import java.io.IOException;
import java.util.List;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import org.eclipse.jetty.websocket.extensions.Extension;
import org.eclipse.jetty.websocket.server.WebSocketServer;
/**
* WebSocket Handshake for spec <a href="https://tools.ietf.org/html/draft-hixie-thewebsocketprotocol-76">Hixie-76 Draft</a>.
* <p>
* Most often seen in use by Safari/OSX
*/
public class HandshakeHixie76 implements WebSocketServer.Handshake
{
/** draft-hixie-thewebsocketprotocol-76 - Sec-WebSocket-Draft */
public static final int VERSION = 0;
@Override
public void doHandshakeResponse(HttpServletRequest request, HttpServletResponse response, List<Extension> extensions, String acceptedSubProtocol)
throws IOException
{
// TODO: implement the Hixie76 handshake?
throw new IOException("Not implemented yet");
}
}

View File

@ -0,0 +1,50 @@
package org.eclipse.jetty.websocket.server.handshake;
import java.util.List;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import org.eclipse.jetty.websocket.api.AcceptHash;
import org.eclipse.jetty.websocket.extensions.Extension;
import org.eclipse.jetty.websocket.server.WebSocketServer;
/**
* WebSocket Handshake for <a href="https://tools.ietf.org/html/rfc6455">RFC 6455</a>.
*/
public class HandshakeRFC6455 implements WebSocketServer.Handshake
{
/** RFC 6455 - Sec-WebSocket-Version */
public static final int VERSION = 13;
@Override
public void doHandshakeResponse(HttpServletRequest request, HttpServletResponse response, List<Extension> extensions, String acceptedSubProtocol)
{
String key = request.getHeader("Sec-WebSocket-Key");
if (key == null)
{
throw new IllegalStateException("Missing request header 'Sec-WebSocket-Key'");
}
// build response
response.setHeader("Upgrade","WebSocket");
response.addHeader("Connection","Upgrade");
response.addHeader("Sec-WebSocket-Accept",AcceptHash.hashKey(key));
if (acceptedSubProtocol != null)
{
response.addHeader("Sec-WebSocket-Protocol",acceptedSubProtocol);
}
if (extensions != null)
{
for (Extension ext : extensions)
{
response.addHeader("Sec-WebSocket-Extensions",ext.getConfig().getParameterizedName());
}
}
response.setStatus(HttpServletResponse.SC_SWITCHING_PROTOCOLS);
}
}

View File

@ -11,29 +11,37 @@ import java.net.InetSocketAddress;
import java.net.Socket; import java.net.Socket;
import java.net.SocketAddress; import java.net.SocketAddress;
import java.net.URI; import java.net.URI;
import java.util.concurrent.TimeUnit; import java.nio.ByteBuffer;
import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletRequest;
import org.eclipse.jetty.io.ByteBufferPool;
import org.eclipse.jetty.io.StandardByteBufferPool;
import org.eclipse.jetty.server.SelectChannelConnector; import org.eclipse.jetty.server.SelectChannelConnector;
import org.eclipse.jetty.server.Server; import org.eclipse.jetty.server.Server;
import org.eclipse.jetty.servlet.ServletContextHandler; import org.eclipse.jetty.servlet.ServletContextHandler;
import org.eclipse.jetty.servlet.ServletHolder; import org.eclipse.jetty.servlet.ServletHolder;
import org.eclipse.jetty.util.BufferUtil;
import org.eclipse.jetty.util.IO; import org.eclipse.jetty.util.IO;
import org.eclipse.jetty.websocket.WebSocket; import org.eclipse.jetty.websocket.WebSocket;
import org.eclipse.jetty.websocket.WebSocketGeneratorRFC6455Test; import org.eclipse.jetty.websocket.WebSocketGeneratorRFC6455Test;
import org.eclipse.jetty.websocket.server.helper.MessageSender; import org.eclipse.jetty.websocket.api.StatusCode;
import org.eclipse.jetty.websocket.api.WebSocketPolicy;
import org.eclipse.jetty.websocket.frames.CloseFrame;
import org.eclipse.jetty.websocket.frames.TextFrame;
import org.eclipse.jetty.websocket.generator.Generator;
import org.eclipse.jetty.websocket.parser.Parser;
import org.eclipse.jetty.websocket.server.helper.FrameParseCapture;
import org.junit.AfterClass; import org.junit.AfterClass;
import org.junit.Assert; import org.junit.Assert;
import org.junit.BeforeClass; import org.junit.BeforeClass;
import org.junit.Test; import org.junit.Test;
/** /**
* Test various <a href="http://tools.ietf.org/html/rfc6455">RFC 6455</a> specified requirements placed on * Test various <a href="http://tools.ietf.org/html/rfc6455">RFC 6455</a> specified requirements placed on {@link WebSocketServlet}
* {@link WebSocketServlet}
* <p> * <p>
* This test serves a different purpose than than the {@link WebSocketGeneratorRFC6455Test}, * This test serves a different purpose than than the {@link WebSocketGeneratorRFC6455Test}, {@link WebSocketMessageRFC6455Test}, and
* {@link WebSocketMessageRFC6455Test}, and {@link WebSocketParserRFC6455Test} tests. * {@link WebSocketParserRFC6455Test} tests.
*/ */
public class WebSocketServletRFCTest public class WebSocketServletRFCTest
{ {
@ -64,6 +72,7 @@ public class WebSocketServletRFCTest
// trigger a WebSocket server terminated close. // trigger a WebSocket server terminated close.
if (data.equals("CRASH")) if (data.equals("CRASH"))
{ {
System.out.printf("Got OnTextMessage");
throw new RuntimeException("Something bad happened"); throw new RuntimeException("Something bad happened");
} }
@ -131,6 +140,14 @@ public class WebSocketServletRFCTest
} }
} }
private void read(InputStream in, ByteBuffer buf) throws IOException
{
while ((in.available() > 0) && (buf.remaining() > 0))
{
buf.put((byte)in.read());
}
}
private String readResponseHeader(InputStream in) throws IOException private String readResponseHeader(InputStream in) throws IOException
{ {
InputStreamReader isr = new InputStreamReader(in); InputStreamReader isr = new InputStreamReader(in);
@ -153,34 +170,79 @@ public class WebSocketServletRFCTest
} }
/** /**
* Test the requirement of responding with server terminated close code 1011 when there is an unhandled (internal * Test the requirement of responding with server terminated close code 1011 when there is an unhandled (internal server error) being produced by the
* server error) being produced by the extended WebSocketServlet. * extended WebSocketServlet.
*/ */
@Test @Test
public void testResponseOnInternalError() throws Exception public void testResponseOnInternalError() throws Exception
{ {
// WebSocketClientFactory clientFactory = new WebSocketClientFactory(); Socket socket = new Socket();
// clientFactory.start(); SocketAddress endpoint = new InetSocketAddress(serverUri.getHost(),serverUri.getPort());
socket.connect(endpoint);
// WebSocketClient wsc = clientFactory.newWebSocketClient(); // acting as client
MessageSender sender = new MessageSender(); WebSocketPolicy policy = WebSocketPolicy.newClientPolicy();
// wsc.open(serverUri,sender); ByteBufferPool bufferPool = new StandardByteBufferPool(policy.getBufferSize());
Generator generator = new Generator(bufferPool,policy);
Parser parser = new Parser(policy);
FrameParseCapture capture = new FrameParseCapture();
parser.addListener(capture);
StringBuilder req = new StringBuilder();
req.append("GET / HTTP/1.1\r\n");
req.append(String.format("Host: %s:%d\r\n",serverUri.getHost(),serverUri.getPort()));
req.append("Sec-WebSocket-Key: dGhlIHNhbXBsZSBub25jZQ==\r\n");
req.append("Upgrade: WebSocket\r\n");
req.append("Connection: Upgrade\r\n");
req.append("Sec-WebSocket-Version: 13\r\n"); // RFC 6455
req.append("\r\n");
OutputStream out = null;
InputStream in = null;
try try
{ {
sender.awaitConnect(); out = socket.getOutputStream();
in = socket.getInputStream();
sender.sendMessage("CRASH"); // Write request
out.write(req.toString().getBytes());
out.flush();
// Give servlet 500 millisecond to process messages // Read response header
TimeUnit.MILLISECONDS.sleep(500); String respHeader = readResponseHeader(in);
// System.out.println("RESPONSE: " + respHeader);
Assert.assertThat("WebSocket should be closed",sender.isConnected(),is(false)); Assert.assertThat("Response Code",respHeader,startsWith("HTTP/1.1 101 Switching Protocols"));
Assert.assertThat("WebSocket close clode",sender.getCloseCode(),is(1011)); Assert.assertThat("Response Header Upgrade",respHeader,containsString("Upgrade: WebSocket\r\n"));
// Assert.assertThat("Response Header Connection",respHeader,containsString("Connection: Upgrade\r\n"));
// Generate text frame
TextFrame txt = new TextFrame("CRASH");
ByteBuffer txtbuf = generator.generate(txt);
txtbuf.flip();
// Write Text Frame
BufferUtil.writeTo(txtbuf,out);
// Read frame (hopefully close frame)
ByteBuffer closeFrame = ByteBuffer.allocate(20);
System.out.println("Reading from in");
read(in,closeFrame);
// Parse Frame
parser.parse(closeFrame);
capture.assertNoErrors();
capture.assertHasFrame(CloseFrame.class,1);
CloseFrame cf = (CloseFrame)capture.getFrames().get(0);
Assert.assertThat("Close Frame.status code",cf.getStatusCode(),is(StatusCode.SERVER_ERROR));
} }
finally finally
{ {
sender.close(); IO.close(in);
IO.close(out);
socket.close();
} }
} }
@ -190,9 +252,6 @@ public class WebSocketServletRFCTest
@Test @Test
public void testResponseOnInvalidVersion() throws Exception public void testResponseOnInvalidVersion() throws Exception
{ {
// Using straight Socket to accomplish this as jetty's WebSocketClient
// doesn't allow the use of invalid versions. (obviously)
Socket socket = new Socket(); Socket socket = new Socket();
SocketAddress endpoint = new InetSocketAddress(serverUri.getHost(),serverUri.getPort()); SocketAddress endpoint = new InetSocketAddress(serverUri.getHost(),serverUri.getPort());
socket.connect(endpoint); socket.connect(endpoint);
@ -221,7 +280,7 @@ public class WebSocketServletRFCTest
// System.out.println("RESPONSE: " + respHeader); // System.out.println("RESPONSE: " + respHeader);
Assert.assertThat("Response Code",respHeader,startsWith("HTTP/1.1 400 Unsupported websocket version specification")); Assert.assertThat("Response Code",respHeader,startsWith("HTTP/1.1 400 Unsupported websocket version specification"));
Assert.assertThat("Response Header Versions",respHeader,containsString("Sec-WebSocket-Version: 13, 8, 6, 0\r\n")); Assert.assertThat("Response Header Versions",respHeader,containsString("Sec-WebSocket-Version: 13, 0\r\n"));
} }
finally finally
{ {

View File

@ -0,0 +1,19 @@
package org.eclipse.jetty.websocket.server.examples;
import javax.servlet.http.HttpServletRequest;
import org.eclipse.jetty.websocket.WebSocket;
import org.eclipse.jetty.websocket.server.WebSocketServlet;
/**
* Example servlet for most basic form.
*/
@SuppressWarnings("serial")
public class MyEchoServlet extends WebSocketServlet
{
@Override
public WebSocket doWebSocketConnect(HttpServletRequest request, String protocol)
{
return new MyEchoSocket();
}
}

View File

@ -0,0 +1,36 @@
package org.eclipse.jetty.websocket.server.examples;
import java.io.IOException;
import org.eclipse.jetty.websocket.WebSocket;
public class MyEchoSocket implements WebSocket, WebSocket.OnTextMessage
{
private Connection conn;
@Override
public void onClose(int closeCode, String message)
{
/* do nothing */
}
@Override
public void onMessage(String data)
{
try
{
// echo the data back
conn.sendMessage(data);
}
catch (IOException e)
{
e.printStackTrace();
}
}
@Override
public void onOpen(Connection connection)
{
this.conn = connection;
}
}

View File

@ -0,0 +1,94 @@
package org.eclipse.jetty.websocket.server.helper;
import static org.hamcrest.Matchers.*;
import java.util.ArrayList;
import java.util.List;
import org.eclipse.jetty.util.log.Log;
import org.eclipse.jetty.util.log.Logger;
import org.eclipse.jetty.websocket.api.WebSocketException;
import org.eclipse.jetty.websocket.frames.BaseFrame;
import org.eclipse.jetty.websocket.parser.Parser;
import org.junit.Assert;
public class FrameParseCapture implements Parser.Listener
{
private static final Logger LOG = Log.getLogger(FrameParseCapture.class);
private List<BaseFrame> frames = new ArrayList<>();
private List<WebSocketException> errors = new ArrayList<>();
public void assertHasErrors(Class<? extends WebSocketException> errorType, int expectedCount)
{
Assert.assertThat(errorType.getSimpleName(),getErrorCount(errorType),is(expectedCount));
}
public void assertHasFrame(Class<? extends BaseFrame> frameType)
{
Assert.assertThat(frameType.getSimpleName(),getFrameCount(frameType),greaterThanOrEqualTo(1));
}
public void assertHasFrame(Class<? extends BaseFrame> frameType, int expectedCount)
{
Assert.assertThat(frameType.getSimpleName(),getFrameCount(frameType),is(expectedCount));
}
public void assertHasNoFrames()
{
Assert.assertThat("Has no frames",frames.size(),is(0));
}
public void assertNoErrors()
{
Assert.assertThat("Has no errors",errors.size(),is(0));
}
public int getErrorCount(Class<? extends WebSocketException> errorType)
{
int count = 0;
for (WebSocketException error : errors)
{
if (errorType.isInstance(error))
{
count++;
}
}
return count;
}
public List<WebSocketException> getErrors()
{
return errors;
}
public int getFrameCount(Class<? extends BaseFrame> frameType)
{
int count = 0;
for (BaseFrame frame : frames)
{
if (frameType.isInstance(frame))
{
count++;
}
}
return count;
}
public List<BaseFrame> getFrames()
{
return frames;
}
@Override
public void onFrame(BaseFrame frame)
{
frames.add(frame);
}
@Override
public void onWebSocketException(WebSocketException e)
{
LOG.warn(e);
errors.add(e);
}
}