Package org.apache.qpid.server.queue

Source Code of org.apache.qpid.server.queue.ClusteredQueue

/*
*
* Licensed to the Apache Software Foundation (ASF) under one
* or more contributor license agreements.  See the NOTICE file
* distributed with this work for additional information
* regarding copyright ownership.  The ASF licenses this file
* to you 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.apache.qpid.server.queue;

import org.apache.log4j.Logger;
import org.apache.qpid.AMQException;
import org.apache.qpid.framing.BasicCancelBody;
import org.apache.qpid.framing.QueueDeleteBody;
import org.apache.qpid.server.cluster.*;
import org.apache.qpid.server.cluster.util.LogMessage;
import org.apache.qpid.server.protocol.AMQProtocolSession;

import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.Executor;

/**
* Represents a shared queue in a cluster. The key difference is that as well as any
* local consumers, there may be consumers for this queue on other members of the
* cluster.
*
*/
public class ClusteredQueue extends AMQQueue
{
    private static final Logger _logger = Logger.getLogger(ClusteredQueue.class);
    private final ConcurrentHashMap<SimpleMemberHandle, RemoteSubscriptionImpl> _peers = new ConcurrentHashMap<SimpleMemberHandle, RemoteSubscriptionImpl>();
    private final GroupManager _groupMgr;
    private final NestedSubscriptionManager _subscriptions;

    public ClusteredQueue(GroupManager groupMgr, String name, boolean durable, String owner, boolean autoDelete, QueueRegistry queueRegistry)
            throws AMQException
    {
        super(name, durable, owner, autoDelete, queueRegistry, new ClusteredSubscriptionManager());
        _groupMgr = groupMgr;
        _subscriptions = ((ClusteredSubscriptionManager) getSubscribers()).getAllSubscribers();
    }

    public ClusteredQueue(GroupManager groupMgr, String name, boolean durable, String owner, boolean autoDelete, QueueRegistry queueRegistry, Executor asyncDelivery)
            throws AMQException
    {
        super(name, durable, owner, autoDelete, queueRegistry, asyncDelivery, new ClusteredSubscriptionManager(),
              new SubscriptionImpl.Factory());
        _groupMgr = groupMgr;
        _subscriptions = ((ClusteredSubscriptionManager) getSubscribers()).getAllSubscribers();
    }

    public void deliver(AMQMessage message) throws AMQException
    {
        _logger.info(new LogMessage("{0} delivered to clustered queue {1}", message, this));
        super.deliver(message);
    }

    protected void autodelete() throws AMQException
    {
        if(!_subscriptions.hasActiveSubscribers())
        {
            //delete locally:
            delete();

            //send deletion request to all other members:
            QueueDeleteBody request = new QueueDeleteBody();
            request.queue = getName();
            _groupMgr.broadcast(new SimpleSendable(request));
        }
    }

    public void unregisterProtocolSession(AMQProtocolSession ps, int channel, String consumerTag) throws AMQException
    {
        //handle locally:
        super.unregisterProtocolSession(ps, channel, consumerTag);

        //signal other members:
        BasicCancelBody request = new BasicCancelBody();
        request.consumerTag = getName();
        _groupMgr.broadcast(new SimpleSendable(request));
    }

    public void addRemoteSubcriber(MemberHandle peer)
    {
        _logger.info(new LogMessage("Added remote subscriber for {0} to clustered queue {1}", peer, this));
        //find (or create) a matching subscriber for the peer then increment the count
        getSubscriber(key(peer), true).increment();
    }

    public void removeRemoteSubscriber(MemberHandle peer)
    {
        //find a matching subscriber for the peer then decrement the count
        //if count is now zero, remove the subscriber
        SimpleMemberHandle key = key(peer);
        RemoteSubscriptionImpl s = getSubscriber(key, true);
        if (s == null)
        {
            throw new RuntimeException("No subscriber for " + peer);
        }
        if (s.decrement())
        {
            _peers.remove(key);
            _subscriptions.removeSubscription(s);
        }
    }

    public void removeAllRemoteSubscriber(MemberHandle peer)
    {
        SimpleMemberHandle key = key(peer);
        RemoteSubscriptionImpl s = getSubscriber(key, true);
        _peers.remove(key);
        _subscriptions.removeSubscription(s);
    }

    private RemoteSubscriptionImpl getSubscriber(SimpleMemberHandle key, boolean create)
    {
        RemoteSubscriptionImpl s = _peers.get(key);
        if (s == null && create)
        {
            return addSubscriber(key, new RemoteSubscriptionImpl(_groupMgr, key));
        }
        else
        {
            return s;
        }
    }

    private RemoteSubscriptionImpl addSubscriber(SimpleMemberHandle key, RemoteSubscriptionImpl s)
    {
        RemoteSubscriptionImpl other = _peers.putIfAbsent(key, s);
        if (other == null)
        {
            _subscriptions.addSubscription(s);
            new SubscriberCleanup(key, this, _groupMgr);
            return s;
        }
        else
        {
            return other;
        }
    }

    private SimpleMemberHandle key(MemberHandle peer)
    {
        return peer instanceof SimpleMemberHandle ? (SimpleMemberHandle) peer : (SimpleMemberHandle) SimpleMemberHandle.resolve(peer);
    }

    static boolean isFromBroker(AMQMessage msg)
    {
        return ClusteredProtocolSession.isPayloadFromPeer(msg);
    }
}
TOP

Related Classes of org.apache.qpid.server.queue.ClusteredQueue

TOP
Copyright © 2018 www.massapi.com. All rights reserved.
All source code are property of their respective owners. Java is a trademark of Sun Microsystems, Inc and owned by ORACLE Inc. Contact coftware#gmail.com.