@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
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 "
|
|
10
|
-
import { defer } from "
|
|
9
|
+
import { iife } from "../util/iife"
|
|
10
|
+
import { defer } from "../util/defer"
|
|
11
11
|
import { Config } from "../config/config"
|
|
12
|
-
import { PermissionNext } from "
|
|
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
|
-
|
|
5
|
+
return (): T => {
|
|
6
6
|
if (loaded) return value as T
|
|
7
|
-
|
|
8
|
-
|
|
9
|
-
|
|
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
|
}
|
package/src/util/path.ts
ADDED
|
@@ -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
|
+
}
|
package/src/util/slug.ts
ADDED
|
@@ -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
|
+
}
|
package/src/worktree/index.ts
CHANGED
|
@@ -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 "
|
|
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 "
|
|
15
|
-
import { GlobalBus } from "
|
|
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
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