com.github.sinsinpub.pero.backend.ConnectBackendHandler.java Source code

Java tutorial

Introduction

Here is the source code for com.github.sinsinpub.pero.backend.ConnectBackendHandler.java

Source

/*
 * Copyright 2012 The Netty Project
 *
 * The Netty Project 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 com.github.sinsinpub.pero.backend;

import io.netty.bootstrap.Bootstrap;
import io.netty.channel.Channel;
import io.netty.channel.ChannelFuture;
import io.netty.channel.ChannelFutureListener;
import io.netty.channel.ChannelHandler;
import io.netty.channel.ChannelHandlerContext;
import io.netty.channel.ChannelOption;
import io.netty.channel.SimpleChannelInboundHandler;
import io.netty.channel.socket.nio.NioSocketChannel;
import io.netty.handler.codec.socks.SocksCmdRequest;
import io.netty.handler.codec.socks.SocksCmdResponse;
import io.netty.handler.codec.socks.SocksCmdStatus;
import io.netty.handler.proxy.ProxyHandler;
import io.netty.handler.proxy.Socks5ProxyHandler;
import io.netty.util.concurrent.Future;
import io.netty.util.concurrent.GenericFutureListener;
import io.netty.util.concurrent.Promise;

import java.net.InetSocketAddress;
import java.util.concurrent.TimeUnit;
import java.util.concurrent.atomic.AtomicBoolean;
import java.util.concurrent.atomic.AtomicLong;

import javax.inject.Singleton;

import jodd.petite.meta.PetiteBean;
import jodd.util.StringUtil;

import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.stereotype.Component;

import com.github.sinsinpub.pero.config.AppProps;
import com.github.sinsinpub.pero.utils.NettyChannelUtils;

@PetiteBean("connectBackendHandler")
@Component("connectBackendHandler")
@Singleton
@ChannelHandler.Sharable
public final class ConnectBackendHandler extends SimpleChannelInboundHandler<SocksCmdRequest> {

    private static final Logger logger = LoggerFactory.getLogger(ConnectBackendHandler.class);
    static final AtomicLong nekoKilled = new AtomicLong();
    private int connectTimeoutMillis = AppProps.PROPS.getInteger("upstream.socks5.connectTimeoutMillis", 10000);

    public ConnectBackendHandler() {
    }

    @Override
    public void channelRead0(final ChannelHandlerContext ctx, final SocksCmdRequest request) throws Exception {
        final Channel inboundChannel = ctx.channel();
        final Promise<Channel> outboundPromise = newOutboundPromise(ctx, request);
        final int maxProxies = AppProps.PROPS.getInteger("upstream.socks5.count", 1);
        final AtomicBoolean isFinalSuccess = new AtomicBoolean(false);
        final AtomicBoolean isRetryOccured = new AtomicBoolean(false);
        for (int i = 1; i <= maxProxies; i++) {
            final boolean isFirstOne = (i == 1);
            final boolean isFinalOne = (i == maxProxies);
            if (!isFirstOne) {
                isRetryOccured.set(true);
            }
            try {
                final ProxyHandler upstreamProxyHandler = newUpstreamProxyHandler(i);
                final Bootstrap b = newConnectionBootstrap(inboundChannel, outboundPromise, upstreamProxyHandler);
                logger.info(String.format("Connecting backend %s:%s via %s", request.host(), request.port(),
                        upstreamProxyHandler != null ? upstreamProxyHandler.proxyAddress() : "direct"));
                ChannelFuture connFuture = b.connect(request.host(), request.port());
                connFuture.addListener(new ChannelFutureListener() {
                    @Override
                    public void operationComplete(ChannelFuture future) throws Exception {
                        if (future.isSuccess()) {
                            // Connection established use handler provided results
                            if (upstreamProxyHandler == null) {
                                logger.info("Backend connected directly");
                            } else {
                                logger.info("Backend connected via: " + upstreamProxyHandler.proxyAddress());
                            }
                        } else {
                            // Close the connection if the connection attempt has failed.
                            if (isFinalOne) {
                                logger.error("Backend connection failed: " + future.cause(), future.cause());
                                ctx.channel().writeAndFlush(
                                        new SocksCmdResponse(SocksCmdStatus.FAILURE, request.addressType()));
                                NettyChannelUtils.closeOnFlush(ctx.channel());
                            } else {
                                logger.warn("Backend connection failed: {}", future.cause().toString());
                            }
                        }
                    }
                });
                connFuture.await(getConnectTimeoutMillis(), TimeUnit.MILLISECONDS);
                if (connFuture.isSuccess()) {
                    isFinalSuccess.set(true);
                    break;
                }
            } catch (Exception e) {
                logger.error("Connecting exception {} caught:", e);
            }
        }
        if (isFinalSuccess.get() && isRetryOccured.get()) {
            logger.warn("Protected from Error-Neko: {} times", nekoKilled.incrementAndGet());
        }
    }

    /**
     * Create new promised callback on outbound channel operation complete.
     * 
     * @param ctx
     * @param request
     * @return Promise
     */
    protected Promise<Channel> newOutboundPromise(final ChannelHandlerContext ctx, final SocksCmdRequest request) {
        final Promise<Channel> promise = ctx.executor().newPromise();
        promise.addListener(new GenericFutureListener<Future<Channel>>() {
            @Override
            public void operationComplete(final Future<Channel> future) throws Exception {
                final Channel outboundChannel = future.getNow();
                if (future.isSuccess()) {
                    ctx.channel().writeAndFlush(new SocksCmdResponse(SocksCmdStatus.SUCCESS, request.addressType()))
                            .addListener(new ChannelFutureListener() {
                                @Override
                                public void operationComplete(ChannelFuture channelFuture) {
                                    ctx.pipeline().remove(ConnectBackendHandler.this);
                                    outboundChannel.pipeline().addLast(new RelayTrafficHandler(ctx.channel()));
                                    ctx.pipeline().addLast(new RelayTrafficHandler(outboundChannel));
                                }
                            });
                } else {
                    ctx.channel()
                            .writeAndFlush(new SocksCmdResponse(SocksCmdStatus.FAILURE, request.addressType()));
                    NettyChannelUtils.closeOnFlush(ctx.channel());
                }
            }
        });
        return promise;
    }

    /**
     * Create new instance of proxy handler as it is not Sharable.
     * 
     * @return Socks5ProxyHandler
     */
    protected ProxyHandler newUpstreamProxyHandler(int index) {
        final ProxyHandler upstreamProxyHandler;
        final StringBuilder attrPrefix = new StringBuilder("upstream.socks5.");
        if (index > 0) {
            attrPrefix.append(String.valueOf(index)).append(".");
        }
        String host = AppProps.PROPS.getValue(attrPrefix.toString().concat("host"));
        if (StringUtil.isEmpty(host)) {
            upstreamProxyHandler = null;
        } else {
            int port = AppProps.PROPS.getInteger(attrPrefix.toString().concat("port"), 1080);
            upstreamProxyHandler = new Socks5ProxyHandler(new InetSocketAddress(host, port));
            upstreamProxyHandler.setConnectTimeoutMillis(getConnectTimeoutMillis());
        }
        return upstreamProxyHandler;
    }

    /**
     * Create new connection context to connect real back-end (may be another proxy).
     * 
     * @param inboundChannel
     * @param outboundPromise
     * @param upstreamProxyHandler
     * @return Bootstrap
     */
    protected Bootstrap newConnectionBootstrap(final Channel inboundChannel, final Promise<Channel> outboundPromise,
            final ProxyHandler upstreamProxyHandler) {
        final Bootstrap b = new Bootstrap();
        b.group(inboundChannel.eventLoop()).channel(NioSocketChannel.class)
                .option(ChannelOption.CONNECT_TIMEOUT_MILLIS, getConnectTimeoutMillis())
                .option(ChannelOption.SO_KEEPALIVE, true)
                .handler(new ProxyChannelInitializer(outboundPromise, upstreamProxyHandler));
        return b;
    }

    @Override
    public void exceptionCaught(ChannelHandlerContext ctx, Throwable cause) throws Exception {
        // NettyChannelUtils.closeOnFlush(ctx.channel());
        logger.error("Unexpected exception from backend proxy:", cause);
    }

    public int getConnectTimeoutMillis() {
        return connectTimeoutMillis;
    }

    public void setConnectTimeoutMillis(int connectTimeoutMillis) {
        this.connectTimeoutMillis = connectTimeoutMillis;
    }

}