我有一个Grails应用程序,它有一个简单的Websocket端点,如下所示:
@WebListener
@ServerEndpoint("/chatEndpoint/{chatId}")
public class WebsocketChatroomEndpoint implements ServletContextListener {
static ConfigObject config
@Override
public void contextInitialized(ServletContextEvent sce) {
ServletContext servletContext = sce.servletContext
ServerContainer serverContainer = servletContext.getAttribute("javax.websocket.server.ServerContainer")
try {
ApplicationContext ctx = (ApplicationContext) servletContext.getAttribute(GA.APPLICATION_CONTEXT)
config = ctx.grailsApplication.config
serverContainer.defaultMaxSessionIdleTimeout = 0
} catch (IOException e) {
log.error(e.message, e)
}
}
@Override
public void contextDestroyed(ServletContextEvent servletContextEvent) {
}
@OnOpen
public void onOpen(Session userSession, @PathParam("chatId") String chatId) {
// Do some stuff
}
@OnMessage
public String onMessage(String message, Session userSession) throws IOException {
// Handle message received
}
@OnClose
public void onClose(Session userSession, CloseReason reason) {
// Handle close
}
@OnError
public void onError(Throwable t) {
// Handle error
}
}
端点工作正常,但我遇到的问题是,在部署到Tomcat 8时,空闲连接会在60秒后断开连接。
Tomcat Websocket How-To表示可以通过设置以下servlet上下文初始化参数来更改此值:org.apache.tomcat.websocket.executorKeepAliveTimeSeconds
。
我已将此参数设置在web.xml
文件中,如下所示:
<context-param>
<param-name>org.apache.tomcat.websocket.executorKeepAliveTimeSeconds</param-name>
<param-value>1000</param-value>
</context-param>
而且,当我在Listener的contextInitialized
方法中访问参数时,该值设置正确。
@Override
public void contextInitialized(ServletContextEvent sce) {
ServletContext servletContext = sce.servletContext
// Logs "1000"
log.debug(servletContext.getInitParameter("org.apache.tomcat.websocket.executorKeepAliveTimeSeconds"))
}
问题是,无论我将此值设置为什么,端点在60秒后仍会超时。我已经尝试在Tomcat的context.xml
文件中设置值,并在我的监听器中以编程方式设置它。有没有人知道如何覆盖Tomcat的默认值60秒?
答案 0 :(得分:3)
我已经解决了这个问题,在对此进行排查时,我显然不在意。
我有与Tomcat的nginx反向代理连接。 Tomcat正确设置了值,但nginx中的proxy_read_timeout
默认为60秒。在我的nginx站点配置中增加此值可以解决问题。