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

org.infinispan.counter.impl.manager.CounterConfigurationManager Maven / Gradle / Ivy

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

import static org.infinispan.counter.configuration.ConvertUtil.parsedConfigToConfig;
import static org.infinispan.counter.impl.Utils.validateStrongCounterBounds;
import static org.infinispan.counter.logging.Log.CONTAINER;

import java.util.Collection;
import java.util.Collections;
import java.util.List;
import java.util.Map;
import java.util.Objects;
import java.util.concurrent.CompletableFuture;
import java.util.concurrent.atomic.AtomicBoolean;
import java.util.stream.Collectors;

import org.infinispan.AdvancedCache;
import org.infinispan.Cache;
import org.infinispan.configuration.global.GlobalConfiguration;
import org.infinispan.counter.api.CounterConfiguration;
import org.infinispan.counter.configuration.AbstractCounterConfiguration;
import org.infinispan.counter.configuration.CounterManagerConfiguration;
import org.infinispan.counter.impl.CounterModuleLifecycle;
import org.infinispan.globalstate.GlobalConfigurationManager;
import org.infinispan.globalstate.ScopeFilter;
import org.infinispan.globalstate.ScopedState;
import org.infinispan.lifecycle.ComponentStatus;
import org.infinispan.manager.EmbeddedCacheManager;
import org.infinispan.notifications.Listener;
import org.infinispan.notifications.cachelistener.annotation.CacheEntryCreated;
import org.infinispan.notifications.cachelistener.annotation.CacheEntryModified;
import org.infinispan.notifications.cachelistener.event.Event;
import org.infinispan.stream.CacheCollectors;
import org.infinispan.util.concurrent.BlockingManager;
import org.infinispan.util.concurrent.CompletableFutures;

/**
 * Stores all the defined counter's configuration.
 * 

* It uses the state-{@link Cache} to distribute the counter's configuration among the nodes in the cluster, and the {@link CounterConfigurationStorage} to persist persistent configurations. *

* Note that the {@link Cache} doesn't persist anything. * * @author Pedro Ruivo * @since 9.2 */ public class CounterConfigurationManager { public static final String COUNTER_SCOPE = "counter"; private final AtomicBoolean counterCacheStarted = new AtomicBoolean(false); private final EmbeddedCacheManager cacheManager; private final List configuredCounters; private final CounterConfigurationStorage storage; private volatile AdvancedCache stateCache; private volatile CounterConfigurationListener listener; CounterConfigurationManager(EmbeddedCacheManager cacheManager, CounterConfigurationStorage storage) { this.cacheManager = cacheManager; this.storage = storage; GlobalConfiguration globalConfig = SecurityActions.getCacheManagerConfiguration(cacheManager); CounterManagerConfiguration counterManagerConfig = globalConfig.module(CounterManagerConfiguration.class); this.configuredCounters = counterManagerConfig == null ? Collections.emptyList() : counterManagerConfig.counters(); } private static ScopedState stateKey(String counterName) { return new ScopedState(COUNTER_SCOPE, counterName); } /** * It checks for existing defined configurations in {@link CounterConfigurationStorage} and in the {@link Cache}. *

* If any is found, it starts the counter's {@link Cache}. */ public void start() { this.stateCache = cacheManager .getCache(GlobalConfigurationManager.CONFIG_STATE_CACHE_NAME) .getAdvancedCache(); listener = new CounterConfigurationListener(); Map persisted = storage.loadAll(); persisted.forEach((name, cfg) -> stateCache.putIfAbsent(stateKey(name), cfg)); counterCacheStarted.set(false); stateCache.addListener(listener, new ScopeFilter(COUNTER_SCOPE), null); if (!persisted.isEmpty() || stateCache.keySet().stream() .anyMatch(scopedState -> COUNTER_SCOPE.equals(scopedState.getScope()))) { //we have counter defined startCounterCache(); } } /** * Removes the listener for new coming defined counter's. *

* The persistence is done on the fly when the configuration is defined. */ public void stop() { counterCacheStarted.set(true); //avoid starting the counter cache if it hasn't started yet if (listener != null && stateCache != null) { stateCache.removeListener(listener); } listener = null; stateCache = null; } /** * It defines a new counter with the {@link CounterConfiguration}. * * @param name the counter's name. * @param configuration the counter's {@link CounterConfiguration}. * @return {@code true} if the counter doesn't exist. {@code false} otherwise. */ public CompletableFuture defineConfiguration(String name, CounterConfiguration configuration) { //first, check if valid. validateConfiguration(configuration); return checkGlobalConfiguration(name) .thenCompose(fileConfig -> { if (fileConfig != null) { //counter configuration exists in configuration file! return CompletableFuture.completedFuture(Boolean.FALSE); } //put configuration in cache return stateCache.putIfAbsentAsync(stateKey(name), configuration) .thenCompose(cfg -> { if (cfg == null) { BlockingManager blockingManager = SecurityActions.getGlobalComponentRegistry(cacheManager).getComponent(BlockingManager.class); //the counter was created. we need to persist it (if the counter is persistent) return blockingManager.supplyBlocking(() -> { storage.store(name, configuration); return Boolean.TRUE; }, name); } else { //already defined. return CompletableFutures.completedFalse(); } }); }); } /** * Remove a configuration * * @param name the name of the counter * * @return true if the configuration was removed */ CompletableFuture removeConfiguration(String name) { return stateCache.removeAsync(stateKey(name)).thenApply(Objects::nonNull); } /** * @return all the defined counter's name, even the one in Infinispan's configuration. */ public Collection getCounterNames() { Collection countersName = stateCache.keySet().stream() .filter(new ScopeFilter(COUNTER_SCOPE)) .map(ScopedState::getName) .collect(CacheCollectors.serializableCollector(Collectors::toSet)); configuredCounters.stream().map(AbstractCounterConfiguration::name).forEach(countersName::add); return Collections.unmodifiableCollection(countersName); } /** * Returns the counter's configuration. * * @param name the counter's name. * @return the counter's {@link CounterConfiguration} or {@code null} if not defined. */ CompletableFuture getConfiguration(String name) { return stateCache.getAsync(stateKey(name)).thenCompose(existingConfiguration -> { if (existingConfiguration == null) { return checkGlobalConfiguration(name); } else { return CompletableFuture.completedFuture(existingConfiguration); } }); } /** * Checks if the counter is defined in the Infinispan's configuration file. * * @param name the counter's name. * @return {@code null} if the counter isn't defined yet. {@link CounterConfiguration} if it is defined. */ private CompletableFuture checkGlobalConfiguration(String name) { for (AbstractCounterConfiguration config : configuredCounters) { if (config.name().equals(name)) { CounterConfiguration cConfig = parsedConfigToConfig(config); return stateCache.putIfAbsentAsync(stateKey(name), cConfig) .thenApply(configuration -> { if (configuration == null) { storage.store(name, cConfig); return cConfig; } return configuration; }); } } return CompletableFutures.completedNull(); } /** * Invoked when a new configuration is stored in the state-cache, it boots the counter's cache. */ private void createCounter() { if (stateCache.getStatus() == ComponentStatus.RUNNING) { startCounterCache(); } } /** * Validates the counter's configuration. * * @param configuration the {@link CounterConfiguration} to be validated. */ private void validateConfiguration(CounterConfiguration configuration) { storage.validatePersistence(configuration); switch (configuration.type()) { case BOUNDED_STRONG: validateStrongCounterBounds(configuration.lowerBound(), configuration.initialValue(), configuration.upperBound()); break; case WEAK: if (configuration.concurrencyLevel() < 1) { throw CONTAINER.invalidConcurrencyLevel(configuration.concurrencyLevel()); } break; } } private void startCounterCache() { if (counterCacheStarted.compareAndSet(false, true)) { BlockingManager blockingManager = SecurityActions.getGlobalComponentRegistry(cacheManager) .getComponent(BlockingManager.class); blockingManager.runBlocking(() -> { String oldName = Thread.currentThread().getName(); try { GlobalConfiguration configuration = SecurityActions.getCacheManagerConfiguration(cacheManager); String threadName = "CounterCacheStartThread," + configuration.transport().nodeName(); SecurityActions.setThreadName(threadName); cacheManager.getCache(CounterModuleLifecycle.COUNTER_CACHE_NAME); } finally { SecurityActions.setThreadName(oldName); } }, CounterModuleLifecycle.COUNTER_CACHE_NAME); } } //can be async! @Listener(observation = Listener.Observation.POST, sync = false) private class CounterConfigurationListener { @CacheEntryModified @CacheEntryCreated public void onNewCounterConfiguration(Event event) { createCounter(); } } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy