com.netflix.archaius.property.DefaultPropertyContainer Maven / Gradle / Ivy
/**
* Copyright 2015 Netflix, Inc.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package com.netflix.archaius.property;
import java.math.BigDecimal;
import java.math.BigInteger;
import java.util.concurrent.CopyOnWriteArrayList;
import java.util.concurrent.TimeUnit;
import java.util.concurrent.atomic.AtomicInteger;
import java.util.concurrent.atomic.AtomicReference;
import java.util.concurrent.atomic.AtomicStampedReference;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import com.netflix.archaius.api.Config;
import com.netflix.archaius.api.Property;
import com.netflix.archaius.api.PropertyContainer;
import com.netflix.archaius.api.PropertyListener;
import com.netflix.archaius.property.ListenerManager.ListenerUpdater;
/**
* Implementation of PropertyContainer which reuses the same object for each
* type. This implementation assumes that each fast property is mostly accessed
* as the same type but allows for additional types to be deserialized.
* Instead of incurring the overhead for caching in a hash map, the objects are
* stored in a CopyOnWriteArrayList and items are retrieved via a linear scan.
*
* Once created a PropertyContainer property cannot be removed. However, listeners may be
* added and removed.
*
* @author elandau
*
*/
public class DefaultPropertyContainer implements PropertyContainer {
private final Logger LOG = LoggerFactory.getLogger(DefaultPropertyContainer.class);
enum Type {
INTEGER (int.class, Integer.class),
BYTE (byte.class, Byte.class),
SHORT (short.class, Short.class),
DOUBLE (double.class, Double.class),
FLOAT (float.class, Float.class),
BOOLEAN (boolean.class, Boolean.class),
LONG (long.class, Long.class),
STRING (String.class),
BIG_DECIMAL (BigDecimal.class),
BIG_INTEGER (BigInteger.class),
CUSTOM (); // Must be last
private final Class>[] types;
Type(Class> ... type) {
types = type;
}
static Type fromClass(Class> clazz) {
for (Type type : values()) {
for (Class> cls : type.types) {
if (cls.equals(clazz)) {
return type;
}
}
}
return CUSTOM;
}
}
/**
* The property name
*/
private final String key;
/**
* Config from which property values are resolved
*/
private final Config config;
/**
* Cache for each type attached to this property.
*/
private final CopyOnWriteArrayList> cache = new CopyOnWriteArrayList>();
/**
* Listeners are tracked globally as an optimization so it is not necessary to iterate through all
* property containers when the listeners need to be invoked since the expectation is to have far
* less listeners than property containers.
*/
private final ListenerManager listeners;
/**
* Reference to the externally managed master version used as the dirty flag
*/
private final AtomicInteger masterVersion;
private volatile long lastUpdateTimeInMillis = 0;
public DefaultPropertyContainer(String key, Config config, AtomicInteger version, ListenerManager listeners) {
this.key = key;
this.config = config;
this.listeners = listeners;
this.masterVersion = version;
}
abstract class CachedProperty implements Property {
@Override
public int hashCode() {
final int prime = 31;
int result = 1;
result = prime * result + ((defaultValue == null) ? 0 : defaultValue.hashCode());
result = prime * result + type;
return result;
}
@Override
public boolean equals(Object obj) {
if (this == obj)
return true;
if (obj == null)
return false;
if (getClass() != obj.getClass())
return false;
CachedProperty other = (CachedProperty) obj;
if (defaultValue == null) {
if (other.defaultValue != null)
return false;
} else if (!defaultValue.equals(other.defaultValue))
return false;
if (type != other.type)
return false;
return true;
}
private final AtomicStampedReference cache = new AtomicStampedReference<>(null, -1);
private final int type;
private final T defaultValue;
CachedProperty(Type type, T defaultValue) {
this.type = type.ordinal();
this.defaultValue = defaultValue;
}
public void addListener(final PropertyListener listener) {
listeners.add(listener, new ListenerUpdater() {
private AtomicReference last = new AtomicReference(null);
@Override
public void update() {
final T prev = last.get();
final T value;
try {
value = get();
}
catch (Exception e) {
listener.onParseError(e);
return;
}
if (prev != value) {
if (last.compareAndSet(prev, value)) {
listener.onChange(value);
}
}
}
});
}
@Override
public String getKey() {
return DefaultPropertyContainer.this.key;
}
public void removeListener(PropertyListener listener) {
listeners.remove(listener);
}
/**
* Fetch the latest version of the property. If not up to date then resolve to the latest
* value, inline.
*
* TODO: Make resolving property value an offline task
*
* @return
*/
public T get() {
int cacheVersion = cache.getStamp();
int latestVersion = masterVersion.get();
if (cacheVersion != latestVersion) {
T currentValue = cache.getReference();
T newValue = null;
try {
newValue = resolveCurrent();
}
catch (Exception e) {
LOG.warn("Unable to get current version of property '{}'. Error: {}", key, e.getMessage());
}
if (cache.compareAndSet(currentValue, newValue, cacheVersion, latestVersion)) {
// Slight race condition here but not important enough to warrent locking
lastUpdateTimeInMillis = System.currentTimeMillis();
return firstNonNull(newValue, defaultValue);
}
}
return firstNonNull(cache.getReference(), defaultValue);
}
public long getLastUpdateTime(TimeUnit units) {
return units.convert(lastUpdateTimeInMillis, TimeUnit.MILLISECONDS);
}
private T firstNonNull(T first, T second) {
return first == null ? second : first;
}
/**
* Resolve to the most recent value
* @return
* @throws Exception
*/
protected abstract T resolveCurrent() throws Exception;
private DefaultPropertyContainer getOuterType() {
return DefaultPropertyContainer.this;
}
}
/**
* Add a new property to the end of the array list but first check
* to see if it already exists.
* @param newProperty
* @return
*/
@SuppressWarnings("unchecked")
private CachedProperty add(final CachedProperty newProperty) {
// TODO(nikos): This while() looks like it's redundant
// since we are only calling add() after a get().
while (!cache.add(newProperty)) {
for (CachedProperty> property : cache) {
if (property.equals(newProperty)) {
return (CachedProperty) property;
}
}
}
return newProperty;
}
@Override
public Property asString(final String defaultValue) {
return add(new CachedProperty(Type.STRING, defaultValue) {
@Override
protected String resolveCurrent() throws Exception {
return config.getString(key, null);
}
});
}
@Override
public Property asInteger(final Integer defaultValue) {
return add(new CachedProperty(Type.INTEGER, defaultValue) {
@Override
protected Integer resolveCurrent() throws Exception {
return config.getInteger(key, null);
}
});
}
@Override
public Property asLong(final Long defaultValue) {
return add(new CachedProperty(Type.LONG, defaultValue) {
@Override
protected Long resolveCurrent() throws Exception {
return config.getLong(key, null);
}
});
}
@Override
public Property asDouble(final Double defaultValue) {
return add(new CachedProperty(Type.DOUBLE, defaultValue) {
@Override
protected Double resolveCurrent() throws Exception {
return config.getDouble(key, null);
}
});
}
@Override
public Property asFloat(final Float defaultValue) {
return add(new CachedProperty(Type.FLOAT, defaultValue) {
@Override
protected Float resolveCurrent() throws Exception {
return config.getFloat(key, null);
}
});
}
@Override
public Property asShort(final Short defaultValue) {
return add(new CachedProperty(Type.SHORT, defaultValue) {
@Override
protected Short resolveCurrent() throws Exception {
return config.getShort(key, null);
}
});
}
@Override
public Property asByte(final Byte defaultValue) {
return add(new CachedProperty(Type.BYTE, defaultValue) {
@Override
protected Byte resolveCurrent() throws Exception {
return config.getByte(key, defaultValue);
}
});
}
@Override
public Property asBigDecimal(final BigDecimal defaultValue) {
return add(new CachedProperty(Type.BIG_DECIMAL, defaultValue) {
@Override
protected BigDecimal resolveCurrent() throws Exception {
return config.getBigDecimal(key, defaultValue);
}
});
}
@Override
public Property asBoolean(final Boolean defaultValue) {
return add(new CachedProperty(Type.BOOLEAN, defaultValue) {
@Override
protected Boolean resolveCurrent() throws Exception {
return config.getBoolean(key, defaultValue);
}
});
}
@Override
public Property asBigInteger(final BigInteger defaultValue) {
return add(new CachedProperty(Type.BIG_INTEGER, defaultValue) {
@Override
protected BigInteger resolveCurrent() throws Exception {
return config.getBigInteger(key, defaultValue);
}
});
}
/**
* No caching for custom types.
*/
@SuppressWarnings("unchecked")
@Override
public Property asType(final Class type, final T defaultValue) {
switch (Type.fromClass(type)) {
case INTEGER:
return (Property) asInteger((Integer)defaultValue);
case BYTE:
return (Property) asByte((Byte)defaultValue);
case SHORT:
return (Property) asShort((Short)defaultValue);
case DOUBLE:
return (Property) asDouble((Double)defaultValue);
case FLOAT:
return (Property) asFloat((Float)defaultValue);
case BOOLEAN:
return (Property) asBoolean((Boolean)defaultValue);
case STRING:
return (Property) asString((String)defaultValue);
case LONG:
return (Property) asLong((Long)defaultValue);
case BIG_DECIMAL:
return (Property) asBigDecimal((BigDecimal)defaultValue);
case BIG_INTEGER:
return (Property) asBigInteger((BigInteger)defaultValue);
default: {
CachedProperty prop = add(new CachedProperty(Type.CUSTOM, defaultValue) {
@Override
protected T resolveCurrent() throws Exception {
return config.get(type, key, defaultValue);
}
});
return prop;
}
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy