func_argparse
                                
                                 func_argparse copied to clipboard
                                
                                    func_argparse copied to clipboard
                            
                            
                            
                        Generate nice CLI from a function signature.
func_argparse
Generate a nice command line interface for a list of functions or a module, leveraging your doc-strings and type annotations.
Never worry about your Argparser being out of sync with your code.
Try it with pip install func_argparse.
Example
In a 'hello.py' file:
"""Say hello or goodbye to the user."""
import func_argparse
def hello(user: str, times: int = None):
    """Say hello.
    Arguments:
        user: name of the user
    """
    print(f"Hello {user}" * (1 if times is None else times))
def bye(user: str, see_you: float = 1.0):
    """Say goodbye."""
    print(f"Goodbye {user}, see you in {see_you:.1f} days")
if __name__ == "__main__":
    func_argparse.main()
From CLI:
$ python hello.py hello --user gwenzek
Hello gwenzek
$ python hello.py hello --user gwenzek --times 2
Hello gwenzekHello gwenzek
$ python hello.py bye --user gwenzek --see_you 12.345
Goodbye gwenzek, see you in 12.3 days
$ python hello.py hello -u gwenzek -t 1
Hello gwenzek
$ python hello.py --help
usage: hello.py [-h] {hello,bye} ...
Say hello or goodbye to the user.
positional arguments:
  {hello,bye}
    hello      Say hello.
    bye        Say goodbye.
optional arguments:
  -h, --help   show this help message and exit
$ python hello.py bye --help
usage: hello.py bye [-h] -u USER [-s SEE_YOU]
optional arguments:
  -h, --help            show this help message and exit
  -u USER, --user USER
  -s SEE_YOU, --see_you SEE_YOU
                        (default=1.0)
Gotchas
- func_argparsegenerates classics- argparse.Argparser, you can mix and match them with hand-written parsers.
- func_argparse.main()creates one CLI command by "public" function from a file / module.
- func_argparse.main(status, push)creates one CLI command for the two functions- statusand- push.
- func_argparse.main(ls=list_files)creates one CLI command- lsfor the function- list_files.
- Use func_argparse.single_main(my_main)if you only have one entry point in your file.
- All functions arguments need a type hint.
- Arguments without default value will be marked as required.
- A boolean argument awill generate two flags:--aand--no-a.
- A boolean argument with no default value will be assumed to default to False.
- The first argument starting with letter awill also be available with the flag-a.
- The function docstring will be parsed to extract the help messages.
- First line is used as help message for the function
- First line starting with awill be used to extract the documentation for argumenta. Spaces, dashes and columns will be stripped before displaying.
 
- Some kind of functions (notably builtins and C-function) can't be inspected and we can't generate Argparser for them.
- You can't have a function argument named __command.
- If you don't like the generated parser, you can modify it using overridefunction.
Alternatives
Here are other alternatives you might be interested in.
- argparse: the builtin library upon which func_argparseis built. Grants a very precise control on the CLI but is a bit verbose and prone to go out-of-sync with the code.
- fire: also generates parser with introspection but doesn't leverage types. So the types of arguments is determined at parse time. Can generate completion files.
- click: uses function annotations to generate the CLI.
Contibuting
All contributions are welcome.
Code formatting is enforced with isort, black.
Types annotations are required for the main module and checked with mypy.
Tests are run with pytest.
Run pip install ".[dev]" to install the required modules
Run ./tools.sh all to format your code and run mypy and pytest.
TODOs
- Add all parameters from add_argumenttooverride
- Make it easy to parse a comma separated list of argument
- Add an OOP interface by sub-classing ArgumentParser