catvayor
2551da6388
All checks were successful
Check meta / check_dns (pull_request) Successful in 19s
Check meta / check_meta (pull_request) Successful in 20s
Check workflows / check_workflows (pull_request) Successful in 32s
Build all the nodes / ap01 (pull_request) Successful in 1m35s
Build all the nodes / bridge01 (pull_request) Successful in 2m4s
Build all the nodes / geo01 (pull_request) Successful in 2m18s
Build all the nodes / hypervisor01 (pull_request) Successful in 2m7s
Build all the nodes / geo02 (pull_request) Successful in 2m19s
Build all the nodes / hypervisor02 (pull_request) Successful in 1m35s
Build all the nodes / netcore02 (pull_request) Successful in 34s
Build all the nodes / hypervisor03 (pull_request) Successful in 1m33s
Build all the nodes / compute01 (pull_request) Successful in 4m6s
Build all the nodes / rescue01 (pull_request) Successful in 2m15s
Build all the nodes / storage01 (pull_request) Successful in 2m20s
Build all the nodes / tower01 (pull_request) Successful in 1m49s
Build all the nodes / vault01 (pull_request) Successful in 2m5s
Build the shell / build-shell (pull_request) Successful in 32s
Run pre-commit on all files / pre-commit (pull_request) Successful in 36s
Build all the nodes / web02 (pull_request) Successful in 1m45s
Build all the nodes / web01 (pull_request) Successful in 2m24s
Build all the nodes / web03 (pull_request) Successful in 1m47s
Build all the nodes / ap01 (push) Successful in 1m20s
Build all the nodes / bridge01 (push) Successful in 2m20s
Build all the nodes / geo01 (push) Successful in 2m22s
Build all the nodes / hypervisor01 (push) Successful in 2m24s
Build all the nodes / geo02 (push) Successful in 2m24s
Build all the nodes / netcore02 (push) Successful in 28s
Build all the nodes / hypervisor02 (push) Successful in 1m36s
Build all the nodes / compute01 (push) Successful in 3m14s
Build all the nodes / hypervisor03 (push) Successful in 1m39s
Build all the nodes / tower01 (push) Successful in 1m40s
Build all the nodes / storage01 (push) Successful in 2m12s
Build all the nodes / rescue01 (push) Successful in 2m15s
Build all the nodes / vault01 (push) Successful in 1m57s
Build the shell / build-shell (push) Successful in 39s
Run pre-commit on all files / pre-commit (push) Successful in 42s
Build all the nodes / web02 (push) Successful in 1m44s
Build all the nodes / web01 (push) Successful in 2m23s
Build all the nodes / web03 (push) Successful in 1m40s
276 lines
8.1 KiB
Nix
276 lines
8.1 KiB
Nix
# SPDX-FileCopyrightText: 2024 Lubin Bailly <lubin.bailly@dgnum.eu>
|
|
#
|
|
# SPDX-License-Identifier: EUPL-1.2
|
|
|
|
{
|
|
lib,
|
|
config,
|
|
pkgs,
|
|
...
|
|
}:
|
|
let
|
|
inherit (lib)
|
|
attrNames
|
|
concatMapStringsSep
|
|
concatStringsSep
|
|
escapeXML
|
|
filter
|
|
getExe
|
|
hasPrefix
|
|
hasSuffix
|
|
head
|
|
importJSON
|
|
mapAttrs'
|
|
mapAttrsToList
|
|
mkDefault
|
|
mkEnableOption
|
|
mkIf
|
|
mkOption
|
|
optionalString
|
|
pathIsDirectory
|
|
removePrefix
|
|
;
|
|
inherit (lib.strings)
|
|
sanitizeDerivationName
|
|
;
|
|
inherit (lib.types)
|
|
attrs
|
|
attrsOf
|
|
deferredModule
|
|
listOf
|
|
path
|
|
submodule
|
|
str
|
|
;
|
|
yaml = pkgs.formats.yaml { };
|
|
json = pkgs.formats.json { };
|
|
cfg = config.services.extranix;
|
|
|
|
module-eval =
|
|
module-name: module:
|
|
let
|
|
ignored-eval = lib.evalModules {
|
|
modules = module.ignored-modules;
|
|
inherit (module) specialArgs;
|
|
};
|
|
ignored-opts-doc = pkgs.nixosOptionsDoc { inherit (ignored-eval) options; };
|
|
ignored-opts = importJSON "${ignored-opts-doc.optionsJSON}/share/doc/nixos/options.json";
|
|
eval = lib.evalModules {
|
|
modules = module.paths ++ module.ignored-modules;
|
|
inherit (module) specialArgs;
|
|
};
|
|
opts-doc = pkgs.nixosOptionsDoc { inherit (eval) options; };
|
|
opts = importJSON "${opts-doc.optionsJSON}/share/doc/nixos/options.json";
|
|
filtered-opts = removeAttrs opts (attrNames ignored-opts);
|
|
path-translation =
|
|
let
|
|
translations = map (
|
|
{ base, url }:
|
|
{
|
|
url = "${url}${optionalString (!hasSuffix "/" url) "/"}";
|
|
base =
|
|
let
|
|
base1 = toString base;
|
|
in
|
|
base1 + (optionalString (!hasSuffix "/" base1) "/");
|
|
}
|
|
) module.path-translations;
|
|
in
|
|
path:
|
|
let
|
|
fullPath = path + (optionalString (pathIsDirectory path) "/default.nix");
|
|
fitting = filter ({ base, ... }: hasPrefix base fullPath) translations;
|
|
translate-info = head (
|
|
fitting
|
|
++ [
|
|
(throw (
|
|
"${fullPath} is not in any base path of ${module-name}. Base paths are "
|
|
+ concatMapStringsSep "\n" ({ base, ... }: base) translations
|
|
))
|
|
]
|
|
);
|
|
innerPath = removePrefix translate-info.base fullPath;
|
|
in
|
|
{
|
|
name = "<${innerPath}>";
|
|
url = "${translate-info.url}${innerPath}";
|
|
};
|
|
result' = json.generate "options-extranix-fileDesc.json" {
|
|
last_update = "-/-";
|
|
options = mapAttrsToList (title: val: {
|
|
inherit title;
|
|
inherit (val)
|
|
type
|
|
readOnly
|
|
loc
|
|
;
|
|
descriptionHTML = pkgs.runCommand "option-${title}.html" { } ''
|
|
${getExe pkgs.pandoc} -f markdown-raw_html ${pkgs.writeText "option-${title}.md" val.description} > $out
|
|
'';
|
|
description = escapeXML val.description;
|
|
example = escapeXML (val.example.text or "");
|
|
default = escapeXML (val.default.text or "");
|
|
declarations = map path-translation val.declarations;
|
|
}) filtered-opts;
|
|
};
|
|
result =
|
|
pkgs.runCommand "options-extranix.json"
|
|
{
|
|
nativeBuildInputs = [ pkgs.jq ];
|
|
}
|
|
''
|
|
jq -r '.options[].descriptionHTML | "--rawfile\n" + . + "\n" + .' ${result'} | xargs \
|
|
jq -c '.options |= map(.descriptionHTML as $desc | .descriptionHTML |= $ARGS.named.[$desc])' ${result'} \
|
|
> $out
|
|
'';
|
|
in
|
|
result;
|
|
|
|
options-files = mapAttrs' (name: value: {
|
|
name = sanitizeDerivationName name;
|
|
value = module-eval name value;
|
|
}) cfg.modules;
|
|
|
|
webroot = pkgs.callPackage ./webroot.nix {
|
|
inherit options-files;
|
|
inherit (cfg) static-data;
|
|
settings = yaml.generate "config.yaml" cfg.settings;
|
|
hugo-theme-extranix-options-search = pkgs.callPackage ./hugo-theme-extranix-options-search.nix { };
|
|
};
|
|
in
|
|
{
|
|
options.services.extranix = {
|
|
enable = mkEnableOption "extranix documentation";
|
|
modules = mkOption {
|
|
type =
|
|
|
|
attrsOf (submodule {
|
|
options = {
|
|
specialArgs = mkOption {
|
|
type = attrs;
|
|
default = { };
|
|
description = ''
|
|
Special arguments to give to evalModules.
|
|
'';
|
|
};
|
|
paths = mkOption {
|
|
type = listOf deferredModule;
|
|
description = ''
|
|
Modules to from which to document options.
|
|
'';
|
|
};
|
|
ignored-modules = mkOption {
|
|
type = listOf deferredModule;
|
|
default = [ ];
|
|
description = ''
|
|
Modules required to make modules of `paths` valid.
|
|
'';
|
|
example = ''
|
|
import "''${infra-modulesPath}/module-list.nix"
|
|
'';
|
|
};
|
|
path-translations = mkOption {
|
|
type = listOf (submodule {
|
|
options = {
|
|
base = mkOption {
|
|
type = path;
|
|
description = ''
|
|
Base path of some module files to be documented.
|
|
'';
|
|
};
|
|
url = mkOption {
|
|
type = str;
|
|
description = ''
|
|
Url root to use for files on this path.
|
|
'';
|
|
};
|
|
};
|
|
});
|
|
description = ''
|
|
Rules to convert file paths to urls in the documentation to indicate where
|
|
the option is declared.
|
|
'';
|
|
};
|
|
};
|
|
});
|
|
description = ''
|
|
Sets of modules to be documented separately. The identifier to give for
|
|
`settings.params.release_current_stable` (which is the default module shown) is the key after
|
|
passing through `sanitizeDerivationName`.
|
|
'';
|
|
};
|
|
settings = mkOption {
|
|
inherit (yaml) type;
|
|
description = ''
|
|
Settings for the `config.yaml` for the hugo instantiation.
|
|
'';
|
|
example = {
|
|
baseUrl = "https://example.org/";
|
|
title = "Module documentation";
|
|
languageCode = "en-us";
|
|
params = {
|
|
release_current_stable = "Some-Module";
|
|
logo = "logo.png";
|
|
footer_credits_line = ''
|
|
Based on <a href="https://github.com/mipmip/home-manager-option-search">Home Manager Option Search</a>
|
|
'';
|
|
footer_copyright_line = ''
|
|
Made by catvayor for the <a href="https://dgnum.eu">DGNum</a>.
|
|
'';
|
|
main_menu = [
|
|
{
|
|
name = ''Sources'';
|
|
url = "https://git.example.org/";
|
|
}
|
|
];
|
|
};
|
|
};
|
|
};
|
|
static-data = mkOption {
|
|
type = path;
|
|
description = ''
|
|
Static files for the website. Should have `images/favicon.png` for favicon.
|
|
'';
|
|
};
|
|
host = mkOption {
|
|
type = str;
|
|
description = ''
|
|
Hostname of the service.
|
|
'';
|
|
};
|
|
};
|
|
config = mkIf cfg.enable {
|
|
services = {
|
|
extranix.settings = {
|
|
theme = "extranix-options-search";
|
|
params = {
|
|
releases = mapAttrsToList (name: _: {
|
|
inherit name;
|
|
value = sanitizeDerivationName name;
|
|
}) cfg.modules;
|
|
release_current_stable = mkDefault (head (attrNames options-files));
|
|
};
|
|
};
|
|
nginx = {
|
|
enable = true;
|
|
virtualHosts.${cfg.host}.locations."/".alias = "${webroot}/";
|
|
};
|
|
};
|
|
assertions = [
|
|
{
|
|
assertion = cfg.modules != { };
|
|
message = ''
|
|
`services.extranix` can't be enabled without any modules to document.
|
|
'';
|
|
}
|
|
{
|
|
assertion = options-files ? ${cfg.settings.params.release_current_stable};
|
|
message = ''
|
|
`services.extranix.settings.params.release_current_stable` should be the
|
|
`sanitizeDerivationName` of a key of `services.extranix.modules`, here one of:
|
|
+ ${concatStringsSep "\n + " (attrNames options-files)}
|
|
'';
|
|
}
|
|
];
|
|
};
|
|
}
|