org.apache.tajo.rpc.RpcConnectionPool.java Source code

Java tutorial

Introduction

Here is the source code for org.apache.tajo.rpc.RpcConnectionPool.java

Source

/**
 * 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.tajo.rpc;

import org.apache.commons.logging.Log;
import org.apache.commons.logging.LogFactory;
import io.netty.channel.ConnectTimeoutException;
import io.netty.util.internal.logging.CommonsLoggerFactory;
import io.netty.util.internal.logging.InternalLoggerFactory;

import java.net.InetSocketAddress;
import java.util.HashMap;
import java.util.Map;

public class RpcConnectionPool {
    private static final Log LOG = LogFactory.getLog(RpcConnectionPool.class);

    private Map<RpcConnectionKey, NettyClientBase> connections = new HashMap<RpcConnectionKey, NettyClientBase>();

    private static RpcConnectionPool instance;
    private final Object lockObject = new Object();

    public final static int RPC_RETRIES = 3;

    private RpcConnectionPool() {
    }

    public synchronized static RpcConnectionPool getPool() {
        if (instance == null) {
            InternalLoggerFactory.setDefaultFactory(new CommonsLoggerFactory());
            instance = new RpcConnectionPool();
        }
        return instance;
    }

    private NettyClientBase makeConnection(RpcConnectionKey rpcConnectionKey)
            throws NoSuchMethodException, ClassNotFoundException, ConnectTimeoutException {
        NettyClientBase client;
        if (rpcConnectionKey.asyncMode) {
            client = new AsyncRpcClient(rpcConnectionKey, RPC_RETRIES);
        } else {
            client = new BlockingRpcClient(rpcConnectionKey, RPC_RETRIES);
        }
        return client;
    }

    public static final long DEFAULT_TIMEOUT = 3000;
    public static final long DEFAULT_INTERVAL = 500;

    public NettyClientBase getConnection(InetSocketAddress addr, Class<?> protocolClass, boolean asyncMode)
            throws NoSuchMethodException, ClassNotFoundException, ConnectTimeoutException {
        return getConnection(addr, protocolClass, asyncMode, DEFAULT_TIMEOUT, DEFAULT_INTERVAL);
    }

    public NettyClientBase getConnection(InetSocketAddress addr, Class<?> protocolClass, boolean asyncMode,
            long timeout, long interval)
            throws NoSuchMethodException, ClassNotFoundException, ConnectTimeoutException {
        RpcConnectionKey key = new RpcConnectionKey(addr, protocolClass, asyncMode);

        RpcUtils.Timer timer = new RpcUtils.Timer(timeout);
        for (; !timer.isTimedOut(); timer.elapsed()) {
            NettyClientBase client;
            synchronized (lockObject) {
                client = connections.get(key);
                if (client == null) {
                    connections.put(key, client = makeConnection(key));
                }
            }
            if (client.acquire(timer.remaining())) {
                return client;
            }
            timer.interval(interval);
        }

        throw new ConnectTimeoutException("Failed to get connection for " + timeout + " msec");
    }

    public void releaseConnection(NettyClientBase client) {
        if (client != null) {
            release(client, false);
        }
    }

    public void closeConnection(NettyClientBase client) {
        if (client != null) {
            release(client, true);
        }
    }

    private void release(NettyClientBase client, boolean close) {
        try {
            if (returnToPool(client, close)) {
                if (LOG.isDebugEnabled()) {
                    LOG.debug("Closing connection [" + client.getKey() + "]");
                }
                client.close();
            }
            if (LOG.isDebugEnabled()) {
                LOG.debug("Current Connections in pool [" + connections.size() + "]");
            }
        } catch (Exception e) {
            LOG.error("Can't close connection:" + client.getKey() + ":" + e.getMessage(), e);
        }
    }

    // return true if the connection should be closed
    private boolean returnToPool(NettyClientBase client, boolean close) {
        synchronized (lockObject) {
            if (client.release() && (close || !client.isConnected())) {
                connections.remove(client.getKey());
                return true;
            }
        }
        return false;
    }

    public void close() {
        if (LOG.isDebugEnabled()) {
            LOG.debug("Pool Closed");
        }

        synchronized (lockObject) {
            for (NettyClientBase eachClient : connections.values()) {
                try {
                    eachClient.close();
                } catch (Exception e) {
                    LOG.error("close client pool error", e);
                }
            }
            connections.clear();
        }
    }

    public void shutdown() {
        close();
        RpcChannelFactory.shutdownGracefully();
    }

    static class RpcConnectionKey {
        final InetSocketAddress addr;
        final Class<?> protocolClass;
        final boolean asyncMode;

        final String description;

        public RpcConnectionKey(InetSocketAddress addr, Class<?> protocolClass, boolean asyncMode) {
            this.addr = addr;
            this.protocolClass = protocolClass;
            this.asyncMode = asyncMode;
            this.description = "[" + protocolClass + "] " + addr + "," + asyncMode;
        }

        @Override
        public String toString() {
            return description;
        }

        @Override
        public boolean equals(Object obj) {
            if (!(obj instanceof RpcConnectionKey)) {
                return false;
            }

            return toString().equals(obj.toString());
        }

        @Override
        public int hashCode() {
            return description.hashCode();
        }
    }
}