All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.vmware.xenon.common.http.netty.NettyHttpServiceClient Maven / Gradle / Ivy

There is a newer version: 1.6.18
Show newest version
/*
 * Copyright (c) 2014-2015 VMware, Inc. All Rights Reserved.
 *
 * 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 com.vmware.xenon.common.http.netty;

import java.net.URI;
import java.net.URISyntaxException;
import java.util.EnumSet;
import java.util.Map;
import java.util.Map.Entry;
import java.util.SortedMap;
import java.util.concurrent.CancellationException;
import java.util.concurrent.ConcurrentSkipListMap;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.ScheduledExecutorService;
import java.util.concurrent.TimeUnit;
import java.util.concurrent.TimeoutException;
import java.util.function.BiConsumer;
import java.util.function.BiFunction;
import java.util.logging.Level;
import java.util.logging.Logger;
import javax.net.ssl.SSLContext;
import javax.net.ssl.SSLException;

import io.netty.bootstrap.Bootstrap;
import io.netty.buffer.ByteBuf;
import io.netty.buffer.Unpooled;
import io.netty.channel.Channel;
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.HttpVersion;
import io.netty.handler.codec.http.cookie.ClientCookieDecoder;
import io.netty.handler.codec.http.cookie.Cookie;
import io.netty.handler.codec.http2.Http2SecurityUtil;
import io.netty.handler.ssl.ApplicationProtocolConfig;
import io.netty.handler.ssl.ApplicationProtocolNames;
import io.netty.handler.ssl.SslContext;
import io.netty.handler.ssl.SslContextBuilder;
import io.netty.handler.ssl.SupportedCipherSuiteFilter;
import io.netty.util.AsciiString;

import com.vmware.xenon.common.Operation;
import com.vmware.xenon.common.Operation.AuthorizationContext;
import com.vmware.xenon.common.Operation.CompletionHandler;
import com.vmware.xenon.common.Operation.OperationOption;
import com.vmware.xenon.common.Operation.SocketContext;
import com.vmware.xenon.common.OperationContext;
import com.vmware.xenon.common.Service.Action;
import com.vmware.xenon.common.ServiceClient;
import com.vmware.xenon.common.ServiceErrorResponse;
import com.vmware.xenon.common.ServiceErrorResponse.ErrorDetail;
import com.vmware.xenon.common.ServiceHost;
import com.vmware.xenon.common.ServiceHost.ServiceHostState;
import com.vmware.xenon.common.UriUtils;
import com.vmware.xenon.common.Utils;
import com.vmware.xenon.common.http.netty.NettyChannelPool.NettyChannelGroupKey;

/**
 * Asynchronous request / response client with concurrent connection management
 */
public class NettyHttpServiceClient implements ServiceClient {
    /**
     * Number of maximum parallel connections to a remote host. Idle connections are groomed but if
     * this limit is set too high, and we are talking to many remote hosts, we can possibly exceed
     * the process file descriptor limit
     */
    public static final int DEFAULT_CONNECTIONS_PER_HOST = ServiceClient.DEFAULT_CONNECTION_LIMIT_PER_HOST;

    public static final String CHANNELPOOL_NAME_DEFAULT = "channelpool-default";
    public static final String CHANNELPOOL_NAME_HTTP2 = "channelpool-http2";
    public static final String CHANNELPOOL_NAME_SSL_DEFAULT = "channelpool-ssl-default";
    public static final String CHANNELPOOL_NAME_SSL_HTTP2 = "channelpool-ssl-http2";

    public static final Logger LOGGER = Logger.getLogger(ServiceClient.class
            .getName());
    private static final String ENV_VAR_NAME_HTTP_PROXY = "http_proxy";

    static final AsciiString TRANSACTION_ID_HEADER_ASCII = new AsciiString(
            Operation.TRANSACTION_ID_HEADER);

    static final AsciiString CONTEXT_ID_HEADER_ASCII = new AsciiString(
            Operation.CONTEXT_ID_HEADER);

    static final AsciiString AUTH_TOKEN_HEADER_ASCII = new AsciiString(
            Operation.REQUEST_AUTH_TOKEN_HEADER);

    static final AsciiString DEFAULT_MEDIA_TYPE_ASCII = new AsciiString(
            Operation.MEDIA_TYPE_EVERYTHING_WILDCARDS);

    static final AsciiString PRAGMA_DIRECTIVE_REPLICATED_ASCII = new AsciiString(
            Operation.PRAGMA_DIRECTIVE_REPLICATED);

    static final AsciiString MEDIA_TYPE_JSON_ASCII = new AsciiString(
            Operation.MEDIA_TYPE_APPLICATION_JSON);

    static final AsciiString MEDIA_TYPE_KRYO_OCTET_STREAM_ASCII = new AsciiString(
            Operation.MEDIA_TYPE_APPLICATION_KRYO_OCTET_STREAM);

    private static final int MEDIA_TYPE_APPLICATION_PREFIX_LENGTH = 12;

    private URI httpProxy;
    private AsciiString userAgentAscii;

    private NettyChannelPool sslChannelPool;
    private NettyChannelPool channelPool;
    private NettyChannelPool http2SslChannelPool;
    private NettyChannelPool http2ChannelPool;
    private SortedMap pendingRequests = new ConcurrentSkipListMap<>();

    private ScheduledExecutorService scheduledExecutor;

    private ExecutorService executor;

    private SslContext http2SslContext;
    private SSLContext sslContext;

    private ServiceHost host;

    private CookieJar cookieJar = new CookieJar();

    private boolean isStarted;

    private boolean warnHttp2DisablingConnectionSharing = false;

    private BiConsumer onChannelInitialization;

    private BiConsumer onBootstrapInitialization;

    private BiFunction onPoolSelection = (op, pool) -> pool;

    private BiFunction onPoolSelectionForFailure = (ctx, pool) -> pool;

    private final Object startSync = new Object();

    public static ServiceClient create(String userAgent,
            ExecutorService executor,
            ScheduledExecutorService scheduledExecutor) throws URISyntaxException {
        return create(userAgent, executor, scheduledExecutor, null);
    }

    public static ServiceClient create(String userAgent,
            ExecutorService executor,
            ScheduledExecutorService scheduledExecutor,
            ServiceHost host) throws URISyntaxException {
        NettyHttpServiceClient sc = new NettyHttpServiceClient();
        sc.userAgentAscii = new AsciiString(userAgent);
        sc.scheduledExecutor = scheduledExecutor;
        sc.executor = executor;
        sc.host = host;
        sc.channelPool = new NettyChannelPool(CHANNELPOOL_NAME_DEFAULT);
        sc.http2ChannelPool = new NettyChannelPool(CHANNELPOOL_NAME_HTTP2);
        sc.sslChannelPool = new NettyChannelPool(CHANNELPOOL_NAME_SSL_DEFAULT);
        sc.http2SslChannelPool = new NettyChannelPool(CHANNELPOOL_NAME_SSL_HTTP2);
        String proxy = System.getenv(ENV_VAR_NAME_HTTP_PROXY);
        if (proxy != null) {
            sc.setHttpProxy(new URI(proxy));
        }

        sc.setPendingRequestQueueLimit(ServiceClient.DEFAULT_PENDING_REQUEST_QUEUE_LIMIT);
        sc.setConnectionLimitPerTag(ServiceClient.CONNECTION_TAG_DEFAULT,
                DEFAULT_CONNECTIONS_PER_HOST);
        sc.setConnectionLimitPerTag(ServiceClient.CONNECTION_TAG_HTTP2_DEFAULT,
                DEFAULT_CONNECTION_LIMIT_PER_TAG);
        sc.setRequestPayloadSizeLimit(ServiceClient.REQUEST_PAYLOAD_SIZE_LIMIT);

        return sc;
    }

    private String buildThreadTag() {
        if (this.host != null) {
            return UriUtils.extendUri(this.host.getUri(), "netty-client").toString();
        }
        return getClass().getSimpleName() + ":" + Utils.getSystemNowMicrosUtc();
    }

    @Override
    public void start() {
        synchronized (this.startSync) {
            if (this.isStarted) {
                return;
            }
            this.isStarted = true;
        }

        if (NettyChannelContext.isALPNEnabled() && this.http2SslContext == null) {
            try {
                this.http2SslContext = SslContextBuilder.forClient()
                        .ciphers(Http2SecurityUtil.CIPHERS, SupportedCipherSuiteFilter.INSTANCE)
                        .applicationProtocolConfig(new ApplicationProtocolConfig(
                                ApplicationProtocolConfig.Protocol.ALPN,
                                ApplicationProtocolConfig.SelectorFailureBehavior.NO_ADVERTISE,
                                ApplicationProtocolConfig.SelectedListenerFailureBehavior.ACCEPT,
                                ApplicationProtocolNames.HTTP_2))
                        .build();
            } catch (SSLException e) {
                throw new RuntimeException("SSL context builder exception", e);
            }
        }

        this.channelPool.setThreadTag(buildThreadTag());
        this.channelPool.setThreadCount(Utils.DEFAULT_IO_THREAD_COUNT);
        this.channelPool.setExecutor(this.executor);
        this.channelPool.setOnChannelInitialization(this.onChannelInitialization);
        this.channelPool.setOnBootstrapInitialization(this.onBootstrapInitialization);
        this.channelPool.start();

        // We make a separate pool for HTTP/2. We want to have only one connection per host
        // when using HTTP/2 since HTTP/2 multiplexes streams on a single connection.
        this.http2ChannelPool.setThreadTag(buildThreadTag());
        this.http2ChannelPool.setThreadCount(Utils.DEFAULT_IO_THREAD_COUNT);
        this.http2ChannelPool.setExecutor(this.executor);
        this.http2ChannelPool.setHttp2Only();
        this.http2ChannelPool.setOnChannelInitialization(this.onChannelInitialization);
        this.http2ChannelPool.setOnBootstrapInitialization(this.onBootstrapInitialization);
        this.http2ChannelPool.start();

        if (this.sslContext != null) {
            this.sslChannelPool.setThreadTag(buildThreadTag());
            this.sslChannelPool.setThreadCount(Utils.DEFAULT_IO_THREAD_COUNT);
            this.sslChannelPool.setExecutor(this.executor);
            this.sslChannelPool.setSSLContext(this.sslContext);
            this.sslChannelPool.setOnChannelInitialization(this.onChannelInitialization);
            this.sslChannelPool.setOnBootstrapInitialization(this.onBootstrapInitialization);
            this.sslChannelPool.start();
        }

        if (this.http2SslContext != null) {
            this.http2SslChannelPool.setThreadTag(buildThreadTag());
            this.http2SslChannelPool.setThreadCount(Utils.DEFAULT_IO_THREAD_COUNT);
            this.http2SslChannelPool.setExecutor(this.executor);
            this.http2SslChannelPool.setHttp2Only();
            this.http2SslChannelPool.setHttp2SslContext(this.http2SslContext);
            this.http2SslChannelPool.setOnChannelInitialization(this.onChannelInitialization);
            this.http2SslChannelPool.setOnBootstrapInitialization(this.onBootstrapInitialization);
            this.http2SslChannelPool.start();
        }

        if (this.host != null) {
            MaintenanceProxyService.start(this.host, this::handleMaintenance);
        }
    }

    @Override
    public void stop() {
        // In practice, it's safe not to synchronize here, but this make Findbugs happy.
        synchronized (this.startSync) {
            if (this.isStarted == false) {
                return;
            }
            this.isStarted = false;
        }

        this.channelPool.stop();
        this.sslChannelPool.stop();
        this.http2ChannelPool.stop();
        this.pendingRequests.clear();
    }

    public ServiceClient setHttpProxy(URI proxy) {
        this.httpProxy = proxy;
        return this;
    }

    @Override
    public void send(Operation op) {
        sendRequest(op);
    }

    @Override
    public void sendRequest(Operation op) {
        if (!validateOperation(op)) {
            return;
        }

        Operation clone = op.clone();

        setExpiration(clone);
        setCookies(clone);

        OperationContext ctx = OperationContext.getOperationContext();

        try {
            // First attempt in process delivery to co-located host
            if (!op.isRemote()) {
                if (this.host != null && this.host.handleRequest(clone)) {
                    return;
                }
            }
            sendRemote(clone);
        } finally {
            // we must restore the operation context after each send, since
            // it can be reset by the host, depending on queuing and dispatching behavior
            OperationContext.restoreOperationContext(ctx);
        }
    }

    private void setExpiration(Operation op) {
        if (op.getExpirationMicrosUtc() != 0) {
            return;
        }
        long expMicros = this.host != null ? this.host.getOperationTimeoutMicros()
                : ServiceHostState.DEFAULT_OPERATION_TIMEOUT_MICROS;
        op.setExpiration(Utils.fromNowMicrosUtc(expMicros));
    }

    private void setCookies(Operation clone) {
        if (this.cookieJar.isEmpty()) {
            return;
        }
        // Set cookies for outbound request
        Map cookies = this.cookieJar.list(clone.getUri());
        if (cookies == null || cookies.isEmpty()) {
            return;
        }

        Map existingCookies = clone.getCookies();
        if (existingCookies == null) {
            clone.setCookies(cookies);
        } else {
            cookies.forEach(existingCookies::putIfAbsent);
        }
    }

    private void startTracking(Operation op) {
        this.pendingRequests.put(op.getId(), op);
    }

    void stopTracking(Operation op) {
        this.pendingRequests.remove(op.getId());
    }

    private void updateCookieJarFromResponseHeaders(Operation op) {
        String value = op.getResponseHeaderAsIs(Operation.SET_COOKIE_HEADER);
        if (value == null) {
            return;
        }

        Cookie cookie = ClientCookieDecoder.LAX.decode(value);
        if (cookie == null) {
            return;
        }

        this.cookieJar.add(op.getUri(), cookie);
    }

    private void sendRemote(Operation op) {
        startTracking(op);

        // Determine the remote host address, port number
        // and uri scheme (http or https)
        String remoteHost = op.getUri().getHost();
        String scheme = op.getUri().getScheme();
        int port = op.getUri().getPort();

        if (this.httpProxy != null && !ServiceHost.LOCAL_HOST.equals(remoteHost)) {
            remoteHost = this.httpProxy.getHost();
            port = this.httpProxy.getPort();
            scheme = this.httpProxy.getScheme();
        }

        boolean isHttpScheme = false;
        boolean isHttpsScheme = scheme.equals(UriUtils.HTTPS_SCHEME);
        if (!isHttpsScheme) {
            isHttpScheme = scheme.equals(UriUtils.HTTP_SCHEME);
        }

        if (!isHttpScheme && !isHttpsScheme) {
            op.setRetryCount(0);
            fail(new IllegalArgumentException(
                    "Scheme is not supported: " + op.getUri().getScheme()), op, op.getBodyRaw());
            return;
        }

        if (isHttpsScheme && this.getSSLContext() == null) {
            op.setRetryCount(0);
            fail(new IllegalArgumentException(
                    "HTTPS not enabled, set SSL context before starting client:" + op.getUri()),
                    op, op.getBodyRaw());
            return;
        }

        // if there are no ports specified, choose the default ports http or https
        if (port == -1) {
            port = isHttpScheme ? UriUtils.HTTP_DEFAULT_PORT : UriUtils.HTTPS_DEFAULT_PORT;
        }

        if (op.isConnectionSharing() && isHttpsScheme && !NettyChannelContext.isALPNEnabled()) {
            op.setConnectionSharing(false);
            if (!this.warnHttp2DisablingConnectionSharing) {
                this.warnHttp2DisablingConnectionSharing = true;
                LOGGER.warning(
                        "HTTP/2 requests are not supported on HTTPS. Falling back to HTTP1.1");
            }
        }

        // Determine the channel pool used for this request.
        NettyChannelPool pool = this.channelPool;

        if (op.isConnectionSharing()) {
            if (isHttpsScheme) {
                pool = this.http2SslChannelPool;
            } else {
                pool = this.http2ChannelPool;
            }
        } else if (isHttpsScheme) {
            pool = this.sslChannelPool;
        }

        // allow subclass to choose a pool for sending op
        pool = this.onPoolSelection.apply(op, pool);

        connectChannel(pool, op, remoteHost, port);
    }

    private void connectChannel(NettyChannelPool pool, Operation op,
            String remoteHost, int port) {
        op.nestCompletion((o, e) -> {
            if (o.getStatusCode() == Operation.STATUS_CODE_TIMEOUT) {
                failWithTimeout(op, op.getBodyRaw());
                return;
            }
            if (e != null) {
                Object originalBody = op.getBodyRaw();
                ServiceErrorResponse rsp = null;
                if (o.hasBody() && (o.getBodyRaw() instanceof ServiceErrorResponse)) {
                    rsp = (ServiceErrorResponse) o.getBodyRaw();
                    if (rsp.details == null) {
                        rsp.details = EnumSet.of(ErrorDetail.SHOULD_RETRY);
                    } else {
                        rsp.details.add(ErrorDetail.SHOULD_RETRY);
                    }
                } else {
                    rsp = ServiceErrorResponse.create(e, Operation.STATUS_CODE_BAD_REQUEST,
                            EnumSet.of(ErrorDetail.SHOULD_RETRY));
                }
                op.setBodyNoCloning(rsp);
                fail(e, op, originalBody);
                return;
            }
            sendHttpRequest(op);
        });

        NettyChannelGroupKey key = NettyChannelPool.buildLookupKey(
                op.getConnectionTag(), remoteHost, port, pool.isHttp2Only());
        pool.connectOrReuse(key, op);
    }

    private void sendHttpRequest(Operation op) {
        final Object originalBody = op.getBodyRaw();
        try {
            byte[] body = Utils.encodeBody(op, true);
            if (op.getContentLength() > getRequestPayloadSizeLimit()) {
                String error = String.format("Content length %d, limit is %d",
                        op.getContentLength(), getRequestPayloadSizeLimit());
                Exception e = new IllegalArgumentException(error);
                op.setBody(ServiceErrorResponse.create(e, Operation.STATUS_CODE_BAD_REQUEST));
                fail(e, op, originalBody);
                return;
            }

            String pathAndQuery;
            String path = op.getUri().getPath();
            String query = op.getUri().getRawQuery();
            String userInfo = op.getUri().getRawUserInfo();
            path = path == null || path.isEmpty() ? "/" : path;
            if (query != null) {
                pathAndQuery = path + "?" + query;
            } else {
                pathAndQuery = path;
            }

            boolean useHttp2 = op.isConnectionSharing();
            if (this.httpProxy != null || useHttp2 || userInfo != null) {
                pathAndQuery = op.getUri().toString();
            }

            NettyFullHttpRequest request = null;
            HttpMethod method = toHttpMethod(op.getAction());
            if (body == null || body.length == 0) {
                request = new NettyFullHttpRequest(HttpVersion.HTTP_1_1, method, pathAndQuery,
                        Unpooled.EMPTY_BUFFER, false);
            } else {
                ByteBuf content = Unpooled.wrappedBuffer(body, 0, (int) op.getContentLength());
                request = new NettyFullHttpRequest(HttpVersion.HTTP_1_1, method, pathAndQuery,
                        content, false);
            }

            HttpHeaders httpHeaders = request.headers();

            /*
             * NOTE: Pay close attention to calls that access the operation request headers, since
             * they will cause a memory allocation. We avoid the allocation by first checking if
             * the operation has any custom headers to begin with, then we check for the specific
             * header
             */
            boolean hasRequestHeaders = op.hasRequestHeaders();

            if (useHttp2) {
                // when operation is cloned, it may contain original streamId header. remove it.
                if (hasRequestHeaders) {
                    op.getAndRemoveRequestHeaderAsIs(Operation.STREAM_ID_HEADER);
                }
                // We set the operation so that once a streamId is assigned, we can record
                // the correspondence between the streamId and operation: this will let us
                // handle responses properly later.
                request.setOperation(op);
            }

            String pragmaValue = op.getAndRemoveRequestHeaderAsIs(Operation.PRAGMA_HEADER);
            String acceptValue = op.getAndRemoveRequestHeaderAsIs(Operation.ACCEPT_HEADER);
            String authTokenValue = op
                    .getAndRemoveRequestHeaderAsIs(Operation.REQUEST_AUTH_TOKEN_HEADER);

            if (op.isFromReplication() && pragmaValue == null) {
                httpHeaders.add(HttpHeaderNames.PRAGMA, PRAGMA_DIRECTIVE_REPLICATED_ASCII);
            } else if (pragmaValue != null) {
                httpHeaders.add(HttpHeaderNames.PRAGMA, pragmaValue);
            }

            if (op.getTransactionId() != null) {
                httpHeaders.add(TRANSACTION_ID_HEADER_ASCII, op.getTransactionId());
            }

            if (op.getContextId() != null) {
                httpHeaders.add(CONTEXT_ID_HEADER_ASCII, op.getContextId());
            }

            hasRequestHeaders = op.hasRequestHeaders();

            String connectionHeaderValue = op.getRequestHeaderAsIs(Operation.CONNECTION_HEADER);

            boolean isXenonToXenon = op.isFromReplication() || op.isForwarded();
            if (hasRequestHeaders) {
                // remove all headers that we will set explicitly
                op.getAndRemoveRequestHeaderAsIs(Operation.CONTENT_LENGTH_HEADER);
                op.getAndRemoveRequestHeaderAsIs(Operation.CONTENT_TYPE_HEADER);
                op.getAndRemoveRequestHeaderAsIs(Operation.CONNECTION_HEADER);
                op.getAndRemoveRequestHeaderAsIs(Operation.REFERER_HEADER);
                op.getAndRemoveRequestHeaderAsIs(Operation.HOST_HEADER);
                op.getAndRemoveRequestHeaderAsIs(Operation.USER_AGENT_HEADER);
                hasRequestHeaders = op.hasRequestHeaders();
            }

            if (hasRequestHeaders) {
                for (Entry nameValue : op.getRequestHeaders().entrySet()) {
                    httpHeaders.add(nameValue.getKey(), nameValue.getValue());
                }
            }

            if (authTokenValue != null) {
                httpHeaders.add(AUTH_TOKEN_HEADER_ASCII, authTokenValue);
            } else {
                AuthorizationContext ctx = op.getAuthorizationContext();
                if (ctx != null && ctx.getToken() != null) {
                    httpHeaders.add(AUTH_TOKEN_HEADER_ASCII, ctx.getToken());
                }
            }

            setAsciiContentType(op, httpHeaders);

            httpHeaders.add(HttpHeaderNames.CONTENT_LENGTH,
                    Long.toString(op.getContentLength()));

            if (op.isKeepAlive()) {
                httpHeaders.add(HttpHeaderNames.CONNECTION, HttpHeaderValues.KEEP_ALIVE);
            } else if (connectionHeaderValue != null) {
                // If connection header is specified and keep-alive is not enabled, put back original value
                httpHeaders.add(HttpHeaderNames.CONNECTION, connectionHeaderValue);
            }

            if (op.hasReferer() && !op.isFromReplication()) {
                httpHeaders.add(HttpHeaderNames.REFERER, op.getRefererAsString());
            }

            if (!isXenonToXenon) {
                if (op.getCookies() != null) {
                    String header = CookieJar.encodeCookies(op.getCookies());
                    httpHeaders.set(HttpHeaderNames.COOKIE, header);
                }

                request.headers().set(HttpHeaderNames.USER_AGENT, this.userAgentAscii);
                if (acceptValue == null) {
                    httpHeaders.add(HttpHeaderNames.ACCEPT, DEFAULT_MEDIA_TYPE_ASCII);
                } else {
                    httpHeaders.add(HttpHeaderNames.ACCEPT, acceptValue);
                }

                httpHeaders.add(HttpHeaderNames.HOST, op.getUri().getHost());
            } else {
                if (acceptValue != null) {
                    httpHeaders.add(HttpHeaderNames.ACCEPT, acceptValue);
                }
            }

            if (LOGGER.isLoggable(Level.FINEST)) {
                logRequestFraming(op, request);
            }

            boolean doCookieJarUpdate = !isXenonToXenon;
            op.nestCompletion((o, e) -> {
                if (e != null) {
                    fail(e, op, originalBody);
                    return;
                }

                stopTracking(op);

                if (doCookieJarUpdate) {
                    updateCookieJarFromResponseHeaders(o);
                }

                // After request is sent control is transferred to the
                // NettyHttpServerResponseHandler. The response handler will nest completions
                // and call complete() when response is received, which will invoke this completion
                op.complete();
            });

            op.toggleOption(OperationOption.SOCKET_ACTIVE, true);
            op.getSocketContext().writeHttpRequest(request);
        } catch (Exception e) {
            op.setBody(ServiceErrorResponse.create(e, Operation.STATUS_CODE_BAD_REQUEST,
                    EnumSet.of(ErrorDetail.SHOULD_RETRY)));
            fail(e, op, originalBody);
        }
    }

    private void setAsciiContentType(Operation op, HttpHeaders httpHeaders) {
        String contentType = op.getContentType();
        /*
         * Optimization that saves char sequence to byte[] conversion on the content type, which
         * is non trivial overhead on small HTTP requests.
         * We first check for a specific char at a index in the string, then the hash code to nearly
         * eliminate the false positives from hash code alone.
         */
        if (contentType.length() >= MEDIA_TYPE_APPLICATION_PREFIX_LENGTH
                && contentType.charAt(MEDIA_TYPE_APPLICATION_PREFIX_LENGTH) == 'k'
                && Operation.MEDIA_TYPE_APPLICATION_KRYO_OCTET_STREAM.hashCode() == contentType
                        .hashCode()) {
            httpHeaders.add(HttpHeaderNames.CONTENT_TYPE, MEDIA_TYPE_KRYO_OCTET_STREAM_ASCII);
        } else if (contentType.length() >= MEDIA_TYPE_APPLICATION_PREFIX_LENGTH
                && contentType.charAt(MEDIA_TYPE_APPLICATION_PREFIX_LENGTH) == 'j'
                && Operation.MEDIA_TYPE_APPLICATION_JSON.hashCode() == contentType.hashCode()) {
            httpHeaders.add(HttpHeaderNames.CONTENT_TYPE, MEDIA_TYPE_JSON_ASCII);
        } else {
            httpHeaders.add(HttpHeaderNames.CONTENT_TYPE, op.getContentType());
        }
    }

    private void logRequestFraming(Operation op, NettyFullHttpRequest request) {
        StringBuilder s = new StringBuilder();
        s.append(op.getAction().toString())
                .append(" ")
                .append(op.getUri().toString())
                .append("\n");
        request.headers().forEach((e) -> {
            s.append(e.toString()).append("\n");
        });
        LOGGER.info(s.toString());
    }

    private static HttpMethod toHttpMethod(Action a) {
        switch (a) {
        case DELETE:
            return HttpMethod.DELETE;
        case GET:
            return HttpMethod.GET;
        case OPTIONS:
            return HttpMethod.OPTIONS;
        case PATCH:
            return HttpMethod.PATCH;
        case POST:
            return HttpMethod.POST;
        case PUT:
            return HttpMethod.PUT;
        default:
            throw new IllegalArgumentException("unknown method " + a);

        }
    }

    private void failWithTimeout(Operation op, Object originalBody) {
        Throwable e = new TimeoutException(op.getUri() + ":" + op.getExpirationMicrosUtc());
        op.setStatusCode(Operation.STATUS_CODE_TIMEOUT);
        fail(e, op, originalBody);
    }

    private void fail(Throwable e, Operation op, Object originalBody) {
        stopTracking(op);
        SocketContext ctx = op.getSocketContext();
        if (ctx != null && ctx instanceof NettyChannelContext) {
            NettyChannelContext nettyCtx = (NettyChannelContext) op.getSocketContext();
            NettyChannelPool pool;
            Runnable r = null;
            if (nettyCtx.getProtocol() == NettyChannelContext.Protocol.HTTP2) {
                nettyCtx.removeStreamForOperation(op);
                if (this.http2SslChannelPool != null
                        && this.http2SslChannelPool.isContextInUse(nettyCtx)) {
                    pool = this.http2SslChannelPool;
                } else {
                    pool = this.http2ChannelPool;
                }
                // allow user to choose a pool for op failure
                NettyChannelPool finalPool = this.onPoolSelectionForFailure.apply(nettyCtx, pool);
                // For HTTP/2, we maintain multiple streams, so we don't close the connection.
                r = () -> finalPool.returnOrClose(nettyCtx, false);
            } else {
                op.setSocketContext(null);
                if (this.sslChannelPool != null
                        && this.sslChannelPool.isContextInUse(nettyCtx)) {
                    pool = this.sslChannelPool;
                } else {
                    pool = this.channelPool;
                }
                // allow user to choose a pool for op failure
                NettyChannelPool finalPool = this.onPoolSelectionForFailure.apply(nettyCtx, pool);
                r = () -> finalPool.returnOrClose(nettyCtx, !op.isKeepAlive());
            }

            ExecutorService exec = this.host != null ? this.host.getExecutor() : this.executor;
            if (exec != null) {
                exec.execute(r);
            } else {
                r.run();
            }
        }

        if (this.scheduledExecutor.isShutdown()) {
            op.fail(new CancellationException("Host is stopping"));
            return;
        }

        boolean isRetryRequested = op.getRetryCount() > 0 && op.decrementRetriesRemaining() >= 0;

        if (isRetryRequested) {
            if (op.getStatusCode() >= Operation.STATUS_CODE_SERVER_FAILURE_THRESHOLD) {
                isRetryRequested = false;
            } else if (op.getStatusCode() == Operation.STATUS_CODE_CONFLICT) {
                isRetryRequested = false;
            } else if (op.getStatusCode() == Operation.STATUS_CODE_UNAUTHORIZED) {
                isRetryRequested = false;
            } else if (op.getStatusCode() == Operation.STATUS_CODE_NOT_FOUND) {
                isRetryRequested = false;
            } else if (op.getStatusCode() == Operation.STATUS_CODE_FORBIDDEN) {
                isRetryRequested = false;
            }
        }

        if (!isRetryRequested) {
            LOGGER.fine(() -> String.format("Send of %d, from %s to %s failed with %s",
                    op.getId(), op.getRefererAsString(), op.getUri(), e));
            op.fail(e);
            return;
        }

        LOGGER.info(String.format("Retry %d of request %d from %s to %s due to %s",
                op.getRetryCount() - op.getRetriesRemaining(), op.getId(), op.getRefererAsString(),
                op.getUri(), e));

        int delaySeconds = op.getRetryCount() - op.getRetriesRemaining();

        // restore status code and body, then restart send state machine
        // (connect, encode, write to channel)
        op.setStatusCode(Operation.STATUS_CODE_OK).setBodyNoCloning(originalBody);

        this.scheduledExecutor.schedule(() -> {
            startTracking(op);
            sendRemote(op);
        }, delaySeconds, TimeUnit.SECONDS);
    }

    private static boolean validateOperation(Operation op) {
        if (op == null) {
            throw new IllegalArgumentException("Operation is required");
        }

        Throwable e = null;
        if (op.getUri() == null) {
            e = new IllegalArgumentException("Uri is required");
        } else if (op.getUri().getHost() == null) {
            e = new IllegalArgumentException("Missing host in URI");
        } else if (op.getAction() == null) {
            e = new IllegalArgumentException("Action is required");
        } else if (!op.hasReferer()) {
            e = new IllegalArgumentException("Referer is required");
        } else {
            boolean needsBody = op.getAction() != Action.GET && op.getAction() != Action.DELETE &&
                    op.getAction() != Action.POST && op.getAction() != Action.OPTIONS;
            if (!op.hasBody() && needsBody) {
                e = new IllegalArgumentException("Body is required");
            }
        }

        if (e == null) {
            return true;
        }

        CompletionHandler c = op.getCompletion();
        if (c != null) {
            c.handle(op, e);
            return false;
        }
        throw new RuntimeException(e);
    }

    @Override
    public void handleMaintenance(Operation op) {
        long now = Utils.getSystemNowMicrosUtc();
        if (this.sslChannelPool != null) {
            this.sslChannelPool.handleMaintenance(Operation.createPost(op.getUri()));
        }
        if (this.http2ChannelPool != null) {
            this.http2ChannelPool.handleMaintenance(Operation.createPost(op.getUri()));
        }
        if (this.http2SslChannelPool != null) {
            this.http2SslChannelPool.handleMaintenance(Operation.createPost(op.getUri()));
        }
        this.channelPool.handleMaintenance(Operation.createPost(op.getUri()));

        failExpiredRequests(now);
        op.complete();
    }

    /**
     * Periodically check our pending request sorted map for expired operations. Since
     * maintenance (this method) runs in parallel with the connect and send state machine,
     * we need to be careful on how we fail expired operations. The connect() method
     * uses nestCompletion() which is meant to be used in a asynchronous, but isolated
     * flow over a single operation, where only one stage acts on the operation at a time.
     * We violate this design requirement, since we want to avoid locks, and we want to
     * leverage the maintenance interval. So, we run two passes, first marking the operation
     * with a status code, allowing the parallel connect/send pipeline to avoid triggering
     * a complete() and non atomic roll back of the nested completions.
     */
    private void failExpiredRequests(long now) {
        if (this.pendingRequests.isEmpty()) {
            return;
        }


        // We do a limited search of pending operation, in each maintenance period, to
        // determine if any have expired. The operations are kept in a sorted map,
        // with the key being the operation id. The operation id increments monotonically
        // so we are effectively traversing oldest to newest. We can not assume if operation
        // with id k, with k << n, has expired, also operation with id n has, since operations
        // have different expirations. We limit the search so we don't take too much time when
        // millions of operations are pending
        final int searchLimit = 1000;
        final long epsilonMicros = TimeUnit.SECONDS.toMicros(1);
        int expiredCount = 0;
        int forcedExpiredCount = 0;
        int i = 0;
        for (Operation o : this.pendingRequests.values()) {
            if (i++ >= searchLimit) {
                break;
            }
            long exp = o.getExpirationMicrosUtc();

            if (exp > now) {
                continue;
            }

            // Bad HTTP/2 connections will not close until idle detection kicks in, which can
            // be much longer than operation expiration. We force expiration even if the operation
            // has not been written to the channel, if it has expired and some additional time has
            // passed.
            boolean forceExpiration = o.hasOption(OperationOption.CONNECTION_SHARING) &&
                    now - exp > epsilonMicros;

            if (!forceExpiration && !o.hasOption(OperationOption.SOCKET_ACTIVE)) {
                continue;
            }
            o.fail(Operation.STATUS_CODE_TIMEOUT);
            expiredCount++;
            if (forceExpiration) {
                forcedExpiredCount++;
            }
        }

        if (expiredCount == 0) {
            return;
        }
        LOGGER.info("Failed expired operations, count: " + expiredCount + " forced count: "
                + forcedExpiredCount);
    }

    /**
     * @see ServiceClient#setPendingRequestQueueLimit(int)
     */
    @Override
    public ServiceClient setPendingRequestQueueLimit(int limit) {
        this.channelPool.setPendingRequestQueueLimit(limit);
        if (this.sslChannelPool != null) {
            this.sslChannelPool.setPendingRequestQueueLimit(limit);
        }
        if (this.http2ChannelPool != null) {
            this.http2ChannelPool.setPendingRequestQueueLimit(limit);
        }
        if (this.http2SslChannelPool != null) {
            this.http2SslChannelPool.setPendingRequestQueueLimit(limit);
        }
        return this;
    }

    /**
     * @see ServiceClient#getPendingRequestQueueLimit()
     */
    @Override
    public int getPendingRequestQueueLimit() {
        return this.channelPool.getPendingRequestQueueLimit();
    }

    /**
     * @see ServiceClient#setConnectionLimitPerTag(String, int)
     */
    @Override
    public ServiceClient setConnectionLimitPerTag(String tag, int limit) {
        this.channelPool.setConnectionLimitPerTag(tag, limit);
        if (this.sslChannelPool != null) {
            this.sslChannelPool.setConnectionLimitPerTag(tag, limit);
        }
        if (this.http2ChannelPool != null) {
            this.http2ChannelPool.setConnectionLimitPerTag(tag, limit);
        }
        if (this.http2SslChannelPool != null) {
            this.http2SslChannelPool.setConnectionLimitPerTag(tag, limit);
        }
        return this;
    }

    /**
     * @see ServiceClient#getConnectionLimitPerTag(String)
     */
    @Override
    public int getConnectionLimitPerTag(String tag) {
        return this.channelPool.getConnectionLimitPerTag(tag);
    }

    public ServiceClient setHttp2SslContext(SslContext context) {
        this.http2SslContext = context;
        return this;
    }

    public SslContext getHttp2SslContext() {
        return this.http2SslContext;
    }

    @Override
    public ServiceClient setSSLContext(SSLContext context) {
        this.sslContext = context;
        return this;
    }

    @Override
    public SSLContext getSSLContext() {
        return this.sslContext;
    }

    public NettyChannelPool getChannelPool() {
        return this.channelPool;
    }

    public NettyChannelPool getHttp2ChannelPool() {
        return this.http2ChannelPool;
    }

    public NettyChannelPool getSslChannelPool() {
        return this.sslChannelPool;
    }

    public NettyChannelContext getInUseHttp2SslContext(String tag, String host, int port) {
        if (this.http2SslChannelPool == null) {
            throw new IllegalStateException("Internal error: No HTTP/2 SSL channel pool");
        }
        return this.http2SslChannelPool.getFirstValidHttp2Context(tag, host, port);
    }

    /**
     * Find the HTTP/2 context that is currently being used to talk to a given host.
     * This is intended for infrastructure test purposes.
     */
    public NettyChannelContext getInUseHttp2Context(String tag, String host, int port) {
        if (this.http2ChannelPool == null) {
            throw new IllegalStateException("Internal error: no HTTP/2 channel pool");
        }
        return this.http2ChannelPool.getFirstValidHttp2Context(tag, host, port);
    }

    @Override
    public ConnectionPoolMetrics getConnectionPoolMetrics(boolean http2) {
        if (http2) {
            return getPoolMetrics(this.http2ChannelPool, this.http2SslChannelPool);
        } else {
            return getPoolMetrics(this.channelPool, this.sslChannelPool);
        }
    }

    private ConnectionPoolMetrics getPoolMetrics(NettyChannelPool p1, NettyChannelPool p2) {
        ConnectionPoolMetrics metrics = null;
        if (p1 != null) {
            metrics = p1.getConnectionTagInfo(null);
        }
        if (p2 != null) {
            ConnectionPoolMetrics cpm = p2.getConnectionTagInfo(null);
            if (metrics == null) {
                metrics = cpm;
            } else if (cpm != null) {
                metrics.inUseConnectionCount += cpm.inUseConnectionCount;
                metrics.availableConnectionCount += cpm.availableConnectionCount;
                metrics.pendingRequestCount += cpm.pendingRequestCount;
            }
        }
        return metrics;
    }

    @Override
    public ConnectionPoolMetrics getConnectionPoolMetricsPerTag(String tag) {
        if (tag == null) {
            throw new IllegalArgumentException("tag is required");
        }

        ConnectionPoolMetrics tagInfo = null;
        if (this.http2ChannelPool != null) {
            tagInfo = this.http2ChannelPool.getConnectionTagInfo(tag);
        }
        ConnectionPoolMetrics secureTagInfo = null;
        if (this.http2SslChannelPool != null) {
            secureTagInfo = this.http2SslChannelPool.getConnectionTagInfo(tag);
        }

        if (tagInfo == null) {
            tagInfo = secureTagInfo;
        } else if (secureTagInfo != null) {
            tagInfo.inUseConnectionCount += secureTagInfo.inUseConnectionCount;
            tagInfo.pendingRequestCount += secureTagInfo.pendingRequestCount;
            tagInfo.availableConnectionCount += secureTagInfo.availableConnectionCount;
        }

        if (tagInfo != null) {
            return tagInfo;
        }

        if (this.channelPool != null) {
            tagInfo = this.channelPool.getConnectionTagInfo(tag);
        }
        if (this.sslChannelPool != null) {
            secureTagInfo = this.sslChannelPool.getConnectionTagInfo(tag);
        }

        if (tagInfo == null) {
            tagInfo = secureTagInfo;
        } else if (secureTagInfo != null) {
            tagInfo.inUseConnectionCount += secureTagInfo.inUseConnectionCount;
            tagInfo.pendingRequestCount += secureTagInfo.pendingRequestCount;
            tagInfo.availableConnectionCount += secureTagInfo.availableConnectionCount;
        }

        return tagInfo;
    }

    /**
     * Count how many HTTP/2 contexts we have. There may be more than one if we have
     * an exhausted connection that hasn't been cleaned up yet.
     * This is intended for infrastructure test purposes.
     */
    public int getInUseContextCount(String tag, String host, int port) {
        if (this.http2ChannelPool == null) {
            throw new IllegalStateException("Internal error: no HTTP/2 channel pool");
        }
        return this.http2ChannelPool.getHttp2ActiveContextCount(tag, host, port);
    }

    /**
     * Infrastructure testing use only: do not use this in production
     */
    public void clearCookieJar() {
        this.cookieJar = new CookieJar();
    }

    @Override
    public int getRequestPayloadSizeLimit() {
        return this.channelPool.getRequestPayloadSizeLimit();
    }

    @Override
    public ServiceClient setRequestPayloadSizeLimit(int limit) {
        synchronized (this.startSync) {
            if (this.isStarted) {
                throw new IllegalStateException("Already started");
            }

            this.channelPool.setRequestPayloadSizeLimit(limit);
            if (this.sslChannelPool != null) {
                this.sslChannelPool.setRequestPayloadSizeLimit(limit);
            }
            if (this.http2ChannelPool != null) {
                this.http2ChannelPool.setRequestPayloadSizeLimit(limit);
            }
            if (this.http2SslChannelPool != null) {
                this.http2SslChannelPool.setRequestPayloadSizeLimit(limit);
            }
        }

        return this;
    }

    /**
     * Set callback that can configure netty channel/pipeline at channel initialization
     * @param onChannelInitialization callback
     */
    public void setOnChannelInitialization(BiConsumer onChannelInitialization) {
        this.onChannelInitialization = onChannelInitialization;
    }

    /**
     * Set a callback that can configure netty bootstrap at {@link NettyChannelPool} start.
     * @param onBootstrapInitialization callback
     */
    public void setOnBootstrapInitialization(BiConsumer onBootstrapInitialization) {
        this.onBootstrapInitialization = onBootstrapInitialization;
    }

    /**
     * Set a callback that chooses channel pool to use for the operation.
     * @param onPoolSelection callback
     */
    public void setOnPoolSelection(BiFunction onPoolSelection) {
        this.onPoolSelection = onPoolSelection;
    }

    /**
     * Set a callback that chooses channel pool to use for the operation failure.
     * @param onPoolSelectionForFailure callback
     */
    public void setOnPoolSelectionForFailure(BiFunction onPoolSelectionForFailure) {
        this.onPoolSelectionForFailure = onPoolSelectionForFailure;
    }

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy