2006-02-08 13:21:16 +00:00
|
|
|
#include "get-drvs.hh"
|
|
|
|
#include "nixexpr-ast.hh"
|
|
|
|
|
|
|
|
|
2006-02-08 16:15:07 +00:00
|
|
|
typedef set<Expr> Exprs;
|
|
|
|
|
|
|
|
|
|
|
|
static bool getDerivation(EvalState & state, Expr e,
|
|
|
|
DrvInfos & drvs, Exprs & doneExprs)
|
2006-02-08 13:21:16 +00:00
|
|
|
{
|
|
|
|
ATermList es;
|
|
|
|
e = evalExpr(state, e);
|
|
|
|
if (!matchAttrs(e, es)) return false;
|
|
|
|
|
|
|
|
ATermMap attrs;
|
|
|
|
queryAllAttrs(e, attrs, false);
|
|
|
|
|
|
|
|
Expr a = attrs.get("type");
|
|
|
|
if (!a || evalString(state, a) != "derivation") return false;
|
|
|
|
|
2006-02-08 16:15:07 +00:00
|
|
|
/* Remove spurious duplicates (e.g., an attribute set like `rec {
|
|
|
|
x = derivation {...}; y = x;}'. */
|
|
|
|
if (doneExprs.find(e) != doneExprs.end()) return true;
|
|
|
|
doneExprs.insert(e);
|
|
|
|
|
|
|
|
DrvInfo drv;
|
|
|
|
|
2006-02-08 13:21:16 +00:00
|
|
|
a = attrs.get("name");
|
|
|
|
if (!a) throw badTerm("derivation name missing", e);
|
|
|
|
drv.name = evalString(state, a);
|
|
|
|
|
|
|
|
a = attrs.get("system");
|
|
|
|
if (!a)
|
|
|
|
drv.system = "unknown";
|
|
|
|
else
|
|
|
|
drv.system = evalString(state, a);
|
|
|
|
|
|
|
|
drv.attrs = attrs;
|
|
|
|
|
2006-02-08 16:15:07 +00:00
|
|
|
drvs.push_back(drv);
|
2006-02-08 13:21:16 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2006-02-08 16:15:07 +00:00
|
|
|
bool getDerivation(EvalState & state, Expr e, DrvInfo & drv)
|
|
|
|
{
|
|
|
|
Exprs doneExprs;
|
|
|
|
DrvInfos drvs;
|
|
|
|
bool result = getDerivation(state, e, drvs, doneExprs);
|
|
|
|
if (result) drv = drvs.front();
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static void getDerivations(EvalState & state, Expr e,
|
|
|
|
DrvInfos & drvs, Exprs & doneExprs)
|
2006-02-08 13:21:16 +00:00
|
|
|
{
|
|
|
|
ATermList es;
|
|
|
|
DrvInfo drv;
|
|
|
|
|
|
|
|
e = evalExpr(state, e);
|
|
|
|
|
2006-02-08 15:22:30 +00:00
|
|
|
if (getDerivation(state, e, drv)) {
|
2006-02-08 13:21:16 +00:00
|
|
|
drvs.push_back(drv);
|
2006-02-08 15:22:30 +00:00
|
|
|
return;
|
|
|
|
}
|
2006-02-08 13:21:16 +00:00
|
|
|
|
2006-02-08 15:22:30 +00:00
|
|
|
if (matchAttrs(e, es)) {
|
2006-02-08 13:21:16 +00:00
|
|
|
ATermMap drvMap;
|
|
|
|
queryAllAttrs(e, drvMap);
|
|
|
|
for (ATermIterator i(drvMap.keys()); i; ++i) {
|
|
|
|
debug(format("evaluating attribute `%1%'") % aterm2String(*i));
|
2006-02-08 16:15:07 +00:00
|
|
|
getDerivation(state, drvMap.get(*i), drvs, doneExprs);
|
2006-02-08 13:21:16 +00:00
|
|
|
}
|
2006-02-08 15:22:30 +00:00
|
|
|
return;
|
2006-02-08 13:21:16 +00:00
|
|
|
}
|
|
|
|
|
2006-02-08 15:22:30 +00:00
|
|
|
if (matchList(e, es)) {
|
2006-02-08 13:21:16 +00:00
|
|
|
for (ATermIterator i(es); i; ++i) {
|
|
|
|
debug(format("evaluating list element"));
|
2006-02-08 16:15:07 +00:00
|
|
|
if (!getDerivation(state, *i, drvs, doneExprs))
|
|
|
|
getDerivations(state, *i, drvs, doneExprs);
|
2006-02-08 13:21:16 +00:00
|
|
|
}
|
2006-02-08 15:22:30 +00:00
|
|
|
return;
|
2006-02-08 13:21:16 +00:00
|
|
|
}
|
|
|
|
|
2006-02-08 15:22:30 +00:00
|
|
|
ATermList formals;
|
|
|
|
ATerm body, pos;
|
|
|
|
if (matchFunction(e, formals, body, pos)) {
|
|
|
|
for (ATermIterator i(formals); i; ++i) {
|
|
|
|
Expr name, def;
|
|
|
|
if (matchNoDefFormal(*i, name))
|
|
|
|
throw Error(format("expression evaluates to a function with no-default arguments (`%1%')")
|
|
|
|
% aterm2String(name));
|
|
|
|
else if (!matchDefFormal(*i, name, def))
|
|
|
|
abort(); /* can't happen */
|
|
|
|
}
|
2006-02-08 16:15:07 +00:00
|
|
|
getDerivations(state, makeCall(e, makeAttrs(ATermMap())), drvs, doneExprs);
|
2006-02-08 15:22:30 +00:00
|
|
|
return;
|
|
|
|
}
|
2006-02-08 13:21:16 +00:00
|
|
|
|
2006-02-08 15:22:30 +00:00
|
|
|
throw Error("expression does not evaluate to a derivation (or a set or list of those)");
|
|
|
|
}
|
2006-02-08 16:15:07 +00:00
|
|
|
|
|
|
|
|
|
|
|
void getDerivations(EvalState & state, Expr e, DrvInfos & drvs)
|
|
|
|
{
|
|
|
|
Exprs doneExprs;
|
|
|
|
getDerivations(state, e, drvs, doneExprs);
|
|
|
|
}
|