会话管理深入探讨
OpenClaw 在以下领域端到端地管理会话:
- 会话路由 (入站消息如何映射到
sessionKey) - 会话存储 (
sessions.json) 及其跟踪内容 - 对话记录持久化 (
*.jsonl) 及其结构 - 对话记录清理 (运行前的提供商特定修正)
- 上下文限制 (上下文窗口与已跟踪令牌的对比)
- 压缩 (手动和自动压缩) 以及挂钩压缩前工作的位置
- 静默维护 (不应产生用户可见输出的内存写入)
如果您想先了解更高层次的概述,请从以下内容开始:
单一真实来源:Gateway(网关)
Section titled “单一真实来源:Gateway(网关)”OpenClaw 的设计围绕拥有会话状态的单个 Gateway(网关) 进程 展开。
- UI (macOS 应用、Web 控制界面、TUI) 应查询 Gateway(网关) 以获取会话列表和令牌计数。
- 在远程模式下,会话文件位于远程主机上;“检查本地 Mac 文件”无法反映 Gateway(网关) 正在使用的实际内容。
两个持久化层
Section titled “两个持久化层”OpenClaw 在两层中持久化会话:
-
会话存储 (
sessions.json)- 键/值映射:
sessionKey -> SessionEntry - 小型、可变、可安全编辑(或删除条目)
- 跟踪会话元数据(当前会话 ID、最后活动时间、切换开关、令牌计数器等)
- 键/值映射:
-
对话记录 (
<sessionId>.jsonl)- 具有树结构的仅追加对话记录(条目具有
id+parentId) - 存储实际的对话 + 工具调用 + 压缩摘要
- 用于为未来的轮次重建模型上下文
- 一旦活动对话记录超过检查点大小上限,就会跳过大型压缩前调试检查点,从而避免第二次巨大的
.checkpoint.*.jsonl复制。
- 具有树结构的仅追加对话记录(条目具有
Gateway(网关) 历史记录读取器应避免物化整个逐字稿,除非表面层明确需要任意历史访问。首页历史记录、嵌入式聊天历史记录、重启恢复以及令牌/使用情况检查使用有界的尾部读取。完整的逐字稿扫描通过异步逐字稿索引进行,该索引按文件路径加上 Gateway(网关)mtimeMs/size 进行缓存,并在并发读取器之间共享。
对于每个代理,在 Gateway(网关) 主机上:
- 存储:
~/.openclaw/agents/<agentId>/sessions/sessions.json - 逐字稿:
~/.openclaw/agents/<agentId>/sessions/<sessionId>.jsonl- Telegram 主题会话:Telegram
.../<sessionId>-topic-<threadId>.jsonl
- Telegram 主题会话:Telegram
OpenClaw 通过 OpenClawsrc/config/sessions.ts 解析这些路径。
存储维护和磁盘控制
Section titled “存储维护和磁盘控制”会话持久化具有针对 sessions.json、逐字稿构件和轨迹附属文件的自动维护控制 (session.maintenance):
mode:warn(默认)或enforcepruneAfter:陈旧条目的年龄截止(默认30d)maxEntries:限制sessions.json中的条目数(默认500)resetArchiveRetention:*.reset.<timestamp>逐字稿存档的保留期(默认:与pruneAfter相同;false禁用清理)maxDiskBytes:可选的会话目录预算highWaterBytes:清理后的可选目标(80%的默认maxDiskBytes)
常规 Gateway(网关) 写入流经每个存储区的会话写入器,该写入器在不获取运行时文件锁的情况下序列化进程内变更。热路径修补辅助程序在持有该写入器槽位时借用经过验证的可变缓存,因此不会针对每个元数据更新克隆或重新读取大型 Gateway(网关)sessions.json 文件。运行时代码应优先使用 updateSessionStore(...) 或 updateSessionStoreEntry(...)Gateway(网关);直接全量存储保存是兼容性和离线维护工具。当 Gateway(网关) 可访问时,非试运行 openclaw sessions cleanup 和 openclaw agents deleteGateway(网关) 将存储变更委托给 Gateway(网关),以便清理操作加入同一写入器队列;--store <path> 是用于直接文件维护的显式离线修复路径。maxEntriesGateway(网关) 清理仍针对生产级上限进行批处理,因此存储区可能会在下次高水位清理将其重写回之前短暂超过配置的上限。会话存储读取不会在 Gateway(网关) 启动期间修剪或限制条目;请使用写入或 openclaw sessions cleanup --enforce 进行清理。即使未配置磁盘预算,openclaw sessions cleanup --enforce 仍会立即应用配置的上限,并修剪旧的未引用的脚本、检查点和轨迹产物。
维护操作会保留持久的对外会话指针(例如群组会话和线程范围的聊天会话),但在超过配置的时长、计数或磁盘预算时,仍可移除针对 cron、hooks、heartbeat、ACP 和子代理的合成运行时条目。
OpenClaw 不再在 Gateway(网关) 写入期间创建自动 OpenClawsessions.json.bak.*Gateway(网关) 轮换备份。传统的 session.maintenance.rotateBytes 键将被忽略,并且 openclaw doctor --fix 会将其从旧版配置中移除。
记录变更会在记录文件上使用会话写入锁。获取锁最多等待 session.writeLock.acquireTimeoutMs,然后才会提示忙碌会话错误;默认值为 60000 毫秒。仅当在慢速机器上合法的准备工作、清理、压缩或记录镜像工作争用时间更长时,才调高此值。session.writeLock.staleMs 控制何时可以将现有锁视为过期并重新声明;默认值为 1800000 毫秒。session.writeLock.maxHoldMs 控制进程内看门狗释放阈值;默认值为 300000 毫秒。紧急环境覆盖变量为 OPENCLAW_SESSION_WRITE_LOCK_ACQUIRE_TIMEOUT_MS、OPENCLAW_SESSION_WRITE_LOCK_STALE_MS 和 OPENCLAW_SESSION_WRITE_LOCK_MAX_HOLD_MS。
磁盘预算清理的执行顺序 (mode: "enforce"):
- 首先移除最旧的已存档、孤立的 transcript 或孤立的轨迹工件。
- 如果仍高于目标,则驱逐最旧的会话条目及其 transcript/trajectory 文件。
- 继续直到使用量处于或低于
highWaterBytes。
在 mode: "warn"OpenClaw 中,OpenClaw 会报告潜在的驱逐操作,但不会更改存储/文件。
按需运行维护:
openclaw sessions cleanup --dry-runopenclaw sessions cleanup --enforceCron 会话和运行日志
Section titled “Cron 会话和运行日志”独立的 cron 运行也会创建会话条目/transcripts,并且它们具有专门的保留控制:
cron.sessionRetention(默认24h)从会话存储中修剪旧的独立 cron 运行会话(false禁用)。cron.runLog.maxBytes+cron.runLog.keepLines修剪~/.openclaw/cron/runs/<jobId>.jsonl文件(默认值:2_000_000字节和2000行)。
当 cron 强制创建一个新的独立运行会话时,它会在写入新行之前清理之前的 cron:<jobId> 会话条目。它会保留安全的偏好设置,例如 thinking/fast/verbose 设置、标签以及用户明确选择的模型/身份验证覆盖。它会丢弃环境对话上下文,例如渠道/组路由、发送或队列策略、提升、来源以及 ACP 运行时绑定,这样新的独立运行就无法从较旧的运行中继承过时的传递或运行时权限。
会话密钥 (sessionKey)
Section titled “会话密钥 (sessionKey)”sessionKey 标识您处于哪个对话存储桶(路由 + 隔离)。
常见模式:
- 主/直接聊天(每个代理):
agent:<agentId>:<mainKey>(默认main) - 组:
agent:<agentId>:<channel>:group:<id> - 房间/渠道(Discord/Slack):
agent:<agentId>:<channel>:channel:<id>或...:room:<id> - Cron:
cron:<job.id> - Webhook:
hook:<uuid>(除非被覆盖)
标准规则记录在 /concepts/会话 中。
会话 ID (sessionId)
Section titled “会话 ID (sessionId)”每个 sessionKey 指向一个当前的 sessionId(继续对话的记录文件)。
经验法则:
- 重置 (
/new,/reset) 会为该sessionKey创建一个新的sessionId。 - 每日重置(默认为网关主机当地时间凌晨 4:00)会在重置边界后的下一条消息上创建一个新的
sessionId。 - 空闲过期 (
session.reset.idleMinutes或旧版session.idleMinutes) 会在消息在空闲窗口后到达时创建一个新的sessionId。当同时配置了每日和空闲时,以先过期的为准。 - 系统事件(心跳、cron 唤醒、exec 通知、网关维护)可能会修改会话行,但不会延长每日/空闲重置的新鲜度。重置轮转会在构建新的提示之前,丢弃上一个会话排队的系统事件通知。
- 父级分支策略 在创建线程或子代理分支时使用 PI 的活动分支。如果该分支过大,OpenClaw 将使用隔离上下文启动子进程,而不是失败或继承无法使用的历史记录。大小策略是自动的;旧的 OpenClaw
session.parentForkMaxTokens配置已被openclaw doctor --fix移除。
实现细节:该决策发生在 src/auto-reply/reply/session.ts 的 initSessionState() 中。
会话存储架构 (sessions.json)
Section titled “会话存储架构 (sessions.json)”存储的值类型是 src/config/sessions.ts 中的 SessionEntry。
关键字段(非详尽列表):
sessionId:当前记录 ID(除非设置了sessionFile,否则文件名由此派生)sessionStartedAt:当前sessionId的开始时间戳;每日重置 新鲜度使用此值。旧行可能从 JSONL 会话头部派生它。lastInteractionAt:最后一次真实用户/渠道交互时间戳;空闲重置 新鲜度使用此值,因此心跳、cron 和 exec 事件不会保持会话 存活。没有此字段的旧行将回退到恢复的会话开始 时间以进行空闲新鲜度检查。updatedAt:最后一次存储行变更时间戳,用于列出、修剪和 记账。它不是每日/空闲重置新鲜度的权威依据。sessionFile:可选的显式记录路径覆盖chatType:direct | group | room(有助于 UI 和发送策略)provider、subject、room、space、displayName:用于群组/渠道标签的元数据- 切换开关:
thinkingLevel、verboseLevel、reasoningLevel、elevatedLevelsendPolicy(每个会话的覆盖)
- 模型选择:
providerOverride、modelOverride、authProfileOverride
- 令牌计数器(尽力而为 / 取决于提供商):
inputTokens、outputTokens、totalTokens、contextTokens
compactionCount:该会话键完成自动压缩的频率memoryFlushAt:上次预压缩内存刷新的时间戳memoryFlushCompactionCount:上次刷新运行时的压缩计数
该存储区可以安全编辑,但 Gateway(网关) 是权威来源:当会话运行时,它可能会重写或重新填充条目。
Transcript 结构 (*.jsonl)
Section titled “Transcript 结构 (*.jsonl)”Transcript 由 @earendil-works/pi-coding-agent 的 SessionManager 管理。
该文件是 JSONL:
- 第一行:会话标头 (
type: "session",包括id、cwd、timestamp、可选的parentSession) - 然后:带有
id+parentId(树)的会话条目
值得注意的条目类型:
message:用户/助手/工具结果 消息custom_message:扩展注入的、会进入模型上下文的消息(可对 UI 隐藏)custom:不进入模型上下文的扩展状态compaction:持久化的压缩摘要,包含firstKeptEntryId和tokensBeforebranch_summary:导航树分支时的持久化摘要
OpenClaw 故意不“修正”transcript;Gateway 使用 OpenClawGateway(网关)SessionManager 来读取/写入它们。
上下文窗口与跟踪的 Token
Section titled “上下文窗口与跟踪的 Token”有两个不同的概念很重要:
- 模型上下文窗口:每个模型的硬性上限(模型可见的 Token)
- 会话存储计数器:写入
sessions.json的滚动统计信息(用于 /status 和仪表板)
如果您正在调整限制:
- 上下文窗口来自模型目录(并且可以通过配置覆盖)。
- 存储中的
contextTokens是一个运行时估算/报告值;不要将其视为严格的保证。
欲了解更多,请参阅 /token-use。
压缩:它是什么
Section titled “压缩:它是什么”压缩将较旧的对话摘要为 transcript 中持久化的 compaction 条目,并保持最近的消息完整。
压缩后,未来的轮次将看到:
- 压缩摘要
firstKeptEntryId之后的消息
压缩是持久化的(与会话修剪不同)。请参阅 /concepts/会话-pruning。
压缩块边界和工具配对
Section titled “压缩块边界和工具配对”当 OpenClaw 将长 transcript 拆分为压缩块时,它会确保
助手工具调用与其匹配的 OpenClawtoolResult 条目保持配对。
- 如果令牌共享分割点位于工具调用及其结果之间,OpenClaw 会将边界移动到助手工具调用消息,而不是将这对消息分开。
- 如果尾部的工具结果块会导致分块超过目标大小,OpenClaw 会保留该挂起的工具块,并保持未摘要的尾部完整。
- 中止/错误的工具调用块不会保持挂起的分割状态。
当自动压缩发生时(Pi 运行时)
Section titled “当自动压缩发生时(Pi 运行时)”在嵌入式 Pi 代理中,自动压缩在以下两种情况下触发:
- 溢出恢复:模型返回上下文溢出错误
(
request_too_large,context length exceeded,input exceeds the maximum number of tokens,input token count exceeds the maximum number of input tokens,input is too long for the 模型,ollama error: context length exceeded, 以及类似的提供商特定变体) → 压缩 → 重试。 - 阈值维护:在一次成功的轮次之后,当满足以下条件时:
contextTokens > contextWindow - reserveTokens
其中:
contextWindow是模型的上下文窗口reserveTokens是为提示词和下一次模型输出保留的余量
这些是 Pi 运行时语义(OpenClaw 消费事件,但 Pi 决定何时进行压缩)。
当设置了 OpenClawagents.defaults.compaction.maxActiveTranscriptBytesOpenClaw 且活动记录文件达到该大小时,OpenClaw 也可以在开始下一次运行之前触发预检本地压缩。这是针对本地重新打开成本的文件大小保护,而非原始归档:OpenClaw 仍然会运行常规的语义压缩,并且它需要 truncateAfterCompaction,以便压缩后的摘要可以成为新的后续记录。
对于嵌入式 Pi 运行,agents.defaults.compaction.midTurnPrecheck.enabled: trueOpenClaw
增加了一个可选的工具循环保护。在追加工具结果之后和下一次模型调用之前,OpenClaw 使用在回合开始时使用的相同预检预算逻辑来估算提示词压力。如果上下文不再适合,该保护不会在 Pi 的 transformContext 钩子内部进行压缩。它会引发一个结构化的回合中预检信号,停止当前的提示词提交,并让外层运行循环使用现有的恢复路径:如果足够则截断过大的工具结果,或者触发配置的压缩模式并重试。该选项默认禁用,并且适用于 default 和 safeguard
压缩模式,包括提供商支持的安全保护压缩。
这与 maxActiveTranscriptBytes 无关:字节大小保护在回合打开之前运行,而回合中预检在追加新的工具结果后,于嵌入式 Pi 工具循环的稍后阶段运行。
压缩设置 (reserveTokens, keepRecentTokens)
Section titled “压缩设置 (reserveTokens, keepRecentTokens)”Pi 的压缩设置位于 Pi 设置中:
{ compaction: { enabled: true, reserveTokens: 16384, keepRecentTokens: 20000, },}OpenClaw 还为嵌入式运行强制执行一个安全下限:
- 如果
compaction.reserveTokens < reserveTokensFloorOpenClaw,OpenClaw 会将其提升。 - 默认下限为
20000个 token。 - 设置
agents.defaults.compaction.reserveTokensFloor: 0可禁用下限。 - 如果它已经更高,OpenClaw 将保持原样。
- 手动
/compact会遵守明确的agents.defaults.compaction.keepRecentTokens并保留 Pi 的最近尾部切断点。如果没有明确的保留预算, 手动压缩仍然是一个硬检查点,重建的上下文从 新摘要开始。 - 设置
agents.defaults.compaction.midTurnPrecheck.enabled: true可在新的工具结果之后、下一次模型调用之前运行 可选的工具循环预检查。这只是一个触发器;摘要生成仍使用配置的 压缩路径。它与maxActiveTranscriptBytes无关,后者是一个 回合开始时的活动记录字节数守卫。 - 将
agents.defaults.compaction.maxActiveTranscriptBytes设置为一个字节值或 字符串(例如"20mb"),以便在活动记录变大时, 在回合之前运行本地压缩。此守卫仅在同时启用了truncateAfterCompaction时才会激活。将其保留未设置或设置0可 禁用。 - 当启用
agents.defaults.compaction.truncateAfterCompactionOpenClaw 时, OpenClaw 会在压缩后将活动记录轮换为压缩后的继任者 JSONL。 旧的完整记录将被归档,并从压缩检查点进行链接,而不是就地重写。
原因:在压缩变得不可避免之前,为多轮次“内部维护”(如内存写入)留出足够的余量。
实现:ensurePiCompactionReserveTokens() 在 src/agents/pi-settings.ts 中
(从 src/agents/pi-embedded-runner.ts 调用)。
可插拔压缩提供商
Section titled “可插拔压缩提供商”插件可以通过插件 API 上的 registerCompactionProvider()API 注册压缩提供商。当 agents.defaults.compaction.provider 设置为注册的提供商 ID 时,安全防护扩展会将摘要生成委托给该提供商,而不是内置的 summarizeInStages 管道。
providerLLM:注册的压缩提供商插件的 ID。保留未设置以使用默认的 LLM 摘要生成。- 设置
provider会强制执行mode: "safeguard"。 - 提供商接收与内置路径相同的压缩指令和标识符保留策略。
- 保护措施仍会在提供商输出后保留最近轮次和拆分轮次的后缀上下文。
- 内置保护摘要会使用新消息重新提炼先前的摘要, 而不是逐字保留完整的先前摘要。
- 安全防护模式默认启用摘要质量审计;设置
qualityGuard.enabled: false可跳过输出格式错误时的重试行为。 - 如果提供商失败或返回空结果,OpenClaw 将自动回退到内置的 LLM 摘要。
- 中止/超时信号将被重新抛出(而不是被吞没),以尊重调用者的取消请求。
来源:src/plugins/compaction-provider.ts,src/agents/pi-hooks/compaction-safeguard.ts。
用户可见界面
Section titled “用户可见界面”您可以通过以下方式观察压缩和会话状态:
/status(在任何聊天会话中)openclaw statusCLI (CLI)openclaw sessions/sessions --json- Gateway(网关) 日志(Gateway(网关)
pnpm gateway:watch或openclaw logs --follow):embedded run auto-compaction start+complete - 详细模式:
🧹 Auto-compaction complete+ 压缩计数
静默维护(NO_REPLY)
Section titled “静默维护(NO_REPLY)”OpenClaw 支持用于后台任务的“静默”轮次,其中用户不应看到中间输出。
约定:
- 助手以其输出以确切的静默令牌
NO_REPLY/no_reply开头,以表示“不向用户传送回复”。 - OpenClaw 会在交付层中剥离/抑制它。
- 确切的静默令牌抑制不区分大小写,因此当整个负载仅为静默令牌时,
NO_REPLY和no_reply均有效。 - 这仅适用于真正的后台/无交付轮次;它不是普通可操作用户请求的捷径。
自 2026.1.10OpenClaw 起,当部分块以 NO_REPLY 开头时,OpenClaw 还会抑制 草稿/输入流,这样静默操作就不会在轮次中途泄漏部分输出。
预压缩“内存刷新”(已实现)
Section titled “预压缩“内存刷新”(已实现)”目标:在自动压缩发生之前,运行一个静默代理轮次,将持久状态写入磁盘(例如,代理工作区中的 memory/YYYY-MM-DD.md),以便压缩无法擦除关键上下文。
OpenClaw 使用 预阈值刷新 方法:
- 监控会话上下文的使用情况。
- 当它跨越一个“软阈值”(低于 Pi 的压缩阈值)时,向代理运行一个静默的“立即写入内存”指令。
- 使用确切的静默令牌
NO_REPLY/no_reply,以便用户 什么都看不到。
配置(agents.defaults.compaction.memoryFlush):
enabled(默认值:true)model(用于刷新轮次的可选精确提供商/模型覆盖,例如ollama/qwen3:8b)softThresholdTokens(默认值:4000)prompt(刷新轮次的用户消息)systemPrompt(为刷新轮次附加的额外系统提示)
注意:
- 默认提示/系统提示包含一个
NO_REPLY提示以抑制 传送。 - 当设置了
model时,刷新轮次将使用该模型,而不继承活动会话的回退链,因此仅限本地的维护不会静默回退到付费的对话模型。 - 刷新在每个压缩周期运行一次(在
sessions.json中跟踪)。 - 清理仅针对嵌入式 Pi 会话运行(CLI 后端会跳过它)。
- 当会话工作区为只读时(
workspaceAccess: "ro"或"none"),将跳过刷新。 - 有关工作区文件布局和写入模式,请参阅 Memory。
Pi 还在扩展 API 中暴露了 session_before_compactAPIOpenClawGateway(网关) 钩子,但目前 OpenClaw 的刷新逻辑位于 Gateway 端。
故障排查清单
Section titled “故障排查清单”- 会话密钥错误?首先查看 /concepts/会话 并确认
/status中的sessionKey。 - 存储与脚本不匹配?从 Gateway(网关)
openclaw status确认 Gateway 主机和存储路径。 - 压缩垃圾信息?请检查:
- 模型上下文窗口(太小)
- 压缩设置(对于模型窗口而言
reserveTokens太高可能会导致更早的压缩) - 工具结果膨胀:启用/调优会话修剪
- 静默轮次泄露?确认回复以
NO_REPLY开头(不区分大小写的精确令牌),并且您使用的是包含流抑制修复的构建版本。