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

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

// ========================================================================
// Copyright 2008 Mort Bay Consulting Pty. Ltd.
// ------------------------------------------------------------------------
// 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.io.IOException;
import java.lang.reflect.Method;
import java.util.Map;
import java.util.concurrent.ConcurrentHashMap;

import org.cometd.bayeux.Bayeux;
import org.cometd.bayeux.Message;
import org.cometd.bayeux.Session;
import org.cometd.bayeux.client.SessionChannel;
import org.cometd.bayeux.client.SessionChannel.SubscriberListener;
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.ServerSession;
import org.cometd.bayeux.server.ServerMessage.Mutable;
import org.cometd.bayeux.server.ServerSession.MessageListener;
import org.cometd.common.ChannelId;
import org.eclipse.jetty.util.component.LifeCycle;
import org.eclipse.jetty.util.log.Log;
import org.eclipse.jetty.util.log.Logger;
import org.eclipse.jetty.util.thread.QueuedThreadPool;
import org.eclipse.jetty.util.thread.ThreadPool;

/* ------------------------------------------------------------ */
/**
 * Abstract Bayeux Service class. This is a base class to assist with the
 * creation of server side @ link Bayeux} clients that provide services to
 * remote Bayeux clients. The class provides a Bayeux {@link Client} and
 * {@link Listener} together with convenience methods to map subscriptions to
 * methods on the derived class and to send responses to those methods.
 * 
 * 

* If a {@link #set_threadPool(ThreadPool)} is set, then messages are handled in * their own threads. This is desirable if the handling of a message can take * considerable time and it is desired not to hold up the delivering thread * (typically a HTTP request handling thread). * *

* If the BayeuxService is constructed asynchronously (the default), then * messages are delivered unsynchronized and multiple simultaneous calls to * handling methods may occur. * *

* If the BayeuxService is constructed as a synchronous service, then message * delivery is synchronized on the internal {@link Client} instances used and * only a single call will be made to the handler method (unless a thread pool * is used). * * @see MessageListener * @author gregw * */ public abstract class BayeuxService { private final String _name; private final BayeuxServerImpl _bayeux; private final LocalSession _session; private final Map _methods=new ConcurrentHashMap(); private final Map _wild=new ConcurrentHashMap(); private ThreadPool _threadPool; private boolean _seeOwn=false; private final Logger _logger; /* ------------------------------------------------------------ */ /** * Instantiate the service. Typically the derived constructor will call @ * #subscribe(String, String)} to map subscriptions to methods. * * @param bayeux * The bayeux instance. * @param name * The name of the service (used as client ID prefix). */ public BayeuxService(BayeuxServer bayeux, String name) { this(bayeux,name,0); } /* ------------------------------------------------------------ */ /** * Instantiate the service. Typically the derived constructor will call @ * #subscribe(String, String)} to map subscriptions to methods. * * @param bayeux * The bayeux instance. * @param name * The name of the service (used as client ID prefix). * @param maxThreads * The size of a ThreadPool to create to handle messages. */ public BayeuxService(BayeuxServer bayeux, String name, int maxThreads) { if (maxThreads > 0) setThreadPool(new QueuedThreadPool(maxThreads)); _name=name; _bayeux=(BayeuxServerImpl)bayeux; _session=_bayeux.newLocalSession(name); try { _session.handshake(); } catch(IOException e) { throw new RuntimeException(e); } _logger=((BayeuxServerImpl)bayeux).getLogger(); } /* ------------------------------------------------------------ */ public BayeuxServer getBayeux() { return _bayeux; } /* ------------------------------------------------------------ */ /** * @deprecated use {@link #getLocalSession()} or {@link #getServerSession()} */ public LocalSession getClient() { return _session; } /* ------------------------------------------------------------ */ public LocalSession getLocalSession() { return _session; } /* ------------------------------------------------------------ */ public ServerSession getServerSession() { return _session.getServerSession(); } /* ------------------------------------------------------------ */ public ThreadPool getThreadPool() { return _threadPool; } /* ------------------------------------------------------------ */ /** * Set the threadpool. If the {@link ThreadPool} is a {@link LifeCycle}, * then it is started by this method. * * @param pool */ 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; } /* ------------------------------------------------------------ */ public boolean isSeeOwnPublishes() { return _seeOwn; } /* ------------------------------------------------------------ */ public void setSeeOwnPublishes(boolean own) { _seeOwn=own; } /* ------------------------------------------------------------ */ /** * Subscribe to a channel. Subscribe to channel and map a method to handle * received messages. The method must have a unique name and one of the * following signatures: *

    *
  • myMethod(ServerSession from,Object data)
  • *
  • myMethod(ServerSession from,Object data,String|Object id)
  • *
  • myMethod(ServerSession from,String channel,Object data,String|Object id) *
  • * * * The data parameter can be typed if the type of the data object published * by the client is known (typically Map). If the type of the * data parameter is {@link Message} then the message object itself is * passed rather than just the data. *

    * Typically a service will subscribe to a channel in the "/service/**" * space which is not a broadcast channel. Messages published to these * channels are only delivered to server side clients like this service. * *

    * Any object returned by a mapped subscription method is delivered to the * calling client and not broadcast. If the method returns void or null, * then no response is sent. A mapped subscription method may also call * {@link #send(Client, String, Object, String)} to deliver a response * message(s) to different clients and/or channels. It may also publish * methods via the normal {@link Bayeux} API. *

    * * * @param channelId * The channel to subscribe to * @param methodName * The name of the method on this object to call when messages * are recieved. */ protected void subscribe(String channelId, String methodName) { if (_logger.isDebugEnabled()) _logger.debug("subscribe "+_name+"#"+methodName+" to "+channelId); 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 2or3 parameters"); if (!ServerSession.class.isAssignableFrom(method.getParameterTypes()[0])) throw new IllegalArgumentException("Method '" + methodName + "' does not have Session as first parameter"); ServerChannel channel=_bayeux.getChannel(channelId,true); if (channel.isWild()) _wild.put(new ChannelId(channel.getId()),method); else _methods.put(channelId,method); final Method invoke=method; channel.addListener(new ServerChannel.MessageListener() { @Override public boolean onMessage(ServerSession from, ServerChannel channel, Mutable message) { if (_seeOwn || from != getServerSession()) invoke(invoke,from,message); return true; } }); } /* ------------------------------------------------------------ */ /** * Send data to a individual 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. However to the target client, the message appears as * if it was broadcast. *

    * Typcially this method is only required if a service method sends * response(s) to channels other than the subscribed channel. If the * response is to be sent to the subscribed channel, then the data can * simply be returned from the subscription method. * * @param toClient * The target client * @param onChannel * The channel the message is for * @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, Object id) { toClient.deliver(_session.getServerSession(),onChannel,data,id); } /* ------------------------------------------------------------ */ /** * Handle Exception. This method is called when a mapped subscription method * throws and exception while handling a message. * * @param fromClient * @param toClient * @param msg * @param th */ protected void exception(String method,ServerSession fromClient, LocalSession toClient, ServerMessage msg, Throwable th) { System.err.println(method+": "+msg); th.printStackTrace(); } /* ------------------------------------------------------------ */ private void invoke(final Method method, final ServerSession fromClient, final ServerMessage msg) { if (_logger.isDebugEnabled()) _logger.debug("invoke "+_name+"#"+method.getName()+" from "+fromClient+" with "+msg.getData()); if (_threadPool == null) doInvoke(method,fromClient,msg); else { msg.incRef(); _threadPool.dispatch(new Runnable() { public void run() { try { doInvoke(method,fromClient,msg); } finally { msg.decRef(); } } }); } } /* ------------------------------------------------------------ */ private void doInvoke(Method method, ServerSession fromClient, ServerMessage msg) { String channel=msg.getChannel(); Object data=msg.getData(); Object id=msg.getId(); if (method != null) { try { Class[] args=method.getParameterTypes(); Object arg=Message.class.isAssignableFrom(args[1])?msg:data; Object reply=null; switch(method.getParameterTypes().length) { case 2: reply=method.invoke(this,fromClient,arg); break; case 3: if (id!=null && String.class.equals(method.getParameterTypes()[2])) id=id.toString(); reply=method.invoke(this,fromClient,arg,id); break; case 4: if (id!=null && String.class.equals(method.getParameterTypes()[3])) id=id.toString(); reply=method.invoke(this,fromClient,channel,arg,id); break; } 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); } } } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy