gsd-pi 2.8.2 → 2.9.0
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/README.md +2 -1
- package/dist/cli.js +5 -0
- package/dist/loader.js +1 -1
- package/dist/update-check.d.ts +24 -0
- package/dist/update-check.js +93 -0
- package/node_modules/@gsd/pi-coding-agent/dist/core/extensions/types.d.ts +4 -2
- package/node_modules/@gsd/pi-coding-agent/dist/core/extensions/types.d.ts.map +1 -1
- package/node_modules/@gsd/pi-coding-agent/dist/core/extensions/types.js.map +1 -1
- package/node_modules/@gsd/pi-coding-agent/dist/core/lsp/client.d.ts +46 -0
- package/node_modules/@gsd/pi-coding-agent/dist/core/lsp/client.d.ts.map +1 -0
- package/node_modules/@gsd/pi-coding-agent/dist/core/lsp/client.js +758 -0
- package/node_modules/@gsd/pi-coding-agent/dist/core/lsp/client.js.map +1 -0
- package/node_modules/@gsd/pi-coding-agent/dist/core/lsp/config.d.ts +23 -0
- package/node_modules/@gsd/pi-coding-agent/dist/core/lsp/config.d.ts.map +1 -0
- package/node_modules/@gsd/pi-coding-agent/dist/core/lsp/config.js +267 -0
- package/node_modules/@gsd/pi-coding-agent/dist/core/lsp/config.js.map +1 -0
- package/node_modules/@gsd/pi-coding-agent/dist/core/lsp/edits.d.ts +17 -0
- package/node_modules/@gsd/pi-coding-agent/dist/core/lsp/edits.d.ts.map +1 -0
- package/node_modules/@gsd/pi-coding-agent/dist/core/lsp/edits.js +101 -0
- package/node_modules/@gsd/pi-coding-agent/dist/core/lsp/edits.js.map +1 -0
- package/node_modules/@gsd/pi-coding-agent/dist/core/lsp/helpers.d.ts +15 -0
- package/node_modules/@gsd/pi-coding-agent/dist/core/lsp/helpers.d.ts.map +1 -0
- package/node_modules/@gsd/pi-coding-agent/dist/core/lsp/helpers.js +46 -0
- package/node_modules/@gsd/pi-coding-agent/dist/core/lsp/helpers.js.map +1 -0
- package/node_modules/@gsd/pi-coding-agent/dist/core/lsp/index.d.ts +35 -0
- package/node_modules/@gsd/pi-coding-agent/dist/core/lsp/index.d.ts.map +1 -0
- package/node_modules/@gsd/pi-coding-agent/dist/core/lsp/index.js +709 -0
- package/node_modules/@gsd/pi-coding-agent/dist/core/lsp/index.js.map +1 -0
- package/node_modules/@gsd/pi-coding-agent/dist/core/lsp/lsp-integration.test.d.ts +2 -0
- package/node_modules/@gsd/pi-coding-agent/dist/core/lsp/lsp-integration.test.d.ts.map +1 -0
- package/node_modules/@gsd/pi-coding-agent/dist/core/lsp/lsp-integration.test.js +308 -0
- package/node_modules/@gsd/pi-coding-agent/dist/core/lsp/lsp-integration.test.js.map +1 -0
- package/node_modules/@gsd/pi-coding-agent/dist/core/lsp/lspmux.d.ts +34 -0
- package/node_modules/@gsd/pi-coding-agent/dist/core/lsp/lspmux.d.ts.map +1 -0
- package/node_modules/@gsd/pi-coding-agent/dist/core/lsp/lspmux.js +136 -0
- package/node_modules/@gsd/pi-coding-agent/dist/core/lsp/lspmux.js.map +1 -0
- package/node_modules/@gsd/pi-coding-agent/dist/core/lsp/types.d.ts +262 -0
- package/node_modules/@gsd/pi-coding-agent/dist/core/lsp/types.d.ts.map +1 -0
- package/node_modules/@gsd/pi-coding-agent/dist/core/lsp/types.js +64 -0
- package/node_modules/@gsd/pi-coding-agent/dist/core/lsp/types.js.map +1 -0
- package/node_modules/@gsd/pi-coding-agent/dist/core/lsp/utils.d.ts +50 -0
- package/node_modules/@gsd/pi-coding-agent/dist/core/lsp/utils.d.ts.map +1 -0
- package/node_modules/@gsd/pi-coding-agent/dist/core/lsp/utils.js +574 -0
- package/node_modules/@gsd/pi-coding-agent/dist/core/lsp/utils.js.map +1 -0
- package/node_modules/@gsd/pi-coding-agent/dist/core/slash-commands.d.ts.map +1 -1
- package/node_modules/@gsd/pi-coding-agent/dist/core/slash-commands.js +1 -0
- package/node_modules/@gsd/pi-coding-agent/dist/core/slash-commands.js.map +1 -1
- package/node_modules/@gsd/pi-coding-agent/dist/core/tools/index.d.ts +13 -0
- package/node_modules/@gsd/pi-coding-agent/dist/core/tools/index.d.ts.map +1 -1
- package/node_modules/@gsd/pi-coding-agent/dist/core/tools/index.js +4 -0
- package/node_modules/@gsd/pi-coding-agent/dist/core/tools/index.js.map +1 -1
- package/node_modules/@gsd/pi-coding-agent/dist/modes/interactive/components/settings-selector.d.ts +10 -1
- package/node_modules/@gsd/pi-coding-agent/dist/modes/interactive/components/settings-selector.d.ts.map +1 -1
- package/node_modules/@gsd/pi-coding-agent/dist/modes/interactive/components/settings-selector.js +2 -2
- package/node_modules/@gsd/pi-coding-agent/dist/modes/interactive/components/settings-selector.js.map +1 -1
- package/node_modules/@gsd/pi-coding-agent/dist/modes/interactive/interactive-mode.d.ts +2 -0
- package/node_modules/@gsd/pi-coding-agent/dist/modes/interactive/interactive-mode.d.ts.map +1 -1
- package/node_modules/@gsd/pi-coding-agent/dist/modes/interactive/interactive-mode.js +80 -1
- package/node_modules/@gsd/pi-coding-agent/dist/modes/interactive/interactive-mode.js.map +1 -1
- package/node_modules/@gsd/pi-coding-agent/dist/modes/rpc/rpc-mode.js +1 -1
- package/node_modules/@gsd/pi-coding-agent/dist/modes/rpc/rpc-mode.js.map +1 -1
- package/node_modules/@gsd/pi-coding-agent/dist/modes/rpc/rpc-types.d.ts +5 -0
- package/node_modules/@gsd/pi-coding-agent/dist/modes/rpc/rpc-types.d.ts.map +1 -1
- package/node_modules/@gsd/pi-coding-agent/dist/modes/rpc/rpc-types.js.map +1 -1
- package/node_modules/@gsd/pi-coding-agent/src/core/extensions/types.ts +4 -2
- package/node_modules/@gsd/pi-coding-agent/src/core/lsp/client.ts +880 -0
- package/node_modules/@gsd/pi-coding-agent/src/core/lsp/config.ts +325 -0
- package/node_modules/@gsd/pi-coding-agent/src/core/lsp/defaults.json +456 -0
- package/node_modules/@gsd/pi-coding-agent/src/core/lsp/edits.ts +109 -0
- package/node_modules/@gsd/pi-coding-agent/src/core/lsp/helpers.ts +54 -0
- package/node_modules/@gsd/pi-coding-agent/src/core/lsp/index.ts +943 -0
- package/node_modules/@gsd/pi-coding-agent/src/core/lsp/lsp-integration.test.ts +407 -0
- package/node_modules/@gsd/pi-coding-agent/src/core/lsp/lsp.md +33 -0
- package/node_modules/@gsd/pi-coding-agent/src/core/lsp/lspmux.ts +199 -0
- package/node_modules/@gsd/pi-coding-agent/src/core/lsp/types.ts +421 -0
- package/node_modules/@gsd/pi-coding-agent/src/core/lsp/utils.ts +682 -0
- package/node_modules/@gsd/pi-coding-agent/src/core/slash-commands.ts +1 -0
- package/node_modules/@gsd/pi-coding-agent/src/core/tools/index.ts +10 -0
- package/node_modules/@gsd/pi-coding-agent/src/modes/interactive/components/settings-selector.ts +2 -2
- package/node_modules/@gsd/pi-coding-agent/src/modes/interactive/interactive-mode.ts +94 -2
- package/node_modules/@gsd/pi-coding-agent/src/modes/rpc/rpc-mode.ts +2 -2
- package/node_modules/@gsd/pi-coding-agent/src/modes/rpc/rpc-types.ts +2 -1
- package/package.json +1 -1
- package/packages/pi-coding-agent/dist/core/extensions/types.d.ts +4 -2
- package/packages/pi-coding-agent/dist/core/extensions/types.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/core/extensions/types.js.map +1 -1
- package/packages/pi-coding-agent/dist/core/lsp/client.d.ts +46 -0
- package/packages/pi-coding-agent/dist/core/lsp/client.d.ts.map +1 -0
- package/packages/pi-coding-agent/dist/core/lsp/client.js +758 -0
- package/packages/pi-coding-agent/dist/core/lsp/client.js.map +1 -0
- package/packages/pi-coding-agent/dist/core/lsp/config.d.ts +23 -0
- package/packages/pi-coding-agent/dist/core/lsp/config.d.ts.map +1 -0
- package/packages/pi-coding-agent/dist/core/lsp/config.js +267 -0
- package/packages/pi-coding-agent/dist/core/lsp/config.js.map +1 -0
- package/packages/pi-coding-agent/dist/core/lsp/edits.d.ts +17 -0
- package/packages/pi-coding-agent/dist/core/lsp/edits.d.ts.map +1 -0
- package/packages/pi-coding-agent/dist/core/lsp/edits.js +101 -0
- package/packages/pi-coding-agent/dist/core/lsp/edits.js.map +1 -0
- package/packages/pi-coding-agent/dist/core/lsp/helpers.d.ts +15 -0
- package/packages/pi-coding-agent/dist/core/lsp/helpers.d.ts.map +1 -0
- package/packages/pi-coding-agent/dist/core/lsp/helpers.js +46 -0
- package/packages/pi-coding-agent/dist/core/lsp/helpers.js.map +1 -0
- package/packages/pi-coding-agent/dist/core/lsp/index.d.ts +35 -0
- package/packages/pi-coding-agent/dist/core/lsp/index.d.ts.map +1 -0
- package/packages/pi-coding-agent/dist/core/lsp/index.js +709 -0
- package/packages/pi-coding-agent/dist/core/lsp/index.js.map +1 -0
- package/packages/pi-coding-agent/dist/core/lsp/lsp-integration.test.d.ts +2 -0
- package/packages/pi-coding-agent/dist/core/lsp/lsp-integration.test.d.ts.map +1 -0
- package/packages/pi-coding-agent/dist/core/lsp/lsp-integration.test.js +308 -0
- package/packages/pi-coding-agent/dist/core/lsp/lsp-integration.test.js.map +1 -0
- package/packages/pi-coding-agent/dist/core/lsp/lspmux.d.ts +34 -0
- package/packages/pi-coding-agent/dist/core/lsp/lspmux.d.ts.map +1 -0
- package/packages/pi-coding-agent/dist/core/lsp/lspmux.js +136 -0
- package/packages/pi-coding-agent/dist/core/lsp/lspmux.js.map +1 -0
- package/packages/pi-coding-agent/dist/core/lsp/types.d.ts +262 -0
- package/packages/pi-coding-agent/dist/core/lsp/types.d.ts.map +1 -0
- package/packages/pi-coding-agent/dist/core/lsp/types.js +64 -0
- package/packages/pi-coding-agent/dist/core/lsp/types.js.map +1 -0
- package/packages/pi-coding-agent/dist/core/lsp/utils.d.ts +50 -0
- package/packages/pi-coding-agent/dist/core/lsp/utils.d.ts.map +1 -0
- package/packages/pi-coding-agent/dist/core/lsp/utils.js +574 -0
- package/packages/pi-coding-agent/dist/core/lsp/utils.js.map +1 -0
- package/packages/pi-coding-agent/dist/core/slash-commands.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/core/slash-commands.js +1 -0
- package/packages/pi-coding-agent/dist/core/slash-commands.js.map +1 -1
- package/packages/pi-coding-agent/dist/core/tools/index.d.ts +13 -0
- package/packages/pi-coding-agent/dist/core/tools/index.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/core/tools/index.js +4 -0
- package/packages/pi-coding-agent/dist/core/tools/index.js.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/components/settings-selector.d.ts +10 -1
- package/packages/pi-coding-agent/dist/modes/interactive/components/settings-selector.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/components/settings-selector.js +2 -2
- package/packages/pi-coding-agent/dist/modes/interactive/components/settings-selector.js.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.d.ts +2 -0
- package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.js +80 -1
- package/packages/pi-coding-agent/dist/modes/interactive/interactive-mode.js.map +1 -1
- package/packages/pi-coding-agent/dist/modes/rpc/rpc-mode.js +1 -1
- package/packages/pi-coding-agent/dist/modes/rpc/rpc-mode.js.map +1 -1
- package/packages/pi-coding-agent/dist/modes/rpc/rpc-types.d.ts +5 -0
- package/packages/pi-coding-agent/dist/modes/rpc/rpc-types.d.ts.map +1 -1
- package/packages/pi-coding-agent/dist/modes/rpc/rpc-types.js.map +1 -1
- package/packages/pi-coding-agent/src/core/extensions/types.ts +4 -2
- package/packages/pi-coding-agent/src/core/lsp/client.ts +880 -0
- package/packages/pi-coding-agent/src/core/lsp/config.ts +325 -0
- package/packages/pi-coding-agent/src/core/lsp/defaults.json +456 -0
- package/packages/pi-coding-agent/src/core/lsp/edits.ts +109 -0
- package/packages/pi-coding-agent/src/core/lsp/helpers.ts +54 -0
- package/packages/pi-coding-agent/src/core/lsp/index.ts +943 -0
- package/packages/pi-coding-agent/src/core/lsp/lsp-integration.test.ts +407 -0
- package/packages/pi-coding-agent/src/core/lsp/lsp.md +33 -0
- package/packages/pi-coding-agent/src/core/lsp/lspmux.ts +199 -0
- package/packages/pi-coding-agent/src/core/lsp/types.ts +421 -0
- package/packages/pi-coding-agent/src/core/lsp/utils.ts +682 -0
- package/packages/pi-coding-agent/src/core/slash-commands.ts +1 -0
- package/packages/pi-coding-agent/src/core/tools/index.ts +10 -0
- package/packages/pi-coding-agent/src/modes/interactive/components/settings-selector.ts +2 -2
- package/packages/pi-coding-agent/src/modes/interactive/interactive-mode.ts +94 -2
- package/packages/pi-coding-agent/src/modes/rpc/rpc-mode.ts +2 -2
- package/packages/pi-coding-agent/src/modes/rpc/rpc-types.ts +2 -1
- package/src/resources/extensions/ask-user-questions.ts +42 -2
- package/src/resources/extensions/bg-shell/index.ts +34 -37
- package/src/resources/extensions/browser-tools/core.d.ts +205 -0
- package/src/resources/extensions/browser-tools/index.ts +2 -2
- package/src/resources/extensions/browser-tools/refs.ts +1 -1
- package/src/resources/extensions/browser-tools/tools/session.ts +1 -1
- package/src/resources/extensions/context7/index.ts +2 -2
- package/src/resources/extensions/get-secrets-from-user.ts +3 -2
- package/src/resources/extensions/google-search/index.ts +1 -1
- package/src/resources/extensions/gsd/auto.ts +126 -12
- package/src/resources/extensions/gsd/commands.ts +218 -3
- package/src/resources/extensions/gsd/doctor.ts +1 -1
- package/src/resources/extensions/gsd/git-service.ts +163 -13
- package/src/resources/extensions/gsd/guided-flow.ts +19 -9
- package/src/resources/extensions/gsd/index.ts +17 -7
- package/src/resources/extensions/gsd/preferences.ts +1 -1
- package/src/resources/extensions/gsd/tests/git-service.test.ts +226 -0
- package/src/resources/extensions/gsd/tests/migrate-command.test.ts +2 -2
- package/src/resources/extensions/gsd/tests/migrate-transformer.test.ts +1 -1
- package/src/resources/extensions/gsd/tests/migrate-writer-integration.test.ts +10 -10
- package/src/resources/extensions/gsd/tests/next-milestone-id.test.ts +87 -0
- package/src/resources/extensions/gsd/tests/worktree.test.ts +352 -0
- package/src/resources/extensions/gsd/types.ts +1 -0
- package/src/resources/extensions/gsd/worktree.ts +20 -1
- package/src/resources/extensions/mac-tools/index.ts +1 -1
- package/src/resources/extensions/search-the-web/command-search-provider.ts +1 -1
- package/src/resources/extensions/search-the-web/format.ts +1 -1
- package/src/resources/extensions/search-the-web/index.ts +5 -5
- package/src/resources/extensions/search-the-web/native-search.ts +5 -6
- package/src/resources/extensions/search-the-web/tool-fetch-page.ts +7 -7
- package/src/resources/extensions/search-the-web/tool-llm-context.ts +11 -11
- package/src/resources/extensions/search-the-web/tool-search.ts +10 -10
- package/src/resources/extensions/shared/interview-ui.ts +2 -2
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"client.js","sourceRoot":"","sources":["../../../src/core/lsp/client.ts"],"names":[],"mappings":"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","sourcesContent":["import { spawn } from \"node:child_process\";\nimport * as fsPromises from \"node:fs/promises\";\nimport type { Writable } from \"node:stream\";\nimport { killProcessTree } from \"../../utils/shell.js\";\nimport { ToolAbortError, isEnoent, throwIfAborted, untilAborted } from \"./helpers.js\";\nimport { applyWorkspaceEdit } from \"./edits.js\";\nimport { getLspmuxCommand, isLspmuxSupported } from \"./lspmux.js\";\nimport type {\n\tDiagnostic,\n\tLspClient,\n\tLspJsonRpcNotification,\n\tLspJsonRpcRequest,\n\tLspJsonRpcResponse,\n\tServerConfig,\n\tWorkspaceEdit,\n} from \"./types.js\";\nimport { detectLanguageId, fileToUri } from \"./utils.js\";\n\n// =============================================================================\n// Client State\n// =============================================================================\n\nconst clients = new Map<string, LspClient>();\nconst clientLocks = new Map<string, Promise<LspClient>>();\nconst fileOperationLocks = new Map<string, Promise<void>>();\n\n// Idle timeout configuration (disabled by default)\nlet idleTimeoutMs: number | null = null;\nlet idleCheckInterval: ReturnType<typeof setInterval> | null = null;\nconst IDLE_CHECK_INTERVAL_MS = 60 * 1000;\n\n/**\n * Configure the idle timeout for LSP clients.\n */\nexport function setIdleTimeout(ms: number | null | undefined): void {\n\tidleTimeoutMs = ms ?? null;\n\n\tif (idleTimeoutMs && idleTimeoutMs > 0) {\n\t\tstartIdleChecker();\n\t} else {\n\t\tstopIdleChecker();\n\t}\n}\n\nfunction startIdleChecker(): void {\n\tif (idleCheckInterval) return;\n\tidleCheckInterval = setInterval(() => {\n\t\tif (!idleTimeoutMs) return;\n\t\tconst now = Date.now();\n\t\tfor (const [key, client] of Array.from(clients.entries())) {\n\t\t\tif (now - client.lastActivity > idleTimeoutMs) {\n\t\t\t\tshutdownClient(key);\n\t\t\t}\n\t\t}\n\t}, IDLE_CHECK_INTERVAL_MS);\n}\n\nfunction stopIdleChecker(): void {\n\tif (idleCheckInterval) {\n\t\tclearInterval(idleCheckInterval);\n\t\tidleCheckInterval = null;\n\t}\n}\n\n// =============================================================================\n// Client Capabilities\n// =============================================================================\n\nconst CLIENT_CAPABILITIES = {\n\ttextDocument: {\n\t\tsynchronization: {\n\t\t\tdidSave: true,\n\t\t\tdynamicRegistration: false,\n\t\t\twillSave: false,\n\t\t\twillSaveWaitUntil: false,\n\t\t},\n\t\thover: {\n\t\t\tcontentFormat: [\"markdown\", \"plaintext\"],\n\t\t\tdynamicRegistration: false,\n\t\t},\n\t\tdefinition: {\n\t\t\tdynamicRegistration: false,\n\t\t\tlinkSupport: true,\n\t\t},\n\t\ttypeDefinition: {\n\t\t\tdynamicRegistration: false,\n\t\t\tlinkSupport: true,\n\t\t},\n\t\timplementation: {\n\t\t\tdynamicRegistration: false,\n\t\t\tlinkSupport: true,\n\t\t},\n\t\treferences: {\n\t\t\tdynamicRegistration: false,\n\t\t},\n\t\tdocumentSymbol: {\n\t\t\tdynamicRegistration: false,\n\t\t\thierarchicalDocumentSymbolSupport: true,\n\t\t\tsymbolKind: {\n\t\t\t\tvalueSet: [1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 17, 18, 19, 20, 21, 22, 23, 24, 25, 26],\n\t\t\t},\n\t\t},\n\t\trename: {\n\t\t\tdynamicRegistration: false,\n\t\t\tprepareSupport: true,\n\t\t},\n\t\tcodeAction: {\n\t\t\tdynamicRegistration: false,\n\t\t\tcodeActionLiteralSupport: {\n\t\t\t\tcodeActionKind: {\n\t\t\t\t\tvalueSet: [\n\t\t\t\t\t\t\"quickfix\",\n\t\t\t\t\t\t\"refactor\",\n\t\t\t\t\t\t\"refactor.extract\",\n\t\t\t\t\t\t\"refactor.inline\",\n\t\t\t\t\t\t\"refactor.rewrite\",\n\t\t\t\t\t\t\"source\",\n\t\t\t\t\t\t\"source.organizeImports\",\n\t\t\t\t\t\t\"source.fixAll\",\n\t\t\t\t\t],\n\t\t\t\t},\n\t\t\t},\n\t\t\tresolveSupport: {\n\t\t\t\tproperties: [\"edit\"],\n\t\t\t},\n\t\t},\n\t\tformatting: {\n\t\t\tdynamicRegistration: false,\n\t\t},\n\t\trangeFormatting: {\n\t\t\tdynamicRegistration: false,\n\t\t},\n\t\tpublishDiagnostics: {\n\t\t\trelatedInformation: true,\n\t\t\tversionSupport: false,\n\t\t\ttagSupport: { valueSet: [1, 2] },\n\t\t\tcodeDescriptionSupport: true,\n\t\t\tdataSupport: true,\n\t\t},\n\t},\n\tworkspace: {\n\t\tapplyEdit: true,\n\t\tworkspaceEdit: {\n\t\t\tdocumentChanges: true,\n\t\t\tresourceOperations: [\"create\", \"rename\", \"delete\"],\n\t\t\tfailureHandling: \"textOnlyTransactional\",\n\t\t},\n\t\tconfiguration: true,\n\t\tsymbol: {\n\t\t\tdynamicRegistration: false,\n\t\t\tsymbolKind: {\n\t\t\t\tvalueSet: [1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 17, 18, 19, 20, 21, 22, 23, 24, 25, 26],\n\t\t\t},\n\t\t},\n\t},\n\texperimental: {\n\t\tsnippetTextEdit: true,\n\t},\n};\n\n// =============================================================================\n// LSP Message Protocol\n// =============================================================================\n\nfunction parseMessage(\n\tbuffer: Buffer,\n): { message: LspJsonRpcResponse | LspJsonRpcNotification | null; remaining: Buffer } | null {\n\tconst headerEndIndex = findHeaderEnd(buffer);\n\tif (headerEndIndex === -1) return null;\n\n\tconst headerText = new TextDecoder().decode(buffer.slice(0, headerEndIndex));\n\tconst contentLengthMatch = headerText.match(/Content-Length: (\\d+)/i);\n\tif (!contentLengthMatch) return null;\n\n\tconst contentLength = Number.parseInt(contentLengthMatch[1], 10);\n\tconst messageStart = headerEndIndex + 4; // Skip \\r\\n\\r\\n\n\tconst messageEnd = messageStart + contentLength;\n\n\tif (buffer.length < messageEnd) return null;\n\n\tconst messageBytes = buffer.subarray(messageStart, messageEnd);\n\tconst messageText = new TextDecoder().decode(messageBytes);\n\tconst remaining = Buffer.from(buffer.subarray(messageEnd));\n\n\tlet message: LspJsonRpcResponse | LspJsonRpcNotification;\n\ttry {\n\t\tmessage = JSON.parse(messageText);\n\t} catch {\n\t\t// Malformed JSON from LSP server — skip this message and advance past it\n\t\treturn { message: null, remaining };\n\t}\n\n\treturn { message, remaining };\n}\n\nfunction findHeaderEnd(buffer: Uint8Array): number {\n\tfor (let i = 0; i < buffer.length - 3; i++) {\n\t\tif (buffer[i] === 13 && buffer[i + 1] === 10 && buffer[i + 2] === 13 && buffer[i + 3] === 10) {\n\t\t\treturn i;\n\t\t}\n\t}\n\treturn -1;\n}\n\nasync function writeMessage(\n\tstdin: Writable | null,\n\tmessage: LspJsonRpcRequest | LspJsonRpcNotification | LspJsonRpcResponse,\n): Promise<void> {\n\tif (!stdin) {\n\t\tthrow new Error(\"LSP process stdin is not available\");\n\t}\n\tconst content = JSON.stringify(message);\n\tconst header = `Content-Length: ${Buffer.byteLength(content, \"utf-8\")}\\r\\n\\r\\n`;\n\treturn new Promise((resolve, reject) => {\n\t\tstdin.write(header + content, (err?: Error | null) => {\n\t\t\tif (err) reject(err);\n\t\t\telse resolve();\n\t\t});\n\t});\n}\n\n// =============================================================================\n// Message Reader\n// =============================================================================\n\nasync function startMessageReader(client: LspClient): Promise<void> {\n\tif (client.isReading) return;\n\tclient.isReading = true;\n\n\tconst stdout = client.proc.stdout;\n\tif (!stdout) {\n\t\tclient.isReading = false;\n\t\treturn;\n\t}\n\n\treturn new Promise<void>((resolve) => {\n\t\tstdout.on(\"data\", async (chunk: Buffer) => {\n\t\t\tconst currentBuffer: Buffer = Buffer.concat([client.messageBuffer, chunk]);\n\t\t\tclient.messageBuffer = currentBuffer;\n\n\t\t\tlet workingBuffer = currentBuffer;\n\t\t\tlet parsed = parseMessage(workingBuffer);\n\t\t\twhile (parsed) {\n\t\t\t\tconst { message, remaining } = parsed;\n\t\t\t\tworkingBuffer = remaining;\n\n\t\t\t\tif (!message) {\n\t\t\t\t\tparsed = parseMessage(workingBuffer);\n\t\t\t\t\tcontinue;\n\t\t\t\t}\n\n\t\t\t\tif (\"id\" in message && message.id !== undefined) {\n\t\t\t\t\tconst pending = client.pendingRequests.get(message.id);\n\t\t\t\t\tif (pending) {\n\t\t\t\t\t\tclient.pendingRequests.delete(message.id);\n\t\t\t\t\t\tif (\"error\" in message && message.error) {\n\t\t\t\t\t\t\tpending.reject(new Error(`LSP error: ${message.error.message}`));\n\t\t\t\t\t\t} else {\n\t\t\t\t\t\t\tpending.resolve(message.result);\n\t\t\t\t\t\t}\n\t\t\t\t\t} else if (\"method\" in message) {\n\t\t\t\t\t\tawait handleServerRequest(client, message as LspJsonRpcRequest);\n\t\t\t\t\t}\n\t\t\t\t} else if (\"method\" in message) {\n\t\t\t\t\tif (message.method === \"textDocument/publishDiagnostics\" && message.params) {\n\t\t\t\t\t\tconst params = message.params as { uri: string; diagnostics: Diagnostic[] };\n\t\t\t\t\t\tclient.diagnostics.set(params.uri, params.diagnostics);\n\t\t\t\t\t\tclient.diagnosticsVersion += 1;\n\t\t\t\t\t}\n\t\t\t\t}\n\n\t\t\t\tparsed = parseMessage(workingBuffer);\n\t\t\t}\n\n\t\t\tclient.messageBuffer = workingBuffer;\n\t\t});\n\n\t\tstdout.on(\"end\", () => {\n\t\t\tclient.isReading = false;\n\t\t\tresolve();\n\t\t});\n\n\t\tstdout.on(\"error\", () => {\n\t\t\tclient.isReading = false;\n\t\t\tresolve();\n\t\t});\n\t});\n}\n\n// =============================================================================\n// Server Request Handlers\n// =============================================================================\n\nasync function handleConfigurationRequest(client: LspClient, message: LspJsonRpcRequest): Promise<void> {\n\tif (typeof message.id !== \"number\") return;\n\tconst params = message.params as { items?: Array<{ section?: string }> };\n\tconst items = params?.items ?? [];\n\tconst result = items.map(item => {\n\t\tconst section = item.section ?? \"\";\n\t\treturn client.config.settings?.[section] ?? {};\n\t});\n\tawait sendResponse(client, message.id, result, \"workspace/configuration\");\n}\n\nasync function handleApplyEditRequest(client: LspClient, message: LspJsonRpcRequest): Promise<void> {\n\tif (typeof message.id !== \"number\") return;\n\tconst params = message.params as { edit?: WorkspaceEdit };\n\tif (!params?.edit) {\n\t\tawait sendResponse(\n\t\t\tclient,\n\t\t\tmessage.id,\n\t\t\t{ applied: false, failureReason: \"No edit provided\" },\n\t\t\t\"workspace/applyEdit\",\n\t\t);\n\t\treturn;\n\t}\n\n\ttry {\n\t\tawait applyWorkspaceEdit(params.edit, client.cwd);\n\t\tawait sendResponse(client, message.id, { applied: true }, \"workspace/applyEdit\");\n\t} catch (err: unknown) {\n\t\tawait sendResponse(client, message.id, { applied: false, failureReason: String(err) }, \"workspace/applyEdit\");\n\t}\n}\n\nasync function handleServerRequest(client: LspClient, message: LspJsonRpcRequest): Promise<void> {\n\tif (message.method === \"workspace/configuration\") {\n\t\tawait handleConfigurationRequest(client, message);\n\t\treturn;\n\t}\n\tif (message.method === \"workspace/applyEdit\") {\n\t\tawait handleApplyEditRequest(client, message);\n\t\treturn;\n\t}\n\tif (typeof message.id !== \"number\") return;\n\tawait sendResponse(client, message.id, null, message.method, {\n\t\tcode: -32601,\n\t\tmessage: `Method not found: ${message.method}`,\n\t});\n}\n\nasync function sendResponse(\n\tclient: LspClient,\n\tid: number,\n\tresult: unknown,\n\t_method: string,\n\terror?: { code: number; message: string; data?: unknown },\n): Promise<void> {\n\tconst response: LspJsonRpcResponse = {\n\t\tjsonrpc: \"2.0\",\n\t\tid,\n\t\t...(error ? { error } : { result }),\n\t};\n\n\ttry {\n\t\tawait writeMessage(client.proc.stdin, response);\n\t} catch {\n\t\t// Failed to respond to server request\n\t}\n}\n\n// =============================================================================\n// Stderr Buffer\n// =============================================================================\n\nasync function startStderrReader(client: LspClient): Promise<void> {\n\tconst stderr = client.proc.stderr;\n\tif (!stderr) return;\n\n\treturn new Promise<void>((resolve) => {\n\t\tstderr.on(\"data\", (chunk: Buffer) => {\n\t\t\tconst text = chunk.toString(\"utf-8\");\n\t\t\tclient.stderrBuffer += text;\n\t\t\tif (client.stderrBuffer.length > 4096) {\n\t\t\t\tclient.stderrBuffer = client.stderrBuffer.slice(-4096);\n\t\t\t}\n\t\t});\n\n\t\tstderr.on(\"end\", () => {\n\t\t\tresolve();\n\t\t});\n\n\t\tstderr.on(\"error\", () => {\n\t\t\tresolve();\n\t\t});\n\t});\n}\n\n// =============================================================================\n// Client Management\n// =============================================================================\n\n/** Timeout for warmup initialize requests (5 seconds) */\nexport const WARMUP_TIMEOUT_MS = 5000;\n\n/**\n * Get or create an LSP client for the given server configuration and working directory.\n */\nexport async function getOrCreateClient(config: ServerConfig, cwd: string, initTimeoutMs?: number): Promise<LspClient> {\n\tconst key = `${config.command}:${cwd}`;\n\n\tconst existingClient = clients.get(key);\n\tif (existingClient) {\n\t\texistingClient.lastActivity = Date.now();\n\t\treturn existingClient;\n\t}\n\n\tconst existingLock = clientLocks.get(key);\n\tif (existingLock) {\n\t\treturn existingLock;\n\t}\n\n\tconst clientPromise = (async () => {\n\t\tconst baseCommand = config.resolvedCommand ?? config.command;\n\t\tconst baseArgs = config.args ?? [];\n\n\t\t// Wrap with lspmux if available and supported\n\t\tconst { command, args, env } = isLspmuxSupported(baseCommand)\n\t\t\t? await getLspmuxCommand(baseCommand, baseArgs)\n\t\t\t: { command: baseCommand, args: baseArgs };\n\n\t\tconst proc = spawn(command, args, {\n\t\t\tcwd,\n\t\t\tstdio: [\"pipe\", \"pipe\", \"pipe\"],\n\t\t\tenv: env ? { ...process.env, ...env } : undefined,\n\t\t});\n\n\t\tconst exitedPromise = new Promise<number>((resolve) => {\n\t\t\tproc.on(\"exit\", (code: number | null) => resolve(code ?? 1));\n\t\t});\n\n\t\tconst client: LspClient = {\n\t\t\tname: key,\n\t\t\tcwd,\n\t\t\tproc: {\n\t\t\t\tstdin: proc.stdin,\n\t\t\t\tstdout: proc.stdout,\n\t\t\t\tstderr: proc.stderr,\n\t\t\t\tpid: proc.pid ?? 0,\n\t\t\t\texitCode: null,\n\t\t\t\texited: exitedPromise,\n\t\t\t\tkill: (signal?: number) => proc.kill(signal),\n\t\t\t},\n\t\t\tconfig,\n\t\t\trequestId: 0,\n\t\t\tdiagnostics: new Map(),\n\t\t\tdiagnosticsVersion: 0,\n\t\t\topenFiles: new Map(),\n\t\t\tpendingRequests: new Map(),\n\t\t\tmessageBuffer: Buffer.alloc(0),\n\t\t\tisReading: false,\n\t\t\tlastActivity: Date.now(),\n\t\t\tstderrBuffer: \"\",\n\t\t};\n\t\tclients.set(key, client);\n\n\t\t// Register crash recovery\n\t\texitedPromise.then((code: number) => {\n\t\t\tclient.proc.exitCode = code;\n\t\t\tclients.delete(key);\n\t\t\tclientLocks.delete(key);\n\n\t\t\tif (client.pendingRequests.size > 0) {\n\t\t\t\tconst stderr = client.stderrBuffer.trim();\n\t\t\t\tconst err = new Error(\n\t\t\t\t\tstderr ? `LSP server exited (code ${code}): ${stderr}` : `LSP server exited unexpectedly (code ${code})`,\n\t\t\t\t);\n\t\t\t\tfor (const pending of client.pendingRequests.values()) {\n\t\t\t\t\tpending.reject(err);\n\t\t\t\t}\n\t\t\t\tclient.pendingRequests.clear();\n\t\t\t}\n\t\t});\n\n\t\t// Start background readers\n\t\tstartMessageReader(client);\n\t\tstartStderrReader(client);\n\n\t\ttry {\n\t\t\tconst initResult = (await sendRequest(\n\t\t\t\tclient,\n\t\t\t\t\"initialize\",\n\t\t\t\t{\n\t\t\t\t\tprocessId: process.pid,\n\t\t\t\t\trootUri: fileToUri(cwd),\n\t\t\t\t\trootPath: cwd,\n\t\t\t\t\tcapabilities: CLIENT_CAPABILITIES,\n\t\t\t\t\tinitializationOptions: config.initOptions ?? {},\n\t\t\t\t\tworkspaceFolders: [{ uri: fileToUri(cwd), name: cwd.split(\"/\").pop() ?? \"workspace\" }],\n\t\t\t\t},\n\t\t\t\tundefined, // signal\n\t\t\t\tinitTimeoutMs,\n\t\t\t)) as { capabilities?: unknown };\n\n\t\t\tif (!initResult) {\n\t\t\t\tthrow new Error(\"Failed to initialize LSP: no response\");\n\t\t\t}\n\n\t\t\tclient.serverCapabilities = initResult.capabilities as LspClient[\"serverCapabilities\"];\n\n\t\t\tawait sendNotification(client, \"initialized\", {});\n\n\t\t\treturn client;\n\t\t} catch (err) {\n\t\t\tclients.delete(key);\n\t\t\tclientLocks.delete(key);\n\t\t\ttry {\n\t\t\t\tkillProcessTree(proc.pid ?? 0);\n\t\t\t} catch {\n\t\t\t\tproc.kill();\n\t\t\t}\n\t\t\tthrow err;\n\t\t} finally {\n\t\t\tclientLocks.delete(key);\n\t\t}\n\t})();\n\n\tclientLocks.set(key, clientPromise);\n\treturn clientPromise;\n}\n\n/**\n * Ensure a file is opened in the LSP client.\n */\nexport async function ensureFileOpen(client: LspClient, filePath: string, signal?: AbortSignal): Promise<void> {\n\tthrowIfAborted(signal);\n\tconst uri = fileToUri(filePath);\n\tconst lockKey = `${client.name}:${uri}`;\n\n\tif (client.openFiles.has(uri)) {\n\t\treturn;\n\t}\n\n\tconst existingLock = fileOperationLocks.get(lockKey);\n\tif (existingLock) {\n\t\tawait untilAborted(signal, () => existingLock);\n\t\treturn;\n\t}\n\n\tconst openPromise = (async () => {\n\t\tthrowIfAborted(signal);\n\t\tif (client.openFiles.has(uri)) {\n\t\t\treturn;\n\t\t}\n\n\t\tlet content: string;\n\t\ttry {\n\t\t\tcontent = await fsPromises.readFile(filePath, \"utf-8\");\n\t\t\tthrowIfAborted(signal);\n\t\t} catch (err: unknown) {\n\t\t\tif (isEnoent(err)) return;\n\t\t\tthrow err;\n\t\t}\n\t\tconst languageId = detectLanguageId(filePath);\n\t\tthrowIfAborted(signal);\n\n\t\tawait sendNotification(client, \"textDocument/didOpen\", {\n\t\t\ttextDocument: {\n\t\t\t\turi,\n\t\t\t\tlanguageId,\n\t\t\t\tversion: 1,\n\t\t\t\ttext: content,\n\t\t\t},\n\t\t});\n\n\t\tclient.openFiles.set(uri, { version: 1, languageId });\n\t\tclient.lastActivity = Date.now();\n\t})();\n\n\tfileOperationLocks.set(lockKey, openPromise);\n\ttry {\n\t\tawait openPromise;\n\t} finally {\n\t\tfileOperationLocks.delete(lockKey);\n\t}\n}\n\n/**\n * Sync in-memory content to the LSP client without reading from disk.\n */\nexport async function syncContent(\n\tclient: LspClient,\n\tfilePath: string,\n\tcontent: string,\n\tsignal?: AbortSignal,\n): Promise<void> {\n\tconst uri = fileToUri(filePath);\n\tconst lockKey = `${client.name}:${uri}`;\n\tthrowIfAborted(signal);\n\n\tconst existingLock = fileOperationLocks.get(lockKey);\n\tif (existingLock) {\n\t\tawait untilAborted(signal, () => existingLock);\n\t}\n\n\tconst syncPromise = (async () => {\n\t\tclient.diagnostics.delete(uri);\n\n\t\tconst info = client.openFiles.get(uri);\n\n\t\tif (!info) {\n\t\t\tconst languageId = detectLanguageId(filePath);\n\t\t\tthrowIfAborted(signal);\n\t\t\tawait sendNotification(client, \"textDocument/didOpen\", {\n\t\t\t\ttextDocument: {\n\t\t\t\t\turi,\n\t\t\t\t\tlanguageId,\n\t\t\t\t\tversion: 1,\n\t\t\t\t\ttext: content,\n\t\t\t\t},\n\t\t\t});\n\t\t\tclient.openFiles.set(uri, { version: 1, languageId });\n\t\t\tclient.lastActivity = Date.now();\n\t\t\treturn;\n\t\t}\n\n\t\tconst version = ++info.version;\n\t\tthrowIfAborted(signal);\n\t\tawait sendNotification(client, \"textDocument/didChange\", {\n\t\t\ttextDocument: { uri, version },\n\t\t\tcontentChanges: [{ text: content }],\n\t\t});\n\t\tclient.lastActivity = Date.now();\n\t})();\n\n\tfileOperationLocks.set(lockKey, syncPromise);\n\ttry {\n\t\tawait syncPromise;\n\t} finally {\n\t\tfileOperationLocks.delete(lockKey);\n\t}\n}\n\n/**\n * Notify LSP that a file was saved.\n */\nexport async function notifySaved(client: LspClient, filePath: string, signal?: AbortSignal): Promise<void> {\n\tconst uri = fileToUri(filePath);\n\tconst info = client.openFiles.get(uri);\n\tif (!info) return;\n\n\tthrowIfAborted(signal);\n\tawait sendNotification(client, \"textDocument/didSave\", {\n\t\ttextDocument: { uri },\n\t});\n\tclient.lastActivity = Date.now();\n}\n\n/**\n * Refresh a file in the LSP client.\n */\nexport async function refreshFile(client: LspClient, filePath: string, signal?: AbortSignal): Promise<void> {\n\tthrowIfAborted(signal);\n\tconst uri = fileToUri(filePath);\n\tconst lockKey = `${client.name}:${uri}`;\n\n\tconst existingLock = fileOperationLocks.get(lockKey);\n\tif (existingLock) {\n\t\tawait untilAborted(signal, () => existingLock);\n\t}\n\n\tconst refreshPromise = (async () => {\n\t\tthrowIfAborted(signal);\n\t\tconst info = client.openFiles.get(uri);\n\n\t\tif (!info) {\n\t\t\tawait ensureFileOpen(client, filePath, signal);\n\t\t\treturn;\n\t\t}\n\n\t\tlet content: string;\n\t\ttry {\n\t\t\tcontent = await fsPromises.readFile(filePath, \"utf-8\");\n\t\t\tthrowIfAborted(signal);\n\t\t} catch (err: unknown) {\n\t\t\tif (isEnoent(err)) return;\n\t\t\tthrow err;\n\t\t}\n\t\tconst version = ++info.version;\n\t\tthrowIfAborted(signal);\n\n\t\tawait sendNotification(client, \"textDocument/didChange\", {\n\t\t\ttextDocument: { uri, version },\n\t\t\tcontentChanges: [{ text: content }],\n\t\t});\n\t\tthrowIfAborted(signal);\n\n\t\tawait sendNotification(client, \"textDocument/didSave\", {\n\t\t\ttextDocument: { uri },\n\t\t\ttext: content,\n\t\t});\n\n\t\tclient.lastActivity = Date.now();\n\t})();\n\n\tfileOperationLocks.set(lockKey, refreshPromise);\n\ttry {\n\t\tawait refreshPromise;\n\t} finally {\n\t\tfileOperationLocks.delete(lockKey);\n\t}\n}\n\n/**\n * Shutdown a specific client by key.\n */\nexport function shutdownClient(key: string): void {\n\tconst client = clients.get(key);\n\tif (!client) return;\n\n\tfor (const pending of Array.from(client.pendingRequests.values())) {\n\t\tpending.reject(new Error(\"LSP client shutdown\"));\n\t}\n\tclient.pendingRequests.clear();\n\n\tsendRequest(client, \"shutdown\", null).catch(() => {});\n\n\ttry {\n\t\tkillProcessTree(client.proc.pid);\n\t} catch {\n\t\tclient.proc.kill();\n\t}\n\tclients.delete(key);\n}\n\n// =============================================================================\n// LSP Protocol Methods\n// =============================================================================\n\nconst DEFAULT_REQUEST_TIMEOUT_MS = 30000;\n\nexport async function sendRequest(\n\tclient: LspClient,\n\tmethod: string,\n\tparams: unknown,\n\tsignal?: AbortSignal,\n\ttimeoutMs: number = DEFAULT_REQUEST_TIMEOUT_MS,\n): Promise<unknown> {\n\tconst id = ++client.requestId;\n\tif (signal?.aborted) {\n\t\tconst reason = signal.reason instanceof Error ? signal.reason : new ToolAbortError();\n\t\treturn Promise.reject(reason);\n\t}\n\n\tconst request: LspJsonRpcRequest = {\n\t\tjsonrpc: \"2.0\",\n\t\tid,\n\t\tmethod,\n\t\tparams,\n\t};\n\n\tclient.lastActivity = Date.now();\n\n\tconst { promise, resolve, reject } = Promise.withResolvers<unknown>();\n\tlet timeout: NodeJS.Timeout | undefined;\n\tconst cleanup = () => {\n\t\tif (signal) {\n\t\t\tsignal.removeEventListener(\"abort\", abortHandler);\n\t\t}\n\t};\n\tconst abortHandler = () => {\n\t\tif (client.pendingRequests.has(id)) {\n\t\t\tclient.pendingRequests.delete(id);\n\t\t}\n\t\tvoid sendNotification(client, \"$/cancelRequest\", { id }).catch(() => {});\n\t\tif (timeout) clearTimeout(timeout);\n\t\tcleanup();\n\t\tconst reason = signal?.reason instanceof Error ? signal.reason : new ToolAbortError();\n\t\treject(reason);\n\t};\n\n\ttimeout = setTimeout(() => {\n\t\tif (client.pendingRequests.has(id)) {\n\t\t\tclient.pendingRequests.delete(id);\n\t\t\tconst err = new Error(`LSP request ${method} timed out after ${timeoutMs}ms`);\n\t\t\tcleanup();\n\t\t\treject(err);\n\t\t}\n\t}, timeoutMs);\n\tif (signal) {\n\t\tsignal.addEventListener(\"abort\", abortHandler, { once: true });\n\t\tif (signal.aborted) {\n\t\t\tabortHandler();\n\t\t\treturn promise;\n\t\t}\n\t}\n\n\tclient.pendingRequests.set(id, {\n\t\tresolve: (result: unknown) => {\n\t\t\tif (timeout) clearTimeout(timeout);\n\t\t\tcleanup();\n\t\t\tresolve(result);\n\t\t},\n\t\treject: (err: Error) => {\n\t\t\tif (timeout) clearTimeout(timeout);\n\t\t\tcleanup();\n\t\t\treject(err);\n\t\t},\n\t\tmethod,\n\t});\n\n\twriteMessage(client.proc.stdin, request).catch((err: Error) => {\n\t\tif (timeout) clearTimeout(timeout);\n\t\tclient.pendingRequests.delete(id);\n\t\tcleanup();\n\t\treject(err);\n\t});\n\treturn promise;\n}\n\nexport async function sendNotification(client: LspClient, method: string, params: unknown): Promise<void> {\n\tconst notification: LspJsonRpcNotification = {\n\t\tjsonrpc: \"2.0\",\n\t\tmethod,\n\t\tparams,\n\t};\n\n\tclient.lastActivity = Date.now();\n\tawait writeMessage(client.proc.stdin, notification);\n}\n\n/**\n * Shutdown all LSP clients.\n */\nexport function shutdownAll(): void {\n\tconst clientsToShutdown = Array.from(clients.values());\n\tclients.clear();\n\n\tconst err = new Error(\"LSP client shutdown\");\n\tfor (const client of clientsToShutdown) {\n\t\tconst reqs = Array.from(client.pendingRequests.values());\n\t\tclient.pendingRequests.clear();\n\t\tfor (const pending of reqs) {\n\t\t\tpending.reject(err);\n\t\t}\n\n\t\tvoid (async () => {\n\t\t\tconst timeout = new Promise<void>(resolve => setTimeout(resolve, 5_000));\n\t\t\tconst result = sendRequest(client, \"shutdown\", null).catch(() => {});\n\t\t\tawait Promise.race([result, timeout]);\n\t\t\ttry {\n\t\t\t\tkillProcessTree(client.proc.pid);\n\t\t\t} catch {\n\t\t\t\tclient.proc.kill();\n\t\t\t}\n\t\t})().catch(() => {});\n\t}\n}\n\n/** Status of an LSP server */\nexport interface LspServerStatus {\n\tname: string;\n\tstatus: \"connecting\" | \"ready\" | \"error\";\n\tfileTypes: string[];\n\terror?: string;\n}\n\nexport function getActiveClients(): LspServerStatus[] {\n\treturn Array.from(clients.values()).map(client => ({\n\t\tname: client.config.command,\n\t\tstatus: \"ready\" as const,\n\t\tfileTypes: client.config.fileTypes,\n\t}));\n}\n\n// =============================================================================\n// Process Cleanup\n// =============================================================================\n\nif (typeof process !== \"undefined\") {\n\tprocess.on(\"beforeExit\", shutdownAll);\n\tprocess.on(\"SIGINT\", () => {\n\t\tshutdownAll();\n\t\tprocess.exit(0);\n\t});\n\tprocess.on(\"SIGTERM\", () => {\n\t\tshutdownAll();\n\t\tprocess.exit(0);\n\t});\n}\n"]}
|
|
@@ -0,0 +1,23 @@
|
|
|
1
|
+
import type { ServerConfig } from "./types.js";
|
|
2
|
+
export interface LspConfig {
|
|
3
|
+
servers: Record<string, ServerConfig>;
|
|
4
|
+
/** Idle timeout in milliseconds. If set, LSP clients will be shutdown after this period of inactivity. Disabled by default. */
|
|
5
|
+
idleTimeoutMs?: number;
|
|
6
|
+
}
|
|
7
|
+
export declare function hasRootMarkers(cwd: string, markers: string[]): boolean;
|
|
8
|
+
export declare function resolveCommand(command: string, cwd: string): string | null;
|
|
9
|
+
/**
|
|
10
|
+
* Load LSP configuration.
|
|
11
|
+
*
|
|
12
|
+
* Priority (highest to lowest):
|
|
13
|
+
* 1. Project root: lsp.json/.lsp.json/lsp.yml/.lsp.yml/lsp.yaml/.lsp.yaml
|
|
14
|
+
* 2. Project config dir: {CONFIG_DIR_NAME}/lsp.* (+ hidden variants)
|
|
15
|
+
* 3. User config dir: ~/{CONFIG_DIR_NAME}/agent/lsp.* (+ hidden variants)
|
|
16
|
+
* 4. User home root: ~/lsp.*, ~/.lsp.*
|
|
17
|
+
* 5. Auto-detect from project markers + available binaries
|
|
18
|
+
*/
|
|
19
|
+
export declare function loadConfig(cwd: string): LspConfig;
|
|
20
|
+
export declare function getServersForFile(config: LspConfig, filePath: string): Array<[string, ServerConfig]>;
|
|
21
|
+
export declare function getServerForFile(config: LspConfig, filePath: string): [string, ServerConfig] | null;
|
|
22
|
+
export declare function hasCapability(config: ServerConfig, capability: keyof NonNullable<ServerConfig["capabilities"]>): boolean;
|
|
23
|
+
//# sourceMappingURL=config.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"config.d.ts","sourceRoot":"","sources":["../../../src/core/lsp/config.ts"],"names":[],"mappings":"AASA,OAAO,KAAK,EAAE,YAAY,EAAE,MAAM,YAAY,CAAC;AAK/C,MAAM,WAAW,SAAS;IACzB,OAAO,EAAE,MAAM,CAAC,MAAM,EAAE,YAAY,CAAC,CAAC;IACtC,+HAA+H;IAC/H,aAAa,CAAC,EAAE,MAAM,CAAC;CACvB;AA6HD,wBAAgB,cAAc,CAAC,GAAG,EAAE,MAAM,EAAE,OAAO,EAAE,MAAM,EAAE,GAAG,OAAO,CAmBtE;AAsBD,wBAAgB,cAAc,CAAC,OAAO,EAAE,MAAM,EAAE,GAAG,EAAE,MAAM,GAAG,MAAM,GAAG,IAAI,CAW1E;AAkCD;;;;;;;;;GASG;AACH,wBAAgB,UAAU,CAAC,GAAG,EAAE,MAAM,GAAG,SAAS,CA6CjD;AAMD,wBAAgB,iBAAiB,CAAC,MAAM,EAAE,SAAS,EAAE,QAAQ,EAAE,MAAM,GAAG,KAAK,CAAC,CAAC,MAAM,EAAE,YAAY,CAAC,CAAC,CAsBpG;AAED,wBAAgB,gBAAgB,CAAC,MAAM,EAAE,SAAS,EAAE,QAAQ,EAAE,MAAM,GAAG,CAAC,MAAM,EAAE,YAAY,CAAC,GAAG,IAAI,CAGnG;AAED,wBAAgB,aAAa,CAC5B,MAAM,EAAE,YAAY,EACpB,UAAU,EAAE,MAAM,WAAW,CAAC,YAAY,CAAC,cAAc,CAAC,CAAC,GACzD,OAAO,CAET"}
|
|
@@ -0,0 +1,267 @@
|
|
|
1
|
+
import * as fs from "node:fs";
|
|
2
|
+
import { createRequire } from "node:module";
|
|
3
|
+
import * as os from "node:os";
|
|
4
|
+
import * as path from "node:path";
|
|
5
|
+
import { spawnSync } from "node:child_process";
|
|
6
|
+
import YAML from "yaml";
|
|
7
|
+
import { globSync } from "glob";
|
|
8
|
+
import { CONFIG_DIR_NAME } from "../../config.js";
|
|
9
|
+
import { isRecord } from "./helpers.js";
|
|
10
|
+
const require = createRequire(import.meta.url);
|
|
11
|
+
const DEFAULTS = require("./defaults.json");
|
|
12
|
+
// =============================================================================
|
|
13
|
+
// Default Server Configuration Loading
|
|
14
|
+
// =============================================================================
|
|
15
|
+
const PID_TOKEN = "$PID";
|
|
16
|
+
function parseConfigContent(content, filePath) {
|
|
17
|
+
const extension = path.extname(filePath).toLowerCase();
|
|
18
|
+
if (extension === ".yaml" || extension === ".yml") {
|
|
19
|
+
return YAML.parse(content);
|
|
20
|
+
}
|
|
21
|
+
return JSON.parse(content);
|
|
22
|
+
}
|
|
23
|
+
function normalizeConfig(value) {
|
|
24
|
+
if (!isRecord(value))
|
|
25
|
+
return null;
|
|
26
|
+
const idleTimeoutMs = typeof value.idleTimeoutMs === "number" ? value.idleTimeoutMs : undefined;
|
|
27
|
+
const rawServers = value.servers;
|
|
28
|
+
if (isRecord(rawServers)) {
|
|
29
|
+
return { servers: rawServers, idleTimeoutMs };
|
|
30
|
+
}
|
|
31
|
+
const servers = Object.fromEntries(Object.entries(value).filter(([key]) => key !== "idleTimeoutMs"));
|
|
32
|
+
return { servers, idleTimeoutMs };
|
|
33
|
+
}
|
|
34
|
+
function normalizeStringArray(value) {
|
|
35
|
+
if (!Array.isArray(value))
|
|
36
|
+
return null;
|
|
37
|
+
const items = value.filter((entry) => typeof entry === "string" && entry.length > 0);
|
|
38
|
+
return items.length > 0 ? items : null;
|
|
39
|
+
}
|
|
40
|
+
function normalizeServerConfig(name, config) {
|
|
41
|
+
const command = typeof config.command === "string" && config.command.length > 0 ? config.command : null;
|
|
42
|
+
const fileTypes = normalizeStringArray(config.fileTypes);
|
|
43
|
+
const rootMarkers = normalizeStringArray(config.rootMarkers);
|
|
44
|
+
if (!command || !fileTypes || !rootMarkers) {
|
|
45
|
+
return null;
|
|
46
|
+
}
|
|
47
|
+
const args = Array.isArray(config.args)
|
|
48
|
+
? config.args.filter((entry) => typeof entry === "string")
|
|
49
|
+
: undefined;
|
|
50
|
+
return {
|
|
51
|
+
...config,
|
|
52
|
+
command,
|
|
53
|
+
args,
|
|
54
|
+
fileTypes,
|
|
55
|
+
rootMarkers,
|
|
56
|
+
};
|
|
57
|
+
}
|
|
58
|
+
function readConfigFile(filePath) {
|
|
59
|
+
try {
|
|
60
|
+
const content = fs.readFileSync(filePath, "utf-8");
|
|
61
|
+
const parsed = parseConfigContent(content, filePath);
|
|
62
|
+
return normalizeConfig(parsed);
|
|
63
|
+
}
|
|
64
|
+
catch {
|
|
65
|
+
return null;
|
|
66
|
+
}
|
|
67
|
+
}
|
|
68
|
+
function coerceServerConfigs(servers) {
|
|
69
|
+
const result = {};
|
|
70
|
+
for (const [name, config] of Object.entries(servers)) {
|
|
71
|
+
const normalized = normalizeServerConfig(name, config);
|
|
72
|
+
if (normalized) {
|
|
73
|
+
result[name] = normalized;
|
|
74
|
+
}
|
|
75
|
+
}
|
|
76
|
+
return result;
|
|
77
|
+
}
|
|
78
|
+
function mergeServers(base, overrides) {
|
|
79
|
+
const merged = { ...base };
|
|
80
|
+
for (const [name, config] of Object.entries(overrides)) {
|
|
81
|
+
if (merged[name]) {
|
|
82
|
+
const candidate = { ...merged[name], ...config };
|
|
83
|
+
const normalized = normalizeServerConfig(name, candidate);
|
|
84
|
+
if (normalized) {
|
|
85
|
+
merged[name] = normalized;
|
|
86
|
+
}
|
|
87
|
+
}
|
|
88
|
+
else {
|
|
89
|
+
const normalized = normalizeServerConfig(name, config);
|
|
90
|
+
if (normalized) {
|
|
91
|
+
merged[name] = normalized;
|
|
92
|
+
}
|
|
93
|
+
}
|
|
94
|
+
}
|
|
95
|
+
return merged;
|
|
96
|
+
}
|
|
97
|
+
function applyRuntimeDefaults(servers) {
|
|
98
|
+
const updated = { ...servers };
|
|
99
|
+
if (updated.omnisharp?.args) {
|
|
100
|
+
const args = updated.omnisharp.args.map((arg) => (arg === PID_TOKEN ? String(process.pid) : arg));
|
|
101
|
+
updated.omnisharp = { ...updated.omnisharp, args };
|
|
102
|
+
}
|
|
103
|
+
return updated;
|
|
104
|
+
}
|
|
105
|
+
// =============================================================================
|
|
106
|
+
// Configuration Loading
|
|
107
|
+
// =============================================================================
|
|
108
|
+
export function hasRootMarkers(cwd, markers) {
|
|
109
|
+
for (const marker of markers) {
|
|
110
|
+
if (marker.includes("*")) {
|
|
111
|
+
try {
|
|
112
|
+
const matches = globSync(marker, { cwd, nodir: false });
|
|
113
|
+
if (matches.length > 0) {
|
|
114
|
+
return true;
|
|
115
|
+
}
|
|
116
|
+
}
|
|
117
|
+
catch {
|
|
118
|
+
// Failed to resolve glob root marker
|
|
119
|
+
}
|
|
120
|
+
continue;
|
|
121
|
+
}
|
|
122
|
+
const filePath = path.join(cwd, marker);
|
|
123
|
+
if (fs.existsSync(filePath)) {
|
|
124
|
+
return true;
|
|
125
|
+
}
|
|
126
|
+
}
|
|
127
|
+
return false;
|
|
128
|
+
}
|
|
129
|
+
// =============================================================================
|
|
130
|
+
// Local Binary Resolution
|
|
131
|
+
// =============================================================================
|
|
132
|
+
const LOCAL_BIN_PATHS = [
|
|
133
|
+
{ markers: ["package.json", "package-lock.json", "yarn.lock", "pnpm-lock.yaml"], binDir: "node_modules/.bin" },
|
|
134
|
+
{ markers: ["pyproject.toml", "requirements.txt", "setup.py", "Pipfile"], binDir: ".venv/bin" },
|
|
135
|
+
{ markers: ["pyproject.toml", "requirements.txt", "setup.py", "Pipfile"], binDir: "venv/bin" },
|
|
136
|
+
{ markers: ["pyproject.toml", "requirements.txt", "setup.py", "Pipfile"], binDir: ".env/bin" },
|
|
137
|
+
{ markers: ["Gemfile", "Gemfile.lock"], binDir: "vendor/bundle/bin" },
|
|
138
|
+
{ markers: ["Gemfile", "Gemfile.lock"], binDir: "bin" },
|
|
139
|
+
{ markers: ["go.mod", "go.sum"], binDir: "bin" },
|
|
140
|
+
];
|
|
141
|
+
function which(command) {
|
|
142
|
+
const result = spawnSync("which", [command], { encoding: "utf-8" });
|
|
143
|
+
if (result.status !== 0)
|
|
144
|
+
return null;
|
|
145
|
+
return result.stdout.trim() || null;
|
|
146
|
+
}
|
|
147
|
+
export function resolveCommand(command, cwd) {
|
|
148
|
+
for (const { markers, binDir } of LOCAL_BIN_PATHS) {
|
|
149
|
+
if (hasRootMarkers(cwd, markers)) {
|
|
150
|
+
const localPath = path.join(cwd, binDir, command);
|
|
151
|
+
if (fs.existsSync(localPath)) {
|
|
152
|
+
return localPath;
|
|
153
|
+
}
|
|
154
|
+
}
|
|
155
|
+
}
|
|
156
|
+
return which(command);
|
|
157
|
+
}
|
|
158
|
+
/**
|
|
159
|
+
* Configuration file search paths (in priority order).
|
|
160
|
+
*/
|
|
161
|
+
function getConfigPaths(cwd) {
|
|
162
|
+
const filenames = ["lsp.json", ".lsp.json", "lsp.yaml", ".lsp.yaml", "lsp.yml", ".lsp.yml"];
|
|
163
|
+
const paths = [];
|
|
164
|
+
// Project root files (highest priority)
|
|
165
|
+
for (const filename of filenames) {
|
|
166
|
+
paths.push(path.join(cwd, filename));
|
|
167
|
+
}
|
|
168
|
+
// Project config directory
|
|
169
|
+
const projectConfigDir = path.join(cwd, CONFIG_DIR_NAME);
|
|
170
|
+
for (const filename of filenames) {
|
|
171
|
+
paths.push(path.join(projectConfigDir, filename));
|
|
172
|
+
}
|
|
173
|
+
// User config directory
|
|
174
|
+
const userConfigDir = path.join(os.homedir(), CONFIG_DIR_NAME, "agent");
|
|
175
|
+
for (const filename of filenames) {
|
|
176
|
+
paths.push(path.join(userConfigDir, filename));
|
|
177
|
+
}
|
|
178
|
+
// User home root files (lowest priority fallback)
|
|
179
|
+
for (const filename of filenames) {
|
|
180
|
+
paths.push(path.join(os.homedir(), filename));
|
|
181
|
+
}
|
|
182
|
+
return paths;
|
|
183
|
+
}
|
|
184
|
+
/**
|
|
185
|
+
* Load LSP configuration.
|
|
186
|
+
*
|
|
187
|
+
* Priority (highest to lowest):
|
|
188
|
+
* 1. Project root: lsp.json/.lsp.json/lsp.yml/.lsp.yml/lsp.yaml/.lsp.yaml
|
|
189
|
+
* 2. Project config dir: {CONFIG_DIR_NAME}/lsp.* (+ hidden variants)
|
|
190
|
+
* 3. User config dir: ~/{CONFIG_DIR_NAME}/agent/lsp.* (+ hidden variants)
|
|
191
|
+
* 4. User home root: ~/lsp.*, ~/.lsp.*
|
|
192
|
+
* 5. Auto-detect from project markers + available binaries
|
|
193
|
+
*/
|
|
194
|
+
export function loadConfig(cwd) {
|
|
195
|
+
let mergedServers = coerceServerConfigs(DEFAULTS);
|
|
196
|
+
const configPaths = getConfigPaths(cwd).reverse();
|
|
197
|
+
let hasOverrides = false;
|
|
198
|
+
let idleTimeoutMs;
|
|
199
|
+
for (const configPath of configPaths) {
|
|
200
|
+
const parsed = readConfigFile(configPath);
|
|
201
|
+
if (!parsed)
|
|
202
|
+
continue;
|
|
203
|
+
const hasServerOverrides = Object.keys(parsed.servers).length > 0;
|
|
204
|
+
if (hasServerOverrides) {
|
|
205
|
+
hasOverrides = true;
|
|
206
|
+
mergedServers = mergeServers(mergedServers, parsed.servers);
|
|
207
|
+
}
|
|
208
|
+
if (parsed.idleTimeoutMs !== undefined) {
|
|
209
|
+
idleTimeoutMs = parsed.idleTimeoutMs;
|
|
210
|
+
}
|
|
211
|
+
}
|
|
212
|
+
if (!hasOverrides) {
|
|
213
|
+
const detected = {};
|
|
214
|
+
const defaultsWithRuntime = applyRuntimeDefaults(mergedServers);
|
|
215
|
+
for (const [name, config] of Object.entries(defaultsWithRuntime)) {
|
|
216
|
+
if (!hasRootMarkers(cwd, config.rootMarkers))
|
|
217
|
+
continue;
|
|
218
|
+
const resolved = resolveCommand(config.command, cwd);
|
|
219
|
+
if (!resolved)
|
|
220
|
+
continue;
|
|
221
|
+
detected[name] = { ...config, resolvedCommand: resolved };
|
|
222
|
+
}
|
|
223
|
+
return { servers: detected, idleTimeoutMs };
|
|
224
|
+
}
|
|
225
|
+
const mergedWithRuntime = applyRuntimeDefaults(mergedServers);
|
|
226
|
+
const available = {};
|
|
227
|
+
for (const [name, config] of Object.entries(mergedWithRuntime)) {
|
|
228
|
+
if (config.disabled)
|
|
229
|
+
continue;
|
|
230
|
+
const resolved = resolveCommand(config.command, cwd);
|
|
231
|
+
if (!resolved)
|
|
232
|
+
continue;
|
|
233
|
+
available[name] = { ...config, resolvedCommand: resolved };
|
|
234
|
+
}
|
|
235
|
+
return { servers: available, idleTimeoutMs };
|
|
236
|
+
}
|
|
237
|
+
// =============================================================================
|
|
238
|
+
// Server Selection
|
|
239
|
+
// =============================================================================
|
|
240
|
+
export function getServersForFile(config, filePath) {
|
|
241
|
+
const ext = path.extname(filePath).toLowerCase();
|
|
242
|
+
const fileName = path.basename(filePath).toLowerCase();
|
|
243
|
+
const matches = [];
|
|
244
|
+
for (const [name, serverConfig] of Object.entries(config.servers)) {
|
|
245
|
+
const supportsFile = serverConfig.fileTypes.some(fileType => {
|
|
246
|
+
const normalized = fileType.toLowerCase();
|
|
247
|
+
return normalized === ext || normalized === fileName;
|
|
248
|
+
});
|
|
249
|
+
if (supportsFile) {
|
|
250
|
+
matches.push([name, serverConfig]);
|
|
251
|
+
}
|
|
252
|
+
}
|
|
253
|
+
// Sort: primary servers (non-linters) first, then linters
|
|
254
|
+
return matches.sort((a, b) => {
|
|
255
|
+
const aIsLinter = a[1].isLinter ? 1 : 0;
|
|
256
|
+
const bIsLinter = b[1].isLinter ? 1 : 0;
|
|
257
|
+
return aIsLinter - bIsLinter;
|
|
258
|
+
});
|
|
259
|
+
}
|
|
260
|
+
export function getServerForFile(config, filePath) {
|
|
261
|
+
const servers = getServersForFile(config, filePath);
|
|
262
|
+
return servers.length > 0 ? servers[0] : null;
|
|
263
|
+
}
|
|
264
|
+
export function hasCapability(config, capability) {
|
|
265
|
+
return config.capabilities?.[capability] === true;
|
|
266
|
+
}
|
|
267
|
+
//# sourceMappingURL=config.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"config.js","sourceRoot":"","sources":["../../../src/core/lsp/config.ts"],"names":[],"mappings":"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","sourcesContent":["import * as fs from \"node:fs\";\nimport { createRequire } from \"node:module\";\nimport * as os from \"node:os\";\nimport * as path from \"node:path\";\nimport { spawnSync } from \"node:child_process\";\nimport YAML from \"yaml\";\nimport { globSync } from \"glob\";\nimport { CONFIG_DIR_NAME } from \"../../config.js\";\nimport { isRecord } from \"./helpers.js\";\nimport type { ServerConfig } from \"./types.js\";\n\nconst require = createRequire(import.meta.url);\nconst DEFAULTS = require(\"./defaults.json\") as Record<string, Partial<ServerConfig>>;\n\nexport interface LspConfig {\n\tservers: Record<string, ServerConfig>;\n\t/** Idle timeout in milliseconds. If set, LSP clients will be shutdown after this period of inactivity. Disabled by default. */\n\tidleTimeoutMs?: number;\n}\n\n// =============================================================================\n// Default Server Configuration Loading\n// =============================================================================\n\nconst PID_TOKEN = \"$PID\";\n\ninterface NormalizedConfig {\n\tservers: Record<string, Partial<ServerConfig>>;\n\tidleTimeoutMs?: number;\n}\n\nfunction parseConfigContent(content: string, filePath: string): unknown {\n\tconst extension = path.extname(filePath).toLowerCase();\n\tif (extension === \".yaml\" || extension === \".yml\") {\n\t\treturn YAML.parse(content) as unknown;\n\t}\n\treturn JSON.parse(content) as unknown;\n}\n\nfunction normalizeConfig(value: unknown): NormalizedConfig | null {\n\tif (!isRecord(value)) return null;\n\n\tconst idleTimeoutMs = typeof value.idleTimeoutMs === \"number\" ? value.idleTimeoutMs : undefined;\n\tconst rawServers = value.servers;\n\n\tif (isRecord(rawServers)) {\n\t\treturn { servers: rawServers as Record<string, Partial<ServerConfig>>, idleTimeoutMs };\n\t}\n\n\tconst servers = Object.fromEntries(Object.entries(value).filter(([key]) => key !== \"idleTimeoutMs\")) as Record<\n\t\tstring,\n\t\tPartial<ServerConfig>\n\t>;\n\n\treturn { servers, idleTimeoutMs };\n}\n\nfunction normalizeStringArray(value: unknown): string[] | null {\n\tif (!Array.isArray(value)) return null;\n\tconst items = value.filter((entry): entry is string => typeof entry === \"string\" && entry.length > 0);\n\treturn items.length > 0 ? items : null;\n}\n\nfunction normalizeServerConfig(name: string, config: Partial<ServerConfig>): ServerConfig | null {\n\tconst command = typeof config.command === \"string\" && config.command.length > 0 ? config.command : null;\n\tconst fileTypes = normalizeStringArray(config.fileTypes);\n\tconst rootMarkers = normalizeStringArray(config.rootMarkers);\n\n\tif (!command || !fileTypes || !rootMarkers) {\n\t\treturn null;\n\t}\n\n\tconst args = Array.isArray(config.args)\n\t\t? config.args.filter((entry): entry is string => typeof entry === \"string\")\n\t\t: undefined;\n\n\treturn {\n\t\t...config,\n\t\tcommand,\n\t\targs,\n\t\tfileTypes,\n\t\trootMarkers,\n\t};\n}\n\nfunction readConfigFile(filePath: string): NormalizedConfig | null {\n\ttry {\n\t\tconst content = fs.readFileSync(filePath, \"utf-8\");\n\t\tconst parsed = parseConfigContent(content, filePath);\n\t\treturn normalizeConfig(parsed);\n\t} catch {\n\t\treturn null;\n\t}\n}\n\nfunction coerceServerConfigs(servers: Record<string, Partial<ServerConfig>>): Record<string, ServerConfig> {\n\tconst result: Record<string, ServerConfig> = {};\n\tfor (const [name, config] of Object.entries(servers)) {\n\t\tconst normalized = normalizeServerConfig(name, config);\n\t\tif (normalized) {\n\t\t\tresult[name] = normalized;\n\t\t}\n\t}\n\treturn result;\n}\n\nfunction mergeServers(\n\tbase: Record<string, ServerConfig>,\n\toverrides: Record<string, Partial<ServerConfig>>,\n): Record<string, ServerConfig> {\n\tconst merged: Record<string, ServerConfig> = { ...base };\n\tfor (const [name, config] of Object.entries(overrides)) {\n\t\tif (merged[name]) {\n\t\t\tconst candidate = { ...merged[name], ...config };\n\t\t\tconst normalized = normalizeServerConfig(name, candidate);\n\t\t\tif (normalized) {\n\t\t\t\tmerged[name] = normalized;\n\t\t\t}\n\t\t} else {\n\t\t\tconst normalized = normalizeServerConfig(name, config);\n\t\t\tif (normalized) {\n\t\t\t\tmerged[name] = normalized;\n\t\t\t}\n\t\t}\n\t}\n\treturn merged;\n}\n\nfunction applyRuntimeDefaults(servers: Record<string, ServerConfig>): Record<string, ServerConfig> {\n\tconst updated: Record<string, ServerConfig> = { ...servers };\n\n\tif (updated.omnisharp?.args) {\n\t\tconst args = updated.omnisharp.args.map((arg: string) => (arg === PID_TOKEN ? String(process.pid) : arg));\n\t\tupdated.omnisharp = { ...updated.omnisharp, args };\n\t}\n\n\treturn updated;\n}\n\n// =============================================================================\n// Configuration Loading\n// =============================================================================\n\nexport function hasRootMarkers(cwd: string, markers: string[]): boolean {\n\tfor (const marker of markers) {\n\t\tif (marker.includes(\"*\")) {\n\t\t\ttry {\n\t\t\t\tconst matches = globSync(marker, { cwd, nodir: false });\n\t\t\t\tif (matches.length > 0) {\n\t\t\t\t\treturn true;\n\t\t\t\t}\n\t\t\t} catch {\n\t\t\t\t// Failed to resolve glob root marker\n\t\t\t}\n\t\t\tcontinue;\n\t\t}\n\t\tconst filePath = path.join(cwd, marker);\n\t\tif (fs.existsSync(filePath)) {\n\t\t\treturn true;\n\t\t}\n\t}\n\treturn false;\n}\n\n// =============================================================================\n// Local Binary Resolution\n// =============================================================================\n\nconst LOCAL_BIN_PATHS: Array<{ markers: string[]; binDir: string }> = [\n\t{ markers: [\"package.json\", \"package-lock.json\", \"yarn.lock\", \"pnpm-lock.yaml\"], binDir: \"node_modules/.bin\" },\n\t{ markers: [\"pyproject.toml\", \"requirements.txt\", \"setup.py\", \"Pipfile\"], binDir: \".venv/bin\" },\n\t{ markers: [\"pyproject.toml\", \"requirements.txt\", \"setup.py\", \"Pipfile\"], binDir: \"venv/bin\" },\n\t{ markers: [\"pyproject.toml\", \"requirements.txt\", \"setup.py\", \"Pipfile\"], binDir: \".env/bin\" },\n\t{ markers: [\"Gemfile\", \"Gemfile.lock\"], binDir: \"vendor/bundle/bin\" },\n\t{ markers: [\"Gemfile\", \"Gemfile.lock\"], binDir: \"bin\" },\n\t{ markers: [\"go.mod\", \"go.sum\"], binDir: \"bin\" },\n];\n\nfunction which(command: string): string | null {\n\tconst result = spawnSync(\"which\", [command], { encoding: \"utf-8\" });\n\tif (result.status !== 0) return null;\n\treturn result.stdout.trim() || null;\n}\n\nexport function resolveCommand(command: string, cwd: string): string | null {\n\tfor (const { markers, binDir } of LOCAL_BIN_PATHS) {\n\t\tif (hasRootMarkers(cwd, markers)) {\n\t\t\tconst localPath = path.join(cwd, binDir, command);\n\t\t\tif (fs.existsSync(localPath)) {\n\t\t\t\treturn localPath;\n\t\t\t}\n\t\t}\n\t}\n\n\treturn which(command);\n}\n\n/**\n * Configuration file search paths (in priority order).\n */\nfunction getConfigPaths(cwd: string): string[] {\n\tconst filenames = [\"lsp.json\", \".lsp.json\", \"lsp.yaml\", \".lsp.yaml\", \"lsp.yml\", \".lsp.yml\"];\n\tconst paths: string[] = [];\n\n\t// Project root files (highest priority)\n\tfor (const filename of filenames) {\n\t\tpaths.push(path.join(cwd, filename));\n\t}\n\n\t// Project config directory\n\tconst projectConfigDir = path.join(cwd, CONFIG_DIR_NAME);\n\tfor (const filename of filenames) {\n\t\tpaths.push(path.join(projectConfigDir, filename));\n\t}\n\n\t// User config directory\n\tconst userConfigDir = path.join(os.homedir(), CONFIG_DIR_NAME, \"agent\");\n\tfor (const filename of filenames) {\n\t\tpaths.push(path.join(userConfigDir, filename));\n\t}\n\n\t// User home root files (lowest priority fallback)\n\tfor (const filename of filenames) {\n\t\tpaths.push(path.join(os.homedir(), filename));\n\t}\n\n\treturn paths;\n}\n\n/**\n * Load LSP configuration.\n *\n * Priority (highest to lowest):\n * 1. Project root: lsp.json/.lsp.json/lsp.yml/.lsp.yml/lsp.yaml/.lsp.yaml\n * 2. Project config dir: {CONFIG_DIR_NAME}/lsp.* (+ hidden variants)\n * 3. User config dir: ~/{CONFIG_DIR_NAME}/agent/lsp.* (+ hidden variants)\n * 4. User home root: ~/lsp.*, ~/.lsp.*\n * 5. Auto-detect from project markers + available binaries\n */\nexport function loadConfig(cwd: string): LspConfig {\n\tlet mergedServers = coerceServerConfigs(DEFAULTS);\n\n\tconst configPaths = getConfigPaths(cwd).reverse();\n\tlet hasOverrides = false;\n\n\tlet idleTimeoutMs: number | undefined;\n\tfor (const configPath of configPaths) {\n\t\tconst parsed = readConfigFile(configPath);\n\t\tif (!parsed) continue;\n\t\tconst hasServerOverrides = Object.keys(parsed.servers).length > 0;\n\t\tif (hasServerOverrides) {\n\t\t\thasOverrides = true;\n\t\t\tmergedServers = mergeServers(mergedServers, parsed.servers);\n\t\t}\n\t\tif (parsed.idleTimeoutMs !== undefined) {\n\t\t\tidleTimeoutMs = parsed.idleTimeoutMs;\n\t\t}\n\t}\n\n\tif (!hasOverrides) {\n\t\tconst detected: Record<string, ServerConfig> = {};\n\t\tconst defaultsWithRuntime = applyRuntimeDefaults(mergedServers);\n\n\t\tfor (const [name, config] of Object.entries(defaultsWithRuntime)) {\n\t\t\tif (!hasRootMarkers(cwd, config.rootMarkers)) continue;\n\t\t\tconst resolved = resolveCommand(config.command, cwd);\n\t\t\tif (!resolved) continue;\n\t\t\tdetected[name] = { ...config, resolvedCommand: resolved };\n\t\t}\n\n\t\treturn { servers: detected, idleTimeoutMs };\n\t}\n\n\tconst mergedWithRuntime = applyRuntimeDefaults(mergedServers);\n\tconst available: Record<string, ServerConfig> = {};\n\n\tfor (const [name, config] of Object.entries(mergedWithRuntime)) {\n\t\tif (config.disabled) continue;\n\t\tconst resolved = resolveCommand(config.command, cwd);\n\t\tif (!resolved) continue;\n\t\tavailable[name] = { ...config, resolvedCommand: resolved };\n\t}\n\n\treturn { servers: available, idleTimeoutMs };\n}\n\n// =============================================================================\n// Server Selection\n// =============================================================================\n\nexport function getServersForFile(config: LspConfig, filePath: string): Array<[string, ServerConfig]> {\n\tconst ext = path.extname(filePath).toLowerCase();\n\tconst fileName = path.basename(filePath).toLowerCase();\n\tconst matches: Array<[string, ServerConfig]> = [];\n\n\tfor (const [name, serverConfig] of Object.entries(config.servers)) {\n\t\tconst supportsFile = serverConfig.fileTypes.some(fileType => {\n\t\t\tconst normalized = fileType.toLowerCase();\n\t\t\treturn normalized === ext || normalized === fileName;\n\t\t});\n\n\t\tif (supportsFile) {\n\t\t\tmatches.push([name, serverConfig]);\n\t\t}\n\t}\n\n\t// Sort: primary servers (non-linters) first, then linters\n\treturn matches.sort((a, b) => {\n\t\tconst aIsLinter = a[1].isLinter ? 1 : 0;\n\t\tconst bIsLinter = b[1].isLinter ? 1 : 0;\n\t\treturn aIsLinter - bIsLinter;\n\t});\n}\n\nexport function getServerForFile(config: LspConfig, filePath: string): [string, ServerConfig] | null {\n\tconst servers = getServersForFile(config, filePath);\n\treturn servers.length > 0 ? servers[0] : null;\n}\n\nexport function hasCapability(\n\tconfig: ServerConfig,\n\tcapability: keyof NonNullable<ServerConfig[\"capabilities\"]>,\n): boolean {\n\treturn config.capabilities?.[capability] === true;\n}\n"]}
|
|
@@ -0,0 +1,17 @@
|
|
|
1
|
+
import type { TextEdit, WorkspaceEdit } from "./types.js";
|
|
2
|
+
/**
|
|
3
|
+
* Apply text edits to a string in-memory.
|
|
4
|
+
* Edits are applied in reverse order (bottom-to-top) to preserve line/character indices.
|
|
5
|
+
*/
|
|
6
|
+
export declare function applyTextEditsToString(content: string, edits: TextEdit[]): string;
|
|
7
|
+
/**
|
|
8
|
+
* Apply text edits to a file.
|
|
9
|
+
* Edits are applied in reverse order (bottom-to-top) to preserve line/character indices.
|
|
10
|
+
*/
|
|
11
|
+
export declare function applyTextEdits(filePath: string, edits: TextEdit[]): Promise<void>;
|
|
12
|
+
/**
|
|
13
|
+
* Apply a workspace edit (collection of file changes).
|
|
14
|
+
* Returns array of applied change descriptions.
|
|
15
|
+
*/
|
|
16
|
+
export declare function applyWorkspaceEdit(edit: WorkspaceEdit, cwd: string): Promise<string[]>;
|
|
17
|
+
//# sourceMappingURL=edits.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"edits.d.ts","sourceRoot":"","sources":["../../../src/core/lsp/edits.ts"],"names":[],"mappings":"AAEA,OAAO,KAAK,EAAwD,QAAQ,EAAE,aAAa,EAAE,MAAM,YAAY,CAAC;AAOhH;;;GAGG;AACH,wBAAgB,sBAAsB,CAAC,OAAO,EAAE,MAAM,EAAE,KAAK,EAAE,QAAQ,EAAE,GAAG,MAAM,CA4BjF;AAED;;;GAGG;AACH,wBAAsB,cAAc,CAAC,QAAQ,EAAE,MAAM,EAAE,KAAK,EAAE,QAAQ,EAAE,GAAG,OAAO,CAAC,IAAI,CAAC,CAIvF;AAMD;;;GAGG;AACH,wBAAsB,kBAAkB,CAAC,IAAI,EAAE,aAAa,EAAE,GAAG,EAAE,MAAM,GAAG,OAAO,CAAC,MAAM,EAAE,CAAC,CA+C5F"}
|
|
@@ -0,0 +1,101 @@
|
|
|
1
|
+
import * as fs from "node:fs/promises";
|
|
2
|
+
import path from "node:path";
|
|
3
|
+
import { uriToFile } from "./utils.js";
|
|
4
|
+
// =============================================================================
|
|
5
|
+
// Text Edit Application
|
|
6
|
+
// =============================================================================
|
|
7
|
+
/**
|
|
8
|
+
* Apply text edits to a string in-memory.
|
|
9
|
+
* Edits are applied in reverse order (bottom-to-top) to preserve line/character indices.
|
|
10
|
+
*/
|
|
11
|
+
export function applyTextEditsToString(content, edits) {
|
|
12
|
+
const lines = content.split("\n");
|
|
13
|
+
// Sort edits in reverse order (bottom-to-top, right-to-left)
|
|
14
|
+
const sortedEdits = [...edits].sort((a, b) => {
|
|
15
|
+
if (a.range.start.line !== b.range.start.line) {
|
|
16
|
+
return b.range.start.line - a.range.start.line;
|
|
17
|
+
}
|
|
18
|
+
return b.range.start.character - a.range.start.character;
|
|
19
|
+
});
|
|
20
|
+
for (const edit of sortedEdits) {
|
|
21
|
+
const { start, end } = edit.range;
|
|
22
|
+
// Single-line edit: replace substring within same line
|
|
23
|
+
if (start.line === end.line) {
|
|
24
|
+
const line = lines[start.line] || "";
|
|
25
|
+
lines[start.line] = line.slice(0, start.character) + edit.newText + line.slice(end.character);
|
|
26
|
+
}
|
|
27
|
+
else {
|
|
28
|
+
// Multi-line edit: splice across multiple lines
|
|
29
|
+
const startLine = lines[start.line] || "";
|
|
30
|
+
const endLine = lines[end.line] || "";
|
|
31
|
+
const newContent = startLine.slice(0, start.character) + edit.newText + endLine.slice(end.character);
|
|
32
|
+
lines.splice(start.line, end.line - start.line + 1, ...newContent.split("\n"));
|
|
33
|
+
}
|
|
34
|
+
}
|
|
35
|
+
return lines.join("\n");
|
|
36
|
+
}
|
|
37
|
+
/**
|
|
38
|
+
* Apply text edits to a file.
|
|
39
|
+
* Edits are applied in reverse order (bottom-to-top) to preserve line/character indices.
|
|
40
|
+
*/
|
|
41
|
+
export async function applyTextEdits(filePath, edits) {
|
|
42
|
+
const content = await fs.readFile(filePath, "utf-8");
|
|
43
|
+
const result = applyTextEditsToString(content, edits);
|
|
44
|
+
await fs.writeFile(filePath, result);
|
|
45
|
+
}
|
|
46
|
+
// =============================================================================
|
|
47
|
+
// Workspace Edit Application
|
|
48
|
+
// =============================================================================
|
|
49
|
+
/**
|
|
50
|
+
* Apply a workspace edit (collection of file changes).
|
|
51
|
+
* Returns array of applied change descriptions.
|
|
52
|
+
*/
|
|
53
|
+
export async function applyWorkspaceEdit(edit, cwd) {
|
|
54
|
+
const applied = [];
|
|
55
|
+
// Handle changes map (legacy format)
|
|
56
|
+
if (edit.changes) {
|
|
57
|
+
for (const [uri, textEdits] of Object.entries(edit.changes)) {
|
|
58
|
+
const filePath = uriToFile(uri);
|
|
59
|
+
await applyTextEdits(filePath, textEdits);
|
|
60
|
+
applied.push(`Applied ${textEdits.length} edit(s) to ${path.relative(cwd, filePath)}`);
|
|
61
|
+
}
|
|
62
|
+
}
|
|
63
|
+
// Handle documentChanges array (modern format)
|
|
64
|
+
if (edit.documentChanges) {
|
|
65
|
+
for (const change of edit.documentChanges) {
|
|
66
|
+
if ("textDocument" in change && change.textDocument && "edits" in change && change.edits) {
|
|
67
|
+
// TextDocumentEdit
|
|
68
|
+
const docChange = change;
|
|
69
|
+
const filePath = uriToFile(docChange.textDocument.uri);
|
|
70
|
+
const textEdits = docChange.edits.filter((e) => "range" in e && "newText" in e);
|
|
71
|
+
await applyTextEdits(filePath, textEdits);
|
|
72
|
+
applied.push(`Applied ${textEdits.length} edit(s) to ${path.relative(cwd, filePath)}`);
|
|
73
|
+
}
|
|
74
|
+
else if ("kind" in change && change.kind) {
|
|
75
|
+
// Resource operations
|
|
76
|
+
if (change.kind === "create") {
|
|
77
|
+
const createOp = change;
|
|
78
|
+
const filePath = uriToFile(createOp.uri);
|
|
79
|
+
await fs.writeFile(filePath, "");
|
|
80
|
+
applied.push(`Created ${path.relative(cwd, filePath)}`);
|
|
81
|
+
}
|
|
82
|
+
else if (change.kind === "rename") {
|
|
83
|
+
const renameOp = change;
|
|
84
|
+
const oldPath = uriToFile(renameOp.oldUri);
|
|
85
|
+
const newPath = uriToFile(renameOp.newUri);
|
|
86
|
+
await fs.mkdir(path.dirname(newPath), { recursive: true });
|
|
87
|
+
await fs.rename(oldPath, newPath);
|
|
88
|
+
applied.push(`Renamed ${path.relative(cwd, oldPath)} → ${path.relative(cwd, newPath)}`);
|
|
89
|
+
}
|
|
90
|
+
else if (change.kind === "delete") {
|
|
91
|
+
const deleteOp = change;
|
|
92
|
+
const filePath = uriToFile(deleteOp.uri);
|
|
93
|
+
await fs.rm(filePath, { recursive: true });
|
|
94
|
+
applied.push(`Deleted ${path.relative(cwd, filePath)}`);
|
|
95
|
+
}
|
|
96
|
+
}
|
|
97
|
+
}
|
|
98
|
+
}
|
|
99
|
+
return applied;
|
|
100
|
+
}
|
|
101
|
+
//# sourceMappingURL=edits.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"edits.js","sourceRoot":"","sources":["../../../src/core/lsp/edits.ts"],"names":[],"mappings":"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","sourcesContent":["import * as fs from \"node:fs/promises\";\nimport path from \"node:path\";\nimport type { CreateFile, DeleteFile, RenameFile, TextDocumentEdit, TextEdit, WorkspaceEdit } from \"./types.js\";\nimport { uriToFile } from \"./utils.js\";\n\n// =============================================================================\n// Text Edit Application\n// =============================================================================\n\n/**\n * Apply text edits to a string in-memory.\n * Edits are applied in reverse order (bottom-to-top) to preserve line/character indices.\n */\nexport function applyTextEditsToString(content: string, edits: TextEdit[]): string {\n\tconst lines = content.split(\"\\n\");\n\n\t// Sort edits in reverse order (bottom-to-top, right-to-left)\n\tconst sortedEdits = [...edits].sort((a, b) => {\n\t\tif (a.range.start.line !== b.range.start.line) {\n\t\t\treturn b.range.start.line - a.range.start.line;\n\t\t}\n\t\treturn b.range.start.character - a.range.start.character;\n\t});\n\n\tfor (const edit of sortedEdits) {\n\t\tconst { start, end } = edit.range;\n\n\t\t// Single-line edit: replace substring within same line\n\t\tif (start.line === end.line) {\n\t\t\tconst line = lines[start.line] || \"\";\n\t\t\tlines[start.line] = line.slice(0, start.character) + edit.newText + line.slice(end.character);\n\t\t} else {\n\t\t\t// Multi-line edit: splice across multiple lines\n\t\t\tconst startLine = lines[start.line] || \"\";\n\t\t\tconst endLine = lines[end.line] || \"\";\n\t\t\tconst newContent = startLine.slice(0, start.character) + edit.newText + endLine.slice(end.character);\n\t\t\tlines.splice(start.line, end.line - start.line + 1, ...newContent.split(\"\\n\"));\n\t\t}\n\t}\n\n\treturn lines.join(\"\\n\");\n}\n\n/**\n * Apply text edits to a file.\n * Edits are applied in reverse order (bottom-to-top) to preserve line/character indices.\n */\nexport async function applyTextEdits(filePath: string, edits: TextEdit[]): Promise<void> {\n\tconst content = await fs.readFile(filePath, \"utf-8\");\n\tconst result = applyTextEditsToString(content, edits);\n\tawait fs.writeFile(filePath, result);\n}\n\n// =============================================================================\n// Workspace Edit Application\n// =============================================================================\n\n/**\n * Apply a workspace edit (collection of file changes).\n * Returns array of applied change descriptions.\n */\nexport async function applyWorkspaceEdit(edit: WorkspaceEdit, cwd: string): Promise<string[]> {\n\tconst applied: string[] = [];\n\n\t// Handle changes map (legacy format)\n\tif (edit.changes) {\n\t\tfor (const [uri, textEdits] of Object.entries(edit.changes)) {\n\t\t\tconst filePath = uriToFile(uri);\n\t\t\tawait applyTextEdits(filePath, textEdits);\n\t\t\tapplied.push(`Applied ${textEdits.length} edit(s) to ${path.relative(cwd, filePath)}`);\n\t\t}\n\t}\n\n\t// Handle documentChanges array (modern format)\n\tif (edit.documentChanges) {\n\t\tfor (const change of edit.documentChanges) {\n\t\t\tif (\"textDocument\" in change && change.textDocument && \"edits\" in change && change.edits) {\n\t\t\t\t// TextDocumentEdit\n\t\t\t\tconst docChange = change as TextDocumentEdit;\n\t\t\t\tconst filePath = uriToFile(docChange.textDocument.uri);\n\t\t\t\tconst textEdits = docChange.edits.filter((e): e is TextEdit => \"range\" in e && \"newText\" in e);\n\t\t\t\tawait applyTextEdits(filePath, textEdits);\n\t\t\t\tapplied.push(`Applied ${textEdits.length} edit(s) to ${path.relative(cwd, filePath)}`);\n\t\t\t} else if (\"kind\" in change && change.kind) {\n\t\t\t\t// Resource operations\n\t\t\t\tif (change.kind === \"create\") {\n\t\t\t\t\tconst createOp = change as CreateFile;\n\t\t\t\t\tconst filePath = uriToFile(createOp.uri);\n\t\t\t\t\tawait fs.writeFile(filePath, \"\");\n\t\t\t\t\tapplied.push(`Created ${path.relative(cwd, filePath)}`);\n\t\t\t\t} else if (change.kind === \"rename\") {\n\t\t\t\t\tconst renameOp = change as RenameFile;\n\t\t\t\t\tconst oldPath = uriToFile(renameOp.oldUri);\n\t\t\t\t\tconst newPath = uriToFile(renameOp.newUri);\n\t\t\t\t\tawait fs.mkdir(path.dirname(newPath), { recursive: true });\n\t\t\t\t\tawait fs.rename(oldPath, newPath);\n\t\t\t\t\tapplied.push(`Renamed ${path.relative(cwd, oldPath)} → ${path.relative(cwd, newPath)}`);\n\t\t\t\t} else if (change.kind === \"delete\") {\n\t\t\t\t\tconst deleteOp = change as DeleteFile;\n\t\t\t\t\tconst filePath = uriToFile(deleteOp.uri);\n\t\t\t\t\tawait fs.rm(filePath, { recursive: true });\n\t\t\t\t\tapplied.push(`Deleted ${path.relative(cwd, filePath)}`);\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\t}\n\n\treturn applied;\n}\n"]}
|
|
@@ -0,0 +1,15 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Local helpers replacing @oh-my-pi/pi-utils and tool-errors/tool-timeouts imports.
|
|
3
|
+
*/
|
|
4
|
+
export declare class ToolAbortError extends Error {
|
|
5
|
+
constructor();
|
|
6
|
+
}
|
|
7
|
+
export declare function throwIfAborted(signal?: AbortSignal): void;
|
|
8
|
+
export declare function isEnoent(err: unknown): boolean;
|
|
9
|
+
export declare function isRecord(v: unknown): v is Record<string, unknown>;
|
|
10
|
+
export declare function clampTimeout(timeout?: number): number;
|
|
11
|
+
/**
|
|
12
|
+
* Run a promise, rejecting if the signal aborts.
|
|
13
|
+
*/
|
|
14
|
+
export declare function untilAborted<T>(signal: AbortSignal | undefined, fn: () => Promise<T>): Promise<T>;
|
|
15
|
+
//# sourceMappingURL=helpers.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"helpers.d.ts","sourceRoot":"","sources":["../../../src/core/lsp/helpers.ts"],"names":[],"mappings":"AAAA;;GAEG;AAEH,qBAAa,cAAe,SAAQ,KAAK;;CAKxC;AAED,wBAAgB,cAAc,CAAC,MAAM,CAAC,EAAE,WAAW,GAAG,IAAI,CAIzD;AAED,wBAAgB,QAAQ,CAAC,GAAG,EAAE,OAAO,GAAG,OAAO,CAE9C;AAED,wBAAgB,QAAQ,CAAC,CAAC,EAAE,OAAO,GAAG,CAAC,IAAI,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,CAEjE;AAED,wBAAgB,YAAY,CAAC,OAAO,CAAC,EAAE,MAAM,GAAG,MAAM,CAErD;AAED;;GAEG;AACH,wBAAsB,YAAY,CAAC,CAAC,EAAE,MAAM,EAAE,WAAW,GAAG,SAAS,EAAE,EAAE,EAAE,MAAM,OAAO,CAAC,CAAC,CAAC,GAAG,OAAO,CAAC,CAAC,CAAC,CAqBvG"}
|