6

我在tomcat中使用Guice-servlets和websocket创建了一个示例webapp,现在一旦使用guice过滤器websocket停止工作

基本信息:

在我的 web.xml 中,我使用初始化 GuiceservletGuiceBasedListener

<web-app>
     <listener>
        <listener-class>test.GuiceBasedListener</listener-class>
    </listener>          
    <filter>
        <filter-name>guiceFilter</filter-name>
        <filter-class>com.google.inject.servlet.GuiceFilter</filter-class>
    </filter>
    <filter-mapping>
        <filter-name>guiceFilter</filter-name>
        <url-pattern>/*</url-pattern>
    </filter-mapping>
</web-app>

GuieBasedListener将所有请求绑定/*到的代码MyDispatcher

public class GuiceBasedListener extends GuiceServletContextListener {
    protected Injector getInjector() {
        return Guice.createInjector( new ServletModule() {
            @Override
            protected void configureServlets() {
                bind(MyDispatcher.class).asEagerSingleton();
                serve("/*").with(MyDispatcher.class);//////IMPORTANT LINE//
            }
        });}}

MyDispatcher仅以字符串响应的代码

public class MyDispatcher extends HttpServlet {    
    @Inject private Injector injector;
    public MyDispatcher() {}    
    public void service(ServletRequest req, ServletResponse resp) throws IOException, ServletException {
        resp.getOutputStream().print("SUCCESS:" + req);
    }
}

我还有一个用于 Websocket 的 @ServerEndPoint

@ServerEndpoint(value = "/websocket/chat2")
public class WebSocket{
....
    @OnOpen
    public void start(Session session) {        
        System.out.println("Staring:"+this);
   }
....
}

观察:

  1. 现在,如果我运行应用程序并点击http://app:8080/test它会返回SUCCESS
  2. 但是,如果我尝试使用 ws://app:8080/websocket/chat2 连接到 websocket,它会失败
  3. 现在,如果我serve("/*").with(MyDispatcher.class);基本上评论如果我们关闭 guice 路由 websocket 开始工作

  4. 如果我关闭 guice-servlet 但在 web.xml 中添加一个 servlet 映射,如下所示 websocket 仍然有效

    < servlet-mapping > < servlet-name > HelloWorld< / servlet-name > < url-pattern > /* < / url-pattern > < / servlet-mapping >

我错过了什么或做错了什么?


编辑:

观察-连续:

  1. 我所做的是定义了一个简单的过滤器,它只响应FILTER.

public void doFilter(ServletRequest request, ServletResponse response, FilterChain chain) throws IOException, ServletException { response.getOutputStream().print("FILTER"); }

并将我的 web.xml 更改为

<web-app>           
    <filter>
        <filter-name>myFilter</filter-name>
        <filter-class>test.MyFilter</filter-class>
    </filter>
    <filter-mapping>
        <filter-name>myFilter</filter-name>
        <url-pattern>/*</url-pattern>
    </filter-mapping>       
</web-app>

现在按预期http://localhost:8080/app/x返回。FILTER但是尝试与 websocket 连接失败,因为请求显示了这样的内容。我还注意到,当我更改字符串时,在响应更改中返回内容长度,这意味着在 tomcat 为 websocket 处理它之前MyFilter已到达请求。MyFilter

在此处输入图像描述

  1. 我将 web.xml 更改为下面,guice 和 websocket 现在工作正常.. 所以我认为 Guice 不尊重在 GuiceFilter 之后注册的 WsFilter

    <filter>
    <filter-name>myFilter</filter-name>
    <filter-class>org.apache.tomcat.websocket.server.WsFilter</filter-class>
    </filter>
    <filter-mapping>
        <filter-name>myFilter</filter-name>
        <url-pattern>/*</url-pattern>
    </filter-mapping> 
    
     <filter>
        <filter-name>guiceFilter</filter-name>
        <filter-class>com.google.inject.servlet.GuiceFilter</filter-class>
    </filter>
    <filter-mapping>
        <filter-name>guiceFilter</filter-name>
        <url-pattern>/*</url-pattern>
    </filter-mapping>
    

TOMCAT 8.0、Window 7、Java 1.7、Guice 4.0、Guice-servlet-4.0

4

1 回答 1

3

这对我来说也像是一个 Guice 问题(正如评论中已经提到的)。在同一个应用程序中使用 servlet 和 WebSocket 应该不是问题,即使 servlet 映射覆盖/*.

关于 servlet 和过滤器的 2 件事:

  • 如前所述,当收到 websocket 升级请求(一种非常特殊的 http 请求)时,通常 Tomcat 不应该让它通过过滤器。看起来 Guice 以某种方式打破了这一点。
  • 过滤器按其映射在 web.xml 中定义的顺序应用。

所以,如果 WsFilter 在前,它会先拦截请求,然后检查是否是 WebSocket 升级请求
如果它确实是一个 WebSocket 连接,过滤器不会将它传递给链的其余部分
如果它是另一种类型的请求(GET、POST...),它将传递它,然后 Guice 会做它的事情。

(所以你在这里找到了第一个解决方案)

如果 Guice 过滤器是第一个,并且您使用serve("/*")...,那么它会破坏您的 WS。

如果你注释掉serve("/*")...,那么 Guice 过滤器是不是第一个都没有关系,WsFilter 甚至可以不存在:你的 WS 可以到达(单独建立 GuiceFilter 是可以的)。

所以 Guice 在 servlet 映射之上有自己的“拦截层”,我认为这就是破坏 WebSockets 的原因。我不知道 Guice 中是否存在错误或需要修复的任何内容(我的意思是可能,但不知道具体是什么),但您可以指定 Guice 的例外情况(与 中的 servlet 映射不同web.xml)。
替换这个:

serve("/*").with(TestServlet.class);

有了这个:

serveRegex("/(?!websocket/).*").with(TestServlet.class);
// Regex that accepts /.* but excludes /websocket/.*

这样,您可以将 servlet 映射保持在 上/*,并WsFilter在不需要时将其删除。(经过测试,对我有用)

所以这是第二种解决方案,它的优点是它还允许为非 WebSocket 的东西指定异常。

于 2017-05-31T20:03:06.947 回答