From a40826eae6d0dd3b60166844a3be048a6dfee40f Mon Sep 17 00:00:00 2001 From: Michael Gibney Date: Fri, 20 Dec 2024 16:12:37 -0500 Subject: [PATCH] SOLR-17597: Fix `CaffeineCache.put()` ramBytes decrement (#2917) --- .../org/apache/solr/search/CaffeineCache.java | 36 ++++++++++--------- .../apache/solr/search/TestCaffeineCache.java | 4 +-- 2 files changed, 22 insertions(+), 18 deletions(-) diff --git a/solr/core/src/java/org/apache/solr/search/CaffeineCache.java b/solr/core/src/java/org/apache/solr/search/CaffeineCache.java index 1d921030ff7..091dd984fa5 100644 --- a/solr/core/src/java/org/apache/solr/search/CaffeineCache.java +++ b/solr/core/src/java/org/apache/solr/search/CaffeineCache.java @@ -232,7 +232,7 @@ private V computeAsync(K key, IOFunction mappingFunction // We reserved the slot, so we do the work V value = mappingFunction.apply(key); future.complete(value); // This will update the weight and expiration - recordRamBytes(key, null, value); + recordRamBytes(key, value); inserts.increment(); return value; } catch (Error | RuntimeException | IOException e) { @@ -262,7 +262,7 @@ public V computeIfAbsent(K key, IOFunction mappingFuncti if (value == null) { return null; } - recordRamBytes(key, null, value); + recordRamBytes(key, value); inserts.increment(); return value; }); @@ -275,30 +275,34 @@ public V computeIfAbsent(K key, IOFunction mappingFuncti public V put(K key, V val) { inserts.increment(); V old = cache.asMap().put(key, val); - recordRamBytes(key, old, val); + // ramBytes decrement for `old` happens via #onRemoval + if (val != old) { + // NOTE: this is conditional on `val != old` in order to work around a + // behavior in the Caffeine library: where there is reference equality + // between `val` and `old`, caffeine does _not_ invoke RemovalListener, + // so the entry is not decremented for the replaced value (hence we + // don't need to increment ram bytes for the entry either). + recordRamBytes(key, val); + } return old; } /** - * Update the estimate of used memory + * Update the estimate of used memory. + * + *

NOTE: old value (in the event of replacement) adjusts {@link #ramBytes} via {@link + * #onRemoval(Object, Object, RemovalCause)} * * @param key the cache key - * @param oldValue the old cached value to decrement estimate (can be null) * @param newValue the new cached value to increment estimate */ - private void recordRamBytes(K key, V oldValue, V newValue) { + private void recordRamBytes(K key, V newValue) { ramBytes.add( RamUsageEstimator.sizeOfObject(newValue, RamUsageEstimator.QUERY_DEFAULT_RAM_BYTES_USED)); - if (oldValue == null) { - ramBytes.add( - RamUsageEstimator.sizeOfObject(key, RamUsageEstimator.QUERY_DEFAULT_RAM_BYTES_USED)); - ramBytes.add(RamUsageEstimator.LINKED_HASHTABLE_RAM_BYTES_PER_ENTRY); - if (async) ramBytes.add(RAM_BYTES_PER_FUTURE); - } else { - ramBytes.add( - -RamUsageEstimator.sizeOfObject( - oldValue, RamUsageEstimator.QUERY_DEFAULT_RAM_BYTES_USED)); - } + ramBytes.add( + RamUsageEstimator.sizeOfObject(key, RamUsageEstimator.QUERY_DEFAULT_RAM_BYTES_USED)); + ramBytes.add(RamUsageEstimator.LINKED_HASHTABLE_RAM_BYTES_PER_ENTRY); + if (async) ramBytes.add(RAM_BYTES_PER_FUTURE); } @Override diff --git a/solr/core/src/test/org/apache/solr/search/TestCaffeineCache.java b/solr/core/src/test/org/apache/solr/search/TestCaffeineCache.java index 87ff29d5e6a..b4c69600fbd 100644 --- a/solr/core/src/test/org/apache/solr/search/TestCaffeineCache.java +++ b/solr/core/src/test/org/apache/solr/search/TestCaffeineCache.java @@ -308,7 +308,7 @@ public void testRamBytesSync() throws IOException { cache.put(0, "test"); long nonEmptySize = cache.ramBytesUsed(); - cache.put(0, "test"); + cache.put(0, random().nextBoolean() ? "test" : "rest"); assertEquals(nonEmptySize, cache.ramBytesUsed()); cache.remove(0); @@ -341,7 +341,7 @@ public void testRamBytesAsync() throws IOException { cache.put(0, "test"); long nonEmptySize = cache.ramBytesUsed(); - cache.put(0, "test"); + cache.put(0, random().nextBoolean() ? "test" : "rest"); assertEquals(nonEmptySize, cache.ramBytesUsed()); cache.remove(0);