2013-09-02 13:18:15 +00:00
|
|
|
|
#include <iostream>
|
|
|
|
|
#include <cstdlib>
|
|
|
|
|
|
2013-09-06 11:01:02 +00:00
|
|
|
|
#include <setjmp.h>
|
|
|
|
|
|
2013-09-02 13:18:15 +00:00
|
|
|
|
#include <readline/readline.h>
|
|
|
|
|
#include <readline/history.h>
|
|
|
|
|
|
|
|
|
|
#include "shared.hh"
|
|
|
|
|
#include "eval.hh"
|
2013-09-02 15:53:58 +00:00
|
|
|
|
#include "eval-inline.hh"
|
|
|
|
|
#include "store-api.hh"
|
2013-09-02 16:18:27 +00:00
|
|
|
|
#include "common-opts.hh"
|
2013-09-02 13:18:15 +00:00
|
|
|
|
|
|
|
|
|
using namespace std;
|
|
|
|
|
using namespace nix;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
string programId = "nix-repl";
|
|
|
|
|
|
|
|
|
|
|
2013-09-02 15:53:58 +00:00
|
|
|
|
struct NixRepl
|
|
|
|
|
{
|
|
|
|
|
string curDir;
|
|
|
|
|
EvalState state;
|
|
|
|
|
|
|
|
|
|
StaticEnv staticEnv;
|
|
|
|
|
Env * env;
|
|
|
|
|
int displ;
|
|
|
|
|
|
|
|
|
|
NixRepl();
|
|
|
|
|
void mainLoop();
|
|
|
|
|
void processLine(string line);
|
2013-09-02 16:18:27 +00:00
|
|
|
|
void addAttrsToScope(Value & attrs);
|
|
|
|
|
void addVarToScope(const Symbol & name, Value * v);
|
2013-09-02 15:53:58 +00:00
|
|
|
|
Expr * parseString(string s);
|
2013-09-02 16:00:48 +00:00
|
|
|
|
void evalString(string s, Value & v);
|
2013-09-02 15:53:58 +00:00
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
2013-09-02 13:18:15 +00:00
|
|
|
|
void printHelp()
|
|
|
|
|
{
|
|
|
|
|
std::cout << "Usage: nix-repl\n";
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2013-09-06 11:01:02 +00: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);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2013-09-02 13:18:15 +00:00
|
|
|
|
bool getLine(string & line)
|
|
|
|
|
{
|
2013-09-06 11:01:02 +00:00
|
|
|
|
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 {
|
|
|
|
|
char * s = readline("nix-repl> ");
|
|
|
|
|
if (!s) return false;
|
|
|
|
|
line = chomp(string(s));
|
|
|
|
|
free(s);
|
2013-09-06 11:14:28 +00:00
|
|
|
|
if (line != "") {
|
|
|
|
|
add_history(line.c_str());
|
|
|
|
|
append_history(1, 0);
|
|
|
|
|
}
|
2013-09-06 11:01:02 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
_isInterrupted = 0;
|
|
|
|
|
|
|
|
|
|
if (sigaction(SIGINT, &old, 0))
|
|
|
|
|
throw SysError("restoring handler for SIGINT");
|
|
|
|
|
|
2013-09-02 13:18:15 +00:00
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2013-09-02 16:18:27 +00: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 15:53:58 +00:00
|
|
|
|
NixRepl::NixRepl()
|
|
|
|
|
: staticEnv(false, &state.staticBaseEnv)
|
2013-09-02 13:18:15 +00:00
|
|
|
|
{
|
2013-09-02 15:53:58 +00:00
|
|
|
|
curDir = absPath(".");
|
|
|
|
|
|
|
|
|
|
env = &state.allocEnv(32768);
|
|
|
|
|
env->up = &state.baseEnv;
|
|
|
|
|
displ = 0;
|
|
|
|
|
|
|
|
|
|
store = openStore();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void NixRepl::mainLoop()
|
|
|
|
|
{
|
|
|
|
|
std::cerr << "Welcome to Nix version " << NIX_VERSION << ". Type :? for help." << std::endl << std::endl;
|
2013-09-02 13:18:15 +00:00
|
|
|
|
|
2013-09-06 11:14:28 +00:00
|
|
|
|
using_history();
|
|
|
|
|
read_history(0);
|
|
|
|
|
|
2013-09-02 13:18:15 +00:00
|
|
|
|
while (true) {
|
|
|
|
|
string line;
|
|
|
|
|
if (!getLine(line)) break;
|
|
|
|
|
|
|
|
|
|
try {
|
2013-09-02 16:18:27 +00:00
|
|
|
|
processLine(removeWhitespace(line));
|
2013-09-02 13:18:15 +00:00
|
|
|
|
} catch (Error & e) {
|
|
|
|
|
printMsg(lvlError, e.msg());
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
std::cout << std::endl;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
std::cout << std::endl;
|
|
|
|
|
}
|
2013-09-02 15:53:58 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void NixRepl::processLine(string line)
|
|
|
|
|
{
|
2013-09-06 11:01:02 +00:00
|
|
|
|
if (line == "") return;
|
|
|
|
|
|
2013-09-02 15:53:58 +00:00
|
|
|
|
if (string(line, 0, 2) == ":a") {
|
|
|
|
|
Value v;
|
2013-09-02 16:00:48 +00:00
|
|
|
|
evalString(string(line, 2), v);
|
2013-09-02 16:18:27 +00:00
|
|
|
|
addAttrsToScope(v);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
else if (string(line, 0, 2) == ":l") {
|
|
|
|
|
state.resetFileCache();
|
|
|
|
|
Path path = lookupFileArg(state, removeWhitespace(string(line, 2)));
|
|
|
|
|
Value v, v2;
|
|
|
|
|
state.evalFile(path, v);
|
|
|
|
|
Bindings bindings;
|
|
|
|
|
state.autoCallFunction(bindings, v, v2);
|
|
|
|
|
addAttrsToScope(v2);
|
2013-09-02 15:53:58 +00:00
|
|
|
|
}
|
|
|
|
|
|
2013-09-02 16:00:48 +00:00
|
|
|
|
else if (string(line, 0, 2) == ":t") {
|
|
|
|
|
Value v;
|
|
|
|
|
evalString(string(line, 2), v);
|
|
|
|
|
std::cout << showType(v) << std::endl;
|
|
|
|
|
}
|
|
|
|
|
|
2013-09-02 15:53:58 +00:00
|
|
|
|
else if (string(line, 0, 1) == ":") {
|
|
|
|
|
throw Error(format("unknown command ‘%1%’") % string(line, 0, 2));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
else {
|
|
|
|
|
Value v;
|
2013-09-02 16:00:48 +00:00
|
|
|
|
evalString(line, v);
|
2013-09-02 15:53:58 +00:00
|
|
|
|
state.strictForceValue(v);
|
|
|
|
|
std::cout << v << std::endl;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2013-09-02 16:18:27 +00:00
|
|
|
|
void NixRepl::addAttrsToScope(Value & attrs)
|
|
|
|
|
{
|
|
|
|
|
state.forceAttrs(attrs);
|
|
|
|
|
foreach (Bindings::iterator, i, *attrs.attrs)
|
|
|
|
|
addVarToScope(i->name, i->value);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void NixRepl::addVarToScope(const Symbol & name, Value * v)
|
2013-09-02 15:53:58 +00:00
|
|
|
|
{
|
|
|
|
|
staticEnv.vars[name] = displ;
|
|
|
|
|
env->values[displ++] = v;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
Expr * NixRepl::parseString(string s)
|
|
|
|
|
{
|
|
|
|
|
Expr * e = state.parseExprFromString(s, curDir, staticEnv);
|
|
|
|
|
return e;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2013-09-02 16:00:48 +00:00
|
|
|
|
void NixRepl::evalString(string s, Value & v)
|
|
|
|
|
{
|
|
|
|
|
Expr * e = parseString(s);
|
|
|
|
|
e->eval(state, *env, v);
|
|
|
|
|
state.forceValue(v);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2013-09-02 15:53:58 +00:00
|
|
|
|
void run(nix::Strings args)
|
|
|
|
|
{
|
|
|
|
|
NixRepl repl;
|
|
|
|
|
repl.mainLoop();
|
|
|
|
|
}
|