2003-03-13 16:28:32 +00:00
|
|
|
#include <iostream>
|
2003-03-14 16:43:14 +00:00
|
|
|
|
2003-06-17 21:12:58 +00:00
|
|
|
#include "config.h"
|
2003-03-14 16:43:14 +00:00
|
|
|
|
2003-06-17 21:12:58 +00:00
|
|
|
#include "globals.hh"
|
|
|
|
#include "values.hh"
|
|
|
|
#include "eval.hh"
|
2003-06-20 10:40:25 +00:00
|
|
|
#include "archive.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-06-20 14:11:31 +00:00
|
|
|
typedef enum { atpHash, atpName, atpPath, atpUnknown } ArgType;
|
2003-03-14 16:43:14 +00:00
|
|
|
|
2003-06-20 14:11:31 +00:00
|
|
|
static ArgType argType = atpUnknown;
|
|
|
|
|
|
|
|
|
|
|
|
/* Nix syntax:
|
|
|
|
|
|
|
|
nix [OPTIONS...] [ARGUMENTS...]
|
|
|
|
|
|
|
|
Operations:
|
|
|
|
|
|
|
|
--evaluate / -e: evaluate values
|
|
|
|
--delete / -d: delete values
|
|
|
|
--query / -q: query stored values
|
|
|
|
--add: add values
|
2003-06-23 14:08:34 +00:00
|
|
|
|
|
|
|
--dump: dump a value as a Nix archive
|
|
|
|
--restore: restore a value from a Nix archive
|
|
|
|
|
2003-06-20 14:11:31 +00:00
|
|
|
--init: initialise the Nix database
|
2003-06-23 14:08:34 +00:00
|
|
|
--verify: verify Nix structures
|
|
|
|
|
2003-06-20 14:11:31 +00:00
|
|
|
--version: output version information
|
|
|
|
--help: display help
|
|
|
|
|
|
|
|
Source selection for operations that work on values:
|
|
|
|
|
|
|
|
--file / -f: by file name
|
|
|
|
--hash / -h: by hash
|
|
|
|
--name / -n: by symbolic name
|
|
|
|
|
|
|
|
Query suboptions:
|
|
|
|
|
|
|
|
Selection:
|
|
|
|
|
|
|
|
--all / -a: query all stored values, otherwise given values
|
|
|
|
|
|
|
|
Information:
|
|
|
|
|
|
|
|
--info / -i: general value information
|
|
|
|
|
|
|
|
Options:
|
|
|
|
|
|
|
|
--verbose / -v: verbose operation
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
|
|
/* Parse the `-f' / `-h' / `-n' flags, i.e., the type of value
|
|
|
|
arguments. These flags are deleted from the referenced vector. */
|
|
|
|
void getArgType(Strings & flags)
|
|
|
|
{
|
|
|
|
for (Strings::iterator it = flags.begin();
|
|
|
|
it != flags.end(); )
|
|
|
|
{
|
|
|
|
string arg = *it;
|
|
|
|
ArgType tp;
|
|
|
|
if (arg == "--hash" || arg == "-h")
|
|
|
|
tp = atpHash;
|
|
|
|
else if (arg == "--name" || arg == "-n")
|
|
|
|
tp = atpName;
|
|
|
|
else if (arg == "--file" || arg == "-f")
|
|
|
|
tp = atpPath;
|
|
|
|
else {
|
|
|
|
it++;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if (argType != atpUnknown)
|
|
|
|
throw UsageError("only one argument type specified may be specified");
|
|
|
|
argType = tp;
|
|
|
|
it = flags.erase(it);
|
2003-03-20 16:53:00 +00:00
|
|
|
}
|
2003-06-20 14:11:31 +00:00
|
|
|
if (argType == atpUnknown)
|
|
|
|
throw UsageError("argument type not specified");
|
2003-03-14 16:43:14 +00:00
|
|
|
}
|
|
|
|
|
2003-03-13 16:28:32 +00:00
|
|
|
|
2003-06-17 21:12:58 +00:00
|
|
|
/* Evaluate values. */
|
|
|
|
static void opEvaluate(Strings opFlags, Strings opArgs)
|
2003-03-14 16:43:14 +00:00
|
|
|
{
|
2003-06-20 14:11:31 +00:00
|
|
|
getArgType(opFlags);
|
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-03-28 10:33:34 +00:00
|
|
|
{
|
2003-06-20 14:11:31 +00:00
|
|
|
Hash hash;
|
|
|
|
if (argType == atpHash)
|
|
|
|
hash = parseHash(*it);
|
|
|
|
else if (argType == atpName)
|
|
|
|
throw Error("not implemented");
|
|
|
|
else if (argType == atpPath)
|
|
|
|
hash = addValue(*it);
|
2003-06-17 21:12:58 +00:00
|
|
|
Expr e = ATmake("Deref(Hash(<str>))", ((string) hash).c_str());
|
|
|
|
cerr << printExpr(evalValue(e)) << endl;
|
2003-04-02 15:34:05 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
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-20 14:11:31 +00:00
|
|
|
getArgType(opFlags);
|
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++)
|
|
|
|
{
|
|
|
|
Hash hash;
|
|
|
|
if (argType == atpHash)
|
|
|
|
hash = parseHash(*it);
|
|
|
|
else if (argType == atpName)
|
|
|
|
throw Error("not implemented");
|
|
|
|
else
|
|
|
|
throw Error("invalid argument type");
|
|
|
|
deleteValue(hash);
|
|
|
|
}
|
2003-04-02 15:34:05 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2003-06-17 21:12:58 +00:00
|
|
|
/* Add values to the Nix values directory and print the hashes of
|
|
|
|
those values. */
|
|
|
|
static void opAdd(Strings opFlags, Strings opArgs)
|
2003-04-02 15:34:05 +00:00
|
|
|
{
|
2003-06-20 14:11:31 +00:00
|
|
|
getArgType(opFlags);
|
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++)
|
|
|
|
cout << (string) addValue(*it) << endl;
|
2003-03-13 16:28:32 +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-06-23 14:08:34 +00:00
|
|
|
if (write(STDOUT_FILENO, (char *) data, len) != (ssize_t) len)
|
2003-06-23 13:27:59 +00:00
|
|
|
throw SysError("writing to stdout");
|
2003-06-18 14:34:43 +00:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
|
2003-06-23 14:08:34 +00:00
|
|
|
/* Dump a value as a Nix archive. The archive is written to standard
|
|
|
|
output. */
|
2003-06-18 14:34:43 +00:00
|
|
|
static void opDump(Strings opFlags, Strings opArgs)
|
|
|
|
{
|
2003-06-20 14:11:31 +00:00
|
|
|
getArgType(opFlags);
|
2003-06-18 14:34:43 +00:00
|
|
|
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();
|
|
|
|
string path;
|
|
|
|
|
|
|
|
if (argType == atpHash)
|
|
|
|
path = queryValuePath(parseHash(arg));
|
|
|
|
else if (argType == atpName)
|
|
|
|
throw Error("not implemented");
|
|
|
|
else if (argType == atpPath)
|
|
|
|
path = arg;
|
|
|
|
|
2003-06-23 14:08:34 +00: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 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-06-17 21:12:58 +00:00
|
|
|
/* Initialize, process arguments, and dispatch to the right
|
|
|
|
operation. */
|
2003-06-20 14:11:31 +00:00
|
|
|
void run(int argc, char * * argv)
|
2003-03-24 17:49:56 +00:00
|
|
|
{
|
2003-06-17 21:12:58 +00:00
|
|
|
/* Setup Nix paths. */
|
|
|
|
nixValues = NIX_VALUES_DIR;
|
|
|
|
nixLogDir = NIX_LOG_DIR;
|
|
|
|
nixDB = (string) NIX_STATE_DIR + "/nixstate.db";
|
2003-03-24 17:49:56 +00:00
|
|
|
|
2003-06-20 14:11:31 +00:00
|
|
|
/* Put the arguments in a vector. */
|
|
|
|
Strings args;
|
|
|
|
while (argc--) args.push_back(*argv++);
|
|
|
|
args.erase(args.begin());
|
|
|
|
|
|
|
|
/* Expand compound dash options (i.e., `-qlf' -> `-q -l -f'). */
|
|
|
|
for (Strings::iterator it = args.begin();
|
|
|
|
it != args.end(); )
|
|
|
|
{
|
|
|
|
string arg = *it;
|
|
|
|
if (arg.length() > 2 && arg[0] == '-' && arg[1] != '-') {
|
|
|
|
for (unsigned int i = 1; i < arg.length(); i++)
|
|
|
|
args.insert(it, (string) "-" + arg[i]);
|
|
|
|
it = args.erase(it);
|
|
|
|
} else it++;
|
|
|
|
}
|
|
|
|
|
|
|
|
Strings opFlags, opArgs;
|
|
|
|
Operation op = 0;
|
|
|
|
|
2003-06-17 21:12:58 +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. */
|
2003-03-24 17:49:56 +00:00
|
|
|
|
2003-06-20 14:11:31 +00:00
|
|
|
for (Strings::iterator it = args.begin();
|
|
|
|
it != args.end(); it++)
|
|
|
|
{
|
|
|
|
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-06-17 21:12:58 +00:00
|
|
|
if (arg == "--evaluate" || arg == "-e")
|
|
|
|
op = opEvaluate;
|
|
|
|
else if (arg == "--delete" || arg == "-d")
|
|
|
|
op = opDelete;
|
|
|
|
else if (arg == "--add")
|
|
|
|
op = opAdd;
|
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;
|
|
|
|
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-03-20 16:53:00 +00:00
|
|
|
int main(int argc, char * * argv)
|
|
|
|
{
|
2003-06-17 21:12:58 +00:00
|
|
|
/* ATerm setup. */
|
2003-04-01 14:00:47 +00:00
|
|
|
ATerm bottomOfStack;
|
|
|
|
ATinit(argc, argv, &bottomOfStack);
|
|
|
|
|
2003-06-17 21:12:58 +00:00
|
|
|
try {
|
2003-06-20 14:11:31 +00:00
|
|
|
run(argc, argv);
|
2003-03-20 16:53:00 +00:00
|
|
|
} catch (UsageError & e) {
|
|
|
|
cerr << "error: " << e.what() << endl
|
2003-06-17 21:12:58 +00:00
|
|
|
<< "Try `nix --help' for more information.\n";
|
2003-03-14 16:43:14 +00:00
|
|
|
return 1;
|
2003-03-20 16:53:00 +00:00
|
|
|
} catch (exception & e) {
|
|
|
|
cerr << "error: " << e.what() << endl;
|
2003-03-13 16:28:32 +00:00
|
|
|
return 1;
|
|
|
|
}
|
2003-03-14 16:43:14 +00:00
|
|
|
|
|
|
|
return 0;
|
2003-03-13 16:28:32 +00:00
|
|
|
}
|