aboutsummaryrefslogtreecommitdiff
path: root/src/nix-env/nix-env.cc
diff options
context:
space:
mode:
authorEelco Dolstra <e.dolstra@tudelft.nl>2007-09-17 16:08:24 +0000
committerEelco Dolstra <e.dolstra@tudelft.nl>2007-09-17 16:08:24 +0000
commit055608227f5a003825a905b0eb2aea39cba5ca16 (patch)
treefd3f4a9e9e6dd0df64e208b74f84ac7d79a67808 /src/nix-env/nix-env.cc
parent3339f854473863c8780e537626f457be0b2b33e2 (diff)
* nix-env: allow ~/.nix-defexpr to be a directory. If it is, then the
Nix expressions in that directory are combined into an attribute set {file1 = import file1; file2 = import file2; ...}, i.e. each Nix expression is an attribute with the file name as the attribute name. Also recurses into directories. * nix-env: removed the "--import" (-I) option which set the ~/.nix-defexpr symlink. * nix-channel: don't use "nix-env --import", instead symlink ~/.nix-defexpr/channels. So finally nix-channel --update doesn't override any default Nix expressions but combines with them. This means that you can have (say) a local Nixpkgs SVN tree and use it as a default for nix-env: $ ln -s .../path-to-nixpkgs-tree ~/.nix-defexpr/nixpkgs_svn and be subscribed to channels (including Nixpkgs) at the same time. (If there is any ambiguity, the -A flag can be used to disambiguate, e.g. "nix-env -i -A nixpkgs_svn.pan".)
Diffstat (limited to 'src/nix-env/nix-env.cc')
-rw-r--r--src/nix-env/nix-env.cc68
1 files changed, 48 insertions, 20 deletions
diff --git a/src/nix-env/nix-env.cc b/src/nix-env/nix-env.cc
index e0de2812c..89a92fdbb 100644
--- a/src/nix-env/nix-env.cc
+++ b/src/nix-env/nix-env.cc
@@ -22,6 +22,8 @@
#include <iostream>
#include <sstream>
+#include <sys/types.h>
+#include <sys/stat.h>
#include <unistd.h>
@@ -72,11 +74,53 @@ void printHelp()
}
+static bool isNixExpr(const Path & path)
+{
+ struct stat st;
+ if (stat(path.c_str(), &st) == -1)
+ throw SysError(format("getting information about `%1%'") % path);
+
+ return !S_ISDIR(st.st_mode) || pathExists(path + "/default.nix");
+}
+
+
+static void getAllExprs(EvalState & state,
+ const Path & path, ATermMap & attrs)
+{
+ Strings names = readDirectory(path);
+
+ for (Strings::iterator i = names.begin(); i != names.end(); ++i) {
+ Path path2 = path + "/" + *i;
+ if (isNixExpr(path2))
+ attrs.set(toATerm(*i), makeAttrRHS(
+ parseExprFromFile(state, absPath(path2)), makeNoPos()));
+ else
+ getAllExprs(state, path2, attrs);
+ }
+}
+
+
+static Expr loadSourceExpr(EvalState & state, const Path & path)
+{
+ if (isNixExpr(path)) return parseExprFromFile(state, absPath(path));
+
+ /* The path is a directory. Put the Nix expressions in the
+ directory in an attribute set, with the file name of each
+ expression as the attribute name. Recurse into subdirectories
+ (but keep the attribute set flat, not nested, to make it easier
+ for a user to have a ~/.nix-defexpr directory that includes
+ some system-wide directory). */
+ ATermMap attrs;
+ attrs.set(toATerm("_combineChannels"), makeAttrRHS(eTrue, makeNoPos()));
+ getAllExprs(state, path, attrs);
+ return makeAttrs(attrs);
+}
+
+
static void loadDerivations(EvalState & state, Path nixExprPath,
string systemFilter, const ATermMap & autoArgs, DrvInfos & elems)
{
- getDerivations(state,
- parseExprFromFile(state, absPath(nixExprPath)), "", autoArgs, elems);
+ getDerivations(state, loadSourceExpr(state, nixExprPath), "", autoArgs, elems);
/* Filter out all derivations not applicable to the current
system. */
@@ -365,9 +409,7 @@ static void queryInstSources(EvalState & state,
(import ./foo.nix)' = `(import ./foo.nix).bar'. */
case srcNixExprs: {
-
- Expr e1 = parseExprFromFile(state,
- absPath(instSource.nixExprPath));
+ Expr e1 = loadSourceExpr(state, instSource.nixExprPath);
for (Strings::const_iterator i = args.begin();
i != args.end(); ++i)
@@ -429,7 +471,7 @@ static void queryInstSources(EvalState & state,
i != args.end(); ++i)
getDerivations(state,
findAlongAttrPath(state, *i, instSource.autoArgs,
- parseExprFromFile(state, instSource.nixExprPath)),
+ loadSourceExpr(state, instSource.nixExprPath)),
"", instSource.autoArgs, elems);
break;
}
@@ -1218,18 +1260,6 @@ static void opDeleteGenerations(Globals & globals,
}
-static void opDefaultExpr(Globals & globals,
- Strings opFlags, Strings opArgs)
-{
- if (opFlags.size() > 0)
- throw UsageError(format("unknown flag `%1%'") % opFlags.front());
- if (opArgs.size() != 1)
- throw UsageError(format("exactly one argument expected"));
-
- switchLink(getDefNixExprPath(), absPath(opArgs.front()));
-}
-
-
static string needArg(Strings::iterator & i,
Strings & args, const string & arg)
{
@@ -1286,8 +1316,6 @@ void run(Strings args)
op = opSet;
else if (arg == "--query" || arg == "-q")
op = opQuery;
- else if (arg == "--import" || arg == "-I") /* !!! bad name */
- op = opDefaultExpr;
else if (arg == "--profile" || arg == "-p")
globals.profile = absPath(needArg(i, args, arg));
else if (arg == "--file" || arg == "-f")