Timothy Ward created FELIX-5310:
-----------------------------------
Summary: Felix HTTP Jetty WebSockets do not work out of the box
Key: FELIX-5310
URL: https://issues.apache.org/jira/browse/FELIX-5310
Project: Felix
Issue Type: Bug
Components: HTTP Service
Affects Versions: http.jetty-3.2.0
Reporter: Timothy Ward
I have encountered two significant problems when trying to use the Jetty
WebSocketServlet with the Jetty based Felix HTTP service whiteboard. Both
problems occur in the init method of my servlet.
1. I have to set the TCCL for the Jetty Web Socket implementation to be able
to find its own internal implementation of Websockets (specifically
org.eclipse.jetty.websocket.server.WebSocketServerFactory). This is really
stupid, as the TCCL I have to use is the Felix HTTP Jetty bundle's class loader!
2. The Jetty Web Socket implementation does one (and only one) check to check
that it's running on Jetty. It looks as though the only reason for this is to
get hold of an Executor which is available via a getter. This check involves
trying to establish the Jetty Context by casting the ServletContext to a Jetty
internal type. This does not work as the HTTP Whiteboard wrappers the
ServletContext. I can work around this by delaying initialisation to the first
request so that there is a context to get hold of, but this also sucks :(
In summary, my servlet has to look like this, most of which should not be
necessary!
public class EchoServlet extends WebSocketServlet {
private static final Logger LOGGER =
LoggerFactory.getLogger(EchoServlet.class);
private final AtomicBoolean firstCall = new AtomicBoolean(true);
private final CountDownLatch initBarrier = new CountDownLatch(1);
@Override
public void init() throws ServletException {
LOGGER.info("The Echo servlet has been initialized, but we delay
initialization until the first request so that a Jetty Context is available");
}
@Override
public void service(ServletRequest arg0, ServletResponse arg1) throws
ServletException, IOException {
if(firstCall.compareAndSet(true, false)) {
try {
delayedInit();
} finally {
initBarrier.countDown();
}
} else {
try {
initBarrier.await();
} catch (InterruptedException e) {
throw new ServletException("Timed out waiting for
initialisation", e);
}
}
super.service(arg0, arg1);
}
private void delayedInit() throws ServletException {
// Overide the TCCL so that the internal factory can be found
// Jetty tries to use ServiceLoader, and their fallback is to
// use TCCL, it would be better if we could provide a loader...
Thread currentThread = Thread.currentThread();
ClassLoader tccl = currentThread.getContextClassLoader();
currentThread.setContextClassLoader(WebSocketServlet.class.getClassLoader());
try {
super.init();
} finally {
currentThread.setContextClassLoader(tccl);
}
}
@Override
public void configure(WebSocketServletFactory wsf) {
wsf.setCreator((req,res) -> new WebSocketAdaptor() {
public void onWebSocketText(String message) {
getRemote().sendStringByFuture("Echo: " + message);
}
});
}
}
--
This message was sent by Atlassian JIRA
(v6.3.4#6332)