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

org.cloudfoundry.reactor._DefaultConnectionContext Maven / Gradle / Ivy

The newest version!
/*
 * Copyright 2013-2021 the original author or authors.
 *
 * 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 org.cloudfoundry.reactor;

import com.fasterxml.jackson.databind.DeserializationFeature;
import com.fasterxml.jackson.databind.ObjectMapper;
import com.fasterxml.jackson.databind.deser.DeserializationProblemHandler;
import com.fasterxml.jackson.datatype.jdk8.Jdk8Module;
import io.netty.buffer.PooledByteBufAllocator;
import io.netty.handler.logging.LogLevel;
import io.netty.handler.ssl.SslContextBuilder;
import org.cloudfoundry.Nullable;
import org.cloudfoundry.reactor.util.ByteBufAllocatorMetricProviderWrapper;
import org.cloudfoundry.reactor.util.DefaultSslCertificateTruster;
import org.cloudfoundry.reactor.util.SslCertificateTruster;
import org.cloudfoundry.reactor.util.StaticTrustManagerFactory;
import org.immutables.value.Value;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import reactor.core.publisher.Mono;
import reactor.netty.http.client.HttpClient;
import reactor.netty.resources.ConnectionProvider;
import reactor.netty.resources.LoopResources;
import reactor.netty.tcp.SslProvider;

import jakarta.annotation.PostConstruct;
import jakarta.annotation.PreDestroy;
import javax.management.JMException;
import javax.management.MalformedObjectNameException;
import javax.management.ObjectName;
import javax.net.ssl.SSLException;
import javax.net.ssl.TrustManagerFactory;
import java.lang.management.ManagementFactory;
import java.time.Duration;
import java.util.List;
import java.util.Optional;
import java.util.function.UnaryOperator;

import static com.fasterxml.jackson.annotation.JsonInclude.Include.NON_NULL;
import static io.netty.channel.ChannelOption.CONNECT_TIMEOUT_MILLIS;
import static io.netty.channel.ChannelOption.SO_KEEPALIVE;
import static io.netty.channel.ChannelOption.SO_RCVBUF;
import static io.netty.channel.ChannelOption.SO_SNDBUF;

/**
 * The default implementation of the {@link ConnectionContext} interface. This is the implementation that should be used for most non-testing cases.
 */
@Value.Immutable
abstract class _DefaultConnectionContext implements ConnectionContext {

    private static final int DEFAULT_PORT = 443;

    private static final int SEND_RECEIVE_BUFFER_SIZE = 10 * 1024 * 1024;

    private final Logger logger = LoggerFactory.getLogger("cloudfoundry-client");

    /**
     * Disposes resources created to service this connection context
     */
    @PreDestroy
    public final void dispose() {
        getConnectionProvider().ifPresent(ConnectionProvider::dispose);
        getThreadPool().dispose();

        try {
            ObjectName name = getByteBufAllocatorObjectName();

            if (ManagementFactory.getPlatformMBeanServer().isRegistered(name)) {
                ManagementFactory.getPlatformMBeanServer().unregisterMBean(name);
            }
        } catch (JMException e) {
            this.logger.error("Unable to register ByteBufAllocator MBean", e);
        }
    }

    @Override
    public abstract Optional getCacheDuration();

    /**
     * The number of connections to use when processing requests and responses. Setting this to {@code null} disables connection pooling.
     */
    @Nullable
    @Value.Default
    public Integer getConnectionPoolSize() {
        return ConnectionProvider.DEFAULT_POOL_MAX_CONNECTIONS;
    }

    @Override
    @Value.Default
    public HttpClient getHttpClient() {
        HttpClient client = configureHttpClient(createHttpClient().compress(true)
            .secure(this::configureSsl));

        return getAdditionalHttpClientConfiguration().map(configuration -> configuration.apply(client))
            .orElse(client);
    }

    @Override
    @Value.Default
    public Long getInvalidTokenRetries() {
        return 5L;
    }

    @Override
    @Value.Default
    public ObjectMapper getObjectMapper() {
        ObjectMapper objectMapper = new ObjectMapper().disable(DeserializationFeature.FAIL_ON_UNKNOWN_PROPERTIES)
            .enable(DeserializationFeature.ACCEPT_SINGLE_VALUE_AS_ARRAY)
            .registerModule(new Jdk8Module())
            .setSerializationInclusion(NON_NULL);

        getProblemHandlers().forEach(objectMapper::addHandler);

        return objectMapper;
    }

    @Override
    @Value.Default
    public RootProvider getRootProvider() {
        return DelegatingRootProvider.builder()
            .apiHost(getApiHost())
            .objectMapper(getObjectMapper())
            .port(getPort())
            .secure(getSecure())
            .build();
    }

    /**
     * The number of worker threads to use when processing requests and responses
     */
    @Value.Default
    public Integer getThreadPoolSize() {
        return LoopResources.DEFAULT_IO_WORKER_COUNT;
    }

    @Override
    public Mono trust(String host, int port) {
        return getSslCertificateTruster()
            .map(t -> t.trust(host, port, Duration.ofSeconds(30)))
            .orElse(Mono.empty());
    }

    /**
     * Additional configuration for the underlying HttpClient
     */
    abstract Optional> getAdditionalHttpClientConfiguration();

    /**
     * The hostname of the API root. Typically something like {@code api.run.pivotal.io}.
     */
    abstract String getApiHost();

    /**
     * The {@code CONNECT_TIMEOUT_MILLIS} value
     */
    abstract Optional getConnectTimeout();

    @Value.Derived
    Optional getConnectionProvider() {
        ConnectionProvider.Builder builder = ConnectionProvider.builder("cloudfoundry-client");

        return Optional.ofNullable(getConnectionPoolSize())
            .map(connectionPoolSize -> builder.maxConnections(connectionPoolSize)
                .pendingAcquireMaxCount(-1)
                .build());
    }

    /**
     * The {@code SO_KEEPALIVE} value
     */
    abstract Optional getKeepAlive();

    /**
     * The port for the Cloud Foundry instance. Defaults to {@code 443}.
     */
    abstract Optional getPort();

    /**
     * Jackson deserialization problem handlers. Typically only used for testing.
     */
    abstract List getProblemHandlers();

    /**
     * The (optional) proxy configuration
     */
    abstract Optional getProxyConfiguration();

    /**
     * Whether the connection to the root API should be secure (i.e. using HTTPS). Defaults to {@code true}.
     */
    abstract Optional getSecure();

    /**
     * Whether to skip SSL certificate validation for all hosts reachable from the API host. Defaults to {@code false}.
     */
    abstract Optional getSkipSslValidation();

    @Value.Derived
    Optional getSslCertificateTruster() {
        if (getSkipSslValidation().orElse(false)) {
            return Optional.of(new DefaultSslCertificateTruster(getProxyConfiguration(), getThreadPool()));
        } else {
            return Optional.empty();
        }
    }

    /**
     * The timeout for the SSL close notify flush
     */
    abstract Optional getSslCloseNotifyFlushTimeout();

    /**
     * THe timeout for the SSL close notify read
     */
    abstract Optional getSslCloseNotifyReadTimeout();

    /**
     * The timeout for the SSL handshake negotiation
     */
    abstract Optional getSslHandshakeTimeout();

    @Value.Derived
    LoopResources getThreadPool() {
        return LoopResources.create("cloudfoundry-client", getThreadPoolSize(), true);
    }

    @PostConstruct
    void monitorByteBufAllocator() {
        try {
            ObjectName name = getByteBufAllocatorObjectName();

            if (ManagementFactory.getPlatformMBeanServer().isRegistered(name)) {
                this.logger.warn("MBean '{}' is already registered and will be removed. You should only have a single DefaultConnectionContext per endpoint.", name);
                ManagementFactory.getPlatformMBeanServer().unregisterMBean(name);
            }

            ManagementFactory.getPlatformMBeanServer().registerMBean(new ByteBufAllocatorMetricProviderWrapper(PooledByteBufAllocator.DEFAULT), name);
        } catch (JMException e) {
            this.logger.error("Unable to register ByteBufAllocator MBean", e);
        }
    }

    private HttpClient configureConnectTimeout(HttpClient client) {
        return getConnectTimeout()
            .map(connectTimeout -> client.option(CONNECT_TIMEOUT_MILLIS, (int) connectTimeout.toMillis()))
            .orElse(client);
    }

    private HttpClient configureHttpClient(HttpClient client) {
        client = configureProxy(client);
        client = client.runOn(getThreadPool())
            .option(SO_SNDBUF, SEND_RECEIVE_BUFFER_SIZE)
            .option(SO_RCVBUF, SEND_RECEIVE_BUFFER_SIZE);
        client = configureKeepAlive(client);
        client = client.wiretap("cloudfoundry-client.wire", LogLevel.TRACE);

        return configureConnectTimeout(client);
    }

    private HttpClient configureKeepAlive(HttpClient client) {
        return getKeepAlive()
            .map(keepAlive -> client.option(SO_KEEPALIVE, keepAlive))
            .orElse(client);
    }

    private HttpClient configureProxy(HttpClient client) {
        return getProxyConfiguration()
            .map(proxyConfiguration -> proxyConfiguration.configure(client))
            .orElse(client);
    }

    private void configureSsl(SslProvider.SslContextSpec ssl){
        try{
            SslProvider.Builder builder = ssl.sslContext(createSslContextBuilder().build());

            getSslCloseNotifyReadTimeout().ifPresent(builder::closeNotifyReadTimeout);
            getSslHandshakeTimeout().ifPresent(builder::handshakeTimeout);
            getSslCloseNotifyFlushTimeout().ifPresent(builder::closeNotifyFlushTimeout);
        } catch (SSLException e) {
            this.logger.error("Unable to configure SSL", e);
        }
    }

    private HttpClient createHttpClient() {
        return getConnectionProvider()
            .map(HttpClient::create)
            .orElse(HttpClient.create());
    }

    private SslContextBuilder createSslContextBuilder() {
        SslContextBuilder sslContextBuilder = SslContextBuilder.forClient();

        getSslCertificateTruster()
            .map(this::createTrustManagerFactory)
            .ifPresent(sslContextBuilder::trustManager);

        return sslContextBuilder;
    }

    private TrustManagerFactory createTrustManagerFactory(SslCertificateTruster sslCertificateTruster) {
        return new StaticTrustManagerFactory(sslCertificateTruster);
    }

    private ObjectName getByteBufAllocatorObjectName() throws MalformedObjectNameException {
        return ObjectName.getInstance(String.format("org.cloudfoundry.reactor:type=ByteBufAllocator,endpoint=%s/%d", getApiHost(), getPort().orElse(DEFAULT_PORT)));
    }

}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy