2020-11-09 14:40:10 +00:00
|
|
|
|
#include "drv-output-substitution-goal.hh"
|
2021-10-28 14:43:54 +00:00
|
|
|
|
#include "finally.hh"
|
2020-11-09 14:40:10 +00:00
|
|
|
|
#include "worker.hh"
|
|
|
|
|
#include "substitution-goal.hh"
|
2021-10-28 14:43:54 +00:00
|
|
|
|
#include "callback.hh"
|
2020-11-09 14:40:10 +00:00
|
|
|
|
|
|
|
|
|
namespace nix {
|
|
|
|
|
|
2022-03-09 11:25:35 +00:00
|
|
|
|
DrvOutputSubstitutionGoal::DrvOutputSubstitutionGoal(
|
|
|
|
|
const DrvOutput & id,
|
|
|
|
|
Worker & worker,
|
|
|
|
|
RepairFlag repair,
|
|
|
|
|
std::optional<ContentAddress> ca)
|
|
|
|
|
: Goal(worker, DerivedPath::Opaque { StorePath::dummy })
|
2020-11-09 14:40:10 +00:00
|
|
|
|
, id(id)
|
|
|
|
|
{
|
|
|
|
|
state = &DrvOutputSubstitutionGoal::init;
|
|
|
|
|
name = fmt("substitution of '%s'", id.to_string());
|
|
|
|
|
trace("created");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void DrvOutputSubstitutionGoal::init()
|
|
|
|
|
{
|
|
|
|
|
trace("init");
|
2021-05-19 14:27:09 +00:00
|
|
|
|
|
|
|
|
|
/* If the derivation already exists, we’re done */
|
|
|
|
|
if (worker.store.queryRealisation(id)) {
|
|
|
|
|
amDone(ecSuccess);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
2020-11-09 14:40:10 +00:00
|
|
|
|
subs = settings.useSubstitutes ? getDefaultSubstituters() : std::list<ref<Store>>();
|
|
|
|
|
tryNext();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void DrvOutputSubstitutionGoal::tryNext()
|
|
|
|
|
{
|
2022-03-08 18:50:46 +00:00
|
|
|
|
trace("trying next substituter");
|
2020-11-09 14:40:10 +00:00
|
|
|
|
|
|
|
|
|
if (subs.size() == 0) {
|
|
|
|
|
/* None left. Terminate this goal and let someone else deal
|
|
|
|
|
with it. */
|
|
|
|
|
debug("drv output '%s' is required, but there is no substituter that can provide it", id.to_string());
|
|
|
|
|
|
|
|
|
|
/* Hack: don't indicate failure if there were no substituters.
|
|
|
|
|
In that case the calling derivation should just do a
|
|
|
|
|
build. */
|
|
|
|
|
amDone(substituterFailed ? ecFailed : ecNoSubstituters);
|
|
|
|
|
|
|
|
|
|
if (substituterFailed) {
|
|
|
|
|
worker.failedSubstitutions++;
|
|
|
|
|
worker.updateProgress();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
2021-10-28 14:43:54 +00:00
|
|
|
|
sub = subs.front();
|
2020-11-09 14:40:10 +00:00
|
|
|
|
subs.pop_front();
|
|
|
|
|
|
|
|
|
|
// FIXME: Make async
|
2021-10-28 14:43:54 +00:00
|
|
|
|
// outputInfo = sub->queryRealisation(id);
|
|
|
|
|
outPipe.create();
|
|
|
|
|
promise = decltype(promise)();
|
|
|
|
|
|
|
|
|
|
sub->queryRealisation(
|
|
|
|
|
id, { [&](std::future<std::shared_ptr<const Realisation>> res) {
|
|
|
|
|
try {
|
|
|
|
|
Finally updateStats([this]() { outPipe.writeSide.close(); });
|
|
|
|
|
promise.set_value(res.get());
|
|
|
|
|
} catch (...) {
|
|
|
|
|
promise.set_exception(std::current_exception());
|
|
|
|
|
}
|
|
|
|
|
} });
|
|
|
|
|
|
|
|
|
|
worker.childStarted(shared_from_this(), {outPipe.readSide.get()}, true, false);
|
|
|
|
|
|
|
|
|
|
state = &DrvOutputSubstitutionGoal::realisationFetched;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void DrvOutputSubstitutionGoal::realisationFetched()
|
|
|
|
|
{
|
|
|
|
|
worker.childTerminated(this);
|
|
|
|
|
|
|
|
|
|
try {
|
|
|
|
|
outputInfo = promise.get_future().get();
|
|
|
|
|
} catch (std::exception & e) {
|
|
|
|
|
printError(e.what());
|
|
|
|
|
substituterFailed = true;
|
|
|
|
|
}
|
|
|
|
|
|
2020-11-09 14:40:10 +00:00
|
|
|
|
if (!outputInfo) {
|
2021-10-28 14:43:54 +00:00
|
|
|
|
return tryNext();
|
2020-11-09 14:40:10 +00:00
|
|
|
|
}
|
|
|
|
|
|
2021-05-19 14:27:09 +00:00
|
|
|
|
for (const auto & [depId, depPath] : outputInfo->dependentRealisations) {
|
|
|
|
|
if (depId != id) {
|
|
|
|
|
if (auto localOutputInfo = worker.store.queryRealisation(depId);
|
|
|
|
|
localOutputInfo && localOutputInfo->outPath != depPath) {
|
|
|
|
|
warn(
|
2021-06-22 08:46:29 +00:00
|
|
|
|
"substituter '%s' has an incompatible realisation for '%s', ignoring.\n"
|
|
|
|
|
"Local: %s\n"
|
|
|
|
|
"Remote: %s",
|
2021-05-19 14:27:09 +00:00
|
|
|
|
sub->getUri(),
|
2021-06-22 08:46:29 +00:00
|
|
|
|
depId.to_string(),
|
|
|
|
|
worker.store.printStorePath(localOutputInfo->outPath),
|
|
|
|
|
worker.store.printStorePath(depPath)
|
|
|
|
|
);
|
2021-05-19 14:27:09 +00:00
|
|
|
|
tryNext();
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
addWaitee(worker.makeDrvOutputSubstitutionGoal(depId));
|
2021-05-19 12:51:34 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2020-11-09 14:40:10 +00:00
|
|
|
|
addWaitee(worker.makePathSubstitutionGoal(outputInfo->outPath));
|
|
|
|
|
|
|
|
|
|
if (waitees.empty()) outPathValid();
|
|
|
|
|
else state = &DrvOutputSubstitutionGoal::outPathValid;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void DrvOutputSubstitutionGoal::outPathValid()
|
|
|
|
|
{
|
|
|
|
|
assert(outputInfo);
|
2022-03-08 18:50:46 +00:00
|
|
|
|
trace("output path substituted");
|
2020-11-09 14:40:10 +00:00
|
|
|
|
|
|
|
|
|
if (nrFailed > 0) {
|
|
|
|
|
debug("The output path of the derivation output '%s' could not be substituted", id.to_string());
|
|
|
|
|
amDone(nrNoSubstituters > 0 || nrIncompleteClosure > 0 ? ecIncompleteClosure : ecFailed);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
worker.store.registerDrvOutput(*outputInfo);
|
|
|
|
|
finished();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void DrvOutputSubstitutionGoal::finished()
|
|
|
|
|
{
|
|
|
|
|
trace("finished");
|
|
|
|
|
amDone(ecSuccess);
|
|
|
|
|
}
|
|
|
|
|
|
2022-02-25 15:00:00 +00:00
|
|
|
|
std::string DrvOutputSubstitutionGoal::key()
|
2020-11-09 14:40:10 +00:00
|
|
|
|
{
|
|
|
|
|
/* "a$" ensures substitution goals happen before derivation
|
|
|
|
|
goals. */
|
|
|
|
|
return "a$" + std::string(id.to_string());
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void DrvOutputSubstitutionGoal::work()
|
|
|
|
|
{
|
|
|
|
|
(this->*state)();
|
|
|
|
|
}
|
|
|
|
|
|
2021-10-28 14:43:54 +00:00
|
|
|
|
void DrvOutputSubstitutionGoal::handleEOF(int fd)
|
|
|
|
|
{
|
|
|
|
|
if (fd == outPipe.readSide.get()) worker.wakeUp(shared_from_this());
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2020-11-09 14:40:10 +00:00
|
|
|
|
}
|