@pugi/cli 0.1.0-beta.4 → 0.1.0-beta.41

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 (250) hide show
  1. package/THIRD_PARTY_NOTICES.md +40 -0
  2. package/assets/pugi-mascot.ansi +15 -25
  3. package/bin/run.js +33 -1
  4. package/dist/commands/jobs-watch.js +201 -0
  5. package/dist/commands/jobs.js +15 -0
  6. package/dist/commands/smoke.js +133 -0
  7. package/dist/core/agent-progress/cleanup.js +134 -0
  8. package/dist/core/agent-progress/schema.js +144 -0
  9. package/dist/core/agent-progress/writer.js +101 -0
  10. package/dist/core/artifact-chain/dispatcher.js +148 -0
  11. package/dist/core/artifact-chain/exporter.js +164 -0
  12. package/dist/core/artifact-chain/state.js +243 -0
  13. package/dist/core/artifact-chain/steps.js +169 -0
  14. package/dist/core/auth/ensure-authenticated.js +129 -0
  15. package/dist/core/auth/env-provider.js +238 -0
  16. package/dist/core/auto-update/channels.js +122 -0
  17. package/dist/core/auto-update/checker.js +241 -0
  18. package/dist/core/auto-update/state.js +235 -0
  19. package/dist/core/bare-mode/index.js +107 -0
  20. package/dist/core/bash-classifier.js +108 -1
  21. package/dist/core/checkpoint/resumer.js +149 -0
  22. package/dist/core/checkpoint/rewinder.js +291 -0
  23. package/dist/core/codegraph/decision-store.js +248 -0
  24. package/dist/core/codegraph/detect-repo.js +459 -0
  25. package/dist/core/codegraph/install.js +134 -0
  26. package/dist/core/codegraph/offer-hook.js +220 -0
  27. package/dist/core/compact/auto-trigger.js +96 -0
  28. package/dist/core/compact/buffer-rewriter.js +115 -0
  29. package/dist/core/compact/summarizer.js +208 -0
  30. package/dist/core/compact/token-counter.js +108 -0
  31. package/dist/core/consensus/diff-capture.js +73 -0
  32. package/dist/core/context/index.js +7 -0
  33. package/dist/core/context/markdown-traverse.js +255 -0
  34. package/dist/core/cost/rate-card.js +129 -0
  35. package/dist/core/cost/tracker.js +221 -0
  36. package/dist/core/denial-tracking/index.js +8 -0
  37. package/dist/core/denial-tracking/state.js +264 -0
  38. package/dist/core/diagnostics/probe-runner.js +93 -0
  39. package/dist/core/diagnostics/probes/api.js +46 -0
  40. package/dist/core/diagnostics/probes/auth.js +86 -0
  41. package/dist/core/diagnostics/probes/bare-mode.js +42 -0
  42. package/dist/core/diagnostics/probes/cli-version.js +127 -0
  43. package/dist/core/diagnostics/probes/config.js +72 -0
  44. package/dist/core/diagnostics/probes/denial-tracking.js +57 -0
  45. package/dist/core/diagnostics/probes/disk.js +81 -0
  46. package/dist/core/diagnostics/probes/git.js +65 -0
  47. package/dist/core/diagnostics/probes/mcp.js +75 -0
  48. package/dist/core/diagnostics/probes/node.js +59 -0
  49. package/dist/core/diagnostics/probes/pnpm.js +36 -0
  50. package/dist/core/diagnostics/probes/pugi-md.js +89 -0
  51. package/dist/core/diagnostics/probes/session.js +74 -0
  52. package/dist/core/diagnostics/probes/status-snapshot.js +488 -0
  53. package/dist/core/diagnostics/probes/workspace.js +63 -0
  54. package/dist/core/diagnostics/types.js +70 -0
  55. package/dist/core/dispatch/cache-cleanup.js +197 -0
  56. package/dist/core/dispatch/cache-handoff.js +295 -0
  57. package/dist/core/edits/dispatch.js +218 -2
  58. package/dist/core/edits/journal.js +199 -0
  59. package/dist/core/edits/layer-d-ast.js +557 -14
  60. package/dist/core/edits/verify-hook.js +273 -0
  61. package/dist/core/edits/worktree.js +322 -0
  62. package/dist/core/engine/anvil-client.js +115 -5
  63. package/dist/core/engine/budgets.js +98 -0
  64. package/dist/core/engine/context-prefix.js +155 -0
  65. package/dist/core/engine/intent.js +260 -0
  66. package/dist/core/engine/native-pugi.js +860 -211
  67. package/dist/core/engine/prompts.js +88 -2
  68. package/dist/core/engine/strip-internal-fields.js +124 -0
  69. package/dist/core/engine/tool-bridge.js +1045 -36
  70. package/dist/core/feedback/queue.js +177 -0
  71. package/dist/core/feedback/submitter.js +145 -0
  72. package/dist/core/file-cache.js +113 -1
  73. package/dist/core/hooks/events.js +44 -0
  74. package/dist/core/hooks/index.js +15 -0
  75. package/dist/core/hooks/registry.js +213 -0
  76. package/dist/core/hooks/runner.js +236 -0
  77. package/dist/core/hooks/v2/event-emitter.js +115 -0
  78. package/dist/core/hooks/v2/executor.js +282 -0
  79. package/dist/core/hooks/v2/index.js +25 -0
  80. package/dist/core/hooks/v2/lifecycle.js +104 -0
  81. package/dist/core/hooks/v2/loader.js +216 -0
  82. package/dist/core/hooks/v2/matcher.js +125 -0
  83. package/dist/core/hooks/v2/trust.js +143 -0
  84. package/dist/core/hooks/v2/types.js +86 -0
  85. package/dist/core/lsp/cache.js +105 -0
  86. package/dist/core/lsp/client.js +776 -0
  87. package/dist/core/lsp/language-detect.js +66 -0
  88. package/dist/core/lsp/post-edit-diagnostics.js +171 -0
  89. package/dist/core/mcp/client.js +75 -6
  90. package/dist/core/mcp/http-server.js +553 -0
  91. package/dist/core/mcp/orchestrator-tools.js +662 -0
  92. package/dist/core/mcp/permission.js +190 -0
  93. package/dist/core/mcp/registry.js +24 -2
  94. package/dist/core/mcp/server-tools.js +219 -0
  95. package/dist/core/mcp/server.js +397 -0
  96. package/dist/core/memory/dual-write.js +416 -0
  97. package/dist/core/memory/phase1-kinds.js +20 -0
  98. package/dist/core/memory-sync/queue.js +158 -0
  99. package/dist/core/onboarding/ensure-initialized.js +133 -0
  100. package/dist/core/onboarding/marker.js +111 -0
  101. package/dist/core/onboarding/telemetry-state.js +108 -0
  102. package/dist/core/output-style/presets.js +176 -0
  103. package/dist/core/output-style/state.js +185 -0
  104. package/dist/core/permissions/auto-classifier.js +124 -0
  105. package/dist/core/permissions/circuit-breaker.js +83 -0
  106. package/dist/core/permissions/gate.js +278 -0
  107. package/dist/core/permissions/index.js +20 -0
  108. package/dist/core/permissions/mode.js +174 -0
  109. package/dist/core/permissions/state.js +241 -0
  110. package/dist/core/permissions/tool-class.js +93 -0
  111. package/dist/core/prd-check/parser.js +215 -0
  112. package/dist/core/prd-check/reporter.js +127 -0
  113. package/dist/core/prd-check/session-review.js +557 -0
  114. package/dist/core/prd-check/verifiers.js +223 -0
  115. package/dist/core/pugi-md/context-injector.js +76 -0
  116. package/dist/core/pugi-md/walk-up.js +207 -0
  117. package/dist/core/release-notes/parser.js +241 -0
  118. package/dist/core/release-notes/state.js +116 -0
  119. package/dist/core/repl/history.js +11 -1
  120. package/dist/core/repl/model-pricing.js +135 -0
  121. package/dist/core/repl/session.js +1899 -38
  122. package/dist/core/repl/slash-commands.js +406 -21
  123. package/dist/core/repl/store/session-store.js +31 -2
  124. package/dist/core/repl/workspace-context.js +22 -0
  125. package/dist/core/repo-map/build.js +125 -0
  126. package/dist/core/repo-map/cache.js +185 -0
  127. package/dist/core/repo-map/extractor.js +254 -0
  128. package/dist/core/repo-map/formatter.js +145 -0
  129. package/dist/core/repo-map/scanner.js +211 -0
  130. package/dist/core/retry-budget/budget.js +284 -0
  131. package/dist/core/retry-budget/index.js +5 -0
  132. package/dist/core/session.js +92 -0
  133. package/dist/core/settings.js +80 -0
  134. package/dist/core/share/formatter.js +271 -0
  135. package/dist/core/share/redactor.js +221 -0
  136. package/dist/core/share/uploader.js +267 -0
  137. package/dist/core/skills/defaults.js +457 -0
  138. package/dist/core/smoke/headless-driver.js +174 -0
  139. package/dist/core/smoke/orchestrator.js +194 -0
  140. package/dist/core/smoke/runner.js +238 -0
  141. package/dist/core/smoke/scenario-parser.js +316 -0
  142. package/dist/core/subagents/dispatcher-real.js +600 -0
  143. package/dist/core/subagents/dispatcher.js +113 -24
  144. package/dist/core/subagents/index.js +18 -5
  145. package/dist/core/subagents/isolation-matrix.js +213 -0
  146. package/dist/core/subagents/spawn.js +19 -4
  147. package/dist/core/telemetry/emitter.js +229 -0
  148. package/dist/core/telemetry/queue.js +251 -0
  149. package/dist/core/theme/context.js +91 -0
  150. package/dist/core/theme/presets.js +228 -0
  151. package/dist/core/theme/state.js +181 -0
  152. package/dist/core/todos/invariant.js +10 -0
  153. package/dist/core/todos/state.js +177 -0
  154. package/dist/core/transport/version-interceptor.js +166 -0
  155. package/dist/core/vim/keymap.js +288 -0
  156. package/dist/core/vim/state.js +92 -0
  157. package/dist/index.js +28 -0
  158. package/dist/runtime/bootstrap.js +190 -0
  159. package/dist/runtime/cli.js +3073 -321
  160. package/dist/runtime/commands/cancel.js +231 -0
  161. package/dist/runtime/commands/chain.js +489 -0
  162. package/dist/runtime/commands/codegraph-status.js +227 -0
  163. package/dist/runtime/commands/compact.js +297 -0
  164. package/dist/runtime/commands/cost.js +199 -0
  165. package/dist/runtime/commands/delegate.js +242 -11
  166. package/dist/runtime/commands/dispatch.js +126 -0
  167. package/dist/runtime/commands/doctor.js +390 -0
  168. package/dist/runtime/commands/feedback.js +184 -0
  169. package/dist/runtime/commands/hooks.js +184 -0
  170. package/dist/runtime/commands/lsp.js +368 -0
  171. package/dist/runtime/commands/mcp.js +879 -0
  172. package/dist/runtime/commands/memory.js +508 -0
  173. package/dist/runtime/commands/model.js +237 -0
  174. package/dist/runtime/commands/onboarding.js +275 -0
  175. package/dist/runtime/commands/patch.js +128 -0
  176. package/dist/runtime/commands/permissions.js +112 -0
  177. package/dist/runtime/commands/plan.js +143 -0
  178. package/dist/runtime/commands/prd-check.js +285 -0
  179. package/dist/runtime/commands/redo-blob-store.js +92 -0
  180. package/dist/runtime/commands/redo.js +361 -0
  181. package/dist/runtime/commands/release-notes.js +229 -0
  182. package/dist/runtime/commands/repo-map.js +95 -0
  183. package/dist/runtime/commands/report.js +299 -0
  184. package/dist/runtime/commands/resume.js +118 -0
  185. package/dist/runtime/commands/review-consensus.js +17 -2
  186. package/dist/runtime/commands/rewind.js +333 -0
  187. package/dist/runtime/commands/sessions.js +163 -0
  188. package/dist/runtime/commands/share.js +316 -0
  189. package/dist/runtime/commands/status.js +186 -0
  190. package/dist/runtime/commands/stickers.js +82 -0
  191. package/dist/runtime/commands/style.js +194 -0
  192. package/dist/runtime/commands/theme.js +196 -0
  193. package/dist/runtime/commands/undo.js +32 -0
  194. package/dist/runtime/commands/update.js +289 -0
  195. package/dist/runtime/commands/vim.js +140 -0
  196. package/dist/runtime/commands/worktree.js +177 -0
  197. package/dist/runtime/headless-repl.js +195 -0
  198. package/dist/runtime/headless.js +543 -0
  199. package/dist/runtime/load-hooks-or-exit.js +71 -0
  200. package/dist/runtime/plan-decompose.js +531 -0
  201. package/dist/runtime/version.js +65 -0
  202. package/dist/tools/agent-tool.js +229 -0
  203. package/dist/tools/apply-patch.js +556 -0
  204. package/dist/tools/ask-user-question.js +213 -0
  205. package/dist/tools/ask-user.js +115 -0
  206. package/dist/tools/file-tools.js +85 -14
  207. package/dist/tools/lsp-tools.js +189 -0
  208. package/dist/tools/mcp-tool.js +260 -0
  209. package/dist/tools/multi-edit.js +361 -0
  210. package/dist/tools/powershell.js +156 -0
  211. package/dist/tools/registry.js +51 -0
  212. package/dist/tools/skill-tool.js +96 -0
  213. package/dist/tools/tasks.js +208 -0
  214. package/dist/tools/todo-write.js +184 -0
  215. package/dist/tools/web-fetch.js +147 -2
  216. package/dist/tools/web-search.js +458 -0
  217. package/dist/tui/agent-progress-card.js +111 -0
  218. package/dist/tui/agent-tree.js +10 -0
  219. package/dist/tui/ask-modal.js +2 -2
  220. package/dist/tui/ask-user-question-prompt.js +192 -0
  221. package/dist/tui/compact-banner.js +81 -0
  222. package/dist/tui/conversation-pane.js +82 -8
  223. package/dist/tui/cost-table.js +111 -0
  224. package/dist/tui/doctor-table.js +46 -0
  225. package/dist/tui/feedback-prompt.js +156 -0
  226. package/dist/tui/input-box.js +69 -2
  227. package/dist/tui/markdown-render.js +4 -4
  228. package/dist/tui/onboarding-wizard.js +240 -0
  229. package/dist/tui/permissions-picker.js +86 -0
  230. package/dist/tui/render.js +35 -0
  231. package/dist/tui/repl-render.js +303 -13
  232. package/dist/tui/repl-splash.js +2 -2
  233. package/dist/tui/repl.js +72 -14
  234. package/dist/tui/splash.js +1 -1
  235. package/dist/tui/status-bar.js +94 -16
  236. package/dist/tui/status-table.js +7 -0
  237. package/dist/tui/stickers-art.js +136 -0
  238. package/dist/tui/style-table.js +28 -0
  239. package/dist/tui/theme-table.js +29 -0
  240. package/dist/tui/tool-stream-pane.js +52 -3
  241. package/dist/tui/update-banner.js +20 -2
  242. package/dist/tui/vim-input.js +267 -0
  243. package/docs/examples/codegraph.mcp.json +10 -0
  244. package/package.json +12 -6
  245. package/test/scenarios/codegen-create-file.scenario.txt +13 -0
  246. package/test/scenarios/compact-force.scenario.txt +11 -0
  247. package/test/scenarios/identity.scenario.txt +11 -0
  248. package/test/scenarios/persona-handoff.scenario.txt +11 -0
  249. package/test/scenarios/walkback.scenario.txt +12 -0
  250. package/dist/core/engine/compaction-hook.js +0 -154
@@ -0,0 +1,20 @@
1
+ /**
2
+ * Permission gate (Leak L6) public surface.
3
+ *
4
+ * Re-exports the canonical 4-mode types, the tool-class classifier,
5
+ * the dispatch gate, and the workspace + global session-state helpers
6
+ * so callers import from one place:
7
+ *
8
+ * import { gate, resolveMode, PermissionDenied } from '<...>/permissions/index.js';
9
+ *
10
+ * Keeps the internal file split (mode / tool-class / gate / state)
11
+ * invisible to consumers — those files are an implementation detail
12
+ * the engine adapter does not need to know about.
13
+ */
14
+ export { DEFAULT_PERMISSION_MODE, PERMISSION_MODE_GLOSS, PERMISSION_MODES, isPermissionMode, nextPermissionMode, parsePermissionMode, toLegacyMode, } from './mode.js';
15
+ export { classifyAutoMode, listAutoAllowPatterns, listAutoDenyPatterns, } from './auto-classifier.js';
16
+ export { evaluateCircuitBreaker, listCircuitBreakerPatterns, } from './circuit-breaker.js';
17
+ export { getToolClass, listBuiltInToolClasses, } from './tool-class.js';
18
+ export { ASK_OPTIONS, PermissionDenied, applyAskAnswer, createAskAlwaysCache, gate, } from './gate.js';
19
+ export { getCurrentMode, getGlobalDefaultMode, getPreviousMode, globalConfigPath, resolveMode, sessionStatePath, setCurrentMode, setGlobalDefaultMode, setPreviousMode, } from './state.js';
20
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1,174 @@
1
+ /**
2
+ * Permission modes — Wave 7 canonical 6-mode taxonomy (Claude Code parity).
3
+ *
4
+ * Pugi α6 shipped a 4-mode taxonomy (`plan | ask | allow | bypass`) that
5
+ * proved fine для daily use but diverged from Claude Code's 6-mode
6
+ * surface (`default | acceptEdits | plan | auto | dontAsk |
7
+ * bypassPermissions`). Wave 7 Sprint 1 epic #2 closes the parity gap so
8
+ * operators coming from Claude Code see the same mode names + same
9
+ * Shift+Tab cycle.
10
+ *
11
+ * Canonical 6 modes (Wave 7):
12
+ *
13
+ * - `default` — every tool call asks the operator. Safe
14
+ * ground state, replaces α6 `ask`.
15
+ * - `acceptEdits` — auto-allow write/edit on workspace files,
16
+ * ask for everything else (bash, dispatch).
17
+ * Matches CC's "trust file edits только".
18
+ * - `plan` — read-only proposal mode. Write/dispatch
19
+ * refused with deterministic sentinel; the
20
+ * model surfaces a plan, не executes.
21
+ * - `auto` — classifier decides per-call. Phase 1
22
+ * (this PR) ships regex allowlist (safe
23
+ * commands) + regex denylist (destructive).
24
+ * Anything else falls back to ask.
25
+ * - `dontAsk` — auto-allow everything except `permissions.deny`
26
+ * list. Replaces α6 `allow`.
27
+ * - `bypassPermissions` — skip ALL checks including deny list +
28
+ * policy hooks. Has a circuit-breaker for
29
+ * catastrophic patterns (rm -rf /, fork bomb,
30
+ * dd if=/) that refuses regardless of mode.
31
+ * Replaces α6 `bypass`.
32
+ *
33
+ * Backwards-compat aliases: the α6 short names (`ask`, `allow`, `bypass`)
34
+ * map to the new canonical names via `MODE_ALIASES`. `parsePermissionMode`
35
+ * resolves aliases so existing session.json files keep working.
36
+ *
37
+ * Rename mapping (α6 → α7):
38
+ * ask → default
39
+ * allow → dontAsk
40
+ * bypass → bypassPermissions
41
+ * (plan, acceptEdits, auto unchanged / new)
42
+ */
43
+ /**
44
+ * Closed list — used by Shift+Tab cycle (in order), input validation,
45
+ * and slash-command help. Order matches Claude Code's documented
46
+ * Shift+Tab progression: default → acceptEdits → plan → auto → dontAsk
47
+ * → bypassPermissions → wrap к default.
48
+ */
49
+ export const PERMISSION_MODES = Object.freeze([
50
+ 'default',
51
+ 'acceptEdits',
52
+ 'plan',
53
+ 'auto',
54
+ 'dontAsk',
55
+ 'bypassPermissions',
56
+ ]);
57
+ /**
58
+ * Default mode applied when no `--mode` flag, no per-workspace session
59
+ * state, and no `defaultPermissionMode` в `~/.pugi/config.json`. We
60
+ * default cautious (`default` mode = prompt every call) — an operator
61
+ * who has not configured anything is treated as a new operator who
62
+ * deserves visibility into every tool call.
63
+ */
64
+ export const DEFAULT_PERMISSION_MODE = 'default';
65
+ /**
66
+ * Backwards-compat aliases: α6 short names map to α7 canonical names.
67
+ * `parsePermissionMode` consults this table так existing session.json
68
+ * files + scripts that pass `--mode ask` keep working without breaking.
69
+ *
70
+ * Aliases are one-way: persistence writes the canonical name, so a
71
+ * session that started on `ask` is migrated to `default` on next save.
72
+ */
73
+ const MODE_ALIASES = Object.freeze({
74
+ ask: 'default',
75
+ allow: 'dontAsk',
76
+ bypass: 'bypassPermissions',
77
+ });
78
+ /**
79
+ * Type guard для arbitrary string input (CLI flag, session.json
80
+ * deserialization). Returns false for casing variants — caller is
81
+ * expected to lowercase before testing. Aliases are NOT accepted by
82
+ * this predicate; use `parsePermissionMode` for alias resolution.
83
+ */
84
+ export function isPermissionMode(value) {
85
+ return typeof value === 'string' && PERMISSION_MODES.includes(value);
86
+ }
87
+ /**
88
+ * Parse + validate a mode string. Returns null для invalid input so the
89
+ * caller can surface a typed error (`unknown mode: <value>`) instead of
90
+ * throwing from a parse helper.
91
+ *
92
+ * Resolves α6 aliases (`ask` → `default`, `allow` → `dontAsk`,
93
+ * `bypass` → `bypassPermissions`) for backwards compatibility.
94
+ *
95
+ * Case-handling: lowercases for canonical names but matches camelCase
96
+ * names case-insensitively too (так `acceptedits` resolves to
97
+ * `acceptEdits`). The aliases table covers the legacy lowercase tokens.
98
+ */
99
+ export function parsePermissionMode(value) {
100
+ const trimmed = value.trim();
101
+ if (trimmed.length === 0)
102
+ return null;
103
+ // Direct canonical match first (preserves camelCase capitalisation).
104
+ if (isPermissionMode(trimmed))
105
+ return trimmed;
106
+ // Case-insensitive canonical match — operator typed `acceptedits`.
107
+ const lower = trimmed.toLowerCase();
108
+ const canonical = PERMISSION_MODES.find((m) => m.toLowerCase() === lower);
109
+ if (canonical)
110
+ return canonical;
111
+ // α6 alias fallthrough.
112
+ const aliased = MODE_ALIASES[lower];
113
+ if (aliased)
114
+ return aliased;
115
+ return null;
116
+ }
117
+ /**
118
+ * Wave 7 Shift+Tab cycle — advance to the next mode in the canonical
119
+ * order, wrapping from the last back to the first. Pure helper so the
120
+ * TUI binding can call it without re-implementing the cycle logic.
121
+ */
122
+ export function nextPermissionMode(current) {
123
+ const idx = PERMISSION_MODES.indexOf(current);
124
+ if (idx === -1)
125
+ return DEFAULT_PERMISSION_MODE;
126
+ const next = PERMISSION_MODES[(idx + 1) % PERMISSION_MODES.length];
127
+ return next ?? DEFAULT_PERMISSION_MODE;
128
+ }
129
+ /**
130
+ * Map the canonical 6-mode taxonomy to the legacy SDK enum used by
131
+ * `@pugi/sdk::permissionModeSchema`. The SDK enum already contains all
132
+ * 6 names so the map is identity для the modes that align; the two
133
+ * α6-only legacy names (`ask`, `allow`) are not part of the canonical
134
+ * set anymore — `default` and `dontAsk` are их replacements.
135
+ *
136
+ * Callers that need the legacy enum (existing bash classifier, settings
137
+ * persistence) should funnel through this helper so the mapping stays
138
+ * в one place.
139
+ */
140
+ export function toLegacyMode(mode) {
141
+ switch (mode) {
142
+ case 'default':
143
+ // SDK enum doesn't carry `default`; the closest legacy semantic is
144
+ // `ask` (prompt-every-call). Persistence layers that round-trip
145
+ // через the SDK enum get back `ask`, which `parsePermissionMode`
146
+ // re-maps to `default` via the alias table. Round-trip safe.
147
+ return 'ask';
148
+ case 'acceptEdits':
149
+ return 'acceptEdits';
150
+ case 'plan':
151
+ return 'plan';
152
+ case 'auto':
153
+ return 'auto';
154
+ case 'dontAsk':
155
+ return 'dontAsk';
156
+ case 'bypassPermissions':
157
+ return 'bypassPermissions';
158
+ }
159
+ }
160
+ /**
161
+ * One-line human-readable summary surfaced by the `/permissions` table,
162
+ * the Ink picker, and `pugi --help` text. Each line carries a safety
163
+ * hint ("safe-by-default" / "use carefully" / "power-user only") so an
164
+ * operator скимming the picker knows the risk profile at a glance.
165
+ */
166
+ export const PERMISSION_MODE_GLOSS = Object.freeze({
167
+ default: 'Prompt before every tool call. Safe-by-default for new operators.',
168
+ acceptEdits: 'Auto-allow file edit/write; ask for bash + dispatch. Safe-by-default.',
169
+ plan: 'Read-only — propose, never execute. Write + dispatch refused.',
170
+ auto: 'Classifier decides per call (safe regex allowlist; falls back к ask). Use carefully.',
171
+ dontAsk: 'Execute tools without prompts; deny-list still applies. Use carefully.',
172
+ bypassPermissions: 'Skip ALL checks AND policy hooks; circuit-breaker on catastrophic patterns. Power-user only.',
173
+ });
174
+ //# sourceMappingURL=mode.js.map
@@ -0,0 +1,241 @@
1
+ /**
2
+ * Per-workspace permission-mode session state — Leak L6.
3
+ *
4
+ * State lives in `.pugi/session.json` under the workspace root. The
5
+ * file is read on first `getCurrentMode()` call (cached for the
6
+ * process lifetime) and written atomically via tmp+rename on
7
+ * `setCurrentMode()` so a kill mid-write does not corrupt the JSON.
8
+ *
9
+ * Resolution order for the effective mode on a fresh process:
10
+ * 1. CLI flag (`pugi --mode plan`) — passed via `resolveMode` arg;
11
+ * not read from disk here.
12
+ * 2. Workspace session state — `<root>/.pugi/session.json` field
13
+ * `permissionMode`.
14
+ * 3. Global config — `~/.pugi/config.json` field
15
+ * `defaultPermissionMode`.
16
+ * 4. Hard default `ask`.
17
+ *
18
+ * This module owns layers 2 + 3. The CLI arg parser owns layer 1; both
19
+ * funnel into `resolveMode()` which performs the merge.
20
+ */
21
+ import { existsSync, mkdirSync, readFileSync, renameSync, writeFileSync } from 'node:fs';
22
+ import { dirname, resolve } from 'node:path';
23
+ import { homedir } from 'node:os';
24
+ import { z } from 'zod';
25
+ import { DEFAULT_PERMISSION_MODE, parsePermissionMode, } from './mode.js';
26
+ /**
27
+ * Wave 7: zod enum for the canonical 6-mode taxonomy. Includes α6
28
+ * aliases (`ask`, `allow`, `bypass`) as accepted input — Zod parses
29
+ * them, the helpers below remap к canonical names before returning к
30
+ * the caller. Persistence always writes the canonical name so the file
31
+ * migrates forward on next save.
32
+ */
33
+ const permissionModeEnum = z.enum([
34
+ // Canonical Wave 7 names.
35
+ 'default',
36
+ 'acceptEdits',
37
+ 'plan',
38
+ 'auto',
39
+ 'dontAsk',
40
+ 'bypassPermissions',
41
+ // α6 backwards-compat aliases — resolved via parsePermissionMode.
42
+ 'ask',
43
+ 'allow',
44
+ 'bypass',
45
+ ]);
46
+ const sessionStateSchema = z
47
+ .object({
48
+ permissionMode: permissionModeEnum.optional(),
49
+ /**
50
+ * Leak L7: snapshot of the mode that was active immediately BEFORE
51
+ * the operator typed `/plan` (or `/plan <prompt>`). `/plan --back`
52
+ * pops this snapshot and restores it. Cleared after a successful
53
+ * pop so a second `/plan --back` does not double-revert.
54
+ */
55
+ previousPermissionMode: permissionModeEnum.optional(),
56
+ })
57
+ .partial()
58
+ .passthrough();
59
+ const globalConfigSchema = z
60
+ .object({
61
+ defaultPermissionMode: permissionModeEnum.optional(),
62
+ })
63
+ .partial()
64
+ .passthrough();
65
+ const SESSION_FILE = '.pugi/session.json';
66
+ /**
67
+ * Return the path to the workspace session-state file.
68
+ */
69
+ export function sessionStatePath(workspaceRoot) {
70
+ return resolve(workspaceRoot, SESSION_FILE);
71
+ }
72
+ /**
73
+ * Return the path to the user-global config file. Uses HOME env when
74
+ * present (test fixtures, CI) so we never accidentally hit the real
75
+ * user-global file in spec runs.
76
+ */
77
+ export function globalConfigPath(homeDir = homedir()) {
78
+ return resolve(homeDir, '.pugi/config.json');
79
+ }
80
+ /**
81
+ * Read the workspace's saved permission mode. Returns null when the
82
+ * file is absent OR the field is unset; the caller layers in CLI + env
83
+ * + global config defaults to produce the effective mode.
84
+ *
85
+ * Never throws on JSON parse / schema errors — a malformed session
86
+ * file should not break the gate. The defensive `try/catch` returns
87
+ * null and lets the caller fall through to the next layer.
88
+ */
89
+ export function getCurrentMode(workspaceRoot) {
90
+ const path = sessionStatePath(workspaceRoot);
91
+ if (!existsSync(path))
92
+ return null;
93
+ try {
94
+ const raw = readFileSync(path, 'utf8');
95
+ const parsed = sessionStateSchema.parse(JSON.parse(raw));
96
+ if (typeof parsed.permissionMode !== 'string')
97
+ return null;
98
+ // Wave 7: parsePermissionMode resolves α6 aliases (`ask`, `allow`,
99
+ // `bypass`) to their canonical Wave 7 names. A session file written
100
+ // by α6.x is silently upgraded on read.
101
+ return parsePermissionMode(parsed.permissionMode);
102
+ }
103
+ catch {
104
+ return null;
105
+ }
106
+ }
107
+ /**
108
+ * Persist the workspace's permission mode. Creates the `.pugi/` dir
109
+ * when missing; preserves any unrelated keys in the file (passthrough
110
+ * schema). Atomic tmp+rename so a kill mid-write does not corrupt the
111
+ * JSON.
112
+ */
113
+ export function setCurrentMode(workspaceRoot, mode) {
114
+ const path = sessionStatePath(workspaceRoot);
115
+ mkdirSync(dirname(path), { recursive: true });
116
+ const existing = existsSync(path)
117
+ ? safeParseObject(readFileSync(path, 'utf8'))
118
+ : {};
119
+ const next = { ...existing, permissionMode: mode };
120
+ const tmpPath = `${path}.tmp`;
121
+ writeFileSync(tmpPath, `${JSON.stringify(next, null, 2)}\n`, { encoding: 'utf8', mode: 0o600 });
122
+ renameSync(tmpPath, path);
123
+ }
124
+ /**
125
+ * Leak L7 — read the snapshot of the mode that was active before the
126
+ * most-recent `/plan` (or `pugi plan`) entry. Returns null when the
127
+ * file is absent OR the field is unset. Same defensive behaviour as
128
+ * `getCurrentMode`: a malformed session file never breaks the slash
129
+ * command — the worst case is `/plan --back` reports "no previous
130
+ * mode to restore" and the operator picks the target mode explicitly.
131
+ */
132
+ export function getPreviousMode(workspaceRoot) {
133
+ const path = sessionStatePath(workspaceRoot);
134
+ if (!existsSync(path))
135
+ return null;
136
+ try {
137
+ const raw = readFileSync(path, 'utf8');
138
+ const parsed = sessionStateSchema.parse(JSON.parse(raw));
139
+ if (typeof parsed.previousPermissionMode !== 'string')
140
+ return null;
141
+ return parsePermissionMode(parsed.previousPermissionMode);
142
+ }
143
+ catch {
144
+ return null;
145
+ }
146
+ }
147
+ /**
148
+ * Leak L7 — record the mode that was active immediately before the
149
+ * operator switched to plan. The runtime calls this AT `/plan` entry
150
+ * with the current mode (whatever `resolveMode` returned). Atomic
151
+ * tmp+rename keeps the snapshot consistent if the process is killed
152
+ * mid-write. Pass `null` to clear the snapshot (used after a
153
+ * successful `/plan --back` so a second `--back` does not loop).
154
+ */
155
+ export function setPreviousMode(workspaceRoot, mode) {
156
+ const path = sessionStatePath(workspaceRoot);
157
+ mkdirSync(dirname(path), { recursive: true });
158
+ const existing = existsSync(path)
159
+ ? safeParseObject(readFileSync(path, 'utf8'))
160
+ : {};
161
+ const next = { ...existing };
162
+ if (mode === null) {
163
+ delete next.previousPermissionMode;
164
+ }
165
+ else {
166
+ next.previousPermissionMode = mode;
167
+ }
168
+ const tmpPath = `${path}.tmp`;
169
+ writeFileSync(tmpPath, `${JSON.stringify(next, null, 2)}\n`, { encoding: 'utf8', mode: 0o600 });
170
+ renameSync(tmpPath, path);
171
+ }
172
+ /**
173
+ * Read `~/.pugi/config.json::defaultPermissionMode`. Returns null when
174
+ * the file is absent / the field is unset; same defensive behaviour
175
+ * as `getCurrentMode` — a malformed global config never breaks the gate.
176
+ */
177
+ export function getGlobalDefaultMode(homeDir = homedir()) {
178
+ const path = globalConfigPath(homeDir);
179
+ if (!existsSync(path))
180
+ return null;
181
+ try {
182
+ const raw = readFileSync(path, 'utf8');
183
+ const parsed = globalConfigSchema.parse(JSON.parse(raw));
184
+ if (typeof parsed.defaultPermissionMode !== 'string')
185
+ return null;
186
+ return parsePermissionMode(parsed.defaultPermissionMode);
187
+ }
188
+ catch {
189
+ return null;
190
+ }
191
+ }
192
+ /**
193
+ * Persist `~/.pugi/config.json::defaultPermissionMode`. Used by the
194
+ * `/permissions <mode> --persist` flow so a future fresh session
195
+ * defaults to the same mode without an explicit `--mode` flag.
196
+ */
197
+ export function setGlobalDefaultMode(mode, homeDir = homedir()) {
198
+ const path = globalConfigPath(homeDir);
199
+ mkdirSync(dirname(path), { recursive: true });
200
+ const existing = existsSync(path)
201
+ ? safeParseObject(readFileSync(path, 'utf8'))
202
+ : {};
203
+ const next = { ...existing, defaultPermissionMode: mode };
204
+ const tmpPath = `${path}.tmp`;
205
+ writeFileSync(tmpPath, `${JSON.stringify(next, null, 2)}\n`, { encoding: 'utf8', mode: 0o600 });
206
+ renameSync(tmpPath, path);
207
+ }
208
+ export function resolveMode(options) {
209
+ if (options.cliFlag) {
210
+ const flag = parsePermissionMode(options.cliFlag);
211
+ if (flag)
212
+ return flag;
213
+ }
214
+ const workspace = getCurrentMode(options.workspaceRoot);
215
+ if (workspace)
216
+ return workspace;
217
+ const global = getGlobalDefaultMode(options.homeDir);
218
+ if (global)
219
+ return global;
220
+ return DEFAULT_PERMISSION_MODE;
221
+ }
222
+ /**
223
+ * Defensive helper — parse JSON to an object; non-object payload (top-
224
+ * level array, primitive) collapses to an empty object so the merge
225
+ * doesn't surface a TypeError. The `setCurrentMode` / `setGlobalDefaultMode`
226
+ * helpers only write objects, so a non-object existing file is corrupted
227
+ * and we explicitly reset it rather than appending into a non-object.
228
+ */
229
+ function safeParseObject(raw) {
230
+ try {
231
+ const parsed = JSON.parse(raw);
232
+ if (parsed && typeof parsed === 'object' && !Array.isArray(parsed)) {
233
+ return parsed;
234
+ }
235
+ return {};
236
+ }
237
+ catch {
238
+ return {};
239
+ }
240
+ }
241
+ //# sourceMappingURL=state.js.map
@@ -0,0 +1,93 @@
1
+ /**
2
+ * Tool side-effect classification — Leak L6.
3
+ *
4
+ * Three classes drive the canonical 4-mode permission gate:
5
+ *
6
+ * - `read` — observe-only. Plan mode allows; ask still prompts;
7
+ * allow + bypass execute silently. Examples: read,
8
+ * grep, glob, web_fetch, web_search, skills_list.
9
+ * - `write` — mutates workspace, journal, or operator screen with
10
+ * visible side effects. Plan mode refuses; ask prompts;
11
+ * allow + bypass execute. Examples: write, edit, bash,
12
+ * multi_edit, task_*. `ask_user_question` is also
13
+ * classed as `write` because it interrupts the
14
+ * dispatcher's flow control and demands operator
15
+ * attention — plan mode should not prompt operators.
16
+ * - `dispatch` — spawns a child subagent or off-tree task. Plan mode
17
+ * refuses (a write-capable child violates plan-mode's
18
+ * read-only contract); ask prompts; allow + bypass
19
+ * execute. Example: `agent`.
20
+ *
21
+ * Unknown tool names default to `write` — deny-first safety. A stale
22
+ * schema entry that the gate has not been told about should not silently
23
+ * pass in plan mode just because the gate doesn't recognise it.
24
+ */
25
+ /**
26
+ * Closed map of every built-in tool name -> side-effect class. The
27
+ * source of truth for the four standard modes; mirrored against the
28
+ * `WIRED_TOOLS` set in `core/engine/tool-bridge.ts` so an unrecognised
29
+ * tool surfaces as the safe deny-first `write` default.
30
+ *
31
+ * MCP tools follow the `mcp__<server>__<tool>` namespace and are
32
+ * uniformly classed via `getToolClass` because per-tool annotations are
33
+ * not yet a part of the MCP spec — treating them as `write` is the
34
+ * conservative default until server-side metadata is trustworthy.
35
+ */
36
+ const BUILT_IN_TOOL_CLASSES = Object.freeze({
37
+ // Read-only observations.
38
+ read: 'read',
39
+ grep: 'read',
40
+ glob: 'read',
41
+ ls: 'read',
42
+ search: 'read',
43
+ web_fetch: 'read',
44
+ web_search: 'read',
45
+ file_cache_check: 'read',
46
+ skills_list: 'read',
47
+ skill: 'read',
48
+ task_get: 'read',
49
+ task_list: 'read',
50
+ // Mutating actions.
51
+ write: 'write',
52
+ edit: 'write',
53
+ multi_edit: 'write',
54
+ bash: 'write',
55
+ task_create: 'write',
56
+ task_update: 'write',
57
+ todo_write: 'write',
58
+ // `ask_user_question` halts the loop and demands operator attention.
59
+ // Plan mode should not interrupt — class as write so the gate refuses
60
+ // it in plan mode but ask + allow + bypass execute normally.
61
+ ask_user_question: 'write',
62
+ // Dispatch — spawn a child agent. Refused in plan mode regardless of
63
+ // the child's role tier (the engine adapter applies role-based
64
+ // capability filtering, but the gate refuses dispatch up front so a
65
+ // plan-mode session cannot leak a writeable child).
66
+ agent: 'dispatch',
67
+ pugi_delegate: 'dispatch',
68
+ sub_agent_spawn: 'dispatch',
69
+ });
70
+ const MCP_TOOL_PREFIX = 'mcp__';
71
+ /**
72
+ * Resolve the class for a tool name. Unknown names default to `write`
73
+ * (deny-first). MCP tools (any name prefixed with `mcp__`) default to
74
+ * `write` for the same conservative reason — the MCP spec lacks
75
+ * per-tool annotations today.
76
+ */
77
+ export function getToolClass(toolName) {
78
+ const builtIn = BUILT_IN_TOOL_CLASSES[toolName];
79
+ if (builtIn)
80
+ return builtIn;
81
+ if (toolName.startsWith(MCP_TOOL_PREFIX))
82
+ return 'write';
83
+ return 'write';
84
+ }
85
+ /**
86
+ * Expose the built-in class map for diagnostic surfaces (`pugi doctor`,
87
+ * test fixtures). Caller MUST NOT mutate — the object is already frozen
88
+ * so any attempt throws in strict mode.
89
+ */
90
+ export function listBuiltInToolClasses() {
91
+ return BUILT_IN_TOOL_CLASSES;
92
+ }
93
+ //# sourceMappingURL=tool-class.js.map