The async actor model you didn't know you needed
Project description
Papyra
Durable persistence, retention and compaction for actor systems
Documentation: https://papyra.dymmond.com 📚
Source Code: https://github.com/dymmond/papyra
The official supported version is always the latest released.
Durable persistence, retention, recovery, and observability for actor systems.
Papyra is a production-grade persistence layer designed specifically for actor-based runtimes. It provides durable system history, audits, dead-letter tracking, retention + compaction, startup health checks, recovery orchestration, and operational tooling.
Papyra is not an actor framework. It's the persistence and observability backbone that makes an actor system operationally safe.
Why Papyra?
Actor systems are excellent at concurrency and fault isolation, but production operators eventually need:
- A durable history of lifecycle events (start/stop/crash/restart)
- Audit snapshots for “what's running and what's broken?”
- Dead letters for undeliverable messages
- Retention to prevent unbounded growth
- Compaction to physically reclaim disk space
- Startup checks and deterministic recovery
- Metrics for observability
- CLI tools for real-world operations
Papyra solves this explicitly and safely.
Features
🧱 Persistence backends
- JSON NDJSON file backend (simple, readable, portable)
- Rotating files backend (bounded disk usage)
- Redis Streams backend (production, distributed, consumer-groups)
- In-memory backend (tests, ephemeral)
♻️ Retention & compaction
- Record-count, age, and size-based retention
- Explicit physical compaction / vacuum
- Crash-safe atomic rewrite semantics where applicable
🩺 Health, startup checks & recovery
- Scan for corruption / anomalies
- Recovery modes: IGNORE / REPAIR / QUARANTINE
- Startup orchestration to guarantee a clean persistence layer before actors start
📊 Metrics & integration
- Backend metrics (writes, errors, scans, recoveries, compactions)
- CLI metrics output
- Optional OpenTelemetry integration
🛠️ CLI
persistence scan | recover | compact | inspect | startup-checkdoctor runinspect events | audits | dead-letters | summarymetrics …
Installation
pip install papyra
Optional extras:
pip install papyra[redis]
ActorSystem + Papyra: the mental model
An ActorSystem emits observable facts while it runs:
- Events: lifecycle transitions (started, stopped, crashed, restarted)
- Audits: point-in-time health snapshots (counts, registry status, dead letters)
- Dead letters: messages that couldn't be delivered
Papyra persists these facts using the configured backend.
A key guarantee: startup checks happen before any actor is allowed to run.
If the persistence layer is corrupted and startup mode is strict, ActorSystem.start() fails.
Quickstart (with ActorSystem)
1) Pick a persistence backend
JSON file backend
from papyra.persistence.json import JsonFilePersistence
persistence = JsonFilePersistence("./papyra.ndjson")
Redis Streams backend
from papyra.persistence.backends.redis import RedisStreamsConfig, RedisStreamsPersistence
persistence = RedisStreamsPersistence(
RedisStreamsConfig(url="redis://localhost:6379/0", prefix="papyra", system_id="local")
)
2) Start the ActorSystem with startup checks
from papyra.system import ActorSystem
from papyra.persistence.startup import PersistenceStartupConfig, PersistenceStartupMode
from papyra.persistence.models import PersistenceRecoveryConfig, PersistenceRecoveryMode
system = ActorSystem(
persistence=persistence,
# Ensure the persistence layer is clean *before* any actor starts
persistence_startup=PersistenceStartupConfig(
mode=PersistenceStartupMode.RECOVER,
recovery=PersistenceRecoveryConfig(mode=PersistenceRecoveryMode.REPAIR),
),
)
await system.start()
fail_on_anomaly→ start fails if corruption is detectedrecover→ attempt recovery, then require a clean post-scanignore/scan_only→ don't fail startup
3) Spawn actors after the system starts
from papyra.actor import Actor
class Echo(Actor):
async def receive(self, message):
return message
ref = system.spawn(Echo, name="echo")
4) Shut down cleanly
await system.aclose()
Operational CLI
Health check (Doctor)
papyra doctor run
Fail hard if there are anomalies:
papyra doctor run --mode fail_on_anomaly
Attempt recovery:
papyra doctor run --mode recover --recovery-mode repair
Persistence maintenance
Scan:
papyra persistence scan --path ./papyra.ndjson
Recover:
papyra persistence recover --mode repair --path ./papyra.ndjson
Compact:
papyra persistence compact --path ./papyra.ndjson
Inspect summary:
papyra persistence inspect --path ./papyra.ndjson --show-metrics
Documentation
The full documentation covers:
- Core concepts and actor lifecycle observability
- All persistence backends (JSON, rotation, Redis, memory)
- Retention and compaction strategies
- Failure scenarios and recovery playbooks
- Startup guarantees
- Metrics + OpenTelemetry integration
- Extending Papyra with custom backends
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 papyra-0.1.0.tar.gz.
File metadata
- Download URL: papyra-0.1.0.tar.gz
- Upload date:
- Size: 86.8 kB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: Hatch/1.16.2 cpython/3.10.19 HTTPX/0.28.1
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
7f0c88430c046182dd7a9229fbe28ff322fbf48a51786a1cb4b9b4bee42decea
|
|
| MD5 |
db1aea91e6d4ce6ba82809d103168b07
|
|
| BLAKE2b-256 |
a257f41224a10dbb83e8400992f2317b1a282b5f8391068e800c7b930fcd372a
|
File details
Details for the file papyra-0.1.0-py3-none-any.whl.
File metadata
- Download URL: papyra-0.1.0-py3-none-any.whl
- Upload date:
- Size: 115.3 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: Hatch/1.16.2 cpython/3.10.19 HTTPX/0.28.1
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
edfb1fbf4ba194f53d0ee0706f585cd8bd2d2acc31e3d4bcdf97154bb108ca85
|
|
| MD5 |
935babe52ef5e54471bcac7d94b0d5d9
|
|
| BLAKE2b-256 |
279b1cc4c8b5c7a84fbec10711580579fae2e337a56f97697181c0a2d573e7ea
|