Providers¶
Introduction¶
Providers are the core of waku
dependency injection system.
Idea behind a provider is that it can be injected as a dependency into other provider constructors,
allowing objects to form various relationships with each other.
waku
responsibility is to "wire up" all the providers using DI framework and manage the lifecycle its lifecycle.
This way you can focus on writing your application logic.
Dependency Injection¶
waku
is designed to be modular and extensible.
To support this principle, it provides a flexible dependency injection (DI) system that integrates seamlessly
with various DI frameworks. waku
itself acts like an IoC-container,
allowing you to register and resolve dependencies using modules system.
Note
Instead of relying on a specific DI framework, waku
uses an interface called DependencyProvider
.
This allows you to choose any DI framework you prefer (see Included Dependency Providers)
or even create your own provider.
What is Dependency Injection?¶
Dependency Injection (DI) is a design pattern that addresses the issue of tightly coupled code by decoupling the creation and management of dependencies from the classes that rely on them. In traditional approaches, classes directly instantiate their dependencies, resulting in rigid, hard-to-maintain code. DI solves this problem by enabling dependencies to be supplied externally, typically through mechanisms like constructor or setter injection.
By shifting the responsibility of dependency management outside the class, DI promotes loose coupling, allowing classes to focus on their core functionality rather than how dependencies are created. This separation enhances maintainability, testability, and flexibility, as dependencies can be easily swapped or modified without altering the class's code. Ultimately, DI improves system design by reducing interdependencies and making code more modular and scalable.
Manual DI Example
Here, a MockClient
is injected into Service
, making it easy to test Service
in isolation without relying
on a real client implementation.
What is IoC-container?¶
An IoC container is a framework that automates object creation and dependency management based on the Inversion of Control (IoC) principle. It centralizes the configuration and instantiation of components, reducing tight coupling and simplifying code maintenance. By handling dependency resolution, an IoC container promotes modular, testable, and scalable application design.
With power of IoC-container you can leverage all the benefits of DI without manually managing dependencies.
Providers¶
Provider
is an object that holds dependency metadata, such as its type, lifetime scope and factory.
In waku
there are four types of providers, for one for each scope:
Each provider take two arguments:
factory
: type or callable that returns or yields an instance of the dependency.type_
: type of the dependency. If not provided, it will be inferred from the factory function's return type.
Note
Object
provider is a special case, it first argument named object
instead of a factory
because you should
pass already instantiated object directly, not a factory function.
Scopes¶
waku
supports four different lifetime scopes for providers, inspired by
the service lifetimes
in .NET Core’s DI system.
Transient¶
Dependency defined with the Transient
provider are created each time they’re requested.
Scoped¶
Dependency defined with the Scoped
provider are created once per dependency provider context entrance and disposed
when the context exits.
Singleton¶
Dependency defined with the Singleton
provider are created the first time they’re requested and disposed when the
application lifecycle ends.
Object¶
Dependency defined with the Object
provider behave like Singleton
, but you must provide the implementation instance
directly to the provide and manage its lifecycle manually, outside the IoC-container.
Where and how to inject dependencies?¶
To inject dependencies with waku
you need:
- Register them to
providers
with desired scope in modules. - Identify your application entrypoints and decorate them with
@inject
. - Add dependencies as arguments to your entrypoint signature using
Injected
type hint.
Included Dependency Providers¶
waku
includes out-of-the-box support for several popular DI frameworks through its dependency provider system.
Aioinject¶
waku
dependency provider interface is heavily inspired by Aioinject,
making it our recommended default choice.
Aioinject integrates seamlessly with waku
and offers all the necessary features:
- Support for all providers scopes (transient, singleton, scoped, object)
- Container lifecycle management
- Providers overriding
- Custom context passing
Available by installing waku
with aioinject
extra or by directly installing aioinject
:
Basic Usage¶
Custom Container Configuration¶
You can provide your own pre-configured aioinject
container:
Dishka¶
Currently not supported but planned.
Writing Custom Dependency Provider¶
To create custom dependency provider you need to implement DependencyProvider
interface.