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

com.oracle.bmc.http.client.jersey.JerseyHttpClientBuilder Maven / Gradle / Ivy

/**
 * Copyright (c) 2016, 2024, Oracle and/or its affiliates.  All rights reserved.
 * This software is dual-licensed to you under the Universal Permissive License (UPL) 1.0 as shown at https://oss.oracle.com/licenses/upl or Apache License 2.0 as shown at http://www.apache.org/licenses/LICENSE-2.0. You may choose either license.
 */
package com.oracle.bmc.http.client.jersey;

import com.oracle.bmc.http.client.internal.ClientThreadFactory;
import com.oracle.bmc.http.client.jersey.internal.IdleConnectionMonitor;
import com.oracle.bmc.serialization.jackson.JacksonSerializer;
import com.oracle.bmc.http.client.ClientProperty;
import com.oracle.bmc.http.client.HttpClient;
import com.oracle.bmc.http.client.HttpClientBuilder;
import com.oracle.bmc.http.client.KeyStoreWithPassword;
import com.oracle.bmc.http.client.ProxyConfiguration;
import com.oracle.bmc.http.client.RequestInterceptor;
import com.oracle.bmc.http.client.StandardClientProperties;
import com.oracle.bmc.http.client.jersey.internal.DaemonClientAsyncExecutorProvider;
import org.apache.http.ConnectionReuseStrategy;
import org.apache.http.client.HttpRequestRetryHandler;
import org.apache.http.client.config.RequestConfig;
import org.apache.http.config.Registry;
import org.apache.http.config.RegistryBuilder;
import org.apache.http.conn.HttpClientConnectionManager;
import org.apache.http.conn.socket.ConnectionSocketFactory;
import org.apache.http.conn.socket.PlainConnectionSocketFactory;
import org.apache.http.conn.ssl.SSLConnectionSocketFactory;
import org.apache.http.impl.NoConnectionReuseStrategy;
import org.apache.http.impl.client.DefaultHttpRequestRetryHandler;
import org.apache.http.impl.conn.PoolingHttpClientConnectionManager;
import org.apache.http.protocol.RequestContent;
import org.glassfish.jersey.apache.connector.ApacheClientProperties;
import org.glassfish.jersey.apache.connector.ApacheConnectionClosingStrategy;
import org.glassfish.jersey.apache.connector.ApacheConnectorProvider;
import org.glassfish.jersey.apache.connector.ApacheHttpClientBuilderConfigurator;
import org.glassfish.jersey.client.ClientConfig;
import org.glassfish.jersey.client.ClientProperties;
import org.glassfish.jersey.client.HttpUrlConnectorProvider;
import org.glassfish.jersey.client.RequestEntityProcessing;
import org.glassfish.jersey.internal.InternalProperties;
import org.glassfish.jersey.jackson.internal.jackson.jaxrs.json.JacksonJaxbJsonProvider;
import org.glassfish.jersey.jackson.internal.jackson.jaxrs.json.JacksonJsonProvider;
import org.glassfish.jersey.spi.ExecutorServiceProvider;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import javax.net.ssl.HostnameVerifier;
import javax.net.ssl.SSLContext;
import javax.ws.rs.client.Client;
import javax.ws.rs.client.ClientBuilder;
import javax.ws.rs.client.WebTarget;
import java.net.InetSocketAddress;
import java.net.URI;
import java.net.URISyntaxException;
import java.security.KeyStore;
import java.time.Duration;
import java.util.ArrayList;
import java.util.Collection;
import java.util.Comparator;
import java.util.HashMap;
import java.util.Map;
import java.util.stream.Collectors;

import static com.oracle.bmc.http.client.jersey.internal.IdleConnectionMonitor.DEFAULT_IDLE_CONNECTION_MONITOR_THREAD_WAIT_TIME_IN_SECONDS;

final class JerseyHttpClientBuilder implements HttpClientBuilder {
    private static final Logger LOG = LoggerFactory.getLogger(JerseyHttpClientBuilder.class);
    private static final int DEFAULT_MAX_ASYNC_THREADS = 50;

    /** The default {@link ClientBuilderDecorator} simply calls {@link ClientBuilder#build()}. */
    private static final ClientBuilderDecorator SIMPLE_DECORATOR =
            clientBuilder -> clientBuilder.build();

    public static ConnectionReuseStrategy DEFAULT_CONNECTION_REUSE_STRATEGY;

    private static boolean DEFAULT_IDLE_CONNECTION_MONITOR_THREAD_ENABLED;

    static {
        DEFAULT_IDLE_CONNECTION_MONITOR_THREAD_ENABLED =
                Boolean.parseBoolean(
                        System.getProperty(
                                "oci.javasdk.apache.idle.connection.monitor.thread.enabled",
                                "true"));
        if (!DEFAULT_IDLE_CONNECTION_MONITOR_THREAD_ENABLED) {
            DEFAULT_CONNECTION_REUSE_STRATEGY = new NoConnectionReuseStrategy();
        }
    }

    private static int DEFAULT_IDLE_CONNECTION_MONITOR_THREAD_IDLE_TIMEOUT_IN_SECONDS = 20;

    public static HttpRequestRetryHandler DEFAULT_REQUEST_RETRY_HANDLER =
            new DefaultHttpRequestRetryHandler(0, false);

    public static final JacksonJsonProvider JACKSON_JSON_PROVIDER =
            new JacksonJaxbJsonProvider(
                    JacksonSerializer.getDefaultObjectMapper(),
                    JacksonJaxbJsonProvider.DEFAULT_ANNOTATIONS);

    private final Collection> requestInterceptors =
            new ArrayList<>();
    private final Map properties = new HashMap<>();
    private URI baseUri;
    private String baseUriString;
    private boolean isApacheNonBufferingClient = false;
    private KeyStoreWithPassword keyStore;
    private KeyStore trustStore;
    private HostnameVerifier hostnameVerifier;
    private SSLContext sslContext;
    private boolean useApacheConnector = true;
    private ExecutorServiceProvider executorServiceProvider = null;
    private boolean useJerseyDefaultExecutorServiceProvider = false;
    private ClientBuilderDecorator decorator = SIMPLE_DECORATOR;

    private boolean shouldSetDefaultConnectionManagerForApacheConnector = false;

    private boolean apacheIdleConnectionMonitorThreadEnabled = false;
    private int apacheIdleConnectionMonitorThreadWaitTimeInSeconds;
    private int apacheIdleConnectionMonitorThreadIdleTimeoutInSeconds;
    private HttpClientConnectionManager httpClientConnectionManager;

    JerseyHttpClientBuilder() {
        // buffer by default, for signing and better error messages.
        properties.put(
                ClientProperties.REQUEST_ENTITY_PROCESSING, RequestEntityProcessing.BUFFERED);
        if (shouldUseApacheConnector()) {
            properties.put(
                    ApacheClientProperties.REQUEST_CONFIG,
                    RequestConfig.custom()
                            .setContentCompressionEnabled(false)
                            .setExpectContinueEnabled(false)
                            .build());
            properties.put(
                    ApacheClientProperties.CONNECTION_CLOSING_STRATEGY,
                    new ApacheConnectionClosingStrategy.ImmediateClosingStrategy());

            // later, before constructing the client, we should set the connection manager,
            // unless the user has set a different connection manager already using
            // properties.put(ApacheClientProperties.CONNECTION_MANAGER, cm)
            this.shouldSetDefaultConnectionManagerForApacheConnector = true;

            this.apacheIdleConnectionMonitorThreadEnabled =
                    DEFAULT_IDLE_CONNECTION_MONITOR_THREAD_ENABLED;

            this.apacheIdleConnectionMonitorThreadWaitTimeInSeconds =
                    DEFAULT_IDLE_CONNECTION_MONITOR_THREAD_WAIT_TIME_IN_SECONDS;

            this.apacheIdleConnectionMonitorThreadIdleTimeoutInSeconds =
                    DEFAULT_IDLE_CONNECTION_MONITOR_THREAD_IDLE_TIMEOUT_IN_SECONDS;

            property(
                    com.oracle.bmc.http.client.jersey.ApacheClientProperties
                            .CONNECTION_MANAGER_SHARED,
                    false);
            property(
                    com.oracle.bmc.http.client.jersey.ApacheClientProperties.RETRY_HANDLER,
                    DEFAULT_REQUEST_RETRY_HANDLER);
            property(
                    com.oracle.bmc.http.client.jersey.ApacheClientProperties.REUSE_STRATEGY,
                    DEFAULT_CONNECTION_REUSE_STRATEGY);
        }
    }

    @Override
    public HttpClientBuilder baseUri(URI uri) {
        this.baseUri = uri;
        return this;
    }

    @Override
    public HttpClientBuilder baseUri(String uri) {
        this.baseUriString = uri;
        return this;
    }

    @Override
    public  HttpClientBuilder property(ClientProperty key, T value) {
        if (key == StandardClientProperties.ASYNC_POOL_SIZE) {
            properties.put(ClientProperties.ASYNC_THREADPOOL_SIZE, value);
        } else if (key == StandardClientProperties.READ_TIMEOUT) {
            properties.put(ClientProperties.READ_TIMEOUT, (int) ((Duration) value).toMillis());
        } else if (key == StandardClientProperties.CONNECT_TIMEOUT) {
            properties.put(ClientProperties.CONNECT_TIMEOUT, (int) ((Duration) value).toMillis());
        } else if (key == StandardClientProperties.KEY_STORE) {
            keyStore = (KeyStoreWithPassword) value;
        } else if (key == StandardClientProperties.TRUST_STORE) {
            trustStore = (KeyStore) value;
        } else if (key == StandardClientProperties.HOSTNAME_VERIFIER) {
            hostnameVerifier = (HostnameVerifier) value;
        } else if (key == StandardClientProperties.SSL_CONTEXT) {
            sslContext = (SSLContext) value;
        } else if (key == StandardClientProperties.BUFFER_REQUEST) {
            if ((Boolean) value) {
                properties.put(
                        ClientProperties.REQUEST_ENTITY_PROCESSING,
                        RequestEntityProcessing.BUFFERED);
            } else {
                if (shouldUseApacheConnector()) {
                    LOG.debug("Configuring non-buffering for ApacheConnectorProvider");
                    isApacheNonBufferingClient = true;
                } else {
                    LOG.debug("Configuring non-buffering for HttpUrlConnectorProvider");
                    properties.put(ClientProperties.REQUEST_ENTITY_PROCESSING, null);
                    properties.put(ApacheClientProperties.RETRY_HANDLER, null);
                    properties.put(ApacheClientProperties.REUSE_STRATEGY, null);
                    properties.put(ApacheClientProperties.CONNECTION_MANAGER, null);
                    properties.put(ApacheClientProperties.CONNECTION_CLOSING_STRATEGY, null);
                    properties.put(ApacheClientProperties.CONNECTION_MANAGER_SHARED, null);

                    // Use fixed length streaming when possible to allow large uploads without
                    // buffering.
                    properties.put(HttpUrlConnectorProvider.USE_FIXED_LENGTH_STREAMING, true);
                }
            }
        } else if (key == StandardClientProperties.PROXY) {
            ProxyConfiguration proxy = (ProxyConfiguration) value;
            String scheme;
            switch (proxy.getProxy().type()) {
                case HTTP:
                    scheme = "http";
                    break;
                case SOCKS:
                    scheme = "socks";
                    break;
                default:
                    throw new IllegalArgumentException(
                            "Unsupported proxy type: " + proxy.getProxy().type());
            }
            InetSocketAddress address = (InetSocketAddress) proxy.getProxy().address();
            try {
                properties.put(
                        ClientProperties.PROXY_URI,
                        new URI(
                                scheme,
                                null,
                                address.getHostString(),
                                address.getPort(),
                                null,
                                null,
                                null));
            } catch (URISyntaxException e) {
                throw new IllegalArgumentException(e);
            }
            if (proxy.getUsername() != null) {
                properties.put(ClientProperties.PROXY_USERNAME, proxy.getUsername());
            }
            if (proxy.getPassword() != null) {
                properties.put(
                        ClientProperties.PROXY_PASSWORD, String.valueOf(proxy.getPassword()));
            }
        } else if (key == JerseyClientProperties.USE_APACHE_CONNECTOR) {
            useApacheConnector = (Boolean) value;
        } else if (key == JerseyClientProperties.USE_JERSEY_DEFAULT_EXECUTOR_SERVICE_PROVIDER) {
            useJerseyDefaultExecutorServiceProvider = (((Boolean) value) == Boolean.TRUE);
        } else if (key == JerseyClientProperties.EXECUTOR_SERVICE_PROVIDER) {
            executorServiceProvider = (ExecutorServiceProvider) value;
        } else if (key == JerseyClientProperties.APACHE_IDLE_CONNECTION_MONITOR_THREAD_ENABLED) {
            apacheIdleConnectionMonitorThreadEnabled = (Boolean) value;
        } else if (key
                == JerseyClientProperties.APACHE_IDLE_CONNECTION_MONITOR_THREAD_WAIT_TIME_SECONDS) {
            apacheIdleConnectionMonitorThreadWaitTimeInSeconds = (Integer) value;
        } else if (key
                == JerseyClientProperties
                        .APACHE_IDLE_CONNECTION_MONITOR_THREAD_IDLE_TIMEOUT_SECONDS) {
            apacheIdleConnectionMonitorThreadIdleTimeoutInSeconds = (Integer) value;
        } else if (key == ClientBuilderDecorator.PROPERTY) {
            decorator = (ClientBuilderDecorator) value;
        } else if (key instanceof JerseyClientProperty) {
            String jerseyProperty = ((JerseyClientProperty) key).jerseyProperty;
            properties.put(jerseyProperty, value);
            if (com.oracle.bmc.http.client.jersey.ApacheClientProperties.CONNECTION_MANAGER
                    .jerseyProperty.equals(jerseyProperty)) {
                // user has set connection manager already, don't set default anymore
                shouldSetDefaultConnectionManagerForApacheConnector = false;
            }
        } else {
            throw new IllegalArgumentException(
                    "Unknown or unsupported HTTP client property " + key);
        }
        return this;
    }

    @Override
    public HttpClientBuilder registerRequestInterceptor(
            int priority, RequestInterceptor interceptor) {
        requestInterceptors.add(new PrioritizedValue<>(priority, interceptor));
        return this;
    }

    @Override
    public HttpClient build() {
        if (shouldSetDefaultConnectionManagerForApacheConnector) {
            PoolingHttpClientConnectionManager cm =
                    buildDefaultPoolingHttpClientConnectionManagerForApacheConnector(sslContext);
            properties.put(ApacheClientProperties.CONNECTION_MANAGER, cm);
        }
        Object connectionManagerObject = properties.get(ApacheClientProperties.CONNECTION_MANAGER);

        httpClientConnectionManager = null;
        if (apacheIdleConnectionMonitorThreadEnabled && connectionManagerObject != null) {
            httpClientConnectionManager = (HttpClientConnectionManager) connectionManagerObject;
            IdleConnectionMonitor.registerConnectionManager(
                    httpClientConnectionManager,
                    apacheIdleConnectionMonitorThreadWaitTimeInSeconds,
                    apacheIdleConnectionMonitorThreadIdleTimeoutInSeconds);
        }

        ClientBuilder clientBuilder = ClientBuilder.newBuilder();
        ClientConfig clientConfig = new ClientConfig();
        if (shouldUseApacheConnector()) {
            LOG.info("Setting connector provider to ApacheConnectorProvider");
            clientConfig.connectorProvider(new ApacheConnectorProvider());
            // need to configure the client so that it doesn't fail if we provide Content-Length
            // headers etc.
            // this will overwrite them instead
            clientConfig.register(
                    new ApacheHttpClientBuilderConfigurator() {
                        @Override
                        public org.apache.http.impl.client.HttpClientBuilder configure(
                                org.apache.http.impl.client.HttpClientBuilder httpClientBuilder) {
                            return httpClientBuilder.addInterceptorLast(new RequestContent(true));
                        }
                    });
        } else {
            LOG.info("Setting connector provider to HttpUrlConnectorProvider");
            // 1) enable workaround for 'patch' requests
            HttpUrlConnectorProvider provider =
                    new HttpUrlConnectorProvider().useSetMethodWorkaround();
            clientConfig.connectorProvider(provider);
        }
        clientBuilder.withConfig(clientConfig);
        clientBuilder.register(JACKSON_JSON_PROVIDER);

        if (executorServiceProvider == null && !useJerseyDefaultExecutorServiceProvider) {
            // not told to use Jersey's default
            if (!properties.containsKey(ClientProperties.ASYNC_THREADPOOL_SIZE)) {
                // register default async thread pool size
                LOG.info(
                        "Using {} with default async thread pool size {}, since no other executorServiceProvider was given and the async thread pool size had not been set specifically",
                        DaemonClientAsyncExecutorProvider.class.getName(),
                        DEFAULT_MAX_ASYNC_THREADS);
                properties.put(ClientProperties.ASYNC_THREADPOOL_SIZE, DEFAULT_MAX_ASYNC_THREADS);
            }
        }

        for (Map.Entry prop : properties.entrySet()) {
            clientBuilder.property(prop.getKey(), prop.getValue());
        }
        clientBuilder.property(InternalProperties.JSON_FEATURE, "JacksonFeature");
        if (keyStore != null) {
            clientBuilder.keyStore(keyStore.getKeyStore(), keyStore.getPassword());
        }
        if (hostnameVerifier != null) {
            clientBuilder.hostnameVerifier(hostnameVerifier);
        }
        if (sslContext == null) {
            if (trustStore != null) {
                clientBuilder.trustStore(trustStore);
            }
        } else {
            if (trustStore != null) {
                throw new IllegalStateException(
                        "Cannot set trust store when SSL context is also set");
            }
            clientBuilder.sslContext(sslContext);
        }

        // build the client, using the decorator
        Client client = decorator.finish(clientBuilder);

        if (executorServiceProvider != null) {
            client.register(executorServiceProvider);
        } else {
            // no specific ExecutorServiceProvider set
            if (!useJerseyDefaultExecutorServiceProvider) {
                // not told to use Jersey's default
                client.register(DaemonClientAsyncExecutorProvider.class);
            }
        }

        if (LOG.isTraceEnabled()) {
            String collectedProperties = null;
            if (client.getConfiguration() != null
                    && client.getConfiguration().getProperties() != null) {
                collectedProperties =
                        client.getConfiguration().getProperties().entrySet().stream()
                                .sorted(Comparator.comparing(e -> e.getKey()))
                                .map(e -> "['" + e.getKey() + "':'" + e.getValue() + "']")
                                .collect(Collectors.joining(", "));
            }
            LOG.trace(
                    "Creating client '{}' with 'isApacheNonBufferingClient'='{}' and properties: {}",
                    client.getClass().getName(),
                    isApacheNonBufferingClient,
                    collectedProperties);
        }

        WebTarget baseTarget;
        if (baseUri != null) {
            baseTarget = client.target(baseUri);
        } else {
            baseTarget = client.target(baseUriString);
        }

        return new JerseyHttpClient(
                client,
                baseTarget,
                requestInterceptors.stream()
                        .sorted(Comparator.comparingInt(p -> p.priority))
                        .map(p -> p.value)
                        .collect(Collectors.toList()),
                isApacheNonBufferingClient,
                httpClientConnectionManager);
    }

    private boolean shouldUseApacheConnector() {
        return JerseyHttpProvider.isApacheDependencyPresent && useApacheConnector;
    }

    private static class PrioritizedValue {
        final int priority;
        final T value;

        PrioritizedValue(int priority, T value) {
            this.priority = priority;
            this.value = value;
        }
    }

    /**
     * Build the default {@link PoolingHttpClientConnectionManager} used for the Apache Connector.
     *
     * @param sslContext SSL context, or null if none
     * @return PoolingHttpClientConnectionManager
     */
    public static PoolingHttpClientConnectionManager
            buildDefaultPoolingHttpClientConnectionManagerForApacheConnector(
                    SSLContext sslContext) {
        PoolingHttpClientConnectionManager cm;
        if (sslContext != null) {
            Registry socketFactoryRegistry =
                    RegistryBuilder.create()
                            .register("http", PlainConnectionSocketFactory.getSocketFactory())
                            .register("https", new SSLConnectionSocketFactory(sslContext))
                            .build();
            cm = new PoolingHttpClientConnectionManager(socketFactoryRegistry);
        } else {
            cm = new PoolingHttpClientConnectionManager();
        }
        cm.setMaxTotal(50);
        cm.setDefaultMaxPerRoute(50);
        return cm;
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy