WebSocket Container

This chapter describes the concepts, functions, and configuration of the WebSocket container.

1. Overview

The WebSocket container is included in the web engine and one exists for every web context.The WebSocket container provided by JEUS complies with JSR 356 and Java API for WebSocket. Thus, the server’s WebSocket service must be written according to the API defined by the relevant standard. But, this document does not include detailed information about WebSocket RFC6455 and Java API for WebSocket standards.

  1. Since the HTML5 WebSocket API is a client library, it is irrelevant to WebtoB and JEUS.

  2. The Java API for WebSocket is included in Jakarta EE 8. Therefore, reference Jakarta EE 8 libraries to develop applications based on Jakarta EE 8.

The basic functions of the WebSocket container are to deploy WebSocket Server Endpoint objects included in the web context and connect a WebSocket Handshake request from the client to a Server Endpoint object that it maps to. Once a client and a WebSocket container are connected, WebSocket sessions are created and the life cycle of the session objects are also managed.

The role of a service developer is to implement a WebSocket server endpoint class (jakarta.websocket.Endpoint) and a configuration class (jakarta.websocket.server.ServerApplicationConfig), and package them into the web application.

2. Constraints

The following are the constraints for using WebSocket containers in JEUS 9.

  • Basically available for HTTP listeners. To use websocket containers in association with WebtoB, refer to the manual of WebtoB.

  • Detailed settings are possible under the <websocket> tag in jeus-web-dd.xml, but websockets can be used without those settings.

  • The client API provided by the Java API for WebSocket standard is provided through lib/system/jeus-websocket-client.jar. To use another vendor’s client module, if the full name of the another vendor’s provider class in the META-INF/service/jakarta.websocket.ContainerProvider file of the above jar file is specified, then the module can be used via the service loader.

3. WebSocket Container Functions

This chapter describes the additional functions of the WebSocket container.

3.1. WebSocket UserProperties Failover

A WebSocket container provides space in memory for storing data per WebSocket session. The map object (hereafter UserProperties) obtained by calling the jakarta.websocket.Session.getUserProperties() API contains the data.

Assume that WebSocket applications (web context including the WebSocket Server Endpoint) are deployed to two different servers. If data are saved in the UserProperties by using the WebSocket and the server is abnormally terminated, all data will be lost. However, if the data are backed up on another server, the UserProperties can be recovered by connecting the WebSocket to the backup server even if the server is abnormally terminated. From the user perspective, the WebSocket service can be used seamlessly. This is called WebSocket UserProperties Failover or Distributed WebSocket UserProperties.

The existing user can get the data in UserProperties stored in advance, but the failed over WebSocket endpoint is an endpoint that exists in a different server (not the existing server), and WebSocket Session is newly created. Therefore, this function must be used in very limited cases.

Considerations

The following conditions must be met to use this function.

  • WebSocket UserProperties Failover operates based on the HTTP session service. The WebSocket session cluster must be supported. For more information about the session cluster, refer to Session Cluster Modes in JEUS Session Management Guide.

  • The following must be added to the jeus-web-dd.xml configuration file of the WebSocket application.

    WebSocket UserProperties Failover Setting : <jeus-web-dd.xml>
    <websocket>
        <distributed-userProperties>
            <enabled>true</enabled>
        </distributed-userProperties>
    </websocket>

    In general, the WebSocket session and HTTP session are not integrated for use.

  • Data in the UserProperties object must be serializable.

  • WebSocket connection is initiated by an HTTP request. The cookie header of the WebSocket Handshake request must include the JSESSIONID. The HTTP session that the JSESSIONID points to must be created in the web context that the server endpoint belongs to.

    For instance, when the JSP that calls the HTML5 WebSocket API is called, the HTTP session is created to send JSESSIONID to the cookie header. When HTML5 WebSocket JavaScript is executed in the web browser, the WebSocket Handshake request is sent to JEUS. The cookie header must be included in the request header. WebSocket UserProperties Failover cannot be used if the WebSocket client does not support this.

    Firefox 30.0 attaches the cookie header to the WebSocket Handshake request when HTML5 WebSocket API is used.

    GET /service/chat HTTP/1.1
    Connection: keep-alive, Upgrade
    Cookie: JSESSIONID=FheDy8e0bOTPO7KNqdeJ7Eps8j51CaQqcRWHvpYo9mdVw1BCSwlwrFiyrclsolkr.amV1czcvc2VydmVyMg==
    Sec-WebSocket-Key: Csv/FCQo1g1eZfqMtPd8+g==
    Sec-WebSocket-Version: 13
    Upgrade: websocket
    User-Agent: Mozilla/5.0 (Windows NT 6.1; WOW64; rv:30.0) Gecko/20100101 Firefox/30.0 FirePHP/0.7.4

When this function is used with the WebtoB reverse proxy, the session routing must be configured in the reverse proxy configuration. For detailed information, refer to the WebtoB guide.

Considerations

Note the following when using this function.

  • The HTTP session’s timeout and the WebSocket session’s timeout are not related.

    Especially in the case of a WebSocket session, since it is dependent on the HTTP session, the HTTP session properties cannot be modified from the WebSocket container level. When serializable data are added(put) to the UserProperties object, depending on the circumstances the HTTP session can time out first which prevents failover from occurring. HTTP session’s timeout must be adjusted appropriately taking such situations into consideration.

    If the HTTP session times out when adding (put) serializable data to the UserProperties, a WARNING level message will be logged.

  • The UserProperties of WebSocket sessions are not shared between different web contexts.

3.2. Other Functions

Information that cannot be obtained through the jakarta.websocket.Session API

The following information can be obtained through Map<String, Object> which is a return value of Session.getUserProperties().

Item Description

jeus.websocket.remoteAddr(String)

If Forwarded or X-Forwarded-For exists in an HTTP request header, the header value will be used.

If not, a HttpServletRequest.getRemoteAddr() value will be used.

jeus.websocket.remoteHost(String)

If Forwarded or X-Forwarded-For exists in an HTTP request header, the header value will be used.

If Forwarded or X-Forwarded-For exists in an HTTP request header, the header value will be used. If not, a HttpServletRequest.getRemoteHost() value will be used.

jeus.websocket.remotePort(String)

If Forwarded or X-Forwarded-For exists in an HTTP request header, a port value included in the header will be used. If a header does not have a port value, this property will be not supported.

If there is no header, a HttpServletRequest.getRemotePort() value will be used. If the value is less than or equal to 0, it will be invalid.

4. Configuring WebSocket Container

Configure the WebSocket container settings in jeus-web-dd.xml of each web application.

Web Context Configuration File: <jeus-web-dd.xml>
<jeus-web-dd xmlns="http://www.tmaxsoft.com/xml/ns/jeus" version="9">
    <websocket>
      <max-incoming-binary-message-buffer-size>8192</max-incoming-binary-message-buffer-size>
      <max-incoming-text-message-buffer-size>8192</max-incoming-text-message-buffer-size>
      <max-session-idle-timeout-in-millis>1800000</max-session-idle-timeout-in-millis>
      <monitoring-period-in-millis>300000</monitoring-period-in-millis>
      <blocking-send-timeout-in-millis>10000</blocking-send-timeout-in-millis>
      <async-send-timeout-in-millis>30000</async-send-timeout-in-millis>
      <websocket-executor>
          <min>10</min>
          <max>30</max>
          <keep-alive-time>60000</keep-alive-time>
          <queue-size>4096</queue-size>
      </websocket-executor>
      <distributed-userProperties>
          <enabled>false</enabled>
          <use-write-through-policy>false</use-write-through-policy>
      </distributed-userProperties>
      <init-param>
          <name>name</name>
          <value>value</value>
      </init-param>
    </websocket>
    <upgrade>
        <upgrade-executor>
            <min>0</min>
            <max>30</max>
            <keep-alive-time>60000</keep-alive-time>
            <queue-size>4096</queue-size>
        </upgrade-executor>
    </upgrade>
</jeus-web-dd>

The following describes each configuration tag.

  • <websocket>

    Tag Description

    <max-incoming-binary-message-buffer-size>

    Maximum size of the buffer used for text messages transmitted from the client.

    If the message size is larger than this value, the 1009 error occurs and the WebSocket session is closed.

    <max-incoming-text-message-buffer-size>

    Maximum size of the buffer used for text messages transmitted from the client.

    If the message size is larger than this value, the 1009 error occurs and the WebSocket session is closed.

    <max-session-idle-timeout-in-millis>

    Timeout for closing idle WebSocket sessions.

    If the value is larger than zero and smaller than 1000, it is set to 1000. (Default value: 30 minutes (1800000ms))

    <monitoring-period-in-millis>

    Cycle for checking the WebSocket session timeout.

    If the value is smaller than 1000, it is set to 1000. (Default value: 5 minutes (300000ms))

    <blocking-send-timeout-in-millis>

    Timeout for waiting for a synchronous send.

    Applied when using jakarta.websocket.RemoteEndpoint.Basic.

    (Default value: 10 seconds)

    <async-send-timeout-in-millis>

    Timeout for waiting for messages awaiting to be sent asynchronously.

    Returned by jakarta.websocket.WebSocketContainer.getDefaultAsyncSendTimeout().

    <websocket-executor>

    Deprecated. Use <upgrade-executor> under <upgrade> instead.

    <distributed-userProperties>

    WebSocket Session Failover related settings depending on what is defined in the jakarta.websocket.Session.getUserProperties() method.

    Set the following sub-tags.

    • <enabled>: Option to use WebSocket session failover. This option is usually not used because it needs to connect to an HTTP session.

    • <use-write-through-policy>: Option to wait for backup server synchronization when processing put/remove into/from UserProperties of a WebSocket session. This option is usually not used, and the synchronization is processed in background.

    <init-param>

    Additional configurations used for a WebSocket container.

    Set a parameter name to <name> and a value to <value>.

  • <upgrade><upgrade-executor>

    A thread pool configuration used internally by the container to process Upgrade inbound messages. It is mainly used for processing the HTTP Upgrade handler during NIO Servlet processing, WebSocket endpoint processing, and handling the SendHandler during WebSocket asynchronous send operations.

    Tag Description

    <keep-alive-time>

    Timeout for removing idle threads from a thread pool. If set to 0, threads are not removed. (Default value: 1 minute (60000ms))

    <queue-size>

    Size of a queue that stores tasks processed by a thread pool. (Default value: 4096)

5. Using Spring WebSocket

This section describes how to use Spring WebSocket in JEUS.

Spring WebSocket offers an additional feature within the Spring Framework, leveraging the WebSocket container provided by the web engine. This WebSocket container adheres to JSR 356, the Java API for WebSocket, and offers compatible interfaces. However, importing the WebSocket container may vary depending on the vendor.

Since Spring WebSocket supports importing the WebSocket container from only specific vendors, JEUS provides an additional library for utilizing Spring WebSocket. This library requires specific configurations.

The library provided with JEUS is compatible with the Spring Framework 4.2.0 or later.

Add the following configuration for the Spring WebSocket-supporting library.

Web Context Configuration File: <jeus-web-dd.xml>
<jeus-web-dd xmlns="http://www.tmaxsoft.com/xml/ns/jeus" version="9">
    <library-ref>
        <library-name>spring-support</library-name>
        <specification-version>
            <value>4.2.0.RELEASE</value>
        </specification-version>
        <failon-error>true</failon-error>
    </library-ref>
</jeus-web-dd>

To use the Spring WebSocket-supporting library, specify the websocket handshake-handler as jeus.spring.websocket.JeusHandshakeHandler.

Spring Context Configuration File: <spring-context.xml>
<beans xmlns="http://www.springframework.org/schema/beans"
    xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
    xmlns:websocket="http://www.springframework.org/schema/websocket"
    xmlns:schemaLocation="
        http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-4.2.xsd
        http://www.springframework.org/schema/websocket http://www.springframework.org/schema/websocket/spring-websocket-4.2.xsd">

    <websocket:handlers>
        <websocket:mapping path="/chat" handler="chatHandler"/>
        <websocket:handshake-handler ref="handshakeHandler"/>
    </websocket:handlers>

    <bean id="chatHandler" class="com.tmax.jeus.ChatHandler"/>
    <bean id="handshakeHandler" class="jeus.spring.websocket.JeusHandshakeHandler"/>
</beans>