io.bdeploy.bhive.op.PruneOperation Maven / Gradle / Ivy
Show all versions of api Show documentation
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;
}
}
}