Skip to main content

Lightweight dependency injection container for Python.

Project description

philiprehberger-di

Tests PyPI version Last updated

Lightweight dependency injection container for Python.

Installation

pip install philiprehberger-di

Usage

from philiprehberger_di import Container

container = Container()
container.register(Logger)
logger = container.resolve(Logger)

Singletons

container.register(Database, singleton=True)

a = container.resolve(Database)
b = container.resolve(Database)
assert a is b

Custom Factories

container.register(Cache, factory=lambda: Cache(max_size=256))
cache = container.resolve(Cache)

Recursive Resolution

class Service:
    def __init__(self, db: Database, logger: Logger) -> None:
        self.db = db
        self.logger = logger

container.register(Database)
container.register(Logger)
container.register(Service)
service = container.resolve(Service)  # db and logger are injected automatically

Inject Decorator

from philiprehberger_di import Container, inject

container = Container()
container.register(Logger, singleton=True)

@inject(container)
def handle_request(logger: Logger) -> str:
    logger.log("request handled")
    return "ok"

handle_request()  # logger is resolved and injected automatically

Lifecycle Hooks

container.register(
    Database,
    singleton=True,
    on_create=lambda db: db.connect(),
    on_destroy=lambda db: db.disconnect(),
)

db = container.resolve(Database)  # on_create called after creation
container.reset()                 # on_destroy called before clearing singletons

Circular Dependency Detection

The container detects circular dependencies during resolution and raises a CircularDependencyError with the full dependency chain:

from philiprehberger_di import CircularDependencyError

class A:
    def __init__(self, b: B) -> None: ...

class B:
    def __init__(self, a: A) -> None: ...

container.register(A)
container.register(B)

try:
    container.resolve(A)
except CircularDependencyError as e:
    print(e)  # Circular dependency detected: A -> B -> A
    print(e.chain)  # [A, B, A]

Scoped Lifetime

Services registered with Lifetime.SCOPED are singletons within a scope but differ across scopes:

from philiprehberger_di import Container, Lifetime

container = Container()
container.register(RequestContext, lifetime=Lifetime.SCOPED)
container.register(Logger, lifetime=Lifetime.SINGLETON)

with container.create_scope() as scope:
    ctx1 = scope.resolve(RequestContext)
    ctx2 = scope.resolve(RequestContext)
    assert ctx1 is ctx2  # same within the scope

# on_destroy hooks are called when the scope exits

Lazy Resolution

Defer construction of an expensive service until first use:

from philiprehberger_di import Container, Lifetime

container = Container()
container.register(ExpensiveClient, lifetime=Lifetime.SINGLETON)

proxy = container.lazy(ExpensiveClient)
# ExpensiveClient is NOT yet constructed

result = proxy.do_thing()  # constructed on first access, cached afterwards

API

Function / Class Description
Container() Create a new dependency injection container
container.register(cls, factory?, singleton?, lifetime?, on_create?, on_destroy?) Register a class with optional factory, lifetime, and lifecycle hooks
container.resolve(cls) Resolve an instance, recursively injecting dependencies
container.lazy(cls) Return a Lazy[cls] proxy that resolves on first use
container.create_scope() Create a child scope for scoped lifetime management
container.reset() Call on_destroy for singletons with hooks, then clear the cache
inject(container) Decorator that resolves type-hinted params from the container
Lifetime.TRANSIENT New instance on every resolve (default)
Lifetime.SINGLETON Single shared instance across the container
Lifetime.SCOPED Single instance per scope, transient without a scope
CircularDependencyError Raised when a circular dependency chain is detected
Scope Child scope returned by create_scope(), used as a context manager
Lazy[T] Proxy that resolves the underlying service on first access

Development

pip install -e .
python -m pytest tests/ -v

Support

If you find this project useful:

Star the repo

🐛 Report issues

💡 Suggest features

❤️ Sponsor development

🌐 All Open Source Projects

💻 GitHub Profile

🔗 LinkedIn Profile

License

MIT

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

philiprehberger_di-0.4.0.tar.gz (9.9 kB view details)

Uploaded Source

Built Distribution

If you're not sure about the file name format, learn more about wheel file names.

philiprehberger_di-0.4.0-py3-none-any.whl (7.4 kB view details)

Uploaded Python 3

File details

Details for the file philiprehberger_di-0.4.0.tar.gz.

File metadata

  • Download URL: philiprehberger_di-0.4.0.tar.gz
  • Upload date:
  • Size: 9.9 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/6.2.0 CPython/3.12.13

File hashes

Hashes for philiprehberger_di-0.4.0.tar.gz
Algorithm Hash digest
SHA256 2e13e974ca42234b1ecb10263845ab2c0629c978c0e6bbed0eec0f253287e3f5
MD5 2560a077045e93f4d5f1ded6c2e21548
BLAKE2b-256 527669e6fac3b5c98a405f46bca3f4ab525e61ae55a52f3efbc47c2581d3bab7

See more details on using hashes here.

File details

Details for the file philiprehberger_di-0.4.0-py3-none-any.whl.

File metadata

File hashes

Hashes for philiprehberger_di-0.4.0-py3-none-any.whl
Algorithm Hash digest
SHA256 04d11d94db5638286271035ee52eebf7b2f98d38f16254fec178a5847055fb76
MD5 6c4e59f9fdedcccdec632e3e1a0d0bc6
BLAKE2b-256 d1250ac1271564a7a58a645999214fe9fa364e2653f81590a6b8ef6842f8a12d

See more details on using hashes here.

Supported by

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