Apps¶
Apps can be added to commands using the quo.console.app()
decorator.
Apps in quo are profoundly configurable and ought not to be mistaken for positional arguments.
How to name Apps
¶
For the purpose of uniformity, a name is chosen in the following order
In the event that a name is not prefixed, therefore it is used as the Python argument name
2. If there is at least one name prefixed with – the first one given is used as the name. To get the Python argument name, the chosen name is converted to lower case, up to two dashes are removed as the prefix, and other dashes are converted to underscores.
Note
Apps are given as position arguments to the decorator.
from quo import print
from quo.console import app
from quo.console import command
@command()
@app('-s', '--string-to-echo')
def echo(string_to_echo):
print(string_to_echo)
from quo import print
from quo.console import app
from quo.console import command
@command()
@app('-s', '--string-to-echo', 'string')
def echo(string):
print(string)
"-f", "--foo-bar"
, the name isfoo_bar
"-x"
, the name isx
"-f", "--filename", "dest"
, the name isdest
"--CamelCase"
, the name iscamelcase
"-f", "--fb"
, the name isf
"-f", "--foo-bar"
, the name isf
Basic Value Apps
¶
The most basic app is a value app. These apps accept one
argument which is a value. If no type is provided, the type of the default
value is used. If no default value is provided, the type is assumed to be
STRING
. Unless a name is explicitly specified, the name of the
parameter is the first long option defined; otherwise the first short one is
used. By default, apps are not required, however to make an app required,
simply pass in required=True as an argument to the decorator.
from quo import print
from quo.console import app
from quo.console import command
@command()
@app('-n', default=1)
def dots(n):
print('.' * n)
# How to make an app required
from quo import print
from quo.console import app
from quo.console import command
@command()
@app('-n', required=True, type=int)
def dots(n):
print('.' * n)
# How to use a Python reserved word such as `from` as a parameter
from quo import print
from quo.console import app
from quo.console import command
@command()
@app('--from', '-f', '--from_')
@app('--to', '-t')
def reserved_param_name(from_, to):
print(f"from {from_} to {to}")
In this case the app is of type INT
because the default value
is an integer.
To show the default values when showing command help, use show_default=True
from quo import print
from quo.console import app
from quo.console import command
@command()
@app('-n', default=1, show_default=True)
def dots(n):
print('.' * n)
Multi Value Apps
¶
Sometimes, you have apps that take more than one arg. For apps,
only a fixed number of arguments is supported. This can be configured by
the nargs
parameter. The values are then stored as a tuple.
from quo import print
from quo.console import app
from quo.console import command
@command()
@app('--pos', nargs=2, type=float)
def findme(pos):
a, b = pos
print(f"{a} / {b}")
Tuples as Multi Value Apps
¶
As you can see that by using nargs set to a specific number each item in the resulting tuple is of the same type. This might not be what you want. Commonly you might want to use different types for different indexes in the tuple. For this you can directly specify a tuple as type:
from quo import print
from quo.console import app, command
@command()
@app('--item', type=(str, int))
def putitem(item):
name, id = item
print(f"name={name} id={id}")
By using a tuple literal as type, nargs gets automatically set to the
length of the tuple and the quo.types.Tuple
type is automatically
used. The above example is thus equivalent to this:
from quo import print
from quo.console import app, command
from quo.types import Tuple
@quo.command()
@quo.app('--item', nargs=2, type=Tuple([str, int]))
def putitem(item):
name, id = item
print(f"name={name} id={id}")
Multiple Apps
¶
Similarly to nargs
, there is also the case of wanting to support a
parameter being provided multiple times and have all the values recorded –
not just the last one. For instance, git commit -m foo -m bar
would
record two lines for the commit message: foo
and bar
. This can be
accomplished with the multiple
flag:
Example:
from quo import print
from quo.console import app, command
@command()
@app('--message', '-m', multiple=True)
def commit(message):
print('\n'.join(message))
When passing a default
with multiple=True
, the default value
must be a list or tuple, otherwise it will be interpreted as a list of
single characters.
@app("--format", multiple=True, default=["json"])
Counting
¶
In some very rare circumstances, it is interesting to use the repetition of apps to count an integer up. This can be used for verbosity flags, for instance:
from quo import print
from quo.console import app, command
@command()
@app('-v', '--verbose', count=True)
def log(verbose):
print(f"Verbosity: {verbose}")
Boolean Flags
¶
Boolean flags are apps that can be enabled or disabled. This can be
accomplished by defining two flags in one go separated by a slash (/
)
for enabling or disabling the app. (If a slash is in an app string,
quo automatically knows that it’s a boolean flag and will pass
is_flag=True
.) quo always wants you to provide an enable
and disable flag so that you can change the default later.
Example:
import sys
from quo import print
from quo.console import app, command
@command()
@app('--shout/--no-shout', default=False)
def info(shout):
rv = sys.platform
if shout:
rv = rv.upper() + '!!!!111'
print(rv)
If you really don’t want an off-switch, you can just define one and manually inform quo that something is a flag:
import sys
from quo import print
from quo.console import app, command
@command()
@app('--shout', is_flag=True)
def info(shout):
rv = sys.platform
if shout:
rv = rv.upper() + '!!!!111'
print(rv)
Note that if a slash is contained in your app already (for instance, if
you use Windows-style parameters where /
is the prefix character), you
can alternatively split the parameters through ;
instead:
from quo import print
from quo.console import app, command
@command()
@app('/debug;/no-debug')
def log(debug):
print(f"debug={debug}")
if __name__ == '__main__':
log()
If you want to define an alias for the second apo only, then you will need to use leading whitespace to disambiguate the format string:
Example:
import sys
from quo import print
from quo.console import app, command
@command()
@app('--shout/--no-shout', ' /-S', default=False)
def info(shout):
rv = sys.platform
if shout:
rv = rv.upper() + '!!!!111'
print(rv)
Feature Switches
¶
In addition to boolean flags, there are also feature switches. These are
implemented by setting multiple apps to the same parameter name and
defining a flag value. Note that by providing the flag_value
parameter,
quo will implicitly set is_flag=True
.
To set a default flag, assign a value of True to the flag that should be the default.
import sys
from quo import print
from quo.console import app, command
@command()
@app('--upper', 'transformation', flag_value='upper',default=True)
@app('--lower', 'transformation', flag_value='lower')
def info(transformation):
print(getattr(sys.platform, transformation)())
Choice Apps
¶
Sometimes, you want to have a parameter be a choice of a list of values.
In that case you can use Choice
type. It can be instantiated
with a list of valid values. The originally passed choice will be returned,
not the str passed on the command line. Token normalization functions and
case_sensitive=False
can cause the two to be different but still match.
Example:
from quo import print
from quo.console import app, command
from quo.types import Choice
@command()
@qapp('--hash-type', type=Choice(['MD5', 'SHA1'], case_sensitive=False))
def digest(hash_type):
print(hash_type)
Only pass the choices as list or tuple. Other iterables (like generators) may lead to unexpected results.
Choices work with apps that have multiple=True
. If a default
value is given with multiple=True
, it should be a list or tuple of
valid choices.
Choices should be unique after considering the effects of
case_sensitive
and any specified token normalization function.
Prompting
¶
In some cases, you want parameters that can be provided from the command line, but if not provided, ask for user input instead. This can be implemented with quo by defining a prompt string.
Example:
from quo import print
from quo.console import app, command
@command()
@app('--name', prompt=True)
def hello(name):
print(f"Hello {name}!")
If you are not happy with the default prompt string, you can ask for a different one:
from quo import print
from quo.console import app, command
@command()
@app('--name', prompt='Your name please')
def hello(name):
print(f"Hello {name}!")
It is advised that prompt not be used in conjunction with the multiple flag set to True. Instead, prompt in the function interactively.
By default, the user will be prompted for an input if one was not passed through the command line. To turn this behavior off, see Optional Value.
Password Prompts
¶
quo also supports hidden prompts and asking for confirmation. This is useful for password input:
import codecs
from quo import print
from quo.console import app, command
@command()
@app("--password", prompt=True, hide=True, affirm==True)
def encode(password):
print(f"encoded: {codecs.encode(password, 'rot13')}")
Dynamic Defaults for Prompts
¶
The auto_envvar_prefix
and default_map
apps for the context
allow the program to read option values from the environment or a
configuration file. However, this overrides the prompting mechanism, so
that the user does not get the app to change the value interactively.
If you want to let the user configure the default value, but still be prompted if the app isn’t specified on the command line, you can do so by supplying a callable as the default value. For example, to get a default from the environment:
import os
from quo import print
from quo.console import app, command
@command()
@app("--username", prompt= True, default=lambda: os.environ.get("USER", ""))
def hello(username):
print(f"Hello, {username}!")
To describe what the default value will be, set it in show_default
.
import os
from quo import print
from quo.console import app, command
@command()
@app("--username", prompt=True, default=lambda: os.environ.get("USER", ""), show_default="current user")
def hello(username):
print(f"Hello, {username}!")
Callbacks and Eager Apps
¶
Sometimes, you want a parameter to completely change the execution flow.
For instance, this is the case when you want to have a --version
parameter that prints out the version and then exits the application.
In such cases, you need two concepts: eager parameters and a callback. An
eager parameter is a parameter that is handled before others, and a
callback is what executes after the parameter is handled. The eagerness
is necessary so that an earlier required parameter does not produce an
error message. For instance, if --version
was not eager and a
parameter --foo
was required and defined before, you would need to
specify it for --version
to work. For more information, see
callback-evaluation-order.
A callback is a function that is invoked with two parameters: the current
Clime
and the value. The context provides some useful features
such as quitting the application and gives access to other already
processed parameters.
Here an example for a --version
flag:
from quo import print
from quo.console import app, command
def print_version(clime, param, value):
if not value or clime.parse:
return
print('Version 1.0')
clime.exit()
@command()
@app('--version', is_flag=True, callback=print_version, expose_value=False, is_eager=True)
def hello():
print('Hello World!')
The expose_value parameter prevents the pretty pointless version
parameter from being passed to the callback. If that was not specified, a
boolean would be passed to the hello script. The parse
flag is applied to the context if quo wants to parse the command line
without any destructive behavior that would change the execution flow. In
this case, because we would exit the program, we instead do nothing.
Yes Parameters
¶
For dangerous operations, it’s very useful to be able to ask a user for
confirmation. This can be done by adding a boolean --yes
flag and
asking for confirmation if the user did not provide it and to fail in a
callback:
from quo import echo
from quo.console import app, command
def abort_if_false(clime, param, value):
if not value:
clime.abort()
@command()
@app('--yes', is_flag=True, callback=abort_if_false, expose_value=False, prompt='Are you sure you want to drop the db?')
def dropdb():
echo('Dropped all tables!')
Values from Environment Variables
¶
A very useful feature of quo is the ability to accept parameters from
environment variables in addition to regular parameters. This allows
tools to be automated much easier. For instance, you might want to pass
a configuration file with a --config
parameter but also support exporting
a TOOL_CONFIG=hello.cfg
key-value pair for a nicer development
experience.
This is supported by quo in two ways. One is to automatically build
environment variables which is supported for apps only. To enable this
feature, the auto_envvar_prefix
parameter needs to be passed to the
script that is invoked. Each command and parameter is then added as an
uppercase underscore-separated variable. If you have a subcommand
called run
taking an app called reload
and the prefix is
WEB
, then the variable is WEB_RUN_RELOAD
.
Example usage:
from quo import echo
from quo.console import app, command
@command()
@app('--username')
def greet(username):
echo(f'Hello {username}!')
if __name__ == '__main__':
greet(auto_envvar_prefix='GREETER')
When using auto_envvar_prefix
with command groups, the command name
needs to be included in the environment variable, between the prefix and
the parameter name, i.e. PREFIX_COMMAND_VARIABLE
. If you have a
subcommand called run-server
taking an app called host
and
the prefix is WEB
, then the variable is WEB_RUN_SERVER_HOST
.
Example:
from quo import echo
from quo.console import app, tether
@tether()
@app('--debug/--no-debug')
def cli(debug):
echo(f"Debug mode is {'on' if debug else 'off'}")
@cli.command()
@app('--username')
def greet(username):
echo(f"Hello {username}!")
if __name__ == '__main__':
cli(auto_envvar_prefix='GREETER')
The second option is to manually pull values in from specific environment variables by defining the name of the environment variable on the app.
Example usage:
from quo import echo
from quo.console import app, command
@command()
@app('--username', envvar='USERNAME')
def greet(username):
echo(f"Hello {username}!")
if __name__ == '__main__':
greet()
In that case it can also be a list of different environment variables where the first one is picked.
Multiple Values from Environment Values
¶
As apps can accept multiple values, pulling in such values from
environment variables (which are strings) is a bit more complex. The way
quo solves this is by leaving it up to the type to customize this
behavior. For both multiple
and nargs
with values other than
1
, quo will invoke the ParamType.split_envvar_value()
method to
perform the splitting.
The default implementation for all types is to split on whitespace. The
exceptions to this rule are the quo.types.File
and quo.types.Path
types
which both split according to the operating system’s path splitting rules.
On Unix systems like Linux and OS X, the splitting happens for those on
every colon (:
), and for Windows, on every semicolon (;
).
Example usage:
from quo import echo
from quo.console import app, command
from quo.types import Path
@command()
@app('paths', '--path', envvar='PATHS', multiple=True, type=Path())
def perform(paths):
for path in paths:
echo(path)
if __name__ == '__main__':
perform()
Other Prefix Characters
¶
quo can deal with alternative prefix characters other than --
for
apps. This is for instance useful if you want to handle slashes as
parameters /
or something similar.
from quo import echo
from quo.console import app, command
@command()
@app('+w/-w')
def chmod(w):
echo(f"writable={w}")
if __name__ == '__main__':
chmod()
Note that if you are using /
as prefix character and you want to use a
boolean flag you need to separate it with ;
instead of /
:
from quo import echo
from quo.console import app, command
@command()
@app('/debug;/no-debug')
def log(debug):
echo(f"debug={debug}")
if __name__ == '__main__':
log()
Range Apps
¶
The IntRange
type extends the INT
type to ensure the
value is contained in the given range. The FloatRange
type does
the same for FLOAT
.
If min
or max
is omitted, that side is unbounded. Any value in
that direction is accepted. By default, both bounds are closed, which
means the boundary value is included in the accepted range. min_open
and max_open
can be used to exclude that boundary from the range.
If clamp
mode is enabled, a value that is outside the range is set
to the boundary instead of failing. For example, the range 0, 5
would return 5
for the value 10
, or 0
for the value -1
.
When using FloatRange
, clamp
can only be enabled if both
bounds are closed (the default).
from quo import echo
from quo.console import app, command
from quo.types import IntRange
@command()
@app("--count", type= IntRange(0, 20, clamp=True))
@app("--digit", type= IntRange(0, 9))
def repeat(count, digit):
echo(str(digit) * count)
Callbacks for Validation
¶
If you want to apply custom validation logic, you can do this in the parameter callbacks. These callbacks can both modify values as well as raise errors if the validation does not work. The callback runs after type conversion. It is called for all sources, including prompts.
from quo import echo
from quo.console import app, command
from quo.types import BadParameter, UNPROCESSED
def validate_rolls(clime, param, value):
if isinstance(value, tuple):
return value
try:
rolls, _, dice = value.partition("d")
return int(dice), int(rolls)
except ValueError:
raise BadParameter("format must be 'NdM'")
@command()
@app( "--rolls", type = UNPROCESSED, callback=validate_rolls, default="1d6", prompt=True)
def roll(rolls):
sides, times = rolls
echo(f"Rolling a {sides}-sided dice {times} time(s)")
Optional Value
¶
Providing the value to an app can be made optional, in which case
providing only the app’s flag without a value will either show a
prompt or use its flag_value
.
Setting is_flag=False, flag_value=value
tells quo that the app
can still be passed a value, but if only the flag is given the
flag_value
is used.
from quo import print
from quo.console import app
from quo.console import command
@command()
@app("--name", is_flag=False, flag_value="Flag", default="Default")
def hello(name):
print(f"Hello, {name}!")
If the app has prompt
enabled, then setting
prompt_required=False
tells quo to only show the prompt if the
app’s flag is given, instead of if the app is not provided at all.
from quo import print
from quo.console import app, command
@command()
@app('--name', prompt=True, prompt_required=False, default="Default")
def hello(name):
print(f"Hello {name}!")
If required=True
, then the option will still prompt if it is not
given, but it will also prompt if only the flag is given.