Reusable persistence and repository toolkit
Project description
persistence-kit
Reusable persistence toolkit with async repository implementations for:
memorymongo(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 interfacessettings/: shared settings, parsers, and persistence constantsapi/: reusable API exceptions, handlers, and route loading helpersbootstrap/: startup helpers, configuration registry, and seed orchestrationutils/: transversal helpers such as upsert utilitiesstorage/: reusable object storage contracts and local/S3 adapterssecurity/: reusable identity provider contracts, Cognito/memory adapters, and JWT verifiersrepository/: concrete repository implementations by backendrepository_factory/: entity registry, repository creation, and populated view repository
Recommended rule:
- import from
persistence_kitwhen 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_repopersistence_kit.repository_factory.factory.repository_factorypersistence_kit.repository_factory.registry.entity_registrypersistence_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:
IdentityProviderandTokenVerifier: application-facing protocols.MemorySecurityProvider: local identity provider for tests/sandbox environments.CognitoIdentityProvider: AWS Cognito identity provider adapter.MemoryJwtVerifierandCognitoJwtVerifier: 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
- Define your entities as dataclasses.
- Register them in a local bootstrap such as
register_defaults. - Call
set_registry_initializer(register_defaults)during application startup. - Resolve repositories through
get_repo(...),get_repo_view(...), or FastAPI providers. - Use
ConfigRegistryandSeederProvideronly as shared bootstrap infrastructure. The concrete registrations remain in the host app.
Supported Environment Variables
REPO_DATABASE=memory|mongo|postgresMONGO_DSNMONGO_DBPOSTGRES_USERPOSTGRES_PASSWORDPOSTGRES_HOSTPOSTGRES_PORTPOSTGRES_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
mainHEAD
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 Packagemanually from the Actions tab. - Enter a
version(for example0.1.1.dev1or0.1.2.dev1). - Choose target repository:
testpypi(recommended) orpypi. - The workflow patches
pyproject.tomlversion only inside the CI run, builds, and publishes. - No commit and no PR are required for this preview publish flow.
Important:
- Prefer
*.devNversions 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.devNto TestPyPI from local machine. - Official release: publish
0.x.yto 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
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 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
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
d68822485c5ee192f4022f2503ebcbfca4d93761a5080ab63f177b7ebda685d2
|
|
| MD5 |
c2f3f0d297d7e23a816c09c53c3221b0
|
|
| BLAKE2b-256 |
aacdcd663eca334180f276d324e2a7cb37f3e350103dac4c75c6cc1d7a4cc38e
|
Provenance
The following attestation bundles were made for persistence_kit-3.2.0.tar.gz:
Publisher:
publish-pypi.yml on AndresFSerrano/persistence-kit
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
persistence_kit-3.2.0.tar.gz -
Subject digest:
d68822485c5ee192f4022f2503ebcbfca4d93761a5080ab63f177b7ebda685d2 - Sigstore transparency entry: 1285708051
- Sigstore integration time:
-
Permalink:
AndresFSerrano/persistence-kit@4991ee60a123ac27863cd98571a1cd89ba8dafb8 -
Branch / Tag:
refs/tags/v3.2.0 - Owner: https://github.com/AndresFSerrano
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
publish-pypi.yml@4991ee60a123ac27863cd98571a1cd89ba8dafb8 -
Trigger Event:
release
-
Statement type:
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
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
beaaf3714111649242f332ca81a15301cc62cfbb32d58a62a12a55db29567c00
|
|
| MD5 |
b4d9bbeca3b75dcd098e4cd5c53b2ec4
|
|
| BLAKE2b-256 |
cbf4f963d31ce3ad878ae97d349aa4dda9629e0499123912454281702c5f6dc5
|
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
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
persistence_kit-3.2.0-py3-none-any.whl -
Subject digest:
beaaf3714111649242f332ca81a15301cc62cfbb32d58a62a12a55db29567c00 - Sigstore transparency entry: 1285708156
- Sigstore integration time:
-
Permalink:
AndresFSerrano/persistence-kit@4991ee60a123ac27863cd98571a1cd89ba8dafb8 -
Branch / Tag:
refs/tags/v3.2.0 - Owner: https://github.com/AndresFSerrano
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
publish-pypi.yml@4991ee60a123ac27863cd98571a1cd89ba8dafb8 -
Trigger Event:
release
-
Statement type: