diff --git a/nixos/doc/manual/development/settings-options.section.md b/nixos/doc/manual/development/settings-options.section.md index b37cd1d345252a..2d76deae88cbd1 100644 --- a/nixos/doc/manual/development/settings-options.section.md +++ b/nixos/doc/manual/development/settings-options.section.md @@ -343,6 +343,20 @@ have a predefined type and string generator already declared under and returning a set with TOML-specific attributes `type` and `generate` as specified [below](#pkgs-formats-result). +`pkgs.formats.xml` { format ? "badgerfish", withHeader ? true} + +: A function taking an attribute set with values + and returning a set with XML-specific attributes `type` and + `generate` as specified [below](#pkgs-formats-result). + + `format` + + : Input format. Because XML can not be translated one-to-one, we have to use intermediate formats. Possible values `["badgerfish"]`. + + `withHeader` + + : Outputs the xml with header. + `pkgs.formats.elixirConf { elixir ? pkgs.elixir }` : A function taking an attribute set with values diff --git a/pkgs/pkgs-lib/formats.nix b/pkgs/pkgs-lib/formats.nix index 5bf51a7e6fe69e..22e4b287de6047 100644 --- a/pkgs/pkgs-lib/formats.nix +++ b/pkgs/pkgs-lib/formats.nix @@ -544,4 +544,64 @@ rec { '') {}; }; + xml = + { + format ? "badgerfish", + withHeader ? true, + }: + if format == "badgerfish" then + { + type = let + valueType = nullOr (oneOf [ + bool + int + float + str + path + (attrsOf valueType) + (listOf valueType) + ]) // { + description = "XML value"; + }; + in valueType; + + generate = + name: value: + pkgs.callPackage ( + { + runCommand, + python3, + libxml2Python, + }: + runCommand name + { + nativeBuildInputs = [ + python3 + python3.pkgs.xmltodict + libxml2Python + ]; + value = builtins.toJSON value; + pythonGen = '' + import json + import os + import xmltodict + + with open(os.environ["valuePath"], "r") as f: + print(xmltodict.unparse(json.load(f), full_document=${toString withHeader}, pretty=True, indent=" " * 2)) + ''; + passAsFile = [ + "value" + "pythonGen" + ]; + preferLocalBuild = true; + } + '' + python3 "$pythonGenPath" > $out + xmllint $out > /dev/null + '' + ) { }; + } + else + throw "Unknown format: ${format}"; + } diff --git a/pkgs/pkgs-lib/tests/formats.nix b/pkgs/pkgs-lib/tests/formats.nix index 1c6f741353faa7..ec16bb97dd180a 100644 --- a/pkgs/pkgs-lib/tests/formats.nix +++ b/pkgs/pkgs-lib/tests/formats.nix @@ -610,4 +610,31 @@ in runBuildTests { ''; }; + badgerfishToXmlGenerate = shouldPass { + format = formats.xml { }; + input = { + root = { + "@id" = "123"; + "@class" = "example"; + child1 = { + "@name" = "child1Name"; + "#text" = "text node"; + }; + child2 = { + grandchild = "This is a grandchild text node."; + }; + nulltest = null; + }; + }; + expected = '' + + + text node + + This is a grandchild text node. + + + + ''; + }; }