2014-06-06 222 views
3

在这些日子里,我尝试通过使用Jetty来实现WebSocket服务器。Jetty - 独立WebSocket服务器

我在Jetty 9.2.0.v20140526的发行版中创建了一个名为“WebSocketServer”的Jetty项目demo-base

之后,我写了一些代码来实现WebSocket机制,并将所有代码导出到war文件,以将其推送到“WebSocketServer”的webapps文件夹。当我java -jar ..<jetty.home>/start.jar,这是可行的。但是,在我创建了一个到这个WebSocket项目的新连接之后,发生了一些错误代码。

java.lang.ClassCastException: org.eclipse.jetty.server.HttpConnection cannot be cast to org.eclipse.jetty.server.HttpConnection 
at org.eclipse.jetty.websocket.server.WebSocketServerFactory.acceptWebSocket(WebSocketServerFactory.java:175) 
at org.eclipse.jetty.websocket.server.WebSocketServerFactory.acceptWebSocket(WebSocketServerFactory.java:148) 
at org.eclipse.jetty.websocket.servlet.WebSocketServlet.service(WebSocketServlet.java:151) 
at javax.servlet.http.HttpServlet.service(HttpServlet.java:790) 
at org.eclipse.jetty.servlet.ServletHolder.handle(ServletHolder.java:751) 
at org.eclipse.jetty.servlet.ServletHandler.doHandle(ServletHandler.java:566) 
at org.eclipse.jetty.server.handler.ScopedHandler.handle(ScopedHandler.java:143) 
at org.eclipse.jetty.security.SecurityHandler.handle(SecurityHandler.java:578) 
at org.eclipse.jetty.server.session.SessionHandler.doHandle(SessionHandler.java:221) 
at org.eclipse.jetty.server.handler.ContextHandler.doHandle(ContextHandler.java:1111) 
at org.eclipse.jetty.servlet.ServletHandler.doScope(ServletHandler.java:498) 
at org.eclipse.jetty.server.session.SessionHandler.doScope(SessionHandler.java:183) 
at org.eclipse.jetty.server.handler.ContextHandler.doScope(ContextHandler.java:1045) 
at org.eclipse.jetty.server.handler.ScopedHandler.handle(ScopedHandler.java:141) 
at org.eclipse.jetty.server.handler.ContextHandlerCollection.handle(ContextHandlerCollection.java:199) 
at org.eclipse.jetty.server.handler.HandlerCollection.handle(HandlerCollection.java:109) 
at org.eclipse.jetty.server.handler.HandlerWrapper.handle(HandlerWrapper.java:98) 
at org.eclipse.jetty.server.Server.handle(Server.java:461) 
at org.eclipse.jetty.server.HttpChannel.handle(HttpChannel.java:284) 
at org.eclipse.jetty.server.HttpConnection.onFillable(HttpConnection.java:244) 
at org.eclipse.jetty.io.AbstractConnection$2.run(AbstractConnection.java:534) 
at org.eclipse.jetty.util.thread.QueuedThreadPool.runJob(QueuedThreadPool.java:607) 
at org.eclipse.jetty.util.thread.QueuedThreadPool$3.run(QueuedThreadPool.java:536) 
at java.lang.Thread.run(Thread.java:744) 

我不知道发生了什么事?以下代码是我编写的构建简单WebSocket服务器的代码。

的Servlet:

@SuppressWarnings("serial") 
public class XYZWebSocketServlet extends WebSocketServlet{ 
    @Override 
    public void configure(WebSocketServletFactory factory) { 
     factory.getPolicy().setIdleTimeout(600000); 
     factory.register(XYZWebSocketEvent.class); 
    } 
} 

事件:

@WebSocket 
public class XYZWebSocketEvent { 
    private Session session; 

    @OnWebSocketConnect 
    public void onConnect(Session sess) { 
     session = sess; 

     // Get parameters while client connect to server 
     Map<String,List<String>> parameters = session.getUpgradeRequest().getParameterMap(); 
     String encyptedID = parameters.get("ID").get(0); 

     System.out.println("Connect: " + session.getRemoteAddress().getPort()); 
     try { 
      session.setIdleTimeout(600000); 
      session.getRemote().sendString("Hello!"); 
     } catch (IOException ioe) { 
      ioe.printStackTrace(); 
     } 
    } 

    @OnWebSocketMessage 
    public void onMessage(String message) { 
      try { 
       session.getRemote().sendString("Message: " + message); 
      } 
      catch (Exception ex) { 
      } 
    } 

    @OnWebSocketClose 
    public void onClose(int statusCode, String reason) { 
      try { 
       session.getRemote().sendString("Close: statusCode=" + statusCode + ", reason=" +reason); 
     } 
      catch (Exception ex) { 
      } 
    } 

    @OnWebSocketError 
    public void onError(Throwable t) { 
     System.out.println("Error: " + t.getMessage()); 
    } 

    public Session getSession() { 
     return this.session; 
    } 
} 

web.xml中:

<?xml version="1.0" encoding="UTF-8"?> 
<web-app> 
    <display-name>WebSocket application</display-name> 
    <servlet> 
     <servlet-name>XYZWebSocketServlet</servlet-name> 
     <servlet-class>com.newKinpo.servlet.XYZWebSocketServlet</servlet-class> 
    </servlet> 

    <servlet-mapping> 
     <servlet-name>XYZWebSocketServlet</servlet-name> 
     <url-pattern>/events/*</url-pattern> 
    </servlet-mapping> 
</web-app> 

有什么错吗?感谢您的关注。

回答

0

我有类似的问题,我找到了原因和解决方案。嵌入式Jetty服务器由SUN类加载器加载(稍后将调用它为系统类加载器)在扫描webapp目录后,我的应用程序由WebApp类加载器加载,当涉及到WebSocketServerFactory时,它由WebApp加载。然而,从请求中获得的org.eclipse.jetty.server.HttpConnection对象是由系统类加载器加载的。

根据https://wiki.eclipse.org/Jetty/Reference/Jetty_Classloading jetty websocket软件包被视为系统类软件包,不应由WebApp加载(如果已由系统加载)。

解决方案是在初始化jetty服务器时强制加载包org.eclipse.jetty.websocket。

我刚刚创建了例如WebSocketHandler的虚拟实例。有很多选项可以强制包装加载,但它们与此问题无关。

+0

你能否提供关于虚拟实例的更多细节 – Yogesh