All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.palantir.atlasdb.cache.DefaultOffHeapCache Maven / Gradle / Ivy

/*
 * (c) Copyright 2020 Palantir Technologies Inc. All rights reserved.
 *
 * 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.
 */

package com.palantir.atlasdb.cache;

import com.codahale.metrics.Counter;
import com.codahale.metrics.Gauge;
import com.codahale.metrics.Meter;
import com.codahale.metrics.MetricRegistry;
import com.google.common.collect.ImmutableMap;
import com.google.common.collect.Maps;
import com.google.common.collect.Sets;
import com.google.common.util.concurrent.Futures;
import com.google.protobuf.ByteString;
import com.palantir.atlasdb.autobatch.Autobatchers;
import com.palantir.atlasdb.autobatch.CoalescingRequestFunction;
import com.palantir.atlasdb.autobatch.DisruptorAutobatcher;
import com.palantir.atlasdb.persistent.api.PersistentStore;
import com.palantir.common.annotations.ImmutablesStyles.PackageVisibleImmutablesStyle;
import com.palantir.common.streams.KeyedStream;
import com.palantir.logsafe.exceptions.SafeIllegalArgumentException;
import com.palantir.logsafe.logger.SafeLogger;
import com.palantir.logsafe.logger.SafeLoggerFactory;
import com.palantir.tritium.metrics.registry.MetricName;
import com.palantir.tritium.metrics.registry.TaggedMetricRegistry;
import java.util.List;
import java.util.Map;
import java.util.Optional;
import java.util.Set;
import java.util.concurrent.atomic.AtomicInteger;
import java.util.concurrent.atomic.AtomicReference;
import java.util.function.LongSupplier;
import java.util.stream.Collectors;
import org.immutables.value.Value;

public final class DefaultOffHeapCache implements OffHeapCache {
    private static final SafeLogger log = SafeLoggerFactory.get(DefaultOffHeapCache.class);
    private static final String BATCHER_PURPOSE = "off-heap-cache";
    private static final MetricName CACHE_HIT = constructCacheMetricName("cacheHit");
    private static final MetricName CACHE_MISS = constructCacheMetricName("cacheMiss");
    private static final MetricName CACHE_NUKE = constructCacheMetricName("cacheNuke");
    private static final MetricName CACHE_SIZE = constructCacheMetricName("cacheSize");

    private final PersistentStore persistentStore;
    private final EntryMapper entryMapper;
    private final LongSupplier maxSize;
    private final AtomicReference cacheDescriptor = new AtomicReference<>();
    private final DisruptorAutobatcher, Void> valuePutter;
    private final Meter cacheHit;
    private final Meter cacheMiss;
    private final Counter cacheNuke;

    public interface EntryMapper {
        ByteString serializeKey(K key);

        K deserializeKey(ByteString key);

        ByteString serializeValue(K key, V value);

        V deserializeValue(ByteString key, ByteString value);
    }

    public static  OffHeapCache create(
            PersistentStore persistentStore,
            EntryMapper entryMapper,
            TaggedMetricRegistry taggedMetricRegistry,
            LongSupplier maxSize) {
        PersistentStore.Handle handle = persistentStore.createSpace();

        CacheDescriptor cacheDescriptor = ImmutableCacheDescriptor.builder()
                .currentSize(new AtomicInteger())
                .handle(handle)
                .build();

        return new DefaultOffHeapCache<>(persistentStore, entryMapper, cacheDescriptor, maxSize, taggedMetricRegistry);
    }

    private DefaultOffHeapCache(
            PersistentStore persistentStore,
            EntryMapper entryMapper,
            CacheDescriptor cacheDescriptor,
            LongSupplier maxSize,
            TaggedMetricRegistry taggedMetricRegistry) {
        this.persistentStore = persistentStore;
        this.entryMapper = entryMapper;
        this.cacheDescriptor.set(cacheDescriptor);
        this.maxSize = maxSize;
        this.cacheHit = taggedMetricRegistry.meter(CACHE_HIT);
        this.cacheMiss = taggedMetricRegistry.meter(CACHE_MISS);
        this.cacheNuke = taggedMetricRegistry.counter(CACHE_NUKE);
        this.valuePutter = Autobatchers.coalescing(new WriteBatcher<>(this))
                .safeLoggablePurpose(BATCHER_PURPOSE)
                .build();
        Gauge cacheSizeGauge =
                () -> this.cacheDescriptor.get().currentSize().intValue();
        taggedMetricRegistry.registerWithReplacement(CACHE_SIZE, cacheSizeGauge);
    }

    @Override
    public void clear() {
        CacheDescriptor proposedCacheDescriptor = createNamespaceAndConstructCacheProposal(persistentStore);

        CacheDescriptor previous = cacheDescriptor.getAndUpdate(prev -> proposedCacheDescriptor);
        if (previous != null) {
            persistentStore.dropStoreSpace(previous.handle());
        }
    }

    @Override
    public void put(K key, V value) {
        Futures.getUnchecked(valuePutter.apply(Maps.immutableEntry(key, value)));
    }

    @Override
    public Optional get(K key) {
        ByteString serializedKey = entryMapper.serializeKey(key);
        Optional value = persistentStore.get(cacheDescriptor.get().handle(), serializedKey);
        getCacheMeter(value.isPresent()).mark();
        return value.map(v -> entryMapper.deserializeValue(serializedKey, v));
    }

    private Meter getCacheMeter(boolean cacheOutcome) {
        return cacheOutcome ? cacheHit : cacheMiss;
    }

    private static CacheDescriptor createNamespaceAndConstructCacheProposal(PersistentStore persistentStore) {
        PersistentStore.Handle proposal = persistentStore.createSpace();
        return ImmutableCacheDescriptor.builder()
                .currentSize(new AtomicInteger())
                .handle(proposal)
                .build();
    }

    private static MetricName constructCacheMetricName(String metricSuffix) {
        return MetricName.builder()
                .safeName(MetricRegistry.name(DefaultOffHeapCache.class, metricSuffix))
                .build();
    }

    private static class WriteBatcher implements CoalescingRequestFunction, Void> {
        DefaultOffHeapCache offHeapCache;

        WriteBatcher(DefaultOffHeapCache offHeapCache) {
            this.offHeapCache = offHeapCache;
        }

        @Override
        public Map, Void> apply(Set> request) {
            CacheDescriptor cacheDescriptor = offHeapCache.cacheDescriptor.get();
            if (cacheDescriptor.currentSize().get() >= offHeapCache.maxSize.getAsLong()) {
                offHeapCache.cacheNuke.inc();
                offHeapCache.clear();
            }
            cacheDescriptor = offHeapCache.cacheDescriptor.get();
            Set> serializedRequest =
                    request.stream().map(this::serializeEntry).collect(Collectors.toSet());
            try {
                List toWrite =
                        serializedRequest.stream().map(Map.Entry::getKey).collect(Collectors.toList());
                Map response =
                        offHeapCache.persistentStore.get(cacheDescriptor.handle(), toWrite);

                int sizeIncrease = Sets.difference(request, response.entrySet()).size();
                cacheDescriptor.currentSize().addAndGet(sizeIncrease);
                offHeapCache.persistentStore.put(cacheDescriptor.handle(), ImmutableMap.copyOf(serializedRequest));
            } catch (SafeIllegalArgumentException exception) {
                // happens when a store is dropped by a concurrent call to clear
                log.warn("Clear called concurrently, writing failed", exception);
            }
            return KeyedStream.of(request.stream()).map(value -> null).collectToMap();
        }

        private Map.Entry serializeEntry(Map.Entry entry) {
            return Maps.immutableEntry(
                    offHeapCache.entryMapper.serializeKey(entry.getKey()),
                    offHeapCache.entryMapper.serializeValue(entry.getKey(), entry.getValue()));
        }
    }

    @Value.Immutable
    @PackageVisibleImmutablesStyle
    interface CacheDescriptor {
        AtomicInteger currentSize();

        PersistentStore.Handle handle();
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy