Skip to main content

Python client for exact Redis-backed bounded consumption.

Project description

EscrowMint logo

EscrowMint Python

Exact, Redis-backed bounded consumption for shared quotas.

CI GitHub Release License Python Coverage

EscrowMint Python is for cases where many threads, processes, or services need to consume from the same global quota without letting it go below zero.

Good fits:

  • prepaid credits
  • inventory reservation
  • budget caps
  • worker permit pools
  • campaign spend controls

Why EscrowMint

EscrowMint is not a generic counter library. It is a quota and reservation library with application-level semantics:

  • exact bounded decrement
  • idempotent consume
  • reservation with TTL
  • commit and cancel flow
  • crash recovery via lazy expiry reclaim

Install

Install from PyPI:

uv add escrowmint

or:

pip install escrowmint

Quickstart

from escrowmint import Client

client = Client.from_url("redis://localhost:6379/0")

result = client.try_consume(
    "wallet:123",
    5,
    idempotency_key="req-001",
)

print(result.applied)      # True
print(result.remaining)    # remaining global quota

Crash-Safe Reservation

from escrowmint import Client, ReservationExpired

client = Client.from_url("redis://localhost:6379/0")

reservation = client.reserve(
    "wallet:123",
    10,
    ttl_ms=30_000,
)

try:
    result = client.commit("wallet:123", reservation.reservation_id)
except ReservationExpired:
    # the hold expired and the quota was released
    ...

If a worker crashes after reserve but before commit, the held quota is released after TTL expiry on the next mutation or get_state call for that same resource.

Current API

client.try_consume(resource, amount, idempotency_key=None)
client.reserve(resource, amount, ttl_ms=..., reservation_id=None)
client.commit(resource, reservation_id)
client.cancel(resource, reservation_id)
client.get_state(resource)

Chunk Lease Path

EscrowMint also ships an explicit chunk-lease lifecycle for hot resources:

lease = client.allocate_chunk(
    "wallet:123",
    100,
    owner_id="worker-a",
    ttl_ms=30_000,
)

result = client.consume_chunk("wallet:123", lease.lease_id, 5, owner_id="worker-a")
lease = client.renew_chunk("wallet:123", lease.lease_id, owner_id="worker-a", ttl_ms=30_000)
lease = client.release_chunk("wallet:123", lease.lease_id, owner_id="worker-a")

This is the authoritative distributed chunk path. It keeps chunk state in Redis and supports expiry reclaim, renew, release, and worker ownership checks.

How It Works

  • Redis remains the source of truth for each resource.
  • Lua scripts make each operation atomic.
  • Reservations move units from available to reserved.
  • Pending reservations are indexed by expiry time in Redis.
  • Expired reservations are reclaimed lazily in bounded batches on the next touch of that resource.
  • Terminal reservation outcomes are moved into short-lived receipt keys so the hot reservation hash stays small.

Direct Path and Chunk Lease Path

EscrowMint currently ships both models.

The direct path is the shared-resource path:

  • try_consume, reserve, commit, and cancel
  • exact bounded updates against the resource's shared state
  • the simplest way to get correctness and crash recovery

The chunk lease path adds a worker-owned lease layer on top of that model:

  • allocate_chunk, consume_chunk, renew_chunk, release_chunk, and get_chunk
  • explicit escrow or chunk allocation per worker
  • better control over hot-resource ownership, refill, expiry, and reclaim
  • more operational complexity than the direct path

Choose the direct path when you want the simplest exact path.

Choose the chunk lease path when a resource benefits from explicit worker-level quota management.

The current chunk lease implementation is an authoritative Redis-backed lease lifecycle. It improves the state model for hot resources, but it does not automatically become a no-Redis local fast path. If you want fewer Redis round trips than the shipped chunk API provides, you can layer an in-process chunk consumer on top of the authoritative lease lifecycle.

See docs/CHUNK_LEASES.md.

Development

EscrowMint Python uses uv.

uv sync --dev
uv run ruff check
uv run pytest
uv build

Notes:

Support

  • Python: 3.8+
  • Redis: intended for modern Redis deployments that support Lua scripting and standard key expiry semantics
  • Stability: the package is approaching its first public 0.1.x release; expect the core API to be much more stable than the surrounding release/docs tooling

Release Process

EscrowMint Python uses Conventional Commits and Release Please for semantic versioning and release notes.

  • fix: -> patch release
  • feat: -> minor release
  • feat!: or BREAKING CHANGE: -> major release

When releasable commits land on main, Release Please opens or updates a release PR. Merging that PR updates CHANGELOG.md, creates the vX.Y.Z tag, and creates the GitHub release notes. The dedicated release workflow then verifies the tagged code and publishes the package to PyPI.

Docs

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

escrowmint-0.3.3.tar.gz (9.5 kB view details)

Uploaded Source

Built Distribution

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

escrowmint-0.3.3-py3-none-any.whl (11.3 kB view details)

Uploaded Python 3

File details

Details for the file escrowmint-0.3.3.tar.gz.

File metadata

  • Download URL: escrowmint-0.3.3.tar.gz
  • Upload date:
  • Size: 9.5 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: uv/0.11.7 {"installer":{"name":"uv","version":"0.11.7","subcommand":["publish"]},"python":null,"implementation":{"name":null,"version":null},"distro":{"name":"Ubuntu","version":"24.04","id":"noble","libc":null},"system":{"name":null,"release":null},"cpu":null,"openssl_version":null,"setuptools_version":null,"rustc_version":null,"ci":true}

File hashes

Hashes for escrowmint-0.3.3.tar.gz
Algorithm Hash digest
SHA256 7d1577f845a6c3058e7d0b8c898279ed9153914de84dc5d01e26620d7613f21b
MD5 2a973c8ce9021d6dbb660410c09220c7
BLAKE2b-256 1b2f07a3dfa293c9415ac8268f75ffd01d44285e6143d9e2da2c363d4e9202d3

See more details on using hashes here.

File details

Details for the file escrowmint-0.3.3-py3-none-any.whl.

File metadata

  • Download URL: escrowmint-0.3.3-py3-none-any.whl
  • Upload date:
  • Size: 11.3 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: uv/0.11.7 {"installer":{"name":"uv","version":"0.11.7","subcommand":["publish"]},"python":null,"implementation":{"name":null,"version":null},"distro":{"name":"Ubuntu","version":"24.04","id":"noble","libc":null},"system":{"name":null,"release":null},"cpu":null,"openssl_version":null,"setuptools_version":null,"rustc_version":null,"ci":true}

File hashes

Hashes for escrowmint-0.3.3-py3-none-any.whl
Algorithm Hash digest
SHA256 8a1ca4e3b4fcb41faeed239d457bdbc6e22f338e615392d7be0b09c81a1c6d4e
MD5 01e09e1915956e27bb5f0482139352f0
BLAKE2b-256 1890d1d24c2dfb04edf4b116a3e4d1938107a2dae4e6c28349725a589a318d8b

See more details on using hashes here.

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