Skip to main content

Async Python client for the Max messenger Bot API

Project description

python-max-bot

PyPI Python License: MIT

Async Python client for the Max messenger Bot API.

A thin, stable, Pydantic-modeled wrapper. No bot framework, no FSM, no middleware — just the typed API surface for messaging, uploads, updates, webhooks, and chat introspection.

Install

pip install python-max-bot

Quickstart

import asyncio
from max_bot_api import MaxClient

async def main():
    async with MaxClient(token="YOUR_BOT_TOKEN") as client:
        # Send a plain message
        await client.send_message(chat_id=42, text="Hello, Max!")

        # Send a message with an image
        with open("photo.jpg", "rb") as f:
            img = await client.upload_image(f.read(), filename="photo.jpg")
        await client.send_message(
            chat_id=42,
            text="<b>Look at this</b>",
            format="html",
            attachments=[img],
        )

asyncio.run(main())

Features

Feature Status
Send / edit / delete messages
Long-poll updates (get_updates)
Image / video / audio / file uploads
Inline keyboards
Chat metadata (get_chat)
HTML and Markdown formatting
Typed exceptions per HTTP status
Auto-retry / backoff
Bot introspection (get_me, members, admins)
Webhook subscription endpoints
Action indicators (typing, sending photo, …)
Bot framework (handlers, FSM) ❌ out of scope

Polling for updates

Long-poll the API and walk the marker cursor:

import asyncio
from max_bot_api import MaxClient

async def main():
    async with MaxClient(token="YOUR_BOT_TOKEN") as client:
        marker: int | None = None
        while True:
            page = await client.get_updates(marker=marker, timeout=30)
            for update in page.updates:
                # `update` is a discriminated-union Pydantic model — check
                # `update.update_type` or use isinstance on the subtypes.
                print(update)
            marker = page.marker

asyncio.run(main())

get_updates(timeout=30) blocks server-side for up to 30 seconds waiting for new events; pass the returned marker back on the next call to advance the cursor. While a webhook subscription is active, this method is disabled — pick one transport, not both.

Introspection

from max_bot_api import MaxClient

async with MaxClient(token) as client:
    # Who am I?
    me = await client.get_me()
    print(me.user_id, me.username, me.first_name)

    # Walk a paginated member list
    page = await client.get_chat_members(chat_id=42, count=50)
    while True:
        for member in page.members:
            print(member.user_id, member.first_name)
        if page.marker is None:
            break
        page = await client.get_chat_members(chat_id=42, marker=page.marker, count=50)

    # Filter mode — fetch only specific users (overrides marker/count server-side)
    page = await client.get_chat_members(chat_id=42, user_ids=[100, 200])

    # Admins (bot must itself be an admin in the chat)
    admins = await client.get_chat_admins(chat_id=42)
    for admin in admins.members:
        print(admin.user_id, admin.permissions)

    # "Am I admin here?" preflight before doing admin-only operations
    self_membership = await client.get_my_chat_membership(42)
    if self_membership.is_admin:
        ...

Action indicators

Send a typing or sending-photo indicator while you do work:

from max_bot_api import MaxClient, ChatAction

async with MaxClient(token) as client:
    await client.send_action(42, ChatAction.TYPING_ON)
    reply = await compute_expensive_reply(...)
    await client.send_message(chat_id=42, text=reply)

Available actions: TYPING_ON, SENDING_PHOTO, SENDING_VIDEO, SENDING_AUDIO, SENDING_FILE.

Errors

from max_bot_api import (
    MaxClient,
    MaxAuthError,
    MaxRateLimitError,
    MaxTransportError,
)

async with MaxClient(token=...) as client:
    try:
        await client.send_message(chat_id=42, text="hi")
    except MaxAuthError:
        # Bad token — surface to user
        ...
    except MaxRateLimitError as e:
        await asyncio.sleep(e.retry_after or 1)
    except MaxTransportError:
        # Network problem; retry your way
        ...

subscribe, unsubscribe, and send_action can also raise MaxBadResponseError — the API returned 2xx but with {"success": false, "message": "..."} in the body. It inherits from MaxError (not MaxAPIError, since the HTTP call itself succeeded), and .message carries the server's explanation.

Retries

Retries are opt-in. Pass a RetryPolicy to the client constructor:

from max_bot_api import MaxClient, RetryPolicy

async with MaxClient(token, retry=RetryPolicy()) as client:
    await client.get_chat(42)               # retries on 5xx and transport errors
    await client.send_message(chat_id=42, text="hi")  # only retries on transport errors

The default policy: 3 attempts, exponential backoff (1s, 2s, 4s with ±25% jitter), capped at 30s per wait. Tune via RetryPolicy(max_attempts=..., backoff_initial=..., backoff_multiplier=..., backoff_max=..., jitter=...).

Read methods (get_messages, get_updates, get_chat, request_upload_url) retry on both transport errors and 5xx responses. Write methods (send_message, edit_message, delete_message, the upload POST) only retry on transport errors — a 5xx during a write could mean the server processed it, so blind retry could double-apply.

429 responses always retry. The library waits at least as long as the server's Retry-After header asks; backoff_max does not clamp the server's instruction.

Without retry=, behavior is identical to v0.1 — one attempt per call.

Webhooks

Register and manage webhook subscriptions (the receiver server is yours to run):

from max_bot_api import MaxClient, UpdateType

async with MaxClient(token) as client:
    await client.subscribe(
        url="https://my.server/max-webhook",
        update_types=[UpdateType.MESSAGE_CREATED],
        secret="my-shared-secret-12345",
    )

    subs = await client.get_subscriptions()
    for s in subs:
        print(s.url, s.update_types)

    await client.unsubscribe(url="https://my.server/max-webhook")

The URL must be HTTPS — subscribe() raises ValueError on http:// locally before any network call. While a subscription is active, long-polling via get_updates() is disabled.

Built with Claude

This project is developed in collaboration with Claude, Anthropic's AI assistant — design, code, and tests. Commits are co-authored. The repo ships with a CLAUDE.md and .claude/settings.json so any contributor running Claude on the codebase picks up the same conventions automatically. See the v0.1 design for the full collaboration workflow.

Links

License

MIT — see LICENSE.

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

python_max_bot-0.4.0.tar.gz (30.1 kB view details)

Uploaded Source

Built Distribution

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

python_max_bot-0.4.0-py3-none-any.whl (22.6 kB view details)

Uploaded Python 3

File details

Details for the file python_max_bot-0.4.0.tar.gz.

File metadata

  • Download URL: python_max_bot-0.4.0.tar.gz
  • Upload date:
  • Size: 30.1 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/6.1.0 CPython/3.13.12

File hashes

Hashes for python_max_bot-0.4.0.tar.gz
Algorithm Hash digest
SHA256 69d2631b44a916e99cecf3d12106f5369a8d125721e0a88a6f8831d4d4e235e1
MD5 05ebe73f31731aef3d580f9ac6867dc2
BLAKE2b-256 fb10a7fc839715e54ba8ce758619187e183d3c28956069169d5980b7e7972572

See more details on using hashes here.

Provenance

The following attestation bundles were made for python_max_bot-0.4.0.tar.gz:

Publisher: release.yml on Kitaeza/python-max-bot

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

File details

Details for the file python_max_bot-0.4.0-py3-none-any.whl.

File metadata

  • Download URL: python_max_bot-0.4.0-py3-none-any.whl
  • Upload date:
  • Size: 22.6 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/6.1.0 CPython/3.13.12

File hashes

Hashes for python_max_bot-0.4.0-py3-none-any.whl
Algorithm Hash digest
SHA256 fe11c396e1bf5baf1e8ba352fd44da2d0988efb1b79eaa90e6efdf5113df1201
MD5 cc0d0d8403b005972376b2c15b952f40
BLAKE2b-256 739473202c358fc62462cb81ee811a236b96b21713320cc73ccbe0b37212ad92

See more details on using hashes here.

Provenance

The following attestation bundles were made for python_max_bot-0.4.0-py3-none-any.whl:

Publisher: release.yml on Kitaeza/python-max-bot

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

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