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

org.cometd.server.AbstractService Maven / Gradle / Ivy

/*
 * Copyright (c) 2010 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.lang.reflect.Method;
import java.util.Map;
import java.util.concurrent.ConcurrentHashMap;

import org.cometd.bayeux.Message;
import org.cometd.bayeux.Session;
import org.cometd.bayeux.server.BayeuxServer;
import org.cometd.bayeux.server.LocalSession;
import org.cometd.bayeux.server.ServerChannel;
import org.cometd.bayeux.server.ServerMessage;
import org.cometd.bayeux.server.ServerMessage.Mutable;
import org.cometd.bayeux.server.ServerSession;
import org.eclipse.jetty.util.component.LifeCycle;
import org.eclipse.jetty.util.thread.QueuedThreadPool;
import org.eclipse.jetty.util.thread.ThreadPool;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

/**
 * 

{@link AbstractService} provides convenience methods to assist with the * creation of a CometD services.

*

A CometD service runs application code whenever a message is received on * a particular channel.

*

Specifically it provides:

*
    *
  • Mapping of channel subscriptions to method invocation on the derived * service class.
  • *
  • Optional use of a thread pool used for method invocation if handling * can take considerable time and it is desired not to hold up the delivering * thread (typically a HTTP request handling thread).
  • *
  • The objects returned from method invocation are delivered back to the * calling client in a private message.
  • *
*

Subclasses should call {@link #addService(String, String)} in order to * map channel subscriptions to method invocations, usually in the subclass * constructor.

*

Each CometD service has an associated {@link LocalSession} that can be * used as the source for messages published via * {@link ServerChannel#publish(Session, Mutable)} or * {@link ServerSession#deliver(Session, Mutable)}.

* * @see {@link BayeuxServer#newLocalSession(String)} as an alternative to {@link AbstractService}. */ public abstract class AbstractService { protected final Logger _logger = LoggerFactory.getLogger(getClass()); private final Map invokers = new ConcurrentHashMap(); private final String _name; private final BayeuxServerImpl _bayeux; private final LocalSession _session; private ThreadPool _threadPool; private boolean _seeOwn = false; /** *

Instantiates a CometD service with the given name.

* * @param bayeux The BayeuxServer instance. * @param name The name of the service (used as client ID prefix). */ public AbstractService(BayeuxServer bayeux, String name) { this(bayeux, name, 0); } /** *

Instantiate a CometD service with the given name and max number of pooled threads.

* * @param bayeux The BayeuxServer instance. * @param name The name of the service (used as client ID prefix). * @param maxThreads The max size of a ThreadPool to create to handle messages. */ public AbstractService(BayeuxServer bayeux, String name, int maxThreads) { _name = name; _bayeux = (BayeuxServerImpl)bayeux; _session = _bayeux.newLocalSession(name); _session.handshake(); if (maxThreads > 0) setThreadPool(new QueuedThreadPool(maxThreads)); } public BayeuxServer getBayeux() { return _bayeux; } /** * @return The {@link LocalSession} associated with this CometD service */ public LocalSession getLocalSession() { return _session; } /** * @return The {@link ServerSession} of the {@link LocalSession} associated * with this CometD service */ public ServerSession getServerSession() { return _session.getServerSession(); } /** * @return The thread pool associated with this CometD service, or null * @see #AbstractService(BayeuxServer, String, int) */ public ThreadPool getThreadPool() { return _threadPool; } /** *

Sets the thread pool associated to this CometD service.

*

If the {@link ThreadPool} is a {@link LifeCycle} instance, * and it is not already started, then it will started.

* * @param pool The ThreadPool */ public void setThreadPool(ThreadPool pool) { try { if (pool instanceof LifeCycle) if (!((LifeCycle)pool).isStarted()) ((LifeCycle)pool).start(); } catch (Exception e) { throw new IllegalStateException(e); } _threadPool = pool; } /** * @return whether this CometD service receives messages published by itself * on channels it is subscribed to (defaults to false). * @see #setSeeOwnPublishes(boolean) */ public boolean isSeeOwnPublishes() { return _seeOwn; } /** * @param seeOwnPublishes whether this CometD service receives messages published by itself * on channels it is subscribed to (defaults to false). * @see #isSeeOwnPublishes() */ public void setSeeOwnPublishes(boolean seeOwnPublishes) { _seeOwn = seeOwnPublishes; } /** *

Maps the method of a subclass with the given name to a * {@link ServerChannel.MessageListener} on the given channel, so that the method * is invoked for each message received on the channel.

*

The channel name may be a {@link ServerChannel#isWild() wildcard channel name}.

*

The method must have a unique name and one of the following signatures:

*
    *
  • myMethod(ServerSession from, Object data)
  • *
  • myMethod(ServerSession from, Object data, String messageId)
  • *
  • myMethod(ServerSession from, String channel, Object data, String messageId)
  • *
*

The data parameter can be a specific type if the type of * the data object published by the client is known by the server. * If it is not known will be Map<String, Object>.

*

If the type of the data parameter is {@link Message} (or a subinterface * such as {@link ServerMessage.Mutable} then the message object itself is * passed rather than just the message's data.

*

Typically a service will be used to a channel in the /service/** * space which is not a broadcast channel.

*

Any object returned by a mapped method is delivered back to the * client that sent the message and not broadcasted. If the method returns void or null, * then no response is sent.

*

A mapped method may also call {@link #send(ServerSession, String, Object, String)} * to deliver message(s) to specific clients and/or channels.

*

A mapped method may also publish to different channels via * {@link ServerChannel#publish(Session, Mutable)}.

* * @param channelName The channel to listen to * @param methodName The name of the method on this subclass to call when messages * are received on the channel * @see #removeService(String, String) */ protected void addService(String channelName, String methodName) { _logger.debug("Mapping {}#{} to {}", new Object[]{_name, methodName, channelName}); Method method = null; Class c = this.getClass(); while (c != null && c != Object.class) { Method[] methods = c.getDeclaredMethods(); for (int i = methods.length; i-- > 0; ) { if (methodName.equals(methods[i].getName())) { if (method != null) throw new IllegalArgumentException("Multiple methods called '" + methodName + "'"); method = methods[i]; } } c = c.getSuperclass(); } if (method == null) throw new NoSuchMethodError(methodName); int params = method.getParameterTypes().length; if (params < 2 || params > 4) throw new IllegalArgumentException("Method '" + methodName + "' does not have 2, 3 or 4 parameters"); if (!ServerSession.class.isAssignableFrom(method.getParameterTypes()[0])) throw new IllegalArgumentException("Method '" + methodName + "' does not have Session as first parameter"); _bayeux.createIfAbsent(channelName); ServerChannel channel = _bayeux.getChannel(channelName); Invoker invoker = new Invoker(channelName, method); invokers.put(methodName, invoker); channel.addListener(invoker); } /** *

Unmaps the method with the given name that has been mapped to the given channel.

* * @param channelName The channel name * @param methodName The name of the method to unmap * @see #addService(String, String) * @see #removeService(String) */ protected void removeService(String channelName, String methodName) { ServerChannel channel = _bayeux.getChannel(channelName); if (channel != null) { Invoker invoker = invokers.remove(methodName); channel.removeListener(invoker); } } /** *

Unmaps all the methods that have been mapped to the given channel.

* * @param channelName The channel name * @see #addService(String, String) * @see #removeService(String, String) */ protected void removeService(String channelName) { ServerChannel channel = _bayeux.getChannel(channelName); if (channel != null) { for (Invoker invoker : invokers.values()) { if (invoker.channelName.equals(channelName)) channel.removeListener(invoker); } } } /** *

Sends data to an individual remote client.

*

The data passed is sent to the client * as the "data" member of a message with the given channel and id. The * message is not published on the channel and is thus not broadcast to all * channel subscribers, but instead delivered directly to the target client.

*

Typically this method is only required if a service method sends * response(s) to clients other than the sender, or on different channels. * If the response is to be sent to the sender on the same channel, * then the data can simply be the return value of the method.

* * @param toClient The target client * @param onChannel The channel of the message * @param data The data of the message * @param id The id of the message (or null for a random id). */ protected void send(ServerSession toClient, String onChannel, Object data, String id) { toClient.deliver(_session.getServerSession(), onChannel, data, id); } /** *

Handles exceptions during the invocation of a mapped method.

*

This method is called when a mapped method throws and exception while handling a message.

* * @param method the name of the method invoked that threw an exception * @param fromClient the remote session that sent the message * @param toClient the local session associated to this service * @param msg the message sent by the remote session * @param x the exception thrown */ protected void exception(String method, ServerSession fromClient, LocalSession toClient, ServerMessage msg, Throwable x) { _logger.info("Exception while invoking " + _name + "#" + method + " from " + fromClient + " with " + msg, x); } private void invoke(final Method method, final ServerSession fromClient, final ServerMessage msg) { _logger.debug("Invoking {}#{} from {} with {}", new Object[]{_name, method.getName(), fromClient, msg}); ThreadPool threadPool = getThreadPool(); if (threadPool == null) doInvoke(method, fromClient, msg); else { threadPool.dispatch(new Runnable() { public void run() { doInvoke(method, fromClient, msg); } }); } } protected void doInvoke(Method method, ServerSession fromClient, ServerMessage msg) { String channel = msg.getChannel(); Object data = msg.getData(); String id = msg.getId(); if (method != null) { try { Class[] parameterTypes = method.getParameterTypes(); int messageParameterIndex = parameterTypes.length == 4 ? 2 : 1; Object messageArgument = data; if (Message.class.isAssignableFrom(parameterTypes[messageParameterIndex])) messageArgument = msg; boolean accessible = method.isAccessible(); Object reply = null; try { method.setAccessible(true); switch (method.getParameterTypes().length) { case 2: reply = method.invoke(this, fromClient, messageArgument); break; case 3: reply = method.invoke(this, fromClient, messageArgument, id); break; case 4: reply = method.invoke(this, fromClient, channel, messageArgument, id); break; } } finally { method.setAccessible(accessible); } if (reply != null) send(fromClient, channel, reply, id); } catch (Exception e) { exception(method.toString(), fromClient, _session, msg, e); } catch (Error e) { exception(method.toString(), fromClient, _session, msg, e); } } } private class Invoker implements ServerChannel.MessageListener { private final String channelName; private final Method method; public Invoker(String channelName, Method method) { this.channelName = channelName; this.method = method; } public boolean onMessage(ServerSession from, ServerChannel channel, Mutable message) { if (isSeeOwnPublishes() || from != getServerSession()) invoke(method, from, message); return true; } } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy