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-06 12:58:53 +00:00
|
|
|
|
#include "get-drvs.hh"
|
|
|
|
|
#include "derivations.hh"
|
2013-09-06 19:00:36 +00:00
|
|
|
|
#include "affinity.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;
|
2013-09-06 17:51:59 +00:00
|
|
|
|
StringSet varNames;
|
|
|
|
|
|
|
|
|
|
StringSet completions;
|
|
|
|
|
StringSet::iterator curCompletion;
|
2013-09-02 15:53:58 +00:00
|
|
|
|
|
|
|
|
|
NixRepl();
|
2013-09-06 13:20:06 +00:00
|
|
|
|
void mainLoop(const Strings & args);
|
2013-09-06 17:51:59 +00:00
|
|
|
|
void completePrefix(string prefix);
|
|
|
|
|
bool getLine(string & line);
|
2013-09-02 15:53:58 +00:00
|
|
|
|
void processLine(string line);
|
2013-09-06 13:20:06 +00:00
|
|
|
|
void loadFile(const Path & path);
|
2013-09-02 16:18:27 +00:00
|
|
|
|
void addAttrsToScope(Value & attrs);
|
2013-09-09 11:56:53 +00:00
|
|
|
|
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-09 09:14:43 +00:00
|
|
|
|
|
|
|
|
|
typedef set<Value *> ValuesSeen;
|
2013-09-06 22:35:54 +00:00
|
|
|
|
std::ostream & printValue(std::ostream & str, Value & v, unsigned int maxDepth);
|
2013-09-09 09:14:43 +00:00
|
|
|
|
std::ostream & printValue(std::ostream & str, Value & v, unsigned int maxDepth, ValuesSeen & seen);
|
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-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();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2013-09-06 13:20:06 +00:00
|
|
|
|
void NixRepl::mainLoop(const Strings & args)
|
2013-09-02 15:53:58 +00:00
|
|
|
|
{
|
2013-09-06 13:20:06 +00: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 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) {
|
2013-09-06 12:58:53 +00:00
|
|
|
|
printMsg(lvlError, "error: " + e.msg());
|
2013-09-06 11:20:35 +00:00
|
|
|
|
} catch (Interrupted & e) {
|
2013-09-06 12:58:53 +00:00
|
|
|
|
printMsg(lvlError, "error: " + e.msg());
|
2013-09-02 13:18:15 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
std::cout << std::endl;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
std::cout << std::endl;
|
|
|
|
|
}
|
2013-09-02 15:53:58 +00:00
|
|
|
|
|
|
|
|
|
|
2013-09-06 17:51:59 +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);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* 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();
|
|
|
|
|
|
2013-09-09 10:00:33 +00:00
|
|
|
|
size_t dot = prefix.rfind('.');
|
|
|
|
|
|
|
|
|
|
if (dot == string::npos) {
|
|
|
|
|
/* This is a variable name; look it up in the current scope. */
|
|
|
|
|
StringSet::iterator i = varNames.lower_bound(prefix);
|
|
|
|
|
while (i != varNames.end()) {
|
|
|
|
|
if (string(*i, 0, prefix.size()) != prefix) break;
|
|
|
|
|
completions.insert(*i);
|
|
|
|
|
i++;
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
try {
|
|
|
|
|
/* This is an expression that should evaluate to an
|
|
|
|
|
attribute set. Evaluate it to get the names of the
|
|
|
|
|
attributes. */
|
|
|
|
|
string expr(prefix, 0, dot);
|
|
|
|
|
string prefix2 = string(prefix, dot + 1);
|
|
|
|
|
|
|
|
|
|
Expr * e = parseString(expr);
|
|
|
|
|
Value v;
|
|
|
|
|
e->eval(state, *env, v);
|
|
|
|
|
state.forceAttrs(v);
|
|
|
|
|
|
|
|
|
|
foreach (Bindings::iterator, i, *v.attrs) {
|
|
|
|
|
string name = i->name;
|
|
|
|
|
if (string(name, 0, prefix2.size()) != prefix2) continue;
|
|
|
|
|
completions.insert(expr + "." + name);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
} catch (ParseError & e) {
|
|
|
|
|
// Quietly ignore parse errors.
|
|
|
|
|
}catch (EvalError & e) {
|
|
|
|
|
// Quietly ignore evaluation errors.
|
|
|
|
|
}
|
2013-09-06 17:51:59 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2013-09-06 19:00:36 +00:00
|
|
|
|
static int runProgram(const string & program, const Strings & args)
|
|
|
|
|
{
|
|
|
|
|
std::vector<const char *> cargs; /* careful with c_str()! */
|
|
|
|
|
cargs.push_back(program.c_str());
|
|
|
|
|
for (Strings::const_iterator i = args.begin(); i != args.end(); ++i)
|
|
|
|
|
cargs.push_back(i->c_str());
|
|
|
|
|
cargs.push_back(0);
|
|
|
|
|
|
|
|
|
|
Pid pid;
|
|
|
|
|
pid = fork();
|
|
|
|
|
if (pid == -1) throw SysError("forking");
|
|
|
|
|
if (pid == 0) {
|
|
|
|
|
restoreAffinity();
|
|
|
|
|
execvp(program.c_str(), (char * *) &cargs[0]);
|
|
|
|
|
_exit(1);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return pid.wait(true);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2013-09-09 11:56:53 +00:00
|
|
|
|
bool isVarName(const string & s)
|
|
|
|
|
{
|
|
|
|
|
// FIXME: not quite correct.
|
|
|
|
|
foreach (string::const_iterator, i, s)
|
|
|
|
|
if (!((*i >= 'a' && *i <= 'z') ||
|
|
|
|
|
(*i >= 'A' && *i <= 'Z') ||
|
|
|
|
|
(*i >= '0' && *i <= '9') ||
|
|
|
|
|
*i == '_' || *i == '\''))
|
|
|
|
|
return false;
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
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-06 13:05:18 +00:00
|
|
|
|
string command = string(line, 0, 2);
|
|
|
|
|
|
2013-09-09 11:22:33 +00:00
|
|
|
|
if (command == ":?") {
|
|
|
|
|
cout << "The following commands are available:\n"
|
|
|
|
|
<< "\n"
|
2013-09-09 11:56:53 +00:00
|
|
|
|
<< " <expr> Evaluate and print expression\n"
|
|
|
|
|
<< " <x> = <expr> Bind expression to variable\n"
|
|
|
|
|
<< " :a <expr> Add attributes from resulting set to scope\n"
|
|
|
|
|
<< " :b <expr> Build derivation\n"
|
|
|
|
|
<< " :l <path> Load Nix expression and add it to scope\n"
|
|
|
|
|
<< " :p <expr> Evaluate and print expression recursively\n"
|
|
|
|
|
<< " :s <expr> Build dependencies of derivation, then start nix-shell\n"
|
|
|
|
|
<< " :t <expr> Describe result of evaluation\n";
|
2013-09-09 11:22:33 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
else if (command == ":a") {
|
2013-09-02 15:53:58 +00:00
|
|
|
|
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);
|
|
|
|
|
}
|
|
|
|
|
|
2013-09-06 13:05:18 +00:00
|
|
|
|
else if (command == ":l") {
|
2013-09-02 16:18:27 +00:00
|
|
|
|
state.resetFileCache();
|
2013-09-06 13:20:06 +00:00
|
|
|
|
loadFile(removeWhitespace(string(line, 2)));
|
2013-09-02 15:53:58 +00:00
|
|
|
|
}
|
|
|
|
|
|
2013-09-06 13:05:18 +00:00
|
|
|
|
else if (command == ":t") {
|
2013-09-02 16:00:48 +00:00
|
|
|
|
Value v;
|
|
|
|
|
evalString(string(line, 2), v);
|
|
|
|
|
std::cout << showType(v) << std::endl;
|
|
|
|
|
}
|
|
|
|
|
|
2013-09-06 13:05:18 +00:00
|
|
|
|
else if (command == ":b" || command == ":s") {
|
2013-09-06 12:58:53 +00: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 13:05:18 +00: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. */
|
2013-09-06 19:00:36 +00:00
|
|
|
|
if (runProgram("nix-store", Strings{"-r", drvPath}) != 0) return;
|
2013-09-06 13:05:18 +00:00
|
|
|
|
Derivation drv = parseDerivation(readFile(drvPath));
|
2013-09-06 19:00:36 +00:00
|
|
|
|
std::cout << std::endl << "this derivation produced the following outputs:" << std::endl;
|
2013-09-06 13:05:18 +00:00
|
|
|
|
foreach (DerivationOutputs::iterator, i, drv.outputs)
|
|
|
|
|
std::cout << format(" %1% -> %2%") % i->first % i->second.path << std::endl;
|
2013-09-06 19:00:36 +00:00
|
|
|
|
} else
|
|
|
|
|
runProgram("nix-shell", Strings{drvPath});
|
2013-09-02 15:53:58 +00:00
|
|
|
|
}
|
|
|
|
|
|
2013-09-06 22:35:54 +00:00
|
|
|
|
else if (command == ":p") {
|
|
|
|
|
Value v;
|
|
|
|
|
evalString(string(line, 2), v);
|
|
|
|
|
printValue(std::cout, v, 1000000000) << std::endl;
|
|
|
|
|
}
|
|
|
|
|
|
2013-09-06 12:58:53 +00:00
|
|
|
|
else if (string(line, 0, 1) == ":")
|
|
|
|
|
throw Error(format("unknown command ‘%1%’") % string(line, 0, 2));
|
|
|
|
|
|
2013-09-02 15:53:58 +00:00
|
|
|
|
else {
|
2013-09-09 11:56:53 +00:00
|
|
|
|
size_t p = line.find('=');
|
|
|
|
|
string name;
|
|
|
|
|
if (p != string::npos &&
|
|
|
|
|
isVarName(name = removeWhitespace(string(line, 0, p))))
|
|
|
|
|
{
|
|
|
|
|
Expr * e = parseString(string(line, p + 1));
|
|
|
|
|
Value & v(*state.allocValue());
|
|
|
|
|
v.type = tThunk;
|
|
|
|
|
v.thunk.env = env;
|
|
|
|
|
v.thunk.expr = e;
|
|
|
|
|
addVarToScope(state.symbols.create(name), v);
|
|
|
|
|
} else {
|
|
|
|
|
Value v;
|
|
|
|
|
evalString(line, v);
|
|
|
|
|
printValue(std::cout, v, 1) << std::endl;
|
|
|
|
|
}
|
2013-09-02 15:53:58 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2013-09-06 13:20:06 +00: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 16:18:27 +00:00
|
|
|
|
void NixRepl::addAttrsToScope(Value & attrs)
|
|
|
|
|
{
|
|
|
|
|
state.forceAttrs(attrs);
|
|
|
|
|
foreach (Bindings::iterator, i, *attrs.attrs)
|
2013-09-09 11:56:53 +00:00
|
|
|
|
addVarToScope(i->name, *i->value);
|
2013-09-06 13:20:06 +00:00
|
|
|
|
std::cout << format("Added %1% variables.") % attrs.attrs->size() << std::endl;
|
2013-09-02 16:18:27 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2013-09-09 11:56:53 +00:00
|
|
|
|
void NixRepl::addVarToScope(const Symbol & name, Value & v)
|
2013-09-02 15:53:58 +00:00
|
|
|
|
{
|
|
|
|
|
staticEnv.vars[name] = displ;
|
2013-09-09 11:56:53 +00:00
|
|
|
|
env->values[displ++] = &v;
|
2013-09-06 17:51:59 +00:00
|
|
|
|
varNames.insert((string) name);
|
2013-09-02 15:53:58 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
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-09 09:14:43 +00:00
|
|
|
|
std::ostream & NixRepl::printValue(std::ostream & str, Value & v, unsigned int maxDepth)
|
|
|
|
|
{
|
|
|
|
|
ValuesSeen seen;
|
|
|
|
|
return printValue(str, v, maxDepth, seen);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2013-09-06 22:35:54 +00:00
|
|
|
|
// FIXME: lot of cut&paste from Nix's eval.cc.
|
2013-09-09 09:14:43 +00:00
|
|
|
|
std::ostream & NixRepl::printValue(std::ostream & str, Value & v, unsigned int maxDepth, ValuesSeen & seen)
|
2013-09-06 22:35:54 +00:00
|
|
|
|
{
|
|
|
|
|
str.flush();
|
|
|
|
|
checkInterrupt();
|
|
|
|
|
|
|
|
|
|
state.forceValue(v);
|
|
|
|
|
|
|
|
|
|
switch (v.type) {
|
|
|
|
|
|
|
|
|
|
case tInt:
|
|
|
|
|
str << v.integer;
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case tBool:
|
|
|
|
|
str << (v.boolean ? "true" : "false");
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case tString:
|
|
|
|
|
str << "\"";
|
|
|
|
|
for (const char * i = v.string.s; *i; i++)
|
|
|
|
|
if (*i == '\"' || *i == '\\') str << "\\" << *i;
|
|
|
|
|
else if (*i == '\n') str << "\\n";
|
|
|
|
|
else if (*i == '\r') str << "\\r";
|
|
|
|
|
else if (*i == '\t') str << "\\t";
|
|
|
|
|
else str << *i;
|
|
|
|
|
str << "\"";
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case tPath:
|
|
|
|
|
str << v.path; // !!! escaping?
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case tNull:
|
|
|
|
|
str << "null";
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case tAttrs: {
|
2013-09-09 09:14:43 +00:00
|
|
|
|
seen.insert(&v);
|
|
|
|
|
|
2013-09-06 22:35:54 +00:00
|
|
|
|
bool isDrv = state.isDerivation(v);
|
|
|
|
|
if (isDrv) str << "(derivation ";
|
|
|
|
|
str << "{ ";
|
|
|
|
|
|
|
|
|
|
if (maxDepth > 0) {
|
|
|
|
|
typedef std::map<string, Value *> Sorted;
|
|
|
|
|
Sorted sorted;
|
|
|
|
|
foreach (Bindings::iterator, i, *v.attrs)
|
|
|
|
|
sorted[i->name] = i->value;
|
|
|
|
|
|
|
|
|
|
/* If this is a derivation, then don't show the
|
|
|
|
|
self-references ("all", "out", etc.). */
|
|
|
|
|
StringSet hidden;
|
|
|
|
|
if (isDrv) {
|
|
|
|
|
hidden.insert("all");
|
|
|
|
|
Bindings::iterator i = v.attrs->find(state.sOutputs);
|
|
|
|
|
if (i == v.attrs->end())
|
|
|
|
|
hidden.insert("out");
|
|
|
|
|
else {
|
|
|
|
|
state.forceList(*i->value);
|
|
|
|
|
for (unsigned int j = 0; j < i->value->list.length; ++j)
|
|
|
|
|
hidden.insert(state.forceStringNoCtx(*i->value->list.elems[j]));
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2013-09-09 09:34:54 +00:00
|
|
|
|
foreach (Sorted::iterator, i, sorted) {
|
|
|
|
|
str << i->first << " = ";
|
2013-09-09 09:14:43 +00:00
|
|
|
|
if (hidden.find(i->first) != hidden.end())
|
2013-09-09 09:34:54 +00:00
|
|
|
|
str << "«...»";
|
2013-09-09 09:14:43 +00:00
|
|
|
|
else if (seen.find(i->second) != seen.end())
|
2013-09-09 09:34:54 +00:00
|
|
|
|
str << "«repeated»";
|
2013-09-06 22:35:54 +00:00
|
|
|
|
else
|
2013-09-09 09:34:54 +00:00
|
|
|
|
try {
|
|
|
|
|
printValue(str, *i->second, maxDepth - 1, seen);
|
|
|
|
|
} catch (AssertionError & e) {
|
|
|
|
|
str << "«error: " << e.msg() << "»";
|
|
|
|
|
}
|
|
|
|
|
str << "; ";
|
|
|
|
|
}
|
2013-09-06 22:35:54 +00:00
|
|
|
|
|
|
|
|
|
} else
|
|
|
|
|
str << "... ";
|
|
|
|
|
|
|
|
|
|
str << "}";
|
|
|
|
|
if (isDrv) str << ")";
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
case tList:
|
2013-09-09 09:14:43 +00:00
|
|
|
|
seen.insert(&v);
|
|
|
|
|
|
2013-09-06 22:35:54 +00:00
|
|
|
|
str << "[ ";
|
|
|
|
|
if (maxDepth > 0)
|
2013-09-09 09:14:43 +00:00
|
|
|
|
for (unsigned int n = 0; n < v.list.length; ++n) {
|
|
|
|
|
if (seen.find(v.list.elems[n]) != seen.end())
|
2013-09-09 09:34:54 +00:00
|
|
|
|
str << "«repeated»";
|
2013-09-09 09:14:43 +00:00
|
|
|
|
else
|
2013-09-09 09:34:54 +00:00
|
|
|
|
try {
|
|
|
|
|
printValue(str, *v.list.elems[n], maxDepth - 1, seen);
|
|
|
|
|
} catch (AssertionError & e) {
|
|
|
|
|
str << "«error: " << e.msg() << "»";
|
|
|
|
|
}
|
|
|
|
|
str << " ";
|
2013-09-09 09:14:43 +00:00
|
|
|
|
}
|
2013-09-06 22:35:54 +00:00
|
|
|
|
else
|
|
|
|
|
str << "... ";
|
|
|
|
|
str << "]";
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case tLambda:
|
|
|
|
|
str << "«lambda»";
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case tPrimOp:
|
|
|
|
|
str << "«primop»";
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case tPrimOpApp:
|
|
|
|
|
str << "«primop-app»";
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
default:
|
|
|
|
|
str << "«unknown»";
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return str;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2013-09-06 13:20:06 +00:00
|
|
|
|
void run(Strings args)
|
2013-09-02 15:53:58 +00:00
|
|
|
|
{
|
|
|
|
|
NixRepl repl;
|
2013-09-06 13:20:06 +00:00
|
|
|
|
repl.mainLoop(args);
|
2013-09-02 15:53:58 +00:00
|
|
|
|
}
|