Skip to main content

Library for building modular applications

Project description

pipeline status coverage report PyPI Status PyPI Version PyPI Python PyPI License PyPI Format

Applipy

A library for building modular applications.

pip install applipy

Applipy lets you:

  • implement self-contained or interdependant Modules that you can tell applipy to load for the application through the config file
  • implement AppHandles: instances that implement application lifecycle methods
  • modules can register multiple AppHandles that will be run concurrently
  • load your application from a configuration file
  • applipy will gracefully manage the lifecycle of your application
  • define protocol handlers for a given URI scheme in configuration values. With these you can implement secrets that can be accessed by the application through the configuration.

Usage

An application can be defined by using a JSON (or YAML, if pyyaml is installed).

# dev.yaml
app:
  name: demo
  modules:
  - applipy_http.HttpModule
  - applipy_prometheus.PrometheusModule

logging.level: DEBUG

http.servers:
- host: 0.0.0.0
  port: 8080

Save a file dev.yaml with the contents in the snipet above and run the following commands:

$ pip install pyyaml applipy applipy_http applipy_prometheus
$ applipy

The configuration file above defines an application named demo that installs the applipy web module and the Prometheus module.

You can try it by going to http://localhost:8080. To see some metrics you have to call at least twice on the http://0.0.0.0:8080/metrics endpoint.

Applipy will search for a configuration file named ${APPLIPY_CONFIG_PATH}/${APPLIPY_ENV}.json (and ${APPLIPY_CONFIG_PATH}/${APPLIPY_ENV}.yaml, if pyyaml is installed). The default values are: APPLIPY_ENV=dev and APPLIPY_CONFIG_PATH=..

Another option is to indicate the file directly, using applipy -f ./path/to/file.yaml.

AppHandle

AppHandle is the interface through wich applipy manages the lifecycle of the application. An AppHandle implementation looks like this:

# demo_app.py
from asyncio import get_event_loop
from logging import Logger

from applipy import AppHandle


class MyDemoApp(AppHandle):

    def __init__(self, logger: Logger, name: str):
        self.logger = logger.getChild(name)

    async def on_init(self):
        self.logger.info('initialize resources')
        self.future = get_event_loop().create_future()

    async def on_start(self):
        self.logger.info('run long lived application here')
        await self.future

    async def on_shutdown(self):
        self.logger.info('close and release resources')
        self.future.set_result(None)

As you can see above there is three methods exposed by AppHandles that let applipy run your application.

Applipy is capable of running multiple AppHandles concurrently, taking advantage of async in python.

The lifecycle of an application follow the following diagram:

  Application.start()
           │
           ▼
  AppHandle.on_init()
           │
           ▼
  AppHandle.on_start()        applipy cli
           │                SIGINT   SIGTERM   [Your own stopping mechanism]
           ▼                   └────────┴───────┬───────────┘
start_long_running_task()                       ▼
           │                            Application.stop()
           ▼                                    │
 AppHandle.on_shutdown()◄───────────────────────┘
           │
           ▼
  wait(shutdown_timeout)
           │
           ▼
   cancel_all_tasks()

shutdown_timeout can be set in the app configuration using the key app.shutdown_timeout_seconds. Defaults to 1 second.

It is expected that AppHandle.on_shutdown() will be used to indicate any long running tasks to stop and cleanup, and do any required cleanup itself.

Generally, AppHandle implementations are added to the applipy application by including the modules they are part of and registering the AppHandle in the module configure() function.

Modules

In applipy, modules are the building blocks of an application. They allow to bind instances/classes/providers to types by exposing the an applipy_inject.Injector's bind() function, register application handles by exposing the Application's register() function and define dependencies across modules.

An example of a module implementation looks like this:

# mymodule.py

from applipy import Config, Module, LoggingModule
from demo_app import MyDemoApp


class MyModule(Module):

    def __init__(self, config: Config):
        self._config = config

    def configure(self, bind, register):
        bind(str, 'ModuleDemo')
        register(MyDemoApp)

    @classmethod
    def depends_on(cls):
        return (LoggingModule,)

The way you add modules to an application is through the configuration file by defining a list of fully qualified names of Module implementations with the key app.modules:

app:
  modules:
    - applipy_http.HttpModule
    - applipy_prometheus.PrometheusModule
    - mymodule.MyModule

Modules can only receive one parameter in their constructor and it is a Config instance, as shown in the code above. If your module does not need access to the configuration, you can just not implement a __init__ or have it not have arguments (other than self).

The configure() method is run by the applipy Application when it is started and its purpose is to allow for binding types and registering application handles. Check the extended Module documentation in /docs/module.md.

Finally, the depends_on() class method returns a tuple of the module types the module depends on. In the example above, because the application handle registered by the module requires a logging.Logger, the module declares a dependency with the LoggingModule because we know that it binds the logging.Logger type.

More

For a deeper dive on the features and details feel free to check the /docs subdirectory and the code itself!

Project details


Download files

Download the file for your platform. If you're not sure which to choose, learn more about installing packages.

Source Distribution

applipy-2.5.0.tar.gz (15.7 kB view details)

Uploaded Source

Built Distribution

applipy-2.5.0-py3-none-any.whl (12.9 kB view details)

Uploaded Python 3

File details

Details for the file applipy-2.5.0.tar.gz.

File metadata

  • Download URL: applipy-2.5.0.tar.gz
  • Upload date:
  • Size: 15.7 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/4.0.2 CPython/3.12.1

File hashes

Hashes for applipy-2.5.0.tar.gz
Algorithm Hash digest
SHA256 e23ef48f9db6b98e1f65fdb2c02640b5374097b55575a2541d91a4cd9e1eecaf
MD5 fb88b48a1b71c5ffaa231435d8f018bf
BLAKE2b-256 cfb2e1fbbaa6773b969afe483de432041ac7a610c4ac8576f8ad5ef439ddeb8e

See more details on using hashes here.

File details

Details for the file applipy-2.5.0-py3-none-any.whl.

File metadata

  • Download URL: applipy-2.5.0-py3-none-any.whl
  • Upload date:
  • Size: 12.9 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/4.0.2 CPython/3.12.1

File hashes

Hashes for applipy-2.5.0-py3-none-any.whl
Algorithm Hash digest
SHA256 0f9cc46438f6991f1b36734220a0e253cb5181efe56cc1cab332054e7eaa8d6a
MD5 b1dcd39096f59d005181698da7aeba90
BLAKE2b-256 fe62ca7b0a77e9c4ee6bd66c83c7b73733fd3ca66a70719df00c1bf8e7bc9f45

See more details on using hashes here.

Supported by

AWS AWS Cloud computing and Security Sponsor Datadog Datadog Monitoring Fastly Fastly CDN Google Google Download Analytics Microsoft Microsoft PSF Sponsor Pingdom Pingdom Monitoring Sentry Sentry Error logging StatusPage StatusPage Status page