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
|
@@ -1,89 +0,0 @@
|
|
|
1
|
-
import { expect, test } from "bun:test"
|
|
2
|
-
import { ConfigMarkdown } from "../../src/config/markdown"
|
|
3
|
-
|
|
4
|
-
const template = `This is a @valid/path/to/a/file and it should also match at
|
|
5
|
-
the beginning of a line:
|
|
6
|
-
|
|
7
|
-
@another-valid/path/to/a/file
|
|
8
|
-
|
|
9
|
-
but this is not:
|
|
10
|
-
|
|
11
|
-
- Adds a "Co-authored-by:" footer which clarifies which AI agent
|
|
12
|
-
helped create this commit, using an appropriate \`noreply@...\`
|
|
13
|
-
or \`noreply@anthropic.com\` email address.
|
|
14
|
-
|
|
15
|
-
We also need to deal with files followed by @commas, ones
|
|
16
|
-
with @file-extensions.md, even @multiple.extensions.bak,
|
|
17
|
-
hidden directories like @.config/ or files like @.bashrc
|
|
18
|
-
and ones at the end of a sentence like @foo.md.
|
|
19
|
-
|
|
20
|
-
Also shouldn't forget @/absolute/paths.txt with and @/without/extensions,
|
|
21
|
-
as well as @~/home-files and @~/paths/under/home.txt.
|
|
22
|
-
|
|
23
|
-
If the reference is \`@quoted/in/backticks\` then it shouldn't match at all.`
|
|
24
|
-
|
|
25
|
-
const matches = ConfigMarkdown.files(template)
|
|
26
|
-
|
|
27
|
-
test("should extract exactly 12 file references", () => {
|
|
28
|
-
expect(matches.length).toBe(12)
|
|
29
|
-
})
|
|
30
|
-
|
|
31
|
-
test("should extract valid/path/to/a/file", () => {
|
|
32
|
-
expect(matches[0][1]).toBe("valid/path/to/a/file")
|
|
33
|
-
})
|
|
34
|
-
|
|
35
|
-
test("should extract another-valid/path/to/a/file", () => {
|
|
36
|
-
expect(matches[1][1]).toBe("another-valid/path/to/a/file")
|
|
37
|
-
})
|
|
38
|
-
|
|
39
|
-
test("should extract paths ignoring comma after", () => {
|
|
40
|
-
expect(matches[2][1]).toBe("commas")
|
|
41
|
-
})
|
|
42
|
-
|
|
43
|
-
test("should extract a path with a file extension and comma after", () => {
|
|
44
|
-
expect(matches[3][1]).toBe("file-extensions.md")
|
|
45
|
-
})
|
|
46
|
-
|
|
47
|
-
test("should extract a path with multiple dots and comma after", () => {
|
|
48
|
-
expect(matches[4][1]).toBe("multiple.extensions.bak")
|
|
49
|
-
})
|
|
50
|
-
|
|
51
|
-
test("should extract hidden directory", () => {
|
|
52
|
-
expect(matches[5][1]).toBe(".config/")
|
|
53
|
-
})
|
|
54
|
-
|
|
55
|
-
test("should extract hidden file", () => {
|
|
56
|
-
expect(matches[6][1]).toBe(".bashrc")
|
|
57
|
-
})
|
|
58
|
-
|
|
59
|
-
test("should extract a file ignoring period at end of sentence", () => {
|
|
60
|
-
expect(matches[7][1]).toBe("foo.md")
|
|
61
|
-
})
|
|
62
|
-
|
|
63
|
-
test("should extract an absolute path with an extension", () => {
|
|
64
|
-
expect(matches[8][1]).toBe("/absolute/paths.txt")
|
|
65
|
-
})
|
|
66
|
-
|
|
67
|
-
test("should extract an absolute path without an extension", () => {
|
|
68
|
-
expect(matches[9][1]).toBe("/without/extensions")
|
|
69
|
-
})
|
|
70
|
-
|
|
71
|
-
test("should extract an absolute path in home directory", () => {
|
|
72
|
-
expect(matches[10][1]).toBe("~/home-files")
|
|
73
|
-
})
|
|
74
|
-
|
|
75
|
-
test("should extract an absolute path under home directory", () => {
|
|
76
|
-
expect(matches[11][1]).toBe("~/paths/under/home.txt")
|
|
77
|
-
})
|
|
78
|
-
|
|
79
|
-
test("should not match when preceded by backtick", () => {
|
|
80
|
-
const backtickTest = "This `@should/not/match` should be ignored"
|
|
81
|
-
const backtickMatches = ConfigMarkdown.files(backtickTest)
|
|
82
|
-
expect(backtickMatches.length).toBe(0)
|
|
83
|
-
})
|
|
84
|
-
|
|
85
|
-
test("should not match email addresses", () => {
|
|
86
|
-
const emailTest = "Contact user@example.com for help"
|
|
87
|
-
const emailMatches = ConfigMarkdown.files(emailTest)
|
|
88
|
-
expect(emailMatches.length).toBe(0)
|
|
89
|
-
})
|
package/test/file/ignore.test.ts
DELETED
|
@@ -1,10 +0,0 @@
|
|
|
1
|
-
import { test, expect } from "bun:test"
|
|
2
|
-
import { FileIgnore } from "../../src/file/ignore"
|
|
3
|
-
|
|
4
|
-
test("match nested and non-nested", () => {
|
|
5
|
-
expect(FileIgnore.match("node_modules/index.js")).toBe(true)
|
|
6
|
-
expect(FileIgnore.match("node_modules")).toBe(true)
|
|
7
|
-
expect(FileIgnore.match("node_modules/")).toBe(true)
|
|
8
|
-
expect(FileIgnore.match("node_modules/bar")).toBe(true)
|
|
9
|
-
expect(FileIgnore.match("node_modules/bar/")).toBe(true)
|
|
10
|
-
})
|
|
@@ -1,115 +0,0 @@
|
|
|
1
|
-
import { test, expect, describe } from "bun:test"
|
|
2
|
-
import path from "path"
|
|
3
|
-
import { Filesystem } from "../../src/util/filesystem"
|
|
4
|
-
import { File } from "../../src/file"
|
|
5
|
-
import { Instance } from "../../src/project/instance"
|
|
6
|
-
import { tmpdir } from "../fixture/fixture"
|
|
7
|
-
|
|
8
|
-
describe("Filesystem.contains", () => {
|
|
9
|
-
test("allows paths within project", () => {
|
|
10
|
-
expect(Filesystem.contains("/project", "/project/src")).toBe(true)
|
|
11
|
-
expect(Filesystem.contains("/project", "/project/src/file.ts")).toBe(true)
|
|
12
|
-
expect(Filesystem.contains("/project", "/project")).toBe(true)
|
|
13
|
-
})
|
|
14
|
-
|
|
15
|
-
test("blocks ../ traversal", () => {
|
|
16
|
-
expect(Filesystem.contains("/project", "/project/../etc")).toBe(false)
|
|
17
|
-
expect(Filesystem.contains("/project", "/project/src/../../etc")).toBe(false)
|
|
18
|
-
expect(Filesystem.contains("/project", "/etc/passwd")).toBe(false)
|
|
19
|
-
})
|
|
20
|
-
|
|
21
|
-
test("blocks absolute paths outside project", () => {
|
|
22
|
-
expect(Filesystem.contains("/project", "/etc/passwd")).toBe(false)
|
|
23
|
-
expect(Filesystem.contains("/project", "/tmp/file")).toBe(false)
|
|
24
|
-
expect(Filesystem.contains("/home/user/project", "/home/user/other")).toBe(false)
|
|
25
|
-
})
|
|
26
|
-
|
|
27
|
-
test("handles prefix collision edge cases", () => {
|
|
28
|
-
expect(Filesystem.contains("/project", "/project-other/file")).toBe(false)
|
|
29
|
-
expect(Filesystem.contains("/project", "/projectfile")).toBe(false)
|
|
30
|
-
})
|
|
31
|
-
})
|
|
32
|
-
|
|
33
|
-
/*
|
|
34
|
-
* Integration tests for File.read() and File.list() path traversal protection.
|
|
35
|
-
*
|
|
36
|
-
* These tests verify the HTTP API code path is protected. The HTTP endpoints
|
|
37
|
-
* in server.ts (GET /file/content, GET /file) call File.read()/File.list()
|
|
38
|
-
* directly - they do NOT go through ReadTool or the agent permission layer.
|
|
39
|
-
*
|
|
40
|
-
* This is a SEPARATE code path from ReadTool, which has its own checks.
|
|
41
|
-
*/
|
|
42
|
-
describe("File.read path traversal protection", () => {
|
|
43
|
-
test("rejects ../ traversal attempting to read /etc/passwd", async () => {
|
|
44
|
-
await using tmp = await tmpdir({
|
|
45
|
-
init: async (dir) => {
|
|
46
|
-
await Bun.write(path.join(dir, "allowed.txt"), "allowed content")
|
|
47
|
-
},
|
|
48
|
-
})
|
|
49
|
-
|
|
50
|
-
await Instance.provide({
|
|
51
|
-
directory: tmp.path,
|
|
52
|
-
fn: async () => {
|
|
53
|
-
await expect(File.read("../../../etc/passwd")).rejects.toThrow("Access denied: path escapes project directory")
|
|
54
|
-
},
|
|
55
|
-
})
|
|
56
|
-
})
|
|
57
|
-
|
|
58
|
-
test("rejects deeply nested traversal", async () => {
|
|
59
|
-
await using tmp = await tmpdir()
|
|
60
|
-
|
|
61
|
-
await Instance.provide({
|
|
62
|
-
directory: tmp.path,
|
|
63
|
-
fn: async () => {
|
|
64
|
-
await expect(File.read("src/nested/../../../../../../../etc/passwd")).rejects.toThrow(
|
|
65
|
-
"Access denied: path escapes project directory",
|
|
66
|
-
)
|
|
67
|
-
},
|
|
68
|
-
})
|
|
69
|
-
})
|
|
70
|
-
|
|
71
|
-
test("allows valid paths within project", async () => {
|
|
72
|
-
await using tmp = await tmpdir({
|
|
73
|
-
init: async (dir) => {
|
|
74
|
-
await Bun.write(path.join(dir, "valid.txt"), "valid content")
|
|
75
|
-
},
|
|
76
|
-
})
|
|
77
|
-
|
|
78
|
-
await Instance.provide({
|
|
79
|
-
directory: tmp.path,
|
|
80
|
-
fn: async () => {
|
|
81
|
-
const result = await File.read("valid.txt")
|
|
82
|
-
expect(result.content).toBe("valid content")
|
|
83
|
-
},
|
|
84
|
-
})
|
|
85
|
-
})
|
|
86
|
-
})
|
|
87
|
-
|
|
88
|
-
describe("File.list path traversal protection", () => {
|
|
89
|
-
test("rejects ../ traversal attempting to list /etc", async () => {
|
|
90
|
-
await using tmp = await tmpdir()
|
|
91
|
-
|
|
92
|
-
await Instance.provide({
|
|
93
|
-
directory: tmp.path,
|
|
94
|
-
fn: async () => {
|
|
95
|
-
await expect(File.list("../../../etc")).rejects.toThrow("Access denied: path escapes project directory")
|
|
96
|
-
},
|
|
97
|
-
})
|
|
98
|
-
})
|
|
99
|
-
|
|
100
|
-
test("allows valid subdirectory listing", async () => {
|
|
101
|
-
await using tmp = await tmpdir({
|
|
102
|
-
init: async (dir) => {
|
|
103
|
-
await Bun.write(path.join(dir, "subdir", "file.txt"), "content")
|
|
104
|
-
},
|
|
105
|
-
})
|
|
106
|
-
|
|
107
|
-
await Instance.provide({
|
|
108
|
-
directory: tmp.path,
|
|
109
|
-
fn: async () => {
|
|
110
|
-
const result = await File.list("subdir")
|
|
111
|
-
expect(Array.isArray(result)).toBe(true)
|
|
112
|
-
},
|
|
113
|
-
})
|
|
114
|
-
})
|
|
115
|
-
})
|
package/test/fixture/fixture.ts
DELETED
|
@@ -1,45 +0,0 @@
|
|
|
1
|
-
import { $ } from "bun"
|
|
2
|
-
import * as fs from "fs/promises"
|
|
3
|
-
import os from "os"
|
|
4
|
-
import path from "path"
|
|
5
|
-
import type { Config } from "../../src/config/config"
|
|
6
|
-
|
|
7
|
-
// Strip null bytes from paths (defensive fix for CI environment issues)
|
|
8
|
-
function sanitizePath(p: string): string {
|
|
9
|
-
return p.replace(/\0/g, "")
|
|
10
|
-
}
|
|
11
|
-
|
|
12
|
-
type TmpDirOptions<T> = {
|
|
13
|
-
git?: boolean
|
|
14
|
-
config?: Partial<Config.Info>
|
|
15
|
-
init?: (dir: string) => Promise<T>
|
|
16
|
-
dispose?: (dir: string) => Promise<T>
|
|
17
|
-
}
|
|
18
|
-
export async function tmpdir<T>(options?: TmpDirOptions<T>) {
|
|
19
|
-
const dirpath = sanitizePath(path.join(os.tmpdir(), "opencode-test-" + Math.random().toString(36).slice(2)))
|
|
20
|
-
await fs.mkdir(dirpath, { recursive: true })
|
|
21
|
-
if (options?.git) {
|
|
22
|
-
await $`git init`.cwd(dirpath).quiet()
|
|
23
|
-
await $`git commit --allow-empty -m "root commit ${dirpath}"`.cwd(dirpath).quiet()
|
|
24
|
-
}
|
|
25
|
-
if (options?.config) {
|
|
26
|
-
await Bun.write(
|
|
27
|
-
path.join(dirpath, "opencode.json"),
|
|
28
|
-
JSON.stringify({
|
|
29
|
-
$schema: "https://opencode.ai/config.json",
|
|
30
|
-
...options.config,
|
|
31
|
-
}),
|
|
32
|
-
)
|
|
33
|
-
}
|
|
34
|
-
const extra = await options?.init?.(dirpath)
|
|
35
|
-
const realpath = sanitizePath(await fs.realpath(dirpath))
|
|
36
|
-
const result = {
|
|
37
|
-
[Symbol.asyncDispose]: async () => {
|
|
38
|
-
await options?.dispose?.(dirpath)
|
|
39
|
-
// await fs.rm(dirpath, { recursive: true, force: true })
|
|
40
|
-
},
|
|
41
|
-
path: realpath,
|
|
42
|
-
extra: extra as T,
|
|
43
|
-
}
|
|
44
|
-
return result
|
|
45
|
-
}
|
|
@@ -1,77 +0,0 @@
|
|
|
1
|
-
// Simple JSON-RPC 2.0 LSP-like fake server over stdio
|
|
2
|
-
// Implements a minimal LSP handshake and triggers a request upon notification
|
|
3
|
-
|
|
4
|
-
const net = require("net")
|
|
5
|
-
|
|
6
|
-
let nextId = 1
|
|
7
|
-
|
|
8
|
-
function encode(message) {
|
|
9
|
-
const json = JSON.stringify(message)
|
|
10
|
-
const header = `Content-Length: ${Buffer.byteLength(json, "utf8")}\r\n\r\n`
|
|
11
|
-
return Buffer.concat([Buffer.from(header, "utf8"), Buffer.from(json, "utf8")])
|
|
12
|
-
}
|
|
13
|
-
|
|
14
|
-
function decodeFrames(buffer) {
|
|
15
|
-
const results = []
|
|
16
|
-
let idx
|
|
17
|
-
while ((idx = buffer.indexOf("\r\n\r\n")) !== -1) {
|
|
18
|
-
const header = buffer.slice(0, idx).toString("utf8")
|
|
19
|
-
const m = /Content-Length:\s*(\d+)/i.exec(header)
|
|
20
|
-
const len = m ? parseInt(m[1], 10) : 0
|
|
21
|
-
const bodyStart = idx + 4
|
|
22
|
-
const bodyEnd = bodyStart + len
|
|
23
|
-
if (buffer.length < bodyEnd) break
|
|
24
|
-
const body = buffer.slice(bodyStart, bodyEnd).toString("utf8")
|
|
25
|
-
results.push(body)
|
|
26
|
-
buffer = buffer.slice(bodyEnd)
|
|
27
|
-
}
|
|
28
|
-
return { messages: results, rest: buffer }
|
|
29
|
-
}
|
|
30
|
-
|
|
31
|
-
let readBuffer = Buffer.alloc(0)
|
|
32
|
-
|
|
33
|
-
process.stdin.on("data", (chunk) => {
|
|
34
|
-
readBuffer = Buffer.concat([readBuffer, chunk])
|
|
35
|
-
const { messages, rest } = decodeFrames(readBuffer)
|
|
36
|
-
readBuffer = rest
|
|
37
|
-
for (const m of messages) handle(m)
|
|
38
|
-
})
|
|
39
|
-
|
|
40
|
-
function send(msg) {
|
|
41
|
-
process.stdout.write(encode(msg))
|
|
42
|
-
}
|
|
43
|
-
|
|
44
|
-
function sendRequest(method, params) {
|
|
45
|
-
const id = nextId++
|
|
46
|
-
send({ jsonrpc: "2.0", id, method, params })
|
|
47
|
-
return id
|
|
48
|
-
}
|
|
49
|
-
|
|
50
|
-
function handle(raw) {
|
|
51
|
-
let data
|
|
52
|
-
try {
|
|
53
|
-
data = JSON.parse(raw)
|
|
54
|
-
} catch {
|
|
55
|
-
return
|
|
56
|
-
}
|
|
57
|
-
if (data.method === "initialize") {
|
|
58
|
-
send({ jsonrpc: "2.0", id: data.id, result: { capabilities: {} } })
|
|
59
|
-
return
|
|
60
|
-
}
|
|
61
|
-
if (data.method === "initialized") {
|
|
62
|
-
return
|
|
63
|
-
}
|
|
64
|
-
if (data.method === "workspace/didChangeConfiguration") {
|
|
65
|
-
return
|
|
66
|
-
}
|
|
67
|
-
if (data.method === "test/trigger") {
|
|
68
|
-
const method = data.params && data.params.method
|
|
69
|
-
if (method) sendRequest(method, {})
|
|
70
|
-
return
|
|
71
|
-
}
|
|
72
|
-
if (typeof data.id !== "undefined") {
|
|
73
|
-
// Respond OK to any request from client to keep transport flowing
|
|
74
|
-
send({ jsonrpc: "2.0", id: data.id, result: null })
|
|
75
|
-
return
|
|
76
|
-
}
|
|
77
|
-
}
|
package/test/ide/ide.test.ts
DELETED
|
@@ -1,82 +0,0 @@
|
|
|
1
|
-
import { describe, expect, test, afterEach } from "bun:test"
|
|
2
|
-
import { Ide } from "../../src/ide"
|
|
3
|
-
|
|
4
|
-
describe("ide", () => {
|
|
5
|
-
const original = structuredClone(process.env)
|
|
6
|
-
|
|
7
|
-
afterEach(() => {
|
|
8
|
-
Object.keys(process.env).forEach((key) => {
|
|
9
|
-
delete process.env[key]
|
|
10
|
-
})
|
|
11
|
-
Object.assign(process.env, original)
|
|
12
|
-
})
|
|
13
|
-
|
|
14
|
-
test("should detect Visual Studio Code", () => {
|
|
15
|
-
process.env["TERM_PROGRAM"] = "vscode"
|
|
16
|
-
process.env["GIT_ASKPASS"] = "/path/to/Visual Studio Code.app/Contents/Resources/app/extensions/git/dist/askpass.sh"
|
|
17
|
-
|
|
18
|
-
expect(Ide.ide()).toBe("Visual Studio Code")
|
|
19
|
-
})
|
|
20
|
-
|
|
21
|
-
test("should detect Visual Studio Code Insiders", () => {
|
|
22
|
-
process.env["TERM_PROGRAM"] = "vscode"
|
|
23
|
-
process.env["GIT_ASKPASS"] =
|
|
24
|
-
"/Applications/Visual Studio Code - Insiders.app/Contents/Resources/app/extensions/git/dist/askpass.sh"
|
|
25
|
-
|
|
26
|
-
expect(Ide.ide()).toBe("Visual Studio Code - Insiders")
|
|
27
|
-
})
|
|
28
|
-
|
|
29
|
-
test("should detect Cursor", () => {
|
|
30
|
-
process.env["TERM_PROGRAM"] = "vscode"
|
|
31
|
-
process.env["GIT_ASKPASS"] = "/path/to/Cursor.app/Contents/Resources/app/extensions/git/dist/askpass.sh"
|
|
32
|
-
|
|
33
|
-
expect(Ide.ide()).toBe("Cursor")
|
|
34
|
-
})
|
|
35
|
-
|
|
36
|
-
test("should detect VSCodium", () => {
|
|
37
|
-
process.env["TERM_PROGRAM"] = "vscode"
|
|
38
|
-
process.env["GIT_ASKPASS"] = "/path/to/VSCodium.app/Contents/Resources/app/extensions/git/dist/askpass.sh"
|
|
39
|
-
|
|
40
|
-
expect(Ide.ide()).toBe("VSCodium")
|
|
41
|
-
})
|
|
42
|
-
|
|
43
|
-
test("should detect Windsurf", () => {
|
|
44
|
-
process.env["TERM_PROGRAM"] = "vscode"
|
|
45
|
-
process.env["GIT_ASKPASS"] = "/path/to/Windsurf.app/Contents/Resources/app/extensions/git/dist/askpass.sh"
|
|
46
|
-
|
|
47
|
-
expect(Ide.ide()).toBe("Windsurf")
|
|
48
|
-
})
|
|
49
|
-
|
|
50
|
-
test("should return unknown when TERM_PROGRAM is not vscode", () => {
|
|
51
|
-
process.env["TERM_PROGRAM"] = "iTerm2"
|
|
52
|
-
process.env["GIT_ASKPASS"] =
|
|
53
|
-
"/Applications/Visual Studio Code - Insiders.app/Contents/Resources/app/extensions/git/dist/askpass.sh"
|
|
54
|
-
|
|
55
|
-
expect(Ide.ide()).toBe("unknown")
|
|
56
|
-
})
|
|
57
|
-
|
|
58
|
-
test("should return unknown when GIT_ASKPASS does not contain IDE name", () => {
|
|
59
|
-
process.env["TERM_PROGRAM"] = "vscode"
|
|
60
|
-
process.env["GIT_ASKPASS"] = "/path/to/unknown/askpass.sh"
|
|
61
|
-
|
|
62
|
-
expect(Ide.ide()).toBe("unknown")
|
|
63
|
-
})
|
|
64
|
-
|
|
65
|
-
test("should recognize vscode-insiders OPENCODE_CALLER", () => {
|
|
66
|
-
process.env["OPENCODE_CALLER"] = "vscode-insiders"
|
|
67
|
-
|
|
68
|
-
expect(Ide.alreadyInstalled()).toBe(true)
|
|
69
|
-
})
|
|
70
|
-
|
|
71
|
-
test("should recognize vscode OPENCODE_CALLER", () => {
|
|
72
|
-
process.env["OPENCODE_CALLER"] = "vscode"
|
|
73
|
-
|
|
74
|
-
expect(Ide.alreadyInstalled()).toBe(true)
|
|
75
|
-
})
|
|
76
|
-
|
|
77
|
-
test("should return false for unknown OPENCODE_CALLER", () => {
|
|
78
|
-
process.env["OPENCODE_CALLER"] = "unknown"
|
|
79
|
-
|
|
80
|
-
expect(Ide.alreadyInstalled()).toBe(false)
|
|
81
|
-
})
|
|
82
|
-
})
|