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.8.0.tar.gz (90.6 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.8.0-py3-none-any.whl (15.6 kB view details)

Uploaded Python 3

File details

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

File metadata

  • Download URL: axio_transport_codex-0.8.0.tar.gz
  • Upload date:
  • Size: 90.6 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.8.0.tar.gz
Algorithm Hash digest
SHA256 ca8250d3750c5561682ea4243ce25f9ed7548d011a456985090c0cef952e6bc3
MD5 b520b4910c10a6d9d57f4201e8a4fd57
BLAKE2b-256 f7f4ccdb782872f828fe4a6d8117fd87ed05ca6d2dad82874211897f1f4a55ac

See more details on using hashes here.

Provenance

The following attestation bundles were made for axio_transport_codex-0.8.0.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.8.0-py3-none-any.whl.

File metadata

File hashes

Hashes for axio_transport_codex-0.8.0-py3-none-any.whl
Algorithm Hash digest
SHA256 b1140d633bffc463b37b2ad8d4d6eafda0cd8e49d6223752dcf22b93beb13efe
MD5 abc273db4c2cec1c99a0d20ed7232ca5
BLAKE2b-256 c32f057e2c6139423604aeb1816ec85f0a26f8009f45feff182d69da6b32f4cc

See more details on using hashes here.

Provenance

The following attestation bundles were made for axio_transport_codex-0.8.0-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