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

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

There is a newer version: 8.0.6
Show newest version
// ========================================================================
// Copyright 2006 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.util.ArrayList;
import java.util.List;
import java.util.Queue;

import org.cometd.Bayeux;
import org.cometd.Channel;
import org.cometd.Client;
import org.cometd.ClientListener;
import org.cometd.DeliverListener;
import org.cometd.Extension;
import org.cometd.Message;
import org.cometd.MessageListener;
import org.cometd.QueueListener;
import org.cometd.RemoveListener;
import org.eclipse.jetty.util.ArrayQueue;
import org.eclipse.jetty.util.LazyList;
import org.eclipse.jetty.util.ajax.JSON;
import org.eclipse.jetty.util.log.Log;

/* ------------------------------------------------------------ */
/**
 *
 * @author gregw
 */
public class ClientImpl implements Client
{
    private String _id;
    private String _type;
    private int _responsesPending;
    private ChannelImpl[] _subscriptions=new ChannelImpl[0]; // copy on write
    private RemoveListener[] _rListeners; // copy on write
    private MessageListener[] _syncMListeners; // copy on write
    private MessageListener[] _asyncMListeners; // copy on write
    private QueueListener[] _qListeners; // copy on write
    private DeliverListener[] _dListeners; // copy on write
    protected AbstractBayeux _bayeux;
    private String _browserId;
    private JSON.Literal _advice;
    private int _batch;
    private int _maxQueue;
    private ArrayQueue _queue=new ArrayQueue(8,16,this);
    private long _timeout;
    private long _interval;
    private int _lag;
    private Extension[] _extensions;

    private boolean _deliverViaMetaConnectOnly;
    private volatile boolean _isExpired;

    // manipulated and synchronized by AbstractBayeux
    int _adviseVersion;

    /* ------------------------------------------------------------ */
    protected ClientImpl(AbstractBayeux bayeux)
    {
        _bayeux=bayeux;
        _maxQueue=bayeux.getMaxClientQueue();
        _bayeux.addClient(this,null);
        if (_bayeux.isLogInfo())
            _bayeux.logInfo("newClient: " + this);
    }

    /* ------------------------------------------------------------ */
    protected ClientImpl(AbstractBayeux bayeux, String idPrefix)
    {
        _bayeux=bayeux;
        _maxQueue=0;

        _bayeux.addClient(this,idPrefix);

        if (_bayeux.isLogInfo())
            _bayeux.logInfo("newClient: " + this);
    }

    /* ------------------------------------------------------------ */
    public void addExtension(Extension ext)
    {
        _extensions=(Extension[])LazyList.addToArray(_extensions,ext,Extension.class);
    }

    public void removeExtension(Extension ext)
    {
        _extensions=(Extension[])LazyList.removeFromArray(_extensions,ext);
    }

    /* ------------------------------------------------------------ */
    Extension[] getExtensions()
    {
        return _extensions;
    }

    /* ------------------------------------------------------------ */
    public void deliver(Client from, String toChannel, Object data, String id)
    {
        MessageImpl message=_bayeux.newMessage();
        message.put(Bayeux.CHANNEL_FIELD,toChannel);
        message.put(Bayeux.DATA_FIELD,data);
        if (id != null)
            message.put(Bayeux.ID_FIELD,id);

        Message m=_bayeux.extendSendBayeux(from,message);
        if (m != null)
            doDelivery(from,m);
        if (m instanceof MessageImpl)
            ((MessageImpl)m).decRef();
    }

    /* ------------------------------------------------------------ */
    public void deliverLazy(Client from, String toChannel, Object data, String id)
    {
        MessageImpl message=_bayeux.newMessage();
        message.put(Bayeux.CHANNEL_FIELD,toChannel);
        message.put(Bayeux.DATA_FIELD,data);
        if (id != null)
            message.put(Bayeux.ID_FIELD,id);
        message.setLazy(true);
        Message m=_bayeux.extendSendBayeux(from,message);
        if (m != null)
            doDelivery(from,m);
        if (m instanceof MessageImpl)
            ((MessageImpl)m).decRef();
    }

    /* ------------------------------------------------------------ */
    protected void doDelivery(Client from, final Message msg)
    {
        final Message message=_bayeux.extendSendClient(from,this,msg);
        if (message == null)
            return;

        MessageListener[] alisteners=null;
        synchronized(this)
        {
            if (_maxQueue < 0)
            {
                // No queue limit, so always queue the message
                ((MessageImpl)message).incRef();
                _queue.addUnsafe(message);
            }
            else
            {
                // We have a queue limit,
                boolean queue;
                if (_queue.size() >= _maxQueue)
                {
                    // We are over the limit, so consult listeners
                    if (_qListeners != null && _qListeners.length > 0)
                    {
                        queue=true;
                        for (QueueListener l : _qListeners)
                            queue &= notifyQueueListener(l, from, message);
                    }
                    else
                        queue=false;
                }
                else
                    // we are under limit, so queue the messages.
                    queue=true;

                // queue the message if we are meant to
                if (queue)
                {
                    ((MessageImpl)message).incRef();
                    _queue.addUnsafe(message);
                }
            }

            // deliver synchronized
            if (_syncMListeners != null)
                for (MessageListener l : _syncMListeners)
                    notifyMessageListener(l, from, message);
            alisteners=_asyncMListeners;

            if (_batch == 0 && _responsesPending < 1 && _queue.size() > 0)
            {
                if (((MessageImpl)message).isLazy())
                    lazyResume();
                else
                    resume();
            }
        }

        // deliver unsynchronized
        if (alisteners != null)
            for (MessageListener l : alisteners)
                notifyMessageListener(l, from, message);
    }

    private boolean notifyQueueListener(QueueListener listener, Client from, Message message)
    {
        try
        {
            return listener.queueMaxed(from, this, message);
        }
        catch (Throwable x)
        {
            Log.debug(x);
            return false;
        }
    }

    private void notifyMessageListener(MessageListener listener, Client from, Message message)
    {
        try
        {
            listener.deliver(from, this, message);
        }
        catch (Throwable x)
        {
            Log.debug(x);
        }
    }

    /* ------------------------------------------------------------ */
    public void doDeliverListeners()
    {
        synchronized(this)
        {
            if (_dListeners != null)
                for (DeliverListener l : _dListeners)
                    notifyDeliverListener(l, _queue);
        }
    }

    private void notifyDeliverListener(DeliverListener listener, Queue queue)
    {
        try
        {
            listener.deliver(this, queue);
        }
        catch (Throwable x)
        {
            Log.debug(x);
        }
    }

    /* ------------------------------------------------------------ */
    public void setMetaConnectDeliveryOnly(boolean deliverViaMetaConnectOnly)
    {
        _deliverViaMetaConnectOnly=deliverViaMetaConnectOnly;
    }

    /* ------------------------------------------------------------ */
    public boolean isMetaConnectDeliveryOnly()
    {
        return _deliverViaMetaConnectOnly;
    }

    /* ------------------------------------------------------------ */
    public void startBatch()
    {
        synchronized(this)
        {
            _batch++;
        }
    }

    /* ------------------------------------------------------------ */
    public void endBatch()
    {
        synchronized(this)
        {
            if (--_batch == 0 && _responsesPending < 1)
            {
                batch:switch(_queue.size())
                {
                    case 0:
                        break;
                    case 1:
                        if (((MessageImpl)_queue.get(0)).isLazy())
                            lazyResume();
                        else
                            resume();
                        break;
                    default:
                        for (int i=_queue.size();i-->0;)
                        {
                            if (!((MessageImpl)_queue.get(i)).isLazy())
                            {
                                resume();
                                break batch;
                            }
                        }
                        lazyResume();
                }
            }
        }
    }

    /* ------------------------------------------------------------ */
    public String getConnectionType()
    {
        return _type;
    }

    /* ------------------------------------------------------------ */
    /*
     * (non-Javadoc)
     *
     * @see org.cometd.server.C#getId()
     */
    public String getId()
    {
        return _id;
    }

    /* ------------------------------------------------------------ */
    public boolean hasMessages()
    {
        return _queue.size() > 0;
    }

    /* ------------------------------------------------------------ */
    public boolean hasNonLazyMessages()
    {
        synchronized(this)
        {
            for (int i=_queue.size(); i-- > 0;)
            {
                if (!((MessageImpl)_queue.getUnsafe(i)).isLazy())
                    return true;
            }
        }
        return false;
    }

    /* ------------------------------------------------------------ */
    public boolean isLocal()
    {
        return true;
    }

    /* ------------------------------------------------------------ */
    /*
     * (non-Javadoc)
     *
     * @see org.cometd.Client#disconnect()
     */
    public void disconnect()
    {
        synchronized(this)
        {
            if (_bayeux.hasClient(_id))
                remove(false);
        }
    }

    /* ------------------------------------------------------------ */
    /*
     * (non-Javadoc)
     *
     * @see dojox.cometd.Client#remove(boolean)
     */
    public void remove(boolean timeout)
    {
        _isExpired=timeout;
        Client client=_bayeux.removeClient(_id);

        if (client != null && _bayeux.isLogInfo())
            _bayeux.logInfo("Remove client " + client + " timeout=" + timeout);

        final String browser_id;
        final RemoveListener[] listeners;
        synchronized(this)
        {
            browser_id=_browserId;
            _browserId=null;
            listeners=_rListeners;
        }

        if (browser_id != null)
            _bayeux.clientOffBrowser(browser_id,_id);
        if (listeners != null)
            for (RemoveListener l : listeners)
                notifyRemoveListener(l, _id, timeout);

        resume();
    }

    private void notifyRemoveListener(RemoveListener listener, String clientId, boolean timeout)
    {
        try
        {
            listener.removed(clientId, timeout);
        }
        catch (Throwable x)
        {
            Log.debug(x);
        }
    }

    /* ------------------------------------------------------------ */
    public boolean isExpired()
    {
        return _isExpired;
    }

    /* ------------------------------------------------------------ */
    public int responded()
    {
        synchronized(this)
        {
            return _responsesPending--;
        }
    }

    /* ------------------------------------------------------------ */
    public int responsePending()
    {
        synchronized(this)
        {
            return ++_responsesPending;
        }
    }

    /* ------------------------------------------------------------ */
    /**
     * Called by deliver to resume anything waiting on this client lazily
     */
    public void lazyResume()
    {
    }

    /* ------------------------------------------------------------ */
    /**
     * Called by deliver to resume anything waiting on this client.
     */
    public void resume()
    {
    }

    /* ------------------------------------------------------------ */
    /*
     * @return the number of messages queued
     */
    public int getMessages()
    {
        return _queue.size();
    }

    /* ------------------------------------------------------------ */
    public List takeMessages()
    {
        synchronized(this)
        {
            ArrayList list=new ArrayList(_queue);
            _queue.clear();
            return list;
        }
    }

    /* ------------------------------------------------------------ */
    public void returnMessages(List messages)
    {
        synchronized(this)
        {
            _queue.addAll(0,messages);
        }
    }

    /* ------------------------------------------------------------ */
    @Override
    public String toString()
    {
        return _id;
    }

    /* ------------------------------------------------------------ */
    protected void addSubscription(ChannelImpl channel)
    {
        synchronized(this)
        {
            _subscriptions=(ChannelImpl[])LazyList.addToArray(_subscriptions,channel,null);
        }
    }

    /* ------------------------------------------------------------ */
    protected void removeSubscription(ChannelImpl channel)
    {
        synchronized(this)
        {
            _subscriptions=(ChannelImpl[])LazyList.removeFromArray(_subscriptions,channel);
        }
    }

    /* ------------------------------------------------------------ */
    protected void setConnectionType(String type)
    {
        synchronized(this)
        {
            _type=type;
        }
    }

    /* ------------------------------------------------------------ */
    protected void setId(String id)
    {
        synchronized(this)
        {
            _id=id;
        }
    }

    /* ------------------------------------------------------------ */
    public void unsubscribeAll()
    {
        ChannelImpl[] subscriptions;
        synchronized(this)
        {
            subscriptions=_subscriptions;
            _subscriptions=new ChannelImpl[0];
        }
        for (ChannelImpl channel : subscriptions)
            channel.unsubscribe(this);

    }

    /* ------------------------------------------------------------ */
    public void setBrowserId(String id)
    {
        if (_browserId != null && !_browserId.equals(id))
            _bayeux.clientOffBrowser(_browserId,_id);
        _browserId=id;
        if (_browserId != null)
            _bayeux.clientOnBrowser(_browserId,_id);
    }

    /* ------------------------------------------------------------ */
    public String getBrowserId()
    {
        return _browserId;
    }

    /* ------------------------------------------------------------ */
    @Override
    public boolean equals(Object o)
    {
        if (!(o instanceof Client))
            return false;
        return getId().equals(((Client)o).getId());
    }

    /* ------------------------------------------------------------ */
    /**
     * Get the advice specific for this Client
     *
     * @return advice specific for this client or null
     */
    public JSON.Literal getAdvice()
    {
        return _advice;
    }

    /* ------------------------------------------------------------ */
    /**
     * @param advice
     *            specific for this client
     */
    public void setAdvice(JSON.Literal advice)
    {
        _advice=advice;
    }

    /* ------------------------------------------------------------ */
    public void addListener(ClientListener listener)
    {
        synchronized(this)
        {
            if (listener instanceof MessageListener)
            {
                if (listener instanceof MessageListener.Synchronous)
                    _syncMListeners=(MessageListener[])LazyList.addToArray(_syncMListeners,listener,MessageListener.class);
                else
                    _asyncMListeners=(MessageListener[])LazyList.addToArray(_asyncMListeners,listener,MessageListener.class);
            }

            if (listener instanceof RemoveListener)
                _rListeners=(RemoveListener[])LazyList.addToArray(_rListeners,listener,RemoveListener.class);

            if (listener instanceof QueueListener)
                _qListeners=(QueueListener[])LazyList.addToArray(_qListeners,listener,QueueListener.class);

            if (listener instanceof DeliverListener)
                _dListeners=(DeliverListener[])LazyList.addToArray(_dListeners,listener,DeliverListener.class);
        }
    }

    /* ------------------------------------------------------------ */
    public void removeListener(ClientListener listener)
    {
        synchronized(this)
        {
            if (listener instanceof MessageListener)
            {
                _syncMListeners=(MessageListener[])LazyList.removeFromArray(_syncMListeners,listener);
                _asyncMListeners=(MessageListener[])LazyList.removeFromArray(_asyncMListeners,listener);
            }

            if (listener instanceof RemoveListener)
                _rListeners=(RemoveListener[])LazyList.removeFromArray(_rListeners,listener);

            if (listener instanceof QueueListener)
                _qListeners=(QueueListener[])LazyList.removeFromArray(_qListeners,listener);
        }
    }

    /* ------------------------------------------------------------ */
    public long getInterval()
    {
        return _interval;
    }

    /* ------------------------------------------------------------ */
    /**
     * Set per client interval
     *
     * @param intervalMS
     *            timeout in MS for longpoll duration or 0 to use default from
     *            {@link AbstractBayeux#getMaxInterval()}.
     */
    public void setInterval(long intervalMS)
    {
        _interval=intervalMS;
    }

    /* ------------------------------------------------------------ */
    public long getTimeout()
    {
        return _timeout;
    }

    /* ------------------------------------------------------------ */
    /**
     * Set per client timeout
     *
     * @param timeoutMS
     *            timeout in MS for longpoll duration or 0 to use default from
     *            {@link AbstractBayeux#getTimeout()}.
     */
    public void setTimeout(long timeoutMS)
    {
        _timeout=timeoutMS;
    }

    /* ------------------------------------------------------------ */
    public void setMaxQueue(int maxQueue)
    {
        _maxQueue=maxQueue;
    }

    /* ------------------------------------------------------------ */
    public int getMaxQueue()
    {
        return _maxQueue;
    }

    /* ------------------------------------------------------------ */
    public Queue getQueue()
    {
        return _queue;
    }

    /* ------------------------------------------------------------ */
    /**
     * @see org.cometd.server.ext.TimesyncExtension
     * @return The lag in ms as measured by an extension like the
     *         TimesyncExtension
     */
    public int getLag()
    {
        return _lag;
    }

    /* ------------------------------------------------------------ */
    /**
     * @see org.cometd.server.ext.TimesyncExtension
     * @param lag
     *            in ms
     */
    public void setLag(int lag)
    {
        _lag=lag;
    }

    /* ------------------------------------------------------------ */
    /**
     * Get the subscribed to channels
     *
     * @return A copied array of the channels to which this client is subscribed
     */
    public Channel[] getSubscriptions()
    {
        ChannelImpl[] subscriptions=_subscriptions;
        if (subscriptions == null)
            return null;
        Channel[] channels=new Channel[subscriptions.length];
        System.arraycopy(subscriptions,0,channels,0,subscriptions.length);
        return channels;
    }

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy