rird 1.0.200
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/bin/opencode +336 -0
- package/bin/pty-wrapper.js +285 -0
- package/bunfig.toml +4 -0
- package/facebook_ads_library.png +0 -0
- package/nul`nif +0 -0
- package/package.json +111 -0
- package/parsers-config.ts +239 -0
- package/rird-1.0.199.tgz +0 -0
- package/script/build-windows.ts +54 -0
- package/script/build.ts +167 -0
- package/script/postinstall.mjs +544 -0
- package/script/publish-registries.ts +187 -0
- package/script/publish.ts +72 -0
- package/script/schema.ts +47 -0
- package/src/acp/README.md +164 -0
- package/src/acp/agent.ts +1063 -0
- package/src/acp/session.ts +101 -0
- package/src/acp/types.ts +22 -0
- package/src/agent/agent.ts +367 -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 +10 -0
- package/src/agent/prompt/title.txt +36 -0
- package/src/auth/index.ts +70 -0
- package/src/bun/index.ts +114 -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 +88 -0
- package/src/cli/cmd/agent.ts +256 -0
- package/src/cli/cmd/auth.ts +391 -0
- package/src/cli/cmd/cmd.ts +7 -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 +43 -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 +1400 -0
- package/src/cli/cmd/import.ts +98 -0
- package/src/cli/cmd/mcp.ts +654 -0
- package/src/cli/cmd/models.ts +77 -0
- package/src/cli/cmd/pr.ts +112 -0
- package/src/cli/cmd/run.ts +368 -0
- package/src/cli/cmd/serve.ts +31 -0
- package/src/cli/cmd/session.ts +106 -0
- package/src/cli/cmd/stats.ts +298 -0
- package/src/cli/cmd/tui/app.tsx +696 -0
- package/src/cli/cmd/tui/attach.ts +30 -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 +245 -0
- package/src/cli/cmd/tui/component/dialog-provider.tsx +224 -0
- package/src/cli/cmd/tui/component/dialog-session-list.tsx +102 -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 +35 -0
- package/src/cli/cmd/tui/component/prompt/autocomplete.tsx +574 -0
- package/src/cli/cmd/tui/component/prompt/history.tsx +108 -0
- package/src/cli/cmd/tui/component/prompt/index.tsx +1090 -0
- package/src/cli/cmd/tui/component/prompt/stash.tsx +101 -0
- package/src/cli/cmd/tui/component/tips.ts +27 -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 +354 -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 +74 -0
- package/src/cli/cmd/tui/context/sync.tsx +372 -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/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/palenight.json +222 -0
- package/src/cli/cmd/tui/context/theme/rird.json +245 -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 +1109 -0
- package/src/cli/cmd/tui/event.ts +40 -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 +1864 -0
- package/src/cli/cmd/tui/routes/session/sidebar.tsx +318 -0
- package/src/cli/cmd/tui/spawn.ts +60 -0
- package/src/cli/cmd/tui/thread.ts +142 -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-help.tsx +38 -0
- package/src/cli/cmd/tui/ui/dialog-prompt.tsx +77 -0
- package/src/cli/cmd/tui/ui/dialog-select.tsx +332 -0
- package/src/cli/cmd/tui/ui/dialog.tsx +170 -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/terminal.ts +114 -0
- package/src/cli/cmd/tui/worker.ts +63 -0
- package/src/cli/cmd/uninstall.ts +344 -0
- package/src/cli/cmd/upgrade.ts +100 -0
- package/src/cli/cmd/web.ts +84 -0
- package/src/cli/error.ts +56 -0
- package/src/cli/ui.ts +84 -0
- package/src/cli/upgrade.ts +25 -0
- package/src/command/index.ts +80 -0
- package/src/command/template/initialize.txt +10 -0
- package/src/command/template/review.txt +97 -0
- package/src/config/config.ts +995 -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 +328 -0
- package/src/file/ripgrep.ts +393 -0
- package/src/file/time.ts +64 -0
- package/src/file/watcher.ts +103 -0
- package/src/flag/flag.ts +46 -0
- package/src/format/formatter.ts +315 -0
- package/src/format/index.ts +137 -0
- package/src/global/index.ts +52 -0
- package/src/id/id.ts +73 -0
- package/src/ide/index.ts +76 -0
- package/src/index.ts +240 -0
- package/src/installation/index.ts +239 -0
- package/src/lsp/client.ts +229 -0
- package/src/lsp/index.ts +485 -0
- package/src/lsp/language.ts +116 -0
- package/src/lsp/server.ts +1895 -0
- package/src/mcp/auth.ts +135 -0
- package/src/mcp/index.ts +690 -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/index.ts +199 -0
- package/src/plugin/index.ts +91 -0
- package/src/project/bootstrap.ts +31 -0
- package/src/project/instance.ts +78 -0
- package/src/project/project.ts +221 -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 +11 -0
- package/src/provider/models.ts +106 -0
- package/src/provider/provider.ts +1071 -0
- package/src/provider/sdk/openai-compatible/src/README.md +5 -0
- package/src/provider/sdk/openai-compatible/src/index.ts +2 -0
- package/src/provider/sdk/openai-compatible/src/openai-compatible-provider.ts +100 -0
- package/src/provider/sdk/openai-compatible/src/responses/convert-to-openai-responses-input.ts +303 -0
- package/src/provider/sdk/openai-compatible/src/responses/map-openai-responses-finish-reason.ts +22 -0
- package/src/provider/sdk/openai-compatible/src/responses/openai-config.ts +18 -0
- package/src/provider/sdk/openai-compatible/src/responses/openai-error.ts +22 -0
- package/src/provider/sdk/openai-compatible/src/responses/openai-responses-api-types.ts +207 -0
- package/src/provider/sdk/openai-compatible/src/responses/openai-responses-language-model.ts +1713 -0
- package/src/provider/sdk/openai-compatible/src/responses/openai-responses-prepare-tools.ts +177 -0
- package/src/provider/sdk/openai-compatible/src/responses/openai-responses-settings.ts +1 -0
- package/src/provider/sdk/openai-compatible/src/responses/tool/code-interpreter.ts +88 -0
- package/src/provider/sdk/openai-compatible/src/responses/tool/file-search.ts +128 -0
- package/src/provider/sdk/openai-compatible/src/responses/tool/image-generation.ts +115 -0
- package/src/provider/sdk/openai-compatible/src/responses/tool/local-shell.ts +65 -0
- package/src/provider/sdk/openai-compatible/src/responses/tool/web-search-preview.ts +104 -0
- package/src/provider/sdk/openai-compatible/src/responses/tool/web-search.ts +103 -0
- package/src/provider/transform.ts +455 -0
- package/src/pty/index.ts +231 -0
- package/src/security/guardrails.test.ts +341 -0
- package/src/security/guardrails.ts +558 -0
- package/src/security/index.ts +19 -0
- package/src/server/error.ts +36 -0
- package/src/server/project.ts +79 -0
- package/src/server/server.ts +2642 -0
- package/src/server/tui.ts +71 -0
- package/src/session/compaction.ts +223 -0
- package/src/session/index.ts +461 -0
- package/src/session/llm.ts +201 -0
- package/src/session/message-v2.ts +690 -0
- package/src/session/message.ts +189 -0
- package/src/session/processor.ts +409 -0
- package/src/session/prompt/act-switch.txt +5 -0
- package/src/session/prompt/anthropic-20250930.txt +166 -0
- package/src/session/prompt/anthropic.txt +85 -0
- package/src/session/prompt/anthropic_spoof.txt +1 -0
- package/src/session/prompt/beast.txt +103 -0
- package/src/session/prompt/codex.txt +304 -0
- package/src/session/prompt/copilot-gpt-5.txt +138 -0
- package/src/session/prompt/gemini.txt +85 -0
- package/src/session/prompt/max-steps.txt +16 -0
- package/src/session/prompt/plan-reminder-anthropic.txt +35 -0
- package/src/session/prompt/plan.txt +24 -0
- package/src/session/prompt/polaris.txt +84 -0
- package/src/session/prompt/qwen.txt +106 -0
- package/src/session/prompt.ts +1509 -0
- package/src/session/retry.ts +86 -0
- package/src/session/revert.ts +108 -0
- package/src/session/sensitive-filter.test.ts +327 -0
- package/src/session/sensitive-filter.ts +466 -0
- package/src/session/status.ts +76 -0
- package/src/session/summary.ts +194 -0
- package/src/session/system.ts +120 -0
- package/src/session/todo.ts +37 -0
- package/src/share/share-next.ts +194 -0
- package/src/share/share.ts +87 -0
- package/src/shell/shell.ts +67 -0
- package/src/skill/index.ts +1 -0
- package/src/skill/skill.ts +83 -0
- package/src/snapshot/index.ts +197 -0
- package/src/storage/storage.ts +226 -0
- package/src/tests/agent.test.ts +308 -0
- package/src/tests/build-guards.test.ts +267 -0
- package/src/tests/config.test.ts +664 -0
- package/src/tests/tool-registry.test.ts +589 -0
- package/src/tool/bash.ts +317 -0
- package/src/tool/bash.txt +158 -0
- package/src/tool/batch.ts +175 -0
- package/src/tool/batch.txt +24 -0
- package/src/tool/codesearch.ts +168 -0
- package/src/tool/codesearch.txt +12 -0
- package/src/tool/edit.ts +675 -0
- package/src/tool/edit.txt +10 -0
- package/src/tool/glob.ts +65 -0
- package/src/tool/glob.txt +6 -0
- package/src/tool/grep.ts +121 -0
- package/src/tool/grep.txt +8 -0
- package/src/tool/invalid.ts +17 -0
- package/src/tool/ls.ts +110 -0
- package/src/tool/ls.txt +1 -0
- package/src/tool/lsp-diagnostics.ts +26 -0
- package/src/tool/lsp-diagnostics.txt +1 -0
- package/src/tool/lsp-hover.ts +31 -0
- package/src/tool/lsp-hover.txt +1 -0
- package/src/tool/lsp.ts +87 -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 +233 -0
- package/src/tool/patch.txt +1 -0
- package/src/tool/read.ts +219 -0
- package/src/tool/read.txt +12 -0
- package/src/tool/registry.ts +162 -0
- package/src/tool/skill.ts +100 -0
- package/src/tool/task.ts +136 -0
- package/src/tool/task.txt +51 -0
- package/src/tool/todo.ts +39 -0
- package/src/tool/todoread.txt +14 -0
- package/src/tool/todowrite.txt +167 -0
- package/src/tool/tool.ts +71 -0
- package/src/tool/webfetch.ts +198 -0
- package/src/tool/webfetch.txt +13 -0
- package/src/tool/websearch.ts +180 -0
- package/src/tool/websearch.txt +11 -0
- package/src/tool/write.ts +110 -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 +11 -0
- package/src/util/license.ts +325 -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/sst-env.d.ts +9 -0
- package/test/agent/agent.test.ts +146 -0
- package/test/bun.test.ts +53 -0
- package/test/cli/github-remote.test.ts +80 -0
- package/test/config/agent-color.test.ts +66 -0
- package/test/config/config.test.ts +535 -0
- package/test/config/markdown.test.ts +89 -0
- package/test/file/ignore.test.ts +10 -0
- package/test/fixture/fixture.ts +36 -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/preload.ts +57 -0
- package/test/project/project.test.ts +72 -0
- package/test/provider/provider.test.ts +1809 -0
- package/test/provider/transform.test.ts +411 -0
- package/test/session/retry.test.ts +111 -0
- package/test/session/session.test.ts +71 -0
- package/test/skill/skill.test.ts +131 -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 +434 -0
- package/test/tool/grep.test.ts +108 -0
- package/test/tool/patch.test.ts +259 -0
- package/test/tool/read.test.ts +42 -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,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,63 @@
|
|
|
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
|
+
|
|
10
|
+
await Log.init({
|
|
11
|
+
print: process.argv.includes("--print-logs"),
|
|
12
|
+
dev: Installation.isLocal(),
|
|
13
|
+
level: (() => {
|
|
14
|
+
if (Installation.isLocal()) return "DEBUG"
|
|
15
|
+
return "INFO"
|
|
16
|
+
})(),
|
|
17
|
+
})
|
|
18
|
+
|
|
19
|
+
process.on("unhandledRejection", (e) => {
|
|
20
|
+
Log.Default.error("rejection", {
|
|
21
|
+
e: e instanceof Error ? e.message : e,
|
|
22
|
+
})
|
|
23
|
+
})
|
|
24
|
+
|
|
25
|
+
process.on("uncaughtException", (e) => {
|
|
26
|
+
Log.Default.error("exception", {
|
|
27
|
+
e: e instanceof Error ? e.message : e,
|
|
28
|
+
})
|
|
29
|
+
})
|
|
30
|
+
|
|
31
|
+
let server: Bun.Server<BunWebSocketData>
|
|
32
|
+
export const rpc = {
|
|
33
|
+
async server(input: { port: number; hostname: string }) {
|
|
34
|
+
if (server) await server.stop(true)
|
|
35
|
+
try {
|
|
36
|
+
server = Server.listen(input)
|
|
37
|
+
return {
|
|
38
|
+
url: server.url.toString(),
|
|
39
|
+
}
|
|
40
|
+
} catch (e) {
|
|
41
|
+
console.error(e)
|
|
42
|
+
throw e
|
|
43
|
+
}
|
|
44
|
+
},
|
|
45
|
+
async checkUpgrade(input: { directory: string }) {
|
|
46
|
+
await Instance.provide({
|
|
47
|
+
directory: input.directory,
|
|
48
|
+
init: InstanceBootstrap,
|
|
49
|
+
fn: async () => {
|
|
50
|
+
await upgrade().catch(() => {})
|
|
51
|
+
},
|
|
52
|
+
})
|
|
53
|
+
},
|
|
54
|
+
async shutdown() {
|
|
55
|
+
Log.Default.info("worker shutting down")
|
|
56
|
+
await Instance.disposeAll()
|
|
57
|
+
// TODO: this should be awaited, but ws connections are
|
|
58
|
+
// causing this to hang, need to revisit this
|
|
59
|
+
server.stop(true)
|
|
60
|
+
},
|
|
61
|
+
}
|
|
62
|
+
|
|
63
|
+
Rpc.listen(rpc)
|
|
@@ -0,0 +1,344 @@
|
|
|
1
|
+
import type { Argv } from "yargs"
|
|
2
|
+
import { UI } from "../ui"
|
|
3
|
+
import * as prompts from "@clack/prompts"
|
|
4
|
+
import { Installation } from "../../installation"
|
|
5
|
+
import { Global } from "../../global"
|
|
6
|
+
import { $ } from "bun"
|
|
7
|
+
import fs from "fs/promises"
|
|
8
|
+
import path from "path"
|
|
9
|
+
import os from "os"
|
|
10
|
+
|
|
11
|
+
interface UninstallArgs {
|
|
12
|
+
keepConfig: boolean
|
|
13
|
+
keepData: boolean
|
|
14
|
+
dryRun: boolean
|
|
15
|
+
force: boolean
|
|
16
|
+
}
|
|
17
|
+
|
|
18
|
+
interface RemovalTargets {
|
|
19
|
+
directories: Array<{ path: string; label: string; keep: boolean }>
|
|
20
|
+
shellConfig: string | null
|
|
21
|
+
binary: string | null
|
|
22
|
+
}
|
|
23
|
+
|
|
24
|
+
export const UninstallCommand = {
|
|
25
|
+
command: "uninstall",
|
|
26
|
+
describe: "uninstall rird and remove all related files",
|
|
27
|
+
builder: (yargs: Argv) =>
|
|
28
|
+
yargs
|
|
29
|
+
.option("keep-config", {
|
|
30
|
+
alias: "c",
|
|
31
|
+
type: "boolean",
|
|
32
|
+
describe: "keep configuration files",
|
|
33
|
+
default: false,
|
|
34
|
+
})
|
|
35
|
+
.option("keep-data", {
|
|
36
|
+
alias: "d",
|
|
37
|
+
type: "boolean",
|
|
38
|
+
describe: "keep session data and snapshots",
|
|
39
|
+
default: false,
|
|
40
|
+
})
|
|
41
|
+
.option("dry-run", {
|
|
42
|
+
type: "boolean",
|
|
43
|
+
describe: "show what would be removed without removing",
|
|
44
|
+
default: false,
|
|
45
|
+
})
|
|
46
|
+
.option("force", {
|
|
47
|
+
alias: "f",
|
|
48
|
+
type: "boolean",
|
|
49
|
+
describe: "skip confirmation prompts",
|
|
50
|
+
default: false,
|
|
51
|
+
}),
|
|
52
|
+
|
|
53
|
+
handler: async (args: UninstallArgs) => {
|
|
54
|
+
UI.empty()
|
|
55
|
+
UI.println(UI.logo(" "))
|
|
56
|
+
UI.empty()
|
|
57
|
+
prompts.intro("Uninstall RIRD")
|
|
58
|
+
|
|
59
|
+
const method = await Installation.method()
|
|
60
|
+
prompts.log.info(`Installation method: ${method}`)
|
|
61
|
+
|
|
62
|
+
const targets = await collectRemovalTargets(args, method)
|
|
63
|
+
|
|
64
|
+
await showRemovalSummary(targets, method)
|
|
65
|
+
|
|
66
|
+
if (!args.force && !args.dryRun) {
|
|
67
|
+
const confirm = await prompts.confirm({
|
|
68
|
+
message: "Are you sure you want to uninstall?",
|
|
69
|
+
initialValue: false,
|
|
70
|
+
})
|
|
71
|
+
if (!confirm || prompts.isCancel(confirm)) {
|
|
72
|
+
prompts.outro("Cancelled")
|
|
73
|
+
return
|
|
74
|
+
}
|
|
75
|
+
}
|
|
76
|
+
|
|
77
|
+
if (args.dryRun) {
|
|
78
|
+
prompts.log.warn("Dry run - no changes made")
|
|
79
|
+
prompts.outro("Done")
|
|
80
|
+
return
|
|
81
|
+
}
|
|
82
|
+
|
|
83
|
+
await executeUninstall(method, targets)
|
|
84
|
+
|
|
85
|
+
prompts.outro("Done")
|
|
86
|
+
},
|
|
87
|
+
}
|
|
88
|
+
|
|
89
|
+
async function collectRemovalTargets(args: UninstallArgs, method: Installation.Method): Promise<RemovalTargets> {
|
|
90
|
+
const directories: RemovalTargets["directories"] = [
|
|
91
|
+
{ path: Global.Path.data, label: "Data", keep: args.keepData },
|
|
92
|
+
{ path: Global.Path.cache, label: "Cache", keep: false },
|
|
93
|
+
{ path: Global.Path.config, label: "Config", keep: args.keepConfig },
|
|
94
|
+
{ path: Global.Path.state, label: "State", keep: false },
|
|
95
|
+
]
|
|
96
|
+
|
|
97
|
+
const shellConfig = method === "curl" ? await getShellConfigFile() : null
|
|
98
|
+
const binary = method === "curl" ? process.execPath : null
|
|
99
|
+
|
|
100
|
+
return { directories, shellConfig, binary }
|
|
101
|
+
}
|
|
102
|
+
|
|
103
|
+
async function showRemovalSummary(targets: RemovalTargets, method: Installation.Method) {
|
|
104
|
+
prompts.log.message("The following will be removed:")
|
|
105
|
+
|
|
106
|
+
for (const dir of targets.directories) {
|
|
107
|
+
const exists = await fs
|
|
108
|
+
.access(dir.path)
|
|
109
|
+
.then(() => true)
|
|
110
|
+
.catch(() => false)
|
|
111
|
+
if (!exists) continue
|
|
112
|
+
|
|
113
|
+
const size = await getDirectorySize(dir.path)
|
|
114
|
+
const sizeStr = formatSize(size)
|
|
115
|
+
const status = dir.keep ? UI.Style.TEXT_DIM + "(keeping)" : ""
|
|
116
|
+
const prefix = dir.keep ? "○" : "✓"
|
|
117
|
+
|
|
118
|
+
prompts.log.info(` ${prefix} ${dir.label}: ${shortenPath(dir.path)} ${UI.Style.TEXT_DIM}(${sizeStr})${status}`)
|
|
119
|
+
}
|
|
120
|
+
|
|
121
|
+
if (targets.binary) {
|
|
122
|
+
prompts.log.info(` ✓ Binary: ${shortenPath(targets.binary)}`)
|
|
123
|
+
}
|
|
124
|
+
|
|
125
|
+
if (targets.shellConfig) {
|
|
126
|
+
prompts.log.info(` ✓ Shell PATH in ${shortenPath(targets.shellConfig)}`)
|
|
127
|
+
}
|
|
128
|
+
|
|
129
|
+
if (method !== "curl" && method !== "unknown") {
|
|
130
|
+
const cmds: Record<string, string> = {
|
|
131
|
+
npm: "npm uninstall -g rird",
|
|
132
|
+
pnpm: "pnpm uninstall -g rird",
|
|
133
|
+
bun: "bun remove -g rird",
|
|
134
|
+
yarn: "yarn global remove rird",
|
|
135
|
+
brew: "brew uninstall rird",
|
|
136
|
+
}
|
|
137
|
+
prompts.log.info(` ✓ Package: ${cmds[method] || method}`)
|
|
138
|
+
}
|
|
139
|
+
}
|
|
140
|
+
|
|
141
|
+
async function executeUninstall(method: Installation.Method, targets: RemovalTargets) {
|
|
142
|
+
const spinner = prompts.spinner()
|
|
143
|
+
const errors: string[] = []
|
|
144
|
+
|
|
145
|
+
for (const dir of targets.directories) {
|
|
146
|
+
if (dir.keep) {
|
|
147
|
+
prompts.log.step(`Skipping ${dir.label} (--keep-${dir.label.toLowerCase()})`)
|
|
148
|
+
continue
|
|
149
|
+
}
|
|
150
|
+
|
|
151
|
+
const exists = await fs
|
|
152
|
+
.access(dir.path)
|
|
153
|
+
.then(() => true)
|
|
154
|
+
.catch(() => false)
|
|
155
|
+
if (!exists) continue
|
|
156
|
+
|
|
157
|
+
spinner.start(`Removing ${dir.label}...`)
|
|
158
|
+
const err = await fs.rm(dir.path, { recursive: true, force: true }).catch((e) => e)
|
|
159
|
+
if (err) {
|
|
160
|
+
spinner.stop(`Failed to remove ${dir.label}`, 1)
|
|
161
|
+
errors.push(`${dir.label}: ${err.message}`)
|
|
162
|
+
continue
|
|
163
|
+
}
|
|
164
|
+
spinner.stop(`Removed ${dir.label}`)
|
|
165
|
+
}
|
|
166
|
+
|
|
167
|
+
if (targets.shellConfig) {
|
|
168
|
+
spinner.start("Cleaning shell config...")
|
|
169
|
+
const err = await cleanShellConfig(targets.shellConfig).catch((e) => e)
|
|
170
|
+
if (err) {
|
|
171
|
+
spinner.stop("Failed to clean shell config", 1)
|
|
172
|
+
errors.push(`Shell config: ${err.message}`)
|
|
173
|
+
} else {
|
|
174
|
+
spinner.stop("Cleaned shell config")
|
|
175
|
+
}
|
|
176
|
+
}
|
|
177
|
+
|
|
178
|
+
if (method !== "curl" && method !== "unknown") {
|
|
179
|
+
const cmds: Record<string, string[]> = {
|
|
180
|
+
npm: ["npm", "uninstall", "-g", "rird"],
|
|
181
|
+
pnpm: ["pnpm", "uninstall", "-g", "rird"],
|
|
182
|
+
bun: ["bun", "remove", "-g", "rird"],
|
|
183
|
+
yarn: ["yarn", "global", "remove", "rird"],
|
|
184
|
+
brew: ["brew", "uninstall", "rird"],
|
|
185
|
+
}
|
|
186
|
+
|
|
187
|
+
const cmd = cmds[method]
|
|
188
|
+
if (cmd) {
|
|
189
|
+
spinner.start(`Running ${cmd.join(" ")}...`)
|
|
190
|
+
const result = await $`${cmd}`.quiet().nothrow()
|
|
191
|
+
if (result.exitCode !== 0) {
|
|
192
|
+
spinner.stop(`Package manager uninstall failed`, 1)
|
|
193
|
+
prompts.log.warn(`You may need to run manually: ${cmd.join(" ")}`)
|
|
194
|
+
errors.push(`Package manager: exit code ${result.exitCode}`)
|
|
195
|
+
} else {
|
|
196
|
+
spinner.stop("Package removed")
|
|
197
|
+
}
|
|
198
|
+
}
|
|
199
|
+
}
|
|
200
|
+
|
|
201
|
+
if (method === "curl" && targets.binary) {
|
|
202
|
+
UI.empty()
|
|
203
|
+
prompts.log.message("To finish removing the binary, run:")
|
|
204
|
+
prompts.log.info(` rm "${targets.binary}"`)
|
|
205
|
+
|
|
206
|
+
const binDir = path.dirname(targets.binary)
|
|
207
|
+
if (binDir.includes(".opencode")) {
|
|
208
|
+
prompts.log.info(` rmdir "${binDir}" 2>/dev/null`)
|
|
209
|
+
}
|
|
210
|
+
}
|
|
211
|
+
|
|
212
|
+
if (errors.length > 0) {
|
|
213
|
+
UI.empty()
|
|
214
|
+
prompts.log.warn("Some operations failed:")
|
|
215
|
+
for (const err of errors) {
|
|
216
|
+
prompts.log.error(` ${err}`)
|
|
217
|
+
}
|
|
218
|
+
}
|
|
219
|
+
|
|
220
|
+
UI.empty()
|
|
221
|
+
prompts.log.success("Thank you for using RIRD!")
|
|
222
|
+
}
|
|
223
|
+
|
|
224
|
+
async function getShellConfigFile(): Promise<string | null> {
|
|
225
|
+
const shell = path.basename(process.env.SHELL || "bash")
|
|
226
|
+
const home = os.homedir()
|
|
227
|
+
const xdgConfig = process.env.XDG_CONFIG_HOME || path.join(home, ".config")
|
|
228
|
+
|
|
229
|
+
const configFiles: Record<string, string[]> = {
|
|
230
|
+
fish: [path.join(xdgConfig, "fish", "config.fish")],
|
|
231
|
+
zsh: [
|
|
232
|
+
path.join(home, ".zshrc"),
|
|
233
|
+
path.join(home, ".zshenv"),
|
|
234
|
+
path.join(xdgConfig, "zsh", ".zshrc"),
|
|
235
|
+
path.join(xdgConfig, "zsh", ".zshenv"),
|
|
236
|
+
],
|
|
237
|
+
bash: [
|
|
238
|
+
path.join(home, ".bashrc"),
|
|
239
|
+
path.join(home, ".bash_profile"),
|
|
240
|
+
path.join(home, ".profile"),
|
|
241
|
+
path.join(xdgConfig, "bash", ".bashrc"),
|
|
242
|
+
path.join(xdgConfig, "bash", ".bash_profile"),
|
|
243
|
+
],
|
|
244
|
+
ash: [path.join(home, ".ashrc"), path.join(home, ".profile")],
|
|
245
|
+
sh: [path.join(home, ".profile")],
|
|
246
|
+
}
|
|
247
|
+
|
|
248
|
+
const candidates = configFiles[shell] || configFiles.bash
|
|
249
|
+
|
|
250
|
+
for (const file of candidates) {
|
|
251
|
+
const exists = await fs
|
|
252
|
+
.access(file)
|
|
253
|
+
.then(() => true)
|
|
254
|
+
.catch(() => false)
|
|
255
|
+
if (!exists) continue
|
|
256
|
+
|
|
257
|
+
const content = await Bun.file(file)
|
|
258
|
+
.text()
|
|
259
|
+
.catch(() => "")
|
|
260
|
+
if (content.includes("# opencode") || content.includes(".opencode/bin")) {
|
|
261
|
+
return file
|
|
262
|
+
}
|
|
263
|
+
}
|
|
264
|
+
|
|
265
|
+
return null
|
|
266
|
+
}
|
|
267
|
+
|
|
268
|
+
async function cleanShellConfig(file: string) {
|
|
269
|
+
const content = await Bun.file(file).text()
|
|
270
|
+
const lines = content.split("\n")
|
|
271
|
+
|
|
272
|
+
const filtered: string[] = []
|
|
273
|
+
let skip = false
|
|
274
|
+
|
|
275
|
+
for (const line of lines) {
|
|
276
|
+
const trimmed = line.trim()
|
|
277
|
+
|
|
278
|
+
if (trimmed === "# opencode") {
|
|
279
|
+
skip = true
|
|
280
|
+
continue
|
|
281
|
+
}
|
|
282
|
+
|
|
283
|
+
if (skip) {
|
|
284
|
+
skip = false
|
|
285
|
+
if (trimmed.includes(".opencode/bin") || trimmed.includes("fish_add_path")) {
|
|
286
|
+
continue
|
|
287
|
+
}
|
|
288
|
+
}
|
|
289
|
+
|
|
290
|
+
if (
|
|
291
|
+
(trimmed.startsWith("export PATH=") && trimmed.includes(".opencode/bin")) ||
|
|
292
|
+
(trimmed.startsWith("fish_add_path") && trimmed.includes(".opencode"))
|
|
293
|
+
) {
|
|
294
|
+
continue
|
|
295
|
+
}
|
|
296
|
+
|
|
297
|
+
filtered.push(line)
|
|
298
|
+
}
|
|
299
|
+
|
|
300
|
+
while (filtered.length > 0 && filtered[filtered.length - 1].trim() === "") {
|
|
301
|
+
filtered.pop()
|
|
302
|
+
}
|
|
303
|
+
|
|
304
|
+
const output = filtered.join("\n") + "\n"
|
|
305
|
+
await Bun.write(file, output)
|
|
306
|
+
}
|
|
307
|
+
|
|
308
|
+
async function getDirectorySize(dir: string): Promise<number> {
|
|
309
|
+
let total = 0
|
|
310
|
+
|
|
311
|
+
const walk = async (current: string) => {
|
|
312
|
+
const entries = await fs.readdir(current, { withFileTypes: true }).catch(() => [])
|
|
313
|
+
|
|
314
|
+
for (const entry of entries) {
|
|
315
|
+
const full = path.join(current, entry.name)
|
|
316
|
+
if (entry.isDirectory()) {
|
|
317
|
+
await walk(full)
|
|
318
|
+
continue
|
|
319
|
+
}
|
|
320
|
+
if (entry.isFile()) {
|
|
321
|
+
const stat = await fs.stat(full).catch(() => null)
|
|
322
|
+
if (stat) total += stat.size
|
|
323
|
+
}
|
|
324
|
+
}
|
|
325
|
+
}
|
|
326
|
+
|
|
327
|
+
await walk(dir)
|
|
328
|
+
return total
|
|
329
|
+
}
|
|
330
|
+
|
|
331
|
+
function formatSize(bytes: number): string {
|
|
332
|
+
if (bytes < 1024) return `${bytes} B`
|
|
333
|
+
if (bytes < 1024 * 1024) return `${(bytes / 1024).toFixed(1)} KB`
|
|
334
|
+
if (bytes < 1024 * 1024 * 1024) return `${(bytes / (1024 * 1024)).toFixed(1)} MB`
|
|
335
|
+
return `${(bytes / (1024 * 1024 * 1024)).toFixed(1)} GB`
|
|
336
|
+
}
|
|
337
|
+
|
|
338
|
+
function shortenPath(p: string): string {
|
|
339
|
+
const home = os.homedir()
|
|
340
|
+
if (p.startsWith(home)) {
|
|
341
|
+
return p.replace(home, "~")
|
|
342
|
+
}
|
|
343
|
+
return p
|
|
344
|
+
}
|
|
@@ -0,0 +1,100 @@
|
|
|
1
|
+
import type { Argv } from "yargs"
|
|
2
|
+
import { UI } from "../ui"
|
|
3
|
+
import * as prompts from "@clack/prompts"
|
|
4
|
+
import { Installation } from "../../installation"
|
|
5
|
+
|
|
6
|
+
export const UpgradeCommand = {
|
|
7
|
+
command: "upgrade [target]",
|
|
8
|
+
aliases: ["update"],
|
|
9
|
+
describe: "upgrade rird to the latest or a specific version",
|
|
10
|
+
builder: (yargs: Argv) => {
|
|
11
|
+
return yargs
|
|
12
|
+
.positional("target", {
|
|
13
|
+
describe: "version to upgrade to, for ex '0.1.48' or 'v0.1.48'",
|
|
14
|
+
type: "string",
|
|
15
|
+
})
|
|
16
|
+
.option("method", {
|
|
17
|
+
alias: "m",
|
|
18
|
+
describe: "installation method to use",
|
|
19
|
+
type: "string",
|
|
20
|
+
choices: ["curl", "npm", "pnpm", "bun", "brew"],
|
|
21
|
+
})
|
|
22
|
+
},
|
|
23
|
+
handler: async (args: { target?: string; method?: string }) => {
|
|
24
|
+
UI.empty()
|
|
25
|
+
UI.println(UI.logo(" "))
|
|
26
|
+
UI.empty()
|
|
27
|
+
prompts.intro("Upgrade")
|
|
28
|
+
const detectedMethod = await Installation.method()
|
|
29
|
+
const method = (args.method as Installation.Method) ?? detectedMethod
|
|
30
|
+
if (method === "unknown") {
|
|
31
|
+
prompts.log.warn(`rird is installed to ${process.execPath} and may be managed by a package manager`)
|
|
32
|
+
const install = await prompts.select({
|
|
33
|
+
message: "Install anyways?",
|
|
34
|
+
options: [
|
|
35
|
+
{ label: "Yes", value: true },
|
|
36
|
+
{ label: "No", value: false },
|
|
37
|
+
],
|
|
38
|
+
initialValue: true, // Default to Yes for better UX
|
|
39
|
+
})
|
|
40
|
+
if (!install) {
|
|
41
|
+
prompts.outro("Done")
|
|
42
|
+
return
|
|
43
|
+
}
|
|
44
|
+
}
|
|
45
|
+
prompts.log.info("Using method: " + method)
|
|
46
|
+
|
|
47
|
+
// Try to get latest version, fall back to force reinstall if registry fails
|
|
48
|
+
let target: string
|
|
49
|
+
try {
|
|
50
|
+
target = args.target ? args.target.replace(/^v/, "") : await Installation.latest()
|
|
51
|
+
} catch (latestErr) {
|
|
52
|
+
prompts.log.warn("Could not fetch latest version from registry")
|
|
53
|
+
prompts.log.info("Attempting force reinstall to fix broken installation...")
|
|
54
|
+
const spinner = prompts.spinner()
|
|
55
|
+
spinner.start("Reinstalling...")
|
|
56
|
+
const err = await Installation.forceReinstall().catch((err) => err)
|
|
57
|
+
if (err) {
|
|
58
|
+
spinner.stop("Reinstall failed", 1)
|
|
59
|
+
if (err instanceof Installation.UpgradeFailedError) prompts.log.error(err.data.stderr)
|
|
60
|
+
else if (err instanceof Error) prompts.log.error(err.message)
|
|
61
|
+
prompts.log.info("Manual fix: npm install -g rird-cli@latest")
|
|
62
|
+
} else {
|
|
63
|
+
spinner.stop("Reinstall complete")
|
|
64
|
+
}
|
|
65
|
+
prompts.outro("Done")
|
|
66
|
+
return
|
|
67
|
+
}
|
|
68
|
+
|
|
69
|
+
if (Installation.VERSION === target) {
|
|
70
|
+
prompts.log.warn(`rird upgrade skipped: ${target} is already installed`)
|
|
71
|
+
prompts.outro("Done")
|
|
72
|
+
return
|
|
73
|
+
}
|
|
74
|
+
|
|
75
|
+
prompts.log.info(`From ${Installation.VERSION} -> ${target}`)
|
|
76
|
+
const spinner = prompts.spinner()
|
|
77
|
+
spinner.start("Upgrading...")
|
|
78
|
+
let err = await Installation.upgrade(method, target).catch((err) => err)
|
|
79
|
+
|
|
80
|
+
// If upgrade fails, try force reinstall as fallback
|
|
81
|
+
if (err) {
|
|
82
|
+
spinner.stop("Standard upgrade failed, trying force reinstall...", 1)
|
|
83
|
+
const forceSpinner = prompts.spinner()
|
|
84
|
+
forceSpinner.start("Reinstalling...")
|
|
85
|
+
err = await Installation.forceReinstall().catch((err) => err)
|
|
86
|
+
if (err) {
|
|
87
|
+
forceSpinner.stop("Reinstall failed", 1)
|
|
88
|
+
if (err instanceof Installation.UpgradeFailedError) prompts.log.error(err.data.stderr)
|
|
89
|
+
else if (err instanceof Error) prompts.log.error(err.message)
|
|
90
|
+
prompts.log.info("Manual fix: npm install -g rird-cli@latest")
|
|
91
|
+
prompts.outro("Done")
|
|
92
|
+
return
|
|
93
|
+
}
|
|
94
|
+
forceSpinner.stop("Reinstall complete")
|
|
95
|
+
} else {
|
|
96
|
+
spinner.stop("Upgrade complete")
|
|
97
|
+
}
|
|
98
|
+
prompts.outro("Done")
|
|
99
|
+
},
|
|
100
|
+
}
|
|
@@ -0,0 +1,84 @@
|
|
|
1
|
+
import { Server } from "../../server/server"
|
|
2
|
+
import { UI } from "../ui"
|
|
3
|
+
import { cmd } from "./cmd"
|
|
4
|
+
import open from "open"
|
|
5
|
+
import { networkInterfaces } from "os"
|
|
6
|
+
|
|
7
|
+
function getNetworkIPs() {
|
|
8
|
+
const nets = networkInterfaces()
|
|
9
|
+
const results: string[] = []
|
|
10
|
+
|
|
11
|
+
for (const name of Object.keys(nets)) {
|
|
12
|
+
const net = nets[name]
|
|
13
|
+
if (!net) continue
|
|
14
|
+
|
|
15
|
+
for (const netInfo of net) {
|
|
16
|
+
// Skip internal and non-IPv4 addresses
|
|
17
|
+
if (netInfo.internal || netInfo.family !== "IPv4") continue
|
|
18
|
+
|
|
19
|
+
// Skip Docker bridge networks (typically 172.x.x.x)
|
|
20
|
+
if (netInfo.address.startsWith("172.")) continue
|
|
21
|
+
|
|
22
|
+
results.push(netInfo.address)
|
|
23
|
+
}
|
|
24
|
+
}
|
|
25
|
+
|
|
26
|
+
return results
|
|
27
|
+
}
|
|
28
|
+
|
|
29
|
+
export const WebCommand = cmd({
|
|
30
|
+
command: "web",
|
|
31
|
+
builder: (yargs) =>
|
|
32
|
+
yargs
|
|
33
|
+
.option("port", {
|
|
34
|
+
alias: ["p"],
|
|
35
|
+
type: "number",
|
|
36
|
+
describe: "port to listen on",
|
|
37
|
+
default: 0,
|
|
38
|
+
})
|
|
39
|
+
.option("hostname", {
|
|
40
|
+
type: "string",
|
|
41
|
+
describe: "hostname to listen on",
|
|
42
|
+
default: "127.0.0.1",
|
|
43
|
+
}),
|
|
44
|
+
describe: "starts a headless rird server",
|
|
45
|
+
handler: async (args) => {
|
|
46
|
+
const hostname = args.hostname
|
|
47
|
+
const port = args.port
|
|
48
|
+
const server = Server.listen({
|
|
49
|
+
port,
|
|
50
|
+
hostname,
|
|
51
|
+
})
|
|
52
|
+
UI.empty()
|
|
53
|
+
UI.println(UI.logo(" "))
|
|
54
|
+
UI.empty()
|
|
55
|
+
|
|
56
|
+
if (hostname === "0.0.0.0") {
|
|
57
|
+
// Show localhost for local access
|
|
58
|
+
const localhostUrl = `http://localhost:${server.port}`
|
|
59
|
+
UI.println(UI.Style.TEXT_INFO_BOLD + " Local access: ", UI.Style.TEXT_NORMAL, localhostUrl)
|
|
60
|
+
|
|
61
|
+
// Show network IPs for remote access
|
|
62
|
+
const networkIPs = getNetworkIPs()
|
|
63
|
+
if (networkIPs.length > 0) {
|
|
64
|
+
for (const ip of networkIPs) {
|
|
65
|
+
UI.println(
|
|
66
|
+
UI.Style.TEXT_INFO_BOLD + " Network access: ",
|
|
67
|
+
UI.Style.TEXT_NORMAL,
|
|
68
|
+
`http://${ip}:${server.port}`,
|
|
69
|
+
)
|
|
70
|
+
}
|
|
71
|
+
}
|
|
72
|
+
|
|
73
|
+
// Open localhost in browser
|
|
74
|
+
open(localhostUrl.toString()).catch(() => {})
|
|
75
|
+
} else {
|
|
76
|
+
const displayUrl = server.url.toString()
|
|
77
|
+
UI.println(UI.Style.TEXT_INFO_BOLD + " Web interface: ", UI.Style.TEXT_NORMAL, displayUrl)
|
|
78
|
+
open(displayUrl).catch(() => {})
|
|
79
|
+
}
|
|
80
|
+
|
|
81
|
+
await new Promise(() => {})
|
|
82
|
+
await server.stop()
|
|
83
|
+
},
|
|
84
|
+
})
|