All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.cometd.common.AbstractClientSession Maven / Gradle / Ivy

/*
 * Copyright (c) 2008 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.common;

import java.io.IOException;
import java.util.List;
import java.util.Map;
import java.util.Set;
import java.util.TreeMap;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.ConcurrentMap;
import java.util.concurrent.CopyOnWriteArrayList;
import java.util.concurrent.atomic.AtomicInteger;
import java.util.concurrent.atomic.AtomicLong;
import org.cometd.bayeux.Channel;
import org.cometd.bayeux.ChannelId;
import org.cometd.bayeux.MarkedReference;
import org.cometd.bayeux.Message;
import org.cometd.bayeux.Promise;
import org.cometd.bayeux.client.ClientSession;
import org.cometd.bayeux.client.ClientSessionChannel;
import org.eclipse.jetty.util.Attributes;
import org.eclipse.jetty.util.component.Dumpable;
import org.eclipse.jetty.util.component.DumpableCollection;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

/**
 * 

Partial implementation of {@link ClientSession}.

*

It handles extensions and batching, and provides utility methods to be used by subclasses.

*/ public abstract class AbstractClientSession implements ClientSession, Dumpable { private static final Logger _logger = LoggerFactory.getLogger(ClientSession.class); private static final AtomicLong _idGen = new AtomicLong(0); private final List _extensions = new CopyOnWriteArrayList<>(); private final Attributes _attributes = new Attributes.Lazy(); private final ConcurrentMap _channels = new ConcurrentHashMap<>(); private final Map _callbacks = new ConcurrentHashMap<>(); private final Map _subscribers = new ConcurrentHashMap<>(); private final Map _remoteCalls = new ConcurrentHashMap<>(); private final AtomicInteger _batch = new AtomicInteger(); protected AbstractClientSession() { } protected String newMessageId() { return String.valueOf(_idGen.incrementAndGet()); } @Override public void addExtension(Extension extension) { _extensions.add(extension); } @Override public void removeExtension(Extension extension) { _extensions.remove(extension); } @Override public List getExtensions() { return List.copyOf(_extensions); } protected void extendOutgoing(Message.Mutable message, Promise promise) { AsyncFoldLeft.reverseRun(_extensions, true, (result, extension, loop) -> { if (result) { try { extension.outgoing(this, message, Promise.from(loop::proceed, failure -> { _logger.info("Exception reported by extension " + extension, failure); loop.proceed(true); })); } catch (Throwable x) { _logger.info("Exception thrown by extension " + extension, x); loop.proceed(true); } } else { loop.leave(false); } }, promise); } protected void extendIncoming(Message.Mutable message, Promise promise) { AsyncFoldLeft.run(_extensions, true, (result, extension, loop) -> { if (result) { try { extension.incoming(this, message, Promise.from(loop::proceed, failure -> { _logger.info("Exception reported by extension " + extension, failure); loop.proceed(true); })); } catch (Throwable x) { _logger.info("Exception thrown by extension " + extension, x); loop.proceed(true); } } else { loop.leave(false); } }, promise); } protected abstract ChannelId newChannelId(String channelId); protected abstract AbstractSessionChannel newChannel(ChannelId channelId); @Override public ClientSessionChannel getChannel(String channelName) { return getChannel(channelName, null); } public ClientSessionChannel getChannel(ChannelId channelId) { return getChannel(channelId.getId(), channelId); } private ClientSessionChannel getChannel(String channelName, ChannelId channelId) { AbstractSessionChannel channel = _channels.get(channelName); if (channel != null) { return channel; } if (channelId == null) { // Creating the ChannelId will also normalize the channelName. channelId = newChannelId(channelName); return getChannel(channelId); } AbstractSessionChannel newChannel = newChannel(channelId); channel = _channels.putIfAbsent(channelId.getId(), newChannel); if (channel == null) { channel = newChannel; } return channel; } protected ConcurrentMap getChannels() { return _channels; } @Override public void startBatch() { _batch.incrementAndGet(); } protected abstract void sendBatch(); @Override public boolean endBatch() { if (_batch.decrementAndGet() == 0) { sendBatch(); return true; } return false; } @Override public void batch(Runnable batch) { startBatch(); try { batch.run(); } finally { endBatch(); } } protected boolean isBatching() { return _batch.get() > 0; } @Override public Object getAttribute(String name) { return _attributes.getAttribute(name); } @Override public Set getAttributeNames() { return _attributes.getAttributeNameSet(); } @Override public Object removeAttribute(String name) { Object old = _attributes.getAttribute(name); _attributes.removeAttribute(name); return old; } @Override public void setAttribute(String name, Object value) { _attributes.setAttribute(name, value); } @Override public void remoteCall(String target, Object data, MessageListener callback) { if (!target.startsWith("/")) { target = "/" + target; } String channelName = "/service" + target; Message.Mutable message = newMessage(); String messageId = newMessageId(); message.setId(messageId); message.setChannel(channelName); message.setData(data); _remoteCalls.put(messageId, callback); send(message); } protected abstract void send(Message.Mutable message); protected Message.Mutable newMessage() { return new HashMapMessage(); } protected void resetSubscriptions() { for (AbstractSessionChannel ch : _channels.values()) { ch.resetSubscriptions(); } } /** *

Receives a message (from the server) and process it.

*

Processing the message involves calling the receive {@link Extension extensions} * and the channel {@link ClientSessionChannel.ClientSessionChannelListener listeners}.

* * @param message the message received. * @param promise the promise notified of the receive processing */ public void receive(Message.Mutable message, Promise promise) { String channelName = message.getChannel(); if (channelName == null) { promise.fail(new IllegalArgumentException("Bayeux message must have a channel: " + message)); return; } if (Channel.META_SUBSCRIBE.equals(channelName)) { // Remove the subscriber if the subscription fails. ClientSessionChannel.MessageListener subscriber = unregisterSubscriber(message.getId()); if (!message.isSuccessful()) { String subscription = (String)message.get(Message.SUBSCRIPTION_FIELD); // Avoid NPE when the server sends an invalid message with no subscription field. if (subscription != null) { MarkedReference channelRef = getReleasableChannel(subscription); AbstractSessionChannel channel = channelRef.getReference(); channel.removeSubscription(subscriber); if (channelRef.isMarked()) { channel.release(); } } } } extendIncoming(message, Promise.from(pass -> { if (pass) { if (message.isPublishReply() && message.isSuccessful()) { notifyListeners(message); } else { if (!handleRemoteCall(message)) { notifyListeners(message); } } } else { unregisterCallback(message.getId()); } promise.succeed(null); }, promise::fail)); } private boolean handleRemoteCall(Message.Mutable message) { String messageId = message.getId(); if (messageId != null) { MessageListener listener = _remoteCalls.remove(messageId); if (listener != null) { notifyMessageListener(listener, message); return true; } } return false; } private void notifyMessageListener(MessageListener listener, Message.Mutable message) { try { listener.onMessage(message); } catch (Throwable x) { _logger.info("Exception while invoking listener " + listener, x); } } protected void notifyListeners(Message.Mutable message) { if (message.isMeta() || message.isPublishReply()) { String messageId = message.getId(); ClientSession.MessageListener callback = unregisterCallback(messageId); if (callback != null) { notifyCallback(callback, message); } } ChannelId channelId = newChannelId(message.getChannel()); for (String channelName : channelId.getAllIds()) { MarkedReference channelRef = getReleasableChannel(channelName); AbstractSessionChannel channel = channelRef.getReference(); channel.notifyMessageListeners(message); if (channelRef.isMarked()) { channel.release(); } } } protected void notifyCallback(ClientSession.MessageListener callback, Message.Mutable message) { try { callback.onMessage(message); } catch (Throwable x) { _logger.info("Exception while invoking callback " + callback, x); } } private MarkedReference getReleasableChannel(String id) { // Use getChannels().get(channelName) instead of getChannel(channelName) // to avoid to cache channels that can be released immediately. AbstractSessionChannel channel = ChannelId.isMeta(id) ? (AbstractSessionChannel)getChannel(id) : getChannels().get(id); if (channel != null) { return new MarkedReference<>(channel, false); } return new MarkedReference<>(newChannel(newChannelId(id)), true); } protected void registerCallback(String messageId, ClientSession.MessageListener callback) { if (callback != null) { _callbacks.put(messageId, callback); } } protected ClientSession.MessageListener unregisterCallback(String messageId) { if (messageId == null) { return null; } return _callbacks.remove(messageId); } protected void registerSubscriber(String messageId, ClientSessionChannel.MessageListener subscriber) { if (subscriber != null) { _subscribers.put(messageId, subscriber); } } protected ClientSessionChannel.MessageListener unregisterSubscriber(String messageId) { if (messageId == null) { return null; } return _subscribers.remove(messageId); } @Override public void dump(Appendable out, String indent) throws IOException { Dumpable.dumpObjects(out, indent, this, _attributes, new DumpableCollection("extensions", _extensions), new DumpableCollection("channels", new TreeMap<>(_channels).values())); } /** *

A channel scoped to a {@link ClientSession}.

*/ protected abstract class AbstractSessionChannel implements ClientSessionChannel, Dumpable { private final ChannelId _id; private final Attributes _attributes = new Attributes.Lazy(); private final CopyOnWriteArrayList _subscriptions = new CopyOnWriteArrayList<>(); private final AtomicInteger _subscriptionCount = new AtomicInteger(); private final CopyOnWriteArrayList _listeners = new CopyOnWriteArrayList<>(); private volatile boolean _released; protected AbstractSessionChannel(ChannelId id) { _id = id; } @Override public ChannelId getChannelId() { return _id; } @Override public void addListener(ClientSessionChannelListener listener) { throwIfReleased(); _listeners.add(listener); } @Override public void removeListener(ClientSessionChannelListener listener) { throwIfReleased(); _listeners.remove(listener); } @Override public List getListeners() { return List.copyOf(_listeners); } @Override public void publish(Object data, ClientSession.MessageListener callback) { if (data instanceof Message.Mutable) { publish((Message.Mutable)data, callback); } else { Message.Mutable message = newMessage(); message.setData(data); publish(message, callback); } } @Override public void publish(Message.Mutable message, ClientSession.MessageListener callback) { throwIfReleased(); String messageId = newMessageId(); message.setId(messageId); message.setChannel(getId()); registerCallback(messageId, callback); send(message); } @Override public boolean subscribe(Message.Mutable message, MessageListener listener, ClientSession.MessageListener callback) { throwIfReleased(); _subscriptions.add(listener); int count = _subscriptionCount.incrementAndGet(); if (count == 1) { sendSubscribe(message, listener, callback); return true; } else { nonFirstSubscribe(message, listener, callback); return false; } } protected void sendSubscribe(Message.Mutable message, MessageListener listener, ClientSession.MessageListener callback) { if (message == null) { message = newMessage(); } String messageId = newMessageId(); message.setId(messageId); message.setChannel(Channel.META_SUBSCRIBE); message.put(Message.SUBSCRIPTION_FIELD, getId()); registerSubscriber(messageId, listener); registerCallback(messageId, callback); send(message); } protected void nonFirstSubscribe(Message.Mutable message, MessageListener listener, ClientSession.MessageListener callback) { if (callback != null) { Message.Mutable reply = newMessage(); reply.setSuccessful(true); reply.setId(newMessageId()); reply.setChannel(Channel.META_SUBSCRIBE); reply.put(Message.SUBSCRIPTION_FIELD, getId()); notifyCallback(callback, reply); } } @Override public boolean unsubscribe(Message.Mutable message, MessageListener listener, ClientSession.MessageListener callback) { boolean removedLast = removeSubscription(listener); if (removedLast) { sendUnSubscribe(message, listener, callback); return true; } else { nonLastUnSubscribe(message, listener, callback); return false; } } private boolean removeSubscription(MessageListener listener) { throwIfReleased(); boolean removed = _subscriptions.remove(listener); if (removed) { return _subscriptionCount.decrementAndGet() == 0; } return false; } protected void sendUnSubscribe(Message.Mutable message, MessageListener listener, ClientSession.MessageListener callback) { if (message == null) { message = newMessage(); } String messageId = newMessageId(); message.setId(messageId); message.setChannel(Channel.META_UNSUBSCRIBE); message.put(Message.SUBSCRIPTION_FIELD, getId()); registerCallback(messageId, callback); send(message); } protected void nonLastUnSubscribe(Message.Mutable message, MessageListener listener, ClientSession.MessageListener callback) { if (callback != null) { Message.Mutable reply = newMessage(); reply.setSuccessful(true); reply.setId(newMessageId()); reply.setChannel(Channel.META_UNSUBSCRIBE); reply.put(Message.SUBSCRIPTION_FIELD, getId()); notifyCallback(callback, reply); } } @Override public void unsubscribe() { throwIfReleased(); for (MessageListener listener : _subscriptions) { unsubscribe(listener); } } @Override public List getSubscribers() { return List.copyOf(_subscriptions); } @Override public boolean release() { if (_released) { return false; } if (_subscriptions.isEmpty() && _listeners.isEmpty()) { boolean removed = _channels.remove(getId(), this); _released = removed; return removed; } return false; } @Override public boolean isReleased() { return _released; } protected void resetSubscriptions() { throwIfReleased(); for (MessageListener l : _subscriptions) { if (_subscriptions.remove(l)) { _subscriptionCount.decrementAndGet(); } } } @Override public String getId() { return _id.getId(); } @Override public boolean isDeepWild() { return _id.isDeepWild(); } @Override public boolean isMeta() { return _id.isMeta(); } @Override public boolean isService() { return _id.isService(); } @Override public boolean isBroadcast() { return !isMeta() && !isService(); } @Override public boolean isWild() { return _id.isWild(); } protected void notifyMessageListeners(Message message) { throwIfReleased(); for (ClientSessionChannelListener listener : _listeners) { if (listener instanceof ClientSessionChannel.MessageListener) { notifyOnMessage((MessageListener)listener, message); } } for (MessageListener listener : _subscriptions) { if (!message.isPublishReply()) { notifyOnMessage(listener, message); } } } protected void notifyOnMessage(MessageListener listener, Message message) { throwIfReleased(); try { listener.onMessage(this, message); } catch (Throwable x) { _logger.info("Exception while invoking listener " + listener, x); } } @Override public void setAttribute(String name, Object value) { throwIfReleased(); _attributes.setAttribute(name, value); } @Override public Object getAttribute(String name) { throwIfReleased(); return _attributes.getAttribute(name); } @Override public Set getAttributeNames() { throwIfReleased(); return _attributes.getAttributeNameSet(); } @Override public Object removeAttribute(String name) { throwIfReleased(); Object old = getAttribute(name); _attributes.removeAttribute(name); return old; } protected void throwIfReleased() { if (isReleased()) { throw new IllegalStateException("Channel " + this + " has been released"); } } @Override public String dumpSelf() { return getId(); } @Override public void dump(Appendable out, String indent) throws IOException { Dumpable.dumpObjects(out, indent, this, new DumpableCollection("listeners", _listeners), new DumpableCollection("subscribers", _subscriptions)); } @Override public String toString() { return String.format("%s@%x[%s][%s]", getClass().getSimpleName(), hashCode(), getChannelId(), AbstractClientSession.this); } } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy