Issue #3478 - changes from review
ExtensionStack.negotiate now differentiates between incorrect extension config offered or incorrect config from negotiation adding more BadMessageException cases Signed-off-by: lachan-roberts <lachlan@webtide.com>
This commit is contained in:
parent
eb4f7c000f
commit
0d570dd6b6
|
@ -156,18 +156,10 @@ public class JavaxWebSocketClientContainer extends JavaxWebSocketContainer imple
|
||||||
upgradeRequest.addListener(jsrUpgradeListener);
|
upgradeRequest.addListener(jsrUpgradeListener);
|
||||||
|
|
||||||
for (Extension ext : clientEndpointConfig.getExtensions())
|
for (Extension ext : clientEndpointConfig.getExtensions())
|
||||||
{
|
|
||||||
if (!getExtensionRegistry().isAvailable(ext.getName()))
|
|
||||||
{
|
|
||||||
throw new IllegalArgumentException("Requested extension [" + ext.getName() + "] is not installed");
|
|
||||||
}
|
|
||||||
upgradeRequest.addExtensions(new JavaxWebSocketExtensionConfig(ext));
|
upgradeRequest.addExtensions(new JavaxWebSocketExtensionConfig(ext));
|
||||||
}
|
|
||||||
|
|
||||||
if (clientEndpointConfig.getPreferredSubprotocols().size() > 0)
|
if (clientEndpointConfig.getPreferredSubprotocols().size() > 0)
|
||||||
{
|
|
||||||
upgradeRequest.setSubProtocols(clientEndpointConfig.getPreferredSubprotocols());
|
upgradeRequest.setSubProtocols(clientEndpointConfig.getPreferredSubprotocols());
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
try
|
try
|
||||||
|
|
|
@ -20,7 +20,7 @@
|
||||||
# org.eclipse.jetty.util.log.class=org.eclipse.jetty.util.log.Slf4jLog
|
# org.eclipse.jetty.util.log.class=org.eclipse.jetty.util.log.Slf4jLog
|
||||||
org.eclipse.jetty.util.log.class=org.eclipse.jetty.util.log.StdErrLog
|
org.eclipse.jetty.util.log.class=org.eclipse.jetty.util.log.StdErrLog
|
||||||
org.eclipse.jetty.LEVEL=WARN
|
org.eclipse.jetty.LEVEL=WARN
|
||||||
org.eclipse.jetty.websocket.tests.LEVEL=DEBUG
|
# org.eclipse.jetty.websocket.tests.LEVEL=DEBUG
|
||||||
# org.eclipse.jetty.util.log.stderr.LONG=true
|
# org.eclipse.jetty.util.log.stderr.LONG=true
|
||||||
# org.eclipse.jetty.server.AbstractConnector.LEVEL=DEBUG
|
# org.eclipse.jetty.server.AbstractConnector.LEVEL=DEBUG
|
||||||
# org.eclipse.jetty.io.WriteFlusher.LEVEL=DEBUG
|
# org.eclipse.jetty.io.WriteFlusher.LEVEL=DEBUG
|
||||||
|
|
|
@ -260,7 +260,7 @@ public abstract class ClientUpgradeRequest extends HttpRequest implements Respon
|
||||||
throw new HttpResponseException("Invalid Sec-WebSocket-Accept hash (was:" + respHash + ", expected:" + expectedHash + ")", response);
|
throw new HttpResponseException("Invalid Sec-WebSocket-Accept hash (was:" + respHash + ", expected:" + expectedHash + ")", response);
|
||||||
|
|
||||||
// Parse the Negotiated Extensions
|
// Parse the Negotiated Extensions
|
||||||
List<ExtensionConfig> extensions = new ArrayList<>();
|
List<ExtensionConfig> negotiatedExtensions = new ArrayList<>();
|
||||||
HttpField extField = response.getHeaders().getField(HttpHeader.SEC_WEBSOCKET_EXTENSIONS);
|
HttpField extField = response.getHeaders().getField(HttpHeader.SEC_WEBSOCKET_EXTENSIONS);
|
||||||
if (extField != null)
|
if (extField != null)
|
||||||
{
|
{
|
||||||
|
@ -272,7 +272,7 @@ public abstract class ClientUpgradeRequest extends HttpRequest implements Respon
|
||||||
QuotedStringTokenizer tok = new QuotedStringTokenizer(extVal, ",");
|
QuotedStringTokenizer tok = new QuotedStringTokenizer(extVal, ",");
|
||||||
while (tok.hasMoreTokens())
|
while (tok.hasMoreTokens())
|
||||||
{
|
{
|
||||||
extensions.add(ExtensionConfig.parse(tok.nextToken()));
|
negotiatedExtensions.add(ExtensionConfig.parse(tok.nextToken()));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -280,7 +280,7 @@ public abstract class ClientUpgradeRequest extends HttpRequest implements Respon
|
||||||
|
|
||||||
// Verify the Negotiated Extensions
|
// Verify the Negotiated Extensions
|
||||||
List<ExtensionConfig> offeredExtensions = getExtensions();
|
List<ExtensionConfig> offeredExtensions = getExtensions();
|
||||||
for (ExtensionConfig config : extensions)
|
for (ExtensionConfig config : negotiatedExtensions)
|
||||||
{
|
{
|
||||||
if (config.getName().startsWith("@"))
|
if (config.getName().startsWith("@"))
|
||||||
continue;
|
continue;
|
||||||
|
@ -289,7 +289,7 @@ public abstract class ClientUpgradeRequest extends HttpRequest implements Respon
|
||||||
if (numMatch < 1)
|
if (numMatch < 1)
|
||||||
throw new WebSocketException("Upgrade failed: Sec-WebSocket-Extensions contained extension not requested");
|
throw new WebSocketException("Upgrade failed: Sec-WebSocket-Extensions contained extension not requested");
|
||||||
|
|
||||||
numMatch = extensions.stream().filter(c -> config.getName().equalsIgnoreCase(c.getName())).count();
|
numMatch = negotiatedExtensions.stream().filter(c -> config.getName().equalsIgnoreCase(c.getName())).count();
|
||||||
if (numMatch > 1)
|
if (numMatch > 1)
|
||||||
throw new WebSocketException("Upgrade failed: Sec-WebSocket-Extensions contained more than one extension of the same name");
|
throw new WebSocketException("Upgrade failed: Sec-WebSocket-Extensions contained more than one extension of the same name");
|
||||||
}
|
}
|
||||||
|
@ -297,7 +297,7 @@ public abstract class ClientUpgradeRequest extends HttpRequest implements Respon
|
||||||
// Negotiate the extension stack
|
// Negotiate the extension stack
|
||||||
HttpClient httpClient = wsClient.getHttpClient();
|
HttpClient httpClient = wsClient.getHttpClient();
|
||||||
ExtensionStack extensionStack = new ExtensionStack(wsClient.getExtensionRegistry());
|
ExtensionStack extensionStack = new ExtensionStack(wsClient.getExtensionRegistry());
|
||||||
extensionStack.negotiate(wsClient.getObjectFactory(), httpClient.getByteBufferPool(), extensions);
|
extensionStack.negotiate(wsClient.getObjectFactory(), httpClient.getByteBufferPool(), offeredExtensions, negotiatedExtensions);
|
||||||
|
|
||||||
// Get the negotiated subprotocol
|
// Get the negotiated subprotocol
|
||||||
String negotiatedSubProtocol = null;
|
String negotiatedSubProtocol = null;
|
||||||
|
|
|
@ -87,11 +87,7 @@ public class WebSocketCoreClient extends ContainerLifeCycle implements FrameHand
|
||||||
public CompletableFuture<FrameHandler.CoreSession> connect(ClientUpgradeRequest request) throws IOException
|
public CompletableFuture<FrameHandler.CoreSession> connect(ClientUpgradeRequest request) throws IOException
|
||||||
{
|
{
|
||||||
if (!isStarted())
|
if (!isStarted())
|
||||||
{
|
|
||||||
throw new IllegalStateException(WebSocketCoreClient.class.getSimpleName() + "@" + this.hashCode() + " is not started");
|
throw new IllegalStateException(WebSocketCoreClient.class.getSimpleName() + "@" + this.hashCode() + " is not started");
|
||||||
}
|
|
||||||
|
|
||||||
// TODO: add HttpClient delayed/on-demand start - See Issue #1516
|
|
||||||
|
|
||||||
// Validate Requested Extensions
|
// Validate Requested Extensions
|
||||||
for (ExtensionConfig reqExt : request.getExtensions())
|
for (ExtensionConfig reqExt : request.getExtensions())
|
||||||
|
|
|
@ -25,6 +25,7 @@ import java.util.List;
|
||||||
import java.util.ListIterator;
|
import java.util.ListIterator;
|
||||||
import java.util.stream.Collectors;
|
import java.util.stream.Collectors;
|
||||||
|
|
||||||
|
import org.eclipse.jetty.http.BadMessageException;
|
||||||
import org.eclipse.jetty.io.ByteBufferPool;
|
import org.eclipse.jetty.io.ByteBufferPool;
|
||||||
import org.eclipse.jetty.util.Callback;
|
import org.eclipse.jetty.util.Callback;
|
||||||
import org.eclipse.jetty.util.DecoratedObjectFactory;
|
import org.eclipse.jetty.util.DecoratedObjectFactory;
|
||||||
|
@ -38,6 +39,7 @@ import org.eclipse.jetty.websocket.core.ExtensionConfig;
|
||||||
import org.eclipse.jetty.websocket.core.Frame;
|
import org.eclipse.jetty.websocket.core.Frame;
|
||||||
import org.eclipse.jetty.websocket.core.IncomingFrames;
|
import org.eclipse.jetty.websocket.core.IncomingFrames;
|
||||||
import org.eclipse.jetty.websocket.core.OutgoingFrames;
|
import org.eclipse.jetty.websocket.core.OutgoingFrames;
|
||||||
|
import org.eclipse.jetty.websocket.core.WebSocketException;
|
||||||
import org.eclipse.jetty.websocket.core.WebSocketExtensionRegistry;
|
import org.eclipse.jetty.websocket.core.WebSocketExtensionRegistry;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -107,20 +109,37 @@ public class ExtensionStack implements IncomingFrames, OutgoingFrames, Dumpable
|
||||||
* <p>
|
* <p>
|
||||||
* For the list of negotiated extensions, use {@link #getNegotiatedExtensions()}
|
* For the list of negotiated extensions, use {@link #getNegotiatedExtensions()}
|
||||||
*
|
*
|
||||||
* @param configs the configurations being requested
|
* @param offeredConfigs the configurations being requested by the client
|
||||||
|
* @param negotiatedConfigs the configurations accepted by the server
|
||||||
*/
|
*/
|
||||||
public void negotiate(DecoratedObjectFactory objectFactory, ByteBufferPool bufferPool, List<ExtensionConfig> configs)
|
public void negotiate(DecoratedObjectFactory objectFactory, ByteBufferPool bufferPool, List<ExtensionConfig> offeredConfigs, List<ExtensionConfig> negotiatedConfigs)
|
||||||
{
|
{
|
||||||
if (LOG.isDebugEnabled())
|
if (LOG.isDebugEnabled())
|
||||||
LOG.debug("Extension Configs={}", configs);
|
LOG.debug("Extension Configs={}", negotiatedConfigs);
|
||||||
|
|
||||||
this.extensions = new ArrayList<>();
|
this.extensions = new ArrayList<>();
|
||||||
|
|
||||||
String rsvClaims[] = new String[3];
|
String rsvClaims[] = new String[3];
|
||||||
|
|
||||||
for (ExtensionConfig config : configs)
|
for (ExtensionConfig config : negotiatedConfigs)
|
||||||
{
|
{
|
||||||
Extension ext = factory.newInstance(objectFactory, bufferPool, config);
|
Extension ext;
|
||||||
|
|
||||||
|
try
|
||||||
|
{
|
||||||
|
ext = factory.newInstance(objectFactory, bufferPool, config);
|
||||||
|
}
|
||||||
|
catch (Throwable t)
|
||||||
|
{
|
||||||
|
for (ExtensionConfig offered : offeredConfigs)
|
||||||
|
{
|
||||||
|
if (offered.getParameterizedName().equals(config.getParameterizedName()))
|
||||||
|
throw new BadMessageException("offered extension had bad parameters: ", t);
|
||||||
|
}
|
||||||
|
|
||||||
|
throw new WebSocketException("negotiated extension had bad parameters: ", t);
|
||||||
|
}
|
||||||
|
|
||||||
if (ext == null)
|
if (ext == null)
|
||||||
{
|
{
|
||||||
// Extension not present on this side
|
// Extension not present on this side
|
||||||
|
|
|
@ -27,6 +27,7 @@ import java.util.stream.Collectors;
|
||||||
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.BadMessageException;
|
||||||
import org.eclipse.jetty.http.HttpField;
|
import org.eclipse.jetty.http.HttpField;
|
||||||
import org.eclipse.jetty.http.HttpHeader;
|
import org.eclipse.jetty.http.HttpHeader;
|
||||||
import org.eclipse.jetty.http.QuotedCSV;
|
import org.eclipse.jetty.http.QuotedCSV;
|
||||||
|
@ -55,13 +56,16 @@ public class Negotiation
|
||||||
private String subprotocol;
|
private String subprotocol;
|
||||||
private ExtensionStack extensionStack;
|
private ExtensionStack extensionStack;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @throws BadMessageException if there is any errors parsing the upgrade request
|
||||||
|
*/
|
||||||
public Negotiation(
|
public Negotiation(
|
||||||
Request baseRequest,
|
Request baseRequest,
|
||||||
HttpServletRequest request,
|
HttpServletRequest request,
|
||||||
HttpServletResponse response,
|
HttpServletResponse response,
|
||||||
WebSocketExtensionRegistry registry,
|
WebSocketExtensionRegistry registry,
|
||||||
DecoratedObjectFactory objectFactory,
|
DecoratedObjectFactory objectFactory,
|
||||||
ByteBufferPool bufferPool)
|
ByteBufferPool bufferPool) throws BadMessageException
|
||||||
{
|
{
|
||||||
this.baseRequest = baseRequest;
|
this.baseRequest = baseRequest;
|
||||||
this.request = request;
|
this.request = request;
|
||||||
|
@ -77,73 +81,80 @@ public class Negotiation
|
||||||
QuotedCSV extensions = null;
|
QuotedCSV extensions = null;
|
||||||
QuotedCSV subprotocols = null;
|
QuotedCSV subprotocols = null;
|
||||||
|
|
||||||
for (HttpField field : baseRequest.getHttpFields())
|
try
|
||||||
{
|
{
|
||||||
if (field.getHeader() != null)
|
for (HttpField field : baseRequest.getHttpFields())
|
||||||
{
|
{
|
||||||
switch (field.getHeader())
|
if (field.getHeader() != null)
|
||||||
{
|
{
|
||||||
case UPGRADE:
|
switch (field.getHeader())
|
||||||
if (upgrade == null && "websocket".equalsIgnoreCase(field.getValue()))
|
{
|
||||||
upgrade = Boolean.TRUE;
|
case UPGRADE:
|
||||||
break;
|
if (upgrade == null && "websocket".equalsIgnoreCase(field.getValue()))
|
||||||
|
upgrade = Boolean.TRUE;
|
||||||
|
break;
|
||||||
|
|
||||||
case CONNECTION:
|
case CONNECTION:
|
||||||
if (connectionCSVs == null)
|
if (connectionCSVs == null)
|
||||||
connectionCSVs = new QuotedCSV();
|
connectionCSVs = new QuotedCSV();
|
||||||
connectionCSVs.addValue(field.getValue());
|
connectionCSVs.addValue(field.getValue());
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case SEC_WEBSOCKET_KEY:
|
case SEC_WEBSOCKET_KEY:
|
||||||
key = field.getValue();
|
key = field.getValue();
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case SEC_WEBSOCKET_VERSION:
|
case SEC_WEBSOCKET_VERSION:
|
||||||
version = field.getValue();
|
version = field.getValue();
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case SEC_WEBSOCKET_EXTENSIONS:
|
case SEC_WEBSOCKET_EXTENSIONS:
|
||||||
if (extensions == null)
|
if (extensions == null)
|
||||||
extensions = new QuotedCSV(field.getValue());
|
extensions = new QuotedCSV(field.getValue());
|
||||||
else
|
else
|
||||||
extensions.addValue(field.getValue());
|
extensions.addValue(field.getValue());
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case SEC_WEBSOCKET_SUBPROTOCOL:
|
case SEC_WEBSOCKET_SUBPROTOCOL:
|
||||||
if (subprotocols == null)
|
if (subprotocols == null)
|
||||||
subprotocols = new QuotedCSV(field.getValue());
|
subprotocols = new QuotedCSV(field.getValue());
|
||||||
else
|
else
|
||||||
subprotocols.addValue(field.getValue());
|
subprotocols.addValue(field.getValue());
|
||||||
break;
|
break;
|
||||||
|
|
||||||
default:
|
default:
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
this.version = version;
|
||||||
|
this.key = key;
|
||||||
|
this.upgrade = upgrade != null && connectionCSVs != null && connectionCSVs.getValues().stream().anyMatch(s -> s.equalsIgnoreCase("Upgrade"));
|
||||||
|
|
||||||
|
Set<String> available = registry.getAvailableExtensionNames();
|
||||||
|
offeredExtensions = extensions == null
|
||||||
|
? Collections.emptyList()
|
||||||
|
: extensions.getValues().stream()
|
||||||
|
.map(ExtensionConfig::parse)
|
||||||
|
.filter(ec -> available.contains(ec.getName().toLowerCase()) && !ec.getName().startsWith("@"))
|
||||||
|
.collect(Collectors.toList());
|
||||||
|
|
||||||
|
offeredSubprotocols = subprotocols == null
|
||||||
|
? Collections.emptyList()
|
||||||
|
: subprotocols.getValues();
|
||||||
|
|
||||||
|
negotiatedExtensions = new ArrayList<>();
|
||||||
|
for (ExtensionConfig config : offeredExtensions)
|
||||||
|
{
|
||||||
|
long matches = negotiatedExtensions.stream()
|
||||||
|
.filter(negotiatedConfig -> negotiatedConfig.getName().equals(config.getName())).count();
|
||||||
|
if (matches == 0)
|
||||||
|
negotiatedExtensions.add(config);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
catch (Throwable t)
|
||||||
this.version = version;
|
|
||||||
this.key = key;
|
|
||||||
this.upgrade = upgrade != null && connectionCSVs != null && connectionCSVs.getValues().stream().anyMatch(s -> s.equalsIgnoreCase("Upgrade"));
|
|
||||||
|
|
||||||
Set<String> available = registry.getAvailableExtensionNames();
|
|
||||||
offeredExtensions = extensions == null
|
|
||||||
?Collections.emptyList()
|
|
||||||
:extensions.getValues().stream()
|
|
||||||
.map(ExtensionConfig::parse)
|
|
||||||
.filter(ec -> available.contains(ec.getName().toLowerCase()) && !ec.getName().startsWith("@"))
|
|
||||||
.collect(Collectors.toList());
|
|
||||||
|
|
||||||
offeredSubprotocols = subprotocols == null
|
|
||||||
?Collections.emptyList()
|
|
||||||
:subprotocols.getValues();
|
|
||||||
|
|
||||||
negotiatedExtensions = new ArrayList<>();
|
|
||||||
for (ExtensionConfig config : offeredExtensions)
|
|
||||||
{
|
{
|
||||||
long matches = negotiatedExtensions.stream()
|
throw new BadMessageException("Invalid Handshake Request", t);
|
||||||
.filter(negotiatedConfig->negotiatedConfig.getName().equals(config.getName())).count();
|
|
||||||
if (matches == 0)
|
|
||||||
negotiatedExtensions.add(config);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -217,7 +228,7 @@ public class Negotiation
|
||||||
{
|
{
|
||||||
// Extension stack can decide to drop any of these extensions or their parameters
|
// Extension stack can decide to drop any of these extensions or their parameters
|
||||||
extensionStack = new ExtensionStack(registry);
|
extensionStack = new ExtensionStack(registry);
|
||||||
extensionStack.negotiate(objectFactory, bufferPool, negotiatedExtensions);
|
extensionStack.negotiate(objectFactory, bufferPool, offeredExtensions, negotiatedExtensions);
|
||||||
negotiatedExtensions = extensionStack.getNegotiatedExtensions();
|
negotiatedExtensions = extensionStack.getNegotiatedExtensions();
|
||||||
|
|
||||||
if (extensionStack.hasNegotiatedExtensions())
|
if (extensionStack.hasNegotiatedExtensions())
|
||||||
|
|
|
@ -24,6 +24,7 @@ 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.BadMessageException;
|
||||||
import org.eclipse.jetty.http.HttpField;
|
import org.eclipse.jetty.http.HttpField;
|
||||||
import org.eclipse.jetty.http.HttpHeader;
|
import org.eclipse.jetty.http.HttpHeader;
|
||||||
import org.eclipse.jetty.http.HttpMethod;
|
import org.eclipse.jetty.http.HttpMethod;
|
||||||
|
@ -47,6 +48,7 @@ import org.eclipse.jetty.websocket.core.ExtensionConfig;
|
||||||
import org.eclipse.jetty.websocket.core.FrameHandler;
|
import org.eclipse.jetty.websocket.core.FrameHandler;
|
||||||
import org.eclipse.jetty.websocket.core.WebSocketConstants;
|
import org.eclipse.jetty.websocket.core.WebSocketConstants;
|
||||||
import org.eclipse.jetty.websocket.core.WebSocketException;
|
import org.eclipse.jetty.websocket.core.WebSocketException;
|
||||||
|
import org.eclipse.jetty.websocket.core.internal.ExtensionStack;
|
||||||
import org.eclipse.jetty.websocket.core.internal.Negotiated;
|
import org.eclipse.jetty.websocket.core.internal.Negotiated;
|
||||||
import org.eclipse.jetty.websocket.core.internal.WebSocketChannel;
|
import org.eclipse.jetty.websocket.core.internal.WebSocketChannel;
|
||||||
import org.eclipse.jetty.websocket.core.internal.WebSocketConnection;
|
import org.eclipse.jetty.websocket.core.internal.WebSocketConnection;
|
||||||
|
@ -119,11 +121,7 @@ public final class RFC6455Handshaker implements Handshaker
|
||||||
}
|
}
|
||||||
|
|
||||||
if (negotiation.getKey() == null)
|
if (negotiation.getKey() == null)
|
||||||
{
|
throw new BadMessageException("not upgraded no key");
|
||||||
if (LOG.isDebugEnabled())
|
|
||||||
LOG.debug("not upgraded no key {}", baseRequest);
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Negotiate the FrameHandler
|
// Negotiate the FrameHandler
|
||||||
FrameHandler handler = negotiator.negotiate(negotiation);
|
FrameHandler handler = negotiator.negotiate(negotiation);
|
||||||
|
@ -150,7 +148,8 @@ public final class RFC6455Handshaker implements Handshaker
|
||||||
// Check for handler
|
// Check for handler
|
||||||
if (handler == null)
|
if (handler == null)
|
||||||
{
|
{
|
||||||
LOG.warn("not upgraded: no frame handler provided {}", baseRequest);
|
if (LOG.isDebugEnabled())
|
||||||
|
LOG.debug("not upgraded: no frame handler provided {}", baseRequest);
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -182,11 +181,14 @@ public final class RFC6455Handshaker implements Handshaker
|
||||||
throw new WebSocketException("Upgrade failed: multiple negotiated extensions of the same name");
|
throw new WebSocketException("Upgrade failed: multiple negotiated extensions of the same name");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Create and Negotiate the ExtensionStack
|
||||||
|
ExtensionStack extensionStack = negotiation.getExtensionStack();
|
||||||
|
|
||||||
Negotiated negotiated = new Negotiated(
|
Negotiated negotiated = new Negotiated(
|
||||||
baseRequest.getHttpURI().toURI(),
|
baseRequest.getHttpURI().toURI(),
|
||||||
subprotocol,
|
subprotocol,
|
||||||
baseRequest.isSecure(),
|
baseRequest.isSecure(),
|
||||||
negotiation.getExtensionStack(),
|
extensionStack,
|
||||||
WebSocketConstants.SPEC_VERSION_STRING);
|
WebSocketConstants.SPEC_VERSION_STRING);
|
||||||
|
|
||||||
// Create the Channel
|
// Create the Channel
|
||||||
|
@ -203,10 +205,7 @@ public final class RFC6455Handshaker implements Handshaker
|
||||||
if (LOG.isDebugEnabled())
|
if (LOG.isDebugEnabled())
|
||||||
LOG.debug("connection {}", connection);
|
LOG.debug("connection {}", connection);
|
||||||
if (connection == null)
|
if (connection == null)
|
||||||
{
|
throw new WebSocketException("not upgraded: no connection");
|
||||||
LOG.warn("not upgraded: no connection {}", baseRequest);
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
for (Connection.Listener listener : connector.getBeans(Connection.Listener.class))
|
for (Connection.Listener listener : connector.getBeans(Connection.Listener.class))
|
||||||
connection.addListener(listener);
|
connection.addListener(listener);
|
||||||
|
|
|
@ -18,6 +18,10 @@
|
||||||
|
|
||||||
package org.eclipse.jetty.websocket.core;
|
package org.eclipse.jetty.websocket.core;
|
||||||
|
|
||||||
|
import java.nio.ByteBuffer;
|
||||||
|
import java.util.LinkedList;
|
||||||
|
import java.util.stream.Stream;
|
||||||
|
|
||||||
import org.eclipse.jetty.io.ByteBufferPool;
|
import org.eclipse.jetty.io.ByteBufferPool;
|
||||||
import org.eclipse.jetty.io.MappedByteBufferPool;
|
import org.eclipse.jetty.io.MappedByteBufferPool;
|
||||||
import org.eclipse.jetty.util.DecoratedObjectFactory;
|
import org.eclipse.jetty.util.DecoratedObjectFactory;
|
||||||
|
@ -29,10 +33,6 @@ import org.junit.jupiter.params.ParameterizedTest;
|
||||||
import org.junit.jupiter.params.provider.Arguments;
|
import org.junit.jupiter.params.provider.Arguments;
|
||||||
import org.junit.jupiter.params.provider.MethodSource;
|
import org.junit.jupiter.params.provider.MethodSource;
|
||||||
|
|
||||||
import java.nio.ByteBuffer;
|
|
||||||
import java.util.LinkedList;
|
|
||||||
import java.util.stream.Stream;
|
|
||||||
|
|
||||||
import static org.junit.jupiter.api.Assertions.assertThrows;
|
import static org.junit.jupiter.api.Assertions.assertThrows;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -64,7 +64,7 @@ public class GeneratorFrameFlagsTest
|
||||||
public void setup(Frame invalidFrame)
|
public void setup(Frame invalidFrame)
|
||||||
{
|
{
|
||||||
ExtensionStack exStack = new ExtensionStack(new WebSocketExtensionRegistry());
|
ExtensionStack exStack = new ExtensionStack(new WebSocketExtensionRegistry());
|
||||||
exStack.negotiate(new DecoratedObjectFactory(), bufferPool, new LinkedList<>());
|
exStack.negotiate(new DecoratedObjectFactory(), bufferPool, new LinkedList<>(), new LinkedList<>());
|
||||||
this.channel = new WebSocketChannel(new AbstractTestFrameHandler(), Behavior.CLIENT, Negotiated.from(exStack));
|
this.channel = new WebSocketChannel(new AbstractTestFrameHandler(), Behavior.CLIENT, Negotiated.from(exStack));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -56,7 +56,7 @@ public class GeneratorTest
|
||||||
{
|
{
|
||||||
ByteBufferPool bufferPool = new MappedByteBufferPool();
|
ByteBufferPool bufferPool = new MappedByteBufferPool();
|
||||||
ExtensionStack exStack = new ExtensionStack(new WebSocketExtensionRegistry());
|
ExtensionStack exStack = new ExtensionStack(new WebSocketExtensionRegistry());
|
||||||
exStack.negotiate(new DecoratedObjectFactory(), bufferPool, new LinkedList<>());
|
exStack.negotiate(new DecoratedObjectFactory(), bufferPool, new LinkedList<>(), new LinkedList<>());
|
||||||
return new WebSocketChannel(new AbstractTestFrameHandler(), behavior, Negotiated.from(exStack));
|
return new WebSocketChannel(new AbstractTestFrameHandler(), behavior, Negotiated.from(exStack));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -59,7 +59,7 @@ public class ParserCapture
|
||||||
|
|
||||||
ByteBufferPool bufferPool = new MappedByteBufferPool();
|
ByteBufferPool bufferPool = new MappedByteBufferPool();
|
||||||
ExtensionStack exStack = new ExtensionStack(new WebSocketExtensionRegistry());
|
ExtensionStack exStack = new ExtensionStack(new WebSocketExtensionRegistry());
|
||||||
exStack.negotiate(new DecoratedObjectFactory(), bufferPool, new LinkedList<>());
|
exStack.negotiate(new DecoratedObjectFactory(), bufferPool, new LinkedList<>(), new LinkedList<>());
|
||||||
this.channel = new WebSocketChannel(new AbstractTestFrameHandler(), behavior, Negotiated.from(exStack));
|
this.channel = new WebSocketChannel(new AbstractTestFrameHandler(), behavior, Negotiated.from(exStack));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -420,7 +420,7 @@ public class DeflateFrameExtensionTest extends AbstractExtensionTest
|
||||||
{
|
{
|
||||||
ByteBufferPool bufferPool = new MappedByteBufferPool();
|
ByteBufferPool bufferPool = new MappedByteBufferPool();
|
||||||
ExtensionStack exStack = new ExtensionStack(new WebSocketExtensionRegistry());
|
ExtensionStack exStack = new ExtensionStack(new WebSocketExtensionRegistry());
|
||||||
exStack.negotiate(new DecoratedObjectFactory(), bufferPool, new LinkedList<>());
|
exStack.negotiate(new DecoratedObjectFactory(), bufferPool, new LinkedList<>(), new LinkedList<>());
|
||||||
|
|
||||||
WebSocketChannel channel = new WebSocketChannel(new AbstractTestFrameHandler(), Behavior.SERVER, Negotiated.from(exStack));
|
WebSocketChannel channel = new WebSocketChannel(new AbstractTestFrameHandler(), Behavior.SERVER, Negotiated.from(exStack));
|
||||||
channel.setMaxFrameSize(maxMessageSize);
|
channel.setMaxFrameSize(maxMessageSize);
|
||||||
|
|
|
@ -75,7 +75,7 @@ public class ExtensionStackTest
|
||||||
// 1 extension
|
// 1 extension
|
||||||
List<ExtensionConfig> configs = new ArrayList<>();
|
List<ExtensionConfig> configs = new ArrayList<>();
|
||||||
configs.add(ExtensionConfig.parse("identity"));
|
configs.add(ExtensionConfig.parse("identity"));
|
||||||
stack.negotiate(objectFactory, bufferPool, configs);
|
stack.negotiate(objectFactory, bufferPool, configs, configs);
|
||||||
|
|
||||||
// Setup Listeners
|
// Setup Listeners
|
||||||
IncomingFrames session = new IncomingFramesCapture();
|
IncomingFrames session = new IncomingFramesCapture();
|
||||||
|
@ -99,7 +99,7 @@ public class ExtensionStackTest
|
||||||
List<ExtensionConfig> configs = new ArrayList<>();
|
List<ExtensionConfig> configs = new ArrayList<>();
|
||||||
configs.add(ExtensionConfig.parse("identity; id=A"));
|
configs.add(ExtensionConfig.parse("identity; id=A"));
|
||||||
configs.add(ExtensionConfig.parse("identity; id=B"));
|
configs.add(ExtensionConfig.parse("identity; id=B"));
|
||||||
stack.negotiate(objectFactory, bufferPool, configs);
|
stack.negotiate(objectFactory, bufferPool, configs, configs);
|
||||||
|
|
||||||
// Setup Listeners
|
// Setup Listeners
|
||||||
IncomingFrames session = new IncomingFramesCapture();
|
IncomingFrames session = new IncomingFramesCapture();
|
||||||
|
@ -130,7 +130,7 @@ public class ExtensionStackTest
|
||||||
{
|
{
|
||||||
String chromeRequest = "permessage-deflate; client_max_window_bits, x-webkit-deflate-frame";
|
String chromeRequest = "permessage-deflate; client_max_window_bits, x-webkit-deflate-frame";
|
||||||
List<ExtensionConfig> requestedConfigs = ExtensionConfig.parseList(chromeRequest);
|
List<ExtensionConfig> requestedConfigs = ExtensionConfig.parseList(chromeRequest);
|
||||||
stack.negotiate(objectFactory, bufferPool, requestedConfigs);
|
stack.negotiate(objectFactory, bufferPool, requestedConfigs, requestedConfigs);
|
||||||
|
|
||||||
List<ExtensionConfig> negotiated = stack.getNegotiatedExtensions();
|
List<ExtensionConfig> negotiated = stack.getNegotiatedExtensions();
|
||||||
String response = ExtensionConfig.toHeaderValue(negotiated);
|
String response = ExtensionConfig.toHeaderValue(negotiated);
|
||||||
|
|
|
@ -154,7 +154,7 @@ public class ExtensionTool
|
||||||
{
|
{
|
||||||
ByteBufferPool bufferPool = new MappedByteBufferPool();
|
ByteBufferPool bufferPool = new MappedByteBufferPool();
|
||||||
ExtensionStack exStack = new ExtensionStack(new WebSocketExtensionRegistry());
|
ExtensionStack exStack = new ExtensionStack(new WebSocketExtensionRegistry());
|
||||||
exStack.negotiate(new DecoratedObjectFactory(), bufferPool, new LinkedList<>());
|
exStack.negotiate(new DecoratedObjectFactory(), bufferPool, new LinkedList<>(), new LinkedList<>());
|
||||||
WebSocketChannel channel = new WebSocketChannel(new AbstractTestFrameHandler(), Behavior.SERVER, Negotiated.from(exStack));
|
WebSocketChannel channel = new WebSocketChannel(new AbstractTestFrameHandler(), Behavior.SERVER, Negotiated.from(exStack));
|
||||||
return channel;
|
return channel;
|
||||||
}
|
}
|
||||||
|
|
|
@ -22,7 +22,6 @@ import java.net.HttpCookie;
|
||||||
import java.net.InetSocketAddress;
|
import java.net.InetSocketAddress;
|
||||||
import java.net.SocketAddress;
|
import java.net.SocketAddress;
|
||||||
import java.net.URI;
|
import java.net.URI;
|
||||||
import java.net.URISyntaxException;
|
|
||||||
import java.security.Principal;
|
import java.security.Principal;
|
||||||
import java.security.cert.X509Certificate;
|
import java.security.cert.X509Certificate;
|
||||||
import java.util.ArrayList;
|
import java.util.ArrayList;
|
||||||
|
@ -38,6 +37,7 @@ import javax.servlet.http.Cookie;
|
||||||
import javax.servlet.http.HttpServletRequest;
|
import javax.servlet.http.HttpServletRequest;
|
||||||
import javax.servlet.http.HttpSession;
|
import javax.servlet.http.HttpSession;
|
||||||
|
|
||||||
|
import org.eclipse.jetty.http.BadMessageException;
|
||||||
import org.eclipse.jetty.http.HttpHeader;
|
import org.eclipse.jetty.http.HttpHeader;
|
||||||
import org.eclipse.jetty.websocket.core.ExtensionConfig;
|
import org.eclipse.jetty.websocket.core.ExtensionConfig;
|
||||||
import org.eclipse.jetty.websocket.core.WebSocketConstants;
|
import org.eclipse.jetty.websocket.core.WebSocketConstants;
|
||||||
|
@ -57,21 +57,28 @@ public class ServletUpgradeRequest
|
||||||
private List<HttpCookie> cookies;
|
private List<HttpCookie> cookies;
|
||||||
private Map<String, List<String>> parameterMap;
|
private Map<String, List<String>> parameterMap;
|
||||||
|
|
||||||
public ServletUpgradeRequest(Negotiation negotiation) throws URISyntaxException
|
public ServletUpgradeRequest(Negotiation negotiation) throws BadMessageException
|
||||||
{
|
{
|
||||||
this.negotiation = negotiation;
|
this.negotiation = negotiation;
|
||||||
HttpServletRequest httpRequest = negotiation.getRequest();
|
HttpServletRequest httpRequest = negotiation.getRequest();
|
||||||
this.queryString = httpRequest.getQueryString();
|
this.queryString = httpRequest.getQueryString();
|
||||||
this.secure = httpRequest.isSecure();
|
this.secure = httpRequest.isSecure();
|
||||||
|
|
||||||
// TODO why is this URL and not URI?
|
try
|
||||||
StringBuffer uri = httpRequest.getRequestURL();
|
{
|
||||||
// WHY?
|
// TODO why is this URL and not URI?
|
||||||
if (this.queryString != null)
|
StringBuffer uri = httpRequest.getRequestURL();
|
||||||
uri.append("?").append(this.queryString);
|
// WHY?
|
||||||
uri.replace(0, uri.indexOf(":"), secure?"wss":"ws");
|
if (this.queryString != null)
|
||||||
this.requestURI = new URI(uri.toString());
|
uri.append("?").append(this.queryString);
|
||||||
this.request = new UpgradeHttpServletRequest(httpRequest);
|
uri.replace(0, uri.indexOf(":"), secure ? "wss" : "ws");
|
||||||
|
this.requestURI = new URI(uri.toString());
|
||||||
|
this.request = new UpgradeHttpServletRequest(httpRequest);
|
||||||
|
}
|
||||||
|
catch (Throwable t)
|
||||||
|
{
|
||||||
|
throw new BadMessageException("Bad WebSocket UpgradeRequest", t);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|
|
@ -19,7 +19,6 @@
|
||||||
package org.eclipse.jetty.websocket.servlet;
|
package org.eclipse.jetty.websocket.servlet;
|
||||||
|
|
||||||
import java.io.IOException;
|
import java.io.IOException;
|
||||||
import java.net.URISyntaxException;
|
|
||||||
import java.util.function.Consumer;
|
import java.util.function.Consumer;
|
||||||
|
|
||||||
import javax.servlet.ServletContext;
|
import javax.servlet.ServletContext;
|
||||||
|
@ -32,7 +31,6 @@ import org.eclipse.jetty.http.pathmap.PathSpec;
|
||||||
import org.eclipse.jetty.http.pathmap.RegexPathSpec;
|
import org.eclipse.jetty.http.pathmap.RegexPathSpec;
|
||||||
import org.eclipse.jetty.http.pathmap.ServletPathSpec;
|
import org.eclipse.jetty.http.pathmap.ServletPathSpec;
|
||||||
import org.eclipse.jetty.http.pathmap.UriTemplatePathSpec;
|
import org.eclipse.jetty.http.pathmap.UriTemplatePathSpec;
|
||||||
import org.eclipse.jetty.io.RuntimeIOException;
|
|
||||||
import org.eclipse.jetty.server.handler.ContextHandler;
|
import org.eclipse.jetty.server.handler.ContextHandler;
|
||||||
import org.eclipse.jetty.util.component.Dumpable;
|
import org.eclipse.jetty.util.component.Dumpable;
|
||||||
import org.eclipse.jetty.util.component.LifeCycle;
|
import org.eclipse.jetty.util.component.LifeCycle;
|
||||||
|
@ -217,7 +215,7 @@ public class WebSocketMapping implements Dumpable, LifeCycle.Listener
|
||||||
return mapping.getResource();
|
return mapping.getResource();
|
||||||
}
|
}
|
||||||
|
|
||||||
public boolean upgrade(HttpServletRequest request, HttpServletResponse response, FrameHandler.Customizer defaultCustomizer)
|
public boolean upgrade(HttpServletRequest request, HttpServletResponse response, FrameHandler.Customizer defaultCustomizer) throws IOException
|
||||||
{
|
{
|
||||||
// Since this may be a filter, we need to be smart about determining the target path.
|
// Since this may be a filter, we need to be smart about determining the target path.
|
||||||
// We should rely on the Container for stripping path parameters and its ilk before
|
// We should rely on the Container for stripping path parameters and its ilk before
|
||||||
|
@ -240,14 +238,7 @@ public class WebSocketMapping implements Dumpable, LifeCycle.Listener
|
||||||
LOG.debug("WebSocket Negotiated detected on {} for endpoint {}", target, negotiator);
|
LOG.debug("WebSocket Negotiated detected on {} for endpoint {}", target, negotiator);
|
||||||
|
|
||||||
// We have an upgrade request
|
// We have an upgrade request
|
||||||
try
|
return handshaker.upgradeRequest(negotiator, request, response, defaultCustomizer);
|
||||||
{
|
|
||||||
return handshaker.upgradeRequest(negotiator, request, response, defaultCustomizer);
|
|
||||||
}
|
|
||||||
catch (IOException e)
|
|
||||||
{
|
|
||||||
throw new WebSocketException(e);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private class Negotiator extends WebSocketNegotiator.AbstractNegotiator
|
private class Negotiator extends WebSocketNegotiator.AbstractNegotiator
|
||||||
|
@ -269,7 +260,7 @@ public class WebSocketMapping implements Dumpable, LifeCycle.Listener
|
||||||
|
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public FrameHandler negotiate(Negotiation negotiation)
|
public FrameHandler negotiate(Negotiation negotiation) throws IOException
|
||||||
{
|
{
|
||||||
ServletContext servletContext = negotiation.getRequest().getServletContext();
|
ServletContext servletContext = negotiation.getRequest().getServletContext();
|
||||||
if (servletContext == null)
|
if (servletContext == null)
|
||||||
|
@ -304,14 +295,6 @@ public class WebSocketMapping implements Dumpable, LifeCycle.Listener
|
||||||
|
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
catch (IOException e)
|
|
||||||
{
|
|
||||||
throw new RuntimeIOException(e);
|
|
||||||
}
|
|
||||||
catch (URISyntaxException e)
|
|
||||||
{
|
|
||||||
throw new RuntimeIOException("Unable to negotiate websocket due to mangled request URI", e);
|
|
||||||
}
|
|
||||||
finally
|
finally
|
||||||
{
|
{
|
||||||
Thread.currentThread().setContextClassLoader(old);
|
Thread.currentThread().setContextClassLoader(old);
|
||||||
|
|
Loading…
Reference in New Issue