io.castle.client.internal.config.ConfigurationLoader Maven / Gradle / Ivy
Show all versions of castle-java Show documentation
package io.castle.client.internal.config;
import com.google.common.base.Splitter;
import io.castle.client.Castle;
import io.castle.client.internal.backend.CastleBackendProvider;
import io.castle.client.model.AuthenticateAction;
import io.castle.client.model.AuthenticateFailoverStrategy;
import io.castle.client.model.CastleSdkConfigurationException;
import java.io.InputStream;
import java.net.URL;
import java.util.Properties;
/**
* Loads the SDK configuration from the environment or classpath, granting precedence to the environment.
*
* The following criteria are used when loading a configuration value:
*
* - environment values take precedence over values written in properties files;
*
- if no environment value is provided, then a value from classpath properties file is taken;
*
- If neither an environment variable nor a value in a properties file is provided, default values are set for some
* fields in the resulting {@link CastleConfiguration}.
*
* It is necessary to at least provide values for the API Secret, since it does not have a
* default value.
*
* Properties are loaded from file {@code castle_sdk.properties} by default.
* In order to specify a different file, set the {@code CASTLE_PROPERTIES_FILE} environmental variable.
*/
class ConfigurationLoader {
/**
* An instance of Properties that can be used instead of environmental variables or a properties file.
*/
private final Properties castleConfigurationProperties;
/**
* Creates a configurationLoader that will load a castleConfiguration from an instance of Properties.
*
* @param properties properties containing the values that will be passed to the castleConfiguration loaded.
*/
ConfigurationLoader(Properties properties) {
this.castleConfigurationProperties = properties;
}
/**
* Creates a configurationLoader that will load a castleConfiguration from a properties file.
*
* A properties file in the classpath will be loaded if the {@code CASTLE_PROPERTIES_FILE} is set to its name.
* If this variable is not set, then the file named {@code castle_sdk.properties} in the classpath will be loaded,
* provided that it exists.
*/
ConfigurationLoader() {
this(loadPropertiesFile());
}
private static Properties loadPropertiesFile() {
String propertyFile = getPropertiesFilePath();
Properties loaded = new Properties();
URL configFile = Castle.class.getClassLoader().getResource(propertyFile);
if (configFile != null) {
InputStream resourceAsStream = Castle.class.getClassLoader().getResourceAsStream(propertyFile);
loaded = new PropertiesReader().loadPropertiesFromStream(loaded, resourceAsStream);
}
return loaded;
}
private static String getPropertiesFilePath() {
String propertiesFile = System.getenv("CASTLE_PROPERTIES_FILE");
if (propertiesFile != null) {
return propertiesFile;
} else {
return "castle_sdk.properties";
}
}
/**
* Loads the application level configuration for the Castle SDK with values taken from environmental variables,
* a properties file or default values in that order of precedence.
*
* When wrong failover strategy or backend, null is returned instead.
*
* @return a CastleConfiguration instance
* @throws CastleSdkConfigurationException if at least one of apiSecret or castleAppId is not provided in either the
* environment or the properties file in the classpath
* @throws NumberFormatException if the provided timeout environmental variable or property is not
* parsable into an int
*/
public CastleConfiguration loadConfiguration() throws CastleSdkConfigurationException, NumberFormatException {
CastleConfigurationBuilder builder = loadConfigurationBuilder();
return builder.build();
}
public CastleConfigurationBuilder loadConfigurationBuilder() {
String envApiSecret = loadConfigurationValue(
castleConfigurationProperties,
"api_secret",
"CASTLE_API_SECRET"
);
// Legacy env name for secret
if (envApiSecret == null) {
envApiSecret = loadConfigurationValue(
castleConfigurationProperties,
"api_secret",
"CASTLE_SDK_API_SECRET"
);
}
String castleAppId = loadConfigurationValue(
castleConfigurationProperties,
"app_id",
"CASTLE_SDK_APP_ID"
);
String allowListValue = loadConfigurationValue(
castleConfigurationProperties,
"allow_list",
"CASTLE_SDK_ALLOWLIST_HEADERS"
);
String denyListValue = loadConfigurationValue(
castleConfigurationProperties,
"deny_list",
"CASTLE_SDK_DENYLIST_HEADERS"
);
String timeoutValue = loadConfigurationValue(
castleConfigurationProperties,
"timeout",
"CASTLE_SDK_TIMEOUT"
);
String backendProviderValue = loadConfigurationValue(
castleConfigurationProperties,
"backend_provider",
"CASTLE_SDK_BACKEND_PROVIDER"
);
String authenticateFailoverStrategyValue = loadConfigurationValue(
castleConfigurationProperties,
"failover_strategy",
"CASTLE_SDK_AUTHENTICATE_FAILOVER_STRATEGY"
);
String apiBaseUrl = loadConfigurationValue(
castleConfigurationProperties,
"base_url",
"CASTLE_SDK_BASE_URL"
);
String logHttpRequests = loadConfigurationValue(
castleConfigurationProperties,
"log_http",
"CASTLE_SDK_LOG_HTTP"
);
String ipHeadersValue = loadConfigurationValue(
castleConfigurationProperties,
"ip_headers",
"CASTLE_SDK_IP_HEADERS"
);
CastleConfigurationBuilder builder = CastleConfigurationBuilder
.defaultConfigBuilder()
.withApiSecret(envApiSecret)
.withCastleAppId(castleAppId);
if (apiBaseUrl != null) {
builder.withApiBaseUrl(apiBaseUrl);
} else {
builder.withDefaultApiBaseUrl();
}
if (allowListValue != null) {
builder.withAllowListHeaders(Splitter.on(",").splitToList(allowListValue));
}
if (denyListValue != null) {
builder.withDenyListHeaders(Splitter.on(",").splitToList(denyListValue));
}
if (timeoutValue != null) {
// might throw NumberFormatException if string is not parsable to int
int timeout = Integer.parseInt(timeoutValue);
builder.withTimeout(timeout);
}
if (authenticateFailoverStrategyValue != null) {
if (authenticateFailoverStrategyValue.compareTo("throw") == 0) {
builder.withAuthenticateFailoverStrategy(new AuthenticateFailoverStrategy());
} else {
builder.withAuthenticateFailoverStrategy(
new AuthenticateFailoverStrategy(
AuthenticateAction.fromAction(authenticateFailoverStrategyValue)
)
);
}
} else {
builder.withDefaultAuthenticateFailoverStrategy();
}
if (backendProviderValue != null) {
builder.withBackendProvider(
CastleBackendProvider.valueOf(backendProviderValue)
);
}
if (logHttpRequests != null) {
builder.withLogHttpRequests(Boolean.valueOf(logHttpRequests));
}
if (ipHeadersValue != null) {
builder.withIPHeaders(Splitter.on(",").splitToList(ipHeadersValue));
}
return builder;
}
private String loadConfigurationValue(Properties properties, String propertyName, String environmentName) {
String getenv = System.getenv(environmentName);
if (getenv == null) {
getenv = properties.getProperty(propertyName);
}
return getenv;
}
}