2024-05-01 22:14:04 +00:00
|
|
|
|
/*
|
|
|
|
|
Functions that generate widespread file
|
|
|
|
|
formats from nix data structures.
|
2024-06-30 08:16:52 +00:00
|
|
|
|
|
2024-05-01 22:14:04 +00:00
|
|
|
|
They all follow a similar interface:
|
|
|
|
|
generator { config-attrs } data
|
2024-06-30 08:16:52 +00:00
|
|
|
|
|
2024-05-01 22:14:04 +00:00
|
|
|
|
`config-attrs` are “holes” in the generators
|
|
|
|
|
with sensible default implementations that
|
|
|
|
|
can be overwritten. The default implementations
|
|
|
|
|
are mostly generators themselves, called with
|
|
|
|
|
their respective default values; they can be reused.
|
2024-06-30 08:16:52 +00:00
|
|
|
|
|
2024-05-01 22:14:04 +00:00
|
|
|
|
Tests can be found in ./tests/misc.nix
|
|
|
|
|
Documentation in the manual, #sec-generators
|
|
|
|
|
*/
|
|
|
|
|
{ lib }:
|
|
|
|
|
|
|
|
|
|
let
|
|
|
|
|
inherit (lib)
|
|
|
|
|
addErrorContext
|
|
|
|
|
assertMsg
|
|
|
|
|
attrNames
|
|
|
|
|
concatLists
|
|
|
|
|
concatMapStringsSep
|
|
|
|
|
concatStrings
|
|
|
|
|
concatStringsSep
|
|
|
|
|
const
|
|
|
|
|
elem
|
|
|
|
|
escape
|
|
|
|
|
filter
|
|
|
|
|
flatten
|
|
|
|
|
foldl
|
|
|
|
|
functionArgs # Note: not the builtin; considers `__functor` in attrsets.
|
|
|
|
|
gvariant
|
|
|
|
|
hasInfix
|
|
|
|
|
head
|
|
|
|
|
id
|
|
|
|
|
init
|
|
|
|
|
isAttrs
|
|
|
|
|
isBool
|
|
|
|
|
isDerivation
|
|
|
|
|
isFloat
|
|
|
|
|
isFunction # Note: not the builtin; considers `__functor` in attrsets.
|
|
|
|
|
isInt
|
|
|
|
|
isList
|
|
|
|
|
isPath
|
|
|
|
|
isString
|
|
|
|
|
last
|
|
|
|
|
length
|
|
|
|
|
mapAttrs
|
|
|
|
|
mapAttrsToList
|
|
|
|
|
optionals
|
|
|
|
|
recursiveUpdate
|
|
|
|
|
replaceStrings
|
|
|
|
|
reverseList
|
|
|
|
|
splitString
|
|
|
|
|
tail
|
|
|
|
|
toList
|
|
|
|
|
;
|
|
|
|
|
|
|
|
|
|
inherit (lib.strings)
|
|
|
|
|
escapeNixIdentifier
|
|
|
|
|
floatToString
|
|
|
|
|
match
|
|
|
|
|
split
|
|
|
|
|
toJSON
|
|
|
|
|
typeOf
|
|
|
|
|
;
|
|
|
|
|
|
|
|
|
|
## -- HELPER FUNCTIONS & DEFAULTS --
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
Convert a value to a sensible default string representation.
|
|
|
|
|
The builtin `toString` function has some strange defaults,
|
|
|
|
|
suitable for bash scripts but not much else.
|
|
|
|
|
*/
|
|
|
|
|
mkValueStringDefault =
|
|
|
|
|
{ }:
|
|
|
|
|
v:
|
|
|
|
|
let
|
|
|
|
|
err = t: v: abort ("generators.mkValueStringDefault: " + "${t} not supported: ${toPretty { } v}");
|
|
|
|
|
in
|
|
|
|
|
if isInt v then
|
|
|
|
|
toString v
|
|
|
|
|
# convert derivations to store paths
|
|
|
|
|
else if isDerivation v then
|
|
|
|
|
toString v
|
|
|
|
|
# we default to not quoting strings
|
|
|
|
|
else if isString v then
|
|
|
|
|
v
|
|
|
|
|
# isString returns "1", which is not a good default
|
|
|
|
|
else if true == v then
|
|
|
|
|
"true"
|
|
|
|
|
# here it returns to "", which is even less of a good default
|
|
|
|
|
else if false == v then
|
|
|
|
|
"false"
|
|
|
|
|
else if null == v then
|
|
|
|
|
"null"
|
|
|
|
|
# if you have lists you probably want to replace this
|
|
|
|
|
else if isList v then
|
|
|
|
|
err "lists" v
|
|
|
|
|
# same as for lists, might want to replace
|
|
|
|
|
else if isAttrs v then
|
|
|
|
|
err "attrsets" v
|
|
|
|
|
# functions can’t be printed of course
|
|
|
|
|
else if isFunction v then
|
|
|
|
|
err "functions" v
|
|
|
|
|
# Floats currently can't be converted to precise strings,
|
|
|
|
|
# condition warning on nix version once this isn't a problem anymore
|
|
|
|
|
# See https://github.com/NixOS/nix/pull/3480
|
|
|
|
|
else if isFloat v then
|
|
|
|
|
floatToString v
|
|
|
|
|
else
|
|
|
|
|
err "this value is" (toString v);
|
|
|
|
|
|
2024-06-30 08:16:52 +00:00
|
|
|
|
/*
|
2024-05-01 22:14:04 +00:00
|
|
|
|
Generate a line of key k and value v, separated by
|
|
|
|
|
character sep. If sep appears in k, it is escaped.
|
|
|
|
|
Helper for synaxes with different separators.
|
|
|
|
|
|
|
|
|
|
mkValueString specifies how values should be formatted.
|
|
|
|
|
|
|
|
|
|
mkKeyValueDefault {} ":" "f:oo" "bar"
|
|
|
|
|
> "f\:oo:bar"
|
2024-06-30 08:16:52 +00:00
|
|
|
|
*/
|
2024-05-01 22:14:04 +00:00
|
|
|
|
mkKeyValueDefault =
|
2024-06-30 08:16:52 +00:00
|
|
|
|
{
|
2024-05-01 22:14:04 +00:00
|
|
|
|
mkValueString ? mkValueStringDefault { },
|
2024-06-30 08:16:52 +00:00
|
|
|
|
}:
|
2024-05-01 22:14:04 +00:00
|
|
|
|
sep: k: v:
|
|
|
|
|
"${escape [ sep ] k}${sep}${mkValueString v}";
|
|
|
|
|
|
|
|
|
|
## -- FILE FORMAT GENERATORS --
|
|
|
|
|
|
2024-06-30 08:16:52 +00:00
|
|
|
|
/*
|
2024-05-01 22:14:04 +00:00
|
|
|
|
Generate a key-value-style config file from an attrset.
|
|
|
|
|
|
|
|
|
|
mkKeyValue is the same as in toINI.
|
|
|
|
|
*/
|
|
|
|
|
toKeyValue =
|
2024-06-30 08:16:52 +00:00
|
|
|
|
{
|
2024-05-01 22:14:04 +00:00
|
|
|
|
mkKeyValue ? mkKeyValueDefault { } "=",
|
|
|
|
|
listsAsDuplicateKeys ? false,
|
|
|
|
|
indent ? "",
|
|
|
|
|
}:
|
|
|
|
|
let
|
|
|
|
|
mkLine = k: v: indent + mkKeyValue k v + "\n";
|
|
|
|
|
mkLines =
|
|
|
|
|
if listsAsDuplicateKeys then
|
|
|
|
|
k: v: map (mkLine k) (if isList v then v else [ v ])
|
2024-06-30 08:16:52 +00:00
|
|
|
|
else
|
2024-05-01 22:14:04 +00:00
|
|
|
|
k: v: [ (mkLine k v) ];
|
2024-06-30 08:16:52 +00:00
|
|
|
|
in
|
2024-05-01 22:14:04 +00:00
|
|
|
|
attrs: concatStrings (concatLists (mapAttrsToList mkLines attrs));
|
2024-06-30 08:16:52 +00:00
|
|
|
|
|
|
|
|
|
/*
|
2024-05-01 22:14:04 +00:00
|
|
|
|
Generate an INI-style config file from an
|
|
|
|
|
attrset of sections to an attrset of key-value pairs.
|
2024-06-30 08:16:52 +00:00
|
|
|
|
|
2024-05-01 22:14:04 +00:00
|
|
|
|
generators.toINI {} {
|
|
|
|
|
foo = { hi = "${pkgs.hello}"; ciao = "bar"; };
|
|
|
|
|
baz = { "also, integers" = 42; };
|
2024-06-30 08:16:52 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
> [baz]
|
2024-05-01 22:14:04 +00:00
|
|
|
|
> also, integers=42
|
2024-06-30 08:16:52 +00:00
|
|
|
|
>
|
|
|
|
|
> [foo]
|
2024-05-01 22:14:04 +00:00
|
|
|
|
> ciao=bar
|
|
|
|
|
> hi=/nix/store/y93qql1p5ggfnaqjjqhxcw0vqw95rlz0-hello-2.10
|
2024-06-30 08:16:52 +00:00
|
|
|
|
|
2024-05-01 22:14:04 +00:00
|
|
|
|
The mk* configuration attributes can generically change
|
|
|
|
|
the way sections and key-value strings are generated.
|
2024-06-30 08:16:52 +00:00
|
|
|
|
|
2024-05-01 22:14:04 +00:00
|
|
|
|
For more examples see the test cases in ./tests/misc.nix.
|
2024-06-30 08:16:52 +00:00
|
|
|
|
*/
|
|
|
|
|
toINI =
|
|
|
|
|
{
|
2024-05-01 22:14:04 +00:00
|
|
|
|
# apply transformations (e.g. escapes) to section names
|
|
|
|
|
mkSectionName ? (
|
2024-06-30 08:16:52 +00:00
|
|
|
|
name:
|
|
|
|
|
escape [
|
|
|
|
|
"["
|
|
|
|
|
"]"
|
|
|
|
|
] name
|
|
|
|
|
),
|
2024-05-01 22:14:04 +00:00
|
|
|
|
# format a setting line from key and value
|
|
|
|
|
mkKeyValue ? mkKeyValueDefault { } "=",
|
|
|
|
|
# allow lists as values for duplicate keys
|
|
|
|
|
listsAsDuplicateKeys ? false,
|
2024-06-30 08:16:52 +00:00
|
|
|
|
}:
|
2024-05-01 22:14:04 +00:00
|
|
|
|
attrsOfAttrs:
|
2024-06-30 08:16:52 +00:00
|
|
|
|
let
|
2024-05-01 22:14:04 +00:00
|
|
|
|
# map function to string for each key val
|
|
|
|
|
mapAttrsToStringsSep =
|
|
|
|
|
sep: mapFn: attrs:
|
|
|
|
|
concatStringsSep sep (mapAttrsToList mapFn attrs);
|
|
|
|
|
mkSection =
|
|
|
|
|
sectName: sectValues:
|
|
|
|
|
''
|
|
|
|
|
[${mkSectionName sectName}]
|
|
|
|
|
''
|
|
|
|
|
+ toKeyValue { inherit mkKeyValue listsAsDuplicateKeys; } sectValues;
|
|
|
|
|
in
|
|
|
|
|
# map input to ini sections
|
|
|
|
|
mapAttrsToStringsSep "\n" mkSection attrsOfAttrs;
|
2024-06-30 08:16:52 +00:00
|
|
|
|
|
2024-05-01 22:14:04 +00:00
|
|
|
|
/*
|
|
|
|
|
Generate an INI-style config file from an attrset
|
|
|
|
|
specifying the global section (no header), and an
|
|
|
|
|
attrset of sections to an attrset of key-value pairs.
|
2024-06-30 08:16:52 +00:00
|
|
|
|
|
2024-05-01 22:14:04 +00:00
|
|
|
|
generators.toINIWithGlobalSection {} {
|
|
|
|
|
globalSection = {
|
|
|
|
|
someGlobalKey = "hi";
|
|
|
|
|
};
|
|
|
|
|
sections = {
|
|
|
|
|
foo = { hi = "${pkgs.hello}"; ciao = "bar"; };
|
|
|
|
|
baz = { "also, integers" = 42; };
|
|
|
|
|
}
|
2024-06-30 08:16:52 +00:00
|
|
|
|
|
2024-05-01 22:14:04 +00:00
|
|
|
|
> someGlobalKey=hi
|
|
|
|
|
>
|
|
|
|
|
> [baz]
|
|
|
|
|
> also, integers=42
|
|
|
|
|
>
|
|
|
|
|
> [foo]
|
|
|
|
|
> ciao=bar
|
|
|
|
|
> hi=/nix/store/y93qql1p5ggfnaqjjqhxcw0vqw95rlz0-hello-2.10
|
2024-06-30 08:16:52 +00:00
|
|
|
|
|
2024-05-01 22:14:04 +00:00
|
|
|
|
The mk* configuration attributes can generically change
|
|
|
|
|
the way sections and key-value strings are generated.
|
2024-06-30 08:16:52 +00:00
|
|
|
|
|
2024-05-01 22:14:04 +00:00
|
|
|
|
For more examples see the test cases in ./tests/misc.nix.
|
2024-06-30 08:16:52 +00:00
|
|
|
|
|
2024-05-01 22:14:04 +00:00
|
|
|
|
If you don’t need a global section, you can also use
|
|
|
|
|
`generators.toINI` directly, which only takes
|
|
|
|
|
the part in `sections`.
|
|
|
|
|
*/
|
|
|
|
|
toINIWithGlobalSection =
|
|
|
|
|
{
|
|
|
|
|
# apply transformations (e.g. escapes) to section names
|
|
|
|
|
mkSectionName ? (
|
|
|
|
|
name:
|
|
|
|
|
escape [
|
|
|
|
|
"["
|
|
|
|
|
"]"
|
|
|
|
|
] name
|
|
|
|
|
),
|
|
|
|
|
# format a setting line from key and value
|
|
|
|
|
mkKeyValue ? mkKeyValueDefault { } "=",
|
|
|
|
|
# allow lists as values for duplicate keys
|
|
|
|
|
listsAsDuplicateKeys ? false,
|
|
|
|
|
}:
|
|
|
|
|
{
|
|
|
|
|
globalSection,
|
|
|
|
|
sections ? { },
|
|
|
|
|
}:
|
|
|
|
|
(
|
|
|
|
|
if globalSection == { } then
|
|
|
|
|
""
|
|
|
|
|
else
|
|
|
|
|
(toKeyValue { inherit mkKeyValue listsAsDuplicateKeys; } globalSection) + "\n"
|
|
|
|
|
)
|
|
|
|
|
+ (toINI { inherit mkSectionName mkKeyValue listsAsDuplicateKeys; } sections);
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
Generate a git-config file from an attrset.
|
2024-06-30 08:16:52 +00:00
|
|
|
|
|
2024-05-01 22:14:04 +00:00
|
|
|
|
It has two major differences from the regular INI format:
|
2024-06-30 08:16:52 +00:00
|
|
|
|
|
2024-05-01 22:14:04 +00:00
|
|
|
|
1. values are indented with tabs
|
|
|
|
|
2. sections can have sub-sections
|
2024-06-30 08:16:52 +00:00
|
|
|
|
|
2024-05-01 22:14:04 +00:00
|
|
|
|
generators.toGitINI {
|
|
|
|
|
url."ssh://git@github.com/".insteadOf = "https://github.com";
|
|
|
|
|
user.name = "edolstra";
|
|
|
|
|
}
|
2024-06-30 08:16:52 +00:00
|
|
|
|
|
2024-05-01 22:14:04 +00:00
|
|
|
|
> [url "ssh://git@github.com/"]
|
|
|
|
|
> insteadOf = "https://github.com"
|
|
|
|
|
>
|
|
|
|
|
> [user]
|
|
|
|
|
> name = "edolstra"
|
|
|
|
|
*/
|
|
|
|
|
toGitINI =
|
|
|
|
|
attrs:
|
|
|
|
|
let
|
|
|
|
|
mkSectionName =
|
|
|
|
|
name:
|
|
|
|
|
let
|
|
|
|
|
containsQuote = hasInfix ''"'' name;
|
|
|
|
|
sections = splitString "." name;
|
|
|
|
|
section = head sections;
|
|
|
|
|
subsections = tail sections;
|
|
|
|
|
subsection = concatStringsSep "." subsections;
|
|
|
|
|
in
|
|
|
|
|
if containsQuote || subsections == [ ] then name else ''${section} "${subsection}"'';
|
|
|
|
|
|
|
|
|
|
mkValueString =
|
|
|
|
|
v:
|
|
|
|
|
let
|
|
|
|
|
escapedV = ''"${
|
|
|
|
|
replaceStrings
|
|
|
|
|
[
|
|
|
|
|
"\n"
|
|
|
|
|
" "
|
|
|
|
|
''"''
|
|
|
|
|
"\\"
|
|
|
|
|
]
|
|
|
|
|
[
|
|
|
|
|
"\\n"
|
|
|
|
|
"\\t"
|
|
|
|
|
''\"''
|
|
|
|
|
"\\\\"
|
|
|
|
|
]
|
|
|
|
|
v
|
|
|
|
|
}"'';
|
|
|
|
|
in
|
|
|
|
|
mkValueStringDefault { } (if isString v then escapedV else v);
|
|
|
|
|
|
|
|
|
|
# generation for multiple ini values
|
|
|
|
|
mkKeyValue =
|
|
|
|
|
k: v:
|
|
|
|
|
let
|
|
|
|
|
mkKeyValue = mkKeyValueDefault { inherit mkValueString; } " = " k;
|
|
|
|
|
in
|
|
|
|
|
concatStringsSep "\n" (map (kv: "\t" + mkKeyValue kv) (toList v));
|
|
|
|
|
|
|
|
|
|
# converts { a.b.c = 5; } to { "a.b".c = 5; } for toINI
|
|
|
|
|
gitFlattenAttrs =
|
|
|
|
|
let
|
|
|
|
|
recurse =
|
|
|
|
|
path: value:
|
|
|
|
|
if isAttrs value && !isDerivation value then
|
|
|
|
|
mapAttrsToList (name: value: recurse ([ name ] ++ path) value) value
|
|
|
|
|
else if length path > 1 then
|
|
|
|
|
{ ${concatStringsSep "." (reverseList (tail path))}.${head path} = value; }
|
|
|
|
|
else
|
|
|
|
|
{ ${head path} = value; };
|
|
|
|
|
in
|
|
|
|
|
attrs: foldl recursiveUpdate { } (flatten (recurse [ ] attrs));
|
|
|
|
|
|
|
|
|
|
toINI_ = toINI { inherit mkKeyValue mkSectionName; };
|
|
|
|
|
in
|
|
|
|
|
toINI_ (gitFlattenAttrs attrs);
|
|
|
|
|
|
|
|
|
|
# mkKeyValueDefault wrapper that handles dconf INI quirks.
|
|
|
|
|
# The main differences of the format is that it requires strings to be quoted.
|
|
|
|
|
mkDconfKeyValue = mkKeyValueDefault { mkValueString = v: toString (gvariant.mkValue v); } "=";
|
|
|
|
|
|
|
|
|
|
# Generates INI in dconf keyfile style. See https://help.gnome.org/admin/system-admin-guide/stable/dconf-keyfiles.html.en
|
|
|
|
|
# for details.
|
|
|
|
|
toDconfINI = toINI { mkKeyValue = mkDconfKeyValue; };
|
|
|
|
|
|
|
|
|
|
withRecursion =
|
|
|
|
|
{
|
|
|
|
|
# If this option is not null, the given value will stop evaluating at a certain depth
|
|
|
|
|
depthLimit,
|
|
|
|
|
# If this option is true, an error will be thrown, if a certain given depth is exceeded
|
|
|
|
|
throwOnDepthLimit ? true,
|
|
|
|
|
}:
|
|
|
|
|
assert isInt depthLimit;
|
|
|
|
|
let
|
|
|
|
|
specialAttrs = [
|
|
|
|
|
"__functor"
|
|
|
|
|
"__functionArgs"
|
|
|
|
|
"__toString"
|
|
|
|
|
"__pretty"
|
|
|
|
|
];
|
|
|
|
|
stepIntoAttr = evalNext: name: if elem name specialAttrs then id else evalNext;
|
|
|
|
|
transform =
|
|
|
|
|
depth:
|
|
|
|
|
if depthLimit != null && depth > depthLimit then
|
|
|
|
|
if throwOnDepthLimit then
|
|
|
|
|
throw "Exceeded maximum eval-depth limit of ${toString depthLimit} while trying to evaluate with `generators.withRecursion'!"
|
2024-06-30 08:16:52 +00:00
|
|
|
|
else
|
2024-05-01 22:14:04 +00:00
|
|
|
|
const "<unevaluated>"
|
2024-06-30 08:16:52 +00:00
|
|
|
|
else
|
2024-05-01 22:14:04 +00:00
|
|
|
|
id;
|
2024-06-30 08:16:52 +00:00
|
|
|
|
mapAny =
|
2024-05-01 22:14:04 +00:00
|
|
|
|
depth: v:
|
2024-06-30 08:16:52 +00:00
|
|
|
|
let
|
2024-05-01 22:14:04 +00:00
|
|
|
|
evalNext = x: mapAny (depth + 1) (transform (depth + 1) x);
|
2024-06-30 08:16:52 +00:00
|
|
|
|
in
|
2024-05-01 22:14:04 +00:00
|
|
|
|
if isAttrs v then
|
|
|
|
|
mapAttrs (stepIntoAttr evalNext) v
|
|
|
|
|
else if isList v then
|
|
|
|
|
map evalNext v
|
2024-06-30 08:16:52 +00:00
|
|
|
|
else
|
2024-05-01 22:14:04 +00:00
|
|
|
|
transform (depth + 1) v;
|
2024-06-30 08:16:52 +00:00
|
|
|
|
in
|
2024-05-01 22:14:04 +00:00
|
|
|
|
mapAny 0;
|
2024-06-30 08:16:52 +00:00
|
|
|
|
|
|
|
|
|
/*
|
2024-05-01 22:14:04 +00:00
|
|
|
|
Pretty print a value, akin to `builtins.trace`.
|
|
|
|
|
Should probably be a builtin as well.
|
|
|
|
|
The pretty-printed string should be suitable for rendering default values
|
|
|
|
|
in the NixOS manual. In particular, it should be as close to a valid Nix expression
|
|
|
|
|
as possible.
|
2024-06-30 08:16:52 +00:00
|
|
|
|
*/
|
2024-05-01 22:14:04 +00:00
|
|
|
|
toPretty =
|
2024-06-30 08:16:52 +00:00
|
|
|
|
{
|
|
|
|
|
/*
|
2024-05-01 22:14:04 +00:00
|
|
|
|
If this option is true, attrsets like { __pretty = fn; val = …; }
|
|
|
|
|
will use fn to convert val to a pretty printed representation.
|
|
|
|
|
(This means fn is type Val -> String.)
|
2024-06-30 08:16:52 +00:00
|
|
|
|
*/
|
2024-05-01 22:14:04 +00:00
|
|
|
|
allowPrettyValues ? false,
|
|
|
|
|
# If this option is true, the output is indented with newlines for attribute sets and lists
|
|
|
|
|
multiline ? true,
|
|
|
|
|
# Initial indentation level
|
|
|
|
|
indent ? "",
|
2024-06-30 08:16:52 +00:00
|
|
|
|
}:
|
|
|
|
|
let
|
|
|
|
|
go =
|
2024-05-01 22:14:04 +00:00
|
|
|
|
indent: v:
|
2024-06-30 08:16:52 +00:00
|
|
|
|
let
|
2024-05-01 22:14:04 +00:00
|
|
|
|
introSpace = if multiline then "\n${indent} " else " ";
|
|
|
|
|
outroSpace = if multiline then "\n${indent}" else " ";
|
2024-06-30 08:16:52 +00:00
|
|
|
|
in
|
2024-05-01 22:14:04 +00:00
|
|
|
|
if isInt v then
|
|
|
|
|
toString v
|
|
|
|
|
# toString loses precision on floats, so we use toJSON instead. This isn't perfect
|
|
|
|
|
# as the resulting string may not parse back as a float (e.g. 42, 1e-06), but for
|
|
|
|
|
# pretty-printing purposes this is acceptable.
|
|
|
|
|
else if isFloat v then
|
|
|
|
|
builtins.toJSON v
|
|
|
|
|
else if isString v then
|
|
|
|
|
let
|
|
|
|
|
lines = filter (v: !isList v) (split "\n" v);
|
|
|
|
|
escapeSingleline = escape [
|
2024-06-30 08:16:52 +00:00
|
|
|
|
"\\"
|
|
|
|
|
"\""
|
|
|
|
|
"\${"
|
|
|
|
|
];
|
2024-05-01 22:14:04 +00:00
|
|
|
|
escapeMultiline =
|
|
|
|
|
replaceStrings
|
2024-06-30 08:16:52 +00:00
|
|
|
|
[
|
|
|
|
|
"\${"
|
|
|
|
|
"''"
|
|
|
|
|
]
|
|
|
|
|
[
|
|
|
|
|
"''\${"
|
|
|
|
|
"'''"
|
|
|
|
|
];
|
2024-05-01 22:14:04 +00:00
|
|
|
|
singlelineResult = "\"" + concatStringsSep "\\n" (map escapeSingleline lines) + "\"";
|
|
|
|
|
multilineResult =
|
2024-06-30 08:16:52 +00:00
|
|
|
|
let
|
2024-05-01 22:14:04 +00:00
|
|
|
|
escapedLines = map escapeMultiline lines;
|
|
|
|
|
# The last line gets a special treatment: if it's empty, '' is on its own line at the "outer"
|
|
|
|
|
# indentation level. Otherwise, '' is appended to the last line.
|
|
|
|
|
lastLine = last escapedLines;
|
2024-06-30 08:16:52 +00:00
|
|
|
|
in
|
|
|
|
|
"''"
|
2024-05-01 22:14:04 +00:00
|
|
|
|
+ introSpace
|
|
|
|
|
+ concatStringsSep introSpace (init escapedLines)
|
|
|
|
|
+ (if lastLine == "" then outroSpace else introSpace + lastLine)
|
2024-06-30 08:16:52 +00:00
|
|
|
|
+ "''";
|
2024-05-01 22:14:04 +00:00
|
|
|
|
in
|
|
|
|
|
if multiline && length lines > 1 then multilineResult else singlelineResult
|
|
|
|
|
else if true == v then
|
|
|
|
|
"true"
|
|
|
|
|
else if false == v then
|
|
|
|
|
"false"
|
|
|
|
|
else if null == v then
|
|
|
|
|
"null"
|
|
|
|
|
else if isPath v then
|
|
|
|
|
toString v
|
|
|
|
|
else if isList v then
|
|
|
|
|
if v == [ ] then
|
2024-06-30 08:16:52 +00:00
|
|
|
|
"[ ]"
|
2024-05-01 22:14:04 +00:00
|
|
|
|
else
|
|
|
|
|
"[" + introSpace + concatMapStringsSep introSpace (go (indent + " ")) v + outroSpace + "]"
|
|
|
|
|
else if isFunction v then
|
|
|
|
|
let
|
|
|
|
|
fna = functionArgs v;
|
|
|
|
|
showFnas = concatStringsSep ", " (
|
|
|
|
|
mapAttrsToList (name: hasDefVal: if hasDefVal then name + "?" else name) fna
|
|
|
|
|
);
|
|
|
|
|
in
|
|
|
|
|
if fna == { } then "<function>" else "<function, args: {${showFnas}}>"
|
|
|
|
|
else if isAttrs v then
|
|
|
|
|
# apply pretty values if allowed
|
|
|
|
|
if allowPrettyValues && v ? __pretty && v ? val then
|
|
|
|
|
v.__pretty v.val
|
|
|
|
|
else if v == { } then
|
2024-06-30 08:16:52 +00:00
|
|
|
|
"{ }"
|
2024-05-01 22:14:04 +00:00
|
|
|
|
else if v ? type && v.type == "derivation" then
|
|
|
|
|
"<derivation ${v.name or "???"}>"
|
|
|
|
|
else
|
|
|
|
|
"{"
|
|
|
|
|
+ introSpace
|
|
|
|
|
+ concatStringsSep introSpace (
|
|
|
|
|
mapAttrsToList (
|
|
|
|
|
name: value:
|
|
|
|
|
"${escapeNixIdentifier name} = ${
|
|
|
|
|
addErrorContext "while evaluating an attribute `${name}`" (go (indent + " ") value)
|
|
|
|
|
};"
|
|
|
|
|
) v
|
|
|
|
|
)
|
|
|
|
|
+ outroSpace
|
|
|
|
|
+ "}"
|
|
|
|
|
else
|
|
|
|
|
abort "generators.toPretty: should never happen (v = ${v})";
|
|
|
|
|
in
|
|
|
|
|
go indent;
|
|
|
|
|
|
|
|
|
|
# PLIST handling
|
|
|
|
|
toPlist =
|
|
|
|
|
{ }:
|
|
|
|
|
v:
|
|
|
|
|
let
|
|
|
|
|
expr =
|
|
|
|
|
ind: x:
|
|
|
|
|
if x == null then
|
|
|
|
|
""
|
|
|
|
|
else if isBool x then
|
|
|
|
|
bool ind x
|
|
|
|
|
else if isInt x then
|
|
|
|
|
int ind x
|
|
|
|
|
else if isString x then
|
|
|
|
|
str ind x
|
|
|
|
|
else if isList x then
|
|
|
|
|
list ind x
|
|
|
|
|
else if isAttrs x then
|
|
|
|
|
attrs ind x
|
|
|
|
|
else if isPath x then
|
|
|
|
|
str ind (toString x)
|
|
|
|
|
else if isFloat x then
|
|
|
|
|
float ind x
|
2024-06-30 08:16:52 +00:00
|
|
|
|
else
|
2024-05-01 22:14:04 +00:00
|
|
|
|
abort "generators.toPlist: should never happen (v = ${v})";
|
2024-06-30 08:16:52 +00:00
|
|
|
|
|
2024-05-01 22:14:04 +00:00
|
|
|
|
literal = ind: x: ind + x;
|
2024-06-30 08:16:52 +00:00
|
|
|
|
|
2024-05-01 22:14:04 +00:00
|
|
|
|
bool = ind: x: literal ind (if x then "<true/>" else "<false/>");
|
|
|
|
|
int = ind: x: literal ind "<integer>${toString x}</integer>";
|
|
|
|
|
str = ind: x: literal ind "<string>${x}</string>";
|
|
|
|
|
key = ind: x: literal ind "<key>${x}</key>";
|
|
|
|
|
float = ind: x: literal ind "<real>${toString x}</real>";
|
2024-06-30 08:16:52 +00:00
|
|
|
|
|
2024-05-01 22:14:04 +00:00
|
|
|
|
indent = ind: expr "\t${ind}";
|
2024-06-30 08:16:52 +00:00
|
|
|
|
|
2024-05-01 22:14:04 +00:00
|
|
|
|
item = ind: concatMapStringsSep "\n" (indent ind);
|
2024-06-30 08:16:52 +00:00
|
|
|
|
|
2024-05-01 22:14:04 +00:00
|
|
|
|
list =
|
|
|
|
|
ind: x:
|
|
|
|
|
concatStringsSep "\n" [
|
|
|
|
|
(literal ind "<array>")
|
|
|
|
|
(item ind x)
|
|
|
|
|
(literal ind "</array>")
|
|
|
|
|
];
|
2024-06-30 08:16:52 +00:00
|
|
|
|
|
2024-05-01 22:14:04 +00:00
|
|
|
|
attrs =
|
|
|
|
|
ind: x:
|
|
|
|
|
concatStringsSep "\n" [
|
|
|
|
|
(literal ind "<dict>")
|
|
|
|
|
(attr ind x)
|
|
|
|
|
(literal ind "</dict>")
|
|
|
|
|
];
|
2024-06-30 08:16:52 +00:00
|
|
|
|
|
2024-05-01 22:14:04 +00:00
|
|
|
|
attr =
|
|
|
|
|
let
|
|
|
|
|
attrFilter = name: value: name != "_module" && value != null;
|
|
|
|
|
in
|
|
|
|
|
ind: x:
|
|
|
|
|
concatStringsSep "\n" (
|
|
|
|
|
flatten (
|
|
|
|
|
mapAttrsToList (
|
|
|
|
|
name: value:
|
|
|
|
|
optionals (attrFilter name value) [
|
|
|
|
|
(key "\t${ind}" name)
|
|
|
|
|
(expr "\t${ind}" value)
|
|
|
|
|
]
|
|
|
|
|
) x
|
2024-06-30 08:16:52 +00:00
|
|
|
|
)
|
2024-05-01 22:14:04 +00:00
|
|
|
|
);
|
2024-06-30 08:16:52 +00:00
|
|
|
|
|
2024-05-01 22:14:04 +00:00
|
|
|
|
in
|
|
|
|
|
''
|
|
|
|
|
<?xml version="1.0" encoding="UTF-8"?>
|
|
|
|
|
<!DOCTYPE plist PUBLIC "-//Apple Computer//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd">
|
|
|
|
|
<plist version="1.0">
|
|
|
|
|
${expr "" v}
|
|
|
|
|
</plist>'';
|
2024-06-30 08:16:52 +00:00
|
|
|
|
|
2024-05-01 22:14:04 +00:00
|
|
|
|
/*
|
|
|
|
|
Translate a simple Nix expression to Dhall notation.
|
|
|
|
|
Note that integers are translated to Integer and never
|
|
|
|
|
the Natural type.
|
|
|
|
|
*/
|
|
|
|
|
toDhall =
|
|
|
|
|
{ }@args:
|
|
|
|
|
v:
|
|
|
|
|
let
|
|
|
|
|
concatItems = concatStringsSep ", ";
|
|
|
|
|
in
|
|
|
|
|
if isAttrs v then
|
|
|
|
|
"{ ${concatItems (mapAttrsToList (key: value: "${key} = ${toDhall args value}") v)} }"
|
|
|
|
|
else if isList v then
|
|
|
|
|
"[ ${concatItems (map (toDhall args) v)} ]"
|
|
|
|
|
else if isInt v then
|
|
|
|
|
"${if v < 0 then "" else "+"}${toString v}"
|
|
|
|
|
else if isBool v then
|
|
|
|
|
(if v then "True" else "False")
|
|
|
|
|
else if isFunction v then
|
|
|
|
|
abort "generators.toDhall: cannot convert a function to Dhall"
|
|
|
|
|
else if v == null then
|
|
|
|
|
abort "generators.toDhall: cannot convert a null to Dhall"
|
|
|
|
|
else
|
|
|
|
|
toJSON v;
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
Translate a simple Nix expression to Lua representation with occasional
|
|
|
|
|
Lua-inlines that can be constructed by mkLuaInline function.
|
2024-06-30 08:16:52 +00:00
|
|
|
|
|
2024-05-01 22:14:04 +00:00
|
|
|
|
Configuration:
|
|
|
|
|
* multiline - by default is true which results in indented block-like view.
|
|
|
|
|
* indent - initial indent.
|
|
|
|
|
* asBindings - by default generate single value, but with this use attrset to set global vars.
|
2024-06-30 08:16:52 +00:00
|
|
|
|
|
2024-05-01 22:14:04 +00:00
|
|
|
|
Attention:
|
|
|
|
|
Regardless of multiline parameter there is no trailing newline.
|
2024-06-30 08:16:52 +00:00
|
|
|
|
|
2024-05-01 22:14:04 +00:00
|
|
|
|
Example:
|
|
|
|
|
generators.toLua {}
|
2024-06-30 08:16:52 +00:00
|
|
|
|
{
|
2024-05-01 22:14:04 +00:00
|
|
|
|
cmd = [ "typescript-language-server" "--stdio" ];
|
|
|
|
|
settings.workspace.library = mkLuaInline ''vim.api.nvim_get_runtime_file("", true)'';
|
2024-06-30 08:16:52 +00:00
|
|
|
|
}
|
|
|
|
|
->
|
2024-05-01 22:14:04 +00:00
|
|
|
|
{
|
|
|
|
|
["cmd"] = {
|
|
|
|
|
"typescript-language-server",
|
|
|
|
|
"--stdio"
|
2024-06-30 08:16:52 +00:00
|
|
|
|
},
|
2024-05-01 22:14:04 +00:00
|
|
|
|
["settings"] = {
|
|
|
|
|
["workspace"] = {
|
|
|
|
|
["library"] = (vim.api.nvim_get_runtime_file("", true))
|
2024-06-30 08:16:52 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
2024-05-01 22:14:04 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Type:
|
|
|
|
|
toLua :: AttrSet -> Any -> String
|
|
|
|
|
*/
|
|
|
|
|
toLua =
|
|
|
|
|
{
|
|
|
|
|
# If this option is true, the output is indented with newlines for attribute sets and lists
|
|
|
|
|
multiline ? true,
|
|
|
|
|
# Initial indentation level
|
|
|
|
|
indent ? "",
|
|
|
|
|
# Interpret as variable bindings
|
|
|
|
|
asBindings ? false,
|
|
|
|
|
}@args:
|
|
|
|
|
v:
|
|
|
|
|
let
|
|
|
|
|
innerIndent = "${indent} ";
|
|
|
|
|
introSpace = if multiline then "\n${innerIndent}" else " ";
|
|
|
|
|
outroSpace = if multiline then "\n${indent}" else " ";
|
|
|
|
|
innerArgs = args // {
|
|
|
|
|
indent = if asBindings then indent else innerIndent;
|
|
|
|
|
asBindings = false;
|
|
|
|
|
};
|
|
|
|
|
concatItems = concatStringsSep ",${introSpace}";
|
|
|
|
|
isLuaInline =
|
|
|
|
|
{
|
|
|
|
|
_type ? null,
|
|
|
|
|
...
|
|
|
|
|
}:
|
|
|
|
|
_type == "lua-inline";
|
|
|
|
|
|
|
|
|
|
generatedBindings =
|
|
|
|
|
assert assertMsg (badVarNames == [ ]) "Bad Lua var names: ${toPretty { } badVarNames}";
|
|
|
|
|
concatStrings (mapAttrsToList (key: value: "${indent}${key} = ${toLua innerArgs value}\n") v);
|
|
|
|
|
|
|
|
|
|
# https://en.wikibooks.org/wiki/Lua_Programming/variable#Variable_names
|
|
|
|
|
matchVarName = match "[[:alpha:]_][[:alnum:]_]*(\\.[[:alpha:]_][[:alnum:]_]*)*";
|
|
|
|
|
badVarNames = filter (name: matchVarName name == null) (attrNames v);
|
|
|
|
|
in
|
|
|
|
|
if asBindings then
|
|
|
|
|
generatedBindings
|
|
|
|
|
else if v == null then
|
|
|
|
|
"nil"
|
|
|
|
|
else if isInt v || isFloat v || isString v || isBool v then
|
|
|
|
|
toJSON v
|
|
|
|
|
else if isList v then
|
|
|
|
|
(
|
|
|
|
|
if v == [ ] then
|
|
|
|
|
"{}"
|
|
|
|
|
else
|
|
|
|
|
"{${introSpace}${concatItems (map (value: "${toLua innerArgs value}") v)}${outroSpace}}"
|
|
|
|
|
)
|
|
|
|
|
else if isAttrs v then
|
|
|
|
|
(
|
|
|
|
|
if isLuaInline v then
|
|
|
|
|
"(${v.expr})"
|
|
|
|
|
else if v == { } then
|
|
|
|
|
"{}"
|
|
|
|
|
else if isDerivation v then
|
|
|
|
|
''"${toString v}"''
|
|
|
|
|
else
|
|
|
|
|
"{${introSpace}${
|
|
|
|
|
concatItems (mapAttrsToList (key: value: "[${toJSON key}] = ${toLua innerArgs value}") v)
|
|
|
|
|
}${outroSpace}}"
|
|
|
|
|
)
|
|
|
|
|
else
|
|
|
|
|
abort "generators.toLua: type ${typeOf v} is unsupported";
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
Mark string as Lua expression to be inlined when processed by toLua.
|
|
|
|
|
|
|
|
|
|
Type:
|
|
|
|
|
mkLuaInline :: String -> AttrSet
|
|
|
|
|
*/
|
|
|
|
|
mkLuaInline = expr: {
|
|
|
|
|
_type = "lua-inline";
|
|
|
|
|
inherit expr;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
in
|
|
|
|
|
|
|
|
|
|
# Everything in this attrset is the public interface of the file.
|
|
|
|
|
{
|
|
|
|
|
inherit
|
|
|
|
|
mkDconfKeyValue
|
|
|
|
|
mkKeyValueDefault
|
|
|
|
|
mkLuaInline
|
|
|
|
|
mkValueStringDefault
|
|
|
|
|
toDconfINI
|
|
|
|
|
toDhall
|
|
|
|
|
toGitINI
|
|
|
|
|
toINI
|
|
|
|
|
toINIWithGlobalSection
|
|
|
|
|
toKeyValue
|
|
|
|
|
toLua
|
|
|
|
|
toPlist
|
|
|
|
|
toPretty
|
|
|
|
|
withRecursion
|
|
|
|
|
;
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
Generates JSON from an arbitrary (non-function) value.
|
|
|
|
|
* For more information see the documentation of the builtin.
|
|
|
|
|
*/
|
|
|
|
|
toJSON = { }: toJSON;
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
YAML has been a strict superset of JSON since 1.2, so we
|
|
|
|
|
* use toJSON. Before it only had a few differences referring
|
|
|
|
|
* to implicit typing rules, so it should work with older
|
|
|
|
|
* parsers as well.
|
|
|
|
|
*/
|
|
|
|
|
toYAML = { }: toJSON;
|
|
|
|
|
}
|