A minimalist, zero-dependency Inversion of Control (IoC) container for Python.
Project description
📦 pico-ioc: A Robust, Async-Native IoC Container for Python
pico-ioc is a lightweight, async-ready, decorator-driven IoC container built for clarity, testability, and performance. It brings Inversion of Control and dependency injection to Python in a deterministic, modern, and framework-agnostic way.
🐍 Requires Python 3.11+
⚖️ Core Principles
- Single Purpose – Do one thing: dependency management.
- Declarative – Use simple decorators (
@component,@factory,@provides,@configured) instead of complex config files. - Deterministic – No hidden scanning or side-effects; everything flows from an explicit
init(). - Async-Native – Fully supports async providers, async lifecycle hooks (
__ainit__), and async interceptors. - Fail-Fast – Detects missing bindings and circular dependencies at bootstrap (
init()). - Testable by Design – Use
overridesandprofilesto swap components instantly. - Zero Core Dependencies – Built entirely on the Python standard library. Optional features may require external packages (see Installation).
🚀 Why pico-ioc?
As Python systems evolve, wiring dependencies by hand becomes fragile and unmaintainable. pico-ioc eliminates that friction by letting you declare how components relate — not how they’re created.
| Feature | Manual Wiring | With pico-ioc |
|---|---|---|
| Object creation | svc = Service(Repo(Config())) |
svc = container.get(Service) |
| Replacing deps | Monkey-patch | overrides={Repo: FakeRepo()} |
| Coupling | Tight | Loose |
| Testing | Painful | Instant |
| Async support | Manual | Built-in (aget, __ainit__) |
🧩 Highlights (v2.2+)
- Unified Configuration: Use
@configuredto bind both flat (ENV-like) and tree (YAML/JSON) sources via theconfiguration(...)builder (ADR-0010). - Extensible Scanning: Use
CustomScannerto hook into the discovery phase and register functions or custom decorators (ADR-0011). - Async-aware AOP: Method interceptors via
@intercepted_by. - Scoped resolution: singleton, prototype, request, session, transaction, and custom scopes.
- Tree-based configuration: Advanced mapping with reusable adapters (
Annotated[Union[...], Discriminator(...)]). - Observable context: Built-in stats, health checks (
@health), observer hooks (ContainerObserver), and dependency graph export.
📦 Installation
pip install pico-ioc
Optional extras:
-
YAML configuration support (requires PyYAML)
pip install pico-ioc[yaml]
-
Dependency graph export as DOT/SVG (requires Graphviz)
pip install pico-ioc[graphviz]
⚠️ Important Note
Breaking Behavior in Scope Management (v2.1.3+): Scope LRU Eviction has been removed to guarantee data integrity.
- Frameworks (pico-fastapi): Handled automatically.
- Manual usage: You must explicitly call
container._caches.cleanup_scope("scope_name", scope_id)when a context ends to prevent memory leaks.
⚙️ Quick Example (Unified Configuration)
import os
from dataclasses import dataclass
from pico_ioc import component, configured, configuration, init, EnvSource
# 1. Define configuration with @configured
@configured(prefix="APP_", mapping="auto") # Auto-detects flat mapping
@dataclass
class Config:
db_url: str = "sqlite:///demo.db"
# 2. Define components
@component
class Repo:
def __init__(self, cfg: Config): # Inject config
self.cfg = cfg
def fetch(self):
return f"fetching from {self.cfg.db_url}"
@component
class Service:
def __init__(self, repo: Repo): # Inject Repo
self.repo = repo
def run(self):
return self.repo.fetch()
# --- Example Setup ---
os.environ['APP_DB_URL'] = 'postgresql://user:pass@host/db'
# 3. Build configuration context
config_ctx = configuration(
EnvSource(prefix="") # Read APP_DB_URL from environment
)
# 4. Initialize container
container = init(modules=[__name__], config=config_ctx) # Pass context via 'config'
# 5. Get and use the service
svc = container.get(Service)
print(svc.run())
# --- Cleanup ---
del os.environ['APP_DB_URL']
Output:
fetching from postgresql://user:pass@host/db
🧪 Testing with Overrides
class FakeRepo:
def fetch(self): return "fake-data"
# Build configuration context (might be empty or specific for test)
test_config_ctx = configuration()
# Use overrides during init
container = init(
modules=[__name__],
config=test_config_ctx,
overrides={Repo: FakeRepo()} # Replace Repo with FakeRepo
)
svc = container.get(Service)
assert svc.run() == "fake-data"
🧰 Profiles
Use profiles to enable/disable components or configuration branches conditionally.
# Enable "test" profile when bootstrapping the container
container = init(
modules=[__name__],
profiles=["test"]
)
Profiles are typically referenced in decorators or configuration mappings to include/exclude components and bindings.
⚡ Async Components
pico-ioc supports async lifecycle and resolution.
import asyncio
from pico_ioc import component, init
@component
class AsyncRepo:
async def __ainit__(self):
# e.g., open async connections
self.ready = True
async def fetch(self):
return "async-data"
async def main():
container = init(modules=[__name__])
repo = await container.aget(AsyncRepo) # Async resolution
print(await repo.fetch())
# Graceful async shutdown (calls @cleanup async methods)
await container.ashutdown()
asyncio.run(main())
__ainit__runs after construction if defined.- Use
container.aget(Type)to resolve components that require async initialization. - Use
await container.ashutdown()to close resources cleanly.
🩺 Lifecycle & AOP
import time
from pico_ioc import component, init, intercepted_by, MethodInterceptor, MethodCtx
# Define an interceptor component
@component
class LogInterceptor(MethodInterceptor):
def invoke(self, ctx: MethodCtx, call_next):
print(f"→ calling {ctx.cls.__name__}.{ctx.name}")
start = time.perf_counter()
try:
res = call_next(ctx)
duration = (time.perf_counter() - start) * 1000
print(f"← {ctx.cls.__name__}.{ctx.name} done ({duration:.2f}ms)")
return res
except Exception as e:
duration = (time.perf_counter() - start) * 1000
print(f"← {ctx.cls.__name__}.{ctx.name} failed ({duration:.2f}ms): {e}")
raise
@component
class Demo:
@intercepted_by(LogInterceptor) # Apply the interceptor
def work(self):
print(" Working...")
time.sleep(0.01)
return "ok"
# Initialize container (must scan module containing interceptor too)
c = init(modules=[__name__])
result = c.get(Demo).work()
print(f"Result: {result}")
👁️ Observability & Cleanup
-
Export a dependency graph in DOT format:
c = init(modules=[...]) c.export_graph("dependencies.dot") # Writes directly to file
-
Health checks:
- Annotate health probes inside components with
@healthfor container-level reporting. - The container exposes health information that can be queried in observability tooling.
- Annotate health probes inside components with
-
Container cleanup:
- For sync apps:
container.shutdown() - For async apps:
await container.ashutdown()
- For sync apps:
Use cleanup in application shutdown hooks to release resources deterministically.
📖 Documentation
The full documentation is available within the docs/ directory of the project repository. Start with docs/README.md for navigation.
- Getting Started:
docs/getting-started.md - User Guide:
docs/user-guide/README.md - Advanced Features:
docs/advanced-features/README.md - Observability:
docs/observability/README.md - Cookbook (Patterns):
docs/cookbook/README.md - Architecture:
docs/architecture/README.md - API Reference:
docs/api-reference/README.md - ADR Index:
docs/adr/README.md
🧩 Development
pip install tox
tox
🧾 Changelog
See CHANGELOG.md — Significant redesigns and features in v2.0+.
Latest: v2.2.5 (2026-04-25) — fixes the eager
singleton resolver to honor optional dependencies (T | None and
parameters with defaults), matching ADR-0006. Previously these failed at
instantiation time despite passing static validation.
AI Coding Skills
Install Claude Code or OpenAI Codex skills for AI-assisted development with pico-ioc:
curl -sL https://raw.githubusercontent.com/dperezcabrera/pico-skills/main/install.sh | bash -s -- ioc
| Command | Description |
|---|---|
/add-component |
Add components, factories, interceptors, event subscribers, settings |
/add-tests |
Generate tests for pico components |
All skills: curl -sL https://raw.githubusercontent.com/dperezcabrera/pico-skills/main/install.sh | bash
See pico-skills for details.
📜 License
MIT — LICENSE
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 pico_ioc-2.2.5.tar.gz.
File metadata
- Download URL: pico_ioc-2.2.5.tar.gz
- Upload date:
- Size: 286.3 kB
- Tags: Source
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.12
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
46842497a9fd9f818d6accbc6c651e9bd1d26475a2bd92a0161f9d3acd27e099
|
|
| MD5 |
3d19c75cd84e0e56758c5d005e08fd0e
|
|
| BLAKE2b-256 |
b505853562d2d460aad14fb1d19a50e9f98e69e5b95666040137030cf735f288
|
Provenance
The following attestation bundles were made for pico_ioc-2.2.5.tar.gz:
Publisher:
publish-to-pypi.yml on dperezcabrera/pico-ioc
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
pico_ioc-2.2.5.tar.gz -
Subject digest:
46842497a9fd9f818d6accbc6c651e9bd1d26475a2bd92a0161f9d3acd27e099 - Sigstore transparency entry: 1383673600
- Sigstore integration time:
-
Permalink:
dperezcabrera/pico-ioc@f52bb2cedea67e2c4d4626a955c8dd5da6f81a5b -
Branch / Tag:
refs/tags/v2.2.5 - Owner: https://github.com/dperezcabrera
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
publish-to-pypi.yml@f52bb2cedea67e2c4d4626a955c8dd5da6f81a5b -
Trigger Event:
release
-
Statement type:
File details
Details for the file pico_ioc-2.2.5-py3-none-any.whl.
File metadata
- Download URL: pico_ioc-2.2.5-py3-none-any.whl
- Upload date:
- Size: 58.7 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.12
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
483d22defd0a79e645a5cd93b78fb143081c70bf711a1c4d79b66002c480f6aa
|
|
| MD5 |
36d2aa0d37d3c09b75149c152c15ea92
|
|
| BLAKE2b-256 |
89f893084bf0dc52deb19989b57fc4cb199e21bc6f5d38d16a694df6249cbb89
|
Provenance
The following attestation bundles were made for pico_ioc-2.2.5-py3-none-any.whl:
Publisher:
publish-to-pypi.yml on dperezcabrera/pico-ioc
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
pico_ioc-2.2.5-py3-none-any.whl -
Subject digest:
483d22defd0a79e645a5cd93b78fb143081c70bf711a1c4d79b66002c480f6aa - Sigstore transparency entry: 1383673695
- Sigstore integration time:
-
Permalink:
dperezcabrera/pico-ioc@f52bb2cedea67e2c4d4626a955c8dd5da6f81a5b -
Branch / Tag:
refs/tags/v2.2.5 - Owner: https://github.com/dperezcabrera
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
publish-to-pypi.yml@f52bb2cedea67e2c4d4626a955c8dd5da6f81a5b -
Trigger Event:
release
-
Statement type: