org.infinispan.counter.impl.weak.WeakCounterImpl Maven / Gradle / Ivy
Show all versions of infinispan-clustered-counter
package org.infinispan.counter.impl.weak;
import static org.infinispan.counter.impl.Util.awaitCounterOperation;
import static org.infinispan.counter.impl.Utils.getPersistenceMode;
import java.math.BigInteger;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.concurrent.CompletableFuture;
import org.infinispan.AdvancedCache;
import org.infinispan.Cache;
import org.infinispan.commons.util.Util;
import org.infinispan.counter.api.CounterConfiguration;
import org.infinispan.counter.api.CounterEvent;
import org.infinispan.counter.api.CounterListener;
import org.infinispan.counter.api.CounterType;
import org.infinispan.counter.api.Handle;
import org.infinispan.counter.api.SyncWeakCounter;
import org.infinispan.counter.api.WeakCounter;
import org.infinispan.counter.impl.SyncWeakCounterAdapter;
import org.infinispan.counter.impl.entries.CounterKey;
import org.infinispan.counter.impl.entries.CounterValue;
import org.infinispan.counter.impl.function.AddFunction;
import org.infinispan.counter.impl.function.CreateAndAddFunction;
import org.infinispan.counter.impl.function.ReadFunction;
import org.infinispan.counter.impl.function.RemoveFunction;
import org.infinispan.counter.impl.function.ResetFunction;
import org.infinispan.counter.impl.listener.CounterEventGenerator;
import org.infinispan.counter.impl.listener.CounterEventImpl;
import org.infinispan.counter.impl.listener.CounterManagerNotificationManager;
import org.infinispan.counter.impl.listener.TopologyChangeListener;
import org.infinispan.distribution.LocalizedCacheTopology;
import org.infinispan.functional.FunctionalMap;
import org.infinispan.functional.impl.FunctionalMapImpl;
import org.infinispan.functional.impl.ReadOnlyMapImpl;
import org.infinispan.functional.impl.ReadWriteMapImpl;
import org.infinispan.util.ByteString;
import org.infinispan.util.concurrent.CompletableFutures;
import net.jcip.annotations.GuardedBy;
/**
* A weak consistent counter implementation.
*
* Implementation: The counter is split in multiple keys and they are stored in the cache.
*
* Write: A write operation will pick a key to update. If the node is a primary owner of one of the key, that key is
* chosen based on thread-id. This will take advantage of faster write operations. If the node is not a primary owner,
* one of the key in key set is chosen.
*
* Read: A read operation needs to read all the key set (including the remote keys). This is slower than atomic
* counter.
*
* Weak Read: A snapshot of all the keys values is kept locally and they are updated via cluster listeners.
*
* Reset: The reset operation is not atomic and intermediate results may be observed.
*
* @author Pedro Ruivo
* @since 9.0
*/
public class WeakCounterImpl implements WeakCounter, CounterEventGenerator, TopologyChangeListener {
@GuardedBy("entries")
private final Entry[] entries;
private final AdvancedCache cache;
private final FunctionalMap.ReadWriteMap readWriteMap;
private final CounterManagerNotificationManager notificationManager;
private final FunctionalMap.ReadOnlyMap readOnlyMap;
private final CounterConfiguration configuration;
private final CounterConfiguration zeroConfiguration;
private final KeySelector selector;
public WeakCounterImpl(String counterName, AdvancedCache cache,
CounterConfiguration configuration, CounterManagerNotificationManager notificationManager) {
this.cache = cache;
this.notificationManager = notificationManager;
FunctionalMapImpl functionalMap = FunctionalMapImpl.create(cache)
.withParams(getPersistenceMode(configuration.storage()));
this.readWriteMap = ReadWriteMapImpl.create(functionalMap);
this.readOnlyMap = ReadOnlyMapImpl.create(functionalMap);
this.entries = initKeys(counterName, configuration.concurrencyLevel());
this.selector = cache.getCacheConfiguration().clustering().cacheMode().isClustered() ?
new ClusteredKeySelector(entries) :
new LocalKeySelector(entries);
this.configuration = configuration;
this.zeroConfiguration = CounterConfiguration.builder(CounterType.WEAK)
.concurrencyLevel(configuration.concurrencyLevel()).storage(configuration.storage()).initialValue(0)
.build();
}
private static T get(int hash, T[] array) {
return array[hash & (array.length - 1)];
}
private static Entry[] initKeys(String counterName, int concurrencyLevel) {
ByteString name = ByteString.fromString(counterName);
int size = Util.findNextHighestPowerOfTwo(concurrencyLevel);
Entry[] entries = new Entry[size];
for (int i = 0; i < size; ++i) {
entries[i] = new Entry(new WeakCounterKey(name, i));
}
return entries;
}
/**
* It removes a weak counter from the {@code cache}, identified by the {@code counterName}.
*
* @param cache The {@link Cache} to remove the counter from.
* @param configuration The counter's configuration.
* @param counterName The counter's name.
*/
public static void removeWeakCounter(Cache cache, CounterConfiguration configuration,
String counterName) {
ByteString counterNameByteString = ByteString.fromString(counterName);
for (int i = 0; i < configuration.concurrencyLevel(); ++i) {
cache.remove(new WeakCounterKey(counterNameByteString, i));
}
}
/**
* Initializes the key set.
*
* Only one key will have the initial value and the remaining is zero.
*/
public void init() {
registerListener();
for (int i = 0; i < entries.length; ++i) {
final int index = i;
awaitCounterOperation(readOnlyMap.eval(entries[index].key, ReadFunction.getInstance())
.thenAccept(value -> initEntry(index, value)));
}
selector.updatePreferredKeys();
}
@Override
public String getName() {
return counterName().toString();
}
@Override
public long getValue() {
//return the initial value if it doesn't have a valid snapshot!
Long snapshot = getCachedValue();
return snapshot == null ? configuration.initialValue() : snapshot;
}
@Override
public CompletableFuture add(long delta) {
WeakCounterKey key = findKey();
return readWriteMap.eval(key, new AddFunction<>(delta))
.thenCompose(counterValue -> handleAddResult(key, counterValue, delta));
}
@Override
public CompletableFuture reset() {
final int size = entries.length;
CompletableFuture[] futures = new CompletableFuture[size];
for (int i = 0; i < size; ++i) {
futures[i] = readWriteMap.eval(entries[i].key, ResetFunction.getInstance());
}
return CompletableFuture.allOf(futures);
}
@Override
public Handle addListener(T listener) {
return notificationManager.registerUserListener(counterName(), listener);
}
@Override
public CounterConfiguration getConfiguration() {
return configuration;
}
@Override
public CounterEvent generate(CounterKey key, CounterValue value) {
//we need to synchronize this.
//if it receives 2 events concurrently (e.g. 2 increments), it can generate duplicated events!
synchronized (entries) {
assert key instanceof WeakCounterKey;
int index = ((WeakCounterKey) key).getIndex();
long newValue = value == null ?
defaultValueOfIndex(index) :
value.getValue();
Long base = getCachedValue(index);
Long oldValue = entries[index].update(newValue);
return base == null || oldValue == null || oldValue == newValue ?
null :
CounterEventImpl.create(base + oldValue, base + newValue);
}
}
@Override
public CompletableFuture remove() {
final int size = entries.length;
CompletableFuture[] futures = new CompletableFuture[size];
for (int i = 0; i < size; ++i) {
futures[i] = readWriteMap.eval(entries[i].key, RemoveFunction.getInstance());
}
return CompletableFuture.allOf(futures);
}
@Override
public SyncWeakCounter sync() {
return new SyncWeakCounterAdapter(this);
}
public void destroyAndRemove() {
removeListener();
awaitCounterOperation(remove());
}
@Override
public void topologyChanged() {
selector.updatePreferredKeys();
}
/**
* Debug only!
*/
public WeakCounterKey[] getPreferredKeys() {
return selector.getPreferredKeys();
}
/**
* Debug only!
*/
public WeakCounterKey[] getKeys() {
WeakCounterKey[] keys = new WeakCounterKey[entries.length];
for (int i = 0; i < keys.length; ++i) {
keys[i] = entries[i].key;
}
return keys;
}
@Override
public String toString() {
return "WeakCounter{" +
"counterName=" + counterName() +
'}';
}
private long defaultValueOfIndex(int index) {
return index == 0 ? configuration.initialValue() : 0;
}
private void initEntry(int index, Long value) {
if (value == null) {
value = defaultValueOfIndex(index);
}
synchronized (entries) {
entries[index].init(value);
}
}
private Long getCachedValue() {
synchronized (entries) {
long value = 0;
int index = 0;
try {
for (; index < entries.length; ++index) {
Long toAdd = entries[index].snapshot;
if (toAdd == null) {
//we don't have a valid snapshot
return null;
}
value = Math.addExact(value, toAdd);
}
} catch (ArithmeticException e) {
return getCachedValue0(index, value, -1);
}
return value;
}
}
private Long getCachedValue(int skipIndex) {
synchronized (entries) {
long value = 0;
int index = 0;
try {
for (; index < entries.length; ++index) {
if (index == skipIndex) {
continue;
}
Long toAdd = entries[index].snapshot;
if (toAdd == null) {
//we don't have a valid snapshot
return null;
}
value = Math.addExact(value, toAdd);
}
} catch (ArithmeticException e) {
return getCachedValue0(index, value, skipIndex);
}
return value;
}
}
private Long getCachedValue0(int index, long value, int skipIndex) {
BigInteger currentValue = BigInteger.valueOf(value);
do {
Long toAdd = entries[index++].snapshot;
if (toAdd == null) {
//we don't have a valid snapshot
return null;
}
currentValue = currentValue.add(BigInteger.valueOf(toAdd));
if (index == skipIndex) {
index++;
}
} while (index < entries.length);
try {
return currentValue.longValue();
} catch (ArithmeticException e) {
return currentValue.signum() > 0 ? Long.MAX_VALUE : Long.MIN_VALUE;
}
}
private CompletableFuture handleAddResult(WeakCounterKey key, CounterValue value, long delta) {
if (value == null) {
//first time
CounterConfiguration createConfiguration = key.getIndex() == 0 ?
configuration :
zeroConfiguration;
return readWriteMap.eval(key, new CreateAndAddFunction<>(createConfiguration, delta))
.thenApply(value1 -> null);
} else {
return CompletableFutures.completedNull();
}
}
private void registerListener() {
notificationManager.registerCounter(counterName(), this, this);
}
private void removeListener() {
notificationManager.removeCounter(counterName());
}
private WeakCounterKey findKey() {
return selector.findKey((int) Thread.currentThread().getId());
}
private ByteString counterName() {
return entries[0].key.getCounterName();
}
private static class Entry {
final WeakCounterKey key;
@GuardedBy("entries")
volatile Long snapshot;
private Entry(WeakCounterKey key) {
this.key = key;
}
private void init(long initialValue) {
if (snapshot == null) {
snapshot = initialValue;
}
}
private Long update(long value) {
Long old = snapshot;
snapshot = value;
return old;
}
}
private interface KeySelector {
WeakCounterKey findKey(int hash);
void updatePreferredKeys();
WeakCounterKey[] getPreferredKeys();
}
private static class LocalKeySelector implements KeySelector {
private final Entry[] entries;
private LocalKeySelector(Entry[] entries) {
this.entries = entries;
}
@Override
public WeakCounterKey findKey(int hash) {
return get(hash, entries).key;
}
@Override
public void updatePreferredKeys() {
//no-op, everything is local
}
@Override
public WeakCounterKey[] getPreferredKeys() {
return Arrays.stream(entries).map(entry -> entry.key).toArray(WeakCounterKey[]::new);
}
}
private class ClusteredKeySelector implements KeySelector {
private final Entry[] entries;
private volatile WeakCounterKey[] preferredKeys; //null when no keys available
private ClusteredKeySelector(Entry[] entries) {
this.entries = entries;
}
@Override
public WeakCounterKey findKey(int hash) {
WeakCounterKey[] copy = preferredKeys;
if (copy == null) {
return get(hash, entries).key;
} else if (copy.length == 1) {
return copy[0];
} else {
return get(hash, copy);
}
}
@Override
public void updatePreferredKeys() {
ArrayList preferredKeys = new ArrayList<>(entries.length);
LocalizedCacheTopology topology = cache.getDistributionManager().getCacheTopology();
for (Entry entry : entries) {
if (topology.getDistribution(entry.key).isPrimary()) {
preferredKeys.add(entry.key);
}
}
this.preferredKeys = preferredKeys.isEmpty() ?
null :
preferredKeys.toArray(new WeakCounterKey[preferredKeys.size()]);
}
@Override
public WeakCounterKey[] getPreferredKeys() {
return preferredKeys;
}
}
}