click icon indicating copy to clipboard operation
click copied to clipboard

WIP: Help page customization low level api

Open Rowlando13 opened this issue 3 months ago • 0 comments

Work in Progress ...

General

  • help kwarg stays as is.

  • for Commands and Groups, there are new kwargs

    • custom_help : None | Subclass of CustomHelp = a custom help class
    • custom_help_settings : None | Subclass of CustomHelpLoad = for lightly customized custom help classes
    • custom_help_settings_call: None | function = for heavily customized help classes
  • There are two general types of help page, Group help page and Command help page.

  • Help pages may be invoked directly, or shown following an error.

    • ??? custom_help_cascade : bool = True whether or not a custom help formatter cascades to sub command or sub groups. I expect the normal usage pattern to be set one custom fomatter per command line application, so just set in one place, at the top group. However, I can see good reason to allow this to be turned off.

Questions

  • How do we uphold arbitrary nesting guarantees while allowing for maximal customization?
  • Is this general idea feasible with Clicks current internal design, that is, are we able to expose everything a custom help page creator might want without exposing too many internals? My first thought is to try fix some of the entangling in Click by having invoke(top level) call the command/group to see if help has been invoked, then in invoke tell command to dump relevant data in a very flat way. The data is passed in as kwarg to CustomHelp constructor. CustomHelp creates the help page in a very stand alone way. Invoke calls CustomHelp.render() and exits.
  • How does this impact creating error messages?

Examples

In Click

# in click
class InvocationSource(Enum, auto):
    COMMAND = auto()
    GROUP = auto()
    BOTH = auto()

class CustomHelpLoad(ABC):
    """Customization available to User"""


    @abstractmethod
    def __init__(self, arg_1, arg_2,):
        """By having a separate class, instead of dict, user can see 
        customization available on hover in editor.
        """


    @abstractmethod
    def dump(self,) -> dict:
        """Dump everything given by user to dictionary.""" 


class CustomHelp(ABC):
    """ In a relatively standalone way, make a help page.
    Must handle help pages and errors. 
    """

    @abstractmethod
    def __init__(
            self, 
            ctx: Context, 
            # Not right but too sick to fix issue.
            invocation_source_info: dict, 
            invocation_source: InvocationSource,  
            error: None|dict = None
            ):
        """ Load needed properties. 
        
        To support truly dynamic customization, the help page might need 
        any number of things, so a basically everything is injected as a 
        kwarg. Identify, what your implementation actually needs and 
        store it in a property. 
        
        Set default settings dict.
        """
        

    @abstractmethod
    def inject_settings(self, settings: CustomHelpLoad):
        """Load in the customized settings by merging default settings dict 
        with customized one.
        """
        
        #self.settings.update(CustomHelpLoad.dump())


    @abstractmethod
    def create(self):
        """Actually make the help page. Recommended to break into create error
        and create help."""
        

    @abstractmethod
    def render(self):
        """"Render output to the terminal"""

# click plugin implementation
class _ClickBuiltInHelpLoad(CustomHelpLoad):
    """"Base implementation will likely offer no customization."""
    
    def dump(self):
        return dict()


class _ClickBuiltInHelp(CustomHelp): 
    """"Implementing this will guide data and signature of ABC constructor"""

    def __init__(self):
        self.settings = {}

    def inject_settings(self, settings: _ClickBuiltInHelpLoad):
        self.settings.update(_ClickBuiltInHelpLoad.dump())

Click's own plugin classes will be private. When custom help is released, then a guide will be made that uses this as an example. But that will be just an example of how to implement the ABC. We will not expose this, so we can evolve it freely.

Plugin

class RichHelp(CustomHelp):
    """Rich implementation of help page."""

# By convention, named <CustomHelpName>Load
class RichHelpLoad(CustomHelpLoad):
    """Settings available for user customization."""


    def __init__(self, color='blue', mode='normal'):
        pass

    def dump(self):
        return dict()

User Code

import click 
from click_help_rich import RichHelp
from click_help_rich import RichHelpLoad


# drop in help customization 
@click.command(
    'one', 
    custom_help=RichHelp,
    custom_help_settings=RichHelpLoad(color='red'),
)
def one():
    pass 

# help heavily customized
# Much nicer for readability to have the option to factor into a function  
# Also this limits import time side effects. 
def rich_help_very_custom():
    return RichHelpLoad(
        color='red',
        mode='sharp',
    )


@click.command(
    'two', 
    custom_help=RichHelp,
    custom_help_settings_call=rich_help_very_custom,
)
def two():
    pass 

Rowlando13 avatar Sep 28 '25 20:09 Rowlando13