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

com.bazaarvoice.ostrich.pool.ServicePoolBuilder Maven / Gradle / Ivy

package com.bazaarvoice.ostrich.pool;

import com.bazaarvoice.ostrich.HostDiscovery;
import com.bazaarvoice.ostrich.HostDiscoverySource;
import com.bazaarvoice.ostrich.LoadBalanceAlgorithm;
import com.bazaarvoice.ostrich.RetryPolicy;
import com.bazaarvoice.ostrich.ServiceFactory;
import com.bazaarvoice.ostrich.healthcheck.ExponentialBackoffHealthCheckRetryDelay;
import com.bazaarvoice.ostrich.healthcheck.HealthCheckRetryDelay;
import com.bazaarvoice.ostrich.loadbalance.RandomAlgorithm;
import com.bazaarvoice.ostrich.partition.IdentityPartitionFilter;
import com.bazaarvoice.ostrich.partition.PartitionFilter;
import com.bazaarvoice.ostrich.partition.PartitionKey;
import com.codahale.metrics.MetricRegistry;
import com.google.common.annotations.VisibleForTesting;
import com.google.common.base.Strings;
import com.google.common.base.Throwables;
import com.google.common.base.Ticker;
import com.google.common.collect.Lists;
import com.google.common.util.concurrent.ThreadFactoryBuilder;

import java.io.IOException;
import java.util.List;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
import java.util.concurrent.ScheduledExecutorService;
import java.util.concurrent.ThreadFactory;
import java.util.concurrent.TimeUnit;

import static com.google.common.base.Preconditions.checkArgument;
import static com.google.common.base.Preconditions.checkNotNull;
import static java.lang.String.format;

public class ServicePoolBuilder {
    private static final int DEFAULT_NUM_HEALTH_CHECK_THREADS = 1;
    private static final HealthCheckRetryDelay DEFAULT_HEALTH_CHECK_RETRY_POLICY = new ExponentialBackoffHealthCheckRetryDelay(100, 10_000, TimeUnit.MILLISECONDS);

    private final Class _serviceType;
    private final List _hostDiscoverySources = Lists.newArrayList();
    private boolean _closeHostDiscovery;
    private ServiceFactory _serviceFactory;
    private String _serviceName;
    private ScheduledExecutorService _healthCheckExecutor;
    private ServiceCachingPolicy _cachingPolicy;
    private PartitionFilter _partitionFilter = new IdentityPartitionFilter();
    private PartitionContextSupplier _partitionContextSupplier = new EmptyPartitionContextSupplier();
    private LoadBalanceAlgorithm _loadBalanceAlgorithm = new RandomAlgorithm();
    private MetricRegistry _metrics;
    private ExecutorService _asyncExecutor;
    private HealthCheckRetryDelay _healthCheckRetryDelay = DEFAULT_HEALTH_CHECK_RETRY_POLICY;

    public static  ServicePoolBuilder create(Class serviceType) {
        return new ServicePoolBuilder<>(serviceType);
    }

    private ServicePoolBuilder(Class serviceType) {
        _serviceType = checkNotNull(serviceType);
    }

    /**
     * Adds a {@link HostDiscoverySource} instance to the builder.  Multiple instances of {@code HostDiscoverySource}
     * may be specified.  The service pool will query the sources in the order they were registered and use the first
     * non-null {@link HostDiscovery} returned for the service name provided by the
     * {@link ServiceFactory#getServiceName()} method of the factory configured by {@link #withServiceFactory}.
     * 

* Note that using this method will cause the ServicePoolBuilder to call * {@link HostDiscoverySource#forService(String serviceName)} when {@link #build()} is called and pass the returned * {@link HostDiscovery} to the new {@code ServicePool}. Subsequently calling {@link ServicePool#close()} will in * turn call {@link HostDiscovery#close()} on the passed instance. * * @param hostDiscoverySource a host discovery source to use to find the {@link HostDiscovery} when constructing * the {@link ServicePool} * @return this */ public ServicePoolBuilder withHostDiscoverySource(HostDiscoverySource hostDiscoverySource) { checkNotNull(hostDiscoverySource); return withHostDiscovery(hostDiscoverySource, true); } /** * Adds a {@link HostDiscovery} instance to the builder. The service pool will use this {@code HostDiscovery} * instance unless a preceding {@link HostDiscoverySource} provides a non-null instance of {@code HostDiscovery} for * a given service name. *

* Once this method is called, any subsequent calls to host discovery-related methods on this builder instance are * ignored (because this non-null discovery will always be returned). *

* Note that callers of this method are responsible for calling {@link HostDiscovery#close} on the passed instance. * * @param hostDiscovery the host discovery instance to use in the built {@link ServicePool} * @return this */ public ServicePoolBuilder withHostDiscovery(final HostDiscovery hostDiscovery) { checkNotNull(hostDiscovery); HostDiscoverySource hostDiscoverySource = new HostDiscoverySource() { @Override public HostDiscovery forService(String serviceName) { return hostDiscovery; } }; return withHostDiscovery(hostDiscoverySource, false); } private ServicePoolBuilder withHostDiscovery(HostDiscoverySource source, boolean close) { _hostDiscoverySources.add(close ? new ClosingHostDiscoverySource(source) : source); return this; } /** * Adds a {@code ServiceFactory} instance to the builder. The {@code ServiceFactory#configure} method will be * called at this time to allow the {@code ServiceFactory} to set service pool settings on the builder. *

* @param serviceFactory the ServiceFactory to use * @return this */ public ServicePoolBuilder withServiceFactory(ServiceFactory serviceFactory) { checkNotNull(serviceFactory); checkArgument(!Strings.isNullOrEmpty(serviceFactory.getServiceName()), "Service name must be configured"); _serviceFactory = serviceFactory; _serviceName = serviceFactory.getServiceName(); _serviceFactory.configure(this); return this; } /** * Adds a {@code ScheduledExecutorService} instance to the builder for use in executing health checks. *

* Adding an executor is optional. If one isn't specified then one will be created and used automatically. * * @param executor The {@code ScheduledExecutorService} to use * @return this */ public ServicePoolBuilder withHealthCheckExecutor(ScheduledExecutorService executor) { _healthCheckExecutor = checkNotNull(executor); return this; } /** * Adds an {@code ExecutorService} instance to the builder for use in executing asynchronous requests. The executor * is not used unless an asynchronous pool is built with the {@link #buildAsync} method. *

* Adding an executor is optional. If one isn't specified then one will be created and used automatically when * {@code buildAsync} is called. * * @param executor The {@code ExecutorService} to use * @return this */ public ServicePoolBuilder withAsyncExecutor(ExecutorService executor) { _asyncExecutor = checkNotNull(executor); return this; } /** * Enables caching of service instances in the built {@link ServicePool}. *

* Specifying a caching policy is optional. If one isn't specified then a default one that doesn't cache service * instances will be created and used automatically. * * @param policy The {@link ServiceCachingPolicy} to use * @return this */ public ServicePoolBuilder withCachingPolicy(ServiceCachingPolicy policy) { _cachingPolicy = checkNotNull(policy); return this; } /** * Uses the specified partition filter on every service pool operation to narrow down the set of end points that * may be used to service a particular request. * * @param partitionFilter The {@link PartitionFilter} to use * @return this */ public ServicePoolBuilder withPartitionFilter(PartitionFilter partitionFilter) { _partitionFilter = checkNotNull(partitionFilter); return this; } /** * Makes the built proxy generate partition context based on the {@link PartitionKey} annotation * on method arguments in class {@code S}. *

* If {@code S} is not annotated, or annotated differently than desired, consider using * {@link #withPartitionContextAnnotationsFrom(Class)} instead. *

* NOTE: This is only useful if building a proxy with {@link #buildProxy(com.bazaarvoice.ostrich.RetryPolicy)}. If * partition context is necessary with a normal service pool, then can be provided directly by calling * {@link com.bazaarvoice.ostrich.ServicePool#execute(com.bazaarvoice.ostrich.PartitionContext, * com.bazaarvoice.ostrich.RetryPolicy, com.bazaarvoice.ostrich.ServiceCallback)}. * * @return this */ public ServicePoolBuilder withPartitionContextAnnotations() { return withPartitionContextAnnotationsFrom(_serviceType); } /** * Uses {@link PartitionKey} annotations from the specified class to generate partition context in the built proxy. *

* NOTE: This is only useful if building a proxy with {@link #buildProxy(com.bazaarvoice.ostrich.RetryPolicy)}. If * partition context is necessary with a normal service pool, then can be provided directly by calling * {@link com.bazaarvoice.ostrich.ServicePool#execute(com.bazaarvoice.ostrich.PartitionContext, * com.bazaarvoice.ostrich.RetryPolicy, com.bazaarvoice.ostrich.ServiceCallback)}. * * @param annotatedServiceClass A service class with {@link PartitionKey} annotations. * @return this */ public ServicePoolBuilder withPartitionContextAnnotationsFrom(Class annotatedServiceClass) { checkNotNull(annotatedServiceClass); _partitionContextSupplier = new AnnotationPartitionContextSupplier(_serviceType, annotatedServiceClass); return this; } /** * Sets the {@code LoadBalanceAlgorithm} that should be used for this service. * * @param algorithm A load balance algorithm to choose between available end points for the service. * @return this */ public ServicePoolBuilder withLoadBalanceAlgorithm(LoadBalanceAlgorithm algorithm) { _loadBalanceAlgorithm = checkNotNull(algorithm); return this; } /** Sets the {@code MetricRegistry} that should be used for this service. * * @param metrics The metric registry to use. * @return this */ public ServicePoolBuilder withMetricRegistry(MetricRegistry metrics) { _metrics = checkNotNull(metrics); return this; } /** * Sets the {@code HealthCheckRetryPolicy} that should be used for this service. * * @param healthCheckRetryDelay retry policy to use * @return this */ public ServicePoolBuilder withHealthCheckRetryPolicy(HealthCheckRetryDelay healthCheckRetryDelay) { _healthCheckRetryDelay = checkNotNull(healthCheckRetryDelay); return this; } /** * Builds a {@code com.bazaarvoice.ostrich.ServicePool}. * * @return The {@code com.bazaarvoice.ostrich.ServicePool} that was constructed. */ public com.bazaarvoice.ostrich.ServicePool build() { return buildInternal(); } /** * Builds a {@code com.bazaarvoice.ostrich.AsyncServicePool}. * * @return The {@code com.bazaarvoice.ostrich.AsyncServicePool} that was constructed. */ public com.bazaarvoice.ostrich.AsyncServicePool buildAsync() { ServicePool pool = buildInternal(); boolean shutdownAsyncExecutorOnClose = (_asyncExecutor == null); if (_asyncExecutor == null) { ThreadFactory threadFactory = new ThreadFactoryBuilder() .setNameFormat(_serviceName + "-AsyncExecutorThread-%d") .setDaemon(true) .build(); _asyncExecutor = Executors.newCachedThreadPool(threadFactory); } return new AsyncServicePool<>(Ticker.systemTicker(), pool, true, _asyncExecutor, shutdownAsyncExecutorOnClose, _metrics); } /** * Builds a dynamic proxy that wraps a {@code ServicePool} and implements the service interface directly. This is * appropriate for stateless services where it's sensible for the same retry policy to apply to every method. *

* It is the caller's responsibility to shutdown the service pool when they're done with it by casting the proxy * to {@link java.io.Closeable} and calling the {@link java.io.Closeable#close()} method. * * @param retryPolicy The retry policy to apply for every service call. * @return The dynamic proxy instance that implements the service interface {@code S} and the * {@link java.io.Closeable} interface. */ public S buildProxy(RetryPolicy retryPolicy) { return ServicePoolProxy.create(_serviceType, retryPolicy, build(), _partitionContextSupplier, true); } @VisibleForTesting ServicePool buildInternal() { checkNotNull(_serviceFactory); checkNotNull(_metrics); HostDiscovery hostDiscovery = findHostDiscovery(_serviceName); boolean shutdownHealthCheckExecutorOnClose = (_healthCheckExecutor == null); try { if (_cachingPolicy == null) { _cachingPolicy = ServiceCachingPolicyBuilder.NO_CACHING; } if (_healthCheckExecutor == null) { _healthCheckExecutor = Executors.newScheduledThreadPool(DEFAULT_NUM_HEALTH_CHECK_THREADS, new ThreadFactoryBuilder() .setNameFormat(_serviceName + "-HealthCheckThread-%d") .setDaemon(true) .build()); } ServicePool servicePool = new ServicePool<>(Ticker.systemTicker(), hostDiscovery, _closeHostDiscovery, _serviceFactory, _cachingPolicy, _partitionFilter, _loadBalanceAlgorithm, _healthCheckExecutor, shutdownHealthCheckExecutorOnClose, _healthCheckRetryDelay, _metrics); _closeHostDiscovery = false; return servicePool; } catch (Throwable t) { if (shutdownHealthCheckExecutorOnClose && _healthCheckExecutor != null) { _healthCheckExecutor.shutdownNow(); _healthCheckExecutor = null; } try { if (_closeHostDiscovery) { hostDiscovery.close(); } } catch (IOException e) { // NOP } finally { _closeHostDiscovery = false; } throw Throwables.propagate(t); } } private HostDiscovery findHostDiscovery(String serviceName) { for (HostDiscoverySource source : _hostDiscoverySources) { HostDiscovery hostDiscovery = source.forService(serviceName); if (hostDiscovery != null) { return hostDiscovery; } } throw new IllegalStateException(format("No HostDiscovery found for service: %s", serviceName)); } private class ClosingHostDiscoverySource implements HostDiscoverySource { private HostDiscoverySource _wrappedSource; public ClosingHostDiscoverySource(HostDiscoverySource wrappedSource) { _wrappedSource = wrappedSource; } @Override public HostDiscovery forService(String serviceName) { HostDiscovery hostDiscovery = _wrappedSource.forService(serviceName); if (hostDiscovery != null) { _closeHostDiscovery = true; } return hostDiscovery; } } }