org.vootoo.client.netty.connect.SimpleConnectionPool.java Source code

Java tutorial

Introduction

Here is the source code for org.vootoo.client.netty.connect.SimpleConnectionPool.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.vootoo.client.netty.connect;

import io.netty.bootstrap.Bootstrap;
import io.netty.channel.Channel;
import io.netty.channel.ChannelFuture;
import io.netty.channel.ChannelOption;
import io.netty.channel.local.LocalAddress;
import io.netty.util.concurrent.Future;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.vootoo.client.netty.SolrClientChannelPoolHandler;
import org.vootoo.client.netty.HandlerConfig;

import java.net.InetSocketAddress;
import java.net.SocketAddress;

/**
 * @author chenlb on 2015-06-12 09:55.
 */
public class SimpleConnectionPool extends io.netty.channel.pool.SimpleChannelPool implements ConnectionPool {
    private static final Logger logger = LoggerFactory.getLogger(SimpleConnectionPool.class);

    protected static final int DEFAULT_CONNECT_TIMEOUT = 2000;

    protected final SocketAddress socketAddress;
    protected final String host;
    protected final int port;

    protected int connectTimeout; //default 2s
    protected ConnectionPoolContext poolContext;

    /**
     * auto use {@link SolrClientChannelPoolHandler} for solr netty channel handler {@link org.vootoo.client.netty.SolrClientHandler}
     */
    public SimpleConnectionPool(Bootstrap bootstrap, SocketAddress remoteAddress) {
        this(bootstrap, HandlerConfig.DEFAULT_CONFIG, remoteAddress, DEFAULT_CONNECT_TIMEOUT);
    }

    public SimpleConnectionPool(Bootstrap bootstrap, HandlerConfig handlerConfig, SocketAddress remoteAddress) {
        this(bootstrap, handlerConfig, remoteAddress, DEFAULT_CONNECT_TIMEOUT);
    }

    public SimpleConnectionPool(Bootstrap bootstrap, HandlerConfig handlerConfig, SocketAddress remoteAddress,
            int connectTimeout) {
        super(bootstrap, new SolrClientChannelPoolHandler(handlerConfig, remoteAddress));
        this.connectTimeout = connectTimeout;
        this.socketAddress = remoteAddress;
        if (remoteAddress instanceof InetSocketAddress) {
            InetSocketAddress inetSocketAddress = (InetSocketAddress) remoteAddress;
            host = inetSocketAddress.getAddress().getHostAddress();
            port = inetSocketAddress.getPort();
        } else if (remoteAddress instanceof LocalAddress) {
            LocalAddress localAddress = (LocalAddress) remoteAddress;
            int myPort = -1;
            try {
                myPort = Integer.parseInt(localAddress.id());
            } catch (NumberFormatException e) {
            }

            host = "local";
            port = myPort;
        } else {
            throw new IllegalArgumentException("SocketAddress must be '" + InetSocketAddress.class.getName()
                    + "' or '" + LocalAddress.class.getName() + "' (sub) class");
        }

        poolContext = new ConnectionPoolContext(handlerConfig.getResponsePromiseContainer());
    }

    @Override
    protected ChannelFuture connectChannel(Bootstrap bs) {
        bs.option(ChannelOption.CONNECT_TIMEOUT_MILLIS, connectTimeout);
        return bs.connect(socketAddress);
    }

    public Channel acquireConnect() throws NettyConnectLessException {
        Future<Channel> future = acquire();

        // see https://netty.io/4.0/api/io/netty/channel/ChannelFuture.html
        // use bootstrap.option(ChannelOption.CONNECT_TIMEOUT_MILLIS, connectTimeout);
        // so await without timeout
        future.awaitUninterruptibly();

        assert future.isDone();

        if (future.isCancelled()) {
            // Connection attempt cancelled by user
            throw new NettyConnectLessException("connection cancelled tcp=" + socketAddress);
        } else if (!future.isSuccess()) {
            throw new NettyConnectLessException(
                    "connect tcp=" + socketAddress + " fail within " + connectTimeout + "ms time!", future.cause());
        } else {
            // Connection established successfully
            Channel channel = future.getNow();

            if (logger.isDebugEnabled()) {
                logger.debug("acquire connect success channel={}", channel);
            }

            assert channel != null;

            if (channel == null) {
                throw new NettyConnectLessException("connect tcp=" + socketAddress + " fail within "
                        + connectTimeout + "ms time, future.getNow return null!");
            }

            return channel;
        }
    }

    @Override
    public void releaseConnect(Channel channel) {
        Future<Void> future = release(channel);

        //TODO  wait future done ?
    }

    @Override
    public ConnectionPoolContext poolContext() {
        return poolContext;
    }

    @Override
    public String channelHost() {
        return host;
    }

    @Override
    public int channelPort() {
        return port;
    }

    public void setConnectTimeout(int connectTimeout) {
        if (connectTimeout < 10) {
            throw new IllegalArgumentException("connectTimeout can't < 10ms");
        }
        this.connectTimeout = connectTimeout;
    }

}