Skip to content

hugochinchilla/pysyringe

Repository files navigation

PySyringe

Tests Coverage PyPI version

An opinionated dependency injection library for Python.

A container that does not rely on adding decorators to your domain classes. It only wraps views in the infrastructure layer to keep your domain and app layer decoupled from the framework and the container.

✨ Features

  • 🚀 Zero-decorator DI: keep your domain clean; inject only at call sites.
  • 🏭 Factory-based wiring: resolve by return type annotations on your factory.
  • 🧩 Inference-based construction: auto-wire constructor dependencies by type hints.
  • 🧪 Test-friendly mocks: replace any dependency per test with use_mock(...).
  • 🔒 Thread-safe mocks: mocks are stored per-thread; aliases and blacklists are global.
  • 🧰 Aliases & blacklists: map interfaces to implementations and mark types as non-creatable.
  • Resolution cache: caches factory lookups and constructor introspection (not instances).
  • 100% test coverage: comprehensive test suite ensuring reliability and correctness.

📦 Installation

pip install pysyringe

pysyringe on PyPI

🚀 Usage

1) Define a factory with return-type annotations

Your factory can be any class. The container inspects public methods and uses the return-type annotation to know what it can provide.

from myapp.domain import EmailSenderInterface
from myapp.infra import LoggingEmailSender, SmtpEmailSender


class Factory:
    def __init__(self, environment: str) -> None:
        self.environment = environment

    def get_mailer(self) -> EmailSenderInterface:
        if self.environment == "production":
            return SmtpEmailSender("mta.example.org", 25)
        return LoggingEmailSender()

2) Create the container

from os import getenv
from pysyringe.container import Container

factory = Factory(getenv("ENVIRONMENT", "development"))
container = Container(factory)

3) Configure blacklist and aliases

  • never_provide(type) blacklists types you don't want the container to try to resolve when processing functions decorated with @container.inject. This is helpful to prevent the container from attempting to build framework types (e.g., Django/Starlette requests) via constructor inference.

  • alias(interface, implementation) is a simple way to map an interface to a concrete class without having to provide a factory method. The container will build the implementation using constructor introspection, recursively resolving its dependencies.

from django.core.http import HttpRequest, HttpResponse
from myapp.domain import CalendarInterface
from myapp.infra import Calendar

# Never try to construct these framework types during injection
container.never_provide(HttpRequest)
container.never_provide(HttpResponse)

# Map an interface to a concrete implementation (no factory needed)
container.alias(CalendarInterface, Calendar)

4) Inject at the call site

Use @container.inject to supply arguments by type. Only parameters that can be resolved will be injected; the rest stay as normal function parameters. The decorator returns a partial function with all the dependencies wired; you are responsible for passing the remaining parameters to your methods.

A complete Flask example that prints the current time using an alias:

# app.py
from datetime import datetime, timezone
from flask import Flask
from pysyringe.container import Container


# 1) Define an interface and an implementation in your app
class CalendarInterface:
    def now(self) -> datetime:
        raise NotImplementedError


class Calendar(CalendarInterface):
    def now(self) -> datetime:
        return datetime.now(timezone.utc)


# 2) Create the container and configure an alias
container = Container()                       # No factory needed for this example
container.alias(CalendarInterface, Calendar)  # resolve CalendarInterface -> Calendar


# 3) Write your application and leverage the container.inject to provide dependencies
app = Flask(__name__)

@app.get("/now")
@container.inject
def get_now(calendar: CalendarInterface):
    return {"now": calendar.now().isoformat()}

if __name__ == "__main__":
    app.run(debug=True)

5) Test with mocks

Mocks are thread-local. Configure them per-test and clear afterwards.

import pytest
from pysyringe.container import Container
from myapp.domain import UserRepository
from myapp.usecases import SignupUserService
from myapp.infra.testing import InMemoryUserRepository


@pytest.fixture(autouse=True)
def clear_container_mocks_after_each_test():
    yield
    container.clear_mocks()


def test_create_user():
    user_repository = InMemoryUserRepository()
    container.use_mock(UserRepository, user_repository)
    service = container.provide(SignupUserService)

    service.signup("John Doe", "[email protected]")

    assert user_repository.get_by_email("[email protected]")

⚡ Resolution cache

PySyringe includes a lightweight resolution cache to speed up dependency resolution without caching instances.

What is cached:

  • A precomputed map of factory methods keyed by their return type (built once at Container initialization) for O(1) lookups.
  • Constructor parameter introspection is LRU-cached to avoid repeated signature parsing and type disambiguation.

What is NOT cached:

  • Resolved instances. The cache accelerates how dependencies are located and wired, not the objects produced.

This means singleton semantics or any custom sharing strategy you define remain unchanged. The cache only reduces overhead during resolution.

🧷 Singleton helper

Use the built-in singleton helper in your factory methods when you want shared instances keyed by constructor arguments.

from pysyringe.singleton import singleton
from pysyringe.container import Container


class DatabaseClient:
    def __init__(self, connection_string: str) -> None:
        self.connection_string = connection_string


class Factory:
    def get_database_client(self) -> DatabaseClient:
        # Use singleton to ensure the same connection string gets the same client
        return singleton(DatabaseClient, "postgresql://localhost:5432/mydb")


# Container will resolve DatabaseClient through factory methods
container = Container(Factory())

# Multiple calls to provide() return the same instance for the same connection string
client1 = container.provide(DatabaseClient)
client2 = container.provide(DatabaseClient)
assert client1 is client2  # Same instance

Notes:

  • The cache key includes: the class, positional args, and keyword args (order-independent for keywords).
  • Use singleton in factory methods when you want argument-keyed singletons within your DI container.
  • Perfect for database connections, HTTP clients, or any resource that should be shared per configuration.

🔒 Thread safety

The Container is thread-safe with respect to mocks. Mocks configured after the container has been created are stored in thread-local storage, so changes made to mocks in one thread do not affect other threads.

  • Shared across all threads: alias(...), never_provide(...), and the factory configuration (methods on your factory used for resolution).
  • Thread-local: use_mock(...) and clear_mocks() operate only on the calling thread's mock store.

Implications:

  • Using use_mock(SomeType, mock_instance) in one thread will not change what another thread receives for SomeType.
  • Calling clear_mocks() clears only the current thread's mocks.
  • To share a behavior globally across threads, prefer alias(...) or implement a factory method instead of relying on mocks.

About

An opinionated dependecy injection container for Python

Resources

License

Stars

Watchers

Forks

Packages

No packages published