bincode-cli 1.0.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/README.md +15 -0
- package/bin/bincode +98 -0
- package/bunfig.toml +4 -0
- package/package.json +124 -0
- package/parsers-config.ts +239 -0
- package/script/build.ts +167 -0
- package/script/postinstall.mjs +206 -0
- package/script/publish.ts +99 -0
- package/script/schema.ts +47 -0
- package/src/acp/README.md +164 -0
- package/src/acp/agent.ts +1051 -0
- package/src/acp/session.ts +101 -0
- package/src/acp/types.ts +22 -0
- package/src/agent/agent.ts +398 -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/bineric-login.ts +506 -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/auth-check.ts +61 -0
- package/src/cli/bootstrap.ts +21 -0
- package/src/cli/cmd/acp.ts +88 -0
- package/src/cli/cmd/agent.ts +256 -0
- package/src/cli/cmd/auth.ts +436 -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 +1399 -0
- package/src/cli/cmd/import.ts +98 -0
- package/src/cli/cmd/login.ts +112 -0
- package/src/cli/cmd/logout.ts +38 -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 +669 -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 +123 -0
- package/src/cli/cmd/tui/component/dialog-mcp.tsx +86 -0
- package/src/cli/cmd/tui/component/dialog-model.tsx +223 -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-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/logo.tsx +32 -0
- package/src/cli/cmd/tui/component/prompt/autocomplete.tsx +560 -0
- package/src/cli/cmd/tui/component/prompt/history.tsx +108 -0
- package/src/cli/cmd/tui/component/prompt/index.tsx +1052 -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 +339 -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/bincode.json +245 -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/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 +105 -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 +141 -0
- package/src/cli/cmd/tui/routes/session/index.tsx +1888 -0
- package/src/cli/cmd/tui/routes/session/sidebar.tsx +321 -0
- package/src/cli/cmd/tui/spawn.ts +60 -0
- package/src/cli/cmd/tui/thread.ts +120 -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 +330 -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 +67 -0
- package/src/cli/cmd/web.ts +84 -0
- package/src/cli/error.ts +55 -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 +217 -0
- package/src/installation/index.ts +196 -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 +654 -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 +101 -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 +101 -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/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 +458 -0
- package/src/session/llm.ts +201 -0
- package/src/session/message-v2.ts +659 -0
- package/src/session/message.ts +189 -0
- package/src/session/processor.ts +409 -0
- package/src/session/prompt/anthropic-20250930.txt +166 -0
- package/src/session/prompt/anthropic.txt +104 -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/polaris.txt +106 -0
- package/src/session/prompt/qwen.txt +109 -0
- package/src/session/prompt.ts +1446 -0
- package/src/session/retry.ts +86 -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 +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/tool/bash.ts +306 -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 +138 -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 +60 -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 +187 -0
- package/src/tool/webfetch.txt +13 -0
- package/src/tool/websearch.ts +150 -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/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/tsconfig.json +16 -0
|
@@ -0,0 +1,436 @@
|
|
|
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 "@bincode-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: "bincode 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/bincode`).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
|
+
Bineric: 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
|
+
Bineric: "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 configure it in bincode.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 === "bincode") {
|
|
345
|
+
prompts.log.info("Create an api key at https://bincode.ai/auth")
|
|
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
|
+
// Handle Bineric provider with browser-based login
|
|
353
|
+
if (provider === "Bineric" || provider === "bineric") {
|
|
354
|
+
try {
|
|
355
|
+
const { BinericLogin } = await import("../../auth/bineric-login")
|
|
356
|
+
const baseUrl = await BinericLogin.getBaseUrl()
|
|
357
|
+
const { token } = await BinericLogin.login(baseUrl)
|
|
358
|
+
await BinericLogin.updateConfig(token)
|
|
359
|
+
prompts.log.success("Logged in successfully")
|
|
360
|
+
await BinericLogin.stop()
|
|
361
|
+
prompts.outro("Done")
|
|
362
|
+
return
|
|
363
|
+
} catch (error) {
|
|
364
|
+
prompts.log.error(`Login failed: ${error instanceof Error ? error.message : String(error)}`)
|
|
365
|
+
await import("../../auth/bineric-login").then((m) => m.BinericLogin.stop())
|
|
366
|
+
prompts.outro("Done")
|
|
367
|
+
return
|
|
368
|
+
}
|
|
369
|
+
}
|
|
370
|
+
|
|
371
|
+
const key = await prompts.password({
|
|
372
|
+
message: "Enter your API key",
|
|
373
|
+
validate: (x) => (x && x.length > 0 ? undefined : "Required"),
|
|
374
|
+
})
|
|
375
|
+
if (prompts.isCancel(key)) throw new UI.CancelledError()
|
|
376
|
+
await Auth.set(provider, {
|
|
377
|
+
type: "api",
|
|
378
|
+
key,
|
|
379
|
+
})
|
|
380
|
+
|
|
381
|
+
prompts.outro("Done")
|
|
382
|
+
},
|
|
383
|
+
})
|
|
384
|
+
},
|
|
385
|
+
})
|
|
386
|
+
|
|
387
|
+
export const AuthLogoutCommand = cmd({
|
|
388
|
+
command: "logout",
|
|
389
|
+
describe: "log out from a configured provider",
|
|
390
|
+
async handler() {
|
|
391
|
+
await Instance.provide({
|
|
392
|
+
directory: process.cwd(),
|
|
393
|
+
async fn() {
|
|
394
|
+
UI.empty()
|
|
395
|
+
prompts.intro("Remove credential")
|
|
396
|
+
|
|
397
|
+
const { BinericLogin } = await import("../../auth/bineric-login")
|
|
398
|
+
const hasBinericToken = await BinericLogin.hasApiKey()
|
|
399
|
+
const credentials = await Auth.all().then((x) => Object.entries(x))
|
|
400
|
+
const database = await ModelsDev.get()
|
|
401
|
+
|
|
402
|
+
const choices = credentials.map(([key, value]) => ({
|
|
403
|
+
label: (database[key]?.name || key) + UI.Style.TEXT_DIM + " (" + value.type + ")",
|
|
404
|
+
value: key,
|
|
405
|
+
}))
|
|
406
|
+
|
|
407
|
+
if (hasBinericToken) {
|
|
408
|
+
choices.push({ value: "Bineric", label: "Bineric" + UI.Style.TEXT_DIM + " (bincode.json)" })
|
|
409
|
+
}
|
|
410
|
+
|
|
411
|
+
if (choices.length === 0) {
|
|
412
|
+
prompts.log.error("No credentials found")
|
|
413
|
+
prompts.outro("Done")
|
|
414
|
+
return
|
|
415
|
+
}
|
|
416
|
+
|
|
417
|
+
const providerID = await prompts.select({
|
|
418
|
+
message: "Select provider",
|
|
419
|
+
options: choices,
|
|
420
|
+
})
|
|
421
|
+
|
|
422
|
+
if (prompts.isCancel(providerID)) throw new UI.CancelledError()
|
|
423
|
+
|
|
424
|
+
if (providerID === "Bineric") {
|
|
425
|
+
await BinericLogin.removeToken()
|
|
426
|
+
prompts.log.success("Logged out from Bineric")
|
|
427
|
+
} else {
|
|
428
|
+
await Auth.remove(providerID)
|
|
429
|
+
prompts.log.success(`Logged out from ${database[providerID]?.name || providerID}`)
|
|
430
|
+
}
|
|
431
|
+
|
|
432
|
+
prompts.outro("Logout successful")
|
|
433
|
+
},
|
|
434
|
+
})
|
|
435
|
+
},
|
|
436
|
+
})
|
|
@@ -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
|
+
})
|