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

com.yahoo.vespa.hosted.provision.persistence.CuratorDatabase Maven / Gradle / Ivy

// Copyright 2017 Yahoo Holdings. Licensed under the terms of the Apache 2.0 license. See LICENSE in the project root.
package com.yahoo.vespa.hosted.provision.persistence;

import com.google.common.collect.ImmutableList;
import com.yahoo.path.Path;
import com.yahoo.transaction.NestedTransaction;
import com.yahoo.vespa.curator.Curator;
import com.yahoo.vespa.curator.Lock;
import com.yahoo.vespa.curator.recipes.CuratorCounter;
import com.yahoo.vespa.curator.transaction.CuratorTransaction;

import java.time.Duration;
import java.util.Arrays;
import java.util.List;
import java.util.Map;
import java.util.Optional;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.atomic.AtomicReference;

/**
 * This encapsulated the curator database of the node repo.
 * It serves reads from an in-memory cache of the content which is invalidated when changed on another node
 * using a global, shared counter. The counter is updated on all write operations, ensured by wrapping write
 * operations in a 2pc transaction containing the counter update.
 *
 * @author bratseth
 */
public class CuratorDatabase {

    private final Curator curator;

    /** A shared atomic counter which is incremented every time we write to the curator database */
    private final CuratorCounter changeGenerationCounter;

    /** A partial cache of the Curator database, which is only valid if generations match */
    private final AtomicReference cache = new AtomicReference<>();

    /** Whether we should return data from the cache or always read fro ZooKeeper */
    private final boolean useCache;

    private final Object cacheCreationLock = new Object();

    /**
     * All keys, to allow reentrancy.
     * This will grow forever with the number of applications seen, but this should be too slow to be a problem.
     */
    private final ConcurrentHashMap locks = new ConcurrentHashMap<>();

    /**
     * Creates a curator database
     *
     * @param curator the curator instance
     * @param root the file system root of the db
     */
    public CuratorDatabase(Curator curator, Path root, boolean useCache) {
        this.useCache = useCache;
        this.curator = curator;
        changeGenerationCounter = new CuratorCounter(curator, root.append("changeCounter").getAbsolute());
        cache.set(newCache(changeGenerationCounter.get()));
    }

    /** Create a reentrant lock */
    // Locks are not cached in the in-memory state
    public Lock lock(Path path, Duration timeout) {
        Lock lock = locks.computeIfAbsent(path, (pathArg) -> new Lock(pathArg.getAbsolute(), curator));
        lock.acquire(timeout);
        return lock;
    }

    // --------- Write operations ------------------------------------------------------------------------------
    // These must either create a nested transaction ending in a counter increment or not depend on prior state

    /**
     * Creates a new curator transaction against this database and adds it to the given nested transaction.
     * Important: It is the nested transaction which must be committed - never the curator transaction directly.
     */
    public CuratorTransaction newCuratorTransactionIn(NestedTransaction transaction) {
        // Add a counting transaction first, to make sure we always invalidate the current state on any transaction commit
        transaction.add(new EagerCountingCuratorTransaction(changeGenerationCounter), CuratorTransaction.class);
        CuratorTransaction curatorTransaction = new CuratorTransaction(curator);
        transaction.add(curatorTransaction);
        return curatorTransaction;
    }

    /** Creates a path in curator and all its parents as necessary. If the path already exists this does nothing. */
    // As this operation does not depend on the prior state we do not need to increment the write counter
    public void create(Path path) {
        curator.create(path);
    }

    // --------- Read operations -------------------------------------------------------------------------------
    // These can read from the memory file system, which accurately mirrors the ZooKeeper content IF

    /** Returns the immediate, local names of the children under this node in any order */
    public List getChildren(Path path) { return getCache().getChildren(path); }

    public Optional getData(Path path) { return getCache().getData(path); }

    private static class CacheAndGeneration {
        public CacheAndGeneration(CuratorDatabaseCache cache, long generation)
        {
            this.cache = cache;
            this.generation = generation;
        }
        public boolean expired() {
            return generation != cache.generation();
        }
        public CuratorDatabaseCache validCache() {
            if (expired()) {
                throw new IllegalStateException("The cache has generation " + cache.generation() +
                        " while the root genration counter in zookeeper says " + generation +
                        ". That is totally unacceptable and must be a sever programming error in my close vicinity.");
            }
            return cache;
        }

        private CuratorDatabaseCache cache;
        private long generation;
    }
    private CacheAndGeneration getCacheSnapshot() {
        return new CacheAndGeneration(cache.get(), changeGenerationCounter.get());
    }
    private CuratorDatabaseCache getCache() {
        CacheAndGeneration cacheAndGeneration = getCacheSnapshot();
        while (cacheAndGeneration.expired()) {
            synchronized (cacheCreationLock) { // Prevent a race for creating new caches
                cacheAndGeneration = getCacheSnapshot();
                if (cacheAndGeneration.expired()) {
                    cache.set(newCache(changeGenerationCounter.get()));
                    cacheAndGeneration = getCacheSnapshot();
                }
            }
        }
        return cacheAndGeneration.validCache();
    }

    /** Caches must only be instantiated using this method */
    private CuratorDatabaseCache newCache(long generation) {
        return useCache ? new CuratorDatabaseCache(generation, curator) : new DeactivatedCache(generation, curator);
    }

    /**
     * A thread safe partial snapshot of the curator database content with a given generation.
     * This is merely a recording of what Curator returned at various points in time when 
     * it had the counter at this generation.
     */
    private static class CuratorDatabaseCache {

        private final long generation;
        
        /** The curator instance used to fetch missing data */
        protected final Curator curator;

        // The data of this partial state mirror. The amount of curator state mirrored in this may grow
        // over time by multiple threads. Growing is the only operation permitted by this.
        // The content of the map is immutable.
        private final Map> children = new ConcurrentHashMap<>();
        private final Map> data = new ConcurrentHashMap<>();

        /** Create an empty snapshot at a given generation (as an empty snapshot is a valid partial snapshot) */
        public CuratorDatabaseCache(long generation, Curator curator) {
            this.generation = generation;
            this.curator = curator;
        }

        public long generation() { return generation; }

        /**
         * Returns the children of this path, which may be empty.
         * Returns null only if it is not present in this state mirror
         */
        public List getChildren(Path path) { 
            return children.computeIfAbsent(path, key -> ImmutableList.copyOf(curator.getChildren(path)));
        }

        /**
         * Returns the content of this child - which may be empty.
         * Returns null only if it is not present in this state mirror
         */
        public Optional getData(Path path) {
            return data.computeIfAbsent(path, key -> curator.getData(path).map(data -> Arrays.copyOf(data, data.length)));
        }

    }

    /** An implementation of the curator database cache which does no caching */
    private static class DeactivatedCache extends CuratorDatabaseCache {
        
        public DeactivatedCache(long generation, Curator curator) { super(generation, curator); }

        @Override
        public List getChildren(Path path) { return curator.getChildren(path); }

        @Override
        public Optional getData(Path path) { return curator.getData(path); }

    }

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy