org.cometd.bayeux.server.ServerSession Maven / Gradle / Ivy
/*
* Copyright (c) 2008-2022 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.bayeux.server;
import java.util.List;
import java.util.Queue;
import java.util.Set;
import org.cometd.bayeux.Bayeux;
import org.cometd.bayeux.Promise;
import org.cometd.bayeux.Session;
/**
* Objects implementing this interface are the server-side representation of remote Bayeux clients.
* {@link ServerSession} contains the queue of messages to be delivered to the client; messages are
* normally queued on a {@link ServerSession} by publishing them to a channel to which the session is
* subscribed (via {@link ServerChannel#publish(Session, ServerMessage.Mutable, Promise)}.
* The {@link #deliver(Session, ServerMessage.Mutable, Promise)} and {@link #deliver(Session, String, Object, Promise)}
* methods may be used to directly queue messages to a session without publishing them to all subscribers
* of a channel.
*/
public interface ServerSession extends Session {
/**
* Adds the given extension to this session.
*
* @param extension the extension to add
* @see #removeExtension(Extension)
*/
public void addExtension(Extension extension);
/**
* Removes the given extension from this session.
*
* @param extension the extension to remove
* @see #addExtension(Extension)
*/
public void removeExtension(Extension extension);
/**
* @return an immutable list of extensions present in this ServerSession instance
* @see #addExtension(Extension)
*/
public List getExtensions();
/**
* Adds the given listener to this session.
*
* @param listener the listener to add
* @see #removeListener(ServerSessionListener)
*/
public void addListener(ServerSessionListener listener);
/**
* Removes the given listener from this session.
*
* @param listener the listener to remove
* @see #addListener(ServerSessionListener)
*/
public void removeListener(ServerSessionListener listener);
/**
* @return the ServerTransport associated with this session
*/
public ServerTransport getServerTransport();
/**
* @return whether this is a session for a local client on server-side
*/
public boolean isLocalSession();
/**
* @return the {@link LocalSession} associated with this session,
* or null if this is a session representing a remote client.
*/
public LocalSession getLocalSession();
/**
* Delivers the given message to this session.
* This is different from {@link ServerChannel#publish(Session, ServerMessage.Mutable, Promise)}
* as the message is delivered only to this session and
* not to all subscribers of the channel.
* The message should still have a channel id specified, so that the ClientSession
* may identify the listeners the message should be delivered to.
*
* @param sender the session delivering the message
* @param message the message to deliver
* @param promise the promise to notify with the result of the deliver
* @see #deliver(Session, String, Object, Promise)
*/
public void deliver(Session sender, ServerMessage.Mutable message, Promise promise);
/**
* Delivers the given information to this session.
*
* @param sender the session delivering the message
* @param channel the channel of the message
* @param data the data of the message
* @param promise the promise to notify with the result of the deliver
* @see #deliver(Session, ServerMessage.Mutable, Promise)
*/
public void deliver(Session sender, String channel, Object data, Promise promise);
/**
* @return the set of channels to which this session is subscribed to
*/
public Set getSubscriptions();
/**
* @return The string indicating the client user agent, or null if not known
*/
public String getUserAgent();
/**
* @return the period of time, in milliseconds, that the client associated with this session
* will wait before issuing a connect message, or -1 if the default value is used
* @see ServerTransport#getInterval()
* @see #setInterval(long)
*/
public long getInterval();
/**
* @param interval the period of time, in milliseconds, that the client
* associated with this session will wait before issuing a connect message
*/
public void setInterval(long interval);
/**
* @return the period of time, in milliseconds, that the server will hold connect messages
* for this session or -1 if the default value is used
* @see ServerTransport#getTimeout()
*/
public long getTimeout();
/**
* @param timeout the period of time, in milliseconds, that the server will hold connect
* messages for this session
*/
public void setTimeout(long timeout);
/**
* @return the max period of time, in milliseconds, that the server waits before expiring
* the session when the client does not send messages to the server, or -1 if the default
* value is used
*/
public long getMaxInterval();
/**
* @param maxInterval the max period of time, in milliseconds, that the server waits
* before expiring the session
*/
public void setMaxInterval(long maxInterval);
/**
* @return whether delivery of messages only happens via the {@code /meta/connect} channel
*/
public boolean isMetaConnectDeliveryOnly();
/**
* Sets whether delivery of messages only happen via the {@code /meta/connect} channel.
* Transports that use more than one connection to the server (e.g. HTTP transports)
* may deliver some messages in one connection and some in another connection, causing
* messages to arrive out-of-order.
* Forcing messages to be delivered only via the {@code /meta/connect} channel guarantees
* server-to-client message ordering.
*
* @param metaConnectDeliveryOnly whether delivery of messages only happens via the
* {@code /meta/connect} channel
*/
public void setMetaConnectDeliveryOnly(boolean metaConnectDeliveryOnly);
/**
* @return whether this session sends messages back to itself
*/
public boolean isBroadcastToPublisher();
/**
* Sets whether this session sends messages back to itself.
*
* @param broadcastToPublisher whether this session sends messages back to itself
*/
public void setBroadcastToPublisher(boolean broadcastToPublisher);
/**
* Common interface for {@link ServerSession} listeners.
* Specific sub-interfaces define what kind of event listeners will be notified.
*/
public interface ServerSessionListener extends Bayeux.BayeuxListener {
}
/**
* Listener objects that implement this interface will be notified of session addition.
*/
public interface AddedListener extends ServerSessionListener {
/**
* Callback method invoked when the session is added to a {@link BayeuxServer}.
*
* @param session the added session
* @param message the message that caused the session addition
*/
public void added(ServerSession session, ServerMessage message);
}
/**
* Listeners objects that implement this interface will be notified of session removal.
*/
public interface RemovedListener extends ServerSessionListener {
/**
* Callback invoked when the session is removed.
*
* @param session the removed session
* @param message the message that caused the session removal, or null
* @param timeout whether the session has been removed due to a timeout
*/
public void removed(ServerSession session, ServerMessage message, boolean timeout);
}
/**
* Listeners objects that implement this interface will be notified of message sending.
*/
public interface MessageListener extends ServerSessionListener {
/**
* Callback invoked when a message is sent.
* Implementers can decide to notify the promise with false to signal that the
* message should not be further processed, meaning that other session listeners
* will not be notified and that the message will be discarded for this session.
*
* @param session the session that will receive the message
* @param sender the session that sent the message
* @param message the message sent
* @param promise the promise to notify whether the processing of the message should continue
*/
public default void onMessage(ServerSession session, ServerSession sender, ServerMessage message, Promise promise) {
promise.succeed(onMessage(session, sender, message));
}
/**
* Blocking version of {@link #onMessage(ServerSession, ServerSession, ServerMessage, Promise)}.
*
* @param session the session that will receive the message
* @param sender the session that sent the message
* @param message the message sent
* @return whether the processing of the message should continue
*/
public default boolean onMessage(ServerSession session, ServerSession sender, ServerMessage message) {
return true;
}
}
/**
* Listener objects that implement this interface will be notified when a message
* is queued in the session queue.
* This is a restricted listener interface, see {@link QueueMaxedListener}.
*/
public interface QueueListener extends ServerSessionListener {
/**
* Callback invoked when a message is queued in the session queue.
*
* @param sender the ServerSession that sends the message, may be null.
* @param message the message being queued
*/
public void queued(ServerSession sender, ServerMessage message);
}
/**
* Listeners objects that implement this interface will be notified when the session queue
* is being drained to actually deliver the messages.
* This is a restricted listener interface, see {@link QueueMaxedListener}.
*/
public interface DeQueueListener extends ServerSessionListener {
/**
* Callback invoked to notify that the queue of messages and the message replies are
* about to be sent to the remote client.
* This is the last chance to process the queue, to remove duplicates or merge messages,
* and to process the replies.
*
* @param session the session whose messages are being sent
* @param queue the queue of messages to send
* @param replies the message replies to send
*/
public default void deQueue(ServerSession session, Queue queue, List replies) {
deQueue(session, queue);
}
/**
* Callback invoked to notify that the queue of messages is about to be sent to the
* remote client.
* This is the last chance to process the queue and remove duplicates or merge messages.
*
* @param session the session whose messages are being sent
* @param queue the queue of messages to send
*/
public void deQueue(ServerSession session, Queue queue);
}
/**
* Listeners objects that implement this interface will be notified when the session queue is full.
* This is a restricted listener interface because implementers are invoked while holding
* the session lock and therefore are restricted in the type of operations they may perform; in particular,
* publishing a message to another session may end up in a deadlock.
*/
public interface QueueMaxedListener extends ServerSessionListener {
/**
* Callback invoked to notify when the message queue is exceeding the value
* configured for the transport with the option "maxQueue".
* Implementers may modify the queue, for example by removing or merging messages.
*
* @param session the session that will receive the message
* @param queue the session's message queue
* @param sender the session that is sending the messages
* @param message the message that exceeded the max queue capacity
* @return true if the message should be added to the session queue
*/
public boolean queueMaxed(ServerSession session, Queue queue, ServerSession sender, ServerMessage message);
}
/**
* Listeners objects that implement this interface will be notified when a {@code /meta/connect}
* message is suspended by the server, and when it is subsequently resumed.
*/
public interface HeartBeatListener extends ServerSessionListener {
/**
* Callback invoked to notify that a {@code /meta/connect} message has been suspended.
*
* @param session the session that received the {@code /meta/connect} message
* @param message the {@code /meta/connect} message
* @param timeout the time, in milliseconds, the server will hold the message if not otherwise resumed
*/
public default void onSuspended(ServerSession session, ServerMessage message, long timeout) {
}
/**
* Callback invoked to notify that a {@code /meta/connect} message has been resumed.
*
* @param session the session that received the {@code /meta/connect} message
* @param message the {@code /meta/connect} message
* @param timeout whether the {@code /meta/connect} message was resumed after the whole timeout
*/
public default void onResumed(ServerSession session, ServerMessage message, boolean timeout) {
}
}
/**
* Extension API for {@link ServerSession}.
* Implementations of this interface allow to modify incoming and outgoing messages
* for a particular session, before any other processing performed by the implementation
* but after {@link BayeuxServer.Extension} processing.
* Multiple extensions can be registered; the extension receive methods
* are invoked in registration order, while the extension send methods are
* invoked in registration reverse order.
*
* @see ServerSession#addExtension(Extension)
* @see BayeuxServer.Extension
*/
public interface Extension {
/**
* Callback method invoked every time a message is incoming.
*
* @param session the session that sent the message
* @param message the incoming message
* @param promise the promise to notify whether message processing should continue
*/
public default void incoming(ServerSession session, ServerMessage.Mutable message, Promise promise) {
promise.succeed(message.isMeta() ? rcvMeta(session, message) : rcv(session, message));
}
/**
* Blocking version of {@link #incoming(ServerSession, ServerMessage.Mutable, Promise)}
* for non-meta messages.
*
* @param session the session that sent the message
* @param message the incoming message
* @return whether message processing should continue
*/
public default boolean rcv(ServerSession session, ServerMessage.Mutable message) {
return true;
}
/**
* Blocking version of {@link #incoming(ServerSession, ServerMessage.Mutable, Promise)}
* for meta messages.
*
* @param session the session that sent the message
* @param message the incoming message
* @return whether message processing should continue
*/
public default boolean rcvMeta(ServerSession session, ServerMessage.Mutable message) {
return true;
}
/**
* Callback method invoked every time a message is outgoing.
*
* @param sender the session that sent the message or null
* @param session the session receiving the message
* @param message the outgoing message
* @param promise the promise to notify with the message to send or null to not send the message
*/
public default void outgoing(ServerSession sender, ServerSession session, ServerMessage.Mutable message, Promise promise) {
if (message.isMeta()) {
promise.succeed(sendMeta(sender, session, message) ? message : null);
} else {
ServerMessage result = send(sender, session, message);
if (result instanceof ServerMessage.Mutable) {
promise.succeed((ServerMessage.Mutable)result);
} else if (result == null) {
promise.succeed(null);
} else {
promise.fail(new IllegalArgumentException());
}
}
}
/**
* Blocking version of {@link #outgoing(ServerSession, ServerSession, ServerMessage.Mutable, Promise)}
* for non-meta messages.
*
* @param sender the session that sent the message or null
* @param session the session receiving the message
* @param message the outgoing message
* @return the message to send or null to not send the message
*/
public default ServerMessage send(ServerSession sender, ServerSession session, ServerMessage message) {
return message;
}
/**
* Blocking version of {@link #outgoing(ServerSession, ServerSession, ServerMessage.Mutable, Promise)}
* for meta messages.
*
* @param sender the session that sent the message or null
* @param session the session receiving the message
* @param message the outgoing message
* @return whether message processing should continue
*/
public default boolean sendMeta(ServerSession sender, ServerSession session, ServerMessage.Mutable message) {
return true;
}
}
}