Application Controllers¶
Cement defines a controller interface called IController, but does not enable any default handlers that implement the interface.
Using application controllers is not necessary, but enables rapid development
by wrapping pieces of the framework like adding arguments, and linking
commands with functions to name a few. The examples below use the
CementBaseController
for examples. It is important to note that this
class also requires that your application’s argument_handler be the
ArgParseArgumentHandler
. That said, the CementBaseController
is
relatively useless when used directly and therefore should be used as a Base
class to create your own application controllers from.
The following controllers are included and maintained with Cement:
Please reference the IController interface documentation for writing your own controller.
Example Application Base Controller¶
This example demonstrates the use of application controllers that handle command dispatch and rapid development.
from cement.core import backend,
from cement.core.foundation import CementApp
from cement.core.controller import CementBaseController, expose
# define an application base controller
class MyAppBaseController(CementBaseController):
class Meta:
label = 'base'
description = "My Application does amazing things!"
epilog = "This is the text at the bottom of --help."
config_defaults = dict(
foo='bar',
some_other_option='my default value',
)
arguments = [
(['-f', '--foo'],
dict(action='store', help='the notorious foo option')),
(['-C'],
dict(action='store_true', help='the big c option')),
]
@expose(hide=True, aliases=['run'])
def default(self):
self.app.log.info('Inside base.default function.')
if self.app.pargs.foo:
self.app.log.info("Recieved option 'foo' with value '%s'." % \
self.app.pargs.foo)
@expose(help="this command does relatively nothing useful.")
def command1(self):
self.app.log.info("Inside base.command1 function.")
@expose(aliases=['cmd2'], help="more of nothing.")
def command2(self):
self.app.log.info("Inside base.command2 function.")
class MyApp(CementApp):
class Meta:
label = 'example'
base_controller = MyAppBaseController
with MyApp() as app:
app.run()
As you can see, we’re able to build out the core functionality of our app via a controller class. Lets see what this looks like:
$ python example.py --help
usage: example.py <CMD> -opt1 --opt2=VAL [arg1] [arg2] ...
My Application does amazing things!
commands:
command1
this command does relatively nothing useful.
command2 (aliases: cmd2)
more of nothing.
optional arguments:
-h, --help show this help message and exit
--debug toggle debug output
--quiet suppress all output
--foo FOO the notorious foo option
-C the big C option
This is the text at the bottom of --help.
$ python example2.py
INFO: Inside base.default function.
$ python example2.py command1
INFO: Inside base.command1 function.
$ python example2.py cmd2
INFO: Inside base.command2 function.
Additional Controllers and Namespaces¶
Any number of additional controllers can be added to your application after a
base controller is created. Additionally, these controllers can be
stacked
onto the base controller (or any other controller) in one of two
ways:
embedded
- The controllers commands and arguments are included under the parent controllers name space.nested
- The controller label is added as a sub-command under the parent controllers namespace (effectively this is a sub-command with additional sub-sub-commands under it)
For example, The base
controller is accessed when calling example.py
directly. Any commands under the base
controller would be accessible as
example.py <cmd1>
, or example.py <cmd2>
, etc. An embedded
controller will merge its commands and options into the base
controller
namespace and appear to be part of the base controller... meaning you would
still access the embedded
controllers commands as
example.py <embedded_cmd1>
, etc (same for options).
For nested
controllers, a prefix will be created with that controllers
label under its parents namespace. Therefore you would access that
controllers commands and options as
example.py <controller_label> <controller_cmd1>
.
See the Multiple Stacked Controllers example for more help.