Files
snowfall-lib/snowfall-lib/default.nix
anntnzrb 70ad5e14ae style(nix): apply nix fmt
Normalize formatting with the treefmt-based nix fmt configuration.
2026-03-16 14:30:34 -05:00

148 lines
4.2 KiB
Nix

# NOTE: The role of this file is to bootstrap the
# Snowfall library. There is some duplication shared between this
# file and the library itself due to the library needing to pass through
# another extended library for its own applications.
core-inputs: user-options:
let
raw-snowfall-config = user-options.snowfall or { };
snowfall-config = raw-snowfall-config // {
inherit (user-options) src;
root = raw-snowfall-config.root or user-options.src;
namespace = raw-snowfall-config.namespace or "internal";
meta = {
name = raw-snowfall-config.meta.name or null;
title = raw-snowfall-config.meta.title or null;
};
};
user-inputs = user-options.inputs // {
inherit (user-options) src;
};
inherit (core-inputs.nixpkgs.lib)
assertMsg
fix
filterAttrs
mergeAttrs
foldr
recursiveUpdate
callPackageWith
isFunction
;
# Recursively merge a list of attribute sets.
# Type: [Attrs] -> Attrs
# Usage: merge-deep [{ x = 1; } { x = 2; }]
# result: { x = 2; }
merge-deep = foldr recursiveUpdate { };
# Merge the root of a list of attribute sets.
# Type: [Attrs] -> Attrs
# Usage: merge-shallow [{ x = 1; } { x = 2; }]
# result: { x = 2; }
merge-shallow = foldr mergeAttrs { };
# Transform an attribute set of inputs into an attribute set where
# the values are the inputs' `lib` attribute. Entries without a `lib`
# attribute are removed.
# Type: Attrs -> Attrs
# Usage: get-lib { x = nixpkgs; y = {}; }
# result: { x = nixpkgs.lib; }
get-libs =
attrs:
foldr (
name: acc:
let
value = attrs.${name};
in
if builtins.isAttrs (value.lib or null) then acc // { ${name} = value.lib; } else acc
) { } (builtins.attrNames attrs);
# Remove the `self` attribute from an attribute set.
# Type: Attrs -> Attrs
# Usage: without-self { self = {}; x = true; }
# result: { x = true; }
without-self = attrs: builtins.removeAttrs attrs [ "self" ];
core-inputs-libs = get-libs (without-self core-inputs);
user-inputs-libs = get-libs (without-self user-inputs);
# NOTE: This root is different to accommodate the creation
# of a fake user-lib in order to run documentation on this flake.
snowfall-lib-root = "${core-inputs.src}/snowfall-lib";
snowfall-lib-dirs =
let
files = builtins.readDir snowfall-lib-root;
dirs = filterAttrs (_name: kind: kind == "directory") files;
names = builtins.attrNames dirs;
in
names;
snowfall-lib = fix (
snowfall-lib:
let
attrs = {
inherit
snowfall-lib
snowfall-config
core-inputs
user-inputs
;
};
libs = builtins.map (dir: import "${snowfall-lib-root}/${dir}" attrs) snowfall-lib-dirs;
in
merge-deep libs
);
snowfall-top-level-lib = filterAttrs (_name: value: !builtins.isAttrs value) snowfall-lib;
base-lib = merge-shallow [
core-inputs.nixpkgs.lib
core-inputs-libs
user-inputs-libs
snowfall-top-level-lib
{ snowfall = snowfall-lib; }
];
user-lib-root = "${user-inputs.src}/lib";
user-lib-modules = snowfall-lib.fs.get-default-nix-files-recursive user-lib-root;
user-lib = fix (
user-lib:
let
attrs = {
inherit (user-options) inputs;
snowfall-inputs = core-inputs;
inherit (snowfall-config) namespace;
lib = merge-shallow [
base-lib
{ ${snowfall-config.namespace} = user-lib; }
];
};
libs = builtins.map (
path:
let
imported-module = import path;
in
if isFunction imported-module then
callPackageWith attrs path { }
# the only difference is that there is no `override` and `overrideDerivation` on returned value
else
imported-module
) user-lib-modules;
in
merge-deep libs
);
lib = merge-deep [
base-lib
user-lib
];
user-inputs-has-self = builtins.elem "self" (builtins.attrNames user-inputs);
user-inputs-has-src = builtins.elem "src" (builtins.attrNames user-inputs);
in
assert (assertMsg user-inputs-has-self "Missing attribute `self` for mkLib.");
assert (assertMsg user-inputs-has-src "Missing attribute `src` for mkLib.");
lib