org.cometd.server.ext.AcknowledgedMessagesSessionExtension 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.server.ext;
import java.util.ArrayList;
import java.util.Collections;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.Queue;
import java.util.concurrent.CopyOnWriteArrayList;
import org.cometd.bayeux.Channel;
import org.cometd.bayeux.Message;
import org.cometd.bayeux.server.ServerMessage;
import org.cometd.bayeux.server.ServerMessage.Mutable;
import org.cometd.bayeux.server.ServerSession;
import org.cometd.bayeux.server.ServerSession.Extension;
import org.cometd.server.ServerSessionImpl;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
/**
* Tracks the batch id of messages sent to a client.
*/
public class AcknowledgedMessagesSessionExtension implements Extension, ServerSession.DeQueueListener, ServerSession.QueueListener {
private static final Logger _logger = LoggerFactory.getLogger(AcknowledgedMessagesSessionExtension.class);
private final List _listeners = new CopyOnWriteArrayList<>();
private final Map _batches = new HashMap<>();
private final ServerSessionImpl _session;
private final BatchArrayQueue _queue;
private long _lastBatch;
public AcknowledgedMessagesSessionExtension(ServerSession session) {
_session = (ServerSessionImpl)session;
_queue = new BatchArrayQueue<>(16, _session.getLock());
_session.setMetaConnectDeliveryOnly(true);
_session.addListener(this);
}
void addListeners(List listeners) {
_listeners.addAll(listeners);
}
public void addListener(AcknowledgedMessagesExtension.Listener listener) {
_listeners.add(listener);
}
public void removeListener(AcknowledgedMessagesExtension.Listener listener) {
_listeners.remove(listener);
}
@Override
public boolean rcv(ServerSession from, Mutable message) {
return true;
}
@Override
public boolean rcvMeta(ServerSession session, Mutable message) {
if (Channel.META_CONNECT.equals(message.getChannel())) {
Map ext = message.getExt(false);
if (ext != null) {
Number batchValue = (Number)ext.get("ack");
if (batchValue != null) {
processBatch(batchValue.longValue());
updateAdvice(message);
}
}
}
return true;
}
private void updateAdvice(Mutable message) {
synchronized (_session.getLock()) {
if (!_session.hasNonLazyMessages() && _session.getQueue().size() != _queue.size()) {
Map advice = message.getAdvice(true);
if (advice.get(Message.TIMEOUT_FIELD) == null) {
advice.put(Message.TIMEOUT_FIELD, 0L);
if (_logger.isDebugEnabled()) {
_logger.debug("Forcing advice: { timeout: 0 } for {}", _session);
}
}
}
}
}
protected void processBatch(long batch) {
synchronized (_session.getLock()) {
if (_logger.isDebugEnabled()) {
_logger.debug("Processing batch: last={}, client={}, server={} for {}", _lastBatch, batch, _queue.getBatch(), _session);
}
_lastBatch = batch;
_queue.clearToBatch(batch);
notifyBatchReceive(_session, batch);
}
}
@Override
public ServerMessage send(ServerSession sender, ServerSession session, ServerMessage message) {
// Too early to do anything with the message.
// Other extensions and/or listener may modify/veto it.
return message;
}
@Override
public void queued(ServerSession sender, ServerMessage message) {
// This method is called after all the extensions and the other
// listeners, so only here are sure that the message is not vetoed.
synchronized (_session.getLock()) {
_queue.offer(message);
if (_logger.isDebugEnabled()) {
_logger.debug("Stored at batch {} {} for {}", _queue.getBatch(), message, _session);
}
}
}
@Override
public boolean sendMeta(ServerSession sender, ServerSession to, Mutable message) {
String channel = message.getChannel();
Map ext = message.getExt(true);
if (channel.equals(Channel.META_HANDSHAKE)) {
if (_session.isAllowMessageDeliveryDuringHandshake()) {
long batch = closeBatch(message);
Map ack = new HashMap<>(3);
ack.put("enabled", true);
ack.put("batch", batch);
ext.put("ack", ack);
if (_logger.isDebugEnabled()) {
_logger.debug("Sending batch {} for {}", batch, _session);
}
} else {
ext.put("ack", Boolean.TRUE);
}
} else if (channel.equals(Channel.META_CONNECT)) {
long batch = closeBatch(message);
ext.put("ack", batch);
if (_logger.isDebugEnabled()) {
_logger.debug("Sending batch {} for {}", batch, _session);
}
}
return true;
}
private long closeBatch(Mutable message) {
synchronized (_session.getLock()) {
long batch = _queue.getBatch();
_batches.put(message.getId(), batch);
_queue.nextBatch();
return batch;
}
}
@Override
public void deQueue(ServerSession session, Queue queue, List replies) {
Mutable reply = null;
for (Mutable r : replies) {
String channel = r.getChannel();
if (Channel.META_HANDSHAKE.equals(channel) || Channel.META_CONNECT.equals(channel)) {
reply = r;
break;
}
}
if (reply != null) {
long batch = _batches.remove(reply.getId());
synchronized (_session.getLock()) {
if (_logger.isDebugEnabled()) {
_logger.debug("Dequeuing {}/{} messages until batch {} for {} on {}", queue.size(), _queue.size(), batch, reply, _session);
}
queue.clear();
_queue.exportMessagesToBatch(queue, batch);
notifyBatchSend(_session, queue, batch);
}
}
}
@Override
public void deQueue(ServerSession session, Queue queue) {
}
protected void importMessages(ServerSessionImpl session) {
synchronized (_session.getLock()) {
_queue.addAll(session.getQueue());
}
}
private void notifyBatchSend(ServerSession session, Queue queue, long batch) {
List messages = null;
for (AcknowledgedMessagesExtension.Listener listener : _listeners) {
if (messages == null) {
messages = Collections.unmodifiableList(new ArrayList<>(queue));
}
try {
listener.onBatchSend(session, messages, batch);
} catch (Throwable x) {
_logger.info("Exception while invoking listener " + listener, x);
}
}
}
private void notifyBatchReceive(ServerSession session, long batch) {
for (AcknowledgedMessagesExtension.Listener listener : _listeners) {
try {
listener.onBatchReceive(session, batch);
} catch (Throwable x) {
_logger.info("Exception while invoking listener " + listener, x);
}
}
}
// Used only in tests.
BatchArrayQueue getBatchArrayQueue() {
return _queue;
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy