org.cometd.websocket.server.AbstractWebSocketTransport Maven / Gradle / Ivy
/*
* Copyright (c) 2008-2014 the original author or authors.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package org.cometd.websocket.server;
import java.io.IOException;
import java.text.ParseException;
import java.util.ArrayList;
import java.util.Collections;
import java.util.List;
import java.util.Queue;
import java.util.concurrent.Executor;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
import java.util.concurrent.ScheduledExecutorService;
import java.util.concurrent.ScheduledFuture;
import java.util.concurrent.TimeUnit;
import org.cometd.bayeux.Channel;
import org.cometd.bayeux.Message;
import org.cometd.bayeux.server.BayeuxContext;
import org.cometd.bayeux.server.ServerMessage;
import org.cometd.bayeux.server.ServerSession;
import org.cometd.server.AbstractServerTransport;
import org.cometd.server.BayeuxServerImpl;
import org.cometd.server.ServerSessionImpl;
import org.eclipse.jetty.util.Callback;
import org.eclipse.jetty.util.ConcurrentArrayQueue;
import org.eclipse.jetty.util.IteratingCallback;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
public abstract class AbstractWebSocketTransport extends AbstractServerTransport
{
public static final String NAME = "websocket";
public static final String PREFIX = "ws";
public static final String PROTOCOL_OPTION = "protocol";
public static final String MESSAGES_PER_FRAME_OPTION = "messagesPerFrame";
public static final String BUFFER_SIZE_OPTION = "bufferSize";
public static final String MAX_MESSAGE_SIZE_OPTION = "maxMessageSize";
public static final String IDLE_TIMEOUT_OPTION = "idleTimeout";
public static final String THREAD_POOL_MAX_SIZE = "threadPoolMaxSize";
public static final String COMETD_URL_MAPPING = "cometdURLMapping";
private final ThreadLocal _bayeuxContext = new ThreadLocal<>();
private Executor _executor;
private ScheduledExecutorService _scheduler;
private String _protocol;
private int _messagesPerFrame;
protected AbstractWebSocketTransport(BayeuxServerImpl bayeux)
{
super(bayeux, NAME);
setOptionPrefix(PREFIX);
}
@Override
public void init()
{
super.init();
_executor = newExecutor();
_scheduler = newScheduledExecutor();
_protocol = getOption(PROTOCOL_OPTION, null);
_messagesPerFrame = getOption(MESSAGES_PER_FRAME_OPTION, 1);
}
@Override
public void destroy()
{
_scheduler.shutdownNow();
Executor threadPool = _executor;
if (threadPool instanceof ExecutorService)
((ExecutorService)threadPool).shutdown();
super.destroy();
}
protected Executor newExecutor()
{
int size = getOption(THREAD_POOL_MAX_SIZE, 64);
return Executors.newFixedThreadPool(size);
}
protected ScheduledExecutorService newScheduledExecutor()
{
return Executors.newSingleThreadScheduledExecutor();
}
public Executor getExecutor()
{
return _executor;
}
public ScheduledExecutorService getScheduler()
{
return _scheduler;
}
public String getProtocol()
{
return _protocol;
}
public int getMessagesPerFrame()
{
return _messagesPerFrame;
}
protected boolean checkProtocol(List serverProtocols, List clientProtocols)
{
if (serverProtocols.isEmpty())
return true;
for (String clientProtocol : clientProtocols)
{
if (serverProtocols.contains(clientProtocol))
return true;
}
return false;
}
@Override
public BayeuxContext getContext()
{
return _bayeuxContext.get();
}
protected List normalizeURLMapping(String urlMapping)
{
String[] mappings = urlMapping.split(",");
List result = new ArrayList<>(mappings.length);
for (String mapping : mappings)
{
if (mapping.endsWith("/*"))
mapping = mapping.substring(0, mapping.length() - 2);
if (!mapping.startsWith("/"))
mapping = "/" + mapping;
result.add(mapping);
}
return result;
}
protected void handleJSONParseException(S wsSession, ServerSession session, String json, Throwable exception)
{
_logger.warn("Error parsing JSON: " + json, exception);
}
protected void handleException(S wsSession, ServerSession session, Throwable exception)
{
if (_logger.isDebugEnabled())
_logger.debug("", exception);
}
protected abstract void send(S wsSession, ServerSession session, String data, Callback callback);
protected void onClose(int code, String reason)
{
}
protected abstract class AbstractWebSocketScheduler implements AbstractServerTransport.Scheduler
{
protected final Logger _logger = LoggerFactory.getLogger(getClass().getName() + "." + Integer.toHexString(System.identityHashCode(this)));
private final Flusher flusher = new Flusher();
private final BayeuxContext _context;
private volatile ServerSessionImpl _session;
private ServerMessage.Mutable _connectReply;
private ScheduledFuture> _connectTask;
protected AbstractWebSocketScheduler(BayeuxContext context)
{
_context = context;
}
protected void send(S wsSession, List messages, int batchSize, Callback callback)
{
if (messages.isEmpty())
{
callback.succeeded();
return;
}
int size = messages.size();
int batch = Math.min(batchSize, size);
// Assume 4 fields of 48 chars per message
int capacity = batch * 4 * 48;
StringBuilder builder = new StringBuilder(capacity);
builder.append("[");
if (batch == 1)
{
// Common path.
ServerMessage serverMessage = messages.remove(0);
builder.append(serverMessage.getJSON());
}
else
{
boolean comma = false;
for (int b = 0; b < batch; ++b)
{
ServerMessage serverMessage = messages.get(b);
if (comma)
builder.append(",");
comma = true;
builder.append(serverMessage.getJSON());
}
if (batch == size)
messages.clear();
else
messages.subList(0, batch).clear();
}
builder.append("]");
AbstractWebSocketTransport.this.send(wsSession, _session, builder.toString(), callback);
}
protected void onClose(int code, String reason)
{
final ServerSessionImpl session = _session;
if (session != null)
{
// There is no need to call BayeuxServerImpl.removeServerSession(),
// because the connection may have been closed for a reload, so
// just null out the current session to have it retrieved again
_session = null;
session.startIntervalTimeout(getInterval());
cancelMetaConnectTask(session);
}
if (_logger.isDebugEnabled())
_logger.debug("Closing {}/{} - {}", code, reason, session);
AbstractWebSocketTransport.this.onClose(code, reason);
}
protected void onError(Throwable failure)
{
_logger.info("WebSocket Error", failure);
}
protected boolean cancelMetaConnectTask(ServerSessionImpl session)
{
final ScheduledFuture> connectTask;
synchronized (session.getLock())
{
connectTask = _connectTask;
_connectTask = null;
}
if (connectTask == null)
return false;
if (_logger.isDebugEnabled())
_logger.debug("Cancelling meta connect task {}", connectTask);
connectTask.cancel(false);
return true;
}
protected void onMessage(S wsSession, String data)
{
_bayeuxContext.set(_context);
getBayeux().setCurrentTransport(AbstractWebSocketTransport.this);
try
{
ServerMessage.Mutable[] messages = parseMessages(data);
if (_logger.isDebugEnabled())
_logger.debug("Received {}", data);
processMessages(wsSession, messages);
}
catch (ParseException x)
{
close(1011, x.toString());
handleJSONParseException(wsSession, _session, data, x);
}
catch (Throwable x)
{
close(1011, x.toString());
handleException(wsSession, _session, x);
}
finally
{
_bayeuxContext.set(null);
getBayeux().setCurrentTransport(null);
}
}
private void processMessages(S wsSession, ServerMessage.Mutable[] messages) throws IOException
{
ServerSessionImpl session = _session;
boolean startInterval = false;
boolean send = true;
List queue = Collections.emptyList();
List replies = new ArrayList<>(messages.length);
for (int i = 0; i < messages.length; i++)
{
ServerMessage.Mutable message = messages[i];
if (_logger.isDebugEnabled())
_logger.debug("Processing {}", message);
// Get the session from the message
String clientId = message.getClientId();
if (session == null || !session.getId().equals(clientId))
_session = session = (ServerSessionImpl)getBayeux().getSession(message.getClientId());
// Session expired concurrently ?
if (session != null && !session.isHandshook())
_session = session = null;
switch (message.getChannel())
{
case Channel.META_HANDSHAKE:
{
if (messages.length > 1)
throw new IOException();
ServerMessage.Mutable reply = processMetaHandshake(session, message);
if (reply != null)
session = (ServerSessionImpl)getBayeux().getSession(reply.getClientId());
replies.add(processReply(session, reply));
break;
}
case Channel.META_CONNECT:
{
if (messages.length > 1)
throw new IOException();
ServerMessage.Mutable reply = processMetaConnect(session, message);
replies.add(processReply(session, reply));
send = startInterval = reply != null;
if (send && session != null)
{
if (isMetaConnectDeliveryOnly() || session.isMetaConnectDeliveryOnly())
queue = session.takeQueue();
}
break;
}
default:
{
ServerMessage.Mutable reply = getBayeux().handle(session, message);
replies.add(processReply(session, reply));
break;
}
}
}
if (send)
send(wsSession, session, startInterval, queue, replies);
}
private ServerMessage.Mutable processMetaHandshake(ServerSessionImpl session, ServerMessage.Mutable message)
{
ServerMessage.Mutable reply = getBayeux().handle(session, message);
if (reply != null && reply.isSuccessful())
{
session = (ServerSessionImpl)getBayeux().getSession(reply.getClientId());
if (session != null)
session.setScheduler(this);
}
return reply;
}
private ServerMessage.Mutable processMetaConnect(ServerSessionImpl session, ServerMessage.Mutable message)
{
// Remember the connected status before handling the message.
boolean wasConnected = session != null && session.isConnected();
ServerMessage.Mutable reply = getBayeux().handle(session, message);
if (reply != null && session != null)
{
if (reply.isSuccessful() && session.isConnected())
{
// We need to set the scheduler again, in case the connection
// has temporarily broken and we have created a new scheduler.
session.setScheduler(this);
// If we deliver only via meta connect and we have messages, then reply.
boolean metaConnectDelivery = isMetaConnectDeliveryOnly() || session.isMetaConnectDeliveryOnly();
boolean hasMessages = session.hasNonLazyMessages();
boolean replyToMetaConnect = hasMessages && metaConnectDelivery;
if (!replyToMetaConnect)
{
long timeout = session.calculateTimeout(getTimeout());
boolean holdMetaConnect = timeout > 0 && wasConnected;
if (holdMetaConnect)
{
// Decide atomically if we need to hold the meta connect or not
// In schedule() we decide atomically if reply to the meta connect.
synchronized (session.getLock())
{
if (!session.hasNonLazyMessages())
{
if (cancelMetaConnectTask(session))
{
if (_logger.isDebugEnabled())
_logger.debug("Cancelled unresponded meta connect {}", _connectReply);
}
_connectReply = reply;
// Delay the connect reply until timeout.
long expiration = TimeUnit.NANOSECONDS.toMillis(System.nanoTime()) + timeout;
_connectTask = getScheduler().schedule(new MetaConnectReplyTask(reply, expiration), timeout, TimeUnit.MILLISECONDS);
if (_logger.isDebugEnabled())
_logger.debug("Scheduled meta connect {}", _connectTask);
reply = null;
}
}
}
}
}
if (reply != null && session.isDisconnected())
reply.getAdvice(true).put(Message.RECONNECT_FIELD, Message.RECONNECT_NONE_VALUE);
}
return reply;
}
private ServerMessage.Mutable processReply(ServerSessionImpl session, ServerMessage.Mutable reply)
{
if (reply != null)
{
reply = getBayeux().extendReply(session, session, reply);
if (reply != null)
getBayeux().freeze(reply);
}
return reply;
}
protected void send(S wsSession, ServerSessionImpl session, boolean startInterval, List queue, List replies)
{
if (_logger.isDebugEnabled())
_logger.debug("Sending {}, replies={}, messages={}", session, replies, queue);
flusher.queue(new Entry<>(wsSession, session, startInterval, queue, replies));
flusher.iterate();
}
protected abstract void close(int code, String reason);
@Override
public void cancel()
{
final ServerSessionImpl session = _session;
if (session != null)
{
if (cancelMetaConnectTask(session))
close(1000, "Cancel");
}
}
@Override
public void schedule()
{
// This method may be called concurrently, for example when
// two clients publish concurrently on the same channel.
schedule(false, null);
}
protected abstract void schedule(boolean timeout, ServerMessage.Mutable expiredConnectReply);
protected void schedule(S wsSession, boolean timeout, ServerMessage.Mutable expiredConnectReply)
{
// This method may be executed concurrently by threads triggered by
// schedule() and by the timeout thread that replies to the meta connect.
ServerSessionImpl session = _session;
try
{
if (session == null)
{
if (_logger.isDebugEnabled())
_logger.debug("No session, skipping reply {}", expiredConnectReply);
return;
}
// Decide atomically if we have to reply to the meta connect
// We need to guarantee the metaConnectDeliverOnly semantic
// and allow only one thread to reply to the meta connect
// otherwise we may have out of order delivery.
boolean metaConnectDelivery = isMetaConnectDeliveryOnly() || session.isMetaConnectDeliveryOnly();
boolean reply = false;
ServerMessage.Mutable connectReply;
synchronized (session.getLock())
{
connectReply = _connectReply;
if (timeout && connectReply != expiredConnectReply)
{
// We had a second meta connect arrived while we were expiring the first:
// just ignore to reply to the first connect as if we were able to cancel it
if (_logger.isDebugEnabled())
_logger.debug("Flushing skipped replies that do not match: {} != {}", connectReply, expiredConnectReply);
return;
}
if (connectReply == null)
{
if (metaConnectDelivery)
{
// If we need to deliver only via meta connect, but we
// do not have one outstanding, wait until it arrives
if (_logger.isDebugEnabled())
_logger.debug("Flushing skipped since metaConnectDelivery={}, metaConnectReply={}", metaConnectDelivery, connectReply);
return;
}
}
else
{
if (timeout || metaConnectDelivery || !session.isConnected())
{
// We will reply to the meta connect, so cancel the timeout task
cancelMetaConnectTask(session);
_connectReply = null;
reply = true;
}
}
}
List replies = Collections.emptyList();
if (reply)
{
if (session.isDisconnected())
connectReply.getAdvice(true).put(Message.RECONNECT_FIELD, Message.RECONNECT_NONE_VALUE);
connectReply = processReply(session, connectReply);
replies = new ArrayList<>(1);
replies.add(connectReply);
}
List queue = session.takeQueue();
if (_logger.isDebugEnabled())
_logger.debug("Flushing {} timeout={} metaConnectDelivery={}, metaConnectReply={}, messages={}", session, timeout, metaConnectDelivery, connectReply, queue);
send(wsSession, session, reply, queue, replies);
}
catch (Throwable x)
{
close(1011, x.toString());
handleException(wsSession, session, x);
}
}
private class MetaConnectReplyTask implements Runnable
{
private final ServerMessage.Mutable _connectReply;
private final long _connectExpiration;
private MetaConnectReplyTask(ServerMessage.Mutable connectReply, long connectExpiration)
{
this._connectReply = connectReply;
this._connectExpiration = connectExpiration;
}
@Override
public void run()
{
long now = TimeUnit.NANOSECONDS.toMillis(System.nanoTime());
long delay = now - _connectExpiration;
if (_logger.isDebugEnabled())
{
if (delay > 5000) // TODO: make the max delay a parameter ?
_logger.debug("/meta/connect {} expired {} ms too late", _connectReply, delay);
}
// Send the meta connect response after timeout.
// We *must* execute the next schedule() otherwise
// the client will timeout the meta connect, so we
// do not care about flipping the _scheduling field.
schedule(true, _connectReply);
}
}
private class Flusher extends IteratingCallback
{
private Queue> _entries = new ConcurrentArrayQueue<>();
private boolean queue(Entry entry)
{
return _entries.offer(entry);
}
@Override
protected Action process() throws Exception
{
Entry entry = _entries.peek();
if (_logger.isDebugEnabled())
_logger.debug("Processing {}", entry);
if (entry == null)
return Action.IDLE;
S wsSession = entry._wsSession;
List queue = entry._queue;
if (!queue.isEmpty())
{
// Under load, it is possible that we have many bayeux messages and
// that these would generate a large websocket message that the client
// could not handle, so we need to split the messages into batches.
int size = queue.size();
int messagesPerFrame = getMessagesPerFrame();
int batchSize = messagesPerFrame > 0 ? Math.min(messagesPerFrame, size) : size;
if (_logger.isDebugEnabled())
_logger.debug("Processing queue, batch size {}: {}", batchSize, queue);
send(wsSession, queue, batchSize, this);
return Action.SCHEDULED;
}
_entries.poll();
// Start the interval timeout after writing the messages
// since they may take time to be written, even in case
// of exceptions to make sure the session can be swept.
if (entry._startInterval)
{
ServerSessionImpl session = entry._session;
if (session != null)
session.startIntervalTimeout(getInterval());
}
List replies = entry._replies;
if (_logger.isDebugEnabled())
_logger.debug("Processing replies {}", replies);
send(wsSession, replies, replies.size(), this);
return Action.SCHEDULED;
}
@Override
protected void completed()
{
// This IteratingCallback never completes.
}
}
private class Entry
{
private final S _wsSession;
private final ServerSessionImpl _session;
private final boolean _startInterval;
private final List _queue;
private final List _replies;
private Entry(S wsSession, ServerSessionImpl session, boolean startInterval, List queue, List replies)
{
this._wsSession = wsSession;
this._session = session;
this._startInterval = startInterval;
this._queue = queue;
this._replies = replies;
}
@Override
public String toString()
{
return String.format("%s@%x[messages=%d,replies=%d]",
getClass().getSimpleName(),
hashCode(),
_queue.size(),
_replies.size());
}
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy