chad-code 1.3.1 → 1.3.2
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.
- package/README.md +55 -6
- package/package.json +36 -98
- package/AGENTS.md +0 -27
- package/Dockerfile +0 -18
- package/README.npm.md +0 -64
- package/bunfig.toml +0 -7
- package/eslint.config.js +0 -29
- package/parsers-config.ts +0 -253
- package/script/build.ts +0 -167
- package/script/publish-registries.ts +0 -187
- package/script/publish.ts +0 -93
- package/script/schema.ts +0 -47
- package/src/acp/README.md +0 -164
- package/src/acp/agent.ts +0 -1086
- package/src/acp/session.ts +0 -101
- package/src/acp/types.ts +0 -22
- package/src/agent/agent.ts +0 -253
- package/src/agent/generate.txt +0 -75
- package/src/agent/prompt/compaction.txt +0 -12
- package/src/agent/prompt/explore.txt +0 -18
- package/src/agent/prompt/summary.txt +0 -11
- package/src/agent/prompt/title.txt +0 -36
- package/src/auth/index.ts +0 -70
- package/src/bun/index.ts +0 -130
- package/src/bus/bus-event.ts +0 -43
- package/src/bus/global.ts +0 -10
- package/src/bus/index.ts +0 -105
- package/src/cli/bootstrap.ts +0 -17
- package/src/cli/cmd/acp.ts +0 -69
- package/src/cli/cmd/agent.ts +0 -257
- package/src/cli/cmd/auth.ts +0 -132
- package/src/cli/cmd/cmd.ts +0 -7
- package/src/cli/cmd/debug/agent.ts +0 -28
- package/src/cli/cmd/debug/config.ts +0 -15
- package/src/cli/cmd/debug/file.ts +0 -91
- package/src/cli/cmd/debug/index.ts +0 -45
- package/src/cli/cmd/debug/lsp.ts +0 -48
- package/src/cli/cmd/debug/ripgrep.ts +0 -83
- package/src/cli/cmd/debug/scrap.ts +0 -15
- package/src/cli/cmd/debug/skill.ts +0 -15
- package/src/cli/cmd/debug/snapshot.ts +0 -48
- package/src/cli/cmd/export.ts +0 -88
- package/src/cli/cmd/generate.ts +0 -38
- package/src/cli/cmd/github.ts +0 -32
- package/src/cli/cmd/import.ts +0 -98
- package/src/cli/cmd/mcp.ts +0 -670
- package/src/cli/cmd/models.ts +0 -42
- package/src/cli/cmd/pr.ts +0 -112
- package/src/cli/cmd/run.ts +0 -374
- package/src/cli/cmd/serve.ts +0 -16
- package/src/cli/cmd/session.ts +0 -135
- package/src/cli/cmd/stats.ts +0 -402
- package/src/cli/cmd/tui/app.tsx +0 -705
- package/src/cli/cmd/tui/attach.ts +0 -32
- package/src/cli/cmd/tui/component/border.tsx +0 -21
- package/src/cli/cmd/tui/component/dialog-agent.tsx +0 -31
- package/src/cli/cmd/tui/component/dialog-command.tsx +0 -124
- package/src/cli/cmd/tui/component/dialog-mcp.tsx +0 -86
- package/src/cli/cmd/tui/component/dialog-model.tsx +0 -232
- package/src/cli/cmd/tui/component/dialog-provider.tsx +0 -228
- package/src/cli/cmd/tui/component/dialog-session-list.tsx +0 -115
- package/src/cli/cmd/tui/component/dialog-session-rename.tsx +0 -31
- package/src/cli/cmd/tui/component/dialog-stash.tsx +0 -86
- package/src/cli/cmd/tui/component/dialog-status.tsx +0 -162
- package/src/cli/cmd/tui/component/dialog-tag.tsx +0 -44
- package/src/cli/cmd/tui/component/dialog-theme-list.tsx +0 -50
- package/src/cli/cmd/tui/component/did-you-know.tsx +0 -85
- package/src/cli/cmd/tui/component/logo.tsx +0 -43
- package/src/cli/cmd/tui/component/prompt/autocomplete.tsx +0 -654
- package/src/cli/cmd/tui/component/prompt/history.tsx +0 -108
- package/src/cli/cmd/tui/component/prompt/index.tsx +0 -1078
- package/src/cli/cmd/tui/component/prompt/stash.tsx +0 -101
- package/src/cli/cmd/tui/component/textarea-keybindings.ts +0 -73
- package/src/cli/cmd/tui/component/tips.ts +0 -92
- package/src/cli/cmd/tui/component/todo-item.tsx +0 -32
- package/src/cli/cmd/tui/context/args.tsx +0 -14
- package/src/cli/cmd/tui/context/directory.ts +0 -13
- package/src/cli/cmd/tui/context/exit.tsx +0 -23
- package/src/cli/cmd/tui/context/helper.tsx +0 -25
- package/src/cli/cmd/tui/context/keybind.tsx +0 -101
- package/src/cli/cmd/tui/context/kv.tsx +0 -49
- package/src/cli/cmd/tui/context/local.tsx +0 -392
- package/src/cli/cmd/tui/context/prompt.tsx +0 -18
- package/src/cli/cmd/tui/context/route.tsx +0 -46
- package/src/cli/cmd/tui/context/sdk.tsx +0 -75
- package/src/cli/cmd/tui/context/sync.tsx +0 -384
- package/src/cli/cmd/tui/context/theme/aura.json +0 -69
- package/src/cli/cmd/tui/context/theme/ayu.json +0 -80
- package/src/cli/cmd/tui/context/theme/catppuccin-frappe.json +0 -233
- package/src/cli/cmd/tui/context/theme/catppuccin-macchiato.json +0 -233
- package/src/cli/cmd/tui/context/theme/catppuccin.json +0 -112
- package/src/cli/cmd/tui/context/theme/chad.json +0 -245
- package/src/cli/cmd/tui/context/theme/cobalt2.json +0 -228
- package/src/cli/cmd/tui/context/theme/cursor.json +0 -249
- package/src/cli/cmd/tui/context/theme/dracula.json +0 -219
- package/src/cli/cmd/tui/context/theme/everforest.json +0 -241
- package/src/cli/cmd/tui/context/theme/flexoki.json +0 -237
- package/src/cli/cmd/tui/context/theme/github.json +0 -233
- package/src/cli/cmd/tui/context/theme/gruvbox.json +0 -95
- package/src/cli/cmd/tui/context/theme/kanagawa.json +0 -77
- package/src/cli/cmd/tui/context/theme/lucent-orng.json +0 -227
- package/src/cli/cmd/tui/context/theme/material.json +0 -235
- package/src/cli/cmd/tui/context/theme/matrix.json +0 -77
- package/src/cli/cmd/tui/context/theme/mercury.json +0 -252
- package/src/cli/cmd/tui/context/theme/monokai.json +0 -221
- package/src/cli/cmd/tui/context/theme/nightowl.json +0 -221
- package/src/cli/cmd/tui/context/theme/nord.json +0 -223
- package/src/cli/cmd/tui/context/theme/one-dark.json +0 -84
- package/src/cli/cmd/tui/context/theme/orng.json +0 -245
- package/src/cli/cmd/tui/context/theme/osaka-jade.json +0 -93
- package/src/cli/cmd/tui/context/theme/palenight.json +0 -222
- package/src/cli/cmd/tui/context/theme/rosepine.json +0 -234
- package/src/cli/cmd/tui/context/theme/solarized.json +0 -223
- package/src/cli/cmd/tui/context/theme/synthwave84.json +0 -226
- package/src/cli/cmd/tui/context/theme/tokyonight.json +0 -243
- package/src/cli/cmd/tui/context/theme/vercel.json +0 -245
- package/src/cli/cmd/tui/context/theme/vesper.json +0 -218
- package/src/cli/cmd/tui/context/theme/zenburn.json +0 -223
- package/src/cli/cmd/tui/context/theme.tsx +0 -1137
- package/src/cli/cmd/tui/event.ts +0 -46
- package/src/cli/cmd/tui/routes/home.tsx +0 -138
- package/src/cli/cmd/tui/routes/session/dialog-fork-from-timeline.tsx +0 -64
- package/src/cli/cmd/tui/routes/session/dialog-message.tsx +0 -109
- package/src/cli/cmd/tui/routes/session/dialog-subagent.tsx +0 -26
- package/src/cli/cmd/tui/routes/session/dialog-timeline.tsx +0 -47
- package/src/cli/cmd/tui/routes/session/footer.tsx +0 -88
- package/src/cli/cmd/tui/routes/session/header.tsx +0 -125
- package/src/cli/cmd/tui/routes/session/index.tsx +0 -1814
- package/src/cli/cmd/tui/routes/session/permission.tsx +0 -416
- package/src/cli/cmd/tui/routes/session/sidebar.tsx +0 -318
- package/src/cli/cmd/tui/spawn.ts +0 -48
- package/src/cli/cmd/tui/thread.ts +0 -111
- package/src/cli/cmd/tui/ui/dialog-alert.tsx +0 -57
- package/src/cli/cmd/tui/ui/dialog-confirm.tsx +0 -83
- package/src/cli/cmd/tui/ui/dialog-export-options.tsx +0 -204
- package/src/cli/cmd/tui/ui/dialog-help.tsx +0 -38
- package/src/cli/cmd/tui/ui/dialog-prompt.tsx +0 -77
- package/src/cli/cmd/tui/ui/dialog-select.tsx +0 -345
- package/src/cli/cmd/tui/ui/dialog.tsx +0 -171
- package/src/cli/cmd/tui/ui/link.tsx +0 -28
- package/src/cli/cmd/tui/ui/spinner.ts +0 -368
- package/src/cli/cmd/tui/ui/toast.tsx +0 -100
- package/src/cli/cmd/tui/util/clipboard.ts +0 -127
- package/src/cli/cmd/tui/util/editor.ts +0 -32
- package/src/cli/cmd/tui/util/signal.ts +0 -7
- package/src/cli/cmd/tui/util/terminal.ts +0 -114
- package/src/cli/cmd/tui/util/transcript.ts +0 -98
- package/src/cli/cmd/tui/worker.ts +0 -68
- package/src/cli/cmd/uninstall.ts +0 -344
- package/src/cli/cmd/upgrade.ts +0 -67
- package/src/cli/cmd/web.ts +0 -73
- package/src/cli/error.ts +0 -56
- package/src/cli/network.ts +0 -53
- package/src/cli/ui.ts +0 -87
- package/src/cli/upgrade.ts +0 -25
- package/src/command/index.ts +0 -131
- package/src/command/template/initialize.txt +0 -10
- package/src/command/template/review.txt +0 -97
- package/src/config/config.ts +0 -1124
- package/src/config/markdown.ts +0 -41
- package/src/env/index.ts +0 -26
- package/src/file/ignore.ts +0 -83
- package/src/file/index.ts +0 -411
- package/src/file/ripgrep.ts +0 -402
- package/src/file/time.ts +0 -64
- package/src/file/watcher.ts +0 -117
- package/src/flag/flag.ts +0 -52
- package/src/format/formatter.ts +0 -359
- package/src/format/index.ts +0 -137
- package/src/global/index.ts +0 -55
- package/src/id/id.ts +0 -73
- package/src/ide/index.ts +0 -77
- package/src/index.ts +0 -159
- package/src/installation/index.ts +0 -198
- package/src/lsp/client.ts +0 -252
- package/src/lsp/index.ts +0 -485
- package/src/lsp/language.ts +0 -119
- package/src/lsp/server.ts +0 -2023
- package/src/mcp/auth.ts +0 -135
- package/src/mcp/index.ts +0 -874
- package/src/mcp/oauth-callback.ts +0 -200
- package/src/mcp/oauth-provider.ts +0 -154
- package/src/patch/index.ts +0 -622
- package/src/permission/arity.ts +0 -163
- package/src/permission/index.ts +0 -210
- package/src/permission/next.ts +0 -268
- package/src/plugin/index.ts +0 -106
- package/src/project/bootstrap.ts +0 -31
- package/src/project/instance.ts +0 -78
- package/src/project/project.ts +0 -263
- package/src/project/state.ts +0 -65
- package/src/project/vcs.ts +0 -76
- package/src/provider/auth.ts +0 -143
- package/src/provider/models-macro.ts +0 -4
- package/src/provider/models.ts +0 -77
- package/src/provider/provider.ts +0 -516
- package/src/provider/transform.ts +0 -114
- package/src/pty/index.ts +0 -212
- package/src/server/error.ts +0 -36
- package/src/server/mdns.ts +0 -57
- package/src/server/project.ts +0 -79
- package/src/server/server.ts +0 -2866
- package/src/server/tui.ts +0 -71
- package/src/session/compaction.ts +0 -225
- package/src/session/index.ts +0 -469
- package/src/session/llm.ts +0 -213
- package/src/session/message-v2.ts +0 -742
- package/src/session/message.ts +0 -189
- package/src/session/processor.ts +0 -402
- package/src/session/prompt/anthropic-20250930.txt +0 -166
- package/src/session/prompt/anthropic.txt +0 -105
- package/src/session/prompt/anthropic_spoof.txt +0 -1
- package/src/session/prompt/beast.txt +0 -147
- package/src/session/prompt/build-switch.txt +0 -5
- package/src/session/prompt/codex.txt +0 -318
- package/src/session/prompt/copilot-gpt-5.txt +0 -143
- package/src/session/prompt/gemini.txt +0 -155
- package/src/session/prompt/max-steps.txt +0 -16
- package/src/session/prompt/plan-reminder-anthropic.txt +0 -67
- package/src/session/prompt/plan.txt +0 -26
- package/src/session/prompt/qwen.txt +0 -109
- package/src/session/prompt.ts +0 -1621
- package/src/session/retry.ts +0 -90
- package/src/session/revert.ts +0 -108
- package/src/session/status.ts +0 -76
- package/src/session/summary.ts +0 -194
- package/src/session/system.ts +0 -108
- package/src/session/todo.ts +0 -37
- package/src/share/share-next.ts +0 -194
- package/src/share/share.ts +0 -23
- package/src/shell/shell.ts +0 -67
- package/src/skill/index.ts +0 -1
- package/src/skill/skill.ts +0 -124
- package/src/snapshot/index.ts +0 -197
- package/src/storage/storage.ts +0 -226
- package/src/tool/bash.ts +0 -262
- package/src/tool/bash.txt +0 -116
- package/src/tool/batch.ts +0 -175
- package/src/tool/batch.txt +0 -24
- package/src/tool/codesearch.ts +0 -132
- package/src/tool/codesearch.txt +0 -12
- package/src/tool/edit.ts +0 -655
- package/src/tool/edit.txt +0 -10
- package/src/tool/glob.ts +0 -75
- package/src/tool/glob.txt +0 -6
- package/src/tool/grep.ts +0 -132
- package/src/tool/grep.txt +0 -8
- package/src/tool/invalid.ts +0 -17
- package/src/tool/ls.ts +0 -119
- package/src/tool/ls.txt +0 -1
- package/src/tool/lsp.ts +0 -94
- package/src/tool/lsp.txt +0 -19
- package/src/tool/multiedit.ts +0 -46
- package/src/tool/multiedit.txt +0 -41
- package/src/tool/patch.ts +0 -210
- package/src/tool/patch.txt +0 -1
- package/src/tool/read.ts +0 -191
- package/src/tool/read.txt +0 -12
- package/src/tool/registry.ts +0 -137
- package/src/tool/skill.ts +0 -77
- package/src/tool/task.ts +0 -167
- package/src/tool/task.txt +0 -60
- package/src/tool/todo.ts +0 -53
- package/src/tool/todoread.txt +0 -14
- package/src/tool/todowrite.txt +0 -167
- package/src/tool/tool.ts +0 -73
- package/src/tool/webfetch.ts +0 -182
- package/src/tool/webfetch.txt +0 -13
- package/src/tool/websearch.ts +0 -144
- package/src/tool/websearch.txt +0 -11
- package/src/tool/write.ts +0 -84
- package/src/tool/write.txt +0 -8
- package/src/util/archive.ts +0 -16
- package/src/util/color.ts +0 -19
- package/src/util/context.ts +0 -25
- package/src/util/defer.ts +0 -12
- package/src/util/eventloop.ts +0 -20
- package/src/util/filesystem.ts +0 -83
- package/src/util/fn.ts +0 -11
- package/src/util/iife.ts +0 -3
- package/src/util/keybind.ts +0 -102
- package/src/util/lazy.ts +0 -18
- package/src/util/locale.ts +0 -81
- package/src/util/lock.ts +0 -98
- package/src/util/log.ts +0 -180
- package/src/util/queue.ts +0 -32
- package/src/util/rpc.ts +0 -42
- package/src/util/scrap.ts +0 -10
- package/src/util/signal.ts +0 -12
- package/src/util/timeout.ts +0 -14
- package/src/util/token.ts +0 -7
- package/src/util/wildcard.ts +0 -54
- package/src/worktree/index.ts +0 -217
- package/sst-env.d.ts +0 -9
- package/test/agent/agent.test.ts +0 -448
- package/test/bun.test.ts +0 -53
- package/test/cli/github-action.test.ts +0 -129
- package/test/cli/github-remote.test.ts +0 -80
- package/test/cli/tui/transcript.test.ts +0 -297
- package/test/config/agent-color.test.ts +0 -66
- package/test/config/config.test.ts +0 -870
- package/test/config/markdown.test.ts +0 -89
- package/test/file/ignore.test.ts +0 -10
- package/test/file/path-traversal.test.ts +0 -115
- package/test/fixture/fixture.ts +0 -45
- package/test/fixture/lsp/fake-lsp-server.js +0 -77
- package/test/ide/ide.test.ts +0 -82
- package/test/keybind.test.ts +0 -421
- package/test/lsp/client.test.ts +0 -95
- package/test/mcp/headers.test.ts +0 -153
- package/test/patch/patch.test.ts +0 -348
- package/test/permission/arity.test.ts +0 -33
- package/test/permission/next.test.ts +0 -652
- package/test/preload.ts +0 -63
- package/test/project/project.test.ts +0 -120
- package/test/provider/amazon-bedrock.test.ts +0 -236
- package/test/provider/provider.test.ts +0 -2127
- package/test/provider/transform.test.ts +0 -980
- package/test/server/session-select.test.ts +0 -78
- package/test/session/compaction.test.ts +0 -251
- package/test/session/message-v2.test.ts +0 -570
- package/test/session/retry.test.ts +0 -131
- package/test/session/revert-compact.test.ts +0 -285
- package/test/session/session.test.ts +0 -71
- package/test/skill/skill.test.ts +0 -185
- package/test/snapshot/snapshot.test.ts +0 -939
- package/test/tool/__snapshots__/tool.test.ts.snap +0 -9
- package/test/tool/bash.test.ts +0 -232
- package/test/tool/grep.test.ts +0 -109
- package/test/tool/patch.test.ts +0 -261
- package/test/tool/read.test.ts +0 -167
- package/test/util/iife.test.ts +0 -36
- package/test/util/lazy.test.ts +0 -50
- package/test/util/timeout.test.ts +0 -21
- package/test/util/wildcard.test.ts +0 -55
- package/tsconfig.json +0 -16
- /package/{script/postinstall.mjs → postinstall.mjs} +0 -0
package/src/permission/arity.ts
DELETED
|
@@ -1,163 +0,0 @@
|
|
|
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
|
-
}
|
package/src/permission/index.ts
DELETED
|
@@ -1,210 +0,0 @@
|
|
|
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
|
-
}
|
package/src/permission/next.ts
DELETED
|
@@ -1,268 +0,0 @@
|
|
|
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
|
-
}
|