Skip to main content

ChatGPT (Codex) OAuth transport for Axio using Responses API

Project description

axio-transport-codex

PyPI Python License: MIT

ChatGPT OAuth transport for axio using the OpenAI Responses API.

Authenticates via the same OAuth2 PKCE flow used by the ChatGPT desktop client - no API key purchase required; your ChatGPT subscription covers usage. Implements the OpenAI Responses API (not the legacy chat completions endpoint).

Features

  • No API key purchase - authenticates with your ChatGPT account via OAuth2 PKCE
  • Responses API - uses OpenAI's newer Responses endpoint (/backend-api/codex/responses)
  • Automatic token refresh - access tokens are silently refreshed before expiry; callers notified via on_auth_refresh
  • Full streaming - incremental text, reasoning, and tool-call events via SSE
  • Tool calling - works with all axio tool handlers; parallel tool calls enabled
  • Retry logic - automatic backoff on 429 and 5xx responses; honours Retry-After header
  • aiohttp-based - non-blocking I/O throughout

Installation

pip install axio-transport-codex

With the TUI settings screen (ChatGPT sign-in dialog):

pip install "axio-transport-codex[tui]"

Authentication

CodexTransport authenticates with ChatGPT using OAuth2 PKCE. The transport stores an access token (api_key), a refresh token, and an expiry timestamp. It refreshes the access token automatically when it is within 30 seconds of expiry.

Running the OAuth flow

Use run_oauth_flow() from axio_transport_codex.oauth to obtain tokens for the first time. It opens a browser window for ChatGPT sign-in and waits for the callback on http://localhost:1455/auth/callback.

import asyncio
from axio_transport_codex.oauth import run_oauth_flow
from axio_transport_codex import CodexTransport

async def main() -> None:
    tokens = await run_oauth_flow()
    # tokens = {
    #   "access_token": "...",
    #   "refresh_token": "...",
    #   "expires_at": "1234567890",   # Unix timestamp as string
    #   "account_id": "...",
    # }

    transport = CodexTransport(
        api_key=tokens["access_token"],
        refresh_token=tokens["refresh_token"],
        expires_at=tokens["expires_at"],
        account_id=tokens["account_id"],
    )

Persisting tokens across sessions

Pass an on_auth_refresh callback to receive updated credentials whenever the access token is refreshed. Save the returned dict to disk or a secrets store and restore it on the next run.

import json, pathlib, aiohttp
from axio_transport_codex import CodexTransport

CRED_FILE = pathlib.Path("~/.config/axio/codex.json").expanduser()

async def save_tokens(tokens: dict[str, str]) -> None:
    CRED_FILE.parent.mkdir(parents=True, exist_ok=True)
    CRED_FILE.write_text(json.dumps(tokens))

async def main() -> None:
    creds = json.loads(CRED_FILE.read_text()) if CRED_FILE.exists() else {}

    async with aiohttp.ClientSession() as session:
        transport = CodexTransport(
            api_key=creds.get("api_key", ""),
            refresh_token=creds.get("refresh_token", ""),
            expires_at=creds.get("expires_at", ""),
            account_id=creds.get("account_id", ""),
            session=session,
            on_auth_refresh=save_tokens,
        )
        # use transport with an Agent ...

Refreshing tokens manually

from axio_transport_codex.oauth import refresh_access_token

tokens = await refresh_access_token(refresh_token="...")
# tokens = {"access_token": ..., "refresh_token": ..., "expires_at": ..., "account_id": ...}

Usage

import asyncio
import aiohttp
from axio.agent import Agent
from axio.context import MemoryContextStore
from axio.events import TextDelta
from axio_transport_codex import CodexTransport, CODEX_MODELS

async def main() -> None:
    async with aiohttp.ClientSession() as session:
        transport = CodexTransport(
            api_key="<your-chatgpt-access-token>",
            refresh_token="<your-refresh-token>",
            expires_at="<unix-timestamp-string>",
            account_id="<your-account-id>",
            model=CODEX_MODELS["gpt-4.1"],
            session=session,
        )
        agent = Agent(system="You are a helpful assistant.", tools=[], transport=transport)
        ctx = MemoryContextStore()
        async for event in agent.run_stream("Explain the Rust borrow checker.", ctx):
            if isinstance(event, TextDelta):
                print(event.delta, end="", flush=True)
        print()

asyncio.run(main())

The session parameter is required for streaming. Create an aiohttp.ClientSession in an async context and pass it to the transport.

Configuration reference

CodexTransport is a dataclass with the following fields:

Field Type Default Description
name str "ChatGPT (Codex)" Display name (used by TUI)
api_key str "" ChatGPT OAuth access token
refresh_token str "" OAuth refresh token for silent renewal
expires_at str "" Access token expiry as a Unix timestamp string
account_id str "" ChatGPT account/organisation ID (sent as ChatGPT-Account-ID header)
base_url str "https://chatgpt.com/backend-api/codex" API base URL
model ModelSpec CODEX_MODELS["gpt-4.1"] Active model
models ModelRegistry all CODEX_MODELS Available models
session aiohttp.ClientSession | None None HTTP session (required for streaming)
on_auth_refresh Callable[[dict[str, str]], Awaitable[None]] | None None Callback invoked with fresh credentials after token refresh
max_retries int 10 Maximum retry attempts on 429 / 5xx
retry_base_delay float 5.0 Base delay in seconds for exponential backoff

Models

Model ID Context Max output Capabilities
gpt-4.1 1,047,576 32,768 text, vision, tool use
gpt-4.1-mini 1,047,576 32,768 text, vision, tool use
gpt-4.1-nano 1,047,576 32,768 text, tool use
gpt-4o 128,000 16,384 text, vision, tool use
gpt-4o-mini 128,000 16,384 text, vision, tool use
o4-mini 200,000 100,000 text, reasoning, tool use
o3 200,000 100,000 text, reasoning, tool use
o3-mini 200,000 100,000 text, reasoning, tool use

The default model is gpt-4.1.

fetch_models()

await transport.fetch_models() queries the Codex /models endpoint for the list of models your account has access to. If the request fails or the account is not configured, it falls back to the built-in CODEX_MODELS registry. Models not found in the built-in registry are added with basic text and tool_use capabilities.

session and api_key must be set before calling fetch_models().

Plugin registration

When installed, this package registers itself via entry points so axio-tui discovers it automatically:

[project.entry-points."axio.transport"]
codex = "axio_transport_codex:CodexTransport"

[project.entry-points."axio.transport.settings"]
codex = "axio_transport_codex.tui:CodexSettingsScreen"

Part of the axio ecosystem

axio · axio-transport-openai · axio-transport-anthropic · axio-tui

License

MIT

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

axio_transport_codex-0.9.4.tar.gz (90.8 kB view details)

Uploaded Source

Built Distribution

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

axio_transport_codex-0.9.4-py3-none-any.whl (15.9 kB view details)

Uploaded Python 3

File details

Details for the file axio_transport_codex-0.9.4.tar.gz.

File metadata

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

File hashes

Hashes for axio_transport_codex-0.9.4.tar.gz
Algorithm Hash digest
SHA256 acae6de4f9e61b0a749b1e9e8093743c231335c2a4f241a0ad695e15297a4605
MD5 90e9c627b556aae0196a0ae7a58aaf8d
BLAKE2b-256 fcf6d0047aca15724536c41dc71a15efa9293d2600576b595cd0b2ff5323a181

See more details on using hashes here.

Provenance

The following attestation bundles were made for axio_transport_codex-0.9.4.tar.gz:

Publisher: publish.yml on mosquito/axio-agent

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

File details

Details for the file axio_transport_codex-0.9.4-py3-none-any.whl.

File metadata

File hashes

Hashes for axio_transport_codex-0.9.4-py3-none-any.whl
Algorithm Hash digest
SHA256 c48d56074dafc3d3f742d37cb5949e62b9a0edb07dc389e2a3a5aeacb2a91838
MD5 a6a29fd5e2594a52dfc4d4ab3f00577c
BLAKE2b-256 9780820109fa5d318504fe63a9c6bea8b7b7336dc0d332fb06b7ee04f9dfae12

See more details on using hashes here.

Provenance

The following attestation bundles were made for axio_transport_codex-0.9.4-py3-none-any.whl:

Publisher: publish.yml on mosquito/axio-agent

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