chad-code 1.3.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.
Files changed (338) hide show
  1. package/AGENTS.md +27 -0
  2. package/Dockerfile +18 -0
  3. package/README.md +15 -0
  4. package/README.npm.md +64 -0
  5. package/bin/chad-code +84 -0
  6. package/bunfig.toml +7 -0
  7. package/eslint.config.js +29 -0
  8. package/package.json +107 -0
  9. package/parsers-config.ts +253 -0
  10. package/script/build.ts +167 -0
  11. package/script/postinstall.mjs +122 -0
  12. package/script/publish-registries.ts +187 -0
  13. package/script/publish.ts +93 -0
  14. package/script/schema.ts +47 -0
  15. package/src/acp/README.md +164 -0
  16. package/src/acp/agent.ts +1086 -0
  17. package/src/acp/session.ts +101 -0
  18. package/src/acp/types.ts +22 -0
  19. package/src/agent/agent.ts +253 -0
  20. package/src/agent/generate.txt +75 -0
  21. package/src/agent/prompt/compaction.txt +12 -0
  22. package/src/agent/prompt/explore.txt +18 -0
  23. package/src/agent/prompt/summary.txt +11 -0
  24. package/src/agent/prompt/title.txt +36 -0
  25. package/src/auth/index.ts +70 -0
  26. package/src/bun/index.ts +130 -0
  27. package/src/bus/bus-event.ts +43 -0
  28. package/src/bus/global.ts +10 -0
  29. package/src/bus/index.ts +105 -0
  30. package/src/cli/bootstrap.ts +17 -0
  31. package/src/cli/cmd/acp.ts +69 -0
  32. package/src/cli/cmd/agent.ts +257 -0
  33. package/src/cli/cmd/auth.ts +132 -0
  34. package/src/cli/cmd/cmd.ts +7 -0
  35. package/src/cli/cmd/debug/agent.ts +28 -0
  36. package/src/cli/cmd/debug/config.ts +15 -0
  37. package/src/cli/cmd/debug/file.ts +91 -0
  38. package/src/cli/cmd/debug/index.ts +45 -0
  39. package/src/cli/cmd/debug/lsp.ts +48 -0
  40. package/src/cli/cmd/debug/ripgrep.ts +83 -0
  41. package/src/cli/cmd/debug/scrap.ts +15 -0
  42. package/src/cli/cmd/debug/skill.ts +15 -0
  43. package/src/cli/cmd/debug/snapshot.ts +48 -0
  44. package/src/cli/cmd/export.ts +88 -0
  45. package/src/cli/cmd/generate.ts +38 -0
  46. package/src/cli/cmd/github.ts +32 -0
  47. package/src/cli/cmd/import.ts +98 -0
  48. package/src/cli/cmd/mcp.ts +670 -0
  49. package/src/cli/cmd/models.ts +42 -0
  50. package/src/cli/cmd/pr.ts +112 -0
  51. package/src/cli/cmd/run.ts +374 -0
  52. package/src/cli/cmd/serve.ts +16 -0
  53. package/src/cli/cmd/session.ts +135 -0
  54. package/src/cli/cmd/stats.ts +402 -0
  55. package/src/cli/cmd/tui/app.tsx +705 -0
  56. package/src/cli/cmd/tui/attach.ts +32 -0
  57. package/src/cli/cmd/tui/component/border.tsx +21 -0
  58. package/src/cli/cmd/tui/component/dialog-agent.tsx +31 -0
  59. package/src/cli/cmd/tui/component/dialog-command.tsx +124 -0
  60. package/src/cli/cmd/tui/component/dialog-mcp.tsx +86 -0
  61. package/src/cli/cmd/tui/component/dialog-model.tsx +232 -0
  62. package/src/cli/cmd/tui/component/dialog-provider.tsx +228 -0
  63. package/src/cli/cmd/tui/component/dialog-session-list.tsx +115 -0
  64. package/src/cli/cmd/tui/component/dialog-session-rename.tsx +31 -0
  65. package/src/cli/cmd/tui/component/dialog-stash.tsx +86 -0
  66. package/src/cli/cmd/tui/component/dialog-status.tsx +162 -0
  67. package/src/cli/cmd/tui/component/dialog-tag.tsx +44 -0
  68. package/src/cli/cmd/tui/component/dialog-theme-list.tsx +50 -0
  69. package/src/cli/cmd/tui/component/did-you-know.tsx +85 -0
  70. package/src/cli/cmd/tui/component/logo.tsx +43 -0
  71. package/src/cli/cmd/tui/component/prompt/autocomplete.tsx +654 -0
  72. package/src/cli/cmd/tui/component/prompt/history.tsx +108 -0
  73. package/src/cli/cmd/tui/component/prompt/index.tsx +1078 -0
  74. package/src/cli/cmd/tui/component/prompt/stash.tsx +101 -0
  75. package/src/cli/cmd/tui/component/textarea-keybindings.ts +73 -0
  76. package/src/cli/cmd/tui/component/tips.ts +92 -0
  77. package/src/cli/cmd/tui/component/todo-item.tsx +32 -0
  78. package/src/cli/cmd/tui/context/args.tsx +14 -0
  79. package/src/cli/cmd/tui/context/directory.ts +13 -0
  80. package/src/cli/cmd/tui/context/exit.tsx +23 -0
  81. package/src/cli/cmd/tui/context/helper.tsx +25 -0
  82. package/src/cli/cmd/tui/context/keybind.tsx +101 -0
  83. package/src/cli/cmd/tui/context/kv.tsx +49 -0
  84. package/src/cli/cmd/tui/context/local.tsx +392 -0
  85. package/src/cli/cmd/tui/context/prompt.tsx +18 -0
  86. package/src/cli/cmd/tui/context/route.tsx +46 -0
  87. package/src/cli/cmd/tui/context/sdk.tsx +75 -0
  88. package/src/cli/cmd/tui/context/sync.tsx +384 -0
  89. package/src/cli/cmd/tui/context/theme/aura.json +69 -0
  90. package/src/cli/cmd/tui/context/theme/ayu.json +80 -0
  91. package/src/cli/cmd/tui/context/theme/catppuccin-frappe.json +233 -0
  92. package/src/cli/cmd/tui/context/theme/catppuccin-macchiato.json +233 -0
  93. package/src/cli/cmd/tui/context/theme/catppuccin.json +112 -0
  94. package/src/cli/cmd/tui/context/theme/chad.json +245 -0
  95. package/src/cli/cmd/tui/context/theme/cobalt2.json +228 -0
  96. package/src/cli/cmd/tui/context/theme/cursor.json +249 -0
  97. package/src/cli/cmd/tui/context/theme/dracula.json +219 -0
  98. package/src/cli/cmd/tui/context/theme/everforest.json +241 -0
  99. package/src/cli/cmd/tui/context/theme/flexoki.json +237 -0
  100. package/src/cli/cmd/tui/context/theme/github.json +233 -0
  101. package/src/cli/cmd/tui/context/theme/gruvbox.json +95 -0
  102. package/src/cli/cmd/tui/context/theme/kanagawa.json +77 -0
  103. package/src/cli/cmd/tui/context/theme/lucent-orng.json +227 -0
  104. package/src/cli/cmd/tui/context/theme/material.json +235 -0
  105. package/src/cli/cmd/tui/context/theme/matrix.json +77 -0
  106. package/src/cli/cmd/tui/context/theme/mercury.json +252 -0
  107. package/src/cli/cmd/tui/context/theme/monokai.json +221 -0
  108. package/src/cli/cmd/tui/context/theme/nightowl.json +221 -0
  109. package/src/cli/cmd/tui/context/theme/nord.json +223 -0
  110. package/src/cli/cmd/tui/context/theme/one-dark.json +84 -0
  111. package/src/cli/cmd/tui/context/theme/orng.json +245 -0
  112. package/src/cli/cmd/tui/context/theme/osaka-jade.json +93 -0
  113. package/src/cli/cmd/tui/context/theme/palenight.json +222 -0
  114. package/src/cli/cmd/tui/context/theme/rosepine.json +234 -0
  115. package/src/cli/cmd/tui/context/theme/solarized.json +223 -0
  116. package/src/cli/cmd/tui/context/theme/synthwave84.json +226 -0
  117. package/src/cli/cmd/tui/context/theme/tokyonight.json +243 -0
  118. package/src/cli/cmd/tui/context/theme/vercel.json +245 -0
  119. package/src/cli/cmd/tui/context/theme/vesper.json +218 -0
  120. package/src/cli/cmd/tui/context/theme/zenburn.json +223 -0
  121. package/src/cli/cmd/tui/context/theme.tsx +1137 -0
  122. package/src/cli/cmd/tui/event.ts +46 -0
  123. package/src/cli/cmd/tui/routes/home.tsx +138 -0
  124. package/src/cli/cmd/tui/routes/session/dialog-fork-from-timeline.tsx +64 -0
  125. package/src/cli/cmd/tui/routes/session/dialog-message.tsx +109 -0
  126. package/src/cli/cmd/tui/routes/session/dialog-subagent.tsx +26 -0
  127. package/src/cli/cmd/tui/routes/session/dialog-timeline.tsx +47 -0
  128. package/src/cli/cmd/tui/routes/session/footer.tsx +88 -0
  129. package/src/cli/cmd/tui/routes/session/header.tsx +125 -0
  130. package/src/cli/cmd/tui/routes/session/index.tsx +1814 -0
  131. package/src/cli/cmd/tui/routes/session/permission.tsx +416 -0
  132. package/src/cli/cmd/tui/routes/session/sidebar.tsx +318 -0
  133. package/src/cli/cmd/tui/spawn.ts +48 -0
  134. package/src/cli/cmd/tui/thread.ts +111 -0
  135. package/src/cli/cmd/tui/ui/dialog-alert.tsx +57 -0
  136. package/src/cli/cmd/tui/ui/dialog-confirm.tsx +83 -0
  137. package/src/cli/cmd/tui/ui/dialog-export-options.tsx +204 -0
  138. package/src/cli/cmd/tui/ui/dialog-help.tsx +38 -0
  139. package/src/cli/cmd/tui/ui/dialog-prompt.tsx +77 -0
  140. package/src/cli/cmd/tui/ui/dialog-select.tsx +345 -0
  141. package/src/cli/cmd/tui/ui/dialog.tsx +171 -0
  142. package/src/cli/cmd/tui/ui/link.tsx +28 -0
  143. package/src/cli/cmd/tui/ui/spinner.ts +368 -0
  144. package/src/cli/cmd/tui/ui/toast.tsx +100 -0
  145. package/src/cli/cmd/tui/util/clipboard.ts +127 -0
  146. package/src/cli/cmd/tui/util/editor.ts +32 -0
  147. package/src/cli/cmd/tui/util/signal.ts +7 -0
  148. package/src/cli/cmd/tui/util/terminal.ts +114 -0
  149. package/src/cli/cmd/tui/util/transcript.ts +98 -0
  150. package/src/cli/cmd/tui/worker.ts +68 -0
  151. package/src/cli/cmd/uninstall.ts +344 -0
  152. package/src/cli/cmd/upgrade.ts +67 -0
  153. package/src/cli/cmd/web.ts +73 -0
  154. package/src/cli/error.ts +56 -0
  155. package/src/cli/network.ts +53 -0
  156. package/src/cli/ui.ts +87 -0
  157. package/src/cli/upgrade.ts +25 -0
  158. package/src/command/index.ts +131 -0
  159. package/src/command/template/initialize.txt +10 -0
  160. package/src/command/template/review.txt +97 -0
  161. package/src/config/config.ts +1124 -0
  162. package/src/config/markdown.ts +41 -0
  163. package/src/env/index.ts +26 -0
  164. package/src/file/ignore.ts +83 -0
  165. package/src/file/index.ts +411 -0
  166. package/src/file/ripgrep.ts +402 -0
  167. package/src/file/time.ts +64 -0
  168. package/src/file/watcher.ts +117 -0
  169. package/src/flag/flag.ts +52 -0
  170. package/src/format/formatter.ts +359 -0
  171. package/src/format/index.ts +137 -0
  172. package/src/global/index.ts +55 -0
  173. package/src/id/id.ts +73 -0
  174. package/src/ide/index.ts +77 -0
  175. package/src/index.ts +159 -0
  176. package/src/installation/index.ts +198 -0
  177. package/src/lsp/client.ts +252 -0
  178. package/src/lsp/index.ts +485 -0
  179. package/src/lsp/language.ts +119 -0
  180. package/src/lsp/server.ts +2023 -0
  181. package/src/mcp/auth.ts +135 -0
  182. package/src/mcp/index.ts +874 -0
  183. package/src/mcp/oauth-callback.ts +200 -0
  184. package/src/mcp/oauth-provider.ts +154 -0
  185. package/src/patch/index.ts +622 -0
  186. package/src/permission/arity.ts +163 -0
  187. package/src/permission/index.ts +210 -0
  188. package/src/permission/next.ts +268 -0
  189. package/src/plugin/index.ts +106 -0
  190. package/src/project/bootstrap.ts +31 -0
  191. package/src/project/instance.ts +78 -0
  192. package/src/project/project.ts +263 -0
  193. package/src/project/state.ts +65 -0
  194. package/src/project/vcs.ts +76 -0
  195. package/src/provider/auth.ts +143 -0
  196. package/src/provider/models-macro.ts +4 -0
  197. package/src/provider/models.ts +77 -0
  198. package/src/provider/provider.ts +516 -0
  199. package/src/provider/transform.ts +114 -0
  200. package/src/pty/index.ts +212 -0
  201. package/src/server/error.ts +36 -0
  202. package/src/server/mdns.ts +57 -0
  203. package/src/server/project.ts +79 -0
  204. package/src/server/server.ts +2866 -0
  205. package/src/server/tui.ts +71 -0
  206. package/src/session/compaction.ts +225 -0
  207. package/src/session/index.ts +469 -0
  208. package/src/session/llm.ts +213 -0
  209. package/src/session/message-v2.ts +742 -0
  210. package/src/session/message.ts +189 -0
  211. package/src/session/processor.ts +402 -0
  212. package/src/session/prompt/anthropic-20250930.txt +166 -0
  213. package/src/session/prompt/anthropic.txt +105 -0
  214. package/src/session/prompt/anthropic_spoof.txt +1 -0
  215. package/src/session/prompt/beast.txt +147 -0
  216. package/src/session/prompt/build-switch.txt +5 -0
  217. package/src/session/prompt/codex.txt +318 -0
  218. package/src/session/prompt/copilot-gpt-5.txt +143 -0
  219. package/src/session/prompt/gemini.txt +155 -0
  220. package/src/session/prompt/max-steps.txt +16 -0
  221. package/src/session/prompt/plan-reminder-anthropic.txt +67 -0
  222. package/src/session/prompt/plan.txt +26 -0
  223. package/src/session/prompt/qwen.txt +109 -0
  224. package/src/session/prompt.ts +1621 -0
  225. package/src/session/retry.ts +90 -0
  226. package/src/session/revert.ts +108 -0
  227. package/src/session/status.ts +76 -0
  228. package/src/session/summary.ts +194 -0
  229. package/src/session/system.ts +108 -0
  230. package/src/session/todo.ts +37 -0
  231. package/src/share/share-next.ts +194 -0
  232. package/src/share/share.ts +23 -0
  233. package/src/shell/shell.ts +67 -0
  234. package/src/skill/index.ts +1 -0
  235. package/src/skill/skill.ts +124 -0
  236. package/src/snapshot/index.ts +197 -0
  237. package/src/storage/storage.ts +226 -0
  238. package/src/tool/bash.ts +262 -0
  239. package/src/tool/bash.txt +116 -0
  240. package/src/tool/batch.ts +175 -0
  241. package/src/tool/batch.txt +24 -0
  242. package/src/tool/codesearch.ts +132 -0
  243. package/src/tool/codesearch.txt +12 -0
  244. package/src/tool/edit.ts +655 -0
  245. package/src/tool/edit.txt +10 -0
  246. package/src/tool/glob.ts +75 -0
  247. package/src/tool/glob.txt +6 -0
  248. package/src/tool/grep.ts +132 -0
  249. package/src/tool/grep.txt +8 -0
  250. package/src/tool/invalid.ts +17 -0
  251. package/src/tool/ls.ts +119 -0
  252. package/src/tool/ls.txt +1 -0
  253. package/src/tool/lsp.ts +94 -0
  254. package/src/tool/lsp.txt +19 -0
  255. package/src/tool/multiedit.ts +46 -0
  256. package/src/tool/multiedit.txt +41 -0
  257. package/src/tool/patch.ts +210 -0
  258. package/src/tool/patch.txt +1 -0
  259. package/src/tool/read.ts +191 -0
  260. package/src/tool/read.txt +12 -0
  261. package/src/tool/registry.ts +137 -0
  262. package/src/tool/skill.ts +77 -0
  263. package/src/tool/task.ts +167 -0
  264. package/src/tool/task.txt +60 -0
  265. package/src/tool/todo.ts +53 -0
  266. package/src/tool/todoread.txt +14 -0
  267. package/src/tool/todowrite.txt +167 -0
  268. package/src/tool/tool.ts +73 -0
  269. package/src/tool/webfetch.ts +182 -0
  270. package/src/tool/webfetch.txt +13 -0
  271. package/src/tool/websearch.ts +144 -0
  272. package/src/tool/websearch.txt +11 -0
  273. package/src/tool/write.ts +84 -0
  274. package/src/tool/write.txt +8 -0
  275. package/src/util/archive.ts +16 -0
  276. package/src/util/color.ts +19 -0
  277. package/src/util/context.ts +25 -0
  278. package/src/util/defer.ts +12 -0
  279. package/src/util/eventloop.ts +20 -0
  280. package/src/util/filesystem.ts +83 -0
  281. package/src/util/fn.ts +11 -0
  282. package/src/util/iife.ts +3 -0
  283. package/src/util/keybind.ts +102 -0
  284. package/src/util/lazy.ts +18 -0
  285. package/src/util/locale.ts +81 -0
  286. package/src/util/lock.ts +98 -0
  287. package/src/util/log.ts +180 -0
  288. package/src/util/queue.ts +32 -0
  289. package/src/util/rpc.ts +42 -0
  290. package/src/util/scrap.ts +10 -0
  291. package/src/util/signal.ts +12 -0
  292. package/src/util/timeout.ts +14 -0
  293. package/src/util/token.ts +7 -0
  294. package/src/util/wildcard.ts +54 -0
  295. package/src/worktree/index.ts +217 -0
  296. package/sst-env.d.ts +9 -0
  297. package/test/agent/agent.test.ts +448 -0
  298. package/test/bun.test.ts +53 -0
  299. package/test/cli/github-action.test.ts +129 -0
  300. package/test/cli/github-remote.test.ts +80 -0
  301. package/test/cli/tui/transcript.test.ts +297 -0
  302. package/test/config/agent-color.test.ts +66 -0
  303. package/test/config/config.test.ts +870 -0
  304. package/test/config/markdown.test.ts +89 -0
  305. package/test/file/ignore.test.ts +10 -0
  306. package/test/file/path-traversal.test.ts +115 -0
  307. package/test/fixture/fixture.ts +45 -0
  308. package/test/fixture/lsp/fake-lsp-server.js +77 -0
  309. package/test/ide/ide.test.ts +82 -0
  310. package/test/keybind.test.ts +421 -0
  311. package/test/lsp/client.test.ts +95 -0
  312. package/test/mcp/headers.test.ts +153 -0
  313. package/test/patch/patch.test.ts +348 -0
  314. package/test/permission/arity.test.ts +33 -0
  315. package/test/permission/next.test.ts +652 -0
  316. package/test/preload.ts +63 -0
  317. package/test/project/project.test.ts +120 -0
  318. package/test/provider/amazon-bedrock.test.ts +236 -0
  319. package/test/provider/provider.test.ts +2127 -0
  320. package/test/provider/transform.test.ts +980 -0
  321. package/test/server/session-select.test.ts +78 -0
  322. package/test/session/compaction.test.ts +251 -0
  323. package/test/session/message-v2.test.ts +570 -0
  324. package/test/session/retry.test.ts +131 -0
  325. package/test/session/revert-compact.test.ts +285 -0
  326. package/test/session/session.test.ts +71 -0
  327. package/test/skill/skill.test.ts +185 -0
  328. package/test/snapshot/snapshot.test.ts +939 -0
  329. package/test/tool/__snapshots__/tool.test.ts.snap +9 -0
  330. package/test/tool/bash.test.ts +232 -0
  331. package/test/tool/grep.test.ts +109 -0
  332. package/test/tool/patch.test.ts +261 -0
  333. package/test/tool/read.test.ts +167 -0
  334. package/test/util/iife.test.ts +36 -0
  335. package/test/util/lazy.test.ts +50 -0
  336. package/test/util/timeout.test.ts +21 -0
  337. package/test/util/wildcard.test.ts +55 -0
  338. package/tsconfig.json +16 -0
@@ -0,0 +1,163 @@
1
+ export namespace BashArity {
2
+ export function prefix(tokens: string[]) {
3
+ for (let len = tokens.length; len > 0; len--) {
4
+ const prefix = tokens.slice(0, len).join(" ")
5
+ const arity = ARITY[prefix]
6
+ if (arity !== undefined) return tokens.slice(0, arity)
7
+ }
8
+ if (tokens.length === 0) return []
9
+ return tokens.slice(0, 1)
10
+ }
11
+
12
+ /* Generated with following prompt:
13
+ You are generating a dictionary of command-prefix arities for bash-style commands.
14
+ This dictionary is used to identify the "human-understandable command" from an input shell command.### **RULES (follow strictly)**1. Each entry maps a **command prefix string → number**, representing how many **tokens** define the command.
15
+ 2. **Flags NEVER count as tokens**. Only subcommands count.
16
+ 3. **Longest matching prefix wins**.
17
+ 4. **Only include a longer prefix if its arity is different from what the shorter prefix already implies**. * Example: If `git` is 2, then do **not** include `git checkout`, `git commit`, etc. unless they require *different* arity.
18
+ 5. The output must be a **single JSON object**. Each entry should have a comment with an example real world matching command. DO NOT MAKE ANY OTHER COMMENTS. Should be alphabetical
19
+ 6. Include the **most commonly used commands** across many stacks and languages. More is better.### **Semantics examples*** `touch foo.txt` → `touch` (arity 1, explicitly listed)
20
+ * `git checkout main` → `git checkout` (because `git` has arity 2)
21
+ * `npm install` → `npm install` (because `npm` has arity 2)
22
+ * `npm run dev` → `npm run dev` (because `npm run` has arity 3)
23
+ * `python script.py` → `python script.py` (default: whole input, not in dictionary)### **Now generate the dictionary.**
24
+ */
25
+ const ARITY: Record<string, number> = {
26
+ cat: 1, // cat file.txt
27
+ cd: 1, // cd /path/to/dir
28
+ chmod: 1, // chmod 755 script.sh
29
+ chown: 1, // chown user:group file.txt
30
+ cp: 1, // cp source.txt dest.txt
31
+ echo: 1, // echo "hello world"
32
+ env: 1, // env
33
+ export: 1, // export PATH=/usr/bin
34
+ grep: 1, // grep pattern file.txt
35
+ kill: 1, // kill 1234
36
+ killall: 1, // killall process
37
+ ln: 1, // ln -s source target
38
+ ls: 1, // ls -la
39
+ mkdir: 1, // mkdir new-dir
40
+ mv: 1, // mv old.txt new.txt
41
+ ps: 1, // ps aux
42
+ pwd: 1, // pwd
43
+ rm: 1, // rm file.txt
44
+ rmdir: 1, // rmdir empty-dir
45
+ sleep: 1, // sleep 5
46
+ source: 1, // source ~/.bashrc
47
+ tail: 1, // tail -f log.txt
48
+ touch: 1, // touch file.txt
49
+ unset: 1, // unset VAR
50
+ which: 1, // which node
51
+ aws: 3, // aws s3 ls
52
+ az: 3, // az storage blob list
53
+ bazel: 2, // bazel build
54
+ brew: 2, // brew install node
55
+ bun: 2, // bun install
56
+ "bun run": 3, // bun run dev
57
+ "bun x": 3, // bun x vite
58
+ cargo: 2, // cargo build
59
+ "cargo add": 3, // cargo add tokio
60
+ "cargo run": 3, // cargo run main
61
+ cdk: 2, // cdk deploy
62
+ cf: 2, // cf push app
63
+ cmake: 2, // cmake build
64
+ composer: 2, // composer require laravel
65
+ consul: 2, // consul members
66
+ "consul kv": 3, // consul kv get config/app
67
+ crictl: 2, // crictl ps
68
+ deno: 2, // deno run server.ts
69
+ "deno task": 3, // deno task dev
70
+ doctl: 3, // doctl kubernetes cluster list
71
+ docker: 2, // docker run nginx
72
+ "docker builder": 3, // docker builder prune
73
+ "docker compose": 3, // docker compose up
74
+ "docker container": 3, // docker container ls
75
+ "docker image": 3, // docker image prune
76
+ "docker network": 3, // docker network inspect
77
+ "docker volume": 3, // docker volume ls
78
+ eksctl: 2, // eksctl get clusters
79
+ "eksctl create": 3, // eksctl create cluster
80
+ firebase: 2, // firebase deploy
81
+ flyctl: 2, // flyctl deploy
82
+ gcloud: 3, // gcloud compute instances list
83
+ gh: 3, // gh pr list
84
+ git: 2, // git checkout main
85
+ "git config": 3, // git config user.name
86
+ "git remote": 3, // git remote add origin
87
+ "git stash": 3, // git stash pop
88
+ go: 2, // go build
89
+ gradle: 2, // gradle build
90
+ helm: 2, // helm install mychart
91
+ heroku: 2, // heroku logs
92
+ hugo: 2, // hugo new site blog
93
+ ip: 2, // ip link show
94
+ "ip addr": 3, // ip addr show
95
+ "ip link": 3, // ip link set eth0 up
96
+ "ip netns": 3, // ip netns exec foo bash
97
+ "ip route": 3, // ip route add default via 1.1.1.1
98
+ kind: 2, // kind delete cluster
99
+ "kind create": 3, // kind create cluster
100
+ kubectl: 2, // kubectl get pods
101
+ "kubectl kustomize": 3, // kubectl kustomize overlays/dev
102
+ "kubectl rollout": 3, // kubectl rollout restart deploy/api
103
+ kustomize: 2, // kustomize build .
104
+ make: 2, // make build
105
+ mc: 2, // mc ls myminio
106
+ "mc admin": 3, // mc admin info myminio
107
+ minikube: 2, // minikube start
108
+ mongosh: 2, // mongosh test
109
+ mysql: 2, // mysql -u root
110
+ mvn: 2, // mvn compile
111
+ ng: 2, // ng generate component home
112
+ npm: 2, // npm install
113
+ "npm exec": 3, // npm exec vite
114
+ "npm init": 3, // npm init vue
115
+ "npm run": 3, // npm run dev
116
+ "npm view": 3, // npm view react version
117
+ nvm: 2, // nvm use 18
118
+ nx: 2, // nx build
119
+ openssl: 2, // openssl genrsa 2048
120
+ "openssl req": 3, // openssl req -new -key key.pem
121
+ "openssl x509": 3, // openssl x509 -in cert.pem
122
+ pip: 2, // pip install numpy
123
+ pipenv: 2, // pipenv install flask
124
+ pnpm: 2, // pnpm install
125
+ "pnpm dlx": 3, // pnpm dlx create-next-app
126
+ "pnpm exec": 3, // pnpm exec vite
127
+ "pnpm run": 3, // pnpm run dev
128
+ poetry: 2, // poetry add requests
129
+ podman: 2, // podman run alpine
130
+ "podman container": 3, // podman container ls
131
+ "podman image": 3, // podman image prune
132
+ psql: 2, // psql -d mydb
133
+ pulumi: 2, // pulumi up
134
+ "pulumi stack": 3, // pulumi stack output
135
+ pyenv: 2, // pyenv install 3.11
136
+ python: 2, // python -m venv env
137
+ rake: 2, // rake db:migrate
138
+ rbenv: 2, // rbenv install 3.2.0
139
+ "redis-cli": 2, // redis-cli ping
140
+ rustup: 2, // rustup update
141
+ serverless: 2, // serverless invoke
142
+ sfdx: 3, // sfdx force:org:list
143
+ skaffold: 2, // skaffold dev
144
+ sls: 2, // sls deploy
145
+ sst: 2, // sst deploy
146
+ swift: 2, // swift build
147
+ systemctl: 2, // systemctl restart nginx
148
+ terraform: 2, // terraform apply
149
+ "terraform workspace": 3, // terraform workspace select prod
150
+ tmux: 2, // tmux new -s dev
151
+ turbo: 2, // turbo run build
152
+ ufw: 2, // ufw allow 22
153
+ vault: 2, // vault login
154
+ "vault auth": 3, // vault auth list
155
+ "vault kv": 3, // vault kv get secret/api
156
+ vercel: 2, // vercel deploy
157
+ volta: 2, // volta install node
158
+ wp: 2, // wp plugin install
159
+ yarn: 2, // yarn add react
160
+ "yarn dlx": 3, // yarn dlx create-react-app
161
+ "yarn run": 3, // yarn run dev
162
+ }
163
+ }
@@ -0,0 +1,210 @@
1
+ import { BusEvent } from "@/bus/bus-event"
2
+ import { Bus } from "@/bus"
3
+ import z from "zod"
4
+ import { Log } from "../util/log"
5
+ import { Identifier } from "../id/id"
6
+ import { Plugin } from "../plugin"
7
+ import { Instance } from "../project/instance"
8
+ import { Wildcard } from "../util/wildcard"
9
+
10
+ export namespace Permission {
11
+ const log = Log.create({ service: "permission" })
12
+
13
+ function toKeys(pattern: Info["pattern"], type: string): string[] {
14
+ return pattern === undefined ? [type] : Array.isArray(pattern) ? pattern : [pattern]
15
+ }
16
+
17
+ function covered(keys: string[], approved: Record<string, boolean>): boolean {
18
+ const pats = Object.keys(approved)
19
+ return keys.every((k) => pats.some((p) => Wildcard.match(k, p)))
20
+ }
21
+
22
+ export const Info = z
23
+ .object({
24
+ id: z.string(),
25
+ type: z.string(),
26
+ pattern: z.union([z.string(), z.array(z.string())]).optional(),
27
+ sessionID: z.string(),
28
+ messageID: z.string(),
29
+ callID: z.string().optional(),
30
+ message: z.string(),
31
+ metadata: z.record(z.string(), z.any()),
32
+ time: z.object({
33
+ created: z.number(),
34
+ }),
35
+ })
36
+ .meta({
37
+ ref: "Permission",
38
+ })
39
+ export type Info = z.infer<typeof Info>
40
+
41
+ export const Event = {
42
+ Updated: BusEvent.define("permission.updated", Info),
43
+ Replied: BusEvent.define(
44
+ "permission.replied",
45
+ z.object({
46
+ sessionID: z.string(),
47
+ permissionID: z.string(),
48
+ response: z.string(),
49
+ }),
50
+ ),
51
+ }
52
+
53
+ const state = Instance.state(
54
+ () => {
55
+ const pending: {
56
+ [sessionID: string]: {
57
+ [permissionID: string]: {
58
+ info: Info
59
+ resolve: () => void
60
+ reject: (e: any) => void
61
+ }
62
+ }
63
+ } = {}
64
+
65
+ const approved: {
66
+ [sessionID: string]: {
67
+ [permissionID: string]: boolean
68
+ }
69
+ } = {}
70
+
71
+ return {
72
+ pending,
73
+ approved,
74
+ }
75
+ },
76
+ async (state) => {
77
+ for (const pending of Object.values(state.pending)) {
78
+ for (const item of Object.values(pending)) {
79
+ item.reject(new RejectedError(item.info.sessionID, item.info.id, item.info.callID, item.info.metadata))
80
+ }
81
+ }
82
+ },
83
+ )
84
+
85
+ export function pending() {
86
+ return state().pending
87
+ }
88
+
89
+ export function list() {
90
+ const { pending } = state()
91
+ const result: Info[] = []
92
+ for (const items of Object.values(pending)) {
93
+ for (const item of Object.values(items)) {
94
+ result.push(item.info)
95
+ }
96
+ }
97
+ return result.sort((a, b) => a.id.localeCompare(b.id))
98
+ }
99
+
100
+ export async function ask(input: {
101
+ type: Info["type"]
102
+ message: Info["message"]
103
+ pattern?: Info["pattern"]
104
+ callID?: Info["callID"]
105
+ sessionID: Info["sessionID"]
106
+ messageID: Info["messageID"]
107
+ metadata: Info["metadata"]
108
+ }) {
109
+ const { pending, approved } = state()
110
+ log.info("asking", {
111
+ sessionID: input.sessionID,
112
+ messageID: input.messageID,
113
+ toolCallID: input.callID,
114
+ pattern: input.pattern,
115
+ })
116
+ const approvedForSession = approved[input.sessionID] || {}
117
+ const keys = toKeys(input.pattern, input.type)
118
+ if (covered(keys, approvedForSession)) return
119
+ const info: Info = {
120
+ id: Identifier.ascending("permission"),
121
+ type: input.type,
122
+ pattern: input.pattern,
123
+ sessionID: input.sessionID,
124
+ messageID: input.messageID,
125
+ callID: input.callID,
126
+ message: input.message,
127
+ metadata: input.metadata,
128
+ time: {
129
+ created: Date.now(),
130
+ },
131
+ }
132
+
133
+ switch (
134
+ await Plugin.trigger("permission.ask", info, {
135
+ status: "ask",
136
+ }).then((x) => x.status)
137
+ ) {
138
+ case "deny":
139
+ throw new RejectedError(info.sessionID, info.id, info.callID, info.metadata)
140
+ case "allow":
141
+ return
142
+ }
143
+
144
+ pending[input.sessionID] = pending[input.sessionID] || {}
145
+ return new Promise<void>((resolve, reject) => {
146
+ pending[input.sessionID][info.id] = {
147
+ info,
148
+ resolve,
149
+ reject,
150
+ }
151
+ Bus.publish(Event.Updated, info)
152
+ })
153
+ }
154
+
155
+ export const Response = z.enum(["once", "always", "reject"])
156
+ export type Response = z.infer<typeof Response>
157
+
158
+ export function respond(input: { sessionID: Info["sessionID"]; permissionID: Info["id"]; response: Response }) {
159
+ log.info("response", input)
160
+ const { pending, approved } = state()
161
+ const match = pending[input.sessionID]?.[input.permissionID]
162
+ if (!match) return
163
+ delete pending[input.sessionID][input.permissionID]
164
+ Bus.publish(Event.Replied, {
165
+ sessionID: input.sessionID,
166
+ permissionID: input.permissionID,
167
+ response: input.response,
168
+ })
169
+ if (input.response === "reject") {
170
+ match.reject(new RejectedError(input.sessionID, input.permissionID, match.info.callID, match.info.metadata))
171
+ return
172
+ }
173
+ match.resolve()
174
+ if (input.response === "always") {
175
+ approved[input.sessionID] = approved[input.sessionID] || {}
176
+ const approveKeys = toKeys(match.info.pattern, match.info.type)
177
+ for (const k of approveKeys) {
178
+ approved[input.sessionID][k] = true
179
+ }
180
+ const items = pending[input.sessionID]
181
+ if (!items) return
182
+ for (const item of Object.values(items)) {
183
+ const itemKeys = toKeys(item.info.pattern, item.info.type)
184
+ if (covered(itemKeys, approved[input.sessionID])) {
185
+ respond({
186
+ sessionID: item.info.sessionID,
187
+ permissionID: item.info.id,
188
+ response: input.response,
189
+ })
190
+ }
191
+ }
192
+ }
193
+ }
194
+
195
+ export class RejectedError extends Error {
196
+ constructor(
197
+ public readonly sessionID: string,
198
+ public readonly permissionID: string,
199
+ public readonly toolCallID?: string,
200
+ public readonly metadata?: Record<string, any>,
201
+ public readonly reason?: string,
202
+ ) {
203
+ super(
204
+ reason !== undefined
205
+ ? reason
206
+ : `The user rejected permission to use this specific tool call. You may try again with different parameters.`,
207
+ )
208
+ }
209
+ }
210
+ }
@@ -0,0 +1,268 @@
1
+ import { Bus } from "@/bus"
2
+ import { BusEvent } from "@/bus/bus-event"
3
+ import { Config } from "@/config/config"
4
+ import { Identifier } from "@/id/id"
5
+ import { Instance } from "@/project/instance"
6
+ import { Storage } from "@/storage/storage"
7
+ import { fn } from "@/util/fn"
8
+ import { Log } from "@/util/log"
9
+ import { Wildcard } from "@/util/wildcard"
10
+ import z from "zod"
11
+
12
+ export namespace PermissionNext {
13
+ const log = Log.create({ service: "permission" })
14
+
15
+ export const Action = z.enum(["allow", "deny", "ask"]).meta({
16
+ ref: "PermissionAction",
17
+ })
18
+ export type Action = z.infer<typeof Action>
19
+
20
+ export const Rule = z
21
+ .object({
22
+ permission: z.string(),
23
+ pattern: z.string(),
24
+ action: Action,
25
+ })
26
+ .meta({
27
+ ref: "PermissionRule",
28
+ })
29
+ export type Rule = z.infer<typeof Rule>
30
+
31
+ export const Ruleset = Rule.array().meta({
32
+ ref: "PermissionRuleset",
33
+ })
34
+ export type Ruleset = z.infer<typeof Ruleset>
35
+
36
+ export function fromConfig(permission: Config.Permission) {
37
+ const ruleset: Ruleset = []
38
+ for (const [key, value] of Object.entries(permission)) {
39
+ if (typeof value === "string") {
40
+ ruleset.push({
41
+ permission: key,
42
+ action: value,
43
+ pattern: "*",
44
+ })
45
+ continue
46
+ }
47
+ ruleset.push(...Object.entries(value).map(([pattern, action]) => ({ permission: key, pattern, action })))
48
+ }
49
+ return ruleset
50
+ }
51
+
52
+ export function merge(...rulesets: Ruleset[]): Ruleset {
53
+ return rulesets.flat()
54
+ }
55
+
56
+ export const Request = z
57
+ .object({
58
+ id: Identifier.schema("permission"),
59
+ sessionID: Identifier.schema("session"),
60
+ permission: z.string(),
61
+ patterns: z.string().array(),
62
+ metadata: z.record(z.string(), z.any()),
63
+ always: z.string().array(),
64
+ tool: z
65
+ .object({
66
+ messageID: z.string(),
67
+ callID: z.string(),
68
+ })
69
+ .optional(),
70
+ })
71
+ .meta({
72
+ ref: "PermissionRequest",
73
+ })
74
+
75
+ export type Request = z.infer<typeof Request>
76
+
77
+ export const Reply = z.enum(["once", "always", "reject"])
78
+ export type Reply = z.infer<typeof Reply>
79
+
80
+ export const Approval = z.object({
81
+ projectID: z.string(),
82
+ patterns: z.string().array(),
83
+ })
84
+
85
+ export const Event = {
86
+ Asked: BusEvent.define("permission.asked", Request),
87
+ Replied: BusEvent.define(
88
+ "permission.replied",
89
+ z.object({
90
+ sessionID: z.string(),
91
+ requestID: z.string(),
92
+ reply: Reply,
93
+ }),
94
+ ),
95
+ }
96
+
97
+ const state = Instance.state(async () => {
98
+ const projectID = Instance.project.id
99
+ const stored = await Storage.read<Ruleset>(["permission", projectID]).catch(() => [] as Ruleset)
100
+
101
+ const pending: Record<
102
+ string,
103
+ {
104
+ info: Request
105
+ resolve: () => void
106
+ reject: (e: any) => void
107
+ }
108
+ > = {}
109
+
110
+ return {
111
+ pending,
112
+ approved: stored,
113
+ }
114
+ })
115
+
116
+ export const ask = fn(
117
+ Request.partial({ id: true }).extend({
118
+ ruleset: Ruleset,
119
+ }),
120
+ async (input) => {
121
+ const s = await state()
122
+ const { ruleset, ...request } = input
123
+ for (const pattern of request.patterns ?? []) {
124
+ const rule = evaluate(request.permission, pattern, ruleset, s.approved)
125
+ log.info("evaluated", { permission: request.permission, pattern, action: rule })
126
+ if (rule.action === "deny")
127
+ throw new DeniedError(ruleset.filter((r) => Wildcard.match(request.permission, r.permission)))
128
+ if (rule.action === "ask") {
129
+ const id = input.id ?? Identifier.ascending("permission")
130
+ return new Promise<void>((resolve, reject) => {
131
+ const info: Request = {
132
+ id,
133
+ ...request,
134
+ }
135
+ s.pending[id] = {
136
+ info,
137
+ resolve,
138
+ reject,
139
+ }
140
+ Bus.publish(Event.Asked, info)
141
+ })
142
+ }
143
+ if (rule.action === "allow") continue
144
+ }
145
+ },
146
+ )
147
+
148
+ export const reply = fn(
149
+ z.object({
150
+ requestID: Identifier.schema("permission"),
151
+ reply: Reply,
152
+ message: z.string().optional(),
153
+ }),
154
+ async (input) => {
155
+ const s = await state()
156
+ const existing = s.pending[input.requestID]
157
+ if (!existing) return
158
+ delete s.pending[input.requestID]
159
+ Bus.publish(Event.Replied, {
160
+ sessionID: existing.info.sessionID,
161
+ requestID: existing.info.id,
162
+ reply: input.reply,
163
+ })
164
+ if (input.reply === "reject") {
165
+ existing.reject(input.message ? new CorrectedError(input.message) : new RejectedError())
166
+ // Reject all other pending permissions for this session
167
+ const sessionID = existing.info.sessionID
168
+ for (const [id, pending] of Object.entries(s.pending)) {
169
+ if (pending.info.sessionID === sessionID) {
170
+ delete s.pending[id]
171
+ Bus.publish(Event.Replied, {
172
+ sessionID: pending.info.sessionID,
173
+ requestID: pending.info.id,
174
+ reply: "reject",
175
+ })
176
+ pending.reject(new RejectedError())
177
+ }
178
+ }
179
+ return
180
+ }
181
+ if (input.reply === "once") {
182
+ existing.resolve()
183
+ return
184
+ }
185
+ if (input.reply === "always") {
186
+ for (const pattern of existing.info.always) {
187
+ s.approved.push({
188
+ permission: existing.info.permission,
189
+ pattern,
190
+ action: "allow",
191
+ })
192
+ }
193
+
194
+ existing.resolve()
195
+
196
+ const sessionID = existing.info.sessionID
197
+ for (const [id, pending] of Object.entries(s.pending)) {
198
+ if (pending.info.sessionID !== sessionID) continue
199
+ const ok = pending.info.patterns.every(
200
+ (pattern) => evaluate(pending.info.permission, pattern, s.approved).action === "allow",
201
+ )
202
+ if (!ok) continue
203
+ delete s.pending[id]
204
+ Bus.publish(Event.Replied, {
205
+ sessionID: pending.info.sessionID,
206
+ requestID: pending.info.id,
207
+ reply: "always",
208
+ })
209
+ pending.resolve()
210
+ }
211
+
212
+ // TODO: we don't save the permission ruleset to disk yet until there's
213
+ // UI to manage it
214
+ // await Storage.write(["permission", Instance.project.id], s.approved)
215
+ return
216
+ }
217
+ },
218
+ )
219
+
220
+ export function evaluate(permission: string, pattern: string, ...rulesets: Ruleset[]): Rule {
221
+ const merged = merge(...rulesets)
222
+ log.info("evaluate", { permission, pattern, ruleset: merged })
223
+ const match = merged.findLast(
224
+ (rule) => Wildcard.match(permission, rule.permission) && Wildcard.match(pattern, rule.pattern),
225
+ )
226
+ return match ?? { action: "ask", permission, pattern: "*" }
227
+ }
228
+
229
+ const EDIT_TOOLS = ["edit", "write", "patch", "multiedit"]
230
+
231
+ export function disabled(tools: string[], ruleset: Ruleset): Set<string> {
232
+ const result = new Set<string>()
233
+ for (const tool of tools) {
234
+ const permission = EDIT_TOOLS.includes(tool) ? "edit" : tool
235
+ if (evaluate(permission, "*", ruleset).action === "deny") {
236
+ result.add(tool)
237
+ }
238
+ }
239
+ return result
240
+ }
241
+
242
+ /** User rejected without message - halts execution */
243
+ export class RejectedError extends Error {
244
+ constructor() {
245
+ super(`The user rejected permission to use this specific tool call.`)
246
+ }
247
+ }
248
+
249
+ /** User rejected with message - continues with guidance */
250
+ export class CorrectedError extends Error {
251
+ constructor(message: string) {
252
+ super(`The user rejected permission to use this specific tool call with the following feedback: ${message}`)
253
+ }
254
+ }
255
+
256
+ /** Auto-rejected by config rule - halts execution */
257
+ export class DeniedError extends Error {
258
+ constructor(public readonly ruleset: Ruleset) {
259
+ super(
260
+ `The user has specified a rule which prevents you from using this specific tool call. Here are some of the relevant rules ${JSON.stringify(ruleset)}`,
261
+ )
262
+ }
263
+ }
264
+
265
+ export async function list() {
266
+ return state().then((x) => Object.values(x.pending).map((x) => x.info))
267
+ }
268
+ }