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,391 @@
|
|
|
1
|
+
import { Auth } from "../../auth"
|
|
2
|
+
import { cmd } from "./cmd"
|
|
3
|
+
import * as prompts from "@clack/prompts"
|
|
4
|
+
import { UI } from "../ui"
|
|
5
|
+
import { ModelsDev } from "../../provider/models"
|
|
6
|
+
import { map, pipe, sortBy, values } from "remeda"
|
|
7
|
+
import path from "path"
|
|
8
|
+
import os from "os"
|
|
9
|
+
import { Config } from "../../config/config"
|
|
10
|
+
import { Global } from "../../global"
|
|
11
|
+
import { Plugin } from "../../plugin"
|
|
12
|
+
import { Instance } from "../../project/instance"
|
|
13
|
+
import type { Hooks } from "@opencode-ai/plugin"
|
|
14
|
+
|
|
15
|
+
type PluginAuth = NonNullable<Hooks["auth"]>
|
|
16
|
+
|
|
17
|
+
/**
|
|
18
|
+
* Handle plugin-based authentication flow.
|
|
19
|
+
* Returns true if auth was handled, false if it should fall through to default handling.
|
|
20
|
+
*/
|
|
21
|
+
async function handlePluginAuth(plugin: { auth: PluginAuth }, provider: string): Promise<boolean> {
|
|
22
|
+
let index = 0
|
|
23
|
+
if (plugin.auth.methods.length > 1) {
|
|
24
|
+
const method = await prompts.select({
|
|
25
|
+
message: "Login method",
|
|
26
|
+
options: [
|
|
27
|
+
...plugin.auth.methods.map((x, index) => ({
|
|
28
|
+
label: x.label,
|
|
29
|
+
value: index.toString(),
|
|
30
|
+
})),
|
|
31
|
+
],
|
|
32
|
+
})
|
|
33
|
+
if (prompts.isCancel(method)) throw new UI.CancelledError()
|
|
34
|
+
index = parseInt(method)
|
|
35
|
+
}
|
|
36
|
+
const method = plugin.auth.methods[index]
|
|
37
|
+
|
|
38
|
+
// Handle prompts for all auth types
|
|
39
|
+
await new Promise((resolve) => setTimeout(resolve, 10))
|
|
40
|
+
const inputs: Record<string, string> = {}
|
|
41
|
+
if (method.prompts) {
|
|
42
|
+
for (const prompt of method.prompts) {
|
|
43
|
+
if (prompt.condition && !prompt.condition(inputs)) {
|
|
44
|
+
continue
|
|
45
|
+
}
|
|
46
|
+
if (prompt.type === "select") {
|
|
47
|
+
const value = await prompts.select({
|
|
48
|
+
message: prompt.message,
|
|
49
|
+
options: prompt.options,
|
|
50
|
+
})
|
|
51
|
+
if (prompts.isCancel(value)) throw new UI.CancelledError()
|
|
52
|
+
inputs[prompt.key] = value
|
|
53
|
+
} else {
|
|
54
|
+
const value = await prompts.text({
|
|
55
|
+
message: prompt.message,
|
|
56
|
+
placeholder: prompt.placeholder,
|
|
57
|
+
validate: prompt.validate ? (v) => prompt.validate!(v ?? "") : undefined,
|
|
58
|
+
})
|
|
59
|
+
if (prompts.isCancel(value)) throw new UI.CancelledError()
|
|
60
|
+
inputs[prompt.key] = value
|
|
61
|
+
}
|
|
62
|
+
}
|
|
63
|
+
}
|
|
64
|
+
|
|
65
|
+
if (method.type === "oauth") {
|
|
66
|
+
const authorize = await method.authorize(inputs)
|
|
67
|
+
|
|
68
|
+
if (authorize.url) {
|
|
69
|
+
prompts.log.info("Go to: " + authorize.url)
|
|
70
|
+
}
|
|
71
|
+
|
|
72
|
+
if (authorize.method === "auto") {
|
|
73
|
+
if (authorize.instructions) {
|
|
74
|
+
prompts.log.info(authorize.instructions)
|
|
75
|
+
}
|
|
76
|
+
const spinner = prompts.spinner()
|
|
77
|
+
spinner.start("Waiting for authorization...")
|
|
78
|
+
const result = await authorize.callback()
|
|
79
|
+
if (result.type === "failed") {
|
|
80
|
+
spinner.stop("Failed to authorize", 1)
|
|
81
|
+
}
|
|
82
|
+
if (result.type === "success") {
|
|
83
|
+
const saveProvider = result.provider ?? provider
|
|
84
|
+
if ("refresh" in result) {
|
|
85
|
+
const { type: _, provider: __, refresh, access, expires, ...extraFields } = result
|
|
86
|
+
await Auth.set(saveProvider, {
|
|
87
|
+
type: "oauth",
|
|
88
|
+
refresh,
|
|
89
|
+
access,
|
|
90
|
+
expires,
|
|
91
|
+
...extraFields,
|
|
92
|
+
})
|
|
93
|
+
}
|
|
94
|
+
if ("key" in result) {
|
|
95
|
+
await Auth.set(saveProvider, {
|
|
96
|
+
type: "api",
|
|
97
|
+
key: result.key,
|
|
98
|
+
})
|
|
99
|
+
}
|
|
100
|
+
spinner.stop("Login successful")
|
|
101
|
+
}
|
|
102
|
+
}
|
|
103
|
+
|
|
104
|
+
if (authorize.method === "code") {
|
|
105
|
+
const code = await prompts.text({
|
|
106
|
+
message: "Paste the authorization code here: ",
|
|
107
|
+
validate: (x) => (x && x.length > 0 ? undefined : "Required"),
|
|
108
|
+
})
|
|
109
|
+
if (prompts.isCancel(code)) throw new UI.CancelledError()
|
|
110
|
+
const result = await authorize.callback(code)
|
|
111
|
+
if (result.type === "failed") {
|
|
112
|
+
prompts.log.error("Failed to authorize")
|
|
113
|
+
}
|
|
114
|
+
if (result.type === "success") {
|
|
115
|
+
const saveProvider = result.provider ?? provider
|
|
116
|
+
if ("refresh" in result) {
|
|
117
|
+
const { type: _, provider: __, refresh, access, expires, ...extraFields } = result
|
|
118
|
+
await Auth.set(saveProvider, {
|
|
119
|
+
type: "oauth",
|
|
120
|
+
refresh,
|
|
121
|
+
access,
|
|
122
|
+
expires,
|
|
123
|
+
...extraFields,
|
|
124
|
+
})
|
|
125
|
+
}
|
|
126
|
+
if ("key" in result) {
|
|
127
|
+
await Auth.set(saveProvider, {
|
|
128
|
+
type: "api",
|
|
129
|
+
key: result.key,
|
|
130
|
+
})
|
|
131
|
+
}
|
|
132
|
+
prompts.log.success("Login successful")
|
|
133
|
+
}
|
|
134
|
+
}
|
|
135
|
+
|
|
136
|
+
prompts.outro("Done")
|
|
137
|
+
return true
|
|
138
|
+
}
|
|
139
|
+
|
|
140
|
+
if (method.type === "api") {
|
|
141
|
+
if (method.authorize) {
|
|
142
|
+
const result = await method.authorize(inputs)
|
|
143
|
+
if (result.type === "failed") {
|
|
144
|
+
prompts.log.error("Failed to authorize")
|
|
145
|
+
}
|
|
146
|
+
if (result.type === "success") {
|
|
147
|
+
const saveProvider = result.provider ?? provider
|
|
148
|
+
await Auth.set(saveProvider, {
|
|
149
|
+
type: "api",
|
|
150
|
+
key: result.key,
|
|
151
|
+
})
|
|
152
|
+
prompts.log.success("Login successful")
|
|
153
|
+
}
|
|
154
|
+
prompts.outro("Done")
|
|
155
|
+
return true
|
|
156
|
+
}
|
|
157
|
+
}
|
|
158
|
+
|
|
159
|
+
return false
|
|
160
|
+
}
|
|
161
|
+
|
|
162
|
+
export const AuthCommand = cmd({
|
|
163
|
+
command: "auth",
|
|
164
|
+
describe: "manage credentials",
|
|
165
|
+
builder: (yargs) =>
|
|
166
|
+
yargs.command(AuthLoginCommand).command(AuthLogoutCommand).command(AuthListCommand).demandCommand(),
|
|
167
|
+
async handler() {},
|
|
168
|
+
})
|
|
169
|
+
|
|
170
|
+
export const AuthListCommand = cmd({
|
|
171
|
+
command: "list",
|
|
172
|
+
aliases: ["ls"],
|
|
173
|
+
describe: "list providers",
|
|
174
|
+
async handler() {
|
|
175
|
+
UI.empty()
|
|
176
|
+
const authPath = path.join(Global.Path.data, "auth.json")
|
|
177
|
+
const homedir = os.homedir()
|
|
178
|
+
const displayPath = authPath.startsWith(homedir) ? authPath.replace(homedir, "~") : authPath
|
|
179
|
+
prompts.intro(`Credentials ${UI.Style.TEXT_DIM}${displayPath}`)
|
|
180
|
+
const results = Object.entries(await Auth.all())
|
|
181
|
+
const database = await ModelsDev.get()
|
|
182
|
+
|
|
183
|
+
for (const [providerID, result] of results) {
|
|
184
|
+
const name = database[providerID]?.name || providerID
|
|
185
|
+
prompts.log.info(`${name} ${UI.Style.TEXT_DIM}${result.type}`)
|
|
186
|
+
}
|
|
187
|
+
|
|
188
|
+
prompts.outro(`${results.length} credentials`)
|
|
189
|
+
|
|
190
|
+
// Environment variables section
|
|
191
|
+
const activeEnvVars: Array<{ provider: string; envVar: string }> = []
|
|
192
|
+
|
|
193
|
+
for (const [providerID, provider] of Object.entries(database)) {
|
|
194
|
+
for (const envVar of provider.env) {
|
|
195
|
+
if (process.env[envVar]) {
|
|
196
|
+
activeEnvVars.push({
|
|
197
|
+
provider: provider.name || providerID,
|
|
198
|
+
envVar,
|
|
199
|
+
})
|
|
200
|
+
}
|
|
201
|
+
}
|
|
202
|
+
}
|
|
203
|
+
|
|
204
|
+
if (activeEnvVars.length > 0) {
|
|
205
|
+
UI.empty()
|
|
206
|
+
prompts.intro("Environment")
|
|
207
|
+
|
|
208
|
+
for (const { provider, envVar } of activeEnvVars) {
|
|
209
|
+
prompts.log.info(`${provider} ${UI.Style.TEXT_DIM}${envVar}`)
|
|
210
|
+
}
|
|
211
|
+
|
|
212
|
+
prompts.outro(`${activeEnvVars.length} environment variable` + (activeEnvVars.length === 1 ? "" : "s"))
|
|
213
|
+
}
|
|
214
|
+
},
|
|
215
|
+
})
|
|
216
|
+
|
|
217
|
+
export const AuthLoginCommand = cmd({
|
|
218
|
+
command: "login [url]",
|
|
219
|
+
describe: "log in to a provider",
|
|
220
|
+
builder: (yargs) =>
|
|
221
|
+
yargs.positional("url", {
|
|
222
|
+
describe: "RIRD auth provider",
|
|
223
|
+
type: "string",
|
|
224
|
+
}),
|
|
225
|
+
async handler(args) {
|
|
226
|
+
await Instance.provide({
|
|
227
|
+
directory: process.cwd(),
|
|
228
|
+
async fn() {
|
|
229
|
+
UI.empty()
|
|
230
|
+
prompts.intro("Add credential")
|
|
231
|
+
if (args.url) {
|
|
232
|
+
const wellknown = await fetch(`${args.url}/.well-known/opencode`).then((x) => x.json() as any)
|
|
233
|
+
prompts.log.info(`Running \`${wellknown.auth.command.join(" ")}\``)
|
|
234
|
+
const proc = Bun.spawn({
|
|
235
|
+
cmd: wellknown.auth.command,
|
|
236
|
+
stdout: "pipe",
|
|
237
|
+
})
|
|
238
|
+
const exit = await proc.exited
|
|
239
|
+
if (exit !== 0) {
|
|
240
|
+
prompts.log.error("Failed")
|
|
241
|
+
prompts.outro("Done")
|
|
242
|
+
return
|
|
243
|
+
}
|
|
244
|
+
const token = await new Response(proc.stdout).text()
|
|
245
|
+
await Auth.set(args.url, {
|
|
246
|
+
type: "wellknown",
|
|
247
|
+
key: wellknown.auth.env,
|
|
248
|
+
token: token.trim(),
|
|
249
|
+
})
|
|
250
|
+
prompts.log.success("Logged into " + args.url)
|
|
251
|
+
prompts.outro("Done")
|
|
252
|
+
return
|
|
253
|
+
}
|
|
254
|
+
await ModelsDev.refresh().catch(() => {})
|
|
255
|
+
|
|
256
|
+
const config = await Config.get()
|
|
257
|
+
|
|
258
|
+
const disabled = new Set(config.disabled_providers ?? [])
|
|
259
|
+
const enabled = config.enabled_providers ? new Set(config.enabled_providers) : undefined
|
|
260
|
+
|
|
261
|
+
const providers = await ModelsDev.get().then((x) => {
|
|
262
|
+
const filtered: Record<string, (typeof x)[string]> = {}
|
|
263
|
+
for (const [key, value] of Object.entries(x)) {
|
|
264
|
+
if ((enabled ? enabled.has(key) : true) && !disabled.has(key)) {
|
|
265
|
+
filtered[key] = value
|
|
266
|
+
}
|
|
267
|
+
}
|
|
268
|
+
return filtered
|
|
269
|
+
})
|
|
270
|
+
|
|
271
|
+
const priority: Record<string, number> = {
|
|
272
|
+
opencode: 0,
|
|
273
|
+
anthropic: 1,
|
|
274
|
+
"github-copilot": 2,
|
|
275
|
+
openai: 3,
|
|
276
|
+
google: 4,
|
|
277
|
+
openrouter: 5,
|
|
278
|
+
vercel: 6,
|
|
279
|
+
}
|
|
280
|
+
let provider = await prompts.autocomplete({
|
|
281
|
+
message: "Select provider",
|
|
282
|
+
maxItems: 8,
|
|
283
|
+
options: [
|
|
284
|
+
...pipe(
|
|
285
|
+
providers,
|
|
286
|
+
values(),
|
|
287
|
+
sortBy(
|
|
288
|
+
(x) => priority[x.id] ?? 99,
|
|
289
|
+
(x) => x.name ?? x.id,
|
|
290
|
+
),
|
|
291
|
+
map((x) => ({
|
|
292
|
+
label: x.name,
|
|
293
|
+
value: x.id,
|
|
294
|
+
hint: {
|
|
295
|
+
opencode: "recommended",
|
|
296
|
+
anthropic: "Claude Max or API key",
|
|
297
|
+
}[x.id],
|
|
298
|
+
})),
|
|
299
|
+
),
|
|
300
|
+
{
|
|
301
|
+
value: "other",
|
|
302
|
+
label: "Other",
|
|
303
|
+
},
|
|
304
|
+
],
|
|
305
|
+
})
|
|
306
|
+
|
|
307
|
+
if (prompts.isCancel(provider)) throw new UI.CancelledError()
|
|
308
|
+
|
|
309
|
+
const plugin = await Plugin.list().then((x) => x.find((x) => x.auth?.provider === provider))
|
|
310
|
+
if (plugin && plugin.auth) {
|
|
311
|
+
const handled = await handlePluginAuth({ auth: plugin.auth }, provider)
|
|
312
|
+
if (handled) return
|
|
313
|
+
}
|
|
314
|
+
|
|
315
|
+
if (provider === "other") {
|
|
316
|
+
provider = await prompts.text({
|
|
317
|
+
message: "Enter provider id",
|
|
318
|
+
validate: (x) => (x && x.match(/^[0-9a-z-]+$/) ? undefined : "a-z, 0-9 and hyphens only"),
|
|
319
|
+
})
|
|
320
|
+
if (prompts.isCancel(provider)) throw new UI.CancelledError()
|
|
321
|
+
provider = provider.replace(/^@ai-sdk\//, "")
|
|
322
|
+
if (prompts.isCancel(provider)) throw new UI.CancelledError()
|
|
323
|
+
|
|
324
|
+
// Check if a plugin provides auth for this custom provider
|
|
325
|
+
const customPlugin = await Plugin.list().then((x) => x.find((x) => x.auth?.provider === provider))
|
|
326
|
+
if (customPlugin && customPlugin.auth) {
|
|
327
|
+
const handled = await handlePluginAuth({ auth: customPlugin.auth }, provider)
|
|
328
|
+
if (handled) return
|
|
329
|
+
}
|
|
330
|
+
|
|
331
|
+
prompts.log.warn(
|
|
332
|
+
`This only stores a credential for ${provider} - you will need to configure it in rird.json, check the docs for examples.`,
|
|
333
|
+
)
|
|
334
|
+
}
|
|
335
|
+
|
|
336
|
+
if (provider === "amazon-bedrock") {
|
|
337
|
+
prompts.log.info(
|
|
338
|
+
"Amazon bedrock can be configured with standard AWS environment variables like AWS_BEARER_TOKEN_BEDROCK, AWS_PROFILE or AWS_ACCESS_KEY_ID",
|
|
339
|
+
)
|
|
340
|
+
prompts.outro("Done")
|
|
341
|
+
return
|
|
342
|
+
}
|
|
343
|
+
|
|
344
|
+
if (provider === "rird") {
|
|
345
|
+
prompts.log.info("Create an api key at https://rird.ai")
|
|
346
|
+
}
|
|
347
|
+
|
|
348
|
+
if (provider === "vercel") {
|
|
349
|
+
prompts.log.info("You can create an api key at https://vercel.link/ai-gateway-token")
|
|
350
|
+
}
|
|
351
|
+
|
|
352
|
+
const key = await prompts.password({
|
|
353
|
+
message: "Enter your API key",
|
|
354
|
+
validate: (x) => (x && x.length > 0 ? undefined : "Required"),
|
|
355
|
+
})
|
|
356
|
+
if (prompts.isCancel(key)) throw new UI.CancelledError()
|
|
357
|
+
await Auth.set(provider, {
|
|
358
|
+
type: "api",
|
|
359
|
+
key,
|
|
360
|
+
})
|
|
361
|
+
|
|
362
|
+
prompts.outro("Done")
|
|
363
|
+
},
|
|
364
|
+
})
|
|
365
|
+
},
|
|
366
|
+
})
|
|
367
|
+
|
|
368
|
+
export const AuthLogoutCommand = cmd({
|
|
369
|
+
command: "logout",
|
|
370
|
+
describe: "log out from a configured provider",
|
|
371
|
+
async handler() {
|
|
372
|
+
UI.empty()
|
|
373
|
+
const credentials = await Auth.all().then((x) => Object.entries(x))
|
|
374
|
+
prompts.intro("Remove credential")
|
|
375
|
+
if (credentials.length === 0) {
|
|
376
|
+
prompts.log.error("No credentials found")
|
|
377
|
+
return
|
|
378
|
+
}
|
|
379
|
+
const database = await ModelsDev.get()
|
|
380
|
+
const providerID = await prompts.select({
|
|
381
|
+
message: "Select provider",
|
|
382
|
+
options: credentials.map(([key, value]) => ({
|
|
383
|
+
label: (database[key]?.name || key) + UI.Style.TEXT_DIM + " (" + value.type + ")",
|
|
384
|
+
value: key,
|
|
385
|
+
})),
|
|
386
|
+
})
|
|
387
|
+
if (prompts.isCancel(providerID)) throw new UI.CancelledError()
|
|
388
|
+
await Auth.remove(providerID)
|
|
389
|
+
prompts.outro("Logout successful")
|
|
390
|
+
},
|
|
391
|
+
})
|
|
@@ -0,0 +1,15 @@
|
|
|
1
|
+
import { EOL } from "os"
|
|
2
|
+
import { Config } from "../../../config/config"
|
|
3
|
+
import { bootstrap } from "../../bootstrap"
|
|
4
|
+
import { cmd } from "../cmd"
|
|
5
|
+
|
|
6
|
+
export const ConfigCommand = cmd({
|
|
7
|
+
command: "config",
|
|
8
|
+
builder: (yargs) => yargs,
|
|
9
|
+
async handler() {
|
|
10
|
+
await bootstrap(process.cwd(), async () => {
|
|
11
|
+
const config = await Config.get()
|
|
12
|
+
process.stdout.write(JSON.stringify(config, null, 2) + EOL)
|
|
13
|
+
})
|
|
14
|
+
},
|
|
15
|
+
})
|
|
@@ -0,0 +1,91 @@
|
|
|
1
|
+
import { EOL } from "os"
|
|
2
|
+
import { File } from "../../../file"
|
|
3
|
+
import { bootstrap } from "../../bootstrap"
|
|
4
|
+
import { cmd } from "../cmd"
|
|
5
|
+
import { Ripgrep } from "@/file/ripgrep"
|
|
6
|
+
|
|
7
|
+
const FileSearchCommand = cmd({
|
|
8
|
+
command: "search <query>",
|
|
9
|
+
builder: (yargs) =>
|
|
10
|
+
yargs.positional("query", {
|
|
11
|
+
type: "string",
|
|
12
|
+
demandOption: true,
|
|
13
|
+
description: "Search query",
|
|
14
|
+
}),
|
|
15
|
+
async handler(args) {
|
|
16
|
+
await bootstrap(process.cwd(), async () => {
|
|
17
|
+
const results = await File.search({ query: args.query })
|
|
18
|
+
process.stdout.write(results.join(EOL) + EOL)
|
|
19
|
+
})
|
|
20
|
+
},
|
|
21
|
+
})
|
|
22
|
+
|
|
23
|
+
const FileReadCommand = cmd({
|
|
24
|
+
command: "read <path>",
|
|
25
|
+
builder: (yargs) =>
|
|
26
|
+
yargs.positional("path", {
|
|
27
|
+
type: "string",
|
|
28
|
+
demandOption: true,
|
|
29
|
+
description: "File path to read",
|
|
30
|
+
}),
|
|
31
|
+
async handler(args) {
|
|
32
|
+
await bootstrap(process.cwd(), async () => {
|
|
33
|
+
const content = await File.read(args.path)
|
|
34
|
+
process.stdout.write(JSON.stringify(content, null, 2) + EOL)
|
|
35
|
+
})
|
|
36
|
+
},
|
|
37
|
+
})
|
|
38
|
+
|
|
39
|
+
const FileStatusCommand = cmd({
|
|
40
|
+
command: "status",
|
|
41
|
+
builder: (yargs) => yargs,
|
|
42
|
+
async handler() {
|
|
43
|
+
await bootstrap(process.cwd(), async () => {
|
|
44
|
+
const status = await File.status()
|
|
45
|
+
process.stdout.write(JSON.stringify(status, null, 2) + EOL)
|
|
46
|
+
})
|
|
47
|
+
},
|
|
48
|
+
})
|
|
49
|
+
|
|
50
|
+
const FileListCommand = cmd({
|
|
51
|
+
command: "list <path>",
|
|
52
|
+
builder: (yargs) =>
|
|
53
|
+
yargs.positional("path", {
|
|
54
|
+
type: "string",
|
|
55
|
+
demandOption: true,
|
|
56
|
+
description: "File path to list",
|
|
57
|
+
}),
|
|
58
|
+
async handler(args) {
|
|
59
|
+
await bootstrap(process.cwd(), async () => {
|
|
60
|
+
const files = await File.list(args.path)
|
|
61
|
+
process.stdout.write(JSON.stringify(files, null, 2) + EOL)
|
|
62
|
+
})
|
|
63
|
+
},
|
|
64
|
+
})
|
|
65
|
+
|
|
66
|
+
const FileTreeCommand = cmd({
|
|
67
|
+
command: "tree [dir]",
|
|
68
|
+
builder: (yargs) =>
|
|
69
|
+
yargs.positional("dir", {
|
|
70
|
+
type: "string",
|
|
71
|
+
description: "Directory to tree",
|
|
72
|
+
default: process.cwd(),
|
|
73
|
+
}),
|
|
74
|
+
async handler(args) {
|
|
75
|
+
const files = await Ripgrep.tree({ cwd: args.dir, limit: 200 })
|
|
76
|
+
console.log(files)
|
|
77
|
+
},
|
|
78
|
+
})
|
|
79
|
+
|
|
80
|
+
export const FileCommand = cmd({
|
|
81
|
+
command: "file",
|
|
82
|
+
builder: (yargs) =>
|
|
83
|
+
yargs
|
|
84
|
+
.command(FileReadCommand)
|
|
85
|
+
.command(FileStatusCommand)
|
|
86
|
+
.command(FileListCommand)
|
|
87
|
+
.command(FileSearchCommand)
|
|
88
|
+
.command(FileTreeCommand)
|
|
89
|
+
.demandCommand(),
|
|
90
|
+
async handler() {},
|
|
91
|
+
})
|
|
@@ -0,0 +1,43 @@
|
|
|
1
|
+
import { Global } from "../../../global"
|
|
2
|
+
import { bootstrap } from "../../bootstrap"
|
|
3
|
+
import { cmd } from "../cmd"
|
|
4
|
+
import { ConfigCommand } from "./config"
|
|
5
|
+
import { FileCommand } from "./file"
|
|
6
|
+
import { LSPCommand } from "./lsp"
|
|
7
|
+
import { RipgrepCommand } from "./ripgrep"
|
|
8
|
+
import { ScrapCommand } from "./scrap"
|
|
9
|
+
import { SkillCommand } from "./skill"
|
|
10
|
+
import { SnapshotCommand } from "./snapshot"
|
|
11
|
+
|
|
12
|
+
export const DebugCommand = cmd({
|
|
13
|
+
command: "debug",
|
|
14
|
+
builder: (yargs) =>
|
|
15
|
+
yargs
|
|
16
|
+
.command(ConfigCommand)
|
|
17
|
+
.command(LSPCommand)
|
|
18
|
+
.command(RipgrepCommand)
|
|
19
|
+
.command(FileCommand)
|
|
20
|
+
.command(ScrapCommand)
|
|
21
|
+
.command(SkillCommand)
|
|
22
|
+
.command(SnapshotCommand)
|
|
23
|
+
.command(PathsCommand)
|
|
24
|
+
.command({
|
|
25
|
+
command: "wait",
|
|
26
|
+
async handler() {
|
|
27
|
+
await bootstrap(process.cwd(), async () => {
|
|
28
|
+
await new Promise((resolve) => setTimeout(resolve, 1_000 * 60 * 60 * 24))
|
|
29
|
+
})
|
|
30
|
+
},
|
|
31
|
+
})
|
|
32
|
+
.demandCommand(),
|
|
33
|
+
async handler() {},
|
|
34
|
+
})
|
|
35
|
+
|
|
36
|
+
const PathsCommand = cmd({
|
|
37
|
+
command: "paths",
|
|
38
|
+
handler() {
|
|
39
|
+
for (const [key, value] of Object.entries(Global.Path)) {
|
|
40
|
+
console.log(key.padEnd(10), value)
|
|
41
|
+
}
|
|
42
|
+
},
|
|
43
|
+
})
|
|
@@ -0,0 +1,48 @@
|
|
|
1
|
+
import { LSP } from "../../../lsp"
|
|
2
|
+
import { bootstrap } from "../../bootstrap"
|
|
3
|
+
import { cmd } from "../cmd"
|
|
4
|
+
import { Log } from "../../../util/log"
|
|
5
|
+
import { EOL } from "os"
|
|
6
|
+
|
|
7
|
+
export const LSPCommand = cmd({
|
|
8
|
+
command: "lsp",
|
|
9
|
+
builder: (yargs) =>
|
|
10
|
+
yargs.command(DiagnosticsCommand).command(SymbolsCommand).command(DocumentSymbolsCommand).demandCommand(),
|
|
11
|
+
async handler() {},
|
|
12
|
+
})
|
|
13
|
+
|
|
14
|
+
const DiagnosticsCommand = cmd({
|
|
15
|
+
command: "diagnostics <file>",
|
|
16
|
+
builder: (yargs) => yargs.positional("file", { type: "string", demandOption: true }),
|
|
17
|
+
async handler(args) {
|
|
18
|
+
await bootstrap(process.cwd(), async () => {
|
|
19
|
+
await LSP.touchFile(args.file, true)
|
|
20
|
+
await Bun.sleep(1000)
|
|
21
|
+
process.stdout.write(JSON.stringify(await LSP.diagnostics(), null, 2) + EOL)
|
|
22
|
+
})
|
|
23
|
+
},
|
|
24
|
+
})
|
|
25
|
+
|
|
26
|
+
export const SymbolsCommand = cmd({
|
|
27
|
+
command: "symbols <query>",
|
|
28
|
+
builder: (yargs) => yargs.positional("query", { type: "string", demandOption: true }),
|
|
29
|
+
async handler(args) {
|
|
30
|
+
await bootstrap(process.cwd(), async () => {
|
|
31
|
+
using _ = Log.Default.time("symbols")
|
|
32
|
+
const results = await LSP.workspaceSymbol(args.query)
|
|
33
|
+
process.stdout.write(JSON.stringify(results, null, 2) + EOL)
|
|
34
|
+
})
|
|
35
|
+
},
|
|
36
|
+
})
|
|
37
|
+
|
|
38
|
+
export const DocumentSymbolsCommand = cmd({
|
|
39
|
+
command: "document-symbols <uri>",
|
|
40
|
+
builder: (yargs) => yargs.positional("uri", { type: "string", demandOption: true }),
|
|
41
|
+
async handler(args) {
|
|
42
|
+
await bootstrap(process.cwd(), async () => {
|
|
43
|
+
using _ = Log.Default.time("document-symbols")
|
|
44
|
+
const results = await LSP.documentSymbol(args.uri)
|
|
45
|
+
process.stdout.write(JSON.stringify(results, null, 2) + EOL)
|
|
46
|
+
})
|
|
47
|
+
},
|
|
48
|
+
})
|
|
@@ -0,0 +1,83 @@
|
|
|
1
|
+
import { EOL } from "os"
|
|
2
|
+
import { Ripgrep } from "../../../file/ripgrep"
|
|
3
|
+
import { Instance } from "../../../project/instance"
|
|
4
|
+
import { bootstrap } from "../../bootstrap"
|
|
5
|
+
import { cmd } from "../cmd"
|
|
6
|
+
|
|
7
|
+
export const RipgrepCommand = cmd({
|
|
8
|
+
command: "rg",
|
|
9
|
+
builder: (yargs) => yargs.command(TreeCommand).command(FilesCommand).command(SearchCommand).demandCommand(),
|
|
10
|
+
async handler() {},
|
|
11
|
+
})
|
|
12
|
+
|
|
13
|
+
const TreeCommand = cmd({
|
|
14
|
+
command: "tree",
|
|
15
|
+
builder: (yargs) =>
|
|
16
|
+
yargs.option("limit", {
|
|
17
|
+
type: "number",
|
|
18
|
+
}),
|
|
19
|
+
async handler(args) {
|
|
20
|
+
await bootstrap(process.cwd(), async () => {
|
|
21
|
+
process.stdout.write((await Ripgrep.tree({ cwd: Instance.directory, limit: args.limit })) + EOL)
|
|
22
|
+
})
|
|
23
|
+
},
|
|
24
|
+
})
|
|
25
|
+
|
|
26
|
+
const FilesCommand = cmd({
|
|
27
|
+
command: "files",
|
|
28
|
+
builder: (yargs) =>
|
|
29
|
+
yargs
|
|
30
|
+
.option("query", {
|
|
31
|
+
type: "string",
|
|
32
|
+
description: "Filter files by query",
|
|
33
|
+
})
|
|
34
|
+
.option("glob", {
|
|
35
|
+
type: "string",
|
|
36
|
+
description: "Glob pattern to match files",
|
|
37
|
+
})
|
|
38
|
+
.option("limit", {
|
|
39
|
+
type: "number",
|
|
40
|
+
description: "Limit number of results",
|
|
41
|
+
}),
|
|
42
|
+
async handler(args) {
|
|
43
|
+
await bootstrap(process.cwd(), async () => {
|
|
44
|
+
const files: string[] = []
|
|
45
|
+
for await (const file of Ripgrep.files({
|
|
46
|
+
cwd: Instance.directory,
|
|
47
|
+
glob: args.glob ? [args.glob] : undefined,
|
|
48
|
+
})) {
|
|
49
|
+
files.push(file)
|
|
50
|
+
if (args.limit && files.length >= args.limit) break
|
|
51
|
+
}
|
|
52
|
+
process.stdout.write(files.join(EOL) + EOL)
|
|
53
|
+
})
|
|
54
|
+
},
|
|
55
|
+
})
|
|
56
|
+
|
|
57
|
+
const SearchCommand = cmd({
|
|
58
|
+
command: "search <pattern>",
|
|
59
|
+
builder: (yargs) =>
|
|
60
|
+
yargs
|
|
61
|
+
.positional("pattern", {
|
|
62
|
+
type: "string",
|
|
63
|
+
demandOption: true,
|
|
64
|
+
description: "Search pattern",
|
|
65
|
+
})
|
|
66
|
+
.option("glob", {
|
|
67
|
+
type: "array",
|
|
68
|
+
description: "File glob patterns",
|
|
69
|
+
})
|
|
70
|
+
.option("limit", {
|
|
71
|
+
type: "number",
|
|
72
|
+
description: "Limit number of results",
|
|
73
|
+
}),
|
|
74
|
+
async handler(args) {
|
|
75
|
+
const results = await Ripgrep.search({
|
|
76
|
+
cwd: process.cwd(),
|
|
77
|
+
pattern: args.pattern,
|
|
78
|
+
glob: args.glob as string[] | undefined,
|
|
79
|
+
limit: args.limit,
|
|
80
|
+
})
|
|
81
|
+
process.stdout.write(JSON.stringify(results, null, 2) + EOL)
|
|
82
|
+
},
|
|
83
|
+
})
|
|
@@ -0,0 +1,15 @@
|
|
|
1
|
+
import { EOL } from "os"
|
|
2
|
+
import { Project } from "../../../project/project"
|
|
3
|
+
import { Log } from "../../../util/log"
|
|
4
|
+
import { cmd } from "../cmd"
|
|
5
|
+
|
|
6
|
+
export const ScrapCommand = cmd({
|
|
7
|
+
command: "scrap",
|
|
8
|
+
builder: (yargs) => yargs,
|
|
9
|
+
async handler() {
|
|
10
|
+
const timer = Log.Default.time("scrap")
|
|
11
|
+
const list = await Project.list()
|
|
12
|
+
process.stdout.write(JSON.stringify(list, null, 2) + EOL)
|
|
13
|
+
timer.stop()
|
|
14
|
+
},
|
|
15
|
+
})
|