@stonerzju/opencode 1.2.17 → 1.2.19
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/bin/opencode +29 -157
- package/package.json +29 -29
- package/src/acp/agent.ts +4 -4
- package/src/acp/session.ts +1 -1
- package/src/agent/agent.ts +3 -3
- package/src/bun/index.ts +2 -2
- package/src/cli/cmd/acp.ts +3 -3
- package/src/cli/cmd/debug/file.ts +1 -1
- package/src/cli/cmd/github.ts +2 -2
- package/src/cli/cmd/pr.ts +1 -1
- package/src/cli/cmd/tui/app.tsx +24 -24
- package/src/cli/cmd/tui/attach.ts +3 -3
- package/src/cli/cmd/tui/component/dialog-agent.tsx +3 -3
- package/src/cli/cmd/tui/component/dialog-command.tsx +3 -3
- package/src/cli/cmd/tui/component/dialog-mcp.tsx +5 -5
- package/src/cli/cmd/tui/component/dialog-model.tsx +4 -4
- package/src/cli/cmd/tui/component/dialog-provider.tsx +4 -4
- package/src/cli/cmd/tui/component/dialog-session-list.tsx +5 -5
- package/src/cli/cmd/tui/component/dialog-session-rename.tsx +3 -3
- package/src/cli/cmd/tui/component/dialog-skill.tsx +3 -3
- package/src/cli/cmd/tui/component/dialog-stash.tsx +3 -3
- package/src/cli/cmd/tui/component/dialog-status.tsx +2 -2
- package/src/cli/cmd/tui/component/dialog-tag.tsx +3 -3
- package/src/cli/cmd/tui/component/logo.tsx +2 -2
- package/src/cli/cmd/tui/component/prompt/autocomplete.tsx +6 -6
- package/src/cli/cmd/tui/component/prompt/frecency.tsx +2 -2
- package/src/cli/cmd/tui/component/prompt/history.tsx +2 -2
- package/src/cli/cmd/tui/component/prompt/index.tsx +14 -14
- package/src/cli/cmd/tui/component/prompt/stash.tsx +2 -2
- package/src/cli/cmd/tui/component/textarea-keybindings.ts +1 -1
- package/src/cli/cmd/tui/component/tips.tsx +1 -1
- package/src/cli/cmd/tui/context/directory.ts +1 -1
- package/src/cli/cmd/tui/context/exit.tsx +1 -1
- package/src/cli/cmd/tui/context/keybind.tsx +2 -2
- package/src/cli/cmd/tui/context/kv.tsx +2 -2
- package/src/cli/cmd/tui/context/local.tsx +6 -6
- package/src/cli/cmd/tui/context/sync.tsx +4 -4
- package/src/cli/cmd/tui/context/theme/opencode.json +245 -0
- package/src/cli/cmd/tui/context/theme.tsx +2 -2
- package/src/cli/cmd/tui/context/tui-config.tsx +1 -1
- package/src/cli/cmd/tui/event.ts +2 -2
- package/src/cli/cmd/tui/routes/home.tsx +6 -6
- package/src/cli/cmd/tui/routes/session/dialog-fork-from-timeline.tsx +6 -6
- package/src/cli/cmd/tui/routes/session/dialog-message.tsx +6 -6
- package/src/cli/cmd/tui/routes/session/dialog-subagent.tsx +2 -2
- package/src/cli/cmd/tui/routes/session/dialog-timeline.tsx +3 -3
- package/src/cli/cmd/tui/routes/session/header.tsx +5 -5
- package/src/cli/cmd/tui/routes/session/index.tsx +32 -32
- package/src/cli/cmd/tui/routes/session/permission.tsx +4 -4
- package/src/cli/cmd/tui/routes/session/sidebar.tsx +4 -4
- package/src/cli/cmd/tui/thread.ts +9 -9
- package/src/cli/cmd/tui/ui/dialog-confirm.tsx +1 -1
- package/src/cli/cmd/tui/ui/dialog-help.tsx +2 -2
- package/src/cli/cmd/tui/ui/dialog-select.tsx +5 -5
- package/src/cli/cmd/tui/ui/dialog.tsx +3 -3
- package/src/cli/cmd/tui/ui/toast.tsx +1 -1
- package/src/cli/cmd/tui/util/editor.ts +3 -3
- package/src/cli/cmd/tui/util/transcript.ts +1 -1
- package/src/cli/cmd/tui/worker.ts +10 -10
- package/src/cli/error.ts +1 -1
- package/src/cli/ui.ts +1 -1
- package/src/cli/upgrade.ts +4 -4
- package/src/command/index.ts +1 -1
- package/src/config/config.ts +10 -10
- package/src/config/markdown.ts +1 -1
- package/src/config/migrate-tui-config.ts +5 -5
- package/src/config/paths.ts +4 -4
- package/src/config/tui.ts +4 -4
- package/src/control/control.sql.ts +1 -1
- package/src/control/index.ts +1 -1
- package/src/control-plane/adaptors/worktree.ts +1 -1
- package/src/control-plane/session-proxy-middleware.ts +1 -1
- package/src/control-plane/workspace.sql.ts +1 -1
- package/src/control-plane/workspace.ts +7 -7
- package/src/file/index.ts +1 -1
- package/src/file/ripgrep.ts +2 -2
- package/src/file/watcher.ts +5 -5
- package/src/format/formatter.ts +1 -1
- package/src/ide/index.ts +3 -3
- package/src/index.ts +1 -1
- package/src/installation/index.ts +3 -3
- package/src/lsp/client.ts +3 -3
- package/src/lsp/index.ts +3 -3
- package/src/mcp/index.ts +4 -4
- package/src/permission/index.ts +2 -2
- package/src/permission/next.ts +10 -10
- package/src/plugin/codex.ts +1 -1
- package/src/plugin/copilot.ts +2 -2
- package/src/plugin/index.ts +1 -1
- package/src/project/bootstrap.ts +2 -2
- package/src/project/instance.ts +4 -4
- package/src/project/project.sql.ts +1 -1
- package/src/project/project.ts +5 -5
- package/src/project/state.ts +1 -1
- package/src/project/vcs.ts +4 -4
- package/src/provider/auth.ts +4 -4
- package/src/provider/error.ts +1 -1
- package/src/provider/models-snapshot.ts +2 -0
- package/src/provider/models.ts +1 -1
- package/src/provider/provider.ts +2 -2
- package/src/provider/transform.ts +2 -2
- package/src/pty/index.ts +5 -5
- package/src/question/index.ts +5 -5
- package/src/server/event.ts +1 -1
- package/src/server/mdns.ts +1 -1
- package/src/server/routes/global.ts +3 -3
- package/src/server/routes/permission.ts +1 -1
- package/src/server/routes/pty.ts +1 -1
- package/src/server/routes/session.ts +4 -4
- package/src/server/routes/tui.ts +1 -1
- package/src/server/server.ts +3 -3
- package/src/session/compaction.ts +7 -7
- package/src/session/index.ts +10 -10
- package/src/session/instruction.ts +1 -1
- package/src/session/llm.ts +11 -11
- package/src/session/message-v2.ts +10 -10
- package/src/session/message.ts +1 -1
- package/src/session/processor.ts +10 -10
- package/src/session/prompt.ts +8 -8
- package/src/session/retry.ts +2 -2
- package/src/session/revert.ts +1 -1
- package/src/session/session.sql.ts +3 -3
- package/src/session/status.ts +3 -3
- package/src/session/summary.ts +5 -5
- package/src/session/system.ts +1 -1
- package/src/session/todo.ts +2 -2
- package/src/share/share-next.ts +7 -7
- package/src/share/share.sql.ts +1 -1
- package/src/shell/shell.ts +3 -3
- package/src/skill/skill.ts +6 -6
- package/src/storage/db.ts +1 -1
- package/src/storage/storage.ts +1 -1
- package/src/tool/bash.ts +6 -6
- package/src/tool/edit.ts +1 -1
- package/src/tool/registry.ts +2 -2
- package/src/tool/skill.ts +1 -1
- package/src/tool/task.ts +3 -3
- package/src/util/array.ts +10 -0
- package/src/util/binary.ts +41 -0
- package/src/util/encode.ts +51 -0
- package/src/util/error.ts +54 -0
- package/src/util/identifier.ts +48 -0
- package/src/util/lazy.ts +4 -16
- package/src/util/path.ts +37 -0
- package/src/util/retry.ts +41 -0
- package/src/util/slug.ts +74 -0
- package/src/worktree/index.ts +3 -3
- package/AGENTS.md +0 -10
- package/BUN_SHELL_MIGRATION_PLAN.md +0 -136
- package/Dockerfile +0 -18
- package/README.md +0 -15
- package/bunfig.toml +0 -7
- package/drizzle.config.ts +0 -10
- package/script/build.ts +0 -224
- package/script/check-migrations.ts +0 -16
- package/script/postinstall.mjs +0 -131
- package/script/publish.ts +0 -181
- package/script/schema.ts +0 -63
- package/script/seed-e2e.ts +0 -50
- package/sst-env.d.ts +0 -10
- package/test/AGENTS.md +0 -81
- package/test/acp/agent-interface.test.ts +0 -51
- package/test/acp/event-subscription.test.ts +0 -683
- package/test/agent/agent.test.ts +0 -689
- package/test/bun.test.ts +0 -53
- package/test/cli/github-action.test.ts +0 -197
- package/test/cli/github-remote.test.ts +0 -80
- package/test/cli/import.test.ts +0 -38
- package/test/cli/plugin-auth-picker.test.ts +0 -120
- package/test/cli/tui/transcript.test.ts +0 -322
- package/test/config/agent-color.test.ts +0 -71
- package/test/config/config.test.ts +0 -1886
- package/test/config/fixtures/empty-frontmatter.md +0 -4
- package/test/config/fixtures/frontmatter.md +0 -28
- package/test/config/fixtures/markdown-header.md +0 -11
- package/test/config/fixtures/no-frontmatter.md +0 -1
- package/test/config/fixtures/weird-model-id.md +0 -13
- package/test/config/markdown.test.ts +0 -228
- package/test/config/tui.test.ts +0 -510
- package/test/control-plane/session-proxy-middleware.test.ts +0 -147
- package/test/control-plane/sse.test.ts +0 -56
- package/test/control-plane/workspace-server-sse.test.ts +0 -65
- package/test/control-plane/workspace-sync.test.ts +0 -97
- package/test/file/ignore.test.ts +0 -10
- package/test/file/index.test.ts +0 -394
- package/test/file/path-traversal.test.ts +0 -198
- package/test/file/ripgrep.test.ts +0 -39
- package/test/file/time.test.ts +0 -361
- package/test/fixture/db.ts +0 -11
- package/test/fixture/fixture.ts +0 -45
- package/test/fixture/lsp/fake-lsp-server.js +0 -77
- package/test/fixture/skills/agents-sdk/SKILL.md +0 -152
- package/test/fixture/skills/agents-sdk/references/callable.md +0 -92
- package/test/fixture/skills/cloudflare/SKILL.md +0 -211
- package/test/fixture/skills/index.json +0 -6
- 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/mcp/oauth-browser.test.ts +0 -249
- package/test/memory/abort-leak.test.ts +0 -136
- package/test/patch/patch.test.ts +0 -348
- package/test/permission/arity.test.ts +0 -33
- package/test/permission/next.test.ts +0 -689
- package/test/permission-task.test.ts +0 -319
- package/test/plugin/auth-override.test.ts +0 -44
- package/test/plugin/codex.test.ts +0 -123
- package/test/preload.ts +0 -80
- package/test/project/project.test.ts +0 -348
- package/test/project/worktree-remove.test.ts +0 -65
- package/test/provider/amazon-bedrock.test.ts +0 -446
- package/test/provider/copilot/convert-to-copilot-messages.test.ts +0 -523
- package/test/provider/copilot/copilot-chat-model.test.ts +0 -592
- package/test/provider/gitlab-duo.test.ts +0 -262
- package/test/provider/provider.test.ts +0 -2220
- package/test/provider/transform.test.ts +0 -2353
- package/test/pty/pty-output-isolation.test.ts +0 -140
- package/test/question/question.test.ts +0 -300
- package/test/scheduler.test.ts +0 -73
- package/test/server/global-session-list.test.ts +0 -89
- package/test/server/session-list.test.ts +0 -90
- package/test/server/session-select.test.ts +0 -78
- package/test/session/compaction.test.ts +0 -423
- package/test/session/instruction.test.ts +0 -170
- package/test/session/llm.test.ts +0 -667
- package/test/session/message-v2.test.ts +0 -924
- package/test/session/prompt.test.ts +0 -211
- package/test/session/retry.test.ts +0 -188
- package/test/session/revert-compact.test.ts +0 -285
- package/test/session/session.test.ts +0 -71
- package/test/session/structured-output-integration.test.ts +0 -233
- package/test/session/structured-output.test.ts +0 -385
- package/test/skill/discovery.test.ts +0 -110
- package/test/skill/skill.test.ts +0 -388
- package/test/snapshot/snapshot.test.ts +0 -1180
- package/test/storage/json-migration.test.ts +0 -846
- package/test/tool/__snapshots__/tool.test.ts.snap +0 -9
- package/test/tool/apply_patch.test.ts +0 -566
- package/test/tool/bash.test.ts +0 -402
- package/test/tool/edit.test.ts +0 -496
- package/test/tool/external-directory.test.ts +0 -127
- package/test/tool/fixtures/large-image.png +0 -0
- package/test/tool/fixtures/models-api.json +0 -38413
- package/test/tool/grep.test.ts +0 -110
- package/test/tool/question.test.ts +0 -107
- package/test/tool/read.test.ts +0 -504
- package/test/tool/registry.test.ts +0 -122
- package/test/tool/skill.test.ts +0 -112
- package/test/tool/truncation.test.ts +0 -160
- package/test/tool/webfetch.test.ts +0 -100
- package/test/tool/write.test.ts +0 -348
- package/test/util/filesystem.test.ts +0 -443
- package/test/util/format.test.ts +0 -59
- package/test/util/glob.test.ts +0 -164
- package/test/util/iife.test.ts +0 -36
- package/test/util/lazy.test.ts +0 -50
- package/test/util/lock.test.ts +0 -72
- package/test/util/process.test.ts +0 -59
- package/test/util/timeout.test.ts +0 -21
- package/test/util/wildcard.test.ts +0 -90
- package/tsconfig.json +0 -16
|
@@ -1,348 +0,0 @@
|
|
|
1
|
-
import { describe, expect, mock, test } from "bun:test"
|
|
2
|
-
import { Project } from "../../src/project/project"
|
|
3
|
-
import { Log } from "../../src/util/log"
|
|
4
|
-
import { $ } from "bun"
|
|
5
|
-
import path from "path"
|
|
6
|
-
import { tmpdir } from "../fixture/fixture"
|
|
7
|
-
import { Filesystem } from "../../src/util/filesystem"
|
|
8
|
-
import { GlobalBus } from "../../src/bus/global"
|
|
9
|
-
|
|
10
|
-
Log.init({ print: false })
|
|
11
|
-
|
|
12
|
-
const gitModule = await import("../../src/util/git")
|
|
13
|
-
const originalGit = gitModule.git
|
|
14
|
-
|
|
15
|
-
type Mode = "none" | "rev-list-fail" | "top-fail" | "common-dir-fail"
|
|
16
|
-
let mode: Mode = "none"
|
|
17
|
-
|
|
18
|
-
mock.module("../../src/util/git", () => ({
|
|
19
|
-
git: (args: string[], opts: { cwd: string; env?: Record<string, string> }) => {
|
|
20
|
-
const cmd = ["git", ...args].join(" ")
|
|
21
|
-
if (
|
|
22
|
-
mode === "rev-list-fail" &&
|
|
23
|
-
cmd.includes("git rev-list") &&
|
|
24
|
-
cmd.includes("--max-parents=0") &&
|
|
25
|
-
cmd.includes("--all")
|
|
26
|
-
) {
|
|
27
|
-
return Promise.resolve({
|
|
28
|
-
exitCode: 128,
|
|
29
|
-
text: () => Promise.resolve(""),
|
|
30
|
-
stdout: Buffer.from(""),
|
|
31
|
-
stderr: Buffer.from("fatal"),
|
|
32
|
-
})
|
|
33
|
-
}
|
|
34
|
-
if (mode === "top-fail" && cmd.includes("git rev-parse") && cmd.includes("--show-toplevel")) {
|
|
35
|
-
return Promise.resolve({
|
|
36
|
-
exitCode: 128,
|
|
37
|
-
text: () => Promise.resolve(""),
|
|
38
|
-
stdout: Buffer.from(""),
|
|
39
|
-
stderr: Buffer.from("fatal"),
|
|
40
|
-
})
|
|
41
|
-
}
|
|
42
|
-
if (mode === "common-dir-fail" && cmd.includes("git rev-parse") && cmd.includes("--git-common-dir")) {
|
|
43
|
-
return Promise.resolve({
|
|
44
|
-
exitCode: 128,
|
|
45
|
-
text: () => Promise.resolve(""),
|
|
46
|
-
stdout: Buffer.from(""),
|
|
47
|
-
stderr: Buffer.from("fatal"),
|
|
48
|
-
})
|
|
49
|
-
}
|
|
50
|
-
return originalGit(args, opts)
|
|
51
|
-
},
|
|
52
|
-
}))
|
|
53
|
-
|
|
54
|
-
async function withMode(next: Mode, run: () => Promise<void>) {
|
|
55
|
-
const prev = mode
|
|
56
|
-
mode = next
|
|
57
|
-
try {
|
|
58
|
-
await run()
|
|
59
|
-
} finally {
|
|
60
|
-
mode = prev
|
|
61
|
-
}
|
|
62
|
-
}
|
|
63
|
-
|
|
64
|
-
async function loadProject() {
|
|
65
|
-
return (await import("../../src/project/project")).Project
|
|
66
|
-
}
|
|
67
|
-
|
|
68
|
-
describe("Project.fromDirectory", () => {
|
|
69
|
-
test("should handle git repository with no commits", async () => {
|
|
70
|
-
const p = await loadProject()
|
|
71
|
-
await using tmp = await tmpdir()
|
|
72
|
-
await $`git init`.cwd(tmp.path).quiet()
|
|
73
|
-
|
|
74
|
-
const { project } = await p.fromDirectory(tmp.path)
|
|
75
|
-
|
|
76
|
-
expect(project).toBeDefined()
|
|
77
|
-
expect(project.id).toBe("global")
|
|
78
|
-
expect(project.vcs).toBe("git")
|
|
79
|
-
expect(project.worktree).toBe(tmp.path)
|
|
80
|
-
|
|
81
|
-
const opencodeFile = path.join(tmp.path, ".git", "opencode")
|
|
82
|
-
const fileExists = await Filesystem.exists(opencodeFile)
|
|
83
|
-
expect(fileExists).toBe(false)
|
|
84
|
-
})
|
|
85
|
-
|
|
86
|
-
test("should handle git repository with commits", async () => {
|
|
87
|
-
const p = await loadProject()
|
|
88
|
-
await using tmp = await tmpdir({ git: true })
|
|
89
|
-
|
|
90
|
-
const { project } = await p.fromDirectory(tmp.path)
|
|
91
|
-
|
|
92
|
-
expect(project).toBeDefined()
|
|
93
|
-
expect(project.id).not.toBe("global")
|
|
94
|
-
expect(project.vcs).toBe("git")
|
|
95
|
-
expect(project.worktree).toBe(tmp.path)
|
|
96
|
-
|
|
97
|
-
const opencodeFile = path.join(tmp.path, ".git", "opencode")
|
|
98
|
-
const fileExists = await Filesystem.exists(opencodeFile)
|
|
99
|
-
expect(fileExists).toBe(true)
|
|
100
|
-
})
|
|
101
|
-
|
|
102
|
-
test("keeps git vcs when rev-list exits non-zero with empty output", async () => {
|
|
103
|
-
const p = await loadProject()
|
|
104
|
-
await using tmp = await tmpdir()
|
|
105
|
-
await $`git init`.cwd(tmp.path).quiet()
|
|
106
|
-
|
|
107
|
-
await withMode("rev-list-fail", async () => {
|
|
108
|
-
const { project } = await p.fromDirectory(tmp.path)
|
|
109
|
-
expect(project.vcs).toBe("git")
|
|
110
|
-
expect(project.id).toBe("global")
|
|
111
|
-
expect(project.worktree).toBe(tmp.path)
|
|
112
|
-
})
|
|
113
|
-
})
|
|
114
|
-
|
|
115
|
-
test("keeps git vcs when show-toplevel exits non-zero with empty output", async () => {
|
|
116
|
-
const p = await loadProject()
|
|
117
|
-
await using tmp = await tmpdir({ git: true })
|
|
118
|
-
|
|
119
|
-
await withMode("top-fail", async () => {
|
|
120
|
-
const { project, sandbox } = await p.fromDirectory(tmp.path)
|
|
121
|
-
expect(project.vcs).toBe("git")
|
|
122
|
-
expect(project.worktree).toBe(tmp.path)
|
|
123
|
-
expect(sandbox).toBe(tmp.path)
|
|
124
|
-
})
|
|
125
|
-
})
|
|
126
|
-
|
|
127
|
-
test("keeps git vcs when git-common-dir exits non-zero with empty output", async () => {
|
|
128
|
-
const p = await loadProject()
|
|
129
|
-
await using tmp = await tmpdir({ git: true })
|
|
130
|
-
|
|
131
|
-
await withMode("common-dir-fail", async () => {
|
|
132
|
-
const { project, sandbox } = await p.fromDirectory(tmp.path)
|
|
133
|
-
expect(project.vcs).toBe("git")
|
|
134
|
-
expect(project.worktree).toBe(tmp.path)
|
|
135
|
-
expect(sandbox).toBe(tmp.path)
|
|
136
|
-
})
|
|
137
|
-
})
|
|
138
|
-
})
|
|
139
|
-
|
|
140
|
-
describe("Project.fromDirectory with worktrees", () => {
|
|
141
|
-
test("should set worktree to root when called from root", async () => {
|
|
142
|
-
const p = await loadProject()
|
|
143
|
-
await using tmp = await tmpdir({ git: true })
|
|
144
|
-
|
|
145
|
-
const { project, sandbox } = await p.fromDirectory(tmp.path)
|
|
146
|
-
|
|
147
|
-
expect(project.worktree).toBe(tmp.path)
|
|
148
|
-
expect(sandbox).toBe(tmp.path)
|
|
149
|
-
expect(project.sandboxes).not.toContain(tmp.path)
|
|
150
|
-
})
|
|
151
|
-
|
|
152
|
-
test("should set worktree to root when called from a worktree", async () => {
|
|
153
|
-
const p = await loadProject()
|
|
154
|
-
await using tmp = await tmpdir({ git: true })
|
|
155
|
-
|
|
156
|
-
const worktreePath = path.join(tmp.path, "..", path.basename(tmp.path) + "-worktree")
|
|
157
|
-
try {
|
|
158
|
-
await $`git worktree add ${worktreePath} -b test-branch-${Date.now()}`.cwd(tmp.path).quiet()
|
|
159
|
-
|
|
160
|
-
const { project, sandbox } = await p.fromDirectory(worktreePath)
|
|
161
|
-
|
|
162
|
-
expect(project.worktree).toBe(tmp.path)
|
|
163
|
-
expect(sandbox).toBe(worktreePath)
|
|
164
|
-
expect(project.sandboxes).toContain(worktreePath)
|
|
165
|
-
expect(project.sandboxes).not.toContain(tmp.path)
|
|
166
|
-
} finally {
|
|
167
|
-
await $`git worktree remove ${worktreePath}`
|
|
168
|
-
.cwd(tmp.path)
|
|
169
|
-
.quiet()
|
|
170
|
-
.catch(() => {})
|
|
171
|
-
}
|
|
172
|
-
})
|
|
173
|
-
|
|
174
|
-
test("should accumulate multiple worktrees in sandboxes", async () => {
|
|
175
|
-
const p = await loadProject()
|
|
176
|
-
await using tmp = await tmpdir({ git: true })
|
|
177
|
-
|
|
178
|
-
const worktree1 = path.join(tmp.path, "..", path.basename(tmp.path) + "-wt1")
|
|
179
|
-
const worktree2 = path.join(tmp.path, "..", path.basename(tmp.path) + "-wt2")
|
|
180
|
-
try {
|
|
181
|
-
await $`git worktree add ${worktree1} -b branch-${Date.now()}`.cwd(tmp.path).quiet()
|
|
182
|
-
await $`git worktree add ${worktree2} -b branch-${Date.now() + 1}`.cwd(tmp.path).quiet()
|
|
183
|
-
|
|
184
|
-
await p.fromDirectory(worktree1)
|
|
185
|
-
const { project } = await p.fromDirectory(worktree2)
|
|
186
|
-
|
|
187
|
-
expect(project.worktree).toBe(tmp.path)
|
|
188
|
-
expect(project.sandboxes).toContain(worktree1)
|
|
189
|
-
expect(project.sandboxes).toContain(worktree2)
|
|
190
|
-
expect(project.sandboxes).not.toContain(tmp.path)
|
|
191
|
-
} finally {
|
|
192
|
-
await $`git worktree remove ${worktree1}`
|
|
193
|
-
.cwd(tmp.path)
|
|
194
|
-
.quiet()
|
|
195
|
-
.catch(() => {})
|
|
196
|
-
await $`git worktree remove ${worktree2}`
|
|
197
|
-
.cwd(tmp.path)
|
|
198
|
-
.quiet()
|
|
199
|
-
.catch(() => {})
|
|
200
|
-
}
|
|
201
|
-
})
|
|
202
|
-
})
|
|
203
|
-
|
|
204
|
-
describe("Project.discover", () => {
|
|
205
|
-
test("should discover favicon.png in root", async () => {
|
|
206
|
-
const p = await loadProject()
|
|
207
|
-
await using tmp = await tmpdir({ git: true })
|
|
208
|
-
const { project } = await p.fromDirectory(tmp.path)
|
|
209
|
-
|
|
210
|
-
const pngData = Buffer.from([0x89, 0x50, 0x4e, 0x47, 0x0d, 0x0a, 0x1a, 0x0a])
|
|
211
|
-
await Bun.write(path.join(tmp.path, "favicon.png"), pngData)
|
|
212
|
-
|
|
213
|
-
await p.discover(project)
|
|
214
|
-
|
|
215
|
-
const updated = Project.get(project.id)
|
|
216
|
-
expect(updated).toBeDefined()
|
|
217
|
-
expect(updated!.icon).toBeDefined()
|
|
218
|
-
expect(updated!.icon?.url).toStartWith("data:")
|
|
219
|
-
expect(updated!.icon?.url).toContain("base64")
|
|
220
|
-
expect(updated!.icon?.color).toBeUndefined()
|
|
221
|
-
})
|
|
222
|
-
|
|
223
|
-
test("should not discover non-image files", async () => {
|
|
224
|
-
const p = await loadProject()
|
|
225
|
-
await using tmp = await tmpdir({ git: true })
|
|
226
|
-
const { project } = await p.fromDirectory(tmp.path)
|
|
227
|
-
|
|
228
|
-
await Bun.write(path.join(tmp.path, "favicon.txt"), "not an image")
|
|
229
|
-
|
|
230
|
-
await p.discover(project)
|
|
231
|
-
|
|
232
|
-
const updated = Project.get(project.id)
|
|
233
|
-
expect(updated).toBeDefined()
|
|
234
|
-
expect(updated!.icon).toBeUndefined()
|
|
235
|
-
})
|
|
236
|
-
})
|
|
237
|
-
|
|
238
|
-
describe("Project.update", () => {
|
|
239
|
-
test("should update name", async () => {
|
|
240
|
-
await using tmp = await tmpdir({ git: true })
|
|
241
|
-
const { project } = await Project.fromDirectory(tmp.path)
|
|
242
|
-
|
|
243
|
-
const updated = await Project.update({
|
|
244
|
-
projectID: project.id,
|
|
245
|
-
name: "New Project Name",
|
|
246
|
-
})
|
|
247
|
-
|
|
248
|
-
expect(updated.name).toBe("New Project Name")
|
|
249
|
-
|
|
250
|
-
const fromDb = Project.get(project.id)
|
|
251
|
-
expect(fromDb?.name).toBe("New Project Name")
|
|
252
|
-
})
|
|
253
|
-
|
|
254
|
-
test("should update icon url", async () => {
|
|
255
|
-
await using tmp = await tmpdir({ git: true })
|
|
256
|
-
const { project } = await Project.fromDirectory(tmp.path)
|
|
257
|
-
|
|
258
|
-
const updated = await Project.update({
|
|
259
|
-
projectID: project.id,
|
|
260
|
-
icon: { url: "https://example.com/icon.png" },
|
|
261
|
-
})
|
|
262
|
-
|
|
263
|
-
expect(updated.icon?.url).toBe("https://example.com/icon.png")
|
|
264
|
-
|
|
265
|
-
const fromDb = Project.get(project.id)
|
|
266
|
-
expect(fromDb?.icon?.url).toBe("https://example.com/icon.png")
|
|
267
|
-
})
|
|
268
|
-
|
|
269
|
-
test("should update icon color", async () => {
|
|
270
|
-
await using tmp = await tmpdir({ git: true })
|
|
271
|
-
const { project } = await Project.fromDirectory(tmp.path)
|
|
272
|
-
|
|
273
|
-
const updated = await Project.update({
|
|
274
|
-
projectID: project.id,
|
|
275
|
-
icon: { color: "#ff0000" },
|
|
276
|
-
})
|
|
277
|
-
|
|
278
|
-
expect(updated.icon?.color).toBe("#ff0000")
|
|
279
|
-
|
|
280
|
-
const fromDb = Project.get(project.id)
|
|
281
|
-
expect(fromDb?.icon?.color).toBe("#ff0000")
|
|
282
|
-
})
|
|
283
|
-
|
|
284
|
-
test("should update commands", async () => {
|
|
285
|
-
await using tmp = await tmpdir({ git: true })
|
|
286
|
-
const { project } = await Project.fromDirectory(tmp.path)
|
|
287
|
-
|
|
288
|
-
const updated = await Project.update({
|
|
289
|
-
projectID: project.id,
|
|
290
|
-
commands: { start: "npm run dev" },
|
|
291
|
-
})
|
|
292
|
-
|
|
293
|
-
expect(updated.commands?.start).toBe("npm run dev")
|
|
294
|
-
|
|
295
|
-
const fromDb = Project.get(project.id)
|
|
296
|
-
expect(fromDb?.commands?.start).toBe("npm run dev")
|
|
297
|
-
})
|
|
298
|
-
|
|
299
|
-
test("should throw error when project not found", async () => {
|
|
300
|
-
await using tmp = await tmpdir({ git: true })
|
|
301
|
-
|
|
302
|
-
await expect(
|
|
303
|
-
Project.update({
|
|
304
|
-
projectID: "nonexistent-project-id",
|
|
305
|
-
name: "Should Fail",
|
|
306
|
-
}),
|
|
307
|
-
).rejects.toThrow("Project not found: nonexistent-project-id")
|
|
308
|
-
})
|
|
309
|
-
|
|
310
|
-
test("should emit GlobalBus event on update", async () => {
|
|
311
|
-
await using tmp = await tmpdir({ git: true })
|
|
312
|
-
const { project } = await Project.fromDirectory(tmp.path)
|
|
313
|
-
|
|
314
|
-
let eventFired = false
|
|
315
|
-
let eventPayload: any = null
|
|
316
|
-
|
|
317
|
-
GlobalBus.on("event", (data) => {
|
|
318
|
-
eventFired = true
|
|
319
|
-
eventPayload = data
|
|
320
|
-
})
|
|
321
|
-
|
|
322
|
-
await Project.update({
|
|
323
|
-
projectID: project.id,
|
|
324
|
-
name: "Updated Name",
|
|
325
|
-
})
|
|
326
|
-
|
|
327
|
-
expect(eventFired).toBe(true)
|
|
328
|
-
expect(eventPayload.payload.type).toBe("project.updated")
|
|
329
|
-
expect(eventPayload.payload.properties.name).toBe("Updated Name")
|
|
330
|
-
})
|
|
331
|
-
|
|
332
|
-
test("should update multiple fields at once", async () => {
|
|
333
|
-
await using tmp = await tmpdir({ git: true })
|
|
334
|
-
const { project } = await Project.fromDirectory(tmp.path)
|
|
335
|
-
|
|
336
|
-
const updated = await Project.update({
|
|
337
|
-
projectID: project.id,
|
|
338
|
-
name: "Multi Update",
|
|
339
|
-
icon: { url: "https://example.com/favicon.ico", color: "#00ff00" },
|
|
340
|
-
commands: { start: "make start" },
|
|
341
|
-
})
|
|
342
|
-
|
|
343
|
-
expect(updated.name).toBe("Multi Update")
|
|
344
|
-
expect(updated.icon?.url).toBe("https://example.com/favicon.ico")
|
|
345
|
-
expect(updated.icon?.color).toBe("#00ff00")
|
|
346
|
-
expect(updated.commands?.start).toBe("make start")
|
|
347
|
-
})
|
|
348
|
-
})
|
|
@@ -1,65 +0,0 @@
|
|
|
1
|
-
import { describe, expect, test } from "bun:test"
|
|
2
|
-
import { $ } from "bun"
|
|
3
|
-
import fs from "fs/promises"
|
|
4
|
-
import path from "path"
|
|
5
|
-
import { Instance } from "../../src/project/instance"
|
|
6
|
-
import { Worktree } from "../../src/worktree"
|
|
7
|
-
import { Filesystem } from "../../src/util/filesystem"
|
|
8
|
-
import { tmpdir } from "../fixture/fixture"
|
|
9
|
-
|
|
10
|
-
describe("Worktree.remove", () => {
|
|
11
|
-
test("continues when git remove exits non-zero after detaching", async () => {
|
|
12
|
-
await using tmp = await tmpdir({ git: true })
|
|
13
|
-
const root = tmp.path
|
|
14
|
-
const name = `remove-regression-${Date.now().toString(36)}`
|
|
15
|
-
const branch = `opencode/${name}`
|
|
16
|
-
const dir = path.join(root, "..", name)
|
|
17
|
-
|
|
18
|
-
await $`git worktree add --no-checkout -b ${branch} ${dir}`.cwd(root).quiet()
|
|
19
|
-
await $`git reset --hard`.cwd(dir).quiet()
|
|
20
|
-
|
|
21
|
-
const real = (await $`which git`.quiet().text()).trim()
|
|
22
|
-
expect(real).toBeTruthy()
|
|
23
|
-
|
|
24
|
-
const bin = path.join(root, "bin")
|
|
25
|
-
const shim = path.join(bin, "git")
|
|
26
|
-
await fs.mkdir(bin, { recursive: true })
|
|
27
|
-
await Bun.write(
|
|
28
|
-
shim,
|
|
29
|
-
[
|
|
30
|
-
"#!/bin/bash",
|
|
31
|
-
`REAL_GIT=${JSON.stringify(real)}`,
|
|
32
|
-
'if [ "$1" = "worktree" ] && [ "$2" = "remove" ]; then',
|
|
33
|
-
' "$REAL_GIT" "$@" >/dev/null 2>&1',
|
|
34
|
-
' echo "fatal: failed to remove worktree: Directory not empty" >&2',
|
|
35
|
-
" exit 1",
|
|
36
|
-
"fi",
|
|
37
|
-
'exec "$REAL_GIT" "$@"',
|
|
38
|
-
].join("\n"),
|
|
39
|
-
)
|
|
40
|
-
await fs.chmod(shim, 0o755)
|
|
41
|
-
|
|
42
|
-
const prev = process.env.PATH ?? ""
|
|
43
|
-
process.env.PATH = `${bin}${path.delimiter}${prev}`
|
|
44
|
-
|
|
45
|
-
const ok = await (async () => {
|
|
46
|
-
try {
|
|
47
|
-
return await Instance.provide({
|
|
48
|
-
directory: root,
|
|
49
|
-
fn: () => Worktree.remove({ directory: dir }),
|
|
50
|
-
})
|
|
51
|
-
} finally {
|
|
52
|
-
process.env.PATH = prev
|
|
53
|
-
}
|
|
54
|
-
})()
|
|
55
|
-
|
|
56
|
-
expect(ok).toBe(true)
|
|
57
|
-
expect(await Filesystem.exists(dir)).toBe(false)
|
|
58
|
-
|
|
59
|
-
const list = await $`git worktree list --porcelain`.cwd(root).quiet().text()
|
|
60
|
-
expect(list).not.toContain(`worktree ${dir}`)
|
|
61
|
-
|
|
62
|
-
const ref = await $`git show-ref --verify --quiet refs/heads/${branch}`.cwd(root).quiet().nothrow()
|
|
63
|
-
expect(ref.exitCode).not.toBe(0)
|
|
64
|
-
})
|
|
65
|
-
})
|