Codex app-server 是 Codex 面向交互能力更强的客户端提供的接口,例如 Codex VS Code 扩展。当你希望在自己的产品中做深度集成时,就应该使用它:认证、对话历史、审批,以及流式传输的智能体事件都属于它的能力范围。
app-server 的实现已在 Codex GitHub 仓库中开源(openai/codex/codex-rs/app-server)。如需查看 Codex 的完整开源组件列表,请参见开源页面。
协议
和 MCP 一样,codex app-server 支持通过 JSON-RPC 2.0 消息进行双向通信,只是在实际传输时省略了 "jsonrpc":"2.0" 这个字段。
支持两种传输方式:
stdio(--listen stdio://,默认):使用按行分隔的 JSON,也就是 JSONL。websocket(--listen ws://IP:PORT,实验性):每个 WebSocket 文本帧承载一条 JSON-RPC 消息。
在 WebSocket 模式下,app-server 使用有界队列。当请求入口队列已满时,服务端会拒绝新的请求,并返回 JSON-RPC 错误码 -32001 和消息 "Server overloaded; retry later."。客户端应采用带抖动的指数退避策略重试。
消息结构
请求消息包含 method、params 和 id:
{ "method": "thread/start", "id": 10, "params": { "model": "gpt-5.4" } }响应会回显同一个 id,并携带 result 或 error:
{ "id": 10, "result": { "thread": { "id": "thr_123" } } }{ "id": 10, "error": { "code": 123, "message": "Something went wrong" } }通知消息不带 id,只包含 method 和 params:
{ "method": "turn/started", "params": { "turn": { "id": "turn_456" } } }你也可以通过 CLI 直接生成 TypeScript schema 或 JSON Schema bundle。生成结果与当前运行的 Codex 版本严格对应:
codex app-server generate-ts --out ./schemas
codex app-server generate-json-schema --out ./schemas快速开始
最小握手流程如下:
- 使用
codex app-server启动服务端(默认stdio传输),或者使用codex app-server --listen ws://127.0.0.1:4500启动服务端(实验性的 WebSocket 传输)。 - 通过所选传输方式连接客户端,然后先发送
initialize,再发送initialized通知。 - 启动一个 thread 和一个 turn,然后持续从当前传输流中读取通知。
下面是一个最小 Node.js / TypeScript 示例:
import { spawn } from "node:child_process";
import readline from "node:readline";
const proc = spawn("codex", ["app-server"], {
stdio: ["pipe", "pipe", "inherit"],
});
const rl = readline.createInterface({ input: proc.stdout });
const send = (message: unknown) => {
proc.stdin.write(`${JSON.stringify(message)}\n`);
};
let threadId: string | null = null;
rl.on("line", (line) => {
const msg = JSON.parse(line) as any;
console.log("server:", msg);
if (msg.id === 1 && msg.result?.thread?.id && !threadId) {
threadId = msg.result.thread.id;
send({
method: "turn/start",
id: 2,
params: {
threadId,
input: [{ type: "text", text: "Summarize this repo." }],
},
});
}
});
send({
method: "initialize",
id: 0,
params: {
clientInfo: {
name: "my_product",
title: "My Product",
version: "0.1.0",
},
},
});
send({ method: "initialized", params: {} });
send({ method: "thread/start", id: 1, params: { model: "gpt-5.4" } });核心抽象
- Thread:用户与 Codex 智能体之间的一条对话。Thread 包含多个 turn。
- Turn:一次用户请求,以及之后智能体执行的整段工作。Turn 包含多个 item,并会流式产出增量更新。
- Item:输入或输出的一个单元,例如用户消息、智能体消息、命令执行、文件修改、工具调用等。
通过 thread 相关 API 创建、列出或归档对话;通过 turn 相关 API 驱动对话,并从 turn 通知中流式读取进度。
生命周期概览
- 每条连接只初始化一次:在打开传输连接后,立即发送带有客户端元数据的
initialize请求,然后发出initialized。在完成这次握手前,服务端会拒绝该连接上的其他任何请求。 - 启动或恢复线程:新对话使用
thread/start;继续已有对话使用thread/resume;要把现有历史分叉为新的 thread id,则使用thread/fork。 - 开始一个 turn:调用
turn/start,传入目标threadId和用户输入。可选字段可以覆盖 model、personality、cwd、沙箱策略等设置。 - 引导进行中的 turn:调用
turn/steer,在不创建新 turn 的前提下,把更多用户输入追加到当前仍在进行中的 turn。 - 流式读取事件:调用
turn/start后,持续从当前传输流中读取通知,例如thread/archived、thread/unarchived、item/started、item/completed、item/agentMessage/delta、工具执行进度等。 - 结束 turn:模型完成后,或在收到
turn/interrupt取消后,服务端会发出带最终状态的turn/completed。
初始化
每条传输连接都必须先发送一次 initialize,然后再通过 initialized 通知确认。在初始化完成前调用其他方法,会收到 Not initialized;在同一连接上重复发送 initialize,则会收到 Already initialized。
服务端会返回它向上游服务声明的 User-Agent 字符串,以及描述运行目标的 platformFamily 和 platformOs。你应通过 clientInfo 来标识自己的集成。
initialize.params.capabilities 还支持 optOutNotificationMethods,允许你为当前连接关闭某些指定通知。匹配是精确匹配,不支持通配符;未知方法名会被静默忽略。
重要:请使用 clientInfo.name 作为你在 OpenAI Compliance Logs Platform 中的客户端标识。如果你正在开发面向企业的全新 Codex 集成,请联系 OpenAI,将它加入已知客户端列表。相关背景可参考 Codex logs reference。
示例:
{
"method": "initialize",
"id": 0,
"params": {
"clientInfo": {
"name": "codex_vscode",
"title": "Codex VS Code Extension",
"version": "0.1.0"
}
}
}关闭部分通知的示例:
{
"method": "initialize",
"id": 1,
"params": {
"clientInfo": {
"name": "my_client",
"title": "My Client",
"version": "0.1.0"
},
"capabilities": {
"experimentalApi": true,
"optOutNotificationMethods": ["thread/started", "item/agentMessage/delta"]
}
}
}实验性 API 选项
部分 app-server 的方法和字段被有意放在 experimentalApi 能力开关后面。
- 省略
capabilities,或显式将experimentalApi设为false,表示你只使用稳定 API;此时服务端会拒绝实验性方法和字段。 - 将
capabilities.experimentalApi设为true,则可启用实验性方法和字段。
示例:
{
"method": "initialize",
"id": 1,
"params": {
"clientInfo": {
"name": "my_client",
"title": "My Client",
"version": "0.1.0"
},
"capabilities": {
"experimentalApi": true
}
}
}如果客户端在未启用 experimentalApi 的情况下发送实验性方法或字段,app-server 会拒绝请求,并返回:
<descriptor> requires experimentalApi capabilityAPI 概览
thread/start:创建新线程;会发出thread/started,并自动为当前线程订阅该线程的 turn / item 事件。thread/resume:按 id 重新打开已有线程,使后续turn/start可继续追加到该线程中。thread/fork:通过复制已保存的历史,把一条线程分叉到新的 thread id;会为新线程发出thread/started。thread/read:按 id 读取已保存线程而不恢复它;设置includeTurns可返回完整 turn 历史。返回的thread对象包含运行时status。thread/list:分页列出已保存线程;支持基于 cursor 的分页,以及modelProviders、sourceKinds、archived和cwd过滤。返回的thread对象包含运行时status。thread/loaded/list:列出当前已加载到内存中的 thread id。thread/name/set:为已加载线程或已持久化的会话记录设置或更新用户可见名称;会发出thread/name/updated。thread/archive:把线程的日志文件移动到归档目录;成功时返回{},并发出thread/archived。thread/unsubscribe:取消当前连接对线程的 turn / item 事件订阅。如果这是最后一个订阅者,服务端会卸载该线程,并发出thread/closed。thread/unarchive:把归档线程的会话记录恢复回活跃会话目录;返回恢复后的thread,并发出thread/unarchived。thread/status/changed:某个已加载线程的运行时status发生变化时发出的通知。thread/compact/start:触发线程的会话历史压缩;立即返回{},进度通过turn/*与item/*通知流出。thread/shellCommand:对某条线程执行用户主动发起的 shell 命令。该命令会在沙箱外以完全访问权限运行,不继承线程原有沙箱策略。thread/backgroundTerminals/clean:停止与某条线程关联的所有后台终端(实验性;需要capabilities.experimentalApi)。thread/rollback:从内存上下文中移除最后 N 个 turn,并持久化一条 rollback 标记;返回更新后的thread。turn/start:向线程追加用户输入并开始 Codex 生成;响应中会返回初始turn,并继续流式发送事件。对于collaborationMode,settings.developer_instructions: null表示“使用该模式的内建指令”。turn/steer:向当前仍在进行中的 turn 追加用户输入;返回已接受的turnId。turn/interrupt:请求取消一个仍在执行中的 turn;成功时返回{},该 turn 最终会以status: "interrupted"结束。review/start:为某条线程启动 Codex 评审流程;会发出enteredReviewMode和exitedReviewModeitem。command/exec:在不创建 thread / turn 的前提下,在服务端沙箱中执行单条命令。command/exec/write:向正在运行的command/exec会话写入stdin字节,或关闭stdin。command/exec/resize:调整一个基于 PTY 的command/exec会话尺寸。command/exec/terminate:停止一个正在运行的command/exec会话。model/list:列出可用模型;可通过includeHidden: true包含带hidden: true的条目,并返回 effort 选项、可选upgrade和inputModalities。experimentalFeature/list:列出功能开关及其生命周期阶段元数据,并支持 cursor 分页。collaborationMode/list:列出协作模式预设(实验性,不分页)。skills/list:按一个或多个cwd列出可用技能(支持forceReload和可选的perCwdExtraUserRoots)。plugin/list:列出已发现的插件市场与插件状态,包括安装 / 认证策略元数据、市场错误、featured plugin id,以及仅开发环境可用的forceRemoteSync选项。plugin/read:按 marketplace path 和插件名读取单个插件的详细信息,包括其内置技能、应用和 MCP 服务名称。plugin/install:从某个插件市场安装插件。plugin/uninstall:卸载已安装的插件。app/list:分页列出可用 apps(connectors),并返回可访问性与启用状态等元数据。skills/config/write:按路径启用或禁用某个技能。mcpServer/oauth/login:为已配置的 MCP 服务启动 OAuth 登录;会返回授权 URL,并在完成后发出mcpServer/oauthLogin/completed。tool/requestUserInput:为工具调用向用户发起 1 到 3 个简短问题的交互请求(实验性);问题可通过isOther提供自由输入选项。config/mcpServer/reload:从磁盘重新加载 MCP 服务配置,并为已加载线程排队刷新。mcpServerStatus/list:列出 MCP 服务、工具、资源和认证状态(支持 cursor + limit 分页)。可通过detail: "full"获取完整数据,或用detail: "toolsAndAuthOnly"省略资源。mcpServer/resource/read:通过已初始化的 MCP 服务读取单个 MCP resource。windowsSandbox/setupStart:为elevated或unelevated模式启动 Windows 沙箱初始化;会快速返回,稍后发出windowsSandbox/setupCompleted。feedback/upload:提交反馈报告(分类、可选原因 / 日志 / 对话 id,以及可选extraLogFiles附件)。config/read:读取磁盘上的最终生效配置,结果已经过配置分层合并。externalAgentConfig/detect:检测可被迁移的外部智能体配置工件,支持includeHome和可选的cwds;每个检测结果都包含cwd(home 级配置时为null)。externalAgentConfig/import:通过显式传入migrationItems及其cwd(home 级为null),应用选中的外部智能体迁移项。config/value/write:把单个配置 key / value 写入用户磁盘上的config.toml。config/batchWrite:以原子方式把多项配置编辑写入用户磁盘上的config.toml。configRequirements/read:读取requirements.toml和 / 或 MDM 中的管理员要求,包括允许列表、固定的featureRequirements,以及驻留 / 网络要求(如果未配置则返回null)。fs/readFile、fs/writeFile、fs/createDirectory、fs/getMetadata、fs/readDirectory、fs/remove、fs/copy:通过 app-server v2 文件系统 API 对绝对路径执行文件操作。
模型
列出模型(model/list)
在渲染模型选择器、推理强度选择器或 personality 选择器前,先调用 model/list 获取当前可用模型和能力。
{ "method": "model/list", "id": 6, "params": { "limit": 20, "includeHidden": false } }
{ "id": 6, "result": {
"data": [{
"id": "gpt-5.4",
"model": "gpt-5.4",
"displayName": "GPT-5.4",
"hidden": false,
"defaultReasoningEffort": "medium",
"supportedReasoningEfforts": [{
"reasoningEffort": "low",
"description": "Lower latency"
}],
"inputModalities": ["text", "image"],
"supportsPersonality": true,
"isDefault": true
}],
"nextCursor": null
} }每条模型记录可能包含:
supportedReasoningEfforts:该模型支持的 effort 选项。defaultReasoningEffort:客户端可采用的建议默认 effort。upgrade:可选的推荐升级模型 id,适合在客户端里做迁移提示。upgradeInfo:可选的升级元数据,适合在客户端里做迁移提示。hidden:该模型是否在默认选择器列表中隐藏。inputModalities:模型支持的输入类型,例如text、image。supportsPersonality:模型是否支持/personality这类人格指令。isDefault:该模型是否是推荐默认值。
默认情况下,model/list 只返回适合在选择器中展示的模型;如果你想在客户端自行过滤,设置 includeHidden: true。
如果某个旧版模型目录没有 inputModalities,为兼容起见,应把它视为 ["text", "image"]。
列出实验性功能(experimentalFeature/list)
这个接口用于发现实验功能开关及其生命周期:
{ "method": "experimentalFeature/list", "id": 7, "params": { "limit": 20 } }
{ "id": 7, "result": {
"data": [{
"name": "unified_exec",
"stage": "beta",
"displayName": "Unified exec",
"description": "Use the unified PTY-backed execution tool.",
"announcement": "Beta rollout for improved command execution reliability.",
"enabled": false,
"defaultEnabled": false
}],
"nextCursor": null
} }stage 的可选值包括:beta、underDevelopment、stable、deprecated、removed。对于非 beta 的功能开关,displayName、description 和 announcement 可能为 null。
线程
thread/read会读取已保存线程,但不会自动订阅它;设置includeTurns可把 turns 一并返回。thread/list支持基于 cursor 的分页,以及modelProviders、sourceKinds、archived、cwd过滤。thread/loaded/list返回当前已加载到内存中的 thread id。thread/archive会把线程的持久化 JSONL 日志移动到 archived 目录。thread/unsubscribe会取消当前连接对已加载线程的订阅,并可能触发thread/closed。thread/unarchive会把归档线程的会话记录恢复到活跃会话目录。thread/compact/start会触发压缩,并立即返回{}。thread/rollback会从内存上下文中移除最后 N 个 turn,并在线程的持久化 JSONL 日志中写入 rollback 标记。
启动或恢复线程
当你需要开启一个全新的 Codex 对话时,调用 thread/start:
{ "method": "thread/start", "id": 10, "params": {
"model": "gpt-5.4",
"cwd": "/Users/me/project",
"approvalPolicy": "never",
"sandbox": "workspaceWrite",
"personality": "friendly",
"serviceName": "my_app_server_client"
} }
{ "id": 10, "result": {
"thread": {
"id": "thr_123",
"preview": "",
"ephemeral": false,
"modelProvider": "openai",
"createdAt": 1730910000
}
} }
{ "method": "thread/started", "params": { "thread": { "id": "thr_123" } } }serviceName 是可选的。当你希望让 app-server 用你的服务名为线程级指标打标签时,再设置它。
如果你想继续某条已保存会话,请使用之前记录下来的 thread.id 调用 thread/resume。它的响应结构与 thread/start 相同;你也可以继续传入 personality 这类与 thread/start 相同的配置覆盖项:
{ "method": "thread/resume", "id": 11, "params": {
"threadId": "thr_123",
"personality": "friendly"
} }
{ "id": 11, "result": { "thread": { "id": "thr_123", "name": "Bug bash notes", "ephemeral": false } } }仅仅执行 thread/resume 本身不会更新 thread.updatedAt,也不会修改持久化记录文件的时间戳;只有开始新的 turn 后,时间戳才会更新。
如果某个已启用的 MCP 服务同时被标记为 required,但它初始化失败,那么 thread/start 和 thread/resume 都会直接失败,而不会在缺少该服务的前提下继续运行。
thread/start 上的 dynamicTools 是实验性字段,需要 capabilities.experimentalApi = true。Codex 会把这些 dynamic tools 持久化到线程的会话记录元数据中;当你调用 thread/resume 且没有重新提供新的 dynamic tools 时,它们也会被恢复出来。
如果你用与原始会话记录中记录不同的模型去恢复线程,Codex 会发出一条警告,并在下一次 turn 中注入一次性模型切换说明。
如果你希望从某个已保存会话分叉出新线程,可对它的 thread.id 调用 thread/fork。这会创建一个新的 thread id,并为该新线程发出 thread/started 通知:
{ "method": "thread/fork", "id": 12, "params": { "threadId": "thr_123" } }
{ "id": 12, "result": { "thread": { "id": "thr_456" } } }
{ "method": "thread/started", "params": { "thread": { "id": "thr_456" } } }当用户可见标题已设置后,app-server 会在 thread/list、thread/read、thread/resume、thread/unarchive 和 thread/rollback 的响应中补全 thread.name。而 thread/start 与 thread/fork 在创建时,则可能暂时省略 name,或返回 null。
读取已保存线程(不恢复)
如果你只想读取已保存线程的数据,但不想恢复该线程,也不想订阅它的事件,请使用 thread/read。
includeTurns:为true时,响应会包含该线程的 turns;为false或省略时,只返回线程摘要。- 返回的
thread对象会带运行时status,取值可能是notLoaded、idle、systemError,或带activeFlags的active。
{ "method": "thread/read", "id": 19, "params": { "threadId": "thr_123", "includeTurns": true } }
{ "id": 19, "result": { "thread": { "id": "thr_123", "name": "Bug bash notes", "ephemeral": false, "status": { "type": "notLoaded" }, "turns": [] } } }与 thread/resume 不同,thread/read 不会把 thread 加载进内存,也不会触发 thread/started。
列出线程(支持分页与过滤)
thread/list 适合驱动历史会话列表界面。结果默认按 createdAt 倒序排列,并且会先应用过滤,再执行分页。
常用参数包括:
cursor:上一次响应返回的不透明字符串;第一页时可省略。limit:如果不设置,服务端会使用一个合理的默认分页大小。sortKey:created_at或updated_at。modelProviders:把结果限制为特定模型提供方;未设置、null或空数组表示不过滤。sourceKinds:把结果限制为特定 thread 来源。若省略或传[],服务端默认只返回交互式来源,也就是cli和vscode。archived:为true时只列出归档线程;为false或省略时,只列出未归档线程。cwd:只返回会话工作目录与该路径完全一致的线程。
sourceKinds 接受以下取值:
clivscodeexecappServersubAgentsubAgentReviewsubAgentCompactsubAgentThreadSpawnsubAgentOtherunknown
示例:
{ "method": "thread/list", "id": 20, "params": {
"cursor": null,
"limit": 25,
"sortKey": "created_at"
} }当 nextCursor 为 null 时,表示已经到达最后一页。
跟踪线程状态变化
只要某个已加载线程的运行时状态发生变化,服务端就会发出 thread/status/changed:
{
"method": "thread/status/changed",
"params": {
"threadId": "thr_123",
"status": { "type": "active", "activeFlags": ["waitingOnApproval"] }
}
}列出已加载线程
thread/loaded/list 返回当前已载入内存的 thread id 列表:
{ "method": "thread/loaded/list", "id": 21 }
{ "id": 21, "result": { "data": ["thr_123", "thr_456"] } }取消订阅已加载线程
thread/unsubscribe 会移除当前连接对某条 thread 的订阅。响应中的 status 可能是以下几种:
unsubscribed:当前连接原本订阅了这条 thread,现在已被移除。notSubscribed:当前连接原本就没有订阅这条 thread。notLoaded:这条 thread 当前并未加载。
如果这次移除的是最后一个订阅者,服务端会卸载该 thread,并发出一条状态切换到 notLoaded 的 thread/status/changed,随后再发出 thread/closed。
归档线程
thread/archive 会把持久化线程日志(磁盘上的 JSONL)移动到 archived sessions 目录。
{ "method": "thread/archive", "id": 22, "params": { "threadId": "thr_b" } }
{ "id": 22, "result": {} }
{ "method": "thread/archived", "params": { "threadId": "thr_b" } }除非你显式传入 archived: true,否则后续的 thread/list 不会再返回这些已归档线程。
取消归档线程
thread/unarchive 会把归档线程移回活跃会话目录:
{ "method": "thread/unarchive", "id": 24, "params": { "threadId": "thr_b" } }
{ "id": 24, "result": { "thread": { "id": "thr_b", "name": "Bug bash notes" } } }
{ "method": "thread/unarchived", "params": { "threadId": "thr_b" } }触发线程压缩
thread/compact/start 会触发一次手动历史压缩。请求会立即返回 {}。
app-server 会在同一个 threadId 上通过标准 turn/* 与 item/* 通知流出进度,其中包括一组 contextCompaction item 生命周期事件(先 item/started,再 item/completed)。
{ "method": "thread/compact/start", "id": 25, "params": { "threadId": "thr_b" } }
{ "id": 25, "result": {} }对线程执行 shell 命令
当你需要让某条线程执行用户主动发起的 shell 命令时,可以使用 thread/shellCommand。请求会立即返回 {},后续进度会继续通过标准 turn/* 与 item/* 通知流出。
这个接口会在沙箱外以完全访问权限执行命令,不会继承线程原有的沙箱策略,因此客户端只应把它暴露给明确由用户主动发起的命令。
如果该线程当前已有活跃 turn,命令会作为该 turn 的辅助动作运行,它的格式化输出也会注入这次 turn 的消息流;如果线程当前空闲,app-server 会为这条 shell 命令启动一个独立 turn。
{ "method": "thread/shellCommand", "id": 26, "params": { "threadId": "thr_b", "command": "git status --short" } }
{ "id": 26, "result": {} }清理后台终端
使用 thread/backgroundTerminals/clean 可以停止与某条线程关联的所有后台终端。这个方法属于实验性能力,需要 capabilities.experimentalApi = true。
{ "method": "thread/backgroundTerminals/clean", "id": 27, "params": { "threadId": "thr_b" } }
{ "id": 27, "result": {} }回滚最近的回合
thread/rollback 会从内存上下文中移除最后 numTurns 个 turn,并在持久化日志中写入 rollback 标记。返回的 thread 会反映 rollback 之后的状态。
{ "method": "thread/rollback", "id": 26, "params": { "threadId": "thr_b", "numTurns": 1 } }
{ "id": 26, "result": { "thread": { "id": "thr_b", "name": "Bug bash notes", "ephemeral": false } } }回合
turn/start 与 turn/steer 负责真正驱动对话。
input 支持三种常见 item:
{ "type": "text", "text": "Explain this diff" }{ "type": "image", "url": "https://.../design.png" }{ "type": "localImage", "path": "/tmp/screenshot.png" }
你可以按 turn 覆盖配置设置,例如 model、effort、personality、cwd、沙箱策略和 summary。只要在某个 turn 中显式指定,这些设置就会成为同一条 thread 后续 turn 的默认值。outputSchema 只对当前 turn 生效。
当 sandboxPolicy.type = "externalSandbox" 时,应将 networkAccess 设为 restricted 或 enabled;当类型是 workspaceWrite 时,networkAccess 仍然是布尔值。
对于 turn/start.collaborationMode,如果设置 settings.developer_instructions: null,表示“使用所选模式的内建 instructions”,而不是清空该模式的 instructions。
沙箱读权限(ReadOnlyAccess)
sandboxPolicy 支持显式声明读权限控制:
readOnly:可选access;默认是{ "type": "fullAccess" },也可以改成受限根目录。workspaceWrite:可选readOnlyAccess;默认也是{ "type": "fullAccess" },也可以改成受限根目录。
受限读权限的结构如下:
{
"type": "restricted",
"includePlatformDefaults": true,
"readableRoots": ["/Users/me/shared-read-only"]
}在 macOS 上,includePlatformDefaults: true 会为受限读会话追加一套精心挑选的平台默认 Seatbelt 策略。这样可以提升工具兼容性,同时避免宽泛地放开整个 /System。
示例:
{ "type": "readOnly", "access": { "type": "fullAccess" } }{
"type": "workspaceWrite",
"writableRoots": ["/Users/me/project"],
"readOnlyAccess": {
"type": "restricted",
"includePlatformDefaults": true,
"readableRoots": ["/Users/me/shared-read-only"]
},
"networkAccess": false
}启动回合
{ "method": "turn/start", "id": 30, "params": {
"threadId": "thr_123",
"input": [ { "type": "text", "text": "Run tests" } ],
"cwd": "/Users/me/project",
"approvalPolicy": "unlessTrusted",
"sandboxPolicy": {
"type": "workspaceWrite",
"writableRoots": ["/Users/me/project"],
"networkAccess": true
},
"model": "gpt-5.4",
"effort": "medium",
"summary": "concise",
"personality": "friendly",
"outputSchema": {
"type": "object",
"properties": { "answer": { "type": "string" } },
"required": ["answer"],
"additionalProperties": false
}
} }引导进行中的回合
使用 turn/steer 可以向当前仍在进行中的 turn 继续追加用户输入。
- 请求中必须带上
expectedTurnId,并且它必须与当前活动 turn 的 id 一致。 - 如果该线程当前没有活动中的 turn,请求会失败。
turn/steer不会发出新的turn/started通知。turn/steer不接受 turn 级覆盖项,例如model、cwd、sandboxPolicy或outputSchema。
{ "method": "turn/steer", "id": 32, "params": {
"threadId": "thr_123",
"input": [ { "type": "text", "text": "Actually focus on failing tests first." } ],
"expectedTurnId": "turn_456"
} }
{ "id": 32, "result": { "turnId": "turn_456" } }启动回合(调用技能)
如果你要显式调用某个技能,推荐同时使用两种信号:
- 在文本输入中写入
$<skill-name>。 - 同时附带一个
skill类型的输入项。
{ "method": "turn/start", "id": 33, "params": {
"threadId": "thr_123",
"input": [
{ "type": "text", "text": "$skill-creator Add a new skill for triaging flaky CI and include step-by-step usage." },
{ "type": "skill", "name": "skill-creator", "path": "/Users/me/.codex/skills/skill-creator/SKILL.md" }
]
} }中断回合
{ "method": "turn/interrupt", "id": 31, "params": { "threadId": "thr_123", "turnId": "turn_456" } }
{ "id": 31, "result": {} }成功后,该 turn 会以 status: "interrupted" 结束。
评审
review/start 会对某条 thread 启动 Codex 评审流程,并流式发送评审相关条目。
支持的评审目标包括:
uncommittedChangesbaseBranch(与某个分支做 diff)commit(评审某个特定 commit)custom(自由输入说明)
默认使用 delivery: "inline",即直接在现有 thread 中产出评审结果;如果你想把评审隔离到新线程,可使用 delivery: "detached"。
请求 / 响应示例:
{ "method": "review/start", "id": 40, "params": {
"threadId": "thr_123",
"delivery": "inline",
"target": { "type": "commit", "sha": "1234567deadbeef", "title": "Polish tui colors" }
} }
{ "id": 40, "result": {
"turn": {
"id": "turn_900",
"status": "inProgress",
"items": [
{ "type": "userMessage", "id": "turn_900", "content": [ { "type": "text", "text": "Review commit 1234567: Polish tui colors" } ] }
],
"error": null
},
"reviewThreadId": "thr_123"
} }如果要做 detached review,请使用 "delivery": "detached"。响应结构保持不变,但 reviewThreadId 会是新建评审线程的 id,因此会与原始 threadId 不同。服务端也会在开始流出评审 turn 之前,先为这个新线程发出一条 thread/started 通知。
Codex 会先流出常规的 turn/started 通知,随后再发出一条带有 enteredReviewMode item 的 item/started:
{
"method": "item/started",
"params": {
"item": {
"type": "enteredReviewMode",
"id": "turn_900",
"review": "current changes"
}
}
}当评审智能体 reviewer 完成后,服务端会发出 item/started 和 item/completed,其中都包含一个带最终评审文本的 exitedReviewMode item:
{
"method": "item/completed",
"params": {
"item": {
"type": "exitedReviewMode",
"id": "turn_900",
"review": "Looks solid overall..."
}
}
}客户端应使用这类通知来渲染评审智能体输出。
命令执行
command/exec 用于在不创建 thread 的情况下,直接在服务端沙箱内执行一条命令:
{ "method": "command/exec", "id": 50, "params": {
"command": ["ls", "-la"],
"cwd": "/Users/me/project",
"sandboxPolicy": { "type": "workspaceWrite" },
"timeoutMs": 10000
} }
{ "id": 50, "result": { "exitCode": 0, "stdout": "...", "stderr": "" } }如果你的服务端进程本身已经运行在外部沙箱内,可以使用 sandboxPolicy.type = "externalSandbox",让 Codex 跳过它自带的沙箱。对于外部沙箱,networkAccess 可设为 restricted(默认)或 enabled;对于 readOnly 和 workspaceWrite,则沿用上面介绍过的 access / readOnlyAccess 结构。
补充说明:
- 服务端会拒绝空的
command数组。 sandboxPolicy接受与turn/start相同的结构,例如dangerFullAccess、readOnly、workspaceWrite、externalSandbox。- 省略
timeoutMs时,会回退到服务端默认超时时间。 - 当
tty: true时,响应里会返回processId,并可继续配合command/exec/write、command/exec/resize、command/exec/terminate。 - 当
streamStdoutStderr: true时,stdout / stderr 会通过command/exec/outputDelta通知持续流出。
读取管理员要求(configRequirements/read)
使用 configRequirements/read 可以查看从 requirements.toml 和 / 或 MDM 加载的当前生效管理员要求。
{ "method": "configRequirements/read", "id": 52, "params": {} }
{ "id": 52, "result": {
"requirements": {
"allowedApprovalPolicies": ["onRequest", "unlessTrusted"],
"allowedSandboxModes": ["readOnly", "workspaceWrite"],
"featureRequirements": {
"personality": true,
"unified_exec": false
},
"network": {
"enabled": true,
"allowedDomains": ["api.openai.com"],
"allowUnixSockets": ["/tmp/example.sock"],
"dangerouslyAllowAllUnixSockets": false
}
}
} }如果当前没有配置任何 requirements,result.requirements 会是 null。有关支持的键和值,请点击查看 requirements.toml。
Windows 沙箱设置(windowsSandbox/setupStart)
自定义 Windows 客户端可以异步触发沙箱初始化,而不是在启动时同步阻塞:
{ "method": "windowsSandbox/setupStart", "id": 53, "params": { "mode": "elevated" } }
{ "id": 53, "result": { "started": true } }随后服务端会发出完成通知:
{
"method": "windowsSandbox/setupCompleted",
"params": { "mode": "elevated", "success": true, "error": null }
}模式值包括:
elevated:执行提升权限的 Windows 沙箱初始化路径。unelevated:执行旧版的初始化 / 预检路径。
事件
事件通知是服务端主动发出的流,用来描述线程生命周期、turn 生命周期,以及其中各个 item 的状态变化。在线程启动或恢复后,客户端应持续从当前传输流中读取 thread/started、thread/archived、thread/unarchived、thread/closed、thread/status/changed、turn/*、item/* 和 serverRequest/resolved 这些通知。
通知退订
客户端可以通过在 initialize.params.capabilities.optOutNotificationMethods 中传入精确的方法名,按连接关闭某些指定通知。
- 只支持精确匹配:例如
item/agentMessage/delta只会关闭这一种方法。 - 未知的方法名会被忽略。
- 适用于当前连接上的
thread/*、turn/*、item/*以及相关的 v2 通知。 - 不适用于请求、响应或错误。
模糊文件搜索事件(实验性)
模糊文件搜索会为每次查询发出:
fuzzyFileSearch/sessionUpdated:{ sessionId, query, files },表示当前查询的匹配结果。fuzzyFileSearch/sessionCompleted:{ sessionId },表示该查询的索引与匹配工作已完成。
Windows 沙箱设置事件
Windows 沙箱初始化相关的异步通知是:
windowsSandbox/setupCompleted:{ mode, success, error },表示某次windowsSandbox/setupStart请求已经完成。
回合事件
常见 turn 事件包括:
turn/started:{ turn },其中包含 turn id、空的items,以及status: "inProgress"。turn/completed:{ turn },其中turn.status可能是completed、interrupted或failed;失败时会带{ error: { message, codexErrorInfo?, additionalDetails? } }。turn/diff/updated:{ threadId, turnId, diff },表示该 turn 目前聚合出的最新 unified diff。turn/plan/updated:{ turnId, explanation?, plan },表示智能体新增或更新了计划;每个plan条目都是{ step, status },其中status取值为pending、inProgress或completed。thread/tokenUsage/updated:当前活动线程的用量更新。
当前版本里,turn/diff/updated 和 turn/plan/updated 即便在 item 事件正在流出时,items 数组通常也是空的。因此客户端应以 item/* 事件作为 turn items 的真实来源。
条目
ThreadItem 是 turn 响应和 item/* 通知里携带的联合类型。常见 item 包括:
userMessage:{ id, content },其中content是用户输入条目列表(text、image、localImage)。agentMessage:{ id, text, phase? },表示智能体当前累计输出;当存在phase时,使用 Responses API 的线协议取值(commentary、final_answer)。plan:{ id, text },表示 plan mode 下提出的计划文本。reasoning:{ id, summary, content },其中summary是流式推送的推理摘要,content是原始 reasoning block。commandExecution:{ id, command, cwd, status, commandActions, aggregatedOutput?, exitCode?, durationMs? }。fileChange:{ id, changes, status },表示建议的文件修改;changes中每项都是{ path, kind, diff }。mcpToolCall:{ id, server, tool, status, arguments, result?, error? }。dynamicToolCall:{ id, tool, arguments, status, contentItems?, success?, durationMs? },表示由客户端执行的动态工具调用。collabToolCall:{ id, tool, status, senderThreadId, receiverThreadId?, newThreadId?, prompt?, agentStatus? }。webSearch:{ id, query, action? },表示智能体发起的网页搜索。imageView:{ id, path },表示智能体调用了图像查看器。enteredReviewMode:{ id, review },表示评审模式已开始。exitedReviewMode:{ id, review },表示评审模式已结束,并附带最终评审文本。contextCompaction:{ id },表示 Codex 正在压缩会话历史。
对于 webSearch.action,其 type 可能是:
search(可带query?、queries?)openPage(可带url?)findInPage(可带url?、pattern?)
旧的 thread/compacted 通知已弃用;现在应使用 contextCompaction item。
所有 items 都遵循统一生命周期:
item/started:某个新工作单元开始时,发出完整item;其中的item.id会与增量事件中的itemId对应。item/completed:该工作单元结束时,发出最终item;应把它视为该 item 的权威最终状态。
条目增量
细粒度增量事件包括:
item/agentMessage/delta:流式追加智能体消息文本。item/plan/delta:流式追加计划文本;最终planitem 不一定与所有 delta 的简单拼接完全一致。item/reasoning/summaryTextDelta:流式追加可读的推理摘要文本;summaryIndex会在开启新摘要段时递增。item/reasoning/summaryPartAdded:标记一段新的推理摘要分段开始。item/reasoning/textDelta:流式追加原始推理文本(如果模型支持)。item/commandExecution/outputDelta:按顺序流出命令的 stdout / stderr 增量。item/fileChange/outputDelta:底层apply_patch工具调用的响应内容。
错误
如果某个 turn 失败,服务端会先发出一个 error 事件,内容为 { error: { message, codexErrorInfo?, additionalDetails? } },然后再以 status: "failed" 结束该 turn。
如果存在上游 HTTP 状态码,它会出现在 codexErrorInfo.httpStatusCode 中。
常见的 codexErrorInfo 取值包括:
ContextWindowExceededUsageLimitExceededHttpConnectionFailed(上游 4xx/5xx 错误)ResponseStreamConnectionFailedResponseStreamDisconnectedResponseTooManyFailedAttemptsBadRequest、Unauthorized、SandboxError、InternalServerError、Other
当存在上游 HTTP 状态码时,服务端会把它透传到对应 codexErrorInfo 变体上的 httpStatusCode 字段。
审批
根据用户当前的 Codex 设置,命令执行和文件变更都可能需要审批。app-server 会向客户端主动发起 JSON-RPC 请求,客户端再返回决策负载。
命令执行的决策包括:
accept、acceptForSession、decline、cancel,或{ "acceptWithExecpolicyAmendment": { "execpolicy_amendment": ["cmd", "..."] } }。文件变更的决策包括:
accept、acceptForSession、decline、cancel。这些请求都会带上
threadId和turnId,客户端应据此将界面状态绑定到当前会话。服务端会根据决策恢复或拒绝相应工作,并以
item/completed结束该 item。
命令执行审批
消息顺序如下:
item/started会发出待审批的commandExecutionitem,其中包含command、cwd等字段。item/commandExecution/requestApproval会包含itemId、threadId、turnId,以及可选的reason、command、cwd、commandActions、proposedExecpolicyAmendment、networkApprovalContext和availableDecisions。当initialize.params.capabilities.experimentalApi = true时,负载还可能包含实验性的additionalPermissions,用于描述按命令申请的额外 sandbox 权限。additionalPermissions中的任何文件系统路径在线路上传输时都为绝对路径。- 客户端返回上述某个命令执行决策。
serverRequest/resolved确认该待处理请求已得到答复或被清理。item/completed返回最终的commandExecutionitem,其status为completed | failed | declined。
当存在 networkApprovalContext 时,这个提示针对的是托管网络访问,而不是普通的 shell 命令审批。当前 v2 schema 会暴露目标 host 和 protocol;客户端应渲染网络专用提示,不要依赖 command 作为对用户有意义的 shell 命令预览。
Codex 会按目标(host、协议和端口)对并发网络审批提示进行分组。因此,app-server 可能只发送一个提示,就放行多个发往同一目标的排队请求;但同一主机上的不同端口仍会分别处理。
文件变更审批
消息顺序如下:
item/started会发出一个fileChangeitem,其中包含建议的changes,且status: "inProgress"。item/fileChange/requestApproval会包含itemId、threadId、turnId,以及可选的reason和grantRoot。- 客户端返回上述某个文件变更决策。
serverRequest/resolved确认该待处理请求已得到答复或被清理。item/completed返回最终的fileChangeitem,其status为completed | failed | declined。
tool/requestUserInput
当客户端响应 item/tool/requestUserInput 时,app-server 会发出带有 { threadId, requestId } 的 serverRequest/resolved。
如果在客户端答复之前,该待处理请求因 turn 启动、turn 完成或 turn 中断而被清理,服务端也会针对这次清理发出同样的通知。
动态工具调用(实验性)
thread/start 上的 dynamicTools 以及对应的 item/tool/call 请求 / 响应流程都属于实验性 API。
当某个动态工具在 turn 期间被调用时,app-server 会依次发出:
item/started,其中item.type = "dynamicToolCall"、status = "inProgress",并带上tool和arguments。item/tool/call,作为服务端发给客户端的请求。- 客户端响应负载,其中包含返回的 content items。
item/completed,其中item.type = "dynamicToolCall",并带上最终status,以及任何返回的contentItems或success值。
MCP 工具调用审批(Apps)
App(connector)工具调用也可能需要审批。当某个 app 工具调用带有副作用时,服务端可能通过 tool/requestUserInput 发起审批,并提供 Accept、Decline、Cancel 等选项。即使某个工具同时声明了权限更低的提示,只要它带有 destructive 工具注解,仍然一定会触发审批。
如果用户选择拒绝或取消,相关 mcpToolCall item 会以错误结束,而不会执行该工具。
技能
要调用某个技能,可在用户文本输入中加入 $<skill-name>。建议同时附带一个 skill 输入项,这样服务端会直接注入完整的技能说明,而不是依赖模型自行解析这个名称。
{
"method": "turn/start",
"id": 101,
"params": {
"threadId": "thread-1",
"input": [
{
"type": "text",
"text": "$skill-creator Add a new skill for triaging flaky CI."
},
{
"type": "skill",
"name": "skill-creator",
"path": "/Users/me/.codex/skills/skill-creator/SKILL.md"
}
]
}
}如果你省略 skill 输入项,模型仍会尝试解析 $skill-name 并查找对应技能,但这通常会增加延迟。
示例:
$skill-creator Add a new skill for triaging flaky CI and include step-by-step usage.使用 skills/list 可获取当前可用技能;你可以按 cwds 作用域查询,也可以通过 forceReload 强制刷新。perCwdExtraUserRoots 则允许你为特定 cwd 增加额外的用户级技能目录。对于那些 cwd 不在 cwds 列表中的条目,app-server 会直接忽略。
服务端会按 cwd 维护缓存;如果你希望强制重新扫描磁盘,设置 forceReload: true。当存在 SKILL.json 时,服务端还会读取其中的 interface 和 dependencies。
要按路径启用或禁用某个技能,可使用:
{
"method": "skills/config/write",
"id": 26,
"params": {
"path": "/Users/me/.codex/skills/skill-creator/SKILL.md",
"enabled": false
}
}Apps(连接器)
使用 app/list 可获取当前可用的 apps。在 CLI / TUI 中,面向用户的选择器是 /apps;在自定义客户端中,则应直接调用 app/list。每个条目都会同时包含 isAccessible(用户是否可访问)和 isEnabled(是否在 config.toml 中启用),这样客户端就能区分安装 / 访问状态与本地启用状态。app 条目还可能包含可选的 branding、appMetadata 和 labels 字段。
示例:
{ "method": "app/list", "id": 50, "params": {
"cursor": null,
"limit": 50,
"threadId": "thread-1",
"forceRefetch": false
} }
{ "id": 50, "result": {
"data": [
{
"id": "demo-app",
"name": "Demo App",
"description": "Example connector for documentation.",
"logoUrl": "https://example.com/demo-app.png",
"logoUrlDark": null,
"distributionChannel": null,
"branding": null,
"appMetadata": null,
"labels": null,
"installUrl": "https://chatgpt.com/apps/demo-app/demo-app",
"isAccessible": true,
"isEnabled": true
}
],
"nextCursor": null
} }如果你传入了 threadId,app 的功能开关判定(features.apps)会基于该 thread 的配置快照;如果不传,则使用最新的全局配置。
app/list 会在“用户可访问的 apps”和“目录来源的 apps”都加载完成后返回。若你想绕过缓存,可设置 forceRefetch: true;缓存只有在刷新成功时才会被替换。
服务端还会在任一来源(用户可访问的 apps 或目录来源的 apps)完成加载时发出 app/list/updated 通知。每条通知都会带上最新合并后的 app 列表。
{
"method": "app/list/updated",
"params": {
"data": [
{
"id": "demo-app",
"name": "Demo App",
"description": "Example connector for documentation.",
"logoUrl": "https://example.com/demo-app.png",
"logoUrlDark": null,
"distributionChannel": null,
"branding": null,
"appMetadata": null,
"labels": null,
"installUrl": "https://chatgpt.com/apps/demo-app/demo-app",
"isAccessible": true,
"isEnabled": true
}
]
}
}要调用某个 app,可在文本输入中插入 $<app-slug>,并添加一个 mention input item,其 path 为 app://<id>(推荐):
{
"method": "turn/start",
"id": 51,
"params": {
"threadId": "thread-1",
"input": [
{
"type": "text",
"text": "$demo-app Pull the latest updates from the team."
},
{
"type": "mention",
"name": "Demo App",
"path": "app://demo-app"
}
]
}
}用于 App 设置的 Config RPC 示例
可使用 config/read、config/value/write 和 config/batchWrite 查看或更新 config.toml 中的 app 控制项。
读取当前生效的 app 配置结构(包括 _default 和按工具覆盖的配置):
{ "method": "config/read", "id": 60, "params": { "includeLayers": false } }
{ "id": 60, "result": {
"config": {
"apps": {
"_default": {
"enabled": true,
"destructive_enabled": true,
"open_world_enabled": true
},
"google_drive": {
"enabled": true,
"destructive_enabled": false,
"default_tools_approval_mode": "prompt",
"tools": {
"files/delete": { "enabled": false, "approval_mode": "approve" }
}
}
}
}
} }更新单个 app 设置:
{
"method": "config/value/write",
"id": 61,
"params": {
"keyPath": "apps.google_drive.default_tools_approval_mode",
"value": "prompt",
"mergeStrategy": "replace"
}
}以原子方式一次应用多项 app 修改:
{
"method": "config/batchWrite",
"id": 62,
"params": {
"edits": [
{
"keyPath": "apps._default.destructive_enabled",
"value": false,
"mergeStrategy": "upsert"
},
{
"keyPath": "apps.google_drive.tools.files/delete.approval_mode",
"value": "approve",
"mergeStrategy": "upsert"
}
]
}
}检测并导入外部智能体配置
使用 externalAgentConfig/detect 可以发现可被迁移的外部智能体配置项,然后将选中的条目传给 externalAgentConfig/import。
探测示例:
{ "method": "externalAgentConfig/detect", "id": 63, "params": {
"includeHome": true,
"cwds": ["/Users/me/project"]
} }
{ "id": 63, "result": {
"items": [
{
"itemType": "AGENTS_MD",
"description": "Import /Users/me/project/CLAUDE.md to /Users/me/project/AGENTS.md.",
"cwd": "/Users/me/project"
},
{
"itemType": "SKILLS",
"description": "Copy skill folders from /Users/me/.claude/skills to /Users/me/.agents/skills.",
"cwd": null
}
]
} }导入示例:
{ "method": "externalAgentConfig/import", "id": 64, "params": {
"migrationItems": [
{
"itemType": "AGENTS_MD",
"description": "Import /Users/me/project/CLAUDE.md to /Users/me/project/AGENTS.md.",
"cwd": "/Users/me/project"
}
]
} }
{ "id": 64, "result": {} }支持的 itemType 取值包括 AGENTS_MD、CONFIG、SKILLS 和 MCP_SERVER_CONFIG。探测结果只会返回仍有待处理工作的条目。例如,如果 AGENTS.md 已存在且非空,就会跳过 AGENTS 迁移;技能导入也不会覆盖已存在的技能目录。
认证接口
JSON-RPC 的 auth/account 接口既包括请求 / 响应方法,也包括服务端主动发出的通知(不带 id)。你可以用这些接口来判断认证状态、启动或取消登录、执行登出,并查看 ChatGPT 速率限制。
认证模式
Codex 支持三种认证模式。当前激活的模式会出现在 account/updated.authMode 中,account/read 也会返回这一信息。
- API key(
apikey):调用方提供 OpenAI API key,Codex 保存它并用于发起 API 请求。 - ChatGPT 托管(
chatgpt):Codex 自行管理 ChatGPT OAuth 流程,持久化令牌并自动刷新。 - ChatGPT 外部令牌(
chatgptAuthTokens):宿主应用直接提供idToken和accessToken。Codex 会将这些令牌保存在内存中,并在需要时要求宿主应用刷新。
API 概览
account/read:读取当前账户信息;也可选择同时刷新令牌。account/login/start:启动登录流程(apiKey、chatgpt或chatgptAuthTokens)。account/login/completed(通知):一次登录尝试结束时发出,无论成功还是失败。account/login/cancel:按loginId取消尚未完成的 ChatGPT 登录流程。account/logout:执行登出;会触发account/updated。account/updated(通知):认证模式发生变化时发出;authMode可能是apikey、chatgpt、chatgptAuthTokens或null。account/chatgptAuthTokens/refresh(服务端请求):在遇到授权错误后,请求宿主应用提供新的外部管理 ChatGPT 令牌。account/rateLimits/read:读取 ChatGPT 速率限制。account/rateLimits/updated(通知):用户的 ChatGPT 速率限制发生变化时发出。mcpServer/oauthLogin/completed(通知):某次mcpServer/oauth/login流程完成后发出,负载中包含{ name, success, error? }。
1)检查认证状态
请求:
{ "method": "account/read", "id": 1, "params": { "refreshToken": false } }响应示例:
{ "id": 1, "result": { "account": null, "requiresOpenaiAuth": false } }{ "id": 1, "result": { "account": null, "requiresOpenaiAuth": true } }{
"id": 1,
"result": { "account": { "type": "apiKey" }, "requiresOpenaiAuth": true }
}{
"id": 1,
"result": {
"account": {
"type": "chatgpt",
"email": "user@example.com",
"planType": "pro"
},
"requiresOpenaiAuth": true
}
}字段说明:
refreshToken(boolean):设为true时,会在 ChatGPT 托管模式下强制刷新一次令牌。在外部令牌模式(chatgptAuthTokens)下,app-server 会忽略这个字段。requiresOpenaiAuth:反映当前激活的 provider 是否要求 OpenAI 凭据;如果为false,说明 Codex 可以在没有 OpenAI 凭据的情况下运行。
2)使用 API key 登录
发送:
{ "method": "account/login/start", "id": 2, "params": { "type": "apiKey", "apiKey": "sk-..." } }预期响应:
{ "id": 2, "result": { "type": "apiKey" } }通知:
{ "method": "account/login/completed", "params": { "loginId": null, "success": true, "error": null } }{ "method": "account/updated", "params": { "authMode": "apikey" } }
3)使用 ChatGPT 登录(浏览器流程)
启动:
{ "method": "account/login/start", "id": 3, "params": { "type": "chatgpt" } }{ "id": 3, "result": { "type": "chatgpt", "loginId": "<uuid>", "authUrl": "https://chatgpt.com/...&redirect_uri=http%3A%2F%2Flocalhost%3A<port>%2Fauth%2Fcallback" } }在浏览器中打开
authUrl;app-server 会托管本地回调地址。等待通知:
{ "method": "account/login/completed", "params": { "loginId": "<uuid>", "success": true, "error": null } }{ "method": "account/updated", "params": { "authMode": "chatgpt" } }
3b)使用外部管理的 ChatGPT 令牌登录(chatgptAuthTokens)
当宿主应用自行管理用户的 ChatGPT 认证生命周期,并直接提供令牌时,应使用这一模式。
发送:
{ "method": "account/login/start", "id": 7, "params": { "type": "chatgptAuthTokens", "idToken": "<jwt>", "accessToken": "<jwt>" } }预期响应:
{ "id": 7, "result": { "type": "chatgptAuthTokens" } }通知:
{ "method": "account/login/completed", "params": { "loginId": null, "success": true, "error": null } }{ "method": "account/updated", "params": { "authMode": "chatgptAuthTokens" } }
当服务端收到 401 Unauthorized 时,它可能向宿主应用请求刷新后的令牌:
{
"method": "account/chatgptAuthTokens/refresh",
"id": 8,
"params": { "reason": "unauthorized", "previousAccountId": "org-123" }
}
{ "id": 8, "result": { "idToken": "<jwt>", "accessToken": "<jwt>" } }在成功收到刷新响应后,服务端会重试原始请求。请求大约会在 10 秒后超时。
4)取消 ChatGPT 登录
{ "method": "account/login/cancel", "id": 4, "params": { "loginId": "<uuid>" } }
{ "method": "account/login/completed", "params": { "loginId": "<uuid>", "success": false, "error": "..." } }5)退出登录
{ "method": "account/logout", "id": 5 }
{ "id": 5, "result": {} }
{ "method": "account/updated", "params": { "authMode": null } }6)速率限制(ChatGPT)
{ "method": "account/rateLimits/read", "id": 6 }
{ "id": 6, "result": {
"rateLimits": {
"limitId": "codex",
"limitName": null,
"primary": { "usedPercent": 25, "windowDurationMins": 15, "resetsAt": 1730947200 },
"secondary": null
},
"rateLimitsByLimitId": {
"codex": {
"limitId": "codex",
"limitName": null,
"primary": { "usedPercent": 25, "windowDurationMins": 15, "resetsAt": 1730947200 },
"secondary": null
},
"codex_other": {
"limitId": "codex_other",
"limitName": "codex_other",
"primary": { "usedPercent": 42, "windowDurationMins": 60, "resetsAt": 1730950800 },
"secondary": null
}
}
} }
{ "method": "account/rateLimits/updated", "params": {
"rateLimits": {
"limitId": "codex",
"primary": { "usedPercent": 31, "windowDurationMins": 15, "resetsAt": 1730948100 }
}
} }字段说明:
rateLimits:向后兼容的单配额桶视图。rateLimitsByLimitId:如果存在,则表示多配额桶视图;按计量使用的limit_id为键,例如codex。limitId:计量配额桶的标识符。limitName:面向用户的可选配额桶标签。usedPercent:当前配额窗口内的使用比例。windowDurationMins:配额窗口长度。resetsAt:下一次重置的 Unix 时间戳(秒)。