diff options
author | Rebecca Turner <rbt@sent.as> | 2024-04-04 16:26:25 -0700 |
---|---|---|
committer | Rebecca Turner <rbt@sent.as> | 2024-04-08 17:11:47 -0700 |
commit | 727b43478cce8ebbd0f58530878d3af80a3ba233 (patch) | |
tree | 07950876b289786c388c28119d55cc9839c27b73 /src/libcmd/repl.cc | |
parent | e55fc5af715020d79ec91c856447303737bf3015 (diff) |
Add `repl-overlays`
Adds a `repl-overlays` option, which specifies files that can overlay
and modify the top-level bindings in `nix repl`. For example, with the
following contents in `~/.config/nix/repl.nix`:
info: final: prev: let
optionalAttrs = predicate: attrs:
if predicate
then attrs
else {};
in
optionalAttrs (prev ? legacyPackages && prev.legacyPackages ? ${info.currentSystem})
{
pkgs = prev.legacyPackages.${info.currentSystem};
}
We can run `nix repl` and use `pkgs` to refer to `legacyPackages.${currentSystem}`:
$ nix repl --repl-overlays ~/.config/nix/repl.nix nixpkgs
Lix 2.90.0
Type :? for help.
Loading installable 'flake:nixpkgs#'...
Added 5 variables.
Loading 'repl-overlays'...
Added 6 variables.
nix-repl> pkgs.bash
«derivation /nix/store/g08b5vkwwh0j8ic9rkmd8mpj878rk62z-bash-5.2p26.drv»
Change-Id: Ic12e0f2f210b2f46e920c33088dfe1083f42391a
Diffstat (limited to 'src/libcmd/repl.cc')
-rw-r--r-- | src/libcmd/repl.cc | 165 |
1 files changed, 163 insertions, 2 deletions
diff --git a/src/libcmd/repl.cc b/src/libcmd/repl.cc index ee3a6e268..4aa46dbc0 100644 --- a/src/libcmd/repl.cc +++ b/src/libcmd/repl.cc @@ -30,6 +30,7 @@ #include "signals.hh" #include "print.hh" #include "progress-bar.hh" +#include "gc-small-vector.hh" #if HAVE_BOEHMGC #define GC_INCLUDE_NEW @@ -118,6 +119,45 @@ struct NixRepl void evalString(std::string s, Value & v); void loadDebugTraceEnv(DebugTrace & dt); + /** + * Load the `repl-overlays` and add the resulting AttrSet to the top-level + * bindings. + */ + void loadReplOverlays(); + + /** + * Get a list of each of the `repl-overlays` (parsed and evaluated). + */ + Value * replOverlays(); + + /** + * Get the Nix function that composes the `repl-overlays` together. + */ + Value * getReplOverlaysEvalFunction(); + + /** + * Cached return value of `getReplOverlaysEvalFunction`. + * + * Note: This is `shared_ptr` to avoid garbage collection. + */ + std::shared_ptr<Value *> replOverlaysEvalFunction = + std::allocate_shared<Value *>(traceable_allocator<Value *>(), nullptr); + + /** + * Get the `info` AttrSet that's passed as the first argument to each + * of the `repl-overlays`. + */ + Value * replInitInfo(); + + /** + * Get the current top-level bindings as an AttrSet. + */ + Value * bindingsToAttrs(); + /** + * Parse a file, evaluate its result, and force the resulting value. + */ + Value * evalFile(SourcePath & path); + void printValue(std::ostream & str, Value & v, unsigned int maxDepth = std::numeric_limits<unsigned int>::max()) @@ -770,14 +810,114 @@ void NixRepl::loadFiles() loadedFiles.clear(); for (auto & i : old) { - notice("Loading '%1%'...", i); + notice("Loading '%1%'...", Magenta(i)); loadFile(i); } for (auto & [i, what] : getValues()) { - notice("Loading installable '%1%'...", what); + notice("Loading installable '%1%'...", Magenta(what)); addAttrsToScope(*i); } + + loadReplOverlays(); +} + +void NixRepl::loadReplOverlays() +{ + if (!evalSettings.replOverlays) { + return; + } + + notice("Loading '%1%'...", Magenta("repl-overlays")); + auto replInitFilesFunction = getReplOverlaysEvalFunction(); + + Value &newAttrs(*state->allocValue()); + SmallValueVector<3> args = {replInitInfo(), bindingsToAttrs(), replOverlays()}; + state->callFunction( + *replInitFilesFunction, + args.size(), + args.data(), + newAttrs, + replInitFilesFunction->determinePos(noPos) + ); + + addAttrsToScope(newAttrs); +} + +Value * NixRepl::getReplOverlaysEvalFunction() +{ + if (replOverlaysEvalFunction && *replOverlaysEvalFunction) { + return *replOverlaysEvalFunction; + } + + auto evalReplInitFilesPath = CanonPath::root + "repl-overlays.nix"; + *replOverlaysEvalFunction = state->allocValue(); + auto code = + #include "repl-overlays.nix.gen.hh" + ; + auto expr = state->parseExprFromString( + code, + SourcePath(evalReplInitFilesPath), + state->staticBaseEnv + ); + + state->eval(expr, **replOverlaysEvalFunction); + + return *replOverlaysEvalFunction; +} + +Value * NixRepl::replOverlays() +{ + Value * replInits(state->allocValue()); + state->mkList(*replInits, evalSettings.replOverlays.get().size()); + Value ** replInitElems = replInits->listElems(); + + size_t i = 0; + for (auto path : evalSettings.replOverlays.get()) { + debug("Loading '%1%' path '%2%'...", "repl-overlays", path); + SourcePath sourcePath((CanonPath(path))); + auto replInit = evalFile(sourcePath); + + if (!replInit->isLambda()) { + state->error<TypeError>( + "Expected `repl-overlays` to be a lambda but found %1%: %2%", + showType(*replInit), + ValuePrinter(*state, *replInit, errorPrintOptions) + ) + .atPos(replInit->determinePos(noPos)) + .debugThrow(); + } + + if (replInit->lambda.fun->hasFormals() + && !replInit->lambda.fun->formals->ellipsis) { + state->error<TypeError>( + "Expected first argument of %1% to have %2% to allow future versions of Lix to add additional attributes to the argument", + "repl-overlays", + "..." + ) + .atPos(replInit->determinePos(noPos)) + .debugThrow(); + } + + replInitElems[i] = replInit; + i++; + } + + + return replInits; +} + +Value * NixRepl::replInitInfo() +{ + auto builder = state->buildBindings(2); + + Value * currentSystem(state->allocValue()); + currentSystem->mkString(evalSettings.getCurrentSystem()); + builder.insert(state->symbols.create("currentSystem"), currentSystem); + + Value * info(state->allocValue()); + info->mkAttrs(builder.finish()); + return info; } @@ -810,6 +950,18 @@ void NixRepl::addVarToScope(const Symbol name, Value & v) varNames.emplace(state->symbols[name]); } +Value * NixRepl::bindingsToAttrs() +{ + auto builder = state->buildBindings(staticEnv->vars.size()); + for (auto & [symbol, displacement] : staticEnv->vars) { + builder.insert(symbol, env->values[displacement]); + } + + Value * attrs(state->allocValue()); + attrs->mkAttrs(builder.finish()); + return attrs; +} + Expr * NixRepl::parseString(std::string s) { @@ -824,6 +976,15 @@ void NixRepl::evalString(std::string s, Value & v) state->forceValue(v, v.determinePos(noPos)); } +Value * NixRepl::evalFile(SourcePath & path) +{ + auto expr = state->parseExprFromFile(path, staticEnv); + Value * result(state->allocValue()); + expr->eval(*state, *env, *result); + state->forceValue(*result, result->determinePos(noPos)); + return result; +} + std::unique_ptr<AbstractNixRepl> AbstractNixRepl::create( const SearchPath & searchPath, nix::ref<Store> store, ref<EvalState> state, |