๐ฆ A minimal dependency injection library for Python
Project description
๐ฆ DuckDI
DuckDI is a minimal, type-safe, and architecture-friendly dependency injection library for Python.
It provides a clean interface to register and resolve dependencies at runtime using a TOML-based configuration, following the duck typing principle: "if it implements the expected methods, itโs good enough."
Ideal for developers who want clarity, zero magic, and full control over dependency resolution.
๐ Features
- โ Clean and lightweight API
- โ Zero runtime dependencies
- โ Fully type-safe (no introspection magic)
- โ Supports singleton and transient resolution
- โ Uses TOML to bind interfaces to adapters
- โ
Works with
ABCand regular classes (no need for Protocols) - โ Clear and informative error messages
- โ
Environment-based configuration (
INJECTIONS_PATH)
๐ฆ Installation
With Poetry:
poetry add duckdi
Or using pip:
pip install duckdi
๐ ๏ธ Usage
1. Define an interface
from duckdi import Interface
from abc import ABC, abstractmethod
@Interface
class IUserRepository(ABC):
@abstractmethod
def get_user(self, user_id: str) -> dict: ...
2. Register an adapter
from duckdi import register
class PostgresUserRepository(IUserRepository):
def get_user(self, user_id: str) -> dict:
return {"id": user_id, "name": "John Doe"}
register(PostgresUserRepository)
You can also register it as a singleton:
register(PostgresUserRepository, is_singleton=True)
3. Create your injection payload
Create a file called injections.toml:
[injections]
"i_user_repository" = "postgres_user_repository"
4. Set the environment variable
Set the injection file path using the INJECTIONS_PATH environment variable:
export INJECTIONS_PATH=./injections.toml
5. Resolve your dependencies
from duckdi import Get
repo = Get(IUserRepository)
user = repo.get_user("123")
print(user) # {'id': '123', 'name': 'John Doe'}
๐ฅ Error Handling
MissingInjectionPayloadError
Raised when no injection payload file is found at the specified path.
InvalidAdapterImplementationError
Raised when the adapter registered does not implement the expected interface.
InterfaceAlreadyRegisteredError
Raised when trying to register the same interface twice.
AdapterAlreadyRegisteredError
Raised when the same adapter is registered more than once.
๐ Project Structure
duckdi/
โโโ pyproject.toml
โโโ README.md
โโโ src/
โ โโโ duckdi/
โ โโโ __init__.py
โ โโโ cli.py
โ โโโ duck.py
โ โโโ errors/
โ โ โโโ __init__.py
โ โ โโโ invalid_adapter_implementation_error.py
โ โ โโโ interface_already_registered_error.py
โ โ โโโ adapter_already_registered_error.py
โ โ โโโ missing_injection_payload_error.py
โ โโโ utils/
โ โ โโโ __init__.py
โ โ โโโ buffer_readers.py
โ โ โโโ to_snake.py
โ โโโ injections/
โ โโโ injections_container.py
โ โโโ injections_payload.py
โโโ tests/
โโโ test_interface.py
โโโ test_register.py
โโโ test_get.py
๐งฉ Advanced Example
You can register multiple adapters and resolve them dynamically based on the TOML mapping:
from duckdi import Interface, register, Get
@Interface
class INotifier:
def send(self, msg: str): ...
class EmailNotifier(INotifier):
def send(self, msg: str):
print(f"Sending email: {msg}")
register(EmailNotifier)
# injections.toml
# [injections]
# "i_notifier" = "email_notifier"
notifier = Get(INotifier)
notifier.send("Hello from DuckDI!")
๐งช Testing
To run tests:
pytest
Or via Makefile:
make test
To check static typing:
make check
๐ License
Licensed under the MIT License.
See the LICENSE file for more information.
๐ค Author
Made with โค๏ธ by PhePato
Pull requests, issues and ideas are always welcome!
Project details
Release history Release notifications | RSS feed
Download files
Download the file for your platform. If you're not sure which to choose, learn more about installing packages.
Source Distribution
Built Distribution
Filter files by name, interpreter, ABI, and platform.
If you're not sure about the file name format, learn more about wheel file names.
Copy a direct link to the current filters
File details
Details for the file duckdi-0.1.11.tar.gz.
File metadata
- Download URL: duckdi-0.1.11.tar.gz
- Upload date:
- Size: 6.4 kB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: poetry/2.1.3 CPython/3.13.4 Darwin/24.0.0
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
72fd00b46a63536e32a6f36b894a547e5fafc0df2c7de9c833c47bb0cb420d14
|
|
| MD5 |
28a6866f7a9cde2ea358683a4f75217d
|
|
| BLAKE2b-256 |
ff9b8d4b28447d12188087f702b8721fbd81182b7e1d77857988b68ca07c1a73
|
File details
Details for the file duckdi-0.1.11-py3-none-any.whl.
File metadata
- Download URL: duckdi-0.1.11-py3-none-any.whl
- Upload date:
- Size: 11.4 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: poetry/2.1.3 CPython/3.13.4 Darwin/24.0.0
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
eec10163f03583e594459387232670bf60531f919944ca7f760fdb25b88065be
|
|
| MD5 |
882a51fa35bc54523b288fa63994e84f
|
|
| BLAKE2b-256 |
669dd2fd258b236946d6ffbdaca439bc7e739f68bb5373f2bd61b9f39d0eeb2b
|