org.wildfly.clustering.infinispan.container.DataContainerFactory Maven / Gradle / Ivy
/*
* Copyright The WildFly Authors
* SPDX-License-Identifier: Apache-2.0
*/
package org.wildfly.clustering.infinispan.container;
import java.util.function.Predicate;
import java.util.function.Supplier;
import org.infinispan.commons.marshall.WrappedBytes;
import org.infinispan.commons.util.EntrySizeCalculator;
import org.infinispan.configuration.cache.ClusteringConfiguration;
import org.infinispan.configuration.cache.MemoryConfiguration;
import org.infinispan.container.DataContainer;
import org.infinispan.container.entries.InternalCacheEntry;
import org.infinispan.container.impl.BoundedSegmentedDataContainer;
import org.infinispan.container.impl.DefaultDataContainer;
import org.infinispan.container.impl.DefaultSegmentedDataContainer;
import org.infinispan.container.impl.InternalDataContainer;
import org.infinispan.container.impl.L1SegmentedDataContainer;
import org.infinispan.container.impl.PeekableTouchableContainerMap;
import org.infinispan.container.impl.PeekableTouchableMap;
import org.infinispan.container.offheap.BoundedOffHeapDataContainer;
import org.infinispan.container.offheap.OffHeapConcurrentMap;
import org.infinispan.container.offheap.OffHeapDataContainer;
import org.infinispan.container.offheap.OffHeapEntryFactory;
import org.infinispan.container.offheap.OffHeapMemoryAllocator;
import org.infinispan.container.offheap.SegmentedBoundedOffHeapDataContainer;
import org.infinispan.eviction.EvictionStrategy;
import org.infinispan.factories.AbstractNamedCacheComponentFactory;
import org.infinispan.factories.AutoInstantiableFactory;
import org.infinispan.factories.annotations.DefaultFactoryFor;
import org.infinispan.factories.annotations.SurvivesRestarts;
/**
* @author Paul Ferraro
*/
@DefaultFactoryFor(classes = InternalDataContainer.class)
@SurvivesRestarts
public class DataContainerFactory extends AbstractNamedCacheComponentFactory implements AutoInstantiableFactory {
@Override
public Object construct(String componentName) {
MemoryConfiguration memory = this.configuration.memory();
EvictionStrategy strategy = memory.whenFull();
// handle case when < 0 value signifies unbounded container or when we are not removal based
if (strategy.isExceptionBased() || !strategy.isEnabled()) {
return this.createUnboundedContainer();
}
DataContainer, ?> container = this.createBoundedDataContainer();
memory.attributes().attribute(MemoryConfiguration.MAX_COUNT).addListener((newSize, oldSize) -> container.resize(newSize.get()));
return container;
}
private DataContainer, ?> createUnboundedContainer() {
ClusteringConfiguration clustering = this.configuration.clustering();
boolean segmented = clustering.cacheMode().needsStateTransfer();
int level = this.configuration.locking().concurrencyLevel();
boolean offHeap = this.configuration.memory().isOffHeap();
if (segmented) {
Supplier> mapSupplier = offHeap ? this::createAndStartOffHeapConcurrentMap : PeekableTouchableContainerMap::new;
int segments = clustering.hash().numSegments();
return clustering.l1().enabled() ? new L1SegmentedDataContainer<>(mapSupplier, segments) : new DefaultSegmentedDataContainer<>(mapSupplier, segments);
}
return offHeap ? new OffHeapDataContainer() : DefaultDataContainer.unBoundedDataContainer(level);
}
@SuppressWarnings("deprecation")
private DataContainer, ?> createBoundedDataContainer() {
ClusteringConfiguration clustering = this.configuration.clustering();
boolean segmented = clustering.cacheMode().needsStateTransfer();
int level = this.configuration.locking().concurrencyLevel();
MemoryConfiguration memory = this.configuration.memory();
boolean offHeap = memory.isOffHeap();
long maxEntries = memory.maxCount();
long maxBytes = memory.maxSizeBytes();
org.infinispan.eviction.EvictionType type = memory.evictionType();
DataContainerConfiguration config = this.configuration.module(DataContainerConfiguration.class);
Predicate> evictable = (config != null) ? config.evictable() : DataContainerConfiguration.EVICTABLE_PREDICATE.getDefaultValue();
if (segmented) {
int segments = clustering.hash().numSegments();
if (offHeap) {
return new SegmentedBoundedOffHeapDataContainer(segments, maxEntries, type);
}
return (maxEntries > 0) ? new BoundedSegmentedDataContainer<>(segments, maxEntries, new EvictableEntrySizeCalculator<>(evictable)) : new BoundedSegmentedDataContainer<>(segments, maxBytes, type);
}
if (offHeap) {
return new BoundedOffHeapDataContainer(maxEntries, type);
}
return (maxEntries > 0) ? new EvictableDataContainer<>(maxEntries, new EvictableEntrySizeCalculator<>(evictable)) : DefaultDataContainer.boundedDataContainer(level, maxBytes, type);
}
private OffHeapConcurrentMap createAndStartOffHeapConcurrentMap() {
OffHeapEntryFactory entryFactory = this.basicComponentRegistry.getComponent(OffHeapEntryFactory.class).wired();
OffHeapMemoryAllocator memoryAllocator = this.basicComponentRegistry.getComponent(OffHeapMemoryAllocator.class).wired();
return new OffHeapConcurrentMap(memoryAllocator, entryFactory, null);
}
public static class EvictableDataContainer extends DefaultDataContainer {
EvictableDataContainer(long size, EntrySizeCalculator super K, ? super InternalCacheEntry> calculator) {
super(size, calculator);
}
}
private static class EvictableEntrySizeCalculator implements EntrySizeCalculator> {
private final Predicate evictable;
EvictableEntrySizeCalculator(Predicate evictable) {
this.evictable = evictable;
}
@Override
public long calculateSize(K key, InternalCacheEntry value) {
return this.evictable.test(key) ? 1 : 0;
}
}
}