2006-09-25 03:07:59 -07:00
|
|
|
# Utility functions.
|
|
|
|
|
2006-11-27 08:58:08 -08:00
|
|
|
let
|
|
|
|
|
2007-01-29 06:53:23 -08:00
|
|
|
inherit (builtins)
|
2008-01-15 19:29:56 -08:00
|
|
|
head tail isList stringLength substring lessThan sub
|
|
|
|
listToAttrs attrNames hasAttr;
|
2006-11-27 08:58:08 -08:00
|
|
|
|
|
|
|
in
|
|
|
|
|
2006-09-25 03:07:59 -07:00
|
|
|
rec {
|
2008-01-28 11:30:13 -08:00
|
|
|
listOfListsToAttrs = ll : builtins.listToAttrs (map (l : { name = (head l); value = (head (tail l)); }) ll);
|
2006-09-25 03:07:59 -07:00
|
|
|
|
2008-01-04 07:06:16 -08:00
|
|
|
|
|
|
|
# Identity function.
|
|
|
|
id = x: x;
|
|
|
|
|
|
|
|
|
2008-01-16 17:31:07 -08:00
|
|
|
# accumulates / merges all attr sets until null is fed.
|
|
|
|
# example: sumArgs id { a = 'a'; x = 'x'; } { y = 'y'; x = 'X'; } null
|
|
|
|
# result : { a = 'a'; x = 'X'; y = 'Y'; }
|
2007-10-27 10:54:20 -07:00
|
|
|
innerSumArgs = f : x : y : (if y == null then (f x)
|
|
|
|
else (innerSumArgs f (x // y)));
|
|
|
|
sumArgs = f : innerSumArgs f {};
|
|
|
|
|
2008-03-06 01:34:36 -08:00
|
|
|
# Advanced sumArgs version. Hm, twice as slow, I'm afraid.
|
|
|
|
# composedArgs id (x:x//{a="b";}) (x:x//{b=x.a + "c";}) null;
|
|
|
|
# {a="b" ; b="bc";};
|
|
|
|
innerComposedArgs = f : x : y : (if y==null then (f x)
|
|
|
|
else (if (builtins.isAttrs y) then
|
|
|
|
(innerComposedArgs f (x//y))
|
|
|
|
else (innerComposedArgs f (y x))));
|
|
|
|
composedArgs = f: innerComposedArgs f {};
|
|
|
|
|
2008-08-20 04:20:32 -07:00
|
|
|
defaultMergeArg = x : y: if builtins.isAttrs y then
|
|
|
|
y
|
2008-08-19 23:45:14 -07:00
|
|
|
else
|
2008-08-20 04:20:32 -07:00
|
|
|
(y x);
|
|
|
|
defaultMerge = x: y: x // (defaultMergeArg x y);
|
2008-08-14 15:12:50 -07:00
|
|
|
sumTwoArgs = f: x: y:
|
2008-08-19 23:45:14 -07:00
|
|
|
f (defaultMerge x y);
|
|
|
|
foldArgs = merger: f: init: x:
|
2008-11-04 13:40:51 -08:00
|
|
|
let arg=(merger init (defaultMergeArg init x)); in
|
|
|
|
# now add the function with composed args already applied to the final attrs
|
|
|
|
setAttrMerge "passthru" {} (f arg) ( x : x // { function = foldArgs merger f arg; } );
|
2008-12-02 04:26:53 -08:00
|
|
|
|
|
|
|
# returns f x // { passthru.fun = y : f (merge x y); } while preserving other passthru names.
|
|
|
|
# example: let ex = applyAndFun (x : removeAttrs x ["fixed"]) (mergeOrApply mergeAttr) {name = 6;};
|
|
|
|
# usage1 = ex.passthru.fun { name = 7; }; # result: { name = 7;}
|
|
|
|
# usage2 = ex.passthru.fun (a: a // {name = __add a.name 1; }); # result: { a = 7; }
|
|
|
|
# fix usage:
|
|
|
|
# usage3a = ex.passthru.fun (a: a // {name2 = a.fixed.toBePassed; }); # usage3a will fail because toBePassed is not yet given
|
|
|
|
# usage3b usage3a.passthru.fun { toBePassed = "foo";}; # result { name = 7; name2 = "foo"; toBePassed = "foo"; fixed = <this attrs>; }
|
|
|
|
applyAndFun = f : merge : x : assert (__isAttrs x || __isFunction x);
|
|
|
|
let takeFix = if (__isFunction x) then x else (attr: merge attr x); in
|
|
|
|
setAttrMerge "passthru" {} (fix (fixed : f (takeFix {inherit fixed;})))
|
|
|
|
( y : y //
|
|
|
|
{
|
|
|
|
fun = z : applyAndFun f merge (fixed: merge (takeFix fixed) z);
|
|
|
|
funMerge = z : applyAndFun f merge (fixed: let e = takeFix fixed; in e // merge e z);
|
|
|
|
} );
|
|
|
|
mergeOrApply = merge : x : y : if (__isFunction y) then y x else merge x y;
|
|
|
|
|
2008-11-04 13:40:51 -08:00
|
|
|
# rec { # an example of how composedArgsAndFun can be used
|
|
|
|
# a = composedArgsAndFun (x : x) { a = ["2"]; meta = { d = "bar";}; };
|
|
|
|
# # meta.d will be lost ! It's your task to preserve it (eg using a merge function)
|
|
|
|
# b = a.passthru.function { a = [ "3" ]; meta = { d2 = "bar2";}; };
|
|
|
|
# # instead of passing/ overriding values you can use a merge function:
|
|
|
|
# c = b.passthru.function ( x: { a = x.a ++ ["4"]; }); # consider using (maybeAttr "a" [] x)
|
|
|
|
# }
|
|
|
|
# result:
|
|
|
|
# {
|
|
|
|
# a = { a = ["2"]; meta = { d = "bar"; }; passthru = { function = .. }; };
|
|
|
|
# b = { a = ["3"]; meta = { d2 = "bar2"; }; passthru = { function = .. }; };
|
|
|
|
# c = { a = ["3" "4"]; meta = { d2 = "bar2"; }; passthru = { function = .. }; };
|
|
|
|
# # c2 is equal to c
|
|
|
|
# }
|
2008-08-20 04:20:32 -07:00
|
|
|
composedArgsAndFun = f: foldArgs defaultMerge f {};
|
2008-08-14 15:04:30 -07:00
|
|
|
|
2008-01-16 17:31:07 -08:00
|
|
|
# example a = pairMap (x : y : x + y) ["a" "b" "c" "d"];
|
|
|
|
# result: ["ab" "cd"]
|
2007-11-05 00:32:20 -08:00
|
|
|
innerPairMap = acc: f: l:
|
|
|
|
if l == [] then acc else
|
|
|
|
innerPairMap (acc ++ [(f (head l)(head (tail l)))])
|
|
|
|
f (tail (tail l));
|
|
|
|
pairMap = innerPairMap [];
|
|
|
|
|
2008-01-04 07:06:16 -08:00
|
|
|
|
2008-02-13 02:09:29 -08:00
|
|
|
|
2006-09-25 03:07:59 -07:00
|
|
|
# "Fold" a binary function `op' between successive elements of
|
|
|
|
# `list' with `nul' as the starting value, i.e., `fold op nul [x_1
|
|
|
|
# x_2 ... x_n] == op x_1 (op x_2 ... (op x_n nul))'. (This is
|
|
|
|
# Haskell's foldr).
|
|
|
|
fold = op: nul: list:
|
|
|
|
if list == []
|
|
|
|
then nul
|
2006-11-27 08:58:08 -08:00
|
|
|
else op (head list) (fold op nul (tail list));
|
2006-09-25 03:07:59 -07:00
|
|
|
|
|
|
|
|
2007-03-05 16:01:27 -08:00
|
|
|
# Concatenate a list of lists.
|
2008-12-02 04:26:20 -08:00
|
|
|
concatList = x : y : x ++ y;
|
|
|
|
concatLists = fold concatList [];
|
2007-03-05 16:01:27 -08:00
|
|
|
|
|
|
|
|
2006-09-25 03:07:59 -07:00
|
|
|
# Concatenate a list of strings.
|
|
|
|
concatStrings =
|
|
|
|
fold (x: y: x + y) "";
|
|
|
|
|
|
|
|
|
2008-02-05 03:41:09 -08:00
|
|
|
# Map and concatenate the result.
|
|
|
|
concatMap = f: list: concatLists (map f list);
|
|
|
|
|
2008-02-14 05:34:55 -08:00
|
|
|
concatMapStrings = f: list: concatStrings (map f list);
|
2008-02-14 05:16:22 -08:00
|
|
|
|
2008-02-05 03:41:09 -08:00
|
|
|
|
2006-10-12 03:53:16 -07:00
|
|
|
# Place an element between each element of a list, e.g.,
|
|
|
|
# `intersperse "," ["a" "b" "c"]' returns ["a" "," "b" "," "c"].
|
|
|
|
intersperse = separator: list:
|
2006-11-27 08:58:08 -08:00
|
|
|
if list == [] || tail list == []
|
2006-10-12 03:53:16 -07:00
|
|
|
then list
|
2006-11-27 08:58:08 -08:00
|
|
|
else [(head list) separator]
|
|
|
|
++ (intersperse separator (tail list));
|
2006-10-12 03:53:16 -07:00
|
|
|
|
2008-01-15 19:29:56 -08:00
|
|
|
toList = x : if (__isList x) then x else [x];
|
2006-10-12 03:53:16 -07:00
|
|
|
|
2007-05-27 07:34:01 -07:00
|
|
|
concatStringsSep = separator: list:
|
|
|
|
concatStrings (intersperse separator list);
|
|
|
|
|
2008-06-04 02:56:11 -07:00
|
|
|
makeLibraryPath = paths: concatStringsSep ":" (map (path: path + "/lib") paths);
|
|
|
|
|
2007-05-27 07:34:01 -07:00
|
|
|
|
2006-09-25 03:07:59 -07:00
|
|
|
# Flatten the argument into a single list; that is, nested lists are
|
|
|
|
# spliced into the top-level lists. E.g., `flatten [1 [2 [3] 4] 5]
|
|
|
|
# == [1 2 3 4 5]' and `flatten 1 == [1]'.
|
|
|
|
flatten = x:
|
2006-11-27 08:58:08 -08:00
|
|
|
if isList x
|
2006-09-25 03:07:59 -07:00
|
|
|
then fold (x: y: (flatten x) ++ y) [] x
|
|
|
|
else [x];
|
|
|
|
|
|
|
|
|
|
|
|
# Return an attribute from nested attribute sets. For instance ["x"
|
|
|
|
# "y"] applied to some set e returns e.x.y, if it exists. The
|
|
|
|
# default value is returned otherwise.
|
2008-01-15 19:29:56 -08:00
|
|
|
# comment: there is also builtins.getAttr ? (is there a better name for this function?)
|
2006-09-25 03:07:59 -07:00
|
|
|
getAttr = attrPath: default: e:
|
2008-01-04 07:06:16 -08:00
|
|
|
let attr = head attrPath;
|
|
|
|
in
|
|
|
|
if attrPath == [] then e
|
|
|
|
else if builtins ? hasAttr && builtins.hasAttr attr e
|
|
|
|
then getAttr (tail attrPath) default (builtins.getAttr attr e)
|
|
|
|
else default;
|
2006-11-27 08:58:08 -08:00
|
|
|
|
2008-02-13 02:09:29 -08:00
|
|
|
# shortcut for getAttr ["name"] default attrs
|
|
|
|
maybeAttr = name: default: attrs:
|
|
|
|
if (__hasAttr name attrs) then (__getAttr name attrs) else default;
|
|
|
|
|
2006-11-27 08:58:08 -08:00
|
|
|
|
|
|
|
# Filter a list using a predicate; that is, return a list containing
|
|
|
|
# every element from `list' for which `pred' returns true.
|
|
|
|
filter = pred: list:
|
|
|
|
fold (x: y: if pred x then [x] ++ y else y) [] list;
|
|
|
|
|
|
|
|
|
2007-01-08 14:49:26 -08:00
|
|
|
# Return true if `list' has an element `x':
|
|
|
|
elem = x: list: fold (a: bs: x == a || bs) false list;
|
|
|
|
|
|
|
|
|
2006-12-20 16:09:40 -08:00
|
|
|
# Find the sole element in the list matching the specified
|
2007-10-03 06:26:24 -07:00
|
|
|
# predicate, returns `default' if no such element exists, or
|
|
|
|
# `multiple' if there are multiple matching elements.
|
|
|
|
findSingle = pred: default: multiple: list:
|
2006-12-20 16:09:40 -08:00
|
|
|
let found = filter pred list;
|
|
|
|
in if found == [] then default
|
2007-10-03 06:26:24 -07:00
|
|
|
else if tail found != [] then multiple
|
2006-12-20 16:09:40 -08:00
|
|
|
else head found;
|
|
|
|
|
|
|
|
|
2008-01-04 07:06:16 -08:00
|
|
|
# Return true iff function `pred' returns true for at least element
|
|
|
|
# of `list'.
|
|
|
|
any = pred: list:
|
|
|
|
if list == [] then false
|
|
|
|
else if pred (head list) then true
|
|
|
|
else any pred (tail list);
|
|
|
|
|
|
|
|
|
|
|
|
# Return true iff function `pred' returns true for all elements of
|
|
|
|
# `list'.
|
|
|
|
all = pred: list:
|
|
|
|
if list == [] then true
|
|
|
|
else if pred (head list) then all pred (tail list)
|
|
|
|
else false;
|
|
|
|
|
2008-01-15 19:29:56 -08:00
|
|
|
# much shorter implementations using map and fold (are lazy as well)
|
|
|
|
# which ones are better?
|
|
|
|
# true if all/ at least one element(s) satisfy f
|
|
|
|
# all = f : l : fold logicalAND true (map f l);
|
|
|
|
# any = f : l : fold logicalOR false (map f l);
|
|
|
|
|
2008-01-04 07:06:16 -08:00
|
|
|
|
2006-11-27 08:58:08 -08:00
|
|
|
# Return true if each element of a list is equal, false otherwise.
|
|
|
|
eqLists = xs: ys:
|
|
|
|
if xs == [] && ys == [] then true
|
|
|
|
else if xs == [] || ys == [] then false
|
|
|
|
else head xs == head ys && eqLists (tail xs) (tail ys);
|
|
|
|
|
2008-01-04 07:06:16 -08:00
|
|
|
|
2007-10-29 03:52:04 -07:00
|
|
|
# Workaround, but works in stable Nix now.
|
|
|
|
eqStrings = a: b: (a+(substring 0 0 b)) == ((substring 0 0 a)+b);
|
2007-01-29 06:53:23 -08:00
|
|
|
|
2008-01-04 07:06:16 -08:00
|
|
|
|
2007-01-29 06:53:23 -08:00
|
|
|
# Determine whether a filename ends in the given suffix.
|
|
|
|
hasSuffix = ext: fileName:
|
|
|
|
let lenFileName = stringLength fileName;
|
|
|
|
lenExt = stringLength ext;
|
|
|
|
in !(lessThan lenFileName lenExt) &&
|
|
|
|
substring (sub lenFileName lenExt) lenFileName fileName == ext;
|
|
|
|
|
2007-10-29 03:52:04 -07:00
|
|
|
hasSuffixHack = a: b: hasSuffix (a+(substring 0 0 b)) ((substring 0 0 a)+b);
|
2008-02-12 02:08:57 -08:00
|
|
|
|
|
|
|
|
2007-01-29 06:53:23 -08:00
|
|
|
# Bring in a path as a source, filtering out all Subversion and CVS
|
|
|
|
# directories, as well as backup files (*~).
|
2007-01-15 01:20:18 -08:00
|
|
|
cleanSource =
|
2007-01-29 06:53:23 -08:00
|
|
|
let filter = name: type: let baseName = baseNameOf (toString name); in ! (
|
|
|
|
# Filter out Subversion and CVS directories.
|
2008-02-12 02:08:57 -08:00
|
|
|
(type == "directory" && (baseName == ".svn" || baseName == "CVS")) ||
|
2007-01-29 06:53:23 -08:00
|
|
|
# Filter out backup files.
|
2008-02-12 02:08:57 -08:00
|
|
|
(hasSuffix "~" baseName)
|
2007-01-29 06:53:23 -08:00
|
|
|
);
|
2007-01-15 01:20:18 -08:00
|
|
|
in src: builtins.filterSource filter src;
|
|
|
|
|
2007-05-20 13:24:43 -07:00
|
|
|
|
2008-01-04 07:06:16 -08:00
|
|
|
# Get all files ending with the specified suffices from the given
|
|
|
|
# directory. E.g. `sourceFilesBySuffices ./dir [".xml" ".c"]'.
|
|
|
|
sourceFilesBySuffices = path: exts:
|
|
|
|
let filter = name: type:
|
|
|
|
let base = baseNameOf (toString name);
|
|
|
|
in type != "directory" && any (ext: hasSuffix ext base) exts;
|
|
|
|
in builtins.filterSource filter path;
|
|
|
|
|
|
|
|
|
2007-05-20 13:24:43 -07:00
|
|
|
# Return a singleton list or an empty list, depending on a boolean
|
|
|
|
# value. Useful when building lists with optional elements
|
|
|
|
# (e.g. `++ optional (system == "i686-linux") flashplayer').
|
|
|
|
optional = cond: elem: if cond then [elem] else [];
|
|
|
|
|
2007-06-09 12:45:55 -07:00
|
|
|
|
2008-02-08 03:48:06 -08:00
|
|
|
# Return a list or an empty list, dependening on a boolean value.
|
|
|
|
optionals = cond: elems: if cond then elems else [];
|
2008-02-10 08:44:43 -08:00
|
|
|
|
2008-04-27 16:17:24 -07:00
|
|
|
optionalString = cond: string: if cond then string else "";
|
2007-06-09 12:45:55 -07:00
|
|
|
|
2008-11-22 16:19:06 -08:00
|
|
|
# Return the second argument if the first one is true or the empty version
|
|
|
|
# of the second argument.
|
|
|
|
ifEnable = cond: val:
|
|
|
|
if cond then val
|
|
|
|
else if builtins.isList val then []
|
|
|
|
else if builtins.isAttrs val then {}
|
|
|
|
# else if builtins.isString val then ""
|
2008-11-22 17:23:32 -08:00
|
|
|
else if (val == true || val == false) then false
|
2008-11-22 16:19:06 -08:00
|
|
|
else null;
|
|
|
|
|
2007-06-09 12:45:55 -07:00
|
|
|
# Return a list of integers from `first' up to and including `last'.
|
|
|
|
range = first: last:
|
|
|
|
if builtins.lessThan last first
|
|
|
|
then []
|
|
|
|
else [first] ++ range (builtins.add first 1) last;
|
|
|
|
|
2007-09-11 13:05:54 -07:00
|
|
|
|
2007-08-11 03:34:07 -07:00
|
|
|
# Return true only if there is an attribute and it is true.
|
|
|
|
checkFlag = attrSet: name:
|
|
|
|
if (name == "true") then true else
|
2007-08-11 04:14:36 -07:00
|
|
|
if (name == "false") then false else
|
2007-08-15 13:54:11 -07:00
|
|
|
if (isInList (getAttr ["flags"] [] attrSet) name) then true else
|
2007-08-11 03:34:07 -07:00
|
|
|
getAttr [name] false attrSet ;
|
|
|
|
|
2007-09-11 13:05:54 -07:00
|
|
|
|
2007-08-11 03:34:07 -07:00
|
|
|
logicalOR = x: y: x || y;
|
|
|
|
logicalAND = x: y: x && y;
|
|
|
|
|
2007-09-11 13:05:54 -07:00
|
|
|
|
2007-08-11 03:34:07 -07:00
|
|
|
# Input : attrSet, [ [name default] ... ], name
|
|
|
|
# Output : its value or default.
|
|
|
|
getValue = attrSet: argList: name:
|
2007-08-15 13:54:11 -07:00
|
|
|
( getAttr [name] (if checkFlag attrSet name then true else
|
|
|
|
if argList == [] then null else
|
2007-08-11 03:34:07 -07:00
|
|
|
let x = builtins.head argList; in
|
|
|
|
if (head x) == name then
|
|
|
|
(head (tail x))
|
|
|
|
else (getValue attrSet
|
|
|
|
(tail argList) name)) attrSet );
|
|
|
|
|
2007-09-11 13:05:54 -07:00
|
|
|
|
2007-08-11 03:34:07 -07:00
|
|
|
# Input : attrSet, [[name default] ...], [ [flagname reqs..] ... ]
|
|
|
|
# Output : are reqs satisfied? It's asserted.
|
|
|
|
checkReqs = attrSet : argList : condList :
|
|
|
|
(
|
|
|
|
fold logicalAND true
|
|
|
|
(map (x: let name = (head x) ; in
|
|
|
|
|
|
|
|
((checkFlag attrSet name) ->
|
|
|
|
(fold logicalAND true
|
|
|
|
(map (y: let val=(getValue attrSet argList y); in
|
|
|
|
(val!=null) && (val!=false))
|
|
|
|
(tail x))))) condList)) ;
|
|
|
|
|
|
|
|
|
|
|
|
isInList = list: x:
|
|
|
|
if (list == []) then false else
|
|
|
|
if (x == (head list)) then true else
|
|
|
|
isInList (tail list) x;
|
2007-09-11 13:05:54 -07:00
|
|
|
|
|
|
|
|
2007-08-11 04:14:36 -07:00
|
|
|
uniqList = {inputList, outputList ? []}:
|
|
|
|
if (inputList == []) then outputList else
|
|
|
|
let x=head inputList;
|
|
|
|
newOutputList = outputList ++
|
|
|
|
(if (isInList outputList x) then [] else [x]);
|
|
|
|
in uniqList {outputList=newOutputList;
|
|
|
|
inputList = (tail inputList);};
|
|
|
|
|
2008-01-11 08:58:55 -08:00
|
|
|
uniqListExt = {inputList, outputList ? [],
|
|
|
|
getter ? (x : x), compare ? (x: y: x==y)}:
|
|
|
|
if (inputList == []) then outputList else
|
|
|
|
let x=head inputList;
|
|
|
|
isX = y: (compare (getter y) (getter x));
|
|
|
|
newOutputList = outputList ++
|
|
|
|
(if any isX outputList then [] else [x]);
|
|
|
|
in uniqListExt {outputList=newOutputList;
|
|
|
|
inputList = (tail inputList);
|
|
|
|
inherit getter compare;
|
|
|
|
};
|
|
|
|
|
|
|
|
|
2007-09-11 13:05:54 -07:00
|
|
|
|
2007-08-15 14:17:11 -07:00
|
|
|
condConcat = name: list: checker:
|
|
|
|
if list == [] then name else
|
|
|
|
if checker (head list) then
|
|
|
|
condConcat
|
|
|
|
(name + (head (tail list)))
|
|
|
|
(tail (tail list))
|
|
|
|
checker
|
|
|
|
else condConcat
|
|
|
|
name (tail (tail list)) checker;
|
2008-08-05 10:16:35 -07:00
|
|
|
|
|
|
|
# Merge sets of attributes and use the function f to merge
|
|
|
|
# attributes values.
|
|
|
|
zip = f: sets:
|
|
|
|
builtins.listToAttrs (map (name: {
|
|
|
|
inherit name;
|
|
|
|
value =
|
|
|
|
f name
|
|
|
|
(map (__getAttr name)
|
|
|
|
(filter (__hasAttr name) sets));
|
|
|
|
}) (concatMap builtins.attrNames sets));
|
|
|
|
|
|
|
|
# divide a list in two depending on the evaluation of a predicate.
|
|
|
|
partition = pred:
|
|
|
|
fold (h: t:
|
|
|
|
if pred h
|
|
|
|
then { right = [h] ++ t.right; wrong = t.wrong; }
|
|
|
|
else { right = t.right; wrong = [h] ++ t.wrong; }
|
|
|
|
) { right = []; wrong = []; };
|
|
|
|
|
|
|
|
# Take a function and evaluate it with its own returned value.
|
2008-08-27 06:58:36 -07:00
|
|
|
fix = f:
|
2008-08-05 10:16:35 -07:00
|
|
|
(rec { result = f result; }).result;
|
|
|
|
|
|
|
|
# flatten a list of sets returned by 'f'.
|
|
|
|
# f : function to evaluate each set.
|
|
|
|
# attr : name of the attribute which contains more values.
|
|
|
|
# default: result if 'x' is empty.
|
|
|
|
# x : list of values that have to be processed.
|
|
|
|
uniqFlattenAttr = f: attr: default: x:
|
|
|
|
if x == []
|
|
|
|
then default
|
2008-11-22 16:19:18 -08:00
|
|
|
else
|
|
|
|
let h = f (head x); t = tail x;
|
|
|
|
v = removeAttrs h [attr]; n = getAttr [attr] [] h;
|
|
|
|
in
|
|
|
|
if elem v default
|
2008-08-05 10:16:35 -07:00
|
|
|
then uniqFlattenAttr f attr default t
|
2008-11-22 16:19:18 -08:00
|
|
|
else uniqFlattenAttr f attr (default ++ [v]) (toList n ++ t)
|
2008-08-05 10:16:35 -07:00
|
|
|
;
|
|
|
|
|
2008-11-22 16:19:18 -08:00
|
|
|
/* If. ThenElse. Always. */
|
|
|
|
|
|
|
|
# create "if" statement that can be dealyed on sets until a "then-else" or
|
|
|
|
# "always" set is reached. When an always set is reached by
|
|
|
|
|
|
|
|
isIf = attrs: (typeOf attrs) == "if";
|
|
|
|
mkIf = condition: thenelse:
|
|
|
|
if isIf thenelse then
|
|
|
|
mkIf (condition && thenelse.condition) thenelse.thenelse
|
|
|
|
else {
|
|
|
|
_type = "if";
|
|
|
|
inherit condition thenelse;
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
isThenElse = attrs: (typeOf attrs) == "then-else";
|
|
|
|
mkThenElse = attrs:
|
|
|
|
assert attrs ? thenPart && attrs ? elsePart;
|
|
|
|
attrs // { _type = "then-else"; };
|
|
|
|
|
|
|
|
|
|
|
|
isAlways = attrs: (typeOf attrs) == "always";
|
|
|
|
mkAlways = value: { inherit value; _type = "always"; };
|
|
|
|
|
|
|
|
pushIf = f: attrs:
|
|
|
|
if isIf attrs then pushIf f (
|
|
|
|
let val = attrs.thenelse; in
|
|
|
|
# evaluate the condition.
|
|
|
|
if isThenElse val then
|
|
|
|
if attrs.condition then
|
|
|
|
val.thenPart
|
|
|
|
else
|
|
|
|
val.elsePart
|
|
|
|
# ignore the condition.
|
|
|
|
else if isAlways val then
|
|
|
|
val.value
|
|
|
|
# otherwise
|
|
|
|
else
|
|
|
|
f attrs.condition val)
|
|
|
|
else
|
|
|
|
attrs;
|
|
|
|
|
|
|
|
# take care otherwise you will have to handle this by hand.
|
|
|
|
rmIf = pushIf (condition: val: val);
|
|
|
|
|
|
|
|
evalIf = pushIf (condition: val:
|
|
|
|
# guess: empty else part.
|
|
|
|
ifEnable condition val
|
|
|
|
);
|
|
|
|
|
|
|
|
delayIf = pushIf (condition: val:
|
|
|
|
# rewrite the condition on sub-attributes.
|
|
|
|
mapAttrs (name: mkIf condition) val
|
|
|
|
);
|
|
|
|
|
2007-11-09 10:05:32 -08:00
|
|
|
/* Options. */
|
2008-08-05 10:16:35 -07:00
|
|
|
|
2007-11-09 10:05:32 -08:00
|
|
|
mkOption = attrs: attrs // {_type = "option";};
|
|
|
|
|
2008-10-26 10:44:23 -07:00
|
|
|
typeOf = x: if (__isAttrs x && x ? _type) then x._type else "";
|
2007-11-09 10:05:32 -08:00
|
|
|
|
2008-10-26 10:44:23 -07:00
|
|
|
isOption = attrs: (typeOf attrs) == "option";
|
2008-08-05 10:16:35 -07:00
|
|
|
|
2007-11-09 10:07:14 -08:00
|
|
|
addDefaultOptionValues = defs: opts: opts //
|
2007-11-09 10:05:32 -08:00
|
|
|
builtins.listToAttrs (map (defName:
|
|
|
|
{ name = defName;
|
|
|
|
value =
|
|
|
|
let
|
|
|
|
defValue = builtins.getAttr defName defs;
|
|
|
|
optValue = builtins.getAttr defName opts;
|
|
|
|
in
|
|
|
|
if typeOf defValue == "option"
|
|
|
|
then
|
|
|
|
# `defValue' is an option.
|
|
|
|
if builtins.hasAttr defName opts
|
|
|
|
then builtins.getAttr defName opts
|
|
|
|
else defValue.default
|
|
|
|
else
|
|
|
|
# `defValue' is an attribute set containing options.
|
|
|
|
# So recurse.
|
|
|
|
if builtins.hasAttr defName opts && builtins.isAttrs optValue
|
2007-11-09 10:07:14 -08:00
|
|
|
then addDefaultOptionValues defValue optValue
|
|
|
|
else addDefaultOptionValues defValue {};
|
2007-11-09 10:05:32 -08:00
|
|
|
}
|
|
|
|
) (builtins.attrNames defs));
|
2008-08-05 10:16:35 -07:00
|
|
|
|
|
|
|
mergeDefaultOption = name: list:
|
|
|
|
if list != [] && tail list == [] then head list
|
|
|
|
else if all __isFunction list then x: mergeDefaultOption (map (f: f x) list)
|
|
|
|
else if all __isList list then concatLists list
|
|
|
|
else if all __isAttrs list then mergeAttrs list
|
2008-11-22 16:19:12 -08:00
|
|
|
else if all (x: true == x || false == x) list then fold logicalOR false list
|
2008-08-06 12:24:36 -07:00
|
|
|
else abort "${name}: Cannot merge values.";
|
2008-08-05 10:16:35 -07:00
|
|
|
|
2008-11-16 11:23:00 -08:00
|
|
|
mergeTypedOption = typeName: predicate: merge: name: list:
|
|
|
|
if all predicate list then merge list
|
|
|
|
else abort "${name}: Expect a ${typeName}.";
|
|
|
|
|
|
|
|
mergeEnableOption = mergeTypedOption "boolean"
|
|
|
|
(x: true == x || false == x) (fold logicalOR false);
|
|
|
|
|
|
|
|
mergeListOption = mergeTypedOption "list"
|
|
|
|
__isList concatLists;
|
|
|
|
|
|
|
|
mergeStringOption = mergeTypedOption "string"
|
|
|
|
(x: if builtins ? isString then builtins.isString x else x + "")
|
|
|
|
concatStrings;
|
2008-08-05 10:16:35 -07:00
|
|
|
|
|
|
|
# Merge sets of options and bindings.
|
|
|
|
# noOption: function to call if no option is declared.
|
|
|
|
mergeOptionSets = noOption: path: opts:
|
|
|
|
if all __isAttrs opts then
|
|
|
|
zip (attr: opts:
|
|
|
|
let
|
|
|
|
name = if path == "" then attr else path + "." + attr;
|
|
|
|
test = partition isOption opts;
|
2008-11-16 11:22:47 -08:00
|
|
|
opt = ({ merge = mergeDefaultOption; apply = id; } // head test.right);
|
2008-08-05 10:16:35 -07:00
|
|
|
in
|
2008-11-22 16:19:18 -08:00
|
|
|
if test.right == [] then mergeOptionSets noOption name (map delayIf test.wrong)
|
2008-08-05 10:16:35 -07:00
|
|
|
else if tail test.right != [] then throw "Multiple options for '${name}'."
|
2008-11-16 11:22:47 -08:00
|
|
|
else if test.wrong == [] then opt.apply opt.default
|
2008-11-22 16:19:18 -08:00
|
|
|
else opt.apply (opt.merge name (map evalIf test.wrong))
|
2008-08-05 10:16:35 -07:00
|
|
|
) opts
|
|
|
|
else noOption path opts;
|
|
|
|
|
|
|
|
# Keep all option declarations and add an attribute "name" inside
|
|
|
|
# each option which contains the path that has to be followed to
|
|
|
|
# access it.
|
|
|
|
filterOptionSets = path: opts:
|
|
|
|
if all __isAttrs opts then
|
|
|
|
zip (attr: opts:
|
|
|
|
let
|
|
|
|
name = if path == "" then attr else path + "." + attr;
|
|
|
|
test = partition isOption opts;
|
|
|
|
in
|
2008-11-22 16:19:18 -08:00
|
|
|
if test.right == [] then filterOptionSets name (map delayIf test.wrong)
|
2008-08-06 11:34:11 -07:00
|
|
|
else if tail test.right != [] then throw "Multiple options for '${name}'."
|
|
|
|
else { inherit name; } // (head test.right)
|
2008-08-05 10:16:35 -07:00
|
|
|
) opts
|
|
|
|
else {};
|
|
|
|
|
|
|
|
# Evaluate a list of option sets that would be merged with the
|
|
|
|
# function "merge" which expects two arguments. The attribute named
|
|
|
|
# "require" is used to imports option declarations and bindings.
|
2008-08-27 06:58:36 -07:00
|
|
|
fixOptionSetsFun = merge: pkgs: opts:
|
2008-11-22 16:19:18 -08:00
|
|
|
let
|
|
|
|
# ignore all conditions that are on require attributes.
|
|
|
|
rmRequireIf = conf:
|
|
|
|
let conf2 = delayIf conf; in
|
|
|
|
if conf2 ? require then
|
|
|
|
conf2 // { require = rmIf conf2.require; }
|
|
|
|
else
|
|
|
|
conf2;
|
|
|
|
|
|
|
|
# call configuration "files" with one of the existing convention.
|
|
|
|
optionSet = config: configFun:
|
|
|
|
if __isFunction configFun then
|
|
|
|
let result = configFun { inherit pkgs config; }; in
|
|
|
|
# {pkgs, config, ...}: {..}
|
|
|
|
if builtins.isAttrs result then result
|
|
|
|
# pkgs: config: {..}
|
|
|
|
else configFun pkgs config
|
|
|
|
# {..}
|
|
|
|
else configFun;
|
|
|
|
|
|
|
|
processConfig = config: configFun:
|
|
|
|
rmRequireIf (optionSet config configFun);
|
2008-08-05 10:16:35 -07:00
|
|
|
in
|
2008-11-22 16:19:18 -08:00
|
|
|
config: merge "" (
|
2008-11-22 17:23:32 -08:00
|
|
|
uniqFlattenAttr (processConfig config) "require" [] (toList opts)
|
2008-11-22 16:19:18 -08:00
|
|
|
);
|
2008-08-06 11:34:11 -07:00
|
|
|
|
2008-08-27 06:58:36 -07:00
|
|
|
fixOptionSets = merge: pkgs: opts:
|
|
|
|
fix (fixOptionSetsFun merge pkgs opts);
|
|
|
|
|
2007-11-11 01:27:52 -08:00
|
|
|
optionAttrSetToDocList = (l: attrs:
|
|
|
|
(if (getAttr ["_type"] "" attrs) == "option" then
|
|
|
|
[({
|
2008-08-27 14:19:03 -07:00
|
|
|
#inherit (attrs) description;
|
|
|
|
description = if attrs ? description then attrs.description else
|
2008-08-27 14:23:49 -07:00
|
|
|
throw ("No description ${toString l} : ${whatis attrs}");
|
2007-11-11 01:27:52 -08:00
|
|
|
}
|
|
|
|
//(if attrs ? example then {inherit(attrs) example;} else {} )
|
|
|
|
//(if attrs ? default then {inherit(attrs) default;} else {} )
|
|
|
|
//{name = l;}
|
|
|
|
)]
|
|
|
|
else (concatLists (map (s: (optionAttrSetToDocList
|
|
|
|
(l + (if l=="" then "" else ".") + s) (builtins.getAttr s attrs)))
|
|
|
|
(builtins.attrNames attrs)))));
|
2007-11-09 10:05:32 -08:00
|
|
|
|
2007-11-12 08:42:13 -08:00
|
|
|
innerModifySumArgs = f: x: a: b: if b == null then (f a b) // x else
|
|
|
|
innerModifySumArgs f x (a // b);
|
|
|
|
modifySumArgs = f: x: innerModifySumArgs f x {};
|
|
|
|
|
2007-11-22 12:26:00 -08:00
|
|
|
debugVal = if builtins ? trace then x: (builtins.trace x x) else x: x;
|
|
|
|
debugXMLVal = if builtins ? trace then x: (builtins.trace (builtins.toXML x) x) else x: x;
|
|
|
|
|
2008-02-13 02:09:29 -08:00
|
|
|
# this can help debug your code as well - designed to not produce thousands of lines
|
|
|
|
traceWhatis = x : __trace (whatis x) x;
|
2008-08-20 01:21:54 -07:00
|
|
|
traceMarked = str: x: __trace (str + (whatis x)) x;
|
2008-12-02 04:26:12 -08:00
|
|
|
attrNamesToStr = a : concatStringsSep "; " (map (x : "${x}=") (__attrNames a));
|
|
|
|
whatis = x :
|
2008-02-13 02:09:29 -08:00
|
|
|
if (__isAttrs x) then
|
2008-12-02 04:26:12 -08:00
|
|
|
if (x ? outPath) then "x is a derivation, name ${if x ? name then x.name else "<no name>"}, { ${attrNamesToStr x} }"
|
|
|
|
else "x is attr set { ${attrNamesToStr x} }"
|
2008-02-13 02:09:29 -08:00
|
|
|
else if (__isFunction x) then "x is a function"
|
2008-06-11 23:58:07 -07:00
|
|
|
else if (x == []) then "x is an empty list"
|
2008-02-13 02:09:29 -08:00
|
|
|
else if (__isList x) then "x is a list, first item is : ${whatis (__head x)}"
|
2008-08-20 01:41:53 -07:00
|
|
|
else if (x == true) then "x is boolean true"
|
|
|
|
else if (x == false) then "x is boolean false"
|
2008-08-19 21:34:28 -07:00
|
|
|
else if (x == null) then "x is null"
|
2008-03-02 06:26:40 -08:00
|
|
|
else "x is probably a string starting, starting characters: ${__substring 0 50 x}..";
|
2008-12-02 04:26:05 -08:00
|
|
|
# trace the arguments passed to function and its result
|
|
|
|
traceCall = n : f : a : let t = n2 : x : traceMarked "${n} ${n2}:" x; in t "result" (f (t "arg 1" a));
|
|
|
|
traceCall2 = n : f : a : b : let t = n2 : x : traceMarked "${n} ${n2}:" x; in t "result" (f (t "arg 1" a) (t "arg 2" b));
|
|
|
|
traceCall3 = n : f : a : b : c : let t = n2 : x : traceMarked "${n} ${n2}:" x; in t "result" (f (t "arg 1" a) (t "arg 2" b) (t "arg 3" c));
|
|
|
|
|
2008-02-13 02:09:29 -08:00
|
|
|
|
|
|
|
|
2007-12-01 08:20:23 -08:00
|
|
|
innerClosePropagation = ready: list: if list == [] then ready else
|
|
|
|
if (head list) ? propagatedBuildInputs then
|
|
|
|
innerClosePropagation (ready ++ [(head list)])
|
|
|
|
((head list).propagatedBuildInputs ++ (tail list)) else
|
|
|
|
innerClosePropagation (ready ++ [(head list)]) (tail list);
|
|
|
|
|
|
|
|
closePropagation = list: (uniqList {inputList = (innerClosePropagation [] list);});
|
|
|
|
|
2008-02-13 02:09:29 -08:00
|
|
|
stringToCharacters = s : let l = __stringLength s; in
|
|
|
|
if (__lessThan l 1) then [""] else [(__substring 0 1 s)] ++ stringToCharacters (__substring 1 (__sub l 1) s);
|
|
|
|
|
|
|
|
# should this be implemented as primop ? Yes it should..
|
|
|
|
escapeShellArg = s :
|
|
|
|
let escapeChar = x : if ( x == "'" ) then "'\"'\"'" else x;
|
|
|
|
in "'" + concatStrings (map escapeChar (stringToCharacters s) ) +"'";
|
|
|
|
|
|
|
|
defineShList = name : list : "\n${name}=(${concatStringsSep " " (map escapeShellArg list)})\n";
|
|
|
|
|
2008-10-14 07:01:00 -07:00
|
|
|
# this as well :-) arg: http://foo/bar/bz.ext returns bz.ext
|
|
|
|
dropPath = s :
|
|
|
|
if s == "" then "" else
|
|
|
|
let takeTillSlash = left : c : s :
|
|
|
|
if left == 0 then s
|
|
|
|
else if (__substring left 1 s == "/") then
|
|
|
|
(__substring (__add left 1) (__sub c 1) s)
|
|
|
|
else takeTillSlash (__sub left 1) (__add c 1) s; in
|
|
|
|
takeTillSlash (__sub (__stringLength s) 1) 1 s;
|
|
|
|
|
2008-01-15 19:29:56 -08:00
|
|
|
# calls a function (f attr value ) for each record item. returns a list
|
2008-12-02 04:26:28 -08:00
|
|
|
# should be renamed to mapAttrsFlatten
|
2008-01-15 19:29:56 -08:00
|
|
|
mapRecordFlatten = f : r : map (attr: f attr (builtins.getAttr attr r) ) (attrNames r);
|
|
|
|
|
2008-04-10 17:40:10 -07:00
|
|
|
# maps a function on each attr value
|
2008-08-16 17:06:01 -07:00
|
|
|
# f = attr : value : ..
|
2008-08-17 00:40:35 -07:00
|
|
|
mapAttrs = f : r : listToAttrs ( mapRecordFlatten (a : v : nv a ( f a v ) ) r);
|
2008-04-10 17:40:10 -07:00
|
|
|
|
2008-01-15 19:29:56 -08:00
|
|
|
# to be used with listToAttrs (_a_ttribute _v_alue)
|
2008-01-17 03:08:24 -08:00
|
|
|
nv = name : value : { inherit name value; };
|
2008-01-15 19:29:56 -08:00
|
|
|
# attribute set containing one attribute
|
2008-01-17 03:08:24 -08:00
|
|
|
nvs = name : value : listToAttrs [ (nv name value) ];
|
2008-01-15 19:29:56 -08:00
|
|
|
# adds / replaces an attribute of an attribute set
|
2008-01-17 03:08:24 -08:00
|
|
|
setAttr = set : name : v : set // (nvs name v);
|
2008-01-15 19:29:56 -08:00
|
|
|
|
2008-11-04 13:40:51 -08:00
|
|
|
# setAttrMerge (similar to mergeAttrsWithFunc but only merges the values of a particular name)
|
|
|
|
# setAttrMerge "a" [] { a = [2];} (x : x ++ [3]) -> { a = [2 3]; }
|
|
|
|
# setAttrMerge "a" [] { } (x : x ++ [3]) -> { a = [ 3]; }
|
|
|
|
setAttrMerge = name : default : attrs : f :
|
|
|
|
setAttr attrs name (f (maybeAttr name default attrs));
|
|
|
|
|
2008-01-15 19:29:56 -08:00
|
|
|
# iterates over a list of attributes collecting the attribute attr if it exists
|
|
|
|
catAttrs = attr : l : fold ( s : l : if (hasAttr attr s) then [(builtins.getAttr attr s)] ++ l else l) [] l;
|
|
|
|
|
2008-12-02 04:26:20 -08:00
|
|
|
mergeAttr = x : y : x // y;
|
|
|
|
mergeAttrs = fold mergeAttr {};
|
2008-01-15 19:29:56 -08:00
|
|
|
|
2008-08-09 13:21:33 -07:00
|
|
|
attrVals = nameList : attrSet :
|
|
|
|
map (x: builtins.getAttr x attrSet) nameList;
|
|
|
|
|
2008-01-15 19:29:56 -08:00
|
|
|
# Using f = a : b = b the result is similar to //
|
|
|
|
# merge attributes with custom function handling the case that the attribute
|
|
|
|
# exists in both sets
|
|
|
|
mergeAttrsWithFunc = f : set1 : set2 :
|
|
|
|
fold (n: set : if (__hasAttr n set)
|
|
|
|
then setAttr set n (f (__getAttr n set) (__getAttr n set2))
|
|
|
|
else set )
|
|
|
|
set1 (__attrNames set2);
|
|
|
|
|
|
|
|
# merging two attribute set concatenating the values of same attribute names
|
|
|
|
# eg { a = 7; } { a = [ 2 3 ]; } becomes { a = [ 7 2 3 ]; }
|
|
|
|
mergeAttrsConcatenateValues = mergeAttrsWithFunc ( a : b : (toList a) ++ (toList b) );
|
|
|
|
|
2008-04-28 15:27:03 -07:00
|
|
|
# merges attributes using //, if a name exisits in both attributes
|
|
|
|
# an error will be triggered unless its listed in mergeLists
|
|
|
|
# so you can mergeAttrsNoOverride { buildInputs = [a]; } { buildInputs = [a]; } {} to get
|
|
|
|
# { buildInputs = [a b]; }
|
|
|
|
# merging buildPhase does'nt really make sense. The cases will be rare where appending /prefixing will fit your needs?
|
|
|
|
# in these cases the first buildPhase will override the second one
|
2008-12-02 04:26:28 -08:00
|
|
|
# ! depreceated, use mergeAttrByFunc instead
|
2008-04-28 15:27:03 -07:00
|
|
|
mergeAttrsNoOverride = { mergeLists ? ["buildInputs" "propagatedBuildInputs"],
|
|
|
|
overrideSnd ? [ "buildPhase" ]
|
|
|
|
} : attrs1 : attrs2 :
|
|
|
|
fold (n: set :
|
|
|
|
setAttr set n ( if (__hasAttr n set)
|
|
|
|
then # merge
|
|
|
|
if elem n mergeLists # attribute contains list, merge them by concatenating
|
|
|
|
then (__getAttr n attrs2) ++ (__getAttr n attrs1)
|
|
|
|
else if elem n overrideSnd
|
|
|
|
then __getAttr n attrs1
|
|
|
|
else throw "error mergeAttrsNoOverride, attribute ${n} given in both attributes - no merge func defined"
|
|
|
|
else __getAttr n attrs2 # add attribute not existing in attr1
|
|
|
|
)) attrs1 (__attrNames attrs2);
|
2008-12-02 04:26:36 -08:00
|
|
|
|
|
|
|
|
|
|
|
# example usage:
|
|
|
|
# mergeAttrByFunc {
|
|
|
|
# inherit mergeAttrBy; # defined below
|
|
|
|
# buildInputs = [ a b ];
|
|
|
|
# } {
|
|
|
|
# buildInputs = [ c d ];
|
|
|
|
# };
|
|
|
|
# will result in
|
|
|
|
# { mergeAttrsBy = [...]; buildInputs = [ a b c d ]; }
|
|
|
|
# is used by prepareDerivationArgs and can be used when composing using
|
|
|
|
# foldArgs, composedArgsAndFun or applyAndFun. Example: composableDerivation in all-packages.nix
|
|
|
|
mergeAttrByFunc = x : y :
|
|
|
|
let
|
|
|
|
mergeAttrBy2 = { mergeAttrBy=mergeAttr; }
|
|
|
|
// (maybeAttr "mergeAttrBy" {} x)
|
|
|
|
// (maybeAttr "mergeAttrBy" {} y); in
|
|
|
|
mergeAttrs [
|
|
|
|
x y
|
|
|
|
(mapAttrs ( a : v : # merge special names using given functions
|
|
|
|
if (__hasAttr a x)
|
|
|
|
then if (__hasAttr a y)
|
|
|
|
then v (__getAttr a x) (__getAttr a y) # both have attr, use merge func
|
|
|
|
else (__getAttr a x) # only x has attr
|
|
|
|
else (__getAttr a y) # only y has attr)
|
|
|
|
) (removeAttrs mergeAttrBy2
|
|
|
|
# don't merge attrs which are neither in x nor y
|
|
|
|
(filter (a : (! __hasAttr a x) && (! __hasAttr a y) )
|
|
|
|
(__attrNames mergeAttrBy2))
|
|
|
|
)
|
|
|
|
)
|
|
|
|
];
|
|
|
|
mergeAttrsByFunc = fold mergeAttrByFunc {};
|
|
|
|
# sane defaults (same name as attr name so that inherit can be used)
|
|
|
|
mergeAttrBy = # { buildInputs = concatList; [...]; passthru = mergeAttr; [..]; }
|
|
|
|
listToAttrs (map (n : nv n concatList) [ "buildInputs" "propagatedBuildInputs" "configureFlags" "prePhases" "postAll" ])
|
|
|
|
// listToAttrs (map (n : nv n mergeAttr) [ "passthru" "meta" ]);
|
|
|
|
|
2008-01-15 19:29:56 -08:00
|
|
|
# returns atribute values as a list
|
|
|
|
flattenAttrs = set : map ( attr : builtins.getAttr attr set) (attrNames set);
|
|
|
|
mapIf = cond : f : fold ( x : l : if (cond x) then [(f x)] ++ l else l) [];
|
|
|
|
|
2008-02-13 02:09:29 -08:00
|
|
|
# pick attrs subset_attr_names and apply f
|
|
|
|
subsetmap = f : attrs : subset_attr_names :
|
|
|
|
listToAttrs (fold ( attr : r : if __hasAttr attr attrs
|
|
|
|
then r ++ [ ( nv attr ( f (__getAttr attr attrs) ) ) ] else r ) []
|
|
|
|
subset_attr_names );
|
|
|
|
|
2008-12-02 04:26:43 -08:00
|
|
|
# prepareDerivationArgs tries to make writing configurable derivations easier
|
|
|
|
# example:
|
|
|
|
# prepareDerivationArgs {
|
|
|
|
# mergeAttrBy = {
|
|
|
|
# myScript = x : y : x ++ "\n" ++ y;
|
|
|
|
# };
|
|
|
|
# cfg = {
|
|
|
|
# readlineSupport = true;
|
|
|
|
# };
|
|
|
|
# flags = {
|
|
|
|
# readline = {
|
|
|
|
# set = {
|
|
|
|
# configureFlags = [ "--with-compiler=${compiler}" ];
|
|
|
|
# buildInputs = [ compiler ];
|
|
|
|
# pass = { inherit compiler; READLINE=1; };
|
|
|
|
# assertion = compiler.dllSupport;
|
|
|
|
# myScript = "foo";
|
|
|
|
# };
|
|
|
|
# unset = { configureFlags = ["--without-compiler"]; };
|
|
|
|
# };
|
|
|
|
# };
|
|
|
|
# src = ...
|
|
|
|
# buildPhase = '' ... '';
|
|
|
|
# name = ...
|
|
|
|
# myScript = "bar";
|
|
|
|
# };
|
|
|
|
# if you don't have need for unset you can omit the surrounding set = { .. } attr
|
|
|
|
# all attrs except flags cfg and mergeAttrBy will be merged with the
|
|
|
|
# additional data from flags depending on config settings
|
|
|
|
# It's used in composableDerivation in all-packages.nix. It's also used
|
|
|
|
# heavily in the new python and libs implementation
|
|
|
|
prepareDerivationArgs = args:
|
|
|
|
let args2 = { cfg = {}; flags = {}; } // args;
|
|
|
|
flagName = name : "${name}Support";
|
|
|
|
cfgWithDefaults = (listToAttrs (map (n : nv (flagName n) false) (attrNames args2.flags)))
|
|
|
|
// args2.cfg;
|
|
|
|
opts = flattenAttrs (mapAttrs (a : v :
|
|
|
|
let v2 = if (v ? set || v ? unset) then v else { set = v; };
|
|
|
|
n = if (__getAttr (flagName a) cfgWithDefaults) then "set" else "unset";
|
|
|
|
attr = maybeAttr n {} v2; in
|
|
|
|
if (maybeAttr "assertion" true attr)
|
|
|
|
then attr
|
|
|
|
else throw "assertion of flag ${a} of derivation ${args.name} failed"
|
|
|
|
) args2.flags );
|
|
|
|
in removeAttrs
|
2008-12-02 04:26:53 -08:00
|
|
|
(mergeAttrsByFunc ([args] ++ opts))
|
2008-12-02 04:26:43 -08:00
|
|
|
["flags" "cfg" "mergeAttrBy" "fixed" ]; # fixed may be passed as fix argument or such
|
|
|
|
# supportFlag functions for convinience
|
|
|
|
sFlagEnable = { name, buildInputs ? [], propagatedBuildInputs ? [] } : {
|
|
|
|
set = { configureFlags = "--enable-${name}"; inherit buildInputs; inherit propagatedBuildInputs; };
|
|
|
|
unset = { configureFlags = "--disable-${name}"; };
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
|
2008-01-15 19:29:56 -08:00
|
|
|
# Marc 2nd proposal: (not everything has been tested in detail yet..)
|
2008-12-02 04:26:28 -08:00
|
|
|
# depreceated because it's too complicated. use prepareDerivationArgs instead
|
2008-01-15 19:29:56 -08:00
|
|
|
|
|
|
|
# usage / example
|
|
|
|
# flagConfig = {
|
|
|
|
# } // (enableDisableFeature "flagName" "configure_feature" extraAttrs;)
|
|
|
|
#
|
|
|
|
# is equal to
|
|
|
|
# flagConfig = {
|
|
|
|
# flagName = { cfgOption = "--enable-configure_feature"; } // extraAttrs;
|
|
|
|
# no_flagName = { cfgOption = "--disable-configure_feature"; };
|
|
|
|
enableDisableFeature = flagName : configure_feature : extraAttrs :
|
2008-01-17 03:08:24 -08:00
|
|
|
listToAttrs [ ( nv flagName ({ cfgOption = "--enable-${configure_feature}"; } // extraAttrs ) )
|
|
|
|
( nv "no_${flagName}" ({ cfgOption = "--disable-${configure_feature}"; } ) )];
|
2008-01-15 19:29:56 -08:00
|
|
|
|
|
|
|
# calls chooseOptionsByFlags2 with some preprocessing
|
|
|
|
# chooseOptionsByFlags2 returns an attribute set meant to be used to create new derivaitons.
|
|
|
|
# see mkDerivationByConfiguration in all-packages.nix and the examples given below.
|
|
|
|
# You can just copy paste them into all-packages.nix to test them..
|
|
|
|
|
|
|
|
chooseOptionsByFlags = { flagConfig, args, optionals ? [], defaults ? [],
|
|
|
|
collectExtraPhaseActions ? [] } :
|
|
|
|
let passedOptionals = filter ( x : hasAttr x args ) optionals; # these are in optionals and in args
|
|
|
|
# we simply merge in <optional_name> = { buildInputs = <arg.<optional_name>; pass = <arg.optional_name>; }
|
|
|
|
flagConfigWithOptionals = flagConfig // ( listToAttrs
|
2008-01-17 03:08:24 -08:00
|
|
|
(map ( o : nv o ( { buildInputs = o; pass = nvs o (builtins.getAttr o args); }
|
2008-01-15 19:29:56 -08:00
|
|
|
// getAttr [o] {} flagConfig )
|
|
|
|
)
|
|
|
|
passedOptionals ) );
|
|
|
|
|
|
|
|
in chooseOptionsByFlags2 flagConfigWithOptionals collectExtraPhaseActions args
|
|
|
|
( (getAttr ["flags"] defaults args) ++ passedOptionals);
|
|
|
|
|
|
|
|
chooseOptionsByFlags2 = flagConfig : collectExtraPhaseActions : args : flags :
|
|
|
|
let
|
|
|
|
# helper function
|
|
|
|
collectFlags = # state : flags :
|
|
|
|
fold ( flag : s : (
|
|
|
|
if (hasAttr flag s.result) then s # this state has already been visited
|
|
|
|
else if (! hasAttr flag flagConfig) then throw "unkown flag `${flag}' specified"
|
|
|
|
else let fDesc = (builtins.getAttr flag flagConfig);
|
|
|
|
implied = flatten ( getAttr ["implies"] [] fDesc );
|
|
|
|
blocked = flatten ( getAttr ["blocks"] [] fDesc );
|
|
|
|
# add this flag
|
|
|
|
s2 = s // { result = ( setAttr s.result flag (builtins.getAttr flag flagConfig) );
|
|
|
|
blockedFlagsBy = s.blockedFlagsBy
|
2008-01-17 03:08:24 -08:00
|
|
|
// listToAttrs (map (b: nv b flag ) blocked); };
|
2008-01-15 19:29:56 -08:00
|
|
|
# add implied flags
|
|
|
|
in collectFlags s2 implied
|
|
|
|
));
|
|
|
|
|
|
|
|
# chosen contains flagConfig but only having those attributes elected by flags
|
|
|
|
# (or by implies attributes of elected attributes)
|
|
|
|
options = let stateOpts = collectFlags { blockedFlagsBy = {}; result = {}; }
|
|
|
|
(flags ++ ( if (hasAttr "mandatory" flagConfig) then ["mandatory"] else [] ));
|
|
|
|
# these options have not been chosen (neither by flags nor by implies)
|
|
|
|
unsetOptions = filter ( x : (! hasAttr x stateOpts.result) && (hasAttr ("no_"+x) flagConfig))
|
|
|
|
( attrNames flagConfig );
|
|
|
|
# no add the corresponding no_ attributes as well ..
|
|
|
|
state = collectFlags stateOpts (map ( x : "no_" + x ) unsetOptions);
|
|
|
|
in # check for blockings:
|
|
|
|
assert ( all id ( map ( b: if (hasAttr b state.result)
|
|
|
|
then throw "flag ${b} is blocked by flag ${__getAttr b state.blockedFlagsBy}"
|
|
|
|
else true )
|
|
|
|
(attrNames state.blockedFlagsBy) ) );
|
|
|
|
state.result;
|
|
|
|
flatOptions = flattenAttrs options;
|
|
|
|
|
|
|
|
# helper functions :
|
|
|
|
collectAttrs = attr : catAttrs attr flatOptions;
|
|
|
|
optsConcatStrs = delimiter : attrs : concatStrings
|
|
|
|
( intersperse delimiter (flatten ( collectAttrs attrs ) ) );
|
|
|
|
|
|
|
|
ifStringGetArg = x : if (__isAttrs x) then x # ( TODO implement __isString ?)
|
2008-01-17 03:08:24 -08:00
|
|
|
else nvs x (__getAttr x args);
|
2008-01-15 19:29:56 -08:00
|
|
|
|
|
|
|
in assert ( all id ( mapRecordFlatten ( attr : r : if ( all id ( flatten (getAttr ["assertion"] [] r ) ) )
|
|
|
|
then true else throw "assertion failed flag ${attr}" )
|
|
|
|
options) );
|
|
|
|
( rec {
|
|
|
|
|
|
|
|
#foldOptions = attr: f : start: fold f start (catAttrs attr flatOptions);
|
|
|
|
|
|
|
|
# compared to flags flagsSet does also contain the implied flags.. This makes it easy to write assertions. ( assert args.
|
|
|
|
inherit options flatOptions collectAttrs optsConcatStrs;
|
|
|
|
|
|
|
|
buildInputs = map ( attr: if (! hasAttr attr args) then throw "argument ${attr} is missing!" else (builtins.getAttr attr args) )
|
|
|
|
(flatten (catAttrs "buildInputs" flatOptions));
|
|
|
|
propagatedBuildInputs = map ( attr: if (! hasAttr attr args) then throw "argument ${attr} is missing!" else (builtins.getAttr attr args) )
|
|
|
|
(flatten (catAttrs "propagatedBuildInputs" flatOptions));
|
|
|
|
|
|
|
|
configureFlags = optsConcatStrs " " "cfgOption";
|
|
|
|
|
2008-01-17 03:08:24 -08:00
|
|
|
#flags = listToAttrs (map ( flag: nv flag (hasAttr flag options) ) (attrNames flagConfig) );
|
|
|
|
flags_prefixed = listToAttrs (map ( flag: nv ("flag_set_"+flag) (hasAttr flag options) ) (attrNames flagConfig) );
|
2008-01-15 19:29:56 -08:00
|
|
|
|
|
|
|
pass = mergeAttrs ( map ifStringGetArg ( flatten (collectAttrs "pass") ) );
|
|
|
|
} # now add additional phase actions (see examples)
|
2008-01-17 03:08:24 -08:00
|
|
|
// listToAttrs ( map ( x : nv x (optsConcatStrs "\n" x) ) collectExtraPhaseActions ) );
|
2006-11-27 08:58:08 -08:00
|
|
|
}
|
2008-01-15 19:29:56 -08:00
|
|
|
|
|
|
|
/*
|
|
|
|
TODO: Perhaps it's better to move this documentation / these tests into some extra packages ..
|
|
|
|
|
|
|
|
# ###########################################################################
|
|
|
|
# configuration tutorial .. examples and tests..
|
|
|
|
# Copy this into all-packages.nix and try
|
|
|
|
|
|
|
|
# The following derviations will all fail..
|
|
|
|
# But they will print the passed options so that you can get to know
|
|
|
|
# how these configurations ought to work.
|
|
|
|
# TODO: There is no nice way to pass an otpion yet.
|
|
|
|
# I could imagine something like
|
|
|
|
# flags = [ "flagA" "flagB" { flagC = 4; } ];
|
|
|
|
|
|
|
|
# They are named:
|
|
|
|
# simpleYes, simpleNo,
|
|
|
|
# defaultsimpleYes, defaultsimpleNo
|
|
|
|
# optionalssimpleYes, optionalssimpleNo
|
|
|
|
# bitingsimpleYes can only be ran with -iA blockingBiteMonster
|
|
|
|
# assertionsimpleNo
|
|
|
|
# of course you can use -iA and the attribute name as well to select these examples
|
|
|
|
|
|
|
|
# dummy build input
|
|
|
|
whoGetsTheFlagFirst = gnused;
|
|
|
|
whoGetsTheFlagLast = gnumake;
|
|
|
|
|
|
|
|
# simple example demonstrating containing one flag.
|
|
|
|
# features:
|
|
|
|
# * configure options are passed automatically
|
|
|
|
# * buildInputs are collected (they are special, see the setup script)
|
|
|
|
# * they can be passed by additional name as well using pass = { inherit (args) python }
|
|
|
|
# ( or short (value not attrs) : pass = "python" )
|
|
|
|
# * an attribute named the same way as the flag is added indicating
|
|
|
|
# true/ false (flag has been set/ not set)
|
|
|
|
# * extra phase dependend commands can be added
|
|
|
|
# Its easy to add your own stuff using co.collectAttrs or co.optsConcatStrs
|
|
|
|
# ( perhaps this name will change?)
|
|
|
|
simpleFlagYesNoF = namePrefix : extraFlagAttrs : mkDerivationByConfiguration ( {
|
|
|
|
flagConfig = {
|
|
|
|
flag = { name = namePrefix + "simpleYes";
|
|
|
|
cfgOption = [ "--Yes" "--you-dont-need-a-list" ];
|
|
|
|
buildInputs = [ "whoGetsTheFlagFirst" ];
|
|
|
|
pass = { inherit gnumake; };
|
|
|
|
extraConfigureCmd = "echo Hello, it worked! ";
|
|
|
|
blocks = "bitingMonster";
|
|
|
|
};
|
|
|
|
no_flag = { name = namePrefix + "simpleNo";
|
|
|
|
cfgOption = "--no";
|
|
|
|
implies = ["bitingMonster"];
|
|
|
|
};
|
|
|
|
bitingMonster = {
|
|
|
|
extraConfigureCmd = "echo Ill bite you";
|
|
|
|
};
|
|
|
|
gnutar = { cfgOption="--with-gnutar";
|
|
|
|
# buildInputs and pass will be added automatically if gnutar is added to optionals
|
|
|
|
};
|
|
|
|
# can be used to check configure options of dependencies
|
|
|
|
# eg testFlag = { assertion = [ arg.desktop.flag_set_wmii (! arg.desktop.flag_set_gnome) (! arg.desktops.flag_set_kde ]; }
|
|
|
|
assertionFlag = { assertion = false; }; # assert is nix language keyword
|
|
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
collectExtraPhaseActions = [ "extraConfigureCmd" ];
|
|
|
|
|
|
|
|
extraAttrs = co : {
|
|
|
|
name = ( __head (co.collectAttrs "name") );
|
|
|
|
|
|
|
|
unpackPhase = "
|
|
|
|
echo my name is
|
|
|
|
echo \$name
|
|
|
|
echo
|
|
|
|
echo flag given \\(should be 1 or empty string\\) ?
|
|
|
|
echo \$flag_set_flag
|
|
|
|
echo
|
|
|
|
echo my build inputs are
|
|
|
|
echo \$buildInputs
|
|
|
|
echo
|
|
|
|
echo my configuration flags are
|
|
|
|
echo \$configureFlags
|
|
|
|
echo
|
|
|
|
echo what about gnumake? Did it pass?
|
|
|
|
echo \$gnumake
|
|
|
|
echo
|
|
|
|
echo configurePhase command is
|
|
|
|
echo $\configurePhase
|
|
|
|
echo
|
|
|
|
echo gnutar passed? \\(optional test\\)
|
|
|
|
echo \$gnutar
|
|
|
|
echo
|
|
|
|
echo dying now
|
|
|
|
echo die_Hopefully_Soon
|
|
|
|
";
|
|
|
|
configurePhase = co.extraConfigureCmd;
|
|
|
|
};
|
|
|
|
} // extraFlagAttrs );
|
|
|
|
|
|
|
|
|
|
|
|
simpleYes = simpleFlagYesNoF "" {} {
|
|
|
|
inherit whoGetsTheFlagFirst lib stdenv;
|
|
|
|
flags = ["flag"];
|
|
|
|
};
|
|
|
|
# note the "I'll bite you" because of the implies attribute
|
|
|
|
simpleNo = simpleFlagYesNoF "" {} {
|
|
|
|
inherit whoGetsTheFlagFirst lib stdenv;
|
|
|
|
flags = [];
|
|
|
|
};
|
|
|
|
|
|
|
|
# specifying defaults by adding a default attribute
|
|
|
|
|
|
|
|
yesAgainDefault = simpleFlagYesNoF "default" { defaults = [ "flag" ];} {
|
|
|
|
inherit whoGetsTheFlagFirst lib stdenv;
|
|
|
|
};
|
|
|
|
noAgainOverridingDefault = simpleFlagYesNoF "default" { defaults = [ "flag" ];} {
|
|
|
|
inherit whoGetsTheFlagFirst lib stdenv;
|
|
|
|
flags = [];
|
|
|
|
};
|
|
|
|
|
|
|
|
# requested by Michael Raskin: activate flag automatically if dependency is passed:
|
|
|
|
withGnutarOptional = simpleFlagYesNoF "optionals" { optionals = [ "gnutar" ];} {
|
|
|
|
flags = [ "flag" ]; # I only need to pass this to trigger name optionalssimpleYes
|
|
|
|
inherit whoGetsTheFlagFirst lib stdenv;
|
|
|
|
inherit gnutar;
|
|
|
|
};
|
|
|
|
withoutGnutarOptional = simpleFlagYesNoF "optionals" { optionals = [ "gnutar" ];} {
|
|
|
|
inherit whoGetsTheFlagFirst lib stdenv;
|
|
|
|
};
|
|
|
|
|
|
|
|
# blocking example, this shouldn't even start building:
|
|
|
|
blockingBiteMonster = simpleFlagYesNoF "biting" {} {
|
|
|
|
inherit whoGetsTheFlagFirst lib stdenv;
|
|
|
|
flags = [ "flag" "bitingMonster" ];
|
|
|
|
};
|
|
|
|
|
|
|
|
# assertion example this shouldn't even start building:
|
|
|
|
assertion = simpleFlagYesNoF "assertion" {} {
|
|
|
|
inherit whoGetsTheFlagFirst lib stdenv;
|
|
|
|
flags = [ "assertionFlag" ];
|
|
|
|
};
|
|
|
|
*/
|