2021-10-27 09:05:52 +00:00
|
|
|
{ lib }:
|
|
|
|
let
|
|
|
|
inherit (lib)
|
|
|
|
mkOption
|
2022-05-25 16:09:17 +00:00
|
|
|
mkOptionType
|
|
|
|
defaultFunctor
|
|
|
|
isAttrs
|
|
|
|
isFunction
|
|
|
|
showOption
|
2022-12-17 17:36:15 +00:00
|
|
|
throwIf
|
2021-10-27 09:05:52 +00:00
|
|
|
types
|
2022-12-17 17:36:15 +00:00
|
|
|
warnIf
|
2022-10-16 20:07:14 +00:00
|
|
|
getAttrFromPath
|
|
|
|
setAttrByPath
|
|
|
|
attrByPath
|
|
|
|
optionalAttrs
|
2021-10-27 09:05:52 +00:00
|
|
|
;
|
2022-10-16 20:07:14 +00:00
|
|
|
inherit (lib.modules)
|
|
|
|
mkAliasAndWrapDefsWithPriority;
|
2022-05-25 16:09:17 +00:00
|
|
|
inherit (lib.types)
|
|
|
|
path
|
|
|
|
submoduleWith
|
|
|
|
;
|
2021-10-27 09:05:52 +00:00
|
|
|
|
2022-05-17 21:20:53 +00:00
|
|
|
# Polyfill functionTo to make sure it has type merging.
|
|
|
|
# Remove 2022-12
|
|
|
|
functionTo =
|
|
|
|
let sample = types.functionTo lib.types.str;
|
|
|
|
in
|
|
|
|
if sample.functor.wrapped._type or null == "option-type"
|
|
|
|
then types.functionTo
|
|
|
|
else
|
|
|
|
elemType: lib.mkOptionType {
|
|
|
|
name = "functionTo";
|
|
|
|
description = "function that evaluates to a(n) ${elemType.description}";
|
|
|
|
check = lib.isFunction;
|
|
|
|
merge = loc: defs:
|
|
|
|
fnArgs: (lib.mergeDefinitions (loc ++ [ "[function body]" ]) elemType (map (fn: { inherit (fn) file; value = fn.value fnArgs; }) defs)).mergedValue;
|
|
|
|
getSubOptions = prefix: elemType.getSubOptions (prefix ++ [ "[function body]" ]);
|
|
|
|
getSubModules = elemType.getSubModules;
|
|
|
|
substSubModules = m: functionTo (elemType.substSubModules m);
|
|
|
|
functor = (lib.defaultFunctor "functionTo") // { type = functionTo; wrapped = elemType; };
|
|
|
|
nestedTypes.elemType = elemType;
|
|
|
|
};
|
|
|
|
|
2022-05-25 16:09:17 +00:00
|
|
|
# Polyfill https://github.com/NixOS/nixpkgs/pull/163617
|
|
|
|
deferredModuleWith = lib.deferredModuleWith or (
|
|
|
|
attrs@{ staticModules ? [ ] }: mkOptionType {
|
|
|
|
name = "deferredModule";
|
|
|
|
description = "module";
|
|
|
|
check = x: isAttrs x || isFunction x || path.check x;
|
|
|
|
merge = loc: defs: staticModules ++ map (def: lib.setDefaultModuleLocation "${def.file}, via option ${showOption loc}" def.value) defs;
|
|
|
|
inherit (submoduleWith { modules = staticModules; })
|
|
|
|
getSubOptions
|
|
|
|
getSubModules;
|
|
|
|
substSubModules = m: deferredModuleWith (attrs // {
|
|
|
|
staticModules = m;
|
|
|
|
});
|
|
|
|
functor = defaultFunctor "deferredModuleWith" // {
|
|
|
|
type = deferredModuleWith;
|
|
|
|
payload = {
|
|
|
|
inherit staticModules;
|
|
|
|
};
|
|
|
|
binOp = lhs: rhs: {
|
|
|
|
staticModules = lhs.staticModules ++ rhs.staticModules;
|
|
|
|
};
|
|
|
|
};
|
|
|
|
}
|
|
|
|
);
|
|
|
|
|
2022-12-17 17:36:15 +00:00
|
|
|
errorExample = ''
|
|
|
|
For example:
|
|
|
|
|
|
|
|
outputs = inputs@{ flake-parts, ... }:
|
|
|
|
flake-parts.lib.mkFlake { inherit inputs; } { /* module */ };
|
|
|
|
|
|
|
|
To avoid an infinite recursion, *DO NOT* pass `self.inputs` and
|
|
|
|
*DO NOT* pass `inherit (self) inputs`, but pass the output function
|
|
|
|
arguments as `inputs` like above.
|
|
|
|
'';
|
2022-05-25 16:09:17 +00:00
|
|
|
|
2022-05-25 14:36:33 +00:00
|
|
|
flake-parts-lib = {
|
2021-10-27 09:05:52 +00:00
|
|
|
evalFlakeModule =
|
2022-12-17 17:36:15 +00:00
|
|
|
args@
|
|
|
|
{ inputs ? self.inputs
|
2021-10-27 09:05:52 +00:00
|
|
|
, specialArgs ? { }
|
2022-12-17 17:36:15 +00:00
|
|
|
|
|
|
|
# legacy
|
|
|
|
, self ? inputs.self or (throw ''
|
|
|
|
When invoking flake-parts, you must pass all the flake output arguments,
|
|
|
|
and not just `self.inputs`.
|
|
|
|
|
|
|
|
${errorExample}
|
|
|
|
'')
|
2021-10-27 09:05:52 +00:00
|
|
|
}:
|
2022-12-17 17:36:15 +00:00
|
|
|
throwIf
|
|
|
|
(!args?self && !args?inputs) ''
|
|
|
|
When invoking flake-parts, you must pass in the flake output arguments.
|
2021-10-27 09:05:52 +00:00
|
|
|
|
2022-12-17 17:36:15 +00:00
|
|
|
${errorExample}
|
|
|
|
''
|
|
|
|
warnIf
|
|
|
|
(!args?inputs) ''
|
|
|
|
When invoking flake-parts, it is recommended to pass all the flake output
|
|
|
|
arguments in the `inputs` parameter. If you only pass `self`, it's not
|
|
|
|
possible to use the `inputs` module argument in the module `imports`.
|
|
|
|
|
|
|
|
Please pass the output function arguments. ${errorExample}
|
|
|
|
''
|
|
|
|
|
|
|
|
(module:
|
|
|
|
lib.evalModules {
|
|
|
|
specialArgs = {
|
|
|
|
inherit self flake-parts-lib;
|
|
|
|
inputs = args.inputs or /* legacy, warned above */ self.inputs;
|
|
|
|
} // specialArgs;
|
|
|
|
modules = [ ./all-modules.nix module ];
|
|
|
|
}
|
|
|
|
);
|
2021-11-21 14:47:11 +00:00
|
|
|
|
2022-05-18 11:59:21 +00:00
|
|
|
mkFlake = args: module:
|
2022-05-25 14:36:33 +00:00
|
|
|
(flake-parts-lib.evalFlakeModule args module).config.flake;
|
2022-05-18 11:59:21 +00:00
|
|
|
|
2021-11-21 14:47:11 +00:00
|
|
|
# For extending options in an already declared submodule.
|
|
|
|
# Workaround for https://github.com/NixOS/nixpkgs/issues/146882
|
|
|
|
mkSubmoduleOptions =
|
|
|
|
options:
|
|
|
|
mkOption {
|
|
|
|
type = types.submoduleWith {
|
2022-05-17 08:28:03 +00:00
|
|
|
modules = [{ inherit options; }];
|
2021-11-21 14:47:11 +00:00
|
|
|
};
|
|
|
|
};
|
2022-05-17 08:28:03 +00:00
|
|
|
|
2022-05-13 08:14:10 +00:00
|
|
|
mkPerSystemType =
|
|
|
|
module:
|
2022-05-25 16:09:17 +00:00
|
|
|
deferredModuleWith {
|
|
|
|
staticModules = [ module ];
|
|
|
|
};
|
2022-05-13 08:14:10 +00:00
|
|
|
|
2022-05-13 08:15:21 +00:00
|
|
|
mkPerSystemOption =
|
2022-05-13 08:14:10 +00:00
|
|
|
module:
|
|
|
|
mkOption {
|
2022-05-25 14:36:33 +00:00
|
|
|
type = flake-parts-lib.mkPerSystemType module;
|
2022-05-13 08:14:10 +00:00
|
|
|
};
|
|
|
|
|
2022-10-30 10:52:28 +00:00
|
|
|
# Helper function for defining a per-system option that
|
|
|
|
# gets transposed by the usual flake system logic to a
|
|
|
|
# top-level flake attribute.
|
|
|
|
mkTransposedPerSystemModule = { name, option, file }: {
|
|
|
|
_file = file;
|
|
|
|
|
|
|
|
options = {
|
|
|
|
flake = flake-parts-lib.mkSubmoduleOptions {
|
|
|
|
${name} = mkOption {
|
|
|
|
type = types.lazyAttrsOf option.type;
|
|
|
|
default = { };
|
2022-11-11 06:39:25 +00:00
|
|
|
description = ''
|
2022-10-30 10:52:28 +00:00
|
|
|
See {option}`perSystem.${name}` for description and examples.
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
|
|
|
perSystem = flake-parts-lib.mkPerSystemOption {
|
|
|
|
_file = file;
|
|
|
|
|
|
|
|
options.${name} = option;
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
|
|
|
config = {
|
|
|
|
transposition.${name} = { };
|
|
|
|
};
|
|
|
|
};
|
2022-10-16 20:07:14 +00:00
|
|
|
|
|
|
|
# Needed pending https://github.com/NixOS/nixpkgs/pull/198450
|
|
|
|
mkAliasOptionModule = from: to: { config, options, ... }:
|
|
|
|
let
|
|
|
|
fromOpt = getAttrFromPath from options;
|
|
|
|
toOf = attrByPath to
|
|
|
|
(abort "Renaming error: option `${showOption to}' does not exist.");
|
|
|
|
toType = let opt = attrByPath to { } options; in opt.type or (types.submodule { });
|
|
|
|
in
|
|
|
|
{
|
|
|
|
options = setAttrByPath from (mkOption
|
|
|
|
{
|
|
|
|
visible = true;
|
|
|
|
description = lib.mdDoc "Alias of {option}`${showOption to}`.";
|
|
|
|
apply = x: (toOf config);
|
|
|
|
} // optionalAttrs (toType != null) {
|
|
|
|
type = toType;
|
|
|
|
});
|
|
|
|
config = (mkAliasAndWrapDefsWithPriority (setAttrByPath to) fromOpt);
|
|
|
|
};
|
2021-10-27 09:05:52 +00:00
|
|
|
};
|
2021-11-21 14:47:11 +00:00
|
|
|
|
2021-10-27 09:05:52 +00:00
|
|
|
in
|
2022-05-25 14:36:33 +00:00
|
|
|
flake-parts-lib
|