2024-05-01 22:14:04 +00:00
# Nixpkgs/NixOS option handling.
{ lib }:
let
inherit ( lib )
all
collect
concatLists
concatMap
concatMapStringsSep
filter
foldl'
head
tail
isAttrs
isBool
isDerivation
isFunction
isInt
isList
isString
length
mapAttrs
optional
optionals
take
;
inherit ( lib . attrsets ) attrByPath optionalAttrs ;
inherit ( lib . strings ) concatMapStrings concatStringsSep ;
inherit ( lib . types ) mkOptionType ;
inherit ( lib . lists ) last ;
prioritySuggestion = ''
Use ` lib . mkForce value ` or ` lib . mkDefault value ` to change the priority on any of these definitions .
'' ;
in
rec {
/*
Returns true when the given argument is an option
Type : isOption : : a -> bool
Example :
isOption 1 // = > false
isOption ( mkOption { } ) // = > true
* /
isOption = lib . isType " o p t i o n " ;
/*
Creates an Option attribute set . mkOption accepts an attribute set with the following keys :
All keys default to ` null ` when not given .
Example :
mkOption { } // = > { _type = " o p t i o n " ; }
mkOption { default = " f o o " ; } // = > { _type = " o p t i o n " ; default = " f o o " ; }
* /
mkOption =
{
# Default value used when no definition is given in the configuration.
default ? null ,
# Textual representation of the default, for the manual.
defaultText ? null ,
# Example value used in the manual.
example ? null ,
# String describing the option.
description ? null ,
# Related packages used in the manual (see `genRelatedPackages` in ../nixos/lib/make-options-doc/default.nix).
relatedPackages ? null ,
# Option type, providing type-checking and value merging.
type ? null ,
# Function that converts the option value to something else.
apply ? null ,
# Whether the option is for NixOS developers only.
internal ? null ,
# Whether the option shows up in the manual. Default: true. Use false to hide the option and any sub-options from submodules. Use "shallow" to hide only sub-options.
visible ? null ,
# Whether the option can be set only once
readOnly ? null ,
} @ attrs :
attrs // { _type = " o p t i o n " ; } ;
/*
Creates an Option attribute set for a boolean value option i . e an
option to be toggled on or off :
Example :
mkEnableOption " f o o "
= > { _type = " o p t i o n " ; default = false ; description = " W h e t h e r t o e n a b l e f o o . " ; example = true ; type = { . . . } ; }
* /
mkEnableOption =
# Name for the created option
name :
mkOption {
default = false ;
example = true ;
description = " W h e t h e r t o e n a b l e ${ name } . " ;
type = lib . types . bool ;
} ;
2024-06-30 08:16:52 +00:00
2024-05-01 22:14:04 +00:00
/*
Creates an Option attribute set for an option that specifies the
package a module should use for some purpose .
2024-06-30 08:16:52 +00:00
2024-05-01 22:14:04 +00:00
The package is specified in the third argument under ` default ` as a list of strings
representing its attribute path in nixpkgs ( or another package set ) .
Because of this , you need to pass nixpkgs itself ( usually ` pkgs ` in a module ;
alternatively to nixpkgs itself , another package set ) as the first argument .
2024-06-30 08:16:52 +00:00
2024-05-01 22:14:04 +00:00
If you pass another package set you should set the ` pkgsText ` option .
This option is used to display the expression for the package set . It is ` " p k g s " ` by default .
If your expression is complex you should parenthesize it , as the ` pkgsText ` argument
is usually immediately followed by an attribute lookup ( ` . ` ) .
2024-06-30 08:16:52 +00:00
2024-05-01 22:14:04 +00:00
The second argument may be either a string or a list of strings .
It provides the display name of the package in the description of the generated option
( using only the last element if the passed value is a list )
and serves as the fallback value for the ` default ` argument .
2024-06-30 08:16:52 +00:00
2024-05-01 22:14:04 +00:00
To include extra information in the description , pass ` extraDescription ` to
append arbitrary text to the generated description .
2024-06-30 08:16:52 +00:00
2024-05-01 22:14:04 +00:00
You can also pass an ` example ` value , either a literal string or an attribute path .
2024-06-30 08:16:52 +00:00
2024-05-01 22:14:04 +00:00
The ` default ` argument can be omitted if the provided name is
an attribute of pkgs ( if ` name ` is a string ) or a valid attribute path in pkgs ( if ` name ` is a list ) .
You can also set ` default ` to just a string in which case it is interpreted as an attribute name
( a singleton attribute path , if you will ) .
2024-06-30 08:16:52 +00:00
2024-05-01 22:14:04 +00:00
If you wish to explicitly provide no default , pass ` null ` as ` default ` .
2024-06-30 08:16:52 +00:00
2024-05-01 22:14:04 +00:00
If you want users to be able to set no package , pass ` nullable = true ` .
In this mode a ` default = null ` will not be interpreted as no default and is interpreted literally .
2024-06-30 08:16:52 +00:00
2024-05-01 22:14:04 +00:00
Type : mkPackageOption : : pkgs -> ( string | [ string ] ) -> { nullable ? : : bool , default ? : : string | [ string ] , example ? : : null | string | [ string ] , extraDescription ? : : string , pkgsText ? : : string } -> option
2024-06-30 08:16:52 +00:00
2024-05-01 22:14:04 +00:00
Example :
mkPackageOption pkgs " h e l l o " { }
= > { . . . ; default = pkgs . hello ; defaultText = literalExpression " p k g s . h e l l o " ; description = " T h e h e l l o p a c k a g e t o u s e . " ; type = package ; }
2024-06-30 08:16:52 +00:00
2024-05-01 22:14:04 +00:00
Example :
mkPackageOption pkgs " G H C " {
default = [ " g h c " ] ;
example = " p k g s . h a s k e l l . p a c k a g e s . g h c 9 2 . g h c . w i t h P a c k a g e s ( h k g s : [ h k g s . p r i m e s ] ) " ;
}
= > { . . . ; default = pkgs . ghc ; defaultText = literalExpression " p k g s . g h c " ; description = " T h e G H C p a c k a g e t o u s e . " ; example = literalExpression " p k g s . h a s k e l l . p a c k a g e s . g h c 9 2 . g h c . w i t h P a c k a g e s ( h k g s : [ h k g s . p r i m e s ] ) " ; type = package ; }
2024-06-30 08:16:52 +00:00
2024-05-01 22:14:04 +00:00
Example :
mkPackageOption pkgs [ " p y t h o n 3 P a c k a g e s " " p y t o r c h " ] {
extraDescription = " T h i s i s a n e x a m p l e a n d d o e s n ' t a c t u a l l y d o a n y t h i n g . " ;
}
= > { . . . ; default = pkgs . python3Packages . pytorch ; defaultText = literalExpression " p k g s . p y t h o n 3 P a c k a g e s . p y t o r c h " ; description = " T h e p y t o r c h p a c k a g e t o u s e . T h i s i s a n e x a m p l e a n d d o e s n ' t a c t u a l l y d o a n y t h i n g . " ; type = package ; }
2024-06-30 08:16:52 +00:00
2024-05-01 22:14:04 +00:00
Example :
mkPackageOption pkgs " n u s h e l l " {
nullable = true ;
}
= > { . . . ; default = pkgs . nushell ; defaultText = literalExpression " p k g s . n u s h e l l " ; description = " T h e n u s h e l l p a c k a g e t o u s e . " ; type = nullOr package ; }
2024-06-30 08:16:52 +00:00
2024-05-01 22:14:04 +00:00
Example :
mkPackageOption pkgs " c o r e u t i l s " {
default = null ;
}
= > { . . . ; description = " T h e c o r e u t i l s p a c k a g e t o u s e . " ; type = package ; }
2024-06-30 08:16:52 +00:00
2024-05-01 22:14:04 +00:00
Example :
mkPackageOption pkgs " d b u s " {
nullable = true ;
default = null ;
}
= > { . . . ; default = null ; description = " T h e d b u s p a c k a g e t o u s e . " ; type = nullOr package ; }
2024-06-30 08:16:52 +00:00
2024-05-01 22:14:04 +00:00
Example :
mkPackageOption pkgs . javaPackages " O p e n J F X " {
default = " o p e n j f x 2 0 " ;
pkgsText = " p k g s . j a v a P a c k a g e s " ;
}
= > { . . . ; default = pkgs . javaPackages . openjfx20 ; defaultText = literalExpression " p k g s . j a v a P a c k a g e s . o p e n j f x 2 0 " ; description = " T h e O p e n J F X p a c k a g e t o u s e . " ; type = package ; }
* /
mkPackageOption =
# Package set (an instantiation of nixpkgs such as pkgs in modules or another package set)
pkgs :
# Name for the package, shown in option description
name :
{
# Whether the package can be null, for example to disable installing a package altogether (defaults to false)
nullable ? false ,
# The attribute path where the default package is located (may be omitted, in which case it is copied from `name`)
default ? name ,
# A string or an attribute path to use as an example (may be omitted)
example ? null ,
# Additional text to include in the option description (may be omitted)
extraDescription ? " " ,
# Representation of the package set passed as pkgs (defaults to `"pkgs"`)
pkgsText ? " p k g s " ,
} :
let
name' = if isList name then last name else name ;
default' = if isList default then default else [ default ] ;
defaultText = concatStringsSep " . " default' ;
defaultValue = attrByPath default' ( throw " ${ defaultText } c a n n o t b e f o u n d i n ${ pkgsText } " ) pkgs ;
defaults =
if default != null then
{
default = defaultValue ;
defaultText = literalExpression ( " ${ pkgsText } . " + defaultText ) ;
}
else
optionalAttrs nullable { default = null ; } ;
in
mkOption (
defaults
// {
description =
" T h e ${ name' } p a c k a g e t o u s e . " + ( if extraDescription == " " then " " else " " ) + extraDescription ;
type = with lib . types ; ( if nullable then nullOr else lib . id ) package ;
}
// optionalAttrs ( example != null ) {
example = literalExpression (
if isList example then " ${ pkgsText } . " + concatStringsSep " . " example else example
) ;
}
) ;
/*
Alias of mkPackageOption . Previously used to create options with markdown
documentation , which is no longer required .
* /
mkPackageOptionMD = mkPackageOption ;
/*
This option accepts anything , but it does not produce any result .
2024-06-30 08:16:52 +00:00
2024-05-01 22:14:04 +00:00
This is useful for sharing a module across different module sets
without having to implement similar features as long as the
values of the options are not accessed .
* /
mkSinkUndeclaredOptions =
attrs :
mkOption (
{
internal = true ;
visible = false ;
default = false ;
description = " S i n k f o r o p t i o n d e f i n i t i o n s . " ;
type = mkOptionType {
name = " s i n k " ;
check = x : true ;
merge = loc : defs : false ;
} ;
apply = x : throw " O p t i o n v a l u e i s n o t r e a d a b l e b e c a u s e t h e o p t i o n i s n o t d e c l a r e d . " ;
}
// attrs
) ;
2024-06-30 08:16:52 +00:00
2024-05-01 22:14:04 +00:00
mergeDefaultOption =
loc : defs :
let
list = getValues defs ;
in
if length list == 1 then
head list
else if all isFunction list then
x : mergeDefaultOption loc ( map ( f : f x ) list )
else if all isList list then
concatLists list
else if all isAttrs list then
foldl' lib . mergeAttrs { } list
else if all isBool list then
foldl' lib . or false list
else if all isString list then
lib . concatStrings list
else if all isInt list && all ( x : x == head list ) list then
head list
else
throw " C a n n o t m e r g e d e f i n i t i o n s o f ` ${ showOption loc } ' . D e f i n i t i o n v a l u e s : ${ showDefs defs } " ;
/*
Require a single definition .
WARNING : Does not perform nested checks , as this does not run the merge function !
* /
mergeOneOption = mergeUniqueOption { message = " " ; } ;
/*
Require a single definition .
NOTE : When the type is not checked completely by check , pass a merge function for further checking ( of sub-attributes , etc ) .
* /
mergeUniqueOption =
args @ {
message ,
# WARNING: the default merge function assumes that the definition is a valid (option) value. You MUST pass a merge function if the return value needs to be
# - type checked beyond what .check does (which should be very litte; only on the value head; not attribute values, etc)
# - if you want attribute values to be checked, or list items
# - if you want coercedTo-like behavior to work
merge ? loc : defs : ( head defs ) . value ,
} :
loc : defs :
if length defs == 1 then
merge loc defs
2024-06-30 08:16:52 +00:00
else
2024-05-01 22:14:04 +00:00
assert length defs > 1 ;
throw " T h e o p t i o n ` ${ showOption loc } ' i s d e f i n e d m u l t i p l e t i m e s w h i l e i t ' s e x p e c t e d t o b e u n i q u e . \n ${ message } \n D e f i n i t i o n v a l u e s : ${ showDefs defs } \n ${ prioritySuggestion } " ;
2024-06-30 08:16:52 +00:00
2024-05-01 22:14:04 +00:00
# "Merge" option definitions by checking that they all have the same value.
mergeEqualOption =
loc : defs :
if defs == [ ] then
abort " T h i s c a s e s h o u l d n e v e r h a p p e n . "
# Return early if we only have one element
# This also makes it work for functions, because the foldl' below would try
# to compare the first element with itself, which is false for functions
else if length defs == 1 then
( head defs ) . value
else
( foldl' (
first : def :
if def . value != first . value then
throw " T h e o p t i o n ` ${ showOption loc } ' h a s c o n f l i c t i n g d e f i n i t i o n v a l u e s : ${
showDefs [
first
def
]
} \ n $ { prioritySuggestion } "
else
first
) ( head defs ) ( tail defs ) ) . value ;
/*
Extracts values of all " v a l u e " keys of the given list .
Type : getValues : : [ { value : : a ; } ] -> [ a ]
Example :
getValues [ { value = 1 ; } { value = 2 ; } ] // = > [ 1 2 ]
getValues [ ] // = > [ ]
* /
getValues = map ( x : x . value ) ;
/*
Extracts values of all " f i l e " keys of the given list
Type : getFiles : : [ { file : : a ; } ] -> [ a ]
Example :
getFiles [ { file = " f i l e 1 " ; } { file = " f i l e 2 " ; } ] // = > [ " f i l e 1 " " f i l e 2 " ]
getFiles [ ] // = > [ ]
* /
getFiles = map ( x : x . file ) ;
# Generate documentation template from the list of option declaration like
# the set generated with filterOptionSets.
optionAttrSetToDocList = optionAttrSetToDocList' [ ] ;
optionAttrSetToDocList' =
_ : options :
concatMap (
opt :
let
name = showOption opt . loc ;
docOption =
{
loc = opt . loc ;
inherit name ;
description = opt . description or null ;
declarations = filter ( x : x != unknownModule ) opt . declarations ;
internal = opt . internal or false ;
visible = if ( opt ? visible && opt . visible == " s h a l l o w " ) then true else opt . visible or true ;
readOnly = opt . readOnly or false ;
type = opt . type . description or " u n s p e c i f i e d " ;
}
// optionalAttrs ( opt ? example ) {
example = builtins . addErrorContext " w h i l e e v a l u a t i n g t h e e x a m p l e o f o p t i o n ` ${ name } ` " (
renderOptionValue opt . example
) ;
}
// optionalAttrs ( opt ? defaultText || opt ? default ) {
default = builtins . addErrorContext " w h i l e e v a l u a t i n g t h e ${
if opt ? defaultText then " d e f a u l t T e x t " else " d e f a u l t v a l u e "
} of option ` $ { name } ` " ( r e n d e r O p t i o n V a l u e ( o p t . d e f a u l t T e x t o r o p t . d e f a u l t ) ) ;
}
// optionalAttrs ( opt ? relatedPackages && opt . relatedPackages != null ) {
inherit ( opt ) relatedPackages ;
} ;
subOptions =
let
ss = opt . type . getSubOptions opt . loc ;
in
if ss != { } then optionAttrSetToDocList' opt . loc ss else [ ] ;
subOptionsVisible = docOption . visible && opt . visible or null != " s h a l l o w " ;
in
# To find infinite recursion in NixOS option docs:
# builtins.trace opt.loc
[ docOption ] ++ optionals subOptionsVisible subOptions
) ( collect isOption options ) ;
2024-06-30 08:16:52 +00:00
/*
2024-05-01 22:14:04 +00:00
This function recursively removes all derivation attributes from
` x ` except for the ` name ` attribute .
This is to make the generation of ` options . xml ` much more
efficient : the XML representation of derivations is very large
( on the order of megabytes ) and is not actually used by the
manual generator .
This function was made obsolete by renderOptionValue and is kept for
compatibility with out-of-tree code .
* /
scrubOptionValue =
x :
if isDerivation x then
{
type = " d e r i v a t i o n " ;
drvPath = x . name ;
outPath = x . name ;
name = x . name ;
}
else if isList x then
map scrubOptionValue x
else if isAttrs x then
mapAttrs ( n : v : scrubOptionValue v ) ( removeAttrs x [ " _ a r g s " ] )
else
x ;
/*
Ensures that the given option value ( default or example ) is a ` _type ` d string
by rendering Nix values to ` literalExpression ` s .
* /
renderOptionValue =
v :
if v ? _type && v ? text then
v
else
literalExpression (
lib . generators . toPretty {
multiline = true ;
allowPrettyValues = true ;
} v
) ;
2024-06-30 08:16:52 +00:00
2024-05-01 22:14:04 +00:00
/*
For use in the ` defaultText ` and ` example ` option attributes . Causes the
given string to be rendered verbatim in the documentation as Nix code . This
is necessary for complex values , e . g . functions , or values that depend on
other values or packages .
* /
literalExpression =
text :
if ! isString text then
throw " l i t e r a l E x p r e s s i o n e x p e c t s a s t r i n g . "
else
{
_type = " l i t e r a l E x p r e s s i o n " ;
inherit text ;
} ;
literalExample = lib . warn " l i b . l i t e r a l E x a m p l e i s d e p r e c a t e d , u s e l i b . l i t e r a l E x p r e s s i o n i n s t e a d , o r u s e l i b . l i t e r a l M D f o r a n o n - N i x d e s c r i p t i o n . " literalExpression ;
/*
Transition marker for documentation that's already migrated to markdown
syntax . Has been a no-op for some while and been removed from nixpkgs .
Kept here to alert downstream users who may not be aware of the migration's
completion that it should be removed from modules .
* /
mdDoc = lib . warn " l i b . m d D o c w i l l b e r e m o v e d f r o m n i x p k g s i n 2 4 . 1 1 . O p t i o n d e s c r i p t i o n s a r e n o w i n M a r k d o w n b y d e f a u l t ; y o u c a n r e m o v e a n y r e m a i n i n g u s e s o f l i b . m d D o c . " ;
/*
For use in the ` defaultText ` and ` example ` option attributes . Causes the
given MD text to be inserted verbatim in the documentation , for when
a ` literalExpression ` would be too hard to read .
* /
literalMD =
text :
if ! isString text then
throw " l i t e r a l M D e x p e c t s a s t r i n g . "
else
{
_type = " l i t e r a l M D " ;
inherit text ;
} ;
# Helper functions.
/*
Convert an option , described as a list of the option parts to a
human-readable version .
Example :
( showOption [ " f o o " " b a r " " b a z " ] ) == " f o o . b a r . b a z "
( showOption [ " f o o " " b a r . b a z " " t u x " ] ) == " f o o . \" b a r . b a z \" . t u x "
( showOption [ " w i n d o w M a n a g e r " " 2 b w m " " e n a b l e " ] ) == " w i n d o w M a n a g e r . \" 2 b w m \" . e n a b l e "
Placeholders will not be quoted as they are not actual values :
( showOption [ " f o o " " * " " b a r " ] ) == " f o o . * . b a r "
( showOption [ " f o o " " < n a m e > " " b a r " ] ) == " f o o . < n a m e > . b a r "
* /
showOption =
parts :
let
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 = [
" < n a m e > " # attrsOf (submodule {})
" * " # listOf (submodule {})
" < f u n c t i o n b o d y > " # functionTo
] ;
in
if builtins . elem part specialIdentifiers then part else lib . strings . escapeNixIdentifier part ;
in
( concatStringsSep " . " ) ( map escapeOptionPart parts ) ;
showFiles = files : concatStringsSep " a n d " ( map ( f : " ` ${ f } ' " ) files ) ;
showDefs =
defs :
concatMapStrings (
def :
let
# Pretty print the value for display, if successful
prettyEval = builtins . tryEval (
lib . generators . toPretty { } (
lib . generators . withRecursion {
depthLimit = 10 ;
throwOnDepthLimit = false ;
} def . value
2024-06-30 08:16:52 +00:00
)
2024-05-01 22:14:04 +00:00
) ;
# Split it into its lines
lines = filter ( v : ! isList v ) ( builtins . split " \n " prettyEval . value ) ;
# Only display the first 5 lines, and indent them for better visibility
value = concatStringsSep " \n " ( take 5 lines ++ optional ( length lines > 5 ) " . . . " ) ;
result =
# Don't print any value if evaluating the value strictly fails
if ! prettyEval . success then
" "
# Put it on a new line if it consists of multiple
else if length lines > 1 then
" : \n " + value
else
" : " + value ;
in
" \n - I n ` ${ def . file } ' ${ result } "
) defs ;
showOptionWithDefLocs = opt : ''
$ { showOption opt . loc } , with values defined in :
$ { concatMapStringsSep " \n " ( defFile : " - ${ defFile } " ) opt . files }
'' ;
unknownModule = " < u n k n o w n - f i l e > " ;
}