org.cometd.server.BayeuxServerImpl Maven / Gradle / Ivy
/*
* Copyright (c) 2008-2019 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.server;
import java.io.IOException;
import java.lang.reflect.Constructor;
import java.security.SecureRandom;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collections;
import java.util.HashSet;
import java.util.Iterator;
import java.util.LinkedHashMap;
import java.util.List;
import java.util.ListIterator;
import java.util.Map;
import java.util.Set;
import java.util.TreeMap;
import java.util.TreeSet;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.ConcurrentMap;
import java.util.concurrent.CopyOnWriteArrayList;
import java.util.concurrent.TimeUnit;
import javax.servlet.http.HttpServletRequest;
import org.cometd.bayeux.Channel;
import org.cometd.bayeux.ChannelId;
import org.cometd.bayeux.MarkedReference;
import org.cometd.bayeux.Message;
import org.cometd.bayeux.server.Authorizer;
import org.cometd.bayeux.server.BayeuxContext;
import org.cometd.bayeux.server.BayeuxServer;
import org.cometd.bayeux.server.ConfigurableServerChannel.Initializer;
import org.cometd.bayeux.server.ConfigurableServerChannel.ServerChannelListener;
import org.cometd.bayeux.server.LocalSession;
import org.cometd.bayeux.server.SecurityPolicy;
import org.cometd.bayeux.server.ServerChannel;
import org.cometd.bayeux.server.ServerChannel.MessageListener;
import org.cometd.bayeux.server.ServerMessage;
import org.cometd.bayeux.server.ServerMessage.Mutable;
import org.cometd.bayeux.server.ServerSession;
import org.cometd.bayeux.server.ServerTransport;
import org.cometd.common.JSONContext;
import org.cometd.server.transport.AbstractHttpTransport;
import org.cometd.server.transport.AsyncJSONTransport;
import org.cometd.server.transport.JSONPTransport;
import org.cometd.server.transport.JSONTransport;
import org.eclipse.jetty.util.annotation.ManagedAttribute;
import org.eclipse.jetty.util.annotation.ManagedObject;
import org.eclipse.jetty.util.annotation.ManagedOperation;
import org.eclipse.jetty.util.annotation.Name;
import org.eclipse.jetty.util.component.AbstractLifeCycle;
import org.eclipse.jetty.util.component.ContainerLifeCycle;
import org.eclipse.jetty.util.component.Dumpable;
import org.eclipse.jetty.util.thread.ScheduledExecutorScheduler;
import org.eclipse.jetty.util.thread.Scheduler;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
@ManagedObject("The CometD server")
public class BayeuxServerImpl extends AbstractLifeCycle implements BayeuxServer, Dumpable {
private static final boolean[] VALID = new boolean[256];
static {
VALID[' '] = true;
VALID['!'] = true;
VALID['#'] = true;
VALID['$'] = true;
VALID['('] = true;
VALID[')'] = true;
VALID['*'] = true;
VALID['+'] = true;
VALID['-'] = true;
VALID['.'] = true;
VALID['/'] = true;
VALID['@'] = true;
VALID['_'] = true;
VALID['{'] = true;
VALID['~'] = true;
VALID['}'] = true;
for (int i = '0'; i <= '9'; ++i) {
VALID[i] = true;
}
for (int i = 'A'; i <= 'Z'; ++i) {
VALID[i] = true;
}
for (int i = 'a'; i <= 'z'; ++i) {
VALID[i] = true;
}
}
public static final String ALLOWED_TRANSPORTS_OPTION = "allowedTransports";
public static final String SWEEP_PERIOD_OPTION = "sweepPeriod";
public static final String TRANSPORTS_OPTION = "transports";
public static final String VALIDATE_MESSAGE_FIELDS_OPTION = "validateMessageFields";
public static final String BROADCAST_TO_PUBLISHER_OPTION = "broadcastToPublisher";
private final Logger _logger = LoggerFactory.getLogger(getClass().getName() + "." + Integer.toHexString(System.identityHashCode(this)));
private final SecureRandom _random = new SecureRandom();
private final List _listeners = new CopyOnWriteArrayList<>();
private final List _extensions = new CopyOnWriteArrayList<>();
private final ConcurrentMap _sessions = new ConcurrentHashMap<>();
private final ConcurrentMap _channels = new ConcurrentHashMap<>();
private final Map _transports = new LinkedHashMap<>(); // Order is important
private final List _allowedTransports = new ArrayList<>();
private final ThreadLocal _currentTransport = new ThreadLocal<>();
private final Map _options = new TreeMap<>();
private final Scheduler _scheduler = new ScheduledExecutorScheduler("BayeuxServer" + hashCode() + " Scheduler", false);
private SecurityPolicy _policy = new DefaultSecurityPolicy();
private JSONContext.Server _jsonContext;
private boolean _validation;
private boolean _broadcastToPublisher;
private boolean _detailedDump;
@Override
protected void doStart() throws Exception {
super.doStart();
initializeMetaChannels();
initializeJSONContext();
initializeServerTransports();
_scheduler.start();
long defaultSweepPeriod = 997;
long sweepPeriodOption = getOption(SWEEP_PERIOD_OPTION, defaultSweepPeriod);
if (sweepPeriodOption < 0) {
sweepPeriodOption = defaultSweepPeriod;
}
final long sweepPeriod = sweepPeriodOption;
schedule(new Runnable() {
@Override
public void run() {
sweep();
schedule(this, sweepPeriod);
}
}, sweepPeriod);
_validation = getOption(VALIDATE_MESSAGE_FIELDS_OPTION, true);
_broadcastToPublisher = getOption(BROADCAST_TO_PUBLISHER_OPTION, true);
}
@Override
protected void doStop() throws Exception {
super.doStop();
for (String allowedTransportName : getAllowedTransports()) {
ServerTransport transport = getTransport(allowedTransportName);
if (transport instanceof AbstractServerTransport) {
((AbstractServerTransport)transport).destroy();
}
}
_listeners.clear();
_extensions.clear();
_sessions.clear();
_channels.clear();
_transports.clear();
_allowedTransports.clear();
_options.clear();
_scheduler.stop();
}
protected void initializeMetaChannels() {
createChannelIfAbsent(Channel.META_HANDSHAKE).getReference().addListener(new HandshakeHandler());
createChannelIfAbsent(Channel.META_CONNECT).getReference().addListener(new ConnectHandler());
createChannelIfAbsent(Channel.META_SUBSCRIBE).getReference().addListener(new SubscribeHandler());
createChannelIfAbsent(Channel.META_UNSUBSCRIBE).getReference().addListener(new UnsubscribeHandler());
createChannelIfAbsent(Channel.META_DISCONNECT).getReference().addListener(new DisconnectHandler());
}
protected void initializeJSONContext() throws Exception {
Object option = getOption(AbstractServerTransport.JSON_CONTEXT_OPTION);
if (option == null) {
_jsonContext = new JettyJSONContextServer();
} else {
if (option instanceof String) {
Class> jsonContextClass = Thread.currentThread().getContextClassLoader().loadClass((String)option);
if (JSONContext.Server.class.isAssignableFrom(jsonContextClass)) {
_jsonContext = (JSONContext.Server)jsonContextClass.newInstance();
} else {
throw new IllegalArgumentException("Invalid " + JSONContext.Server.class.getName() + " implementation class");
}
} else if (option instanceof JSONContext.Server) {
_jsonContext = (JSONContext.Server)option;
} else {
throw new IllegalArgumentException("Invalid " + JSONContext.Server.class.getName() + " implementation class");
}
}
_options.put(AbstractServerTransport.JSON_CONTEXT_OPTION, _jsonContext);
}
protected void initializeServerTransports() {
if (_transports.isEmpty()) {
String option = (String)getOption(TRANSPORTS_OPTION);
if (option == null) {
// Order is important, see #findHttpTransport()
ServerTransport transport = newWebSocketTransport();
if (transport != null) {
addTransport(transport);
}
addTransport(newJSONTransport());
addTransport(new JSONPTransport(this));
} else {
for (String className : option.split(",")) {
ServerTransport transport = newServerTransport(className.trim());
if (transport != null) {
addTransport(transport);
}
}
if (_transports.isEmpty()) {
throw new IllegalArgumentException("Option '" + TRANSPORTS_OPTION +
"' does not contain a valid list of server transport class names");
}
}
}
if (_allowedTransports.isEmpty()) {
String option = (String)getOption(ALLOWED_TRANSPORTS_OPTION);
if (option == null) {
_allowedTransports.addAll(_transports.keySet());
} else {
for (String transportName : option.split(",")) {
if (_transports.containsKey(transportName)) {
_allowedTransports.add(transportName);
}
}
if (_allowedTransports.isEmpty()) {
throw new IllegalArgumentException("Option '" + ALLOWED_TRANSPORTS_OPTION +
"' does not contain at least one configured server transport name");
}
}
}
List activeTransports = new ArrayList<>();
for (String transportName : _allowedTransports) {
ServerTransport serverTransport = getTransport(transportName);
if (serverTransport instanceof AbstractServerTransport) {
((AbstractServerTransport)serverTransport).init();
activeTransports.add(serverTransport.getName());
}
}
if (_logger.isDebugEnabled()) {
_logger.debug("Active transports: {}", activeTransports);
}
}
private ServerTransport newWebSocketTransport() {
try {
ClassLoader loader = Thread.currentThread().getContextClassLoader();
loader.loadClass("javax.websocket.server.ServerContainer");
String transportClass = "org.cometd.websocket.server.WebSocketTransport";
ServerTransport transport = newServerTransport(transportClass);
if (transport == null) {
_logger.info("JSR 356 WebSocket classes available, but " + transportClass +
" unavailable: JSR 356 WebSocket transport disabled");
}
return transport;
} catch (Exception x) {
return null;
}
}
private ServerTransport newJSONTransport() {
try {
ClassLoader loader = Thread.currentThread().getContextClassLoader();
loader.loadClass("javax.servlet.ReadListener");
return new AsyncJSONTransport(this);
} catch (Exception x) {
return new JSONTransport(this);
}
}
private ServerTransport newServerTransport(String className) {
try {
ClassLoader loader = Thread.currentThread().getContextClassLoader();
@SuppressWarnings("unchecked")
Class extends ServerTransport> klass = (Class extends ServerTransport>)loader.loadClass(className);
Constructor extends ServerTransport> constructor = klass.getConstructor(BayeuxServerImpl.class);
return constructor.newInstance(this);
} catch (Exception x) {
return null;
}
}
/**
* Entry point to schedule tasks in CometD.
* Subclasses may override and run the task in a {@link java.util.concurrent.Executor},
* rather than in the scheduler thread.
*
* @param task the task to schedule
* @param delay the delay, in milliseconds, to run the task
* @return the task promise
*/
public Scheduler.Task schedule(Runnable task, long delay) {
return _scheduler.schedule(task, delay, TimeUnit.MILLISECONDS);
}
public ChannelId newChannelId(String id) {
ServerChannelImpl channel = _channels.get(id);
if (channel != null) {
return channel.getChannelId();
}
return new ChannelId(id);
}
public Map getOptions() {
return _options;
}
@Override
@ManagedOperation(value = "The value of the given configuration option", impact = "INFO")
public Object getOption(@Name("optionName") String qualifiedName) {
return _options.get(qualifiedName);
}
protected long getOption(String name, long dft) {
Object val = getOption(name);
if (val == null) {
return dft;
}
if (val instanceof Number) {
return ((Number)val).longValue();
}
return Long.parseLong(val.toString());
}
protected boolean getOption(String name, boolean dft) {
Object value = getOption(name);
if (value == null) {
return dft;
}
if (value instanceof Boolean) {
return (Boolean)value;
}
return Boolean.parseBoolean(value.toString());
}
@Override
public Set getOptionNames() {
return _options.keySet();
}
@Override
public void setOption(String qualifiedName, Object value) {
_options.put(qualifiedName, value);
}
public void setOptions(Map options) {
_options.putAll(options);
}
public long randomLong() {
long value = _random.nextLong();
return value < 0 ? -value : value;
}
public void setCurrentTransport(ServerTransport transport) {
_currentTransport.set(transport);
}
@Override
public ServerTransport getCurrentTransport() {
return _currentTransport.get();
}
@Override
public BayeuxContext getContext() {
ServerTransport transport = _currentTransport.get();
return transport == null ? null : transport.getContext();
}
@Override
public SecurityPolicy getSecurityPolicy() {
return _policy;
}
public JSONContext.Server getJSONContext() {
return _jsonContext;
}
@Override
public MarkedReference createChannelIfAbsent(String channelName, Initializer... initializers) {
ChannelId channelId;
boolean initialized = false;
ServerChannelImpl channel = _channels.get(channelName);
if (channel == null) {
// Creating the ChannelId will also normalize the channelName.
channelId = new ChannelId(channelName);
String id = channelId.getId();
if (!id.equals(channelName)) {
channelName = id;
channel = _channels.get(channelName);
}
} else {
channelId = channel.getChannelId();
}
if (channel == null) {
ServerChannelImpl candidate = new ServerChannelImpl(this, channelId);
channel = _channels.putIfAbsent(channelName, candidate);
if (channel == null) {
// My candidate channel was added to the map, so I'd better initialize it
channel = candidate;
if (_logger.isDebugEnabled()) {
_logger.debug("Added channel {}", channel);
}
try {
for (Initializer initializer : initializers) {
notifyConfigureChannel(initializer, channel);
}
for (BayeuxServer.BayeuxServerListener listener : _listeners) {
if (listener instanceof ServerChannel.Initializer) {
notifyConfigureChannel((Initializer)listener, channel);
}
}
} finally {
channel.initialized();
}
for (BayeuxServer.BayeuxServerListener listener : _listeners) {
if (listener instanceof BayeuxServer.ChannelListener) {
notifyChannelAdded((ChannelListener)listener, channel);
}
}
initialized = true;
}
} else {
channel.resetSweeperPasses();
// Double check if the sweeper removed this channel between the check at the top and here.
// This is not 100% fool proof (e.g. this thread is preempted long enough for the sweeper
// to remove the channel, but the alternative is to have a global lock)
_channels.putIfAbsent(channelName, channel);
}
// Another thread may add this channel concurrently, so wait until it is initialized
channel.waitForInitialized();
return new MarkedReference(channel, initialized);
}
private void notifyConfigureChannel(Initializer listener, ServerChannel channel) {
try {
listener.configureChannel(channel);
} catch (Throwable x) {
_logger.info("Exception while invoking listener " + listener, x);
}
}
private void notifyChannelAdded(ChannelListener listener, ServerChannel channel) {
try {
listener.channelAdded(channel);
} catch (Throwable x) {
_logger.info("Exception while invoking listener " + listener, x);
}
}
@Override
public List getSessions() {
return Collections.unmodifiableList(new ArrayList(_sessions.values()));
}
@Override
public boolean removeSession(ServerSession session) {
return removeSession(session, false).getReference() != null;
}
@Override
public ServerSession getSession(String clientId) {
if (clientId == null) {
return null;
}
return _sessions.get(clientId);
}
protected void addServerSession(ServerSessionImpl session, ServerMessage message) {
_sessions.put(session.getId(), session);
for (BayeuxServerListener listener : _listeners) {
if (listener instanceof BayeuxServer.SessionListener) {
notifySessionAdded((SessionListener)listener, session, message);
}
}
}
private void notifySessionAdded(SessionListener listener, ServerSession session, ServerMessage message) {
try {
listener.sessionAdded(session, message);
} catch (Throwable x) {
_logger.info("Exception while invoking listener " + listener, x);
}
}
/**
* @param session the session to remove
* @param timedOut whether the remove reason is server-side expiration
* @return true if the session was removed and was connected
*/
public boolean removeServerSession(ServerSession session, boolean timedOut) {
return removeSession(session, timedOut).isMarked();
}
private MarkedReference removeSession(ServerSession session, boolean timedOut) {
if (_logger.isDebugEnabled()) {
_logger.debug("Removing session {}, timed out: {}", session, timedOut);
}
ServerSessionImpl removed = _sessions.remove(session.getId());
if (removed != session) {
return MarkedReference.empty();
}
// Invoke BayeuxServer.SessionListener first, so that the application
// can be "pre-notified" that a session is being removed before the
// application gets notifications of channel unsubscriptions
for (BayeuxServerListener listener : _listeners) {
if (listener instanceof SessionListener) {
notifySessionRemoved((SessionListener)listener, removed, timedOut);
}
}
boolean connected = removed.removed(timedOut);
return new MarkedReference<>(removed, connected);
}
private void notifySessionRemoved(SessionListener listener, ServerSession session, boolean timedout) {
try {
listener.sessionRemoved(session, timedout);
} catch (Throwable x) {
_logger.info("Exception while invoking listener " + listener, x);
}
}
public ServerSessionImpl newServerSession() {
return new ServerSessionImpl(this);
}
@Override
public LocalSession newLocalSession(String idHint) {
return new LocalSessionImpl(this, idHint);
}
@Override
public ServerMessage.Mutable newMessage() {
return new ServerMessageImpl();
}
public ServerMessage.Mutable newMessage(ServerMessage tocopy) {
ServerMessage.Mutable mutable = newMessage();
for (String key : tocopy.keySet()) {
mutable.put(key, tocopy.get(key));
}
return mutable;
}
@Override
public void setSecurityPolicy(SecurityPolicy securityPolicy) {
_policy = securityPolicy;
}
@Override
public void addExtension(Extension extension) {
_extensions.add(extension);
}
@Override
public void removeExtension(Extension extension) {
_extensions.remove(extension);
}
@Override
public List getExtensions() {
return Collections.unmodifiableList(_extensions);
}
@Override
public void addListener(BayeuxServerListener listener) {
if (listener == null) {
throw new NullPointerException();
}
_listeners.add(listener);
}
@Override
public ServerChannel getChannel(String channelId) {
return getServerChannel(channelId);
}
private ServerChannelImpl getServerChannel(String channelId) {
ServerChannelImpl channel = _channels.get(channelId);
if (channel != null) {
channel.waitForInitialized();
}
return channel;
}
@Override
public List getChannels() {
List result = new ArrayList<>();
for (ServerChannelImpl channel : _channels.values()) {
channel.waitForInitialized();
result.add(channel);
}
return result;
}
@Override
public void removeListener(BayeuxServerListener listener) {
_listeners.remove(listener);
}
public ServerMessage.Mutable handle(ServerSessionImpl session, ServerMessage.Mutable message) {
if (_logger.isDebugEnabled()) {
_logger.debug("> {} {}", message, session);
}
if (_validation) {
validateMessage(message);
}
Mutable reply = createReply(message);
if (!extendRecv(session, message) || session != null && !session.extendRecv(message)) {
error(reply, "404::message deleted");
} else {
if (_logger.isDebugEnabled()) {
_logger.debug(">> {}", message);
}
handle(session, message, reply);
}
if (_logger.isDebugEnabled()) {
_logger.debug("<< {}", reply);
}
return reply;
}
private void handle(ServerSessionImpl session, Mutable message, Mutable reply) {
String channelName = message.getChannel();
if (session == null || session.isDisconnected() ||
(!session.getId().equals(message.getClientId()) && !Channel.META_HANDSHAKE.equals(message.getChannel()))) {
unknownSession(reply);
return;
}
session.cancelExpiration(Channel.META_CONNECT.equals(channelName));
if (channelName == null) {
error(reply, "400::channel missing");
} else {
ServerChannelImpl channel = getServerChannel(channelName);
if (channel == null) {
Authorizer.Result creationResult = isCreationAuthorized(session, message, channelName);
if (creationResult instanceof Authorizer.Result.Denied) {
String denyReason = ((Authorizer.Result.Denied)creationResult).getReason();
error(reply, "403:" + denyReason + ":create denied");
} else {
channel = (ServerChannelImpl)createChannelIfAbsent(channelName).getReference();
}
}
if (channel != null) {
if (channel.isMeta()) {
doPublish(session, channel, message, true);
} else {
Authorizer.Result publishResult = isPublishAuthorized(channel, session, message);
if (publishResult instanceof Authorizer.Result.Denied) {
String denyReason = ((Authorizer.Result.Denied)publishResult).getReason();
error(reply, "403:" + denyReason + ":publish denied");
} else {
reply.setSuccessful(true);
doPublish(session, channel, message, true);
}
}
}
}
}
protected void validateMessage(Mutable message) {
String channel = message.getChannel();
if (!validate(channel)) {
throw new IllegalArgumentException("Invalid message channel: " + channel);
}
String id = message.getId();
if (id != null && !validate(id)) {
throw new IllegalArgumentException("Invalid message id: " + id);
}
}
private boolean validate(String value) {
for (int i = 0; i < value.length(); ++i) {
char c = value.charAt(i);
if (c > 127 || !VALID[c]) {
return false;
}
}
return true;
}
private Authorizer.Result isPublishAuthorized(ServerChannel channel, ServerSession session, ServerMessage message) {
if (_policy != null && !_policy.canPublish(this, session, channel, message)) {
_logger.warn("{} denied Publish@{} by {}", session, channel.getId(), _policy);
return Authorizer.Result.deny("denied_by_security_policy");
}
return isOperationAuthorized(Authorizer.Operation.PUBLISH, session, message, channel.getChannelId());
}
private Authorizer.Result isSubscribeAuthorized(ServerChannel channel, ServerSession session, ServerMessage message) {
if (_policy != null && !_policy.canSubscribe(this, session, channel, message)) {
_logger.warn("{} denied Subscribe@{} by {}", session, channel, _policy);
return Authorizer.Result.deny("denied_by_security_policy");
}
return isOperationAuthorized(Authorizer.Operation.SUBSCRIBE, session, message, channel.getChannelId());
}
private Authorizer.Result isCreationAuthorized(ServerSession session, ServerMessage message, String channel) {
if (_policy != null && !_policy.canCreate(BayeuxServerImpl.this, session, channel, message)) {
_logger.warn("{} denied Create@{} by {}", session, message.getChannel(), _policy);
return Authorizer.Result.deny("denied_by_security_policy");
}
return isOperationAuthorized(Authorizer.Operation.CREATE, session, message, new ChannelId(channel));
}
private Authorizer.Result isOperationAuthorized(Authorizer.Operation operation, ServerSession session, ServerMessage message, ChannelId channelId) {
Authorizer.Result result = isChannelOperationAuthorized(operation, session, message, channelId);
if (result == null) {
result = Authorizer.Result.grant();
if (_logger.isDebugEnabled()) {
_logger.debug("No authorizers, {} for channel {} {}", operation, channelId, result);
}
} else {
if (result.isGranted()) {
if (_logger.isDebugEnabled()) {
_logger.debug("No authorizer denied {} for channel {}, authorization {}", operation, channelId, result);
}
} else if (!result.isDenied()) {
result = Authorizer.Result.deny("denied_by_not_granting");
if (_logger.isDebugEnabled()) {
_logger.debug("No authorizer granted {} for channel {}, authorization {}", operation, channelId, result);
}
}
}
// We need to make sure that this method returns a boolean result (granted or denied)
// but if it's denied, we need to return the object in order to access the deny reason
assert !(result instanceof Authorizer.Result.Ignored);
return result;
}
private Authorizer.Result isChannelOperationAuthorized(Authorizer.Operation operation, ServerSession session, ServerMessage message, ChannelId channelId) {
Authorizer.Result result = null;
List wilds = channelId.getWilds();
for (int i = 0, size = wilds.size(); i <= size; ++i) {
String channelName = i < size ? wilds.get(i) : channelId.getId();
ServerChannelImpl channel = _channels.get(channelName);
if (channel != null) {
Authorizer.Result authz = isChannelOperationAuthorized(channel, operation, session, message, channelId);
if (authz != null) {
if (result == null || authz.isDenied() || authz.isGranted()) {
result = authz;
}
if (authz.isDenied()) {
break;
}
}
}
}
return result;
}
private Authorizer.Result isChannelOperationAuthorized(ServerChannelImpl channel, Authorizer.Operation operation, ServerSession session, ServerMessage message, ChannelId channelId) {
List authorizers = channel.authorizers();
if (authorizers.isEmpty()) {
return null;
}
Authorizer.Result result = Authorizer.Result.ignore();
for (Authorizer authorizer : authorizers) {
Authorizer.Result authorization = authorizer.authorize(operation, channelId, session, message);
if (_logger.isDebugEnabled()) {
_logger.debug("Authorizer {} on channel {} {} {} for channel {}", authorizer, channel, authorization, operation, channelId);
}
if (authorization.isDenied()) {
result = authorization;
break;
} else if (authorization.isGranted()) {
result = authorization;
}
}
return result;
}
protected void doPublish(ServerSessionImpl from, ServerChannelImpl to, final ServerMessage.Mutable mutable, boolean receiving) {
boolean broadcast = to.isBroadcast();
if (broadcast) {
// Do not leak the clientId to other subscribers
// as we are now "sending" this message.
mutable.setClientId(null);
// Reset the messageId to avoid clashes with message-based transports such
// as websocket whose clients may rely on the messageId to match request/responses.
mutable.setId(null);
}
List wildChannels = to.getChannelId().getWilds();
// First notify the channel listeners.
if (!notifyListeners(from, to, mutable, wildChannels)) {
error(mutable.getAssociated(), "404::message deleted");
return;
}
if (broadcast || !receiving) {
if (!extendSend(from, null, mutable)) {
return;
}
// Exactly at this point, we convert the message to JSON and therefore
// any further modification will be lost.
// This is an optimization so that if the message is sent to a million
// subscribers, we generate the JSON only once.
// From now on, user code is passed a ServerMessage reference (and not
// ServerMessage.Mutable), and we attempt to return immutable data
// structures, even if it is not possible to guard against all cases.
// For example, it is impossible to prevent things like
// ((CustomObject)serverMessage.getData()).change() or
// ((Map)serverMessage.getExt().get("map")).put().
freeze(mutable);
}
// Call the wild subscribers, which can only get broadcast messages.
// We need a special treatment in case of subscription to /**, otherwise
// we will deliver meta messages and service messages as if it could be
// possible to subscribe to meta channels and service channels.
if (broadcast) {
Set wildSubscribers = null;
for (String wildName : wildChannels) {
ServerChannelImpl wildChannel = _channels.get(wildName);
if (wildChannel == null) {
continue;
}
Set subscribers = wildChannel.subscribers();
for (ServerSession session : subscribers) {
if (wildSubscribers == null) {
wildSubscribers = new HashSet<>();
}
if (wildSubscribers.add(session.getId())) {
((ServerSessionImpl)session).doDeliver(from, mutable);
}
}
}
// Call the leaf subscribers
Set subscribers = to.subscribers();
for (ServerSession session : subscribers) {
if (wildSubscribers == null || !wildSubscribers.contains(session.getId())) {
((ServerSessionImpl)session).doDeliver(from, mutable);
}
}
} else if (to.isMeta()) {
notifyHandlerListeners(from, to, mutable);
}
}
private boolean notifyListeners(ServerSessionImpl from, ServerChannelImpl to, Mutable mutable, List wildChannels) {
for (int i = 0, size = wildChannels.size(); i <= size; ++i) {
ServerChannelImpl channel = i == size ? to : _channels.get(wildChannels.get(i));
if (channel == null) {
continue;
}
if (channel.isLazy()) {
mutable.setLazy(true);
}
List listeners = channel.listeners();
for (ServerChannelListener listener : listeners) {
if (listener instanceof MessageListener) {
if (!notifyOnMessage((MessageListener)listener, from, to, mutable)) {
return false;
}
}
}
}
return true;
}
private void notifyHandlerListeners(ServerSessionImpl from, ServerChannelImpl to, Mutable mutable) {
List listeners = to.listeners();
for (ServerChannelListener listener : listeners) {
if (listener instanceof HandlerListener) {
((HandlerListener)listener).onMessage(from, mutable);
}
}
}
public void freeze(Mutable mutable) {
if (mutable instanceof ServerMessageImpl) {
ServerMessageImpl message = (ServerMessageImpl)mutable;
if (message.isFrozen()) {
return;
}
String json = _jsonContext.generate(message);
message.freeze(json);
}
}
private boolean notifyOnMessage(MessageListener listener, ServerSession from, ServerChannel to, Mutable mutable) {
try {
return listener.onMessage(from, to, mutable);
} catch (Throwable x) {
_logger.info("Exception while invoking listener " + listener, x);
return true;
}
}
public ServerMessage.Mutable extendReply(ServerSessionImpl from, ServerSessionImpl to, ServerMessage.Mutable reply) {
if (!extendSend(from, to, reply)) {
return null;
}
if (to != null) {
reply = to.extendSend(reply);
}
return reply;
}
protected boolean extendRecv(ServerSession from, ServerMessage.Mutable message) {
for (Extension extension : _extensions) {
boolean proceed = message.isMeta() ?
notifyRcvMeta(extension, from, message) :
notifyRcv(extension, from, message);
if (!proceed) {
return false;
}
}
return true;
}
private boolean notifyRcvMeta(Extension extension, ServerSession from, Mutable message) {
try {
return extension.rcvMeta(from, message);
} catch (Throwable x) {
_logger.info("Exception while invoking extension " + extension, x);
return true;
}
}
private boolean notifyRcv(Extension extension, ServerSession from, Mutable message) {
try {
return extension.rcv(from, message);
} catch (Throwable x) {
_logger.info("Exception while invoking extension " + extension, x);
return true;
}
}
protected boolean extendSend(ServerSession from, ServerSession to, Mutable message) {
// Cannot use listIterator(int): it is not thread safe
ListIterator i = _extensions.listIterator();
while (i.hasNext()) {
i.next();
}
while (i.hasPrevious()) {
final Extension extension = i.previous();
boolean proceed = message.isMeta() ?
notifySendMeta(extension, to, message) :
notifySend(extension, from, to, message);
if (!proceed) {
if (_logger.isDebugEnabled()) {
_logger.debug("Extension {} interrupted message processing for {}", extension, message);
}
return false;
}
}
if (_logger.isDebugEnabled()) {
_logger.debug("< {}", message);
}
return true;
}
private boolean notifySendMeta(Extension extension, ServerSession to, Mutable message) {
try {
return extension.sendMeta(to, message);
} catch (Throwable x) {
_logger.info("Exception while invoking extension " + extension, x);
return true;
}
}
private boolean notifySend(Extension extension, ServerSession from, ServerSession to, Mutable message) {
try {
return extension.send(from, to, message);
} catch (Throwable x) {
_logger.info("Exception while invoking extension " + extension, x);
return true;
}
}
protected boolean removeServerChannel(ServerChannelImpl channel) {
if (_channels.remove(channel.getId(), channel)) {
if (_logger.isDebugEnabled()) {
_logger.debug("Removed channel {}", channel);
}
for (BayeuxServerListener listener : _listeners) {
if (listener instanceof BayeuxServer.ChannelListener) {
notifyChannelRemoved((ChannelListener)listener, channel);
}
}
return true;
}
return false;
}
private void notifyChannelRemoved(ChannelListener listener, ServerChannelImpl channel) {
try {
listener.channelRemoved(channel.getId());
} catch (Throwable x) {
_logger.info("Exception while invoking listener " + listener, x);
}
}
protected List getListeners() {
return Collections.unmodifiableList(_listeners);
}
@Override
public Set getKnownTransportNames() {
return _transports.keySet();
}
@Override
public ServerTransport getTransport(String transport) {
return _transports.get(transport);
}
public ServerTransport addTransport(ServerTransport transport) {
ServerTransport result = _transports.put(transport.getName(), transport);
if (_logger.isDebugEnabled()) {
_logger.debug("Added transport {} from {}", transport.getName(), transport.getClass());
}
return result;
}
public void setTransports(ServerTransport... transports) {
setTransports(Arrays.asList(transports));
}
public void setTransports(List transports) {
_transports.clear();
for (ServerTransport transport : transports) {
addTransport(transport);
}
}
public List getTransports() {
return new ArrayList<>(_transports.values());
}
@SuppressWarnings("ForLoopReplaceableByForEach")
protected AbstractHttpTransport findHttpTransport(HttpServletRequest request) {
// Avoid allocation of the Iterator
for (int i = 0; i < _allowedTransports.size(); ++i) {
String transportName = _allowedTransports.get(i);
ServerTransport serverTransport = getTransport(transportName);
if (serverTransport instanceof AbstractHttpTransport) {
AbstractHttpTransport transport = (AbstractHttpTransport)serverTransport;
if (transport.accept(request)) {
return transport;
}
}
}
return null;
}
@ManagedAttribute(value = "The transports allowed by this server", readonly = true)
@Override
public List getAllowedTransports() {
return Collections.unmodifiableList(_allowedTransports);
}
public void setAllowedTransports(String... allowed) {
setAllowedTransports(Arrays.asList(allowed));
}
public void setAllowedTransports(List allowed) {
if (_logger.isDebugEnabled()) {
_logger.debug("setAllowedTransport {} of {}", allowed, _transports);
}
_allowedTransports.clear();
for (String transport : allowed) {
if (_transports.containsKey(transport)) {
_allowedTransports.add(transport);
}
}
if (_logger.isDebugEnabled()) {
_logger.debug("allowedTransports {}", _allowedTransports);
}
}
@ManagedAttribute(value = "Whether this server broadcast messages to the publisher", readonly = true)
public boolean isBroadcastToPublisher() {
return _broadcastToPublisher;
}
protected void unknownSession(Mutable reply) {
error(reply, "402::Unknown client");
if (Channel.META_HANDSHAKE.equals(reply.getChannel()) || Channel.META_CONNECT.equals(reply.getChannel())) {
Map advice = reply.getAdvice(true);
advice.put(Message.RECONNECT_FIELD, Message.RECONNECT_HANDSHAKE_VALUE);
advice.put(Message.INTERVAL_FIELD, 0L);
}
}
protected void error(ServerMessage.Mutable reply, String error) {
if (reply != null) {
reply.put(Message.ERROR_FIELD, error);
reply.setSuccessful(false);
}
}
protected ServerMessage.Mutable createReply(ServerMessage.Mutable message) {
ServerMessage.Mutable reply = newMessage();
message.setAssociated(reply);
reply.setAssociated(message);
reply.setChannel(message.getChannel());
String id = message.getId();
if (id != null) {
reply.setId(id);
}
Object subscriptions = message.get(Message.SUBSCRIPTION_FIELD);
if (subscriptions != null) {
reply.put(Message.SUBSCRIPTION_FIELD, subscriptions);
}
return reply;
}
private void validateSubscriptions(List subscriptions) {
if (_validation) {
for (String subscription : subscriptions) {
if (!validate(subscription)) {
throw new IllegalArgumentException("Invalid message subscription: " + subscription);
}
}
}
}
@ManagedOperation(value = "Sweeps channels and sessions of this BayeuxServer", impact = "ACTION")
public void sweep() {
for (ServerChannelImpl channel : _channels.values()) {
channel.sweep();
}
for (ServerTransport transport : _transports.values()) {
if (transport instanceof AbstractServerTransport) {
((AbstractServerTransport)transport).sweep();
}
}
long now = System.currentTimeMillis();
for (ServerSessionImpl session : _sessions.values()) {
session.sweep(now);
}
}
@ManagedAttribute("Reports additional details in the dump")
public boolean isDetailedDump() {
return _detailedDump;
}
public void setDetailedDump(boolean detailedDump) {
_detailedDump = detailedDump;
}
@ManagedOperation(value = "Dumps the BayeuxServer state", impact = "INFO")
@Override
public String dump() {
return ContainerLifeCycle.dump(this);
}
@Override
public void dump(Appendable out, String indent) throws IOException {
ContainerLifeCycle.dumpObject(out, this);
List
© 2015 - 2025 Weber Informatics LLC | Privacy Policy