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.6.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.6-py3-none-any.whl (15.9 kB view details)

Uploaded Python 3

File details

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

File metadata

  • Download URL: axio_transport_codex-0.9.6.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.6.tar.gz
Algorithm Hash digest
SHA256 7ffdf58bfce6b975f54a1b953990f4ddb6684baeb434ff08b257b93c5b1e3713
MD5 259bd3fb246189966a4d13212e503533
BLAKE2b-256 0b0f3f82a3f364a881d6baaff2d773abd3d622d56f2b5b1173a6ce502ad83d3d

See more details on using hashes here.

Provenance

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

File metadata

File hashes

Hashes for axio_transport_codex-0.9.6-py3-none-any.whl
Algorithm Hash digest
SHA256 971325be9d90830abeedd2f88ec6be93a8e9fd9c500ca577574f2d371f90f251
MD5 bb18d67a2477307410f271ebbc61f19e
BLAKE2b-256 da31c8a38873cb05b780400750fd9a7ca3b839ed5e3a01b7fc3546dfc5324eb4

See more details on using hashes here.

Provenance

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