2022-03-01 18:31:36 +00:00
|
|
|
#pragma once
|
2023-04-01 03:18:41 +00:00
|
|
|
///@file
|
2022-03-01 18:31:36 +00:00
|
|
|
|
|
|
|
#include "store-api.hh"
|
|
|
|
|
|
|
|
|
|
|
|
namespace nix {
|
|
|
|
|
|
|
|
|
2024-04-22 17:32:21 +00:00
|
|
|
/**
|
|
|
|
* Garbage-collector roots, referring to a store path
|
|
|
|
*/
|
2022-03-01 18:31:36 +00:00
|
|
|
typedef std::unordered_map<StorePath, std::unordered_set<std::string>> Roots;
|
2024-04-22 17:32:21 +00:00
|
|
|
/**
|
|
|
|
* Possible garbage collector roots, referring to any path
|
|
|
|
*/
|
|
|
|
typedef std::unordered_map<Path, std::unordered_set<std::string>> UncheckedRoots;
|
2022-03-01 18:31:36 +00:00
|
|
|
|
|
|
|
|
|
|
|
struct GCOptions
|
|
|
|
{
|
2023-04-07 13:55:28 +00:00
|
|
|
/**
|
|
|
|
* Garbage collector operation:
|
|
|
|
*
|
|
|
|
* - `gcReturnLive`: return the set of paths reachable from
|
|
|
|
* (i.e. in the closure of) the roots.
|
|
|
|
*
|
|
|
|
* - `gcReturnDead`: return the set of paths not reachable from
|
|
|
|
* the roots.
|
|
|
|
*
|
|
|
|
* - `gcDeleteDead`: actually delete the latter set.
|
|
|
|
*
|
|
|
|
* - `gcDeleteSpecific`: delete the paths listed in
|
|
|
|
* `pathsToDelete`, insofar as they are not reachable.
|
|
|
|
*/
|
2022-03-01 18:31:36 +00:00
|
|
|
typedef enum {
|
|
|
|
gcReturnLive,
|
|
|
|
gcReturnDead,
|
|
|
|
gcDeleteDead,
|
|
|
|
gcDeleteSpecific,
|
|
|
|
} GCAction;
|
|
|
|
|
|
|
|
GCAction action{gcDeleteDead};
|
|
|
|
|
2023-04-07 13:55:28 +00:00
|
|
|
/**
|
|
|
|
* If `ignoreLiveness` is set, then reachability from the roots is
|
|
|
|
* ignored (dangerous!). However, the paths must still be
|
|
|
|
* unreferenced *within* the store (i.e., there can be no other
|
|
|
|
* store paths that depend on them).
|
|
|
|
*/
|
2022-03-01 18:31:36 +00:00
|
|
|
bool ignoreLiveness{false};
|
|
|
|
|
2023-04-07 13:55:28 +00:00
|
|
|
/**
|
|
|
|
* For `gcDeleteSpecific`, the paths to delete.
|
|
|
|
*/
|
2022-03-01 18:31:36 +00:00
|
|
|
StorePathSet pathsToDelete;
|
|
|
|
|
2023-04-07 13:55:28 +00:00
|
|
|
/**
|
|
|
|
* Stop after at least `maxFreed` bytes have been freed.
|
|
|
|
*/
|
2022-03-01 18:31:36 +00:00
|
|
|
uint64_t maxFreed{std::numeric_limits<uint64_t>::max()};
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
struct GCResults
|
|
|
|
{
|
2023-04-07 13:55:28 +00:00
|
|
|
/**
|
|
|
|
* Depending on the action, the GC roots, or the paths that would
|
|
|
|
* be or have been deleted.
|
|
|
|
*/
|
2022-03-01 18:31:36 +00:00
|
|
|
PathSet paths;
|
|
|
|
|
2023-04-07 13:55:28 +00:00
|
|
|
/**
|
|
|
|
* For `gcReturnDead`, `gcDeleteDead` and `gcDeleteSpecific`, the
|
|
|
|
* number of bytes that would be or was freed.
|
|
|
|
*/
|
2022-03-01 18:31:36 +00:00
|
|
|
uint64_t bytesFreed = 0;
|
|
|
|
};
|
|
|
|
|
|
|
|
|
2023-03-23 14:06:45 +00:00
|
|
|
/**
|
|
|
|
* Mix-in class for \ref Store "stores" which expose a notion of garbage
|
|
|
|
* collection.
|
|
|
|
*
|
|
|
|
* Garbage collection will allow deleting paths which are not
|
|
|
|
* transitively "rooted".
|
|
|
|
*
|
|
|
|
* The notion of GC roots actually not part of this class.
|
|
|
|
*
|
|
|
|
* - The base `Store` class has `Store::addTempRoot()` because for a store
|
|
|
|
* that doesn't support garbage collection at all, a temporary GC root is
|
|
|
|
* safely implementable as no-op.
|
|
|
|
*
|
|
|
|
* @todo actually this is not so good because stores are *views*.
|
|
|
|
* Some views have only a no-op temp roots even though others to the
|
|
|
|
* same store allow triggering GC. For instance one can't add a root
|
|
|
|
* over ssh, but that doesn't prevent someone from gc-ing that store
|
|
|
|
* accesed via SSH locally).
|
|
|
|
*
|
|
|
|
* - The derived `LocalFSStore` class has `LocalFSStore::addPermRoot`,
|
|
|
|
* which is not part of this class because it relies on the notion of
|
|
|
|
* an ambient file system. There are stores (`ssh-ng://`, for one),
|
|
|
|
* that *do* support garbage collection but *don't* expose any file
|
|
|
|
* system, and `LocalFSStore::addPermRoot` thus does not make sense
|
|
|
|
* for them.
|
|
|
|
*/
|
2022-03-01 18:31:36 +00:00
|
|
|
struct GcStore : public virtual Store
|
|
|
|
{
|
2022-03-09 15:27:48 +00:00
|
|
|
inline static std::string operationName = "Garbage collection";
|
|
|
|
|
2023-04-07 13:55:28 +00:00
|
|
|
/**
|
|
|
|
* Find the roots of the garbage collector. Each root is a pair
|
|
|
|
* `(link, storepath)` where `link` is the path of the symlink
|
|
|
|
* outside of the Nix store that point to `storePath`. If
|
|
|
|
* `censor` is true, privacy-sensitive information about roots
|
|
|
|
* found in `/proc` is censored.
|
|
|
|
*/
|
2022-03-01 18:31:36 +00:00
|
|
|
virtual Roots findRoots(bool censor) = 0;
|
|
|
|
|
2023-04-07 13:55:28 +00:00
|
|
|
/**
|
|
|
|
* Perform a garbage collection.
|
|
|
|
*/
|
2022-03-01 18:31:36 +00:00
|
|
|
virtual void collectGarbage(const GCOptions & options, GCResults & results) = 0;
|
2022-03-08 17:45:19 +00:00
|
|
|
};
|
2022-03-01 18:31:36 +00:00
|
|
|
|
|
|
|
}
|