@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.
Files changed (261) hide show
  1. package/bin/opencode +29 -157
  2. package/package.json +29 -29
  3. package/src/acp/agent.ts +4 -4
  4. package/src/acp/session.ts +1 -1
  5. package/src/agent/agent.ts +3 -3
  6. package/src/bun/index.ts +2 -2
  7. package/src/cli/cmd/acp.ts +3 -3
  8. package/src/cli/cmd/debug/file.ts +1 -1
  9. package/src/cli/cmd/github.ts +2 -2
  10. package/src/cli/cmd/pr.ts +1 -1
  11. package/src/cli/cmd/tui/app.tsx +24 -24
  12. package/src/cli/cmd/tui/attach.ts +3 -3
  13. package/src/cli/cmd/tui/component/dialog-agent.tsx +3 -3
  14. package/src/cli/cmd/tui/component/dialog-command.tsx +3 -3
  15. package/src/cli/cmd/tui/component/dialog-mcp.tsx +5 -5
  16. package/src/cli/cmd/tui/component/dialog-model.tsx +4 -4
  17. package/src/cli/cmd/tui/component/dialog-provider.tsx +4 -4
  18. package/src/cli/cmd/tui/component/dialog-session-list.tsx +5 -5
  19. package/src/cli/cmd/tui/component/dialog-session-rename.tsx +3 -3
  20. package/src/cli/cmd/tui/component/dialog-skill.tsx +3 -3
  21. package/src/cli/cmd/tui/component/dialog-stash.tsx +3 -3
  22. package/src/cli/cmd/tui/component/dialog-status.tsx +2 -2
  23. package/src/cli/cmd/tui/component/dialog-tag.tsx +3 -3
  24. package/src/cli/cmd/tui/component/logo.tsx +2 -2
  25. package/src/cli/cmd/tui/component/prompt/autocomplete.tsx +6 -6
  26. package/src/cli/cmd/tui/component/prompt/frecency.tsx +2 -2
  27. package/src/cli/cmd/tui/component/prompt/history.tsx +2 -2
  28. package/src/cli/cmd/tui/component/prompt/index.tsx +14 -14
  29. package/src/cli/cmd/tui/component/prompt/stash.tsx +2 -2
  30. package/src/cli/cmd/tui/component/textarea-keybindings.ts +1 -1
  31. package/src/cli/cmd/tui/component/tips.tsx +1 -1
  32. package/src/cli/cmd/tui/context/directory.ts +1 -1
  33. package/src/cli/cmd/tui/context/exit.tsx +1 -1
  34. package/src/cli/cmd/tui/context/keybind.tsx +2 -2
  35. package/src/cli/cmd/tui/context/kv.tsx +2 -2
  36. package/src/cli/cmd/tui/context/local.tsx +6 -6
  37. package/src/cli/cmd/tui/context/sync.tsx +4 -4
  38. package/src/cli/cmd/tui/context/theme/opencode.json +245 -0
  39. package/src/cli/cmd/tui/context/theme.tsx +2 -2
  40. package/src/cli/cmd/tui/context/tui-config.tsx +1 -1
  41. package/src/cli/cmd/tui/event.ts +2 -2
  42. package/src/cli/cmd/tui/routes/home.tsx +6 -6
  43. package/src/cli/cmd/tui/routes/session/dialog-fork-from-timeline.tsx +6 -6
  44. package/src/cli/cmd/tui/routes/session/dialog-message.tsx +6 -6
  45. package/src/cli/cmd/tui/routes/session/dialog-subagent.tsx +2 -2
  46. package/src/cli/cmd/tui/routes/session/dialog-timeline.tsx +3 -3
  47. package/src/cli/cmd/tui/routes/session/header.tsx +5 -5
  48. package/src/cli/cmd/tui/routes/session/index.tsx +32 -32
  49. package/src/cli/cmd/tui/routes/session/permission.tsx +4 -4
  50. package/src/cli/cmd/tui/routes/session/sidebar.tsx +4 -4
  51. package/src/cli/cmd/tui/thread.ts +9 -9
  52. package/src/cli/cmd/tui/ui/dialog-confirm.tsx +1 -1
  53. package/src/cli/cmd/tui/ui/dialog-help.tsx +2 -2
  54. package/src/cli/cmd/tui/ui/dialog-select.tsx +5 -5
  55. package/src/cli/cmd/tui/ui/dialog.tsx +3 -3
  56. package/src/cli/cmd/tui/ui/toast.tsx +1 -1
  57. package/src/cli/cmd/tui/util/editor.ts +3 -3
  58. package/src/cli/cmd/tui/util/transcript.ts +1 -1
  59. package/src/cli/cmd/tui/worker.ts +10 -10
  60. package/src/cli/error.ts +1 -1
  61. package/src/cli/ui.ts +1 -1
  62. package/src/cli/upgrade.ts +4 -4
  63. package/src/command/index.ts +1 -1
  64. package/src/config/config.ts +10 -10
  65. package/src/config/markdown.ts +1 -1
  66. package/src/config/migrate-tui-config.ts +5 -5
  67. package/src/config/paths.ts +4 -4
  68. package/src/config/tui.ts +4 -4
  69. package/src/control/control.sql.ts +1 -1
  70. package/src/control/index.ts +1 -1
  71. package/src/control-plane/adaptors/worktree.ts +1 -1
  72. package/src/control-plane/session-proxy-middleware.ts +1 -1
  73. package/src/control-plane/workspace.sql.ts +1 -1
  74. package/src/control-plane/workspace.ts +7 -7
  75. package/src/file/index.ts +1 -1
  76. package/src/file/ripgrep.ts +2 -2
  77. package/src/file/watcher.ts +5 -5
  78. package/src/format/formatter.ts +1 -1
  79. package/src/ide/index.ts +3 -3
  80. package/src/index.ts +1 -1
  81. package/src/installation/index.ts +3 -3
  82. package/src/lsp/client.ts +3 -3
  83. package/src/lsp/index.ts +3 -3
  84. package/src/mcp/index.ts +4 -4
  85. package/src/permission/index.ts +2 -2
  86. package/src/permission/next.ts +10 -10
  87. package/src/plugin/codex.ts +1 -1
  88. package/src/plugin/copilot.ts +2 -2
  89. package/src/plugin/index.ts +1 -1
  90. package/src/project/bootstrap.ts +2 -2
  91. package/src/project/instance.ts +4 -4
  92. package/src/project/project.sql.ts +1 -1
  93. package/src/project/project.ts +5 -5
  94. package/src/project/state.ts +1 -1
  95. package/src/project/vcs.ts +4 -4
  96. package/src/provider/auth.ts +4 -4
  97. package/src/provider/error.ts +1 -1
  98. package/src/provider/models-snapshot.ts +2 -0
  99. package/src/provider/models.ts +1 -1
  100. package/src/provider/provider.ts +2 -2
  101. package/src/provider/transform.ts +2 -2
  102. package/src/pty/index.ts +5 -5
  103. package/src/question/index.ts +5 -5
  104. package/src/server/event.ts +1 -1
  105. package/src/server/mdns.ts +1 -1
  106. package/src/server/routes/global.ts +3 -3
  107. package/src/server/routes/permission.ts +1 -1
  108. package/src/server/routes/pty.ts +1 -1
  109. package/src/server/routes/session.ts +4 -4
  110. package/src/server/routes/tui.ts +1 -1
  111. package/src/server/server.ts +3 -3
  112. package/src/session/compaction.ts +7 -7
  113. package/src/session/index.ts +10 -10
  114. package/src/session/instruction.ts +1 -1
  115. package/src/session/llm.ts +11 -11
  116. package/src/session/message-v2.ts +10 -10
  117. package/src/session/message.ts +1 -1
  118. package/src/session/processor.ts +10 -10
  119. package/src/session/prompt.ts +8 -8
  120. package/src/session/retry.ts +2 -2
  121. package/src/session/revert.ts +1 -1
  122. package/src/session/session.sql.ts +3 -3
  123. package/src/session/status.ts +3 -3
  124. package/src/session/summary.ts +5 -5
  125. package/src/session/system.ts +1 -1
  126. package/src/session/todo.ts +2 -2
  127. package/src/share/share-next.ts +7 -7
  128. package/src/share/share.sql.ts +1 -1
  129. package/src/shell/shell.ts +3 -3
  130. package/src/skill/skill.ts +6 -6
  131. package/src/storage/db.ts +1 -1
  132. package/src/storage/storage.ts +1 -1
  133. package/src/tool/bash.ts +6 -6
  134. package/src/tool/edit.ts +1 -1
  135. package/src/tool/registry.ts +2 -2
  136. package/src/tool/skill.ts +1 -1
  137. package/src/tool/task.ts +3 -3
  138. package/src/util/array.ts +10 -0
  139. package/src/util/binary.ts +41 -0
  140. package/src/util/encode.ts +51 -0
  141. package/src/util/error.ts +54 -0
  142. package/src/util/identifier.ts +48 -0
  143. package/src/util/lazy.ts +4 -16
  144. package/src/util/path.ts +37 -0
  145. package/src/util/retry.ts +41 -0
  146. package/src/util/slug.ts +74 -0
  147. package/src/worktree/index.ts +3 -3
  148. package/AGENTS.md +0 -10
  149. package/BUN_SHELL_MIGRATION_PLAN.md +0 -136
  150. package/Dockerfile +0 -18
  151. package/README.md +0 -15
  152. package/bunfig.toml +0 -7
  153. package/drizzle.config.ts +0 -10
  154. package/script/build.ts +0 -224
  155. package/script/check-migrations.ts +0 -16
  156. package/script/postinstall.mjs +0 -131
  157. package/script/publish.ts +0 -181
  158. package/script/schema.ts +0 -63
  159. package/script/seed-e2e.ts +0 -50
  160. package/sst-env.d.ts +0 -10
  161. package/test/AGENTS.md +0 -81
  162. package/test/acp/agent-interface.test.ts +0 -51
  163. package/test/acp/event-subscription.test.ts +0 -683
  164. package/test/agent/agent.test.ts +0 -689
  165. package/test/bun.test.ts +0 -53
  166. package/test/cli/github-action.test.ts +0 -197
  167. package/test/cli/github-remote.test.ts +0 -80
  168. package/test/cli/import.test.ts +0 -38
  169. package/test/cli/plugin-auth-picker.test.ts +0 -120
  170. package/test/cli/tui/transcript.test.ts +0 -322
  171. package/test/config/agent-color.test.ts +0 -71
  172. package/test/config/config.test.ts +0 -1886
  173. package/test/config/fixtures/empty-frontmatter.md +0 -4
  174. package/test/config/fixtures/frontmatter.md +0 -28
  175. package/test/config/fixtures/markdown-header.md +0 -11
  176. package/test/config/fixtures/no-frontmatter.md +0 -1
  177. package/test/config/fixtures/weird-model-id.md +0 -13
  178. package/test/config/markdown.test.ts +0 -228
  179. package/test/config/tui.test.ts +0 -510
  180. package/test/control-plane/session-proxy-middleware.test.ts +0 -147
  181. package/test/control-plane/sse.test.ts +0 -56
  182. package/test/control-plane/workspace-server-sse.test.ts +0 -65
  183. package/test/control-plane/workspace-sync.test.ts +0 -97
  184. package/test/file/ignore.test.ts +0 -10
  185. package/test/file/index.test.ts +0 -394
  186. package/test/file/path-traversal.test.ts +0 -198
  187. package/test/file/ripgrep.test.ts +0 -39
  188. package/test/file/time.test.ts +0 -361
  189. package/test/fixture/db.ts +0 -11
  190. package/test/fixture/fixture.ts +0 -45
  191. package/test/fixture/lsp/fake-lsp-server.js +0 -77
  192. package/test/fixture/skills/agents-sdk/SKILL.md +0 -152
  193. package/test/fixture/skills/agents-sdk/references/callable.md +0 -92
  194. package/test/fixture/skills/cloudflare/SKILL.md +0 -211
  195. package/test/fixture/skills/index.json +0 -6
  196. package/test/ide/ide.test.ts +0 -82
  197. package/test/keybind.test.ts +0 -421
  198. package/test/lsp/client.test.ts +0 -95
  199. package/test/mcp/headers.test.ts +0 -153
  200. package/test/mcp/oauth-browser.test.ts +0 -249
  201. package/test/memory/abort-leak.test.ts +0 -136
  202. package/test/patch/patch.test.ts +0 -348
  203. package/test/permission/arity.test.ts +0 -33
  204. package/test/permission/next.test.ts +0 -689
  205. package/test/permission-task.test.ts +0 -319
  206. package/test/plugin/auth-override.test.ts +0 -44
  207. package/test/plugin/codex.test.ts +0 -123
  208. package/test/preload.ts +0 -80
  209. package/test/project/project.test.ts +0 -348
  210. package/test/project/worktree-remove.test.ts +0 -65
  211. package/test/provider/amazon-bedrock.test.ts +0 -446
  212. package/test/provider/copilot/convert-to-copilot-messages.test.ts +0 -523
  213. package/test/provider/copilot/copilot-chat-model.test.ts +0 -592
  214. package/test/provider/gitlab-duo.test.ts +0 -262
  215. package/test/provider/provider.test.ts +0 -2220
  216. package/test/provider/transform.test.ts +0 -2353
  217. package/test/pty/pty-output-isolation.test.ts +0 -140
  218. package/test/question/question.test.ts +0 -300
  219. package/test/scheduler.test.ts +0 -73
  220. package/test/server/global-session-list.test.ts +0 -89
  221. package/test/server/session-list.test.ts +0 -90
  222. package/test/server/session-select.test.ts +0 -78
  223. package/test/session/compaction.test.ts +0 -423
  224. package/test/session/instruction.test.ts +0 -170
  225. package/test/session/llm.test.ts +0 -667
  226. package/test/session/message-v2.test.ts +0 -924
  227. package/test/session/prompt.test.ts +0 -211
  228. package/test/session/retry.test.ts +0 -188
  229. package/test/session/revert-compact.test.ts +0 -285
  230. package/test/session/session.test.ts +0 -71
  231. package/test/session/structured-output-integration.test.ts +0 -233
  232. package/test/session/structured-output.test.ts +0 -385
  233. package/test/skill/discovery.test.ts +0 -110
  234. package/test/skill/skill.test.ts +0 -388
  235. package/test/snapshot/snapshot.test.ts +0 -1180
  236. package/test/storage/json-migration.test.ts +0 -846
  237. package/test/tool/__snapshots__/tool.test.ts.snap +0 -9
  238. package/test/tool/apply_patch.test.ts +0 -566
  239. package/test/tool/bash.test.ts +0 -402
  240. package/test/tool/edit.test.ts +0 -496
  241. package/test/tool/external-directory.test.ts +0 -127
  242. package/test/tool/fixtures/large-image.png +0 -0
  243. package/test/tool/fixtures/models-api.json +0 -38413
  244. package/test/tool/grep.test.ts +0 -110
  245. package/test/tool/question.test.ts +0 -107
  246. package/test/tool/read.test.ts +0 -504
  247. package/test/tool/registry.test.ts +0 -122
  248. package/test/tool/skill.test.ts +0 -112
  249. package/test/tool/truncation.test.ts +0 -160
  250. package/test/tool/webfetch.test.ts +0 -100
  251. package/test/tool/write.test.ts +0 -348
  252. package/test/util/filesystem.test.ts +0 -443
  253. package/test/util/format.test.ts +0 -59
  254. package/test/util/glob.test.ts +0 -164
  255. package/test/util/iife.test.ts +0 -36
  256. package/test/util/lazy.test.ts +0 -50
  257. package/test/util/lock.test.ts +0 -72
  258. package/test/util/process.test.ts +0 -59
  259. package/test/util/timeout.test.ts +0 -21
  260. package/test/util/wildcard.test.ts +0 -90
  261. package/tsconfig.json +0 -16
package/src/tool/task.ts CHANGED
@@ -6,10 +6,10 @@ import { MessageV2 } from "../session/message-v2"
6
6
  import { Identifier } from "../id/id"
7
7
  import { Agent } from "../agent/agent"
8
8
  import { SessionPrompt } from "../session/prompt"
9
- import { iife } from "@/util/iife"
10
- import { defer } from "@/util/defer"
9
+ import { iife } from "../util/iife"
10
+ import { defer } from "../util/defer"
11
11
  import { Config } from "../config/config"
12
- import { PermissionNext } from "@/permission/next"
12
+ import { PermissionNext } from "../permission/next"
13
13
 
14
14
  const parameters = z.object({
15
15
  description: z.string().describe("A short (3-5 words) description of the task"),
@@ -0,0 +1,10 @@
1
+ export function findLast<T>(
2
+ items: readonly T[],
3
+ predicate: (item: T, index: number, items: readonly T[]) => boolean,
4
+ ): T | undefined {
5
+ for (let i = items.length - 1; i >= 0; i -= 1) {
6
+ const item = items[i]
7
+ if (predicate(item, i, items)) return item
8
+ }
9
+ return undefined
10
+ }
@@ -0,0 +1,41 @@
1
+ export namespace Binary {
2
+ export function search<T>(array: T[], id: string, compare: (item: T) => string): { found: boolean; index: number } {
3
+ let left = 0
4
+ let right = array.length - 1
5
+
6
+ while (left <= right) {
7
+ const mid = Math.floor((left + right) / 2)
8
+ const midId = compare(array[mid])
9
+
10
+ if (midId === id) {
11
+ return { found: true, index: mid }
12
+ } else if (midId < id) {
13
+ left = mid + 1
14
+ } else {
15
+ right = mid - 1
16
+ }
17
+ }
18
+
19
+ return { found: false, index: left }
20
+ }
21
+
22
+ export function insert<T>(array: T[], item: T, compare: (item: T) => string): T[] {
23
+ const id = compare(item)
24
+ let left = 0
25
+ let right = array.length
26
+
27
+ while (left < right) {
28
+ const mid = Math.floor((left + right) / 2)
29
+ const midId = compare(array[mid])
30
+
31
+ if (midId < id) {
32
+ left = mid + 1
33
+ } else {
34
+ right = mid
35
+ }
36
+ }
37
+
38
+ array.splice(left, 0, item)
39
+ return array
40
+ }
41
+ }
@@ -0,0 +1,51 @@
1
+ export function base64Encode(value: string) {
2
+ const bytes = new TextEncoder().encode(value)
3
+ const binary = Array.from(bytes, (b) => String.fromCharCode(b)).join("")
4
+ return btoa(binary).replace(/\+/g, "-").replace(/\//g, "_").replace(/=/g, "")
5
+ }
6
+
7
+ export function base64Decode(value: string) {
8
+ const binary = atob(value.replace(/-/g, "+").replace(/_/g, "/"))
9
+ const bytes = Uint8Array.from(binary, (c) => c.charCodeAt(0))
10
+ return new TextDecoder().decode(bytes)
11
+ }
12
+
13
+ export async function hash(content: string, algorithm = "SHA-256"): Promise<string> {
14
+ const encoder = new TextEncoder()
15
+ const data = encoder.encode(content)
16
+ const hashBuffer = await crypto.subtle.digest(algorithm, data)
17
+ const hashArray = Array.from(new Uint8Array(hashBuffer))
18
+ const hashHex = hashArray.map((b) => b.toString(16).padStart(2, "0")).join("")
19
+ return hashHex
20
+ }
21
+
22
+ export function checksum(content: string): string | undefined {
23
+ if (!content) return undefined
24
+ let hash = 0x811c9dc5
25
+ for (let i = 0; i < content.length; i++) {
26
+ hash ^= content.charCodeAt(i)
27
+ hash = Math.imul(hash, 0x01000193)
28
+ }
29
+ return (hash >>> 0).toString(36)
30
+ }
31
+
32
+ export function sampledChecksum(content: string, limit = 500_000): string | undefined {
33
+ if (!content) return undefined
34
+ if (content.length <= limit) return checksum(content)
35
+
36
+ const size = 4096
37
+ const points = [
38
+ 0,
39
+ Math.floor(content.length * 0.25),
40
+ Math.floor(content.length * 0.5),
41
+ Math.floor(content.length * 0.75),
42
+ content.length - size,
43
+ ]
44
+ const hashes = points
45
+ .map((point) => {
46
+ const start = Math.max(0, Math.min(content.length - size, point - Math.floor(size / 2)))
47
+ return checksum(content.slice(start, start + size)) ?? ""
48
+ })
49
+ .join(":")
50
+ return `${content.length}:${hashes}`
51
+ }
@@ -0,0 +1,54 @@
1
+ import z from "zod"
2
+
3
+ export abstract class NamedError extends Error {
4
+ abstract schema(): z.core.$ZodType
5
+ abstract toObject(): { name: string; data: any }
6
+
7
+ static create<Name extends string, Data extends z.core.$ZodType>(name: Name, data: Data) {
8
+ const schema = z
9
+ .object({
10
+ name: z.literal(name),
11
+ data,
12
+ })
13
+ .meta({
14
+ ref: name,
15
+ })
16
+ const result = class extends NamedError {
17
+ public static readonly Schema = schema
18
+
19
+ public override readonly name = name as Name
20
+
21
+ constructor(
22
+ public readonly data: z.input<Data>,
23
+ options?: ErrorOptions,
24
+ ) {
25
+ super(name, options)
26
+ this.name = name
27
+ }
28
+
29
+ static isInstance(input: any): input is InstanceType<typeof result> {
30
+ return typeof input === "object" && "name" in input && input.name === name
31
+ }
32
+
33
+ schema() {
34
+ return schema
35
+ }
36
+
37
+ toObject() {
38
+ return {
39
+ name: name,
40
+ data: this.data,
41
+ }
42
+ }
43
+ }
44
+ Object.defineProperty(result, "name", { value: name })
45
+ return result
46
+ }
47
+
48
+ public static readonly Unknown = NamedError.create(
49
+ "UnknownError",
50
+ z.object({
51
+ message: z.string(),
52
+ }),
53
+ )
54
+ }
@@ -0,0 +1,48 @@
1
+ import { randomBytes } from "crypto"
2
+
3
+ export namespace Identifier {
4
+ const LENGTH = 26
5
+
6
+ // State for monotonic ID generation
7
+ let lastTimestamp = 0
8
+ let counter = 0
9
+
10
+ export function ascending() {
11
+ return create(false)
12
+ }
13
+
14
+ export function descending() {
15
+ return create(true)
16
+ }
17
+
18
+ function randomBase62(length: number): string {
19
+ const chars = "0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz"
20
+ let result = ""
21
+ const bytes = randomBytes(length)
22
+ for (let i = 0; i < length; i++) {
23
+ result += chars[bytes[i] % 62]
24
+ }
25
+ return result
26
+ }
27
+
28
+ export function create(descending: boolean, timestamp?: number): string {
29
+ const currentTimestamp = timestamp ?? Date.now()
30
+
31
+ if (currentTimestamp !== lastTimestamp) {
32
+ lastTimestamp = currentTimestamp
33
+ counter = 0
34
+ }
35
+ counter++
36
+
37
+ let now = BigInt(currentTimestamp) * BigInt(0x1000) + BigInt(counter)
38
+
39
+ now = descending ? ~now : now
40
+
41
+ const timeBytes = Buffer.alloc(6)
42
+ for (let i = 0; i < 6; i++) {
43
+ timeBytes[i] = Number((now >> BigInt(40 - 8 * i)) & BigInt(0xff))
44
+ }
45
+
46
+ return timeBytes.toString("hex") + randomBase62(LENGTH - 12)
47
+ }
48
+ }
package/src/util/lazy.ts CHANGED
@@ -2,22 +2,10 @@ export function lazy<T>(fn: () => T) {
2
2
  let value: T | undefined
3
3
  let loaded = false
4
4
 
5
- const result = (): T => {
5
+ return (): T => {
6
6
  if (loaded) return value as T
7
- try {
8
- value = fn()
9
- loaded = true
10
- return value as T
11
- } catch (e) {
12
- // Don't mark as loaded if initialization failed
13
- throw e
14
- }
7
+ loaded = true
8
+ value = fn()
9
+ return value as T
15
10
  }
16
-
17
- result.reset = () => {
18
- loaded = false
19
- value = undefined
20
- }
21
-
22
- return result
23
11
  }
@@ -0,0 +1,37 @@
1
+ export function getFilename(path: string | undefined) {
2
+ if (!path) return ""
3
+ const trimmed = path.replace(/[\/\\]+$/, "")
4
+ const parts = trimmed.split(/[\/\\]/)
5
+ return parts[parts.length - 1] ?? ""
6
+ }
7
+
8
+ export function getDirectory(path: string | undefined) {
9
+ if (!path) return ""
10
+ const trimmed = path.replace(/[\/\\]+$/, "")
11
+ const parts = trimmed.split(/[\/\\]/)
12
+ return parts.slice(0, parts.length - 1).join("/") + "/"
13
+ }
14
+
15
+ export function getFileExtension(path: string | undefined) {
16
+ if (!path) return ""
17
+ const parts = path.split(".")
18
+ return parts[parts.length - 1]
19
+ }
20
+
21
+ export function getFilenameTruncated(path: string | undefined, maxLength: number = 20) {
22
+ const filename = getFilename(path)
23
+ if (filename.length <= maxLength) return filename
24
+ const lastDot = filename.lastIndexOf(".")
25
+ const ext = lastDot <= 0 ? "" : filename.slice(lastDot)
26
+ const available = maxLength - ext.length - 1 // -1 for ellipsis
27
+ if (available <= 0) return filename.slice(0, maxLength - 1) + "…"
28
+ return filename.slice(0, available) + "…" + ext
29
+ }
30
+
31
+ export function truncateMiddle(text: string, maxLength: number = 20) {
32
+ if (text.length <= maxLength) return text
33
+ const available = maxLength - 1 // -1 for ellipsis
34
+ const start = Math.ceil(available / 2)
35
+ const end = Math.floor(available / 2)
36
+ return text.slice(0, start) + "…" + text.slice(-end)
37
+ }
@@ -0,0 +1,41 @@
1
+ export interface RetryOptions {
2
+ attempts?: number
3
+ delay?: number
4
+ factor?: number
5
+ maxDelay?: number
6
+ retryIf?: (error: unknown) => boolean
7
+ }
8
+
9
+ const TRANSIENT_MESSAGES = [
10
+ "load failed",
11
+ "network connection was lost",
12
+ "network request failed",
13
+ "failed to fetch",
14
+ "econnreset",
15
+ "econnrefused",
16
+ "etimedout",
17
+ "socket hang up",
18
+ ]
19
+
20
+ function isTransientError(error: unknown): boolean {
21
+ if (!error) return false
22
+ const message = String(error instanceof Error ? error.message : error).toLowerCase()
23
+ return TRANSIENT_MESSAGES.some((m) => message.includes(m))
24
+ }
25
+
26
+ export async function retry<T>(fn: () => Promise<T>, options: RetryOptions = {}): Promise<T> {
27
+ const { attempts = 3, delay = 500, factor = 2, maxDelay = 10000, retryIf = isTransientError } = options
28
+
29
+ let lastError: unknown
30
+ for (let attempt = 0; attempt < attempts; attempt++) {
31
+ try {
32
+ return await fn()
33
+ } catch (error) {
34
+ lastError = error
35
+ if (attempt === attempts - 1 || !retryIf(error)) throw error
36
+ const wait = Math.min(delay * Math.pow(factor, attempt), maxDelay)
37
+ await new Promise((resolve) => setTimeout(resolve, wait))
38
+ }
39
+ }
40
+ throw lastError
41
+ }
@@ -0,0 +1,74 @@
1
+ export namespace Slug {
2
+ const ADJECTIVES = [
3
+ "brave",
4
+ "calm",
5
+ "clever",
6
+ "cosmic",
7
+ "crisp",
8
+ "curious",
9
+ "eager",
10
+ "gentle",
11
+ "glowing",
12
+ "happy",
13
+ "hidden",
14
+ "jolly",
15
+ "kind",
16
+ "lucky",
17
+ "mighty",
18
+ "misty",
19
+ "neon",
20
+ "nimble",
21
+ "playful",
22
+ "proud",
23
+ "quick",
24
+ "quiet",
25
+ "shiny",
26
+ "silent",
27
+ "stellar",
28
+ "sunny",
29
+ "swift",
30
+ "tidy",
31
+ "witty",
32
+ ] as const
33
+
34
+ const NOUNS = [
35
+ "cabin",
36
+ "cactus",
37
+ "canyon",
38
+ "circuit",
39
+ "comet",
40
+ "eagle",
41
+ "engine",
42
+ "falcon",
43
+ "forest",
44
+ "garden",
45
+ "harbor",
46
+ "island",
47
+ "knight",
48
+ "lagoon",
49
+ "meadow",
50
+ "moon",
51
+ "mountain",
52
+ "nebula",
53
+ "orchid",
54
+ "otter",
55
+ "panda",
56
+ "pixel",
57
+ "planet",
58
+ "river",
59
+ "rocket",
60
+ "sailor",
61
+ "squid",
62
+ "star",
63
+ "tiger",
64
+ "wizard",
65
+ "wolf",
66
+ ] as const
67
+
68
+ export function create() {
69
+ return [
70
+ ADJECTIVES[Math.floor(Math.random() * ADJECTIVES.length)],
71
+ NOUNS[Math.floor(Math.random() * NOUNS.length)],
72
+ ].join("-")
73
+ }
74
+ }
@@ -2,7 +2,7 @@ import { $ } from "bun"
2
2
  import fs from "fs/promises"
3
3
  import path from "path"
4
4
  import z from "zod"
5
- import { NamedError } from "@opencode-ai/util/error"
5
+ import { NamedError } from "../util/error"
6
6
  import { Global } from "../global"
7
7
  import { Instance } from "../project/instance"
8
8
  import { InstanceBootstrap } from "../project/bootstrap"
@@ -11,8 +11,8 @@ import { Database, eq } from "../storage/db"
11
11
  import { ProjectTable } from "../project/project.sql"
12
12
  import { fn } from "../util/fn"
13
13
  import { Log } from "../util/log"
14
- import { BusEvent } from "@/bus/bus-event"
15
- import { GlobalBus } from "@/bus/global"
14
+ import { BusEvent } from "../bus/bus-event"
15
+ import { GlobalBus } from "../bus/global"
16
16
 
17
17
  export namespace Worktree {
18
18
  const log = Log.create({ service: "worktree" })
package/AGENTS.md DELETED
@@ -1,10 +0,0 @@
1
- # opencode database guide
2
-
3
- ## Database
4
-
5
- - **Schema**: Drizzle schema lives in `src/**/*.sql.ts`.
6
- - **Naming**: tables and columns use snake*case; join columns are `<entity>_id`; indexes are `<table>*<column>\_idx`.
7
- - **Migrations**: generated by Drizzle Kit using `drizzle.config.ts` (schema: `./src/**/*.sql.ts`, output: `./migration`).
8
- - **Command**: `bun run db generate --name <slug>`.
9
- - **Output**: creates `migration/<timestamp>_<slug>/migration.sql` and `snapshot.json`.
10
- - **Tests**: migration tests should read the per-folder layout (no `_journal.json`).
@@ -1,136 +0,0 @@
1
- # Bun shell migration plan
2
-
3
- Practical phased replacement of Bun `$` calls.
4
-
5
- ## Goal
6
-
7
- Replace runtime Bun shell template-tag usage in `packages/opencode/src` with a unified `Process` API in `util/process.ts`.
8
-
9
- Keep behavior stable while improving safety, testability, and observability.
10
-
11
- Current baseline from audit:
12
-
13
- - 143 runtime command invocations across 17 files
14
- - 84 are git commands
15
- - Largest hotspots:
16
- - `src/cli/cmd/github.ts` (33)
17
- - `src/worktree/index.ts` (22)
18
- - `src/lsp/server.ts` (21)
19
- - `src/installation/index.ts` (20)
20
- - `src/snapshot/index.ts` (18)
21
-
22
- ## Decisions
23
-
24
- - Extend `src/util/process.ts` (do not create a separate exec module).
25
- - Proceed with phased migration for both git and non-git paths.
26
- - Keep plugin `$` compatibility in 1.x and remove in 2.0.
27
-
28
- ## Non-goals
29
-
30
- - Do not remove plugin `$` compatibility in this effort.
31
- - Do not redesign command semantics beyond what is needed to preserve behavior.
32
-
33
- ## Constraints
34
-
35
- - Keep migration phased, not big-bang.
36
- - Minimize behavioral drift.
37
- - Keep these explicit shell-only exceptions:
38
- - `src/session/prompt.ts` raw command execution
39
- - worktree start scripts in `src/worktree/index.ts`
40
-
41
- ## Process API proposal (`src/util/process.ts`)
42
-
43
- Add higher-level wrappers on top of current spawn support.
44
-
45
- Core methods:
46
-
47
- - `Process.run(cmd, opts)`
48
- - `Process.text(cmd, opts)`
49
- - `Process.lines(cmd, opts)`
50
- - `Process.status(cmd, opts)`
51
- - `Process.shell(command, opts)` for intentional shell execution
52
-
53
- Git helpers:
54
-
55
- - `Process.git(args, opts)`
56
- - `Process.gitText(args, opts)`
57
-
58
- Shared options:
59
-
60
- - `cwd`, `env`, `stdin`, `stdout`, `stderr`, `abort`, `timeout`, `kill`
61
- - `allowFailure` / non-throw mode
62
- - optional redaction + trace metadata
63
-
64
- Standard result shape:
65
-
66
- - `code`, `stdout`, `stderr`, `duration_ms`, `cmd`
67
- - helpers like `text()` and `arrayBuffer()` where useful
68
-
69
- ## Phased rollout
70
-
71
- ### Phase 0: Foundation
72
-
73
- - Implement Process wrappers in `src/util/process.ts`.
74
- - Refactor `src/util/git.ts` to use Process only.
75
- - Add tests for exit handling, timeout, abort, and output capture.
76
-
77
- ### Phase 1: High-impact hotspots
78
-
79
- Migrate these first:
80
-
81
- - `src/cli/cmd/github.ts`
82
- - `src/worktree/index.ts`
83
- - `src/lsp/server.ts`
84
- - `src/installation/index.ts`
85
- - `src/snapshot/index.ts`
86
-
87
- Within each file, migrate git paths first where applicable.
88
-
89
- ### Phase 2: Remaining git-heavy files
90
-
91
- Migrate git-centric call sites to `Process.git*` helpers:
92
-
93
- - `src/file/index.ts`
94
- - `src/project/vcs.ts`
95
- - `src/file/watcher.ts`
96
- - `src/storage/storage.ts`
97
- - `src/cli/cmd/pr.ts`
98
-
99
- ### Phase 3: Remaining non-git files
100
-
101
- Migrate residual non-git usages:
102
-
103
- - `src/cli/cmd/tui/util/clipboard.ts`
104
- - `src/util/archive.ts`
105
- - `src/file/ripgrep.ts`
106
- - `src/tool/bash.ts`
107
- - `src/cli/cmd/uninstall.ts`
108
-
109
- ### Phase 4: Stabilize
110
-
111
- - Remove dead wrappers and one-off patterns.
112
- - Keep plugin `$` compatibility isolated and documented as temporary.
113
- - Create linked 2.0 task for plugin `$` removal.
114
-
115
- ## Validation strategy
116
-
117
- - Unit tests for new `Process` methods and options.
118
- - Integration tests on hotspot modules.
119
- - Smoke tests for install, snapshot, worktree, and GitHub flows.
120
- - Regression checks for output parsing behavior.
121
-
122
- ## Risk mitigation
123
-
124
- - File-by-file PRs with small diffs.
125
- - Preserve behavior first, simplify second.
126
- - Keep shell-only exceptions explicit and documented.
127
- - Add consistent error shaping and logging at Process layer.
128
-
129
- ## Definition of done
130
-
131
- - Runtime Bun `$` usage in `packages/opencode/src` is removed except:
132
- - approved shell-only exceptions
133
- - temporary plugin compatibility path (1.x)
134
- - Git paths use `Process.git*` consistently.
135
- - CI and targeted smoke tests pass.
136
- - 2.0 issue exists for plugin `$` removal.
package/Dockerfile DELETED
@@ -1,18 +0,0 @@
1
- FROM alpine AS base
2
-
3
- # Disable the runtime transpiler cache by default inside Docker containers.
4
- # On ephemeral containers, the cache is not useful
5
- ARG BUN_RUNTIME_TRANSPILER_CACHE_PATH=0
6
- ENV BUN_RUNTIME_TRANSPILER_CACHE_PATH=${BUN_RUNTIME_TRANSPILER_CACHE_PATH}
7
- RUN apk add libgcc libstdc++ ripgrep
8
-
9
- FROM base AS build-amd64
10
- COPY dist/opencode-linux-x64-baseline-musl/bin/opencode /usr/local/bin/opencode
11
-
12
- FROM base AS build-arm64
13
- COPY dist/opencode-linux-arm64-musl/bin/opencode /usr/local/bin/opencode
14
-
15
- ARG TARGETARCH
16
- FROM build-${TARGETARCH}
17
- RUN opencode --version
18
- ENTRYPOINT ["opencode"]
package/README.md DELETED
@@ -1,15 +0,0 @@
1
- # js
2
-
3
- To install dependencies:
4
-
5
- ```bash
6
- bun install
7
- ```
8
-
9
- To run:
10
-
11
- ```bash
12
- bun run index.ts
13
- ```
14
-
15
- This project was created using `bun init` in bun v1.2.12. [Bun](https://bun.sh) is a fast all-in-one JavaScript runtime.
package/bunfig.toml DELETED
@@ -1,7 +0,0 @@
1
- preload = ["@opentui/solid/preload"]
2
-
3
- [test]
4
- preload = ["./test/preload.ts"]
5
- # timeout is not actually parsed from bunfig.toml (see src/bunfig.zig in oven-sh/bun)
6
- # using --timeout in package.json scripts instead
7
- # https://github.com/oven-sh/bun/issues/7789
package/drizzle.config.ts DELETED
@@ -1,10 +0,0 @@
1
- import { defineConfig } from "drizzle-kit"
2
-
3
- export default defineConfig({
4
- dialect: "sqlite",
5
- schema: "./src/**/*.sql.ts",
6
- out: "./migration",
7
- dbCredentials: {
8
- url: "/home/thdxr/.local/share/opencode/opencode.db",
9
- },
10
- })