Working towards a happy websocket-server connection + handshake
This commit is contained in:
parent
e919c0dff8
commit
970d0764f7
|
@ -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));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -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();
|
||||||
|
}
|
||||||
|
}
|
|
@ -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;
|
||||||
|
|
||||||
|
|
|
@ -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 */
|
|
||||||
}
|
|
|
@ -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
|
||||||
|
}
|
||||||
|
}
|
|
@ -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;
|
||||||
|
}
|
||||||
|
}
|
|
@ -15,87 +15,101 @@ 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")))
|
||||||
{
|
{
|
||||||
String origin = request.getHeader("Origin");
|
String origin = request.getHeader("Origin");
|
||||||
if (origin==null)
|
if (origin == null)
|
||||||
{
|
{
|
||||||
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,15 +118,14 @@ 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())
|
||||||
{
|
{
|
||||||
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,9 +138,9 @@ 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)
|
||||||
{
|
{
|
||||||
response.sendError(HttpServletResponse.SC_SERVICE_UNAVAILABLE);
|
response.sendError(HttpServletResponse.SC_SERVICE_UNAVAILABLE);
|
||||||
return false;
|
return false;
|
||||||
|
@ -135,38 +148,38 @@ public class WebSocketServerFactory extends AbstractLifeCycle
|
||||||
}
|
}
|
||||||
|
|
||||||
// Send the upgrade
|
// Send the upgrade
|
||||||
upgrade(request, response, websocket, protocol);
|
upgrade(request,response,websocket,protocol);
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
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();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
protected void doStop() throws Exception
|
protected void doStop() throws Exception
|
||||||
{
|
{
|
||||||
closeConnections();
|
closeConnections();
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @return A modifiable map of extension name to extension class
|
* @return A modifiable map of extension name to extension class
|
||||||
*/
|
*/
|
||||||
public Map<String,Class<? extends Extension>> getExtensionClassesMap()
|
public Map<String, Class<? extends Extension>> getExtensionClassesMap()
|
||||||
{
|
{
|
||||||
return _extensionClasses;
|
return extensionClasses;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -179,34 +192,22 @@ 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)
|
|
||||||
|
for (ExtensionConfig cfg : requested)
|
||||||
{
|
{
|
||||||
QuotedStringTokenizer tok = new QuotedStringTokenizer(rExt,";");
|
Extension extension = newExtension(cfg.getName());
|
||||||
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);
|
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,8 +217,8 @@ 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,16 +235,18 @@ 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];
|
||||||
System.arraycopy(passed, 0, protocols, 0, passed.length);
|
System.arraycopy(passed,0,protocols,0,passed.length);
|
||||||
return protocols;
|
return protocols;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -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);
|
||||||
{
|
// Per RFC 6455 - 4.4 - Supporting Multiple Versions of WebSocket Protocol
|
||||||
// List<Extension> extensions = initExtensions(extensions_requested,
|
// Using the examples as outlined
|
||||||
// 8 - WebSocketConnectionRFC6455.OP_EXT_DATA,
|
response.setHeader("Sec-WebSocket-Version",supportedVersions);
|
||||||
// 16 - WebSocketConnectionRFC6455.OP_EXT_CTRL,
|
response.sendError(HttpStatus.BAD_REQUEST_400,"Unsupported websocket version specification");
|
||||||
// 3);
|
return;
|
||||||
// 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
|
|
||||||
// Using the examples as outlined
|
|
||||||
response.setHeader("Sec-WebSocket-Version", "" + org.eclipse.jetty.websocket.api.WebSocket.VERSION /*+ ", 0"*/);
|
|
||||||
response.setStatus(HttpStatus.BAD_REQUEST_400);
|
|
||||||
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?
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -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
|
||||||
|
* {
|
||||||
|
* @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));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -118,6 +155,6 @@ public abstract class WebSocketServlet extends HttpServlet implements WebSocketS
|
||||||
{
|
{
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
super.service(request, response);
|
super.service(request,response);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -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");
|
||||||
|
}
|
||||||
|
}
|
|
@ -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);
|
||||||
|
}
|
||||||
|
}
|
|
@ -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
|
||||||
{
|
{
|
||||||
|
|
|
@ -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();
|
||||||
|
}
|
||||||
|
}
|
|
@ -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;
|
||||||
|
}
|
||||||
|
}
|
|
@ -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);
|
||||||
|
}
|
||||||
|
}
|
Loading…
Reference in New Issue