diff --git a/.gitignore b/.gitignore index 5bed4f055..a123472de 100644 --- a/.gitignore +++ b/.gitignore @@ -14,3 +14,5 @@ build/ bin/ gradle.properties + +classes/ \ No newline at end of file diff --git a/CHANGELOG.md b/CHANGELOG.md index 0490b87c1..f5067fafa 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -2,6 +2,13 @@ All notable changes to the LaunchDarkly Java SDK will be documented in this file. This project adheres to [Semantic Versioning](http://semver.org). +## [2.2.0] - 2017-04-11 +### Added +- Authentication for proxied http requests is now supported (Basic Auth only) + +### Changed +- Improved Redis connection pool management. + ## [2.1.0] - 2017-03-02 ### Added - LdClientInterface (and its implementation) have a new method: `boolean isFlagKnown(String featureKey)` which checks for a diff --git a/CONTRIBUTING.md b/CONTRIBUTING.md index 23101b8a7..cae601ae5 100644 --- a/CONTRIBUTING.md +++ b/CONTRIBUTING.md @@ -1,4 +1,25 @@ Contributing to the LaunchDarkly SDK for Java ================================================ -We encourage pull-requests and other contributions from the community. We've also published an [SDK contributor's guide](http://docs.launchdarkly.com/v1.0/docs/sdk-contributors-guide) that provides a detailed explanation of how our SDKs work. \ No newline at end of file +We encourage pull-requests and other contributions from the community. We've also published an [SDK contributor's guide](http://docs.launchdarkly.com/v1.0/docs/sdk-contributors-guide) that provides a detailed explanation of how our SDKs work. + + +Testing Proxy Settings +================== +Installation is your own journey, but your squid.conf file should have auth/access sections that look something like this: + +``` +auth_param basic program /usr/local/Cellar/squid/3.5.6/libexec/basic_ncsa_auth /passwords +auth_param basic realm proxy +acl authenticated proxy_auth REQUIRED +http_access allow authenticated +# And finally deny all other access to this proxy +http_access deny all +``` + +The contents of the passwords file is: +``` +user:$apr1$sBfNiLFJ$7h3S84EgJhlbWM3v.90v61 +``` + +The username/password is: user/password diff --git a/build.gradle b/build.gradle index 96bcb7b74..d84e729f9 100644 --- a/build.gradle +++ b/build.gradle @@ -19,20 +19,18 @@ repositories { allprojects { group = 'com.launchdarkly' - version = "2.1.0" + version = "2.2.0" sourceCompatibility = 1.7 targetCompatibility = 1.7 } dependencies { - compile "org.apache.httpcomponents:httpclient:4.5.2" - compile "org.apache.httpcomponents:httpclient-cache:4.5.2" compile "commons-codec:commons-codec:1.10" compile "com.google.code.gson:gson:2.7" compile "com.google.guava:guava:19.0" compile "joda-time:joda-time:2.9.3" compile "org.slf4j:slf4j-api:1.7.21" - compile group: "com.launchdarkly", name: "okhttp-eventsource", version: "1.1.1", changing: true + compile group: "com.launchdarkly", name: "okhttp-eventsource", version: "1.3.0", changing: true compile "redis.clients:jedis:2.9.0" testCompile "org.easymock:easymock:3.4" testCompile 'junit:junit:4.12' diff --git a/src/main/java/com/launchdarkly/client/EventProcessor.java b/src/main/java/com/launchdarkly/client/EventProcessor.java index a578e771c..844f3d64f 100644 --- a/src/main/java/com/launchdarkly/client/EventProcessor.java +++ b/src/main/java/com/launchdarkly/client/EventProcessor.java @@ -1,13 +1,10 @@ package com.launchdarkly.client; import com.google.common.util.concurrent.ThreadFactoryBuilder; -import com.google.gson.Gson; -import org.apache.http.client.config.RequestConfig; -import org.apache.http.client.methods.CloseableHttpResponse; -import org.apache.http.client.methods.HttpPost; -import org.apache.http.entity.StringEntity; -import org.apache.http.impl.client.CloseableHttpClient; -import org.apache.http.impl.client.HttpClients; +import okhttp3.MediaType; +import okhttp3.Request; +import okhttp3.RequestBody; +import okhttp3.Response; import org.slf4j.Logger; import org.slf4j.LoggerFactory; @@ -58,19 +55,10 @@ public void flush() { class Consumer implements Runnable { private final Logger logger = LoggerFactory.getLogger(Consumer.class); - private final CloseableHttpClient client; private final LDConfig config; Consumer(LDConfig config) { this.config = config; - RequestConfig requestConfig = RequestConfig.custom() - .setConnectTimeout(config.connectTimeout) - .setSocketTimeout(config.socketTimeout) - .setProxy(config.proxyHost) - .build(); - client = HttpClients.custom() - .setDefaultRequestConfig(requestConfig) - .build(); } @Override @@ -88,29 +76,28 @@ public void flush() { } private void postEvents(List events) { - CloseableHttpResponse response = null; - Gson gson = new Gson(); - String json = gson.toJson(events); + + String json = LDConfig.gson.toJson(events); logger.debug("Posting " + events.size() + " event(s) to " + config.eventsURI + " with payload: " + json); - HttpPost request = config.postEventsRequest(sdkKey, "/bulk"); - StringEntity entity = new StringEntity(json, "UTF-8"); - entity.setContentType("application/json"); - request.setEntity(entity); + String content = LDConfig.gson.toJson(events); - try { - response = client.execute(request); - if (Util.handleResponse(logger, request, response)) { - logger.debug("Successfully posted " + events.size() + " event(s)."); + Request request = config.getRequestBuilder(sdkKey) + .url(config.eventsURI.toString() + "/bulk") + .post(RequestBody.create(MediaType.parse("application/json; charset=utf-8"), content)) + .addHeader("Content-Type", "application/json") + .build(); + + logger.debug("Posting " + events.size() + " event(s) using request: " + request); + + try (Response response = config.httpClient.newCall(request).execute()) { + if (!response.isSuccessful()) { + logger.info("Got unexpected response when posting events: " + response); + } else { + logger.debug("Events Response: " + response.code()); } } catch (IOException e) { - logger.error("Unhandled exception in LaunchDarkly client attempting to connect to URI: " + config.eventsURI, e); - } finally { - try { - if (response != null) response.close(); - } catch (IOException e) { - logger.error("Unhandled exception in LaunchDarkly client", e); - } + logger.info("Unhandled exception in LaunchDarkly client when posting events to URL: " + request.url(), e); } } } diff --git a/src/main/java/com/launchdarkly/client/FeatureFlag.java b/src/main/java/com/launchdarkly/client/FeatureFlag.java index 7dbaa0810..5fd556a97 100644 --- a/src/main/java/com/launchdarkly/client/FeatureFlag.java +++ b/src/main/java/com/launchdarkly/client/FeatureFlag.java @@ -1,6 +1,5 @@ package com.launchdarkly.client; -import com.google.gson.Gson; import com.google.gson.JsonElement; import com.google.gson.reflect.TypeToken; import org.slf4j.Logger; @@ -14,7 +13,6 @@ class FeatureFlag { private final static Logger logger = LoggerFactory.getLogger(FeatureFlag.class); - private static final Gson gson = new Gson(); private static final Type mapType = new TypeToken>() { }.getType(); @@ -31,11 +29,11 @@ class FeatureFlag { private final boolean deleted; static FeatureFlag fromJson(String json) { - return gson.fromJson(json, FeatureFlag.class); + return LDConfig.gson.fromJson(json, FeatureFlag.class); } static Map fromJsonMap(String json) { - return gson.fromJson(json, mapType); + return LDConfig.gson.fromJson(json, mapType); } FeatureFlag(String key, int version, boolean on, List prerequisites, String salt, List targets, List rules, VariationOrRollout fallthrough, Integer offVariation, List variations, boolean deleted) { diff --git a/src/main/java/com/launchdarkly/client/FeatureRequestor.java b/src/main/java/com/launchdarkly/client/FeatureRequestor.java index 543c93b91..3ce564d97 100644 --- a/src/main/java/com/launchdarkly/client/FeatureRequestor.java +++ b/src/main/java/com/launchdarkly/client/FeatureRequestor.java @@ -1,15 +1,7 @@ package com.launchdarkly.client; -import org.apache.http.client.cache.CacheResponseStatus; -import org.apache.http.client.cache.HttpCacheContext; -import org.apache.http.client.config.RequestConfig; -import org.apache.http.client.methods.CloseableHttpResponse; -import org.apache.http.client.methods.HttpGet; -import org.apache.http.impl.client.CloseableHttpClient; -import org.apache.http.impl.client.cache.CacheConfig; -import org.apache.http.impl.client.cache.CachingHttpClients; -import org.apache.http.impl.conn.PoolingHttpClientConnectionManager; -import org.apache.http.util.EntityUtils; +import okhttp3.Request; +import okhttp3.Response; import org.slf4j.Logger; import org.slf4j.LoggerFactory; @@ -17,110 +9,50 @@ import java.util.Map; class FeatureRequestor { - - public static final String GET_LATEST_FLAGS_PATH = "/sdk/latest-flags"; + private static final Logger logger = LoggerFactory.getLogger(FeatureRequestor.class); + private static final String GET_LATEST_FLAGS_PATH = "/sdk/latest-flags"; private final String sdkKey; private final LDConfig config; - private final CloseableHttpClient client; - private static final Logger logger = LoggerFactory.getLogger(FeatureRequestor.class); FeatureRequestor(String sdkKey, LDConfig config) { this.sdkKey = sdkKey; this.config = config; - this.client = createClient(); - } - - protected CloseableHttpClient createClient() { - CloseableHttpClient client; - PoolingHttpClientConnectionManager manager = new PoolingHttpClientConnectionManager(); - manager.setMaxTotal(100); - manager.setDefaultMaxPerRoute(20); - - CacheConfig cacheConfig = CacheConfig.custom() - .setMaxCacheEntries(1000) - .setMaxObjectSize(131072) - .setSharedCache(false) - .build(); - - RequestConfig requestConfig = RequestConfig.custom() - .setConnectTimeout(config.connectTimeout) - .setSocketTimeout(config.socketTimeout) - .setProxy(config.proxyHost) - .build(); - client = CachingHttpClients.custom() - .setCacheConfig(cacheConfig) - .setConnectionManager(manager) - .setDefaultRequestConfig(requestConfig) - .build(); - return client; } Map getAllFlags() throws IOException { - HttpCacheContext context = HttpCacheContext.create(); - - HttpGet request = config.getRequest(sdkKey, GET_LATEST_FLAGS_PATH); - - CloseableHttpResponse response = null; - try { - logger.debug("Making request: " + request); - response = client.execute(request, context); - - logCacheResponse(context.getCacheResponseStatus()); - if (!Util.handleResponse(logger, request, response)) { - throw new IOException("Failed to fetch flags"); - } - - String json = EntityUtils.toString(response.getEntity()); - logger.debug("Got response: " + response.toString()); - return FeatureFlag.fromJsonMap(json); - } - finally { - try { - if (response != null) response.close(); - } catch (IOException ignored) { - } - } + String body = get(GET_LATEST_FLAGS_PATH); + return FeatureFlag.fromJsonMap(body); } - void logCacheResponse(CacheResponseStatus status) { - switch (status) { - case CACHE_HIT: - logger.debug("A response was generated from the cache with " + - "no requests sent upstream"); - break; - case CACHE_MODULE_RESPONSE: - logger.debug("The response was generated directly by the " + - "caching module"); - break; - case CACHE_MISS: - logger.debug("The response came from an upstream server"); - break; - case VALIDATED: - logger.debug("The response was generated from the cache " + - "after validating the entry with the origin server"); - break; - } + FeatureFlag getFlag(String featureKey) throws IOException { + String body = get(GET_LATEST_FLAGS_PATH + "/" + featureKey); + return FeatureFlag.fromJson(body); } - FeatureFlag getFlag(String featureKey) throws IOException { - HttpCacheContext context = HttpCacheContext.create(); - HttpGet request = config.getRequest(sdkKey, GET_LATEST_FLAGS_PATH + "/" + featureKey); - CloseableHttpResponse response = null; - try { - response = client.execute(request, context); + private String get(String path) throws IOException { + Request request = config.getRequestBuilder(sdkKey) + .url(config.baseURI.toString() + path) + .get() + .build(); - logCacheResponse(context.getCacheResponseStatus()); + logger.debug("Making request: " + request); - if (!Util.handleResponse(logger, request, response)) { - throw new IOException("Failed to fetch flag"); - } - return FeatureFlag.fromJson(EntityUtils.toString(response.getEntity())); - } - finally { - try { - if (response != null) response.close(); - } catch (IOException ignored) { + try (Response response = config.httpClient.newCall(request).execute()) { + String body = response.body().string(); + + if (!response.isSuccessful()) { + if (response.code() == 401) { + logger.error("[401] Invalid SDK key when accessing URI: " + request.url()); + } + throw new IOException("Unexpected response when retrieving Feature Flag(s): " + response + " using url: " + + request.url() + " with body: " + body); } + logger.debug("Get flag(s) response: " + response.toString() + " with body: " + body); + logger.debug("Cache hit count: " + config.httpClient.cache().hitCount() + " Cache network Count: " + config.httpClient.cache().networkCount()); + logger.debug("Cache response: " + response.cacheResponse()); + logger.debug("Network response: " + response.networkResponse()); + + return body; } } } diff --git a/src/main/java/com/launchdarkly/client/LDClient.java b/src/main/java/com/launchdarkly/client/LDClient.java index 50fc98628..33c32246e 100644 --- a/src/main/java/com/launchdarkly/client/LDClient.java +++ b/src/main/java/com/launchdarkly/client/LDClient.java @@ -5,7 +5,6 @@ import com.google.gson.JsonElement; import com.google.gson.JsonPrimitive; import org.apache.commons.codec.binary.Hex; -import org.apache.http.annotation.ThreadSafe; import org.slf4j.Logger; import org.slf4j.LoggerFactory; @@ -28,11 +27,10 @@ * A client for the LaunchDarkly API. Client instances are thread-safe. Applications should instantiate * a single {@code LDClient} for the lifetime of their application. */ -@ThreadSafe public class LDClient implements LDClientInterface { private static final Logger logger = LoggerFactory.getLogger(LDClient.class); private static final String HMAC_ALGORITHM = "HmacSHA256"; - protected static final String CLIENT_VERSION = getClientVersion(); + static final String CLIENT_VERSION = getClientVersion(); private final LDConfig config; private final String sdkKey; diff --git a/src/main/java/com/launchdarkly/client/LDConfig.java b/src/main/java/com/launchdarkly/client/LDConfig.java index ad7ab7837..7be4cf2ff 100644 --- a/src/main/java/com/launchdarkly/client/LDConfig.java +++ b/src/main/java/com/launchdarkly/client/LDConfig.java @@ -1,31 +1,45 @@ package com.launchdarkly.client; -import org.apache.http.HttpHost; -import org.apache.http.client.methods.HttpGet; -import org.apache.http.client.methods.HttpPost; -import org.apache.http.client.utils.URIBuilder; +import com.google.common.io.Files; +import com.google.gson.Gson; +import okhttp3.Authenticator; +import okhttp3.Cache; +import okhttp3.ConnectionPool; +import okhttp3.Credentials; +import okhttp3.OkHttpClient; +import okhttp3.Request; +import okhttp3.Response; +import okhttp3.Route; import org.slf4j.Logger; import org.slf4j.LoggerFactory; +import java.io.File; +import java.io.IOException; +import java.net.InetSocketAddress; +import java.net.Proxy; import java.net.URI; +import java.util.concurrent.TimeUnit; /** * This class exposes advanced configuration options for the {@link LDClient}. Instances of this class must be constructed with a {@link com.launchdarkly.client.LDConfig.Builder}. - * */ public final class LDConfig { + private static final Logger logger = LoggerFactory.getLogger(LDConfig.class); + static final Gson gson = new Gson(); + private static final URI DEFAULT_BASE_URI = URI.create("https://app.launchdarkly.com"); private static final URI DEFAULT_EVENTS_URI = URI.create("https://events.launchdarkly.com"); private static final URI DEFAULT_STREAM_URI = URI.create("https://stream.launchdarkly.com"); private static final int DEFAULT_CAPACITY = 10000; - private static final int DEFAULT_CONNECT_TIMEOUT = 2000; - private static final int DEFAULT_SOCKET_TIMEOUT = 10000; - private static final int DEFAULT_FLUSH_INTERVAL = 5; + private static final int DEFAULT_CONNECT_TIMEOUT_MILLIS = 2000; + private static final int DEFAULT_SOCKET_TIMEOUT_MILLIS = 10000; + private static final int DEFAULT_FLUSH_INTERVAL_SECONDS = 5; private static final long DEFAULT_POLLING_INTERVAL_MILLIS = 1000L; private static final long DEFAULT_START_WAIT_MILLIS = 5000L; private static final int DEFAULT_SAMPLING_INTERVAL = 0; + private static final long DEFAULT_RECONNECT_TIME_MILLIS = 1000; - private static final Logger logger = LoggerFactory.getLogger(LDConfig.class); + private static final long MAX_HTTP_CACHE_SIZE_BYTES = 10 * 1024 * 1024; // 10 MB protected static final LDConfig DEFAULT = new Builder().build(); @@ -33,10 +47,12 @@ public final class LDConfig { final URI eventsURI; final URI streamURI; final int capacity; - final int connectTimeout; - final int socketTimeout; + final int connectTimeoutMillis; + final int socketTimeoutMillis; final int flushInterval; - final HttpHost proxyHost; + final Proxy proxy; + final Authenticator proxyAuthenticator; + final OkHttpClient httpClient; final boolean stream; final FeatureStore featureStore; final boolean useLdd; @@ -50,10 +66,11 @@ protected LDConfig(Builder builder) { this.baseURI = builder.baseURI; this.eventsURI = builder.eventsURI; this.capacity = builder.capacity; - this.connectTimeout = builder.connectTimeout; - this.socketTimeout = builder.socketTimeout; - this.flushInterval = builder.flushInterval; - this.proxyHost = builder.proxyHost(); + this.connectTimeoutMillis = builder.connectTimeoutMillis; + this.socketTimeoutMillis = builder.socketTimeoutMillis; + this.flushInterval = builder.flushIntervalSeconds; + this.proxy = builder.proxy(); + this.proxyAuthenticator = builder.proxyAuthenticator(); this.streamURI = builder.streamURI; this.stream = builder.stream; this.featureStore = builder.featureStore; @@ -66,17 +83,47 @@ protected LDConfig(Builder builder) { } this.startWaitMillis = builder.startWaitMillis; this.samplingInterval = builder.samplingInterval; - this.reconnectTimeMs = builder.reconnectTimeMs; + this.reconnectTimeMs = builder.reconnectTimeMillis; + + File cacheDir = Files.createTempDir(); + Cache cache = new Cache(cacheDir, MAX_HTTP_CACHE_SIZE_BYTES); + + OkHttpClient.Builder httpClientBuilder = new OkHttpClient.Builder() + .cache(cache) + .connectionPool(new ConnectionPool(5, 5, TimeUnit.MINUTES)) + .connectTimeout(connectTimeoutMillis, TimeUnit.MILLISECONDS) + .readTimeout(socketTimeoutMillis, TimeUnit.MILLISECONDS) + .writeTimeout(socketTimeoutMillis, TimeUnit.MILLISECONDS) + .retryOnConnectionFailure(true); + + if (proxy != null) { + httpClientBuilder.proxy(proxy); + if (proxyAuthenticator != null) { + httpClientBuilder.proxyAuthenticator(proxyAuthenticator); + logger.info("Using proxy: " + proxy + " with authentication."); + } else { + logger.info("Using proxy: " + proxy + " without authentication."); + } + } + + httpClient = httpClientBuilder + .build(); + } + + Request.Builder getRequestBuilder(String sdkKey) { + return new Request.Builder() + .addHeader("Authorization", sdkKey) + .addHeader("User-Agent", "JavaClient/" + LDClient.CLIENT_VERSION); } /** * A builder that helps construct {@link com.launchdarkly.client.LDConfig} objects. Builder * calls can be chained, enabling the following pattern: - * + *

*

    * LDConfig config = new LDConfig.Builder()
-   *      .connectTimeout(3)
-   *      .socketTimeout(3)
+   *      .connectTimeoutMillis(3)
+   *      .socketTimeoutMillis(3)
    *      .build()
    * 
*/ @@ -84,13 +131,14 @@ public static class Builder { private URI baseURI = DEFAULT_BASE_URI; private URI eventsURI = DEFAULT_EVENTS_URI; private URI streamURI = DEFAULT_STREAM_URI; - private int connectTimeout = DEFAULT_CONNECT_TIMEOUT; - private int socketTimeout = DEFAULT_SOCKET_TIMEOUT; + private int connectTimeoutMillis = DEFAULT_CONNECT_TIMEOUT_MILLIS; + private int socketTimeoutMillis = DEFAULT_SOCKET_TIMEOUT_MILLIS; private int capacity = DEFAULT_CAPACITY; - private int flushInterval = DEFAULT_FLUSH_INTERVAL; - private String proxyHost; + private int flushIntervalSeconds = DEFAULT_FLUSH_INTERVAL_SECONDS; + private String proxyHost = "localhost"; private int proxyPort = -1; - private String proxyScheme; + private String proxyUsername = null; + private String proxyPassword = null; private boolean stream = true; private boolean useLdd = false; private boolean offline = false; @@ -98,7 +146,7 @@ public static class Builder { private FeatureStore featureStore = new InMemoryFeatureStore(); private long startWaitMillis = DEFAULT_START_WAIT_MILLIS; private int samplingInterval = DEFAULT_SAMPLING_INTERVAL; - private long reconnectTimeMs = DEFAULT_RECONNECT_TIME_MILLIS; + private long reconnectTimeMillis = DEFAULT_RECONNECT_TIME_MILLIS; /** * Creates a builder with all configuration parameters set to the default @@ -108,6 +156,7 @@ public Builder() { /** * Set the base URL of the LaunchDarkly server for this configuration + * * @param baseURI the base URL of the LaunchDarkly server for this configuration * @return the builder */ @@ -118,6 +167,7 @@ public Builder baseURI(URI baseURI) { /** * Set the events URL of the LaunchDarkly server for this configuration + * * @param eventsURI the events URL of the LaunchDarkly server for this configuration * @return the builder */ @@ -128,6 +178,7 @@ public Builder eventsURI(URI eventsURI) { /** * Set the base URL of the LaunchDarkly streaming server for this configuration + * * @param streamURI the base URL of the LaunchDarkly streaming server * @return the builder */ @@ -143,6 +194,7 @@ public Builder featureStore(FeatureStore store) { /** * Set whether streaming mode should be enabled. By default, streaming is enabled. + * * @param stream whether streaming mode should be enabled * @return the builder */ @@ -154,56 +206,56 @@ public Builder stream(boolean stream) { /** * Set the connection timeout in seconds for the configuration. This is the time allowed for the underlying HTTP client to connect * to the LaunchDarkly server. The default is 2 seconds. - * + *

*

Both this method and {@link #connectTimeoutMillis(int) connectTimeoutMillis} affect the same property internally.

* * @param connectTimeout the connection timeout in seconds * @return the builder */ public Builder connectTimeout(int connectTimeout) { - this.connectTimeout = connectTimeout * 1000; + this.connectTimeoutMillis = connectTimeout * 1000; return this; } /** * Set the socket timeout in seconds for the configuration. This is the number of seconds between successive packets that the * client will tolerate before flagging an error. The default is 10 seconds. - * + *

*

Both this method and {@link #socketTimeoutMillis(int) socketTimeoutMillis} affect the same property internally.

* * @param socketTimeout the socket timeout in seconds * @return the builder */ public Builder socketTimeout(int socketTimeout) { - this.socketTimeout = socketTimeout * 1000; + this.socketTimeoutMillis = socketTimeout * 1000; return this; } /** * Set the connection timeout in milliseconds for the configuration. This is the time allowed for the underlying HTTP client to connect * to the LaunchDarkly server. The default is 2000 ms. - * - *

Both this method and {@link #connectTimeout(int) connectTimeout} affect the same property internally.

+ *

+ *

Both this method and {@link #connectTimeout(int) connectTimeoutMillis} affect the same property internally.

* * @param connectTimeoutMillis the connection timeout in milliseconds * @return the builder */ public Builder connectTimeoutMillis(int connectTimeoutMillis) { - this.connectTimeout = connectTimeoutMillis; + this.connectTimeoutMillis = connectTimeoutMillis; return this; } /** * Set the socket timeout in milliseconds for the configuration. This is the number of milliseconds between successive packets that the * client will tolerate before flagging an error. The default is 10,000 milliseconds. - * - *

Both this method and {@link #socketTimeout(int) socketTimeout} affect the same property internally.

+ *

+ *

Both this method and {@link #socketTimeout(int) socketTimeoutMillis} affect the same property internally.

* * @param socketTimeoutMillis the socket timeout in milliseconds * @return the builder */ public Builder socketTimeoutMillis(int socketTimeoutMillis) { - this.socketTimeout = socketTimeoutMillis; + this.socketTimeoutMillis = socketTimeoutMillis; return this; } @@ -215,7 +267,7 @@ public Builder socketTimeoutMillis(int socketTimeoutMillis) { * @return the builder */ public Builder flushInterval(int flushInterval) { - this.flushInterval = flushInterval; + this.flushIntervalSeconds = flushInterval; return this; } @@ -233,9 +285,9 @@ public Builder capacity(int capacity) { /** * Set the host to use as an HTTP proxy for making connections to LaunchDarkly. If this is not set, but - * {@link #proxyPort(int)} or {@link #proxyScheme(String)} are specified, this will default to localhost. + * {@link #proxyPort(int)} is specified, this will default to localhost. *

- * If none of {@link #proxyHost(String)}, {@link #proxyPort(int)} or {@link #proxyScheme(String)} are specified, + * If neither {@link #proxyHost(String)} nor {@link #proxyPort(int)} are specified, * a proxy will not be used, and {@link LDClient} will connect to LaunchDarkly directly. *

* @@ -248,13 +300,7 @@ public Builder proxyHost(String host) { } /** - * Set the port to use for an HTTP proxy for making connections to LaunchDarkly. If not set (but {@link #proxyHost(String)} - * or {@link #proxyScheme(String)} are specified, the default port for the scheme will be used. - *

- *

- * If none of {@link #proxyHost(String)}, {@link #proxyPort(int)} or {@link #proxyScheme(String)} are specified, - * a proxy will not be used, and {@link LDClient} will connect to LaunchDarkly directly. - *

+ * Set the port to use for an HTTP proxy for making connections to LaunchDarkly. This is required for proxied HTTP connections. * * @param port * @return the builder @@ -265,19 +311,37 @@ public Builder proxyPort(int port) { } /** - * Set the scheme to use for an HTTP proxy for making connections to LaunchDarkly. If not set (but {@link #proxyHost(String)} - * or {@link #proxyPort(int)} are specified, the default https scheme will be used. - *

- *

- * If none of {@link #proxyHost(String)}, {@link #proxyPort(int)} or {@link #proxyScheme(String)} are specified, - * a proxy will not be used, and {@link LDClient} will connect to LaunchDarkly directly. - *

+ * Sets the username for the optional HTTP proxy. Only used when {@link LDConfig.Builder#proxyPassword(String)} + * is also called. + * + * @param username + * @return the builder + */ + public Builder proxyUsername(String username) { + this.proxyUsername = username; + return this; + } + + /** + * Sets the password for the optional HTTP proxy. Only used when {@link LDConfig.Builder#proxyUsername(String)} + * is also called. * - * @param scheme + * @param password * @return the builder */ - public Builder proxyScheme(String scheme) { - this.proxyScheme = scheme; + public Builder proxyPassword(String password) { + this.proxyPassword = password; + return this; + } + + /** + * Deprecated. Only HTTP proxies are currently supported. + * + * @param unused + * @return the builder + */ + @Deprecated + public Builder proxyScheme(String unused) { return this; } @@ -321,7 +385,6 @@ public Builder pollingIntervalMillis(long pollingIntervalMillis) { * Setting this to 0 will not block and cause the constructor to return immediately. * Default value: 5000 * - * * @param startWaitMillis milliseconds to wait * @return the builder */ @@ -334,7 +397,7 @@ public Builder startWaitMillis(long startWaitMillis) { * Enable event sampling. When set to the default of zero, sampling is disabled and all events * are sent back to LaunchDarkly. When set to greater than zero, there is a 1 in * samplingInterval chance events will be will be sent. - * + *

*

Example: if you want 5% sampling rate, set samplingInterval to 20. * * @param samplingInterval the sampling interval. @@ -354,21 +417,38 @@ public Builder samplingInterval(int samplingInterval) { * @return the builder */ public Builder reconnectTimeMs(long reconnectTimeMs) { - this.reconnectTimeMs = reconnectTimeMs; + this.reconnectTimeMillis = reconnectTimeMs; return this; } - HttpHost proxyHost() { - if (this.proxyHost == null && this.proxyPort == -1 && this.proxyScheme == null) { + // returns null if none of the proxy bits were configured. Minimum required part: port. + Proxy proxy() { + if (this.proxyPort == -1) { return null; } else { - String hostname = this.proxyHost == null ? "localhost" : this.proxyHost; - String scheme = this.proxyScheme == null ? "https" : this.proxyScheme; - return new HttpHost(hostname, this.proxyPort, scheme); + return new Proxy(Proxy.Type.HTTP, new InetSocketAddress(proxyHost, proxyPort)); } } + Authenticator proxyAuthenticator() { + if (this.proxyUsername != null && this.proxyPassword != null) { + final String credential = Credentials.basic(proxyUsername, proxyPassword); + return new Authenticator() { + public Request authenticate(Route route, Response response) throws IOException { + if (response.request().header("Proxy-Authorization") != null) { + return null; // Give up, we've already failed to authenticate with the proxy. + } else { + return response.request().newBuilder() + .header("Proxy-Authorization", credential) + .build(); + } + } + }; + } + return null; + } + /** * Build the configured {@link com.launchdarkly.client.LDConfig} object * @@ -377,50 +457,5 @@ HttpHost proxyHost() { public LDConfig build() { return new LDConfig(this); } - - } - - private URIBuilder getBuilder() { - return new URIBuilder() - .setScheme(baseURI.getScheme()) - .setHost(baseURI.getHost()) - .setPort(baseURI.getPort()); - } - - private URIBuilder getEventsBuilder() { - return new URIBuilder() - .setScheme(eventsURI.getScheme()) - .setHost(eventsURI.getHost()) - .setPort(eventsURI.getPort()); - } - - HttpGet getRequest(String sdkKey, String path) { - URIBuilder builder = this.getBuilder().setPath(path); - - try { - HttpGet request = new HttpGet(builder.build()); - request.addHeader("Authorization", sdkKey); - request.addHeader("User-Agent", "JavaClient/" + LDClient.CLIENT_VERSION); - - return request; - } catch (Exception e) { - logger.error("Unhandled exception in LaunchDarkly client", e); - return null; - } - } - - HttpPost postEventsRequest(String sdkKey, String path) { - URIBuilder builder = this.getEventsBuilder().setPath(eventsURI.getPath() + path); - - try { - HttpPost request = new HttpPost(builder.build()); - request.addHeader("Authorization", sdkKey); - request.addHeader("User-Agent", "JavaClient/" + LDClient.CLIENT_VERSION); - - return request; - } catch (Exception e) { - logger.error("Unhandled exception in LaunchDarkly client", e); - return null; - } } } \ No newline at end of file diff --git a/src/main/java/com/launchdarkly/client/PollingProcessor.java b/src/main/java/com/launchdarkly/client/PollingProcessor.java index 134f955d7..a0cd8c707 100644 --- a/src/main/java/com/launchdarkly/client/PollingProcessor.java +++ b/src/main/java/com/launchdarkly/client/PollingProcessor.java @@ -1,5 +1,6 @@ package com.launchdarkly.client; +import com.google.common.util.concurrent.SettableFuture; import com.google.common.util.concurrent.ThreadFactoryBuilder; import org.slf4j.Logger; import org.slf4j.LoggerFactory; @@ -38,7 +39,7 @@ public void close() throws IOException { public Future start() { logger.info("Starting LaunchDarkly polling client with interval: " + config.pollingIntervalMillis + " milliseconds"); - final VeryBasicFuture initFuture = new VeryBasicFuture(); + final SettableFuture initFuture = SettableFuture.create(); ThreadFactory threadFactory = new ThreadFactoryBuilder() .setNameFormat("LaunchDarkly-PollingProcessor-%d") .build(); @@ -51,7 +52,7 @@ public void run() { store.init(requestor.getAllFlags()); if (!initialized.getAndSet(true)) { logger.info("Initialized LaunchDarkly client."); - initFuture.completed(null); + initFuture.set(null); } } catch (IOException e) { logger.error("Encountered exception in LaunchDarkly client when retrieving update", e); diff --git a/src/main/java/com/launchdarkly/client/RedisFeatureStore.java b/src/main/java/com/launchdarkly/client/RedisFeatureStore.java index 8694d403d..1a1186b5b 100644 --- a/src/main/java/com/launchdarkly/client/RedisFeatureStore.java +++ b/src/main/java/com/launchdarkly/client/RedisFeatureStore.java @@ -30,7 +30,6 @@ /** * A thread-safe, versioned store for {@link FeatureFlag} objects backed by Redis. Also * supports an optional in-memory cache configuration that can be used to improve performance. - * */ public class RedisFeatureStore implements FeatureStore { private static final Logger logger = LoggerFactory.getLogger(RedisFeatureStore.class); @@ -47,9 +46,9 @@ public class RedisFeatureStore implements FeatureStore { * Creates a new store instance that connects to Redis with the provided host, port, prefix, and cache timeout. Uses a default * connection pool configuration. * - * @param host the host for the Redis connection - * @param port the port for the Redis connection - * @param prefix a namespace prefix for all keys stored in Redis + * @param host the host for the Redis connection + * @param port the port for the Redis connection + * @param prefix a namespace prefix for all keys stored in Redis * @param cacheTimeSecs an optional timeout for the in-memory cache. If set to 0, no in-memory caching will be performed * @deprecated as of 1.1. Please use the {@link RedisFeatureStoreBuilder#build()} for a more flexible way of constructing a {@link RedisFeatureStore}. */ @@ -62,8 +61,8 @@ public RedisFeatureStore(String host, int port, String prefix, long cacheTimeSec * Creates a new store instance that connects to Redis with the provided URI, prefix, and cache timeout. Uses a default * connection pool configuration. * - * @param uri the URI for the Redis connection - * @param prefix a namespace prefix for all keys stored in Redis + * @param uri the URI for the Redis connection + * @param prefix a namespace prefix for all keys stored in Redis * @param cacheTimeSecs an optional timeout for the in-memory cache. If set to 0, no in-memory caching will be performed * @deprecated as of 1.1. Please use the {@link RedisFeatureStoreBuilder#build()} for a more flexible way of constructing a {@link RedisFeatureStore}. */ @@ -75,11 +74,11 @@ public RedisFeatureStore(URI uri, String prefix, long cacheTimeSecs) { /** * Creates a new store instance that connects to Redis with the provided host, port, prefix, cache timeout, and connection pool settings. * - * @param host the host for the Redis connection - * @param port the port for the Redis connection - * @param prefix a namespace prefix for all keys stored in Redis + * @param host the host for the Redis connection + * @param port the port for the Redis connection + * @param prefix a namespace prefix for all keys stored in Redis * @param cacheTimeSecs an optional timeout for the in-memory cache. If set to 0, no in-memory caching will be performed - * @param poolConfig an optional pool config for the Jedis connection pool + * @param poolConfig an optional pool config for the Jedis connection pool * @deprecated as of 1.1. Please use the {@link RedisFeatureStoreBuilder#build()} for a more flexible way of constructing a {@link RedisFeatureStore}. */ @Deprecated @@ -93,10 +92,10 @@ public RedisFeatureStore(String host, int port, String prefix, long cacheTimeSec /** * Creates a new store instance that connects to Redis with the provided URI, prefix, cache timeout, and connection pool settings. * - * @param uri the URI for the Redis connection - * @param prefix a namespace prefix for all keys stored in Redis + * @param uri the URI for the Redis connection + * @param prefix a namespace prefix for all keys stored in Redis * @param cacheTimeSecs an optional timeout for the in-memory cache. If set to 0, no in-memory caching will be performed - * @param poolConfig an optional pool config for the Jedis connection pool + * @param poolConfig an optional pool config for the Jedis connection pool * @deprecated as of 1.1. Please use the {@link RedisFeatureStoreBuilder#build()} for a more flexible way of constructing a {@link RedisFeatureStore}. */ @Deprecated @@ -109,7 +108,7 @@ public RedisFeatureStore(URI uri, String prefix, long cacheTimeSecs, JedisPoolCo /** * Creates a new store instance that connects to Redis based on the provided {@link RedisFeatureStoreBuilder}. - * + *

* See the {@link RedisFeatureStoreBuilder} for information on available configuration options and what they do. * * @param builder the configured builder to construct the store with. @@ -127,7 +126,6 @@ protected RedisFeatureStore(RedisFeatureStoreBuilder builder) { /** * Creates a new store instance that connects to Redis with a default connection (localhost port 6379) and no in-memory cache. - * */ public RedisFeatureStore() { pool = new JedisPool(getPoolConfig(), "localhost"); @@ -168,8 +166,9 @@ public Optional load(String key) throws Exception { /** * Configures the instance to use a "refresh after write" cache. This will not automatically evict stale values, allowing them to be returned if failures * occur when updating them. Optionally set the cache to refresh values asynchronously, which always returns the previously cached value immediately. + * * @param cacheTimeSecs the length of time in seconds, after a {@link FeatureFlag} value is created that it should be refreshed. - * @param asyncRefresh makes the refresh asynchronous or not. + * @param asyncRefresh makes the refresh asynchronous or not. */ private void createRefreshCache(long cacheTimeSecs, boolean asyncRefresh) { ThreadFactory threadFactory = new ThreadFactoryBuilder().setNameFormat(CACHE_REFRESH_THREAD_POOL_NAME_FORMAT).setDaemon(true).build(); @@ -184,6 +183,7 @@ private void createRefreshCache(long cacheTimeSecs, boolean asyncRefresh) { /** * Configures the instance to use an "expire after write" cache. This will evict stale values and block while loading the latest from Redis. + * * @param cacheTimeSecs the length of time in seconds, after a {@link FeatureFlag} value is created that it should be automatically removed. */ private void createExpiringCache(long cacheTimeSecs) { @@ -234,15 +234,16 @@ public FeatureFlag get(String key) { @Override public Map all() { try (Jedis jedis = pool.getResource()) { - Map featuresJson = jedis.hgetAll(featuresKey()); + Map featuresJson = jedis.hgetAll(featuresKey()); Map result = new HashMap<>(); Gson gson = new Gson(); - Type type = new TypeToken() {}.getType(); + Type type = new TypeToken() { + }.getType(); for (Map.Entry entry : featuresJson.entrySet()) { - FeatureFlag featureFlag = gson.fromJson(entry.getValue(), type); - if (!featureFlag.isDeleted()){ + FeatureFlag featureFlag = gson.fromJson(entry.getValue(), type); + if (!featureFlag.isDeleted()) { result.put(entry.getKey(), featureFlag); } } @@ -264,7 +265,7 @@ public void init(Map features) { t.del(featuresKey()); - for (FeatureFlag f: features.values()) { + for (FeatureFlag f : features.values()) { t.hset(featuresKey(), f.getKey(), gson.toJson(f)); } @@ -276,7 +277,7 @@ public void init(Map features) { * Deletes the feature associated with the specified key, if it exists and its version * is less than or equal to the specified version. * - * @param key the key of the feature to be deleted + * @param key the key of the feature to be deleted * @param version the version for the delete operation */ @Override @@ -287,7 +288,7 @@ public void delete(String key, int version) { jedis = pool.getResource(); jedis.watch(featuresKey()); - FeatureFlag feature = getRedis(key); + FeatureFlag feature = getRedis(key, jedis); if (feature != null && feature.getVersion() >= version) { logger.warn("Attempted to delete flag: " + key + " version: " + feature.getVersion() + @@ -303,8 +304,7 @@ public void delete(String key, int version) { if (cache != null) { cache.invalidate(key); } - } - finally { + } finally { if (jedis != null) { jedis.unwatch(); jedis.close(); @@ -327,7 +327,7 @@ public void upsert(String key, FeatureFlag feature) { Gson gson = new Gson(); jedis.watch(featuresKey()); - FeatureFlag f = getRedis(key); + FeatureFlag f = getRedis(key, jedis); if (f != null && f.getVersion() >= feature.getVersion()) { logger.warn("Attempted to update flag: " + key + " version: " + f.getVersion() + @@ -368,6 +368,7 @@ public boolean initialized() { /** * Releases all resources associated with the store. The store must no longer be used once closed. + * * @throws IOException */ public void close() throws IOException { @@ -404,24 +405,29 @@ private Boolean getInit() { } private FeatureFlag getRedis(String key) { - try (Jedis jedis = pool.getResource()){ - Gson gson = new Gson(); - String featureJson = jedis.hget(featuresKey(), key); + try (Jedis jedis = pool.getResource()) { + return getRedis(key, jedis); + } + } - if (featureJson == null) { - logger.debug("[get] Key: " + key + " not found in feature store. Returning null"); - return null; - } + private FeatureFlag getRedis(String key, Jedis jedis) { + Gson gson = new Gson(); + String featureJson = jedis.hget(featuresKey(), key); - Type type = new TypeToken() {}.getType(); - FeatureFlag f = gson.fromJson(featureJson, type); + if (featureJson == null) { + logger.debug("[get] Key: " + key + " not found in feature store. Returning null"); + return null; + } - if (f.isDeleted()) { - logger.debug("[get] Key: " + key + " has been deleted. Returning null"); - return null; - } - return f; + Type type = new TypeToken() { + }.getType(); + FeatureFlag f = gson.fromJson(featureJson, type); + + if (f.isDeleted()) { + logger.debug("[get] Key: " + key + " has been deleted. Returning null"); + return null; } + return f; } private static JedisPoolConfig getPoolConfig() { diff --git a/src/main/java/com/launchdarkly/client/RedisFeatureStoreBuilder.java b/src/main/java/com/launchdarkly/client/RedisFeatureStoreBuilder.java index 6eeb85de7..2d75b2686 100644 --- a/src/main/java/com/launchdarkly/client/RedisFeatureStoreBuilder.java +++ b/src/main/java/com/launchdarkly/client/RedisFeatureStoreBuilder.java @@ -1,6 +1,5 @@ package com.launchdarkly.client; -import org.apache.http.client.utils.URIBuilder; import org.slf4j.Logger; import org.slf4j.LoggerFactory; import redis.clients.jedis.JedisPoolConfig; @@ -56,7 +55,7 @@ public RedisFeatureStoreBuilder(URI uri, long cacheTimeSecs) { * @throws URISyntaxException */ public RedisFeatureStoreBuilder(String scheme, String host, int port, long cacheTimeSecs) throws URISyntaxException { - this.uri = new URIBuilder().setScheme(scheme).setHost(host).setPort(port).build(); + this.uri = new URI(scheme, null, host, port, null, null, null); this.cacheTimeSecs = cacheTimeSecs; } diff --git a/src/main/java/com/launchdarkly/client/StreamProcessor.java b/src/main/java/com/launchdarkly/client/StreamProcessor.java index a27d444db..e42a5c9cd 100644 --- a/src/main/java/com/launchdarkly/client/StreamProcessor.java +++ b/src/main/java/com/launchdarkly/client/StreamProcessor.java @@ -1,5 +1,6 @@ package com.launchdarkly.client; +import com.google.common.util.concurrent.SettableFuture; import com.google.common.util.concurrent.ThreadFactoryBuilder; import com.google.gson.Gson; import com.launchdarkly.eventsource.EventHandler; @@ -7,8 +8,6 @@ import com.launchdarkly.eventsource.MessageEvent; import com.launchdarkly.eventsource.ReadyState; import okhttp3.Headers; - -import org.apache.http.HttpHost; import org.joda.time.DateTime; import org.slf4j.Logger; import org.slf4j.LoggerFactory; @@ -51,7 +50,7 @@ class StreamProcessor implements UpdateProcessor { @Override public Future start() { - final VeryBasicFuture initFuture = new VeryBasicFuture(); + final SettableFuture initFuture = SettableFuture.create(); Headers headers = new Headers.Builder() .add("Authorization", this.sdkKey) @@ -77,7 +76,7 @@ public void onMessage(String name, MessageEvent event) throws Exception { case PUT: store.init(FeatureFlag.fromJsonMap(event.getData())); if (!initialized.getAndSet(true)) { - initFuture.completed(null); + initFuture.set(null); logger.info("Initialized LaunchDarkly client."); } break; @@ -95,7 +94,7 @@ public void onMessage(String name, MessageEvent event) throws Exception { try { store.init(requestor.getAllFlags()); if (!initialized.getAndSet(true)) { - initFuture.completed(null); + initFuture.set(null); logger.info("Initialized LaunchDarkly client."); } } catch (IOException e) { @@ -133,18 +132,12 @@ public void onError(Throwable throwable) { EventSource.Builder builder = new EventSource.Builder(handler, URI.create(config.streamURI.toASCIIString() + "/flags")) .headers(headers) .reconnectTimeMs(config.reconnectTimeMs); - if (config.proxyHost != null) { - int proxyPort = config.proxyHost.getPort(); - if (proxyPort == -1) { - String scheme = config.proxyHost.getSchemeName(); - if (scheme == "http") - proxyPort = 80; - else if (scheme == "https") - proxyPort = 443; - else - logger.error("Unknown proxy scheme: " + scheme); - } - builder.proxy(config.proxyHost.getHostName(), proxyPort); + + if (config.proxy != null) { + builder.proxy(config.proxy); + if (config.proxyAuthenticator != null) { + builder.proxyAuthenticator(config.proxyAuthenticator); + } } es = builder.build(); diff --git a/src/main/java/com/launchdarkly/client/Util.java b/src/main/java/com/launchdarkly/client/Util.java index 2de223022..c3a2643a2 100644 --- a/src/main/java/com/launchdarkly/client/Util.java +++ b/src/main/java/com/launchdarkly/client/Util.java @@ -1,12 +1,8 @@ package com.launchdarkly.client; import com.google.gson.JsonPrimitive; -import org.apache.http.HttpStatus; -import org.apache.http.client.methods.CloseableHttpResponse; -import org.apache.http.client.methods.HttpRequestBase; import org.joda.time.DateTime; import org.joda.time.DateTimeZone; -import org.slf4j.Logger; class Util { /** @@ -14,7 +10,7 @@ class Util { * @param maybeDate wraps either a nubmer or a string that may contain a valid timestamp. * @return null if input is not a valid format. */ - protected static DateTime jsonPrimitiveToDateTime(JsonPrimitive maybeDate) { + static DateTime jsonPrimitiveToDateTime(JsonPrimitive maybeDate) { if (maybeDate.isNumber()) { long millis = maybeDate.getAsLong(); return new DateTime(millis); @@ -28,25 +24,4 @@ protected static DateTime jsonPrimitiveToDateTime(JsonPrimitive maybeDate) { return null; } } - - /** - * Logs an error if the response is not 2xx and returns false. Otherwise returns true. - * - * @param logger for logging responses. - * @param request http request - * @param response http response - * @return whether or not the response's status code is acceptable. - */ - static boolean handleResponse(Logger logger, HttpRequestBase request, CloseableHttpResponse response) { - int statusCode = response.getStatusLine().getStatusCode(); - if (statusCode / 100 == 2) { - return true; - } - if (statusCode == HttpStatus.SC_UNAUTHORIZED) { - logger.error("[401] Invalid SDK key when accessing URI: " + request.getURI().toString()); - } else { - logger.error("[" + statusCode + "] " + response.getStatusLine().getReasonPhrase() + " When accessing URI: " + request.getURI().toString()); - } - return false; - } } diff --git a/src/main/java/com/launchdarkly/client/VeryBasicFuture.java b/src/main/java/com/launchdarkly/client/VeryBasicFuture.java deleted file mode 100644 index 6bcab2811..000000000 --- a/src/main/java/com/launchdarkly/client/VeryBasicFuture.java +++ /dev/null @@ -1,30 +0,0 @@ -package com.launchdarkly.client; - -import org.apache.http.concurrent.BasicFuture; -import org.apache.http.concurrent.FutureCallback; - -import java.util.concurrent.Future; - -/** - * Very Basic {@link Future} implementation extending {@link BasicFuture} with no callback or return value. - */ -public class VeryBasicFuture extends BasicFuture { - - public VeryBasicFuture() { - super(new NoOpFutureCallback()); - } - - static class NoOpFutureCallback implements FutureCallback { - @Override - public void completed(Void result) { - } - - @Override - public void failed(Exception ex) { - } - - @Override - public void cancelled() { - } - } -} diff --git a/src/test/java/com/launchdarkly/client/LDConfigTest.java b/src/test/java/com/launchdarkly/client/LDConfigTest.java index d5078750c..4bc9857d6 100644 --- a/src/test/java/com/launchdarkly/client/LDConfigTest.java +++ b/src/test/java/com/launchdarkly/client/LDConfigTest.java @@ -1,11 +1,12 @@ package com.launchdarkly.client; -import org.apache.http.client.methods.HttpPost; import org.junit.Test; -import java.net.URI; +import java.net.InetSocketAddress; +import java.net.Proxy; import static org.junit.Assert.assertEquals; +import static org.junit.Assert.assertNotNull; import static org.junit.Assert.assertNull; public class LDConfigTest { @@ -13,92 +14,85 @@ public class LDConfigTest { public void testConnectTimeoutSpecifiedInSeconds() { LDConfig config = new LDConfig.Builder().connectTimeout(3).build(); - assertEquals(3000, config.connectTimeout); + assertEquals(3000, config.connectTimeoutMillis); } @Test public void testConnectTimeoutSpecifiedInMilliseconds() { LDConfig config = new LDConfig.Builder().connectTimeoutMillis(3000).build(); - assertEquals(3000, config.connectTimeout); + assertEquals(3000, config.connectTimeoutMillis); } @Test public void testSocketTimeoutSpecifiedInSeconds() { LDConfig config = new LDConfig.Builder().socketTimeout(3).build(); - assertEquals(3000, config.socketTimeout); + assertEquals(3000, config.socketTimeoutMillis); } @Test public void testSocketTimeoutSpecifiedInMilliseconds() { LDConfig config = new LDConfig.Builder().socketTimeoutMillis(3000).build(); - assertEquals(3000, config.socketTimeout); + assertEquals(3000, config.socketTimeoutMillis); } @Test public void testNoProxyConfigured() { LDConfig config = new LDConfig.Builder().build(); - - assertNull(config.proxyHost); + assertNull(config.proxy); + assertNull(config.proxyAuthenticator); } @Test - public void testOnlyProxyPortConfiguredHasPort() { - LDConfig config = new LDConfig.Builder().proxyPort(1234).build(); - - assertEquals(1234, config.proxyHost.getPort()); + public void testOnlyProxyHostConfiguredIsNull() { + LDConfig config = new LDConfig.Builder().proxyHost("bla").build(); + assertNull(config.proxy); } @Test - public void testOnlyProxyPortConfiguredHasDefaultHost() { + public void testOnlyProxyPortConfiguredHasPortAndDefaultHost() { LDConfig config = new LDConfig.Builder().proxyPort(1234).build(); - - assertEquals("localhost", config.proxyHost.getHostName()); - } - - @Test - public void testOnlyProxyPortConfiguredHasDefaultScheme() { - LDConfig config = new LDConfig.Builder().proxyPort(1234).build(); - - assertEquals("https", config.proxyHost.getSchemeName()); - } - - @Test - public void testOnlyProxyHostConfiguredHasDefaultPort() { - LDConfig config = new LDConfig.Builder().proxyHost("myproxy.example.com").build(); - - assertEquals(-1, config.proxyHost.getPort()); + assertEquals(new Proxy(Proxy.Type.HTTP, new InetSocketAddress("localhost", 1234)), config.proxy); } - @Test - public void testProxyHostConfiguredHasHost() { - LDConfig config = new LDConfig.Builder().proxyHost("myproxy.example.com").build(); - - assertEquals("myproxy.example.com", config.proxyHost.getHostName()); + public void testProxy() { + LDConfig config = new LDConfig.Builder() + .proxyHost("localhost2") + .proxyPort(4444) + .build(); + assertEquals(new Proxy(Proxy.Type.HTTP, new InetSocketAddress("localhost2", 4444)), config.proxy); } @Test - public void testProxySchemeConfiguredHasScheme() { - LDConfig config = new LDConfig.Builder().proxyScheme("http").build(); - - assertEquals("http", config.proxyHost.getSchemeName()); + public void testProxyAuth() { + LDConfig config = new LDConfig.Builder() + .proxyHost("localhost2") + .proxyPort(4444) + .proxyUsername("proxyUser") + .proxyPassword("proxyPassword") + .build(); + assertNotNull(config.proxy); + assertNotNull(config.proxyAuthenticator); } @Test - public void testDefaultEventsUriIsConstructedProperly(){ - LDConfig config = new LDConfig.Builder().build(); - - HttpPost post = config.postEventsRequest("dummy-api-key", "/bulk"); - assertEquals("https://events.launchdarkly.com/bulk", post.getURI().toString()); - } - - @Test - public void testCustomEventsUriIsConstructedProperly(){ - LDConfig config = new LDConfig.Builder().eventsURI(URI.create("http://localhost:3000/api/events")).build(); - - HttpPost post = config.postEventsRequest("dummy-api-key", "/bulk"); - assertEquals("http://localhost:3000/api/events/bulk", post.getURI().toString()); + public void testProxyAuthPartialConfig() { + LDConfig config = new LDConfig.Builder() + .proxyHost("localhost2") + .proxyPort(4444) + .proxyUsername("proxyUser") + .build(); + assertNotNull(config.proxy); + assertNull(config.proxyAuthenticator); + + config = new LDConfig.Builder() + .proxyHost("localhost2") + .proxyPort(4444) + .proxyPassword("proxyPassword") + .build(); + assertNotNull(config.proxy); + assertNull(config.proxyAuthenticator); } @Test