2013-09-02 15:18:15 +02:00
|
|
|
|
#include <iostream>
|
|
|
|
|
#include <cstdlib>
|
|
|
|
|
|
2013-09-06 13:01:02 +02:00
|
|
|
|
#include <setjmp.h>
|
|
|
|
|
|
2013-09-02 15:18:15 +02:00
|
|
|
|
#include <readline/readline.h>
|
|
|
|
|
#include <readline/history.h>
|
|
|
|
|
|
|
|
|
|
#include "shared.hh"
|
|
|
|
|
#include "eval.hh"
|
2013-09-02 17:53:58 +02:00
|
|
|
|
#include "eval-inline.hh"
|
|
|
|
|
#include "store-api.hh"
|
2013-09-02 18:18:27 +02:00
|
|
|
|
#include "common-opts.hh"
|
2013-09-06 14:58:53 +02:00
|
|
|
|
#include "get-drvs.hh"
|
|
|
|
|
#include "derivations.hh"
|
2013-09-02 15:18:15 +02:00
|
|
|
|
|
|
|
|
|
using namespace std;
|
|
|
|
|
using namespace nix;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
string programId = "nix-repl";
|
|
|
|
|
|
|
|
|
|
|
2013-09-02 17:53:58 +02:00
|
|
|
|
struct NixRepl
|
|
|
|
|
{
|
|
|
|
|
string curDir;
|
|
|
|
|
EvalState state;
|
|
|
|
|
|
|
|
|
|
StaticEnv staticEnv;
|
|
|
|
|
Env * env;
|
|
|
|
|
int displ;
|
2013-09-06 19:51:59 +02:00
|
|
|
|
StringSet varNames;
|
|
|
|
|
|
|
|
|
|
StringSet completions;
|
|
|
|
|
StringSet::iterator curCompletion;
|
2013-09-02 17:53:58 +02:00
|
|
|
|
|
|
|
|
|
NixRepl();
|
2013-09-06 15:20:06 +02:00
|
|
|
|
void mainLoop(const Strings & args);
|
2013-09-06 19:51:59 +02:00
|
|
|
|
void completePrefix(string prefix);
|
|
|
|
|
bool getLine(string & line);
|
2013-09-02 17:53:58 +02:00
|
|
|
|
void processLine(string line);
|
2013-09-06 15:20:06 +02:00
|
|
|
|
void loadFile(const Path & path);
|
2013-09-02 18:18:27 +02:00
|
|
|
|
void addAttrsToScope(Value & attrs);
|
|
|
|
|
void addVarToScope(const Symbol & name, Value * v);
|
2013-09-02 17:53:58 +02:00
|
|
|
|
Expr * parseString(string s);
|
2013-09-02 18:00:48 +02:00
|
|
|
|
void evalString(string s, Value & v);
|
2013-09-02 17:53:58 +02:00
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
2013-09-02 15:18:15 +02:00
|
|
|
|
void printHelp()
|
|
|
|
|
{
|
|
|
|
|
std::cout << "Usage: nix-repl\n";
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2013-09-02 18:18:27 +02:00
|
|
|
|
string removeWhitespace(string s)
|
|
|
|
|
{
|
|
|
|
|
s = chomp(s);
|
|
|
|
|
size_t n = s.find_first_not_of(" \n\r\t");
|
|
|
|
|
if (n != string::npos) s = string(s, n);
|
|
|
|
|
return s;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2013-09-02 17:53:58 +02:00
|
|
|
|
NixRepl::NixRepl()
|
|
|
|
|
: staticEnv(false, &state.staticBaseEnv)
|
2013-09-02 15:18:15 +02:00
|
|
|
|
{
|
2013-09-02 17:53:58 +02:00
|
|
|
|
curDir = absPath(".");
|
|
|
|
|
|
|
|
|
|
env = &state.allocEnv(32768);
|
|
|
|
|
env->up = &state.baseEnv;
|
|
|
|
|
displ = 0;
|
|
|
|
|
|
|
|
|
|
store = openStore();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2013-09-06 15:20:06 +02:00
|
|
|
|
void NixRepl::mainLoop(const Strings & args)
|
2013-09-02 17:53:58 +02:00
|
|
|
|
{
|
2013-09-06 15:20:06 +02:00
|
|
|
|
std::cout << "Welcome to Nix version " << NIX_VERSION << ". Type :? for help." << std::endl << std::endl;
|
|
|
|
|
|
|
|
|
|
foreach (Strings::const_iterator, i, args) {
|
|
|
|
|
std::cout << format("Loading ‘%1%’...") % *i << std::endl;
|
|
|
|
|
loadFile(*i);
|
|
|
|
|
std::cout << std::endl;
|
|
|
|
|
}
|
2013-09-02 15:18:15 +02:00
|
|
|
|
|
2013-09-06 13:14:28 +02:00
|
|
|
|
using_history();
|
|
|
|
|
read_history(0);
|
|
|
|
|
|
2013-09-02 15:18:15 +02:00
|
|
|
|
while (true) {
|
|
|
|
|
string line;
|
|
|
|
|
if (!getLine(line)) break;
|
|
|
|
|
|
|
|
|
|
try {
|
2013-09-02 18:18:27 +02:00
|
|
|
|
processLine(removeWhitespace(line));
|
2013-09-02 15:18:15 +02:00
|
|
|
|
} catch (Error & e) {
|
2013-09-06 14:58:53 +02:00
|
|
|
|
printMsg(lvlError, "error: " + e.msg());
|
2013-09-06 13:20:35 +02:00
|
|
|
|
} catch (Interrupted & e) {
|
2013-09-06 14:58:53 +02:00
|
|
|
|
printMsg(lvlError, "error: " + e.msg());
|
2013-09-02 15:18:15 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
std::cout << std::endl;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
std::cout << std::endl;
|
|
|
|
|
}
|
2013-09-02 17:53:58 +02:00
|
|
|
|
|
|
|
|
|
|
2013-09-06 19:51:59 +02:00
|
|
|
|
/* Apparently, the only way to get readline() to return on Ctrl-C
|
|
|
|
|
(SIGINT) is to use siglongjmp(). That's fucked up... */
|
|
|
|
|
static sigjmp_buf sigintJmpBuf;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
static void sigintHandler(int signo)
|
|
|
|
|
{
|
|
|
|
|
siglongjmp(sigintJmpBuf, 1);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Oh, if only g++ had nested functions... */
|
|
|
|
|
NixRepl * curRepl;
|
|
|
|
|
|
|
|
|
|
char * completerThunk(const char * s, int state)
|
|
|
|
|
{
|
|
|
|
|
string prefix(s);
|
|
|
|
|
|
|
|
|
|
/* If the prefix has a slash in it, use readline's builtin filename
|
|
|
|
|
completer. */
|
|
|
|
|
if (prefix.find('/') != string::npos)
|
|
|
|
|
return rl_filename_completion_function(s, state);
|
|
|
|
|
|
|
|
|
|
/* Otherwise, return all symbols that start with the prefix. */
|
|
|
|
|
if (state == 0) {
|
|
|
|
|
curRepl->completePrefix(s);
|
|
|
|
|
curRepl->curCompletion = curRepl->completions.begin();
|
|
|
|
|
}
|
|
|
|
|
if (curRepl->curCompletion == curRepl->completions.end()) return 0;
|
|
|
|
|
return strdup((curRepl->curCompletion++)->c_str());
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
bool NixRepl::getLine(string & line)
|
|
|
|
|
{
|
|
|
|
|
struct sigaction act, old;
|
|
|
|
|
act.sa_handler = sigintHandler;
|
|
|
|
|
sigfillset(&act.sa_mask);
|
|
|
|
|
act.sa_flags = 0;
|
|
|
|
|
if (sigaction(SIGINT, &act, &old))
|
|
|
|
|
throw SysError("installing handler for SIGINT");
|
|
|
|
|
|
|
|
|
|
if (sigsetjmp(sigintJmpBuf, 1))
|
|
|
|
|
line = "";
|
|
|
|
|
else {
|
|
|
|
|
curRepl = this;
|
|
|
|
|
rl_completion_entry_function = completerThunk;
|
|
|
|
|
|
|
|
|
|
char * s = readline("nix-repl> ");
|
|
|
|
|
if (!s) return false;
|
|
|
|
|
line = chomp(string(s));
|
|
|
|
|
free(s);
|
|
|
|
|
if (line != "") {
|
|
|
|
|
add_history(line.c_str());
|
|
|
|
|
append_history(1, 0);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
_isInterrupted = 0;
|
|
|
|
|
|
|
|
|
|
if (sigaction(SIGINT, &old, 0))
|
|
|
|
|
throw SysError("restoring handler for SIGINT");
|
|
|
|
|
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void NixRepl::completePrefix(string prefix)
|
|
|
|
|
{
|
|
|
|
|
completions.clear();
|
|
|
|
|
|
|
|
|
|
StringSet::iterator i = varNames.lower_bound(prefix);
|
|
|
|
|
while (i != varNames.end()) {
|
|
|
|
|
if (string(*i, 0, prefix.size()) != prefix) break;
|
|
|
|
|
completions.insert(*i);
|
|
|
|
|
i++;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2013-09-02 17:53:58 +02:00
|
|
|
|
void NixRepl::processLine(string line)
|
|
|
|
|
{
|
2013-09-06 13:01:02 +02:00
|
|
|
|
if (line == "") return;
|
|
|
|
|
|
2013-09-06 15:05:18 +02:00
|
|
|
|
string command = string(line, 0, 2);
|
|
|
|
|
|
|
|
|
|
if (command == ":a") {
|
2013-09-02 17:53:58 +02:00
|
|
|
|
Value v;
|
2013-09-02 18:00:48 +02:00
|
|
|
|
evalString(string(line, 2), v);
|
2013-09-02 18:18:27 +02:00
|
|
|
|
addAttrsToScope(v);
|
|
|
|
|
}
|
|
|
|
|
|
2013-09-06 15:05:18 +02:00
|
|
|
|
else if (command == ":l") {
|
2013-09-02 18:18:27 +02:00
|
|
|
|
state.resetFileCache();
|
2013-09-06 15:20:06 +02:00
|
|
|
|
loadFile(removeWhitespace(string(line, 2)));
|
2013-09-02 17:53:58 +02:00
|
|
|
|
}
|
|
|
|
|
|
2013-09-06 15:05:18 +02:00
|
|
|
|
else if (command == ":t") {
|
2013-09-02 18:00:48 +02:00
|
|
|
|
Value v;
|
|
|
|
|
evalString(string(line, 2), v);
|
|
|
|
|
std::cout << showType(v) << std::endl;
|
|
|
|
|
}
|
|
|
|
|
|
2013-09-06 15:05:18 +02:00
|
|
|
|
else if (command == ":b" || command == ":s") {
|
2013-09-06 14:58:53 +02:00
|
|
|
|
Value v;
|
|
|
|
|
evalString(string(line, 2), v);
|
|
|
|
|
DrvInfo drvInfo;
|
|
|
|
|
if (!getDerivation(state, v, drvInfo, false))
|
|
|
|
|
throw Error("expression does not evaluation to a derivation, so I can't build it");
|
|
|
|
|
Path drvPath = drvInfo.queryDrvPath(state);
|
|
|
|
|
if (drvPath == "" || !store->isValidPath(drvPath))
|
|
|
|
|
throw Error("expression did not evaluate to a valid derivation");
|
2013-09-06 15:05:18 +02:00
|
|
|
|
|
|
|
|
|
if (command == ":b") {
|
|
|
|
|
/* We could do the build in this process using buildPaths(),
|
|
|
|
|
but doing it in a child makes it easier to recover from
|
|
|
|
|
problems / SIGINT. */
|
|
|
|
|
if (system(("nix-store -r " + drvPath + " > /dev/null").c_str()) == -1)
|
|
|
|
|
throw SysError("starting nix-store");
|
|
|
|
|
Derivation drv = parseDerivation(readFile(drvPath));
|
|
|
|
|
std::cout << "this derivation produced the following outputs:" << std::endl;
|
|
|
|
|
foreach (DerivationOutputs::iterator, i, drv.outputs)
|
|
|
|
|
std::cout << format(" %1% -> %2%") % i->first % i->second.path << std::endl;
|
|
|
|
|
} else {
|
|
|
|
|
if (system(("nix-shell " + drvPath).c_str()) == -1)
|
|
|
|
|
throw SysError("starting nix-shell");
|
|
|
|
|
}
|
2013-09-02 17:53:58 +02:00
|
|
|
|
}
|
|
|
|
|
|
2013-09-06 14:58:53 +02:00
|
|
|
|
else if (string(line, 0, 1) == ":")
|
|
|
|
|
throw Error(format("unknown command ‘%1%’") % string(line, 0, 2));
|
|
|
|
|
|
2013-09-02 17:53:58 +02:00
|
|
|
|
else {
|
|
|
|
|
Value v;
|
2013-09-02 18:00:48 +02:00
|
|
|
|
evalString(line, v);
|
2013-09-02 17:53:58 +02:00
|
|
|
|
state.strictForceValue(v);
|
|
|
|
|
std::cout << v << std::endl;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2013-09-06 15:20:06 +02:00
|
|
|
|
void NixRepl::loadFile(const Path & path)
|
|
|
|
|
{
|
|
|
|
|
Value v, v2;
|
|
|
|
|
state.evalFile(lookupFileArg(state, path), v);
|
|
|
|
|
Bindings bindings;
|
|
|
|
|
state.autoCallFunction(bindings, v, v2);
|
|
|
|
|
addAttrsToScope(v2);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2013-09-02 18:18:27 +02:00
|
|
|
|
void NixRepl::addAttrsToScope(Value & attrs)
|
|
|
|
|
{
|
|
|
|
|
state.forceAttrs(attrs);
|
|
|
|
|
foreach (Bindings::iterator, i, *attrs.attrs)
|
|
|
|
|
addVarToScope(i->name, i->value);
|
2013-09-06 15:20:06 +02:00
|
|
|
|
std::cout << format("Added %1% variables.") % attrs.attrs->size() << std::endl;
|
2013-09-02 18:18:27 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void NixRepl::addVarToScope(const Symbol & name, Value * v)
|
2013-09-02 17:53:58 +02:00
|
|
|
|
{
|
|
|
|
|
staticEnv.vars[name] = displ;
|
|
|
|
|
env->values[displ++] = v;
|
2013-09-06 19:51:59 +02:00
|
|
|
|
varNames.insert((string) name);
|
2013-09-02 17:53:58 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
Expr * NixRepl::parseString(string s)
|
|
|
|
|
{
|
|
|
|
|
Expr * e = state.parseExprFromString(s, curDir, staticEnv);
|
|
|
|
|
return e;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2013-09-02 18:00:48 +02:00
|
|
|
|
void NixRepl::evalString(string s, Value & v)
|
|
|
|
|
{
|
|
|
|
|
Expr * e = parseString(s);
|
|
|
|
|
e->eval(state, *env, v);
|
|
|
|
|
state.forceValue(v);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2013-09-06 15:20:06 +02:00
|
|
|
|
void run(Strings args)
|
2013-09-02 17:53:58 +02:00
|
|
|
|
{
|
|
|
|
|
NixRepl repl;
|
2013-09-06 15:20:06 +02:00
|
|
|
|
repl.mainLoop(args);
|
2013-09-02 17:53:58 +02:00
|
|
|
|
}
|