diff --git a/lib/generators.nix b/lib/generators.nix
index e29bf5a02f3d9e..391492309e9ecc 100644
--- a/lib/generators.nix
+++ b/lib/generators.nix
@@ -549,13 +549,17 @@ in rec {
# Inputs
- Options
- : Empty set, there may be configuration options in the future
+ Structured function argument
+
+ : escape (optional, default: `false`)
+ : If this option is true, XML special characters are escaped in string values and keys
Value
: The value to be converted to Plist
*/
- toPlist = {}: v: let
+ toPlist = {
+ escape ? false
+ }: v: let
expr = ind: x:
if x == null then "" else
if isBool x then bool ind x else
@@ -569,10 +573,12 @@ in rec {
literal = ind: x: ind + x;
+ maybeEscapeXML = if escape then escapeXML else x: x;
+
bool = ind: x: literal ind (if x then "" else "");
int = ind: x: literal ind "${toString x}";
- str = ind: x: literal ind "${escapeXML x}";
- key = ind: x: literal ind "${escapeXML x}";
+ str = ind: x: literal ind "${maybeEscapeXML x}";
+ key = ind: x: literal ind "${maybeEscapeXML x}";
float = ind: x: literal ind "${toString x}";
indent = ind: expr "\t${ind}";
@@ -598,7 +604,10 @@ in rec {
(expr "\t${ind}" value)
]) x));
- in ''
+ in
+ # TODO: As discussed in #356502, we should do the actual deprecation in a future release cycle.
+ lib.warnIf (!escape) "Using `lib.generators.toPlist` without escape = true; is deprecated"
+ ''
${expr "" v}
diff --git a/lib/tests/misc.nix b/lib/tests/misc.nix
index fc9bd0a404a966..2128cf17a3334d 100644
--- a/lib/tests/misc.nix
+++ b/lib/tests/misc.nix
@@ -1635,8 +1635,31 @@ runTests {
expected = "«foo»";
};
- testToPlist = {
+ testToPlistUnescaped = {
expr = mapAttrs (const (generators.toPlist { })) {
+ value = {
+ nested.values = {
+ int = 42;
+ float = 0.1337;
+ bool = true;
+ emptystring = "";
+ string = "fn\${o}\"r\\d";
+ newlinestring = "\n";
+ path = /. + "/foo";
+ null_ = null;
+ list = [ 3 4 "test" ];
+ emptylist = [];
+ attrs = { foo = null; "foo b/ar" = "baz"; };
+ emptyattrs = {};
+ "keys are not " = "and < neither are string values";
+ };
+ };
+ };
+ expected = { value = builtins.readFile ./test-to-plist-unescaped-expected.plist; };
+ };
+
+ testToPlistEscaped = {
+ expr = mapAttrs (const (generators.toPlist { escape = true; })) {
value = {
nested.values = {
int = 42;
@@ -1655,7 +1678,7 @@ runTests {
};
};
};
- expected = { value = builtins.readFile ./test-to-plist-expected.plist; };
+ expected = { value = builtins.readFile ./test-to-plist-escaped-expected.plist; };
};
testToLuaEmptyAttrSet = {
diff --git a/lib/tests/test-to-plist-expected.plist b/lib/tests/test-to-plist-escaped-expected.plist
similarity index 100%
rename from lib/tests/test-to-plist-expected.plist
rename to lib/tests/test-to-plist-escaped-expected.plist
diff --git a/lib/tests/test-to-plist-unescaped-expected.plist b/lib/tests/test-to-plist-unescaped-expected.plist
new file mode 100644
index 00000000000000..cad684188bcf1c
--- /dev/null
+++ b/lib/tests/test-to-plist-unescaped-expected.plist
@@ -0,0 +1,48 @@
+
+
+
+
+ nested
+
+ values
+
+ attrs
+
+ foo b/ar
+ baz
+
+ bool
+
+ emptyattrs
+
+
+
+ emptylist
+
+
+
+ emptystring
+
+ float
+ 0.133700
+ int
+ 42
+ keys are not
+ and < neither are string values
+ list
+
+ 3
+ 4
+ test
+
+ newlinestring
+
+
+ path
+ /foo
+ string
+ fn${o}"r\d
+
+
+
+