org.dataloader.ValueCache Maven / Gradle / Ivy
Show all versions of java-dataloader Show documentation
package org.dataloader;
import org.dataloader.annotations.PublicSpi;
import org.dataloader.impl.CompletableFutureKit;
import org.dataloader.impl.NoOpValueCache;
import java.util.ArrayList;
import java.util.List;
import java.util.concurrent.CompletableFuture;
/**
* The {@link ValueCache} is used by data loaders that use caching and want a long-lived or external cache
* of values. The {@link ValueCache} is used as a place to cache values when they come back from an async
* cache store.
*
* It differs from {@link CacheMap} which is in fact a cache of promised values aka {@link CompletableFuture}<V>'s.
*
* {@link ValueCache} is more suited to be a wrapper of a long-lived or externallly cached values. {@link CompletableFuture}s cant
* be easily placed in an external cache outside the JVM say, hence the need for the {@link ValueCache}.
*
* {@link DataLoader}s use a two stage cache strategy if caching is enabled. If the {@link CacheMap} already has the promise to a value
* that is used. If not then the {@link ValueCache} is asked for a value, if it has one then that is returned (and cached as a promise in the {@link CacheMap}.
*
* If there is no value then the key is queued and loaded via the {@link BatchLoader} calls. The returned values will then be stored in
* the {@link ValueCache} and the promises to those values are also stored in the {@link CacheMap}.
*
* The default implementation is a no-op store which replies with the key always missing and doesn't
* store any actual results. This is to avoid duplicating the stored data between the {@link CacheMap}
* out of the box.
*
* The API signature uses {@link CompletableFuture}s because the backing implementation MAY be a remote external cache
* and hence exceptions may happen in retrieving values and they may take time to complete.
*
* @param the type of cache keys
* @param the type of cache values
*
* @author Craig Day
* @author Brad Baker
*/
@PublicSpi
public interface ValueCache {
/**
* Creates a new value cache, using the default no-op implementation.
*
* @param the type of cache keys
* @param the type of cache values
*
* @return the cache store
*/
static ValueCache defaultValueCache() {
//noinspection unchecked
return (ValueCache) NoOpValueCache.NOOP;
}
/**
* Gets the specified key from the value cache. If the key is not present, then the implementation MUST return an exceptionally completed future
* and not null because null is a valid cacheable value. An exceptionally completed future will cause {@link DataLoader} to load the key via batch loading
* instead.
*
*
* @param key the key to retrieve
*
* @return a future containing the cached value (which maybe null) or exceptionally completed future if the key does
* not exist in the cache.
*/
CompletableFuture get(K key);
/**
* Gets the specified keys from the value cache, in a batch call. If your underlying cache cant do batch caching retrieval
* then do not implement this method and it will delegate back to {@link #get(Object)} for you
*
* Each item in the returned list of values is a {@link Try}. If the key could not be found then a failed Try just be returned otherwise
* a successful Try contain the cached value is returned.
*
* You MUST return a List that is the same size as the keys passed in. The code will assert if you do not.
*
* @param keys the list of keys to get cached values for.
*
* @return a future containing a list of {@link Try} cached values for each key passed in.
*/
default CompletableFuture>> getValues(List keys) {
List>> cacheLookups = new ArrayList<>();
for (K key : keys) {
CompletableFuture> cacheTry = Try.tryFuture(get(key));
cacheLookups.add(cacheTry);
}
return CompletableFutureKit.allOf(cacheLookups);
}
/**
* Stores the value with the specified key, or updates it if the key already exists.
*
* @param key the key to store
* @param value the value to store
*
* @return a future containing the stored value for fluent composition
*/
CompletableFuture set(K key, V value);
/**
* Stores the value with the specified keys, or updates it if the keys if they already exist. If your underlying cache cant do batch caching setting
* then do not implement this method and it will delegate back to {@link #set(Object, Object)} for you
*
* @param keys the keys to store
* @param values the values to store
*
* @return a future containing the stored values for fluent composition
*/
default CompletableFuture> setValues(List keys, List values) {
List> cacheSets = new ArrayList<>();
for (int i = 0; i < keys.size(); i++) {
K k = keys.get(i);
V v = values.get(i);
CompletableFuture setCall = set(k, v);
CompletableFuture set = Try.tryFuture(setCall).thenApply(ignored -> v);
cacheSets.add(set);
}
return CompletableFutureKit.allOf(cacheSets);
}
/**
* Deletes the entry with the specified key from the value cache, if it exists.
*
* NOTE: Your implementation MUST not throw exceptions, rather it should return a CompletableFuture that has completed exceptionally. Failure
* to do this may cause the {@link DataLoader} code to not run properly.
*
* @param key the key to delete
*
* @return a void future for error handling and fluent composition
*/
CompletableFuture delete(K key);
/**
* Clears all entries from the value cache.
*
* NOTE: Your implementation MUST not throw exceptions, rather it should return a CompletableFuture that has completed exceptionally. Failure
* to do this may cause the {@link DataLoader} code to not run properly.
*
* @return a void future for error handling and fluent composition
*/
CompletableFuture clear();
}