Make "nix-build -A <derivation>.<output>" do the right thing
For example, given a derivation with outputs "out", "man" and "bin": $ nix-build -A pkg produces ./result pointing to the "out" output; $ nix-build -A pkg.man produces ./result-man pointing to the "man" output; $ nix-build -A pkg.all produces ./result, ./result-man and ./result-bin; $ nix-build -A pkg.all -A pkg2 produces ./result, ./result-man, ./result-bin and ./result-2.
This commit is contained in:
		
							parent
							
								
									a3d6585c5a
								
							
						
					
					
						commit
						46a369ad95
					
				
					 10 changed files with 106 additions and 20 deletions
				
			
		| 
						 | 
				
			
			@ -3197,11 +3197,13 @@ void LocalStore::buildPaths(const PathSet & drvPaths, bool repair)
 | 
			
		|||
    Worker worker(*this);
 | 
			
		||||
 | 
			
		||||
    Goals goals;
 | 
			
		||||
    foreach (PathSet::const_iterator, i, drvPaths)
 | 
			
		||||
        if (isDerivation(*i))
 | 
			
		||||
            goals.insert(worker.makeDerivationGoal(*i, repair));
 | 
			
		||||
    foreach (PathSet::const_iterator, i, drvPaths) {
 | 
			
		||||
        DrvPathWithOutputs i2 = parseDrvPathWithOutputs(*i);
 | 
			
		||||
        if (isDerivation(i2.first))
 | 
			
		||||
            goals.insert(worker.makeDerivationGoal(i2.first, repair));
 | 
			
		||||
        else
 | 
			
		||||
            goals.insert(worker.makeSubstitutionGoal(*i, repair));
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    worker.run(goals);
 | 
			
		||||
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -252,4 +252,21 @@ Hash hashDerivationModulo(StoreAPI & store, Derivation drv)
 | 
			
		|||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
DrvPathWithOutputs parseDrvPathWithOutputs(const string & s)
 | 
			
		||||
{
 | 
			
		||||
    size_t n = s.find("!");
 | 
			
		||||
    return n == s.npos
 | 
			
		||||
        ? DrvPathWithOutputs(s, std::set<string>())
 | 
			
		||||
        : DrvPathWithOutputs(string(s, 0, n), tokenizeString<std::set<string> >(string(s, n + 1), ","));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
Path makeDrvPathWithOutputs(const Path & drvPath, std::set<string> outputs)
 | 
			
		||||
{
 | 
			
		||||
    return outputs.empty()
 | 
			
		||||
        ? drvPath
 | 
			
		||||
        : drvPath + "!" + concatStringsSep(",", outputs);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
}
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -79,4 +79,13 @@ typedef std::map<Path, Hash> DrvHashes;
 | 
			
		|||
 | 
			
		||||
extern DrvHashes drvHashes;
 | 
			
		||||
 | 
			
		||||
/* Split a string specifying a derivation and a set of outputs
 | 
			
		||||
   (/nix/store/hash-foo!out1,out2,...) into the derivation path and
 | 
			
		||||
   the outputs. */
 | 
			
		||||
typedef std::pair<string, std::set<string> > DrvPathWithOutputs;
 | 
			
		||||
DrvPathWithOutputs parseDrvPathWithOutputs(const string & s);
 | 
			
		||||
 | 
			
		||||
Path makeDrvPathWithOutputs(const Path & drvPath, std::set<string> outputs);
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
}
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -82,20 +82,23 @@ void queryMissing(StoreAPI & store, const PathSet & targets,
 | 
			
		|||
            if (done.find(*i) != done.end()) continue;
 | 
			
		||||
            done.insert(*i);
 | 
			
		||||
 | 
			
		||||
            if (isDerivation(*i)) {
 | 
			
		||||
                if (!store.isValidPath(*i)) {
 | 
			
		||||
            DrvPathWithOutputs i2 = parseDrvPathWithOutputs(*i);
 | 
			
		||||
 | 
			
		||||
            if (isDerivation(i2.first)) {
 | 
			
		||||
                if (!store.isValidPath(i2.first)) {
 | 
			
		||||
                    // FIXME: we could try to substitute p.
 | 
			
		||||
                    unknown.insert(*i);
 | 
			
		||||
                    continue;
 | 
			
		||||
                }
 | 
			
		||||
                Derivation drv = derivationFromPath(store, *i);
 | 
			
		||||
                Derivation drv = derivationFromPath(store, i2.first);
 | 
			
		||||
 | 
			
		||||
                PathSet invalid;
 | 
			
		||||
                // FIXME: only fetch the desired outputs
 | 
			
		||||
                foreach (DerivationOutputs::iterator, j, drv.outputs)
 | 
			
		||||
                    if (!store.isValidPath(j->second.path)) invalid.insert(j->second.path);
 | 
			
		||||
                if (invalid.empty()) continue;
 | 
			
		||||
 | 
			
		||||
                todoDrv.insert(*i);
 | 
			
		||||
                todoDrv.insert(i2.first);
 | 
			
		||||
                if (settings.useSubstitutes) query.insert(invalid.begin(), invalid.end());
 | 
			
		||||
            }
 | 
			
		||||
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue