9.6 KiB
9.6 KiB
实施进度
配合
DESIGN.md。本文件只记 phase 状态、决策偏差、文件量、下一步。
最后更新:2026-05-14(Step 6)
状态
| Phase | 标题 | 状态 | 备注 |
|---|---|---|---|
| 1-3 | 骨架 + Skill + run_python | ✅ | 三个 skill;CoreCoder 唯一匹配 edit;敏感 env 过滤 |
| 4 | 演化性能力 | 🟡 | Model Profile + Probing ✅;版本化 prompt 未做 |
| 5 | Eval Suite | ⏸ 不做 | dogfooding 替代,probe 覆盖健康检查 |
| 6 | 长任务工程化 | 🟡 | task + 恢复 ✅;双层记忆 ✅;context 压缩未做 |
| 7 | 打磨 | ❌ | Docker 沙盒 / 更多 skill |
| §7 SaaS | DESIGN §7 路线 | 🟡 | A 事件流化 ✅;B 完工(Step 1 基建 ✅;Step 2 Session ORM ✅;Step 3 TaskState ORM ✅;Step 4 task_dir 双形态 ✅;Step 6 no-subtask 校验 ✅;Step 5 migrate-from-fs 取消)。下一阶 C(Executor + sandbox)或 Phase G(Web UI)。 |
已完成关键能力
- Q1 → 05-06 / Phase 1-4:骨架 / 三 skill / run_python / Model Profile + Probing。ppt v3 加商务红 + apply_brand + Iconify;素材摄取改 markitdown CLI。
- 05-06 / Phase 6 部分:task + state.json + tokens 累计;CLI
tasks+ REPL/status /done /abandon /desc;移除 legacyworkspace/sessions/。 - 05-07 / TUI + task_dir:rich Markdown 渲染;spinner 显实时耗时 + 累计 token;system prompt 注入 task_dir 绝对路径,产物收敛
workspace/tasks/<id>/;.gitignore删 bandaid。 - 05-08 / REPL 切换 + 懒创建:
/resume [last|<id>];build_agent不预占文件;_cleanup_if_empty三条件守门。 - 05-09 → 05-10 / §7 草案 + 导出:DESIGN §7 初版(05-12 重写);
cli.py export <task_id>+core/export_docx.py。 - 05-11 / 原子写 + 双层记忆 + §7 A:
atomic_write_text接管 save;core/memory.py(core.md 入 prompt,extended/* 走索引);loop 事件流化(sink.emit)铺 SSE 路。 - 05-12 / §7 改写:platform/core 多租户方案废弃,改 user-direct(folder-centric、task/messages 入 PG、no-subtask、hard cascade)。
- 05-14 / §7.1 心智模型修正:
Folder-centric→ Task 一等公民 + Dir 文件副视图(双视图正交,dir 不是 task 父容器);task_dir 留空=一次性对话 / 指定=项目化二分语义入文。 - 05-14 / §7 B Step 1 基建:
core/storage/{engine,models}.pySQLAlchemy 2.x ORM(users/tasks/messages/runs/usage_events 5 表)+ alembic(初版 migration0001_initial_schema,GIN/复合索引)+cli db {upgrade,downgrade,current}子命令组 + 本地 sentinel user(00000000-...)+ZCBOT_DB_URL必填(未设给清晰报错,不引导 docker)。已在远端测试 PG 跑通db upgrade head。 - 05-14 / §7 B Step 2 Session ORM:
core/session.py重写,messages 走 PG(append-only,jsonb,idx 严格递增);system prompt 不入库(每次 build_agent 重建);Session.load(task_id, system_prompt=...)resume 接口;ensure_local_task_rowidempotent UPSERT(INSERT ... ON CONFLICT DO NOTHING)在首条非 system 消息前打底 tasks 行。task_id 切换为 UUID(原时间戳格式废弃,旧 workspace 不做兼容)。main.py/cli.py 适配:resolve_task_id(UUID 前缀解析)、_cleanup_if_empty双检查(DB messages + FS 产物)、_list_task_rows改读 PG。core/export_docx.py改从 PG 读 messages。端到端 build/append/resume/cleanup smoke 全绿。取消 Step 5 migrate-from-fs(用户决定不兼容旧 workspace)。 - 05-14 / §7 B Step 3 TaskState ORM:
core/task.py重写,TaskState dataclass 保留为内存 DTO 但落地走 PG ——save()调upsert_task(INSERT ON CONFLICT DO UPDATE,显式 setupdated_at=func.now()),load(task_id)走 SELECT;字段去掉cwd(改读 task_dir,§7 SaaS task_dir-as-identity)。state.json文件全面废除,task_dir 只承担 skill 产物。core/storage/utils.py加upsert_task/update_task工具。main.py::sync_task_tokens改update_task(tokens_p,tokens_c)单字段 UPDATE(ORM-level update 自带 onupdate=func.now())。core/session.py::Session.append的 ensure 调用补传mode/description/reasoning_effort,避免首次 INSERT 后 _list_task_rows 看到空 meta。cli.py全字段从 ORM Task 列读;_cleanup_if_empty去 state.json 特例(任何 FS 文件 / 子目录都算实质痕迹);/done /abandon /desc走 PG。core/export_docx.pymeta 改从TaskState.load(tid)读(asdict 拿到 dict),去 CWD 字段。端到端 smoke:storage UPSERT/UPDATE round-trip + build_agent 懒创建 + Session.append 自动 INSERT 完整 meta + sync_task_tokens 局部 UPDATE + task_state.save UPSERT 保留 task_dir/tokens + export → .docx 37KB 全绿。 - 05-14 / §7 B Step 4 task_dir 双形态:CLI
chat --task-dir <path>支持用户显式指定项目目录(§7.1 task-primary + dir 副视图心智模型落地)—— 留空走默认派生workspace/tasks/<uuid>/,显式走用户路径(绝对或相对 cwd,Path.resolve())。main.py::resolve_task_id增task_dir_arg;resume 时从 PGtasks.task_dir读(SELECT task_dir WHERE task_id=?),空则降级默认派生。新增is_managed_task_dir(td, ws)判断是否在workspace/tasks/<uuid>/模板下,作_cleanup_if_empty保护开关 —— 用户自指定的项目目录绝不 rmtree(可能含用户已有文件);DB 行该删还是删。core/export_docx.py::export_chat_to_docx重构:task_id 升一等参数(从task_dir.name提取改入参传入),task_dir 留空时自动从 PG 读,支持用户目录(非 UUID 命名)正常导出。cli/export与cli.py export子命令均改走_resolve_uuid_or_prefix+ task_id 直传。Smoke 4 路径全绿:default-derived(managed=True, cleanup rmtree)/ --task-dir(managed=False, FS preserved)/ resume reads DB / export 自动 PG 查路径。 - 05-14 / §7 B Step 6 no-subtask 校验:
core/storage/utils.py::check_no_subtask(task_dir, user_id=SENTINEL)—— 同 user 下查new LIKE existing||'/%' OR existing LIKE new||'/%'(task_dir != new过滤掉同 task_dir 同项目多对话场景)。冲突抛NoSubtaskError(ValueError子类),消息带冲突 task 的 UUID 前 8 位 + 它的 task_dir。分隔符容差:SQL 里replace(task_dir, :bs, '/')把存的 Windows\在比较前归一,新值也replace('\\', '/'),跨 OS / 历史数据混合分隔符不漏判;bs通过 bind 参数传(绕开 SQL 字符串转义陷阱)。空 / whitespacetask_dir直接 return(legacy / 未绑项目)。main.py::build_agent在resolve_task_id后、TaskState 构造前调,if not resume单层闸 —— resume 跳过(目录改名走未来 Folder API cascade,这里只拦新建)。cli.py三处 build_agent 调用现有 try/except 直接接住 NoSubtaskError 并友好打印。Smoke 全绿:同 dir 允许 / child 拒 / parent 拒 / sibling 允许 /proj_a_other不误中proj_a(因为用/%而非%)/ 空跳过 / Win\子目录拒 / 混合分隔符(\存 +/查)仍拒 / build_agent 端到端三分支(child raise / same pass / resume bypass)。
关键决策与偏差
| 项 | 决策 | 备注 |
|---|---|---|
| 工具基目录 | cwd(读)+ task_dir(写) | system prompt 同时注入两者绝对路径 |
| Workspace 布局 | tasks/<id>/ + memory/{core.md, extended/} |
memory 跨 task 共享 |
| Eval Suite | 不做 | 个人工具 dogfooding |
| 版本化 prompt | 直接 general_v1.md |
Windows 软链接麻烦,真要切再做 |
| run_python 沙盒 | subprocess + env 过滤 | Docker 在 §7 C 阶段 |
文件清单
core/capabilities.py 71
core/llm.py 89
core/loop.py 152 ← §7 A: sink.emit
core/sinks.py 101 ← §7 A
core/ui.py 38
core/probe.py 243
core/session.py 153 ← §7 B Step 2-3: ORM + ensure 补 meta
core/skills.py 81
core/task.py 82 ← §7 B Step 3: PG-backed TaskState,去 cwd
core/memory.py 76
core/export_docx.py 379 ← §7 B Step 2-4: task_id 升一等
core/storage/__init__.py 27 ← §7 B Step 1-3
core/storage/engine.py 80 ← §7 B Step 1
core/storage/models.py 124 ← §7 B Step 1
core/storage/utils.py 139 ← §7 B Step 3-6: +upsert_task/update_task/check_no_subtask
tools/base.py 34
tools/fs.py 182
tools/shell.py 94
tools/run_python.py 84
tools/skill_tool.py 45
main.py 277 ← §7 B Step 4-6: +is_managed_task_dir / task_dir_arg / no-subtask check
cli.py 538 ← §7 B Step 4: --task-dir / cleanup 保护用户目录
db/migrations/env.py 61 ← §7 B Step 1
db/migrations/versions/
0001_initial_schema.py 125 ← §7 B Step 1
─────────────────────────────────
Python 合计 ~3101 行
加 skills/ppt 脚本 ~600 行 + SKILL.md / references / config / prompts + alembic.ini,总仓库约 3500 行。
下一步候选(性价比排序)
- §7 Phase G Web UI 简洁版(~2-3 天)—— FastAPI + Jinja2 + HTMX + SSE,task list / chat / folder tree / 文件上传下载;依赖 D(HTTP /v1)的 SSE 端点,与 E 无强序。Phase G 也可先上 task list + chat(读 PG)再补 folder tree。
- §7 C Executor + sandbox(~2-3 天)——
run_python/shell→Executor.run(...);本地保留 subprocess executor,SaaS 走 docker;api_key_env→KeyProvider运行时注入。 - Phase 6 context 三层压缩(~1 天)—— 兜底,V4 长上下文一般用不到。
- Phase 7 更多 skill / 模型档案(持续)。
- Proposal mermaid 预渲染(~半天)—— ASCII 透传不够用时再上
mmdc。
§7 B 已完工(Step 1-4 + Step 6,Step 5 取消)。剩余路线:C(Executor)/ D(HTTP API)/ E(auth + UI)/ G(Web UI)/ F(deploy / billing)。