2003-03-13 16:28:32 +00:00
|
|
|
#include <iostream>
|
2003-07-24 08:53:43 +00:00
|
|
|
#include <sstream>
|
2003-03-14 16:43:14 +00:00
|
|
|
|
2003-06-17 21:12:58 +00:00
|
|
|
#include "globals.hh"
|
2003-07-20 19:29:38 +00:00
|
|
|
#include "normalise.hh"
|
2003-06-20 10:40:25 +00:00
|
|
|
#include "archive.hh"
|
2003-07-04 15:42:03 +00:00
|
|
|
#include "shared.hh"
|
2003-03-24 11:50:20 +00:00
|
|
|
|
|
|
|
|
2003-06-17 21:12:58 +00:00
|
|
|
typedef void (* Operation) (Strings opFlags, Strings opArgs);
|
2003-04-02 15:34:05 +00:00
|
|
|
|
|
|
|
|
2003-07-21 14:46:01 +00:00
|
|
|
static bool pathArgs = false;
|
2003-06-20 14:11:31 +00:00
|
|
|
|
|
|
|
|
2003-07-28 12:19:23 +00:00
|
|
|
static void printHelp()
|
|
|
|
{
|
|
|
|
cout <<
|
|
|
|
#include "nix-help.txt.hh"
|
|
|
|
;
|
|
|
|
exit(0);
|
|
|
|
}
|
2003-06-20 14:11:31 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
2003-07-15 22:28:27 +00:00
|
|
|
static FSId argToId(const string & arg)
|
2003-07-08 13:22:08 +00:00
|
|
|
{
|
2003-07-21 14:46:01 +00:00
|
|
|
if (!pathArgs)
|
2003-07-08 13:22:08 +00:00
|
|
|
return parseHash(arg);
|
2003-07-21 14:46:01 +00:00
|
|
|
else {
|
2003-07-15 22:28:27 +00:00
|
|
|
FSId id;
|
2003-07-21 14:46:01 +00:00
|
|
|
if (!queryPathId(arg, id))
|
|
|
|
throw Error(format("don't know id of `%1%'") % arg);
|
2003-07-15 22:28:27 +00:00
|
|
|
return id;
|
2003-07-08 13:22:08 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2003-07-08 10:00:46 +00:00
|
|
|
/* Realise (or install) paths from the given Nix fstate
|
|
|
|
expressions. */
|
2003-07-07 09:29:40 +00:00
|
|
|
static void opInstall(Strings opFlags, Strings opArgs)
|
2003-03-14 16:43:14 +00:00
|
|
|
{
|
2003-06-17 21:12:58 +00:00
|
|
|
if (!opFlags.empty()) throw UsageError("unknown flag");
|
2003-03-13 16:28:32 +00:00
|
|
|
|
2003-06-17 21:12:58 +00:00
|
|
|
for (Strings::iterator it = opArgs.begin();
|
|
|
|
it != opArgs.end(); it++)
|
2003-07-29 09:45:03 +00:00
|
|
|
{
|
|
|
|
FSId id = normaliseFState(argToId(*it));
|
|
|
|
realiseSlice(id);
|
|
|
|
cout << format("%1%\n") % (string) id;
|
|
|
|
}
|
2003-04-02 15:34:05 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2003-07-08 13:22:08 +00:00
|
|
|
/* Delete a path in the Nix store directory. */
|
2003-06-17 21:12:58 +00:00
|
|
|
static void opDelete(Strings opFlags, Strings opArgs)
|
2003-04-02 15:34:05 +00:00
|
|
|
{
|
2003-06-23 14:40:49 +00:00
|
|
|
if (!opFlags.empty()) throw UsageError("unknown flag");
|
2003-06-20 14:11:31 +00:00
|
|
|
|
2003-06-23 14:40:49 +00:00
|
|
|
for (Strings::iterator it = opArgs.begin();
|
|
|
|
it != opArgs.end(); it++)
|
2003-07-08 10:00:46 +00:00
|
|
|
deleteFromStore(absPath(*it));
|
2003-04-02 15:34:05 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2003-07-08 10:00:46 +00:00
|
|
|
/* Add paths to the Nix values directory and print the hashes of those
|
|
|
|
paths. */
|
2003-06-17 21:12:58 +00:00
|
|
|
static void opAdd(Strings opFlags, Strings opArgs)
|
2003-04-02 15:34:05 +00:00
|
|
|
{
|
2003-06-17 21:12:58 +00:00
|
|
|
if (!opFlags.empty()) throw UsageError("unknown flag");
|
2003-04-02 15:34:05 +00:00
|
|
|
|
2003-06-17 21:12:58 +00:00
|
|
|
for (Strings::iterator it = opArgs.begin();
|
|
|
|
it != opArgs.end(); it++)
|
2003-07-04 12:18:06 +00:00
|
|
|
{
|
|
|
|
string path;
|
2003-07-15 22:28:27 +00:00
|
|
|
FSId id;
|
|
|
|
addToStore(*it, path, id);
|
|
|
|
cout << format("%1% %2%\n") % (string) id % path;
|
2003-07-04 12:18:06 +00:00
|
|
|
}
|
2003-03-13 16:28:32 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2003-07-29 10:43:12 +00:00
|
|
|
static string dotQuote(const string & s)
|
2003-07-28 12:19:23 +00:00
|
|
|
{
|
|
|
|
return "\"" + s + "\"";
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2003-07-29 10:43:12 +00:00
|
|
|
FSId maybeNormalise(const FSId & id, bool normalise)
|
|
|
|
{
|
|
|
|
return normalise ? normaliseFState(id) : id;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2003-07-08 13:22:08 +00:00
|
|
|
/* Perform various sorts of queries. */
|
|
|
|
static void opQuery(Strings opFlags, Strings opArgs)
|
|
|
|
{
|
2003-07-29 14:28:17 +00:00
|
|
|
enum { qList, qRequisites, qGenerators, qExpansion, qGraph
|
2003-07-28 12:19:23 +00:00
|
|
|
} query = qList;
|
2003-07-29 10:43:12 +00:00
|
|
|
bool normalise = false;
|
2003-07-29 14:28:17 +00:00
|
|
|
bool includeExprs = true;
|
|
|
|
bool includeSuccessors = false;
|
2003-07-21 14:46:01 +00:00
|
|
|
|
|
|
|
for (Strings::iterator i = opFlags.begin();
|
|
|
|
i != opFlags.end(); i++)
|
2003-07-22 10:24:22 +00:00
|
|
|
if (*i == "--list" || *i == "-l") query = qList;
|
2003-07-29 14:28:17 +00:00
|
|
|
else if (*i == "--requisites" || *i == "-r") query = qRequisites;
|
2003-07-21 14:46:01 +00:00
|
|
|
else if (*i == "--generators" || *i == "-g") query = qGenerators;
|
2003-07-22 10:24:22 +00:00
|
|
|
else if (*i == "--expansion" || *i == "-e") query = qExpansion;
|
2003-07-28 12:19:23 +00:00
|
|
|
else if (*i == "--graph") query = qGraph;
|
2003-07-29 10:43:12 +00:00
|
|
|
else if (*i == "--normalise" || *i == "-n") normalise = true;
|
2003-07-29 14:28:17 +00:00
|
|
|
else if (*i == "--exclude-exprs") includeExprs = false;
|
|
|
|
else if (*i == "--include-successors") includeSuccessors = true;
|
2003-07-21 14:46:01 +00:00
|
|
|
else throw UsageError(format("unknown flag `%1%'") % *i);
|
|
|
|
|
|
|
|
switch (query) {
|
|
|
|
|
2003-07-22 10:24:22 +00:00
|
|
|
case qList: {
|
2003-07-21 14:46:01 +00:00
|
|
|
StringSet paths;
|
|
|
|
for (Strings::iterator i = opArgs.begin();
|
|
|
|
i != opArgs.end(); i++)
|
|
|
|
{
|
2003-07-29 10:43:12 +00:00
|
|
|
Strings paths2 = fstatePaths(
|
|
|
|
maybeNormalise(argToId(*i), normalise));
|
2003-07-21 14:46:01 +00:00
|
|
|
paths.insert(paths2.begin(), paths2.end());
|
|
|
|
}
|
|
|
|
for (StringSet::iterator i = paths.begin();
|
|
|
|
i != paths.end(); i++)
|
|
|
|
cout << format("%s\n") % *i;
|
2003-07-08 13:22:08 +00:00
|
|
|
break;
|
2003-07-10 13:41:28 +00:00
|
|
|
}
|
2003-07-08 13:22:08 +00:00
|
|
|
|
2003-07-29 14:28:17 +00:00
|
|
|
case qRequisites: {
|
2003-07-21 14:46:01 +00:00
|
|
|
StringSet paths;
|
|
|
|
for (Strings::iterator i = opArgs.begin();
|
|
|
|
i != opArgs.end(); i++)
|
|
|
|
{
|
2003-07-29 14:28:17 +00:00
|
|
|
Strings paths2 = fstateRequisites(
|
|
|
|
maybeNormalise(argToId(*i), normalise),
|
|
|
|
includeExprs, includeSuccessors);
|
2003-07-21 14:46:01 +00:00
|
|
|
paths.insert(paths2.begin(), paths2.end());
|
|
|
|
}
|
|
|
|
for (StringSet::iterator i = paths.begin();
|
|
|
|
i != paths.end(); i++)
|
|
|
|
cout << format("%s\n") % *i;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case qGenerators: {
|
|
|
|
FSIds outIds;
|
|
|
|
for (Strings::iterator i = opArgs.begin();
|
|
|
|
i != opArgs.end(); i++)
|
|
|
|
outIds.push_back(argToId(*i));
|
|
|
|
|
|
|
|
FSIds genIds = findGenerators(outIds);
|
|
|
|
|
|
|
|
for (FSIds::iterator i = genIds.begin();
|
|
|
|
i != genIds.end(); i++)
|
2003-07-21 21:31:03 +00:00
|
|
|
cout << format("%s\n") % expandId(*i);
|
2003-07-08 13:22:08 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2003-07-22 10:24:22 +00:00
|
|
|
case qExpansion: {
|
|
|
|
for (Strings::iterator i = opArgs.begin();
|
|
|
|
i != opArgs.end(); i++)
|
|
|
|
/* !!! should not use substitutes; this is a query,
|
|
|
|
it should not have side-effects */
|
|
|
|
cout << format("%s\n") % expandId(parseHash(*i));
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2003-07-28 12:19:23 +00:00
|
|
|
case qGraph: {
|
|
|
|
|
|
|
|
FSIds workList;
|
|
|
|
|
|
|
|
for (Strings::iterator i = opArgs.begin();
|
|
|
|
i != opArgs.end(); i++)
|
|
|
|
workList.push_back(argToId(*i));
|
|
|
|
|
|
|
|
FSIdSet doneSet;
|
|
|
|
|
|
|
|
cout << "digraph G {\n";
|
|
|
|
|
|
|
|
while (!workList.empty()) {
|
|
|
|
FSId id = workList.front();
|
|
|
|
workList.pop_front();
|
|
|
|
|
|
|
|
if (doneSet.find(id) == doneSet.end()) {
|
|
|
|
doneSet.insert(id);
|
|
|
|
|
|
|
|
FState fs = parseFState(termFromId(id));
|
|
|
|
|
2003-07-29 10:53:27 +00:00
|
|
|
string label, shape;
|
2003-07-28 12:19:23 +00:00
|
|
|
|
|
|
|
if (fs.type == FState::fsDerive) {
|
|
|
|
for (FSIds::iterator i = fs.derive.inputs.begin();
|
|
|
|
i != fs.derive.inputs.end(); i++)
|
|
|
|
{
|
|
|
|
workList.push_back(*i);
|
|
|
|
cout << dotQuote(*i) << " -> "
|
|
|
|
<< dotQuote(id) << ";\n";
|
|
|
|
}
|
|
|
|
|
|
|
|
label = "derive";
|
2003-07-29 10:53:27 +00:00
|
|
|
shape = "box";
|
2003-07-28 12:19:23 +00:00
|
|
|
for (StringPairs::iterator i = fs.derive.env.begin();
|
|
|
|
i != fs.derive.env.end(); i++)
|
|
|
|
if (i->first == "name") label = i->second;
|
|
|
|
}
|
|
|
|
|
|
|
|
else if (fs.type == FState::fsSlice) {
|
|
|
|
label = baseNameOf((*fs.slice.elems.begin()).path);
|
2003-07-29 10:53:27 +00:00
|
|
|
shape = "ellipse";
|
|
|
|
if (isHash(string(label, 0, Hash::hashSize * 2)) &&
|
|
|
|
label[Hash::hashSize * 2] == '-')
|
|
|
|
label = string(label, Hash::hashSize * 2 + 1);
|
2003-07-28 12:19:23 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
else abort();
|
|
|
|
|
|
|
|
cout << dotQuote(id) << "[label = "
|
|
|
|
<< dotQuote(label)
|
2003-07-29 10:53:27 +00:00
|
|
|
<< ", shape = " << shape
|
2003-07-28 12:19:23 +00:00
|
|
|
<< "];\n";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
cout << "}\n";
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2003-07-08 13:22:08 +00:00
|
|
|
default:
|
|
|
|
abort();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2003-07-10 18:48:11 +00:00
|
|
|
static void opSuccessor(Strings opFlags, Strings opArgs)
|
|
|
|
{
|
|
|
|
if (!opFlags.empty()) throw UsageError("unknown flag");
|
|
|
|
if (opArgs.size() % 2) throw UsageError("expecting even number of arguments");
|
2003-08-01 15:41:47 +00:00
|
|
|
|
|
|
|
Transaction txn(nixDB); /* !!! this could be a big transaction */
|
2003-07-10 18:48:11 +00:00
|
|
|
for (Strings::iterator i = opArgs.begin();
|
|
|
|
i != opArgs.end(); )
|
|
|
|
{
|
2003-07-15 22:28:27 +00:00
|
|
|
FSId id1 = parseHash(*i++);
|
|
|
|
FSId id2 = parseHash(*i++);
|
2003-08-01 15:41:47 +00:00
|
|
|
registerSuccessor(txn, id1, id2);
|
2003-07-10 18:48:11 +00:00
|
|
|
}
|
2003-08-01 15:41:47 +00:00
|
|
|
txn.commit();
|
2003-07-10 18:48:11 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2003-07-10 15:11:48 +00:00
|
|
|
static void opSubstitute(Strings opFlags, Strings opArgs)
|
|
|
|
{
|
|
|
|
if (!opFlags.empty()) throw UsageError("unknown flag");
|
|
|
|
if (opArgs.size() % 2) throw UsageError("expecting even number of arguments");
|
|
|
|
|
|
|
|
for (Strings::iterator i = opArgs.begin();
|
|
|
|
i != opArgs.end(); )
|
|
|
|
{
|
2003-07-15 22:28:27 +00:00
|
|
|
FSId src = parseHash(*i++);
|
|
|
|
FSId sub = parseHash(*i++);
|
|
|
|
registerSubstitute(src, sub);
|
2003-07-10 15:11:48 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2003-06-18 14:34:43 +00:00
|
|
|
/* A sink that writes dump output to stdout. */
|
|
|
|
struct StdoutSink : DumpSink
|
|
|
|
{
|
|
|
|
virtual void operator ()
|
|
|
|
(const unsigned char * data, unsigned int len)
|
|
|
|
{
|
2003-07-20 21:11:43 +00:00
|
|
|
writeFull(STDOUT_FILENO, data, len);
|
2003-06-18 14:34:43 +00:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
|
2003-07-08 10:00:46 +00:00
|
|
|
/* Dump a path as a Nix archive. The archive is written to standard
|
2003-06-23 14:08:34 +00:00
|
|
|
output. */
|
2003-06-18 14:34:43 +00:00
|
|
|
static void opDump(Strings opFlags, Strings opArgs)
|
|
|
|
{
|
|
|
|
if (!opFlags.empty()) throw UsageError("unknown flag");
|
|
|
|
if (opArgs.size() != 1) throw UsageError("only one argument allowed");
|
|
|
|
|
|
|
|
StdoutSink sink;
|
2003-06-20 14:11:31 +00:00
|
|
|
string arg = *opArgs.begin();
|
2003-07-21 14:46:01 +00:00
|
|
|
string path = pathArgs ? arg : expandId(parseHash(arg));
|
2003-06-20 14:11:31 +00:00
|
|
|
|
2003-06-23 14:08:34 +00:00
|
|
|
dumpPath(path, sink);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/* A source that read restore intput to stdin. */
|
|
|
|
struct StdinSource : RestoreSource
|
|
|
|
{
|
2003-07-20 21:11:43 +00:00
|
|
|
virtual void operator () (unsigned char * data, unsigned int len)
|
2003-06-23 14:08:34 +00:00
|
|
|
{
|
2003-07-20 21:11:43 +00:00
|
|
|
readFull(STDIN_FILENO, data, len);
|
2003-06-23 14:08:34 +00:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
/* Restore a value from a Nix archive. The archive is written to
|
|
|
|
standard input. */
|
|
|
|
static void opRestore(Strings opFlags, Strings opArgs)
|
|
|
|
{
|
|
|
|
if (!opFlags.empty()) throw UsageError("unknown flag");
|
|
|
|
if (opArgs.size() != 1) throw UsageError("only one argument allowed");
|
|
|
|
|
|
|
|
StdinSource source;
|
|
|
|
restorePath(*opArgs.begin(), source);
|
2003-06-18 14:34:43 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2003-06-17 21:12:58 +00:00
|
|
|
/* Initialise the Nix databases. */
|
|
|
|
static void opInit(Strings opFlags, Strings opArgs)
|
2003-05-26 09:44:18 +00:00
|
|
|
{
|
2003-06-17 21:12:58 +00:00
|
|
|
if (!opFlags.empty()) throw UsageError("unknown flag");
|
|
|
|
if (!opArgs.empty())
|
|
|
|
throw UsageError("--init does not have arguments");
|
|
|
|
initDB();
|
2003-03-21 15:53:35 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2003-07-17 12:27:55 +00:00
|
|
|
/* Verify the consistency of the Nix environment. */
|
|
|
|
static void opVerify(Strings opFlags, Strings opArgs)
|
|
|
|
{
|
|
|
|
verifyStore();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2003-07-04 15:42:03 +00:00
|
|
|
/* Scan the arguments; find the operation, set global flags, put all
|
|
|
|
other flags in a list, and put all other arguments in another
|
|
|
|
list. */
|
|
|
|
void run(Strings args)
|
2003-03-24 17:49:56 +00:00
|
|
|
{
|
2003-07-31 13:47:13 +00:00
|
|
|
openDB();
|
|
|
|
|
2003-06-20 14:11:31 +00:00
|
|
|
Strings opFlags, opArgs;
|
|
|
|
Operation op = 0;
|
|
|
|
|
2003-07-24 08:53:43 +00:00
|
|
|
for (Strings::iterator it = args.begin(); it != args.end(); )
|
2003-06-20 14:11:31 +00:00
|
|
|
{
|
2003-07-24 08:53:43 +00:00
|
|
|
string arg = *it++;
|
2003-03-24 17:49:56 +00:00
|
|
|
|
2003-06-17 21:12:58 +00:00
|
|
|
Operation oldOp = op;
|
2003-03-24 17:49:56 +00:00
|
|
|
|
2003-07-07 09:29:40 +00:00
|
|
|
if (arg == "--install" || arg == "-i")
|
|
|
|
op = opInstall;
|
2003-06-17 21:12:58 +00:00
|
|
|
else if (arg == "--delete" || arg == "-d")
|
|
|
|
op = opDelete;
|
2003-07-08 13:22:08 +00:00
|
|
|
else if (arg == "--add" || arg == "-A")
|
2003-06-17 21:12:58 +00:00
|
|
|
op = opAdd;
|
2003-07-08 13:22:08 +00:00
|
|
|
else if (arg == "--query" || arg == "-q")
|
|
|
|
op = opQuery;
|
2003-07-10 18:48:11 +00:00
|
|
|
else if (arg == "--successor")
|
|
|
|
op = opSuccessor;
|
2003-07-10 15:11:48 +00:00
|
|
|
else if (arg == "--substitute")
|
|
|
|
op = opSubstitute;
|
2003-06-18 14:34:43 +00:00
|
|
|
else if (arg == "--dump")
|
|
|
|
op = opDump;
|
2003-06-23 14:08:34 +00:00
|
|
|
else if (arg == "--restore")
|
|
|
|
op = opRestore;
|
2003-06-17 21:12:58 +00:00
|
|
|
else if (arg == "--init")
|
|
|
|
op = opInit;
|
2003-07-17 12:27:55 +00:00
|
|
|
else if (arg == "--verify")
|
|
|
|
op = opVerify;
|
2003-07-21 14:46:01 +00:00
|
|
|
else if (arg == "--path" || arg == "-p")
|
|
|
|
pathArgs = true;
|
2003-07-24 13:43:16 +00:00
|
|
|
else if (arg == "--verbose" || arg == "-v")
|
|
|
|
verbosity = (Verbosity) ((int) verbosity + 1);
|
2003-07-28 12:19:23 +00:00
|
|
|
else if (arg == "--help")
|
|
|
|
printHelp();
|
2003-06-17 21:12:58 +00:00
|
|
|
else if (arg[0] == '-')
|
|
|
|
opFlags.push_back(arg);
|
|
|
|
else
|
|
|
|
opArgs.push_back(arg);
|
2003-05-25 22:42:19 +00:00
|
|
|
|
2003-06-17 21:12:58 +00:00
|
|
|
if (oldOp && oldOp != op)
|
|
|
|
throw UsageError("only one operation may be specified");
|
2003-05-25 22:42:19 +00:00
|
|
|
}
|
|
|
|
|
2003-06-17 21:12:58 +00:00
|
|
|
if (!op) throw UsageError("no operation specified");
|
2003-03-20 16:53:00 +00:00
|
|
|
|
2003-06-17 21:12:58 +00:00
|
|
|
op(opFlags, opArgs);
|
2003-03-20 16:53:00 +00:00
|
|
|
}
|
2003-03-14 16:43:14 +00:00
|
|
|
|
2003-03-24 17:49:56 +00:00
|
|
|
|
2003-07-04 15:42:03 +00:00
|
|
|
string programId = "nix";
|