Skip to main content

Pico-ioc integration for SQLAlchemy. Adds Spring-style transactional support, configuration, and helpers.

Project description

pico-sqlalchemy

PyPI Ask DeepWiki License: MIT CI (tox matrix) codecov Quality Gate Status Duplicated Lines (%) Maintainability Rating Docs Interactive Lab

Pico-SQLAlchemy

Pico-SQLAlchemy integrates Pico-IoC with SQLAlchemy, providing a true inversion of control persistence layer with Spring Data-style declarative features.

It brings constructor-based dependency injection, implicit transaction management, and powerful declarative queries using pure Python and SQLAlchemy’s Async ORM.

Requires Python 3.11+ Async-Native: Built entirely on AsyncSession and create_async_engine. Zero-Boilerplate: Repositories are transactional by default. Declarative Queries: Define SQL or expressions in decorators; the library executes them for you.


Why pico-sqlalchemy?

Most Python apps suffer from manual session handling (async with session...), scattered transaction logic, and verbose repository patterns.

Pico-SQLAlchemy solves this by offering:

Feature SQLAlchemy Default pico-sqlalchemy
Transactions Manual commit() / rollback() Implicit (Auto-managed)
Repositories DIY Classes @repository (Transactional by default)
Queries Manual implementation @query (Declarative execution)
Injection None / Global variables Constructor Injection (IoC)
Pagination Manual calculation Automatic (PageRequest / Page)

Core Features

  • Implicit Transactions: Methods inside @repository are automatically Read-Write transactional.
  • Declarative Queries: Use @query to run SQL or Expressions automatically (defaults to Read-Only).
  • AOP-Based Propagation: REQUIRED, REQUIRES_NEW, MANDATORY, NEVER, etc.
  • Session Lifecycle: Centralized SessionManager handles engine creation and cleanup.
  • Pagination: Built-in support for paged results via @query(paged=True).

Installation

pip install pico-sqlalchemy

You will also need an async database driver:

pip install aiosqlite   # for SQLite
pip install asyncpg     # for PostgreSQL

Quick Example

1. Define Model

from sqlalchemy import Integer, String
from pico_sqlalchemy import AppBase, Mapped, mapped_column

class User(AppBase):
    __tablename__ = "users"
    id: Mapped[int] = mapped_column(Integer, primary_key=True)
    username: Mapped[str] = mapped_column(String(50))

2. Define Repository (The "Magic" Part)

Notice we don't need @transactional here.

  • save: Automatically runs in a Read-Write transaction.
  • find_by_name: Automatically runs in a Read-Only transaction and executes the query logic.
from pico_sqlalchemy import repository, query, SessionManager, get_session

@repository(entity=User)
class UserRepository:
    def __init__(self, manager: SessionManager):
        self.manager = manager

    # IMPLICIT: Read-Write Transaction
    async def save(self, user: User) -> User:
        session = get_session(self.manager)
        session.add(user)
        return user

    # DECLARATIVE: Read-Only Transaction + Auto-Execution
    @query(expr="username = :username", unique=True)
    async def find_by_name(self, username: str) -> User | None:
        ... # Body is ignored; the library executes the query

3. Define Service

Use @transactional here to define business logic boundaries.

from pico_ioc import component
from pico_sqlalchemy import transactional

@component
class UserService:
    def __init__(self, repo: UserRepository):
        self.repo = repo

    @transactional
    async def create(self, name: str) -> User:
        # 1. Check existence (Read-Only tx from repo)
        existing = await self.repo.find_by_name(name)
        if existing:
            raise ValueError("User exists")
            
        # 2. Save new user (Joins current transaction)
        return await self.repo.save(User(username=name))

4. Run it

import asyncio
from pico_ioc import init, configuration, DictSource

config = configuration(DictSource({
    "database": {
        "url": "sqlite+aiosqlite:///:memory:",
        "echo": False
    }
}))

async def main():
    container = init(modules=["pico_sqlalchemy", "__main__"], config=config)
    service = await container.aget(UserService)
    
    user = await service.create("alice")
    print(f"Created: {user.id}")
    
    await container.cleanup_all_async()

if __name__ == "__main__":
    asyncio.run(main())

Transaction Hierarchy & Rules

Pico-SQLAlchemy applies a "Best Effort" strategy to determine transaction configuration. The priority order (highest wins) is:

Priority Decorator Default Mode Use Case
1 (High) @transactional(...) Explicit Config Overriding defaults, Service layer logic.
2 @query(...) Read-Only Efficient data fetching.
3 (Base) @repository Read-Write Default for CRUD (saves, updates, deletes).

Example Scenarios

  1. Plain Method in Repository:

    async def update_user(self): ...
    

    Result: Active Read-Write Transaction (Implicit from @repository).

  2. Query Method:

    @query("SELECT ...")
    async def get_data(self): ...
    

    Result: Active Read-Only Transaction (Implicit from @query).

  3. Manual Override:

    @transactional(read_only=True)
    async def complex_report(self): ...
    

    Result: Active Read-Only Transaction (Explicit override).

Transaction-scoped components (v0.4.0+)

Beyond managing the SQLAlchemy session, the interceptor binds pico-ioc's "transaction" DI scope to the same boundary. A component registered with scope="transaction" is instantiated once per database transaction and torn down (running its @cleanup hooks) when that transaction ends:

@component(scope="transaction")
class UnitOfWorkAudit:
    def __init__(self):
        self.events: list[str] = []

    @cleanup
    def flush(self):
        # runs exactly when the enclosing transaction ends
        ...

A new transaction (REQUIRES_NEW, or REQUIRED with no enclosing transaction) opens a fresh scope; joins reuse the enclosing one — so the session boundary and the DI lifetime are two facets of a single transaction. Requires pico-ioc ≥ 2.2.6.


Declarative Queries in Depth

The @query decorator eliminates boilerplate for common fetches.

Expression Mode (expr)

Requires @repository(entity=Model). Injects the expression into a SELECT * FROM table WHERE ....

@query(expr="age > :min_age", unique=False)
async def find_adults(self, min_age: int) -> list[User]: ...

SQL Mode (sql)

Executes raw SQL. Useful for complex joins or specific DTOs.

@query(sql="SELECT count(*) as cnt FROM users")
async def count_users(self) -> int: ...

Automatic Pagination

Just add paged=True and a page: PageRequest parameter.

from pico_sqlalchemy import Page, PageRequest

@query(expr="active = true", paged=True)
async def find_active(self, page: PageRequest) -> Page[User]: ...

Testing

Testing is simple because you can override the configuration or the components easily using Pico-IoC.

@pytest.mark.asyncio
async def test_service():
    # Setup container with in-memory DB
    container = ... 
    
    service = await container.aget(UserService)
    user = await service.create("test")
    
    assert user.id is not None

Architecture Overview

                 ┌─────────────────────────────┐
                 │          Your App           │
                 └──────────────┬──────────────┘
                                │
                        Constructor Injection
                                │
                 ┌──────────────▼───────────────┐
                 │          Pico-IoC            │
                 └──────────────┬───────────────┘
                                │
                 ┌──────────────▼───────────────┐
                 │       pico-sqlalchemy        │
                 │ 1. Implicit Repo Transactions│
                 │ 2. Declarative @query        │
                 │ 3. Explicit @transactional   │
                 └──────────────┬───────────────┘
                                │
                           SQLAlchemy
                           (Async ORM)

AI Coding Skills

Install Claude Code or OpenAI Codex skills for AI-assisted development with pico-sqlalchemy:

curl -sL https://raw.githubusercontent.com/dperezcabrera/pico-skills/main/install.sh | bash -s -- sqlalchemy
Command Description
/add-repository Add SQLAlchemy entities and repositories with transactions
/add-component Add components, factories, interceptors, 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

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

pico_sqlalchemy-0.4.0.tar.gz (77.8 kB view details)

Uploaded Source

Built Distribution

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

pico_sqlalchemy-0.4.0-py3-none-any.whl (26.9 kB view details)

Uploaded Python 3

File details

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

File metadata

  • Download URL: pico_sqlalchemy-0.4.0.tar.gz
  • Upload date:
  • Size: 77.8 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/6.1.0 CPython/3.13.12

File hashes

Hashes for pico_sqlalchemy-0.4.0.tar.gz
Algorithm Hash digest
SHA256 809ecc3df50a41312154d549f38ad141141862d96fa01bfbebb34151daa2bb3b
MD5 038f764d42f4d15b6e3834a0b88fa42c
BLAKE2b-256 3ab3d7561c04f2d917ac462b8c8e41d404dc94f7c599cf359e416abc9f9c44e7

See more details on using hashes here.

Provenance

The following attestation bundles were made for pico_sqlalchemy-0.4.0.tar.gz:

Publisher: publish-to-pypi.yml on dperezcabrera/pico-sqlalchemy

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

File details

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

File metadata

File hashes

Hashes for pico_sqlalchemy-0.4.0-py3-none-any.whl
Algorithm Hash digest
SHA256 72b4c1a10af9d5468b4c7d3c64c99d8b19bf8e22341aacb971634884e865e766
MD5 7f4122fee0cf4088e9e34be880154379
BLAKE2b-256 a8ccf0a90eb4b1202974c1796182b2b09c97fba27932d05f82cd5a64283f1a40

See more details on using hashes here.

Provenance

The following attestation bundles were made for pico_sqlalchemy-0.4.0-py3-none-any.whl:

Publisher: publish-to-pypi.yml on dperezcabrera/pico-sqlalchemy

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

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