throughline 0.3.25 → 0.4.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -1,5 +1,27 @@
1
1
  # 引き継ぎ発火条件の絞り込み調査 & 実装計画
2
2
 
3
+ > **Status (2026-05-09 update): 本書は履歴扱い**
4
+ >
5
+ > 当時 (2026-04-18) は VSCode 拡張 2.1.112 で `/clear` 後も `source='startup'` に
6
+ > 潰される問題があり、バトン方式 (案 E) を採用した。その後 Claude Code 2.1.105
7
+ > (VSCode `/clear` not clearing context fix) と 2.1.126 (Windows env fix) で
8
+ > 段階的に修正され、**2.1.128 で `source='clear'` が reliable**になった。
9
+ >
10
+ > 現行仕様 (v0.4.1): **baton path (primary)** + auto path (fallback) の 2 経路。
11
+ > typed `/clear` / `/tl` はどちらも UserPromptSubmit hook で baton を書き、次
12
+ > SessionStart が確定的にそのセッションを引き継ぐ。auto path は VSCode 拡張
13
+ > メニュー由来 `/clear` のように UserPromptSubmit に届かない経路のための
14
+ > fallback で、`THROUGHLINE_DISABLE_AUTO_HANDOFF=1` で OFF にできる (typed
15
+ > `/clear` / `/tl` は env と無関係に引き続き発火する)。詳細は
16
+ > [THROUGHLINE_CLEAR_AUTO_HANDOFF_PLAN.md](THROUGHLINE_CLEAR_AUTO_HANDOFF_PLAN.md)。
17
+ >
18
+ > 本書は当時のバトン採用判断を残す履歴ドキュメント。「結局 baton primary に
19
+ > 戻った」という結末は皮肉だが、当時の判断は VSCode 拡張側 source バグへの
20
+ > 適切な対応だった。今回 baton primary を再採択した理由は (a) typed `/clear`
21
+ > が UserPromptSubmit に確実に届く、(b) multi-window で「最新更新セッション
22
+ > ≠ /clear したセッション」になる場合に heuristic が誤った前任を選ばない、
23
+ > の 2 点。
24
+
3
25
  ## Status (2026-04-18 更新)
4
26
 
5
27
  - **Phase 0 実機検証: 完了**
@@ -78,7 +78,8 @@ schema v4 で PostToolUse (`capture-tool`) は廃止、L2/L3 は Stop 内で一
78
78
  | schema v4 migration(bodies 追加、judgments DROP) | [src/db.mjs](../src/db.mjs) |
79
79
  | schema v5 migration(details に kind / source_id 追加、L3 分離書き込み対応) | [src/db.mjs](../src/db.mjs) |
80
80
  | schema v6 migration(handoff_batons テーブル追加、`/tl` バトン引き継ぎ方式) | [src/db.mjs](../src/db.mjs), [src/baton.mjs](../src/baton.mjs) |
81
- | schema v7 migration(`handoff_batons.memo_text` カラム追加、in-flight メモ保存) | [src/db.mjs](../src/db.mjs), [src/baton.mjs](../src/baton.mjs), [src/cli/save-inflight.mjs](../src/cli/save-inflight.mjs) |
81
+ | schema v7 migration(`handoff_batons.memo_text` カラム追加、in-flight メモ保存) | (v8 memo_text drop、save-inflight 削除済み) |
82
+ | schema v8 migration(`handoff_batons.memo_text` drop、`/clear` auto path 化、`save-inflight` / `/tl-trim` / `updateBatonMemo` 削除、注入を L1+L2+L3 refs のみに簡素化) | [src/db.mjs](../src/db.mjs), [src/session-start.mjs](../src/session-start.mjs), [src/resume-context.mjs](../src/resume-context.mjs), [docs/THROUGHLINE_CLEAR_AUTO_HANDOFF_PLAN.md](THROUGHLINE_CLEAR_AUTO_HANDOFF_PLAN.md) |
82
83
  | VSCode `.vscode/tasks.json` の自動プロビジョニング(token-monitor の folderOpen 自動起動) | [src/vscode-task.mjs](../src/vscode-task.mjs) |
83
84
  | Stop フック時の state.usage スナップショット(monitor の「止まって見える」問題の切り分け用) | [src/state-file.mjs](../src/state-file.mjs), [src/turn-processor.mjs](../src/turn-processor.mjs) |
84
85
  | `throughline doctor --session <id-prefix>` セッション診断(state/transcript ズレ検出) | [src/cli/doctor.mjs](../src/cli/doctor.mjs) |
@@ -133,6 +134,8 @@ schema v4 で PostToolUse (`capture-tool`) は廃止、L2/L3 は Stop 内で一
133
134
  | **npm 公開 (v0.3.25): Codex global Stop hook / skill install** | `throughline install` が Claude hooks / slash commands に加えて `~/.codex/hooks.json` に絶対 node + installed `bin/throughline.mjs codex-hook stop` を `async: false` で登録し、`~/.codex/config.toml` の `[features].codex_hooks = true` を有効化し、`~/.codex/skills/throughline` に `$throughline` skill を配置する。Codex App Server / VSCode host の PATH 差分で bare `throughline` が見えない可能性があるため、hook は Caveat と同じ絶対パス型に寄せる。既存 Caveat / Spotter などの Codex hooks は保持し、`throughline uninstall` は Throughline 管理の Codex hook / skill だけを削除する。既に bare command または `async: true` で登録済みの Throughline Codex Stop hook は次回 install で更新する。実環境では `codex exec --json` child thread `019dfd4f-93ff-7522-8f89-bd1e1996c8d7` が Stop hook で自然 capture され、`doctor --codex` の latest DB session が `codex:019dfd4f-93ff-7522-8f89-bd1e1996c8d7` に進むことを確認した。さらに絶対パス型へ更新後、child thread `019dfd5e-1248-7c11-8ddc-97e1b0701e10` でも latest DB session が `codex:019dfd5e-1248-7c11-8ddc-97e1b0701e10` に進むことを確認した。hook shape 変更後に新規開始した VSCode-origin thread `019dfd62-9a9d-7211-bf91-89d8e3fc908e` でも `doctor --codex` の current thread と latest DB session が一致し、自然 Stop hook capture を確認済み。hook shape 変更前から開いていた VSCode-origin parent thread は、変更後の自然 Stop smoke としては扱わない。Caveat 側にも `async: false` Stop hook が動く実測があるため、Codex 側は Caveat と同じ同期 hook 方針に寄せる。`codex-capture` / `codex-summarize` / `codex-resume --memo-stdin` は診断・明示操作 surface として維持し、model-visible smoke は明示 opt-in。2026-05-08 以降、Stop hook auto-refresh は verified usage 90% 以上で guarded rollback / inject を試行し、estimate usage では mutation しない |
134
135
  | **npm 公開 (v0.3.25): Codex-first roadmap** | [THROUGHLINE_CODEX_FIRST_ROADMAP.md](THROUGHLINE_CODEX_FIRST_ROADMAP.md) を追加。次フェーズは Codex primary 実用化、Codex Rewind 互換、Claude 側 finalization の順で進める。Codex primary の L2→L1 backend は Codex CLI を本線とし、`codex-sidecar` は Claude primary からの review / risk-check / second opinion / 互換 L2→L1 経路として整理する |
135
136
  | **npm 公開 (v0.3.25): npm docs packaging** | README から参照する `docs/` と `CHANGELOG.md` を npm `files` に追加。`docs/throughline-handoff-context.example.json` を含め、README の sidecar dry-run 例が tarball 内でも成立するようにする |
137
+ | **npm 公開 (v0.4.0): /clear auto-handoff + memo / save-inflight / /tl-trim retire** | 2026-05-08 Claude Code 2.1.128 で `source='clear'` が reliable になったため、`/clear` で自動引継ぎがデフォルト ON になる auto path を追加。`THROUGHLINE_DISABLE_AUTO_HANDOFF=1` で OFF にできる。`/tl` slash command は明示意思マーカーへ簡素化 (memo 4 項目入力廃止、`save-inflight` CLI 削除、`/tl-trim` slash command 廃止、`updateBatonMemo` 関数削除、`handoff_batons.memo_text` を schema v8 で drop)。注入は L1 + L2 + L3 references のみに簡素化し、memo / 中断直前 thinking セクションを削除。Codex 側 trim path は維持。詳細は [CHANGELOG.md](../CHANGELOG.md) と [THROUGHLINE_CLEAR_AUTO_HANDOFF_PLAN.md](THROUGHLINE_CLEAR_AUTO_HANDOFF_PLAN.md) |
138
+ | **npm 公開 (v0.4.1): typed `/clear` も baton を書く + 2 経路の優先順位入れ替え** | 2026-05-09 `/clear` を UserPromptSubmit hook で検出した時点で当該セッションの `session_id` を `handoff_batons` に書き込み、次 SessionStart が確定的にそのセッションを引き継ぐ。これで multi-window で「最新更新セッション ≠ /clear したセッション」になるシナリオで `findLatestClaudePredecessor` heuristic が誤った前任を選ぶ問題を解消。2 経路の優先順位を **baton path = primary、auto path = fallback** に変更 (auto path は VSCode 拡張メニュー由来など UserPromptSubmit に届かない経路のフォールバック)。`THROUGHLINE_DISABLE_AUTO_HANDOFF=1` は fallback path のみに作用するようになった (typed `/clear` / `/tl` は env と無関係に発火する)。あわせて `.vscode/tasks.json` を git 追跡から外し (gitignore)、`ensureMonitorTaskFile` が hook 発火ごとに絶対パスを書き換える挙動による別環境での dirty diff を解消。`src/prompt-submit.test.mjs` を新設し、`isClearCommand` / `isBatonCommand` 判定 14 件と subprocess+DB 実体テスト 3 件を追加。詳細は [CHANGELOG.md](../CHANGELOG.md) |
136
139
  | **グローバル E2E 検証** | 2026-04-17 別ディレクトリから `throughline doctor` 全緑を確認 |
137
140
 
138
141
  ### ❌ 未完タスク
@@ -0,0 +1,304 @@
1
+ # Throughline `/clear` 自動引継ぎ計画 (TODO 兼)
2
+
3
+ `/clear` をトリガーにした **自動かつ軽量な引継ぎ** を実現する計画。
4
+ 2026-05-08 セッションの議論と実機検証、外部仕様調査に基づく。
5
+ A 案 (= /clear で自動引継ぎ + /tl は逃げ道として残す + /tl-trim 廃止) **採択確定**。
6
+
7
+ > 過去の経緯 (なぜ `/tl` バトンを採用したか) は [INHERITANCE_ON_CLEAR_ONLY.md](INHERITANCE_ON_CLEAR_ONLY.md) を参照。
8
+ > 本書は **2026-05-08 時点の現状検証 + 新理想設計** を扱う。
9
+
10
+ > **2026-05-09 (v0.4.1) update**: 2 経路の優先順位を **入れ替えた**。
11
+ > baton path が **primary**、auto path は **fallback**。理由: typed `/clear`
12
+ > は UserPromptSubmit に届くので baton 書き込みで確定的に当該セッションを
13
+ > 指名できる。一方 VSCode 拡張のメニュー由来 `/clear` は UserPromptSubmit
14
+ > に届かない (= `findLatestClaudePredecessor` heuristic が誤った前任を選ぶ
15
+ > リスクあり) ため、auto path を残してフォールバック化した。typed `/clear`
16
+ > も UserPromptSubmit hook で baton を書く ([src/prompt-submit.mjs](../src/prompt-submit.mjs))。
17
+ > `THROUGHLINE_DISABLE_AUTO_HANDOFF=1` は **fallback path のみに作用** する
18
+ > ようになった (typed `/clear` / `/tl` には効かない)。
19
+
20
+ ---
21
+
22
+ ## 1. 確定した事実 (実機検証済み)
23
+
24
+ ### 1.1 `/clear` 後の SessionStart `source` は 2.1.128 で reliable
25
+
26
+ ```
27
+ inheritance-decision.log (2026-05-08 12:26 検証)
28
+ 12:26:08.481Z source="startup" session=05735717 ← 新 chat 開始
29
+ 12:26:52.257Z source="clear" session=b2addc4a ← /clear 後
30
+ ```
31
+
32
+ - Claude Code `2.1.128` (VSCode native extension, Linux/WSL2) で `/clear` 後の SessionStart は **確実に `source='clear'`** を payload に乗せる
33
+ - 過去の [GitHub issue #49937](https://github.com/anthropics/claude-code/issues/49937) (= VSCode 拡張で /clear 後も `source='startup'` になっていたバグ) は **解決済み**
34
+ - v2.1.105 (VSCode `/clear` not clearing conversation context fix) と v2.1.126 (Windows SessionStart hook env files apply) のリリースで段階的に修正された
35
+
36
+ ### 1.2 SessionStart `source` の 4 値 (公式 docs)
37
+
38
+ | 値 | 意味 |
39
+ |---|---|
40
+ | `startup` | 新 chat / VSCode 再起動 / 別 project / cold start |
41
+ | `resume` | `--resume` / `--continue` / `/resume` |
42
+ | `clear` | `/clear` 直後 |
43
+ | `compact` | 自動 / 手動 compaction 直後 |
44
+
45
+ source: [code.claude.com/docs/en/hooks](https://code.claude.com/docs/en/hooks)
46
+
47
+ ### 1.3 `/clear` 等価の user-defined slash command は **作れない**
48
+
49
+ 調査結果 (claude-code-guide Agent + bundled `claude-code` 検査):
50
+
51
+ - `.claude/commands/*.md` (skill markdown) は **prompt 拡張**であり、built-in `/clear` を programmatic に invoke できない
52
+ - "subcommand" や "exec built-in" 構文は無い
53
+ - `/clear` には built-in alias が存在しない
54
+ - CLI flag `claude clear` も無い
55
+ - cross-platform で「新ウィンドウ起動」する built-in も無い (`/branch` は full history copy で軽量化と矛盾、VSCode URL scheme は VSCode 専用)
56
+
57
+ → **「引継ぎたい /clear」と「reset したい /clear」を `source` 値で区別する手段は無い**
58
+
59
+ ### 1.4 `/rewind` は fork 動作 (caveat 記録済み)
60
+
61
+ - `/rewind` Continue 確認画面に "A new forked conversation will be created after rewinding" と明示
62
+ - 同 thread 内 rollback ではなく、新 fork session id を生成
63
+ - 詳細は caveat `claude-code/claude-code-rewind-fork-conversation-rollback-primitive` を参照
64
+ - 本計画では `/rewind` は **対象外**
65
+
66
+ ---
67
+
68
+ ## 2. 採用する理想設計
69
+
70
+ ### 2.1 引継ぎ発火条件 (2 経路、v0.4.1 で baton primary に変更)
71
+
72
+ | 経路 | 条件 | 起動 |
73
+ |---|---|---|
74
+ | **baton path (primary)** | `handoff_batons` テーブルに TTL (1 時間) 内 baton あり (= ユーザーが `/tl` または `/clear` を打った) | `source` 値関係なく確定的に引継ぎ |
75
+ | **auto path (fallback)** | baton 不在 + `source='clear'` + env `THROUGHLINE_DISABLE_AUTO_HANDOFF` が `'1'` でない | `findLatestClaudePredecessor` heuristic で引継ぎ |
76
+
77
+ 判定ロジック (擬似コード):
78
+
79
+ ```
80
+ on UserPromptSubmit(prompt, session_id, project_path):
81
+ if isBatonCommand(prompt) or isClearCommand(prompt):
82
+ writeBaton(project_path, session_id, now) // typed /clear / /tl が確定的に baton を書く
83
+
84
+ on SessionStart(source, session_id, project_path):
85
+ baton = consumeBaton(project_path) // atomic SELECT + DELETE, TTL 超過は sessionId=null で返る
86
+ if baton.sessionId:
87
+ inject(curated_memory_from(baton.sessionId)) // baton path (primary, env 関係なく発火)
88
+ return
89
+ if source == 'clear' and env.THROUGHLINE_DISABLE_AUTO_HANDOFF != '1':
90
+ predecessor = findLatestClaudePredecessor(project_path, session_id)
91
+ inject(curated_memory_from(predecessor)) // auto path (fallback)
92
+ return
93
+ // 何もしない
94
+ ```
95
+
96
+ `consumeBaton` が先発なので「両方同時成立」は構造上発生しない (= baton ありなら baton 経路、無ければ source 判定)。typed `/clear` も UserPromptSubmit hook で baton を書くため、通常はほぼ常に baton path が走る。auto path は VSCode 拡張のメニュー由来 `/clear` のように UserPromptSubmit に届かない経路のためのフォールバック。
97
+
98
+ ### 2.2 注入内容: L1 + L2 + L3 refs のみ (baton/auto どちらの経路でも同一)
99
+
100
+ 含める:
101
+ - ヘッダ + Reading Contract framing (= Codex 側 `renderCodexRolloutMemoryPreview` の写像)
102
+ - **L1 summaries** (古い turn の一行要約)
103
+ - **L2 bodies** (直近 20 turn の verbatim)
104
+ - **L3 references** (= `throughline detail <時刻>` の取り出しコマンド一覧、Codex 風の `- ${kind}: ${detailCommand}` フォーマット)
105
+ - Continuation Instruction (= 「これは過去ログではなく現在進行中の作業」と明示)
106
+
107
+ 含めない (= 削除):
108
+ - 中断直前の in-flight memo (memo セクション)
109
+ - 中断直前の thinking (extended thinking セクション)
110
+ - 既存の Claude 向け footer の冗長な使い方説明
111
+
112
+ 理由: L2 全文があれば最後の assistant turn 自体に「次に何をしようとしていたか」が含まれている。memo / thinking は redundant。
113
+
114
+ ### 2.3 `/tl` の役割: **残すが簡素化**
115
+
116
+ - `/tl` slash command 自体は **維持** (= 明示意思マーカー = baton path のトリガー)
117
+ - 簡素化:
118
+ - memo 4 項目入力要求を **削除** ([.claude/commands/tl.md](../.claude/commands/tl.md) を「baton 立てるだけ」の最小実装に)
119
+ - `save-inflight` CLI を **削除** (memo を baton.memo_text に保存する役目だった)
120
+ - `handoff_batons.memo_text` 列を **drop** (schema v8 migration)
121
+ - `src/baton.mjs` の `updateBatonMemo` 関数を **削除** (memo_text 列が drop されるため)
122
+ - `prompt-submit.mjs` の baton 書き込み path は **維持** (`UserPromptSubmit` hook で `/tl` 検出 + writeBaton)
123
+ - v0.4.1 で `/clear` も同じ hook で baton を書くように拡張 ([src/prompt-submit.mjs](../src/prompt-submit.mjs) `isClearCommand`)
124
+
125
+ ユーザーから見た `/tl` の使い方:
126
+ - typed `/clear` (デフォルト): `/clear` を打った時点で baton が書かれ、次セッションが確定的に引継ぐ。`/tl` を打つ必要なし
127
+ - VSCode メニュー `/clear` 経由: UserPromptSubmit に届かないので baton は書かれず、auto path (fallback) が `findLatestClaudePredecessor` で前任を選ぶ
128
+ - 非 `/clear` 境界 (新規 chat / VSCode 再起動): `/tl` で baton を立ててから新セッションを開く
129
+ - env で auto OFF: typed `/clear` / `/tl` は引き続き動く (env は fallback 専用)
130
+
131
+ ### 2.4 `/tl-trim` 廃止 (Codex 側を壊さない)
132
+
133
+ - 元機能: memo 入力 + dry-run preview 表示
134
+ - 新仕様で memo 廃止 + 軽量化方針 → 役割なし
135
+ - 削除対象:
136
+ - [.claude/commands/tl-trim.md](../.claude/commands/tl-trim.md) (slash command)
137
+ - [src/cli/trim.mjs](../src/cli/trim.mjs) の **Claude path 部分のみ** 削除 (`describeTrimHost('claude')` ブランチ、Claude 用 memory preview 経路など)
138
+ - 関連 test
139
+ - **維持** (Codex 側を壊さないため):
140
+ - [src/cli/trim.mjs](../src/cli/trim.mjs) の Codex path (`--host codex`, `--codex-thread-id`, `--preflight`, `--execute`, etc.) はすべて維持
141
+ - [src/trim-model.mjs](../src/trim-model.mjs) の `describeTrimHost('codex')` / `buildTrimPlan` の Codex 関連
142
+ - [src/codex-app-server.mjs](../src/codex-app-server.mjs), [src/codex-rollout-memory.mjs](../src/codex-rollout-memory.mjs), `codex-*` CLI 全般
143
+ - `bin/throughline.mjs` の `trim` dispatch (Codex 経路で必要)
144
+ - Codex skill ([codex/skills/throughline](../codex/skills/throughline)) の trim 機能 (= 機能自体は無変更、SKILL.md 内の `/tl-trim` 言及があれば 4 TODO で update)
145
+
146
+ ### 2.5 `THROUGHLINE_DISABLE_AUTO_HANDOFF` env var
147
+
148
+ - 値が `'1'` のとき auto path を skip
149
+ - それ以外の値、または未設定 → auto path 有効 (= デフォルト ON)
150
+ - 判定箇所: [src/session-start.mjs](../src/session-start.mjs)
151
+ - 設定方法: ユーザーが `.bashrc` / `.zshrc` / VSCode terminal env / `~/.claude/settings.json` の `env` セクションで設定
152
+
153
+ ### 2.6 トレードオフ (受容する)
154
+
155
+ - ⚠️ 「reset したい /clear」も auto path で引継ぎ発火する → 受容 (= 不要なら env で OFF)
156
+ - ⚠️ baton TTL は 1 時間 (= 既存仕様維持)
157
+
158
+ ---
159
+
160
+ ## 3. 確定した内部判断
161
+
162
+ ### 3.1 `/rewind` source 検証 — 不要
163
+
164
+ A 採択により本計画は `/rewind` を扱わない。`/rewind` 後の `source` 値は将来要件で再検討。
165
+
166
+ ### 3.2 `handoff_batons` テーブル — 残す + memo_text 列だけ drop
167
+
168
+ - table 自体は baton path で必要なので **維持**
169
+ - `memo_text TEXT` 列を schema v8 migration で drop
170
+ - 既存の memo データは廃棄 (受容)
171
+ - **SQLite DROP COLUMN 互換確認必須**: `ALTER TABLE ... DROP COLUMN` は SQLite 3.35.0+ で利用可。Node.js v22.5+ 同梱の SQLite バージョンで動作するか実装時に検証
172
+
173
+ ### 3.3 旧 `/tl` ユーザー移行 — `/tl` 自体は continue、memo 関連だけ breaking
174
+
175
+ - `/tl` slash command 自体は使い続けられる (簡素化されただけ)
176
+ - 廃止される機能: memo 4 項目入力、`save-inflight` CLI、`/tl-trim`
177
+ - breaking change として CHANGELOG に明示
178
+
179
+ ### 3.4 `source='compact'` 扱い — 引継ぎしない
180
+
181
+ auto-compaction は Claude Code 内部の context 圧縮で、conversation 連続性は host 側が担保している。Throughline 側で別途引継ぎを発火する必要なし。
182
+
183
+ ### 3.5 ログファイルの扱い
184
+
185
+ - `~/.throughline/logs/inflight-memo.log`: `save-inflight` CLI 削除で **新規書き込みなし**。既存ファイルは削除提案を README / CHANGELOG に書く (= 自動削除はしない、ユーザー手動)
186
+ - `~/.throughline/logs/inheritance-decision.log` 内の `baton_has_memo` フィールド: memo 廃止で意味を失うため、`logDecision()` から **削除**
187
+ - `~/.throughline/logs/baton-write.log`: 維持 (= `/tl` baton 書き込みログとして引き続き有用)
188
+
189
+ ---
190
+
191
+ ## 4. 実装 TODO (実装開始可)
192
+
193
+ 優先度順:
194
+
195
+ - [ ] **schema v8 migration** ([src/db.mjs](../src/db.mjs)): `ALTER TABLE handoff_batons DROP COLUMN memo_text`
196
+ - SQLite 3.35.0+ サポート確認 (Node.js v22.5+ 同梱版)
197
+ - 動かない場合は `CREATE TABLE` + `INSERT SELECT` + `DROP` の rebuild migration に切り替え
198
+ - [ ] **`src/baton.mjs`**:
199
+ - `consumeBaton` 戻り値から `memoText` プロパティを削除
200
+ - `updateBatonMemo` 関数を **削除**
201
+ - `BATON_TTL_MS`, `writeBaton`, `consumeBaton` は維持
202
+ - [x] **`src/handoff-record.mjs`**: **維持** (Codex 側 codex-handoff.mjs / codex-resume / codex-handoff-smoke 等が `memory.inflightMemo` / `memory.latestThinking` を参照しているため、削除すると Codex を壊す)。Claude 側で「使わない」のは resume-context.mjs 側で実現済み
203
+ - [ ] **`src/resume-context.mjs`**: 注入テキストを新仕様に書き換え:
204
+ - memo セクション削除
205
+ - thinking セクション削除
206
+ - L3 references 一覧追加 (Codex `renderCodexRolloutMemoryPreview` 形式)
207
+ - footer 簡素化 (Continuation Instruction だけ残す)
208
+ - [ ] **`src/session-start.mjs`** を 2.1 のロジックに改修:
209
+ - `consumeBaton` 先発 → `baton.sessionId` あれば inject
210
+ - 無ければ `source==='clear'` かつ env が `'1'` でない場合に inject
211
+ - それ以外は何もしない
212
+ - `logDecision()` から `baton_has_memo` フィールド削除
213
+ - [ ] **`src/cli/save-inflight.mjs`** 削除
214
+ - [ ] **`bin/throughline.mjs`** の `save-inflight` dispatch 削除 (`trim` dispatch は **維持**)
215
+ - [ ] **`src/prompt-submit.mjs`**: 維持 (baton 書き込み + ensureMonitorTaskFile)
216
+ - [ ] **[.claude/commands/tl.md](../.claude/commands/tl.md)**: memo 4 項目入力要求を削除、純粋に「baton 立てるだけ」の最小実装に書き換え
217
+ - [x] **`/tl-trim` 関連削除**:
218
+ - [.claude/commands/tl-trim.md](../.claude/commands/tl-trim.md) ファイル削除
219
+ - **`src/cli/trim.mjs` 自体は維持**: Codex 経路 (`--host codex`, `--preflight`, `--execute`, `--codex-app-server-bin` 等) と doctor `--trim --host claude` で使う `describeTrimHost('claude')` の dry-run 表示が依存しているため、コード削除はしない (= ユーザーが直接 `throughline trim --host claude --dry-run` を打つ余地は残す。実用は SessionStart 自動経路に置き換わる)
220
+ - [ ] **[src/cli/install.mjs](../src/cli/install.mjs)**: Throughline 管理 slash commands の copy 対象リストから `tl-trim.md` を除外。`tl.md` は維持。`src/cli/install.test.mjs` の関連 test も update
221
+ - [ ] **[bin/throughline.mjs](../bin/throughline.mjs) の `showHelp()` 文言 update**:
222
+ - `save-inflight` 関連 help 文言を削除
223
+ - `/tl-trim` / Claude trim 関連の help 文言を削除
224
+ - Codex trim 関連 (`trim --dry-run`, `--preflight`, `--execute --host codex` など) は **維持**
225
+ - `bin/throughline.mjs` の `save-inflight` dispatch case 削除 (上の TODO と重複するが help text だけ別作業)
226
+ - [ ] **[codex/skills/throughline/SKILL.md](../codex/skills/throughline/SKILL.md)**: `/tl-trim` への言及があれば削除し、`throughline trim --execute --host codex` 直接呼び出しに統一。Codex 側 trim 案内自体は維持
227
+ - [ ] **[.codex-sidecar.yml](../.codex-sidecar.yml)** 確認: `/tl-trim` / `save-inflight` 経路の参照があれば削除。無ければ no-op
228
+ - [ ] **テスト全部更新**:
229
+ - `src/baton.test.mjs` → memo 関連 test 削除、`updateBatonMemo` test 削除
230
+ - `src/session-merger.test.mjs` → source='clear' 自動経路の test 追加
231
+ - `src/resume-context.test.mjs` → memo/thinking 削除を反映
232
+ - `src/handoff-record.test.mjs` → projection 簡素化
233
+ - `src/hook-entrypoints.test.mjs` → save-inflight subprocess test ケース削除 (= 独立ファイルではなく本ファイル内の test)
234
+ - `src/turn-processor.test.mjs` → 既存維持
235
+ - `src/trim-cli.test.mjs` / `src/trim-model.test.mjs` → Claude path 関連テストのみ削除、Codex 経路テストは維持
236
+ - 新規: env var 判定 test、source='clear' auto path test
237
+ - [ ] **docs 更新**:
238
+ - [CLAUDE.md](../CLAUDE.md): 設計の核を書き換え (「`/tl` バトンのみ」→「`source='clear'` 自動 + `/tl` 逃げ道」)
239
+ - [README.md](../README.md): 以下範囲を update:
240
+ - Quick Start: 「`/clear` で自動引継ぎ」中心に書き直し
241
+ - 「Explicit handoff via `/tl`」セクション: `save-inflight` / memo 4 項目要求の記述を削除、`/tl` は逃げ道として簡素な記述に
242
+ - Troubleshoot / How it compares 等の他セクション: `/tl-trim`, `save-inflight`, `inflight-memo.log` への言及をすべて削除
243
+ - `THROUGHLINE_DISABLE_AUTO_HANDOFF` env var 紹介を新規追加
244
+ - 既存 `inflight-memo.log` ファイルは新版で書き込み停止することを README で告知 (= 手動削除提案)
245
+ - [CHANGELOG.md](../CHANGELOG.md): breaking change を明示 (memo 廃止、save-inflight 削除、/tl-trim 削除、`updateBatonMemo` 削除、baton_has_memo フィールド削除)
246
+ - [INHERITANCE_ON_CLEAR_ONLY.md](INHERITANCE_ON_CLEAR_ONLY.md): 「2026-04 段階の検証 → 2026-05 でバグ修正により案 A 成立、本書は履歴扱い」note 追加
247
+ - [PUBLIC_RELEASE_PLAN.md](PUBLIC_RELEASE_PLAN.md): version bump + breaking change 反映
248
+ - [ ] **package.json**: **0.4.0** に bump (semver minor、pre-1.0 の breaking)
249
+ - [ ] **caveat 記録**: 「`/clear` SessionStart `source` は 2.1.128 で reliable、過去 #49937 は fix 済み」を public で記録
250
+
251
+ ---
252
+
253
+ ## 5. 削減できるコード規模 (見積もり)
254
+
255
+ 廃止対象:
256
+ - `src/cli/save-inflight.mjs` (~80 行) → 削除
257
+ - `src/cli/trim.mjs` の Claude path 部分 (~30 行) → 削除 (Codex path は維持)
258
+ - [.claude/commands/tl-trim.md](../.claude/commands/tl-trim.md) (~40 行) → 削除
259
+ - `src/baton.mjs` の `updateBatonMemo` 関数 (~10 行) → 削除
260
+ - `handoff_batons.memo_text` 列 (schema migration、コードへの影響は consumeBaton 戻り値変更のみ)
261
+ - `src/hook-entrypoints.test.mjs` 内 save-inflight test ケース (~30 行) → 削除
262
+ - `resume-context.mjs` の memo/thinking セクション (~30 行) → 削除
263
+ - `handoff-record.mjs` の memo/thinking projection (~50 行) → 削除
264
+ - [.claude/commands/tl.md](../.claude/commands/tl.md) の memo 4 項目要求 (~20 行) → 削除
265
+
266
+ 代わりに追加:
267
+ - `session-start.mjs` の env / source 判定 (~15 行)
268
+ - `resume-context.mjs` の L3 refs framing (~30 行)
269
+
270
+ 純減 ~245 行。
271
+
272
+ ---
273
+
274
+ ## 6. Codex 側との整合 (壊さない)
275
+
276
+ Codex 側 v0.3.25 の以下は本計画で **完全に無変更**:
277
+
278
+ - `codex-capture` / `codex-summarize` / `codex-resume` (Codex primary L1/L2/L3 path)
279
+ - `codex-resume --format handoff` (新規 Codex thread 用 prompt)
280
+ - `trim --execute --host codex` / `--preflight --host codex` (app-server `thread/rollback` + `thread/inject_items`)
281
+ - Codex Stop hook 90% auto-refresh
282
+ - restore-safety / host primitive audit diagnostics
283
+ - Codex skill ([codex/skills/throughline](../codex/skills/throughline)) の trim 機能 (= 機能自体は無変更、SKILL.md 内の `/tl-trim` 言及があれば 4 TODO で update)
284
+ - [src/codex-app-server.mjs](../src/codex-app-server.mjs), [src/codex-rollout-memory.mjs](../src/codex-rollout-memory.mjs)
285
+ - [src/trim-model.mjs](../src/trim-model.mjs) の Codex 関連 (`describeTrimHost('codex')`, `buildTrimPlan` の Codex source path)
286
+
287
+ `codex-resume --memo-stdin` は引き続きユーザーが stdin で memo を流す経路。Throughline DB の baton.memo_text には依存していない (= 列削除の影響なし)。
288
+
289
+ `/tl-trim` 削除に伴って Codex 経由でも slash command としての `/tl-trim` は使えなくなる。Codex 用 trim は `throughline trim --execute --host codex` を **CLI 直接呼ぶ**運用に統一 (Codex skill SKILL.md がそれを案内する)。
290
+
291
+ ---
292
+
293
+ ## 7. 進め方
294
+
295
+ 1. **本計画 確定** (ユーザー A 採択済み、本書 update により方針固定)
296
+ 2. **実装** (上記 TODO 順)
297
+ 3. **テスト + 実機 smoke + commit**
298
+ 4. **publish** (npm 0.4.0 として release)
299
+
300
+ 実機 smoke 手順:
301
+ - 自動引継ぎ ON (デフォルト): /clear → 新セッションで curated memory 注入を確認
302
+ - 自動引継ぎ OFF: env を立てて /clear → 注入されないことを確認
303
+ - baton path: `/tl` を打って新 chat タブで開く → baton 経由で注入を確認
304
+ - Codex 側 regression: `npm test` で既存 Codex test がすべて pass することを確認、`throughline trim --execute --host codex` の CLI 動作も維持
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "throughline",
3
- "version": "0.3.25",
3
+ "version": "0.4.1",
4
4
  "type": "module",
5
5
  "description": "Claude Code hooks plugin for structured context compression (/clear-safe persistent memory)",
6
6
  "keywords": [
package/src/baton.mjs CHANGED
@@ -1,19 +1,20 @@
1
1
  /**
2
2
  * baton.mjs — 引き継ぎバトン管理
3
3
  *
4
- * バトン方式の設計 (docs/INHERITANCE_ON_CLEAR_ONLY.md):
5
- * - ユーザーが旧セッションで /tl スラッシュコマンドを打つ UserPromptSubmit hook が
4
+ * バトン方式の設計 (docs/THROUGHLINE_CLEAR_AUTO_HANDOFF_PLAN.md):
5
+ * - 新仕様では `/clear` 自動引継ぎがデフォルト ON。バトンは「/clear 自動引継ぎを
6
+ * 使わずに明示的に引き継ぎたい」ユーザーのための逃げ道。
7
+ * - ユーザーが旧セッションで `/tl` スラッシュコマンドを打つ → UserPromptSubmit hook が
6
8
  * baton テーブルに (project_path, session_id, created_at) を INSERT OR REPLACE
7
- * - 新セッションの SessionStart hook が baton を消費:
8
- * TTL 1 時間以内 かつ session_id が自分自身でない → 前任として merge
9
- * 期限切れ or 自己指名 破棄
10
- * - 消費は atomic (BEGIN IMMEDIATE トランザクション内で SELECT + DELETE)
9
+ * - 新セッションの SessionStart hook が baton を atomic に消費 (BEGIN IMMEDIATE 内で
10
+ * SELECT + DELETE)。TTL 1 時間以内なら前任として merge、超過は破棄。
11
+ * - 注入する curated memory は L1 + L2 + L3 refs のみ (memo / thinking なし)。
11
12
  *
12
- * なぜバトン方式か:
13
- * - VSCode 拡張では SessionStart payload の source が /clear 後も "startup" に潰される
14
- * ため source 値だけで /clear を識別できない (GitHub issue #49937)
15
- * - 時間差ヒューリスティック (案 D) は誤爆の可能性があり、ユーザー明示の意思表示を
16
- * 引き継ぎ発火の唯一の条件とする方が決定論的
13
+ * 履歴: もともと VSCode 拡張で SessionStart payload の source が /clear 後も
14
+ * "startup" に潰される問題 (#49937) に対する明示意思マーカーとして導入。
15
+ * 2026-05-08 時点で Claude Code 2.1.128 で source='clear' reliable
16
+ * なったため auto path 中心の設計に変わったが、明示意思の signal として
17
+ * baton 仕組み自体は残す。詳細は docs/THROUGHLINE_CLEAR_AUTO_HANDOFF_PLAN.md。
17
18
  */
18
19
 
19
20
  /**
@@ -24,9 +25,7 @@ export const BATON_TTL_MS = 60 * 60 * 1000; // 1 時間
24
25
 
25
26
  /**
26
27
  * 現在セッション (= /tl を発動したセッション) を次回 SessionStart で merge 対象に指名する。
27
- * 同 project_path の既存バトンがあれば session_id / created_at のみ上書き。
28
- * v7 で追加された memo_text は保持する(連続した /tl → save-inflight の順番で
29
- * 呼ばれた場合に、再度 /tl を打った時点で古い memo が消えないようにする)。
28
+ * 同 project_path の既存バトンがあれば session_id / created_at を上書き。
30
29
  *
31
30
  * @param {import('node:sqlite').DatabaseSync} db
32
31
  * @param {{ projectPath: string, sessionId: string, now?: number }} params
@@ -41,43 +40,17 @@ export function writeBaton(db, { projectPath, sessionId, now = Date.now() }) {
41
40
  ).run(projectPath, sessionId, now);
42
41
  }
43
42
 
44
- /**
45
- * 既存バトンの memo_text を更新する。バトンが存在しない場合は NOOP。
46
- * /tl 発動後、現行セッションの Claude が `throughline save-inflight` CLI 経由で
47
- * 呼び出す。memo_text は Markdown 形式の「次の一手 / 現在の方針 / 未解決 /
48
- * 進行中 TODO」をまとめたテキスト。
49
- *
50
- * Windows 互換: ドライブレター(`C:` / `c:`)やパス区切りの差異で
51
- * /tl 書き込み時と save-inflight 呼び出し時の project_path が一致しない
52
- * ケースがあるため、SQLite の COLLATE NOCASE で大小無視で照合する。
53
- *
54
- * @param {import('node:sqlite').DatabaseSync} db
55
- * @param {{ projectPath: string, memoText: string, now?: number }} params
56
- * @returns {{ updated: boolean }}
57
- */
58
- export function updateBatonMemo(db, { projectPath, memoText }) {
59
- const result = db
60
- .prepare(
61
- `UPDATE handoff_batons SET memo_text = ? WHERE project_path = ? COLLATE NOCASE`,
62
- )
63
- .run(memoText, projectPath);
64
- return { updated: (result.changes ?? 0) > 0 };
65
- }
66
-
67
43
  /**
68
44
  * 同 project_path のバトンを読み出して削除する (atomic)。
69
45
  *
70
46
  * 戻り値:
71
- * - { sessionId, ageMs, memoText } : バトン存在 かつ TTL 以内
72
- * - { sessionId: null, skipReason: 'expired', ageMs } : TTL 超過で破棄
47
+ * - { sessionId, ageMs } : バトン存在 かつ TTL 以内
48
+ * - { sessionId: null, skipReason: 'expired', ageMs } : TTL 超過で破棄
73
49
  * - { sessionId: null, skipReason: 'missing' } : バトン無し
74
50
  *
75
- * memoText は /tl 後に save-inflight で書き込まれた in-flight メモ。
76
- * 未保存なら null。
77
- *
78
51
  * @param {import('node:sqlite').DatabaseSync} db
79
52
  * @param {{ projectPath: string, now?: number, ttlMs?: number }} params
80
- * @returns {{ sessionId: string | null, ageMs?: number, memoText?: string | null, skipReason?: 'expired' | 'missing' }}
53
+ * @returns {{ sessionId: string | null, ageMs?: number, skipReason?: 'expired' | 'missing' }}
81
54
  */
82
55
  export function consumeBaton(db, { projectPath, now = Date.now(), ttlMs = BATON_TTL_MS }) {
83
56
  db.exec('BEGIN IMMEDIATE');
@@ -85,7 +58,7 @@ export function consumeBaton(db, { projectPath, now = Date.now(), ttlMs = BATON_
85
58
  // Windows 互換: ドライブレターの大小差を吸収するため COLLATE NOCASE
86
59
  const row = db
87
60
  .prepare(
88
- `SELECT session_id, created_at, memo_text FROM handoff_batons WHERE project_path = ? COLLATE NOCASE`,
61
+ `SELECT session_id, created_at FROM handoff_batons WHERE project_path = ? COLLATE NOCASE`,
89
62
  )
90
63
  .get(projectPath);
91
64
 
@@ -108,7 +81,6 @@ export function consumeBaton(db, { projectPath, now = Date.now(), ttlMs = BATON_
108
81
  return {
109
82
  sessionId: row.session_id,
110
83
  ageMs,
111
- memoText: row.memo_text ?? null,
112
84
  };
113
85
  } catch (err) {
114
86
  try {
@@ -1,7 +1,7 @@
1
1
  import { test } from 'node:test';
2
2
  import assert from 'node:assert/strict';
3
3
  import { DatabaseSync } from 'node:sqlite';
4
- import { writeBaton, consumeBaton, updateBatonMemo, BATON_TTL_MS } from './baton.mjs';
4
+ import { writeBaton, consumeBaton, BATON_TTL_MS } from './baton.mjs';
5
5
 
6
6
  function makeDb() {
7
7
  const db = new DatabaseSync(':memory:');
@@ -9,8 +9,7 @@ function makeDb() {
9
9
  CREATE TABLE handoff_batons (
10
10
  project_path TEXT PRIMARY KEY,
11
11
  session_id TEXT NOT NULL,
12
- created_at INTEGER NOT NULL,
13
- memo_text TEXT
12
+ created_at INTEGER NOT NULL
14
13
  );
15
14
  `);
16
15
  return db;
@@ -99,46 +98,10 @@ test('consumeBaton: scopes per project_path (does not cross-consume)', () => {
99
98
  assert.equal(rows.length, 1);
100
99
  });
101
100
 
102
- test('updateBatonMemo: writes memo_text into existing baton', () => {
103
- const db = makeDb();
104
- writeBaton(db, { projectPath: '/proj', sessionId: 'S1', now: 1000 });
105
- const result = updateBatonMemo(db, { projectPath: '/proj', memoText: '次の一手: X' });
106
- assert.equal(result.updated, true);
107
- const row = db.prepare('SELECT memo_text FROM handoff_batons').get();
108
- assert.equal(row.memo_text, '次の一手: X');
109
- });
110
-
111
- test('updateBatonMemo: is a NOOP when no baton exists (no throw)', () => {
112
- const db = makeDb();
113
- const result = updateBatonMemo(db, { projectPath: '/missing', memoText: 'hello' });
114
- assert.equal(result.updated, false);
115
- });
116
-
117
- test('writeBaton: preserves memo_text when same project_path is re-batoned', () => {
118
- const db = makeDb();
119
- writeBaton(db, { projectPath: '/proj', sessionId: 'S1', now: 1000 });
120
- updateBatonMemo(db, { projectPath: '/proj', memoText: 'preserved memo' });
121
- // 同じ project_path に再度 /tl を打っても memo は残る
122
- writeBaton(db, { projectPath: '/proj', sessionId: 'S2', now: 2000 });
123
- const row = db.prepare('SELECT * FROM handoff_batons').get();
124
- assert.equal(row.session_id, 'S2');
125
- assert.equal(row.created_at, 2000);
126
- assert.equal(row.memo_text, 'preserved memo');
127
- });
128
-
129
- test('consumeBaton: returns memoText when set', () => {
130
- const db = makeDb();
131
- writeBaton(db, { projectPath: '/proj', sessionId: 'S1', now: 1000 });
132
- updateBatonMemo(db, { projectPath: '/proj', memoText: '中断メモ' });
133
- const result = consumeBaton(db, { projectPath: '/proj', now: 1000 + 1000 });
134
- assert.equal(result.sessionId, 'S1');
135
- assert.equal(result.memoText, '中断メモ');
136
- });
137
-
138
- test('consumeBaton: returns memoText=null when not set', () => {
101
+ test('consumeBaton: returns no memoText property in v8 (memo column dropped)', () => {
139
102
  const db = makeDb();
140
103
  writeBaton(db, { projectPath: '/proj', sessionId: 'S1', now: 1000 });
141
104
  const result = consumeBaton(db, { projectPath: '/proj', now: 1000 + 1000 });
142
105
  assert.equal(result.sessionId, 'S1');
143
- assert.equal(result.memoText, null);
106
+ assert.equal('memoText' in result, false, 'memoText property should be removed in v8');
144
107
  });
@@ -18,7 +18,7 @@ import { homedir } from 'node:os';
18
18
 
19
19
  const PACKAGE_ROOT = resolve(dirname(fileURLToPath(import.meta.url)), '..', '..');
20
20
  const SLASH_COMMANDS_SRC = join(PACKAGE_ROOT, '.claude', 'commands');
21
- const SC_SLASH_COMMAND_FILES = ['tl.md', 'sc-detail.md', 'tl-trim.md'];
21
+ const SC_SLASH_COMMAND_FILES = ['tl.md', 'sc-detail.md'];
22
22
  const CODEX_SKILLS_SRC = join(PACKAGE_ROOT, 'codex', 'skills');
23
23
  const CODEX_SKILL_NAMES = ['throughline'];
24
24
  const CODEX_HOOKS_RELATIVE_PATH = ['.codex', 'hooks.json'];
@@ -415,7 +415,7 @@ export async function run(args = []) {
415
415
  console.log('有効な hooks:');
416
416
  console.log(' SessionStart → throughline session-start (セッション記録・バトン消費・引き継ぎ注入)');
417
417
  console.log(' Stop → throughline process-turn (L1 要約 + L2 本文保存 + L3 詳細保存)');
418
- console.log(' UserPromptSubmit → throughline prompt-submit (/tl バトン書き込み)');
418
+ console.log(' UserPromptSubmit → throughline prompt-submit (/tl & /clear バトン書き込み)');
419
419
  if (codex) {
420
420
  console.log(` Codex Stop → ${buildCodexStopHookCommand()} (Codex rollout capture + L1 要約)`);
421
421
  }
@@ -54,7 +54,7 @@ test('global install copies Throughline slash commands to ~/.claude/commands/',
54
54
  const trim = join(home.dir, '.claude', 'commands', 'tl-trim.md');
55
55
  assert.ok(existsSync(tl), 'tl.md should be installed globally');
56
56
  assert.ok(existsSync(sc), 'sc-detail.md should be installed globally');
57
- assert.ok(existsSync(trim), 'tl-trim.md should be installed globally');
57
+ assert.ok(!existsSync(trim), 'tl-trim.md should NOT be installed (deprecated in v0.4.0)');
58
58
  const tlBody = readFileSync(tl, 'utf8');
59
59
  assert.match(tlBody, /Throughline/, 'tl.md content should be real');
60
60
  const settings = JSON.parse(readFileSync(join(home.dir, '.claude', 'settings.json'), 'utf8'));
@@ -261,8 +261,6 @@ test('uninstall removes slash command files', async () => {
261
261
  assert.ok(!existsSync(tl), 'uninstall should remove tl.md');
262
262
  const sc = join(home.dir, '.claude', 'commands', 'sc-detail.md');
263
263
  assert.ok(!existsSync(sc), 'uninstall should remove sc-detail.md');
264
- const trim = join(home.dir, '.claude', 'commands', 'tl-trim.md');
265
- assert.ok(!existsSync(trim), 'uninstall should remove tl-trim.md');
266
264
  const codexSkill = join(home.dir, '.codex', 'skills', 'throughline', 'SKILL.md');
267
265
  assert.ok(!existsSync(codexSkill), 'uninstall should remove Throughline Codex skill');
268
266
  } finally {
@@ -38,10 +38,10 @@ function indexNames(db) {
38
38
  .map((row) => row.name);
39
39
  }
40
40
 
41
- test('schema v7 preserves Claude-facing tables, fields, and unique indexes', async () => {
41
+ test('schema v8 preserves Claude-facing tables, fields, and unique indexes', async () => {
42
42
  await withIsolatedDb((db) => {
43
43
  const version = db.prepare('PRAGMA user_version').get();
44
- assert.equal(version.user_version, 7);
44
+ assert.equal(version.user_version, 8);
45
45
 
46
46
  assert.deepEqual(columnNames(db, 'sessions'), [
47
47
  'session_id',
@@ -87,7 +87,6 @@ test('schema v7 preserves Claude-facing tables, fields, and unique indexes', asy
87
87
  'project_path',
88
88
  'session_id',
89
89
  'created_at',
90
- 'memo_text',
91
90
  ]);
92
91
 
93
92
  const indexes = indexNames(db);
package/src/db.mjs CHANGED
@@ -9,7 +9,7 @@ import { join } from 'path';
9
9
 
10
10
  const DB_DIR = join(homedir(), '.throughline');
11
11
  const DB_PATH = join(DB_DIR, 'throughline.db');
12
- const CURRENT_VERSION = 7;
12
+ const CURRENT_VERSION = 8;
13
13
 
14
14
  let _db = null;
15
15
 
@@ -202,6 +202,19 @@ function initSchema(db) {
202
202
  }
203
203
  }
204
204
 
205
+ // v7 → v8: handoff_batons から memo_text 列を drop。
206
+ // 新仕様 (docs/THROUGHLINE_CLEAR_AUTO_HANDOFF_PLAN.md) で memo 廃止:
207
+ // - /clear 自動引継ぎ (SessionStart source='clear') + /tl baton (memo なし) の 2 経路に
208
+ // - 注入は L1 + L2 + L3 refs のみ
209
+ // - save-inflight CLI / updateBatonMemo 関数も併せて削除
210
+ // SQLite 3.35.0+ で DROP COLUMN サポート (Node.js v22.5+ 同梱版で利用可)。
211
+ if (version < 8) {
212
+ const batonCols = db.prepare('PRAGMA table_info(handoff_batons)').all();
213
+ if (batonCols.some((c) => c.name === 'memo_text')) {
214
+ db.exec('ALTER TABLE handoff_batons DROP COLUMN memo_text');
215
+ }
216
+ }
217
+
205
218
  if (version < CURRENT_VERSION) {
206
219
  db.exec(`PRAGMA user_version = ${CURRENT_VERSION}`);
207
220
  }