Socket
Book a DemoInstallSign in
Socket

tidipy

Package Overview
Dependencies
Maintainers
1
Alerts
File Explorer

Advanced tools

Socket logo

Install Socket

Detect and block malicious and high-risk dependencies

Install

tidipy

A minimalist, opinionated dependency injection framework

0.4.3
pipPyPI
Maintainers
1

TiDIpy

Index

  • Intro
  • In the spirit of real DI
  • Scope lifetime management
  • Scope context
  • The scope tree
  • Integration with fastapi
  • API Reference

Intro

TiDIpy—short for Timber Dependency Injection Python—is a minimalist, opinionated dependency injection framework created by me, Timber. It emphasizes simplicity, ease of use, and a principled approach to DI.

In the spirit of real DI

The essence of dependency injection is to invert the direction of dependency: the client specifies what type of dependencies it requires, while someone else decides which concrete instances to provide. Many DI frameworks blur this distinction by introducing configuration logic directly into the client—who should only declare its dependencies, not manage them. While this can seem convenient, it undermines the core value of dependency injection.

TiDIpy maintains this separation strictly. Consider the following client:

class MyClient:
    def __init__(self, dependency: MyDependency):
        ... 

No TiDIpy-specific elements are required in MyClient. Instead, composition is handled externally in a dedicated composer function:

@composer
def my_client() -> MyClient:
    return MyClient(MyDependency())

This keeps the client clean and focused and ensures that wiring and instantiation remain a separate responsibility of the composition root.

Keeping dependency composition explicitly separate is the core principle of DI—but it does come with the trade-off of writing a bit more code. TiDIpy aims to make defining the composition root as convenient as possible. One key feature that helps with this is auto composition:

auto_compose(MyClient)

This allows TiDIpy to automatically resolve and compose the dependencies in many cases, reducing boilerplate while preserving separation of concerns.

Scope lifetime management

Composition isn't just about specifying how dependencies are created and connected—it's also about managing when they are instantiated and cleaned up. Not every object in the dependency graph needs to be created upfront, and clients should stay unaware of the timing or lifecycle of their dependencies.

TiDIpy handles this through scopes. Every composer in TiDIpy is associated with a scope type—by default, this is root. You can assign a custom scope like this:

@composer(scope_type='request')
def my_dependency() -> MyDependency:
    ...

This means MyDependency will only be available within scopes of type request. To resolve it during a specific request, you'd do:

ensure_scope(scope_id='my-request', scope_type='request')
resolver = get_resolver('my-request')
resolver(MyDependency)

When the request ends, the scope—and any dependencies created within it—can be explicitly cleared:

clear_scope('my-request')

This ensures that scoped dependencies are properly cleaned up, giving you fine-grained control over object lifetimes.

Scope context

A common scenario in dependency injection is when a client depends on an interface that has multiple implementations. In such cases, the composition logic must decide which concrete implementation to provide.

class Repository(ABC):
    @abstractmethod
    get_by_id(id: str):
        ...

class InMemoryRepository(Repository):
    ...
    

class PostgresRepository(Repository):
    ...

If both InMemoryRepository and PostgresRepository have their own composers, then the type Repository becomes ambiguous—TiDIpy won’t know which one to resolve by default.

You can use scope context — key-value tags that express context-specific choices. Composers can have a scope context filter:

@composer(scope_type='app', stage='prod')
def postgres_repository() -> PostgresRepository:
    return PostgresRepository()

@composer(scope_type='app', stage='test')
def in_memory_repository() -> InMemoryRepository:
    return InMemoryRepository()

The line @composer(scope_type='app', stage='prod') means that if the key stage is part of the scope context, this composer will only be considered if it has value prod. Note that we use scope type app here. The root scope has no context, so it is not possible to define composers in the root scope with context filters.

We add context to our scope with:

ensure_scope(scope_id='app', scope_type='app', context={'stage': 'test'})

This now means that the resolver of the app scope will resolve to an InMemoryRepository.

The scope tree

Scopes in the system form a hierarchy. Each scope has exactly one parent. If you don’t explicitly set a parent, it defaults to root.

A scope can resolve dependencies registered in:

  • its own context, and
  • all of its ancestor scopes.

Here's how you can define dependencies in different scopes:

auto_compose(DependencyA)  # Registered in the root scope
auto_compose(DependencyB, scope_type='tenant')  # Registered in a tenant scope
auto_compose(DependencyC, scope_type='request')  # Registered in a request scope

You can nest scopes to reflect lifetime and context relationships. For example, start a request scope within a specific tenant scope:

ensure_scope(scope_id='my-tenant', scope_type='tenant')  # Create tenant scope
ensure_scope(scope_id='my-request', scope_type='request', parent_id='my-tenant')  # Create request scope with tenant as parent

resolver = get_resolver('my-request')  # Get a resolver for the request scope

Once scopes are set up, the resolver automatically looks upward through the scope tree if a dependency isn’t found locally:

resolver(DependencyA)  # Resolved from root scope
resolver(DependencyB)  # Resolved from tenant scope
resolver(DependencyC)  # Resolved from request scope

Why create a child scope? A scope might have children for two main reasons:

  • Shorter lifetime – The child starts later and/or ends sooner than its parent. Example: a scope for handling a web request exists only while the request is processed.
  • Extra context – The child has additional context needed for resolution. Example: an tenant-specific scope may carry configuration without having a shorter lifetime than its parent (application or root).

Integration with fastapi

If you want to stick as closely as possible to the typical FastAPI way of working, you can integrate a FastAPI app by connecting FastAPI’s dependency injection system directly to TiDIpy. See the first FastAPI example. In this approach, you define your controllers as FastAPI “endpoints” in the usual style, while still benefiting from TiDIpy’s dependency resolution. However, this does slightly break TiDIpy’s promise of fully separating clients from the application’s composition layer, since endpoints ends up knowing about the composition.

If you’re comfortable with a slightly less idiomatic FastAPI style, you can instead use the FastApiAdapter and create your own class-based controllers. This pattern is illustrated in the second FastAPI example.

API reference

Everything can be imported from the root of the package:

from TiDIpy import composer, auto_compose, ensure_scope, clear_scope, reset, get_resolver, scan, Resolver

composer

composer is a function decorator that takes as arguments:

  • id: str: the id of this dependency
  • scope_type: str: the types of scopes in which it is available
  • kwargs: str | set[str]: the values in the scope context for which this dependency is available

It decorates a factory function that takes a Resolver as a optional argument.

auto_compose

auto_compose is a function that takes a class as its first arguments and will auto compose based on the typing in its __init__. Apart from this argument, it has the same arguments as composer.

ensure_scope

ensure_scope is a function checks that a scope with the given properties already exists and if not creates one. It takes as arguments:

  • scope_id: str the target scope
  • scope_type: str the type this scope should have
  • parent_id: str the ID of the parent scope (by default root)
  • context: Optional[dict[str,str]] context of the scope (by default empty)

clear_scope

clear_scope is a function that clears a scope and all of its children. It has scope_id: str as argument.

reset

reset is a function that resets TiDI completely. It clears all scopes and forgets all registered composers.

get_resolver

get_resolver is a function that gets the resolver of a certain scope. It takes scope_id: str as argument to indicate the target scope. Its return type is Resolver

Resolver

The basic usage of a Resolver is:

resolve(DatabaseConnection)

which returns something of the same type.

Sometimes, you need multiple composers that return the same type but represent different roles or configurations. By assigning an explicit id to each composer, you can resolve the exact one you need without ambiguity.

@composer(id='primary-db')
def primary_connection() -> DatabaseConnection:
    return DatabaseConnection(dsn='postgresql://primary.db.local')

@composer(id='analytics-db')
def analytics_connection() -> DatabaseConnection:
    return DatabaseConnection(dsn='postgresql://analytics.db.local')

You can resolve them explicitly by ID:

resolve(DatabaseConnection, id='primary-db')

Since IDs are unique, this guarantees an unambiguous resolution.

scan

scan is a function that scans a python module for composers. It takes a module as an argument:

from my_project import composition_root

scan(composition_root)

FAQs

Did you know?

Socket

Socket for GitHub automatically highlights issues in each pull request and monitors the health of all your open source dependencies. Discover the contents of your packages and block harmful activity before you install or update your dependencies.

Install

Related posts

SocketSocket SOC 2 Logo

Product

About

Packages

Stay in touch

Get open source security insights delivered straight into your inbox.

  • Terms
  • Privacy
  • Security

Made with ⚡️ by Socket Inc

U.S. Patent No. 12,346,443 & 12,314,394. Other pending.