Skip to main content

Official Mesa Python SDK

Project description

mesa-sdk

Official Mesa Python SDK.

This is the primary Python SDK for Mesa. It wraps the generated mesa-rest client with ergonomic async resource namespaces and automatic org inference.

Python 3.10+ is required.

Install

pip install mesa-sdk

Quick Start

import asyncio
from mesa_sdk import Mesa

async def main():
    async with Mesa(api_key="mk_...") as mesa:
        repos = await mesa.repos.list()
        print(repos)

asyncio.run(main())

Usage

Org Resolution

The SDK resolves your default organization automatically via /whoami on first use. You can bypass this by passing org to the constructor or overriding per-call:

# Default: org inferred from /whoami (lazy, cached)
mesa = Mesa(api_key="mk_...")
repos = await mesa.repos.list()

# Constructor org bypasses /whoami entirely
mesa = Mesa(api_key="mk_...", org="acme")
repos = await mesa.repos.list()

# Per-call override
repos = await mesa.repos.list(org="other-org")

Repositories

# List
repos = await mesa.repos.list()

# Create
repo = await mesa.repos.create(name="my-repo")

# Get
repo = await mesa.repos.get(repo="my-repo")

# Update
repo = await mesa.repos.update(repo="my-repo", name="renamed")

# Delete
await mesa.repos.delete(repo="my-repo")

Bookmarks

bookmarks = await mesa.bookmarks.list(repo="my-repo")
await mesa.bookmarks.create(repo="my-repo", name="feature-x", change_id="abc123")
await mesa.bookmarks.move(repo="my-repo", bookmark="feature-x", change_id="def456")
await mesa.bookmarks.merge(repo="my-repo", source="feature-x", target="main", message="Merge feature-x into main")
await mesa.bookmarks.delete(repo="my-repo", bookmark="feature-x")

Changes

from mesa_sdk import Author, FileUpsert

changes = await mesa.changes.list(repo="my-repo")
change = await mesa.changes.create(
    repo="my-repo",
    base_change_id="abc123",
    message="Add feature",
    author=Author(name="Alice", email="alice@example.com"),
    files=[FileUpsert(path="hello.txt", content="Hello, world!")],
)
change = await mesa.changes.get(repo="my-repo", change_id="def456")

Content & Diffs

content = await mesa.content.get(repo="my-repo", change_id="abc123")
diff = await mesa.diffs.get(
    repo="my-repo",
    base_change_id="abc123",
    head_change_id="def456",
)

API Keys

keys = await mesa.api_keys.list()
key = await mesa.api_keys.create(name="ci-key", scopes=["read", "write"])
await mesa.api_keys.revoke(key_id=key.id)

Webhook Targets

endpoints = await mesa.webhook_targets.list()
endpoint = await mesa.webhook_targets.create(url="https://example.com/hook", events=["change.created"])
await mesa.webhook_targets.update(webhook_target_id=endpoint.id, events=["push"])
await mesa.webhook_targets.delete(webhook_target_id=endpoint.id)

Webhook Handlers

Register handlers with mesa.webhooks.on(...) and pass the raw request body and headers to mesa.webhooks.receive(...). receive verifies the signature, parses the payload, and dispatches registered handlers.

from fastapi import FastAPI, Request
from mesa_sdk import Mesa

app = FastAPI()
mesa = Mesa(api_key="mk_...", org="acme", webhook_secret="whsec_...")

mesa.webhooks.on("push", lambda event: print(event["data"]["updates"]))

@app.post("/webhooks/mesa")
async def mesa_webhook(request: Request):
    await mesa.webhooks.receive(await request.body(), request.headers)
    return {"ok": True}

Virtual Filesystem

Mount repositories as a local filesystem for direct file I/O. The mount() context manager handles setup and teardown automatically.

async with mesa.fs.mount(repos=["my-repo"]) as fs:
    data = await fs.read("/my-repo/src/main.py")
    await fs.write("/my-repo/src/new_file.py", b"print('hello')")
    entries = await fs.readdir("/my-repo/src")

Read-only Repos

Pass RepoConfig(..., read_only=True) to mount a repo read-only. Writes to it raise OSError: [Errno 30] Read-only file system. A single mount can mix read-only and writable repos.

from mesa_sdk import RepoConfig

async with mesa.fs.mount(repos=[RepoConfig("my-repo", read_only=True)]) as fs:
    data = await fs.read("/my-repo/README.md")

Multiple Repos

Mount several repositories at once. Each repo appears as a top-level directory.

async with mesa.fs.mount(repos=["repo-a", "repo-b"]) as fs:
    a = await fs.read("/repo-a/file.txt")
    b = await fs.read("/repo-b/file.txt")

Pin to Bookmark or Change

Use RepoConfig to pin a mount to a specific bookmark or change.

from mesa_sdk import RepoConfig

async with mesa.fs.mount(repos=[
    RepoConfig("my-repo", bookmark="feature-x"),
    RepoConfig("other-repo", change_id="abc123"),
]) as fs:
    data = await fs.read("/my-repo/file.txt")

Bash

Run shell commands inside the mounted filesystem with fs.bash().

async with mesa.fs.mount(repos=["my-repo"]) as fs:
    bash = fs.bash(env={"FOO": "bar"}, cwd="/my-repo", timeout_ms=30000)
    result = await bash.exec("ls -la")
    print(result.stdout, result.stderr, result.exit_code)

bash.exec() returns an ExecResult with stdout: bytes, stderr: bytes, and exit_code: int.

Changes and Bookmarks (on the Filesystem)

Create and manage changes and bookmarks directly from a mounted filesystem.

async with mesa.fs.mount(repos=["my-repo"]) as fs:
    # Changes
    change = await fs.changes.new("my-repo", bookmark="main")
    change = await fs.changes.edit("my-repo", change_id="abc123")
    changes = await fs.changes.list("my-repo", limit=50)

    # Bookmarks
    await fs.bookmarks.create("my-repo", "feature-y")
    await fs.bookmarks.move("my-repo", "main", change_id=change)
    bookmarks = await fs.bookmarks.list("my-repo")

Disk Cache

Enable on-disk caching to speed up repeated mounts.

from mesa_sdk import DiskCacheConfig

async with mesa.fs.mount(
    repos=["my-repo"],
    disk_cache=DiskCacheConfig(path="/tmp/mesa-cache", max_size_bytes=1_000_000_000),
) as fs:
    data = await fs.read("/my-repo/file.txt")

Filesystem Errors

Filesystem operations raise standard Python exceptions:

Exception Condition
FileNotFoundError Path does not exist
FileExistsError Path already exists (e.g. mkdir without parents)
IsADirectoryError Expected a file, got a directory
NotADirectoryError Expected a directory, got a file
OSError General I/O failure; read-only repos use the read-only filesystem errno
NotImplementedError Operation not supported (e.g. link)

Low-Level REST Access

For operations not covered by the resource namespaces, install and use mesa-rest directly, or call the API with your own HTTP client:

from mesa_rest.api.repo import list_repos
from mesa_rest.client import AuthenticatedClient

client = AuthenticatedClient(
    base_url="https://api.mesa.dev/v1",
    token="mk_...",
    prefix="Bearer",
)
response = await list_repos.asyncio_detailed("acme", client=client)

Configuration

Mesa accepts the following keyword arguments:

Parameter Type Default Description
api_key str | None MESA_API_KEY env var API key for authentication
api_url str https://api.mesa.dev/v1 Base URL for the Mesa API
org str | None Resolved from /whoami Default organization slug
user_agent str | None None Custom user agent suffix
webhook_secret str | None None Secret used by mesa.webhooks.receive(...)

Error Handling

The SDK raises typed exceptions for API errors:

from mesa_sdk import Mesa, NotFoundError, AuthenticationError

async with Mesa() as mesa:
    try:
        repo = await mesa.repos.get(repo="nonexistent")
    except NotFoundError:
        print("Repo not found")
    except AuthenticationError:
        print("Invalid API key")
Exception HTTP Status Description
ValidationError 400, 406 Invalid request parameters
AuthenticationError 401 Invalid or missing API key
AuthorizationError 403 Insufficient permissions
NotFoundError 404 Resource not found
ConflictError 409 Resource conflict
RateLimitError 429 Rate limit exceeded
ServerError 5xx Server-side error

All API exceptions inherit from ApiError, which inherits from MesaError.

Package Relationship

  • mesa-sdk is the ergonomic, main SDK.
  • mesa-rest is the generated REST client used under the hood.

Use mesa-rest directly, or call the API with your own HTTP client, when you need low-level REST access beyond the resource namespaces.

Project details


Download files

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

Source Distributions

No source distribution files available for this release.See tutorial on generating distribution archives.

Built Distributions

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

mesa_sdk-0.37.0-cp310-abi3-musllinux_1_2_x86_64.whl (12.5 MB view details)

Uploaded CPython 3.10+musllinux: musl 1.2+ x86-64

mesa_sdk-0.37.0-cp310-abi3-musllinux_1_2_aarch64.whl (12.2 MB view details)

Uploaded CPython 3.10+musllinux: musl 1.2+ ARM64

mesa_sdk-0.37.0-cp310-abi3-manylinux_2_34_x86_64.whl (12.4 MB view details)

Uploaded CPython 3.10+manylinux: glibc 2.34+ x86-64

mesa_sdk-0.37.0-cp310-abi3-manylinux_2_34_aarch64.whl (12.0 MB view details)

Uploaded CPython 3.10+manylinux: glibc 2.34+ ARM64

mesa_sdk-0.37.0-cp310-abi3-macosx_11_0_arm64.whl (11.0 MB view details)

Uploaded CPython 3.10+macOS 11.0+ ARM64

File details

Details for the file mesa_sdk-0.37.0-cp310-abi3-musllinux_1_2_x86_64.whl.

File metadata

  • Download URL: mesa_sdk-0.37.0-cp310-abi3-musllinux_1_2_x86_64.whl
  • Upload date:
  • Size: 12.5 MB
  • Tags: CPython 3.10+, musllinux: musl 1.2+ x86-64
  • Uploaded using Trusted Publishing? No
  • Uploaded via: uv/0.11.14 {"installer":{"name":"uv","version":"0.11.14","subcommand":["publish"]},"python":null,"implementation":{"name":null,"version":null},"distro":{"name":"Ubuntu","version":"22.04","id":"jammy","libc":null},"system":{"name":null,"release":null},"cpu":null,"openssl_version":null,"setuptools_version":null,"rustc_version":null,"ci":true}

File hashes

Hashes for mesa_sdk-0.37.0-cp310-abi3-musllinux_1_2_x86_64.whl
Algorithm Hash digest
SHA256 64baf3d7360497c79396a9e1267f3936b0f21e934ed61dc9a80c22cb58a88c70
MD5 d896953afeee876287e8dcb610fd78c8
BLAKE2b-256 4209e8e41e5cd8e9fcef013db6a40cf18f205b1816be232d95525e9d6a8a4313

See more details on using hashes here.

File details

Details for the file mesa_sdk-0.37.0-cp310-abi3-musllinux_1_2_aarch64.whl.

File metadata

  • Download URL: mesa_sdk-0.37.0-cp310-abi3-musllinux_1_2_aarch64.whl
  • Upload date:
  • Size: 12.2 MB
  • Tags: CPython 3.10+, musllinux: musl 1.2+ ARM64
  • Uploaded using Trusted Publishing? No
  • Uploaded via: uv/0.11.14 {"installer":{"name":"uv","version":"0.11.14","subcommand":["publish"]},"python":null,"implementation":{"name":null,"version":null},"distro":{"name":"Ubuntu","version":"22.04","id":"jammy","libc":null},"system":{"name":null,"release":null},"cpu":null,"openssl_version":null,"setuptools_version":null,"rustc_version":null,"ci":true}

File hashes

Hashes for mesa_sdk-0.37.0-cp310-abi3-musllinux_1_2_aarch64.whl
Algorithm Hash digest
SHA256 cb2cbb350e236d94af791a32a0eac9dcf8bba6843c159b340ab0b5a821fe5141
MD5 71c26374732167ddca4dbfc5d23fe242
BLAKE2b-256 367c50c28ba426b7f5c2f3c0daa3e931b08471ea2e50c3908c5f3ea62f3fcf7e

See more details on using hashes here.

File details

Details for the file mesa_sdk-0.37.0-cp310-abi3-manylinux_2_34_x86_64.whl.

File metadata

  • Download URL: mesa_sdk-0.37.0-cp310-abi3-manylinux_2_34_x86_64.whl
  • Upload date:
  • Size: 12.4 MB
  • Tags: CPython 3.10+, manylinux: glibc 2.34+ x86-64
  • Uploaded using Trusted Publishing? No
  • Uploaded via: uv/0.11.14 {"installer":{"name":"uv","version":"0.11.14","subcommand":["publish"]},"python":null,"implementation":{"name":null,"version":null},"distro":{"name":"Ubuntu","version":"22.04","id":"jammy","libc":null},"system":{"name":null,"release":null},"cpu":null,"openssl_version":null,"setuptools_version":null,"rustc_version":null,"ci":true}

File hashes

Hashes for mesa_sdk-0.37.0-cp310-abi3-manylinux_2_34_x86_64.whl
Algorithm Hash digest
SHA256 af2031be983c1a887bf12980de2730abf28a681cb12951e635c1eb4d5ba8ae04
MD5 4f09145ea56bb7ddbb0b61e0a0e78b9a
BLAKE2b-256 dd7262d39e463b4f93cff5a095a30746f4502aa289fa7b33b3383541076049e3

See more details on using hashes here.

File details

Details for the file mesa_sdk-0.37.0-cp310-abi3-manylinux_2_34_aarch64.whl.

File metadata

  • Download URL: mesa_sdk-0.37.0-cp310-abi3-manylinux_2_34_aarch64.whl
  • Upload date:
  • Size: 12.0 MB
  • Tags: CPython 3.10+, manylinux: glibc 2.34+ ARM64
  • Uploaded using Trusted Publishing? No
  • Uploaded via: uv/0.11.14 {"installer":{"name":"uv","version":"0.11.14","subcommand":["publish"]},"python":null,"implementation":{"name":null,"version":null},"distro":{"name":"Ubuntu","version":"22.04","id":"jammy","libc":null},"system":{"name":null,"release":null},"cpu":null,"openssl_version":null,"setuptools_version":null,"rustc_version":null,"ci":true}

File hashes

Hashes for mesa_sdk-0.37.0-cp310-abi3-manylinux_2_34_aarch64.whl
Algorithm Hash digest
SHA256 6715966b295cf4f8377e62e111d463e6c6f18cf2f9a149e45c61c6d9d371d79d
MD5 c0859b61584e23f03fe593e550a5304f
BLAKE2b-256 9c1e7a362a4551bda45bd7bb959b3243a3bec87a412fccb86f28b021e925ab43

See more details on using hashes here.

File details

Details for the file mesa_sdk-0.37.0-cp310-abi3-macosx_11_0_arm64.whl.

File metadata

  • Download URL: mesa_sdk-0.37.0-cp310-abi3-macosx_11_0_arm64.whl
  • Upload date:
  • Size: 11.0 MB
  • Tags: CPython 3.10+, macOS 11.0+ ARM64
  • Uploaded using Trusted Publishing? No
  • Uploaded via: uv/0.11.14 {"installer":{"name":"uv","version":"0.11.14","subcommand":["publish"]},"python":null,"implementation":{"name":null,"version":null},"distro":{"name":"Ubuntu","version":"22.04","id":"jammy","libc":null},"system":{"name":null,"release":null},"cpu":null,"openssl_version":null,"setuptools_version":null,"rustc_version":null,"ci":true}

File hashes

Hashes for mesa_sdk-0.37.0-cp310-abi3-macosx_11_0_arm64.whl
Algorithm Hash digest
SHA256 f25ce07c57ec6fcf3c88e9d55610ec2cd9d66bbb3a8728f0c05451f94f4e6ff6
MD5 dabab4397b3f74a683dd0c2449e8075c
BLAKE2b-256 5f305ae3536002f66c3780b3c0314571f4cd008dfacd3ac7f66cdd948270a4eb

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