Skip to content

Instantly share code, notes, and snippets.

@phanirithvij
Forked from roberth/minimod.nix
Created December 17, 2024 13:31
Show Gist options
  • Save phanirithvij/0f24f0e4ad9b1c9950c80699caddf1e6 to your computer and use it in GitHub Desktop.
Save phanirithvij/0f24f0e4ad9b1c9950c80699caddf1e6 to your computer and use it in GitHub Desktop.
Simple and quick module system alternative + thoughts and tasks
/*
minimod: A stripped down module system
TODO Comparison:
- [ ] Come up with a benchmark "logic" using plain old functions and let bindings
- [ ] Write the benchmark for the module system
- [ ] Write the benchmark for POP?
- [ ] Qualitative comparison of extensibility in the context of composable
Nixpkgs packaging logic
TODO Fine-tuning:
- [ ] Try option-driven module merging
- [ ] Try the WIP features listed below
TODO Validation:
- [ ] Write actual packaging logic with this. The examples at the bottom
aren't quite realistic yet.
The decomposition into modules might already suit RFC 92 by decoupling
"package" and "derivation".
Why strip down the module system?
A module system for packaging would be awesome, but the current module system
is too slow. This is ok for configuration management, but hurts packaging
performance too much.
Why is the module system too slow?
- It has many features that we don't really need for packaging. This results
in a system that is _slightly_ too strict (not quite lazy enough) and has
a high constant factor overhead.
- Specifically in NixOS, it imports too many modules. See RFC 22.
A packaging solution based on the existing module system could be designed
to rely on "users" `imports`-ing everything they need, a la RFC 22.
If you want to go fast, you can't bring the kitchen sink.
Which module system features are included?
- mkForce / mkDefault / mkOverride
- types (incompatible for performance reasons)
- value merging
- type merging
- freeform types (module with `_wildcard` field)
Which module system features are intentionally omitted by minimod?
- import resolution (instead, provide a flat list of modules)
- disabledModules (instead, only import what you need; no global module list)
- specialArgs (without import resolution, args don't ever have to be special)
- syntax sugar including
- custom module arguments (no functionArgs quirks, yay!)
- _module.args (instead, use self)
- specialArgs (instead, use the lexical scope)
- shorthand module definitions (instead, value-only modules are the default.
Only module _types_ will be able to carry both "options" and "config" (WIP))
- mkIf (instead, use empty value, e.g. optionalAttrs)
- mkBefore (instead, use attrset if order is important, DAG type?)
- option trees (instead, nest modules)
- checks (instead, rely on testing, which is acceptable because packaging is less end-user than configuration)
- undeclared config value check
- option apply function (instead, add a new option to provide the computed value)
- all options have a value (minimod is config-driven instead of option-driven)
Why did you remove all the good parts?
Well, it's a simplification that tries to only
sacrifice as little as possible while keeping
the useful composition properties of the module
system.
Programming _and maintenance_ should feel the same,
except for the lack of bells and whistles.
I believe some features can be re-added with care. ->
Which module system features are WIP?
- combined options+config (allow module to carry a list of values to always mix in)
- extendModules (to allow exposing an overriding method not unlike `overrideAttrs`,
which can also be used for debugging, exposing internal attrs)
debugModule = moduleArgs: { package.debug.moduleArgs = moduleArgs; };
- optional checking (maybe?)
- first class documentation (seems to be worth adding; does not seem too costly)
Can this object system be rebased onto POP?
Not impossible, but probably not a net benefit.
Comparing the two, they don't seem like a great match.
POP has overlay-style overriding, whereas the module systems use priorities (`mkDefault`)
These are rather distinct solutions to the same problem that aren't really reconcilable.
Allowing both adds both cognitive and machine overhead.
Overriding is inherently about change; not very declarative. A priority system gets out of the way until you use it, whereas `super` is always present.
Can this object system be merged with the existing one?
We can have "submodule" adapters between the two. Maybe the `types` can be merged,
because having two distinct `types` isn't great.
*/
let
# nixpkgs lib
lib = import ./lib;
inherit (lib.modules) defaultPriority;
uniqueMerge = vs:
if builtins.length vs == 1
then builtins.head vs
else
throw "Only a single definition is allowed";
ignorePrio = v: if v._type or null == "override" then v.content else v;
resolvePrio = vs:
if builtins.length vs == 1
then
map ignorePrio vs
else
let
min =
lib.lists.foldl'
(min: v: if v._type or null == "override" then v.priority else defaultPriority) 1000000
vs;
in
if min == defaultPriority then
lib.filter (v: v._type or null != "override" || v.priority == defaultPriority) vs
else
lib.filter (v: v._type or null == "override" && v.priority == min) vs;
types = {
attrs = t: {
name = "attrs";
params = { inherit t; };
merge =
lib.zipAttrsWith
(name: values:
if builtins.length values == 1
then ignorePrio (builtins.head values)
else t.merge (resolvePrio values)
);
typeMerge = tys:
types.attrs (t.typeMerge (map (ty: ty.params.t) tys));
};
list = t: {
name = "list";
params = { inherit t; };
merge = lib.concatLists;
typeMerge = tys:
types.list (t.typeMerge (map (ty: ty.params.t) tys));
};
module = fields: {
name = "module";
params = { inherit fields; };
merge = rawModuleValues:
let
args = { inherit self fields; };
self =
lib.zipAttrsWith
(name: fieldValues:
let fvs = resolvePrio fieldValues;
in
if builtins.length fvs == 1
then builtins.head fvs
else
builtins.addErrorContext "in field ${name}" (
(
fields.${name}.merge or
fields._wildcard.merge or
(throw "Do not know how to merge field ${name}. Perhaps you forgot to declare it in the module, added a value to the wrong module, or mistyped the name ${name}.")
)
fvs
)
)
(map (v: lib.toFunction v args) rawModuleValues);
in
self;
typeMerge = tys:
types.module
(lib.zipAttrsWith
(name: fieldDecls:
builtins.addErrorContext "while merging module field type for ${name}" (
if builtins.length fieldDecls == 1
then builtins.head fieldDecls
else (builtins.head fieldDecls).typeMerge fieldDecls
)
)
(map (ty: ty.params.fields) tys)
);
};
int = {
name = "int";
merge = uniqueMerge;
};
sum = {
name = "sum";
merge = lib.foldl' __add 0;
};
unique = {
name = "unique";
merge = uniqueMerge;
};
package = {
name = "package";
merge = uniqueMerge;
};
};
derivation = with types; module {
derivation = attrs unique;
};
derivationMixIn = { self, ... }:
let
run = builtins.derivationStrict self.derivation;
in
{
derivationPath = run.drvPath;
# By iterating the outputs with genAttrs, we make `attrNames derivationOutputs`
# lazy in all of `derivation.*` except `derivation.outputs`
derivationOutputs = lib.genAttrs (self.derivation.outputs or [ "out" ]) (outputName: run.${outputName});
};
package = with types; module {
package = attrs unique; # freeform type?
};
stdDerivation = with types; module {
buildInputs = list package;
nativeBuildInputs = list package;
n = sum;
meta = module {
timeout = sum;
};
};
stdDerivationMixIn = { self, ... }: {
# set derivation arguments
derivation = {
name = if self?version then self.name + "-" self.version else self.name;
builder = "bash";
args = [ "setup.sh" ];
system = "x86_64-linux";
inherit (self) buildInputs nativeBuildInputs;
};
buildInputs = [ ];
nativeBuildInputs = [ ];
package = self.derivationOutputs // {
name = self.name;
drvPath = self.derivationPath;
};
};
haskellDerivation = with types; module {
haskell = module {
buildTools = list package;
};
};
haskellMixIn = { self, ... }: {
nativeBuildInputs = self.haskell.buildTools or [ ];
};
# NB: partially applied mkPackage memoizes the final fields, so it is worthwhile
# to bind it.
mkPackage = modules:
let inherit (mergeModules modules) merge;
in
mixins: (merge mixins).package;
mergeModules = (types.module { }).typeMerge;
example = mkPackage [ derivation stdDerivation haskellDerivation ] [
haskellMixIn
stdDerivationMixIn
derivationMixIn
{
name = "mypkg";
haskell.buildTools = [ "alex" ];
}
{
buildInputs = [ "libsystemd" ];
}
({ self, ... }: {
nativeBuildInputs = self.buildInputs ++ [ "gcc ${toString self.meta.timeout}" ];
})
{
buildInputs = [ "SDL2" ];
}
{
foo = lib.mkForce "foo";
bar = "bar";
meta.timeout = 1;
two = 2;
n = 1;
}
({ self, ... }: {
n = 2;
meta.timeout = self.two;
foo = "bar";
bar = lib.mkDefault "foo";
})
]
;
in
example
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment