Scaffolding for Brewblox service applications
In order to reduce code duplication between services, generic functionality is implemented here.
For an example on how to implement your own service based on brewblox-service
, see https://github.com/brewblox/brewblox-boilerplate.
Small generic tools are defined here.
brewblox_logger
can be used for creating module-specific loggers. It is not required, but will play a bit nicer with default formatting of the log.
Example:
from brewblox_service import brewblox_logger
LOGGER = brewblox_logger(__name__)
LOGGER.info('hello')
Parses commandline arguments, creates an aiohttp
application object, and runs it.
The shortest implementation is:
parser = service.create_parser('my_service')
config = service.create_config(parser)
app = service.create_app(config)
service.run_app(app)
This will get you a working web application, but it will not have any endpoints.
Applications can configure their own features, and add new commandline arguments.
Example:
class ExtendedConfig(models.BaseServiceConfig):
my_arg: str
parser = service.create_parser('my_service')
parser.add_argument('--my-arg')
config = service.create_config(parser, model=ExtendedConfig)
app = service.create_app(config)
async def setup():
device.setup(app)
api.setup(app)
service.run_app(app, setup())
Many service features are application-scoped. Their lifecycle should span multiple requests, either because they are not request-driven, or because they manage asynchronous I/O operations (such as listening to AMQP messages).
The ServiceFeature
class offers an abstract base class for this behavior. Implementing classes should define startup(app)
and shutdown(app)
functions, and those will be automatically called when the application starts up and shuts down.
Both startup()
and shutdown()
are called in an async context, making them the async counterparts of __init__()
and __del__()
functions.
Features must be constructed after the app is created, but before it starts running. (service.create_app()
and service.run_app(app)
).
It is recommended but not required to add features in an async setup function.
The add()
and get()
functions make it easy to centrally declare a feature, and then use it in any function that has a reference to the aiohttp app.
A very common use case for features is to have the equivalent of a while True
loop.
The RepeaterFeature
class handles the boilerplate involved in setting this up in an async context.
Basic MQTT messaging: publish and listen to events.
To publish data, all you need is the publish(topic, message)
function.
To listen to incoming messages, you can combine subscribe(topic)
with one or more calls to listen(topic, callback)
.
The subscribe/listen functions allow for + and # wildcards to be used.
For a detailed explanation of how to use MQTT topics, see http://www.steves-internet-guide.com/understanding-mqtt-topics/.
For the Brewblox spec on how and where to publish data, see https://brewblox.com/dev/reference/events.html.
Includes top-level convenience functions for publish(topic, message)
, listen(topic, callback)
and subscribe(topic)
.