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:
lachan-roberts 2019-03-22 15:42:39 +11:00
parent eb4f7c000f
commit 0d570dd6b6
15 changed files with 139 additions and 132 deletions

View File

@ -156,19 +156,11 @@ 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
{ {

View File

@ -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

View File

@ -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;

View File

@ -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())

View File

@ -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

View File

@ -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,6 +81,8 @@ public class Negotiation
QuotedCSV extensions = null; QuotedCSV extensions = null;
QuotedCSV subprotocols = null; QuotedCSV subprotocols = null;
try
{
for (HttpField field : baseRequest.getHttpFields()) for (HttpField field : baseRequest.getHttpFields())
{ {
if (field.getHeader() != null) if (field.getHeader() != null)
@ -127,25 +133,30 @@ public class Negotiation
Set<String> available = registry.getAvailableExtensionNames(); Set<String> available = registry.getAvailableExtensionNames();
offeredExtensions = extensions == null offeredExtensions = extensions == null
?Collections.emptyList() ? Collections.emptyList()
:extensions.getValues().stream() : extensions.getValues().stream()
.map(ExtensionConfig::parse) .map(ExtensionConfig::parse)
.filter(ec -> available.contains(ec.getName().toLowerCase()) && !ec.getName().startsWith("@")) .filter(ec -> available.contains(ec.getName().toLowerCase()) && !ec.getName().startsWith("@"))
.collect(Collectors.toList()); .collect(Collectors.toList());
offeredSubprotocols = subprotocols == null offeredSubprotocols = subprotocols == null
?Collections.emptyList() ? Collections.emptyList()
:subprotocols.getValues(); : subprotocols.getValues();
negotiatedExtensions = new ArrayList<>(); negotiatedExtensions = new ArrayList<>();
for (ExtensionConfig config : offeredExtensions) for (ExtensionConfig config : offeredExtensions)
{ {
long matches = negotiatedExtensions.stream() long matches = negotiatedExtensions.stream()
.filter(negotiatedConfig->negotiatedConfig.getName().equals(config.getName())).count(); .filter(negotiatedConfig -> negotiatedConfig.getName().equals(config.getName())).count();
if (matches == 0) if (matches == 0)
negotiatedExtensions.add(config); negotiatedExtensions.add(config);
} }
} }
catch (Throwable t)
{
throw new BadMessageException("Invalid Handshake Request", t);
}
}
public String getKey() public String getKey()
{ {
@ -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())

View File

@ -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);

View File

@ -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));
} }

View File

@ -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));
} }

View File

@ -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));
} }

View File

@ -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);

View File

@ -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);

View File

@ -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;
} }

View File

@ -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,22 +57,29 @@ 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();
try
{
// TODO why is this URL and not URI? // TODO why is this URL and not URI?
StringBuffer uri = httpRequest.getRequestURL(); StringBuffer uri = httpRequest.getRequestURL();
// WHY? // WHY?
if (this.queryString != null) if (this.queryString != null)
uri.append("?").append(this.queryString); uri.append("?").append(this.queryString);
uri.replace(0, uri.indexOf(":"), secure?"wss":"ws"); uri.replace(0, uri.indexOf(":"), secure ? "wss" : "ws");
this.requestURI = new URI(uri.toString()); this.requestURI = new URI(uri.toString());
this.request = new UpgradeHttpServletRequest(httpRequest); this.request = new UpgradeHttpServletRequest(httpRequest);
} }
catch (Throwable t)
{
throw new BadMessageException("Bad WebSocket UpgradeRequest", t);
}
}
/** /**
* @return The {@link X509Certificate} instance at request attribute "javax.servlet.request.X509Certificate" or null. * @return The {@link X509Certificate} instance at request attribute "javax.servlet.request.X509Certificate" or null.

View File

@ -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,15 +238,8 @@ 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);