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

com.palantir.atlasdb.transaction.impl.SweepStrategyManagers Maven / Gradle / Ivy

/*
 * (c) Copyright 2018 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.transaction.impl;

import com.github.benmanes.caffeine.cache.Caffeine;
import com.github.benmanes.caffeine.cache.LoadingCache;
import com.google.common.collect.Maps;
import com.palantir.atlasdb.keyvalue.api.KeyValueService;
import com.palantir.atlasdb.keyvalue.api.TableReference;
import com.palantir.atlasdb.protos.generated.TableMetadataPersistence;
import com.palantir.atlasdb.table.description.SweepStrategy;
import com.palantir.atlasdb.table.description.TableMetadata;
import com.palantir.logsafe.Preconditions;
import com.palantir.logsafe.SafeArg;
import java.time.Duration;
import java.util.Collections;
import java.util.Map;
import java.util.Set;

public final class SweepStrategyManagers {
    private SweepStrategyManagers() {
        //
    }

    public enum CacheWarming {
        FULL,
        NONE,
        ;
    }

    public static SweepStrategyManager createDefault(KeyValueService kvs) {
        return create(kvs, CacheWarming.FULL);
    }

    public static SweepStrategyManager create(KeyValueService kvs, CacheWarming cacheWarming) {
        // Wrap in a RecomputingSupplier for its logic to protect against concurrent initialization
        RecomputingSupplier> sweepStrategySupplierLoadingCache =
                RecomputingSupplier.create(() -> {
                    // On a cache miss, load metadata only for the relevant table. Helpful when many dynamic tables.
                    LoadingCache cache = Caffeine.newBuilder()
                            .expireAfterAccess(Duration.ofDays(1))
                            .build(tableRef -> getSweepStrategy(kvs.getMetadataForTable(tableRef), kvs));

                    // Possibly warm the cache.
                    cache.putAll(getSweepStrategiesForWarmingCache(kvs, cacheWarming));

                    return cache;
                });

        return new SweepStrategyManager() {
            @Override
            public SweepStrategy get(TableReference tableRef) {
                return sweepStrategySupplierLoadingCache.get().get(tableRef);
            }

            @Override
            public void invalidateCaches(Set tableRefs) {
                sweepStrategySupplierLoadingCache.get().invalidateAll(tableRefs);
            }
        };
    }

    public static SweepStrategyManager fromMap(final Map map) {
        return tableRef ->
                Preconditions.checkNotNull(map.get(tableRef), "unknown table", SafeArg.of("tableRef", tableRef));
    }

    public static SweepStrategyManager completelyConservative(KeyValueService kvs) {
        return tableRef -> SweepStrategy.from(TableMetadataPersistence.SweepStrategy.CONSERVATIVE, kvs);
    }

    private static Map getSweepStrategiesForWarmingCache(
            KeyValueService kvs, CacheWarming cacheWarming) {
        switch (cacheWarming) {
            case FULL:
                return Maps.transformValues(
                        kvs.getMetadataForTables(),
                        tableMetadata -> SweepStrategyManagers.getSweepStrategy(tableMetadata, kvs));
            case NONE:
            default:
                return Collections.emptyMap();
        }
    }

    private static SweepStrategy getSweepStrategy(byte[] tableMeta, KeyValueService kvs) {
        if (tableMeta != null && tableMeta.length > 0) {
            return SweepStrategy.from(
                    TableMetadata.BYTES_HYDRATOR.hydrateFromBytes(tableMeta).getSweepStrategy(), kvs);
        } else {
            return SweepStrategy.from(TableMetadataPersistence.SweepStrategy.CONSERVATIVE, kvs);
        }
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy