2023-05-03 07:16:29 +00:00
|
|
|
let
|
|
|
|
inherit (builtins)
|
|
|
|
attrNames attrValues fromJSON listToAttrs mapAttrs
|
|
|
|
concatStringsSep concatMap length lessThan replaceStrings sort;
|
|
|
|
inherit (import ./utils.nix) concatStrings optionalString filterAttrs trim squash unique showSettings;
|
|
|
|
in
|
2020-12-02 22:05:28 +00:00
|
|
|
|
2023-05-03 07:16:29 +00:00
|
|
|
commandDump:
|
2020-09-16 12:55:24 +00:00
|
|
|
|
|
|
|
let
|
|
|
|
|
2023-05-03 07:16:29 +00:00
|
|
|
commandInfo = fromJSON commandDump;
|
|
|
|
|
2022-10-07 16:07:22 +00:00
|
|
|
showCommand = { command, details, filename, toplevel }:
|
2022-08-26 15:40:34 +00:00
|
|
|
let
|
2023-03-21 11:11:32 +00:00
|
|
|
|
2022-08-27 00:44:54 +00:00
|
|
|
result = ''
|
|
|
|
> **Warning** \
|
2023-04-06 15:09:01 +00:00
|
|
|
> This program is
|
|
|
|
> [**experimental**](@docroot@/contributing/experimental-features.md#xp-feature-nix-command)
|
|
|
|
> and its interface is subject to change.
|
2022-08-27 00:44:54 +00:00
|
|
|
|
|
|
|
# Name
|
|
|
|
|
2022-08-27 01:25:12 +00:00
|
|
|
`${command}` - ${details.description}
|
2022-08-27 00:44:54 +00:00
|
|
|
|
|
|
|
# Synopsis
|
|
|
|
|
2022-08-27 01:25:12 +00:00
|
|
|
${showSynopsis command details.args}
|
2022-08-27 00:44:54 +00:00
|
|
|
|
|
|
|
${maybeSubcommands}
|
|
|
|
|
|
|
|
${maybeDocumentation}
|
|
|
|
|
|
|
|
${maybeOptions}
|
|
|
|
'';
|
2023-03-21 11:11:32 +00:00
|
|
|
|
2022-08-26 15:40:34 +00:00
|
|
|
showSynopsis = command: args:
|
|
|
|
let
|
2023-04-16 14:44:03 +00:00
|
|
|
showArgument = arg: "*${arg.label}*" + optionalString (! arg ? arity) "...";
|
2022-08-26 15:40:34 +00:00
|
|
|
arguments = concatStringsSep " " (map showArgument args);
|
|
|
|
in ''
|
|
|
|
`${command}` [*option*...] ${arguments}
|
|
|
|
'';
|
2023-03-21 11:11:32 +00:00
|
|
|
|
2023-04-16 14:44:03 +00:00
|
|
|
maybeSubcommands = optionalString (details ? commands && details.commands != {})
|
|
|
|
''
|
2022-08-26 15:40:34 +00:00
|
|
|
where *subcommand* is one of the following:
|
|
|
|
|
|
|
|
${subcommands}
|
2023-04-16 14:44:03 +00:00
|
|
|
'';
|
2023-03-21 11:11:32 +00:00
|
|
|
|
2022-08-26 15:40:34 +00:00
|
|
|
subcommands = if length categories > 1
|
|
|
|
then listCategories
|
2022-08-27 01:25:12 +00:00
|
|
|
else listSubcommands details.commands;
|
2023-03-21 11:11:32 +00:00
|
|
|
|
2022-08-27 01:25:12 +00:00
|
|
|
categories = sort (x: y: x.id < y.id) (unique (map (cmd: cmd.category) (attrValues details.commands)));
|
2023-03-21 11:11:32 +00:00
|
|
|
|
2022-08-26 15:40:34 +00:00
|
|
|
listCategories = concatStrings (map showCategory categories);
|
2023-03-21 11:11:32 +00:00
|
|
|
|
2022-08-26 15:40:34 +00:00
|
|
|
showCategory = cat: ''
|
|
|
|
**${toString cat.description}:**
|
|
|
|
|
2022-08-27 01:25:12 +00:00
|
|
|
${listSubcommands (filterAttrs (n: v: v.category == cat) details.commands)}
|
2022-08-26 15:40:34 +00:00
|
|
|
'';
|
2023-03-21 11:11:32 +00:00
|
|
|
|
2022-08-26 15:40:34 +00:00
|
|
|
listSubcommands = cmds: concatStrings (attrValues (mapAttrs showSubcommand cmds));
|
2023-03-21 11:11:32 +00:00
|
|
|
|
2022-08-26 15:40:34 +00:00
|
|
|
showSubcommand = name: subcmd: ''
|
|
|
|
* [`${command} ${name}`](./${appendName filename name}.md) - ${subcmd.description}
|
|
|
|
'';
|
2023-03-21 11:11:32 +00:00
|
|
|
|
2023-09-06 07:14:56 +00:00
|
|
|
# TODO: move this confusing special case out of here when implementing #8496
|
2023-04-16 14:44:03 +00:00
|
|
|
maybeDocumentation = optionalString
|
|
|
|
(details ? doc)
|
|
|
|
(replaceStrings ["@stores@"] [storeDocs] details.doc);
|
2023-03-21 11:11:32 +00:00
|
|
|
|
2023-04-16 14:44:03 +00:00
|
|
|
maybeOptions = optionalString (details.flags != {}) ''
|
2022-08-26 15:40:34 +00:00
|
|
|
# Options
|
|
|
|
|
2022-10-07 16:07:22 +00:00
|
|
|
${showOptions details.flags toplevel.flags}
|
2022-08-26 15:40:34 +00:00
|
|
|
'';
|
2023-03-21 11:11:32 +00:00
|
|
|
|
2022-10-07 16:07:22 +00:00
|
|
|
showOptions = options: commonOptions:
|
2022-08-26 15:40:34 +00:00
|
|
|
let
|
2022-10-07 16:07:22 +00:00
|
|
|
allOptions = options // commonOptions;
|
2022-08-26 21:09:19 +00:00
|
|
|
showCategory = cat: ''
|
2023-04-16 14:44:03 +00:00
|
|
|
${optionalString (cat != "") "**${cat}:**"}
|
2022-08-26 21:09:19 +00:00
|
|
|
|
2022-10-07 16:07:22 +00:00
|
|
|
${listOptions (filterAttrs (n: v: v.category == cat) allOptions)}
|
2022-08-26 21:09:19 +00:00
|
|
|
'';
|
|
|
|
listOptions = opts: concatStringsSep "\n" (attrValues (mapAttrs showOption opts));
|
|
|
|
showOption = name: option:
|
|
|
|
let
|
2023-04-16 14:44:03 +00:00
|
|
|
shortName = optionalString
|
|
|
|
(option ? shortName)
|
|
|
|
("/ `-${option.shortName}`");
|
|
|
|
labels = optionalString
|
|
|
|
(option ? labels)
|
|
|
|
(concatStringsSep " " (map (s: "*${s}*") option.labels));
|
2022-08-26 21:09:19 +00:00
|
|
|
in trim ''
|
2023-09-06 07:52:56 +00:00
|
|
|
- <span id="opt-${name}">[`--${name}`](#opt-${name})</span> ${shortName} ${labels}
|
2022-08-26 21:09:19 +00:00
|
|
|
|
|
|
|
${option.description}
|
|
|
|
'';
|
2023-05-03 07:16:29 +00:00
|
|
|
categories = sort lessThan (unique (map (cmd: cmd.category) (attrValues allOptions)));
|
2022-08-26 21:09:19 +00:00
|
|
|
in concatStrings (map showCategory categories);
|
2022-08-27 00:44:54 +00:00
|
|
|
in squash result;
|
2022-08-26 15:40:34 +00:00
|
|
|
|
|
|
|
appendName = filename: name: (if filename == "nix" then "nix3" else filename) + "-" + name;
|
2020-09-16 12:55:24 +00:00
|
|
|
|
2022-10-07 16:07:22 +00:00
|
|
|
processCommand = { command, details, filename, toplevel }:
|
2022-08-27 01:25:12 +00:00
|
|
|
let
|
|
|
|
cmd = {
|
|
|
|
inherit command;
|
|
|
|
name = filename + ".md";
|
2022-10-07 16:07:22 +00:00
|
|
|
value = showCommand { inherit command details filename toplevel; };
|
2022-08-27 01:25:12 +00:00
|
|
|
};
|
|
|
|
subcommand = subCmd: processCommand {
|
|
|
|
command = command + " " + subCmd;
|
|
|
|
details = details.commands.${subCmd};
|
|
|
|
filename = appendName filename subCmd;
|
2022-10-07 16:07:22 +00:00
|
|
|
inherit toplevel;
|
2022-08-27 01:25:12 +00:00
|
|
|
};
|
|
|
|
in [ cmd ] ++ concatMap subcommand (attrNames details.commands or {});
|
|
|
|
|
|
|
|
manpages = processCommand {
|
|
|
|
command = "nix";
|
2023-05-03 07:16:29 +00:00
|
|
|
details = commandInfo.args;
|
2022-08-27 01:25:12 +00:00
|
|
|
filename = "nix";
|
2023-05-03 07:16:29 +00:00
|
|
|
toplevel = commandInfo.args;
|
2022-08-27 01:25:12 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
tableOfContents = let
|
|
|
|
showEntry = page:
|
|
|
|
" - [${page.command}](command-ref/new-cli/${page.name})";
|
2022-09-29 23:41:56 +00:00
|
|
|
in concatStringsSep "\n" (map showEntry manpages) + "\n";
|
2022-08-27 01:25:12 +00:00
|
|
|
|
2023-03-21 11:58:14 +00:00
|
|
|
storeDocs =
|
|
|
|
let
|
2023-04-17 15:58:47 +00:00
|
|
|
showStore = name: { settings, doc, experimentalFeature }:
|
|
|
|
let
|
|
|
|
experimentalFeatureNote = optionalString (experimentalFeature != null) ''
|
|
|
|
> **Warning**
|
|
|
|
> This store is part of an
|
|
|
|
> [experimental feature](@docroot@/contributing/experimental-features.md).
|
|
|
|
|
|
|
|
To use this store, you need to make sure the corresponding experimental feature,
|
|
|
|
[`${experimentalFeature}`](@docroot@/contributing/experimental-features.md#xp-feature-${experimentalFeature}),
|
|
|
|
is enabled.
|
|
|
|
For example, include the following in [`nix.conf`](#):
|
|
|
|
|
|
|
|
```
|
|
|
|
extra-experimental-features = ${experimentalFeature}
|
|
|
|
```
|
|
|
|
'';
|
|
|
|
in ''
|
2023-03-21 11:58:14 +00:00
|
|
|
## ${name}
|
|
|
|
|
2023-03-21 13:03:05 +00:00
|
|
|
${doc}
|
|
|
|
|
2023-04-17 15:58:47 +00:00
|
|
|
${experimentalFeatureNote}
|
|
|
|
|
2023-03-21 13:03:05 +00:00
|
|
|
**Settings**:
|
|
|
|
|
2023-03-23 09:08:49 +00:00
|
|
|
${showSettings { useAnchors = false; } settings}
|
2023-03-21 11:58:14 +00:00
|
|
|
'';
|
2023-05-03 07:16:29 +00:00
|
|
|
in concatStrings (attrValues (mapAttrs showStore commandInfo.stores));
|
2023-03-21 11:58:14 +00:00
|
|
|
|
2022-08-27 01:25:12 +00:00
|
|
|
in (listToAttrs manpages) // { "SUMMARY.md" = tableOfContents; }
|