Official Python SDK for the Slothbox API
Project description
slothbox (Python SDK)
Official Python SDK for the Slothbox API — typed,
sync and async, with retries, cursor pagination, lifecycle waiters, and a
webhook verification toolkit. Feature parity with
@slothbox/sdk is tracked in
PARITY_CHECKLIST.md.
pip install slothbox
Requires Python 3.10+ and an org service-account API key (sk_…, API plan)
from the Slothbox dashboard — the SDK identifies itself on every request and
the API rejects SDK traffic on other credentials (see
SDK identification).
Quickstart
from slothbox import Slothbox
client = Slothbox() # reads SLOTHBOX_API_KEY; or Slothbox(api_key="sk_…")
orgs = client.organizations.list()
envs = client.environments.list(org_id)
# Safe launch: always sends an Idempotency-Key, then polls until `running`.
box = client.environments.launch_and_wait(org_id, {"templateId": template_id})
print(box.env_id, box.status)
Async is the same surface with await:
from slothbox import AsyncSlothbox
async with AsyncSlothbox() as client:
box = await client.environments.launch_and_wait(org_id, {"templateId": template_id})
Request bodies accept either the typed models (slothbox.models) or plain
dicts keyed by the wire (JSON) names, as above.
SDK identification
Every request carries an always-on identification header:
x-slothbox-sdk: slothbox-sdk-python/<version>
The API uses it to identify SDK traffic, which must authenticate with an org
service-account key (API plan). SDK-identified requests on any other
credential are rejected 403 with code sdk_requires_service_key, raised as
PermissionDeniedError. Identification wins: the header is set as a
client-level default (so it also covers direct calls through the
client.generated escape hatch) and re-asserted on every request the
wrapper sends, so it cannot be removed or overridden — not via
RequestOptions(headers=…), not via httpx_args, and not via the generated
with_headers().
Errors
Every non-2xx response raises a typed exception with the API's
machine-readable code, the gateway request id, and validation issues:
from slothbox import ConflictError, RateLimitError
try:
client.environments.launch(org_id, {"templateId": template_id})
except ConflictError as err:
if err.code == "template_not_baked":
client.templates.wait_until_baked(org_id, template_id)
except RateLimitError as err:
print("retry after", err.retry_after, "s — request", err.request_id)
Retries
GET/HEAD/PUT/DELETE are retried on 429/5xx/network errors (3 retries by
default) with capped full-jitter backoff; Retry-After is honored. POST and
PATCH are never blind-retried — a duplicated launch provisions a second
EC2 box — unless the request carries an Idempotency-Key
(environments.launch(..., idempotency_key=…); launch_and_wait always
sends one). Tune with Slothbox(max_retries=…) or per request via
RequestOptions(max_retries=…).
Pagination
The audit and webhook-delivery lists are cursor-paginated; iterating a page walks every page:
for event in client.audit.list_org_events(org_id, limit=100):
...
page = client.webhooks.list_deliveries(org_id, endpoint_id)
while page.has_next_page():
page = page.get_next_page()
(async for on the async client.)
Webhooks
Verify deliveries with the exact raw bytes you received:
from slothbox import parse_webhook_event, WebhookVerificationError
try:
event = parse_webhook_event(raw_body, request_headers, endpoint_secret)
except WebhookVerificationError as err:
return 400, err.code
if event["type"] == "environment.stopped":
...
The verifier implements Standard Webhooks v1 (HMAC-SHA256, secret rotation, ±300s replay window) and is tested against the same vector file as the API's signer and the TS SDK.
Architecture
Per ADR 0001: a pinned
openapi-python-client
generates the typed core (slothbox._generated — committed, httpx + attrs,
regenerated deterministically by scripts/generate.sh and diff-checked in
CI), and the hand-written slothbox package on top is the public
compatibility surface. client.generated exposes the generated
AuthenticatedClient as an escape hatch — note that the generated
set_httpx_client() bypasses auth; pass transports via
Slothbox(httpx_args=…) instead.
Development
python -m venv .venv && .venv/bin/pip install -e '.[dev]'
ruff check . && ruff format --check . # lint
mypy # strict on the wrapper
pytest # 3.10+ supported
scripts/generate.sh # regen the core from openapi.json
Releases are automated — see RELEASING.md.
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 slothbox-0.2.0.tar.gz.
File metadata
- Download URL: slothbox-0.2.0.tar.gz
- Upload date:
- Size: 146.7 kB
- Tags: Source
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.12
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
be0f292e012cae86c5d4c31e1dfbda96c678bf225655bdd392c6b758853be7d8
|
|
| MD5 |
6843ea6aac0258121f68976429477938
|
|
| BLAKE2b-256 |
78db6205e0a3b742b2d170c66c85e665bea99aead6c5da0cf23a0be217e34420
|
Provenance
The following attestation bundles were made for slothbox-0.2.0.tar.gz:
Publisher:
release.yml on sloth-box/sdk-python
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
slothbox-0.2.0.tar.gz -
Subject digest:
be0f292e012cae86c5d4c31e1dfbda96c678bf225655bdd392c6b758853be7d8 - Sigstore transparency entry: 1783164396
- Sigstore integration time:
-
Permalink:
sloth-box/sdk-python@87edbe37e073686b5f10b8802ed897a9bb839187 -
Branch / Tag:
refs/heads/main - Owner: https://github.com/sloth-box
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
release.yml@87edbe37e073686b5f10b8802ed897a9bb839187 -
Trigger Event:
push
-
Statement type:
File details
Details for the file slothbox-0.2.0-py3-none-any.whl.
File metadata
- Download URL: slothbox-0.2.0-py3-none-any.whl
- Upload date:
- Size: 359.8 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 |
f632da9cf572e41e4c0a701dc58cd450c4ec53d6ad8dc633d8fba51e74b67997
|
|
| MD5 |
897d21ebdf817891ce0ffb9fd6bc5b19
|
|
| BLAKE2b-256 |
02abfa93a135e76cfc433a00b214ada9718821d84461d964ecee991eb189c14d
|
Provenance
The following attestation bundles were made for slothbox-0.2.0-py3-none-any.whl:
Publisher:
release.yml on sloth-box/sdk-python
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
slothbox-0.2.0-py3-none-any.whl -
Subject digest:
f632da9cf572e41e4c0a701dc58cd450c4ec53d6ad8dc633d8fba51e74b67997 - Sigstore transparency entry: 1783164460
- Sigstore integration time:
-
Permalink:
sloth-box/sdk-python@87edbe37e073686b5f10b8802ed897a9bb839187 -
Branch / Tag:
refs/heads/main - Owner: https://github.com/sloth-box
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
release.yml@87edbe37e073686b5f10b8802ed897a9bb839187 -
Trigger Event:
push
-
Statement type: