Source code for craft_cli.helptexts

# Copyright 2021-2022 Canonical Ltd.
#
# This program is free software; you can redistribute it and/or
# modify it under the terms of the GNU Lesser General Public
# License version 3 as published by the Free Software Foundation.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
# Lesser General Public License for more details.
#
# You should have received a copy of the GNU Lesser General Public License
# along with this program; if not, write to the Free Software Foundation,
# Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301, USA.

"""Provide all help texts."""

from __future__ import annotations

import argparse
import enum
import re
import textwrap
from operator import attrgetter
from typing import TYPE_CHECKING

if TYPE_CHECKING:
    from craft_cli.dispatcher import BaseCommand, CommandGroup


# if the `help` of any argument (global or for any command, option or parameter) is set to this
# value, the argument will not be shown in help messages; the default is to support the
# non-documented argparse attribute (so if users were using it, will just work) in a secure way
# in case it disappears in the future.
HIDDEN = argparse.SUPPRESS

# max columns used in the terminal
TERMINAL_WIDTH = 72

# generic intro and outro texts
HEADER = """
Usage:
    {appname} [help] <command>
"""

USAGE = """\
Usage: {appname} [options] command [args]...
Try '{full_command} -h' for help.

Error: {error_message}
"""


# the used formats, defaults to first one
OutputFormat = enum.Enum("OutputFormat", "plain markdown")


def _build_item_plain(title: str, text: str, title_space: int) -> list[str]:
    """Prepare an item for the help in plain format, generically a title and a text aligned.

    This is how the plain mode is built:
    - the title starts in column 4 with an extra ':', aligned to the right
    - the text starts in 4 plus the title space; if too wide it's wrapped.
    """
    # wrap the general text to the desired max width (discounting the space for the title,
    # the first 4 spaces, the two spaces to separate title/text, and the ':'
    not_title_space = 7
    text_space = TERMINAL_WIDTH - title_space - not_title_space
    wrapped_lines = textwrap.wrap(text, text_space)

    result: list[str] = []
    # first line goes with the title at column 4
    if wrapped_lines:
        result.append(f"    {title:>{title_space}s}:  {wrapped_lines[0]}")

    # the rest (if any) still aligned but without title
    for line in wrapped_lines[1:]:
        result.append(" " * (title_space + not_title_space) + line)

    return result


[docs] def process_overview_for_markdown(text: str) -> str: """Process a regular overview to be rendered with markdown. In detail: - Join all lines for the same paragraph (as wrapping is responsibility of the renderer) - Dedent and wrap with triple-backtick all indented blocks Paragraphs are separated by empty lines """ lines = [x.rstrip() for x in text.strip().split("\n")] # group all the lines in different blocks, each holding what would be a # paragraph (detected by the empty line that separates them) blocks: list[list[str]] = [[]] for line in lines: if line: blocks[-1].append(line) else: blocks.append([]) # convert each of the block/paragraph into their markdown representation result: list[str] = [] for block in blocks: if block and block[0] and block[0][0] == " ": # it is indented! dedent and wrap with backticks dedented = textwrap.dedent("\n".join(block)) text = f"```text\n{dedented}\n```" else: # regular text text = " ".join(block) # include the processed text and an empty line; this empty line will be a separation # between paragraphs or the final newline at the end of the whole text result.extend((text, "")) return "\n".join(result)
[docs] class HelpBuilder: """Produce the different help texts.""" def __init__( self, appname: str, general_summary: str, command_groups: list[CommandGroup], docs_base_url: str | None = None, ) -> None: """Initialize the help builder. :param appname: The name of the application. :param general_summary: A summary of the application. :param command_groups: The CommandGroups for the application. :param docs_base_url: The base URL for the documentation. """ self.appname = appname self.general_summary = general_summary self.command_groups = command_groups self._docs_base_url = docs_base_url if docs_base_url and docs_base_url.endswith("/"): self._docs_base_url = docs_base_url[:-1]
[docs] def get_usage_message(self, error_message: str, command: str = "") -> str: """Build a usage and error message. The command is the extra string used after the application name to build the full command that will be shown in the usage message; for example, having an application name of "someapp": - if command is "" it will be shown "Try 'appname -h' for help". - if command is "version" it will be shown "Try 'appname version -h' for help" The error message is the specific problem in the given parameters. """ full_command = f"{self.appname} {command}" if command else self.appname return USAGE.format( appname=self.appname, full_command=full_command, error_message=error_message )
[docs] def get_full_help(self, global_options: list[tuple[str, str]]) -> str: """Produce the text for the default help. - global_options: options defined at application level (not in the commands), with the (options, description) structure The help text has the following structure: - usage - summary - common commands listed and described shortly - all commands grouped, just listed - more help and documentation """ textblocks = [] # title textblocks.append(HEADER.format(appname=self.appname)) # summary textblocks.append("Summary:" + textwrap.indent(self.general_summary, " ")) # column alignment is dictated by longest common commands names and groups names max_title_len = 0 # collect common commands common_commands = [] for command_group in self.command_groups: max_title_len = max(len(command_group.name), max_title_len) for cmd in command_group.commands: if cmd.common: common_commands.append(cmd) max_title_len = max(len(cmd.name), max_title_len) for title, _ in global_options: max_title_len = max(len(title), max_title_len) global_lines = ["Global options:"] for title, text in global_options: if text is not HIDDEN: global_lines.extend(_build_item_plain(title, text, max_title_len)) textblocks.append("\n".join(global_lines)) common_lines = ["Starter commands:"] for cmd in sorted(common_commands, key=attrgetter("name")): common_lines.extend(_build_item_plain(cmd.name, cmd.help_msg, max_title_len)) textblocks.append("\n".join(common_lines)) grouped_lines = ["Commands can be classified as follows:"] for command_group in sorted(self.command_groups, key=attrgetter("name")): command_names = [cmd.name for cmd in command_group.commands if not cmd.hidden] if not command_group.ordered: command_names.sort() command_names_str = ", ".join(command_names) grouped_lines.extend( _build_item_plain(command_group.name, command_names_str, max_title_len) ) textblocks.append("\n".join(grouped_lines)) more_help_text = textwrap.dedent( f""" For more information about a command, run '{self.appname} help <command>'. For a summary of all commands, run '{self.appname} help --all'.""" ) # append documentation links to block for more help if self._docs_base_url: more_help_text += ( f"\nFor more information about {self.appname}, check out: {self._docs_base_url}" ) textblocks.append(more_help_text) # join all stripped blocks, leaving ONE empty blank line between return "\n\n".join(block.strip() for block in textblocks) + "\n"
[docs] def get_detailed_help(self, global_options: list[tuple[str, str]]) -> str: """Produce the text for the detailed help. - global_options: options defined at application level (not in the commands), with the (options, description) structure The help text has the following structure: - usage - summary - global options - all commands shown with description, grouped - more help and documentation """ textblocks = [] # title textblocks.append(HEADER.format(appname=self.appname)) # summary textblocks.append("Summary:" + textwrap.indent(self.general_summary, " ")) # column alignment is dictated by longest common commands names and groups names max_title_len = 0 for command_group in self.command_groups: for cmd in command_group.commands: max_title_len = max(len(cmd.name), max_title_len) for title, _ in global_options: max_title_len = max(len(title), max_title_len) global_lines = ["Global options:"] for title, text in global_options: if text is not HIDDEN: global_lines.extend(_build_item_plain(title, text, max_title_len)) textblocks.append("\n".join(global_lines)) textblocks.append("Commands can be classified as follows:") for command_group in self.command_groups: group_lines = [f"{command_group.name}:"] for cmd in command_group.commands: if cmd.hidden: continue group_lines.extend(_build_item_plain(cmd.name, cmd.help_msg, max_title_len)) textblocks.append("\n".join(group_lines)) more_help_text = ( f"For more information about a specific command, run '{self.appname} help <command>'." ) if self._docs_base_url: more_help_text += ( f"\nFor more information about {self.appname}, check out: {self._docs_base_url}" ) textblocks.append(more_help_text) # join all stripped blocks, leaving ONE empty blank line between return "\n\n".join(block.strip() for block in textblocks) + "\n"
def _build_plain_command_help( self, command: BaseCommand, usage: str, parameters: list[tuple[str, str]], options: list[tuple[str, str]], other_command_names: list[str], ) -> list[str]: """Build the command help in its plain version. The help text has the following structure: - usage - summary - positional arguments (only if parameters are not empty) - options - other related commands - help for all commands and documentation """ textblocks = [] textblocks.append( textwrap.dedent( f"""\ Usage: {usage} """ ) ) overview = textwrap.indent(command.overview, " ") # Remove reST-style double backticks # Match _only_ double backticks, never triples overview = re.sub(r"(?<!`)``(?!`)", "", overview) textblocks.append(f"Summary:{overview}") # column alignment is dictated by longest options title max_title_len = max(len(title) for title, text in options) if parameters: # command positional arguments positional_args_lines: list[str] = [] for title, text in parameters: positional_args_lines.extend(_build_item_plain(title, text, max_title_len)) # Only populate if we have collected parameters. if positional_args_lines: textblocks.append("\n".join(["Positional arguments:", *positional_args_lines])) # command options option_lines = ["Options:"] for title, text in options: option_lines.extend(_build_item_plain(title, text, max_title_len)) textblocks.append("\n".join(option_lines)) if other_command_names: see_also_block = ["See also:"] see_also_block.extend((" " + name) for name in sorted(other_command_names)) textblocks.append("\n".join(see_also_block)) # help for all commands more_help_text = f"For a summary of all commands, run '{self.appname} help --all'." if self._docs_base_url: command_url = f"{self._docs_base_url}/reference/commands/{command.name}" more_help_text += f"\nFor more information, check out: {command_url}" textblocks.append(more_help_text) return textblocks def _build_markdown_command_help( self, command: BaseCommand, usage: str, parameters: list[tuple[str, str]], options: list[tuple[str, str]], other_command_names: list[str], ) -> list[str]: """Build the command help in its markdown version. The help text has the following structure: - usage - summary - positional arguments (only if parameters are not empty) - options - other related commands """ textblocks = [] textblocks.append( textwrap.dedent( f"""\ ## Usage: ```text {usage} ``` """ ) ) overview = process_overview_for_markdown(command.overview) textblocks.append(f"## Summary:\n\n{overview}") if parameters: parameters_lines: list[str] = [] # Only keep items that have text for title, text in ((title, text) for title, text in parameters if text): parameters_lines.append(f"| `{title}` | {text} |") # Only populate if we have collected parameters if parameters_lines: header_lines = [ "## Positional arguments:", "| | |", "|-|-|", ] textblocks.append("\n".join(header_lines + parameters_lines)) option_lines = [ "## Options:", "| | |", "|-|-|", ] for title, text in options: option_lines.append(f"| `{title}` | {text} |") textblocks.append("\n".join(option_lines)) if other_command_names: see_also_block = ["## See also:"] see_also_block.extend(f"- `{name}`" for name in sorted(other_command_names)) textblocks.append("\n".join(see_also_block)) return textblocks
[docs] def get_command_help( self, command: BaseCommand, arguments: list[tuple[str, str]], output_format: OutputFormat, ) -> str: """Produce the text for each command's help in any output format. - command: the instantiated command for which help is prepared - arguments: all command options and parameters, with the (name, description) structure; note that any argument with description being `HIDDEN` will be ignored - output_format: the selected output format The help text structure depends of the output format. """ # separate all arguments into the parameters and optional ones, just checking # if first char is a dash parameters = [] options = [] for name, title in arguments: if title is HIDDEN: continue if name[0] == "-": options.append((name, title)) else: parameters.append((name, title)) usage = f"{self.appname} {command.name} [options]" if parameters: usage += " " + " ".join(f"<{parameter[0]}>" for parameter in parameters) for command_group in self.command_groups: if any(isinstance(command, command_class) for command_class in command_group.commands): break else: raise RuntimeError("Internal inconsistency in commands groups") other_command_names = [ c.name for c in command_group.commands if not isinstance(command, c) ] if output_format == OutputFormat.markdown: builder = self._build_markdown_command_help else: builder = self._build_plain_command_help textblocks = builder(command, usage, parameters, options, other_command_names) # join all stripped blocks, leaving ONE empty blank line between return "\n\n".join(block.strip() for block in textblocks) + "\n"