Issue #3458 - testing of Jetty WebSocket API ExtensionConfig
wire up the jetty-api extensions by copying the ExtensionConfig from the jetty-api upgrade request to the core upgrade request make UpgradeListener interface methods default for convenience introduce test to test the functionality of API ExtensionConfig Signed-off-by: Lachlan Roberts <lachlan@webtide.com>
This commit is contained in:
parent
62406cdfa2
commit
20e9770b73
|
@ -22,6 +22,7 @@ import java.net.HttpCookie;
|
|||
import java.net.URI;
|
||||
import java.util.List;
|
||||
import java.util.concurrent.CompletableFuture;
|
||||
import java.util.stream.Collectors;
|
||||
|
||||
import org.eclipse.jetty.client.HttpResponse;
|
||||
import org.eclipse.jetty.http.HttpFields;
|
||||
|
@ -33,6 +34,7 @@ import org.eclipse.jetty.websocket.api.UpgradeRequest;
|
|||
import org.eclipse.jetty.websocket.api.UpgradeResponse;
|
||||
import org.eclipse.jetty.websocket.client.WebSocketClient;
|
||||
import org.eclipse.jetty.websocket.common.JettyWebSocketFrameHandler;
|
||||
import org.eclipse.jetty.websocket.core.ExtensionConfig;
|
||||
import org.eclipse.jetty.websocket.core.FrameHandler;
|
||||
import org.eclipse.jetty.websocket.core.client.ClientUpgradeRequest;
|
||||
import org.eclipse.jetty.websocket.core.client.WebSocketCoreClient;
|
||||
|
@ -74,9 +76,9 @@ public class JettyClientUpgradeRequest extends ClientUpgradeRequest
|
|||
setSubProtocols(request.getSubProtocols());
|
||||
|
||||
// Copy extensions
|
||||
/* TODO or not?
|
||||
setExtensions(request.getExtensions());
|
||||
*/
|
||||
setExtensions(request.getExtensions().stream()
|
||||
.map(c -> new ExtensionConfig(c.getName(), c.getParameters()))
|
||||
.collect(Collectors.toList()));
|
||||
|
||||
// Copy method from upgradeRequest object
|
||||
if (request.getMethod() != null)
|
||||
|
|
|
@ -0,0 +1,136 @@
|
|||
//
|
||||
// ========================================================================
|
||||
// Copyright (c) 1995-2019 Mort Bay Consulting Pty. Ltd.
|
||||
// ------------------------------------------------------------------------
|
||||
// All rights reserved. This program and the accompanying materials
|
||||
// are made available under the terms of the Eclipse Public License v1.0
|
||||
// and Apache License v2.0 which accompanies this distribution.
|
||||
//
|
||||
// The Eclipse Public License is available at
|
||||
// http://www.eclipse.org/legal/epl-v10.html
|
||||
//
|
||||
// The Apache License v2.0 is available at
|
||||
// http://www.opensource.org/licenses/apache2.0.php
|
||||
//
|
||||
// You may elect to redistribute this code under either of these licenses.
|
||||
// ========================================================================
|
||||
//
|
||||
|
||||
package org.eclipse.jetty.websocket.tests;
|
||||
|
||||
import java.net.URI;
|
||||
import java.util.List;
|
||||
import java.util.concurrent.CompletableFuture;
|
||||
import java.util.concurrent.CountDownLatch;
|
||||
import java.util.concurrent.TimeUnit;
|
||||
|
||||
import org.eclipse.jetty.client.HttpRequest;
|
||||
import org.eclipse.jetty.client.HttpResponse;
|
||||
import org.eclipse.jetty.http.HttpHeader;
|
||||
import org.eclipse.jetty.server.Server;
|
||||
import org.eclipse.jetty.server.ServerConnector;
|
||||
import org.eclipse.jetty.servlet.ServletContextHandler;
|
||||
import org.eclipse.jetty.websocket.api.Session;
|
||||
import org.eclipse.jetty.websocket.api.UpgradeRequest;
|
||||
import org.eclipse.jetty.websocket.api.extensions.ExtensionConfig;
|
||||
import org.eclipse.jetty.websocket.client.ClientUpgradeRequest;
|
||||
import org.eclipse.jetty.websocket.client.WebSocketClient;
|
||||
import org.eclipse.jetty.websocket.core.client.UpgradeListener;
|
||||
import org.eclipse.jetty.websocket.server.JettyWebSocketServerContainer;
|
||||
import org.eclipse.jetty.websocket.server.JettyWebSocketServletContainerInitializer;
|
||||
import org.junit.jupiter.api.AfterEach;
|
||||
import org.junit.jupiter.api.BeforeEach;
|
||||
import org.junit.jupiter.api.Test;
|
||||
|
||||
import static org.hamcrest.MatcherAssert.assertThat;
|
||||
import static org.hamcrest.Matchers.is;
|
||||
import static org.junit.jupiter.api.Assertions.assertEquals;
|
||||
import static org.junit.jupiter.api.Assertions.assertNotNull;
|
||||
import static org.junit.jupiter.api.Assertions.assertThrows;
|
||||
import static org.junit.jupiter.api.Assertions.assertTrue;
|
||||
|
||||
public class JettyWebSocketExtensionConfigTest
|
||||
{
|
||||
Server server;
|
||||
WebSocketClient client;
|
||||
|
||||
@BeforeEach
|
||||
public void start() throws Exception
|
||||
{
|
||||
server = new Server();
|
||||
ServerConnector connector = new ServerConnector(server);
|
||||
connector.setPort(8080);
|
||||
server.addConnector(connector);
|
||||
|
||||
ServletContextHandler contextHandler = new ServletContextHandler(ServletContextHandler.SESSIONS);
|
||||
contextHandler.setContextPath("/");
|
||||
server.setHandler(contextHandler);
|
||||
|
||||
JettyWebSocketServerContainer container = JettyWebSocketServletContainerInitializer.configureContext(contextHandler);
|
||||
container.addMapping("/", (req, resp)->
|
||||
{
|
||||
assertEquals(req.getExtensions().stream().filter(e -> e.getName().equals("permessage-deflate")).count(), 1);
|
||||
assertEquals(resp.getExtensions().stream().filter(e -> e.getName().equals("permessage-deflate")).count(), 1);
|
||||
|
||||
ExtensionConfig nonRequestedExtension = ExtensionConfig.parse("identity");
|
||||
assertNotNull(nonRequestedExtension);
|
||||
|
||||
assertThrows(IllegalArgumentException.class,
|
||||
()->resp.setExtensions(List.of(nonRequestedExtension)),
|
||||
"should not allow extensions not requested");
|
||||
|
||||
// Check identity extension was not added because it was not requested
|
||||
assertEquals(resp.getExtensions().stream().filter(config -> config.getName().equals("identity")).count(), 0);
|
||||
assertEquals(resp.getExtensions().stream().filter(e -> e.getName().equals("permessage-deflate")).count(), 1);
|
||||
|
||||
return new EchoSocket();
|
||||
});
|
||||
server.start();
|
||||
|
||||
client = new WebSocketClient();
|
||||
client.start();
|
||||
}
|
||||
|
||||
@AfterEach
|
||||
public void stop() throws Exception
|
||||
{
|
||||
client.stop();
|
||||
server.stop();
|
||||
}
|
||||
|
||||
@Test
|
||||
public void testJettyExtensionConfig() throws Exception
|
||||
{
|
||||
URI uri = URI.create("ws://localhost:8080/filterPath");
|
||||
EventSocket socket = new EventSocket();
|
||||
|
||||
UpgradeRequest request = new ClientUpgradeRequest();
|
||||
request.addExtensions(ExtensionConfig.parse("permessage-deflate"));
|
||||
|
||||
CountDownLatch correctResponseExtensions = new CountDownLatch(1);
|
||||
UpgradeListener listener = new UpgradeListener()
|
||||
{
|
||||
@Override
|
||||
public void onHandshakeResponse(HttpRequest request, HttpResponse response)
|
||||
{
|
||||
|
||||
String extensions = response.getHeaders().get(HttpHeader.SEC_WEBSOCKET_EXTENSIONS);
|
||||
if("permessage-deflate".equals(extensions))
|
||||
correctResponseExtensions.countDown();
|
||||
else
|
||||
throw new IllegalStateException("Unexpected Negotiated Extensions: " + extensions);
|
||||
}
|
||||
};
|
||||
|
||||
CompletableFuture<Session> connect = client.connect(socket, uri, request, listener);
|
||||
try(Session session = connect.get(5, TimeUnit.SECONDS))
|
||||
{
|
||||
session.getRemote().sendString("hello world");
|
||||
}
|
||||
assertTrue(socket.closed.await(5, TimeUnit.SECONDS));
|
||||
assertTrue(correctResponseExtensions.await(5, TimeUnit.SECONDS));
|
||||
|
||||
String msg = socket.receivedMessages.poll();
|
||||
assertThat(msg, is("hello world"));
|
||||
}
|
||||
}
|
|
@ -28,7 +28,8 @@ public interface UpgradeListener
|
|||
*
|
||||
* @param request the request
|
||||
*/
|
||||
void onHandshakeRequest(HttpRequest request);
|
||||
default void onHandshakeRequest(HttpRequest request)
|
||||
{}
|
||||
|
||||
/**
|
||||
* Event that triggers after the Handshake response has been received.
|
||||
|
@ -36,5 +37,6 @@ public interface UpgradeListener
|
|||
* @param request the request that was used
|
||||
* @param response the response that was received
|
||||
*/
|
||||
void onHandshakeResponse(HttpRequest request, HttpResponse response);
|
||||
default void onHandshakeResponse(HttpRequest request, HttpResponse response)
|
||||
{}
|
||||
}
|
||||
|
|
|
@ -18,11 +18,6 @@
|
|||
|
||||
package org.eclipse.jetty.websocket.servlet;
|
||||
|
||||
import org.eclipse.jetty.http.HttpHeader;
|
||||
import org.eclipse.jetty.websocket.core.ExtensionConfig;
|
||||
import org.eclipse.jetty.websocket.core.server.Negotiation;
|
||||
|
||||
import javax.servlet.http.HttpServletResponse;
|
||||
import java.io.IOException;
|
||||
import java.util.ArrayList;
|
||||
import java.util.Collections;
|
||||
|
@ -33,6 +28,12 @@ import java.util.Objects;
|
|||
import java.util.Set;
|
||||
import java.util.stream.Collectors;
|
||||
|
||||
import javax.servlet.http.HttpServletResponse;
|
||||
|
||||
import org.eclipse.jetty.http.HttpHeader;
|
||||
import org.eclipse.jetty.websocket.core.ExtensionConfig;
|
||||
import org.eclipse.jetty.websocket.core.server.Negotiation;
|
||||
|
||||
/**
|
||||
* Servlet Specific UpgradeResponse implementation.
|
||||
*/
|
||||
|
@ -166,6 +167,20 @@ public class ServletUpgradeResponse
|
|||
|
||||
public void setExtensions(List<ExtensionConfig> configs)
|
||||
{
|
||||
for (ExtensionConfig config : configs)
|
||||
{
|
||||
List<ExtensionConfig> collect = negotiation.getOfferedExtensions().stream()
|
||||
.filter(e -> e.getName().equals(config.getName()))
|
||||
.collect(Collectors.toList());
|
||||
|
||||
if (collect.size() == 1)
|
||||
continue;
|
||||
else if (collect.size() == 0)
|
||||
throw new IllegalArgumentException("Extension not a requested extension");
|
||||
else if (collect.size() > 1)
|
||||
throw new IllegalArgumentException("Multiple extensions of the same name");
|
||||
}
|
||||
|
||||
negotiation.setNegotiatedExtensions(configs);
|
||||
}
|
||||
|
||||
|
|
Loading…
Reference in New Issue