Skip to content

Commit

Permalink
Merge pull request #6325 from eclipse/jetty-9.4.x-6287-WebSocketClien…
Browse files Browse the repository at this point in the history
…tClassLoading

 Issue #6287 - fix classloading for WebSocketClient in webapp
  • Loading branch information
lachlan-roberts committed Jun 3, 2021
2 parents 76f5162 + 816b326 commit 121d8c2
Show file tree
Hide file tree
Showing 6 changed files with 511 additions and 4 deletions.
Expand Up @@ -325,6 +325,12 @@ public WebSocketClient getClient()
return client;
}

@Override
public ClassLoader getClassLoader()
{
return client.getClassLoader();
}

public EndpointMetadata getClientEndpointMetadata(Class<?> endpoint, EndpointConfig config)
{
synchronized (endpointClientMetadataCache)
Expand Down
@@ -0,0 +1,217 @@
//
// ========================================================================
// Copyright (c) 1995-2021 Mort Bay Consulting Pty Ltd and others.
// ------------------------------------------------------------------------
// 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.jsr356.server;

import java.net.URI;
import java.util.concurrent.LinkedBlockingQueue;
import java.util.concurrent.TimeUnit;
import javax.servlet.annotation.WebServlet;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import javax.websocket.ClientEndpoint;
import javax.websocket.ContainerProvider;
import javax.websocket.OnMessage;
import javax.websocket.OnOpen;
import javax.websocket.Session;
import javax.websocket.WebSocketContainer;
import javax.websocket.server.ServerEndpoint;

import org.eclipse.jetty.client.HttpClient;
import org.eclipse.jetty.client.api.ContentResponse;
import org.eclipse.jetty.client.api.Response;
import org.eclipse.jetty.http.BadMessageException;
import org.eclipse.jetty.http.HttpStatus;
import org.eclipse.jetty.io.ByteBufferPool;
import org.eclipse.jetty.util.component.ContainerLifeCycle;
import org.eclipse.jetty.webapp.WebAppContext;
import org.eclipse.jetty.websocket.client.WebSocketClient;
import org.eclipse.jetty.websocket.common.WebSocketSession;
import org.eclipse.jetty.websocket.jsr356.ClientContainer;
import org.eclipse.jetty.xml.XmlConfiguration;
import org.junit.jupiter.api.AfterEach;
import org.junit.jupiter.api.Test;

import static org.hamcrest.MatcherAssert.assertThat;
import static org.hamcrest.Matchers.containsString;
import static org.hamcrest.Matchers.is;
import static org.junit.jupiter.api.Assertions.assertNotNull;

public class JavaxClientClassLoaderTest
{
private final WebAppTester server = new WebAppTester();
private HttpClient httpClient;

@FunctionalInterface
interface ThrowingRunnable
{
void run() throws Exception;
}

public void start(ThrowingRunnable configuration) throws Exception
{
configuration.run();
server.start();
httpClient = new HttpClient();
httpClient.start();
}

@AfterEach
public void after() throws Exception
{
httpClient.stop();
server.stop();
}

@ClientEndpoint()
public static class ClientSocket
{
LinkedBlockingQueue<String> textMessages = new LinkedBlockingQueue<>();

@OnOpen
public void onOpen(Session session)
{
session.getAsyncRemote().sendText("ContextClassLoader: " + Thread.currentThread().getContextClassLoader());
}

@OnMessage
public void onMessage(String message)
{
textMessages.add(message);
}
}

@WebServlet("/servlet")
public static class WebSocketClientServlet extends HttpServlet
{
private final WebSocketContainer clientContainer = ContainerProvider.getWebSocketContainer();

@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp)
{
URI wsEchoUri = URI.create("ws://localhost:" + req.getServerPort() + "/echo/");
ClientSocket clientSocket = new ClientSocket();

try (Session ignored = clientContainer.connectToServer(clientSocket, wsEchoUri))
{
String recv = clientSocket.textMessages.poll(5, TimeUnit.SECONDS);
assertNotNull(recv);
resp.setStatus(HttpStatus.OK_200);
resp.getWriter().println(recv);
resp.getWriter().println("ClientClassLoader: " + clientContainer.getClass().getClassLoader());
}
catch (Exception e)
{
throw new RuntimeException(e);
}
}
}

@ServerEndpoint("/")
public static class EchoSocket
{
@OnMessage
public void onMessage(Session session, String message) throws Exception
{
session.getBasicRemote().sendText(message);
}
}

public WebAppTester.WebApp createWebSocketWebapp(String contextName) throws Exception
{
WebAppTester.WebApp app = this.server.createWebApp(contextName);

// We must hide the websocket classes from the webapp if we are to include websocket client jars in WEB-INF/lib.
WebAppContext context = app.getWebAppContext();
context.getServerClasspathPattern().include("org.eclipse.jetty.websocket.");
context.getSystemClasspathPattern().exclude("org.eclipse.jetty.websocket.");

// Copy over the individual jars required for Javax WebSocket.
app.createWebInf();
app.copyLib(WebSocketContainer.class, "websocket-javax-api.jar");
app.copyLib(ClientContainer.class, "websocket-javax-client.jar");
app.copyLib(WebSocketClient.class, "websocket-jetty-client.jar");
app.copyLib(WebSocketSession.class, "websocket-common.jar");
app.copyLib(ContainerLifeCycle.class, "jetty-util.jar");
app.copyLib(Response.class, "jetty-client.jar");
app.copyLib(ByteBufferPool.class, "jetty-io.jar");
app.copyLib(BadMessageException.class, "jetty-http.jar");
app.copyLib(XmlConfiguration.class, "jetty-xml.jar");

return app;
}

@Test
public void websocketProvidedByServer() throws Exception
{
start(() ->
{
WebAppTester.WebApp app1 = server.createWebApp("/app");
app1.createWebInf();
app1.copyClass(WebSocketClientServlet.class);
app1.copyClass(ClientSocket.class);
app1.deploy();

WebAppTester.WebApp app2 = server.createWebApp("/echo");
app2.createWebInf();
app2.copyClass(EchoSocket.class);
app2.deploy();
});

// After hitting each WebApp we will get 200 response if test succeeds.
ContentResponse response = httpClient.GET(server.getServerUri().resolve("/app/servlet"));
assertThat(response.getStatus(), is(HttpStatus.OK_200));

// The ContextClassLoader in the WebSocketClients onOpen was the WebAppClassloader.
assertThat(response.getContentAsString(), containsString("ContextClassLoader: WebAppClassLoader"));

// Verify that we used Servers version of WebSocketClient.
ClassLoader serverClassLoader = server.getServer().getClass().getClassLoader();
assertThat(response.getContentAsString(), containsString("ClientClassLoader: " + serverClassLoader)); }

@Test
public void websocketProvidedByWebApp() throws Exception
{
start(() ->
{
WebAppTester.WebApp app1 = createWebSocketWebapp("/app");
app1.createWebInf();
app1.copyClass(WebSocketClientServlet.class);
app1.copyClass(ClientSocket.class);
app1.copyClass(EchoSocket.class);
app1.deploy();

// Do not exclude JavaxWebSocketConfiguration for this webapp (we need the websocket server classes).
WebAppTester.WebApp app2 = server.createWebApp("/echo");
app2.createWebInf();
app2.copyClass(EchoSocket.class);
app2.deploy();
});

// After hitting each WebApp we will get 200 response if test succeeds.
ContentResponse response = httpClient.GET(server.getServerUri().resolve("/app/servlet"));
assertThat(response.getStatus(), is(HttpStatus.OK_200));

// The ContextClassLoader in the WebSocketClients onOpen was the WebAppClassloader.
assertThat(response.getContentAsString(), containsString("ContextClassLoader: WebAppClassLoader"));

// Verify that we used WebApps version of WebSocketClient.
assertThat(response.getContentAsString(), containsString("ClientClassLoader: WebAppClassLoader"));
}
}

0 comments on commit 121d8c2

Please sign in to comment.