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

io.bdeploy.bhive.BHiveTransactions 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.nio.file.Files;
import java.nio.file.Path;
import java.util.Collections;
import java.util.List;
import java.util.Map;
import java.util.Stack;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.atomic.LongAdder;

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

import io.bdeploy.bhive.model.ObjectId;
import io.bdeploy.bhive.objects.MarkerDatabase;
import io.bdeploy.bhive.objects.ObjectDatabase;
import io.bdeploy.bhive.op.AwaitDirectoryLockOperation;
import io.bdeploy.common.ActivityReporter;
import io.bdeploy.common.util.PathHelper;
import io.bdeploy.common.util.StringHelper;
import io.bdeploy.common.util.UuidHelper;

/**
 * Keeps track of running operations in the JVM.
 * 

* Uses MarkerDatabase to synchronize with other JVMs which might have operations running as well. */ public class BHiveTransactions { private static final Logger log = LoggerFactory.getLogger(BHiveTransactions.class); private static final String TX_PID_FILE = "tx.pid"; private final InheritableThreadLocal> transactions = new InheritableThreadLocal<>(); private final Map dbs = new ConcurrentHashMap<>(); private final BHive hive; private final ActivityReporter reporter; private final Path markerRoot; public BHiveTransactions(BHive hive, Path markerRoot, ActivityReporter reporter) { this.hive = hive; this.markerRoot = markerRoot; this.reporter = reporter; } /** * Transactions are inherited by default. If this is not desired, this method can be used * to explicitly detach the current Thread from its parent (and potential sibling threads). */ public void detachThread() { transactions.set(new Stack<>()); } private Stack getOrCreate() { Stack result = transactions.get(); if (result == null) { result = new Stack<>(); transactions.set(result); } return result; } /** * @param object the object which should be considered "touched", i.e. inserted. */ public void touchObject(ObjectId object) { Stack all = transactions.get(); String id = (all == null || all.isEmpty()) ? null : all.peek(); if (id == null) { throw new IllegalStateException("No transaction active while inserting object."); } MarkerDatabase mdb = dbs.get(id); if (mdb == null) { throw new IllegalStateException("Transaction database missing for transaction " + id); } mdb.addMarker(object); } /** * @return whether the current thread has an associated transaction. */ public boolean hasTransaction() { Stack stack = transactions.get(); return stack != null && !stack.isEmpty(); } /** * Begins a new transaction on this thread. *

* Inserts on the {@link ObjectDatabase} of a {@link BHive} will use this transaction to keep track of objects inserted. * * @return a {@link Transaction} which will cleanup associated resources when closed. */ public Transaction begin() { hive.execute(new AwaitDirectoryLockOperation().setDirectory(markerRoot)); String txid = UuidHelper.randomId(); getOrCreate().push(txid); Path mdbPath = markerRoot.resolve(txid); dbs.put(txid, new MarkerDatabase(mdbPath, reporter)); if (hive.getLockContentSupplier() != null) { String txValidationContent = hive.getLockContentSupplier().get(); try { Files.write(mdbPath.resolve(TX_PID_FILE), Collections.singletonList(txValidationContent)); } catch (IOException e) { log.debug("Cannot write transaction validation information", e); } } if (log.isTraceEnabled()) { log.trace("Starting transaction {}", txid, new RuntimeException("Starting Transaction")); } return () -> { hive.execute(new AwaitDirectoryLockOperation().setDirectory(markerRoot)); Stack stack = transactions.get(); if (stack == null || stack.isEmpty()) { throw new IllegalStateException("No transaction has been started on this thread!"); } String top = stack.peek(); if (!top.equals(txid)) { log.warn("Out-of-order transaction found: {}, expected: {}", top, txid); } if (log.isTraceEnabled()) { log.trace("Ending transaction {}", txid, new RuntimeException("Ending Transaction")); } stack.remove(txid); dbs.remove(txid); Path mdb = mdbPath; if (!Files.isDirectory(mdb)) { return; // nothing to clean. } PathHelper.deleteRecursiveRetry(mdb); }; } /** * This method can be used to detect and clean stale transactions which may keep (potentially damaged) objects alive. * * @return the amount of stale transactions found (and removed). */ public long cleanStaleTransactions() { LongAdder amount = new LongAdder(); try { Files.list(markerRoot).forEach(p -> { if (Files.isDirectory(p) && PathHelper.exists(p.resolve(TX_PID_FILE))) { // TX PID exists, validate. if (hive.getLockContentValidator() == null) { return; // cannot validate if transaction is still valid - should *never* happen. } try { List lines = Files.readAllLines(p.resolve(TX_PID_FILE)); if (!lines.isEmpty() && !StringHelper.isNullOrEmpty(lines.get(0)) && !hive.getLockContentValidator().test(lines.get(0))) { log.warn("Stale transaction detected, removing {}", p.getFileName()); PathHelper.deleteRecursiveRetry(p); amount.increment(); } } catch (IOException e) { log.warn("Problem determining stale transactions", e); } } else if (Files.isDirectory(p)) { // TX PID does NOT exist, this is a *very* old or completely outdated transaction. log.warn("Stale transaction detected, removing {}" + p.getFileName()); PathHelper.deleteRecursiveRetry(p); amount.increment(); } }); } catch (IOException e) { log.warn("Cannot list potentially stale transaction databases", e); } return amount.sum(); } /** * Represents a writing transaction in the BHive. */ public interface Transaction extends AutoCloseable { @Override public void close(); } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy