Skip to main content

Reusable persistence and repository toolkit

Project description

persistence-kit

Reusable persistence toolkit with async repository implementations for:

  • memory
  • mongo (Motor)
  • postgres (SQLAlchemy async + asyncpg)

Documentation:

  • docs/repositories_and_relations.md

Author: Andres Felipe Serrano Barrios

Package Structure

persistence_kit is organized by responsibility:

  • contracts/: repository interfaces
  • settings/: shared settings, parsers, and persistence constants
  • api/: reusable API exceptions, handlers, and route loading helpers
  • bootstrap/: startup helpers, configuration registry, and seed orchestration
  • utils/: transversal helpers such as upsert utilities
  • storage/: reusable object storage contracts and local/S3 adapters
  • security/: reusable identity provider contracts, Cognito/memory adapters, and JWT verifiers
  • repository/: concrete repository implementations by backend
  • repository_factory/: entity registry, repository creation, and populated view repository

Recommended rule:

  • import from persistence_kit when the public facade is enough
  • import from the internal folders only when you need an implementation-specific module

Installation

pip install persistence-kit

The base install keeps optional capabilities out of the dependency graph. Enable only what the host project uses:

pip install "persistence-kit[api]"
pip install "persistence-kit[security]"
pip install "persistence-kit[security-cognito]"
pip install "persistence-kit[storage-s3]"
pip install "persistence-kit[storage-routes]"
pip install "persistence-kit[dynamodb]"

Available capabilities:

  • api: FastAPI exceptions, pagination helpers, route loading, and error handlers.
  • security: memory identity provider and JWT verifier.
  • security-cognito: security plus Cognito and AWS/JWKS dependencies.
  • storage-s3: S3 object storage adapter.
  • storage-routes: FastAPI local export download router.
  • dynamodb: DynamoDB repository backend.
  • all: every optional capability.

Quick Start

from persistence_kit import Database
from persistence_kit.repository_factory import get_repo, register_entity

# register entities during application startup
register_entity(
    "user",
    {
        "entity": User,
        "collection": "users",
        "database": Database.MEMORY,
        "unique": {"email": "email"},
    },
)

repo = get_repo("user")

Public API

Preferred public imports:

from persistence_kit import (
    Repository,
    ViewRepository,
    RepoSettings,
    PersistenceKitSettings,
    AuthProvider,
    ExportStorageProvider,
    Database,
    ConfigRegistry,
    configuration,
    SeederProvider,
    ObjectStorage,
    LocalObjectStorage,
    S3ObjectStorage,
    get_export_storage,
    build_local_export_storage_router,
    IdentityProvider,
    MemorySecurityProvider,
    CognitoIdentityProvider,
    MemoryJwtVerifier,
    CognitoJwtVerifier,
    get_identity_provider,
    get_token_verifier,
    build_api_router,
    handle_service_errors,
    handle_repository_errors,
    NotFoundException,
    ValidationException,
    BusinessRuleException,
    DatabaseException,
)
from persistence_kit.repository_factory import (
    register_entity,
    get_repo,
    get_repo_view,
    provide_repo,
    provide_view_repo,
    set_registry_initializer,
)

Use internal paths only for implementation details, for example:

  • persistence_kit.repository.sqlalchemy_repo.sqlalchemy_repo
  • persistence_kit.repository_factory.factory.repository_factory
  • persistence_kit.repository_factory.registry.entity_registry
  • persistence_kit.repository_factory.view.populating_repository

Object Storage

persistence_kit.storage provides driven adapters for storing generated files or binary objects outside the domain layer:

from persistence_kit.storage import LocalObjectStorage

storage = LocalObjectStorage(
    base_dir=".local",
    public_base_url="http://localhost:8000",
    signing_secret="dev-secret",
)

key = await storage.upload("exports/report.csv", b"id,name\n1,Ada\n", "text/csv")
url = await storage.generate_presigned_url(key)

Available adapters:

  • LocalObjectStorage: stores files under a configured local directory and signs download URLs.
  • S3ObjectStorage: uploads objects to S3 and returns AWS presigned URLs.

Backward-compatible aliases are exported for applications that previously used LocalExportStorageProvider and S3ExportStorageProvider.

get_export_storage(settings) builds and caches the configured adapter from a PersistenceKitSettings instance or a subclass inherited by the host app.

For FastAPI applications, build_local_export_storage_router(...) exposes a reusable local download route. The host app passes its settings provider, optional-current-user dependency, and product authorization callback.

Install persistence-kit[storage-s3] before using S3ObjectStorage and persistence-kit[storage-routes] before using the FastAPI export route.

Security

persistence_kit.security provides reusable driven adapters for application authentication flows:

from persistence_kit.security import MemorySecurityProvider, MemoryJwtVerifier

identity = MemorySecurityProvider(
    jwt_secret="dev-secret-with-enough-length",
    jwt_issuer="local-sandbox",
    seed_role_users=True,
    seed_role_codes=("admin", "operator"),
    seed_user_domain="example.org",
)
verifier = MemoryJwtVerifier(secret="dev-secret-with-enough-length", issuer="local-sandbox")

Available pieces:

  • IdentityProvider and TokenVerifier: application-facing protocols.
  • MemorySecurityProvider: local identity provider for tests/sandbox environments.
  • CognitoIdentityProvider: AWS Cognito identity provider adapter.
  • MemoryJwtVerifier and CognitoJwtVerifier: JWT token verifiers.
  • Registration/login/password-reset result dataclasses and helper functions such as unique_roles.

Host applications should keep domain-specific roles, authorization policies, scope rules, and route permission matrices outside the kit.

Install persistence-kit[security] for the memory provider/JWT verifier and persistence-kit[security-cognito] for Cognito support.

PersistenceKitSettings centralizes common auth, storage, observability, AWS, and job-service settings. Host applications can inherit from it and override only product-specific defaults:

from persistence_kit import Database, PersistenceKitSettings


class Settings(PersistenceKitSettings):
    service_name: str = "my-api"
    key_status_history_database: Database = Database.MONGO
    memory_seed_role_codes: tuple[str, ...] = ("admin", "operator")

persistence_kit.security.factory can build the identity provider and token verifier from that inherited settings object. persistence_kit.storage.factory does the same for export storage, and persistence_kit.storage.routes exposes the reusable FastAPI local export route.

Typical Host Application Flow

  1. Define your entities as dataclasses.
  2. Register them in a local bootstrap such as register_defaults.
  3. Call set_registry_initializer(register_defaults) during application startup.
  4. Resolve repositories through get_repo(...), get_repo_view(...), or FastAPI providers.
  5. Use ConfigRegistry and SeederProvider only as shared bootstrap infrastructure. The concrete registrations remain in the host app.

Supported Environment Variables

  • REPO_DATABASE=memory|mongo|postgres
  • MONGO_DSN
  • MONGO_DB
  • POSTGRES_USER
  • POSTGRES_PASSWORD
  • POSTGRES_HOST
  • POSTGRES_PORT
  • POSTGRES_DB

Local Development

Create the local environment and run tests from the library root:

poetry lock
poetry install --with dev --all-extras
poetry run pytest -q

Current validation baseline:

  • persistence_kit: 335 passed

Publish to PyPI (Manual)

python -m pip install --upgrade build twine
python -m build
python -m twine check dist/*
python -m twine upload dist/*

Automated Publish via GitHub Actions

This repository includes a workflow at .github/workflows/publish-pypi.yml.

It publishes to PyPI when:

  • a GitHub Release is published
  • the release tag points to the current main HEAD

Prerequisite:

  • Configure PyPI Trusted Publishing for this repository and workflow file.

Preview Releases Without PRs

Use .github/workflows/publish-preview.yml to publish directly from GitHub Actions without merging a PR.

How it works:

  • Trigger Publish Preview Package manually from the Actions tab.
  • Enter a version (for example 0.1.1.dev1 or 0.1.2.dev1).
  • Choose target repository: testpypi (recommended) or pypi.
  • The workflow patches pyproject.toml version only inside the CI run, builds, and publishes.
  • No commit and no PR are required for this preview publish flow.

Important:

  • Prefer *.devN versions for preview builds.
  • PyPI/TestPyPI do not allow re-uploading the same file version.

Local Test Releases (No PR Required)

If you want to test changes from your machine in external projects without opening a PR, publish a prerelease from local code to TestPyPI.

1. Create a TestPyPI token

  • Create an API token in TestPyPI.
  • Export it as environment variable:
export TWINE_PASSWORD="pypi-***"

2. Publish from local code

bash ./scripts/publish-local.sh 0.1.1.dev1 testpypi

You can publish another local iteration with a new version:

bash ./scripts/publish-local.sh 0.1.1.dev2 testpypi

3. Install from external projects

pip install --index-url https://test.pypi.org/simple/ --extra-index-url https://pypi.org/simple persistence-kit==0.1.1.dev1

Recommended Release Strategy

  • Local experimental testing: publish 0.x.y.devN to TestPyPI from local machine.
  • Official release: publish 0.x.y to PyPI through GitHub Release (publish-pypi.yml).

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

persistence_kit-3.2.0.tar.gz (47.5 kB view details)

Uploaded Source

Built Distribution

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

persistence_kit-3.2.0-py3-none-any.whl (70.6 kB view details)

Uploaded Python 3

File details

Details for the file persistence_kit-3.2.0.tar.gz.

File metadata

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

File hashes

Hashes for persistence_kit-3.2.0.tar.gz
Algorithm Hash digest
SHA256 d68822485c5ee192f4022f2503ebcbfca4d93761a5080ab63f177b7ebda685d2
MD5 c2f3f0d297d7e23a816c09c53c3221b0
BLAKE2b-256 aacdcd663eca334180f276d324e2a7cb37f3e350103dac4c75c6cc1d7a4cc38e

See more details on using hashes here.

Provenance

The following attestation bundles were made for persistence_kit-3.2.0.tar.gz:

Publisher: publish-pypi.yml on AndresFSerrano/persistence-kit

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

File details

Details for the file persistence_kit-3.2.0-py3-none-any.whl.

File metadata

  • Download URL: persistence_kit-3.2.0-py3-none-any.whl
  • Upload date:
  • Size: 70.6 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/6.1.0 CPython/3.13.12

File hashes

Hashes for persistence_kit-3.2.0-py3-none-any.whl
Algorithm Hash digest
SHA256 beaaf3714111649242f332ca81a15301cc62cfbb32d58a62a12a55db29567c00
MD5 b4d9bbeca3b75dcd098e4cd5c53b2ec4
BLAKE2b-256 cbf4f963d31ce3ad878ae97d349aa4dda9629e0499123912454281702c5f6dc5

See more details on using hashes here.

Provenance

The following attestation bundles were made for persistence_kit-3.2.0-py3-none-any.whl:

Publisher: publish-pypi.yml on AndresFSerrano/persistence-kit

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