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

com.optimizely.ab.OptimizelyHttpClient Maven / Gradle / Ivy

Go to download

The Java SDK for Optimizely Feature Experimentation, Optimizely Full Stack (legacy), and Optimizely Rollouts

The newest version!
/**
 *
 *    Copyright 2019, 2022 Optimizely
 *
 *    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.optimizely.ab;

import com.optimizely.ab.annotations.VisibleForTesting;
import com.optimizely.ab.HttpClientUtils;

import org.apache.http.client.HttpClient;
import org.apache.http.client.HttpRequestRetryHandler;
import org.apache.http.client.ResponseHandler;
import org.apache.http.client.methods.CloseableHttpResponse;
import org.apache.http.client.methods.HttpUriRequest;
import org.apache.http.impl.client.CloseableHttpClient;
import org.apache.http.impl.client.DefaultHttpRequestRetryHandler;
import org.apache.http.impl.client.HttpClientBuilder;
import org.apache.http.impl.client.HttpClients;
import org.apache.http.impl.conn.PoolingHttpClientConnectionManager;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import java.io.Closeable;
import java.io.IOException;
import java.util.concurrent.TimeUnit;

/**
 * Basic HttpClient wrapper to be utilized for fetching the datafile
 * and for posting events through the EventHandler
 *
 * TODO abstract out interface and move into core?
 */
public class OptimizelyHttpClient implements Closeable {

    private static final Logger logger = LoggerFactory.getLogger(OptimizelyHttpClient.class);
    private final CloseableHttpClient httpClient;

    OptimizelyHttpClient(CloseableHttpClient httpClient) {
        this.httpClient = httpClient;
    }

    @VisibleForTesting
    HttpClient getHttpClient() {
        return httpClient;
    }

    public static Builder builder() {
        return new Builder();
    }

    @Override
    public void close() throws IOException {
        this.httpClient.close();
    }

    public  T execute(final HttpUriRequest request, final ResponseHandler responseHandler) throws IOException {
        return httpClient.execute(request, responseHandler);
    }

    public CloseableHttpResponse execute(final HttpUriRequest request) throws IOException {
        return httpClient.execute(request);
    }

    public static class Builder {
        // The following static values are public so that they can be tweaked if necessary.
        // These are the recommended settings for http protocol.  https://hc.apache.org/httpcomponents-client-ga/tutorial/html/connmgmt.html
        // The maximum number of connections allowed across all routes.
        int maxTotalConnections = HttpClientUtils.DEFAULT_MAX_CONNECTIONS;
        // The maximum number of connections allowed for a route
        int maxPerRoute = HttpClientUtils.DEFAULT_MAX_PER_ROUTE;
        // Defines period of inactivity in milliseconds after which persistent connections must be re-validated prior to being leased to the consumer.
        // If this is too long, it's expected to see more requests dropped on staled connections (dropped by the server or networks).
        // We can configure retries (POST for AsyncEventDispatcher) to cover the staled connections.
        int validateAfterInactivity = HttpClientUtils.DEFAULT_VALIDATE_AFTER_INACTIVITY;
        // force-close the connection after this idle time (with 0, eviction is disabled by default)
        long evictConnectionIdleTimePeriod = 0;
        HttpRequestRetryHandler customRetryHandler = null;
        TimeUnit evictConnectionIdleTimeUnit = TimeUnit.MILLISECONDS;
        private int timeoutMillis = HttpClientUtils.CONNECTION_TIMEOUT_MS;


        private Builder() {

        }

        public Builder withMaxTotalConnections(int maxTotalConnections) {
            this.maxTotalConnections = maxTotalConnections;
            return this;
        }

        public Builder withMaxPerRoute(int maxPerRoute) {
            this.maxPerRoute = maxPerRoute;
            return this;
        }

        public Builder withValidateAfterInactivity(int validateAfterInactivity) {
            this.validateAfterInactivity = validateAfterInactivity;
            return this;
        }

        public Builder withEvictIdleConnections(long maxIdleTime, TimeUnit maxIdleTimeUnit) {
            this.evictConnectionIdleTimePeriod = maxIdleTime;
            this.evictConnectionIdleTimeUnit = maxIdleTimeUnit;
            return this;
        }

        // customize retryHandler (DefaultHttpRequestRetryHandler will be used by default)
        public Builder withRetryHandler(HttpRequestRetryHandler retryHandler) {
            this.customRetryHandler = retryHandler;
            return this;
        }
        
        public Builder setTimeoutMillis(int timeoutMillis) {
            this.timeoutMillis = timeoutMillis;
            return this;
        }

        public OptimizelyHttpClient build() {
            PoolingHttpClientConnectionManager poolingHttpClientConnectionManager = new PoolingHttpClientConnectionManager();
            poolingHttpClientConnectionManager.setMaxTotal(maxTotalConnections);
            poolingHttpClientConnectionManager.setDefaultMaxPerRoute(maxPerRoute);
            poolingHttpClientConnectionManager.setValidateAfterInactivity(validateAfterInactivity);

            HttpClientBuilder builder = HttpClients.custom()
                .setDefaultRequestConfig(HttpClientUtils.getDefaultRequestConfigWithTimeout(timeoutMillis))
                .setConnectionManager(poolingHttpClientConnectionManager)
                .disableCookieManagement()
                .useSystemProperties();
            if (customRetryHandler != null) {
                builder.setRetryHandler(customRetryHandler);
            }

            logger.debug("Creating HttpClient with timeout: " + timeoutMillis);

            if (evictConnectionIdleTimePeriod > 0) {
                builder.evictIdleConnections(evictConnectionIdleTimePeriod, evictConnectionIdleTimeUnit);
            }

            CloseableHttpClient closableHttpClient = builder.build();

            return new OptimizelyHttpClient(closableHttpClient);
        }
    }

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy