chad-code 1.3.1
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/AGENTS.md +27 -0
- package/Dockerfile +18 -0
- package/README.md +15 -0
- package/README.npm.md +64 -0
- package/bin/chad-code +84 -0
- package/bunfig.toml +7 -0
- package/eslint.config.js +29 -0
- package/package.json +107 -0
- package/parsers-config.ts +253 -0
- package/script/build.ts +167 -0
- package/script/postinstall.mjs +122 -0
- package/script/publish-registries.ts +187 -0
- package/script/publish.ts +93 -0
- package/script/schema.ts +47 -0
- package/src/acp/README.md +164 -0
- package/src/acp/agent.ts +1086 -0
- package/src/acp/session.ts +101 -0
- package/src/acp/types.ts +22 -0
- package/src/agent/agent.ts +253 -0
- package/src/agent/generate.txt +75 -0
- package/src/agent/prompt/compaction.txt +12 -0
- package/src/agent/prompt/explore.txt +18 -0
- package/src/agent/prompt/summary.txt +11 -0
- package/src/agent/prompt/title.txt +36 -0
- package/src/auth/index.ts +70 -0
- package/src/bun/index.ts +130 -0
- package/src/bus/bus-event.ts +43 -0
- package/src/bus/global.ts +10 -0
- package/src/bus/index.ts +105 -0
- package/src/cli/bootstrap.ts +17 -0
- package/src/cli/cmd/acp.ts +69 -0
- package/src/cli/cmd/agent.ts +257 -0
- package/src/cli/cmd/auth.ts +132 -0
- package/src/cli/cmd/cmd.ts +7 -0
- package/src/cli/cmd/debug/agent.ts +28 -0
- package/src/cli/cmd/debug/config.ts +15 -0
- package/src/cli/cmd/debug/file.ts +91 -0
- package/src/cli/cmd/debug/index.ts +45 -0
- package/src/cli/cmd/debug/lsp.ts +48 -0
- package/src/cli/cmd/debug/ripgrep.ts +83 -0
- package/src/cli/cmd/debug/scrap.ts +15 -0
- package/src/cli/cmd/debug/skill.ts +15 -0
- package/src/cli/cmd/debug/snapshot.ts +48 -0
- package/src/cli/cmd/export.ts +88 -0
- package/src/cli/cmd/generate.ts +38 -0
- package/src/cli/cmd/github.ts +32 -0
- package/src/cli/cmd/import.ts +98 -0
- package/src/cli/cmd/mcp.ts +670 -0
- package/src/cli/cmd/models.ts +42 -0
- package/src/cli/cmd/pr.ts +112 -0
- package/src/cli/cmd/run.ts +374 -0
- package/src/cli/cmd/serve.ts +16 -0
- package/src/cli/cmd/session.ts +135 -0
- package/src/cli/cmd/stats.ts +402 -0
- package/src/cli/cmd/tui/app.tsx +705 -0
- package/src/cli/cmd/tui/attach.ts +32 -0
- package/src/cli/cmd/tui/component/border.tsx +21 -0
- package/src/cli/cmd/tui/component/dialog-agent.tsx +31 -0
- package/src/cli/cmd/tui/component/dialog-command.tsx +124 -0
- package/src/cli/cmd/tui/component/dialog-mcp.tsx +86 -0
- package/src/cli/cmd/tui/component/dialog-model.tsx +232 -0
- package/src/cli/cmd/tui/component/dialog-provider.tsx +228 -0
- package/src/cli/cmd/tui/component/dialog-session-list.tsx +115 -0
- package/src/cli/cmd/tui/component/dialog-session-rename.tsx +31 -0
- package/src/cli/cmd/tui/component/dialog-stash.tsx +86 -0
- package/src/cli/cmd/tui/component/dialog-status.tsx +162 -0
- package/src/cli/cmd/tui/component/dialog-tag.tsx +44 -0
- package/src/cli/cmd/tui/component/dialog-theme-list.tsx +50 -0
- package/src/cli/cmd/tui/component/did-you-know.tsx +85 -0
- package/src/cli/cmd/tui/component/logo.tsx +43 -0
- package/src/cli/cmd/tui/component/prompt/autocomplete.tsx +654 -0
- package/src/cli/cmd/tui/component/prompt/history.tsx +108 -0
- package/src/cli/cmd/tui/component/prompt/index.tsx +1078 -0
- package/src/cli/cmd/tui/component/prompt/stash.tsx +101 -0
- package/src/cli/cmd/tui/component/textarea-keybindings.ts +73 -0
- package/src/cli/cmd/tui/component/tips.ts +92 -0
- package/src/cli/cmd/tui/component/todo-item.tsx +32 -0
- package/src/cli/cmd/tui/context/args.tsx +14 -0
- package/src/cli/cmd/tui/context/directory.ts +13 -0
- package/src/cli/cmd/tui/context/exit.tsx +23 -0
- package/src/cli/cmd/tui/context/helper.tsx +25 -0
- package/src/cli/cmd/tui/context/keybind.tsx +101 -0
- package/src/cli/cmd/tui/context/kv.tsx +49 -0
- package/src/cli/cmd/tui/context/local.tsx +392 -0
- package/src/cli/cmd/tui/context/prompt.tsx +18 -0
- package/src/cli/cmd/tui/context/route.tsx +46 -0
- package/src/cli/cmd/tui/context/sdk.tsx +75 -0
- package/src/cli/cmd/tui/context/sync.tsx +384 -0
- package/src/cli/cmd/tui/context/theme/aura.json +69 -0
- package/src/cli/cmd/tui/context/theme/ayu.json +80 -0
- package/src/cli/cmd/tui/context/theme/catppuccin-frappe.json +233 -0
- package/src/cli/cmd/tui/context/theme/catppuccin-macchiato.json +233 -0
- package/src/cli/cmd/tui/context/theme/catppuccin.json +112 -0
- package/src/cli/cmd/tui/context/theme/chad.json +245 -0
- package/src/cli/cmd/tui/context/theme/cobalt2.json +228 -0
- package/src/cli/cmd/tui/context/theme/cursor.json +249 -0
- package/src/cli/cmd/tui/context/theme/dracula.json +219 -0
- package/src/cli/cmd/tui/context/theme/everforest.json +241 -0
- package/src/cli/cmd/tui/context/theme/flexoki.json +237 -0
- package/src/cli/cmd/tui/context/theme/github.json +233 -0
- package/src/cli/cmd/tui/context/theme/gruvbox.json +95 -0
- package/src/cli/cmd/tui/context/theme/kanagawa.json +77 -0
- package/src/cli/cmd/tui/context/theme/lucent-orng.json +227 -0
- package/src/cli/cmd/tui/context/theme/material.json +235 -0
- package/src/cli/cmd/tui/context/theme/matrix.json +77 -0
- package/src/cli/cmd/tui/context/theme/mercury.json +252 -0
- package/src/cli/cmd/tui/context/theme/monokai.json +221 -0
- package/src/cli/cmd/tui/context/theme/nightowl.json +221 -0
- package/src/cli/cmd/tui/context/theme/nord.json +223 -0
- package/src/cli/cmd/tui/context/theme/one-dark.json +84 -0
- package/src/cli/cmd/tui/context/theme/orng.json +245 -0
- package/src/cli/cmd/tui/context/theme/osaka-jade.json +93 -0
- package/src/cli/cmd/tui/context/theme/palenight.json +222 -0
- package/src/cli/cmd/tui/context/theme/rosepine.json +234 -0
- package/src/cli/cmd/tui/context/theme/solarized.json +223 -0
- package/src/cli/cmd/tui/context/theme/synthwave84.json +226 -0
- package/src/cli/cmd/tui/context/theme/tokyonight.json +243 -0
- package/src/cli/cmd/tui/context/theme/vercel.json +245 -0
- package/src/cli/cmd/tui/context/theme/vesper.json +218 -0
- package/src/cli/cmd/tui/context/theme/zenburn.json +223 -0
- package/src/cli/cmd/tui/context/theme.tsx +1137 -0
- package/src/cli/cmd/tui/event.ts +46 -0
- package/src/cli/cmd/tui/routes/home.tsx +138 -0
- package/src/cli/cmd/tui/routes/session/dialog-fork-from-timeline.tsx +64 -0
- package/src/cli/cmd/tui/routes/session/dialog-message.tsx +109 -0
- package/src/cli/cmd/tui/routes/session/dialog-subagent.tsx +26 -0
- package/src/cli/cmd/tui/routes/session/dialog-timeline.tsx +47 -0
- package/src/cli/cmd/tui/routes/session/footer.tsx +88 -0
- package/src/cli/cmd/tui/routes/session/header.tsx +125 -0
- package/src/cli/cmd/tui/routes/session/index.tsx +1814 -0
- package/src/cli/cmd/tui/routes/session/permission.tsx +416 -0
- package/src/cli/cmd/tui/routes/session/sidebar.tsx +318 -0
- package/src/cli/cmd/tui/spawn.ts +48 -0
- package/src/cli/cmd/tui/thread.ts +111 -0
- package/src/cli/cmd/tui/ui/dialog-alert.tsx +57 -0
- package/src/cli/cmd/tui/ui/dialog-confirm.tsx +83 -0
- package/src/cli/cmd/tui/ui/dialog-export-options.tsx +204 -0
- package/src/cli/cmd/tui/ui/dialog-help.tsx +38 -0
- package/src/cli/cmd/tui/ui/dialog-prompt.tsx +77 -0
- package/src/cli/cmd/tui/ui/dialog-select.tsx +345 -0
- package/src/cli/cmd/tui/ui/dialog.tsx +171 -0
- package/src/cli/cmd/tui/ui/link.tsx +28 -0
- package/src/cli/cmd/tui/ui/spinner.ts +368 -0
- package/src/cli/cmd/tui/ui/toast.tsx +100 -0
- package/src/cli/cmd/tui/util/clipboard.ts +127 -0
- package/src/cli/cmd/tui/util/editor.ts +32 -0
- package/src/cli/cmd/tui/util/signal.ts +7 -0
- package/src/cli/cmd/tui/util/terminal.ts +114 -0
- package/src/cli/cmd/tui/util/transcript.ts +98 -0
- package/src/cli/cmd/tui/worker.ts +68 -0
- package/src/cli/cmd/uninstall.ts +344 -0
- package/src/cli/cmd/upgrade.ts +67 -0
- package/src/cli/cmd/web.ts +73 -0
- package/src/cli/error.ts +56 -0
- package/src/cli/network.ts +53 -0
- package/src/cli/ui.ts +87 -0
- package/src/cli/upgrade.ts +25 -0
- package/src/command/index.ts +131 -0
- package/src/command/template/initialize.txt +10 -0
- package/src/command/template/review.txt +97 -0
- package/src/config/config.ts +1124 -0
- package/src/config/markdown.ts +41 -0
- package/src/env/index.ts +26 -0
- package/src/file/ignore.ts +83 -0
- package/src/file/index.ts +411 -0
- package/src/file/ripgrep.ts +402 -0
- package/src/file/time.ts +64 -0
- package/src/file/watcher.ts +117 -0
- package/src/flag/flag.ts +52 -0
- package/src/format/formatter.ts +359 -0
- package/src/format/index.ts +137 -0
- package/src/global/index.ts +55 -0
- package/src/id/id.ts +73 -0
- package/src/ide/index.ts +77 -0
- package/src/index.ts +159 -0
- package/src/installation/index.ts +198 -0
- package/src/lsp/client.ts +252 -0
- package/src/lsp/index.ts +485 -0
- package/src/lsp/language.ts +119 -0
- package/src/lsp/server.ts +2023 -0
- package/src/mcp/auth.ts +135 -0
- package/src/mcp/index.ts +874 -0
- package/src/mcp/oauth-callback.ts +200 -0
- package/src/mcp/oauth-provider.ts +154 -0
- package/src/patch/index.ts +622 -0
- package/src/permission/arity.ts +163 -0
- package/src/permission/index.ts +210 -0
- package/src/permission/next.ts +268 -0
- package/src/plugin/index.ts +106 -0
- package/src/project/bootstrap.ts +31 -0
- package/src/project/instance.ts +78 -0
- package/src/project/project.ts +263 -0
- package/src/project/state.ts +65 -0
- package/src/project/vcs.ts +76 -0
- package/src/provider/auth.ts +143 -0
- package/src/provider/models-macro.ts +4 -0
- package/src/provider/models.ts +77 -0
- package/src/provider/provider.ts +516 -0
- package/src/provider/transform.ts +114 -0
- package/src/pty/index.ts +212 -0
- package/src/server/error.ts +36 -0
- package/src/server/mdns.ts +57 -0
- package/src/server/project.ts +79 -0
- package/src/server/server.ts +2866 -0
- package/src/server/tui.ts +71 -0
- package/src/session/compaction.ts +225 -0
- package/src/session/index.ts +469 -0
- package/src/session/llm.ts +213 -0
- package/src/session/message-v2.ts +742 -0
- package/src/session/message.ts +189 -0
- package/src/session/processor.ts +402 -0
- package/src/session/prompt/anthropic-20250930.txt +166 -0
- package/src/session/prompt/anthropic.txt +105 -0
- package/src/session/prompt/anthropic_spoof.txt +1 -0
- package/src/session/prompt/beast.txt +147 -0
- package/src/session/prompt/build-switch.txt +5 -0
- package/src/session/prompt/codex.txt +318 -0
- package/src/session/prompt/copilot-gpt-5.txt +143 -0
- package/src/session/prompt/gemini.txt +155 -0
- package/src/session/prompt/max-steps.txt +16 -0
- package/src/session/prompt/plan-reminder-anthropic.txt +67 -0
- package/src/session/prompt/plan.txt +26 -0
- package/src/session/prompt/qwen.txt +109 -0
- package/src/session/prompt.ts +1621 -0
- package/src/session/retry.ts +90 -0
- package/src/session/revert.ts +108 -0
- package/src/session/status.ts +76 -0
- package/src/session/summary.ts +194 -0
- package/src/session/system.ts +108 -0
- package/src/session/todo.ts +37 -0
- package/src/share/share-next.ts +194 -0
- package/src/share/share.ts +23 -0
- package/src/shell/shell.ts +67 -0
- package/src/skill/index.ts +1 -0
- package/src/skill/skill.ts +124 -0
- package/src/snapshot/index.ts +197 -0
- package/src/storage/storage.ts +226 -0
- package/src/tool/bash.ts +262 -0
- package/src/tool/bash.txt +116 -0
- package/src/tool/batch.ts +175 -0
- package/src/tool/batch.txt +24 -0
- package/src/tool/codesearch.ts +132 -0
- package/src/tool/codesearch.txt +12 -0
- package/src/tool/edit.ts +655 -0
- package/src/tool/edit.txt +10 -0
- package/src/tool/glob.ts +75 -0
- package/src/tool/glob.txt +6 -0
- package/src/tool/grep.ts +132 -0
- package/src/tool/grep.txt +8 -0
- package/src/tool/invalid.ts +17 -0
- package/src/tool/ls.ts +119 -0
- package/src/tool/ls.txt +1 -0
- package/src/tool/lsp.ts +94 -0
- package/src/tool/lsp.txt +19 -0
- package/src/tool/multiedit.ts +46 -0
- package/src/tool/multiedit.txt +41 -0
- package/src/tool/patch.ts +210 -0
- package/src/tool/patch.txt +1 -0
- package/src/tool/read.ts +191 -0
- package/src/tool/read.txt +12 -0
- package/src/tool/registry.ts +137 -0
- package/src/tool/skill.ts +77 -0
- package/src/tool/task.ts +167 -0
- package/src/tool/task.txt +60 -0
- package/src/tool/todo.ts +53 -0
- package/src/tool/todoread.txt +14 -0
- package/src/tool/todowrite.txt +167 -0
- package/src/tool/tool.ts +73 -0
- package/src/tool/webfetch.ts +182 -0
- package/src/tool/webfetch.txt +13 -0
- package/src/tool/websearch.ts +144 -0
- package/src/tool/websearch.txt +11 -0
- package/src/tool/write.ts +84 -0
- package/src/tool/write.txt +8 -0
- package/src/util/archive.ts +16 -0
- package/src/util/color.ts +19 -0
- package/src/util/context.ts +25 -0
- package/src/util/defer.ts +12 -0
- package/src/util/eventloop.ts +20 -0
- package/src/util/filesystem.ts +83 -0
- package/src/util/fn.ts +11 -0
- package/src/util/iife.ts +3 -0
- package/src/util/keybind.ts +102 -0
- package/src/util/lazy.ts +18 -0
- package/src/util/locale.ts +81 -0
- package/src/util/lock.ts +98 -0
- package/src/util/log.ts +180 -0
- package/src/util/queue.ts +32 -0
- package/src/util/rpc.ts +42 -0
- package/src/util/scrap.ts +10 -0
- package/src/util/signal.ts +12 -0
- package/src/util/timeout.ts +14 -0
- package/src/util/token.ts +7 -0
- package/src/util/wildcard.ts +54 -0
- package/src/worktree/index.ts +217 -0
- package/sst-env.d.ts +9 -0
- package/test/agent/agent.test.ts +448 -0
- package/test/bun.test.ts +53 -0
- package/test/cli/github-action.test.ts +129 -0
- package/test/cli/github-remote.test.ts +80 -0
- package/test/cli/tui/transcript.test.ts +297 -0
- package/test/config/agent-color.test.ts +66 -0
- package/test/config/config.test.ts +870 -0
- package/test/config/markdown.test.ts +89 -0
- package/test/file/ignore.test.ts +10 -0
- package/test/file/path-traversal.test.ts +115 -0
- package/test/fixture/fixture.ts +45 -0
- package/test/fixture/lsp/fake-lsp-server.js +77 -0
- package/test/ide/ide.test.ts +82 -0
- package/test/keybind.test.ts +421 -0
- package/test/lsp/client.test.ts +95 -0
- package/test/mcp/headers.test.ts +153 -0
- package/test/patch/patch.test.ts +348 -0
- package/test/permission/arity.test.ts +33 -0
- package/test/permission/next.test.ts +652 -0
- package/test/preload.ts +63 -0
- package/test/project/project.test.ts +120 -0
- package/test/provider/amazon-bedrock.test.ts +236 -0
- package/test/provider/provider.test.ts +2127 -0
- package/test/provider/transform.test.ts +980 -0
- package/test/server/session-select.test.ts +78 -0
- package/test/session/compaction.test.ts +251 -0
- package/test/session/message-v2.test.ts +570 -0
- package/test/session/retry.test.ts +131 -0
- package/test/session/revert-compact.test.ts +285 -0
- package/test/session/session.test.ts +71 -0
- package/test/skill/skill.test.ts +185 -0
- package/test/snapshot/snapshot.test.ts +939 -0
- package/test/tool/__snapshots__/tool.test.ts.snap +9 -0
- package/test/tool/bash.test.ts +232 -0
- package/test/tool/grep.test.ts +109 -0
- package/test/tool/patch.test.ts +261 -0
- package/test/tool/read.test.ts +167 -0
- package/test/util/iife.test.ts +36 -0
- package/test/util/lazy.test.ts +50 -0
- package/test/util/timeout.test.ts +21 -0
- package/test/util/wildcard.test.ts +55 -0
- package/tsconfig.json +16 -0
|
@@ -0,0 +1,127 @@
|
|
|
1
|
+
import { $ } from "bun"
|
|
2
|
+
import { platform, release } from "os"
|
|
3
|
+
import clipboardy from "clipboardy"
|
|
4
|
+
import { lazy } from "../../../../util/lazy.js"
|
|
5
|
+
import { tmpdir } from "os"
|
|
6
|
+
import path from "path"
|
|
7
|
+
|
|
8
|
+
export namespace Clipboard {
|
|
9
|
+
export interface Content {
|
|
10
|
+
data: string
|
|
11
|
+
mime: string
|
|
12
|
+
}
|
|
13
|
+
|
|
14
|
+
export async function read(): Promise<Content | undefined> {
|
|
15
|
+
const os = platform()
|
|
16
|
+
|
|
17
|
+
if (os === "darwin") {
|
|
18
|
+
const tmpfile = path.join(tmpdir(), "opencode-clipboard.png")
|
|
19
|
+
try {
|
|
20
|
+
await $`osascript -e 'set imageData to the clipboard as "PNGf"' -e 'set fileRef to open for access POSIX file "${tmpfile}" with write permission' -e 'set eof fileRef to 0' -e 'write imageData to fileRef' -e 'close access fileRef'`
|
|
21
|
+
.nothrow()
|
|
22
|
+
.quiet()
|
|
23
|
+
const file = Bun.file(tmpfile)
|
|
24
|
+
const buffer = await file.arrayBuffer()
|
|
25
|
+
return { data: Buffer.from(buffer).toString("base64"), mime: "image/png" }
|
|
26
|
+
} catch {
|
|
27
|
+
} finally {
|
|
28
|
+
await $`rm -f "${tmpfile}"`.nothrow().quiet()
|
|
29
|
+
}
|
|
30
|
+
}
|
|
31
|
+
|
|
32
|
+
if (os === "win32" || release().includes("WSL")) {
|
|
33
|
+
const script =
|
|
34
|
+
"Add-Type -AssemblyName System.Windows.Forms; $img = [System.Windows.Forms.Clipboard]::GetImage(); if ($img) { $ms = New-Object System.IO.MemoryStream; $img.Save($ms, [System.Drawing.Imaging.ImageFormat]::Png); [System.Convert]::ToBase64String($ms.ToArray()) }"
|
|
35
|
+
const base64 = await $`powershell.exe -command "${script}"`.nothrow().text()
|
|
36
|
+
if (base64) {
|
|
37
|
+
const imageBuffer = Buffer.from(base64.trim(), "base64")
|
|
38
|
+
if (imageBuffer.length > 0) {
|
|
39
|
+
return { data: imageBuffer.toString("base64"), mime: "image/png" }
|
|
40
|
+
}
|
|
41
|
+
}
|
|
42
|
+
}
|
|
43
|
+
|
|
44
|
+
if (os === "linux") {
|
|
45
|
+
const wayland = await $`wl-paste -t image/png`.nothrow().arrayBuffer()
|
|
46
|
+
if (wayland && wayland.byteLength > 0) {
|
|
47
|
+
return { data: Buffer.from(wayland).toString("base64"), mime: "image/png" }
|
|
48
|
+
}
|
|
49
|
+
const x11 = await $`xclip -selection clipboard -t image/png -o`.nothrow().arrayBuffer()
|
|
50
|
+
if (x11 && x11.byteLength > 0) {
|
|
51
|
+
return { data: Buffer.from(x11).toString("base64"), mime: "image/png" }
|
|
52
|
+
}
|
|
53
|
+
}
|
|
54
|
+
|
|
55
|
+
const text = await clipboardy.read().catch(() => {})
|
|
56
|
+
if (text) {
|
|
57
|
+
return { data: text, mime: "text/plain" }
|
|
58
|
+
}
|
|
59
|
+
}
|
|
60
|
+
|
|
61
|
+
const getCopyMethod = lazy(() => {
|
|
62
|
+
const os = platform()
|
|
63
|
+
|
|
64
|
+
if (os === "darwin" && Bun.which("osascript")) {
|
|
65
|
+
console.log("clipboard: using osascript")
|
|
66
|
+
return async (text: string) => {
|
|
67
|
+
const escaped = text.replace(/\\/g, "\\\\").replace(/"/g, '\\"')
|
|
68
|
+
await $`osascript -e 'set the clipboard to "${escaped}"'`.nothrow().quiet()
|
|
69
|
+
}
|
|
70
|
+
}
|
|
71
|
+
|
|
72
|
+
if (os === "linux") {
|
|
73
|
+
if (process.env["WAYLAND_DISPLAY"] && Bun.which("wl-copy")) {
|
|
74
|
+
console.log("clipboard: using wl-copy")
|
|
75
|
+
return async (text: string) => {
|
|
76
|
+
const proc = Bun.spawn(["wl-copy"], { stdin: "pipe", stdout: "ignore", stderr: "ignore" })
|
|
77
|
+
proc.stdin.write(text)
|
|
78
|
+
proc.stdin.end()
|
|
79
|
+
await proc.exited.catch(() => {})
|
|
80
|
+
}
|
|
81
|
+
}
|
|
82
|
+
if (Bun.which("xclip")) {
|
|
83
|
+
console.log("clipboard: using xclip")
|
|
84
|
+
return async (text: string) => {
|
|
85
|
+
const proc = Bun.spawn(["xclip", "-selection", "clipboard"], {
|
|
86
|
+
stdin: "pipe",
|
|
87
|
+
stdout: "ignore",
|
|
88
|
+
stderr: "ignore",
|
|
89
|
+
})
|
|
90
|
+
proc.stdin.write(text)
|
|
91
|
+
proc.stdin.end()
|
|
92
|
+
await proc.exited.catch(() => {})
|
|
93
|
+
}
|
|
94
|
+
}
|
|
95
|
+
if (Bun.which("xsel")) {
|
|
96
|
+
console.log("clipboard: using xsel")
|
|
97
|
+
return async (text: string) => {
|
|
98
|
+
const proc = Bun.spawn(["xsel", "--clipboard", "--input"], {
|
|
99
|
+
stdin: "pipe",
|
|
100
|
+
stdout: "ignore",
|
|
101
|
+
stderr: "ignore",
|
|
102
|
+
})
|
|
103
|
+
proc.stdin.write(text)
|
|
104
|
+
proc.stdin.end()
|
|
105
|
+
await proc.exited.catch(() => {})
|
|
106
|
+
}
|
|
107
|
+
}
|
|
108
|
+
}
|
|
109
|
+
|
|
110
|
+
if (os === "win32") {
|
|
111
|
+
console.log("clipboard: using powershell")
|
|
112
|
+
return async (text: string) => {
|
|
113
|
+
const escaped = text.replace(/"/g, '""')
|
|
114
|
+
await $`powershell -command "Set-Clipboard -Value \"${escaped}\""`.nothrow().quiet()
|
|
115
|
+
}
|
|
116
|
+
}
|
|
117
|
+
|
|
118
|
+
console.log("clipboard: no native support")
|
|
119
|
+
return async (text: string) => {
|
|
120
|
+
await clipboardy.write(text).catch(() => {})
|
|
121
|
+
}
|
|
122
|
+
})
|
|
123
|
+
|
|
124
|
+
export async function copy(text: string): Promise<void> {
|
|
125
|
+
await getCopyMethod()(text)
|
|
126
|
+
}
|
|
127
|
+
}
|
|
@@ -0,0 +1,32 @@
|
|
|
1
|
+
import { defer } from "@/util/defer"
|
|
2
|
+
import { rm } from "node:fs/promises"
|
|
3
|
+
import { tmpdir } from "node:os"
|
|
4
|
+
import { join } from "node:path"
|
|
5
|
+
import { CliRenderer } from "@opentui/core"
|
|
6
|
+
|
|
7
|
+
export namespace Editor {
|
|
8
|
+
export async function open(opts: { value: string; renderer: CliRenderer }): Promise<string | undefined> {
|
|
9
|
+
const editor = process.env["VISUAL"] || process.env["EDITOR"]
|
|
10
|
+
if (!editor) return
|
|
11
|
+
|
|
12
|
+
const filepath = join(tmpdir(), `${Date.now()}.md`)
|
|
13
|
+
await using _ = defer(async () => rm(filepath, { force: true }))
|
|
14
|
+
|
|
15
|
+
await Bun.write(filepath, opts.value)
|
|
16
|
+
opts.renderer.suspend()
|
|
17
|
+
opts.renderer.currentRenderBuffer.clear()
|
|
18
|
+
const parts = editor.split(" ")
|
|
19
|
+
const proc = Bun.spawn({
|
|
20
|
+
cmd: [...parts, filepath],
|
|
21
|
+
stdin: "inherit",
|
|
22
|
+
stdout: "inherit",
|
|
23
|
+
stderr: "inherit",
|
|
24
|
+
})
|
|
25
|
+
await proc.exited
|
|
26
|
+
const content = await Bun.file(filepath).text()
|
|
27
|
+
opts.renderer.currentRenderBuffer.clear()
|
|
28
|
+
opts.renderer.resume()
|
|
29
|
+
opts.renderer.requestRender()
|
|
30
|
+
return content || undefined
|
|
31
|
+
}
|
|
32
|
+
}
|
|
@@ -0,0 +1,7 @@
|
|
|
1
|
+
import { createSignal, type Accessor } from "solid-js"
|
|
2
|
+
import { debounce, type Scheduled } from "@solid-primitives/scheduled"
|
|
3
|
+
|
|
4
|
+
export function createDebouncedSignal<T>(value: T, ms: number): [Accessor<T>, Scheduled<[value: T]>] {
|
|
5
|
+
const [get, set] = createSignal(value)
|
|
6
|
+
return [get, debounce((v: T) => set(() => v), ms)]
|
|
7
|
+
}
|
|
@@ -0,0 +1,114 @@
|
|
|
1
|
+
import { RGBA } from "@opentui/core"
|
|
2
|
+
|
|
3
|
+
export namespace Terminal {
|
|
4
|
+
export type Colors = Awaited<ReturnType<typeof colors>>
|
|
5
|
+
/**
|
|
6
|
+
* Query terminal colors including background, foreground, and palette (0-15).
|
|
7
|
+
* Uses OSC escape sequences to retrieve actual terminal color values.
|
|
8
|
+
*
|
|
9
|
+
* Note: OSC 4 (palette) queries may not work through tmux as responses are filtered.
|
|
10
|
+
* OSC 10/11 (foreground/background) typically work in most environments.
|
|
11
|
+
*
|
|
12
|
+
* Returns an object with background, foreground, and colors array.
|
|
13
|
+
* Any query that fails will be null/empty.
|
|
14
|
+
*/
|
|
15
|
+
export async function colors(): Promise<{
|
|
16
|
+
background: RGBA | null
|
|
17
|
+
foreground: RGBA | null
|
|
18
|
+
colors: RGBA[]
|
|
19
|
+
}> {
|
|
20
|
+
if (!process.stdin.isTTY) return { background: null, foreground: null, colors: [] }
|
|
21
|
+
|
|
22
|
+
return new Promise((resolve) => {
|
|
23
|
+
let background: RGBA | null = null
|
|
24
|
+
let foreground: RGBA | null = null
|
|
25
|
+
const paletteColors: RGBA[] = []
|
|
26
|
+
let timeout: NodeJS.Timeout
|
|
27
|
+
|
|
28
|
+
const cleanup = () => {
|
|
29
|
+
process.stdin.setRawMode(false)
|
|
30
|
+
process.stdin.removeListener("data", handler)
|
|
31
|
+
clearTimeout(timeout)
|
|
32
|
+
}
|
|
33
|
+
|
|
34
|
+
const parseColor = (colorStr: string): RGBA | null => {
|
|
35
|
+
if (colorStr.startsWith("rgb:")) {
|
|
36
|
+
const parts = colorStr.substring(4).split("/")
|
|
37
|
+
return RGBA.fromInts(
|
|
38
|
+
parseInt(parts[0], 16) >> 8, // Convert 16-bit to 8-bit
|
|
39
|
+
parseInt(parts[1], 16) >> 8,
|
|
40
|
+
parseInt(parts[2], 16) >> 8,
|
|
41
|
+
255,
|
|
42
|
+
)
|
|
43
|
+
}
|
|
44
|
+
if (colorStr.startsWith("#")) {
|
|
45
|
+
return RGBA.fromHex(colorStr)
|
|
46
|
+
}
|
|
47
|
+
if (colorStr.startsWith("rgb(")) {
|
|
48
|
+
const parts = colorStr.substring(4, colorStr.length - 1).split(",")
|
|
49
|
+
return RGBA.fromInts(parseInt(parts[0]), parseInt(parts[1]), parseInt(parts[2]), 255)
|
|
50
|
+
}
|
|
51
|
+
return null
|
|
52
|
+
}
|
|
53
|
+
|
|
54
|
+
const handler = (data: Buffer) => {
|
|
55
|
+
const str = data.toString()
|
|
56
|
+
|
|
57
|
+
// Match OSC 11 (background color)
|
|
58
|
+
const bgMatch = str.match(/\x1b]11;([^\x07\x1b]+)/)
|
|
59
|
+
if (bgMatch) {
|
|
60
|
+
background = parseColor(bgMatch[1])
|
|
61
|
+
}
|
|
62
|
+
|
|
63
|
+
// Match OSC 10 (foreground color)
|
|
64
|
+
const fgMatch = str.match(/\x1b]10;([^\x07\x1b]+)/)
|
|
65
|
+
if (fgMatch) {
|
|
66
|
+
foreground = parseColor(fgMatch[1])
|
|
67
|
+
}
|
|
68
|
+
|
|
69
|
+
// Match OSC 4 (palette colors)
|
|
70
|
+
const paletteMatches = str.matchAll(/\x1b]4;(\d+);([^\x07\x1b]+)/g)
|
|
71
|
+
for (const match of paletteMatches) {
|
|
72
|
+
const index = parseInt(match[1])
|
|
73
|
+
const color = parseColor(match[2])
|
|
74
|
+
if (color) paletteColors[index] = color
|
|
75
|
+
}
|
|
76
|
+
|
|
77
|
+
// Return immediately if we have all 16 palette colors
|
|
78
|
+
if (paletteColors.filter((c) => c !== undefined).length === 16) {
|
|
79
|
+
cleanup()
|
|
80
|
+
resolve({ background, foreground, colors: paletteColors })
|
|
81
|
+
}
|
|
82
|
+
}
|
|
83
|
+
|
|
84
|
+
process.stdin.setRawMode(true)
|
|
85
|
+
process.stdin.on("data", handler)
|
|
86
|
+
|
|
87
|
+
// Query background (OSC 11)
|
|
88
|
+
process.stdout.write("\x1b]11;?\x07")
|
|
89
|
+
// Query foreground (OSC 10)
|
|
90
|
+
process.stdout.write("\x1b]10;?\x07")
|
|
91
|
+
// Query palette colors 0-15 (OSC 4)
|
|
92
|
+
for (let i = 0; i < 16; i++) {
|
|
93
|
+
process.stdout.write(`\x1b]4;${i};?\x07`)
|
|
94
|
+
}
|
|
95
|
+
|
|
96
|
+
timeout = setTimeout(() => {
|
|
97
|
+
cleanup()
|
|
98
|
+
resolve({ background, foreground, colors: paletteColors })
|
|
99
|
+
}, 1000)
|
|
100
|
+
})
|
|
101
|
+
}
|
|
102
|
+
|
|
103
|
+
export async function getTerminalBackgroundColor(): Promise<"dark" | "light"> {
|
|
104
|
+
const result = await colors()
|
|
105
|
+
if (!result.background) return "dark"
|
|
106
|
+
|
|
107
|
+
const { r, g, b } = result.background
|
|
108
|
+
// Calculate luminance using relative luminance formula
|
|
109
|
+
const luminance = (0.299 * r + 0.587 * g + 0.114 * b) / 255
|
|
110
|
+
|
|
111
|
+
// Determine if dark or light based on luminance threshold
|
|
112
|
+
return luminance > 0.5 ? "light" : "dark"
|
|
113
|
+
}
|
|
114
|
+
}
|
|
@@ -0,0 +1,98 @@
|
|
|
1
|
+
import type { AssistantMessage, Part, UserMessage } from "@opencode-ai/sdk/v2"
|
|
2
|
+
import { Locale } from "@/util/locale"
|
|
3
|
+
|
|
4
|
+
export type TranscriptOptions = {
|
|
5
|
+
thinking: boolean
|
|
6
|
+
toolDetails: boolean
|
|
7
|
+
assistantMetadata: boolean
|
|
8
|
+
}
|
|
9
|
+
|
|
10
|
+
export type SessionInfo = {
|
|
11
|
+
id: string
|
|
12
|
+
title: string
|
|
13
|
+
time: {
|
|
14
|
+
created: number
|
|
15
|
+
updated: number
|
|
16
|
+
}
|
|
17
|
+
}
|
|
18
|
+
|
|
19
|
+
export type MessageWithParts = {
|
|
20
|
+
info: UserMessage | AssistantMessage
|
|
21
|
+
parts: Part[]
|
|
22
|
+
}
|
|
23
|
+
|
|
24
|
+
export function formatTranscript(
|
|
25
|
+
session: SessionInfo,
|
|
26
|
+
messages: MessageWithParts[],
|
|
27
|
+
options: TranscriptOptions,
|
|
28
|
+
): string {
|
|
29
|
+
let transcript = `# ${session.title}\n\n`
|
|
30
|
+
transcript += `**Session ID:** ${session.id}\n`
|
|
31
|
+
transcript += `**Created:** ${new Date(session.time.created).toLocaleString()}\n`
|
|
32
|
+
transcript += `**Updated:** ${new Date(session.time.updated).toLocaleString()}\n\n`
|
|
33
|
+
transcript += `---\n\n`
|
|
34
|
+
|
|
35
|
+
for (const msg of messages) {
|
|
36
|
+
transcript += formatMessage(msg.info, msg.parts, options)
|
|
37
|
+
transcript += `---\n\n`
|
|
38
|
+
}
|
|
39
|
+
|
|
40
|
+
return transcript
|
|
41
|
+
}
|
|
42
|
+
|
|
43
|
+
export function formatMessage(msg: UserMessage | AssistantMessage, parts: Part[], options: TranscriptOptions): string {
|
|
44
|
+
let result = ""
|
|
45
|
+
|
|
46
|
+
if (msg.role === "user") {
|
|
47
|
+
result += `## User\n\n`
|
|
48
|
+
} else {
|
|
49
|
+
result += formatAssistantHeader(msg, options.assistantMetadata)
|
|
50
|
+
}
|
|
51
|
+
|
|
52
|
+
for (const part of parts) {
|
|
53
|
+
result += formatPart(part, options)
|
|
54
|
+
}
|
|
55
|
+
|
|
56
|
+
return result
|
|
57
|
+
}
|
|
58
|
+
|
|
59
|
+
export function formatAssistantHeader(msg: AssistantMessage, includeMetadata: boolean): string {
|
|
60
|
+
if (!includeMetadata) {
|
|
61
|
+
return `## Assistant\n\n`
|
|
62
|
+
}
|
|
63
|
+
|
|
64
|
+
const duration =
|
|
65
|
+
msg.time.completed && msg.time.created ? ((msg.time.completed - msg.time.created) / 1000).toFixed(1) + "s" : ""
|
|
66
|
+
|
|
67
|
+
return `## Assistant (${Locale.titlecase(msg.agent)} · ${msg.modelID}${duration ? ` · ${duration}` : ""})\n\n`
|
|
68
|
+
}
|
|
69
|
+
|
|
70
|
+
export function formatPart(part: Part, options: TranscriptOptions): string {
|
|
71
|
+
if (part.type === "text" && !part.synthetic) {
|
|
72
|
+
return `${part.text}\n\n`
|
|
73
|
+
}
|
|
74
|
+
|
|
75
|
+
if (part.type === "reasoning") {
|
|
76
|
+
if (options.thinking) {
|
|
77
|
+
return `_Thinking:_\n\n${part.text}\n\n`
|
|
78
|
+
}
|
|
79
|
+
return ""
|
|
80
|
+
}
|
|
81
|
+
|
|
82
|
+
if (part.type === "tool") {
|
|
83
|
+
let result = `\`\`\`\nTool: ${part.tool}\n`
|
|
84
|
+
if (options.toolDetails && part.state.input) {
|
|
85
|
+
result += `\n**Input:**\n\`\`\`json\n${JSON.stringify(part.state.input, null, 2)}\n\`\`\``
|
|
86
|
+
}
|
|
87
|
+
if (options.toolDetails && part.state.status === "completed" && part.state.output) {
|
|
88
|
+
result += `\n**Output:**\n\`\`\`\n${part.state.output}\n\`\`\``
|
|
89
|
+
}
|
|
90
|
+
if (options.toolDetails && part.state.status === "error" && part.state.error) {
|
|
91
|
+
result += `\n**Error:**\n\`\`\`\n${part.state.error}\n\`\`\``
|
|
92
|
+
}
|
|
93
|
+
result += `\n\`\`\`\n\n`
|
|
94
|
+
return result
|
|
95
|
+
}
|
|
96
|
+
|
|
97
|
+
return ""
|
|
98
|
+
}
|
|
@@ -0,0 +1,68 @@
|
|
|
1
|
+
import { Installation } from "@/installation"
|
|
2
|
+
import { Server } from "@/server/server"
|
|
3
|
+
import { Log } from "@/util/log"
|
|
4
|
+
import { Instance } from "@/project/instance"
|
|
5
|
+
import { InstanceBootstrap } from "@/project/bootstrap"
|
|
6
|
+
import { Rpc } from "@/util/rpc"
|
|
7
|
+
import { upgrade } from "@/cli/upgrade"
|
|
8
|
+
import type { BunWebSocketData } from "hono/bun"
|
|
9
|
+
import { Config } from "@/config/config"
|
|
10
|
+
|
|
11
|
+
await Log.init({
|
|
12
|
+
print: process.argv.includes("--print-logs"),
|
|
13
|
+
dev: Installation.isLocal(),
|
|
14
|
+
level: (() => {
|
|
15
|
+
if (Installation.isLocal()) return "DEBUG"
|
|
16
|
+
return "INFO"
|
|
17
|
+
})(),
|
|
18
|
+
})
|
|
19
|
+
|
|
20
|
+
process.on("unhandledRejection", (e) => {
|
|
21
|
+
Log.Default.error("rejection", {
|
|
22
|
+
e: e instanceof Error ? e.message : e,
|
|
23
|
+
})
|
|
24
|
+
})
|
|
25
|
+
|
|
26
|
+
process.on("uncaughtException", (e) => {
|
|
27
|
+
Log.Default.error("exception", {
|
|
28
|
+
e: e instanceof Error ? e.message : e,
|
|
29
|
+
})
|
|
30
|
+
})
|
|
31
|
+
|
|
32
|
+
let server: Bun.Server<BunWebSocketData>
|
|
33
|
+
export const rpc = {
|
|
34
|
+
async server(input: { port: number; hostname: string; mdns?: boolean }) {
|
|
35
|
+
if (server) await server.stop(true)
|
|
36
|
+
try {
|
|
37
|
+
server = Server.listen(input)
|
|
38
|
+
return {
|
|
39
|
+
url: server.url.toString(),
|
|
40
|
+
}
|
|
41
|
+
} catch (e) {
|
|
42
|
+
console.error(e)
|
|
43
|
+
throw e
|
|
44
|
+
}
|
|
45
|
+
},
|
|
46
|
+
async checkUpgrade(input: { directory: string }) {
|
|
47
|
+
await Instance.provide({
|
|
48
|
+
directory: input.directory,
|
|
49
|
+
init: InstanceBootstrap,
|
|
50
|
+
fn: async () => {
|
|
51
|
+
await upgrade().catch(() => {})
|
|
52
|
+
},
|
|
53
|
+
})
|
|
54
|
+
},
|
|
55
|
+
async reload() {
|
|
56
|
+
Config.global.reset()
|
|
57
|
+
await Instance.disposeAll()
|
|
58
|
+
},
|
|
59
|
+
async shutdown() {
|
|
60
|
+
Log.Default.info("worker shutting down")
|
|
61
|
+
await Instance.disposeAll()
|
|
62
|
+
// TODO: this should be awaited, but ws connections are
|
|
63
|
+
// causing this to hang, need to revisit this
|
|
64
|
+
server.stop(true)
|
|
65
|
+
},
|
|
66
|
+
}
|
|
67
|
+
|
|
68
|
+
Rpc.listen(rpc)
|