Skip to content

Commit

Permalink
ensure that the expiration ticker is not used for stats (fixes #1678)
Browse files Browse the repository at this point in the history
  • Loading branch information
ben-manes committed May 13, 2024
1 parent 46d84e0 commit c8ba23a
Show file tree
Hide file tree
Showing 6 changed files with 58 additions and 8 deletions.
1 change: 1 addition & 0 deletions .gitignore
Original file line number Diff line number Diff line change
Expand Up @@ -14,6 +14,7 @@ jitwatch.out
.vscode
.idea
*.iml
*.hprof
hs_err_pid*.log
replay_pid*.log
simulator/src/main/resources/application.conf
Original file line number Diff line number Diff line change
Expand Up @@ -860,7 +860,7 @@ public Caffeine<K, V> ticker(Ticker ticker) {

Ticker getTicker() {
boolean useTicker = expiresVariable() || expiresAfterAccess()
|| expiresAfterWrite() || refreshAfterWrite() || isRecordingStats();
|| expiresAfterWrite() || refreshAfterWrite();
return useTicker
? (ticker == null) ? Ticker.systemTicker() : ticker
: Ticker.disabledTicker();
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -72,7 +72,6 @@ final class UnboundedLocalCache<K, V> implements LocalCache<K, V> {
final boolean isRecordingStats;
final Executor executor;
final boolean isAsync;
final Ticker ticker;

@Nullable Set<K> keySet;
@Nullable Collection<V> values;
Expand All @@ -85,7 +84,6 @@ final class UnboundedLocalCache<K, V> implements LocalCache<K, V> {
this.removalListener = builder.getRemovalListener(isAsync);
this.isRecordingStats = builder.isRecordingStats();
this.executor = builder.getExecutor();
this.ticker = builder.getTicker();
this.isAsync = isAsync;
}

Expand Down Expand Up @@ -237,7 +235,7 @@ void discardRefresh(Object keyReference) {

@Override
public Ticker statsTicker() {
return ticker;
return isRecordingStats ? Ticker.systemTicker() : Ticker.disabledTicker();
}

/* --------------- JDK8+ Map extensions --------------- */
Expand Down Expand Up @@ -1056,7 +1054,6 @@ Object writeReplace() {
SerializationProxy<K, V> proxy = new SerializationProxy<>();
proxy.isRecordingStats = cache.isRecordingStats;
proxy.removalListener = cache.removalListener;
proxy.ticker = cache.ticker;
return proxy;
}
}
Expand Down Expand Up @@ -1208,7 +1205,6 @@ Object writeReplace() {
SerializationProxy<K, V> proxy = new SerializationProxy<>();
proxy.isRecordingStats = cache.isRecordingStats;
proxy.removalListener = cache.removalListener;
proxy.ticker = cache.ticker;
proxy.async = true;
return proxy;
}
Expand Down Expand Up @@ -1264,7 +1260,6 @@ Object writeReplace() {
proxy.isRecordingStats = cache.isRecordingStats();
proxy.removalListener = cache.removalListener;
proxy.cacheLoader = cacheLoader;
proxy.ticker = cache.ticker;
proxy.async = true;
return proxy;
}
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -29,6 +29,10 @@
import static org.junit.Assert.assertThrows;
import static org.mockito.ArgumentMatchers.any;
import static org.mockito.Mockito.doAnswer;
import static org.mockito.Mockito.times;
import static org.mockito.Mockito.verify;
import static org.mockito.Mockito.verifyNoInteractions;
import static org.mockito.Mockito.when;
import static org.slf4j.event.Level.ERROR;
import static org.slf4j.event.Level.WARN;

Expand All @@ -37,6 +41,7 @@
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import java.lang.reflect.Modifier;
import java.time.Duration;
import java.util.ArrayList;
import java.util.Collection;
import java.util.Collections;
Expand All @@ -49,6 +54,7 @@
import java.util.stream.IntStream;

import org.apache.commons.lang3.tuple.Triple;
import org.mockito.Mockito;
import org.mockito.stubbing.Answer;
import org.testng.annotations.Listeners;
import org.testng.annotations.Test;
Expand All @@ -67,11 +73,14 @@
import com.github.benmanes.caffeine.cache.testing.CacheProvider;
import com.github.benmanes.caffeine.cache.testing.CacheSpec;
import com.github.benmanes.caffeine.cache.testing.CacheSpec.CacheExecutor;
import com.github.benmanes.caffeine.cache.testing.CacheSpec.CacheExpiry;
import com.github.benmanes.caffeine.cache.testing.CacheSpec.Compute;
import com.github.benmanes.caffeine.cache.testing.CacheSpec.ExecutorFailure;
import com.github.benmanes.caffeine.cache.testing.CacheSpec.Expire;
import com.github.benmanes.caffeine.cache.testing.CacheSpec.Implementation;
import com.github.benmanes.caffeine.cache.testing.CacheSpec.Listener;
import com.github.benmanes.caffeine.cache.testing.CacheSpec.Population;
import com.github.benmanes.caffeine.cache.testing.CacheSpec.Stats;
import com.github.benmanes.caffeine.cache.testing.CacheValidationListener;
import com.github.benmanes.caffeine.cache.testing.CheckMaxLogLevel;
import com.github.benmanes.caffeine.cache.testing.CheckNoEvictions;
Expand Down Expand Up @@ -859,6 +868,38 @@ public void removalListener_submit_error_log(Cache<Int, Int> cache, CacheContext
assertThat(event.getThrowable().orElseThrow()).isInstanceOf(RejectedExecutionException.class);
}

@CheckNoStats
@Test(dataProvider = "caches")
@CacheSpec(population = Population.EMPTY, refreshAfterWrite = Expire.DISABLED,
expireAfterAccess = Expire.DISABLED, expireAfterWrite = Expire.DISABLED,
expiry = CacheExpiry.DISABLED, stats = Stats.DISABLED)
public void ticker_noStats(CacheContext context) {
var caffeineTicker = Mockito.mock(Ticker.class);
var guavaTicker = Mockito.mock(com.google.common.base.Ticker.class);
when(guavaTicker.read()).thenAnswer(invocation -> context.ticker().read());
when(caffeineTicker.read()).thenAnswer(invocation -> context.ticker().read());
context.ticker().setAutoIncrementStep(Duration.ofSeconds(1));

if (context.isGuava()) {
context.guava().ticker(guavaTicker);
} else {
context.caffeine().ticker(caffeineTicker);
}

Cache<Int, Int> cache = context.build(key -> key);
assertThat(cache.getIfPresent(context.absentKey())).isNull();
assertThat(cache.get(context.absentKey(), key -> null)).isNull();
assertThat(cache.get(context.absentKey(), key -> key)).isNotNull();
assertThat(cache.getIfPresent(context.absentKey())).isNotNull();

if (context.isGuava()) {
// Used for expiration timestamp and ignored
verify(guavaTicker, times(4)).read();
} else {
verifyNoInteractions(caffeineTicker);
}
}

/* --------------- serialize --------------- */

@CacheSpec
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -18,6 +18,7 @@
import static com.github.benmanes.caffeine.cache.LinkedDequeSubject.deque;
import static com.github.benmanes.caffeine.testing.Awaits.await;
import static com.github.benmanes.caffeine.testing.MapSubject.map;
import static com.google.common.truth.Truth.assertThat;

import java.util.Collection;
import java.util.Map;
Expand All @@ -36,6 +37,7 @@
import com.github.benmanes.caffeine.cache.UnboundedLocalCache.UnboundedLocalAsyncCache;
import com.github.benmanes.caffeine.cache.UnboundedLocalCache.UnboundedLocalAsyncLoadingCache;
import com.github.benmanes.caffeine.cache.UnboundedLocalCache.UnboundedLocalManualCache;
import com.github.benmanes.caffeine.cache.stats.StatsCounter;
import com.github.benmanes.caffeine.cache.testing.Weighers;
import com.google.common.collect.ImmutableTable;
import com.google.common.collect.Sets;
Expand Down Expand Up @@ -110,13 +112,24 @@ public void isValid() {
}
}

private void checkStats(LocalCache<?, ?> cache) {
if (cache.isRecordingStats()) {
assertThat(cache.statsTicker()).isSameInstanceAs(Ticker.systemTicker());
assertThat(cache.statsCounter()).isNotSameInstanceAs(StatsCounter.disabledStatsCounter());
} else {
assertThat(cache.statsTicker()).isSameInstanceAs(Ticker.disabledTicker());
assertThat(cache.statsCounter()).isSameInstanceAs(StatsCounter.disabledStatsCounter());
}
}

/* --------------- Bounded --------------- */

private void checkBounded(BoundedLocalCache<Object, Object> bounded) {
drain(bounded);
checkReadBuffer(bounded);

checkCache(bounded);
checkStats(bounded);
checkTimerWheel(bounded);
checkEvictionDeque(bounded);
}
Expand Down Expand Up @@ -409,5 +422,6 @@ private void checkUnbounded(UnboundedLocalCache<?, ?> unbounded) {
check("value").that(value).isNotNull();
checkIfAsyncValue(value);
});
checkStats(unbounded);
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -253,7 +253,6 @@ private void checkUnboundedAsyncLocalLoadingCache(

private void checkUnboundedLocalCache(
UnboundedLocalCache<?, ?> original, UnboundedLocalCache<?, ?> copy) {
check("ticker").that(copy.ticker).isEqualTo(original.ticker);
check("isRecordingStats").that(copy.isRecordingStats).isEqualTo(original.isRecordingStats);

if (original.removalListener == null) {
Expand Down

0 comments on commit c8ba23a

Please sign in to comment.