Clinner
- Version: 1.12.3
- Status: Production/Stable
- Author: José Antonio Perdiguero López
Clinner is a library that provides some useful tools to create command line interfaces for your application
Check Clinner docs.
Features
Can define commands in multiple way:
- List of shell commands such as
["docker build", "docker push"]
. - Python functions.
- Python async functions.
Clinner provides a set of commands ready to use like:
- Black.
- Flake8.
- Isort.
- Nosetest.
- Prospector.
- Pytest.
- Sphinx.
- Tox.
Hooks for injecting variables or add global arguments to your script.
Quick start
Install this package using pip:
pip install clinner
Create a command
from clinner.command import command
@command
def foo(*args, **kwargs):
return True
Create a main file:
from clinner.run.main import Main
if __name__ == '__main__':
sys.exit(Main().run())
Commands
Commands are declared using a decorator to register given functions. Commands are functions with the follow parameters:
func
: Function that will be called when command would be executed.command_type
: Type of the command, could be a bash or python command.args
: Parser arguments for this command.parser_opts
: Command subparser's keywords, such as description.
This decorator allows to be used as a common decorator without arguments, where default type (python) will be used:
@command
def foobar(*args, **kwargs):
pass
Or specifying the type:
@command(command_type=CommandType.PYTHON)
def foobar(*args, **kwargs):
pass
But also is possible to provide command line arguments, as expected by argparse.ArgumentParser.add_argument:
@command(args=((('-f', '--foo'), {'help': 'Foo argument that does nothing'}),
(('--bar',), {'action': 'store_true', 'help': 'Bar argument stored as True'})),
parser_opts={'title': 'foobar_command', 'help': 'Help for foobar_command'})
def foobar(*args, **kwargs):
pass
All commands will be registered in a command register that can be accessed through command.register
. Each entry in
this register is a dictionary with the fields declared at the beginning of this section.
Shell command
Example of running ls -la
shell command.
@command(command_type=CommandType.SHELL)
def lsla(*args, **kwargs):
return [shlex.split("ls -la")]
Python function
Run a python function.
@command
def foo(*args, **kwargs):
return "foo"
Python async function
Run a python async function.
@command
async def bar(*args, **kwargs):
await asyncio.sleep(1)
return "bar"
Main
A main class is defined to ease the creation of command line applications. This class follows the process:
-
Create a parser using argparse.ArgumentParser
for the application:
a) Calling all add_arguments(parser)
methods from all super classes, e.g: clinner.mixins.HealthCheckMixin
.
b) Addding a subparser for each command with their specific arguments.
-
Parse arguments using the argument parser created previously.
-
Inject variables into environment calling all super classes methods whose name starts with inject_
.
Examples
Some Clinner examples.
Simple Main
Example of a simple main with two defined commands foo
and bar
.
import shlex
import sys
from clinner.command import command, Type as CommandType
from clinner.run.main import Main
@command(command_type=CommandType.SHELL,
args=(('-i', '--input'),
('-o', '--output')),
parser_opts={'help': 'Foo command'})
def foo(*args, **kwargs):
"""List of foo commands"""
ls_cmd = shlex.split('ls')
wc_cmd = shlex.split('wc')
wc_cmd += [kwargs['input'], kwargs['output']]
return [ls_cmd, wc_cmd]
@command(command_type=CommandType.PYTHON,
parser_opts={'help': 'Bar command'})
def bar(*args, **kwargs):
"""Do a bar."""
return True
if __name__ == '__main__':
sys.exit(Main().run())
Builder Main
Example of main module with build utilities such as unit tests, lint, sphinx doc, tox and dist packaging:
import sys
from clinner.run import Main
class Build(Main):
commands = (
'clinner.run.commands.black.black',
'clinner.run.commands.flake8.flake8',
'clinner.run.commands.isort.isort',
'clinner.run.commands.pytest.pytest',
'clinner.run.commands.sphinx.sphinx',
'clinner.run.commands.tox.tox',
)
if __name__ == '__main__':
sys.exit(Build().run())
Check Clinner docs to see more advanced examples.