soonix/lib/module.nix

258 lines
7.4 KiB
Nix
Raw Normal View History

2025-08-25 16:45:22 +02:00
{
pkgs,
config,
lib,
...
}: let
inherit (lib) types mkOption concatMapStringsSep;
soonix_lib = import ./. {inherit pkgs;};
inherit (soonix_lib) engines buildAllFiles;
in {
options = {
hooks = mkOption {
type = types.attrsOf (types.submodule ({
name,
config,
...
}: {
options = {
name = mkOption {
type = types.str;
internal = true;
default = name;
};
output = mkOption {
type = types.str;
description = "The relative path where the generated file should be placed";
};
generator = mkOption {
type = types.enum ["nix" "string" "derivation" "gotmpl" "jinja" "template"];
description = "Which engine to use for content generation";
default = "nix";
2025-08-25 16:45:22 +02:00
};
data = mkOption {
type = types.anything;
description = "The input data for the chosen generator";
};
opts = mkOption {
type = types.attrs;
default = {};
description = "Generator-specific options";
};
hook = mkOption {
type = types.submodule {
options = {
mode = mkOption {
type = types.enum ["link" "copy"];
default = "link";
description = "How the file should be managed (link or copy)";
};
gitignore = mkOption {
type = types.bool;
default = true;
description = "Whether to add the output path to .gitignore";
};
extra = mkOption {
type = types.str;
default = "";
description = "Additional bash commands to execute after file operation";
};
};
};
default = {};
description = "Hook-specific options for file management";
};
generatedDerivation = mkOption {
type = types.package;
internal = true;
readOnly = true;
description = "The generated derivation for this file";
};
};
config = {
generatedDerivation =
(engines.${config.generator} or (throw "Generator ${config.generator} not found"))
{
inherit (config) opts data name;
};
};
}));
default = {};
description = "Configuration hooks for file generation and management";
};
shellHook = mkOption {
type = types.str;
readOnly = true;
description = "Generated shell hook script for managing all files";
};
shellHookFile = mkOption {
type = types.package;
readOnly = true;
description = "Generated shell hook script for managing all files";
};
finalFiles = mkOption {
type = types.package;
readOnly = true;
description = "Aggregated derivation containing all managed files";
};
};
config = let
hooks = config.hooks;
hookNames = builtins.attrNames hooks;
runHooks = concatMapStringsSep "\n" (hookName: let
hook = hooks.${hookName};
modes = {
link =
# sh
''
if [[ ! -L "${hook.output}" ]] || [[ "$(readlink "${hook.output}")" != "${hook.generatedDerivation}" ]]; then
_soonix_log "info" "${hookName}" "Creating symlink: ${hook.output} -> ${hook.generatedDerivation}"
mkdir -p "$(dirname "${hook.output}")"
ln -sf "${hook.generatedDerivation}" "${hook.output}"
_changed=true
else
_soonix_log "info" "${hookName}" "Symlink up to date: ${hook.output}"
fi
'';
copy =
# sh
''
if [[ ! -f "${hook.output}" ]] || ! cmp -s "${hook.generatedDerivation}" "${hook.output}"; then
_soonix_log "info" "${hookName}" "Copying file: ${hook.generatedDerivation} -> ${hook.output}"
mkdir -p "$(dirname "${hook.output}")"
cp "${hook.generatedDerivation}" "${hook.output}"
_changed=true
else
_soonix_log "info" "${hookName}" "File up to date: ${hook.output}"
fi
'';
};
optionalGitignore =
if hook.hook.gitignore
then ''
_soonix_add_to_gitignore "${hook.output}"
''
else "";
in
# sh
''
# Process hook: ${hookName}
while IFS= read -r line; do
case "$line" in
UPDATED) _soonix_updated+=("${hookName}") ;;
UPTODATE) _soonix_uptodate+=("${hookName}") ;;
*) echo "$line" ;;
esac
done < <(
set -euo pipefail
_changed=false
${modes.${hook.hook.mode} or (throw "Mode ${hook.hook.mode} doesnt exist")}
# Add to gitignore if requested
${optionalGitignore}
# Run extra commands if file changed
if [[ "$_changed" == "true" && -n "${hook.hook.extra}" ]]; then
_soonix_log "info" "${hookName}" "Running extra command: ${hook.hook.extra}"
eval "${hook.hook.extra}"
fi
if [[ "$_changed" == "true" ]]; then
echo "UPDATED"
else
echo "UPTODATE"
fi
) || {
_soonix_log "error" "${hookName}" "Failed to process hook"
_soonix_failed+=("${hookName}")
}
'')
hookNames;
generatedShellHook =
# sh
''
_soonix_log() {
local level="$1"
local hook="$2"
local message="$3"
[[ "''${SOONIX_LOG-}" == "true" ]] && echo "$level [$hook]: $message" || true
}
_soonix_add_to_gitignore() {
local file="$1"
local gitignore=".gitignore"
if [[ ! -f "$gitignore" ]]; then
touch "$gitignore"
fi
# Check if file is already in gitignore
if ! grep -Fxq "$file" "$gitignore"; then
# Add sentinel comments if not present
if ! grep -q "# soonix" "$gitignore"; then
echo "" >> "$gitignore"
echo "# soonix" >> "$gitignore"
echo "# end soonix" >> "$gitignore"
fi
# Insert the file path before the end comment
${pkgs.gnused}/bin/sed -i "/# end soonix/i /$file" "$gitignore"
2025-08-25 16:45:22 +02:00
fi
}
_soonix_updated=()
_soonix_failed=()
_soonix_uptodate=()
${runHooks}
echo -n "[soonix] " >&2
if [[ ''${#_soonix_updated[@]} -gt 0 ]]; then
echo -n "[updated: ''${_soonix_updated[*]}] " >&2
fi
if [[ ''${#_soonix_uptodate[@]} -gt 0 ]]; then
echo -n "[unchanged: ''${_soonix_uptodate[*]}] " >&2
fi
if [[ ''${#_soonix_failed[@]} -gt 0 ]]; then
echo "[failed: ''${_soonix_failed[*]}]" >&2
exit 1
else
echo ""
fi
'';
allFiles =
lib.mapAttrsToList (name: hook: {
src = hook.generatedDerivation;
path = hook.output;
})
hooks;
in rec {
# nothing to do if no hooks exist
shellHook =
if (builtins.length hookNames > 0)
then generatedShellHook
else "";
shellHookFile = pkgs.writeShellScript "shellHook" shellHook;
finalFiles = buildAllFiles allFiles;
};
}