diff --git a/lib/options.nix b/lib/options.nix index 4f75da5ad51a83..5b22b1b37b86d1 100644 --- a/lib/options.nix +++ b/lib/options.nix @@ -420,20 +420,17 @@ rec { Placeholders will not be quoted as they are not actual values: (showOption ["foo" "*" "bar"]) == "foo.*.bar" (showOption ["foo" "" "bar"]) == "foo..bar" + (showOption ["foo" "" "bar"]) == "foo..bar" */ showOption = parts: let + # If the part is a named placeholder of the form "<...>" don't escape it. + # It may cause misleading escaping if somebody uses literally "<...>" in their option names. + # This is the trade-off to allow for placeholders in option names. + isNamedPlaceholder = builtins.match "\<(.*)\>"; escapeOptionPart = part: - let - # We assume that these are "special values" and not real configuration data. - # If it is real configuration data, it is rendered incorrectly. - specialIdentifiers = [ - "" # attrsOf (submodule {}) - "*" # listOf (submodule {}) - "" # functionTo - ]; - in if builtins.elem part specialIdentifiers - then part - else lib.strings.escapeNixIdentifier part; + if part == "*" || isNamedPlaceholder part != null + then part + else lib.strings.escapeNixIdentifier part; in (concatStringsSep ".") (map escapeOptionPart parts); showFiles = files: concatStringsSep " and " (map (f: "`${f}'") files); diff --git a/lib/tests/misc.nix b/lib/tests/misc.nix index 6357cc3ec1cf51..989733a25089e7 100644 --- a/lib/tests/misc.nix +++ b/lib/tests/misc.nix @@ -1877,6 +1877,44 @@ runTests { expected = [ [ "_module" "args" ] [ "foo" ] [ "foo" "" "bar" ] [ "foo" "bar" ] ]; }; + testAttrsWithName = { + expr = let + eval = evalModules { + modules = [ + { + options = { + foo = lib.mkOption { + type = lib.types.attrsWith { + placeholder = "MyCustomPlaceholder"; + elemType = lib.types.submodule { + options.bar = lib.mkOption { + type = lib.types.int; + default = 42; + }; + }; + }; + }; + }; + } + ]; + }; + opt = eval.options.foo; + in + (opt.type.getSubOptions opt.loc).bar.loc; + expected = [ + "foo" + "" + "bar" + ]; + }; + + testShowOptionWithPlaceholder = { + # , *, should now be escaped. It is used as a placeholder by convention. + # Other symbols should be escaped. `{}` + expr = lib.showOption ["" "" "*" "{foo}"]; + expected = "..*.\"{foo}\""; + }; + testCartesianProductOfEmptySet = { expr = cartesianProduct {}; expected = [ {} ]; diff --git a/lib/tests/modules.sh b/lib/tests/modules.sh index 4c00ecaab605bf..dd3c1e573abdb1 100755 --- a/lib/tests/modules.sh +++ b/lib/tests/modules.sh @@ -391,6 +391,10 @@ checkConfigError 'The option `mergedLazyNonLazy'\'' in `.*'\'' is already declar checkConfigOutput '^11$' config.lazyResult ./lazy-attrsWith.nix checkConfigError 'infinite recursion encountered' config.nonLazyResult ./lazy-attrsWith.nix +# AttrsWith placeholder tests +checkConfigOutput '^"mergedName..nested"$' config.result ./name-merge-attrsWith-1.nix +checkConfigError 'The option .mergedName. in .*\.nix. is already declared in .*\.nix' config.mergedName ./name-merge-attrsWith-2.nix + # Even with multiple assignments, a type error should be thrown if any of them aren't valid checkConfigError 'A definition for option .* is not of type .*' \ config.value ./declare-int-unsigned-value.nix ./define-value-list.nix ./define-value-int-positive.nix diff --git a/lib/tests/modules/name-merge-attrsWith-1.nix b/lib/tests/modules/name-merge-attrsWith-1.nix new file mode 100644 index 00000000000000..6b8fd9cb5b1403 --- /dev/null +++ b/lib/tests/modules/name-merge-attrsWith-1.nix @@ -0,0 +1,51 @@ +{ lib, ... }: +let + inherit (lib) types mkOption; +in +{ + imports = [ + # Module A + ( + { ... }: + { + options.mergedName = mkOption { + default = { }; + type = types.attrsWith { + placeholder = "id"; # <- define placeholder = "id" + elemType = types.submodule { + options.nested = mkOption { + type = types.int; + default = 1; + }; + }; + }; + }; + } + ) + # Module B + ( + { ... }: + { + # defines the default placeholder "name" + # type merging should resolve to "id" + options.mergedName = mkOption { + type = types.attrsOf (types.submodule { }); + }; + } + ) + + # Output + ( + { + options, + ... + }: + { + options.result = mkOption { + default = lib.concatStringsSep "." (options.mergedName.type.getSubOptions options.mergedName.loc) + .nested.loc; + }; + } + ) + ]; +} diff --git a/lib/tests/modules/name-merge-attrsWith-2.nix b/lib/tests/modules/name-merge-attrsWith-2.nix new file mode 100644 index 00000000000000..9f2c7331ddd617 --- /dev/null +++ b/lib/tests/modules/name-merge-attrsWith-2.nix @@ -0,0 +1,38 @@ +{ lib, ... }: +let + inherit (lib) types mkOption; +in +{ + imports = [ + # Module A + ( + { ... }: + { + options.mergedName = mkOption { + default = { }; + type = types.attrsWith { + placeholder = "id"; # <- define placeholder = "id" + elemType = types.submodule { + options.nested = mkOption { + type = types.int; + default = 1; + }; + }; + }; + }; + } + ) + # Module B + ( + { ... }: + { + options.mergedName = mkOption { + type = types.attrsWith { + placeholder = "other"; # <- define placeholder = "other" (conflict) + elemType = types.submodule { }; + }; + }; + } + ) + ]; +} diff --git a/lib/types.nix b/lib/types.nix index dd1e8ab9d01652..097ab1b42dcbeb 100644 --- a/lib/types.nix +++ b/lib/types.nix @@ -608,17 +608,27 @@ rec { lhs.lazy else null; + placeholder = + if lhs.placeholder == rhs.placeholder then + lhs.placeholder + else if lhs.placeholder == "name" then + rhs.placeholder + else if rhs.placeholder == "name" then + lhs.placeholder + else + null; in - if elemType == null || lazy == null then + if elemType == null || lazy == null || placeholder == null then null else { - inherit elemType lazy; + inherit elemType lazy placeholder; }; in { elemType, lazy ? false, + placeholder ? "name", }: mkOptionType { name = if lazy then "lazyAttrsOf" else "attrsOf"; @@ -645,16 +655,16 @@ rec { (pushPositions defs))) ); emptyValue = { value = {}; }; - getSubOptions = prefix: elemType.getSubOptions (prefix ++ [""]); + getSubOptions = prefix: elemType.getSubOptions (prefix ++ ["<${placeholder}>"]); getSubModules = elemType.getSubModules; - substSubModules = m: attrsWith { elemType = elemType.substSubModules m; inherit lazy; }; + substSubModules = m: attrsWith { elemType = elemType.substSubModules m; inherit lazy placeholder; }; functor = defaultFunctor "attrsWith" // { wrappedDeprecationMessage = { loc }: lib.warn '' The deprecated `type.functor.wrapped` attribute of the option `${showOption loc}` is accessed, use `type.nestedTypes.elemType` instead. '' elemType; payload = { # Important!: Add new function attributes here in case of future changes - inherit elemType lazy; + inherit elemType lazy placeholder; }; inherit binOp; }; diff --git a/nixos/doc/manual/development/option-types.section.md b/nixos/doc/manual/development/option-types.section.md index a9c8f2f8a88dcd..814c38657d1b95 100644 --- a/nixos/doc/manual/development/option-types.section.md +++ b/nixos/doc/manual/development/option-types.section.md @@ -411,10 +411,22 @@ Composed types are types that take a type as parameter. `listOf `lazy` : Determines whether the attribute set is lazily evaluated. See: `types.lazyAttrsOf` + `placeholder` (`String`, default: `name` ) + : Placeholder string in documentation for the attribute names. + The default value `name` results in the placeholder `` + **Behavior** - `attrsWith { elemType = t; }` is equivalent to `attrsOf t` - `attrsWith { lazy = true; elemType = t; }` is equivalent to `lazyAttrsOf t` + - `attrsWith { placeholder = "id"; elemType = t; }` + Allows to customize how the option type is displayed + For example `settings...` instead of `settings...` + + ::: {.note} + This is the underlying implementation of `types.attrsOf` and `types.lazyAttrsOf` + ::: + `types.uniq` *`t`*