All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.infinispan.counter.impl.weak.WeakCounterImpl Maven / Gradle / Ivy

There is a newer version: 15.1.0.Dev05
Show newest version
package org.infinispan.counter.impl.weak;

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 java.util.concurrent.CompletionStage;
import java.util.concurrent.atomic.AtomicReferenceFieldUpdater;

import org.infinispan.AdvancedCache;
import org.infinispan.Cache;
import org.infinispan.commons.util.Util;
import org.infinispan.commons.util.concurrent.CompletableFutures;
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.counter.impl.manager.InternalCounterAdmin;
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.commons.util.concurrent.AggregateCompletionStage;
import org.infinispan.commons.util.concurrent.CompletionStages;

/**
 * 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, InternalCounterAdmin { private static final AtomicReferenceFieldUpdater SNAPSHOT_UPDATER = AtomicReferenceFieldUpdater.newUpdater(Entry.class, Long.class, "snapshot"); 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 CompletionStage removeWeakCounter(Cache cache, CounterConfiguration configuration, String counterName) { ByteString counterNameByteString = ByteString.fromString(counterName); AggregateCompletionStage stage = CompletionStages.aggregateCompletionStage(); for (int i = 0; i < configuration.concurrencyLevel(); ++i) { stage.dependsOn(cache.removeAsync(new WeakCounterKey(counterNameByteString, i))); } return stage.freeze(); } /** * Initializes the key set. *

* Only one key will have the initial value and the remaining is zero. */ public CompletionStage init() { registerListener(); AggregateCompletionStage stage = CompletionStages.aggregateCompletionStage(this); for (int i = 0; i < entries.length; ++i) { int index = i; stage.dependsOn(readOnlyMap.eval(entries[index].key, ReadFunction.getInstance()) .thenAccept(value -> initEntry(index, value))); } selector.updatePreferredKeys(); return stage.freeze(); } @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 WeakCounter asWeakCounter() { return this; } @Override public CompletionStage destroy() { removeListener(); return remove(); } @Override public CompletableFuture reset() { AggregateCompletionStage stage = CompletionStages.aggregateCompletionStage(); for (Entry entry : entries) { stage.dependsOn(readWriteMap.eval(entry.key, ResetFunction.getInstance())); } return stage.freeze().toCompletableFuture(); } @Override public CompletionStage value() { return CompletableFuture.completedFuture(getValue()); } @Override public boolean isWeakCounter() { return true; } @Override public Handle addListener(T listener) { return notificationManager.registerUserListener(counterName(), listener); } @Override public CounterConfiguration getConfiguration() { return configuration; } @Override public synchronized 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! 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() { AggregateCompletionStage stage = CompletionStages.aggregateCompletionStage(); for (Entry entry : entries) { stage.dependsOn(readWriteMap.eval(entry.key, RemoveFunction.getInstance())); } return stage.freeze().toCompletableFuture(); } @Override public SyncWeakCounter sync() { return new SyncWeakCounterAdapter(this); } @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); } entries[index].init(value); } private Long getCachedValue() { 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) { 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 final class Entry { final WeakCounterKey key; volatile Long snapshot; private Entry(WeakCounterKey key) { this.key = key; } void init(long initialValue) { SNAPSHOT_UPDATER.compareAndSet(this, null, initialValue); } Long update(long value) { return SNAPSHOT_UPDATER.getAndSet(this, value); } } private interface KeySelector { WeakCounterKey findKey(int hash); void updatePreferredKeys(); WeakCounterKey[] getPreferredKeys(); } private static final 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 final 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; } } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy