forked from lix-project/lix
* Don't allow derivations with fixed and non-fixed outputs.
This commit is contained in:
parent
b2027f70d9
commit
c8606664ab
1 changed files with 40 additions and 45 deletions
|
@ -394,53 +394,48 @@ static void prim_derivationStrict(EvalState & state, Value * * args, Value & v)
|
||||||
throw EvalError(format("derivation names are not allowed to end in `%1%'")
|
throw EvalError(format("derivation names are not allowed to end in `%1%'")
|
||||||
% drvExtension);
|
% drvExtension);
|
||||||
|
|
||||||
/* Construct the "masked" store derivation, which is the final one
|
if (outputHash != "") {
|
||||||
except that in the list of outputs, the output paths are empty,
|
/* Handle fixed-output derivations. */
|
||||||
and the corresponding environment variables have an empty
|
if (outputs.size() != 1 || *(outputs.begin()) != "out")
|
||||||
value. This ensures that changes in the set of output names do
|
throw Error("multiple outputs are not supported in fixed-output derivations");
|
||||||
get reflected in the hash.
|
|
||||||
|
HashType ht = parseHashType(outputHashAlgo);
|
||||||
However, for fixed-output derivations, we can compute the
|
if (ht == htUnknown)
|
||||||
output path directly, so we don't need this. */
|
throw EvalError(format("unknown hash algorithm `%1%'") % outputHashAlgo);
|
||||||
bool fixedOnly = true;
|
Hash h(ht);
|
||||||
foreach (StringSet::iterator, i, outputs) {
|
if (outputHash.size() == h.hashSize * 2)
|
||||||
if (*i != "out" || outputHash == "") {
|
/* hexadecimal representation */
|
||||||
drv.env[*i] = "";
|
h = parseHash(ht, outputHash);
|
||||||
drv.outputs[*i] = DerivationOutput("", "", "");
|
else if (outputHash.size() == hashLength32(h))
|
||||||
fixedOnly = false;
|
/* base-32 representation */
|
||||||
} else {
|
h = parseHash32(ht, outputHash);
|
||||||
/* If an output hash was given, check it, and compute the
|
else
|
||||||
output path. */
|
throw Error(format("hash `%1%' has wrong length for hash type `%2%'")
|
||||||
HashType ht = parseHashType(outputHashAlgo);
|
% outputHash % outputHashAlgo);
|
||||||
if (ht == htUnknown)
|
string s = outputHash;
|
||||||
throw EvalError(format("unknown hash algorithm `%1%'") % outputHashAlgo);
|
outputHash = printHash(h);
|
||||||
Hash h(ht);
|
if (outputHashRecursive) outputHashAlgo = "r:" + outputHashAlgo;
|
||||||
if (outputHash.size() == h.hashSize * 2)
|
|
||||||
/* hexadecimal representation */
|
Path outPath = makeFixedOutputPath(outputHashRecursive, ht, h, drvName);
|
||||||
h = parseHash(ht, outputHash);
|
drv.env["out"] = outPath;
|
||||||
else if (outputHash.size() == hashLength32(h))
|
drv.outputs["out"] = DerivationOutput(outPath, outputHashAlgo, outputHash);
|
||||||
/* base-32 representation */
|
|
||||||
h = parseHash32(ht, outputHash);
|
|
||||||
else
|
|
||||||
throw Error(format("hash `%1%' has wrong length for hash type `%2%'")
|
|
||||||
% outputHash % outputHashAlgo);
|
|
||||||
string s = outputHash;
|
|
||||||
outputHash = printHash(h);
|
|
||||||
if (outputHashRecursive) outputHashAlgo = "r:" + outputHashAlgo;
|
|
||||||
Path outPath = makeFixedOutputPath(outputHashRecursive, ht, h, drvName);
|
|
||||||
drv.env[*i] = outPath;
|
|
||||||
drv.outputs[*i] = DerivationOutput(outPath, outputHashAlgo, outputHash);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Use the masked derivation expression to compute the output
|
else {
|
||||||
path. !!! Isn't it a potential security problem that the name
|
/* Construct the "masked" store derivation, which is the final
|
||||||
of each output path (including the suffix) isn't taken into
|
one except that in the list of outputs, the output paths
|
||||||
account? For instance, changing the suffix for one path
|
are empty, and the corresponding environment variables have
|
||||||
(‘i->first == "out" ...’) doesn't affect the hash of the
|
an empty value. This ensures that changes in the set of
|
||||||
others. Is that exploitable? */
|
output names do get reflected in the hash. */
|
||||||
if (!fixedOnly) {
|
foreach (StringSet::iterator, i, outputs) {
|
||||||
|
drv.env[*i] = "";
|
||||||
|
drv.outputs[*i] = DerivationOutput("", "", "");
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Use the masked derivation expression to compute the output
|
||||||
|
path. */
|
||||||
Hash h = hashDerivationModulo(drv);
|
Hash h = hashDerivationModulo(drv);
|
||||||
|
|
||||||
foreach (DerivationOutputs::iterator, i, drv.outputs)
|
foreach (DerivationOutputs::iterator, i, drv.outputs)
|
||||||
if (i->second.path == "") {
|
if (i->second.path == "") {
|
||||||
Path outPath = makeOutputPath(i->first, h, drvName);
|
Path outPath = makeOutputPath(i->first, h, drvName);
|
||||||
|
@ -448,7 +443,7 @@ static void prim_derivationStrict(EvalState & state, Value * * args, Value & v)
|
||||||
i->second.path = outPath;
|
i->second.path = outPath;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Write the resulting term into the Nix store directory. */
|
/* Write the resulting term into the Nix store directory. */
|
||||||
Path drvPath = writeDerivation(drv, drvName);
|
Path drvPath = writeDerivation(drv, drvName);
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue