2015-05-21 16:21:38 +03:00
|
|
|
#include "store-api.hh"
|
2015-05-21 17:26:03 +03:00
|
|
|
#include "profiles.hh"
|
2015-04-22 15:21:52 +03:00
|
|
|
#include "shared.hh"
|
|
|
|
#include "globals.hh"
|
2020-03-30 15:29:29 +03:00
|
|
|
#include "../nix/legacy.hh"
|
2015-04-22 15:21:52 +03:00
|
|
|
|
|
|
|
#include <iostream>
|
2015-10-18 22:04:24 +03:00
|
|
|
#include <cerrno>
|
2015-04-22 15:21:52 +03:00
|
|
|
|
|
|
|
using namespace nix;
|
|
|
|
|
2015-05-21 17:26:03 +03:00
|
|
|
std::string deleteOlderThan;
|
2015-04-22 15:21:52 +03:00
|
|
|
bool dryRun = false;
|
|
|
|
|
|
|
|
|
|
|
|
/* If `-d' was specified, remove all old generations of all profiles.
|
|
|
|
* Of course, this makes rollbacks to before this point in time
|
|
|
|
* impossible. */
|
|
|
|
|
|
|
|
void removeOldGenerations(std::string dir)
|
|
|
|
{
|
2015-05-21 16:04:05 +03:00
|
|
|
if (access(dir.c_str(), R_OK) != 0) return;
|
|
|
|
|
|
|
|
bool canWrite = access(dir.c_str(), W_OK) == 0;
|
|
|
|
|
2015-04-22 15:21:52 +03:00
|
|
|
for (auto & i : readDirectory(dir)) {
|
|
|
|
checkInterrupt();
|
|
|
|
|
2015-05-21 16:04:05 +03:00
|
|
|
auto path = dir + "/" + i.name;
|
2015-05-21 15:09:34 +03:00
|
|
|
auto type = i.type == DT_UNKNOWN ? getFileType(path) : i.type;
|
2015-04-22 15:21:52 +03:00
|
|
|
|
2015-05-21 16:04:05 +03:00
|
|
|
if (type == DT_LNK && canWrite) {
|
2015-07-17 12:24:25 +03:00
|
|
|
std::string link;
|
|
|
|
try {
|
|
|
|
link = readLink(path);
|
|
|
|
} catch (SysError & e) {
|
|
|
|
if (e.errNo == ENOENT) continue;
|
|
|
|
}
|
2015-04-22 15:21:52 +03:00
|
|
|
if (link.find("link") != string::npos) {
|
2016-09-21 17:11:01 +03:00
|
|
|
printInfo(format("removing old generations of profile %1%") % path);
|
2015-05-21 17:26:03 +03:00
|
|
|
if (deleteOlderThan != "")
|
|
|
|
deleteGenerationsOlderThan(path, deleteOlderThan, dryRun);
|
|
|
|
else
|
|
|
|
deleteOldGenerations(path, dryRun);
|
2015-04-22 15:21:52 +03:00
|
|
|
}
|
|
|
|
} else if (type == DT_DIR) {
|
|
|
|
removeOldGenerations(path);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-10-26 12:35:46 +03:00
|
|
|
static int _main(int argc, char * * argv)
|
2015-04-22 15:21:52 +03:00
|
|
|
{
|
2018-10-26 12:35:46 +03:00
|
|
|
{
|
|
|
|
bool removeOld = false;
|
2015-04-22 15:21:52 +03:00
|
|
|
|
2015-08-21 14:57:53 +03:00
|
|
|
GCOptions options;
|
|
|
|
|
2015-04-22 15:21:52 +03:00
|
|
|
parseCmdLine(argc, argv, [&](Strings::iterator & arg, const Strings::iterator & end) {
|
|
|
|
if (*arg == "--help")
|
|
|
|
showManPage("nix-collect-garbage");
|
|
|
|
else if (*arg == "--version")
|
|
|
|
printVersion("nix-collect-garbage");
|
|
|
|
else if (*arg == "--delete-old" || *arg == "-d") removeOld = true;
|
|
|
|
else if (*arg == "--delete-older-than") {
|
|
|
|
removeOld = true;
|
2015-05-21 17:26:03 +03:00
|
|
|
deleteOlderThan = getArg(*arg, arg, end);
|
2015-04-22 15:21:52 +03:00
|
|
|
}
|
|
|
|
else if (*arg == "--dry-run") dryRun = true;
|
2020-07-30 14:10:49 +03:00
|
|
|
else if (*arg == "--max-freed")
|
|
|
|
options.maxFreed = std::max(getIntArg<int64_t>(*arg, arg, end, true), (int64_t) 0);
|
2015-04-22 15:21:52 +03:00
|
|
|
else
|
2015-08-21 14:57:53 +03:00
|
|
|
return false;
|
2015-04-22 15:21:52 +03:00
|
|
|
return true;
|
|
|
|
});
|
|
|
|
|
2018-02-08 18:26:18 +02:00
|
|
|
initPlugins();
|
|
|
|
|
2015-04-22 15:21:52 +03:00
|
|
|
auto profilesDir = settings.nixStateDir + "/profiles";
|
|
|
|
if (removeOld) removeOldGenerations(profilesDir);
|
|
|
|
|
|
|
|
// Run the actual garbage collector.
|
2015-05-21 16:21:38 +03:00
|
|
|
if (!dryRun) {
|
2016-02-24 18:33:53 +02:00
|
|
|
auto store = openStore();
|
2015-05-21 16:21:38 +03:00
|
|
|
options.action = GCOptions::gcDeleteDead;
|
|
|
|
GCResults results;
|
|
|
|
PrintFreed freed(true, results);
|
|
|
|
store->collectGarbage(options, results);
|
|
|
|
}
|
2018-10-26 12:35:46 +03:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
2015-04-22 15:21:52 +03:00
|
|
|
}
|
2018-10-26 12:35:46 +03:00
|
|
|
|
|
|
|
static RegisterLegacyCommand s1("nix-collect-garbage", _main);
|