WebSockets With Spring, Part 2: WebSocket With SockJS Fallback
Introduction
According to some sources, the WebSocket API is currently (2020) implemented in the most common browsers:
But in addition to outdated browsers (mainly on mobile platforms), there are network intermediaries (proxies, firewalls, routers, etc.) that can prevent WebSocket communication. These intermediaries may not pass HTTP to WebSocket protocol upgrade or may close long-lived connections.
One possible solution to this problem is WebSocket emulation — first trying to use WebSocket and then falling back to HTTP-based techniques that expose the same API.
SockJS
Design
SockJS is one of the existing WebSocket browser fallbacks. SockJS includes a protocol, a JavaScript browser client, and Node.js, Erlang, Ruby servers. There are also third-party SockJS clients and servers for different programming languages and platforms. SockJS was designed to emulate the WebSocket API as close as possible to provide the ability to use WebSocket subprotocols on top of it.
SockJS has the following design considerations:
- simple browser API as close to WebSocket API as possible
- only JavaScript, no Flash/Java plugins
- support of scaling and load balancing techniques
- support of cross-domain communication and cookies
- fast connection establishment
- graceful degradation in case of outdated browsers and restrictive proxies
Gradceful degradation in web development is a design principle that focuses on trying to use the best features that work in newer browsers but falls back on other features that, while not as good, still provides essential functionality in older browsers.
SockJS supports a wide range of browser versions and has at least one long polling and streaming transport for each of them. It is important, that streaming transports support cross-domain communications and cookies. Cross-domain communication is required when SockJS is hosted on a different server than the main web application. Cookies are essential for authentication in web applications and cookie-based sticky sessions in load balancers.
SockJS and the same-origin policy
The same-origin policy is a critical concept in web application security. An origin is a combination of scheme, host name, and port number. This policy describes how a document or script loaded from one origin can interact with a resource from another origin. Some resources such as images, sounds, videos, stylesheets, fonts, iframes can be accessed across origins. Scripts can be loaded across origins as well, but some of their actions (such as cross-origin Ajax calls) are disabled by default.
SockJS and load balancers
SockJS supports sticky sessions in load balancers. With sticky sessions, a load balancer identifies a user and routes all of the requests from this user to a specific server. Among other methods, load balancers can use application cookies for this (for example, JSESSIONID cookie that uses Java Servlet containers for an HTTP session).
Some load balancers do not support WebSocket. In this case, it is necessary to exclude WebSocket from the list of available SockJS transports.
SockJS transports
SockJS transports fall into three categories: native WebSocket, HTTP streaming, HTTP long polling.
WebSocket transport
WebSocket is the transport with the best latency and throughput, and it has built-in support for cross-domain communication.
SockJS exists precisely because some browsers or network intermediaries still do not support WebSocket.
Streaming transports
SockJS streaming transports are based on HTTP 1.1 chunked transfer encoding (the Transfer-Encoding: chunked response header) that allows the browser to receive a single response from the server in many parts.
Every browser supports a different set of streaming transports and they usually do not support cross-domain communication. SockJS overcomes that limitation by using an iframe and communicating with it using cross-document messaging (the technique is known as iframe via postMessage).
SockJS has the following streaming transports:
- xhr-streaming — the transport using XMLHttpRequest object via streaming capability
- xdr-streaming — the transport using XDomainRequest object via streaming capability
- eventsource — the transport using EventSource object (Server-Sent Events)
- iframe-eventsource — the transport using EventSource object (Server-Sent Events) from an iframe via postMessage
- htmlfile — the transport using ActiveXObject HtmlFile object
- iframe-htmlfile — the transport using ActiveXObject HtmlFile object from an iframe via postMessage
XMLHttpRequest (XHR) is a browser interface to make Ajax calls.
XDomainRequest (XDR) is a deprecated browser interface in Internet Explorer to make Ajax calls.
Polling transports
SockJS supports a few polling transports for outdated browsers. SockJS uses slow and outdated JSONP transport when nothing else works.
SockJS has the following polling transports:
- xhr-polling — transport using XMLHttpRequest object
- xdr-polling — transport using XDomainRequest object
- iframe-xhr-polling — transport using XMLHttpRequest object from an iframe via postMessage
- jsonp-polling — transport using JSONP technique
SockJS protocol
The SockJS network protocol consists of two components:
- the opening handshake for identifying the parameters of the SockJS session
- the session for full-duplex communication
Before starting a session, a SockJS client sends a GET request to the /info server URL to obtain basic information from the server. The server responds with the following properties:
- websocket — the property to specify whether the server needs WebSocket transport
- cookie_needed — the property to specify whether the server needs cookie support
- origins — the list of allowed origins
- entropy — the source of entropy for the random number generator
After the handshake, the client decides which transport to use. If it is possible, the client selects WebSocket. If not, in most browsers there is at least one HTTP streaming transport. Otherwise, the client uses some HTTP long polling transport.
During a SockJS session, the client sends requests to the /server/session/transport server URLs, where:
- server — the identifier of a server in a cluster
- session — the identifier of a SockJS session
- transport — the transport type
A SockJS client accepts the following frames:
- “o” — open frame is sent by the server every time a new session is established
- “h” — heartbeat frame is periodically sent (if there is no message flow) by the server to prevent load balancers from closing connection by timeout
- “a” — messages frame is an array of JSON-encoded messages (for example, a[“Hello!”])
- “c” — close frame is sent by the server to close the session; close frame contains a code and a string explaining a reason for closure (for example, c[3000, “Go away!”])
A SockJS server does not define any framing. All incoming data is accepted as incoming messages, either single JSON-encoded messages or an array of JSON-encoded messages (depending on the transport).
Examples
Introduction
The Spring Framework provides support for WebSocket/SockJS clients and servers in the spring-websocket module.
The following example implements full-duplex WebSocket text communication with SockJS fallback between a server and clients. The server and the clients work according to the following algorithm:
- the server sends a one-time message to the client
- the server sends periodic messages to the client
- the server receives messages from a client, logs them, and sends them back to the client
- the client sends aperiodic messages to the server
- the client receives messages from a server and logs them
The server is implemented as a Spring web application with Spring Web MVC framework to handle static web resources. One client is implemented as a JavaScript browser client and another client is implemented as a Java Spring console application.
Java Spring server
Java Spring server consists of two parts: Spring WebSocket events handler and Spring WebSocket/SockJS configuration.
Because the server uses text (not binary) messages, the events handler extends the existing TextWebSocketHandler class as the required implementation of the WebSocketHandler interface. The handler uses the handleTextMessage callback method to receive messages from a client and the sendMessage method to send messages back to the client.
Existing Spring WebSocket event handlers do not support broadcasting messages to many clients. To implement this manually, the afterConnectionEstablished and afterConnectionClosed methods maintain the thread-safe list of active clients. The @Scheduled method broadcasts periodic messages to active clients with the same sendMessage method.
public class ServerWebSocketHandler extends TextWebSocketHandler implements SubProtocolCapable { private final Set<WebSocketSession> sessions = new CopyOnWriteArraySet<>(); @Override
public void afterConnectionEstablished(WebSocketSession session) throws Exception {
logger.info("Server connection opened");
sessions.add(session); TextMessage message = new TextMessage("one-time message from server");
logger.info("Server sends: {}", message);
session.sendMessage(message);
} @Override
public void afterConnectionClosed(WebSocketSession session, CloseStatus status) {
logger.info("Server connection closed: {}", status);
sessions.remove(session);
} @Scheduled(fixedRate = 10000)
void sendPeriodicMessages() throws IOException {
for (WebSocketSession session : sessions) {
if (session.isOpen()) {
String broadcast = "server periodic message " + LocalTime.now();
logger.info("Server sends: {}", broadcast);
session.sendMessage(new TextMessage(broadcast));
}
}
} @Override
public void handleTextMessage(WebSocketSession session, TextMessage message) throws Exception {
String request = message.getPayload();
logger.info("Server received: {}", request); String response = String.format("response from server to '%s'", HtmlUtils.htmlEscape(request));
logger.info("Server sends: {}", response);
session.sendMessage(new TextMessage(response));
} @Override
public void handleTransportError(WebSocketSession session, Throwable exception) {
logger.info("Server transport error: {}", exception.getMessage());
} @Override
public List<String> getSubProtocols() {
return Collections.singletonList("subprotocol.demo.websocket");
}
}
We can notice, that the Spring WebSocket events handlers for the servers with plain WebSocket and with WebSocket with SockJS fallback are the same.
The following Spring configuration enables WebSocket support in the Spring server with the @EnableWebSocket annotation. This configuration also registers the implemented WebSocket handler for the WebSocket endpoint with the SockJS fallback.
@Configuration
@EnableWebSocket
public class ServerWebSocketSockJsConfig implements WebSocketConfigurer { @Override
public void registerWebSocketHandlers(WebSocketHandlerRegistry registry) {
registry.addHandler(webSocketHandler(), "/websocket-sockjs")
.setAllowedOrigins("*")
.withSockJS()
.setWebSocketEnabled(true)
.setHeartbeatTime(25000)
.setDisconnectDelay(5000)
.setClientLibraryUrl("/webjars/sockjs-client/1.1.2/sockjs.js")
.setSessionCookieNeeded(false);
} @Bean
public WebSocketHandler webSocketHandler() {
return new ServerWebSocketHandler();
}
}
This configuration demonstrates some of the configuration properties available for a Spring SockJS server:
- allowedOrigins — this property can be used to specify allowed origins for CORS
- webSocketEnabled — this property can be used to disable the WebSocket transport if the load balancer does not support WebSocket
- heartbeatTime — this property can be used to specify the period with which server (if there is no message flow) sends heartbeat frames to the client to keep the connection from closing
- disconnectDelay — this property can be used to specify a timeout before closing an expired session
- clientLibraryUrl — this property can be used to specify the URL of the SockJS JavaScript client library for the iframe-based transports
- sessionCookieNeeded — this property can be used to specify whether the server needs cookies for load balancing or in Java Servlet containers to use HTTP session
The server is a Spring Boot web application with Spring Web MVC framework to handle static web resources for the JavaScript browser client. However, Spring WebSocket support does not depend on Spring MVC and can be used with any Java Servlet framework.
@SpringBootApplication
@EnableScheduling
public class ServerWebSocketSockJsApplicaion { public static void main(String[] args) {
SpringApplication.run(ServerWebSocketSockJsApplicaion.class, args);
}
}
JavaScript browser client
The JavaScript browser client uses the SockJS object from the SockJS library. It is important, that the client uses the “http” scheme (not the “ws” scheme) to specify the server URL.
When a user clicks the ‘Connect’ button, the client uses the SockJS constructor (with the server URL, the subprotocol, and the selected SockJS transports) to initiate a connection to the server. When the connection is established, the SockJS.onopen callback handler is called.
When the user clicks the ‘Disconnect’ button, the client uses the SockJS.close method to initiate the close of the connection. When the connection is closed, the SockJS.onclose callback handler is called.
let sockJS = null;// 'Connect' button click handler
function connect() {
const option = $("#transports").find('option:selected').val();
const transports = (option === 'all') ? [] : [option]; sockJS = new SockJS('http://localhost:8080/websocket-sockjs',
'subprotocol.demo.websocket', {debug: true, transports: transports}); sockJS.onopen = function () {
log('Client connection opened'); console.log('Subprotocol: ' + sockJS.protocol);
console.log('Extensions: ' + sockJS.extensions);
}; sockJS.onmessage = function (event) {
log('Client received: ' + event.data);
}; sockJS.onerror = function (event) {
log('Client error: ' + event);
}; sockJS.onclose = function (event) {
log('Client connection closed: ' + event.code);
};
}// 'Disconnect' button click handler
function disconnect() {
if (sockJS != null) {
sockJS.close();
sockJS = null;
}
}
When the user clicks the ‘Send’ button, the client uses the SockJS.send method to send a text message to the server.
// 'Send' button click handler
function send() {
var message = $("#request").val();
log('Client sends: ' + message);
sockJS.send(message);
}
When the client receives a message, the SockJS.onmessage callback handler is called. Incoming messages are received and outgoing messages are transmitted independently of each other.
We can notice, that the JavaScript browser client for the clients with plain WebSocket and with WebSocket with SockJS fallback are very similar (they differ only in the creation of the communicating object).
Java Spring client
Java Spring client consists of two parts: Spring WebSocket events handler and Spring WebSocket configuration.
The client (as the server) extends the existing TextWebSocketHandler class. The handler uses the handleTextMessage callback method to receive messages from a server and the sendMessage method to send messages back to the server.
public class ClientWebSocketHandler extends TextWebSocketHandler { @Override
public void afterConnectionEstablished(WebSocketSession session) {
logger.info("Client connection opened");
} @Override
public void afterConnectionClosed(WebSocketSession session, CloseStatus status) {
logger.info("Client connection closed: {}", status);
} @Override
public void handleTextMessage(WebSocketSession session, TextMessage message) {
logger.info("Client received: {}", message);
} @Override
public void handleTransportError(WebSocketSession session, Throwable exception) {
logger.info("Client transport error: {}", exception.getMessage());
}
}
We can notice, that the Spring WebSocket events handlers for the clients (as for the servers) with plain WebSocket and with WebSocket with SockJS fallback are the same.
The following Spring configuration enables WebSocket support in the Spring client. The configuration defines a WebSocketConnectionManager object, that uses two Spring beans:
- the SockJsClient class (from the spring-websocket dependency) as an implementation of the WebSocketClient interface — to connect to the WebSocket server
- the implemented WebSocketHandler — to handle WebSocket events during communication
The SockJsClient object uses two transports:
- the WebSocketTransport object, which supports SockJS WebSocket transport
- the RestTemplateXhrTransport object, which supports SockJS XhrStreaming and XhrPolling transports
@Configuration
public class ClientWebSocketSockJsConfig { @Bean
public WebSocketConnectionManager webSocketConnectionManager() {
WebSocketConnectionManager manager = new WebSocketConnectionManager(
webSocketClient(),
webSocketHandler(),
"http://localhost:8080/websocket-sockjs"
);
manager.setAutoStartup(true);
return manager;
} @Bean
public WebSocketClient webSocketClient() {
List<Transport> transports = new ArrayList<>();
transports.add(new WebSocketTransport(new StandardWebSocketClient()));
transports.add(new RestTemplateXhrTransport());
return new SockJsClient(transports);
} @Bean
public WebSocketHandler webSocketHandler() {
return new ClientWebSocketHandler();
}
}
We can notice, that the Spring WebSocket configuration with plain WebSocket and with WebSocket with SockJS fallback are very similar (they differ only in the used implementation of the WebSocketClient interface).
The client is a console Spring Boot application without Spring Web MVC.
@SpringBootApplication
public class ClientWebSocketSockJsApplication { public static void main(String[] args) {
new SpringApplicationBuilder(ClientWebSocketSockJsApplication.class)
.web(WebApplicationType.NONE)
.run(args);
}
}
Conclusion
There are two strategies to deal with the absence of WebSocket support in browsers and network infrastructure: emulation and extensions. The first strategy (SockJS follows it) is to emulate the WebSocket API as close as possible. The second strategy (most of the other fallbacks follow it) is to build a top-level API and use WebSocket as one of the transports along with Flash/Java plugins or Comet techniques.
Emulation strategy does not provide any additional API on top of the WebSocket protocol and might require additional development. However, this strategy may be beneficial in the long term when all browsers will eventually get WebSocket support, and fallbacks are no longer needed.
Many commercial operators provide their solutions that include WebSocket as one of the protocols: Kaazing WebSocket Gateway, PubNub, Pusher, Ably, etc. Although all these solutions are non-compliant and proprietary, using them can be a reasonable decision to get a business solution right now and do not deal with the Web development that requires a lot of time and expertise.
Complete code examples are available in the GitHub repository.