2003-03-13 17:28:32 +01:00
|
|
|
#include <iostream>
|
2003-03-14 17:43:14 +01:00
|
|
|
|
2003-06-17 23:12:58 +02:00
|
|
|
#include "globals.hh"
|
2003-07-07 09:43:58 +02:00
|
|
|
#include "store.hh"
|
|
|
|
#include "fstate.hh"
|
2003-06-20 12:40:25 +02:00
|
|
|
#include "archive.hh"
|
2003-07-04 17:42:03 +02:00
|
|
|
#include "shared.hh"
|
2003-03-24 12:50:20 +01:00
|
|
|
|
|
|
|
|
2003-06-17 23:12:58 +02:00
|
|
|
typedef void (* Operation) (Strings opFlags, Strings opArgs);
|
2003-04-02 17:34:05 +02:00
|
|
|
|
|
|
|
|
2003-07-08 12:00:46 +02:00
|
|
|
typedef enum { atpHash, atpPath, atpUnknown } ArgType;
|
2003-03-14 17:43:14 +01:00
|
|
|
|
2003-06-20 16:11:31 +02:00
|
|
|
static ArgType argType = atpUnknown;
|
|
|
|
|
|
|
|
|
|
|
|
/* Nix syntax:
|
|
|
|
|
|
|
|
nix [OPTIONS...] [ARGUMENTS...]
|
|
|
|
|
|
|
|
Operations:
|
|
|
|
|
2003-07-08 15:22:08 +02:00
|
|
|
--install / -i: realise an fstate
|
2003-07-08 12:00:46 +02:00
|
|
|
--delete / -d: delete paths from the Nix store
|
|
|
|
--add / -A: copy a path to the Nix store
|
2003-07-08 15:22:08 +02:00
|
|
|
--query / -q: query information
|
2003-06-23 16:08:34 +02:00
|
|
|
|
2003-07-10 20:48:11 +02:00
|
|
|
--successor: register a successor expression
|
2003-07-10 17:11:48 +02:00
|
|
|
--substitute: register a substitute expression
|
|
|
|
|
2003-07-08 12:00:46 +02:00
|
|
|
--dump: dump a path as a Nix archive
|
|
|
|
--restore: restore a path from a Nix archive
|
2003-06-23 16:08:34 +02:00
|
|
|
|
2003-06-20 16:11:31 +02:00
|
|
|
--init: initialise the Nix database
|
2003-06-23 16:08:34 +02:00
|
|
|
--verify: verify Nix structures
|
|
|
|
|
2003-06-20 16:11:31 +02:00
|
|
|
--version: output version information
|
|
|
|
--help: display help
|
|
|
|
|
2003-07-08 12:00:46 +02:00
|
|
|
Source selection for --install, --dump:
|
2003-06-20 16:11:31 +02:00
|
|
|
|
2003-07-10 15:41:28 +02:00
|
|
|
--file / -f: by file name !!! -> path
|
2003-06-20 16:11:31 +02:00
|
|
|
--hash / -h: by hash
|
|
|
|
|
2003-07-08 15:22:08 +02:00
|
|
|
Query flags:
|
|
|
|
|
|
|
|
--path / -p: query the path of an fstate
|
|
|
|
--refs / -r: query paths referenced by an fstate
|
|
|
|
|
2003-06-20 16:11:31 +02:00
|
|
|
Options:
|
|
|
|
|
|
|
|
--verbose / -v: verbose operation
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
2003-07-08 12:00:46 +02:00
|
|
|
/* Parse the `-f' / `-h' / flags, i.e., the type of arguments. These
|
|
|
|
flags are deleted from the referenced vector. */
|
2003-06-25 16:58:56 +02:00
|
|
|
static void getArgType(Strings & flags)
|
2003-06-20 16:11:31 +02:00
|
|
|
{
|
|
|
|
for (Strings::iterator it = flags.begin();
|
|
|
|
it != flags.end(); )
|
|
|
|
{
|
|
|
|
string arg = *it;
|
|
|
|
ArgType tp;
|
2003-07-08 15:22:08 +02:00
|
|
|
if (arg == "--hash" || arg == "-h") tp = atpHash;
|
|
|
|
else if (arg == "--file" || arg == "-f") tp = atpPath;
|
2003-07-08 12:00:46 +02:00
|
|
|
else { it++; continue; }
|
2003-06-20 16:11:31 +02:00
|
|
|
if (argType != atpUnknown)
|
|
|
|
throw UsageError("only one argument type specified may be specified");
|
|
|
|
argType = tp;
|
|
|
|
it = flags.erase(it);
|
2003-03-20 17:53:00 +01:00
|
|
|
}
|
2003-06-20 16:11:31 +02:00
|
|
|
if (argType == atpUnknown)
|
|
|
|
throw UsageError("argument type not specified");
|
2003-03-14 17:43:14 +01:00
|
|
|
}
|
|
|
|
|
2003-03-13 17:28:32 +01:00
|
|
|
|
2003-07-08 15:22:08 +02:00
|
|
|
static Hash argToHash(const string & arg)
|
|
|
|
{
|
|
|
|
if (argType == atpHash)
|
|
|
|
return parseHash(arg);
|
|
|
|
else if (argType == atpPath) {
|
|
|
|
string path;
|
|
|
|
Hash hash;
|
|
|
|
addToStore(arg, path, hash);
|
|
|
|
return hash;
|
|
|
|
}
|
|
|
|
else abort();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2003-07-08 12:00:46 +02:00
|
|
|
/* Realise (or install) paths from the given Nix fstate
|
|
|
|
expressions. */
|
2003-07-07 11:29:40 +02:00
|
|
|
static void opInstall(Strings opFlags, Strings opArgs)
|
2003-03-14 17:43:14 +01:00
|
|
|
{
|
2003-06-20 16:11:31 +02:00
|
|
|
getArgType(opFlags);
|
2003-06-17 23:12:58 +02:00
|
|
|
if (!opFlags.empty()) throw UsageError("unknown flag");
|
2003-03-13 17:28:32 +01:00
|
|
|
|
2003-06-17 23:12:58 +02:00
|
|
|
for (Strings::iterator it = opArgs.begin();
|
|
|
|
it != opArgs.end(); it++)
|
2003-07-09 17:02:03 +02:00
|
|
|
{
|
|
|
|
StringSet paths;
|
2003-07-10 15:41:28 +02:00
|
|
|
realiseFState(hash2fstate(argToHash(*it)), paths);
|
2003-07-09 17:02:03 +02:00
|
|
|
}
|
2003-04-02 17:34:05 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2003-07-08 15:22:08 +02:00
|
|
|
/* Delete a path in the Nix store directory. */
|
2003-06-17 23:12:58 +02:00
|
|
|
static void opDelete(Strings opFlags, Strings opArgs)
|
2003-04-02 17:34:05 +02:00
|
|
|
{
|
2003-06-23 16:40:49 +02:00
|
|
|
if (!opFlags.empty()) throw UsageError("unknown flag");
|
2003-06-20 16:11:31 +02:00
|
|
|
|
2003-06-23 16:40:49 +02:00
|
|
|
for (Strings::iterator it = opArgs.begin();
|
|
|
|
it != opArgs.end(); it++)
|
2003-07-08 12:00:46 +02:00
|
|
|
deleteFromStore(absPath(*it));
|
2003-04-02 17:34:05 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2003-07-08 12:00:46 +02:00
|
|
|
/* Add paths to the Nix values directory and print the hashes of those
|
|
|
|
paths. */
|
2003-06-17 23:12:58 +02:00
|
|
|
static void opAdd(Strings opFlags, Strings opArgs)
|
2003-04-02 17:34:05 +02:00
|
|
|
{
|
2003-06-20 16:11:31 +02:00
|
|
|
getArgType(opFlags);
|
2003-06-17 23:12:58 +02:00
|
|
|
if (!opFlags.empty()) throw UsageError("unknown flag");
|
2003-04-02 17:34:05 +02:00
|
|
|
|
2003-06-17 23:12:58 +02:00
|
|
|
for (Strings::iterator it = opArgs.begin();
|
|
|
|
it != opArgs.end(); it++)
|
2003-07-04 14:18:06 +02:00
|
|
|
{
|
|
|
|
string path;
|
|
|
|
Hash hash;
|
|
|
|
addToStore(*it, path, hash);
|
|
|
|
cout << format("%1% %2%\n") % (string) hash % path;
|
|
|
|
}
|
2003-03-13 17:28:32 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2003-07-08 15:22:08 +02:00
|
|
|
/* Perform various sorts of queries. */
|
|
|
|
static void opQuery(Strings opFlags, Strings opArgs)
|
|
|
|
{
|
|
|
|
enum { qPath, qRefs, qUnknown } query = qPath;
|
|
|
|
|
|
|
|
for (Strings::iterator it = opFlags.begin();
|
|
|
|
it != opFlags.end(); )
|
|
|
|
{
|
|
|
|
string arg = *it;
|
|
|
|
if (arg == "--path" || arg == "-p") query = qPath;
|
|
|
|
else if (arg == "--refs" || arg == "-r") query = qRefs;
|
|
|
|
else { it++; continue; }
|
|
|
|
it = opFlags.erase(it);
|
|
|
|
}
|
|
|
|
|
|
|
|
getArgType(opFlags);
|
|
|
|
if (!opFlags.empty()) throw UsageError("unknown flag");
|
|
|
|
|
|
|
|
for (Strings::iterator it = opArgs.begin();
|
|
|
|
it != opArgs.end(); it++)
|
|
|
|
{
|
|
|
|
Hash hash = argToHash(*it);
|
|
|
|
|
|
|
|
switch (query) {
|
|
|
|
|
2003-07-10 15:41:28 +02:00
|
|
|
case qPath: {
|
|
|
|
StringSet refs;
|
2003-07-08 15:22:08 +02:00
|
|
|
cout << format("%s\n") %
|
2003-07-10 15:41:28 +02:00
|
|
|
(string) fstatePath(realiseFState(termFromHash(hash), refs));
|
2003-07-08 15:22:08 +02:00
|
|
|
break;
|
2003-07-10 15:41:28 +02:00
|
|
|
}
|
2003-07-08 15:22:08 +02:00
|
|
|
|
|
|
|
case qRefs: {
|
2003-07-09 17:02:03 +02:00
|
|
|
StringSet refs;
|
2003-07-10 15:41:28 +02:00
|
|
|
FState fs = hash2fstate(hash);
|
2003-07-09 17:02:03 +02:00
|
|
|
fstateRefs(realiseFState(fs, refs), refs);
|
|
|
|
for (StringSet::iterator j = refs.begin();
|
2003-07-08 15:22:08 +02:00
|
|
|
j != refs.end(); j++)
|
|
|
|
cout << format("%s\n") % *j;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
default:
|
|
|
|
abort();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2003-07-10 20:48:11 +02: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");
|
|
|
|
|
|
|
|
for (Strings::iterator i = opArgs.begin();
|
|
|
|
i != opArgs.end(); )
|
|
|
|
{
|
|
|
|
Hash fsHash = parseHash(*i++);
|
|
|
|
Hash scHash = parseHash(*i++);
|
|
|
|
registerSuccessor(fsHash, scHash);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2003-07-10 17:11:48 +02: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(); )
|
|
|
|
{
|
|
|
|
Hash srcHash = parseHash(*i++);
|
|
|
|
Hash subHash = parseHash(*i++);
|
|
|
|
registerSubstitute(srcHash, subHash);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2003-06-18 16:34:43 +02:00
|
|
|
/* A sink that writes dump output to stdout. */
|
|
|
|
struct StdoutSink : DumpSink
|
|
|
|
{
|
|
|
|
virtual void operator ()
|
|
|
|
(const unsigned char * data, unsigned int len)
|
|
|
|
{
|
2003-06-23 16:08:34 +02:00
|
|
|
if (write(STDOUT_FILENO, (char *) data, len) != (ssize_t) len)
|
2003-06-23 15:27:59 +02:00
|
|
|
throw SysError("writing to stdout");
|
2003-06-18 16:34:43 +02:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
|
2003-07-08 12:00:46 +02:00
|
|
|
/* Dump a path as a Nix archive. The archive is written to standard
|
2003-06-23 16:08:34 +02:00
|
|
|
output. */
|
2003-06-18 16:34:43 +02:00
|
|
|
static void opDump(Strings opFlags, Strings opArgs)
|
|
|
|
{
|
2003-06-20 16:11:31 +02:00
|
|
|
getArgType(opFlags);
|
2003-06-18 16:34:43 +02:00
|
|
|
if (!opFlags.empty()) throw UsageError("unknown flag");
|
|
|
|
if (opArgs.size() != 1) throw UsageError("only one argument allowed");
|
|
|
|
|
|
|
|
StdoutSink sink;
|
2003-06-20 16:11:31 +02:00
|
|
|
string arg = *opArgs.begin();
|
|
|
|
string path;
|
|
|
|
|
2003-07-10 15:41:28 +02:00
|
|
|
if (argType == atpHash) path = expandHash(parseHash(arg));
|
|
|
|
else if (argType == atpPath) path = arg;
|
2003-06-20 16:11:31 +02:00
|
|
|
|
2003-06-23 16:08:34 +02:00
|
|
|
dumpPath(path, sink);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/* A source that read restore intput to stdin. */
|
|
|
|
struct StdinSource : RestoreSource
|
|
|
|
{
|
|
|
|
virtual void operator () (const unsigned char * data, unsigned int len)
|
|
|
|
{
|
|
|
|
ssize_t res = read(STDIN_FILENO, (char *) data, len);
|
|
|
|
if (res == -1)
|
|
|
|
throw SysError("reading from stdin");
|
|
|
|
if (res != (ssize_t) len)
|
|
|
|
throw Error("not enough data available on stdin");
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
/* 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 16:34:43 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2003-06-17 23:12:58 +02:00
|
|
|
/* Initialise the Nix databases. */
|
|
|
|
static void opInit(Strings opFlags, Strings opArgs)
|
2003-05-26 11:44:18 +02:00
|
|
|
{
|
2003-06-17 23:12:58 +02:00
|
|
|
if (!opFlags.empty()) throw UsageError("unknown flag");
|
|
|
|
if (!opArgs.empty())
|
|
|
|
throw UsageError("--init does not have arguments");
|
|
|
|
initDB();
|
2003-03-21 16:53:35 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2003-07-04 17:42:03 +02: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 18:49:56 +01:00
|
|
|
{
|
2003-06-20 16:11:31 +02:00
|
|
|
Strings opFlags, opArgs;
|
|
|
|
Operation op = 0;
|
|
|
|
|
|
|
|
for (Strings::iterator it = args.begin();
|
|
|
|
it != args.end(); it++)
|
|
|
|
{
|
|
|
|
string arg = *it;
|
2003-03-24 18:49:56 +01:00
|
|
|
|
2003-06-17 23:12:58 +02:00
|
|
|
Operation oldOp = op;
|
2003-03-24 18:49:56 +01:00
|
|
|
|
2003-07-07 11:29:40 +02:00
|
|
|
if (arg == "--install" || arg == "-i")
|
|
|
|
op = opInstall;
|
2003-06-17 23:12:58 +02:00
|
|
|
else if (arg == "--delete" || arg == "-d")
|
|
|
|
op = opDelete;
|
2003-07-08 15:22:08 +02:00
|
|
|
else if (arg == "--add" || arg == "-A")
|
2003-06-17 23:12:58 +02:00
|
|
|
op = opAdd;
|
2003-07-08 15:22:08 +02:00
|
|
|
else if (arg == "--query" || arg == "-q")
|
|
|
|
op = opQuery;
|
2003-07-10 20:48:11 +02:00
|
|
|
else if (arg == "--successor")
|
|
|
|
op = opSuccessor;
|
2003-07-10 17:11:48 +02:00
|
|
|
else if (arg == "--substitute")
|
|
|
|
op = opSubstitute;
|
2003-06-18 16:34:43 +02:00
|
|
|
else if (arg == "--dump")
|
|
|
|
op = opDump;
|
2003-06-23 16:08:34 +02:00
|
|
|
else if (arg == "--restore")
|
|
|
|
op = opRestore;
|
2003-06-17 23:12:58 +02:00
|
|
|
else if (arg == "--init")
|
|
|
|
op = opInit;
|
|
|
|
else if (arg[0] == '-')
|
|
|
|
opFlags.push_back(arg);
|
|
|
|
else
|
|
|
|
opArgs.push_back(arg);
|
2003-05-26 00:42:19 +02:00
|
|
|
|
2003-06-17 23:12:58 +02:00
|
|
|
if (oldOp && oldOp != op)
|
|
|
|
throw UsageError("only one operation may be specified");
|
2003-05-26 00:42:19 +02:00
|
|
|
}
|
|
|
|
|
2003-06-17 23:12:58 +02:00
|
|
|
if (!op) throw UsageError("no operation specified");
|
2003-03-20 17:53:00 +01:00
|
|
|
|
2003-06-17 23:12:58 +02:00
|
|
|
op(opFlags, opArgs);
|
2003-03-20 17:53:00 +01:00
|
|
|
}
|
2003-03-14 17:43:14 +01:00
|
|
|
|
2003-03-24 18:49:56 +01:00
|
|
|
|
2003-07-04 17:42:03 +02:00
|
|
|
string programId = "nix";
|