alluxio.conf.InstancedConfiguration Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of alluxio-core-common Show documentation
Show all versions of alluxio-core-common Show documentation
Common utilities shared in Alluxio core modules
/*
* The Alluxio Open Foundation licenses this work under the Apache License, version 2.0
* (the "License"). You may not use this work except in compliance with the License, which is
* available at www.apache.org/licenses/LICENSE-2.0
*
* This software is distributed on an "AS IS" basis, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND,
* either express or implied, as more fully set forth in the License.
*
* See the NOTICE file distributed with this work for information regarding copyright ownership.
*/
package alluxio.conf;
import static alluxio.conf.PropertyKey.CONF_REGEX;
import static alluxio.conf.PropertyKey.REGEX_STRING;
import static com.google.common.base.Preconditions.checkArgument;
import static com.google.common.base.Preconditions.checkState;
import static java.lang.String.format;
import alluxio.exception.ExceptionMessage;
import alluxio.util.ConfigurationUtils;
import alluxio.util.FormatUtils;
import com.google.common.base.Preconditions;
import com.google.common.collect.Maps;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import java.time.Duration;
import java.util.HashMap;
import java.util.HashSet;
import java.util.List;
import java.util.Map;
import java.util.Properties;
import java.util.Set;
import java.util.regex.Matcher;
import javax.annotation.Nonnull;
/**
* Alluxio configuration.
* WARNING: This API is not intended to be used outside of internal Alluxio code and may be
* changed or removed in a future minor release.
*
* Application code should use APIs {@link Configuration}.
*
*/
public class InstancedConfiguration implements AlluxioConfiguration {
private static final Logger LOG = LoggerFactory.getLogger(InstancedConfiguration.class);
/** Source of the truth of all property values (default or customized). */
protected final AlluxioProperties mProperties;
private final boolean mClusterDefaultsLoaded;
/**
* Creates a new instance of {@link InstancedConfiguration}.
*
* WARNING: This API is not intended to be used outside of internal Alluxio code and may be
* changed or removed in a future minor release.
*
* Application code should use {@link Configuration#global()}.
*
* @param properties alluxio properties underlying this configuration
*/
public InstancedConfiguration(AlluxioProperties properties) {
this(properties, false);
}
/**
* Creates a new instance of {@link InstancedConfiguration}.
*
* WARNING: This API is not intended to be used outside of internal Alluxio code and may be
* changed or removed in a future minor release.
*
* Application code should use {@link Configuration#global()}.
*
* @param properties alluxio properties underlying this configuration
* @param clusterDefaultsLoaded Whether or not the properties represent the cluster defaults
*/
public InstancedConfiguration(AlluxioProperties properties, boolean clusterDefaultsLoaded) {
mProperties = properties;
mClusterDefaultsLoaded = clusterDefaultsLoaded;
}
/**
* Return reference to mProperties.
* @return mProperties
*/
public AlluxioProperties getProperties() {
return mProperties;
}
@Override
public AlluxioProperties copyProperties() {
return mProperties.copy();
}
@Override
public Object get(PropertyKey key) {
return get(key, ConfigurationValueOptions.defaults());
}
@Override
public Object get(PropertyKey key, ConfigurationValueOptions options) {
Object value = mProperties.get(key);
if (value == null) {
// if value or default value is not set in configuration for the given key
throw new RuntimeException(ExceptionMessage.UNDEFINED_CONFIGURATION_KEY.getMessage(key));
}
if (!(value instanceof String)) {
return value;
}
if (!options.shouldUseRawValue()) {
try {
value = lookup(key, (String) value);
} catch (UnresolvablePropertyException e) {
throw new RuntimeException("Could not resolve key \""
+ key.getName() + "\": " + e.getMessage(), e);
}
}
if (options.shouldUseDisplayValue()) {
PropertyKey.DisplayType displayType = key.getDisplayType();
switch (displayType) {
case DEFAULT:
break;
case CREDENTIALS:
value = "******";
break;
default:
throw new IllegalStateException(String.format("Invalid displayType %s for property %s",
displayType.name(), key.getName()));
}
}
return value;
}
private boolean isResolvable(PropertyKey key) {
Object value = mProperties.get(key);
// null values are unresolvable
if (value == null) {
return false;
}
try {
// Lookup to resolve any key before simply returning isSet. An exception will be thrown if
// the key can't be resolved or if a lower level value isn't set.
if (value instanceof String) {
lookup(key, (String) value);
}
} catch (UnresolvablePropertyException e) {
return false;
}
return true;
}
@Override
public boolean isSet(PropertyKey key) {
return mProperties.isSet(key) && isResolvable(key);
}
@Override
public boolean isSetByUser(PropertyKey key) {
return mProperties.isSetByUser(key) && isResolvable(key);
}
/**
* Sets the value for the appropriate key in the {@link Properties}.
*
* @param key the key to set
* @param value the value for the key
*/
public void set(PropertyKey key, Object value) {
set(key, value, Source.RUNTIME);
}
/**
* Sets the value for the appropriate key in the {@link Properties}.
*
* @param key the key to set
* @param value the value for the key
*
* @deprecated API to aid property key type transition
*/
@java.lang.Deprecated
public void set(@Nonnull PropertyKey key, @Nonnull String value) {
checkArgument(!value.equals(""),
"The key \"%s\" cannot be have an empty string as a value. Use "
+ "Configuration.unset to remove a key from the configuration.", key);
if (key.validateValue(value)) {
mProperties.put(key, key.formatValue(value), Source.RUNTIME);
} else {
if (key.getType() == PropertyKey.PropertyType.STRING) {
throw new IllegalArgumentException(
format("Invalid value for property key %s: %s", key, value));
}
LOG.warn("The value {} for property key {} is invalid. PropertyKey are now typed "
+ "and require values to be properly typed. Invalid PropertyKey values will not be "
+ "accepted in 3.0", value, key);
mProperties.put(key, key.parseValue(value), Source.RUNTIME);
}
}
/**
* Sets the value for the appropriate key in the {@link Properties} by source.
*
* @param key the key to set
* @param value the value for the key
* @param source the source of the the properties (e.g., system property, default and etc)
*/
public void set(@Nonnull PropertyKey key, @Nonnull Object value, @Nonnull Source source) {
checkArgument(!value.equals(""),
"The key \"%s\" cannot be have an empty string as a value. Use "
+ "Configuration.unset to remove a key from the configuration.", key);
checkArgument(key.validateValue(value),
"Invalid value for property key %s: %s", key, value);
value = key.formatValue(value);
mProperties.put(key, value, source);
}
/**
* Unsets the value for the appropriate key in the {@link Properties}. If the {@link PropertyKey}
* has a default value, it will still be considered set after executing this method.
*
* @param key the key to unset
*/
public void unset(PropertyKey key) {
Preconditions.checkNotNull(key, "key");
mProperties.remove(key);
}
/**
* Merges map of properties into the current alluxio properties.
*
* @param properties map of keys to values
* @param source the source type for these properties
*/
public void merge(Map, ?> properties, Source source) {
mProperties.merge(properties, source);
}
/**
* Merges map of properties into the current alluxio properties.
*
* @param properties map of keys to values
*/
public void merge(AlluxioProperties properties) {
mProperties.merge(properties);
}
@Override
public Set keySet() {
return mProperties.keySet();
}
@Override
public Set userKeySet() {
return mProperties.userKeySet();
}
@Override
public String getString(PropertyKey key) {
if (key.getType() != PropertyKey.PropertyType.STRING) {
LOG.warn("PropertyKey {}'s type is {}, please use proper getter method for the type, "
+ "getString will no longer work for non-STRING property types in 3.0",
key, key.getType());
}
Object value = get(key);
if (value instanceof String) {
return (String) value;
}
return value.toString();
}
@Override
public int getInt(PropertyKey key)
{
checkArgument(key.getType() == PropertyKey.PropertyType.INTEGER);
return (int) get(key);
}
@Override
public long getLong(PropertyKey key) {
// Low-precision types int can be implicitly converted to high-precision types long
// without loss of precision
checkArgument(key.getType() == PropertyKey.PropertyType.LONG
|| key.getType() == PropertyKey.PropertyType.INTEGER);
return ((Number) get(key)).longValue();
}
@Override
public double getDouble(PropertyKey key) {
checkArgument(key.getType() == PropertyKey.PropertyType.DOUBLE);
return (double) get(key);
}
@Override
public boolean getBoolean(PropertyKey key) {
checkArgument(key.getType() == PropertyKey.PropertyType.BOOLEAN);
return (boolean) get(key);
}
@Override
public List getList(PropertyKey key) {
checkArgument(key.getType() == PropertyKey.PropertyType.LIST);
String value = (String) get(key);
return ConfigurationUtils.parseAsList(value, key.getDelimiter());
}
@Override
public > T getEnum(PropertyKey key, Class enumType) {
checkArgument(key.getEnumType().equals(enumType), "PropertyKey %s is not of enum type", key);
return enumType.cast(get(key));
}
@Override
public long getBytes(PropertyKey key) {
checkArgument(key.getType() == PropertyKey.PropertyType.DATASIZE);
return FormatUtils.parseSpaceSize((String) get(key));
}
@Override
public long getMs(PropertyKey key) {
checkArgument(key.getType() == PropertyKey.PropertyType.DURATION);
return FormatUtils.parseTimeSize((String) get(key));
}
@Override
public Duration getDuration(PropertyKey key) {
return Duration.ofMillis(getMs(key));
}
@Override
public Class getClass(PropertyKey key) {
Object value = get(key);
if (value instanceof Class) {
return (Class) value;
}
try {
return (Class) Class.forName((String) value);
} catch (ClassNotFoundException e) {
throw new IllegalStateException(
format("Requested class %s can not be loaded", value));
}
}
@Override
public Map getNestedProperties(PropertyKey prefixKey) {
Map ret = Maps.newHashMap();
for (Map.Entry entry: mProperties.entrySet()) {
String key = entry.getKey().getName();
if (prefixKey.isNested(key)) {
String suffixKey = key.substring(prefixKey.length() + 1);
ret.put(suffixKey, entry.getValue());
}
}
return ret;
}
@Override
public Source getSource(PropertyKey key) {
return mProperties.getSource(key);
}
@Override
public Map toMap(ConfigurationValueOptions opts) {
Map map = new HashMap<>();
// Cannot use Collectors.toMap because we support null keys.
keySet().forEach(key -> map.put(key.getName(), getOrDefault(key, null, opts)));
return map;
}
@Override
public void validate() {
if (!getBoolean(PropertyKey.CONF_VALIDATION_ENABLED)) {
return;
}
for (PropertyKey key : keySet()) {
checkState(
getSource(key).getType() != Source.Type.SITE_PROPERTY || !key.isIgnoredSiteProperty(),
"%s is not accepted in alluxio-site.properties, "
+ "and must be specified as a JVM property. "
+ "If no JVM property is present, Alluxio will use default value '%s'.",
key.getName(), key.getDefaultValue());
if (PropertyKey.isDeprecated(key) && isSetByUser(key)) {
LOG.warn("{} is deprecated. Please avoid using this key in the future. {}", key.getName(),
PropertyKey.getDeprecationMessage(key));
}
}
checkTimeouts();
checkUserFileBufferBytes();
checkZkConfiguration();
checkCheckpointZipConfig();
}
@Override
public boolean clusterDefaultsLoaded() {
return mClusterDefaultsLoaded;
}
@Override
public String hash() {
return mProperties.hash();
}
/**
* Lookup key names to handle ${key} stuff.
*
* @param base the String to look for
* @return resolved String value
*/
protected Object lookup(PropertyKey key, final String base) throws UnresolvablePropertyException {
return lookupRecursively(key, base, new HashSet<>());
}
/**
* Actual recursive lookup replacement.
*
* @param base the string to resolve
* @param seen strings already seen during this lookup, used to prevent unbound recursion
* @return the resolved string
*/
protected Object lookupRecursively(PropertyKey originalKey, String base, Set seen)
throws UnresolvablePropertyException {
// check argument
if (base == null) {
throw new UnresolvablePropertyException("Can't resolve property with null value");
}
String resolved = base;
Object resolvedValue = null;
PropertyKey key = null;
// Lets find pattern match to ${key}.
// TODO(hsaputra): Consider using Apache Commons StrSubstitutor.
Matcher matcher = CONF_REGEX.matcher(base);
while (matcher.find()) {
String match = matcher.group(2).trim();
if (!seen.add(match)) {
throw new RuntimeException(ExceptionMessage.KEY_CIRCULAR_DEPENDENCY.getMessage(match));
}
if (!PropertyKey.isValid(match)) {
throw new RuntimeException(ExceptionMessage.INVALID_CONFIGURATION_KEY.getMessage(match));
}
key = PropertyKey.fromString(match);
Object value = mProperties.get(key);
String stringValue = null;
if (value instanceof String) {
stringValue = String.valueOf(lookupRecursively(key, (String) value , seen));
}
else if (value != null) {
stringValue = String.valueOf(value);
}
seen.remove(match);
if (stringValue == null) {
throw new UnresolvablePropertyException(ExceptionMessage
.UNDEFINED_CONFIGURATION_KEY.getMessage(match));
}
resolved = resolved.replaceFirst(REGEX_STRING, Matcher.quoteReplacement(stringValue));
}
if (key != null) {
resolvedValue = originalKey.parseValue(resolved);
} else {
resolvedValue = resolved;
}
return resolvedValue;
}
/**
* Validates timeout related configuration.
*/
private void checkTimeouts() {
long waitTime = getMs(PropertyKey.MASTER_WORKER_CONNECT_WAIT_TIME);
long retryInterval = getMs(PropertyKey.USER_RPC_RETRY_MAX_SLEEP_MS);
if (waitTime < retryInterval) {
LOG.warn("{}={}ms is smaller than {}={}ms. Workers might not have enough time to register. "
+ "Consider either increasing {} or decreasing {}",
PropertyKey.MASTER_WORKER_CONNECT_WAIT_TIME, waitTime,
PropertyKey.USER_RPC_RETRY_MAX_SLEEP_MS, retryInterval,
PropertyKey.MASTER_WORKER_CONNECT_WAIT_TIME,
PropertyKey.USER_RPC_RETRY_MAX_SLEEP_MS);
}
checkHeartbeatTimeout(PropertyKey.MASTER_STANDBY_HEARTBEAT_INTERVAL,
PropertyKey.MASTER_HEARTBEAT_TIMEOUT);
// Skip checking block worker heartbeat config because the timeout is master-side while the
// heartbeat interval is worker-side.
}
/**
* Checks that the interval is shorter than the timeout.
*
* @param intervalKey property key for an interval
* @param timeoutKey property key for a timeout
*/
private void checkHeartbeatTimeout(PropertyKey intervalKey, PropertyKey timeoutKey) {
long interval = getMs(intervalKey);
long timeout = getMs(timeoutKey);
checkState(interval < timeout,
"heartbeat interval (%s=%s) must be less than heartbeat timeout (%s=%s)", intervalKey,
interval, timeoutKey, timeout);
}
/**
* Validates the user file buffer size is a non-negative number.
*
* @throws IllegalStateException if invalid user file buffer size configuration is encountered
*/
private void checkUserFileBufferBytes() {
if (!isSet(PropertyKey.USER_FILE_BUFFER_BYTES)) { // load from hadoop conf
return;
}
long usrFileBufferBytes = getBytes(PropertyKey.USER_FILE_BUFFER_BYTES);
checkState((usrFileBufferBytes & Integer.MAX_VALUE) == usrFileBufferBytes,
"Invalid value of %s: %s",
PropertyKey.Name.USER_FILE_BUFFER_BYTES, usrFileBufferBytes);
}
/**
* Validates Zookeeper-related configuration and prints warnings for possible sources of error.
*
* @throws IllegalStateException if invalid Zookeeper configuration is encountered
*/
private void checkZkConfiguration() {
checkState(
isSet(PropertyKey.ZOOKEEPER_ADDRESS) == getBoolean(PropertyKey.ZOOKEEPER_ENABLED),
"Inconsistent Zookeeper configuration; %s should be set if and only if %s is true",
PropertyKey.Name.ZOOKEEPER_ADDRESS, PropertyKey.Name.ZOOKEEPER_ENABLED);
}
/**
* @throws IllegalStateException if invalid checkpoint zip configuration parameters are found
*/
private void checkCheckpointZipConfig() {
int compression = getInt(
PropertyKey.MASTER_EMBEDDED_JOURNAL_SNAPSHOT_REPLICATION_COMPRESSION_LEVEL);
if (compression < -1 || compression > 9) {
throw new IllegalStateException(String.format("Zip compression level for property key %s"
+ " must be between -1 and 9 inclusive",
PropertyKey.MASTER_EMBEDDED_JOURNAL_SNAPSHOT_REPLICATION_COMPRESSION_LEVEL.getName()));
}
}
/**
* @return the last update time
*/
public long getLastUpdateTime() {
return mProperties.getLastUpdateTime();
}
protected class UnresolvablePropertyException extends Exception {
public UnresolvablePropertyException(String msg) {
super(msg);
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy