@pugi/cli 0.1.0-beta.5 → 0.1.0-beta.51
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/THIRD_PARTY_NOTICES.md +40 -0
- package/assets/pugi-mascot.ansi +15 -25
- package/assets/pugi-prozr2-mascot.ansi +9 -0
- package/bin/run.js +33 -1
- package/dist/commands/jobs-watch.js +201 -0
- package/dist/commands/jobs.js +15 -0
- package/dist/commands/smoke.js +133 -0
- package/dist/core/agent-progress/cleanup.js +134 -0
- package/dist/core/agent-progress/schema.js +144 -0
- package/dist/core/agent-progress/writer.js +101 -0
- package/dist/core/artifact-chain/dispatcher.js +148 -0
- package/dist/core/artifact-chain/exporter.js +164 -0
- package/dist/core/artifact-chain/state.js +243 -0
- package/dist/core/artifact-chain/steps.js +169 -0
- package/dist/core/auth/ensure-authenticated.js +129 -0
- package/dist/core/auth/env-provider.js +238 -0
- package/dist/core/auto-update/channels.js +122 -0
- package/dist/core/auto-update/checker.js +241 -0
- package/dist/core/auto-update/state.js +235 -0
- package/dist/core/bare-mode/index.js +107 -0
- package/dist/core/bash-classifier.js +400 -4
- package/dist/core/checkpoint/resumer.js +149 -0
- package/dist/core/checkpoint/rewinder.js +291 -0
- package/dist/core/codegraph/decision-store.js +248 -0
- package/dist/core/codegraph/detect-repo.js +459 -0
- package/dist/core/codegraph/install.js +134 -0
- package/dist/core/codegraph/offer-hook.js +220 -0
- package/dist/core/compact/auto-trigger.js +96 -0
- package/dist/core/compact/buffer-rewriter.js +115 -0
- package/dist/core/compact/summarizer.js +208 -0
- package/dist/core/compact/token-counter.js +108 -0
- package/dist/core/consensus/diff-capture.js +112 -3
- package/dist/core/context/index.js +7 -0
- package/dist/core/context/markdown-traverse.js +255 -0
- package/dist/core/cost/rate-card.js +129 -0
- package/dist/core/cost/tracker.js +221 -0
- package/dist/core/denial-tracking/index.js +8 -0
- package/dist/core/denial-tracking/state.js +264 -0
- package/dist/core/diagnostics/probe-runner.js +93 -0
- package/dist/core/diagnostics/probes/api.js +46 -0
- package/dist/core/diagnostics/probes/auth.js +86 -0
- package/dist/core/diagnostics/probes/bare-mode.js +42 -0
- package/dist/core/diagnostics/probes/cli-version.js +127 -0
- package/dist/core/diagnostics/probes/config.js +72 -0
- package/dist/core/diagnostics/probes/denial-tracking.js +57 -0
- package/dist/core/diagnostics/probes/disk.js +81 -0
- package/dist/core/diagnostics/probes/git.js +65 -0
- package/dist/core/diagnostics/probes/hooks.js +118 -0
- package/dist/core/diagnostics/probes/mcp.js +75 -0
- package/dist/core/diagnostics/probes/node.js +59 -0
- package/dist/core/diagnostics/probes/pnpm.js +36 -0
- package/dist/core/diagnostics/probes/pugi-md.js +89 -0
- package/dist/core/diagnostics/probes/sandbox.js +40 -0
- package/dist/core/diagnostics/probes/session.js +74 -0
- package/dist/core/diagnostics/probes/status-snapshot.js +488 -0
- package/dist/core/diagnostics/probes/workspace.js +63 -0
- package/dist/core/diagnostics/types.js +70 -0
- package/dist/core/dispatch/cache-cleanup.js +197 -0
- package/dist/core/dispatch/cache-handoff.js +295 -0
- package/dist/core/edits/dispatch.js +218 -2
- package/dist/core/edits/journal.js +199 -0
- package/dist/core/edits/layer-d-ast.js +557 -14
- package/dist/core/edits/verify-hook.js +273 -0
- package/dist/core/edits/worktree.js +322 -0
- package/dist/core/engine/anvil-client.js +115 -5
- package/dist/core/engine/auto-compact.js +179 -0
- package/dist/core/engine/budgets.js +155 -0
- package/dist/core/engine/context-prefix.js +155 -0
- package/dist/core/engine/intent.js +260 -0
- package/dist/core/engine/native-pugi.js +897 -211
- package/dist/core/engine/prompts.js +88 -2
- package/dist/core/engine/strip-internal-fields.js +124 -0
- package/dist/core/engine/tool-bridge.js +1045 -36
- package/dist/core/feedback/queue.js +177 -0
- package/dist/core/feedback/submitter.js +145 -0
- package/dist/core/file-cache.js +113 -1
- package/dist/core/hooks/events.js +44 -0
- package/dist/core/hooks/index.js +15 -0
- package/dist/core/hooks/registry.js +213 -0
- package/dist/core/hooks/runner.js +236 -0
- package/dist/core/hooks/v2/event-emitter.js +115 -0
- package/dist/core/hooks/v2/executor.js +282 -0
- package/dist/core/hooks/v2/index.js +25 -0
- package/dist/core/hooks/v2/lifecycle.js +104 -0
- package/dist/core/hooks/v2/loader.js +216 -0
- package/dist/core/hooks/v2/matcher.js +125 -0
- package/dist/core/hooks/v2/trust.js +143 -0
- package/dist/core/hooks/v2/types.js +86 -0
- package/dist/core/lsp/cache.js +105 -0
- package/dist/core/lsp/client.js +776 -0
- package/dist/core/lsp/language-detect.js +66 -0
- package/dist/core/lsp/post-edit-diagnostics.js +171 -0
- package/dist/core/mcp/client.js +75 -6
- package/dist/core/mcp/http-server.js +553 -0
- package/dist/core/mcp/orchestrator-tools.js +662 -0
- package/dist/core/mcp/permission.js +190 -0
- package/dist/core/mcp/registry.js +24 -2
- package/dist/core/mcp/server-tools.js +219 -0
- package/dist/core/mcp/server.js +397 -0
- package/dist/core/memory/dual-write.js +416 -0
- package/dist/core/memory/phase1-kinds.js +20 -0
- package/dist/core/memory-sync/queue.js +158 -0
- package/dist/core/onboarding/ensure-initialized.js +133 -0
- package/dist/core/onboarding/marker.js +111 -0
- package/dist/core/onboarding/telemetry-state.js +108 -0
- package/dist/core/output-style/presets.js +176 -0
- package/dist/core/output-style/state.js +185 -0
- package/dist/core/path-security.js +284 -2
- package/dist/core/permissions/auto-classifier.js +124 -0
- package/dist/core/permissions/circuit-breaker.js +83 -0
- package/dist/core/permissions/gate.js +278 -0
- package/dist/core/permissions/index.js +20 -0
- package/dist/core/permissions/mode.js +174 -0
- package/dist/core/permissions/state.js +241 -0
- package/dist/core/permissions/tool-class.js +93 -0
- package/dist/core/prd-check/parser.js +215 -0
- package/dist/core/prd-check/reporter.js +127 -0
- package/dist/core/prd-check/session-review.js +557 -0
- package/dist/core/prd-check/verifiers.js +223 -0
- package/dist/core/pugi-md/context-injector.js +76 -0
- package/dist/core/pugi-md/walk-up.js +207 -0
- package/dist/core/release-notes/parser.js +241 -0
- package/dist/core/release-notes/state.js +116 -0
- package/dist/core/repl/history.js +11 -1
- package/dist/core/repl/model-pricing.js +135 -0
- package/dist/core/repl/session.js +1897 -37
- package/dist/core/repl/slash-commands.js +430 -15
- package/dist/core/repl/store/session-store.js +31 -2
- package/dist/core/repl/workspace-context.js +22 -0
- package/dist/core/repo-map/build.js +125 -0
- package/dist/core/repo-map/cache.js +185 -0
- package/dist/core/repo-map/extractor.js +254 -0
- package/dist/core/repo-map/formatter.js +145 -0
- package/dist/core/repo-map/scanner.js +211 -0
- package/dist/core/retry-budget/budget.js +284 -0
- package/dist/core/retry-budget/index.js +5 -0
- package/dist/core/session.js +92 -0
- package/dist/core/settings.js +80 -0
- package/dist/core/share/formatter.js +271 -0
- package/dist/core/share/redactor.js +221 -0
- package/dist/core/share/uploader.js +267 -0
- package/dist/core/skills/defaults.js +457 -0
- package/dist/core/smoke/headless-driver.js +174 -0
- package/dist/core/smoke/orchestrator.js +194 -0
- package/dist/core/smoke/runner.js +238 -0
- package/dist/core/smoke/scenario-parser.js +316 -0
- package/dist/core/subagents/dispatcher-real.js +600 -0
- package/dist/core/subagents/dispatcher.js +113 -24
- package/dist/core/subagents/index.js +18 -5
- package/dist/core/subagents/isolation-matrix.js +213 -0
- package/dist/core/subagents/spawn.js +19 -4
- package/dist/core/telemetry/emitter.js +229 -0
- package/dist/core/telemetry/queue.js +251 -0
- package/dist/core/theme/context.js +91 -0
- package/dist/core/theme/presets.js +228 -0
- package/dist/core/theme/state.js +181 -0
- package/dist/core/todos/invariant.js +10 -0
- package/dist/core/todos/state.js +177 -0
- package/dist/core/transport/version-interceptor.js +166 -0
- package/dist/core/vim/keymap.js +288 -0
- package/dist/core/vim/state.js +92 -0
- package/dist/core/worktree-manager/cleanup.js +123 -0
- package/dist/core/worktree-manager/manager.js +303 -0
- package/dist/index.js +28 -0
- package/dist/runtime/bootstrap.js +190 -0
- package/dist/runtime/cli.js +3241 -343
- package/dist/runtime/commands/cancel.js +231 -0
- package/dist/runtime/commands/chain.js +489 -0
- package/dist/runtime/commands/codegraph-status.js +227 -0
- package/dist/runtime/commands/compact.js +297 -0
- package/dist/runtime/commands/cost.js +199 -0
- package/dist/runtime/commands/delegate.js +242 -11
- package/dist/runtime/commands/dispatch.js +126 -0
- package/dist/runtime/commands/doctor.js +412 -0
- package/dist/runtime/commands/feedback.js +184 -0
- package/dist/runtime/commands/hooks.js +184 -0
- package/dist/runtime/commands/lsp.js +368 -0
- package/dist/runtime/commands/mcp.js +879 -0
- package/dist/runtime/commands/memory.js +508 -0
- package/dist/runtime/commands/model.js +237 -0
- package/dist/runtime/commands/onboarding.js +275 -0
- package/dist/runtime/commands/patch.js +128 -0
- package/dist/runtime/commands/permissions.js +112 -0
- package/dist/runtime/commands/plan.js +143 -0
- package/dist/runtime/commands/prd-check.js +285 -0
- package/dist/runtime/commands/redo-blob-store.js +92 -0
- package/dist/runtime/commands/redo.js +361 -0
- package/dist/runtime/commands/release-notes.js +229 -0
- package/dist/runtime/commands/repo-map.js +95 -0
- package/dist/runtime/commands/report.js +299 -0
- package/dist/runtime/commands/resume.js +118 -0
- package/dist/runtime/commands/review-consensus.js +17 -2
- package/dist/runtime/commands/rewind.js +333 -0
- package/dist/runtime/commands/sessions.js +163 -0
- package/dist/runtime/commands/share.js +316 -0
- package/dist/runtime/commands/status.js +186 -0
- package/dist/runtime/commands/stickers.js +82 -0
- package/dist/runtime/commands/style.js +194 -0
- package/dist/runtime/commands/theme.js +196 -0
- package/dist/runtime/commands/undo.js +32 -0
- package/dist/runtime/commands/update.js +289 -0
- package/dist/runtime/commands/vim.js +140 -0
- package/dist/runtime/commands/worktree.js +177 -0
- package/dist/runtime/commands/worktrees.js +155 -0
- package/dist/runtime/headless-repl.js +195 -0
- package/dist/runtime/headless.js +543 -0
- package/dist/runtime/load-hooks-or-exit.js +71 -0
- package/dist/runtime/plan-decompose.js +531 -0
- package/dist/runtime/version.js +65 -0
- package/dist/tools/agent-tool.js +229 -0
- package/dist/tools/apply-patch.js +556 -0
- package/dist/tools/ask-user-question.js +213 -0
- package/dist/tools/ask-user.js +115 -0
- package/dist/tools/bash.js +203 -4
- package/dist/tools/file-tools.js +85 -14
- package/dist/tools/lsp-tools.js +189 -0
- package/dist/tools/mcp-tool.js +260 -0
- package/dist/tools/multi-edit.js +361 -0
- package/dist/tools/powershell.js +268 -0
- package/dist/tools/registry.js +51 -0
- package/dist/tools/skill-tool.js +96 -0
- package/dist/tools/tasks.js +208 -0
- package/dist/tools/todo-write.js +184 -0
- package/dist/tools/web-fetch.js +147 -2
- package/dist/tools/web-search.js +458 -0
- package/dist/tui/agent-progress-card.js +111 -0
- package/dist/tui/agent-tree.js +10 -0
- package/dist/tui/ask-modal.js +2 -2
- package/dist/tui/ask-user-question-prompt.js +192 -0
- package/dist/tui/compact-banner.js +81 -0
- package/dist/tui/conversation-pane.js +82 -8
- package/dist/tui/cost-table.js +111 -0
- package/dist/tui/doctor-table.js +46 -0
- package/dist/tui/feedback-prompt.js +156 -0
- package/dist/tui/input-box.js +218 -3
- package/dist/tui/markdown-render.js +4 -4
- package/dist/tui/onboarding-wizard.js +240 -0
- package/dist/tui/permissions-picker.js +86 -0
- package/dist/tui/render.js +35 -0
- package/dist/tui/repl-render.js +313 -35
- package/dist/tui/repl-splash-art.js +1 -1
- package/dist/tui/repl-splash-mascot.js +32 -8
- package/dist/tui/repl-splash.js +2 -2
- package/dist/tui/repl.js +85 -5
- package/dist/tui/splash.js +1 -1
- package/dist/tui/status-bar.js +94 -16
- package/dist/tui/status-table.js +7 -0
- package/dist/tui/stickers-art.js +136 -0
- package/dist/tui/style-table.js +28 -0
- package/dist/tui/theme-table.js +29 -0
- package/dist/tui/thinking-spinner.js +123 -0
- package/dist/tui/tool-stream-pane.js +52 -3
- package/dist/tui/update-banner.js +27 -2
- package/dist/tui/vim-input.js +267 -0
- package/dist/tui/welcome-banner.js +107 -0
- package/dist/tui/welcome-data.js +293 -0
- package/docs/examples/codegraph.mcp.json +10 -0
- package/package.json +13 -7
- package/test/scenarios/codegen-create-file.scenario.txt +13 -0
- package/test/scenarios/compact-force.scenario.txt +11 -0
- package/test/scenarios/identity.scenario.txt +11 -0
- package/test/scenarios/persona-handoff.scenario.txt +11 -0
- package/test/scenarios/walkback.scenario.txt +12 -0
- package/dist/core/engine/compaction-hook.js +0 -154
|
@@ -0,0 +1,776 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* LSP client wrapper — α7.7 Phase 1 (LSP tool + worktree + apply_patch).
|
|
3
|
+
*
|
|
4
|
+
* Wraps a Language Server Protocol server process (started locally via
|
|
5
|
+
* stdio) behind a small async surface used by the LSP tools:
|
|
6
|
+
*
|
|
7
|
+
* - hover(file, line, col)
|
|
8
|
+
* - definition(file, line, col)
|
|
9
|
+
* - references(file, line, col)
|
|
10
|
+
* - diagnostics(file)
|
|
11
|
+
*
|
|
12
|
+
* Why no `vscode-languageserver-protocol` dep:
|
|
13
|
+
*
|
|
14
|
+
* The α6.6+ posture for `apps/pugi-cli` keeps the dep tree intentionally
|
|
15
|
+
* lean (zod, ink, react, undici, tar — that's most of it). Pulling
|
|
16
|
+
* `vscode-languageserver-protocol` + transitive `vscode-jsonrpc` for
|
|
17
|
+
* four operations would expand the install footprint disproportionately.
|
|
18
|
+
* The LSP framing protocol (Content-Length headers + JSON-RPC bodies)
|
|
19
|
+
* is documented in the LSP spec §3 and stable since 2017; we implement
|
|
20
|
+
* the framer ourselves in ~80 LOC. Pulling in the official packages
|
|
21
|
+
* later (when we add 20+ operations) stays an open option — every
|
|
22
|
+
* public surface here mirrors the official type names so a future swap
|
|
23
|
+
* is mechanical.
|
|
24
|
+
*
|
|
25
|
+
* Why we spawn `npx <server>` for stock languages:
|
|
26
|
+
*
|
|
27
|
+
* - typescript-language-server: `npx typescript-language-server --stdio`
|
|
28
|
+
* - pyright: `pyright-langserver --stdio` (operator-installed)
|
|
29
|
+
* - gopls: `gopls` (operator-installed via `go install`)
|
|
30
|
+
* - rust-analyzer: `rust-analyzer` (operator-installed via `rustup`)
|
|
31
|
+
*
|
|
32
|
+
* `npx typescript-language-server` resolves the binary from the
|
|
33
|
+
* workspace's `node_modules/.bin` first, then falls back to the global
|
|
34
|
+
* npm cache. The other servers are operator-installed system binaries
|
|
35
|
+
* (we run `which <name>` once at start and fail loud with
|
|
36
|
+
* `lsp_unavailable` if absent — see `detectServer` below). This keeps
|
|
37
|
+
* `pugi` install-time zero-deps for non-TS workspaces.
|
|
38
|
+
*
|
|
39
|
+
* Lifecycle contract:
|
|
40
|
+
*
|
|
41
|
+
* 1. `startLspClient(lang, opts)` spawns the server, performs the LSP
|
|
42
|
+
* `initialize` handshake, sends `initialized`, and returns a client.
|
|
43
|
+
* 2. Each operation auto-opens the target file via `textDocument/didOpen`
|
|
44
|
+
* on first touch (we keep a `Set<string>` of opened URIs). Files are
|
|
45
|
+
* never closed mid-session — the server is per-CLI-invocation and
|
|
46
|
+
* the process exit cleans up.
|
|
47
|
+
* 3. `stop()` sends `shutdown` + `exit` (best-effort) and SIGKILLs the
|
|
48
|
+
* child after a 1s grace window so the CLI never hangs on a
|
|
49
|
+
* misbehaving server.
|
|
50
|
+
*
|
|
51
|
+
* Cancellation: every async operation accepts an optional CancellationToken
|
|
52
|
+
* (α6.9). On abort, we send LSP `$/cancelRequest` for the in-flight ID and
|
|
53
|
+
* reject the promise with `OperatorAbortedError`.
|
|
54
|
+
*
|
|
55
|
+
* Privacy: requests stay client-side. There is no Anvil round-trip; the
|
|
56
|
+
* server process reads ONLY the files the operator's code actually opens.
|
|
57
|
+
* The `pugi privacy airgapped` mode does not need to gate this surface
|
|
58
|
+
* because the LSP server is local; we still honor the mode by skipping
|
|
59
|
+
* the optional update-check banner inside the CLI command surface.
|
|
60
|
+
*
|
|
61
|
+
* Brand voice: ASCII only, no emoji, no banned words.
|
|
62
|
+
*/
|
|
63
|
+
import { spawn, spawnSync } from 'node:child_process';
|
|
64
|
+
import { pathToFileURL } from 'node:url';
|
|
65
|
+
import { existsSync, readFileSync, realpathSync } from 'node:fs';
|
|
66
|
+
import { resolve, sep } from 'node:path';
|
|
67
|
+
import { OperatorAbortedError } from '../../tools/file-tools.js';
|
|
68
|
+
const LANGUAGE_SERVERS = {
|
|
69
|
+
ts: {
|
|
70
|
+
command: 'npx',
|
|
71
|
+
args: ['--yes', 'typescript-language-server', '--stdio'],
|
|
72
|
+
languageId: 'typescript',
|
|
73
|
+
probe: 'npx',
|
|
74
|
+
},
|
|
75
|
+
js: {
|
|
76
|
+
command: 'npx',
|
|
77
|
+
args: ['--yes', 'typescript-language-server', '--stdio'],
|
|
78
|
+
languageId: 'javascript',
|
|
79
|
+
probe: 'npx',
|
|
80
|
+
},
|
|
81
|
+
py: {
|
|
82
|
+
command: 'pyright-langserver',
|
|
83
|
+
args: ['--stdio'],
|
|
84
|
+
languageId: 'python',
|
|
85
|
+
probe: 'pyright-langserver',
|
|
86
|
+
},
|
|
87
|
+
go: {
|
|
88
|
+
command: 'gopls',
|
|
89
|
+
args: [],
|
|
90
|
+
languageId: 'go',
|
|
91
|
+
probe: 'gopls',
|
|
92
|
+
},
|
|
93
|
+
rust: {
|
|
94
|
+
command: 'rust-analyzer',
|
|
95
|
+
args: [],
|
|
96
|
+
languageId: 'rust',
|
|
97
|
+
probe: 'rust-analyzer',
|
|
98
|
+
},
|
|
99
|
+
};
|
|
100
|
+
const DEFAULT_REQUEST_TIMEOUT_MS = 5_000;
|
|
101
|
+
/**
|
|
102
|
+
* Returned by `startLspClient`. Methods are async; every method honors
|
|
103
|
+
* the optional `CancellationToken` parameter.
|
|
104
|
+
*/
|
|
105
|
+
export class LspClient {
|
|
106
|
+
cwd;
|
|
107
|
+
child;
|
|
108
|
+
server;
|
|
109
|
+
openedFiles = new Set();
|
|
110
|
+
pending = new Map();
|
|
111
|
+
diagnosticsByUri = new Map();
|
|
112
|
+
requestTimeoutMs;
|
|
113
|
+
nextId = 1;
|
|
114
|
+
buffer = Buffer.alloc(0);
|
|
115
|
+
stopped = false;
|
|
116
|
+
constructor(child, server, opts) {
|
|
117
|
+
this.child = child;
|
|
118
|
+
this.server = server;
|
|
119
|
+
this.cwd = opts.cwd;
|
|
120
|
+
this.requestTimeoutMs = opts.requestTimeoutMs ?? DEFAULT_REQUEST_TIMEOUT_MS;
|
|
121
|
+
child.stdout.on('data', (chunk) => this.onStdout(chunk));
|
|
122
|
+
// Stderr is the server's diagnostic channel; we swallow it so a
|
|
123
|
+
// chatty server (e.g. tsserver `debug:` lines) does not leak into
|
|
124
|
+
// the operator-facing CLI stream. Operators can re-run with
|
|
125
|
+
// `PUGI_LSP_DEBUG=1` to surface stderr.
|
|
126
|
+
if (process.env.PUGI_LSP_DEBUG === '1') {
|
|
127
|
+
child.stderr.on('data', (chunk) => process.stderr.write(chunk));
|
|
128
|
+
}
|
|
129
|
+
else {
|
|
130
|
+
child.stderr.on('data', () => { });
|
|
131
|
+
}
|
|
132
|
+
child.on('exit', () => this.onExit());
|
|
133
|
+
// R1 fix (2026-05-26, PR #413 r1, P2 #11): mirror onExit for the
|
|
134
|
+
// 'error' event. A late-fired spawn error (EIO, ENOMEM, etc.) or
|
|
135
|
+
// any unhandled child-process error would otherwise leave
|
|
136
|
+
// in-flight pending requests dangling until their per-request
|
|
137
|
+
// timer fired, which can be up to `requestTimeoutMs` later.
|
|
138
|
+
// Failing fast here matches the exit-time semantics.
|
|
139
|
+
child.on('error', () => this.onExit());
|
|
140
|
+
}
|
|
141
|
+
/**
|
|
142
|
+
* Send `shutdown` + `exit`, then SIGKILL after a 1s grace window so
|
|
143
|
+
* a hung server never blocks CLI termination. Idempotent.
|
|
144
|
+
*/
|
|
145
|
+
async stop() {
|
|
146
|
+
if (this.stopped)
|
|
147
|
+
return;
|
|
148
|
+
this.stopped = true;
|
|
149
|
+
try {
|
|
150
|
+
await this.sendRequest('shutdown', null, undefined);
|
|
151
|
+
this.sendNotification('exit', null);
|
|
152
|
+
}
|
|
153
|
+
catch {
|
|
154
|
+
// Best-effort. A server that ignored shutdown gets the SIGKILL
|
|
155
|
+
// path below.
|
|
156
|
+
}
|
|
157
|
+
const killTimer = setTimeout(() => {
|
|
158
|
+
try {
|
|
159
|
+
this.child.kill('SIGKILL');
|
|
160
|
+
}
|
|
161
|
+
catch {
|
|
162
|
+
// already exited
|
|
163
|
+
}
|
|
164
|
+
}, 1_000);
|
|
165
|
+
killTimer.unref();
|
|
166
|
+
// Reject any in-flight requests so callers do not hang on the
|
|
167
|
+
// shutdown path. We drain the map first to avoid the reject
|
|
168
|
+
// callback re-entering and mutating the map mid-iteration.
|
|
169
|
+
const snapshot = Array.from(this.pending.entries());
|
|
170
|
+
this.pending.clear();
|
|
171
|
+
for (const [, entry] of snapshot) {
|
|
172
|
+
clearTimeout(entry.timer);
|
|
173
|
+
entry.reject(new Error('lsp_stopped'));
|
|
174
|
+
}
|
|
175
|
+
}
|
|
176
|
+
async hover(file, pos, token) {
|
|
177
|
+
return this.withDocument(file, async (uri) => {
|
|
178
|
+
const raw = await this.sendRequest('textDocument/hover', {
|
|
179
|
+
textDocument: { uri },
|
|
180
|
+
position: pos,
|
|
181
|
+
}, token);
|
|
182
|
+
if (raw === null || raw === undefined)
|
|
183
|
+
return { ok: true, value: null };
|
|
184
|
+
const value = normalizeHover(raw);
|
|
185
|
+
return { ok: true, value };
|
|
186
|
+
});
|
|
187
|
+
}
|
|
188
|
+
async definition(file, pos, token) {
|
|
189
|
+
return this.withDocument(file, async (uri) => {
|
|
190
|
+
const raw = await this.sendRequest('textDocument/definition', {
|
|
191
|
+
textDocument: { uri },
|
|
192
|
+
position: pos,
|
|
193
|
+
}, token);
|
|
194
|
+
const locations = normalizeLocations(raw, this.cwd);
|
|
195
|
+
return { ok: true, value: locations };
|
|
196
|
+
});
|
|
197
|
+
}
|
|
198
|
+
async references(file, pos, token) {
|
|
199
|
+
return this.withDocument(file, async (uri) => {
|
|
200
|
+
const raw = await this.sendRequest('textDocument/references', {
|
|
201
|
+
textDocument: { uri },
|
|
202
|
+
position: pos,
|
|
203
|
+
context: { includeDeclaration: true },
|
|
204
|
+
}, token);
|
|
205
|
+
const locations = normalizeLocations(raw, this.cwd);
|
|
206
|
+
return { ok: true, value: locations };
|
|
207
|
+
});
|
|
208
|
+
}
|
|
209
|
+
/**
|
|
210
|
+
* Diagnostics in LSP arrive as PUSH (`textDocument/publishDiagnostics`)
|
|
211
|
+
* not pull. We open the document, wait one short tick for the server
|
|
212
|
+
* to drain its first analysis pass, and return what is cached.
|
|
213
|
+
* Servers that emit diagnostics asynchronously (gopls, rust-analyzer)
|
|
214
|
+
* may take longer; the caller can extend `requestTimeoutMs` to allow
|
|
215
|
+
* for the cold start.
|
|
216
|
+
*/
|
|
217
|
+
async diagnostics(file, token) {
|
|
218
|
+
return this.withDocument(file, async (uri) => {
|
|
219
|
+
// Give the server one analysis cycle before reading the cache.
|
|
220
|
+
// 200ms is enough for a warm tsserver; a cold server returns the
|
|
221
|
+
// empty array and the caller can re-poll.
|
|
222
|
+
await new Promise((res) => {
|
|
223
|
+
const wait = setTimeout(res, 200);
|
|
224
|
+
wait.unref();
|
|
225
|
+
if (token) {
|
|
226
|
+
token.onAbort(() => {
|
|
227
|
+
clearTimeout(wait);
|
|
228
|
+
res();
|
|
229
|
+
});
|
|
230
|
+
}
|
|
231
|
+
});
|
|
232
|
+
if (token && token.isAborted) {
|
|
233
|
+
return { ok: false, reason: 'operator_aborted', detail: 'lspDiagnostics aborted' };
|
|
234
|
+
}
|
|
235
|
+
const cached = this.diagnosticsByUri.get(uri) ?? [];
|
|
236
|
+
return { ok: true, value: cached };
|
|
237
|
+
});
|
|
238
|
+
}
|
|
239
|
+
/**
|
|
240
|
+
* Ensure the file is open server-side, then run `op`. The first call
|
|
241
|
+
* for a path sends `textDocument/didOpen` with the on-disk content;
|
|
242
|
+
* subsequent calls skip the open.
|
|
243
|
+
*/
|
|
244
|
+
async withDocument(file, op) {
|
|
245
|
+
let absPath;
|
|
246
|
+
try {
|
|
247
|
+
absPath = resolve(this.cwd, file);
|
|
248
|
+
if (!absPath.startsWith(this.cwd + sep) && absPath !== this.cwd) {
|
|
249
|
+
return {
|
|
250
|
+
ok: false,
|
|
251
|
+
reason: 'lsp_error',
|
|
252
|
+
detail: `path escapes workspace: ${file}`,
|
|
253
|
+
};
|
|
254
|
+
}
|
|
255
|
+
}
|
|
256
|
+
catch (error) {
|
|
257
|
+
return {
|
|
258
|
+
ok: false,
|
|
259
|
+
reason: 'lsp_error',
|
|
260
|
+
detail: error instanceof Error ? error.message : String(error),
|
|
261
|
+
};
|
|
262
|
+
}
|
|
263
|
+
// R1 fix (2026-05-26, PR #413 r1, Fix 8): realpath containment.
|
|
264
|
+
// Without this gate, a workspace-local symlink (e.g. `alias` ->
|
|
265
|
+
// `/etc/passwd`) passed the lexical `absPath.startsWith(cwd)`
|
|
266
|
+
// check, then `readFileSync(absPath, 'utf8')` happily followed the
|
|
267
|
+
// symlink and shipped `/etc/passwd` into the LSP `textDocument/didOpen`
|
|
268
|
+
// payload. Parity with `applySecurityGate`'s symlink-escape rule:
|
|
269
|
+
// when the file exists, the realpath MUST stay inside the workspace
|
|
270
|
+
// realpath. Missing files (newly-typed paths the operator is
|
|
271
|
+
// querying) skip the check — there's no symlink target to escape.
|
|
272
|
+
if (existsSync(absPath)) {
|
|
273
|
+
try {
|
|
274
|
+
const realRoot = realpathSync.native(this.cwd);
|
|
275
|
+
const realTarget = realpathSync.native(absPath);
|
|
276
|
+
if (realTarget !== realRoot && !realTarget.startsWith(realRoot + sep)) {
|
|
277
|
+
return {
|
|
278
|
+
ok: false,
|
|
279
|
+
reason: 'lsp_error',
|
|
280
|
+
detail: `symlink escapes workspace: ${file} -> ${realTarget}`,
|
|
281
|
+
};
|
|
282
|
+
}
|
|
283
|
+
}
|
|
284
|
+
catch (error) {
|
|
285
|
+
return {
|
|
286
|
+
ok: false,
|
|
287
|
+
reason: 'lsp_error',
|
|
288
|
+
detail: `cannot realpath ${file}: ${error instanceof Error ? error.message : String(error)}`,
|
|
289
|
+
};
|
|
290
|
+
}
|
|
291
|
+
}
|
|
292
|
+
const uri = pathToFileURL(absPath).toString();
|
|
293
|
+
if (!this.openedFiles.has(uri)) {
|
|
294
|
+
try {
|
|
295
|
+
const text = readFileSync(absPath, 'utf8');
|
|
296
|
+
this.sendNotification('textDocument/didOpen', {
|
|
297
|
+
textDocument: {
|
|
298
|
+
uri,
|
|
299
|
+
languageId: this.server.languageId,
|
|
300
|
+
version: 1,
|
|
301
|
+
text,
|
|
302
|
+
},
|
|
303
|
+
});
|
|
304
|
+
this.openedFiles.add(uri);
|
|
305
|
+
}
|
|
306
|
+
catch (error) {
|
|
307
|
+
return {
|
|
308
|
+
ok: false,
|
|
309
|
+
reason: 'lsp_error',
|
|
310
|
+
detail: `cannot read ${file}: ${error instanceof Error ? error.message : String(error)}`,
|
|
311
|
+
};
|
|
312
|
+
}
|
|
313
|
+
}
|
|
314
|
+
try {
|
|
315
|
+
return await op(uri);
|
|
316
|
+
}
|
|
317
|
+
catch (error) {
|
|
318
|
+
if (error instanceof OperatorAbortedError) {
|
|
319
|
+
return { ok: false, reason: 'operator_aborted', detail: error.message };
|
|
320
|
+
}
|
|
321
|
+
if (error instanceof Error && error.message === 'request_timeout') {
|
|
322
|
+
return { ok: false, reason: 'request_timeout', detail: 'lsp request timed out' };
|
|
323
|
+
}
|
|
324
|
+
return {
|
|
325
|
+
ok: false,
|
|
326
|
+
reason: 'lsp_error',
|
|
327
|
+
detail: error instanceof Error ? error.message : String(error),
|
|
328
|
+
};
|
|
329
|
+
}
|
|
330
|
+
}
|
|
331
|
+
sendRequest(method, params, token) {
|
|
332
|
+
const id = this.nextId++;
|
|
333
|
+
const payload = { jsonrpc: '2.0', id, method, params };
|
|
334
|
+
return new Promise((resolveFn, rejectFn) => {
|
|
335
|
+
const timer = setTimeout(() => {
|
|
336
|
+
if (this.pending.has(id)) {
|
|
337
|
+
this.pending.delete(id);
|
|
338
|
+
rejectFn(new Error('request_timeout'));
|
|
339
|
+
}
|
|
340
|
+
}, this.requestTimeoutMs);
|
|
341
|
+
timer.unref();
|
|
342
|
+
this.pending.set(id, { resolve: resolveFn, reject: rejectFn, timer });
|
|
343
|
+
if (token) {
|
|
344
|
+
token.onAbort(() => {
|
|
345
|
+
if (this.pending.has(id)) {
|
|
346
|
+
this.pending.delete(id);
|
|
347
|
+
clearTimeout(timer);
|
|
348
|
+
// Best-effort cancel notification to the server. Some
|
|
349
|
+
// servers ignore $/cancelRequest; the local reject below
|
|
350
|
+
// is what frees the caller.
|
|
351
|
+
try {
|
|
352
|
+
this.sendNotification('$/cancelRequest', { id });
|
|
353
|
+
}
|
|
354
|
+
catch {
|
|
355
|
+
// server may already be gone
|
|
356
|
+
}
|
|
357
|
+
rejectFn(new OperatorAbortedError('lsp_request'));
|
|
358
|
+
}
|
|
359
|
+
});
|
|
360
|
+
}
|
|
361
|
+
try {
|
|
362
|
+
this.writeMessage(payload);
|
|
363
|
+
}
|
|
364
|
+
catch (error) {
|
|
365
|
+
this.pending.delete(id);
|
|
366
|
+
clearTimeout(timer);
|
|
367
|
+
rejectFn(error instanceof Error ? error : new Error(String(error)));
|
|
368
|
+
}
|
|
369
|
+
});
|
|
370
|
+
}
|
|
371
|
+
sendNotification(method, params) {
|
|
372
|
+
this.writeMessage({ jsonrpc: '2.0', method, params });
|
|
373
|
+
}
|
|
374
|
+
writeMessage(payload) {
|
|
375
|
+
const body = JSON.stringify(payload);
|
|
376
|
+
const message = `Content-Length: ${Buffer.byteLength(body, 'utf8')}\r\n\r\n${body}`;
|
|
377
|
+
this.child.stdin.write(message, 'utf8');
|
|
378
|
+
}
|
|
379
|
+
onStdout(chunk) {
|
|
380
|
+
this.buffer = Buffer.concat([this.buffer, chunk]);
|
|
381
|
+
while (true) {
|
|
382
|
+
const headerEnd = this.buffer.indexOf('\r\n\r\n');
|
|
383
|
+
if (headerEnd < 0)
|
|
384
|
+
return;
|
|
385
|
+
const headerText = this.buffer.subarray(0, headerEnd).toString('ascii');
|
|
386
|
+
const lengthMatch = headerText.match(/Content-Length:\s*(\d+)/i);
|
|
387
|
+
if (!lengthMatch || lengthMatch[1] === undefined) {
|
|
388
|
+
// R1 fix (2026-05-26, PR #413 r1, Fix 7): malformed header —
|
|
389
|
+
// instead of nuking the entire buffer (which would discard ANY
|
|
390
|
+
// subsequent valid messages already queued in `this.buffer`),
|
|
391
|
+
// scan forward for the next `Content-Length:` marker and resync
|
|
392
|
+
// from there. A misbehaving server that emits one bad header
|
|
393
|
+
// followed by a normal stream of responses must not freeze the
|
|
394
|
+
// client. When no recoverable next marker is in the buffer, we
|
|
395
|
+
// keep the buffer as-is and wait for more data — the broken
|
|
396
|
+
// bytes will be re-evaluated on the next chunk.
|
|
397
|
+
const nextHeaderIdx = this.buffer.indexOf(Buffer.from('Content-Length:'), 1);
|
|
398
|
+
if (nextHeaderIdx > 0) {
|
|
399
|
+
this.buffer = this.buffer.subarray(nextHeaderIdx);
|
|
400
|
+
continue;
|
|
401
|
+
}
|
|
402
|
+
// No next marker visible — wait for more data, do not nuke the
|
|
403
|
+
// buffer. A subsequent chunk may complete a valid header.
|
|
404
|
+
return;
|
|
405
|
+
}
|
|
406
|
+
const length = Number.parseInt(lengthMatch[1], 10);
|
|
407
|
+
const bodyStart = headerEnd + 4;
|
|
408
|
+
if (this.buffer.length < bodyStart + length)
|
|
409
|
+
return;
|
|
410
|
+
const bodyText = this.buffer.subarray(bodyStart, bodyStart + length).toString('utf8');
|
|
411
|
+
this.buffer = this.buffer.subarray(bodyStart + length);
|
|
412
|
+
try {
|
|
413
|
+
const message = JSON.parse(bodyText);
|
|
414
|
+
this.handleMessage(message);
|
|
415
|
+
}
|
|
416
|
+
catch {
|
|
417
|
+
// Garbage body — drop and continue parsing the next message.
|
|
418
|
+
}
|
|
419
|
+
}
|
|
420
|
+
}
|
|
421
|
+
handleMessage(message) {
|
|
422
|
+
// Response — has `id` and either `result` or `error`.
|
|
423
|
+
if (typeof message['id'] === 'number') {
|
|
424
|
+
const id = message['id'];
|
|
425
|
+
const entry = this.pending.get(id);
|
|
426
|
+
if (!entry)
|
|
427
|
+
return;
|
|
428
|
+
this.pending.delete(id);
|
|
429
|
+
clearTimeout(entry.timer);
|
|
430
|
+
if ('error' in message && message['error']) {
|
|
431
|
+
const err = message['error'];
|
|
432
|
+
entry.reject(new Error(`lsp_error code=${err.code ?? 'unknown'}: ${err.message ?? 'no message'}`));
|
|
433
|
+
return;
|
|
434
|
+
}
|
|
435
|
+
entry.resolve(message['result'] ?? null);
|
|
436
|
+
return;
|
|
437
|
+
}
|
|
438
|
+
// Notification from server.
|
|
439
|
+
if (message['method'] === 'textDocument/publishDiagnostics') {
|
|
440
|
+
const params = message['params'];
|
|
441
|
+
if (!params)
|
|
442
|
+
return;
|
|
443
|
+
const uri = typeof params.uri === 'string' ? params.uri : null;
|
|
444
|
+
if (!uri)
|
|
445
|
+
return;
|
|
446
|
+
const raw = Array.isArray(params.diagnostics) ? params.diagnostics : [];
|
|
447
|
+
const normalized = [];
|
|
448
|
+
for (const item of raw) {
|
|
449
|
+
const parsed = normalizeDiagnostic(item);
|
|
450
|
+
if (parsed)
|
|
451
|
+
normalized.push(parsed);
|
|
452
|
+
}
|
|
453
|
+
this.diagnosticsByUri.set(uri, normalized);
|
|
454
|
+
}
|
|
455
|
+
}
|
|
456
|
+
onExit() {
|
|
457
|
+
if (this.stopped)
|
|
458
|
+
return;
|
|
459
|
+
this.stopped = true;
|
|
460
|
+
const snapshot = Array.from(this.pending.entries());
|
|
461
|
+
this.pending.clear();
|
|
462
|
+
for (const [, entry] of snapshot) {
|
|
463
|
+
clearTimeout(entry.timer);
|
|
464
|
+
entry.reject(new Error('lsp_server_exited'));
|
|
465
|
+
}
|
|
466
|
+
}
|
|
467
|
+
}
|
|
468
|
+
/**
|
|
469
|
+
* Map a short LSP language slug to the settings.json key. β7 L9 — the
|
|
470
|
+
* settings schema spells out the full language name (`typescript`,
|
|
471
|
+
* `python`, ...) for human readability; the short slug (`ts`, `py`) is
|
|
472
|
+
* what every internal call site uses. Keep this map narrow and explicit.
|
|
473
|
+
*/
|
|
474
|
+
const SETTINGS_KEY_BY_LANG = {
|
|
475
|
+
ts: 'typescript',
|
|
476
|
+
js: 'javascript',
|
|
477
|
+
py: 'python',
|
|
478
|
+
go: 'go',
|
|
479
|
+
rust: 'rust',
|
|
480
|
+
};
|
|
481
|
+
/**
|
|
482
|
+
* Report whether the operator has explicitly disabled this language via
|
|
483
|
+
* `.pugi/settings.json::lsp.<language> = false`. Absent section or
|
|
484
|
+
* absent key means "enabled by default" — backwards-compatible with the
|
|
485
|
+
* α7.7 surface that ignored settings entirely. Returns true ONLY when
|
|
486
|
+
* the operator explicitly set the value to false.
|
|
487
|
+
*/
|
|
488
|
+
export function isLspLanguageDisabled(lang, lspSettings) {
|
|
489
|
+
if (!lspSettings)
|
|
490
|
+
return false;
|
|
491
|
+
const key = SETTINGS_KEY_BY_LANG[lang];
|
|
492
|
+
return lspSettings[key] === false;
|
|
493
|
+
}
|
|
494
|
+
/**
|
|
495
|
+
* Probe every registered language server. Operator-facing helper for
|
|
496
|
+
* `pugi lsp servers` — returns one row per language with the binary
|
|
497
|
+
* name, whether it was found on PATH, and whether the settings toggle
|
|
498
|
+
* has explicitly disabled it.
|
|
499
|
+
*/
|
|
500
|
+
export function inspectLspServers(lspSettings) {
|
|
501
|
+
const out = [];
|
|
502
|
+
for (const lang of Object.keys(LANGUAGE_SERVERS)) {
|
|
503
|
+
const server = LANGUAGE_SERVERS[lang];
|
|
504
|
+
out.push({
|
|
505
|
+
language: lang,
|
|
506
|
+
command: server.command + (server.args.length > 0 ? ` ${server.args.join(' ')}` : ''),
|
|
507
|
+
available: detectBinary(server.probe),
|
|
508
|
+
enabled: !isLspLanguageDisabled(lang, lspSettings),
|
|
509
|
+
});
|
|
510
|
+
}
|
|
511
|
+
return out;
|
|
512
|
+
}
|
|
513
|
+
/**
|
|
514
|
+
* Start an LSP client for the given language. Returns either an `LspClient`
|
|
515
|
+
* ready to use, or a structured failure (`lsp_unavailable`,
|
|
516
|
+
* `language_unsupported`).
|
|
517
|
+
*
|
|
518
|
+
* β7 L9: respects `.pugi/settings.json::lsp.<language> = false` —
|
|
519
|
+
* a disabled language reports `lsp_disabled` so the caller surface can
|
|
520
|
+
* tell the operator the binary IS available but settings says no.
|
|
521
|
+
*/
|
|
522
|
+
export async function startLspClient(lang, opts) {
|
|
523
|
+
const server = opts.serverOverride ?? LANGUAGE_SERVERS[lang];
|
|
524
|
+
if (!server) {
|
|
525
|
+
return {
|
|
526
|
+
ok: false,
|
|
527
|
+
reason: 'language_unsupported',
|
|
528
|
+
detail: `no LSP server registered for language: ${lang}`,
|
|
529
|
+
};
|
|
530
|
+
}
|
|
531
|
+
if (!opts.serverOverride && isLspLanguageDisabled(lang, opts.lspSettings)) {
|
|
532
|
+
return {
|
|
533
|
+
ok: false,
|
|
534
|
+
reason: 'lsp_disabled',
|
|
535
|
+
detail: `${lang} is disabled in .pugi/settings.json::lsp.${SETTINGS_KEY_BY_LANG[lang]}. ` +
|
|
536
|
+
`Remove the override (or set it to true) to enable.`,
|
|
537
|
+
};
|
|
538
|
+
}
|
|
539
|
+
if (!opts.serverOverride) {
|
|
540
|
+
const available = detectBinary(server.probe);
|
|
541
|
+
if (!available) {
|
|
542
|
+
return {
|
|
543
|
+
ok: false,
|
|
544
|
+
reason: 'lsp_unavailable',
|
|
545
|
+
detail: `${server.probe} not found on PATH. ` +
|
|
546
|
+
`Install the language server first ` +
|
|
547
|
+
`(see https://pugi.io/docs/cli/lsp for per-language commands).`,
|
|
548
|
+
};
|
|
549
|
+
}
|
|
550
|
+
}
|
|
551
|
+
let child;
|
|
552
|
+
try {
|
|
553
|
+
child = spawn(server.command, [...server.args], {
|
|
554
|
+
cwd: opts.cwd,
|
|
555
|
+
stdio: ['pipe', 'pipe', 'pipe'],
|
|
556
|
+
});
|
|
557
|
+
}
|
|
558
|
+
catch (error) {
|
|
559
|
+
return {
|
|
560
|
+
ok: false,
|
|
561
|
+
reason: 'lsp_unavailable',
|
|
562
|
+
detail: `failed to spawn ${server.command}: ${error instanceof Error ? error.message : String(error)}`,
|
|
563
|
+
};
|
|
564
|
+
}
|
|
565
|
+
// `child_process.spawn` reports a missing binary asynchronously via
|
|
566
|
+
// the 'error' event, NOT via a synchronous throw — the synchronous
|
|
567
|
+
// spawn returns a ChildProcess object even when the binary does not
|
|
568
|
+
// exist. Attach an error listener immediately so the missing-binary
|
|
569
|
+
// case never becomes an uncaught exception. Wait one microtask tick
|
|
570
|
+
// for the event-loop to fire the 'error' event before we attempt
|
|
571
|
+
// the handshake; if the spawn failed, return early with
|
|
572
|
+
// `lsp_unavailable`.
|
|
573
|
+
let spawnError = null;
|
|
574
|
+
child.on('error', (err) => {
|
|
575
|
+
spawnError = err;
|
|
576
|
+
});
|
|
577
|
+
// Yield one tick so Node's spawn-error path lands before we
|
|
578
|
+
// proceed. The error event lives on the same nextTick queue as the
|
|
579
|
+
// initial spawn handshake, so a single setImmediate-equivalent
|
|
580
|
+
// delay is enough to observe it.
|
|
581
|
+
await new Promise((resolveFn) => {
|
|
582
|
+
setImmediate(resolveFn);
|
|
583
|
+
});
|
|
584
|
+
if (spawnError) {
|
|
585
|
+
try {
|
|
586
|
+
child.kill('SIGKILL');
|
|
587
|
+
}
|
|
588
|
+
catch {
|
|
589
|
+
// ignore — process never started
|
|
590
|
+
}
|
|
591
|
+
return {
|
|
592
|
+
ok: false,
|
|
593
|
+
reason: 'lsp_unavailable',
|
|
594
|
+
detail: `failed to spawn ${server.command}: ${spawnError.message}`,
|
|
595
|
+
};
|
|
596
|
+
}
|
|
597
|
+
const client = new LspClient(child, server, opts);
|
|
598
|
+
try {
|
|
599
|
+
await initializeHandshake(client, opts.cwd);
|
|
600
|
+
}
|
|
601
|
+
catch (error) {
|
|
602
|
+
await client.stop();
|
|
603
|
+
if (spawnError) {
|
|
604
|
+
return {
|
|
605
|
+
ok: false,
|
|
606
|
+
reason: 'lsp_unavailable',
|
|
607
|
+
detail: `failed to spawn ${server.command}: ${spawnError.message}`,
|
|
608
|
+
};
|
|
609
|
+
}
|
|
610
|
+
return {
|
|
611
|
+
ok: false,
|
|
612
|
+
reason: 'lsp_error',
|
|
613
|
+
detail: `initialize failed: ${error instanceof Error ? error.message : String(error)}`,
|
|
614
|
+
};
|
|
615
|
+
}
|
|
616
|
+
return { ok: true, value: client };
|
|
617
|
+
}
|
|
618
|
+
async function initializeHandshake(client, cwd) {
|
|
619
|
+
const rootUri = pathToFileURL(cwd).toString();
|
|
620
|
+
// Reach the private send-request / send-notification surface through
|
|
621
|
+
// a typed accessor cast. The two methods are intentionally not part
|
|
622
|
+
// of the public class surface (callers should use `hover`/`definition`
|
|
623
|
+
// etc.), but the handshake is a single-shot bootstrap and exposing
|
|
624
|
+
// the raw methods would weaken the type story.
|
|
625
|
+
const internal = client;
|
|
626
|
+
await internal.sendRequest('initialize', {
|
|
627
|
+
processId: process.pid,
|
|
628
|
+
rootUri,
|
|
629
|
+
capabilities: {
|
|
630
|
+
textDocument: {
|
|
631
|
+
hover: { contentFormat: ['plaintext', 'markdown'] },
|
|
632
|
+
definition: { linkSupport: false },
|
|
633
|
+
references: {},
|
|
634
|
+
publishDiagnostics: { relatedInformation: false },
|
|
635
|
+
},
|
|
636
|
+
},
|
|
637
|
+
workspaceFolders: [{ uri: rootUri, name: 'pugi-workspace' }],
|
|
638
|
+
}, undefined);
|
|
639
|
+
internal.sendNotification('initialized', {});
|
|
640
|
+
}
|
|
641
|
+
function detectBinary(name) {
|
|
642
|
+
// Cross-platform `which` — spawnSync of the binary with --version is
|
|
643
|
+
// too aggressive (some servers don't honor --version). Use `which`
|
|
644
|
+
// on POSIX and `where` on Windows. Failures are non-fatal — we just
|
|
645
|
+
// report unavailable.
|
|
646
|
+
const probe = process.platform === 'win32' ? 'where' : 'which';
|
|
647
|
+
try {
|
|
648
|
+
const result = spawnSync(probe, [name], { stdio: 'ignore' });
|
|
649
|
+
return result.status === 0;
|
|
650
|
+
}
|
|
651
|
+
catch {
|
|
652
|
+
return false;
|
|
653
|
+
}
|
|
654
|
+
}
|
|
655
|
+
function normalizeHover(raw) {
|
|
656
|
+
// LSP `Hover.contents` can be:
|
|
657
|
+
// - string
|
|
658
|
+
// - { kind: 'markdown' | 'plaintext', value: string }
|
|
659
|
+
// - Array<string | { language: string, value: string }>
|
|
660
|
+
if (!raw || typeof raw !== 'object') {
|
|
661
|
+
return { content: String(raw ?? ''), raw };
|
|
662
|
+
}
|
|
663
|
+
const obj = raw;
|
|
664
|
+
const range = parseRange(obj.range);
|
|
665
|
+
const body = obj.contents;
|
|
666
|
+
const result = (() => {
|
|
667
|
+
if (typeof body === 'string')
|
|
668
|
+
return { content: body, raw, ...(range ? { range } : {}) };
|
|
669
|
+
if (Array.isArray(body)) {
|
|
670
|
+
const parts = [];
|
|
671
|
+
for (const item of body) {
|
|
672
|
+
if (typeof item === 'string')
|
|
673
|
+
parts.push(item);
|
|
674
|
+
else if (item && typeof item === 'object' && 'value' in item) {
|
|
675
|
+
const value = item.value;
|
|
676
|
+
if (typeof value === 'string')
|
|
677
|
+
parts.push(value);
|
|
678
|
+
}
|
|
679
|
+
}
|
|
680
|
+
return { content: parts.join('\n'), raw, ...(range ? { range } : {}) };
|
|
681
|
+
}
|
|
682
|
+
if (body && typeof body === 'object' && 'value' in body) {
|
|
683
|
+
const value = body.value;
|
|
684
|
+
return { content: typeof value === 'string' ? value : '', raw, ...(range ? { range } : {}) };
|
|
685
|
+
}
|
|
686
|
+
return { content: '', raw, ...(range ? { range } : {}) };
|
|
687
|
+
})();
|
|
688
|
+
return result;
|
|
689
|
+
}
|
|
690
|
+
function normalizeLocations(raw, cwd) {
|
|
691
|
+
if (!raw)
|
|
692
|
+
return [];
|
|
693
|
+
const items = Array.isArray(raw) ? raw : [raw];
|
|
694
|
+
const out = [];
|
|
695
|
+
for (const item of items) {
|
|
696
|
+
if (!item || typeof item !== 'object')
|
|
697
|
+
continue;
|
|
698
|
+
const obj = item;
|
|
699
|
+
const uri = typeof obj.uri === 'string' ? obj.uri : typeof obj.targetUri === 'string' ? obj.targetUri : null;
|
|
700
|
+
const range = parseRange(obj.range ?? obj.targetRange);
|
|
701
|
+
if (!uri || !range)
|
|
702
|
+
continue;
|
|
703
|
+
let path = '';
|
|
704
|
+
try {
|
|
705
|
+
const url = new URL(uri);
|
|
706
|
+
if (url.protocol === 'file:') {
|
|
707
|
+
const abs = decodeURIComponent(url.pathname);
|
|
708
|
+
if (abs.startsWith(cwd + sep) || abs === cwd) {
|
|
709
|
+
path = abs.slice(cwd.length + 1);
|
|
710
|
+
}
|
|
711
|
+
}
|
|
712
|
+
}
|
|
713
|
+
catch {
|
|
714
|
+
path = '';
|
|
715
|
+
}
|
|
716
|
+
out.push({ uri, path, range });
|
|
717
|
+
}
|
|
718
|
+
return out;
|
|
719
|
+
}
|
|
720
|
+
function parseRange(raw) {
|
|
721
|
+
if (!raw || typeof raw !== 'object')
|
|
722
|
+
return undefined;
|
|
723
|
+
const obj = raw;
|
|
724
|
+
const start = parsePosition(obj.start);
|
|
725
|
+
const end = parsePosition(obj.end);
|
|
726
|
+
if (!start || !end)
|
|
727
|
+
return undefined;
|
|
728
|
+
return { start, end };
|
|
729
|
+
}
|
|
730
|
+
function parsePosition(raw) {
|
|
731
|
+
if (!raw || typeof raw !== 'object')
|
|
732
|
+
return undefined;
|
|
733
|
+
const obj = raw;
|
|
734
|
+
if (typeof obj.line !== 'number' || typeof obj.character !== 'number')
|
|
735
|
+
return undefined;
|
|
736
|
+
return { line: obj.line, character: obj.character };
|
|
737
|
+
}
|
|
738
|
+
function normalizeDiagnostic(raw) {
|
|
739
|
+
if (!raw || typeof raw !== 'object')
|
|
740
|
+
return null;
|
|
741
|
+
const obj = raw;
|
|
742
|
+
const range = parseRange(obj.range);
|
|
743
|
+
if (!range)
|
|
744
|
+
return null;
|
|
745
|
+
if (typeof obj.message !== 'string')
|
|
746
|
+
return null;
|
|
747
|
+
const severityRaw = typeof obj.severity === 'number' ? obj.severity : 1;
|
|
748
|
+
const severity = (severityRaw >= 1 && severityRaw <= 4 ? severityRaw : 1);
|
|
749
|
+
const labels = {
|
|
750
|
+
1: 'error',
|
|
751
|
+
2: 'warning',
|
|
752
|
+
3: 'info',
|
|
753
|
+
4: 'hint',
|
|
754
|
+
};
|
|
755
|
+
const out = {
|
|
756
|
+
severity,
|
|
757
|
+
severityLabel: labels[severity],
|
|
758
|
+
message: obj.message,
|
|
759
|
+
range,
|
|
760
|
+
...(typeof obj.source === 'string' ? { source: obj.source } : {}),
|
|
761
|
+
...(typeof obj.code === 'string' || typeof obj.code === 'number' ? { code: obj.code } : {}),
|
|
762
|
+
};
|
|
763
|
+
return out;
|
|
764
|
+
}
|
|
765
|
+
/**
|
|
766
|
+
* Test-only surface so specs can hand-craft an `LspClient` over a mock
|
|
767
|
+
* stdio pipe without paying for the real `startLspClient` spawn cost.
|
|
768
|
+
* The exported shape mirrors what the constructor needs.
|
|
769
|
+
*/
|
|
770
|
+
export const __test__ = {
|
|
771
|
+
LANGUAGE_SERVERS,
|
|
772
|
+
normalizeHover,
|
|
773
|
+
normalizeLocations,
|
|
774
|
+
normalizeDiagnostic,
|
|
775
|
+
};
|
|
776
|
+
//# sourceMappingURL=client.js.map
|