2018-09-28 15:31:16 +03:00
|
|
|
#include "parsed-derivations.hh"
|
|
|
|
|
2020-03-24 15:26:13 +02:00
|
|
|
#include <nlohmann/json.hpp>
|
2021-05-02 18:24:14 +03:00
|
|
|
#include <regex>
|
|
|
|
#include "json.hh"
|
2020-03-24 15:26:13 +02:00
|
|
|
|
2018-09-28 15:31:16 +03:00
|
|
|
namespace nix {
|
|
|
|
|
2020-06-16 23:20:18 +03:00
|
|
|
ParsedDerivation::ParsedDerivation(const StorePath & drvPath, BasicDerivation & drv)
|
|
|
|
: drvPath(drvPath), drv(drv)
|
2018-09-28 15:31:16 +03:00
|
|
|
{
|
|
|
|
/* Parse the __json attribute, if any. */
|
|
|
|
auto jsonAttr = drv.env.find("__json");
|
|
|
|
if (jsonAttr != drv.env.end()) {
|
|
|
|
try {
|
2020-03-24 15:26:13 +02:00
|
|
|
structuredAttrs = std::make_unique<nlohmann::json>(nlohmann::json::parse(jsonAttr->second));
|
2018-09-28 15:31:16 +03:00
|
|
|
} catch (std::exception & e) {
|
2019-12-05 20:11:09 +02:00
|
|
|
throw Error("cannot process __json attribute of '%s': %s", drvPath.to_string(), e.what());
|
2018-09-28 15:31:16 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-03-24 15:26:13 +02:00
|
|
|
ParsedDerivation::~ParsedDerivation() { }
|
|
|
|
|
2019-02-12 14:43:32 +02:00
|
|
|
std::optional<std::string> ParsedDerivation::getStringAttr(const std::string & name) const
|
2018-09-28 15:31:16 +03:00
|
|
|
{
|
|
|
|
if (structuredAttrs) {
|
|
|
|
auto i = structuredAttrs->find(name);
|
|
|
|
if (i == structuredAttrs->end())
|
|
|
|
return {};
|
|
|
|
else {
|
|
|
|
if (!i->is_string())
|
2019-12-05 20:11:09 +02:00
|
|
|
throw Error("attribute '%s' of derivation '%s' must be a string", name, drvPath.to_string());
|
2018-09-28 15:31:16 +03:00
|
|
|
return i->get<std::string>();
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
auto i = drv.env.find(name);
|
|
|
|
if (i == drv.env.end())
|
|
|
|
return {};
|
|
|
|
else
|
|
|
|
return i->second;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
bool ParsedDerivation::getBoolAttr(const std::string & name, bool def) const
|
|
|
|
{
|
|
|
|
if (structuredAttrs) {
|
|
|
|
auto i = structuredAttrs->find(name);
|
|
|
|
if (i == structuredAttrs->end())
|
|
|
|
return def;
|
|
|
|
else {
|
|
|
|
if (!i->is_boolean())
|
2019-12-05 20:11:09 +02:00
|
|
|
throw Error("attribute '%s' of derivation '%s' must be a Boolean", name, drvPath.to_string());
|
2018-09-28 15:31:16 +03:00
|
|
|
return i->get<bool>();
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
auto i = drv.env.find(name);
|
|
|
|
if (i == drv.env.end())
|
|
|
|
return def;
|
|
|
|
else
|
|
|
|
return i->second == "1";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-02-12 14:43:32 +02:00
|
|
|
std::optional<Strings> ParsedDerivation::getStringsAttr(const std::string & name) const
|
2018-09-28 15:31:16 +03:00
|
|
|
{
|
|
|
|
if (structuredAttrs) {
|
|
|
|
auto i = structuredAttrs->find(name);
|
|
|
|
if (i == structuredAttrs->end())
|
|
|
|
return {};
|
|
|
|
else {
|
|
|
|
if (!i->is_array())
|
2019-12-05 20:11:09 +02:00
|
|
|
throw Error("attribute '%s' of derivation '%s' must be a list of strings", name, drvPath.to_string());
|
2018-09-28 15:31:16 +03:00
|
|
|
Strings res;
|
|
|
|
for (auto j = i->begin(); j != i->end(); ++j) {
|
|
|
|
if (!j->is_string())
|
2019-12-05 20:11:09 +02:00
|
|
|
throw Error("attribute '%s' of derivation '%s' must be a list of strings", name, drvPath.to_string());
|
2018-09-28 15:31:16 +03:00
|
|
|
res.push_back(j->get<std::string>());
|
|
|
|
}
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
auto i = drv.env.find(name);
|
|
|
|
if (i == drv.env.end())
|
|
|
|
return {};
|
|
|
|
else
|
|
|
|
return tokenizeString<Strings>(i->second);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-09-28 16:57:27 +03:00
|
|
|
StringSet ParsedDerivation::getRequiredSystemFeatures() const
|
|
|
|
{
|
|
|
|
StringSet res;
|
|
|
|
for (auto & i : getStringsAttr("requiredSystemFeatures").value_or(Strings()))
|
|
|
|
res.insert(i);
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
2020-08-12 19:32:36 +03:00
|
|
|
bool ParsedDerivation::canBuildLocally(Store & localStore) const
|
2018-09-28 15:31:16 +03:00
|
|
|
{
|
2018-09-28 16:57:27 +03:00
|
|
|
if (drv.platform != settings.thisSystem.get()
|
|
|
|
&& !settings.extraPlatforms.get().count(drv.platform)
|
|
|
|
&& !drv.isBuiltin())
|
|
|
|
return false;
|
|
|
|
|
2021-01-12 02:28:00 +02:00
|
|
|
if (settings.maxBuildJobs.get() == 0
|
|
|
|
&& !drv.isBuiltin())
|
|
|
|
return false;
|
|
|
|
|
2018-09-28 16:57:27 +03:00
|
|
|
for (auto & feature : getRequiredSystemFeatures())
|
2020-08-12 19:32:36 +03:00
|
|
|
if (!localStore.systemFeatures.get().count(feature)) return false;
|
2018-09-28 16:57:27 +03:00
|
|
|
|
|
|
|
return true;
|
2018-09-28 15:31:16 +03:00
|
|
|
}
|
|
|
|
|
2020-08-12 19:32:36 +03:00
|
|
|
bool ParsedDerivation::willBuildLocally(Store & localStore) const
|
2018-09-28 15:31:16 +03:00
|
|
|
{
|
2020-08-12 19:32:36 +03:00
|
|
|
return getBoolAttr("preferLocalBuild") && canBuildLocally(localStore);
|
2018-09-28 15:31:16 +03:00
|
|
|
}
|
|
|
|
|
2019-09-03 17:02:12 +03:00
|
|
|
bool ParsedDerivation::substitutesAllowed() const
|
|
|
|
{
|
|
|
|
return getBoolAttr("allowSubstitutes", true);
|
|
|
|
}
|
|
|
|
|
2021-05-02 18:24:14 +03:00
|
|
|
static std::regex shVarName("[A-Za-z_][A-Za-z0-9_]*");
|
|
|
|
std::optional<StructuredAttrsWithShellRC> ParsedDerivation::generateStructuredAttrs(
|
|
|
|
std::optional<StringMap> inputRewrites, Store & store, const StorePathSet & inputPaths)
|
|
|
|
{
|
|
|
|
auto structuredAttrs = getStructuredAttrs();
|
|
|
|
if (!structuredAttrs) return std::nullopt;
|
|
|
|
|
|
|
|
auto json = *structuredAttrs;
|
|
|
|
|
|
|
|
/* Add an "outputs" object containing the output paths. */
|
|
|
|
nlohmann::json outputs;
|
|
|
|
for (auto & i : drv.outputs) {
|
|
|
|
if (inputRewrites) {
|
|
|
|
/* The placeholder must have a rewrite, so we use it to cover both the
|
|
|
|
cases where we know or don't know the output path ahead of time. */
|
|
|
|
outputs[i.first] = rewriteStrings(hashPlaceholder(i.first), inputRewrites.value());
|
|
|
|
} else {
|
|
|
|
/* This case is only relevant for the nix-shell */
|
|
|
|
outputs[i.first] = hashPlaceholder(i.first);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
json["outputs"] = outputs;
|
|
|
|
|
|
|
|
/* Handle exportReferencesGraph. */
|
|
|
|
auto e = json.find("exportReferencesGraph");
|
|
|
|
if (e != json.end() && e->is_object()) {
|
|
|
|
for (auto i = e->begin(); i != e->end(); ++i) {
|
|
|
|
std::ostringstream str;
|
|
|
|
{
|
|
|
|
JSONPlaceholder jsonRoot(str, true);
|
|
|
|
StorePathSet storePaths;
|
|
|
|
for (auto & p : *i)
|
|
|
|
storePaths.insert(store.parseStorePath(p.get<std::string>()));
|
|
|
|
store.pathInfoToJSON(jsonRoot,
|
|
|
|
store.exportReferences(storePaths, inputPaths), false, true);
|
|
|
|
}
|
|
|
|
json[i.key()] = nlohmann::json::parse(str.str()); // urgh
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* As a convenience to bash scripts, write a shell file that
|
|
|
|
maps all attributes that are representable in bash -
|
|
|
|
namely, strings, integers, nulls, Booleans, and arrays and
|
|
|
|
objects consisting entirely of those values. (So nested
|
|
|
|
arrays or objects are not supported.) */
|
|
|
|
|
|
|
|
auto handleSimpleType = [](const nlohmann::json & value) -> std::optional<std::string> {
|
|
|
|
if (value.is_string())
|
|
|
|
return shellEscape(value);
|
|
|
|
|
|
|
|
if (value.is_number()) {
|
|
|
|
auto f = value.get<float>();
|
|
|
|
if (std::ceil(f) == f)
|
|
|
|
return std::to_string(value.get<int>());
|
|
|
|
}
|
|
|
|
|
|
|
|
if (value.is_null())
|
|
|
|
return std::string("''");
|
|
|
|
|
|
|
|
if (value.is_boolean())
|
|
|
|
return value.get<bool>() ? std::string("1") : std::string("");
|
|
|
|
|
|
|
|
return {};
|
|
|
|
};
|
|
|
|
|
|
|
|
std::string jsonSh;
|
|
|
|
|
|
|
|
for (auto i = json.begin(); i != json.end(); ++i) {
|
|
|
|
|
|
|
|
if (!std::regex_match(i.key(), shVarName)) continue;
|
|
|
|
|
|
|
|
auto & value = i.value();
|
|
|
|
|
|
|
|
auto s = handleSimpleType(value);
|
|
|
|
if (s)
|
|
|
|
jsonSh += fmt("declare %s=%s\n", i.key(), *s);
|
|
|
|
|
|
|
|
else if (value.is_array()) {
|
|
|
|
std::string s2;
|
|
|
|
bool good = true;
|
|
|
|
|
|
|
|
for (auto i = value.begin(); i != value.end(); ++i) {
|
|
|
|
auto s3 = handleSimpleType(i.value());
|
|
|
|
if (!s3) { good = false; break; }
|
|
|
|
s2 += *s3; s2 += ' ';
|
|
|
|
}
|
|
|
|
|
|
|
|
if (good)
|
|
|
|
jsonSh += fmt("declare -a %s=(%s)\n", i.key(), s2);
|
|
|
|
}
|
|
|
|
|
|
|
|
else if (value.is_object()) {
|
|
|
|
std::string s2;
|
|
|
|
bool good = true;
|
|
|
|
|
|
|
|
for (auto i = value.begin(); i != value.end(); ++i) {
|
|
|
|
auto s3 = handleSimpleType(i.value());
|
|
|
|
if (!s3) { good = false; break; }
|
|
|
|
s2 += fmt("[%s]=%s ", shellEscape(i.key()), *s3);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (good)
|
|
|
|
jsonSh += fmt("declare -A %s=(%s)\n", i.key(), s2);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return std::make_pair(jsonSh, json);
|
|
|
|
}
|
2018-09-28 15:31:16 +03:00
|
|
|
}
|