acromusashi.stream.component.rabbitmq.CachingConnectionFactory.java Source code

Java tutorial

Introduction

Here is the source code for acromusashi.stream.component.rabbitmq.CachingConnectionFactory.java

Source

// CHECKSTYLE:OFF
/*
 * Copyright 2002-2012 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 acromusashi.stream.component.rabbitmq;

import java.io.IOException;
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;
import java.util.LinkedList;
import java.util.List;

import org.springframework.amqp.AmqpException;
import org.springframework.amqp.rabbit.connection.AbstractConnectionFactory;
import org.springframework.amqp.rabbit.connection.ChannelProxy;
import org.springframework.amqp.rabbit.connection.Connection;
import org.springframework.amqp.rabbit.connection.ConnectionListener;
import org.springframework.amqp.rabbit.connection.ConnectionProxy;
import org.springframework.amqp.rabbit.connection.RabbitUtils;
import org.springframework.amqp.rabbit.support.PublisherCallbackChannel;
import org.springframework.amqp.rabbit.support.PublisherCallbackChannelImpl;
import org.springframework.util.Assert;
import org.springframework.util.StringUtils;

import com.rabbitmq.client.Address;
import com.rabbitmq.client.Channel;

/**
 * A ConnectionFactory implementation that returns the same Connections from all {@link #createConnection()}
 * calls, and ignores calls to {@link com.rabbitmq.client.Connection#close()} and caches
 * {@link com.rabbitmq.client.Channel}.
 *
 * <p>
 * By default, only one Channel will be cached, with further requested Channels being created and disposed on demand.
 * Consider raising the {@link #setChannelCacheSize(int) "channelCacheSize" value} in case of a high-concurrency
 * environment.
 *
 * <p>
 * <b>NOTE: This ConnectionFactory requires explicit closing of all Channels obtained form its shared Connection.</b>
 * This is the usual recommendation for native Rabbit access code anyway. However, with this ConnectionFactory, its use
 * is mandatory in order to actually allow for Channel reuse.
 *
 * @author Mark Pollack
 * @author Mark Fisher
 * @author Dave Syer
 * @author Gary Russell
 * 
 * TODO Contribute to BaseRepository
 */
public class CachingConnectionFactory extends AbstractConnectionFactory {

    private int channelCacheSize = 1;

    private final LinkedList<ChannelProxy> cachedChannelsNonTransactional = new LinkedList<ChannelProxy>();

    private final LinkedList<ChannelProxy> cachedChannelsTransactional = new LinkedList<ChannelProxy>();

    private volatile boolean active = true;

    private ChannelCachingConnectionProxy connection;

    private volatile boolean publisherConfirms;

    private volatile boolean publisherReturns;

    private FailbackChecker failbackChecker = new FailbackChecker();

    /** Synchronization monitor for the shared Connection */
    private final Object connectionMonitor = new Object();

    /**
     * Create a new CachingConnectionFactory initializing the hostname to be the value returned from
     * InetAddress.getLocalHost(), or "localhost" if getLocalHost() throws an exception.
     */
    public CachingConnectionFactory() {
        this((String) null);
    }

    /**
     * Create a new CachingConnectionFactory given a host name and port.
     *
     * @param hostname the host name to connect to
     * @param port the port number
     */
    public CachingConnectionFactory(String hostname, int port) {
        super(new com.rabbitmq.client.ConnectionFactory());
        if (!StringUtils.hasText(hostname)) {
            hostname = getDefaultHostName();
        }
        setHost(hostname);
        setPort(port);
    }

    /**
     * Create a new CachingConnectionFactory given a port on the hostname returned from InetAddress.getLocalHost(), or
     * "localhost" if getLocalHost() throws an exception.
     *
     * @param port the port number
     */
    public CachingConnectionFactory(int port) {
        this(null, port);
    }

    /**
     * Create a new CachingConnectionFactory given a host name.
     *
     * @param hostname the host name to connect to
     */
    public CachingConnectionFactory(String hostname) {
        this(hostname, com.rabbitmq.client.ConnectionFactory.DEFAULT_AMQP_PORT);
    }

    /**
     * Create a new CachingConnectionFactory for the given target ConnectionFactory.
     *
     * @param rabbitConnectionFactory the target ConnectionFactory
     */
    public CachingConnectionFactory(com.rabbitmq.client.ConnectionFactory rabbitConnectionFactory) {
        super(rabbitConnectionFactory);
    }

    @Override
    public void setAddresses(String addresses) {
        super.setAddresses(addresses);
        this.failbackChecker.setAddresses(addresses);
    }

    public void setChannelCacheSize(int sessionCacheSize) {
        Assert.isTrue(sessionCacheSize >= 1, "Channel cache size must be 1 or higher");
        this.channelCacheSize = sessionCacheSize;
    }

    public int getChannelCacheSize() {
        return this.channelCacheSize;
    }

    public boolean isPublisherConfirms() {
        return this.publisherConfirms;
    }

    public boolean isPublisherReturns() {
        return this.publisherReturns;
    }

    public void setPublisherReturns(boolean publisherReturns) {
        this.publisherReturns = publisherReturns;
    }

    public void setPublisherConfirms(boolean publisherConfirms) {
        this.publisherConfirms = publisherConfirms;
    }

    @Override
    public void setConnectionListeners(List<? extends ConnectionListener> listeners) {
        super.setConnectionListeners(listeners);
        // If the connection is already alive we assume that the new listeners want to be notified
        if (this.connection != null) {
            getConnectionListener().onCreate(this.connection);
        }
    }

    @Override
    public void addConnectionListener(ConnectionListener listener) {
        super.addConnectionListener(listener);
        // If the connection is already alive we assume that the new listener wants to be notified
        if (this.connection != null) {
            listener.onCreate(this.connection);
        }
    }

    public void setFailbackInterval(long interval) {
        Assert.isTrue(interval >= 0, "Failback interbal must be 0 or higher");
        this.failbackChecker.setInterval(interval);
    }

    private boolean shouldFailback() {
        return this.failbackChecker.shuoldFailback();
    }

    private Channel getChannel(boolean transactional) {
        LinkedList<ChannelProxy> channelList = transactional ? this.cachedChannelsTransactional
                : this.cachedChannelsNonTransactional;
        Channel channel = null;
        synchronized (channelList) {
            if (!channelList.isEmpty()) {
                channel = channelList.removeFirst();
            }
        }
        if (channel != null) {
            if (this.logger.isTraceEnabled()) {
                this.logger.trace("Found cached Rabbit Channel");
            }
        } else {
            channel = getCachedChannelProxy(channelList, transactional);
        }
        return channel;
    }

    private ChannelProxy getCachedChannelProxy(LinkedList<ChannelProxy> channelList, boolean transactional) {
        Channel targetChannel = createBareChannel(transactional);

        if (this.logger.isDebugEnabled()) {
            this.logger.debug("Creating cached Rabbit Channel from " + targetChannel);
        }
        getChannelListener().onCreate(targetChannel, transactional);
        Class<?>[] interfaces;
        if (this.publisherConfirms || this.publisherReturns) {
            interfaces = new Class[] { ChannelProxy.class, PublisherCallbackChannel.class };
        } else {
            interfaces = new Class[] { ChannelProxy.class };
        }
        return (ChannelProxy) Proxy.newProxyInstance(ChannelProxy.class.getClassLoader(), interfaces,
                new CachedChannelInvocationHandler(targetChannel, channelList, transactional));
    }

    private synchronized Channel createBareChannel(boolean transactional) {
        // if attempt to fail back, close the caching connection ant channels
        // in this time, others thread may throw exception for the connection loss.
        boolean hasCachedConnection = this.connection != null;
        boolean shouldFailback = shouldFailback();
        if (hasCachedConnection && shouldFailback) {
            if (this.logger.isDebugEnabled()) {
                this.logger.debug("Closing cached Rabbit Connection to failback");
            }
            this.connection.destroy();
        }

        if (hasCachedConnection == false || this.connection.isOpen() == false) {
            // Use createConnection here not doCreateConnection so that the old one is properly disposed
            createConnection();
        }
        Channel channel = this.connection.createBareChannel(transactional);
        if (this.publisherConfirms) {
            try {
                channel.confirmSelect();
            } catch (IOException e) {
                this.logger.error("Could not configure the channel to receive publisher confirms", e);
            }
        }
        if (this.publisherConfirms || this.publisherReturns) {
            if (!(channel instanceof PublisherCallbackChannelImpl)) {
                channel = new PublisherCallbackChannelImpl(channel);
            }
        }
        return channel;
    }

    @Override
    public final Connection createConnection() throws AmqpException {
        synchronized (this.connectionMonitor) {
            if (this.connection == null || !this.connection.isOpen()) {
                Connection bareConnection = super.createBareConnection();
                this.connection = new ChannelCachingConnectionProxy(bareConnection);
                // invoke the listener *after* this.connection is assigned
                getConnectionListener().onCreate(this.connection);

                this.failbackChecker.reserveFailbackIfNessary(bareConnection);
            }
        }
        return this.connection;
    }

    /**
     * Close the underlying shared connection. The provider of this ConnectionFactory needs to care for proper shutdown.
     * <p>
     * As this bean implements DisposableBean, a bean factory will automatically invoke this on destruction of its
     * cached singletons.
     */
    @Override
    public final void destroy() {
        synchronized (this.connectionMonitor) {
            if (this.connection != null) {
                this.connection.destroy();
                this.connection = null;
            }
        }
        reset();
    }

    /**
     * Reset the Channel cache and underlying shared Connection, to be reinitialized on next access.
     */
    protected void reset() {
        this.active = false;
        synchronized (this.cachedChannelsNonTransactional) {
            for (ChannelProxy channel : this.cachedChannelsNonTransactional) {
                try {
                    channel.getTargetChannel().close();
                } catch (Throwable ex) {
                    this.logger.trace("Could not close cached Rabbit Channel", ex);
                }
            }
            this.cachedChannelsNonTransactional.clear();
        }
        synchronized (this.cachedChannelsTransactional) {
            for (ChannelProxy channel : this.cachedChannelsTransactional) {
                try {
                    channel.getTargetChannel().close();
                } catch (Throwable ex) {
                    this.logger.trace("Could not close cached Rabbit Channel", ex);
                }
            }
            this.cachedChannelsTransactional.clear();
        }
        this.active = true;
    }

    @Override
    public String toString() {
        return "CachingConnectionFactory [channelCacheSize=" + this.channelCacheSize + ", host=" + getHost()
                + ", port=" + getPort() + ", active=" + this.active + "]";
    }

    private class CachedChannelInvocationHandler implements InvocationHandler {

        private volatile Channel target;

        private final LinkedList<ChannelProxy> channelList;

        private final Object targetMonitor = new Object();

        private final boolean transactional;

        public CachedChannelInvocationHandler(Channel target, LinkedList<ChannelProxy> channelList,
                boolean transactional) {
            this.target = target;
            this.channelList = channelList;
            this.transactional = transactional;
        }

        @Override
        public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
            String methodName = method.getName();
            if (methodName.equals("txSelect") && !this.transactional) {
                throw new UnsupportedOperationException("Cannot start transaction on non-transactional channel");
            }
            if (methodName.equals("equals")) {
                // Only consider equal when proxies are identical.
                return (proxy == args[0]);
            } else if (methodName.equals("hashCode")) {
                // Use hashCode of Channel proxy.
                return System.identityHashCode(proxy);
            } else if (methodName.equals("toString")) {
                return "Cached Rabbit Channel: " + this.target;
            } else if (methodName.equals("close")) {
                // Handle close method: don't pass the call on.
                if (CachingConnectionFactory.this.active) {
                    synchronized (this.channelList) {
                        if (!RabbitUtils.isPhysicalCloseRequired()
                                && this.channelList.size() < getChannelCacheSize()) {
                            logicalClose((ChannelProxy) proxy);
                            // Remain open in the channel list.
                            return null;
                        }
                    }
                }

                // If we get here, we're supposed to shut down.
                physicalClose();
                return null;
            } else if (methodName.equals("getTargetChannel")) {
                // Handle getTargetChannel method: return underlying Channel.
                return this.target;
            } else if (methodName.equals("isOpen")) {
                // Handle isOpen method: we are closed if the target is closed
                return this.target != null && this.target.isOpen();
            }
            try {
                if (this.target == null || !this.target.isOpen()) {
                    this.target = null;
                }
                synchronized (this.targetMonitor) {
                    boolean shouldFailback = shouldFailback();
                    if (this.target == null || shouldFailback) {
                        this.target = createBareChannel(this.transactional);
                    }
                    return method.invoke(this.target, args);
                }
            } catch (InvocationTargetException ex) {
                if (this.target == null || !this.target.isOpen()) {
                    // Basic re-connection logic...
                    this.target = null;
                    CachingConnectionFactory.this.logger
                            .debug("Detected closed channel on exception.  Re-initializing: " + this.target);
                    synchronized (this.targetMonitor) {
                        if (this.target == null) {
                            this.target = createBareChannel(this.transactional);
                        }
                    }
                }
                throw ex.getTargetException();
            }
        }

        /**
         * GUARDED by channelList
         *
         * @param proxy the channel to close
         */
        private void logicalClose(ChannelProxy proxy) throws Exception {
            if (this.target != null && !this.target.isOpen()) {
                synchronized (this.targetMonitor) {
                    if (this.target != null && !this.target.isOpen()) {
                        this.target = null;
                        return;
                    }
                }
            }
            // Allow for multiple close calls...
            if (!this.channelList.contains(proxy)) {
                if (CachingConnectionFactory.this.logger.isTraceEnabled()) {
                    CachingConnectionFactory.this.logger.trace("Returning cached Channel: " + this.target);
                }
                this.channelList.addLast(proxy);
            }
        }

        private void physicalClose() throws Exception {
            if (CachingConnectionFactory.this.logger.isDebugEnabled()) {
                CachingConnectionFactory.this.logger.debug("Closing cached Channel: " + this.target);
            }
            if (this.target == null) {
                return;
            }
            if (this.target.isOpen()) {
                synchronized (this.targetMonitor) {
                    if (this.target.isOpen()) {
                        this.target.close();
                    }
                    this.target = null;
                }
            }
        }

    }

    private class ChannelCachingConnectionProxy implements Connection, ConnectionProxy {

        private volatile Connection target;

        public ChannelCachingConnectionProxy(Connection target) {
            this.target = target;
        }

        private Channel createBareChannel(boolean transactional) {
            return this.target.createChannel(transactional);
        }

        @Override
        public Channel createChannel(boolean transactional) {
            Channel channel = getChannel(transactional);
            return channel;
        }

        @Override
        public void close() {
        }

        public void destroy() {
            reset();
            if (this.target != null) {
                getConnectionListener().onClose(this.target);
                RabbitUtils.closeConnection(this.target);
            }
            this.target = null;
        }

        @Override
        public boolean isOpen() {
            return this.target != null && this.target.isOpen();
        }

        @Override
        public Connection getTargetConnection() {
            return this.target;
        }

        @Override
        public int hashCode() {
            return 31 + ((this.target == null) ? 0 : this.target.hashCode());
        }

        @Override
        public boolean equals(Object obj) {
            if (this == obj) {
                return true;
            }
            if (obj == null) {
                return false;
            }
            if (getClass() != obj.getClass()) {
                return false;
            }
            ChannelCachingConnectionProxy other = (ChannelCachingConnectionProxy) obj;
            if (this.target == null) {
                if (other.target != null) {
                    return false;
                }
            } else if (!this.target.equals(other.target)) {
                return false;
            }
            return true;
        }

        @Override
        public String toString() {
            return "Shared Rabbit Connection: " + this.target;
        }

    }

    private static class FailbackChecker {

        /** Defalut interval to failback is 10 seconds. */
        private static final long DEFAULT_INTERVAL = 10000L;

        private static final long NOT_NEED_TO_FAILBACK_TIME = 0L;

        private Address[] addresses;

        private long interval = DEFAULT_INTERVAL;

        private long nextTime = NOT_NEED_TO_FAILBACK_TIME;

        public FailbackChecker() {
        }

        public void setAddresses(String addresses) {
            Address[] addressArray = Address.parseAddresses(addresses);
            if (addressArray.length > 0) {
                this.addresses = addressArray;
            }
        }

        /**
         * Check the created connection whether or not it shows the first address of set addresses,
         * And if true, this class considers this connection should be failbacked and reserve to failback.
         * @param connection
         */
        public void reserveFailbackIfNessary(Connection connection) {
            if (this.interval == NOT_NEED_TO_FAILBACK_TIME) {
                return;
            }

            if (this.addresses != null && !isSameAddress(connection, this.addresses[0])) {
                this.nextTime = System.currentTimeMillis() + this.interval;
            } else {
                this.nextTime = NOT_NEED_TO_FAILBACK_TIME;
            }
        }

        private boolean isSameAddress(Connection connection, Address address) {
            String firstHost = address.getHost();
            int firstPort = address.getPort();

            Channel channel = connection.createChannel(false);
            String host = channel.getConnection().getAddress().getHostName();
            String ipAddress = channel.getConnection().getAddress().getHostAddress();
            int port = channel.getConnection().getPort();

            boolean isSameAddress = true;
            if (firstHost.equals(host) == false && firstHost.equals(ipAddress) == false) {
                isSameAddress = false;
            } else if (firstPort != port) {
                isSameAddress = false;
            }

            return isSameAddress;
        }

        /**
         * Tells whether or not the cached connection should be failbacked.
         * @return true if the cached connection should be failbacked.
         */
        public boolean shuoldFailback() {
            if (this.nextTime == NOT_NEED_TO_FAILBACK_TIME || this.interval == NOT_NEED_TO_FAILBACK_TIME) {
                return false;
            }

            long currentTime = System.currentTimeMillis();
            return this.nextTime <= currentTime;
        }

        /**
         * @param interval the interval to set
         */
        public void setInterval(long interval) {
            this.interval = interval;
        }
    }
}
//CHECKSTYLE:ON