# Options Types {#sec-option-types} Option types are a way to put constraints on the values a module option can take. Types are also responsible of how values are merged in case of multiple value definitions. ## Basic Types {#sec-option-types-basic} Basic types are the simplest available types in the module system. Basic types include multiple string types that mainly differ in how definition merging is handled. `types.bool` : A boolean, its values can be `true` or `false`. `types.path` : A filesystem path is anything that starts with a slash when coerced to a string. Even if derivations can be considered as paths, the more specific `types.package` should be preferred. `types.package` : A top-level store path. This can be an attribute set pointing to a store path, like a derivation or a flake input. `types.anything` : A type that accepts any value and recursively merges attribute sets together. This type is recommended when the option type is unknown. ::: {#ex-types-anything .example} ::: {.title} **Example: `types.anything` Example** ::: Two definitions of this type like ```nix { str = lib.mkDefault "foo"; pkg.hello = pkgs.hello; fun.fun = x: x + 1; } ``` ```nix { str = lib.mkIf true "bar"; pkg.gcc = pkgs.gcc; fun.fun = lib.mkForce (x: x + 2); } ``` will get merged to ```nix { str = "bar"; pkg.gcc = pkgs.gcc; pkg.hello = pkgs.hello; fun.fun = x: x + 2; } ``` ::: `types.raw` : A type which doesn't do any checking, merging or nested evaluation. It accepts a single arbitrary value that is not recursed into, making it useful for values coming from outside the module system, such as package sets or arbitrary data. Options of this type are still evaluated according to priorities and conditionals, so `mkForce`, `mkIf` and co. still work on the option value itself, but not for any value nested within it. This type should only be used when checking, merging and nested evaluation are not desirable. `types.optionType` : The type of an option's type. Its merging operation ensures that nested options have the correct file location annotated, and that if possible, multiple option definitions are correctly merged together. The main use case is as the type of the `_module.freeformType` option. `types.attrs` : A free-form attribute set. ::: {.warning} This type will be deprecated in the future because it doesn\'t recurse into attribute sets, silently drops earlier attribute definitions, and doesn\'t discharge `lib.mkDefault`, `lib.mkIf` and co. For allowing arbitrary attribute sets, prefer `types.attrsOf types.anything` instead which doesn\'t have these problems. ::: Integer-related types: `types.int` : A signed integer. `types.ints.{s8, s16, s32}` : Signed integers with a fixed length (8, 16 or 32 bits). They go from −2^n/2 to 2^n/2−1 respectively (e.g. `−128` to `127` for 8 bits). `types.ints.unsigned` : An unsigned integer (that is >= 0). `types.ints.{u8, u16, u32}` : Unsigned integers with a fixed length (8, 16 or 32 bits). They go from 0 to 2^n−1 respectively (e.g. `0` to `255` for 8 bits). `types.ints.positive` : A positive integer (that is > 0). `types.port` : A port number. This type is an alias to `types.ints.u16`. String-related types: `types.str` : A string. Multiple definitions cannot be merged. `types.lines` : A string. Multiple definitions are concatenated with a new line `"\n"`. `types.commas` : A string. Multiple definitions are concatenated with a comma `","`. `types.envVar` : A string. Multiple definitions are concatenated with a collon `":"`. `types.strMatching` : A string matching a specific regular expression. Multiple definitions cannot be merged. The regular expression is processed using `builtins.match`. ## Value Types {#sec-option-types-value} Value types are types that take a value parameter. `types.enum` *`l`* : One element of the list *`l`*, e.g. `types.enum [ "left" "right" ]`. Multiple definitions cannot be merged. `types.separatedString` *`sep`* : A string with a custom separator *`sep`*, e.g. `types.separatedString "|"`. `types.ints.between` *`lowest highest`* : An integer between *`lowest`* and *`highest`* (both inclusive). Useful for creating types like `types.port`. `types.submodule` *`o`* : A set of sub options *`o`*. *`o`* can be an attribute set, a function returning an attribute set, or a path to a file containing such a value. Submodules are used in composed types to create modular options. This is equivalent to `types.submoduleWith { modules = toList o; shorthandOnlyDefinesConfig = true; }`. Submodules are detailed in [Submodule](#section-option-types-submodule). `types.submoduleWith` { *`modules`*, *`specialArgs`* ? {}, *`shorthandOnlyDefinesConfig`* ? false } : Like `types.submodule`, but more flexible and with better defaults. It has parameters - *`modules`* A list of modules to use by default for this submodule type. This gets combined with all option definitions to build the final list of modules that will be included. ::: {.note} Only options defined with this argument are included in rendered documentation. ::: - *`specialArgs`* An attribute set of extra arguments to be passed to the module functions. The option `_module.args` should be used instead for most arguments since it allows overriding. *`specialArgs`* should only be used for arguments that can\'t go through the module fixed-point, because of infinite recursion or other problems. An example is overriding the `lib` argument, because `lib` itself is used to define `_module.args`, which makes using `_module.args` to define it impossible. - *`shorthandOnlyDefinesConfig`* Whether definitions of this type should default to the `config` section of a module (see [Example: Structure of NixOS Modules](#ex-module-syntax)) if it is an attribute set. Enabling this only has a benefit when the submodule defines an option named `config` or `options`. In such a case it would allow the option to be set with `the-submodule.config = "value"` instead of requiring `the-submodule.config.config = "value"`. This is because only when modules *don\'t* set the `config` or `options` keys, all keys are interpreted as option definitions in the `config` section. Enabling this option implicitly puts all attributes in the `config` section. With this option enabled, defining a non-`config` section requires using a function: `the-submodule = { ... }: { options = { ... }; }`. `types.deferredModule` : Whereas `submodule` represents an option tree, `deferredModule` represents a module value, such as a module file or a configuration. It can be set multiple times. Module authors can use its value in `imports`, in `submoduleWith`'s `modules` or in `evalModules`' `modules` parameter, among other places. Note that `imports` must be evaluated before the module fixpoint. Because of this, deferred modules can only be imported into "other" fixpoints, such as submodules. One use case for this type is the type of a "default" module that allow the user to affect all submodules in an `attrsOf submodule` at once. This is more convenient and discoverable than expecting the module user to type-merge with the `attrsOf submodule` option. ## Composed Types {#sec-option-types-composed} Composed types are types that take a type as parameter. `listOf int` and `either int str` are examples of composed types. `types.listOf` *`t`* : A list of *`t`* type, e.g. `types.listOf int`. Multiple definitions are merged with list concatenation. `types.attrsOf` *`t`* : An attribute set of where all the values are of *`t`* type. Multiple definitions result in the joined attribute set. ::: {.note} This type is *strict* in its values, which in turn means attributes cannot depend on other attributes. See ` types.lazyAttrsOf` for a lazy version. ::: `types.lazyAttrsOf` *`t`* : An attribute set of where all the values are of *`t`* type. Multiple definitions result in the joined attribute set. This is the lazy version of `types.attrsOf `, allowing attributes to depend on each other. ::: {.warning} This version does not fully support conditional definitions! With an option `foo` of this type and a definition `foo.attr = lib.mkIf false 10`, evaluating `foo ? attr` will return `true` even though it should be false. Accessing the value will then throw an error. For types *`t`* that have an `emptyValue` defined, that value will be returned instead of throwing an error. So if the type of `foo.attr` was `lazyAttrsOf (nullOr int)`, `null` would be returned instead for the same `mkIf false` definition. ::: `types.nullOr` *`t`* : `null` or type *`t`*. Multiple definitions are merged according to type *`t`*. `types.uniq` *`t`* : Ensures that type *`t`* cannot be merged. It is used to ensure option definitions are declared only once. `types.unique` `{ message = m }` *`t`* : Ensures that type *`t`* cannot be merged. Prints the message *`m`*, after the line `The option