org.restheart.cache.impl.CaffeineCache Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of restheart-commons Show documentation
Show all versions of restheart-commons Show documentation
RESTHeart Commons - Common classes for core components and plugins.
/*-
* ========================LICENSE_START=================================
* restheart-commons
* %%
* Copyright (C) 2019 - 2024 SoftInstigate
* %%
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
* =========================LICENSE_END==================================
*/
package org.restheart.cache.impl;
import java.util.AbstractMap;
import java.util.Map;
import java.util.Optional;
import java.util.concurrent.Executor;
import java.util.concurrent.TimeUnit;
import java.util.function.Consumer;
import org.checkerframework.checker.nullness.qual.NonNull;
import org.checkerframework.checker.nullness.qual.Nullable;
import org.restheart.utils.ThreadsUtils;
import com.github.benmanes.caffeine.cache.Cache;
import com.github.benmanes.caffeine.cache.Caffeine;
import com.github.benmanes.caffeine.cache.RemovalCause;
/**
*
* @author Andrea Di Cesare {@literal }
* @param the class of the keys.
* @param the class of the values (is Optional-ized).
*/
public class CaffeineCache implements org.restheart.cache.Cache {
private static final Executor virtualThreadsExecutor = ThreadsUtils.virtualThreadsExecutor();
private final Cache> wrapped;
public CaffeineCache(long size, EXPIRE_POLICY expirePolicy, long ttl) {
var builder = Caffeine.newBuilder().executor(virtualThreadsExecutor);
builder.maximumSize(size);
if (ttl > 0 && expirePolicy == EXPIRE_POLICY.AFTER_WRITE) {
builder.expireAfterWrite(ttl, TimeUnit.MILLISECONDS);
} else if (ttl > 0 && expirePolicy == EXPIRE_POLICY.AFTER_READ) {
builder.expireAfterAccess(ttl, TimeUnit.MILLISECONDS);
}
wrapped = builder.build();
}
public CaffeineCache(long size, EXPIRE_POLICY expirePolicy, long ttl, Consumer>> remover) {
var builder = Caffeine.newBuilder().executor(virtualThreadsExecutor);
builder.maximumSize(size);
if (ttl > 0 && expirePolicy == EXPIRE_POLICY.AFTER_WRITE) {
builder.expireAfterWrite(ttl, TimeUnit.MILLISECONDS);
} else if (ttl > 0 && expirePolicy == EXPIRE_POLICY.AFTER_READ) {
builder.expireAfterAccess(ttl, TimeUnit.MILLISECONDS);
}
wrapped = builder.removalListener((@Nullable K k, @Nullable Optional v, @NonNull RemovalCause cause) -> {
remover.accept(new AbstractMap.SimpleEntry<>(k, v));
}).build();
}
@Override
public Optional get(K key) {
return wrapped.getIfPresent(key);
}
@Override
public synchronized Optional remove(K key) {
var ret = wrapped.getIfPresent(key);
wrapped.invalidate(key);
return ret;
}
@Override
public void put(K key, V value) {
wrapped.put(key, Optional.ofNullable(value));
}
@Override
public void invalidate(K key) {
wrapped.invalidate(key);
}
@Override
public void invalidateAll() {
wrapped.invalidateAll();
}
@Override
public Map> asMap() {
return wrapped.asMap();
}
@Override
public void cleanUp() {
wrapped.cleanUp();
}
}