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

io.bdeploy.bhive.op.PruneOperation 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.op;

import java.nio.file.DirectoryStream;
import java.nio.file.Files;
import java.nio.file.NoSuchFileException;
import java.nio.file.Path;
import java.util.ArrayList;
import java.util.List;
import java.util.Set;
import java.util.SortedMap;
import java.util.SortedSet;
import java.util.TreeMap;
import java.util.TreeSet;
import java.util.concurrent.atomic.AtomicLong;
import java.util.concurrent.atomic.LongAdder;

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

import io.bdeploy.bhive.BHive;
import io.bdeploy.bhive.model.Manifest;
import io.bdeploy.bhive.model.ObjectId;
import io.bdeploy.bhive.objects.MarkerDatabase;
import io.bdeploy.bhive.objects.ObjectDatabase;
import io.bdeploy.common.ActivityReporter.Activity;
import io.bdeploy.common.audit.AuditRecord;
import io.bdeploy.common.audit.AuditRecord.Severity;

/**
 * Removes dangling (unreferenced) objects from the {@link ObjectDatabase}.
 * 

* Returns a map of removed {@link ObjectId}s along with the size of the removed * underlying file. */ public class PruneOperation extends BHive.Operation> { private static final Logger log = LoggerFactory.getLogger(PruneOperation.class); @Override public SortedMap call() throws Exception { SortedMap result = new TreeMap<>(); AtomicLong max = new AtomicLong(-1); LongAdder current = new LongAdder(); try (Activity activity = getActivityReporter().start("Prune (calculating)", max::get, current::sum)) { // Wait for other operations locking the marker root (e.g. another prune). // The CreateObjectMarkersOperation and ClearObjectMarkersOperation will hold // off until the root is unlocked again, so: // 1) No NEW marker databases will be created and no concurrent prune operations will run. // 2) Existing transactions will be allowed to continue using their existing marker databases. // 3) Upon completion, existing trasactions will block removal of the markers until the root is unlocked. execute(new LockDirectoryOperation().setDirectory(getMarkerRoot())); // need to cleanup all marker databases that are left over... long stale = getTransactions().cleanStaleTransactions(); if (stale > 0) { log.warn("{} stale transactions cleaned", stale); } // make sure there is no outdated information getManifestDatabase().invalidateCaches(); SortedSet all; try { // read existing manifests also inside the lock, so we are sure that the existing // manifests and objects are in a consistent state. Set manifests = execute(new ManifestListOperation()); Set referenced; if (!manifests.isEmpty()) { // we list all object, ignoring manifests which disappeared in the meantime (since the list call). referenced = execute(new ObjectListOperation().addManifest(manifests).ignoreMissingManifest(true)); } else { referenced = new TreeSet<>(); } // collect all the objects directly contained which are not referenced all = new TreeSet<>(); getObjectManager().db(db -> { // ATTENTION: this includes only objects which are DIRECTLY in the BHive, which is exactly what we want :) // (i.e. objects which reside in a pool (see AugmentedObjectDatabase) will not be included in the list. db.walkAllObjects(o -> { if (!referenced.contains(o)) { all.add(o); } }); return null; }); // read all existing marker databases and regard any existing object as referenced. try (DirectoryStream markerDbs = Files.newDirectoryStream(getMarkerRoot())) { for (Path markerDb : markerDbs) { if (Files.isDirectory(markerDb)) { MarkerDatabase mdb = new MarkerDatabase(markerDb, getActivityReporter()); mdb.walkAllObjects(all::remove); } } } List auditList = new ArrayList<>(); activity.activity("Prune (cleaning)"); max.set(all.size()); // delete within the lock, just to be sure that nobody "re-needs" one of the objects. for (ObjectId unreferenced : all) { result.put(unreferenced, getObjectManager().db(x -> { try { long sz = x.getObjectSize(unreferenced); x.removeObject(unreferenced); if (auditList.size() < 50) { auditList.add(unreferenced); } return sz; } catch (NoSuchFileException e) { log.debug("To-be-removed object is no longer existing: {}", unreferenced); return (long) 0; } })); current.increment(); } getAuditor().audit(AuditRecord.Builder.fromSystem().setSeverity(Severity.NORMAL) .setWhat(PruneOperation.class.getName()).setMessage("Removed " + result.size() + " Objects ") .addParameter("removed", auditList.toString()).build()); } finally { // Unlocking the root will allow: // 1) Ongoing operations to continue clearing their markers // 2) Other operations locking the root (e.g. another prune operation). execute(new ReleaseDirectoryLockOperation().setDirectory(getMarkerRoot())); } return result; } } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy