Skip to content

Modularity

Example of how to use modules, dynamic modules and linking them together to build an application.

For full documentation on modules, visit the Modules section.

Code

import asyncio
import logging
from collections.abc import AsyncIterator
from contextlib import asynccontextmanager

from waku import Application
from waku.di import Injected, Scoped, Singleton, inject
from waku.di.contrib.aioinject import AioinjectDependencyProvider
from waku.factory import ApplicationFactory
from waku.modules import DynamicModule, module

logging.basicConfig(level=logging.INFO)
logger = logging.getLogger(__name__)


# Define your providers and modules
class ConfigService:
    def get(self, option: str) -> str:
        return option


@module()
class ConfigModule:
    @classmethod
    def register(cls, env: str = 'dev') -> DynamicModule:
        # You can select providers based on `env` for example
        logger.info('Loading config for env=%s', env)
        return DynamicModule(
            parent_module=cls,
            providers=[Singleton(ConfigService)],
            exports=[ConfigService],
        )


class UserService:
    async def great(self, name: str) -> str:
        return f'Hello, {name}!'


@module(
    providers=[Scoped(UserService)],
    exports=[UserService],
)
class UserModule:
    pass


@module(imports=[UserModule])
class IAMModule:
    pass


@module(imports=[UserModule, IAMModule])
class AdminModule:
    pass


# Define the application composition root module
@module(
    imports=[
        AdminModule,
        ConfigModule.register(env='prod'),
    ],
    exports=[ConfigModule],
)
class AppModule:
    pass


# Define entrypoints
# In real world this can be FastAPI routes, etc.
@inject
async def handler(
    user_service: Injected[UserService],
    config_service: Injected[ConfigService],
) -> None:
    print(await user_service.great('World'))
    print(config_service.get('TEST=1'))


@asynccontextmanager
async def lifespan(_: Application) -> AsyncIterator[None]:
    logger.info('Lifespan startup')
    yield
    logger.info('Lifespan shutdown')


# Create application via factory
def bootstrap() -> Application:
    return ApplicationFactory.create(
        AppModule,
        dependency_provider=AioinjectDependencyProvider(),
        lifespan=[lifespan],
    )


# Run the application
# In real world this would be run by a 3rd party framework like FastAPI
async def main() -> None:
    application = bootstrap()
    async with application, application.container.context():
        await handler()  # type: ignore[call-arg]


if __name__ == '__main__':
    asyncio.run(main())