2019-03-07 18:02:26 +01:00
|
|
|
|
{ lib, pkgs }:
|
2017-11-11 11:52:17 +01:00
|
|
|
|
|
|
|
|
|
|
with lib;
|
|
|
|
|
|
|
2019-03-07 18:02:26 +01:00
|
|
|
|
rec {
|
2019-02-12 16:20:40 +01:00
|
|
|
|
moduleToAttrs = value:
|
|
|
|
|
|
if isAttrs value
|
|
|
|
|
|
then mapAttrs (n: v: moduleToAttrs v) (filterAttrs (n: v: !(hasPrefix "_" n) && v != null) value)
|
|
|
|
|
|
|
|
|
|
|
|
else if isList value
|
|
|
|
|
|
then map (v: moduleToAttrs v) value
|
|
|
|
|
|
|
|
|
|
|
|
else value;
|
|
|
|
|
|
|
2017-11-11 11:52:17 +01:00
|
|
|
|
mkAllDefault = value: priority:
|
|
|
|
|
|
if isAttrs value
|
|
|
|
|
|
then mapAttrs (n: v: mkAllDefault v priority) value
|
|
|
|
|
|
|
|
|
|
|
|
else if isList value
|
|
|
|
|
|
then map (v: mkAllDefault v priority) value
|
|
|
|
|
|
|
|
|
|
|
|
else mkOverride priority value;
|
|
|
|
|
|
|
2021-05-13 17:27:08 -04:00
|
|
|
|
loadYAML = path: importJSON (pkgs.runCommand "yaml-to-json"
|
|
|
|
|
|
{ } "${pkgs.remarshal}/bin/remarshal -i ${path} -if yaml -of json > $out");
|
2017-11-11 11:52:17 +01:00
|
|
|
|
|
2021-05-13 17:27:08 -04:00
|
|
|
|
toYAML = config: builtins.readFile (pkgs.runCommand "to-yaml"
|
|
|
|
|
|
{
|
|
|
|
|
|
buildInputs = [ pkgs.remarshal ];
|
|
|
|
|
|
} ''
|
2019-02-10 21:03:47 +01:00
|
|
|
|
remarshal -i ${pkgs.writeText "to-json" (builtins.toJSON config)} -if json -of yaml > $out
|
2020-01-14 19:11:24 +00:00
|
|
|
|
'');
|
2019-02-10 21:03:47 +01:00
|
|
|
|
|
2021-05-13 17:27:08 -04:00
|
|
|
|
toMultiDocumentYaml = name: documents: pkgs.runCommand name
|
|
|
|
|
|
{
|
|
|
|
|
|
buildInputs = [ pkgs.remarshal ];
|
|
|
|
|
|
}
|
|
|
|
|
|
(concatMapStringsSep "\necho --- >> $out\n"
|
|
|
|
|
|
(d:
|
|
|
|
|
|
"remarshal -i ${builtins.toFile "doc" (builtins.toJSON d)} -if json -of yaml >> $out"
|
|
|
|
|
|
)
|
|
|
|
|
|
documents);
|
2020-04-05 21:25:34 +07:00
|
|
|
|
|
2019-02-10 21:03:47 +01:00
|
|
|
|
toBase64 = value:
|
|
|
|
|
|
builtins.readFile
|
2021-05-13 17:27:08 -04:00
|
|
|
|
(pkgs.runCommand "value-to-b64" { } "echo -n '${value}' | ${pkgs.coreutils}/bin/base64 -w0 > $out");
|
2019-02-10 21:03:47 +01:00
|
|
|
|
|
|
|
|
|
|
exp = base: exp: foldr (value: acc: acc * base) 1 (range 1 exp);
|
|
|
|
|
|
|
2021-05-13 17:27:08 -04:00
|
|
|
|
octalToDecimal = value: (foldr
|
|
|
|
|
|
(char: acc: {
|
|
|
|
|
|
i = acc.i + 1;
|
|
|
|
|
|
value = acc.value + (toInt char) * (exp 8 acc.i);
|
|
|
|
|
|
})
|
|
|
|
|
|
{ i = 0; value = 0; }
|
|
|
|
|
|
(stringToCharacters value)).value;
|
2019-10-06 21:39:10 +02:00
|
|
|
|
|
|
|
|
|
|
submoduleWithSpecialArgs = opts: specialArgs:
|
|
|
|
|
|
let
|
|
|
|
|
|
opts' = toList opts;
|
|
|
|
|
|
inherit (lib.modules) evalModules;
|
|
|
|
|
|
in
|
|
|
|
|
|
mkOptionType rec {
|
|
|
|
|
|
name = "submodule";
|
|
|
|
|
|
check = x: isAttrs x || isFunction x;
|
|
|
|
|
|
merge = loc: defs:
|
|
|
|
|
|
let
|
|
|
|
|
|
coerce = def: if isFunction def then def else { config = def; };
|
2021-05-13 17:27:08 -04:00
|
|
|
|
modules = opts' ++ map (def: { _file = def.file; imports = [ (coerce def.value) ]; }) defs;
|
|
|
|
|
|
in
|
|
|
|
|
|
(evalModules {
|
2019-10-06 21:39:10 +02:00
|
|
|
|
inherit modules specialArgs;
|
|
|
|
|
|
prefix = loc;
|
|
|
|
|
|
}).config;
|
|
|
|
|
|
getSubOptions = prefix: (evalModules
|
2021-05-13 17:27:08 -04:00
|
|
|
|
{
|
|
|
|
|
|
modules = opts'; inherit prefix specialArgs;
|
2019-10-06 21:39:10 +02:00
|
|
|
|
# This is a work-around due to the fact that some sub-modules,
|
|
|
|
|
|
# such as the one included in an attribute set, expects a "args"
|
|
|
|
|
|
# attribute to be given to the sub-module. As the option
|
|
|
|
|
|
# evaluation does not have any specific attribute name, we
|
|
|
|
|
|
# provide a default one for the documentation.
|
|
|
|
|
|
#
|
|
|
|
|
|
# This is mandatory as some option declaration might use the
|
|
|
|
|
|
# "name" attribute given as argument of the submodule and use it
|
|
|
|
|
|
# as the default of option declarations.
|
|
|
|
|
|
#
|
|
|
|
|
|
# Using lookalike unicode single angle quotation marks because
|
|
|
|
|
|
# of the docbook transformation the options receive. In all uses
|
|
|
|
|
|
# > and < wouldn't be encoded correctly so the encoded values
|
|
|
|
|
|
# would be used, and use of `<` and `>` would break the XML document.
|
|
|
|
|
|
# It shouldn't cause an issue since this is cosmetic for the manual.
|
|
|
|
|
|
args.name = "‹name›";
|
|
|
|
|
|
}).options;
|
|
|
|
|
|
getSubModules = opts';
|
|
|
|
|
|
substSubModules = m: submoduleWithSpecialArgs m specialArgs;
|
|
|
|
|
|
functor = (defaultFunctor name) // {
|
|
|
|
|
|
# Merging of submodules is done as part of mergeOptionDecls, as we have to annotate
|
|
|
|
|
|
# each submodule with its location.
|
2021-05-13 17:27:08 -04:00
|
|
|
|
payload = [ ];
|
|
|
|
|
|
binOp = lhs: rhs: [ ];
|
2019-10-06 21:39:10 +02:00
|
|
|
|
};
|
|
|
|
|
|
};
|
2019-10-20 13:36:10 +02:00
|
|
|
|
|
2021-05-13 17:27:08 -04:00
|
|
|
|
coerceListOfSubmodulesToAttrs = submodule: keyFn:
|
|
|
|
|
|
let
|
2019-10-20 13:36:10 +02:00
|
|
|
|
mergeValuesByFn = keyFn: values:
|
2021-05-13 17:27:08 -04:00
|
|
|
|
listToAttrs (map
|
|
|
|
|
|
(value:
|
|
|
|
|
|
nameValuePair (toString (keyFn value)) value
|
|
|
|
|
|
)
|
|
|
|
|
|
values);
|
2019-10-20 13:36:10 +02:00
|
|
|
|
|
|
|
|
|
|
# Either value of type `finalType` or `coercedType`, the latter is
|
|
|
|
|
|
# converted to `finalType` using `coerceFunc`.
|
|
|
|
|
|
coercedTo = coercedType: coerceFunc: finalType:
|
|
|
|
|
|
mkOptionType rec {
|
|
|
|
|
|
name = "coercedTo";
|
|
|
|
|
|
description = "${finalType.description} or ${coercedType.description}";
|
|
|
|
|
|
check = x: finalType.check x || coercedType.check x;
|
|
|
|
|
|
merge = loc: defs:
|
|
|
|
|
|
let
|
|
|
|
|
|
coerceVal = val:
|
|
|
|
|
|
if finalType.check val then
|
|
|
|
|
|
val
|
|
|
|
|
|
else
|
|
|
|
|
|
let coerced = coerceFunc val; in assert finalType.check coerced; coerced;
|
|
|
|
|
|
|
2021-05-13 17:27:08 -04:00
|
|
|
|
in
|
|
|
|
|
|
finalType.merge loc (map (def: def // { value = coerceVal def.value; }) defs);
|
2019-10-20 13:36:10 +02:00
|
|
|
|
getSubOptions = finalType.getSubOptions;
|
|
|
|
|
|
getSubModules = finalType.getSubModules;
|
|
|
|
|
|
substSubModules = m: coercedTo coercedType coerceFunc (finalType.substSubModules m);
|
|
|
|
|
|
typeMerge = t1: t2: null;
|
|
|
|
|
|
functor = (defaultFunctor name) // { wrapped = finalType; };
|
|
|
|
|
|
};
|
2021-05-13 17:27:08 -04:00
|
|
|
|
in
|
|
|
|
|
|
coercedTo
|
2019-10-20 13:36:10 +02:00
|
|
|
|
(types.listOf (types.submodule submodule))
|
|
|
|
|
|
(mergeValuesByFn keyFn)
|
|
|
|
|
|
(types.attrsOf (types.submodule submodule));
|
2017-11-11 11:52:17 +01:00
|
|
|
|
}
|