org.cometd.server.ext.ActivityExtension Maven / Gradle / Ivy
/*
* Copyright (c) 2008-2020 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.ext;
import java.util.concurrent.TimeUnit;
import java.util.concurrent.atomic.AtomicLong;
import org.cometd.bayeux.Channel;
import org.cometd.bayeux.server.BayeuxServer;
import org.cometd.bayeux.server.ServerMessage;
import org.cometd.bayeux.server.ServerSession;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
/**
* Monitors the activity of {@link ServerSession}s and disconnects them after a period of inactivity.
* The inactivity of a particular {@code ServerSession} is determined in two ways:
*
* - Only the client is inactive, that is the client only sends periodic {@code /meta/connect}
* messages but no other messages, while the server may send messages to the client; this is
* configured via {@link Activity#CLIENT}
* - Both the client and the server are inactive, that is neither the client nor the server
* send messages apart the periodic {@code /meta/connect} messages; this is configured via
* {@link Activity#CLIENT_SERVER}
*
* When the inactivity exceeds a configurable {@link #getMaxInactivityPeriod() inactive period},
* the {@code ServerSession} is {@link ServerSession#disconnect() disconnected}.
*/
public class ActivityExtension implements BayeuxServer.Extension {
private final Activity activity;
private final long maxInactivityPeriod;
/**
* Creates an ActivityExtension to be installed in the {@link BayeuxServer}
*
* @param activity the activity to monitor
* @param maxInactivityPeriod the max inactivity period, in milliseconds
*/
public ActivityExtension(Activity activity, long maxInactivityPeriod) {
this.activity = activity;
this.maxInactivityPeriod = maxInactivityPeriod;
}
/**
* @return the activity that is being monitored by this extension
*/
public Activity getActivity() {
return activity;
}
/**
* @return the max inactivity period, in milliseconds
*/
public long getMaxInactivityPeriod() {
return maxInactivityPeriod;
}
@Override
public boolean rcvMeta(ServerSession session, ServerMessage.Mutable message) {
if (Channel.META_HANDSHAKE.equals(message.getChannel())) {
session.addExtension(newSessionExtension(session, message));
}
return true;
}
/**
* Creates a new {@link ServerSession.Extension} that monitors the activity of the given {@link ServerSession}
*
* @param session the {@code ServerSession} to monitor
* @param handshake the handshake message
* @return a new {@code ServerSession.Extension} that monitors the {@code ServerSession} activity
*/
protected ServerSession.Extension newSessionExtension(ServerSession session, ServerMessage handshake) {
return new SessionExtension(getActivity(), getMaxInactivityPeriod());
}
/**
* The possible activity to monitor
*/
public enum Activity {
/**
* Constant that indicates to monitor only client activity for a session
*/
CLIENT,
/**
* Constant that indicates to monitor both client and server activity for a session
*/
CLIENT_SERVER
}
/**
* Monitors the activity of a single {@link ServerSession}, disconnecting it
* when the max inactivity period is exceeded.
*/
public static class SessionExtension implements ServerSession.Extension {
private static final Logger logger = LoggerFactory.getLogger(ActivityExtension.class);
private final AtomicLong lastActivity = new AtomicLong(System.nanoTime());
private final Activity activity;
private final long maxInactivityPeriod;
public SessionExtension(Activity activity, long maxInactivityPeriod) {
this.activity = activity;
this.maxInactivityPeriod = maxInactivityPeriod;
}
/**
* @return the max inactivity period, in milliseconds
*/
public long getMaxInactivityPeriod() {
return maxInactivityPeriod;
}
/**
* @return the last activity timestamp, in nanoseconds
*/
protected long getLastActivity() {
return lastActivity.get();
}
@Override
public boolean rcv(ServerSession session, ServerMessage.Mutable message) {
if (logger.isDebugEnabled()) {
logger.debug("Marking active session {}, received message {}", session, message);
}
markActive();
return true;
}
@Override
public boolean rcvMeta(ServerSession session, ServerMessage.Mutable message) {
if (Channel.META_CONNECT.equals(message.getChannel())) {
if (isInactive()) {
if (logger.isDebugEnabled()) {
logger.debug("Inactive session {}, disconnecting", session);
}
disconnect(session);
}
} else {
if (logger.isDebugEnabled()) {
logger.debug("Marking active session {}, received meta message {}", session, message);
}
markActive();
}
return true;
}
@Override
public ServerMessage send(ServerSession sender, ServerSession session, ServerMessage message) {
if (activity == Activity.CLIENT_SERVER) {
if (logger.isDebugEnabled()) {
logger.debug("Marking active session {}, sending message {}", session, message);
}
markActive();
}
return message;
}
@Override
public boolean sendMeta(ServerSession sender, ServerSession session, ServerMessage.Mutable message) {
if (!Channel.META_CONNECT.equals(message.getChannel()) && activity == Activity.CLIENT_SERVER) {
if (logger.isDebugEnabled()) {
logger.debug("Marking active session {}, sending meta message {}", session, message);
}
markActive();
}
return true;
}
protected void markActive() {
lastActivity.set(System.nanoTime());
}
public boolean isInactive() {
return TimeUnit.NANOSECONDS.toMillis(System.nanoTime() - getLastActivity()) > getMaxInactivityPeriod();
}
protected void disconnect(ServerSession session) {
if (session != null) {
session.disconnect();
}
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy