/**
* diqube: Distributed Query Base.
*
* Copyright (C) 2015 Bastian Gloeckle
*
* This file is part of diqube.
*
* diqube is free software: you can redistribute it and/or modify
* it under the terms of the GNU Affero General Public License as
* published by the Free Software Foundation, either version 3 of the
* License, or (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU Affero General Public License for more details.
*
* You should have received a copy of the GNU Affero General Public License
* along with this program. If not, see <http://www.gnu.org/licenses/>.
*/
package org.diqube.config;
import java.io.FileInputStream;
import java.io.IOException;
import java.io.InputStream;
import java.io.InputStreamReader;
import java.nio.charset.Charset;
import java.util.Collections;
import java.util.HashMap;
import java.util.Map;
import java.util.Properties;
import javax.annotation.PostConstruct;
import org.diqube.config.derived.AbstractDerivedConfigValueProvider;
import org.diqube.config.derived.DerivedConsensusDataDirConfigValueProvider;
import org.diqube.config.derived.DerivedDataDirConfigValueProvider;
import org.diqube.config.derived.DerivedFlattenDiskCacheLocationConfigValueProvider;
import org.diqube.config.derived.DerivedInternalDbDirConfigValueProvider;
import org.diqube.context.AutoInstatiate;
import org.diqube.context.Profiles;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.context.annotation.Profile;
/**
* Loads configuration values from both, a user supplied file and a file on the classpath which contains the default
* values - this class then provides the active configuration values.
*
* @author Bastian Gloeckle
*/
@AutoInstatiate
@Profile(Profiles.CONFIG)
public class ConfigurationManager {
private static final Logger logger = LoggerFactory.getLogger(ConfigurationManager.class);
public static final String TEST_CONFIG_CLASSPATH_FILENAME = "/server-test.properties";
public static final String DEFAULT_CONFIG_CLASSPATH_FILENAME = "/server.properties";
public static final String CUSTOM_PROPERTIES_SYSTEM_PROPERTY = "diqube.properties";
private Map<String, AbstractDerivedConfigValueProvider> derivedValueProviders;
private Properties defaultProperties;
private Properties customProperties;
@PostConstruct
private void initialize() {
derivedValueProviders = new HashMap<>();
derivedValueProviders.put(DerivedConfigKey.FINAL_DATA_DIR, new DerivedDataDirConfigValueProvider());
derivedValueProviders.put(DerivedConfigKey.FINAL_INTERNAL_DB_DIR, new DerivedInternalDbDirConfigValueProvider());
derivedValueProviders.put(DerivedConfigKey.FINAL_CONSENSUS_DATA_DIR,
new DerivedConsensusDataDirConfigValueProvider());
derivedValueProviders.put(DerivedConfigKey.FINAL_FLATTEN_DISK_CACHE_LOCATION,
new DerivedFlattenDiskCacheLocationConfigValueProvider());
derivedValueProviders = Collections.unmodifiableMap(derivedValueProviders);
for (AbstractDerivedConfigValueProvider provider : derivedValueProviders.values())
provider.initialize(derivedValueProviders, this);
String customFileName = System.getProperty(CUSTOM_PROPERTIES_SYSTEM_PROPERTY);
if (customFileName != null) {
customProperties = new Properties();
try {
customProperties.load(new InputStreamReader(new FileInputStream(customFileName), Charset.forName("UTF-8")));
// TODO validate that properties have valid names!
logger.info("Loaded custom properties from {}", customFileName);
} catch (IOException e) {
throw new RuntimeException("Could not load custom properties from " + customFileName);
}
} else {
customProperties = null;
logger.info(
"Did not load custom properties. Use system property '{}' to point to a file containing custom properties.",
CUSTOM_PROPERTIES_SYSTEM_PROPERTY);
}
defaultProperties = new Properties();
try {
String classpathFile = TEST_CONFIG_CLASSPATH_FILENAME;
InputStream classpathStream = this.getClass().getResourceAsStream(classpathFile);
if (classpathStream == null) {
classpathFile = DEFAULT_CONFIG_CLASSPATH_FILENAME;
classpathStream = this.getClass().getResourceAsStream(classpathFile);
}
logger.info("Loading default (fallback) config from {}", classpathFile);
defaultProperties.load(new InputStreamReader(classpathStream, Charset.forName("UTF-8")));
} catch (IOException e) {
throw new RuntimeException("Could not load default config.");
}
}
/**
* @return The value of the given config key (see {@link ConfigKey}), either the one provided by the user or the
* default one.
*/
public String getValue(String configKey) {
if (derivedValueProviders.containsKey(configKey))
return derivedValueProviders.get(configKey).derive();
if (customProperties != null && customProperties.getProperty(configKey) != null)
return customProperties.getProperty(configKey);
return defaultProperties.getProperty(configKey);
}
}