2023-05-03 10:16:29 +03:00
|
|
|
let
|
|
|
|
inherit (builtins)
|
2023-11-30 22:41:47 +02:00
|
|
|
attrNames
|
|
|
|
attrValues
|
|
|
|
concatMap
|
|
|
|
concatStringsSep
|
|
|
|
fromJSON
|
|
|
|
groupBy
|
|
|
|
length
|
|
|
|
lessThan
|
|
|
|
listToAttrs
|
|
|
|
mapAttrs
|
|
|
|
match
|
|
|
|
replaceStrings
|
|
|
|
sort
|
|
|
|
;
|
|
|
|
inherit (import <nix/utils.nix>)
|
|
|
|
attrsToList
|
|
|
|
concatStrings
|
|
|
|
filterAttrs
|
|
|
|
optionalString
|
|
|
|
squash
|
|
|
|
trim
|
|
|
|
unique
|
|
|
|
;
|
|
|
|
showStoreDocs = import <nix/generate-store-info.nix>;
|
2023-05-03 10:16:29 +03:00
|
|
|
in
|
2020-12-03 00:05:28 +02:00
|
|
|
|
2023-10-05 02:20:26 +03:00
|
|
|
inlineHTML: commandDump:
|
2020-09-16 15:55:24 +03:00
|
|
|
|
|
|
|
let
|
|
|
|
|
2023-05-03 10:16:29 +03:00
|
|
|
commandInfo = fromJSON commandDump;
|
|
|
|
|
2022-10-07 19:07:22 +03:00
|
|
|
showCommand = { command, details, filename, toplevel }:
|
2022-08-26 18:40:34 +03:00
|
|
|
let
|
2023-03-21 13:11:32 +02:00
|
|
|
|
2022-08-27 03:44:54 +03:00
|
|
|
result = ''
|
|
|
|
> **Warning** \
|
2023-04-06 18:09:01 +03:00
|
|
|
> This program is
|
|
|
|
> [**experimental**](@docroot@/contributing/experimental-features.md#xp-feature-nix-command)
|
|
|
|
> and its interface is subject to change.
|
2022-08-27 03:44:54 +03:00
|
|
|
|
|
|
|
# Name
|
|
|
|
|
2022-08-27 04:25:12 +03:00
|
|
|
`${command}` - ${details.description}
|
2022-08-27 03:44:54 +03:00
|
|
|
|
|
|
|
# Synopsis
|
|
|
|
|
2022-08-27 04:25:12 +03:00
|
|
|
${showSynopsis command details.args}
|
2022-08-27 03:44:54 +03:00
|
|
|
|
|
|
|
${maybeSubcommands}
|
|
|
|
|
2023-12-01 00:07:09 +02:00
|
|
|
${maybeProse}
|
2022-08-27 03:44:54 +03:00
|
|
|
|
|
|
|
${maybeOptions}
|
|
|
|
'';
|
2023-03-21 13:11:32 +02:00
|
|
|
|
2022-08-26 18:40:34 +03:00
|
|
|
showSynopsis = command: args:
|
|
|
|
let
|
2023-04-16 17:44:03 +03:00
|
|
|
showArgument = arg: "*${arg.label}*" + optionalString (! arg ? arity) "...";
|
2022-08-26 18:40:34 +03:00
|
|
|
arguments = concatStringsSep " " (map showArgument args);
|
|
|
|
in ''
|
2023-09-07 01:31:33 +03:00
|
|
|
`${command}` [*option*...] ${arguments}
|
2022-08-26 18:40:34 +03:00
|
|
|
'';
|
2023-03-21 13:11:32 +02:00
|
|
|
|
2023-04-16 17:44:03 +03:00
|
|
|
maybeSubcommands = optionalString (details ? commands && details.commands != {})
|
2023-09-07 01:31:33 +03:00
|
|
|
''
|
|
|
|
where *subcommand* is one of the following:
|
2022-08-26 18:40:34 +03:00
|
|
|
|
2023-09-07 01:31:33 +03:00
|
|
|
${subcommands}
|
|
|
|
'';
|
2023-03-21 13:11:32 +02:00
|
|
|
|
2022-08-26 18:40:34 +03:00
|
|
|
subcommands = if length categories > 1
|
|
|
|
then listCategories
|
2022-08-27 04:25:12 +03:00
|
|
|
else listSubcommands details.commands;
|
2023-03-21 13:11:32 +02:00
|
|
|
|
2022-08-27 04:25:12 +03:00
|
|
|
categories = sort (x: y: x.id < y.id) (unique (map (cmd: cmd.category) (attrValues details.commands)));
|
2023-03-21 13:11:32 +02:00
|
|
|
|
2022-08-26 18:40:34 +03:00
|
|
|
listCategories = concatStrings (map showCategory categories);
|
2023-03-21 13:11:32 +02:00
|
|
|
|
2022-08-26 18:40:34 +03:00
|
|
|
showCategory = cat: ''
|
|
|
|
**${toString cat.description}:**
|
|
|
|
|
2022-08-27 04:25:12 +03:00
|
|
|
${listSubcommands (filterAttrs (n: v: v.category == cat) details.commands)}
|
2022-08-26 18:40:34 +03:00
|
|
|
'';
|
2023-03-21 13:11:32 +02:00
|
|
|
|
2022-08-26 18:40:34 +03:00
|
|
|
listSubcommands = cmds: concatStrings (attrValues (mapAttrs showSubcommand cmds));
|
2023-03-21 13:11:32 +02:00
|
|
|
|
2022-08-26 18:40:34 +03:00
|
|
|
showSubcommand = name: subcmd: ''
|
|
|
|
* [`${command} ${name}`](./${appendName filename name}.md) - ${subcmd.description}
|
|
|
|
'';
|
2023-03-21 13:11:32 +02:00
|
|
|
|
2023-12-01 00:07:09 +02:00
|
|
|
maybeProse =
|
|
|
|
# FIXME: this is a horrible hack to keep `nix help-stores` working.
|
|
|
|
let
|
|
|
|
help-stores = ''
|
|
|
|
${index}
|
|
|
|
|
|
|
|
${allStores}
|
|
|
|
'';
|
|
|
|
index = replaceStrings
|
2023-12-02 01:56:23 +02:00
|
|
|
[ "@store-types@" "./local-store.md" "./local-daemon-store.md" ]
|
|
|
|
[ storesOverview "#local-store" "#local-daemon-store" ]
|
2023-12-01 00:07:09 +02:00
|
|
|
details.doc;
|
|
|
|
storesOverview =
|
|
|
|
let
|
|
|
|
showEntry = store:
|
|
|
|
"- [${store.name}](#${store.slug})";
|
|
|
|
in
|
|
|
|
concatStringsSep "\n" (map showEntry storesList) + "\n";
|
|
|
|
allStores = concatStringsSep "\n" (attrValues storePages);
|
|
|
|
storePages = listToAttrs
|
|
|
|
(map (s: { name = s.filename; value = s.page; }) storesList);
|
|
|
|
storesList = showStoreDocs {
|
|
|
|
storeInfo = commandInfo.stores;
|
|
|
|
inherit inlineHTML;
|
|
|
|
};
|
2024-07-12 16:17:31 +03:00
|
|
|
hasInfix = infix: content:
|
|
|
|
builtins.stringLength content != builtins.stringLength (replaceStrings [ infix ] [ "" ] content);
|
2023-12-01 00:07:09 +02:00
|
|
|
in
|
|
|
|
optionalString (details ? doc) (
|
2024-07-12 16:17:31 +03:00
|
|
|
# An alternate implementation with builtins.match stack overflowed on some systems.
|
|
|
|
if hasInfix "@store-types@" details.doc
|
2023-12-01 00:07:09 +02:00
|
|
|
then help-stores
|
|
|
|
else details.doc
|
|
|
|
);
|
|
|
|
|
|
|
|
maybeOptions =
|
|
|
|
let
|
|
|
|
allVisibleOptions = filterAttrs
|
|
|
|
(_: o: ! o.hiddenCategory)
|
|
|
|
(details.flags // toplevel.flags);
|
|
|
|
in
|
|
|
|
optionalString (allVisibleOptions != { }) ''
|
|
|
|
# Options
|
|
|
|
|
|
|
|
${showOptions inlineHTML allVisibleOptions}
|
|
|
|
|
|
|
|
> **Note**
|
|
|
|
>
|
|
|
|
> See [`man nix.conf`](@docroot@/command-ref/conf-file.md#command-line-flags) for overriding configuration settings with command line flags.
|
|
|
|
'';
|
2023-03-21 13:11:32 +02:00
|
|
|
|
2023-10-05 02:20:26 +03:00
|
|
|
showOptions = inlineHTML: allOptions:
|
2022-08-26 18:40:34 +03:00
|
|
|
let
|
2023-09-23 07:28:16 +03:00
|
|
|
showCategory = cat: opts: ''
|
2023-09-23 07:35:03 +03:00
|
|
|
${optionalString (cat != "") "## ${cat}"}
|
2022-08-27 00:09:19 +03:00
|
|
|
|
2023-09-23 07:28:16 +03:00
|
|
|
${concatStringsSep "\n" (attrValues (mapAttrs showOption opts))}
|
2023-11-30 22:41:47 +02:00
|
|
|
'';
|
2022-08-27 00:09:19 +03:00
|
|
|
showOption = name: option:
|
|
|
|
let
|
2023-10-05 02:20:26 +03:00
|
|
|
result = trim ''
|
|
|
|
- ${item}
|
|
|
|
|
|
|
|
${option.description}
|
|
|
|
'';
|
|
|
|
item = if inlineHTML
|
|
|
|
then ''<span id="opt-${name}">[`--${name}`](#opt-${name})</span> ${shortName} ${labels}''
|
|
|
|
else "`--${name}` ${shortName} ${labels}";
|
2023-04-16 17:44:03 +03:00
|
|
|
shortName = optionalString
|
|
|
|
(option ? shortName)
|
|
|
|
("/ `-${option.shortName}`");
|
|
|
|
labels = optionalString
|
|
|
|
(option ? labels)
|
|
|
|
(concatStringsSep " " (map (s: "*${s}*") option.labels));
|
2023-10-05 02:20:26 +03:00
|
|
|
in result;
|
2023-09-23 07:28:16 +03:00
|
|
|
categories = mapAttrs
|
2023-09-25 15:20:39 +03:00
|
|
|
# Convert each group from a list of key-value pairs back to an attrset
|
2023-09-23 07:28:16 +03:00
|
|
|
(_: listToAttrs)
|
|
|
|
(groupBy
|
|
|
|
(cmd: cmd.value.category)
|
|
|
|
(attrsToList allOptions));
|
|
|
|
in concatStrings (attrValues (mapAttrs showCategory categories));
|
2022-08-27 03:44:54 +03:00
|
|
|
in squash result;
|
2022-08-26 18:40:34 +03:00
|
|
|
|
|
|
|
appendName = filename: name: (if filename == "nix" then "nix3" else filename) + "-" + name;
|
2020-09-16 15:55:24 +03:00
|
|
|
|
2022-10-07 19:07:22 +03:00
|
|
|
processCommand = { command, details, filename, toplevel }:
|
2022-08-27 04:25:12 +03:00
|
|
|
let
|
|
|
|
cmd = {
|
|
|
|
inherit command;
|
|
|
|
name = filename + ".md";
|
2022-10-07 19:07:22 +03:00
|
|
|
value = showCommand { inherit command details filename toplevel; };
|
2022-08-27 04:25:12 +03:00
|
|
|
};
|
|
|
|
subcommand = subCmd: processCommand {
|
|
|
|
command = command + " " + subCmd;
|
|
|
|
details = details.commands.${subCmd};
|
|
|
|
filename = appendName filename subCmd;
|
2022-10-07 19:07:22 +03:00
|
|
|
inherit toplevel;
|
2022-08-27 04:25:12 +03:00
|
|
|
};
|
|
|
|
in [ cmd ] ++ concatMap subcommand (attrNames details.commands or {});
|
|
|
|
|
|
|
|
manpages = processCommand {
|
|
|
|
command = "nix";
|
2023-05-03 10:16:29 +03:00
|
|
|
details = commandInfo.args;
|
2022-08-27 04:25:12 +03:00
|
|
|
filename = "nix";
|
2023-05-03 10:16:29 +03:00
|
|
|
toplevel = commandInfo.args;
|
2022-08-27 04:25:12 +03:00
|
|
|
};
|
|
|
|
|
|
|
|
tableOfContents = let
|
|
|
|
showEntry = page:
|
|
|
|
" - [${page.command}](command-ref/new-cli/${page.name})";
|
2022-09-30 02:41:56 +03:00
|
|
|
in concatStringsSep "\n" (map showEntry manpages) + "\n";
|
2022-08-27 04:25:12 +03:00
|
|
|
|
|
|
|
in (listToAttrs manpages) // { "SUMMARY.md" = tableOfContents; }
|