@mindfoldhq/trellis 0.5.14 → 0.5.16

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.
Files changed (47) hide show
  1. package/dist/commands/init.d.ts.map +1 -1
  2. package/dist/commands/init.js +70 -41
  3. package/dist/commands/init.js.map +1 -1
  4. package/dist/commands/uninstall.d.ts.map +1 -1
  5. package/dist/commands/uninstall.js +28 -2
  6. package/dist/commands/uninstall.js.map +1 -1
  7. package/dist/commands/update.d.ts.map +1 -1
  8. package/dist/commands/update.js +21 -1
  9. package/dist/commands/update.js.map +1 -1
  10. package/dist/configurators/claude.d.ts.map +1 -1
  11. package/dist/configurators/claude.js +1 -0
  12. package/dist/configurators/claude.js.map +1 -1
  13. package/dist/migrations/manifests/0.5.15.json +9 -0
  14. package/dist/migrations/manifests/0.5.16.json +9 -0
  15. package/dist/migrations/manifests/0.6.0-beta.11.json +9 -0
  16. package/dist/migrations/manifests/0.6.0-beta.12.json +9 -0
  17. package/dist/migrations/manifests/0.6.0-beta.13.json +9 -0
  18. package/dist/migrations/manifests/0.6.0-beta.14.json +9 -0
  19. package/dist/migrations/manifests/0.6.0-beta.15.json +9 -0
  20. package/dist/migrations/manifests/0.6.0-beta.16.json +9 -0
  21. package/dist/migrations/manifests/0.6.0-beta.17.json +9 -0
  22. package/dist/templates/codex/hooks/session-start.py +22 -0
  23. package/dist/templates/codex/hooks.json +1 -1
  24. package/dist/templates/copilot/hooks/session-start.py +24 -0
  25. package/dist/templates/cursor/hooks.json +0 -6
  26. package/dist/templates/shared-hooks/index.d.ts.map +1 -1
  27. package/dist/templates/shared-hooks/index.js +0 -1
  28. package/dist/templates/shared-hooks/index.js.map +1 -1
  29. package/dist/templates/shared-hooks/inject-workflow-state.py +22 -0
  30. package/dist/templates/shared-hooks/session-start.py +25 -8
  31. package/dist/utils/cwd-guard.d.ts +38 -0
  32. package/dist/utils/cwd-guard.d.ts.map +1 -0
  33. package/dist/utils/cwd-guard.js +62 -0
  34. package/dist/utils/cwd-guard.js.map +1 -0
  35. package/dist/utils/file-writer.d.ts +13 -0
  36. package/dist/utils/file-writer.d.ts.map +1 -1
  37. package/dist/utils/file-writer.js +59 -1
  38. package/dist/utils/file-writer.js.map +1 -1
  39. package/dist/utils/manifest-prune.d.ts +61 -0
  40. package/dist/utils/manifest-prune.d.ts.map +1 -0
  41. package/dist/utils/manifest-prune.js +136 -0
  42. package/dist/utils/manifest-prune.js.map +1 -0
  43. package/dist/utils/template-hash.d.ts +32 -6
  44. package/dist/utils/template-hash.d.ts.map +1 -1
  45. package/dist/utils/template-hash.js +53 -31
  46. package/dist/utils/template-hash.js.map +1 -1
  47. package/package.json +1 -1
@@ -73,14 +73,27 @@ Trellis SessionStart 已注入:workflow、当前任务状态、开发者身份
73
73
  Then continue directly with the user's request. This notice is one-shot: do not repeat it after the first assistant reply in the same session.
74
74
  </first-reply-notice>"""
75
75
 
76
- # IMPORTANT: Force stdout to use UTF-8 on Windows
77
- # This fixes UnicodeEncodeError when outputting non-ASCII characters
76
+ # Force UTF-8 on stdin/stdout/stderr on Windows. Default codepage there is
77
+ # cp936 / cp1252 / etc. non-ASCII content (Chinese task names, prd snippets)
78
+ # both in stdin (hook payload from host CLI) and stdout (our emitted blocks)
79
+ # raises UnicodeDecodeError / UnicodeEncodeError. Equivalent to `python -X utf8`
80
+ # but applied per-stream so we don't depend on host CLI's command wiring.
78
81
  if sys.platform.startswith("win"):
79
82
  import io as _io
80
- if hasattr(sys.stdout, "reconfigure"):
81
- sys.stdout.reconfigure(encoding="utf-8", errors="replace") # type: ignore[union-attr]
82
- elif hasattr(sys.stdout, "detach"):
83
- sys.stdout = _io.TextIOWrapper(sys.stdout.detach(), encoding="utf-8", errors="replace") # type: ignore[union-attr]
83
+ for _stream_name in ("stdin", "stdout", "stderr"):
84
+ _stream = getattr(sys, _stream_name, None)
85
+ if _stream is None:
86
+ continue
87
+ if hasattr(_stream, "reconfigure"):
88
+ try:
89
+ _stream.reconfigure(encoding="utf-8", errors="replace") # type: ignore[union-attr]
90
+ except Exception:
91
+ pass
92
+ elif hasattr(_stream, "detach"):
93
+ try:
94
+ setattr(sys, _stream_name, _io.TextIOWrapper(_stream.detach(), encoding="utf-8", errors="replace"))
95
+ except Exception:
96
+ pass
84
97
 
85
98
 
86
99
 
@@ -765,11 +778,15 @@ When the user sends the first message, follow <task-status> and the workflow gui
765
778
  If a task is READY, execute its Next required action without asking whether to continue.
766
779
  </ready>""")
767
780
 
781
+ context_text = output.getvalue()
768
782
  result = {
783
+ # Claude Code / Qoder / CodeBuddy / Droid / Gemini / Copilot format
769
784
  "hookSpecificOutput": {
770
785
  "hookEventName": "SessionStart",
771
- "additionalContext": output.getvalue(),
772
- }
786
+ "additionalContext": context_text,
787
+ },
788
+ # Cursor sessionStart format (top-level snake_case per Cursor docs)
789
+ "additional_context": context_text,
773
790
  }
774
791
 
775
792
  # Output JSON - stdout is already configured for UTF-8
@@ -0,0 +1,38 @@
1
+ /**
2
+ * Homedir guard for destructive commands (init, uninstall).
3
+ *
4
+ * Running `trellis init` / `trellis uninstall` in `$HOME` is catastrophic:
5
+ * platforms like Claude Code, Codex, OpenCode all store global runtime data
6
+ * (`.claude/projects/<sanitized-cwd>/*.jsonl` chat history, `.codex/sessions/`,
7
+ * `.opencode/` caches, etc.) directly in the user's home directory. If
8
+ * trellis manages the same `.{platform}/` config dirs and the hash manifest
9
+ * picks up runtime data, uninstall would later unlink it.
10
+ *
11
+ * Subdirectories of home (`~/Documents/projects/foo/`) are NOT blocked — only
12
+ * exact-home match.
13
+ *
14
+ * Bypass: `TRELLIS_ALLOW_HOMEDIR=1`.
15
+ */
16
+ /**
17
+ * Returns true if `process.cwd()` is exactly the user's home directory.
18
+ *
19
+ * Uses `realpathSync.native()` on both sides so symlinks, `..` segments, and
20
+ * case differences (Windows) don't confuse the comparison. On Windows the
21
+ * comparison is also case-insensitive — `C:\Users\Alice` matches
22
+ * `c:\users\alice`.
23
+ *
24
+ * Permissive on lookup failure: if realpath fails for any reason (broken
25
+ * symlink, EACCES, etc.) we return false so a safety check doesn't crash
26
+ * the command.
27
+ */
28
+ export declare function isCwdHomedir(): boolean;
29
+ /**
30
+ * Error message printed by both `trellis init` and `trellis uninstall` when
31
+ * the homedir guard trips.
32
+ */
33
+ export declare function homedirGuardMessage(commandName: "init" | "uninstall"): string;
34
+ /**
35
+ * Returns true when the bypass env var is set.
36
+ */
37
+ export declare function homedirBypassEnabled(): boolean;
38
+ //# sourceMappingURL=cwd-guard.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"cwd-guard.d.ts","sourceRoot":"","sources":["../../src/utils/cwd-guard.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;;;;;GAcG;AAKH;;;;;;;;;;;GAWG;AACH,wBAAgB,YAAY,IAAI,OAAO,CAYtC;AAED;;;GAGG;AACH,wBAAgB,mBAAmB,CAAC,WAAW,EAAE,MAAM,GAAG,WAAW,GAAG,MAAM,CAS7E;AAED;;GAEG;AACH,wBAAgB,oBAAoB,IAAI,OAAO,CAE9C"}
@@ -0,0 +1,62 @@
1
+ /**
2
+ * Homedir guard for destructive commands (init, uninstall).
3
+ *
4
+ * Running `trellis init` / `trellis uninstall` in `$HOME` is catastrophic:
5
+ * platforms like Claude Code, Codex, OpenCode all store global runtime data
6
+ * (`.claude/projects/<sanitized-cwd>/*.jsonl` chat history, `.codex/sessions/`,
7
+ * `.opencode/` caches, etc.) directly in the user's home directory. If
8
+ * trellis manages the same `.{platform}/` config dirs and the hash manifest
9
+ * picks up runtime data, uninstall would later unlink it.
10
+ *
11
+ * Subdirectories of home (`~/Documents/projects/foo/`) are NOT blocked — only
12
+ * exact-home match.
13
+ *
14
+ * Bypass: `TRELLIS_ALLOW_HOMEDIR=1`.
15
+ */
16
+ import { realpathSync } from "node:fs";
17
+ import * as os from "node:os";
18
+ /**
19
+ * Returns true if `process.cwd()` is exactly the user's home directory.
20
+ *
21
+ * Uses `realpathSync.native()` on both sides so symlinks, `..` segments, and
22
+ * case differences (Windows) don't confuse the comparison. On Windows the
23
+ * comparison is also case-insensitive — `C:\Users\Alice` matches
24
+ * `c:\users\alice`.
25
+ *
26
+ * Permissive on lookup failure: if realpath fails for any reason (broken
27
+ * symlink, EACCES, etc.) we return false so a safety check doesn't crash
28
+ * the command.
29
+ */
30
+ export function isCwdHomedir() {
31
+ try {
32
+ let cwd = realpathSync.native(process.cwd());
33
+ let home = realpathSync.native(os.homedir());
34
+ if (process.platform === "win32") {
35
+ cwd = cwd.toLowerCase();
36
+ home = home.toLowerCase();
37
+ }
38
+ return cwd === home;
39
+ }
40
+ catch {
41
+ return false;
42
+ }
43
+ }
44
+ /**
45
+ * Error message printed by both `trellis init` and `trellis uninstall` when
46
+ * the homedir guard trips.
47
+ */
48
+ export function homedirGuardMessage(commandName) {
49
+ return (`✗ Refusing to run \`trellis ${commandName}\` in your home directory.\n\n` +
50
+ `Trellis manages platform config dirs like .claude/, .codex/, .opencode/, which\n` +
51
+ `in your home directory also contain runtime data from those CLIs (chat history,\n` +
52
+ `session JSONLs, caches). Running here can wipe that data.\n\n` +
53
+ `Run trellis from your project directory instead. If you really want to run in\n` +
54
+ `$HOME, set TRELLIS_ALLOW_HOMEDIR=1.`);
55
+ }
56
+ /**
57
+ * Returns true when the bypass env var is set.
58
+ */
59
+ export function homedirBypassEnabled() {
60
+ return process.env.TRELLIS_ALLOW_HOMEDIR === "1";
61
+ }
62
+ //# sourceMappingURL=cwd-guard.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"cwd-guard.js","sourceRoot":"","sources":["../../src/utils/cwd-guard.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;;;;;GAcG;AAEH,OAAO,EAAE,YAAY,EAAE,MAAM,SAAS,CAAC;AACvC,OAAO,KAAK,EAAE,MAAM,SAAS,CAAC;AAE9B;;;;;;;;;;;GAWG;AACH,MAAM,UAAU,YAAY;IAC1B,IAAI,CAAC;QACH,IAAI,GAAG,GAAG,YAAY,CAAC,MAAM,CAAC,OAAO,CAAC,GAAG,EAAE,CAAC,CAAC;QAC7C,IAAI,IAAI,GAAG,YAAY,CAAC,MAAM,CAAC,EAAE,CAAC,OAAO,EAAE,CAAC,CAAC;QAC7C,IAAI,OAAO,CAAC,QAAQ,KAAK,OAAO,EAAE,CAAC;YACjC,GAAG,GAAG,GAAG,CAAC,WAAW,EAAE,CAAC;YACxB,IAAI,GAAG,IAAI,CAAC,WAAW,EAAE,CAAC;QAC5B,CAAC;QACD,OAAO,GAAG,KAAK,IAAI,CAAC;IACtB,CAAC;IAAC,MAAM,CAAC;QACP,OAAO,KAAK,CAAC;IACf,CAAC;AACH,CAAC;AAED;;;GAGG;AACH,MAAM,UAAU,mBAAmB,CAAC,WAAiC;IACnE,OAAO,CACL,+BAA+B,WAAW,gCAAgC;QAC1E,kFAAkF;QAClF,mFAAmF;QACnF,+DAA+D;QAC/D,iFAAiF;QACjF,qCAAqC,CACtC,CAAC;AACJ,CAAC;AAED;;GAEG;AACH,MAAM,UAAU,oBAAoB;IAClC,OAAO,OAAO,CAAC,GAAG,CAAC,qBAAqB,KAAK,GAAG,CAAC;AACnD,CAAC"}
@@ -4,6 +4,19 @@ export interface WriteOptions {
4
4
  }
5
5
  export declare function setWriteMode(mode: WriteMode): void;
6
6
  export declare function getWriteMode(): WriteMode;
7
+ /**
8
+ * Begin recording every write into the returned Set. Calls accumulate into the
9
+ * same set until `stopRecordingWrites` runs. POSIX relative paths (relative to
10
+ * `cwd`) are stored, matching `.template-hashes.json` keys.
11
+ *
12
+ * Nested recording sessions are NOT supported — the caller must ensure
13
+ * `stopRecordingWrites` runs before the next `startRecordingWrites`. Failure
14
+ * is silent (the second `start` replaces the first set), so callers should
15
+ * always pair start/stop in try/finally.
16
+ */
17
+ export declare function startRecordingWrites(cwd: string): Set<string>;
18
+ /** End recording. Subsequent writes are not captured until `start` is called again. */
19
+ export declare function stopRecordingWrites(): void;
7
20
  /**
8
21
  * Write file with conflict handling
9
22
  * - If file doesn't exist: write directly
@@ -1 +1 @@
1
- {"version":3,"file":"file-writer.d.ts","sourceRoot":"","sources":["../../src/utils/file-writer.ts"],"names":[],"mappings":"AAKA,MAAM,MAAM,SAAS,GAAG,KAAK,GAAG,OAAO,GAAG,MAAM,GAAG,QAAQ,CAAC;AAE5D,MAAM,WAAW,YAAY;IAC3B,IAAI,EAAE,SAAS,CAAC;CACjB;AASD,wBAAgB,YAAY,CAAC,IAAI,EAAE,SAAS,GAAG,IAAI,CAElD;AAED,wBAAgB,YAAY,IAAI,SAAS,CAExC;AA6BD;;;;;;;;GAQG;AACH,wBAAsB,SAAS,CAC7B,QAAQ,EAAE,MAAM,EAChB,OAAO,EAAE,MAAM,EACf,OAAO,CAAC,EAAE;IAAE,UAAU,CAAC,EAAE,OAAO,CAAA;CAAE,GACjC,OAAO,CAAC,OAAO,CAAC,CA8GlB;AAED;;GAEG;AACH,wBAAgB,SAAS,CAAC,OAAO,EAAE,MAAM,GAAG,IAAI,CAE/C"}
1
+ {"version":3,"file":"file-writer.d.ts","sourceRoot":"","sources":["../../src/utils/file-writer.ts"],"names":[],"mappings":"AAOA,MAAM,MAAM,SAAS,GAAG,KAAK,GAAG,OAAO,GAAG,MAAM,GAAG,QAAQ,CAAC;AAE5D,MAAM,WAAW,YAAY;IAC3B,IAAI,EAAE,SAAS,CAAC;CACjB;AASD,wBAAgB,YAAY,CAAC,IAAI,EAAE,SAAS,GAAG,IAAI,CAElD;AAED,wBAAgB,YAAY,IAAI,SAAS,CAExC;AAkBD;;;;;;;;;GASG;AACH,wBAAgB,oBAAoB,CAAC,GAAG,EAAE,MAAM,GAAG,GAAG,CAAC,MAAM,CAAC,CAK7D;AAED,uFAAuF;AACvF,wBAAgB,mBAAmB,IAAI,IAAI,CAG1C;AAsCD;;;;;;;;GAQG;AACH,wBAAsB,SAAS,CAC7B,QAAQ,EAAE,MAAM,EAChB,OAAO,EAAE,MAAM,EACf,OAAO,CAAC,EAAE;IAAE,UAAU,CAAC,EAAE,OAAO,CAAA;CAAE,GACjC,OAAO,CAAC,OAAO,CAAC,CA0HlB;AAED;;GAEG;AACH,wBAAgB,SAAS,CAAC,OAAO,EAAE,MAAM,GAAG,IAAI,CAE/C"}
@@ -2,6 +2,7 @@ import fs from "node:fs";
2
2
  import path from "node:path";
3
3
  import chalk from "chalk";
4
4
  import inquirer from "inquirer";
5
+ import { toPosix } from "./posix.js";
5
6
  // Global write mode (set from CLI options)
6
7
  let globalWriteMode = "ask";
7
8
  export function setWriteMode(mode) {
@@ -10,6 +11,51 @@ export function setWriteMode(mode) {
10
11
  export function getWriteMode() {
11
12
  return globalWriteMode;
12
13
  }
14
+ // ---------------------------------------------------------------------------
15
+ // Write recording
16
+ //
17
+ // `trellis init` uses recording to capture exactly which files were actually
18
+ // written this run (vs skipped because they already existed). The captured
19
+ // set is what `.template-hashes.json` should contain — NOT a blind directory
20
+ // walk of `.codex/` / `.claude/` / etc, which would include user-owned files
21
+ // that pre-dated init. See `pruneOrphanManifestKeys` for the self-heal side
22
+ // of the same contract.
23
+ // ---------------------------------------------------------------------------
24
+ /** When recording is active, every actual `writeFile` disk write appends here. */
25
+ let writeRecorder = null;
26
+ /** Project root used to convert absolute write paths to POSIX-relative keys. */
27
+ let writeRecorderRoot = null;
28
+ /**
29
+ * Begin recording every write into the returned Set. Calls accumulate into the
30
+ * same set until `stopRecordingWrites` runs. POSIX relative paths (relative to
31
+ * `cwd`) are stored, matching `.template-hashes.json` keys.
32
+ *
33
+ * Nested recording sessions are NOT supported — the caller must ensure
34
+ * `stopRecordingWrites` runs before the next `startRecordingWrites`. Failure
35
+ * is silent (the second `start` replaces the first set), so callers should
36
+ * always pair start/stop in try/finally.
37
+ */
38
+ export function startRecordingWrites(cwd) {
39
+ const sink = new Set();
40
+ writeRecorder = sink;
41
+ writeRecorderRoot = cwd;
42
+ return sink;
43
+ }
44
+ /** End recording. Subsequent writes are not captured until `start` is called again. */
45
+ export function stopRecordingWrites() {
46
+ writeRecorder = null;
47
+ writeRecorderRoot = null;
48
+ }
49
+ /** Record a successful write. Called internally by `writeFile`. */
50
+ function recordWrite(absPath) {
51
+ if (!writeRecorder || !writeRecorderRoot)
52
+ return;
53
+ const rel = path.relative(writeRecorderRoot, absPath);
54
+ // Defensive: skip writes outside cwd (no meaningful manifest key).
55
+ if (rel.startsWith("..") || path.isAbsolute(rel))
56
+ return;
57
+ writeRecorder.add(toPosix(rel));
58
+ }
13
59
  /**
14
60
  * Get relative path from cwd for display
15
61
  */
@@ -49,12 +95,15 @@ export async function writeFile(filePath, content, options) {
49
95
  if (options?.executable) {
50
96
  fs.chmodSync(filePath, "755");
51
97
  }
98
+ recordWrite(filePath);
52
99
  return true;
53
100
  }
54
101
  // File exists, check if content is identical
55
102
  const existingContent = fs.readFileSync(filePath, "utf-8");
56
103
  if (existingContent === content) {
57
- // Content identical, skip silently (no output)
104
+ // Content identical, but no disk write happened. Do not record it for
105
+ // init-time manifests: pre-existing user files can legitimately be
106
+ // byte-identical to a Trellis template and still not be Trellis-owned.
58
107
  return false;
59
108
  }
60
109
  // File exists with different content, handle based on mode.
@@ -70,15 +119,22 @@ export async function writeFile(filePath, content, options) {
70
119
  fs.chmodSync(filePath, "755");
71
120
  }
72
121
  console.log(chalk.yellow(` ↻ Overwritten: ${displayPath}`));
122
+ recordWrite(filePath);
73
123
  return true;
74
124
  }
75
125
  if (mode === "skip") {
76
126
  console.log(chalk.gray(` ○ Skipped: ${displayPath} (already exists)`));
127
+ // Skipped: trellis did NOT write this file — caller should not track it
128
+ // in the manifest. This is the AGENTS.md skip-existing case.
77
129
  return false;
78
130
  }
79
131
  if (mode === "append") {
80
132
  appendToFile(filePath, content, options);
81
133
  console.log(chalk.blue(` + Appended: ${displayPath}`));
134
+ // Append: trellis added trellis content to a user-owned file. Tracking
135
+ // is risky here (uninstall would unlink the whole file), so we do NOT
136
+ // record appended files. Users on `--append` get a fresh manifest miss
137
+ // on next update; that's the safer default.
82
138
  return true;
83
139
  }
84
140
  // mode === 'ask': Interactive prompt
@@ -107,6 +163,7 @@ export async function writeFile(filePath, content, options) {
107
163
  fs.chmodSync(filePath, "755");
108
164
  }
109
165
  console.log(chalk.yellow(` ↻ Overwritten: ${displayPath}`));
166
+ recordWrite(filePath);
110
167
  return true;
111
168
  }
112
169
  if (action === "append") {
@@ -126,6 +183,7 @@ export async function writeFile(filePath, content, options) {
126
183
  fs.chmodSync(filePath, "755");
127
184
  }
128
185
  console.log(chalk.yellow(` ↻ Overwritten: ${displayPath}`));
186
+ recordWrite(filePath);
129
187
  return true;
130
188
  }
131
189
  if (action === "append-all") {
@@ -1 +1 @@
1
- {"version":3,"file":"file-writer.js","sourceRoot":"","sources":["../../src/utils/file-writer.ts"],"names":[],"mappings":"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"}
1
+ {"version":3,"file":"file-writer.js","sourceRoot":"","sources":["../../src/utils/file-writer.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,61 @@
1
+ /**
2
+ * Self-heal poisoned `.template-hashes.json` manifests.
3
+ *
4
+ * Versions before this fix walked `.codex/`, `.claude/`, etc. with a blind
5
+ * recursive scan when computing the manifest, so they hashed user-owned
6
+ * runtime data (`.codex/sessions/*`, `.claude/projects/*.jsonl`, pre-existing
7
+ * `AGENTS.md`, user-added `.codex/skills/<custom>/`, …). On uninstall, every
8
+ * manifest entry is unlinked, which silently deletes user data.
9
+ *
10
+ * `pruneOrphanManifestKeys` removes any manifest entry that no current
11
+ * platform configurator owns. The two entry points that consume it are
12
+ * `trellis update` (before migration classification) and `trellis uninstall`
13
+ * (before plan building). Together they ensure existing poisoned manifests
14
+ * self-correct on the next routine command.
15
+ *
16
+ * Rules:
17
+ * - `.trellis/*` entries are ALWAYS kept. `trellis uninstall` removes
18
+ * `.trellis/` wholesale via `fs.rmSync(..., { recursive: true })`, so
19
+ * manifest accuracy there doesn't affect uninstall data-loss. `update`
20
+ * also relies on these entries to detect user-modified workflow files.
21
+ * - Root-level `AGENTS.md` is kept only when it still looks Trellis-managed
22
+ * (contains the managed block markers) or is missing on disk. This
23
+ * self-heals old poisoned manifests for user-owned AGENTS.md files that
24
+ * predated init and were skipped.
25
+ * - Paths referenced by `from`/`to` of any migration manifest entry
26
+ * (rename, rename-dir, delete, safe-file-delete) are preserved. Pruning
27
+ * them would prevent legitimate pending migrations from finding their
28
+ * source/target.
29
+ * - Everything else: if the path is not in the union of
30
+ * `collectPlatformTemplates()` for currently-configured platforms, it is
31
+ * pruned. This matches "files trellis actually wrote during init/update".
32
+ */
33
+ import type { AITool } from "../types/ai-tools.js";
34
+ import type { TemplateHashes } from "../types/migration.js";
35
+ export interface PruneResult {
36
+ /** Manifest keys removed (POSIX-style relative paths). */
37
+ pruned: string[];
38
+ /** The post-prune manifest (saved to disk only when `pruned.length > 0`). */
39
+ hashes: TemplateHashes;
40
+ }
41
+ export interface PruneOptions {
42
+ /**
43
+ * Save the pruned manifest to `.template-hashes.json`. Defaults to true.
44
+ * Callers can pass `false` to compute the prune without mutating disk
45
+ * (dry-run, change-analysis passes).
46
+ */
47
+ persist?: boolean;
48
+ }
49
+ /**
50
+ * Walk the manifest and split it into kept vs pruned entries.
51
+ *
52
+ * @param cwd Project root — used to save the rewritten manifest.
53
+ * @param configuredPlatforms Output of `getConfiguredPlatforms(cwd)` — caller
54
+ * resolves this so we don't have to re-walk the filesystem.
55
+ * @param hashes Already-loaded manifest contents. Passing it in (vs reading
56
+ * from disk) lets the caller chain `loadHashes` → prune → use the result.
57
+ * @param options.persist When true (default), saves the pruned manifest to
58
+ * disk. Pass `false` for dry-run flows.
59
+ */
60
+ export declare function pruneOrphanManifestKeys(cwd: string, configuredPlatforms: readonly AITool[], hashes: TemplateHashes, options?: PruneOptions): PruneResult;
61
+ //# sourceMappingURL=manifest-prune.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"manifest-prune.d.ts","sourceRoot":"","sources":["../../src/utils/manifest-prune.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GA+BG;AAUH,OAAO,KAAK,EAAE,MAAM,EAAE,MAAM,sBAAsB,CAAC;AACnD,OAAO,KAAK,EAAE,cAAc,EAAE,MAAM,uBAAuB,CAAC;AAK5D,MAAM,WAAW,WAAW;IAC1B,0DAA0D;IAC1D,MAAM,EAAE,MAAM,EAAE,CAAC;IACjB,6EAA6E;IAC7E,MAAM,EAAE,cAAc,CAAC;CACxB;AAoDD,MAAM,WAAW,YAAY;IAC3B;;;;OAIG;IACH,OAAO,CAAC,EAAE,OAAO,CAAC;CACnB;AAED;;;;;;;;;;GAUG;AACH,wBAAgB,uBAAuB,CACrC,GAAG,EAAE,MAAM,EACX,mBAAmB,EAAE,SAAS,MAAM,EAAE,EACtC,MAAM,EAAE,cAAc,EACtB,OAAO,GAAE,YAAiB,GACzB,WAAW,CAmCb"}
@@ -0,0 +1,136 @@
1
+ /**
2
+ * Self-heal poisoned `.template-hashes.json` manifests.
3
+ *
4
+ * Versions before this fix walked `.codex/`, `.claude/`, etc. with a blind
5
+ * recursive scan when computing the manifest, so they hashed user-owned
6
+ * runtime data (`.codex/sessions/*`, `.claude/projects/*.jsonl`, pre-existing
7
+ * `AGENTS.md`, user-added `.codex/skills/<custom>/`, …). On uninstall, every
8
+ * manifest entry is unlinked, which silently deletes user data.
9
+ *
10
+ * `pruneOrphanManifestKeys` removes any manifest entry that no current
11
+ * platform configurator owns. The two entry points that consume it are
12
+ * `trellis update` (before migration classification) and `trellis uninstall`
13
+ * (before plan building). Together they ensure existing poisoned manifests
14
+ * self-correct on the next routine command.
15
+ *
16
+ * Rules:
17
+ * - `.trellis/*` entries are ALWAYS kept. `trellis uninstall` removes
18
+ * `.trellis/` wholesale via `fs.rmSync(..., { recursive: true })`, so
19
+ * manifest accuracy there doesn't affect uninstall data-loss. `update`
20
+ * also relies on these entries to detect user-modified workflow files.
21
+ * - Root-level `AGENTS.md` is kept only when it still looks Trellis-managed
22
+ * (contains the managed block markers) or is missing on disk. This
23
+ * self-heals old poisoned manifests for user-owned AGENTS.md files that
24
+ * predated init and were skipped.
25
+ * - Paths referenced by `from`/`to` of any migration manifest entry
26
+ * (rename, rename-dir, delete, safe-file-delete) are preserved. Pruning
27
+ * them would prevent legitimate pending migrations from finding their
28
+ * source/target.
29
+ * - Everything else: if the path is not in the union of
30
+ * `collectPlatformTemplates()` for currently-configured platforms, it is
31
+ * pruned. This matches "files trellis actually wrote during init/update".
32
+ */
33
+ import fs from "node:fs";
34
+ import path from "node:path";
35
+ import { collectPlatformTemplates } from "../configurators/index.js";
36
+ import { FILE_NAMES } from "../constants/paths.js";
37
+ import { getAllMigrations } from "../migrations/index.js";
38
+ import { saveHashes } from "./template-hash.js";
39
+ import { toPosix } from "./posix.js";
40
+ const TRELLIS_BLOCK_START = "<!-- TRELLIS:START -->";
41
+ const TRELLIS_BLOCK_END = "<!-- TRELLIS:END -->";
42
+ /**
43
+ * Compute the union of "what trellis writes" across:
44
+ * - every configured platform's collectTemplates() output
45
+ * - root-level AGENTS.md when it still carries Trellis managed-block markers
46
+ * - every migration manifest's from/to path (preserve so legitimate
47
+ * pending migrations can find their source/target)
48
+ */
49
+ function buildKnownKeys(configuredPlatforms) {
50
+ const known = new Set();
51
+ for (const id of configuredPlatforms) {
52
+ const templates = collectPlatformTemplates(id);
53
+ if (!templates)
54
+ continue;
55
+ for (const key of templates.keys()) {
56
+ known.add(toPosix(key));
57
+ }
58
+ }
59
+ // Preserve any path referenced by a migration: legitimate pending
60
+ // rename/delete operations need to resolve their `from` (and the target's
61
+ // hash record for `to`) even if the current registry doesn't list it.
62
+ for (const migration of getAllMigrations()) {
63
+ if (migration.from)
64
+ known.add(toPosix(migration.from));
65
+ if (migration.to)
66
+ known.add(toPosix(migration.to));
67
+ }
68
+ return known;
69
+ }
70
+ /**
71
+ * Root-level AGENTS.md needs special handling because it has no platform
72
+ * registry owner. New fixed inits record it only when written, but old
73
+ * manifests may contain a user-owned AGENTS.md that init skipped. The
74
+ * managed block markers are the least destructive ownership signal: no
75
+ * markers means preserve the user's file by pruning the stale manifest key.
76
+ */
77
+ function shouldKeepAgentsMd(cwd) {
78
+ const fullPath = path.join(cwd, FILE_NAMES.AGENTS);
79
+ if (!fs.existsSync(fullPath)) {
80
+ return true;
81
+ }
82
+ try {
83
+ const content = fs.readFileSync(fullPath, "utf-8");
84
+ return (content.includes(TRELLIS_BLOCK_START) &&
85
+ content.includes(TRELLIS_BLOCK_END));
86
+ }
87
+ catch {
88
+ return true;
89
+ }
90
+ }
91
+ /**
92
+ * Walk the manifest and split it into kept vs pruned entries.
93
+ *
94
+ * @param cwd Project root — used to save the rewritten manifest.
95
+ * @param configuredPlatforms Output of `getConfiguredPlatforms(cwd)` — caller
96
+ * resolves this so we don't have to re-walk the filesystem.
97
+ * @param hashes Already-loaded manifest contents. Passing it in (vs reading
98
+ * from disk) lets the caller chain `loadHashes` → prune → use the result.
99
+ * @param options.persist When true (default), saves the pruned manifest to
100
+ * disk. Pass `false` for dry-run flows.
101
+ */
102
+ export function pruneOrphanManifestKeys(cwd, configuredPlatforms, hashes, options = {}) {
103
+ const persist = options.persist ?? true;
104
+ const known = buildKnownKeys(configuredPlatforms);
105
+ const pruned = [];
106
+ const kept = {};
107
+ for (const [rawKey, value] of Object.entries(hashes)) {
108
+ const key = toPosix(rawKey);
109
+ // Always preserve .trellis/ entries — they're for the workflow tree
110
+ // which uninstall removes wholesale and which update needs for
111
+ // modified-file detection.
112
+ if (key.startsWith(".trellis/") || key === ".trellis") {
113
+ kept[key] = value;
114
+ continue;
115
+ }
116
+ if (key === FILE_NAMES.AGENTS) {
117
+ if (shouldKeepAgentsMd(cwd)) {
118
+ kept[key] = value;
119
+ }
120
+ else {
121
+ pruned.push(key);
122
+ }
123
+ continue;
124
+ }
125
+ if (known.has(key)) {
126
+ kept[key] = value;
127
+ continue;
128
+ }
129
+ pruned.push(key);
130
+ }
131
+ if (persist && pruned.length > 0) {
132
+ saveHashes(cwd, kept);
133
+ }
134
+ return { pruned, hashes: kept };
135
+ }
136
+ //# sourceMappingURL=manifest-prune.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"manifest-prune.js","sourceRoot":"","sources":["../../src/utils/manifest-prune.ts"],"names":[],"mappings":"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"}
@@ -83,15 +83,41 @@ export declare function matchesOriginalTemplate(cwd: string, relativePath: strin
83
83
  * @returns Map of path to modification status
84
84
  */
85
85
  export declare function getModificationStatus(cwd: string, relativePaths: string[], hashes: TemplateHashes): Map<string, boolean>;
86
+ /** Options accepted by {@link initializeHashes}. */
87
+ export interface InitializeHashesOptions {
88
+ /**
89
+ * POSIX-style relative paths trellis actually wrote during the init run
90
+ * (captured via `startRecordingWrites` in `file-writer.ts`). Only these
91
+ * paths are hashed for the platform/root-level coverage; anything else
92
+ * under `.codex/` / `.claude/` / etc. is left alone, even if it exists
93
+ * on disk. Setting this to `undefined` or an empty set means "no
94
+ * platform/root coverage this run" — historical hashes from earlier
95
+ * runs are preserved via `merge`.
96
+ */
97
+ trackedPaths?: ReadonlySet<string>;
98
+ /**
99
+ * When true, merge `trackedPaths`-derived hashes into the EXISTING manifest
100
+ * instead of replacing it. Used by `handleReinit` "add platform" flow so
101
+ * previously-tracked platforms aren't wiped from the manifest when only
102
+ * a new platform's writes are recorded. Defaults to false (replace).
103
+ */
104
+ merge?: boolean;
105
+ }
86
106
  /**
87
107
  * Initialize template hashes after init
88
108
  *
89
- * Scans all template directories and computes hashes for files.
90
- * This should be called at the end of `trellis init` to enable
91
- * modification detection on subsequent updates.
109
+ * The platform/root section of the manifest comes from `trackedPaths` —
110
+ * the set of POSIX paths that `writeFile` actually wrote (or owned with
111
+ * byte-identical content) during this init run. Avoids the historical bug
112
+ * where a blind directory walk of `.codex/` / `.claude/` swept up
113
+ * user-owned runtime data (chat history, session JSONLs).
92
114
  *
93
- * @param cwd - Working directory
94
- * @returns Number of files hashed
115
+ * `.trellis/` is still walked recursively (with `EXCLUDE_FROM_HASH`) because
116
+ * uninstall removes `.trellis/` wholesale via `rm -rf` regardless of manifest
117
+ * content — accuracy there doesn't affect data-loss, only `trellis update`
118
+ * 3-way-merge fidelity (preserved by the existing walk).
119
+ *
120
+ * @returns Number of files hashed in the final manifest.
95
121
  */
96
- export declare function initializeHashes(cwd: string): number;
122
+ export declare function initializeHashes(cwd: string, options?: InitializeHashesOptions): number;
97
123
  //# sourceMappingURL=template-hash.d.ts.map
@@ -1 +1 @@
1
- {"version":3,"file":"template-hash.d.ts","sourceRoot":"","sources":["../../src/utils/template-hash.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;;;;GAaG;AAQH,OAAO,KAAK,EAAE,cAAc,EAAE,MAAM,uBAAuB,CAAC;AAe5D;;;;;GAKG;AACH,wBAAgB,WAAW,CAAC,OAAO,EAAE,MAAM,GAAG,MAAM,CAGnD;AAoBD;;;;;;;;GAQG;AACH,wBAAgB,UAAU,CAAC,GAAG,EAAE,MAAM,GAAG,cAAc,CA6BtD;AAED;;;;GAIG;AACH,wBAAgB,UAAU,CAAC,GAAG,EAAE,MAAM,EAAE,MAAM,EAAE,cAAc,GAAG,IAAI,CAOpE;AAED;;;;;GAKG;AACH,wBAAgB,YAAY,CAAC,GAAG,EAAE,MAAM,EAAE,KAAK,EAAE,GAAG,CAAC,MAAM,EAAE,MAAM,CAAC,GAAG,IAAI,CAQ1E;AAED;;GAEG;AACH,wBAAgB,kBAAkB,CAAC,GAAG,EAAE,MAAM,EAAE,YAAY,EAAE,MAAM,GAAG,IAAI,CAU1E;AAED;;GAEG;AACH,wBAAgB,UAAU,CAAC,GAAG,EAAE,MAAM,EAAE,YAAY,EAAE,MAAM,GAAG,IAAI,CAKlE;AAED;;GAEG;AACH,wBAAgB,UAAU,CACxB,GAAG,EAAE,MAAM,EACX,OAAO,EAAE,MAAM,EACf,OAAO,EAAE,MAAM,GACd,IAAI,CASN;AAED;;;;;;;GAOG;AACH,wBAAgB,kBAAkB,CAChC,GAAG,EAAE,MAAM,EACX,YAAY,EAAE,MAAM,EACpB,MAAM,EAAE,cAAc,GACrB,OAAO,CAsBT;AAED;;;;;;;;GAQG;AACH,wBAAgB,uBAAuB,CACrC,GAAG,EAAE,MAAM,EACX,YAAY,EAAE,MAAM,EACpB,eAAe,EAAE,MAAM,GACtB,OAAO,CAST;AAED;;;;;;;GAOG;AACH,wBAAgB,qBAAqB,CACnC,GAAG,EAAE,MAAM,EACX,aAAa,EAAE,MAAM,EAAE,EACvB,MAAM,EAAE,cAAc,GACrB,GAAG,CAAC,MAAM,EAAE,OAAO,CAAC,CAQtB;AA0ED;;;;;;;;;GASG;AACH,wBAAgB,gBAAgB,CAAC,GAAG,EAAE,MAAM,GAAG,MAAM,CAwCpD"}
1
+ {"version":3,"file":"template-hash.d.ts","sourceRoot":"","sources":["../../src/utils/template-hash.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;;;;GAaG;AAOH,OAAO,KAAK,EAAE,cAAc,EAAE,MAAM,uBAAuB,CAAC;AAe5D;;;;;GAKG;AACH,wBAAgB,WAAW,CAAC,OAAO,EAAE,MAAM,GAAG,MAAM,CAGnD;AAoBD;;;;;;;;GAQG;AACH,wBAAgB,UAAU,CAAC,GAAG,EAAE,MAAM,GAAG,cAAc,CA6BtD;AAED;;;;GAIG;AACH,wBAAgB,UAAU,CAAC,GAAG,EAAE,MAAM,EAAE,MAAM,EAAE,cAAc,GAAG,IAAI,CAOpE;AAED;;;;;GAKG;AACH,wBAAgB,YAAY,CAAC,GAAG,EAAE,MAAM,EAAE,KAAK,EAAE,GAAG,CAAC,MAAM,EAAE,MAAM,CAAC,GAAG,IAAI,CAQ1E;AAED;;GAEG;AACH,wBAAgB,kBAAkB,CAAC,GAAG,EAAE,MAAM,EAAE,YAAY,EAAE,MAAM,GAAG,IAAI,CAU1E;AAED;;GAEG;AACH,wBAAgB,UAAU,CAAC,GAAG,EAAE,MAAM,EAAE,YAAY,EAAE,MAAM,GAAG,IAAI,CAKlE;AAED;;GAEG;AACH,wBAAgB,UAAU,CACxB,GAAG,EAAE,MAAM,EACX,OAAO,EAAE,MAAM,EACf,OAAO,EAAE,MAAM,GACd,IAAI,CASN;AAED;;;;;;;GAOG;AACH,wBAAgB,kBAAkB,CAChC,GAAG,EAAE,MAAM,EACX,YAAY,EAAE,MAAM,EACpB,MAAM,EAAE,cAAc,GACrB,OAAO,CAsBT;AAED;;;;;;;;GAQG;AACH,wBAAgB,uBAAuB,CACrC,GAAG,EAAE,MAAM,EACX,YAAY,EAAE,MAAM,EACpB,eAAe,EAAE,MAAM,GACtB,OAAO,CAST;AAED;;;;;;;GAOG;AACH,wBAAgB,qBAAqB,CACnC,GAAG,EAAE,MAAM,EACX,aAAa,EAAE,MAAM,EAAE,EACvB,MAAM,EAAE,cAAc,GACrB,GAAG,CAAC,MAAM,EAAE,OAAO,CAAC,CAQtB;AA8DD,oDAAoD;AACpD,MAAM,WAAW,uBAAuB;IACtC;;;;;;;;OAQG;IACH,YAAY,CAAC,EAAE,WAAW,CAAC,MAAM,CAAC,CAAC;IACnC;;;;;OAKG;IACH,KAAK,CAAC,EAAE,OAAO,CAAC;CACjB;AAED;;;;;;;;;;;;;;;GAeG;AACH,wBAAgB,gBAAgB,CAC9B,GAAG,EAAE,MAAM,EACX,OAAO,GAAE,uBAA4B,GACpC,MAAM,CAyDR"}