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

io.bdeploy.bhive.objects.ManifestDatabase Maven / Gradle / Ivy

Go to download

Public API including dependencies, ready to be used for integrations and plugins.

There is a newer version: 7.4.0
Show newest version
/*
 * Copyright (c) SSI Schaefer IT Solutions GmbH
 */
package io.bdeploy.bhive.objects;

import java.io.IOException;
import java.io.InputStream;
import java.io.UncheckedIOException;
import java.nio.file.Files;
import java.nio.file.NoSuchFileException;
import java.nio.file.Path;
import java.nio.file.StandardCopyOption;
import java.nio.file.StandardOpenOption;
import java.util.ArrayList;
import java.util.List;
import java.util.Set;
import java.util.SortedSet;
import java.util.TreeSet;
import java.util.concurrent.ExecutionException;
import java.util.concurrent.Executors;
import java.util.concurrent.ScheduledExecutorService;
import java.util.concurrent.ScheduledFuture;
import java.util.concurrent.TimeUnit;
import java.util.concurrent.atomic.AtomicBoolean;
import java.util.function.Consumer;
import java.util.stream.Stream;

import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import com.google.common.cache.Cache;
import com.google.common.cache.CacheBuilder;
import com.google.common.util.concurrent.UncheckedExecutionException;

import io.bdeploy.bhive.ManifestSpawnListener;
import io.bdeploy.bhive.model.Manifest;
import io.bdeploy.bhive.model.Manifest.Key;
import io.bdeploy.bhive.util.StorageHelper;
import io.bdeploy.common.util.NamedDaemonThreadFactory;
import io.bdeploy.common.util.PathHelper;

/**
 * Stores and manages {@link Manifest}s. Storage happens in files distributed in
 * a database based on name and tag of each {@link Manifest}. This allows
 * concurrent updates to the database even from different processes.
 */
public class ManifestDatabase extends LockableDatabase implements AutoCloseable {

    private static final Logger log = LoggerFactory.getLogger(ManifestDatabase.class);

    private final Path root;
    private final Path tmp;

    private final List listeners = new ArrayList<>();

    private ScheduledFuture schedNotify;
    private final List added = new ArrayList<>();
    private static final ScheduledExecutorService NOTIFY_POOL = Executors.newScheduledThreadPool(2,
            new NamedDaemonThreadFactory("Manifest DB Notifier"));

    /**
     * A cache for Manifest objects which need to actually be loaded from disk.
     * 

* Assuming a max object size of ~4K (manifest includes cached references), this cache would grow to ~10MB. */ private final Cache manifestCache = CacheBuilder.newBuilder().maximumSize(2_500).build(); private final Cache> manifestListCache = CacheBuilder.newBuilder() .expireAfterWrite(10, TimeUnit.MINUTES).build(); /** * @param root the root path of the database, created empty if it does not yet * exist */ public ManifestDatabase(Path root) { super(root); this.root = root; this.tmp = root.resolve(".tmp"); if (!PathHelper.exists(root)) { PathHelper.mkdirs(root); } if (!PathHelper.exists(tmp)) { PathHelper.mkdirs(tmp); } } @Override public void close() { if (schedNotify != null && !schedNotify.isDone()) { schedNotify.cancel(false); } } public void addSpawnListener(ManifestSpawnListener listener) { listeners.add(listener); } public void removeSpawnListener(ManifestSpawnListener listener) { listeners.remove(listener); } public void scheduleNotify(Manifest.Key key) { if (listeners.isEmpty()) { return; } synchronized (added) { if (log.isDebugEnabled()) { log.debug("Adding {} to notify queue", key); } added.add(key); } if (schedNotify != null && !schedNotify.isDone()) { if (log.isDebugEnabled()) { log.debug("Cancel existing scheduled notify"); } schedNotify.cancel(false); } schedNotify = NOTIFY_POOL.schedule(() -> { List copy; synchronized (added) { if (log.isDebugEnabled()) { log.debug("Notify {} listeners: {}", listeners.size(), added); } copy = new ArrayList<>(added); added.clear(); } for (ManifestSpawnListener listener : listeners) { try { listener.spawn(copy); } catch (Exception e) { log.warn("Exception in manifest listener", e); } } }, 100, TimeUnit.MILLISECONDS); } /** * @param key the manifest key * @return the {@link Path} to the file storing the actual {@link Manifest}. */ private Path getPathForKey(Manifest.Key key) { return root.resolve(key.getName()).resolve(key.getTag()); } /** * @param key the manifest key to check * @return whether the {@link Manifest} exists in the database. */ public boolean hasManifest(Manifest.Key key) { return PathHelper.exists(getPathForKey(key)); } /** * Concurrent-save adds a {@link Manifest} to the database. * * @param manifest the manifest to store in the database. * @return true if the manifest was added, false if ignoreExisting was set and the manifest was not * added (by this call). */ public boolean addManifest(Manifest manifest, boolean ignoreExisting) { AtomicBoolean added = new AtomicBoolean(true); locked(() -> { if (hasManifest(manifest.getKey())) { if (ignoreExisting) { added.set(false); return; } else { throw new IllegalArgumentException("Manifest " + manifest.getKey() + " already present."); } } Path pathForKey = getPathForKey(manifest.getKey()); PathHelper.mkdirs(pathForKey.getParent()); // unfortunately there is no better way to detect a 'ZipPath' as the class is not accessible directly. if (pathForKey.getClass().getSimpleName().contains("Zip")) { // in case of ZIP files we cannot move afterwards, so we need to write directly Files.write(pathForKey, StorageHelper.toRawBytes(manifest), StandardOpenOption.SYNC, StandardOpenOption.CREATE, StandardOpenOption.TRUNCATE_EXISTING); } else { Path tmpFile = Files.createTempFile(tmp, "mf-", ".tmp"); try { Files.write(tmpFile, StorageHelper.toRawBytes(manifest), StandardOpenOption.SYNC, StandardOpenOption.CREATE, StandardOpenOption.TRUNCATE_EXISTING); PathHelper.moveRetry(tmpFile, pathForKey, StandardCopyOption.ATOMIC_MOVE); } catch (Throwable t) { PathHelper.deleteRecursiveRetry(tmpFile); throw t; } } manifestCache.put(manifest.getKey(), manifest); updateListCaches(manifest.getKey(), c -> c.add(manifest.getKey())); scheduleNotify(manifest.getKey()); }); return added.get(); } /** * Concurrent-save removes a manifest from the database. Does not remove * underlying objects from any {@link ObjectDatabase}, just removes the manifest * entry (which is the "root-anchor" to those objects). * * @param key the manifest to remove */ public void removeManifest(Manifest.Key key) { locked(() -> { PathHelper.deleteIfExistsRetry(getPathForKey(key)); manifestCache.invalidate(key); updateListCaches(key, c -> c.remove(key)); }); } /** * Allows manipulation of all list cache segments which may contain the given key. */ private void updateListCaches(Manifest.Key key, Consumer> manipulator) { Path resolved = root.resolve(key.getName()).normalize(); synchronized (manifestListCache) { for (Path p : manifestListCache.asMap().keySet()) { if (log.isTraceEnabled()) { log.trace("Path check {} vs. {}", resolved, p); } if (resolved.startsWith(p)) { Set segment = manifestListCache.getIfPresent(p); if (segment != null) { if (log.isTraceEnabled()) { log.trace("Update cache for path {}, manipulate {}", p, key); } manipulator.accept(segment); } } } } } /** * @return all {@link Key}s found in the database's filesystem. */ public Set getAllManifests() { // structure is dir:root/dir:name/dir:name/file:tag return collectManifestsCached(root); } private Set collectManifestsCached(Path r) { Path normalized = r.normalize(); try { synchronized (manifestListCache) { // need to copy the internal list cache, as this will be updated over time. return new TreeSet<>(manifestListCache.get(normalized, () -> collectManifests(normalized))); } } catch (ExecutionException e) { log.warn("Cannot fetch manifest list cache", e); return collectManifests(normalized); // fallback. } } private Set collectManifests(Path scanRoot) { Set result = new TreeSet<>(); try { long xctpCount = 0; do { if (!Files.isDirectory(scanRoot)) { // none exist in the given namespace. return result; } try (Stream walk = Files.walk(scanRoot)) { walk.filter(Files::isRegularFile).forEach(f -> { if (f.startsWith(this.tmp)) { return; } Path rel = root.relativize(f); if (rel.getParent() == null) { // file in the db-root, cannot be a manifest. return; } // windows paths contain '\' - replace it to get proper names. String manifestName = rel.getParent().toString().replace('\\', '/'); String manifestTag = rel.getFileName().toString(); result.add(new Manifest.Key(manifestName, manifestTag)); }); return result; } catch (UncheckedIOException | NoSuchFileException e) { // something was removed in the middle of the walk... retry. if (!(e instanceof NoSuchFileException || e.getCause() instanceof NoSuchFileException || xctpCount++ > 10)) { throw e; } } } while (true); } catch (IOException e) { throw new IllegalStateException("Error reading manifest database", e); } } /** * @param name the name to collect for. May be any number of name segments (must be complete segments). e.g 'my/manifest' will * match the same (but potentially more) than 'my/manifest/name', but none of the two will match 'my/manifestname'. * If the name includes a ':', it is treated as a fully qualified manifest key. * @return the list of manifests which matched the name. */ public Set getAllForName(String name) { if (name.contains(":")) { // if the name is fully qualified, it is either there or not. SortedSet result = new TreeSet<>(); Manifest.Key key = Manifest.Key.parse(name); if (hasManifest(key)) { result.add(key); } return result; } Path namedRoot = root.resolve(name); return collectManifestsCached(namedRoot); } /** * @param key the key of the manifest to load * @return the {@link Manifest} loaded from its backing file. */ public Manifest getManifest(Manifest.Key key) { try { return manifestCache.get(key, () -> { if (!hasManifest(key)) { throw new IllegalArgumentException("Don't have manifest " + key); } try (InputStream is = Files.newInputStream(getPathForKey(key))) { return StorageHelper.fromStream(is, Manifest.class); } catch (IOException e) { throw new IllegalStateException("Cannot read manifest " + key, e); } }); } catch (ExecutionException | UncheckedExecutionException e) { throw new IllegalStateException("Cannot load manifest into cache: " + key, e); } } /** * Invalidates all cached data. */ public void invalidateCaches() { this.manifestCache.invalidateAll(); synchronized (manifestListCache) { this.manifestListCache.invalidateAll(); } } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy