Skip to main content

PyAPIq is a modern Python toolkit for building both synchronous and asynchronous API clients with clean, minimal code and full type safety.

Project description

📦 PyAPIq

PyPI Python Versions License

PyAPIq is a modern Python toolkit for building both synchronous and asynchronous API clients with clean, minimal code and full type safety.

Define endpoints using decorators like @sync_endpoint or @async_endpoint, structure logic with optional namespaces, and leverage Pydantic models for strict request/response validation — all with built-in rate limiting and retries.

Downloads Downloads Downloads

Installation

pip install pyapiq

Quickstart

1. Define your models

Use Pydantic to define request and response schemas with full type safety:

from typing import List
from pydantic import BaseModel


class BulkAccountsRequest(BaseModel):
    account_ids: List[str]


class AccountInfoResponse(BaseModel):
    address: str
    balance: int
    status: str


class BulkAccountsResponse(BaseModel):
    accounts: List[AccountInfoResponse]

2. Define your client

Declare your API client by subclassing AsyncClientAPI and annotating endpoints with @async_endpoint:

from pyapiq import AsyncClientAPI, async_endpoint
from pyapiq.types import HTTPMethod


class AsyncTONAPI(AsyncClientAPI):
    base_url = "https://tonapi.io"
    headers = {"Authorization": "Bearer <YOUR_API_KEY>"}
    version = "v2"
    rps = 1
    max_retries = 2

    @async_endpoint(HTTPMethod.GET)
    async def status(self) -> dict:
        """Check API status (GET /status)"""

    @async_endpoint(HTTPMethod.GET)
    async def rates(self, tokens: str, currencies: str) -> dict:
        """Get token rates (GET /rates?tokens={tokens}&currencies={currencies})"""

Notes:

  • If you prefer synchronous clients, simply use SyncClientAPI with @sync_endpoint instead.
  • For synchronous clients, use SyncClientAPI and @sync_endpoint — interface is fully symmetrical.
  • Method arguments are automatically mapped to path and query parameters. The return value is parsed from JSON and returned as a dict, unless a return_as=Model is specified.

3. Group endpoints with namespaces (optional)

Use AsyncAPINamespace to logically organize endpoints under a common prefix (e.g., /accounts):

from pyapiq import AsyncAPINamespace, async_endpoint
from pyapiq.types import HTTPMethod


class Accounts(AsyncAPINamespace):
    namespace = "accounts"

    @async_endpoint(HTTPMethod.GET, path="/{account_id}", return_as=AccountInfoResponse)
    async def info(self, account_id: str) -> AccountInfoResponse:
        """Retrieve account information by account_id (GET /accounts/{account_id})"""

    @async_endpoint(HTTPMethod.POST, path="/_bulk", return_as=BulkAccountsResponse)
    async def bulk_info(self, payload: BulkAccountsRequest) -> BulkAccountsResponse:
        """Retrieve info for multiple accounts with a Pydantic model (POST /accounts/_bulk)"""

    @async_endpoint(HTTPMethod.POST, path="/_bulk")
    async def bulk_info_dict(self, payload: dict) -> dict:
        """Retrieve info for multiple accounts with a dict payload (POST /accounts/_bulk)"""

Then include the namespace in your main client:

class AsyncTONAPI(AsyncClientAPI):
    base_url = "https://tonapi.io"
    headers = {"Authorization": "Bearer <YOUR_API_KEY>"}
    version = "v2"
    rps = 1
    max_retries = 2

    # ... endpoints above ...

    @property
    def accounts(self) -> Accounts:
        return Accounts(self)

Note: The namespace can be defined with or without a leading slash. It will be joined correctly with endpoint paths. Each namespace instance receives the parent client instance automatically.

4. Usage

async def main():
    tonapi = AsyncTONAPI()

    async with tonapi:
        # GET /status
        status = await tonapi.status()
        print(status)

Note: Always use async with to open and close the session properly. All retries, throttling, and connection reuse are handled under the hood.

API Configuration

All settings are defined as class attributes on your client class:

Name Type Description Default
base_url str Base URL of the API (must start with http:// or https://)
version str Optional API version prefix (e.g. "v1"/v1/... in requests) None
rps int Max requests per second (client-side rate limit) 1
max_retries int Max automatic retries for HTTP 429 (Too Many Requests) 3
headers dict Default headers to send with each request None
cookies dict Default cookies to send with each request None
timeout float Default request timeout in seconds None

Note: The version field is automatically prefixed to all endpoint paths (e.g. /v2/accounts/...). Rate limiting and retries are handled transparently and apply only per-client instance.

Endpoints

  • Use @async_endpoint(method, path=..., return_as=...) to declare each endpoint.

  • All method arguments are automatically mapped:

    • Scalars → query/path parameters
    • dict or Pydantic models → request body
  • return_as=Model lets you parse responses into Pydantic models.

  • If omitted, response is returned as dict.

Note: If path is omitted, the method name becomes the endpoint path (e.g. rates/rates). You can define both flat and namespaced methods together.

Notes

  • Fully asynchronous: all clients and endpoints require async with.
  • Zero boilerplate: request building, error handling, retries, and throttling are automatic.
  • Namespaces help organize large APIs, but are optional.
  • Both dicts and Pydantic models are supported in request payloads.
  • Great for building typed SDKs or internal tools.

Contribution

We welcome your contributions! If you find a bug or have an idea, please open an issue or submit a pull request.

License

Distributed under the MIT License. Use freely for commercial or personal projects.

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

pyapiq-0.1.7.tar.gz (15.9 kB view details)

Uploaded Source

Built Distribution

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

pyapiq-0.1.7-py3-none-any.whl (19.3 kB view details)

Uploaded Python 3

File details

Details for the file pyapiq-0.1.7.tar.gz.

File metadata

  • Download URL: pyapiq-0.1.7.tar.gz
  • Upload date:
  • Size: 15.9 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/6.1.0 CPython/3.10.11

File hashes

Hashes for pyapiq-0.1.7.tar.gz
Algorithm Hash digest
SHA256 8d648a103f2be0229102c78db199cc80a1567e88f1fb0849de6efde71c856fed
MD5 9d1000c9a1a8234f4ca4211882146abe
BLAKE2b-256 dc7702da0d76fef29e35e780c30721a89bad6bfbd19bae23d1c0917a6790c063

See more details on using hashes here.

File details

Details for the file pyapiq-0.1.7-py3-none-any.whl.

File metadata

  • Download URL: pyapiq-0.1.7-py3-none-any.whl
  • Upload date:
  • Size: 19.3 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/6.1.0 CPython/3.10.11

File hashes

Hashes for pyapiq-0.1.7-py3-none-any.whl
Algorithm Hash digest
SHA256 39f67405510d3c4d66455ad19db5e3d75332075fbecca57210c4baa603a9dbed
MD5 9d4fa5b32b4c83b6366aab501682ac59
BLAKE2b-256 d154fd03118d8d85dfcd3193c3b3d60c5521709a27000e6f62cd3e604a6979a6

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