forked from lix-project/lix
Add basic "nix build" command
Currently only builds by attribute from <nixpkgs> or the specified file, e.g. "nix build hello".
This commit is contained in:
parent
cd2196b089
commit
206bbb5dc9
3 changed files with 159 additions and 0 deletions
46
src/nix/build.cc
Normal file
46
src/nix/build.cc
Normal file
|
@ -0,0 +1,46 @@
|
|||
#include "command.hh"
|
||||
#include "common-args.hh"
|
||||
#include "installables.hh"
|
||||
#include "shared.hh"
|
||||
#include "store-api.hh"
|
||||
|
||||
using namespace nix;
|
||||
|
||||
struct CmdBuild : StoreCommand, MixDryRun, MixInstallables
|
||||
{
|
||||
CmdBuild()
|
||||
{
|
||||
}
|
||||
|
||||
std::string name() override
|
||||
{
|
||||
return "build";
|
||||
}
|
||||
|
||||
std::string description() override
|
||||
{
|
||||
return "build a derivation or fetch a store path";
|
||||
}
|
||||
|
||||
void run(ref<Store> store) override
|
||||
{
|
||||
auto elems = evalInstallables(store);
|
||||
|
||||
PathSet pathsToBuild;
|
||||
|
||||
for (auto & elem : elems) {
|
||||
if (elem.isDrv)
|
||||
pathsToBuild.insert(elem.drvPath);
|
||||
else
|
||||
pathsToBuild.insert(elem.outPaths.begin(), elem.outPaths.end());
|
||||
}
|
||||
|
||||
printMissing(store, pathsToBuild);
|
||||
|
||||
if (dryRun) return;
|
||||
|
||||
store->buildPaths(pathsToBuild);
|
||||
}
|
||||
};
|
||||
|
||||
static RegisterCommand r1(make_ref<Command, CmdBuild>());
|
75
src/nix/installables.cc
Normal file
75
src/nix/installables.cc
Normal file
|
@ -0,0 +1,75 @@
|
|||
#include "attr-path.hh"
|
||||
#include "common-opts.hh"
|
||||
#include "derivations.hh"
|
||||
#include "eval-inline.hh"
|
||||
#include "eval.hh"
|
||||
#include "get-drvs.hh"
|
||||
#include "installables.hh"
|
||||
#include "store-api.hh"
|
||||
|
||||
namespace nix {
|
||||
|
||||
UserEnvElems MixInstallables::evalInstallables(ref<Store> store)
|
||||
{
|
||||
UserEnvElems res;
|
||||
|
||||
for (auto & installable : installables) {
|
||||
|
||||
if (std::string(installable, 0, 1) == "/") {
|
||||
|
||||
if (isStorePath(installable)) {
|
||||
|
||||
if (isDerivation(installable)) {
|
||||
UserEnvElem elem;
|
||||
// FIXME: handle empty case, drop version
|
||||
elem.attrPath = {storePathToName(installable)};
|
||||
elem.isDrv = true;
|
||||
elem.drvPath = installable;
|
||||
res.push_back(elem);
|
||||
}
|
||||
|
||||
else {
|
||||
UserEnvElem elem;
|
||||
// FIXME: handle empty case, drop version
|
||||
elem.attrPath = {storePathToName(installable)};
|
||||
elem.isDrv = false;
|
||||
elem.outPaths = {installable};
|
||||
res.push_back(elem);
|
||||
}
|
||||
}
|
||||
|
||||
else
|
||||
throw UsageError(format("don't know what to do with ‘%1%’") % installable);
|
||||
}
|
||||
|
||||
else {
|
||||
|
||||
EvalState state({}, store);
|
||||
|
||||
Expr * e = state.parseExprFromFile(resolveExprPath(lookupFileArg(state, file)));
|
||||
|
||||
Value vRoot;
|
||||
state.eval(e, vRoot);
|
||||
|
||||
std::map<string, string> autoArgs_;
|
||||
Bindings & autoArgs(*evalAutoArgs(state, autoArgs_));
|
||||
|
||||
Value & v(*findAlongAttrPath(state, installable, autoArgs, vRoot));
|
||||
state.forceValue(v);
|
||||
|
||||
DrvInfos drvs;
|
||||
getDerivations(state, v, "", autoArgs, drvs, false);
|
||||
|
||||
for (auto & i : drvs) {
|
||||
UserEnvElem elem;
|
||||
elem.isDrv = true;
|
||||
elem.drvPath = i.queryDrvPath();
|
||||
res.push_back(elem);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
return res;
|
||||
}
|
||||
|
||||
}
|
38
src/nix/installables.hh
Normal file
38
src/nix/installables.hh
Normal file
|
@ -0,0 +1,38 @@
|
|||
#pragma once
|
||||
|
||||
#include "args.hh"
|
||||
|
||||
namespace nix {
|
||||
|
||||
struct UserEnvElem
|
||||
{
|
||||
Strings attrPath;
|
||||
|
||||
// FIXME: should use boost::variant or so.
|
||||
bool isDrv;
|
||||
|
||||
// Derivation case:
|
||||
Path drvPath;
|
||||
StringSet outputNames;
|
||||
|
||||
// Non-derivation case:
|
||||
PathSet outPaths;
|
||||
};
|
||||
|
||||
typedef std::vector<UserEnvElem> UserEnvElems;
|
||||
|
||||
struct MixInstallables : virtual Args
|
||||
{
|
||||
Strings installables;
|
||||
Path file = "<nixpkgs>";
|
||||
|
||||
MixInstallables()
|
||||
{
|
||||
mkFlag('f', "file", "file", "evaluate FILE rather than the default", &file);
|
||||
expectArgs("installables", &installables);
|
||||
}
|
||||
|
||||
UserEnvElems evalInstallables(ref<Store> store);
|
||||
};
|
||||
|
||||
}
|
Loading…
Reference in a new issue