forked from lix-project/lix
Alois Wohlschlager
0dd1d8ca1c
Previously, the progress bar had two subtly different states in which the bar would not actually render, both with their own shortcomings: inactive (which was irreversible) and paused (reversible, but swallowing logs). Furthermore, there was no way of resetting the statistics, so a very bad solution was implemented (243c0f18da
) that would create a new logger for each line of the repl, leaking the previous one and discarding the value of printBuildLogs. Finally, if stderr was not attached to a TTY, the update thread was started even though the logger was not active, violating the invariant required by the destructor (which is not observed because the logger is leaked). In this commit, the two aforementioned states are unified into a single one, which can be exited again, correctly upholds the invariant that the update thread is only running while the progress bar is active, and does not swallow logs. The latter change in behavior is not expected to be a problems in the rare cases where the paused state was used before, since other loggers (like the simple one) don't exhibit it anyway. The startProgressBar/stopProgressBar API is removed due to being a footgun, and a new method for properly resetting the progress is added. Co-Authored-By: Qyriad <qyriad@qyriad.me> Change-Id:I2b7c3eb17d439cd0c16f7b896cfb61239ac7ff3a
167 lines
5.2 KiB
C++
167 lines
5.2 KiB
C++
#include "command.hh"
|
|
#include "common-args.hh"
|
|
#include "shared.hh"
|
|
#include "store-api.hh"
|
|
#include "local-fs-store.hh"
|
|
|
|
#include <nlohmann/json.hpp>
|
|
|
|
using namespace nix;
|
|
|
|
static nlohmann::json derivedPathsToJSON(const DerivedPaths & paths, Store & store)
|
|
{
|
|
auto res = nlohmann::json::array();
|
|
for (auto & t : paths) {
|
|
std::visit([&](const auto & t) {
|
|
res.push_back(t.toJSON(store));
|
|
}, t.raw());
|
|
}
|
|
return res;
|
|
}
|
|
|
|
static nlohmann::json builtPathsWithResultToJSON(const std::vector<BuiltPathWithResult> & buildables, const Store & store)
|
|
{
|
|
auto res = nlohmann::json::array();
|
|
for (auto & b : buildables) {
|
|
std::visit([&](const auto & t) {
|
|
auto j = t.toJSON(store);
|
|
if (b.result) {
|
|
if (b.result->startTime)
|
|
j["startTime"] = b.result->startTime;
|
|
if (b.result->stopTime)
|
|
j["stopTime"] = b.result->stopTime;
|
|
if (b.result->cpuUser)
|
|
j["cpuUser"] = ((double) b.result->cpuUser->count()) / 1000000;
|
|
if (b.result->cpuSystem)
|
|
j["cpuSystem"] = ((double) b.result->cpuSystem->count()) / 1000000;
|
|
}
|
|
res.push_back(j);
|
|
}, b.path.raw());
|
|
}
|
|
return res;
|
|
}
|
|
|
|
// TODO deduplicate with other code also setting such out links.
|
|
static void createOutLinks(const Path& outLink, const std::vector<BuiltPathWithResult>& buildables, LocalFSStore& store2)
|
|
{
|
|
for (const auto & [_i, buildable] : enumerate(buildables)) {
|
|
auto i = _i;
|
|
std::visit(overloaded {
|
|
[&](const BuiltPath::Opaque & bo) {
|
|
std::string symlink = outLink;
|
|
if (i) symlink += fmt("-%d", i);
|
|
store2.addPermRoot(bo.path, absPath(symlink));
|
|
},
|
|
[&](const BuiltPath::Built & bfd) {
|
|
for (auto & output : bfd.outputs) {
|
|
std::string symlink = outLink;
|
|
if (i) symlink += fmt("-%d", i);
|
|
if (output.first != "out") symlink += fmt("-%s", output.first);
|
|
store2.addPermRoot(output.second, absPath(symlink));
|
|
}
|
|
},
|
|
}, buildable.path.raw());
|
|
}
|
|
}
|
|
|
|
struct CmdBuild : InstallablesCommand, MixDryRun, MixJSON, MixProfile
|
|
{
|
|
Path outLink = "result";
|
|
bool printOutputPaths = false;
|
|
BuildMode buildMode = bmNormal;
|
|
|
|
CmdBuild()
|
|
{
|
|
addFlag({
|
|
.longName = "out-link",
|
|
.shortName = 'o',
|
|
.description = "Use *path* as prefix for the symlinks to the build results. It defaults to `result`.",
|
|
.labels = {"path"},
|
|
.handler = {&outLink},
|
|
.completer = completePath
|
|
});
|
|
|
|
addFlag({
|
|
.longName = "no-link",
|
|
.description = "Do not create symlinks to the build results.",
|
|
.handler = {&outLink, Path("")},
|
|
});
|
|
|
|
addFlag({
|
|
.longName = "print-out-paths",
|
|
.description = "Print the resulting output paths",
|
|
.handler = {&printOutputPaths, true},
|
|
});
|
|
|
|
addFlag({
|
|
.longName = "rebuild",
|
|
.description = "Rebuild an already built package and compare the result to the existing store paths.",
|
|
.handler = {&buildMode, bmCheck},
|
|
});
|
|
}
|
|
|
|
std::string description() override
|
|
{
|
|
return "build a derivation or fetch a store path";
|
|
}
|
|
|
|
std::string doc() override
|
|
{
|
|
return
|
|
#include "build.md"
|
|
;
|
|
}
|
|
|
|
void run(ref<Store> store, Installables && installables) override
|
|
{
|
|
if (dryRun) {
|
|
std::vector<DerivedPath> pathsToBuild;
|
|
|
|
for (auto & i : installables)
|
|
for (auto & b : i->toDerivedPaths())
|
|
pathsToBuild.push_back(b.path);
|
|
|
|
printMissing(store, pathsToBuild, lvlError);
|
|
|
|
if (json)
|
|
logger->cout("%s", derivedPathsToJSON(pathsToBuild, *store).dump());
|
|
|
|
return;
|
|
}
|
|
|
|
auto buildables = Installable::build(
|
|
getEvalStore(), store,
|
|
Realise::Outputs,
|
|
installables,
|
|
repair ? bmRepair : buildMode);
|
|
|
|
if (json) logger->cout("%s", builtPathsWithResultToJSON(buildables, *store).dump());
|
|
|
|
if (outLink != "")
|
|
if (auto store2 = store.dynamic_pointer_cast<LocalFSStore>())
|
|
createOutLinks(outLink, buildables, *store2);
|
|
|
|
if (printOutputPaths) {
|
|
logger->pause();
|
|
for (auto & buildable : buildables) {
|
|
std::visit(overloaded {
|
|
[&](const BuiltPath::Opaque & bo) {
|
|
logger->cout(store->printStorePath(bo.path));
|
|
},
|
|
[&](const BuiltPath::Built & bfd) {
|
|
for (auto & output : bfd.outputs) {
|
|
logger->cout(store->printStorePath(output.second));
|
|
}
|
|
},
|
|
}, buildable.path.raw());
|
|
}
|
|
}
|
|
|
|
BuiltPaths buildables2;
|
|
for (auto & b : buildables)
|
|
buildables2.push_back(b.path);
|
|
updateProfile(buildables2);
|
|
}
|
|
};
|
|
|
|
static auto rCmdBuild = registerCommand<CmdBuild>("build");
|