Reuse build products / metrics stored in the database
Previously, if the queue monitor thread encounters a build that Hydra has previously built, it downloaded the output paths from the binary cache, just to determine the build products and metrics. This is very inefficient. In particular, when doing something like merging nixpkgs:staging into nixpkgs:master, the queue monitor thread will be locked up for a long time fetching files from S3, causing the build farm to be mostly idle. Of course this is entirely unnecessary, since the build products/metrics are already in the Hydra database. So now we just look up a previous build with the same output path, and copy the products/metrics.
This commit is contained in:
parent
a7755678fe
commit
f3f661bac1
2 changed files with 73 additions and 2 deletions
|
@ -2,7 +2,6 @@
|
||||||
#include "build-result.hh"
|
#include "build-result.hh"
|
||||||
#include "globals.hh"
|
#include "globals.hh"
|
||||||
|
|
||||||
|
|
||||||
using namespace nix;
|
using namespace nix;
|
||||||
|
|
||||||
|
|
||||||
|
@ -159,7 +158,7 @@ bool State::getQueuedBuilds(Connection & conn, ref<Store> localStore,
|
||||||
all valid. So we mark this as a finished, cached build. */
|
all valid. So we mark this as a finished, cached build. */
|
||||||
if (!step) {
|
if (!step) {
|
||||||
Derivation drv = readDerivation(build->drvPath);
|
Derivation drv = readDerivation(build->drvPath);
|
||||||
BuildOutput res = getBuildOutput(destStore, destStore->getFSAccessor(), drv);
|
BuildOutput res = getBuildOutputCached(conn, destStore, drv);
|
||||||
|
|
||||||
{
|
{
|
||||||
auto mc = startDbUpdate();
|
auto mc = startDbUpdate();
|
||||||
|
@ -531,3 +530,72 @@ void State::processJobsetSharesChange(Connection & conn)
|
||||||
i->second->setShares(row["schedulingShares"].as<unsigned int>());
|
i->second->setShares(row["schedulingShares"].as<unsigned int>());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
BuildOutput State::getBuildOutputCached(Connection & conn, nix::ref<nix::Store> destStore, const nix::Derivation & drv)
|
||||||
|
{
|
||||||
|
{
|
||||||
|
pqxx::work txn(conn);
|
||||||
|
|
||||||
|
for (auto & output : drv.outputs) {
|
||||||
|
auto r = txn.parameterized
|
||||||
|
("select id, buildStatus, releaseName, closureSize, size from Builds b "
|
||||||
|
"join BuildOutputs o on b.id = o.build "
|
||||||
|
"where finished = 1 and (buildStatus = 0 or buildStatus = 6) and path = $1")
|
||||||
|
(output.second.path).exec();
|
||||||
|
if (r.empty()) continue;
|
||||||
|
BuildID id = r[0][0].as<BuildID>();
|
||||||
|
|
||||||
|
printMsg(lvlInfo, format("re-using products of build %d") % id);
|
||||||
|
|
||||||
|
BuildOutput res;
|
||||||
|
res.failed = r[0][1].as<int>() == bsFailedWithOutput;
|
||||||
|
res.releaseName = r[0][2].is_null() ? "" : r[0][2].as<std::string>();
|
||||||
|
res.closureSize = r[0][3].is_null() ? 0 : r[0][3].as<unsigned long long>();
|
||||||
|
res.size = r[0][4].is_null() ? 0 : r[0][4].as<unsigned long long>();
|
||||||
|
|
||||||
|
auto products = txn.parameterized
|
||||||
|
("select type, subtype, fileSize, sha1hash, sha256hash, path, name, defaultPath from BuildProducts where build = $1 order by productnr")
|
||||||
|
(id).exec();
|
||||||
|
|
||||||
|
for (auto row : products) {
|
||||||
|
BuildProduct product;
|
||||||
|
product.type = row[0].as<std::string>();
|
||||||
|
product.subtype = row[1].as<std::string>();
|
||||||
|
if (row[2].is_null())
|
||||||
|
product.isRegular = false;
|
||||||
|
else {
|
||||||
|
product.isRegular = true;
|
||||||
|
product.fileSize = row[2].as<off_t>();
|
||||||
|
}
|
||||||
|
if (!row[3].is_null())
|
||||||
|
product.sha1hash = parseHash(htSHA1, row[3].as<std::string>());
|
||||||
|
if (!row[4].is_null())
|
||||||
|
product.sha256hash = parseHash(htSHA256, row[4].as<std::string>());
|
||||||
|
if (!row[5].is_null())
|
||||||
|
product.path = row[5].as<std::string>();
|
||||||
|
product.name = row[6].as<std::string>();
|
||||||
|
if (!row[7].is_null())
|
||||||
|
product.defaultPath = row[7].as<std::string>();
|
||||||
|
res.products.emplace_back(product);
|
||||||
|
}
|
||||||
|
|
||||||
|
auto metrics = txn.parameterized
|
||||||
|
("select name, unit, value from BuildMetrics where build = $1")
|
||||||
|
(id).exec();
|
||||||
|
|
||||||
|
for (auto row : metrics) {
|
||||||
|
BuildMetric metric;
|
||||||
|
metric.name = row[0].as<std::string>();
|
||||||
|
metric.unit = row[1].is_null() ? "" : row[1].as<std::string>();
|
||||||
|
metric.value = row[2].as<double>();
|
||||||
|
res.metrics.emplace(metric.name, metric);
|
||||||
|
}
|
||||||
|
|
||||||
|
return res;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
return getBuildOutput(destStore, destStore->getFSAccessor(), drv);
|
||||||
|
}
|
||||||
|
|
|
@ -410,6 +410,9 @@ private:
|
||||||
/* Handle cancellation, deletion and priority bumps. */
|
/* Handle cancellation, deletion and priority bumps. */
|
||||||
void processQueueChange(Connection & conn);
|
void processQueueChange(Connection & conn);
|
||||||
|
|
||||||
|
BuildOutput getBuildOutputCached(Connection & conn, nix::ref<nix::Store> destStore,
|
||||||
|
const nix::Derivation & drv);
|
||||||
|
|
||||||
Step::ptr createStep(nix::ref<nix::Store> store,
|
Step::ptr createStep(nix::ref<nix::Store> store,
|
||||||
Connection & conn, Build::ptr build, const nix::Path & drvPath,
|
Connection & conn, Build::ptr build, const nix::Path & drvPath,
|
||||||
Build::ptr referringBuild, Step::ptr referringStep, std::set<nix::Path> & finishedDrvs,
|
Build::ptr referringBuild, Step::ptr referringStep, std::set<nix::Path> & finishedDrvs,
|
||||||
|
|
Loading…
Reference in a new issue