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
package/src/index.ts
ADDED
|
@@ -0,0 +1,217 @@
|
|
|
1
|
+
import yargs from "yargs"
|
|
2
|
+
import { hideBin } from "yargs/helpers"
|
|
3
|
+
import { RunCommand } from "./cli/cmd/run"
|
|
4
|
+
import { GenerateCommand } from "./cli/cmd/generate"
|
|
5
|
+
import { Log } from "./util/log"
|
|
6
|
+
import { AuthCommand } from "./cli/cmd/auth"
|
|
7
|
+
import { LoginCommand } from "./cli/cmd/login"
|
|
8
|
+
import { LogoutCommand } from "./cli/cmd/logout"
|
|
9
|
+
import { AgentCommand } from "./cli/cmd/agent"
|
|
10
|
+
import { UpgradeCommand } from "./cli/cmd/upgrade"
|
|
11
|
+
import { UninstallCommand } from "./cli/cmd/uninstall"
|
|
12
|
+
import { ModelsCommand } from "./cli/cmd/models"
|
|
13
|
+
import { UI } from "./cli/ui"
|
|
14
|
+
import { Installation } from "./installation"
|
|
15
|
+
import { NamedError } from "@bincode-ai/util/error"
|
|
16
|
+
import { FormatError } from "./cli/error"
|
|
17
|
+
import { ServeCommand } from "./cli/cmd/serve"
|
|
18
|
+
import { DebugCommand } from "./cli/cmd/debug"
|
|
19
|
+
import { StatsCommand } from "./cli/cmd/stats"
|
|
20
|
+
import { McpCommand } from "./cli/cmd/mcp"
|
|
21
|
+
import { GithubCommand } from "./cli/cmd/github"
|
|
22
|
+
import { ExportCommand } from "./cli/cmd/export"
|
|
23
|
+
import { ImportCommand } from "./cli/cmd/import"
|
|
24
|
+
import { AttachCommand } from "./cli/cmd/tui/attach"
|
|
25
|
+
import { TuiThreadCommand } from "./cli/cmd/tui/thread"
|
|
26
|
+
import { TuiSpawnCommand } from "./cli/cmd/tui/spawn"
|
|
27
|
+
import { AcpCommand } from "./cli/cmd/acp"
|
|
28
|
+
import { EOL } from "os"
|
|
29
|
+
import { WebCommand } from "./cli/cmd/web"
|
|
30
|
+
import { PrCommand } from "./cli/cmd/pr"
|
|
31
|
+
import { SessionCommand } from "./cli/cmd/session"
|
|
32
|
+
import { checkAuth } from "./cli/auth-check"
|
|
33
|
+
import { Filesystem } from "./util/filesystem"
|
|
34
|
+
|
|
35
|
+
// Load .env file from project root (up the directory tree)
|
|
36
|
+
// Bun automatically loads .env from current directory, but since we run with --cwd packages/bincode,
|
|
37
|
+
// we need to manually load from project root
|
|
38
|
+
async function loadEnvFile() {
|
|
39
|
+
const currentDir = process.cwd()
|
|
40
|
+
const envFiles = await Filesystem.findUp(".env", currentDir)
|
|
41
|
+
if (envFiles.length > 0) {
|
|
42
|
+
// Use the first .env file found (closest to current directory)
|
|
43
|
+
const envPath = envFiles[0]
|
|
44
|
+
try {
|
|
45
|
+
// Use Bun's built-in dotenv parser
|
|
46
|
+
const envContent = await Bun.file(envPath).text()
|
|
47
|
+
const lines = envContent.split("\n")
|
|
48
|
+
for (const line of lines) {
|
|
49
|
+
const trimmed = line.trim()
|
|
50
|
+
// Skip comments and empty lines
|
|
51
|
+
if (!trimmed || trimmed.startsWith("#")) continue
|
|
52
|
+
const match = trimmed.match(/^([^=]+)=(.*)$/)
|
|
53
|
+
if (match) {
|
|
54
|
+
const key = match[1].trim()
|
|
55
|
+
let value = match[2].trim()
|
|
56
|
+
// Remove quotes if present
|
|
57
|
+
if ((value.startsWith('"') && value.endsWith('"')) || (value.startsWith("'") && value.endsWith("'"))) {
|
|
58
|
+
value = value.slice(1, -1)
|
|
59
|
+
}
|
|
60
|
+
// Only set if not already set (env vars take precedence over .env file)
|
|
61
|
+
if (!process.env[key]) {
|
|
62
|
+
process.env[key] = value
|
|
63
|
+
}
|
|
64
|
+
}
|
|
65
|
+
}
|
|
66
|
+
} catch (error) {
|
|
67
|
+
// Silently fail if .env file can't be read
|
|
68
|
+
}
|
|
69
|
+
}
|
|
70
|
+
}
|
|
71
|
+
|
|
72
|
+
// Load .env file before anything else
|
|
73
|
+
await loadEnvFile()
|
|
74
|
+
|
|
75
|
+
process.on("unhandledRejection", (e) => {
|
|
76
|
+
Log.Default.error("rejection", {
|
|
77
|
+
e: e instanceof Error ? e.message : e,
|
|
78
|
+
})
|
|
79
|
+
})
|
|
80
|
+
|
|
81
|
+
process.on("uncaughtException", (e) => {
|
|
82
|
+
Log.Default.error("exception", {
|
|
83
|
+
e: e instanceof Error ? e.message : e,
|
|
84
|
+
})
|
|
85
|
+
})
|
|
86
|
+
|
|
87
|
+
const cli = yargs(hideBin(process.argv))
|
|
88
|
+
.parserConfiguration({ "populate--": true })
|
|
89
|
+
.scriptName("bincode")
|
|
90
|
+
.wrap(100)
|
|
91
|
+
.help("help", "show help")
|
|
92
|
+
.alias("help", "h")
|
|
93
|
+
.version("version", "show version number", Installation.VERSION)
|
|
94
|
+
.alias("version", "v")
|
|
95
|
+
.option("print-logs", {
|
|
96
|
+
describe: "print logs to stderr",
|
|
97
|
+
type: "boolean",
|
|
98
|
+
})
|
|
99
|
+
.option("log-level", {
|
|
100
|
+
describe: "log level",
|
|
101
|
+
type: "string",
|
|
102
|
+
choices: ["DEBUG", "INFO", "WARN", "ERROR"],
|
|
103
|
+
})
|
|
104
|
+
.middleware(async (opts) => {
|
|
105
|
+
await Log.init({
|
|
106
|
+
print: process.argv.includes("--print-logs"),
|
|
107
|
+
dev: Installation.isLocal(),
|
|
108
|
+
level: (() => {
|
|
109
|
+
if (opts.logLevel) return opts.logLevel as Log.Level
|
|
110
|
+
if (Installation.isLocal()) return "DEBUG"
|
|
111
|
+
return "INFO"
|
|
112
|
+
})(),
|
|
113
|
+
})
|
|
114
|
+
|
|
115
|
+
process.env.AGENT = "1"
|
|
116
|
+
process.env.BINCODE = "1"
|
|
117
|
+
|
|
118
|
+
Log.Default.info("bincode", {
|
|
119
|
+
version: Installation.VERSION,
|
|
120
|
+
args: process.argv.slice(2),
|
|
121
|
+
})
|
|
122
|
+
|
|
123
|
+
// Check authentication before running commands (except auth and login commands)
|
|
124
|
+
const command = process.argv[2]
|
|
125
|
+
const authStatus = await checkAuth(command)
|
|
126
|
+
|
|
127
|
+
if (authStatus.needsLogin) {
|
|
128
|
+
UI.error("Authentication required")
|
|
129
|
+
console.error("")
|
|
130
|
+
console.error("Please login first by running:")
|
|
131
|
+
console.error(" bincode login")
|
|
132
|
+
console.error("")
|
|
133
|
+
process.exit(1)
|
|
134
|
+
}
|
|
135
|
+
})
|
|
136
|
+
.usage("\n" + UI.logo())
|
|
137
|
+
.command(LoginCommand)
|
|
138
|
+
.command(LogoutCommand)
|
|
139
|
+
.command(AcpCommand)
|
|
140
|
+
.command(McpCommand)
|
|
141
|
+
.command(TuiThreadCommand)
|
|
142
|
+
.command(TuiSpawnCommand)
|
|
143
|
+
.command(AttachCommand)
|
|
144
|
+
.command(RunCommand)
|
|
145
|
+
.command(GenerateCommand)
|
|
146
|
+
.command(DebugCommand)
|
|
147
|
+
.command(AuthCommand)
|
|
148
|
+
.command(AgentCommand)
|
|
149
|
+
.command(UpgradeCommand)
|
|
150
|
+
.command(UninstallCommand)
|
|
151
|
+
.command(ServeCommand)
|
|
152
|
+
.command(WebCommand)
|
|
153
|
+
.command(ModelsCommand)
|
|
154
|
+
.command(StatsCommand)
|
|
155
|
+
.command(ExportCommand)
|
|
156
|
+
.command(ImportCommand)
|
|
157
|
+
.command(GithubCommand)
|
|
158
|
+
.command(PrCommand)
|
|
159
|
+
.command(SessionCommand)
|
|
160
|
+
.fail((msg) => {
|
|
161
|
+
if (
|
|
162
|
+
msg.startsWith("Unknown argument") ||
|
|
163
|
+
msg.startsWith("Not enough non-option arguments") ||
|
|
164
|
+
msg.startsWith("Invalid values:")
|
|
165
|
+
) {
|
|
166
|
+
cli.showHelp("log")
|
|
167
|
+
}
|
|
168
|
+
process.exit(1)
|
|
169
|
+
})
|
|
170
|
+
.strict()
|
|
171
|
+
|
|
172
|
+
try {
|
|
173
|
+
await cli.parse()
|
|
174
|
+
} catch (e) {
|
|
175
|
+
let data: Record<string, any> = {}
|
|
176
|
+
if (e instanceof NamedError) {
|
|
177
|
+
const obj = e.toObject()
|
|
178
|
+
Object.assign(data, {
|
|
179
|
+
...obj.data,
|
|
180
|
+
})
|
|
181
|
+
}
|
|
182
|
+
|
|
183
|
+
if (e instanceof Error) {
|
|
184
|
+
Object.assign(data, {
|
|
185
|
+
name: e.name,
|
|
186
|
+
message: e.message,
|
|
187
|
+
cause: e.cause?.toString(),
|
|
188
|
+
stack: e.stack,
|
|
189
|
+
})
|
|
190
|
+
}
|
|
191
|
+
|
|
192
|
+
if (e instanceof ResolveMessage) {
|
|
193
|
+
Object.assign(data, {
|
|
194
|
+
name: e.name,
|
|
195
|
+
message: e.message,
|
|
196
|
+
code: e.code,
|
|
197
|
+
specifier: e.specifier,
|
|
198
|
+
referrer: e.referrer,
|
|
199
|
+
position: e.position,
|
|
200
|
+
importKind: e.importKind,
|
|
201
|
+
})
|
|
202
|
+
}
|
|
203
|
+
Log.Default.error("fatal", data)
|
|
204
|
+
const formatted = FormatError(e)
|
|
205
|
+
if (formatted) UI.error(formatted)
|
|
206
|
+
if (formatted === undefined) {
|
|
207
|
+
UI.error("Unexpected error, check log file at " + Log.file() + " for more details" + EOL)
|
|
208
|
+
console.error(e)
|
|
209
|
+
}
|
|
210
|
+
process.exitCode = 1
|
|
211
|
+
} finally {
|
|
212
|
+
// Some subprocesses don't react properly to SIGTERM and similar signals.
|
|
213
|
+
// Most notably, some docker-container-based MCP servers don't handle such signals unless
|
|
214
|
+
// run using `docker run --init`.
|
|
215
|
+
// Explicitly exit to avoid any hanging subprocesses.
|
|
216
|
+
process.exit()
|
|
217
|
+
}
|
|
@@ -0,0 +1,196 @@
|
|
|
1
|
+
import { BusEvent } from "@/bus/bus-event"
|
|
2
|
+
import path from "path"
|
|
3
|
+
import { $ } from "bun"
|
|
4
|
+
import z from "zod"
|
|
5
|
+
import { NamedError } from "@bincode-ai/util/error"
|
|
6
|
+
import { Log } from "../util/log"
|
|
7
|
+
import { iife } from "@/util/iife"
|
|
8
|
+
import { Flag } from "../flag/flag"
|
|
9
|
+
|
|
10
|
+
declare global {
|
|
11
|
+
const BINCODE_VERSION: string
|
|
12
|
+
const BINCODE_CHANNEL: string
|
|
13
|
+
}
|
|
14
|
+
|
|
15
|
+
export namespace Installation {
|
|
16
|
+
const log = Log.create({ service: "installation" })
|
|
17
|
+
|
|
18
|
+
export type Method = Awaited<ReturnType<typeof method>>
|
|
19
|
+
|
|
20
|
+
export const Event = {
|
|
21
|
+
Updated: BusEvent.define(
|
|
22
|
+
"installation.updated",
|
|
23
|
+
z.object({
|
|
24
|
+
version: z.string(),
|
|
25
|
+
}),
|
|
26
|
+
),
|
|
27
|
+
UpdateAvailable: BusEvent.define(
|
|
28
|
+
"installation.update-available",
|
|
29
|
+
z.object({
|
|
30
|
+
version: z.string(),
|
|
31
|
+
}),
|
|
32
|
+
),
|
|
33
|
+
}
|
|
34
|
+
|
|
35
|
+
export const Info = z
|
|
36
|
+
.object({
|
|
37
|
+
version: z.string(),
|
|
38
|
+
latest: z.string(),
|
|
39
|
+
})
|
|
40
|
+
.meta({
|
|
41
|
+
ref: "InstallationInfo",
|
|
42
|
+
})
|
|
43
|
+
export type Info = z.infer<typeof Info>
|
|
44
|
+
|
|
45
|
+
export async function info() {
|
|
46
|
+
return {
|
|
47
|
+
version: VERSION,
|
|
48
|
+
latest: await latest(),
|
|
49
|
+
}
|
|
50
|
+
}
|
|
51
|
+
|
|
52
|
+
export function isPreview() {
|
|
53
|
+
return CHANNEL !== "latest"
|
|
54
|
+
}
|
|
55
|
+
|
|
56
|
+
export function isLocal() {
|
|
57
|
+
return CHANNEL === "local"
|
|
58
|
+
}
|
|
59
|
+
|
|
60
|
+
export async function method() {
|
|
61
|
+
if (process.execPath.includes(path.join(".bincode", "bin"))) return "curl"
|
|
62
|
+
if (process.execPath.includes(path.join(".local", "bin"))) return "curl"
|
|
63
|
+
const exec = process.execPath.toLowerCase()
|
|
64
|
+
|
|
65
|
+
const checks = [
|
|
66
|
+
{
|
|
67
|
+
name: "npm" as const,
|
|
68
|
+
command: () => $`npm list -g --depth=0`.throws(false).quiet().text(),
|
|
69
|
+
},
|
|
70
|
+
{
|
|
71
|
+
name: "yarn" as const,
|
|
72
|
+
command: () => $`yarn global list`.throws(false).quiet().text(),
|
|
73
|
+
},
|
|
74
|
+
{
|
|
75
|
+
name: "pnpm" as const,
|
|
76
|
+
command: () => $`pnpm list -g --depth=0`.throws(false).quiet().text(),
|
|
77
|
+
},
|
|
78
|
+
{
|
|
79
|
+
name: "bun" as const,
|
|
80
|
+
command: () => $`bun pm ls -g`.throws(false).quiet().text(),
|
|
81
|
+
},
|
|
82
|
+
{
|
|
83
|
+
name: "brew" as const,
|
|
84
|
+
command: () => $`brew list --formula bincode`.throws(false).quiet().text(),
|
|
85
|
+
},
|
|
86
|
+
]
|
|
87
|
+
|
|
88
|
+
checks.sort((a, b) => {
|
|
89
|
+
const aMatches = exec.includes(a.name)
|
|
90
|
+
const bMatches = exec.includes(b.name)
|
|
91
|
+
if (aMatches && !bMatches) return -1
|
|
92
|
+
if (!aMatches && bMatches) return 1
|
|
93
|
+
return 0
|
|
94
|
+
})
|
|
95
|
+
|
|
96
|
+
for (const check of checks) {
|
|
97
|
+
const output = await check.command()
|
|
98
|
+
if (output.includes(check.name === "brew" ? "bincode" : "bincode-ai")) {
|
|
99
|
+
return check.name
|
|
100
|
+
}
|
|
101
|
+
}
|
|
102
|
+
|
|
103
|
+
return "unknown"
|
|
104
|
+
}
|
|
105
|
+
|
|
106
|
+
export const UpgradeFailedError = NamedError.create(
|
|
107
|
+
"UpgradeFailedError",
|
|
108
|
+
z.object({
|
|
109
|
+
stderr: z.string(),
|
|
110
|
+
}),
|
|
111
|
+
)
|
|
112
|
+
|
|
113
|
+
async function getBrewFormula() {
|
|
114
|
+
const tapFormula = await $`brew list --formula sst/tap/bincode`.throws(false).quiet().text()
|
|
115
|
+
if (tapFormula.includes("bincode")) return "sst/tap/bincode"
|
|
116
|
+
const coreFormula = await $`brew list --formula bincode`.throws(false).quiet().text()
|
|
117
|
+
if (coreFormula.includes("bincode")) return "bincode"
|
|
118
|
+
return "bincode"
|
|
119
|
+
}
|
|
120
|
+
|
|
121
|
+
export async function upgrade(method: Method, target: string) {
|
|
122
|
+
let cmd
|
|
123
|
+
switch (method) {
|
|
124
|
+
case "curl":
|
|
125
|
+
cmd = $`curl -fsSL https://bincode.ai/install | bash`.env({
|
|
126
|
+
...process.env,
|
|
127
|
+
VERSION: target,
|
|
128
|
+
})
|
|
129
|
+
break
|
|
130
|
+
case "npm":
|
|
131
|
+
cmd = $`npm install -g bincode-ai@${target}`
|
|
132
|
+
break
|
|
133
|
+
case "pnpm":
|
|
134
|
+
cmd = $`pnpm install -g bincode-ai@${target}`
|
|
135
|
+
break
|
|
136
|
+
case "bun":
|
|
137
|
+
cmd = $`bun install -g bincode-ai@${target}`
|
|
138
|
+
break
|
|
139
|
+
case "brew": {
|
|
140
|
+
const formula = await getBrewFormula()
|
|
141
|
+
cmd = $`brew install ${formula}`.env({
|
|
142
|
+
HOMEBREW_NO_AUTO_UPDATE: "1",
|
|
143
|
+
...process.env,
|
|
144
|
+
})
|
|
145
|
+
break
|
|
146
|
+
}
|
|
147
|
+
default:
|
|
148
|
+
throw new Error(`Unknown method: ${method}`)
|
|
149
|
+
}
|
|
150
|
+
const result = await cmd.quiet().throws(false)
|
|
151
|
+
log.info("upgraded", {
|
|
152
|
+
method,
|
|
153
|
+
target,
|
|
154
|
+
stdout: result.stdout.toString(),
|
|
155
|
+
stderr: result.stderr.toString(),
|
|
156
|
+
})
|
|
157
|
+
if (result.exitCode !== 0)
|
|
158
|
+
throw new UpgradeFailedError({
|
|
159
|
+
stderr: result.stderr.toString("utf8"),
|
|
160
|
+
})
|
|
161
|
+
}
|
|
162
|
+
|
|
163
|
+
export const VERSION = typeof BINCODE_VERSION === "string" ? BINCODE_VERSION : "local"
|
|
164
|
+
export const CHANNEL = typeof BINCODE_CHANNEL === "string" ? BINCODE_CHANNEL : "local"
|
|
165
|
+
export const USER_AGENT = `bincode/${CHANNEL}/${VERSION}/${Flag.BINCODE_CLIENT}`
|
|
166
|
+
|
|
167
|
+
export async function latest(installMethod?: Method) {
|
|
168
|
+
const detectedMethod = installMethod || (await method())
|
|
169
|
+
if (detectedMethod === "brew") {
|
|
170
|
+
const formula = await getBrewFormula()
|
|
171
|
+
if (formula === "bincode") {
|
|
172
|
+
return fetch("https://formulae.brew.sh/api/formula/bincode.json")
|
|
173
|
+
.then((res) => {
|
|
174
|
+
if (!res.ok) throw new Error(res.statusText)
|
|
175
|
+
return res.json()
|
|
176
|
+
})
|
|
177
|
+
.then((data: any) => data.versions.stable)
|
|
178
|
+
}
|
|
179
|
+
}
|
|
180
|
+
|
|
181
|
+
const registry = await iife(async () => {
|
|
182
|
+
const r = (await $`npm config get registry`.quiet().nothrow().text()).trim()
|
|
183
|
+
const reg = r || "https://registry.npmjs.org"
|
|
184
|
+
return reg.endsWith("/") ? reg.slice(0, -1) : reg
|
|
185
|
+
})
|
|
186
|
+
const [major] = VERSION.split(".").map((x) => Number(x))
|
|
187
|
+
// const channel = CHANNEL === "latest" ? `latest-${major}` : CHANNEL
|
|
188
|
+
const channel = CHANNEL
|
|
189
|
+
return fetch(`${registry}/bincode-ai/${channel}`)
|
|
190
|
+
.then((res) => {
|
|
191
|
+
if (!res.ok) throw new Error(res.statusText)
|
|
192
|
+
return res.json()
|
|
193
|
+
})
|
|
194
|
+
.then((data: any) => data.version)
|
|
195
|
+
}
|
|
196
|
+
}
|
|
@@ -0,0 +1,229 @@
|
|
|
1
|
+
import { BusEvent } from "@/bus/bus-event"
|
|
2
|
+
import { Bus } from "@/bus"
|
|
3
|
+
import path from "path"
|
|
4
|
+
import { pathToFileURL, fileURLToPath } from "url"
|
|
5
|
+
import { createMessageConnection, StreamMessageReader, StreamMessageWriter } from "vscode-jsonrpc/node"
|
|
6
|
+
import type { Diagnostic as VSCodeDiagnostic } from "vscode-languageserver-types"
|
|
7
|
+
import { Log } from "../util/log"
|
|
8
|
+
import { LANGUAGE_EXTENSIONS } from "./language"
|
|
9
|
+
import z from "zod"
|
|
10
|
+
import type { LSPServer } from "./server"
|
|
11
|
+
import { NamedError } from "@bincode-ai/util/error"
|
|
12
|
+
import { withTimeout } from "../util/timeout"
|
|
13
|
+
import { Instance } from "../project/instance"
|
|
14
|
+
import { Filesystem } from "../util/filesystem"
|
|
15
|
+
|
|
16
|
+
const DIAGNOSTICS_DEBOUNCE_MS = 150
|
|
17
|
+
|
|
18
|
+
export namespace LSPClient {
|
|
19
|
+
const log = Log.create({ service: "lsp.client" })
|
|
20
|
+
|
|
21
|
+
export type Info = NonNullable<Awaited<ReturnType<typeof create>>>
|
|
22
|
+
|
|
23
|
+
export type Diagnostic = VSCodeDiagnostic
|
|
24
|
+
|
|
25
|
+
export const InitializeError = NamedError.create(
|
|
26
|
+
"LSPInitializeError",
|
|
27
|
+
z.object({
|
|
28
|
+
serverID: z.string(),
|
|
29
|
+
}),
|
|
30
|
+
)
|
|
31
|
+
|
|
32
|
+
export const Event = {
|
|
33
|
+
Diagnostics: BusEvent.define(
|
|
34
|
+
"lsp.client.diagnostics",
|
|
35
|
+
z.object({
|
|
36
|
+
serverID: z.string(),
|
|
37
|
+
path: z.string(),
|
|
38
|
+
}),
|
|
39
|
+
),
|
|
40
|
+
}
|
|
41
|
+
|
|
42
|
+
export async function create(input: { serverID: string; server: LSPServer.Handle; root: string }) {
|
|
43
|
+
const l = log.clone().tag("serverID", input.serverID)
|
|
44
|
+
l.info("starting client")
|
|
45
|
+
|
|
46
|
+
const connection = createMessageConnection(
|
|
47
|
+
new StreamMessageReader(input.server.process.stdout as any),
|
|
48
|
+
new StreamMessageWriter(input.server.process.stdin as any),
|
|
49
|
+
)
|
|
50
|
+
|
|
51
|
+
const diagnostics = new Map<string, Diagnostic[]>()
|
|
52
|
+
connection.onNotification("textDocument/publishDiagnostics", (params) => {
|
|
53
|
+
const filePath = Filesystem.normalizePath(fileURLToPath(params.uri))
|
|
54
|
+
l.info("textDocument/publishDiagnostics", {
|
|
55
|
+
path: filePath,
|
|
56
|
+
count: params.diagnostics.length,
|
|
57
|
+
})
|
|
58
|
+
const exists = diagnostics.has(filePath)
|
|
59
|
+
diagnostics.set(filePath, params.diagnostics)
|
|
60
|
+
if (!exists && input.serverID === "typescript") return
|
|
61
|
+
Bus.publish(Event.Diagnostics, { path: filePath, serverID: input.serverID })
|
|
62
|
+
})
|
|
63
|
+
connection.onRequest("window/workDoneProgress/create", (params) => {
|
|
64
|
+
l.info("window/workDoneProgress/create", params)
|
|
65
|
+
return null
|
|
66
|
+
})
|
|
67
|
+
connection.onRequest("workspace/configuration", async () => {
|
|
68
|
+
// Return server initialization options
|
|
69
|
+
return [input.server.initialization ?? {}]
|
|
70
|
+
})
|
|
71
|
+
connection.onRequest("client/registerCapability", async () => {})
|
|
72
|
+
connection.onRequest("client/unregisterCapability", async () => {})
|
|
73
|
+
connection.onRequest("workspace/workspaceFolders", async () => [
|
|
74
|
+
{
|
|
75
|
+
name: "workspace",
|
|
76
|
+
uri: pathToFileURL(input.root).href,
|
|
77
|
+
},
|
|
78
|
+
])
|
|
79
|
+
connection.listen()
|
|
80
|
+
|
|
81
|
+
l.info("sending initialize")
|
|
82
|
+
await withTimeout(
|
|
83
|
+
connection.sendRequest("initialize", {
|
|
84
|
+
rootUri: pathToFileURL(input.root).href,
|
|
85
|
+
processId: input.server.process.pid,
|
|
86
|
+
workspaceFolders: [
|
|
87
|
+
{
|
|
88
|
+
name: "workspace",
|
|
89
|
+
uri: pathToFileURL(input.root).href,
|
|
90
|
+
},
|
|
91
|
+
],
|
|
92
|
+
initializationOptions: {
|
|
93
|
+
...input.server.initialization,
|
|
94
|
+
},
|
|
95
|
+
capabilities: {
|
|
96
|
+
window: {
|
|
97
|
+
workDoneProgress: true,
|
|
98
|
+
},
|
|
99
|
+
workspace: {
|
|
100
|
+
configuration: true,
|
|
101
|
+
},
|
|
102
|
+
textDocument: {
|
|
103
|
+
synchronization: {
|
|
104
|
+
didOpen: true,
|
|
105
|
+
didChange: true,
|
|
106
|
+
},
|
|
107
|
+
publishDiagnostics: {
|
|
108
|
+
versionSupport: true,
|
|
109
|
+
},
|
|
110
|
+
},
|
|
111
|
+
},
|
|
112
|
+
}),
|
|
113
|
+
45_000,
|
|
114
|
+
).catch((err) => {
|
|
115
|
+
l.error("initialize error", { error: err })
|
|
116
|
+
throw new InitializeError(
|
|
117
|
+
{ serverID: input.serverID },
|
|
118
|
+
{
|
|
119
|
+
cause: err,
|
|
120
|
+
},
|
|
121
|
+
)
|
|
122
|
+
})
|
|
123
|
+
|
|
124
|
+
await connection.sendNotification("initialized", {})
|
|
125
|
+
|
|
126
|
+
if (input.server.initialization) {
|
|
127
|
+
await connection.sendNotification("workspace/didChangeConfiguration", {
|
|
128
|
+
settings: input.server.initialization,
|
|
129
|
+
})
|
|
130
|
+
}
|
|
131
|
+
|
|
132
|
+
const files: {
|
|
133
|
+
[path: string]: number
|
|
134
|
+
} = {}
|
|
135
|
+
|
|
136
|
+
const result = {
|
|
137
|
+
root: input.root,
|
|
138
|
+
get serverID() {
|
|
139
|
+
return input.serverID
|
|
140
|
+
},
|
|
141
|
+
get connection() {
|
|
142
|
+
return connection
|
|
143
|
+
},
|
|
144
|
+
notify: {
|
|
145
|
+
async open(input: { path: string }) {
|
|
146
|
+
input.path = path.isAbsolute(input.path) ? input.path : path.resolve(Instance.directory, input.path)
|
|
147
|
+
const file = Bun.file(input.path)
|
|
148
|
+
const text = await file.text()
|
|
149
|
+
const extension = path.extname(input.path)
|
|
150
|
+
const languageId = LANGUAGE_EXTENSIONS[extension] ?? "plaintext"
|
|
151
|
+
|
|
152
|
+
const version = files[input.path]
|
|
153
|
+
if (version !== undefined) {
|
|
154
|
+
const next = version + 1
|
|
155
|
+
files[input.path] = next
|
|
156
|
+
log.info("textDocument/didChange", {
|
|
157
|
+
path: input.path,
|
|
158
|
+
version: next,
|
|
159
|
+
})
|
|
160
|
+
await connection.sendNotification("textDocument/didChange", {
|
|
161
|
+
textDocument: {
|
|
162
|
+
uri: pathToFileURL(input.path).href,
|
|
163
|
+
version: next,
|
|
164
|
+
},
|
|
165
|
+
contentChanges: [{ text }],
|
|
166
|
+
})
|
|
167
|
+
return
|
|
168
|
+
}
|
|
169
|
+
|
|
170
|
+
log.info("textDocument/didOpen", input)
|
|
171
|
+
diagnostics.delete(input.path)
|
|
172
|
+
await connection.sendNotification("textDocument/didOpen", {
|
|
173
|
+
textDocument: {
|
|
174
|
+
uri: pathToFileURL(input.path).href,
|
|
175
|
+
languageId,
|
|
176
|
+
version: 0,
|
|
177
|
+
text,
|
|
178
|
+
},
|
|
179
|
+
})
|
|
180
|
+
files[input.path] = 0
|
|
181
|
+
return
|
|
182
|
+
},
|
|
183
|
+
},
|
|
184
|
+
get diagnostics() {
|
|
185
|
+
return diagnostics
|
|
186
|
+
},
|
|
187
|
+
async waitForDiagnostics(input: { path: string }) {
|
|
188
|
+
const normalizedPath = Filesystem.normalizePath(
|
|
189
|
+
path.isAbsolute(input.path) ? input.path : path.resolve(Instance.directory, input.path),
|
|
190
|
+
)
|
|
191
|
+
log.info("waiting for diagnostics", { path: normalizedPath })
|
|
192
|
+
let unsub: () => void
|
|
193
|
+
let debounceTimer: ReturnType<typeof setTimeout> | undefined
|
|
194
|
+
return await withTimeout(
|
|
195
|
+
new Promise<void>((resolve) => {
|
|
196
|
+
unsub = Bus.subscribe(Event.Diagnostics, (event) => {
|
|
197
|
+
if (event.properties.path === normalizedPath && event.properties.serverID === result.serverID) {
|
|
198
|
+
// Debounce to allow LSP to send follow-up diagnostics (e.g., semantic after syntax)
|
|
199
|
+
if (debounceTimer) clearTimeout(debounceTimer)
|
|
200
|
+
debounceTimer = setTimeout(() => {
|
|
201
|
+
log.info("got diagnostics", { path: normalizedPath })
|
|
202
|
+
unsub?.()
|
|
203
|
+
resolve()
|
|
204
|
+
}, DIAGNOSTICS_DEBOUNCE_MS)
|
|
205
|
+
}
|
|
206
|
+
})
|
|
207
|
+
}),
|
|
208
|
+
3000,
|
|
209
|
+
)
|
|
210
|
+
.catch(() => {})
|
|
211
|
+
.finally(() => {
|
|
212
|
+
if (debounceTimer) clearTimeout(debounceTimer)
|
|
213
|
+
unsub?.()
|
|
214
|
+
})
|
|
215
|
+
},
|
|
216
|
+
async shutdown() {
|
|
217
|
+
l.info("shutting down")
|
|
218
|
+
connection.end()
|
|
219
|
+
connection.dispose()
|
|
220
|
+
input.server.process.kill()
|
|
221
|
+
l.info("shutdown")
|
|
222
|
+
},
|
|
223
|
+
}
|
|
224
|
+
|
|
225
|
+
l.info("initialized")
|
|
226
|
+
|
|
227
|
+
return result
|
|
228
|
+
}
|
|
229
|
+
}
|