lix/src/nix/bundle.cc

134 lines
4.5 KiB
C++
Raw Normal View History

#include "command.hh"
#include "common-args.hh"
#include "shared.hh"
#include "store-api.hh"
#include "local-fs-store.hh"
#include "fs-accessor.hh"
using namespace nix;
2020-07-30 20:03:57 +00:00
struct CmdBundle : InstallableCommand
{
2020-07-30 20:03:57 +00:00
std::string bundler = "github:matthewbauer/nix-bundle";
2020-07-30 20:18:48 +00:00
std::optional<Path> outLink;
2020-07-30 20:03:57 +00:00
CmdBundle()
{
addFlag({
2020-07-30 20:03:57 +00:00
.longName = "bundler",
.description = fmt("Use a custom bundler instead of the default (`%s`).", bundler),
.labels = {"flake-url"},
2020-07-30 20:03:57 +00:00
.handler = {&bundler},
.completer = {[&](size_t, std::string_view prefix) {
completeFlakeRef(getStore(), prefix);
}}
});
addFlag({
.longName = "out-link",
.shortName = 'o',
.description = "Override the name of the symlink to the build result. It defaults to the base name of the app.",
.labels = {"path"},
.handler = {&outLink},
.completer = completePath
});
2020-11-21 04:28:49 +00:00
}
std::string description() override
{
2020-07-30 20:03:57 +00:00
return "bundle an application so that it works outside of the Nix store";
}
2020-12-17 10:45:59 +00:00
std::string doc() override
{
2020-12-17 10:45:59 +00:00
return
#include "bundle.md"
;
}
2020-07-31 15:30:12 +00:00
Category category() override { return catSecondary; }
Strings getDefaultFlakeAttrPaths() override
{
Strings res{"defaultApp." + settings.thisSystem.get()};
for (auto & s : SourceExprCommand::getDefaultFlakeAttrPaths())
res.push_back(s);
return res;
}
Strings getDefaultFlakeAttrPathPrefixes() override
{
Strings res{"apps." + settings.thisSystem.get() + "."};
for (auto & s : SourceExprCommand::getDefaultFlakeAttrPathPrefixes())
res.push_back(s);
return res;
}
void run(ref<Store> store) override
{
auto evalState = getEvalState();
auto app = installable->toApp(*evalState).resolve(getEvalStore(), store);
auto [progFlakeRef, progName] = parseFlakeRefWithFragment(installable->what(), absPath("."));
const flake::LockFlags lockFlagsProg{ .writeLockFile = false };
auto programInstallable = InstallableFlake(this,
evalState, std::move(progFlakeRef),
Strings{progName == "" ? "defaultPackage" : progName},
Strings({"packages."+settings.thisSystem.get()+".","legacyPackages."+settings.thisSystem.get()+"."}), lockFlagsProg);
auto val = programInstallable.toValue(*evalState).first;
2020-07-30 20:03:57 +00:00
auto [bundlerFlakeRef, bundlerName] = parseFlakeRefWithFragment(bundler, absPath("."));
const flake::LockFlags lockFlags{ .writeLockFile = false };
auto bundler = InstallableFlake(this,
2020-07-30 20:03:57 +00:00
evalState, std::move(bundlerFlakeRef),
2020-07-30 20:37:05 +00:00
Strings{bundlerName == "" ? "defaultBundler" : bundlerName},
Strings({"bundlers."}), lockFlags);
auto attrs = evalState->buildBindings(2);
Value & prog = *evalState->allocAttr(*arg, evalState->symbols.create("program"));
evalState->mkAttrs(prog,val->attrs->size());
for (auto &j : *(val->attrs)){
prog.attrs->push_back(j);
}
prog.attrs->sort();
attrs.alloc("system").mkString(settings.thisSystem.get());
auto vRes = evalState->allocValue();
evalState->callFunction(
*bundler.toValue(*evalState).first,
evalState->allocValue()->mkAttrs(attrs),
*vRes, noPos);
if (!evalState->isDerivation(*vRes))
2020-07-30 20:03:57 +00:00
throw Error("the bundler '%s' does not produce a derivation", bundler.what());
auto attr1 = vRes->attrs->get(evalState->sDrvPath);
2020-07-30 20:18:48 +00:00
if (!attr1)
throw Error("the bundler '%s' does not produce a derivation", bundler.what());
PathSet context2;
2020-07-30 20:18:48 +00:00
StorePath drvPath = store->parseStorePath(evalState->coerceToPath(*attr1->pos, *attr1->value, context2));
auto attr2 = vRes->attrs->get(evalState->sOutPath);
2020-07-30 20:18:48 +00:00
if (!attr2)
2020-07-30 20:03:57 +00:00
throw Error("the bundler '%s' does not produce a derivation", bundler.what());
2020-07-30 20:18:48 +00:00
StorePath outPath = store->parseStorePath(evalState->coerceToPath(*attr2->pos, *attr2->value, context2));
2021-04-05 13:48:18 +00:00
store->buildPaths({ DerivedPath::Built { drvPath } });
auto outPathS = store->printStorePath(outPath);
2020-07-30 20:18:48 +00:00
if (!outLink)
outLink = baseNameOf(app.program);
2020-09-03 09:26:36 +00:00
store.dynamic_pointer_cast<LocalFSStore>()->addPermRoot(outPath, absPath(*outLink));
}
};
2020-07-30 20:03:57 +00:00
static auto r2 = registerCommand<CmdBundle>("bundle");