de.ocarthon.core.network.HttpClient.java Source code

Java tutorial

Introduction

Here is the source code for de.ocarthon.core.network.HttpClient.java

Source

/*
 *    Copyright 2015 Ocarthon (Philip Standt)
 *
 *    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 de.ocarthon.core.network;

import de.ocarthon.core.utility.reflection.MethodUtil;
import io.netty.bootstrap.Bootstrap;
import io.netty.buffer.ByteBuf;
import io.netty.buffer.PooledByteBufAllocator;
import io.netty.channel.Channel;
import io.netty.channel.ChannelFactory;
import io.netty.channel.ChannelFuture;
import io.netty.channel.ChannelHandlerContext;
import io.netty.channel.ChannelInitializer;
import io.netty.channel.ChannelOption;
import io.netty.channel.ChannelPipeline;
import io.netty.channel.ReflectiveChannelFactory;
import io.netty.channel.SimpleChannelInboundHandler;
import io.netty.channel.nio.NioEventLoopGroup;
import io.netty.channel.socket.SocketChannel;
import io.netty.channel.socket.nio.NioSocketChannel;
import io.netty.handler.codec.http.DefaultHttpRequest;
import io.netty.handler.codec.http.HttpClientCodec;
import io.netty.handler.codec.http.HttpContent;
import io.netty.handler.codec.http.HttpContentDecompressor;
import io.netty.handler.codec.http.HttpHeaderNames;
import io.netty.handler.codec.http.HttpHeaderValues;
import io.netty.handler.codec.http.HttpHeaders;
import io.netty.handler.codec.http.HttpMethod;
import io.netty.handler.codec.http.HttpObject;
import io.netty.handler.codec.http.HttpRequest;
import io.netty.handler.codec.http.HttpResponse;
import io.netty.handler.codec.http.HttpVersion;
import io.netty.handler.codec.http.multipart.DefaultHttpDataFactory;
import io.netty.handler.codec.http.multipart.HttpDataFactory;
import io.netty.handler.codec.http.multipart.HttpPostRequestEncoder;
import io.netty.handler.codec.http.multipart.MixedFileUpload;
import io.netty.handler.ssl.SslContext;
import io.netty.handler.ssl.util.InsecureTrustManagerFactory;
import io.netty.handler.stream.ChunkedWriteHandler;
import io.netty.util.CharsetUtil;

import javax.net.ssl.TrustManagerFactory;
import java.net.URI;
import java.util.List;
import java.util.Map;

public class HttpClient {
    private static NioEventLoopGroup eventLoopGroup = new NioEventLoopGroup();
    private static HttpDataFactory httpDataFactory = new DefaultHttpDataFactory(DefaultHttpDataFactory.MINSIZE);

    private static ChannelFactory<Channel> channelFactory = new ReflectiveChannelFactory<>(NioSocketChannel.class);
    private static Bootstrap defaultHttpBootstrap;
    private final String[] result = { null };
    private String scheme;
    private String host;
    private int port;
    private Bootstrap bootstrap;
    private Channel channel;
    private boolean useUntrustedConnections = false;
    private SslContext sslCtx;
    private boolean forceReconnect = false;

    public HttpClient(String scheme, String host) {
        this(scheme, host, -1);
    }

    public HttpClient(URI uri) {
        this(uri.getScheme(), uri.getHost(), uri.getPort());
    }

    public HttpClient(String scheme, String host, int port) {
        this.scheme = scheme;
        this.host = host;
        this.port = port;

        checkPortAndScheme();
    }

    private static Bootstrap defaultBootstrap() {
        if (defaultHttpBootstrap == null) {
            defaultHttpBootstrap = new Bootstrap();
            defaultHttpBootstrap.group(eventLoopGroup);
            defaultHttpBootstrap.option(ChannelOption.TCP_NODELAY, true);
            defaultHttpBootstrap.option(ChannelOption.ALLOCATOR, PooledByteBufAllocator.DEFAULT);
            defaultHttpBootstrap.channelFactory(channelFactory);
        }
        return defaultHttpBootstrap;
    }

    public static Bootstrap createBootstrap() {
        Bootstrap bootstrap = defaultBootstrap().clone();
        bootstrap.handler(new ChannelInitializer<SocketChannel>() {
            @Override
            protected void initChannel(SocketChannel ch) throws Exception {
                ChannelPipeline p = ch.pipeline();

                p.addLast("codec", new HttpClientCodec());
                p.addLast("chunkedWriter", new ChunkedWriteHandler());
                p.addLast("gzip", new HttpContentDecompressor());
            }
        });

        return bootstrap;
    }

    public static Bootstrap createBootstrapSsl(SslContext sslCtx) {
        Bootstrap bootstrap = defaultBootstrap().clone();
        bootstrap.handler(new ChannelInitializer<SocketChannel>() {
            @Override
            protected void initChannel(SocketChannel ch) throws Exception {
                ChannelPipeline p = ch.pipeline();

                p.addLast("ssl", sslCtx.newHandler(ch.alloc()));

                p.addLast("codec", new HttpClientCodec());
                p.addLast("chunkedWriter", new ChunkedWriteHandler());
                p.addLast("gzip", new HttpContentDecompressor());
            }
        });

        return bootstrap;
    }

    public synchronized String postRequest(String query, List<Map.Entry<String, String>> postParameters) {
        return postRequest(query, postParameters, null, null, null, null);
    }

    public synchronized String postRequest(String query, List<Map.Entry<String, String>> postParameters,
            String filePostName, String fileName, ByteBuf fileData, String mime) {
        if (bootstrap == null) {
            setupBootstrap();
        }

        if (channel == null || forceReconnect) {
            ChannelFuture cf = bootstrap.connect(host, port);
            forceReconnect = false;
            cf.awaitUninterruptibly();
            channel = cf.channel();

            channel.pipeline().addLast("handler", new SimpleChannelInboundHandler<HttpObject>() {
                @Override
                protected void messageReceived(ChannelHandlerContext ctx, HttpObject msg) throws Exception {
                    if (msg instanceof HttpResponse) {
                        HttpResponse response = ((HttpResponse) msg);
                        String connection = (String) response.headers().get(HttpHeaderNames.CONNECTION);
                        if (connection != null && connection.equalsIgnoreCase(HttpHeaderValues.CLOSE.toString()))
                            forceReconnect = true;
                    }

                    if (msg instanceof HttpContent) {
                        HttpContent chunk = (HttpContent) msg;
                        String message = chunk.content().toString(CharsetUtil.UTF_8);

                        if (!message.isEmpty()) {
                            result[0] = message;

                            synchronized (result) {
                                result.notify();
                            }
                        }
                    }
                }
            });
        }
        boolean isFileAttached = fileData != null && fileData.isReadable();
        HttpRequest request = new DefaultHttpRequest(HttpVersion.HTTP_1_1, HttpMethod.POST,
                scheme + "://" + host + ":" + port + "/" + query);
        HttpPostRequestEncoder bodyReqEncoder;
        try {
            bodyReqEncoder = new HttpPostRequestEncoder(httpDataFactory, request, isFileAttached);

            for (Map.Entry<String, String> entry : postParameters) {
                bodyReqEncoder.addBodyAttribute(entry.getKey(), entry.getValue());
            }

            if (isFileAttached) {
                if (mime == null)
                    mime = "application/octet-stream";

                MixedFileUpload mfu = new MixedFileUpload(filePostName, fileName, mime, "binary", null,
                        fileData.capacity(), DefaultHttpDataFactory.MINSIZE);
                mfu.addContent(fileData, true);
                bodyReqEncoder.addBodyHttpData(mfu);
            }

            HttpHeaders headers = request.headers();
            headers.set(HttpHeaderNames.HOST, host);
            headers.set(HttpHeaderNames.CONNECTION, HttpHeaderValues.KEEP_ALIVE);
            headers.set(HttpHeaderNames.ACCEPT_ENCODING, HttpHeaderValues.GZIP);
            headers.set(HttpHeaderNames.USER_AGENT, "OcarthonCore HttpClient");
            request = bodyReqEncoder.finalizeRequest();
        } catch (Exception e) {
            throw new NullPointerException("key or value is empty or null");
        }

        channel.write(request);

        if (bodyReqEncoder.isChunked()) {
            channel.write(bodyReqEncoder);
        }
        channel.flush();

        synchronized (result) {
            try {
                result.wait();
            } catch (InterruptedException e) {
                return null;
            }
        }

        return result[0];
    }

    private void checkPortAndScheme() {
        if (!scheme.equals("http") && !scheme.equals("https")) {
            throw new IllegalArgumentException("only http and https are supported!");
        }

        if (port == -1) {
            switch (scheme) {
            case "http":
                port = 80;
                return;

            case "https":
                port = 443;
            }
        }

        if (port < 0 || port >= 65536) {
            throw new IllegalArgumentException("invalid port: " + port + " (must be between 0 and 65535))");
        }
    }

    public void allowUntrustedConnections() {
        useUntrustedConnections = true;
    }

    public String getScheme() {
        return scheme;
    }

    public String getHost() {
        return host;
    }

    public int getPort() {
        return port;
    }

    private void setupBootstrap() {
        if (scheme.equals("https")) {
            setupSslContext();
            bootstrap = createBootstrapSsl(sslCtx);
        } else {
            bootstrap = createBootstrap();
        }
    }

    private void setupSslContext() {
        if (useUntrustedConnections) {
            sslCtx = (SslContext) MethodUtil.invoke(SslContext.class, "newClientContext",
                    new Class[] { TrustManagerFactory.class }, InsecureTrustManagerFactory.INSTANCE);
        } else {
            sslCtx = (SslContext) MethodUtil.invoke(SslContext.class, "newClientContext", null);
        }
    }
}