The official Python SDK for the Bird email platform.
Project description
Bird Python SDK
The official Python SDK for the Bird email platform.
Status: in development. The PyPI distribution name is
messagebird-sdk; the import package isbird.
Requires Python 3.10+.
Install
pip install messagebird-sdk # or: uv add messagebird-sdk
This SDK is generated from Bird's public OpenAPI bundle inside Bird's internal monorepo, which is the single source of truth; this repository tracks tagged releases. Generation runs in the monorepo, so
make generatewon't work from a clone here — see CONTRIBUTING.md.
Quickstart
from bird import Bird
client = Bird(api_key="bk_eu1_...") # region inferred from the key prefix
message = client.email.send(
from_="hello@acme.com",
to=["customer@example.com"],
subject="Welcome to Acme",
html="<h1>Welcome</h1>",
)
print(message.id, message.status)
api_key and base_url fall back to the BIRD_API_KEY / BIRD_BASE_URL environment variables, so Bird() with no arguments works when they are set. Use the client as a context manager (with Bird(...) as client:) to close the underlying HTTP connection pool.
# Send
message = client.email.send(from_="hi@acme.com", to=["c@x.com"], subject="Hi", text="hello")
# Fetch
message = client.email.get("em_01krd…")
# List — iterating the page auto-paginates across cursors
for message in client.email.list(status="delivered"):
print(message.id, message.status)
Client-wide email defaults
Defaults fill any unset send field; a per-send value always wins.
client = Bird(
api_key="bk_eu1_...",
email_defaults={"from_": "noreply@acme.com", "reply_to": ["support@acme.com"]},
)
client.email.send(to=["c@x.com"], subject="Receipt", text="…") # uses noreply@acme.com
Webhooks
from bird import Bird, WebhookVerificationError
client = Bird(api_key="bk_eu1_...", webhook_secret="whsec_...")
# In your web handler — pass the RAW request body (bytes) and the request headers
try:
event = client.webhooks.unwrap(request.body, request.headers)
except WebhookVerificationError:
return Response(status=400)
if event.root.type == "email.delivered":
print("delivered:", event.root.data.message_id)
Endpoint management (registering/listing webhook endpoints) is not in this release; it returns once the delivery substrate stabilises.
Errors
Every failure raises a typed exception rooted at BirdError. APIError covers anything that goes wrong issuing a request — including transport failures — so a single except APIError is enough; APIStatusError carries the HTTP status_code.
from bird import APIStatusError, RateLimitError, ValidationError
try:
client.email.send(from_="hi@acme.com", to=["c@x.com"], subject="Hi", text="…")
except RateLimitError as err:
retry_in = err.retry_after # seconds, parsed from Retry-After
except ValidationError as err:
print(err.status_code, err.details) # 422 + per-field detail
except APIStatusError as err:
print(err.status_code, err.code, err.request_id)
Transient failures (timeouts, 429, 5xx) retry automatically with jittered backoff that honors Retry-After; a mutation reuses one idempotency key across attempts, so a retried write never double-applies.
Raw response
Reach the status, headers, and request_id alongside the parsed model:
raw = client.email.with_raw_response.send(from_="hi@acme.com", to=["c@x.com"], subject="Hi", text="…")
print(raw.status_code, raw.request_id)
message = raw.parse()
Async
AsyncBird mirrors Bird method-for-method — await each call and async for over a list:
import asyncio
from bird import AsyncBird
async def main() -> None:
async with AsyncBird(api_key="bk_eu1_...") as client:
await client.email.send(from_="hi@acme.com", to=["c@x.com"], subject="Hi", text="hello")
async for message in client.email.list(status="delivered"):
print(message.id)
asyncio.run(main())
Configuration
| Option | Description |
|---|---|
api_key |
API key; falls back to BIRD_API_KEY. |
region / base_url |
Region (or explicit base URL); falls back to the key prefix / BIRD_BASE_URL. |
timeout, max_retries |
Request timeout and retry budget; overridable per call via options. |
webhook_secret |
Signing secret for webhooks.unwrap. |
email_defaults |
Client-wide send defaults. |
http_client |
Inject your own httpx.Client / AsyncClient. |
client.with_options(...) derives a new client (reusing the connection pool); every method also takes a trailing options for per-call timeout / max_retries / idempotency_key / extra_headers.
Escape hatch
Any endpoint outside the typed surface is reachable through the verb methods, with the same auth, retries, and idempotency:
from bird import EmailMessage
message = client.get("/v1/email/messages/em_01krd…", cast_to=EmailMessage)
client.post("/v1/some/new/endpoint", body={"key": "value"})
Design
The wire models are generated from the OpenAPI spec into bird._generated; this package is the hand-written idiomatic layer on top.
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 messagebird_sdk-0.1.0.tar.gz.
File metadata
- Download URL: messagebird_sdk-0.1.0.tar.gz
- Upload date:
- Size: 27.1 kB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: uv/0.9.2
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
b4f6e084a8bbe04f2b2c6b772fb9c496601a6b45a84fe5ac9aa9bcdc51cd642e
|
|
| MD5 |
1aa1c94b3144d903edb598d51dcdd0a9
|
|
| BLAKE2b-256 |
8a3218208d6253444d63550acb714685722ba9bfe6186ae8811bb8285821f27d
|
File details
Details for the file messagebird_sdk-0.1.0-py3-none-any.whl.
File metadata
- Download URL: messagebird_sdk-0.1.0-py3-none-any.whl
- Upload date:
- Size: 32.8 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: uv/0.9.2
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
058e80f6e3a68efea4261f9eca60de7a39d43a605e458b6bec49c37f1a1ecfd2
|
|
| MD5 |
55dc76738a9f5b18eb6d1f14bd59d6aa
|
|
| BLAKE2b-256 |
163f5062164ccc8078454e8cbbfaf9a4ab6eba39c45c69c76ac5bc505fb71290
|