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

io.bdeploy.bhive.BHive 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
package io.bdeploy.bhive;

import java.io.IOException;
import java.net.URI;
import java.nio.file.FileSystem;
import java.nio.file.FileSystems;
import java.nio.file.Files;
import java.nio.file.Path;
import java.nio.file.Paths;
import java.util.Map;
import java.util.TreeMap;
import java.util.concurrent.Callable;
import java.util.concurrent.ExecutionException;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
import java.util.concurrent.Future;
import java.util.concurrent.TimeUnit;
import java.util.concurrent.atomic.AtomicInteger;
import java.util.function.Predicate;
import java.util.function.Supplier;

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

import com.codahale.metrics.Timer;
import com.google.common.cache.CacheBuilder;
import com.google.common.cache.CacheLoader;
import com.google.common.cache.LoadingCache;

import io.bdeploy.bhive.BHiveTransactions.Transaction;
import io.bdeploy.bhive.audit.AuditParameterExtractor;
import io.bdeploy.bhive.objects.AugmentedObjectDatabase;
import io.bdeploy.bhive.objects.ManifestDatabase;
import io.bdeploy.bhive.objects.ObjectDatabase;
import io.bdeploy.bhive.objects.ObjectManager;
import io.bdeploy.bhive.objects.ReadOnlyObjectDatabase;
import io.bdeploy.bhive.remote.RemoteBHive;
import io.bdeploy.common.ActivityReporter;
import io.bdeploy.common.ActivityReporter.Activity;
import io.bdeploy.common.audit.AuditRecord;
import io.bdeploy.common.audit.AuditRecord.Severity;
import io.bdeploy.common.audit.Auditor;
import io.bdeploy.common.audit.NullAuditor;
import io.bdeploy.common.metrics.Metrics;
import io.bdeploy.common.metrics.Metrics.MetricGroup;
import io.bdeploy.common.util.ExceptionHelper;
import io.bdeploy.common.util.FutureHelper;
import io.bdeploy.common.util.NamedDaemonThreadFactory;
import io.bdeploy.common.util.PathHelper;
import io.bdeploy.common.util.RuntimeAssert;
import io.bdeploy.common.util.Threads;
import io.bdeploy.common.util.ZipHelper;

/**
 * A high level management layer for storage repositories.
 * 

* Encapsulates {@link ObjectDatabase} and {@link ManifestDatabase}, provides * over arching functionality. */ public class BHive implements AutoCloseable, BHiveExecution { private static final String POOLREF = ".poolref"; private static final Logger log = LoggerFactory.getLogger(BHive.class); private final URI uri; private final FileSystem zipFs; private final Path objTmp; private final Path markerTmp; private final BHiveTransactions transactions; private ObjectDatabase objects; private final ManifestDatabase manifests; private final ActivityReporter reporter; private final Auditor auditor; private int parallelism = 4; private boolean auditSlowOps = true; private boolean isPooling = false; private Predicate lockContentValidator = null; private Supplier lockContentSupplier = null; private static final LoadingCache syncCache = CacheBuilder.newBuilder().expireAfterAccess(1, TimeUnit.MINUTES) .build(CacheLoader.from(k -> new Object())); /** * Creates a new hive instance. Supports ZIP and directory hives. *

* To connect to a remote hive instead, use * {@link RemoteBHive#forService(io.bdeploy.common.security.RemoteService, String, ActivityReporter)} */ public BHive(URI uri, Auditor auditor, ActivityReporter reporter) { this.uri = uri; Path relRoot; if (ZipHelper.isZipUri(uri)) { try { if (!uri.getScheme().equals("jar")) { uri = URI.create("jar:" + uri); } Map env = new TreeMap<>(); env.put("create", "true"); env.put("useTempFile", Boolean.TRUE); this.zipFs = FileSystems.newFileSystem(uri, env); } catch (IOException e) { throw new IllegalStateException("cannot open or create ZIP BHive " + uri, e); } relRoot = zipFs.getPath("/"); } else { relRoot = Paths.get(uri); this.zipFs = null; } Path objRoot = relRoot.resolve("objects"); try { objTmp = zipFs == null ? relRoot.resolve("tmp") : Files.createTempDirectory("objdb-"); markerTmp = zipFs == null ? relRoot.resolve("markers") : objTmp.resolve("markers"); PathHelper.mkdirs(markerTmp); } catch (IOException e) { throw new IllegalStateException("Cannot create temporary directory for zipped BHive", e); } this.auditor = auditor == null ? new NullAuditor() : auditor; this.transactions = new BHiveTransactions(this, markerTmp, reporter); if (zipFs != null) { this.objects = new ObjectDatabase(objRoot, objTmp, reporter, transactions); } else { Path pool = getPoolPath(); if (pool != null) { log.trace("Using pool from {}" + pool); this.objects = new AugmentedObjectDatabase(objRoot, objTmp, reporter, transactions, new ReadOnlyObjectDatabase(pool, reporter)); this.isPooling = true; } else { this.objects = new ObjectDatabase(objRoot, objTmp, reporter, transactions); } } this.manifests = new ManifestDatabase(relRoot.resolve("manifests")); this.reporter = reporter; } /** * @param poolPath the pool to enable on this {@link BHive}. * @param force whether to force enabling even if pooling is already configured on the {@link BHive}. */ public synchronized void enablePooling(Path poolPath, boolean force) { if (ZipHelper.isZipUri(uri)) { throw new UnsupportedOperationException("Pooling not supported on ZIP files"); } Path relRoot = Paths.get(uri); Path poolRefFile = relRoot.resolve(POOLREF); if (PathHelper.exists(poolRefFile)) { Path recorded = getPoolPath(); if (!recorded.equals(poolPath) && !force) { throw new UnsupportedOperationException("Pooling is already configured to a different location"); } } PathHelper.mkdirs(poolPath); try { // attach the pool immediately. this.objects = new AugmentedObjectDatabase(relRoot.resolve("objects"), objTmp, reporter, transactions, new ReadOnlyObjectDatabase(poolPath, reporter)); this.isPooling = true; // only persist if attaching the pool worked Files.writeString(poolRefFile, poolPath.toAbsolutePath().normalize().toString()); log.info("Enabled pooling on {}, using pool {}", poolRefFile.getParent(), poolPath); } catch (IOException e) { throw new IllegalStateException("Cannot attach pool configuration to " + poolRefFile, e); } } public synchronized void disablePooling() { Path relRoot = Paths.get(uri); Path poolRefFile = relRoot.resolve(POOLREF); ObjectDatabase newobj = new ObjectDatabase(relRoot.resolve("objects"), objTmp, reporter, transactions); try (Transaction tx = transactions.begin()) { BHivePoolOrganizer.unpoolHive(this, newobj); } catch (Exception e) { throw new IllegalStateException("Cannot unpool BHive", e); } PathHelper.deleteIfExistsRetry(poolRefFile); this.isPooling = false; this.objects = newobj; } /** * @return the configured pool path. Note that this may return a path even though not (yet) pooling due to a missing restart. */ public synchronized Path getPoolPath() { if (ZipHelper.isZipUri(uri)) { return null; } Path poolRefFile = Paths.get(uri).resolve(POOLREF); if (!PathHelper.exists(poolRefFile)) { return null; } try { return Paths.get(Files.readString(poolRefFile).trim()); } catch (Exception e) { log.warn("Cannot read existing pool path!", e); return null; } } /** * @return whether this instance has pooling already enabled. */ public synchronized boolean isPooling() { return isPooling; } public URI getUri() { return uri; } /** * TESTING only, disable unpredictable logs for slow operations depending on machine. */ public void setDisableSlowAudit(boolean disable) { this.auditSlowOps = !disable; } /** * Set the amount of threads to use for parallel-capable file operations. */ public void setParallelism(int parallelism) { this.parallelism = parallelism; } /** * Retrieve the auditor for testing. */ public Auditor getAuditor() { return auditor; } public void addSpawnListener(ManifestSpawnListener listener) { manifests.addSpawnListener(listener); } public void removeSpawnListener(ManifestSpawnListener listener) { manifests.removeSpawnListener(listener); } /** * Sets the supplier that provides the content that is written to a lock file. */ public void setLockContentSupplier(Supplier lockContentSupplier) { this.lockContentSupplier = lockContentSupplier; } /** * Sets the predicate that is used to validate an existing lock file. */ public void setLockContentValidator(Predicate lockContentValidator) { this.lockContentValidator = lockContentValidator; } /** Get the supplier that provides lock file content */ protected Supplier getLockContentSupplier() { return this.lockContentSupplier; } /** Get the predicate that is used to validate an existing lock file. */ protected Predicate getLockContentValidator() { return this.lockContentValidator; } @Override public Object getSynchronizationObject(String name) { try { return syncCache.get(name); } catch (ExecutionException e) { log.warn("Cannot get synchronization object for {}: {}", name, e.toString()); return new Object(); } } /** * Execute the given {@link Operation} on this {@link BHive}. */ @Override public T execute(Operation op) { try { op.initOperation(this); return doExecute(op, 0); } finally { op.closeOperation(); } } /** * Executes the given operation and writes some metrics about the overal execution time. */ private final T doExecute(Operation op, int attempt) { try (Timer.Context timer = Metrics.getMetric(MetricGroup.HIVE).timer(op.getClass().getSimpleName()).time()) { if (op.getClass().getAnnotation(ReadOnlyOperation.class) == null) { auditor.audit(AuditRecord.Builder.fromSystem().setWhat(op.getClass().getSimpleName()) .addParameters(new AuditParameterExtractor().extract(op)).build()); } long start = System.currentTimeMillis(); try { return op.call(); } finally { long timing = System.currentTimeMillis() - start; if (timing > 250 && auditSlowOps) { auditor.audit(AuditRecord.Builder.fromSystem().setWhat(op.getClass().getSimpleName()) .addParameters(new AuditParameterExtractor().extract(op)).setMessage("Long running: " + timing + "ms") .build()); } } } catch (Exception ex) { onOperationFailed(op, ex); if (attempt >= op.retryCount) { throw new IllegalStateException("Operation on hive " + op.hive.getUri() + " failed: " + ex.toString(), ex); } onOperationRetry(op, attempt, ex); return doExecute(op, ++attempt); } } /** Audits the retry of the operation and delays the next retry. */ private void onOperationRetry(Operation op, int attempt, Exception ex) { String retryString = (attempt + 1) + " / " + op.retryCount; auditor.audit(AuditRecord.Builder.fromSystem().setWhat(op.getClass().getSimpleName()).setSeverity(Severity.NORMAL) .setMessage("Retrying operation due to previous failure. Attempt " + retryString).build()); log.warn("Operation failed. Attempt {}", retryString, ex); try (Activity activity = reporter.start("Operation failed (" + retryString + "). Waiting before next retry...", attempt)) { for (int sleep = 0; sleep <= attempt; sleep++) { Threads.sleep(1000); activity.worked(1); } } } /** Audits the failed operation. */ private void onOperationFailed(Operation op, Exception e) { auditor.audit(AuditRecord.Builder.fromSystem().setWhat(op.getClass().getSimpleName()).setSeverity(Severity.ERROR) .addParameters(new AuditParameterExtractor().extract(op)) .setMessage(ExceptionHelper.mapExceptionCausesToReason(e)).build()); } /** * @return the {@link BHiveTransactions} tracker. */ @Override public BHiveTransactions getTransactions() { return transactions; } @Override public void close() { if (zipFs != null) { try { zipFs.close(); } catch (IOException e) { log.warn("Cannot close ZIP FS: {}", uri, e); } PathHelper.deleteRecursiveRetry(objTmp); } manifests.close(); auditor.close(); } /** * Base class for all operations that need to access internals of the * {@link BHive} they are executed on. */ public abstract static class Operation implements Callable, BHiveExecution { private BHive hive; private ObjectManager mgr; private ExecutorService fileOps; private static final AtomicInteger fileOpNum = new AtomicInteger(0); /** Counter how often an operation should be retried. 0 means no retries at all */ private int retryCount = 0; /** * Used internally to associate the operation with the executing hive */ void initOperation(BHive hive) { this.hive = hive; this.fileOps = Executors.newFixedThreadPool(hive.parallelism, new NamedDaemonThreadFactory(() -> "File-OPS-" + fileOpNum.incrementAndGet())); this.mgr = new ObjectManager(hive.objects, hive.manifests, hive.reporter, fileOps); } /** * Dissociates the operation from the associated hive. */ private final void closeOperation() { fileOps.shutdownNow(); hive = null; mgr = null; } /** * @return the {@link ObjectManager} to use when operating on the underlying * {@link ObjectDatabase}. */ protected ObjectManager getObjectManager() { return mgr; } /** * @return the underlying {@link ManifestDatabase} */ protected ManifestDatabase getManifestDatabase() { return hive.manifests; } /** * @return the root path for marker databases which contribute to protected objects. */ protected Path getMarkerRoot() { return hive.markerTmp; } /** * @return the {@link ActivityReporter} to manage {@link Activity}s with. */ protected ActivityReporter getActivityReporter() { return hive.reporter; } /** * @return the {@link Auditor} associated with the current {@link BHive}. */ protected Auditor getAuditor() { return hive.auditor; } /** * Returns the validator to check is a given lock file is still valid. */ protected Predicate getLockContentValidator() { return hive.lockContentValidator; } /** * Returns the supplier that provide the content to be written to the lock file. */ protected Supplier getLockContentSupplier() { return hive.lockContentSupplier; } @Override public BHiveTransactions getTransactions() { return hive.getTransactions(); } @Override public Object getSynchronizationObject(String name) { return hive.getSynchronizationObject(name); } /** * Submit a {@link Runnable} performing a file operation to the pool managing * those operations. * * @param op the operation to run * @return a {@link Future} which can awaited, see * {@link FutureHelper#awaitAll(java.util.Collection)}. */ protected Future submitFileOperation(Runnable op) { return fileOps.submit(op::run); } /** * Execute another {@link Operation} on the {@link BHive} which is currently * associated with this {@link Operation}. */ @Override public X execute(BHive.Operation other) { return hive.execute(other); } /** * Sets the number of times the operation should be retried in case of an exception. The default value is '0' which means * that the operation is not retried on failure. A retry count of '4' means that the operation is executed up to 5 times * before giving up (First run + 4 retries). *

* The default value is '0' which means an operation is not retried in case of an exception. *

* * @param retryCount the number of times to retry the operation */ public Operation setRetryCount(int retryCount) { RuntimeAssert.assertTrue(retryCount >= 0, "Counter must be >=0 but was " + retryCount); this.retryCount = retryCount; return this; } } /** * Base class for operations which require an open transaction, set up by the caller. */ public abstract static class TransactedOperation extends Operation { @Override public final T call() throws Exception { if (!super.getTransactions().hasTransaction()) { throw new IllegalStateException("Operation requires active transaction: " + getClass().getSimpleName()); } return callTransacted(); } /** * Executes the operation. The current thread is guaranteed to be associated with a transaction. */ protected abstract T callTransacted() throws Exception; } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy