Skip to main content

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 is bird.

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 generate won'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.

Email

# 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


Download files

Download the file for your platform. If you're not sure which to choose, learn more about installing packages.

Source Distribution

messagebird_sdk-0.1.0.tar.gz (27.1 kB view details)

Uploaded Source

Built Distribution

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

messagebird_sdk-0.1.0-py3-none-any.whl (32.8 kB view details)

Uploaded Python 3

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

Hashes for messagebird_sdk-0.1.0.tar.gz
Algorithm Hash digest
SHA256 b4f6e084a8bbe04f2b2c6b772fb9c496601a6b45a84fe5ac9aa9bcdc51cd642e
MD5 1aa1c94b3144d903edb598d51dcdd0a9
BLAKE2b-256 8a3218208d6253444d63550acb714685722ba9bfe6186ae8811bb8285821f27d

See more details on using hashes here.

File details

Details for the file messagebird_sdk-0.1.0-py3-none-any.whl.

File metadata

File hashes

Hashes for messagebird_sdk-0.1.0-py3-none-any.whl
Algorithm Hash digest
SHA256 058e80f6e3a68efea4261f9eca60de7a39d43a605e458b6bec49c37f1a1ecfd2
MD5 55dc76738a9f5b18eb6d1f14bd59d6aa
BLAKE2b-256 163f5062164ccc8078454e8cbbfaf9a4ab6eba39c45c69c76ac5bc505fb71290

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