about summary refs log tree commit diff
path: root/nixpkgs/nixos/modules/services/networking/strongswan-swanctl/param-lib.nix
blob: 2bbb39a76049be47d71d811c6551b7bafd69b03c (plain) (blame)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
lib :

with lib;

rec {
  paramsToConf = cfg : ps : mkConf 0 (paramsToRenderedStrings cfg ps);

  # mkConf takes an indentation level (which usually starts at 0) and a nested
  # attribute set of strings and will render that set to a strongswan.conf style
  # configuration format. For example:
  #
  #   mkConf 0 {a = "1"; b = { c = { "foo" = "2"; "bar" = "3"; }; d = "4";};}   =>   ''
  #   a = 1
  #   b {
  #     c {
  #       foo = 2
  #       bar = 3
  #     }
  #     d = 4
  #   }''
  mkConf = indent : ps :
    concatMapStringsSep "\n"
      (name:
        let value = ps.${name};
            indentation = replicate indent " ";
        in
        indentation + (
          if isAttrs value
          then "${name} {\n" +
                 mkConf (indent + 2) value + "\n" +
               indentation + "}"
          else "${name} = ${value}"
        )
      )
      (attrNames ps);

  replicate = n : c : concatStrings (builtins.genList (_x : c) n);

  # `paramsToRenderedStrings cfg ps` converts the NixOS configuration `cfg`
  # (typically the "config" argument of a NixOS module) and the set of
  # parameters `ps` (an attribute set where the values are constructed using the
  # parameter constructors in ./param-constructors.nix) to a nested attribute
  # set of strings (rendered parameters).
  paramsToRenderedStrings = cfg : ps :
    filterEmptySets (
      (mapParamsRecursive (path: name: param:
        let value = attrByPath path null cfg;
        in optionalAttrs (value != null) (param.render name value)
      ) ps));

  filterEmptySets = set : filterAttrs (n: v: (v != null)) (mapAttrs (name: value:
    if isAttrs value
    then let value' = filterEmptySets value;
         in if value' == {}
            then null
            else value'
    else value
  ) set);

  # Recursively map over every parameter in the given attribute set.
  mapParamsRecursive = mapAttrsRecursiveCond' (as: (!(as ? _type && as._type == "param")));

  mapAttrsRecursiveCond' = cond: f: set:
    let
      recurse = path: set:
        let
          g =
            name: value:
            if isAttrs value && cond value
              then { ${name} = recurse (path ++ [name]) value; }
              else f (path ++ [name]) name value;
        in mapAttrs'' g set;
    in recurse [] set;

  mapAttrs'' = f: set:
    foldl' (a: b: a // b) {} (map (attr: f attr set.${attr}) (attrNames set));

  # Extract the options from the given set of parameters.
  paramsToOptions = ps :
    mapParamsRecursive (_path: name: param: { ${name} = param.option; }) ps;

}