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 + + + +