@prmichaelsen/acp-mcp 0.4.1 → 0.5.1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/CHANGELOG.md +31 -0
- package/README.md +2 -0
- package/agent/milestones/milestone-2-bug-fixes-production-readiness.md +159 -0
- package/agent/progress.yaml +92 -13
- package/agent/tasks/task-4-fix-read-file-not-found-bug.md +390 -0
- package/dist/server-factory.js +212 -19
- package/dist/server-factory.js.map +3 -3
- package/dist/server.js +206 -22
- package/dist/server.js.map +3 -3
- package/dist/utils/logger.d.ts +43 -0
- package/package.json +1 -1
- package/src/server-factory.ts +45 -16
- package/src/server.ts +36 -18
- package/src/tools/acp-remote-execute-command.ts +11 -0
- package/src/tools/acp-remote-list-files.ts +7 -4
- package/src/tools/acp-remote-read-file.ts +6 -0
- package/src/tools/acp-remote-write-file.ts +6 -0
- package/src/utils/logger.ts +131 -0
- package/src/utils/ssh-connection.ts +59 -0
package/dist/server.js.map
CHANGED
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
{
|
|
2
2
|
"version": 3,
|
|
3
|
-
"sources": ["../node_modules/dotenv/package.json", "../node_modules/dotenv/lib/main.js", "../src/server.ts", "../src/config.ts", "../src/tools/acp-remote-list-files.ts", "../src/tools/acp-remote-execute-command.ts", "../src/tools/acp-remote-read-file.ts", "../src/tools/acp-remote-write-file.ts", "../src/utils/ssh-connection.ts"],
|
|
4
|
-
"sourcesContent": ["{\n \"name\": \"dotenv\",\n \"version\": \"16.6.1\",\n \"description\": \"Loads environment variables from .env file\",\n \"main\": \"lib/main.js\",\n \"types\": \"lib/main.d.ts\",\n \"exports\": {\n \".\": {\n \"types\": \"./lib/main.d.ts\",\n \"require\": \"./lib/main.js\",\n \"default\": \"./lib/main.js\"\n },\n \"./config\": \"./config.js\",\n \"./config.js\": \"./config.js\",\n \"./lib/env-options\": \"./lib/env-options.js\",\n \"./lib/env-options.js\": \"./lib/env-options.js\",\n \"./lib/cli-options\": \"./lib/cli-options.js\",\n \"./lib/cli-options.js\": \"./lib/cli-options.js\",\n \"./package.json\": \"./package.json\"\n },\n \"scripts\": {\n \"dts-check\": \"tsc --project tests/types/tsconfig.json\",\n \"lint\": \"standard\",\n \"pretest\": \"npm run lint && npm run dts-check\",\n \"test\": \"tap run --allow-empty-coverage --disable-coverage --timeout=60000\",\n \"test:coverage\": \"tap run --show-full-coverage --timeout=60000 --coverage-report=text --coverage-report=lcov\",\n \"prerelease\": \"npm test\",\n \"release\": \"standard-version\"\n },\n \"repository\": {\n \"type\": \"git\",\n \"url\": \"git://github.com/motdotla/dotenv.git\"\n },\n \"homepage\": \"https://github.com/motdotla/dotenv#readme\",\n \"funding\": \"https://dotenvx.com\",\n \"keywords\": [\n \"dotenv\",\n \"env\",\n \".env\",\n \"environment\",\n \"variables\",\n \"config\",\n \"settings\"\n ],\n \"readmeFilename\": \"README.md\",\n \"license\": \"BSD-2-Clause\",\n \"devDependencies\": {\n \"@types/node\": \"^18.11.3\",\n \"decache\": \"^4.6.2\",\n \"sinon\": \"^14.0.1\",\n \"standard\": \"^17.0.0\",\n \"standard-version\": \"^9.5.0\",\n \"tap\": \"^19.2.0\",\n \"typescript\": \"^4.8.4\"\n },\n \"engines\": {\n \"node\": \">=12\"\n },\n \"browser\": {\n \"fs\": false\n }\n}\n", "const fs = require('fs')\nconst path = require('path')\nconst os = require('os')\nconst crypto = require('crypto')\nconst packageJson = require('../package.json')\n\nconst version = packageJson.version\n\nconst LINE = /(?:^|^)\\s*(?:export\\s+)?([\\w.-]+)(?:\\s*=\\s*?|:\\s+?)(\\s*'(?:\\\\'|[^'])*'|\\s*\"(?:\\\\\"|[^\"])*\"|\\s*`(?:\\\\`|[^`])*`|[^#\\r\\n]+)?\\s*(?:#.*)?(?:$|$)/mg\n\n// Parse src into an Object\nfunction parse (src) {\n const obj = {}\n\n // Convert buffer to string\n let lines = src.toString()\n\n // Convert line breaks to same format\n lines = lines.replace(/\\r\\n?/mg, '\\n')\n\n let match\n while ((match = LINE.exec(lines)) != null) {\n const key = match[1]\n\n // Default undefined or null to empty string\n let value = (match[2] || '')\n\n // Remove whitespace\n value = value.trim()\n\n // Check if double quoted\n const maybeQuote = value[0]\n\n // Remove surrounding quotes\n value = value.replace(/^(['\"`])([\\s\\S]*)\\1$/mg, '$2')\n\n // Expand newlines if double quoted\n if (maybeQuote === '\"') {\n value = value.replace(/\\\\n/g, '\\n')\n value = value.replace(/\\\\r/g, '\\r')\n }\n\n // Add to object\n obj[key] = value\n }\n\n return obj\n}\n\nfunction _parseVault (options) {\n options = options || {}\n\n const vaultPath = _vaultPath(options)\n options.path = vaultPath // parse .env.vault\n const result = DotenvModule.configDotenv(options)\n if (!result.parsed) {\n const err = new Error(`MISSING_DATA: Cannot parse ${vaultPath} for an unknown reason`)\n err.code = 'MISSING_DATA'\n throw err\n }\n\n // handle scenario for comma separated keys - for use with key rotation\n // example: DOTENV_KEY=\"dotenv://:key_1234@dotenvx.com/vault/.env.vault?environment=prod,dotenv://:key_7890@dotenvx.com/vault/.env.vault?environment=prod\"\n const keys = _dotenvKey(options).split(',')\n const length = keys.length\n\n let decrypted\n for (let i = 0; i < length; i++) {\n try {\n // Get full key\n const key = keys[i].trim()\n\n // Get instructions for decrypt\n const attrs = _instructions(result, key)\n\n // Decrypt\n decrypted = DotenvModule.decrypt(attrs.ciphertext, attrs.key)\n\n break\n } catch (error) {\n // last key\n if (i + 1 >= length) {\n throw error\n }\n // try next key\n }\n }\n\n // Parse decrypted .env string\n return DotenvModule.parse(decrypted)\n}\n\nfunction _warn (message) {\n console.log(`[dotenv@${version}][WARN] ${message}`)\n}\n\nfunction _debug (message) {\n console.log(`[dotenv@${version}][DEBUG] ${message}`)\n}\n\nfunction _log (message) {\n console.log(`[dotenv@${version}] ${message}`)\n}\n\nfunction _dotenvKey (options) {\n // prioritize developer directly setting options.DOTENV_KEY\n if (options && options.DOTENV_KEY && options.DOTENV_KEY.length > 0) {\n return options.DOTENV_KEY\n }\n\n // secondary infra already contains a DOTENV_KEY environment variable\n if (process.env.DOTENV_KEY && process.env.DOTENV_KEY.length > 0) {\n return process.env.DOTENV_KEY\n }\n\n // fallback to empty string\n return ''\n}\n\nfunction _instructions (result, dotenvKey) {\n // Parse DOTENV_KEY. Format is a URI\n let uri\n try {\n uri = new URL(dotenvKey)\n } catch (error) {\n if (error.code === 'ERR_INVALID_URL') {\n const err = new Error('INVALID_DOTENV_KEY: Wrong format. Must be in valid uri format like dotenv://:key_1234@dotenvx.com/vault/.env.vault?environment=development')\n err.code = 'INVALID_DOTENV_KEY'\n throw err\n }\n\n throw error\n }\n\n // Get decrypt key\n const key = uri.password\n if (!key) {\n const err = new Error('INVALID_DOTENV_KEY: Missing key part')\n err.code = 'INVALID_DOTENV_KEY'\n throw err\n }\n\n // Get environment\n const environment = uri.searchParams.get('environment')\n if (!environment) {\n const err = new Error('INVALID_DOTENV_KEY: Missing environment part')\n err.code = 'INVALID_DOTENV_KEY'\n throw err\n }\n\n // Get ciphertext payload\n const environmentKey = `DOTENV_VAULT_${environment.toUpperCase()}`\n const ciphertext = result.parsed[environmentKey] // DOTENV_VAULT_PRODUCTION\n if (!ciphertext) {\n const err = new Error(`NOT_FOUND_DOTENV_ENVIRONMENT: Cannot locate environment ${environmentKey} in your .env.vault file.`)\n err.code = 'NOT_FOUND_DOTENV_ENVIRONMENT'\n throw err\n }\n\n return { ciphertext, key }\n}\n\nfunction _vaultPath (options) {\n let possibleVaultPath = null\n\n if (options && options.path && options.path.length > 0) {\n if (Array.isArray(options.path)) {\n for (const filepath of options.path) {\n if (fs.existsSync(filepath)) {\n possibleVaultPath = filepath.endsWith('.vault') ? filepath : `${filepath}.vault`\n }\n }\n } else {\n possibleVaultPath = options.path.endsWith('.vault') ? options.path : `${options.path}.vault`\n }\n } else {\n possibleVaultPath = path.resolve(process.cwd(), '.env.vault')\n }\n\n if (fs.existsSync(possibleVaultPath)) {\n return possibleVaultPath\n }\n\n return null\n}\n\nfunction _resolveHome (envPath) {\n return envPath[0] === '~' ? path.join(os.homedir(), envPath.slice(1)) : envPath\n}\n\nfunction _configVault (options) {\n const debug = Boolean(options && options.debug)\n const quiet = options && 'quiet' in options ? options.quiet : true\n\n if (debug || !quiet) {\n _log('Loading env from encrypted .env.vault')\n }\n\n const parsed = DotenvModule._parseVault(options)\n\n let processEnv = process.env\n if (options && options.processEnv != null) {\n processEnv = options.processEnv\n }\n\n DotenvModule.populate(processEnv, parsed, options)\n\n return { parsed }\n}\n\nfunction configDotenv (options) {\n const dotenvPath = path.resolve(process.cwd(), '.env')\n let encoding = 'utf8'\n const debug = Boolean(options && options.debug)\n const quiet = options && 'quiet' in options ? options.quiet : true\n\n if (options && options.encoding) {\n encoding = options.encoding\n } else {\n if (debug) {\n _debug('No encoding is specified. UTF-8 is used by default')\n }\n }\n\n let optionPaths = [dotenvPath] // default, look for .env\n if (options && options.path) {\n if (!Array.isArray(options.path)) {\n optionPaths = [_resolveHome(options.path)]\n } else {\n optionPaths = [] // reset default\n for (const filepath of options.path) {\n optionPaths.push(_resolveHome(filepath))\n }\n }\n }\n\n // Build the parsed data in a temporary object (because we need to return it). Once we have the final\n // parsed data, we will combine it with process.env (or options.processEnv if provided).\n let lastError\n const parsedAll = {}\n for (const path of optionPaths) {\n try {\n // Specifying an encoding returns a string instead of a buffer\n const parsed = DotenvModule.parse(fs.readFileSync(path, { encoding }))\n\n DotenvModule.populate(parsedAll, parsed, options)\n } catch (e) {\n if (debug) {\n _debug(`Failed to load ${path} ${e.message}`)\n }\n lastError = e\n }\n }\n\n let processEnv = process.env\n if (options && options.processEnv != null) {\n processEnv = options.processEnv\n }\n\n DotenvModule.populate(processEnv, parsedAll, options)\n\n if (debug || !quiet) {\n const keysCount = Object.keys(parsedAll).length\n const shortPaths = []\n for (const filePath of optionPaths) {\n try {\n const relative = path.relative(process.cwd(), filePath)\n shortPaths.push(relative)\n } catch (e) {\n if (debug) {\n _debug(`Failed to load ${filePath} ${e.message}`)\n }\n lastError = e\n }\n }\n\n _log(`injecting env (${keysCount}) from ${shortPaths.join(',')}`)\n }\n\n if (lastError) {\n return { parsed: parsedAll, error: lastError }\n } else {\n return { parsed: parsedAll }\n }\n}\n\n// Populates process.env from .env file\nfunction config (options) {\n // fallback to original dotenv if DOTENV_KEY is not set\n if (_dotenvKey(options).length === 0) {\n return DotenvModule.configDotenv(options)\n }\n\n const vaultPath = _vaultPath(options)\n\n // dotenvKey exists but .env.vault file does not exist\n if (!vaultPath) {\n _warn(`You set DOTENV_KEY but you are missing a .env.vault file at ${vaultPath}. Did you forget to build it?`)\n\n return DotenvModule.configDotenv(options)\n }\n\n return DotenvModule._configVault(options)\n}\n\nfunction decrypt (encrypted, keyStr) {\n const key = Buffer.from(keyStr.slice(-64), 'hex')\n let ciphertext = Buffer.from(encrypted, 'base64')\n\n const nonce = ciphertext.subarray(0, 12)\n const authTag = ciphertext.subarray(-16)\n ciphertext = ciphertext.subarray(12, -16)\n\n try {\n const aesgcm = crypto.createDecipheriv('aes-256-gcm', key, nonce)\n aesgcm.setAuthTag(authTag)\n return `${aesgcm.update(ciphertext)}${aesgcm.final()}`\n } catch (error) {\n const isRange = error instanceof RangeError\n const invalidKeyLength = error.message === 'Invalid key length'\n const decryptionFailed = error.message === 'Unsupported state or unable to authenticate data'\n\n if (isRange || invalidKeyLength) {\n const err = new Error('INVALID_DOTENV_KEY: It must be 64 characters long (or more)')\n err.code = 'INVALID_DOTENV_KEY'\n throw err\n } else if (decryptionFailed) {\n const err = new Error('DECRYPTION_FAILED: Please check your DOTENV_KEY')\n err.code = 'DECRYPTION_FAILED'\n throw err\n } else {\n throw error\n }\n }\n}\n\n// Populate process.env with parsed values\nfunction populate (processEnv, parsed, options = {}) {\n const debug = Boolean(options && options.debug)\n const override = Boolean(options && options.override)\n\n if (typeof parsed !== 'object') {\n const err = new Error('OBJECT_REQUIRED: Please check the processEnv argument being passed to populate')\n err.code = 'OBJECT_REQUIRED'\n throw err\n }\n\n // Set process.env\n for (const key of Object.keys(parsed)) {\n if (Object.prototype.hasOwnProperty.call(processEnv, key)) {\n if (override === true) {\n processEnv[key] = parsed[key]\n }\n\n if (debug) {\n if (override === true) {\n _debug(`\"${key}\" is already defined and WAS overwritten`)\n } else {\n _debug(`\"${key}\" is already defined and was NOT overwritten`)\n }\n }\n } else {\n processEnv[key] = parsed[key]\n }\n }\n}\n\nconst DotenvModule = {\n configDotenv,\n _configVault,\n _parseVault,\n config,\n decrypt,\n parse,\n populate\n}\n\nmodule.exports.configDotenv = DotenvModule.configDotenv\nmodule.exports._configVault = DotenvModule._configVault\nmodule.exports._parseVault = DotenvModule._parseVault\nmodule.exports.config = DotenvModule.config\nmodule.exports.decrypt = DotenvModule.decrypt\nmodule.exports.parse = DotenvModule.parse\nmodule.exports.populate = DotenvModule.populate\n\nmodule.exports = DotenvModule\n", "#!/usr/bin/env node\n\nimport { Server } from '@modelcontextprotocol/sdk/server/index.js';\nimport { StdioServerTransport } from '@modelcontextprotocol/sdk/server/stdio.js';\nimport {\n CallToolRequestSchema,\n ListToolsRequestSchema,\n} from '@modelcontextprotocol/sdk/types.js';\nimport { config, loadSSHPrivateKey } from './config.js';\nimport { acpRemoteListFilesTool, handleAcpRemoteListFiles } from './tools/acp-remote-list-files.js';\nimport { acpRemoteExecuteCommandTool, handleAcpRemoteExecuteCommand } from './tools/acp-remote-execute-command.js';\nimport { acpRemoteReadFileTool, handleAcpRemoteReadFile } from './tools/acp-remote-read-file.js';\nimport { acpRemoteWriteFileTool, handleAcpRemoteWriteFile } from './tools/acp-remote-write-file.js';\nimport { SSHConnectionManager } from './utils/ssh-connection.js';\n\nasync function main() {\n // Load SSH private key\n const privateKey = loadSSHPrivateKey();\n\n // Create SSH connection manager\n const sshConnection = new SSHConnectionManager({\n host: config.ssh.host,\n port: config.ssh.port,\n username: config.ssh.username,\n privateKey: privateKey,\n });\n\n // Connect to remote server\n await sshConnection.connect();\n console.error(`Connected to SSH server: ${config.ssh.username}@${config.ssh.host}`);\n\n const server = new Server(\n {\n name: 'acp-mcp',\n version: '0.1.0',\n },\n {\n capabilities: {\n tools: {},\n },\n }\n );\n\n // Register tools\n server.setRequestHandler(ListToolsRequestSchema, async () => ({\n tools: [acpRemoteListFilesTool, acpRemoteExecuteCommandTool, acpRemoteReadFileTool, acpRemoteWriteFileTool],\n }));\n\n server.setRequestHandler(CallToolRequestSchema, async (request) => {\n if (request.params.name === 'acp_remote_list_files') {\n return await handleAcpRemoteListFiles(request.params.arguments, sshConnection);\n }\n if (request.params.name === 'acp_remote_execute_command') {\n return await handleAcpRemoteExecuteCommand(request.params.arguments, sshConnection);\n }\n if (request.params.name === 'acp_remote_read_file') {\n return await handleAcpRemoteReadFile(request.params.arguments, sshConnection);\n }\n if (request.params.name === 'acp_remote_write_file') {\n return await handleAcpRemoteWriteFile(request.params.arguments, sshConnection);\n }\n throw new Error(`Unknown tool: ${request.params.name}`);\n });\n\n // Handle shutdown to cleanup SSH connection\n const cleanup = () => {\n console.error('Shutting down...');\n sshConnection.disconnect();\n process.exit(0);\n };\n\n process.on('SIGINT', cleanup);\n process.on('SIGTERM', cleanup);\n\n // Start server\n const transport = new StdioServerTransport();\n await server.connect(transport);\n console.error('ACP MCP Server running on stdio');\n}\n\nmain().catch((error) => {\n console.error('Server error:', error);\n process.exit(1);\n});\n", "import dotenv from 'dotenv';\nimport { readFileSync } from 'fs';\n\ndotenv.config();\n\nexport interface Config {\n logLevel: string;\n ssh: {\n host: string;\n port: number;\n username: string;\n privateKeyPath: string;\n };\n}\n\nfunction validateConfig(): Config {\n const sshHost = process.env.SSH_HOST;\n const sshUsername = process.env.SSH_USERNAME;\n const sshPrivateKeyPath = process.env.SSH_PRIVATE_KEY_PATH;\n\n if (!sshHost || !sshUsername || !sshPrivateKeyPath) {\n throw new Error(\n 'Missing required SSH configuration. Please set SSH_HOST, SSH_USERNAME, and SSH_PRIVATE_KEY_PATH environment variables.'\n );\n }\n\n return {\n logLevel: process.env.LOG_LEVEL || 'info',\n ssh: {\n host: sshHost,\n port: parseInt(process.env.SSH_PORT || '22', 10),\n username: sshUsername,\n privateKeyPath: sshPrivateKeyPath,\n },\n };\n}\n\nexport const config = validateConfig();\n\n/**\n * Load SSH private key from file\n */\nexport function loadSSHPrivateKey(): string {\n try {\n return readFileSync(config.ssh.privateKeyPath, 'utf-8');\n } catch (error) {\n throw new Error(\n `Failed to load SSH private key from ${config.ssh.privateKeyPath}: ${error instanceof Error ? error.message : String(error)}`\n );\n }\n}\n", "import { Tool } from '@modelcontextprotocol/sdk/types.js';\nimport { SSHConnectionManager } from '../utils/ssh-connection.js';\n\nexport const acpRemoteListFilesTool: Tool = {\n name: 'acp_remote_list_files',\n description: 'List files and directories in a specified path on the remote machine via SSH',\n inputSchema: {\n type: 'object',\n properties: {\n path: {\n type: 'string',\n description: 'The directory path to list files from',\n },\n recursive: {\n type: 'boolean',\n description: 'Whether to list files recursively',\n default: false,\n },\n },\n required: ['path'],\n },\n};\n\ninterface ListFilesArgs {\n path: string;\n recursive?: boolean;\n}\n\n/**\n * Handle the acp_remote_list_files tool invocation\n * Lists files and directories at the specified path on the remote machine via SSH\n * \n * @param args - Tool arguments containing path and recursive flag\n * @param sshConnection - SSH connection manager for remote operations\n */\nexport async function handleAcpRemoteListFiles(\n args: any,\n sshConnection: SSHConnectionManager\n): Promise<{ content: Array<{ type: string; text: string }> }> {\n const { path, recursive = false } = args as ListFilesArgs;\n\n try {\n const files = await listRemoteFiles(sshConnection, path, recursive);\n \n return {\n content: [\n {\n type: 'text',\n text: files.join('\\n'),\n },\n ],\n };\n } catch (error) {\n const errorMessage = error instanceof Error ? error.message : String(error);\n return {\n content: [\n {\n type: 'text',\n text: `Error listing remote files: ${errorMessage}`,\n },\n ],\n };\n }\n}\n\n/**\n * Recursively list files in a remote directory via SSH\n */\nasync function listRemoteFiles(\n ssh: SSHConnectionManager,\n dirPath: string,\n recursive: boolean\n): Promise<string[]> {\n const entries = await ssh.listFiles(dirPath);\n const files: string[] = [];\n\n for (const entry of entries) {\n if (entry.isDirectory) {\n files.push(`${entry.name}/`);\n if (recursive) {\n const fullPath = `${dirPath}/${entry.name}`.replace(/\\/+/g, '/');\n const subFiles = await listRemoteFiles(ssh, fullPath, recursive);\n files.push(...subFiles.map(f => `${entry.name}/${f}`));\n }\n } else {\n files.push(entry.name);\n }\n }\n\n return files.sort();\n}\n", "import { Tool } from '@modelcontextprotocol/sdk/types.js';\nimport { SSHConnectionManager } from '../utils/ssh-connection.js';\n\nexport const acpRemoteExecuteCommandTool: Tool = {\n name: 'acp_remote_execute_command',\n description: 'Execute a shell command on the remote machine via SSH',\n inputSchema: {\n type: 'object',\n properties: {\n command: {\n type: 'string',\n description: 'Shell command to execute',\n },\n cwd: {\n type: 'string',\n description: 'Working directory for command execution (optional)',\n },\n timeout: {\n type: 'number',\n description: 'Timeout in seconds (default: 30)',\n default: 30,\n },\n },\n required: ['command'],\n },\n};\n\ninterface ExecuteCommandArgs {\n command: string;\n cwd?: string;\n timeout?: number;\n}\n\ninterface ExecuteCommandResult {\n stdout: string;\n stderr: string;\n exitCode: number;\n timedOut: boolean;\n}\n\n/**\n * Handle the acp_remote_execute_command tool invocation\n * Executes a shell command on the remote machine via SSH\n * \n * @param args - Tool arguments containing command, cwd, and timeout\n * @param sshConnection - SSH connection manager for remote operations\n */\nexport async function handleAcpRemoteExecuteCommand(\n args: any,\n sshConnection: SSHConnectionManager\n): Promise<{ content: Array<{ type: string; text: string }> }> {\n const { command, cwd, timeout = 30 } = args as ExecuteCommandArgs;\n\n try {\n // Build command with working directory if specified\n const fullCommand = cwd ? `cd ${cwd} && ${command}` : command;\n \n const result = await sshConnection.execWithTimeout(fullCommand, timeout);\n \n // Format output as JSON for structured response\n const output: ExecuteCommandResult = {\n stdout: result.stdout,\n stderr: result.stderr,\n exitCode: result.exitCode,\n timedOut: result.timedOut,\n };\n \n return {\n content: [\n {\n type: 'text',\n text: JSON.stringify(output, null, 2),\n },\n ],\n };\n } catch (error) {\n const errorMessage = error instanceof Error ? error.message : String(error);\n return {\n content: [\n {\n type: 'text',\n text: JSON.stringify({\n stdout: '',\n stderr: errorMessage,\n exitCode: 1,\n timedOut: false,\n }, null, 2),\n },\n ],\n };\n }\n}\n", "import { Tool } from '@modelcontextprotocol/sdk/types.js';\nimport { SSHConnectionManager } from '../utils/ssh-connection.js';\n\nexport const acpRemoteReadFileTool: Tool = {\n name: 'acp_remote_read_file',\n description: 'Read file contents from the remote machine via SSH',\n inputSchema: {\n type: 'object',\n properties: {\n path: {\n type: 'string',\n description: 'Absolute path to file',\n },\n encoding: {\n type: 'string',\n description: 'File encoding (default: utf-8)',\n default: 'utf-8',\n enum: ['utf-8', 'ascii', 'base64'],\n },\n maxSize: {\n type: 'number',\n description: 'Max file size in bytes (default: 1MB)',\n default: 1048576,\n },\n },\n required: ['path'],\n },\n};\n\ninterface ReadFileArgs {\n path: string;\n encoding?: string;\n maxSize?: number;\n}\n\ninterface ReadFileResult {\n content: string;\n size: number;\n encoding: string;\n}\n\n/**\n * Handle the acp_remote_read_file tool invocation\n * Reads file contents from the remote machine via SSH\n * \n * @param args - Tool arguments containing path, encoding, and maxSize\n * @param sshConnection - SSH connection manager for remote operations\n */\nexport async function handleAcpRemoteReadFile(\n args: any,\n sshConnection: SSHConnectionManager\n): Promise<{ content: Array<{ type: string; text: string }> }> {\n const { path, encoding = 'utf-8', maxSize = 1048576 } = args as ReadFileArgs;\n\n try {\n const result = await sshConnection.readFile(path, encoding, maxSize);\n \n const output: ReadFileResult = {\n content: result.content,\n size: result.size,\n encoding: result.encoding,\n };\n \n return {\n content: [\n {\n type: 'text',\n text: JSON.stringify(output, null, 2),\n },\n ],\n };\n } catch (error) {\n const errorMessage = error instanceof Error ? error.message : String(error);\n return {\n content: [\n {\n type: 'text',\n text: JSON.stringify({\n error: errorMessage,\n content: '',\n size: 0,\n encoding,\n }, null, 2),\n },\n ],\n };\n }\n}\n", "import { Tool } from '@modelcontextprotocol/sdk/types.js';\nimport { SSHConnectionManager } from '../utils/ssh-connection.js';\n\nexport const acpRemoteWriteFileTool: Tool = {\n name: 'acp_remote_write_file',\n description: 'Write file contents to the remote machine via SSH',\n inputSchema: {\n type: 'object',\n properties: {\n path: {\n type: 'string',\n description: 'Absolute path to file',\n },\n content: {\n type: 'string',\n description: 'File contents to write',\n },\n encoding: {\n type: 'string',\n description: 'File encoding (default: utf-8)',\n default: 'utf-8',\n },\n createDirs: {\n type: 'boolean',\n description: 'Create parent directories if they don\\'t exist (default: false)',\n default: false,\n },\n backup: {\n type: 'boolean',\n description: 'Backup existing file before overwriting (default: false)',\n default: false,\n },\n },\n required: ['path', 'content'],\n },\n};\n\ninterface WriteFileArgs {\n path: string;\n content: string;\n encoding?: string;\n createDirs?: boolean;\n backup?: boolean;\n}\n\ninterface WriteFileResult {\n success: boolean;\n bytesWritten: number;\n backupPath?: string;\n}\n\n/**\n * Handle the acp_remote_write_file tool invocation\n * Writes file contents to the remote machine via SSH\n * \n * @param args - Tool arguments containing path, content, and options\n * @param sshConnection - SSH connection manager for remote operations\n */\nexport async function handleAcpRemoteWriteFile(\n args: any,\n sshConnection: SSHConnectionManager\n): Promise<{ content: Array<{ type: string; text: string }> }> {\n const { path, content, encoding = 'utf-8', createDirs = false, backup = false } = args as WriteFileArgs;\n\n try {\n const result = await sshConnection.writeFile(path, content, {\n encoding,\n createDirs,\n backup,\n });\n \n const output: WriteFileResult = {\n success: result.success,\n bytesWritten: result.bytesWritten,\n backupPath: result.backupPath,\n };\n \n return {\n content: [\n {\n type: 'text',\n text: JSON.stringify(output, null, 2),\n },\n ],\n };\n } catch (error) {\n const errorMessage = error instanceof Error ? error.message : String(error);\n return {\n content: [\n {\n type: 'text',\n text: JSON.stringify({\n success: false,\n bytesWritten: 0,\n error: errorMessage,\n }, null, 2),\n },\n ],\n };\n }\n}\n", "import { Client, SFTPWrapper } from 'ssh2';\nimport { SSHConfig } from '../types/ssh-config.js';\n\n/**\n * SSH Connection Manager\n * Manages SSH connections and provides SFTP access for remote file operations\n */\nexport class SSHConnectionManager {\n private client: Client;\n private config: SSHConfig;\n private connected: boolean = false;\n\n constructor(config: SSHConfig) {\n this.config = config;\n this.client = new Client();\n }\n\n /**\n * Connect to the remote SSH server\n */\n async connect(): Promise<void> {\n if (this.connected) {\n return;\n }\n\n return new Promise((resolve, reject) => {\n this.client\n .on('ready', () => {\n this.connected = true;\n resolve();\n })\n .on('error', (err) => {\n reject(err);\n })\n .connect({\n host: this.config.host,\n port: this.config.port || 22,\n username: this.config.username,\n privateKey: this.config.privateKey,\n });\n });\n }\n\n /**\n * Execute a command on the remote server\n */\n async exec(command: string): Promise<string> {\n if (!this.connected) {\n await this.connect();\n }\n\n return new Promise((resolve, reject) => {\n this.client.exec(command, (err, stream) => {\n if (err) {\n reject(err);\n return;\n }\n\n let stdout = '';\n let stderr = '';\n\n stream\n .on('close', (code: number) => {\n if (code !== 0) {\n reject(new Error(`Command failed with code ${code}: ${stderr}`));\n } else {\n resolve(stdout);\n }\n })\n .on('data', (data: Buffer) => {\n stdout += data.toString();\n })\n .stderr.on('data', (data: Buffer) => {\n stderr += data.toString();\n });\n });\n });\n }\n\n /**\n * Execute a command on the remote server with timeout support\n */\n async execWithTimeout(\n command: string,\n timeoutSeconds: number = 30\n ): Promise<{ stdout: string; stderr: string; exitCode: number; timedOut: boolean }> {\n if (!this.connected) {\n await this.connect();\n }\n\n const execPromise = new Promise<{ stdout: string; stderr: string; exitCode: number }>((resolve, reject) => {\n this.client.exec(command, (err, stream) => {\n if (err) {\n reject(err);\n return;\n }\n\n let stdout = '';\n let stderr = '';\n\n stream\n .on('close', (code: number) => {\n resolve({ stdout, stderr, exitCode: code });\n })\n .on('data', (data: Buffer) => {\n stdout += data.toString();\n })\n .stderr.on('data', (data: Buffer) => {\n stderr += data.toString();\n });\n });\n });\n\n const timeoutPromise = new Promise<never>((_, reject) => {\n setTimeout(() => {\n reject(new Error('Command execution timed out'));\n }, timeoutSeconds * 1000);\n });\n\n try {\n const result = await Promise.race([execPromise, timeoutPromise]);\n return { ...result, timedOut: false };\n } catch (error) {\n if (error instanceof Error && error.message === 'Command execution timed out') {\n return {\n stdout: '',\n stderr: 'Command execution timed out',\n exitCode: 124,\n timedOut: true,\n };\n }\n throw error;\n }\n }\n\n /**\n * Get SFTP wrapper for file operations\n */\n async getSFTP(): Promise<SFTPWrapper> {\n if (!this.connected) {\n await this.connect();\n }\n\n return new Promise((resolve, reject) => {\n this.client.sftp((err, sftp) => {\n if (err) {\n reject(err);\n } else {\n resolve(sftp);\n }\n });\n });\n }\n\n /**\n * List files in a directory using SFTP\n */\n async listFiles(path: string): Promise<Array<{ name: string; isDirectory: boolean }>> {\n const sftp = await this.getSFTP();\n\n return new Promise((resolve, reject) => {\n sftp.readdir(path, (err, list) => {\n if (err) {\n reject(err);\n return;\n }\n\n const files = list.map((item) => ({\n name: item.filename,\n isDirectory: item.attrs.isDirectory(),\n }));\n\n resolve(files);\n });\n });\n }\n\n /**\n * Read file contents from remote machine\n */\n async readFile(\n path: string,\n encoding: string = 'utf-8',\n maxSize: number = 1048576\n ): Promise<{ content: string; size: number; encoding: string }> {\n const sftp = await this.getSFTP();\n\n return new Promise((resolve, reject) => {\n // First, get file stats to check size\n sftp.stat(path, (err, stats) => {\n if (err) {\n reject(new Error(`File not found or inaccessible: ${path}`));\n return;\n }\n\n if (stats.size > maxSize) {\n reject(new Error(`File too large: ${stats.size} bytes (max: ${maxSize} bytes)`));\n return;\n }\n\n // Read file contents\n sftp.readFile(path, { encoding: encoding as BufferEncoding }, (err, data) => {\n if (err) {\n reject(new Error(`Failed to read file: ${err.message}`));\n return;\n }\n\n resolve({\n content: data.toString(),\n size: stats.size,\n encoding,\n });\n });\n });\n });\n }\n\n /**\n * Write file contents to remote machine\n */\n async writeFile(\n path: string,\n content: string,\n options: {\n encoding?: string;\n createDirs?: boolean;\n backup?: boolean;\n } = {}\n ): Promise<{ success: boolean; bytesWritten: number; backupPath?: string }> {\n const { encoding = 'utf-8', createDirs = false, backup = false } = options;\n const sftp = await this.getSFTP();\n\n return new Promise((resolve, reject) => {\n const writeOperation = () => {\n // Create backup if requested\n if (backup) {\n const backupPath = `${path}.backup`;\n sftp.rename(path, backupPath, (err) => {\n if (err && err.message !== 'No such file') {\n // Ignore \"no such file\" error (file doesn't exist yet)\n reject(new Error(`Failed to create backup: ${err.message}`));\n return;\n }\n \n // Write file\n performWrite(backupPath);\n });\n } else {\n performWrite();\n }\n };\n\n const performWrite = (backupPath?: string) => {\n const buffer = Buffer.from(content, encoding as BufferEncoding);\n const tempPath = `${path}.tmp`;\n\n // Write to temp file first (atomic write)\n sftp.writeFile(tempPath, buffer, (err) => {\n if (err) {\n reject(new Error(`Failed to write file: ${err.message}`));\n return;\n }\n\n // Rename temp file to target (atomic operation)\n sftp.rename(tempPath, path, (err) => {\n if (err) {\n reject(new Error(`Failed to rename temp file: ${err.message}`));\n return;\n }\n\n resolve({\n success: true,\n bytesWritten: buffer.length,\n backupPath,\n });\n });\n });\n };\n\n // Create parent directories if requested\n if (createDirs) {\n const dirPath = path.substring(0, path.lastIndexOf('/'));\n this.exec(`mkdir -p ${dirPath}`).then(() => {\n writeOperation();\n }).catch(reject);\n } else {\n writeOperation();\n }\n });\n }\n\n /**\n * Disconnect from the SSH server\n */\n disconnect(): void {\n if (this.connected) {\n this.client.end();\n this.connected = false;\n }\n }\n\n /**\n * Check if connected\n */\n isConnected(): boolean {\n return this.connected;\n }\n}\n"],
|
|
5
|
-
"mappings": "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",
|
|
3
|
+
"sources": ["../node_modules/dotenv/package.json", "../node_modules/dotenv/lib/main.js", "../src/server.ts", "../src/config.ts", "../src/tools/acp-remote-list-files.ts", "../src/utils/logger.ts", "../src/tools/acp-remote-execute-command.ts", "../src/tools/acp-remote-read-file.ts", "../src/tools/acp-remote-write-file.ts", "../src/utils/ssh-connection.ts"],
|
|
4
|
+
"sourcesContent": ["{\n \"name\": \"dotenv\",\n \"version\": \"16.6.1\",\n \"description\": \"Loads environment variables from .env file\",\n \"main\": \"lib/main.js\",\n \"types\": \"lib/main.d.ts\",\n \"exports\": {\n \".\": {\n \"types\": \"./lib/main.d.ts\",\n \"require\": \"./lib/main.js\",\n \"default\": \"./lib/main.js\"\n },\n \"./config\": \"./config.js\",\n \"./config.js\": \"./config.js\",\n \"./lib/env-options\": \"./lib/env-options.js\",\n \"./lib/env-options.js\": \"./lib/env-options.js\",\n \"./lib/cli-options\": \"./lib/cli-options.js\",\n \"./lib/cli-options.js\": \"./lib/cli-options.js\",\n \"./package.json\": \"./package.json\"\n },\n \"scripts\": {\n \"dts-check\": \"tsc --project tests/types/tsconfig.json\",\n \"lint\": \"standard\",\n \"pretest\": \"npm run lint && npm run dts-check\",\n \"test\": \"tap run --allow-empty-coverage --disable-coverage --timeout=60000\",\n \"test:coverage\": \"tap run --show-full-coverage --timeout=60000 --coverage-report=text --coverage-report=lcov\",\n \"prerelease\": \"npm test\",\n \"release\": \"standard-version\"\n },\n \"repository\": {\n \"type\": \"git\",\n \"url\": \"git://github.com/motdotla/dotenv.git\"\n },\n \"homepage\": \"https://github.com/motdotla/dotenv#readme\",\n \"funding\": \"https://dotenvx.com\",\n \"keywords\": [\n \"dotenv\",\n \"env\",\n \".env\",\n \"environment\",\n \"variables\",\n \"config\",\n \"settings\"\n ],\n \"readmeFilename\": \"README.md\",\n \"license\": \"BSD-2-Clause\",\n \"devDependencies\": {\n \"@types/node\": \"^18.11.3\",\n \"decache\": \"^4.6.2\",\n \"sinon\": \"^14.0.1\",\n \"standard\": \"^17.0.0\",\n \"standard-version\": \"^9.5.0\",\n \"tap\": \"^19.2.0\",\n \"typescript\": \"^4.8.4\"\n },\n \"engines\": {\n \"node\": \">=12\"\n },\n \"browser\": {\n \"fs\": false\n }\n}\n", "const fs = require('fs')\nconst path = require('path')\nconst os = require('os')\nconst crypto = require('crypto')\nconst packageJson = require('../package.json')\n\nconst version = packageJson.version\n\nconst LINE = /(?:^|^)\\s*(?:export\\s+)?([\\w.-]+)(?:\\s*=\\s*?|:\\s+?)(\\s*'(?:\\\\'|[^'])*'|\\s*\"(?:\\\\\"|[^\"])*\"|\\s*`(?:\\\\`|[^`])*`|[^#\\r\\n]+)?\\s*(?:#.*)?(?:$|$)/mg\n\n// Parse src into an Object\nfunction parse (src) {\n const obj = {}\n\n // Convert buffer to string\n let lines = src.toString()\n\n // Convert line breaks to same format\n lines = lines.replace(/\\r\\n?/mg, '\\n')\n\n let match\n while ((match = LINE.exec(lines)) != null) {\n const key = match[1]\n\n // Default undefined or null to empty string\n let value = (match[2] || '')\n\n // Remove whitespace\n value = value.trim()\n\n // Check if double quoted\n const maybeQuote = value[0]\n\n // Remove surrounding quotes\n value = value.replace(/^(['\"`])([\\s\\S]*)\\1$/mg, '$2')\n\n // Expand newlines if double quoted\n if (maybeQuote === '\"') {\n value = value.replace(/\\\\n/g, '\\n')\n value = value.replace(/\\\\r/g, '\\r')\n }\n\n // Add to object\n obj[key] = value\n }\n\n return obj\n}\n\nfunction _parseVault (options) {\n options = options || {}\n\n const vaultPath = _vaultPath(options)\n options.path = vaultPath // parse .env.vault\n const result = DotenvModule.configDotenv(options)\n if (!result.parsed) {\n const err = new Error(`MISSING_DATA: Cannot parse ${vaultPath} for an unknown reason`)\n err.code = 'MISSING_DATA'\n throw err\n }\n\n // handle scenario for comma separated keys - for use with key rotation\n // example: DOTENV_KEY=\"dotenv://:key_1234@dotenvx.com/vault/.env.vault?environment=prod,dotenv://:key_7890@dotenvx.com/vault/.env.vault?environment=prod\"\n const keys = _dotenvKey(options).split(',')\n const length = keys.length\n\n let decrypted\n for (let i = 0; i < length; i++) {\n try {\n // Get full key\n const key = keys[i].trim()\n\n // Get instructions for decrypt\n const attrs = _instructions(result, key)\n\n // Decrypt\n decrypted = DotenvModule.decrypt(attrs.ciphertext, attrs.key)\n\n break\n } catch (error) {\n // last key\n if (i + 1 >= length) {\n throw error\n }\n // try next key\n }\n }\n\n // Parse decrypted .env string\n return DotenvModule.parse(decrypted)\n}\n\nfunction _warn (message) {\n console.log(`[dotenv@${version}][WARN] ${message}`)\n}\n\nfunction _debug (message) {\n console.log(`[dotenv@${version}][DEBUG] ${message}`)\n}\n\nfunction _log (message) {\n console.log(`[dotenv@${version}] ${message}`)\n}\n\nfunction _dotenvKey (options) {\n // prioritize developer directly setting options.DOTENV_KEY\n if (options && options.DOTENV_KEY && options.DOTENV_KEY.length > 0) {\n return options.DOTENV_KEY\n }\n\n // secondary infra already contains a DOTENV_KEY environment variable\n if (process.env.DOTENV_KEY && process.env.DOTENV_KEY.length > 0) {\n return process.env.DOTENV_KEY\n }\n\n // fallback to empty string\n return ''\n}\n\nfunction _instructions (result, dotenvKey) {\n // Parse DOTENV_KEY. Format is a URI\n let uri\n try {\n uri = new URL(dotenvKey)\n } catch (error) {\n if (error.code === 'ERR_INVALID_URL') {\n const err = new Error('INVALID_DOTENV_KEY: Wrong format. Must be in valid uri format like dotenv://:key_1234@dotenvx.com/vault/.env.vault?environment=development')\n err.code = 'INVALID_DOTENV_KEY'\n throw err\n }\n\n throw error\n }\n\n // Get decrypt key\n const key = uri.password\n if (!key) {\n const err = new Error('INVALID_DOTENV_KEY: Missing key part')\n err.code = 'INVALID_DOTENV_KEY'\n throw err\n }\n\n // Get environment\n const environment = uri.searchParams.get('environment')\n if (!environment) {\n const err = new Error('INVALID_DOTENV_KEY: Missing environment part')\n err.code = 'INVALID_DOTENV_KEY'\n throw err\n }\n\n // Get ciphertext payload\n const environmentKey = `DOTENV_VAULT_${environment.toUpperCase()}`\n const ciphertext = result.parsed[environmentKey] // DOTENV_VAULT_PRODUCTION\n if (!ciphertext) {\n const err = new Error(`NOT_FOUND_DOTENV_ENVIRONMENT: Cannot locate environment ${environmentKey} in your .env.vault file.`)\n err.code = 'NOT_FOUND_DOTENV_ENVIRONMENT'\n throw err\n }\n\n return { ciphertext, key }\n}\n\nfunction _vaultPath (options) {\n let possibleVaultPath = null\n\n if (options && options.path && options.path.length > 0) {\n if (Array.isArray(options.path)) {\n for (const filepath of options.path) {\n if (fs.existsSync(filepath)) {\n possibleVaultPath = filepath.endsWith('.vault') ? filepath : `${filepath}.vault`\n }\n }\n } else {\n possibleVaultPath = options.path.endsWith('.vault') ? options.path : `${options.path}.vault`\n }\n } else {\n possibleVaultPath = path.resolve(process.cwd(), '.env.vault')\n }\n\n if (fs.existsSync(possibleVaultPath)) {\n return possibleVaultPath\n }\n\n return null\n}\n\nfunction _resolveHome (envPath) {\n return envPath[0] === '~' ? path.join(os.homedir(), envPath.slice(1)) : envPath\n}\n\nfunction _configVault (options) {\n const debug = Boolean(options && options.debug)\n const quiet = options && 'quiet' in options ? options.quiet : true\n\n if (debug || !quiet) {\n _log('Loading env from encrypted .env.vault')\n }\n\n const parsed = DotenvModule._parseVault(options)\n\n let processEnv = process.env\n if (options && options.processEnv != null) {\n processEnv = options.processEnv\n }\n\n DotenvModule.populate(processEnv, parsed, options)\n\n return { parsed }\n}\n\nfunction configDotenv (options) {\n const dotenvPath = path.resolve(process.cwd(), '.env')\n let encoding = 'utf8'\n const debug = Boolean(options && options.debug)\n const quiet = options && 'quiet' in options ? options.quiet : true\n\n if (options && options.encoding) {\n encoding = options.encoding\n } else {\n if (debug) {\n _debug('No encoding is specified. UTF-8 is used by default')\n }\n }\n\n let optionPaths = [dotenvPath] // default, look for .env\n if (options && options.path) {\n if (!Array.isArray(options.path)) {\n optionPaths = [_resolveHome(options.path)]\n } else {\n optionPaths = [] // reset default\n for (const filepath of options.path) {\n optionPaths.push(_resolveHome(filepath))\n }\n }\n }\n\n // Build the parsed data in a temporary object (because we need to return it). Once we have the final\n // parsed data, we will combine it with process.env (or options.processEnv if provided).\n let lastError\n const parsedAll = {}\n for (const path of optionPaths) {\n try {\n // Specifying an encoding returns a string instead of a buffer\n const parsed = DotenvModule.parse(fs.readFileSync(path, { encoding }))\n\n DotenvModule.populate(parsedAll, parsed, options)\n } catch (e) {\n if (debug) {\n _debug(`Failed to load ${path} ${e.message}`)\n }\n lastError = e\n }\n }\n\n let processEnv = process.env\n if (options && options.processEnv != null) {\n processEnv = options.processEnv\n }\n\n DotenvModule.populate(processEnv, parsedAll, options)\n\n if (debug || !quiet) {\n const keysCount = Object.keys(parsedAll).length\n const shortPaths = []\n for (const filePath of optionPaths) {\n try {\n const relative = path.relative(process.cwd(), filePath)\n shortPaths.push(relative)\n } catch (e) {\n if (debug) {\n _debug(`Failed to load ${filePath} ${e.message}`)\n }\n lastError = e\n }\n }\n\n _log(`injecting env (${keysCount}) from ${shortPaths.join(',')}`)\n }\n\n if (lastError) {\n return { parsed: parsedAll, error: lastError }\n } else {\n return { parsed: parsedAll }\n }\n}\n\n// Populates process.env from .env file\nfunction config (options) {\n // fallback to original dotenv if DOTENV_KEY is not set\n if (_dotenvKey(options).length === 0) {\n return DotenvModule.configDotenv(options)\n }\n\n const vaultPath = _vaultPath(options)\n\n // dotenvKey exists but .env.vault file does not exist\n if (!vaultPath) {\n _warn(`You set DOTENV_KEY but you are missing a .env.vault file at ${vaultPath}. Did you forget to build it?`)\n\n return DotenvModule.configDotenv(options)\n }\n\n return DotenvModule._configVault(options)\n}\n\nfunction decrypt (encrypted, keyStr) {\n const key = Buffer.from(keyStr.slice(-64), 'hex')\n let ciphertext = Buffer.from(encrypted, 'base64')\n\n const nonce = ciphertext.subarray(0, 12)\n const authTag = ciphertext.subarray(-16)\n ciphertext = ciphertext.subarray(12, -16)\n\n try {\n const aesgcm = crypto.createDecipheriv('aes-256-gcm', key, nonce)\n aesgcm.setAuthTag(authTag)\n return `${aesgcm.update(ciphertext)}${aesgcm.final()}`\n } catch (error) {\n const isRange = error instanceof RangeError\n const invalidKeyLength = error.message === 'Invalid key length'\n const decryptionFailed = error.message === 'Unsupported state or unable to authenticate data'\n\n if (isRange || invalidKeyLength) {\n const err = new Error('INVALID_DOTENV_KEY: It must be 64 characters long (or more)')\n err.code = 'INVALID_DOTENV_KEY'\n throw err\n } else if (decryptionFailed) {\n const err = new Error('DECRYPTION_FAILED: Please check your DOTENV_KEY')\n err.code = 'DECRYPTION_FAILED'\n throw err\n } else {\n throw error\n }\n }\n}\n\n// Populate process.env with parsed values\nfunction populate (processEnv, parsed, options = {}) {\n const debug = Boolean(options && options.debug)\n const override = Boolean(options && options.override)\n\n if (typeof parsed !== 'object') {\n const err = new Error('OBJECT_REQUIRED: Please check the processEnv argument being passed to populate')\n err.code = 'OBJECT_REQUIRED'\n throw err\n }\n\n // Set process.env\n for (const key of Object.keys(parsed)) {\n if (Object.prototype.hasOwnProperty.call(processEnv, key)) {\n if (override === true) {\n processEnv[key] = parsed[key]\n }\n\n if (debug) {\n if (override === true) {\n _debug(`\"${key}\" is already defined and WAS overwritten`)\n } else {\n _debug(`\"${key}\" is already defined and was NOT overwritten`)\n }\n }\n } else {\n processEnv[key] = parsed[key]\n }\n }\n}\n\nconst DotenvModule = {\n configDotenv,\n _configVault,\n _parseVault,\n config,\n decrypt,\n parse,\n populate\n}\n\nmodule.exports.configDotenv = DotenvModule.configDotenv\nmodule.exports._configVault = DotenvModule._configVault\nmodule.exports._parseVault = DotenvModule._parseVault\nmodule.exports.config = DotenvModule.config\nmodule.exports.decrypt = DotenvModule.decrypt\nmodule.exports.parse = DotenvModule.parse\nmodule.exports.populate = DotenvModule.populate\n\nmodule.exports = DotenvModule\n", "#!/usr/bin/env node\n\nimport { Server } from '@modelcontextprotocol/sdk/server/index.js';\nimport { StdioServerTransport } from '@modelcontextprotocol/sdk/server/stdio.js';\nimport {\n CallToolRequestSchema,\n ListToolsRequestSchema,\n} from '@modelcontextprotocol/sdk/types.js';\nimport { config, loadSSHPrivateKey } from './config.js';\nimport { acpRemoteListFilesTool, handleAcpRemoteListFiles } from './tools/acp-remote-list-files.js';\nimport { acpRemoteExecuteCommandTool, handleAcpRemoteExecuteCommand } from './tools/acp-remote-execute-command.js';\nimport { acpRemoteReadFileTool, handleAcpRemoteReadFile } from './tools/acp-remote-read-file.js';\nimport { acpRemoteWriteFileTool, handleAcpRemoteWriteFile } from './tools/acp-remote-write-file.js';\nimport { SSHConnectionManager } from './utils/ssh-connection.js';\nimport { logger } from './utils/logger.js';\n\nasync function main() {\n // Load SSH private key\n const privateKey = loadSSHPrivateKey();\n\n // Create SSH connection manager\n const sshConnection = new SSHConnectionManager({\n host: config.ssh.host,\n port: config.ssh.port,\n username: config.ssh.username,\n privateKey: privateKey,\n });\n\n // Connect to remote server\n await sshConnection.connect();\n console.error(`Connected to SSH server: ${config.ssh.username}@${config.ssh.host}`);\n\n const server = new Server(\n {\n name: 'acp-mcp',\n version: '0.1.0',\n },\n {\n capabilities: {\n tools: {},\n },\n }\n );\n\n // Register tools\n server.setRequestHandler(ListToolsRequestSchema, async () => {\n logger.debug('Tool discovery requested');\n const tools = [acpRemoteListFilesTool, acpRemoteExecuteCommandTool, acpRemoteReadFileTool, acpRemoteWriteFileTool];\n logger.debug(`Returning ${tools.length} tools`, { tools: tools.map(t => t.name) });\n return { tools };\n });\n\n server.setRequestHandler(CallToolRequestSchema, async (request) => {\n const startTime = Date.now();\n logger.toolInvoked(request.params.name, request.params.arguments);\n \n try {\n let result;\n \n if (request.params.name === 'acp_remote_list_files') {\n result = await handleAcpRemoteListFiles(request.params.arguments, sshConnection);\n } else if (request.params.name === 'acp_remote_execute_command') {\n result = await handleAcpRemoteExecuteCommand(request.params.arguments, sshConnection);\n } else if (request.params.name === 'acp_remote_read_file') {\n result = await handleAcpRemoteReadFile(request.params.arguments, sshConnection);\n } else if (request.params.name === 'acp_remote_write_file') {\n result = await handleAcpRemoteWriteFile(request.params.arguments, sshConnection);\n } else {\n throw new Error(`Unknown tool: ${request.params.name}`);\n }\n \n const duration = Date.now() - startTime;\n const resultSize = JSON.stringify(result).length;\n logger.toolCompleted(request.params.name, duration, resultSize);\n \n return result;\n } catch (error) {\n logger.toolFailed(request.params.name, error as Error, request.params.arguments);\n throw error;\n }\n });\n\n // Handle shutdown to cleanup SSH connection\n const cleanup = () => {\n logger.info('Shutting down server');\n sshConnection.disconnect();\n process.exit(0);\n };\n\n process.on('SIGINT', cleanup);\n process.on('SIGTERM', cleanup);\n\n // Start server\n const transport = new StdioServerTransport();\n await server.connect(transport);\n logger.info('ACP MCP Server running on stdio');\n}\n\nmain().catch((error) => {\n logger.error('Server startup failed', { error: error.message, stack: error.stack });\n process.exit(1);\n});\n", "import dotenv from 'dotenv';\nimport { readFileSync } from 'fs';\n\ndotenv.config();\n\nexport interface Config {\n logLevel: string;\n ssh: {\n host: string;\n port: number;\n username: string;\n privateKeyPath: string;\n };\n}\n\nfunction validateConfig(): Config {\n const sshHost = process.env.SSH_HOST;\n const sshUsername = process.env.SSH_USERNAME;\n const sshPrivateKeyPath = process.env.SSH_PRIVATE_KEY_PATH;\n\n if (!sshHost || !sshUsername || !sshPrivateKeyPath) {\n throw new Error(\n 'Missing required SSH configuration. Please set SSH_HOST, SSH_USERNAME, and SSH_PRIVATE_KEY_PATH environment variables.'\n );\n }\n\n return {\n logLevel: process.env.LOG_LEVEL || 'info',\n ssh: {\n host: sshHost,\n port: parseInt(process.env.SSH_PORT || '22', 10),\n username: sshUsername,\n privateKeyPath: sshPrivateKeyPath,\n },\n };\n}\n\nexport const config = validateConfig();\n\n/**\n * Load SSH private key from file\n */\nexport function loadSSHPrivateKey(): string {\n try {\n return readFileSync(config.ssh.privateKeyPath, 'utf-8');\n } catch (error) {\n throw new Error(\n `Failed to load SSH private key from ${config.ssh.privateKeyPath}: ${error instanceof Error ? error.message : String(error)}`\n );\n }\n}\n", "import { Tool } from '@modelcontextprotocol/sdk/types.js';\nimport { SSHConnectionManager } from '../utils/ssh-connection.js';\n\nexport const acpRemoteListFilesTool: Tool = {\n name: 'acp_remote_list_files',\n description: 'List files and directories in a specified path on the remote machine via SSH',\n inputSchema: {\n type: 'object',\n properties: {\n path: {\n type: 'string',\n description: 'The directory path to list files from',\n },\n recursive: {\n type: 'boolean',\n description: 'Whether to list files recursively',\n default: false,\n },\n },\n required: ['path'],\n },\n};\n\ninterface ListFilesArgs {\n path: string;\n recursive?: boolean;\n}\n\n/**\n * Handle the acp_remote_list_files tool invocation\n * Lists files and directories at the specified path on the remote machine via SSH\n * \n * @param args - Tool arguments containing path and recursive flag\n * @param sshConnection - SSH connection manager for remote operations\n */\nexport async function handleAcpRemoteListFiles(\n args: any,\n sshConnection: SSHConnectionManager\n): Promise<{ content: Array<{ type: string; text: string }> }> {\n const { path, recursive = false } = args as ListFilesArgs;\n\n try {\n const files = await listRemoteFiles(sshConnection, path, recursive);\n \n return {\n content: [\n {\n type: 'text',\n text: files.join('\\n'),\n },\n ],\n };\n } catch (error) {\n const errorMessage = error instanceof Error ? error.message : String(error);\n return {\n content: [\n {\n type: 'text',\n text: `Error listing remote files: ${errorMessage}`,\n },\n ],\n };\n }\n}\n\n/**\n * Recursively list files in a remote directory via SSH\n * Returns absolute paths for compatibility with acp_remote_read_file\n */\nasync function listRemoteFiles(\n ssh: SSHConnectionManager,\n dirPath: string,\n recursive: boolean\n): Promise<string[]> {\n const entries = await ssh.listFiles(dirPath);\n const files: string[] = [];\n\n for (const entry of entries) {\n // Construct absolute path by combining dirPath with entry name\n const fullPath = `${dirPath}/${entry.name}`.replace(/\\/+/g, '/');\n \n if (entry.isDirectory) {\n files.push(`${fullPath}/`); // Return absolute path with trailing slash\n if (recursive) {\n const subFiles = await listRemoteFiles(ssh, fullPath, recursive);\n files.push(...subFiles); // Sub-files already have absolute paths\n }\n } else {\n files.push(fullPath); // Return absolute path\n }\n }\n\n return files.sort();\n}\n", "/**\n * Logger utility for ACP MCP Server\n * Provides structured logging with configurable log levels\n */\n\nexport type LogLevel = 'error' | 'warn' | 'info' | 'debug' | 'trace';\n\nconst LOG_LEVELS: Record<LogLevel, number> = {\n error: 0,\n warn: 1,\n info: 2,\n debug: 3,\n trace: 4,\n};\n\nclass Logger {\n private level: LogLevel;\n private enabled: boolean;\n\n constructor() {\n // Read configuration from environment variables\n this.level = (process.env.ACP_MCP_LOG_LEVEL as LogLevel) || 'info';\n this.enabled = process.env.ACP_MCP_DEBUG === 'true' || process.env.NODE_ENV === 'development';\n }\n\n private shouldLog(level: LogLevel): boolean {\n if (!this.enabled && level !== 'error' && level !== 'warn') {\n return false;\n }\n return LOG_LEVELS[level] <= LOG_LEVELS[this.level];\n }\n\n private formatMessage(level: LogLevel, message: string, data?: any): string {\n const timestamp = new Date().toISOString();\n const prefix = `[${timestamp}] [${level.toUpperCase()}]`;\n \n if (data !== undefined) {\n const dataStr = typeof data === 'object' ? JSON.stringify(data, null, 2) : String(data);\n return `${prefix} ${message}\\n${dataStr}`;\n }\n \n return `${prefix} ${message}`;\n }\n\n error(message: string, data?: any): void {\n if (this.shouldLog('error')) {\n console.error(this.formatMessage('error', message, data));\n }\n }\n\n warn(message: string, data?: any): void {\n if (this.shouldLog('warn')) {\n console.error(this.formatMessage('warn', message, data));\n }\n }\n\n info(message: string, data?: any): void {\n if (this.shouldLog('info')) {\n console.error(this.formatMessage('info', message, data));\n }\n }\n\n debug(message: string, data?: any): void {\n if (this.shouldLog('debug')) {\n console.error(this.formatMessage('debug', message, data));\n }\n }\n\n trace(message: string, data?: any): void {\n if (this.shouldLog('trace')) {\n console.error(this.formatMessage('trace', message, data));\n }\n }\n\n /**\n * Log tool invocation with parameters\n */\n toolInvoked(toolName: string, params: any, userId?: string): void {\n this.info(`Tool invoked: ${toolName}`);\n this.debug('Tool parameters', { tool: toolName, params, userId });\n }\n\n /**\n * Log tool completion with result summary\n */\n toolCompleted(toolName: string, duration: number, resultSize?: number): void {\n this.info(`Tool completed: ${toolName}`);\n this.debug('Tool performance', { tool: toolName, duration: `${duration}ms`, resultSize });\n }\n\n /**\n * Log tool failure with error details\n */\n toolFailed(toolName: string, error: Error, params?: any): void {\n this.error(`Tool execution failed: ${toolName}`, {\n tool: toolName,\n error: error.message,\n stack: error.stack,\n params,\n });\n }\n\n /**\n * Log SSH command execution\n */\n sshCommand(command: string, cwd?: string, timeout?: number): void {\n this.debug('Executing SSH command', { command, cwd, timeout });\n }\n\n /**\n * Log SSH command result\n */\n sshCommandResult(exitCode: number, duration: number, stdoutSize: number, stderrSize: number): void {\n this.debug('SSH command completed', {\n exitCode,\n duration: `${duration}ms`,\n stdout: `${stdoutSize} bytes`,\n stderr: `${stderrSize} bytes`,\n });\n }\n\n /**\n * Log file operation\n */\n fileOperation(operation: string, path: string, details?: any): void {\n this.info(`File operation: ${operation}`, { path, ...details });\n }\n}\n\n// Export singleton instance\nexport const logger = new Logger();\n", "import { Tool } from '@modelcontextprotocol/sdk/types.js';\nimport { SSHConnectionManager } from '../utils/ssh-connection.js';\nimport { logger } from '../utils/logger.js';\n\nexport const acpRemoteExecuteCommandTool: Tool = {\n name: 'acp_remote_execute_command',\n description: 'Execute a shell command on the remote machine via SSH',\n inputSchema: {\n type: 'object',\n properties: {\n command: {\n type: 'string',\n description: 'Shell command to execute',\n },\n cwd: {\n type: 'string',\n description: 'Working directory for command execution (optional)',\n },\n timeout: {\n type: 'number',\n description: 'Timeout in seconds (default: 30)',\n default: 30,\n },\n },\n required: ['command'],\n },\n};\n\ninterface ExecuteCommandArgs {\n command: string;\n cwd?: string;\n timeout?: number;\n}\n\ninterface ExecuteCommandResult {\n stdout: string;\n stderr: string;\n exitCode: number;\n timedOut: boolean;\n}\n\n/**\n * Handle the acp_remote_execute_command tool invocation\n * Executes a shell command on the remote machine via SSH\n * \n * @param args - Tool arguments containing command, cwd, and timeout\n * @param sshConnection - SSH connection manager for remote operations\n */\nexport async function handleAcpRemoteExecuteCommand(\n args: any,\n sshConnection: SSHConnectionManager\n): Promise<{ content: Array<{ type: string; text: string }> }> {\n const { command, cwd, timeout = 30 } = args as ExecuteCommandArgs;\n\n logger.debug('Executing remote command', { command, cwd, timeout });\n\n try {\n // Build command with working directory if specified\n const fullCommand = cwd ? `cd ${cwd} && ${command}` : command;\n \n const result = await sshConnection.execWithTimeout(fullCommand, timeout);\n \n logger.debug('Command execution result', {\n exitCode: result.exitCode,\n timedOut: result.timedOut,\n stdoutLength: result.stdout.length,\n stderrLength: result.stderr.length,\n });\n \n // Format output as JSON for structured response\n const output: ExecuteCommandResult = {\n stdout: result.stdout,\n stderr: result.stderr,\n exitCode: result.exitCode,\n timedOut: result.timedOut,\n };\n \n return {\n content: [\n {\n type: 'text',\n text: JSON.stringify(output, null, 2),\n },\n ],\n };\n } catch (error) {\n const errorMessage = error instanceof Error ? error.message : String(error);\n logger.error('Command execution error', { command, error: errorMessage });\n return {\n content: [\n {\n type: 'text',\n text: JSON.stringify({\n stdout: '',\n stderr: errorMessage,\n exitCode: 1,\n timedOut: false,\n }, null, 2),\n },\n ],\n };\n }\n}\n", "import { Tool } from '@modelcontextprotocol/sdk/types.js';\nimport { SSHConnectionManager } from '../utils/ssh-connection.js';\nimport { logger } from '../utils/logger.js';\n\nexport const acpRemoteReadFileTool: Tool = {\n name: 'acp_remote_read_file',\n description: 'Read file contents from the remote machine via SSH',\n inputSchema: {\n type: 'object',\n properties: {\n path: {\n type: 'string',\n description: 'Absolute path to file',\n },\n encoding: {\n type: 'string',\n description: 'File encoding (default: utf-8)',\n default: 'utf-8',\n enum: ['utf-8', 'ascii', 'base64'],\n },\n maxSize: {\n type: 'number',\n description: 'Max file size in bytes (default: 1MB)',\n default: 1048576,\n },\n },\n required: ['path'],\n },\n};\n\ninterface ReadFileArgs {\n path: string;\n encoding?: string;\n maxSize?: number;\n}\n\ninterface ReadFileResult {\n content: string;\n size: number;\n encoding: string;\n}\n\n/**\n * Handle the acp_remote_read_file tool invocation\n * Reads file contents from the remote machine via SSH\n * \n * @param args - Tool arguments containing path, encoding, and maxSize\n * @param sshConnection - SSH connection manager for remote operations\n */\nexport async function handleAcpRemoteReadFile(\n args: any,\n sshConnection: SSHConnectionManager\n): Promise<{ content: Array<{ type: string; text: string }> }> {\n const { path, encoding = 'utf-8', maxSize = 1048576 } = args as ReadFileArgs;\n\n logger.debug('Reading remote file', { path, encoding, maxSize });\n\n try {\n const result = await sshConnection.readFile(path, encoding, maxSize);\n \n logger.debug('File read successful', { path, size: result.size });\n \n const output: ReadFileResult = {\n content: result.content,\n size: result.size,\n encoding: result.encoding,\n };\n \n return {\n content: [\n {\n type: 'text',\n text: JSON.stringify(output, null, 2),\n },\n ],\n };\n } catch (error) {\n const errorMessage = error instanceof Error ? error.message : String(error);\n logger.error('File read error', { path, error: errorMessage });\n return {\n content: [\n {\n type: 'text',\n text: JSON.stringify({\n error: errorMessage,\n content: '',\n size: 0,\n encoding,\n }, null, 2),\n },\n ],\n };\n }\n}\n", "import { Tool } from '@modelcontextprotocol/sdk/types.js';\nimport { SSHConnectionManager } from '../utils/ssh-connection.js';\nimport { logger } from '../utils/logger.js';\n\nexport const acpRemoteWriteFileTool: Tool = {\n name: 'acp_remote_write_file',\n description: 'Write file contents to the remote machine via SSH',\n inputSchema: {\n type: 'object',\n properties: {\n path: {\n type: 'string',\n description: 'Absolute path to file',\n },\n content: {\n type: 'string',\n description: 'File contents to write',\n },\n encoding: {\n type: 'string',\n description: 'File encoding (default: utf-8)',\n default: 'utf-8',\n },\n createDirs: {\n type: 'boolean',\n description: 'Create parent directories if they don\\'t exist (default: false)',\n default: false,\n },\n backup: {\n type: 'boolean',\n description: 'Backup existing file before overwriting (default: false)',\n default: false,\n },\n },\n required: ['path', 'content'],\n },\n};\n\ninterface WriteFileArgs {\n path: string;\n content: string;\n encoding?: string;\n createDirs?: boolean;\n backup?: boolean;\n}\n\ninterface WriteFileResult {\n success: boolean;\n bytesWritten: number;\n backupPath?: string;\n}\n\n/**\n * Handle the acp_remote_write_file tool invocation\n * Writes file contents to the remote machine via SSH\n * \n * @param args - Tool arguments containing path, content, and options\n * @param sshConnection - SSH connection manager for remote operations\n */\nexport async function handleAcpRemoteWriteFile(\n args: any,\n sshConnection: SSHConnectionManager\n): Promise<{ content: Array<{ type: string; text: string }> }> {\n const { path, content, encoding = 'utf-8', createDirs = false, backup = false } = args as WriteFileArgs;\n\n logger.debug('Writing remote file', { path, contentSize: content.length, encoding, createDirs, backup });\n\n try {\n const result = await sshConnection.writeFile(path, content, {\n encoding,\n createDirs,\n backup,\n });\n \n logger.debug('File write successful', { path, bytesWritten: result.bytesWritten, backupPath: result.backupPath });\n \n const output: WriteFileResult = {\n success: result.success,\n bytesWritten: result.bytesWritten,\n backupPath: result.backupPath,\n };\n \n return {\n content: [\n {\n type: 'text',\n text: JSON.stringify(output, null, 2),\n },\n ],\n };\n } catch (error) {\n const errorMessage = error instanceof Error ? error.message : String(error);\n logger.error('File write error', { path, error: errorMessage });\n return {\n content: [\n {\n type: 'text',\n text: JSON.stringify({\n success: false,\n bytesWritten: 0,\n error: errorMessage,\n }, null, 2),\n },\n ],\n };\n }\n}\n", "import { Client, SFTPWrapper } from 'ssh2';\nimport { SSHConfig } from '../types/ssh-config.js';\nimport { logger } from './logger.js';\n\n/**\n * SSH Connection Manager\n * Manages SSH connections and provides SFTP access for remote file operations\n */\nexport class SSHConnectionManager {\n private client: Client;\n private config: SSHConfig;\n private connected: boolean = false;\n\n constructor(config: SSHConfig) {\n this.config = config;\n this.client = new Client();\n }\n\n /**\n * Connect to the remote SSH server\n */\n async connect(): Promise<void> {\n if (this.connected) {\n logger.debug('SSH connection already established');\n return;\n }\n\n logger.info('Connecting to SSH server', {\n host: this.config.host,\n port: this.config.port || 22,\n username: this.config.username,\n });\n\n return new Promise((resolve, reject) => {\n this.client\n .on('ready', () => {\n this.connected = true;\n logger.info('SSH connection established', {\n host: this.config.host,\n username: this.config.username,\n });\n resolve();\n })\n .on('error', (err) => {\n logger.error('SSH connection failed', {\n host: this.config.host,\n error: err.message,\n });\n reject(err);\n })\n .connect({\n host: this.config.host,\n port: this.config.port || 22,\n username: this.config.username,\n privateKey: this.config.privateKey,\n });\n });\n }\n\n /**\n * Execute a command on the remote server\n */\n async exec(command: string): Promise<string> {\n if (!this.connected) {\n await this.connect();\n }\n\n return new Promise((resolve, reject) => {\n this.client.exec(command, (err, stream) => {\n if (err) {\n reject(err);\n return;\n }\n\n let stdout = '';\n let stderr = '';\n\n stream\n .on('close', (code: number) => {\n if (code !== 0) {\n reject(new Error(`Command failed with code ${code}: ${stderr}`));\n } else {\n resolve(stdout);\n }\n })\n .on('data', (data: Buffer) => {\n stdout += data.toString();\n })\n .stderr.on('data', (data: Buffer) => {\n stderr += data.toString();\n });\n });\n });\n }\n\n /**\n * Execute a command on the remote server with timeout support\n */\n async execWithTimeout(\n command: string,\n timeoutSeconds: number = 30\n ): Promise<{ stdout: string; stderr: string; exitCode: number; timedOut: boolean }> {\n if (!this.connected) {\n await this.connect();\n }\n\n const startTime = Date.now();\n logger.sshCommand(command, undefined, timeoutSeconds);\n\n const execPromise = new Promise<{ stdout: string; stderr: string; exitCode: number }>((resolve, reject) => {\n this.client.exec(command, (err, stream) => {\n if (err) {\n reject(err);\n return;\n }\n\n let stdout = '';\n let stderr = '';\n\n stream\n .on('close', (code: number) => {\n const duration = Date.now() - startTime;\n logger.sshCommandResult(code, duration, stdout.length, stderr.length);\n resolve({ stdout, stderr, exitCode: code });\n })\n .on('data', (data: Buffer) => {\n stdout += data.toString();\n })\n .stderr.on('data', (data: Buffer) => {\n stderr += data.toString();\n });\n });\n });\n\n const timeoutPromise = new Promise<never>((_, reject) => {\n setTimeout(() => {\n reject(new Error('Command execution timed out'));\n }, timeoutSeconds * 1000);\n });\n\n try {\n const result = await Promise.race([execPromise, timeoutPromise]);\n return { ...result, timedOut: false };\n } catch (error) {\n if (error instanceof Error && error.message === 'Command execution timed out') {\n logger.warn('SSH command timed out', { command, timeout: timeoutSeconds });\n return {\n stdout: '',\n stderr: 'Command execution timed out',\n exitCode: 124,\n timedOut: true,\n };\n }\n logger.error('SSH command execution failed', {\n command,\n error: error instanceof Error ? error.message : String(error),\n });\n throw error;\n }\n }\n\n /**\n * Get SFTP wrapper for file operations\n */\n async getSFTP(): Promise<SFTPWrapper> {\n if (!this.connected) {\n await this.connect();\n }\n\n return new Promise((resolve, reject) => {\n this.client.sftp((err, sftp) => {\n if (err) {\n reject(err);\n } else {\n resolve(sftp);\n }\n });\n });\n }\n\n /**\n * List files in a directory using SFTP\n */\n async listFiles(path: string): Promise<Array<{ name: string; isDirectory: boolean }>> {\n const sftp = await this.getSFTP();\n\n return new Promise((resolve, reject) => {\n sftp.readdir(path, (err, list) => {\n if (err) {\n reject(err);\n return;\n }\n\n const files = list.map((item) => ({\n name: item.filename,\n isDirectory: item.attrs.isDirectory(),\n }));\n\n resolve(files);\n });\n });\n }\n\n /**\n * Read file contents from remote machine\n */\n async readFile(\n path: string,\n encoding: string = 'utf-8',\n maxSize: number = 1048576\n ): Promise<{ content: string; size: number; encoding: string }> {\n const startTime = Date.now();\n logger.fileOperation('read', path, { encoding, maxSize });\n \n const sftp = await this.getSFTP();\n\n return new Promise((resolve, reject) => {\n // First, get file stats to check size\n sftp.stat(path, (err, stats) => {\n if (err) {\n logger.error('File stat failed', { path, error: err.message });\n reject(new Error(`File not found or inaccessible: ${path}`));\n return;\n }\n\n logger.debug('File stat retrieved', { path, size: stats.size });\n\n if (stats.size > maxSize) {\n logger.warn('File too large', { path, size: stats.size, maxSize });\n reject(new Error(`File too large: ${stats.size} bytes (max: ${maxSize} bytes)`));\n return;\n }\n\n // Read file contents\n sftp.readFile(path, { encoding: encoding as BufferEncoding }, (err, data) => {\n if (err) {\n logger.error('File read failed', { path, error: err.message });\n reject(new Error(`Failed to read file: ${err.message}`));\n return;\n }\n\n const duration = Date.now() - startTime;\n logger.debug('File read completed', { path, size: stats.size, duration: `${duration}ms` });\n\n resolve({\n content: data.toString(),\n size: stats.size,\n encoding,\n });\n });\n });\n });\n }\n\n /**\n * Write file contents to remote machine\n */\n async writeFile(\n path: string,\n content: string,\n options: {\n encoding?: string;\n createDirs?: boolean;\n backup?: boolean;\n } = {}\n ): Promise<{ success: boolean; bytesWritten: number; backupPath?: string }> {\n const { encoding = 'utf-8', createDirs = false, backup = false } = options;\n const startTime = Date.now();\n \n logger.fileOperation('write', path, {\n contentSize: content.length,\n encoding,\n createDirs,\n backup,\n });\n \n const sftp = await this.getSFTP();\n\n return new Promise((resolve, reject) => {\n const writeOperation = () => {\n // Create backup if requested\n if (backup) {\n const backupPath = `${path}.backup`;\n sftp.rename(path, backupPath, (err) => {\n if (err && err.message !== 'No such file') {\n // Ignore \"no such file\" error (file doesn't exist yet)\n reject(new Error(`Failed to create backup: ${err.message}`));\n return;\n }\n \n // Write file\n performWrite(backupPath);\n });\n } else {\n performWrite();\n }\n };\n\n const performWrite = (backupPath?: string) => {\n const buffer = Buffer.from(content, encoding as BufferEncoding);\n const tempPath = `${path}.tmp`;\n\n // Write to temp file first (atomic write)\n sftp.writeFile(tempPath, buffer, (err) => {\n if (err) {\n reject(new Error(`Failed to write file: ${err.message}`));\n return;\n }\n\n // Rename temp file to target (atomic operation)\n sftp.rename(tempPath, path, (err) => {\n if (err) {\n logger.error('File rename failed', { tempPath, path, error: err.message });\n reject(new Error(`Failed to rename temp file: ${err.message}`));\n return;\n }\n\n const duration = Date.now() - startTime;\n logger.debug('File write completed', {\n path,\n bytesWritten: buffer.length,\n duration: `${duration}ms`,\n backupPath,\n });\n\n resolve({\n success: true,\n bytesWritten: buffer.length,\n backupPath,\n });\n });\n });\n };\n\n // Create parent directories if requested\n if (createDirs) {\n const dirPath = path.substring(0, path.lastIndexOf('/'));\n this.exec(`mkdir -p ${dirPath}`).then(() => {\n writeOperation();\n }).catch(reject);\n } else {\n writeOperation();\n }\n });\n }\n\n /**\n * Disconnect from the SSH server\n */\n disconnect(): void {\n if (this.connected) {\n logger.info('Disconnecting from SSH server', {\n host: this.config.host,\n username: this.config.username,\n });\n this.client.end();\n this.connected = false;\n }\n }\n\n /**\n * Check if connected\n */\n isConnected(): boolean {\n return this.connected;\n }\n}\n"],
|
|
5
|
+
"mappings": ";;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AAAA;AAAA;AAAA;AAAA,MACE,MAAQ;AAAA,MACR,SAAW;AAAA,MACX,aAAe;AAAA,MACf,MAAQ;AAAA,MACR,OAAS;AAAA,MACT,SAAW;AAAA,QACT,KAAK;AAAA,UACH,OAAS;AAAA,UACT,SAAW;AAAA,UACX,SAAW;AAAA,QACb;AAAA,QACA,YAAY;AAAA,QACZ,eAAe;AAAA,QACf,qBAAqB;AAAA,QACrB,wBAAwB;AAAA,QACxB,qBAAqB;AAAA,QACrB,wBAAwB;AAAA,QACxB,kBAAkB;AAAA,MACpB;AAAA,MACA,SAAW;AAAA,QACT,aAAa;AAAA,QACb,MAAQ;AAAA,QACR,SAAW;AAAA,QACX,MAAQ;AAAA,QACR,iBAAiB;AAAA,QACjB,YAAc;AAAA,QACd,SAAW;AAAA,MACb;AAAA,MACA,YAAc;AAAA,QACZ,MAAQ;AAAA,QACR,KAAO;AAAA,MACT;AAAA,MACA,UAAY;AAAA,MACZ,SAAW;AAAA,MACX,UAAY;AAAA,QACV;AAAA,QACA;AAAA,QACA;AAAA,QACA;AAAA,QACA;AAAA,QACA;AAAA,QACA;AAAA,MACF;AAAA,MACA,gBAAkB;AAAA,MAClB,SAAW;AAAA,MACX,iBAAmB;AAAA,QACjB,eAAe;AAAA,QACf,SAAW;AAAA,QACX,OAAS;AAAA,QACT,UAAY;AAAA,QACZ,oBAAoB;AAAA,QACpB,KAAO;AAAA,QACP,YAAc;AAAA,MAChB;AAAA,MACA,SAAW;AAAA,QACT,MAAQ;AAAA,MACV;AAAA,MACA,SAAW;AAAA,QACT,IAAM;AAAA,MACR;AAAA,IACF;AAAA;AAAA;;;AC7DA;AAAA;AAAA,QAAM,KAAK,UAAQ,IAAI;AACvB,QAAM,OAAO,UAAQ,MAAM;AAC3B,QAAM,KAAK,UAAQ,IAAI;AACvB,QAAM,SAAS,UAAQ,QAAQ;AAC/B,QAAM,cAAc;AAEpB,QAAM,UAAU,YAAY;AAE5B,QAAM,OAAO;AAGb,aAAS,MAAO,KAAK;AACnB,YAAM,MAAM,CAAC;AAGb,UAAI,QAAQ,IAAI,SAAS;AAGzB,cAAQ,MAAM,QAAQ,WAAW,IAAI;AAErC,UAAI;AACJ,cAAQ,QAAQ,KAAK,KAAK,KAAK,MAAM,MAAM;AACzC,cAAM,MAAM,MAAM,CAAC;AAGnB,YAAI,QAAS,MAAM,CAAC,KAAK;AAGzB,gBAAQ,MAAM,KAAK;AAGnB,cAAM,aAAa,MAAM,CAAC;AAG1B,gBAAQ,MAAM,QAAQ,0BAA0B,IAAI;AAGpD,YAAI,eAAe,KAAK;AACtB,kBAAQ,MAAM,QAAQ,QAAQ,IAAI;AAClC,kBAAQ,MAAM,QAAQ,QAAQ,IAAI;AAAA,QACpC;AAGA,YAAI,GAAG,IAAI;AAAA,MACb;AAEA,aAAO;AAAA,IACT;AAEA,aAAS,YAAa,SAAS;AAC7B,gBAAU,WAAW,CAAC;AAEtB,YAAM,YAAY,WAAW,OAAO;AACpC,cAAQ,OAAO;AACf,YAAM,SAAS,aAAa,aAAa,OAAO;AAChD,UAAI,CAAC,OAAO,QAAQ;AAClB,cAAM,MAAM,IAAI,MAAM,8BAA8B,SAAS,wBAAwB;AACrF,YAAI,OAAO;AACX,cAAM;AAAA,MACR;AAIA,YAAM,OAAO,WAAW,OAAO,EAAE,MAAM,GAAG;AAC1C,YAAM,SAAS,KAAK;AAEpB,UAAI;AACJ,eAAS,IAAI,GAAG,IAAI,QAAQ,KAAK;AAC/B,YAAI;AAEF,gBAAM,MAAM,KAAK,CAAC,EAAE,KAAK;AAGzB,gBAAM,QAAQ,cAAc,QAAQ,GAAG;AAGvC,sBAAY,aAAa,QAAQ,MAAM,YAAY,MAAM,GAAG;AAE5D;AAAA,QACF,SAAS,OAAO;AAEd,cAAI,IAAI,KAAK,QAAQ;AACnB,kBAAM;AAAA,UACR;AAAA,QAEF;AAAA,MACF;AAGA,aAAO,aAAa,MAAM,SAAS;AAAA,IACrC;AAEA,aAAS,MAAO,SAAS;AACvB,cAAQ,IAAI,WAAW,OAAO,WAAW,OAAO,EAAE;AAAA,IACpD;AAEA,aAAS,OAAQ,SAAS;AACxB,cAAQ,IAAI,WAAW,OAAO,YAAY,OAAO,EAAE;AAAA,IACrD;AAEA,aAAS,KAAM,SAAS;AACtB,cAAQ,IAAI,WAAW,OAAO,KAAK,OAAO,EAAE;AAAA,IAC9C;AAEA,aAAS,WAAY,SAAS;AAE5B,UAAI,WAAW,QAAQ,cAAc,QAAQ,WAAW,SAAS,GAAG;AAClE,eAAO,QAAQ;AAAA,MACjB;AAGA,UAAI,QAAQ,IAAI,cAAc,QAAQ,IAAI,WAAW,SAAS,GAAG;AAC/D,eAAO,QAAQ,IAAI;AAAA,MACrB;AAGA,aAAO;AAAA,IACT;AAEA,aAAS,cAAe,QAAQ,WAAW;AAEzC,UAAI;AACJ,UAAI;AACF,cAAM,IAAI,IAAI,SAAS;AAAA,MACzB,SAAS,OAAO;AACd,YAAI,MAAM,SAAS,mBAAmB;AACpC,gBAAM,MAAM,IAAI,MAAM,4IAA4I;AAClK,cAAI,OAAO;AACX,gBAAM;AAAA,QACR;AAEA,cAAM;AAAA,MACR;AAGA,YAAM,MAAM,IAAI;AAChB,UAAI,CAAC,KAAK;AACR,cAAM,MAAM,IAAI,MAAM,sCAAsC;AAC5D,YAAI,OAAO;AACX,cAAM;AAAA,MACR;AAGA,YAAM,cAAc,IAAI,aAAa,IAAI,aAAa;AACtD,UAAI,CAAC,aAAa;AAChB,cAAM,MAAM,IAAI,MAAM,8CAA8C;AACpE,YAAI,OAAO;AACX,cAAM;AAAA,MACR;AAGA,YAAM,iBAAiB,gBAAgB,YAAY,YAAY,CAAC;AAChE,YAAM,aAAa,OAAO,OAAO,cAAc;AAC/C,UAAI,CAAC,YAAY;AACf,cAAM,MAAM,IAAI,MAAM,2DAA2D,cAAc,2BAA2B;AAC1H,YAAI,OAAO;AACX,cAAM;AAAA,MACR;AAEA,aAAO,EAAE,YAAY,IAAI;AAAA,IAC3B;AAEA,aAAS,WAAY,SAAS;AAC5B,UAAI,oBAAoB;AAExB,UAAI,WAAW,QAAQ,QAAQ,QAAQ,KAAK,SAAS,GAAG;AACtD,YAAI,MAAM,QAAQ,QAAQ,IAAI,GAAG;AAC/B,qBAAW,YAAY,QAAQ,MAAM;AACnC,gBAAI,GAAG,WAAW,QAAQ,GAAG;AAC3B,kCAAoB,SAAS,SAAS,QAAQ,IAAI,WAAW,GAAG,QAAQ;AAAA,YAC1E;AAAA,UACF;AAAA,QACF,OAAO;AACL,8BAAoB,QAAQ,KAAK,SAAS,QAAQ,IAAI,QAAQ,OAAO,GAAG,QAAQ,IAAI;AAAA,QACtF;AAAA,MACF,OAAO;AACL,4BAAoB,KAAK,QAAQ,QAAQ,IAAI,GAAG,YAAY;AAAA,MAC9D;AAEA,UAAI,GAAG,WAAW,iBAAiB,GAAG;AACpC,eAAO;AAAA,MACT;AAEA,aAAO;AAAA,IACT;AAEA,aAAS,aAAc,SAAS;AAC9B,aAAO,QAAQ,CAAC,MAAM,MAAM,KAAK,KAAK,GAAG,QAAQ,GAAG,QAAQ,MAAM,CAAC,CAAC,IAAI;AAAA,IAC1E;AAEA,aAAS,aAAc,SAAS;AAC9B,YAAM,QAAQ,QAAQ,WAAW,QAAQ,KAAK;AAC9C,YAAM,QAAQ,WAAW,WAAW,UAAU,QAAQ,QAAQ;AAE9D,UAAI,SAAS,CAAC,OAAO;AACnB,aAAK,uCAAuC;AAAA,MAC9C;AAEA,YAAM,SAAS,aAAa,YAAY,OAAO;AAE/C,UAAI,aAAa,QAAQ;AACzB,UAAI,WAAW,QAAQ,cAAc,MAAM;AACzC,qBAAa,QAAQ;AAAA,MACvB;AAEA,mBAAa,SAAS,YAAY,QAAQ,OAAO;AAEjD,aAAO,EAAE,OAAO;AAAA,IAClB;AAEA,aAAS,aAAc,SAAS;AAC9B,YAAM,aAAa,KAAK,QAAQ,QAAQ,IAAI,GAAG,MAAM;AACrD,UAAI,WAAW;AACf,YAAM,QAAQ,QAAQ,WAAW,QAAQ,KAAK;AAC9C,YAAM,QAAQ,WAAW,WAAW,UAAU,QAAQ,QAAQ;AAE9D,UAAI,WAAW,QAAQ,UAAU;AAC/B,mBAAW,QAAQ;AAAA,MACrB,OAAO;AACL,YAAI,OAAO;AACT,iBAAO,oDAAoD;AAAA,QAC7D;AAAA,MACF;AAEA,UAAI,cAAc,CAAC,UAAU;AAC7B,UAAI,WAAW,QAAQ,MAAM;AAC3B,YAAI,CAAC,MAAM,QAAQ,QAAQ,IAAI,GAAG;AAChC,wBAAc,CAAC,aAAa,QAAQ,IAAI,CAAC;AAAA,QAC3C,OAAO;AACL,wBAAc,CAAC;AACf,qBAAW,YAAY,QAAQ,MAAM;AACnC,wBAAY,KAAK,aAAa,QAAQ,CAAC;AAAA,UACzC;AAAA,QACF;AAAA,MACF;AAIA,UAAI;AACJ,YAAM,YAAY,CAAC;AACnB,iBAAWA,SAAQ,aAAa;AAC9B,YAAI;AAEF,gBAAM,SAAS,aAAa,MAAM,GAAG,aAAaA,OAAM,EAAE,SAAS,CAAC,CAAC;AAErE,uBAAa,SAAS,WAAW,QAAQ,OAAO;AAAA,QAClD,SAAS,GAAG;AACV,cAAI,OAAO;AACT,mBAAO,kBAAkBA,KAAI,IAAI,EAAE,OAAO,EAAE;AAAA,UAC9C;AACA,sBAAY;AAAA,QACd;AAAA,MACF;AAEA,UAAI,aAAa,QAAQ;AACzB,UAAI,WAAW,QAAQ,cAAc,MAAM;AACzC,qBAAa,QAAQ;AAAA,MACvB;AAEA,mBAAa,SAAS,YAAY,WAAW,OAAO;AAEpD,UAAI,SAAS,CAAC,OAAO;AACnB,cAAM,YAAY,OAAO,KAAK,SAAS,EAAE;AACzC,cAAM,aAAa,CAAC;AACpB,mBAAW,YAAY,aAAa;AAClC,cAAI;AACF,kBAAM,WAAW,KAAK,SAAS,QAAQ,IAAI,GAAG,QAAQ;AACtD,uBAAW,KAAK,QAAQ;AAAA,UAC1B,SAAS,GAAG;AACV,gBAAI,OAAO;AACT,qBAAO,kBAAkB,QAAQ,IAAI,EAAE,OAAO,EAAE;AAAA,YAClD;AACA,wBAAY;AAAA,UACd;AAAA,QACF;AAEA,aAAK,kBAAkB,SAAS,UAAU,WAAW,KAAK,GAAG,CAAC,EAAE;AAAA,MAClE;AAEA,UAAI,WAAW;AACb,eAAO,EAAE,QAAQ,WAAW,OAAO,UAAU;AAAA,MAC/C,OAAO;AACL,eAAO,EAAE,QAAQ,UAAU;AAAA,MAC7B;AAAA,IACF;AAGA,aAASC,QAAQ,SAAS;AAExB,UAAI,WAAW,OAAO,EAAE,WAAW,GAAG;AACpC,eAAO,aAAa,aAAa,OAAO;AAAA,MAC1C;AAEA,YAAM,YAAY,WAAW,OAAO;AAGpC,UAAI,CAAC,WAAW;AACd,cAAM,+DAA+D,SAAS,+BAA+B;AAE7G,eAAO,aAAa,aAAa,OAAO;AAAA,MAC1C;AAEA,aAAO,aAAa,aAAa,OAAO;AAAA,IAC1C;AAEA,aAAS,QAAS,WAAW,QAAQ;AACnC,YAAM,MAAM,OAAO,KAAK,OAAO,MAAM,GAAG,GAAG,KAAK;AAChD,UAAI,aAAa,OAAO,KAAK,WAAW,QAAQ;AAEhD,YAAM,QAAQ,WAAW,SAAS,GAAG,EAAE;AACvC,YAAM,UAAU,WAAW,SAAS,GAAG;AACvC,mBAAa,WAAW,SAAS,IAAI,GAAG;AAExC,UAAI;AACF,cAAM,SAAS,OAAO,iBAAiB,eAAe,KAAK,KAAK;AAChE,eAAO,WAAW,OAAO;AACzB,eAAO,GAAG,OAAO,OAAO,UAAU,CAAC,GAAG,OAAO,MAAM,CAAC;AAAA,MACtD,SAAS,OAAO;AACd,cAAM,UAAU,iBAAiB;AACjC,cAAM,mBAAmB,MAAM,YAAY;AAC3C,cAAM,mBAAmB,MAAM,YAAY;AAE3C,YAAI,WAAW,kBAAkB;AAC/B,gBAAM,MAAM,IAAI,MAAM,6DAA6D;AACnF,cAAI,OAAO;AACX,gBAAM;AAAA,QACR,WAAW,kBAAkB;AAC3B,gBAAM,MAAM,IAAI,MAAM,iDAAiD;AACvE,cAAI,OAAO;AACX,gBAAM;AAAA,QACR,OAAO;AACL,gBAAM;AAAA,QACR;AAAA,MACF;AAAA,IACF;AAGA,aAAS,SAAU,YAAY,QAAQ,UAAU,CAAC,GAAG;AACnD,YAAM,QAAQ,QAAQ,WAAW,QAAQ,KAAK;AAC9C,YAAM,WAAW,QAAQ,WAAW,QAAQ,QAAQ;AAEpD,UAAI,OAAO,WAAW,UAAU;AAC9B,cAAM,MAAM,IAAI,MAAM,gFAAgF;AACtG,YAAI,OAAO;AACX,cAAM;AAAA,MACR;AAGA,iBAAW,OAAO,OAAO,KAAK,MAAM,GAAG;AACrC,YAAI,OAAO,UAAU,eAAe,KAAK,YAAY,GAAG,GAAG;AACzD,cAAI,aAAa,MAAM;AACrB,uBAAW,GAAG,IAAI,OAAO,GAAG;AAAA,UAC9B;AAEA,cAAI,OAAO;AACT,gBAAI,aAAa,MAAM;AACrB,qBAAO,IAAI,GAAG,0CAA0C;AAAA,YAC1D,OAAO;AACL,qBAAO,IAAI,GAAG,8CAA8C;AAAA,YAC9D;AAAA,UACF;AAAA,QACF,OAAO;AACL,qBAAW,GAAG,IAAI,OAAO,GAAG;AAAA,QAC9B;AAAA,MACF;AAAA,IACF;AAEA,QAAM,eAAe;AAAA,MACnB;AAAA,MACA;AAAA,MACA;AAAA,MACA,QAAAA;AAAA,MACA;AAAA,MACA;AAAA,MACA;AAAA,IACF;AAEA,WAAO,QAAQ,eAAe,aAAa;AAC3C,WAAO,QAAQ,eAAe,aAAa;AAC3C,WAAO,QAAQ,cAAc,aAAa;AAC1C,WAAO,QAAQ,SAAS,aAAa;AACrC,WAAO,QAAQ,UAAU,aAAa;AACtC,WAAO,QAAQ,QAAQ,aAAa;AACpC,WAAO,QAAQ,WAAW,aAAa;AAEvC,WAAO,UAAU;AAAA;AAAA;;;AC/XjB,SAAS,cAAc;AACvB,SAAS,4BAA4B;AACrC;AAAA,EACE;AAAA,EACA;AAAA,OACK;;;ACPP,oBAAmB;AACnB,SAAS,oBAAoB;AAE7B,cAAAC,QAAO,OAAO;AAYd,SAAS,iBAAyB;AAChC,QAAM,UAAU,QAAQ,IAAI;AAC5B,QAAM,cAAc,QAAQ,IAAI;AAChC,QAAM,oBAAoB,QAAQ,IAAI;AAEtC,MAAI,CAAC,WAAW,CAAC,eAAe,CAAC,mBAAmB;AAClD,UAAM,IAAI;AAAA,MACR;AAAA,IACF;AAAA,EACF;AAEA,SAAO;AAAA,IACL,UAAU,QAAQ,IAAI,aAAa;AAAA,IACnC,KAAK;AAAA,MACH,MAAM;AAAA,MACN,MAAM,SAAS,QAAQ,IAAI,YAAY,MAAM,EAAE;AAAA,MAC/C,UAAU;AAAA,MACV,gBAAgB;AAAA,IAClB;AAAA,EACF;AACF;AAEO,IAAM,SAAS,eAAe;AAK9B,SAAS,oBAA4B;AAC1C,MAAI;AACF,WAAO,aAAa,OAAO,IAAI,gBAAgB,OAAO;AAAA,EACxD,SAAS,OAAO;AACd,UAAM,IAAI;AAAA,MACR,uCAAuC,OAAO,IAAI,cAAc,KAAK,iBAAiB,QAAQ,MAAM,UAAU,OAAO,KAAK,CAAC;AAAA,IAC7H;AAAA,EACF;AACF;;;AC/CO,IAAM,yBAA+B;AAAA,EAC1C,MAAM;AAAA,EACN,aAAa;AAAA,EACb,aAAa;AAAA,IACX,MAAM;AAAA,IACN,YAAY;AAAA,MACV,MAAM;AAAA,QACJ,MAAM;AAAA,QACN,aAAa;AAAA,MACf;AAAA,MACA,WAAW;AAAA,QACT,MAAM;AAAA,QACN,aAAa;AAAA,QACb,SAAS;AAAA,MACX;AAAA,IACF;AAAA,IACA,UAAU,CAAC,MAAM;AAAA,EACnB;AACF;AAcA,eAAsB,yBACpB,MACA,eAC6D;AAC7D,QAAM,EAAE,MAAM,YAAY,MAAM,IAAI;AAEpC,MAAI;AACF,UAAM,QAAQ,MAAM,gBAAgB,eAAe,MAAM,SAAS;AAElE,WAAO;AAAA,MACL,SAAS;AAAA,QACP;AAAA,UACE,MAAM;AAAA,UACN,MAAM,MAAM,KAAK,IAAI;AAAA,QACvB;AAAA,MACF;AAAA,IACF;AAAA,EACF,SAAS,OAAO;AACd,UAAM,eAAe,iBAAiB,QAAQ,MAAM,UAAU,OAAO,KAAK;AAC1E,WAAO;AAAA,MACL,SAAS;AAAA,QACP;AAAA,UACE,MAAM;AAAA,UACN,MAAM,+BAA+B,YAAY;AAAA,QACnD;AAAA,MACF;AAAA,IACF;AAAA,EACF;AACF;AAMA,eAAe,gBACb,KACA,SACA,WACmB;AACnB,QAAM,UAAU,MAAM,IAAI,UAAU,OAAO;AAC3C,QAAM,QAAkB,CAAC;AAEzB,aAAW,SAAS,SAAS;AAE3B,UAAM,WAAW,GAAG,OAAO,IAAI,MAAM,IAAI,GAAG,QAAQ,QAAQ,GAAG;AAE/D,QAAI,MAAM,aAAa;AACrB,YAAM,KAAK,GAAG,QAAQ,GAAG;AACzB,UAAI,WAAW;AACb,cAAM,WAAW,MAAM,gBAAgB,KAAK,UAAU,SAAS;AAC/D,cAAM,KAAK,GAAG,QAAQ;AAAA,MACxB;AAAA,IACF,OAAO;AACL,YAAM,KAAK,QAAQ;AAAA,IACrB;AAAA,EACF;AAEA,SAAO,MAAM,KAAK;AACpB;;;ACtFA,IAAM,aAAuC;AAAA,EAC3C,OAAO;AAAA,EACP,MAAM;AAAA,EACN,MAAM;AAAA,EACN,OAAO;AAAA,EACP,OAAO;AACT;AAEA,IAAM,SAAN,MAAa;AAAA,EACH;AAAA,EACA;AAAA,EAER,cAAc;AAEZ,SAAK,QAAS,QAAQ,IAAI,qBAAkC;AAC5D,SAAK,UAAU,QAAQ,IAAI,kBAAkB,UAAU,QAAQ,IAAI,aAAa;AAAA,EAClF;AAAA,EAEQ,UAAU,OAA0B;AAC1C,QAAI,CAAC,KAAK,WAAW,UAAU,WAAW,UAAU,QAAQ;AAC1D,aAAO;AAAA,IACT;AACA,WAAO,WAAW,KAAK,KAAK,WAAW,KAAK,KAAK;AAAA,EACnD;AAAA,EAEQ,cAAc,OAAiB,SAAiB,MAAoB;AAC1E,UAAM,aAAY,oBAAI,KAAK,GAAE,YAAY;AACzC,UAAM,SAAS,IAAI,SAAS,MAAM,MAAM,YAAY,CAAC;AAErD,QAAI,SAAS,QAAW;AACtB,YAAM,UAAU,OAAO,SAAS,WAAW,KAAK,UAAU,MAAM,MAAM,CAAC,IAAI,OAAO,IAAI;AACtF,aAAO,GAAG,MAAM,IAAI,OAAO;AAAA,EAAK,OAAO;AAAA,IACzC;AAEA,WAAO,GAAG,MAAM,IAAI,OAAO;AAAA,EAC7B;AAAA,EAEA,MAAM,SAAiB,MAAkB;AACvC,QAAI,KAAK,UAAU,OAAO,GAAG;AAC3B,cAAQ,MAAM,KAAK,cAAc,SAAS,SAAS,IAAI,CAAC;AAAA,IAC1D;AAAA,EACF;AAAA,EAEA,KAAK,SAAiB,MAAkB;AACtC,QAAI,KAAK,UAAU,MAAM,GAAG;AAC1B,cAAQ,MAAM,KAAK,cAAc,QAAQ,SAAS,IAAI,CAAC;AAAA,IACzD;AAAA,EACF;AAAA,EAEA,KAAK,SAAiB,MAAkB;AACtC,QAAI,KAAK,UAAU,MAAM,GAAG;AAC1B,cAAQ,MAAM,KAAK,cAAc,QAAQ,SAAS,IAAI,CAAC;AAAA,IACzD;AAAA,EACF;AAAA,EAEA,MAAM,SAAiB,MAAkB;AACvC,QAAI,KAAK,UAAU,OAAO,GAAG;AAC3B,cAAQ,MAAM,KAAK,cAAc,SAAS,SAAS,IAAI,CAAC;AAAA,IAC1D;AAAA,EACF;AAAA,EAEA,MAAM,SAAiB,MAAkB;AACvC,QAAI,KAAK,UAAU,OAAO,GAAG;AAC3B,cAAQ,MAAM,KAAK,cAAc,SAAS,SAAS,IAAI,CAAC;AAAA,IAC1D;AAAA,EACF;AAAA;AAAA;AAAA;AAAA,EAKA,YAAY,UAAkB,QAAa,QAAuB;AAChE,SAAK,KAAK,iBAAiB,QAAQ,EAAE;AACrC,SAAK,MAAM,mBAAmB,EAAE,MAAM,UAAU,QAAQ,OAAO,CAAC;AAAA,EAClE;AAAA;AAAA;AAAA;AAAA,EAKA,cAAc,UAAkB,UAAkB,YAA2B;AAC3E,SAAK,KAAK,mBAAmB,QAAQ,EAAE;AACvC,SAAK,MAAM,oBAAoB,EAAE,MAAM,UAAU,UAAU,GAAG,QAAQ,MAAM,WAAW,CAAC;AAAA,EAC1F;AAAA;AAAA;AAAA;AAAA,EAKA,WAAW,UAAkB,OAAc,QAAoB;AAC7D,SAAK,MAAM,0BAA0B,QAAQ,IAAI;AAAA,MAC/C,MAAM;AAAA,MACN,OAAO,MAAM;AAAA,MACb,OAAO,MAAM;AAAA,MACb;AAAA,IACF,CAAC;AAAA,EACH;AAAA;AAAA;AAAA;AAAA,EAKA,WAAW,SAAiB,KAAc,SAAwB;AAChE,SAAK,MAAM,yBAAyB,EAAE,SAAS,KAAK,QAAQ,CAAC;AAAA,EAC/D;AAAA;AAAA;AAAA;AAAA,EAKA,iBAAiB,UAAkB,UAAkB,YAAoB,YAA0B;AACjG,SAAK,MAAM,yBAAyB;AAAA,MAClC;AAAA,MACA,UAAU,GAAG,QAAQ;AAAA,MACrB,QAAQ,GAAG,UAAU;AAAA,MACrB,QAAQ,GAAG,UAAU;AAAA,IACvB,CAAC;AAAA,EACH;AAAA;AAAA;AAAA;AAAA,EAKA,cAAc,WAAmB,MAAc,SAAqB;AAClE,SAAK,KAAK,mBAAmB,SAAS,IAAI,EAAE,MAAM,GAAG,QAAQ,CAAC;AAAA,EAChE;AACF;AAGO,IAAM,SAAS,IAAI,OAAO;;;AC9H1B,IAAM,8BAAoC;AAAA,EAC/C,MAAM;AAAA,EACN,aAAa;AAAA,EACb,aAAa;AAAA,IACX,MAAM;AAAA,IACN,YAAY;AAAA,MACV,SAAS;AAAA,QACP,MAAM;AAAA,QACN,aAAa;AAAA,MACf;AAAA,MACA,KAAK;AAAA,QACH,MAAM;AAAA,QACN,aAAa;AAAA,MACf;AAAA,MACA,SAAS;AAAA,QACP,MAAM;AAAA,QACN,aAAa;AAAA,QACb,SAAS;AAAA,MACX;AAAA,IACF;AAAA,IACA,UAAU,CAAC,SAAS;AAAA,EACtB;AACF;AAsBA,eAAsB,8BACpB,MACA,eAC6D;AAC7D,QAAM,EAAE,SAAS,KAAK,UAAU,GAAG,IAAI;AAEvC,SAAO,MAAM,4BAA4B,EAAE,SAAS,KAAK,QAAQ,CAAC;AAElE,MAAI;AAEF,UAAM,cAAc,MAAM,MAAM,GAAG,OAAO,OAAO,KAAK;AAEtD,UAAM,SAAS,MAAM,cAAc,gBAAgB,aAAa,OAAO;AAEvE,WAAO,MAAM,4BAA4B;AAAA,MACvC,UAAU,OAAO;AAAA,MACjB,UAAU,OAAO;AAAA,MACjB,cAAc,OAAO,OAAO;AAAA,MAC5B,cAAc,OAAO,OAAO;AAAA,IAC9B,CAAC;AAGD,UAAM,SAA+B;AAAA,MACnC,QAAQ,OAAO;AAAA,MACf,QAAQ,OAAO;AAAA,MACf,UAAU,OAAO;AAAA,MACjB,UAAU,OAAO;AAAA,IACnB;AAEA,WAAO;AAAA,MACL,SAAS;AAAA,QACP;AAAA,UACE,MAAM;AAAA,UACN,MAAM,KAAK,UAAU,QAAQ,MAAM,CAAC;AAAA,QACtC;AAAA,MACF;AAAA,IACF;AAAA,EACF,SAAS,OAAO;AACd,UAAM,eAAe,iBAAiB,QAAQ,MAAM,UAAU,OAAO,KAAK;AAC1E,WAAO,MAAM,2BAA2B,EAAE,SAAS,OAAO,aAAa,CAAC;AACxE,WAAO;AAAA,MACL,SAAS;AAAA,QACP;AAAA,UACE,MAAM;AAAA,UACN,MAAM,KAAK,UAAU;AAAA,YACnB,QAAQ;AAAA,YACR,QAAQ;AAAA,YACR,UAAU;AAAA,YACV,UAAU;AAAA,UACZ,GAAG,MAAM,CAAC;AAAA,QACZ;AAAA,MACF;AAAA,IACF;AAAA,EACF;AACF;;;AClGO,IAAM,wBAA8B;AAAA,EACzC,MAAM;AAAA,EACN,aAAa;AAAA,EACb,aAAa;AAAA,IACX,MAAM;AAAA,IACN,YAAY;AAAA,MACV,MAAM;AAAA,QACJ,MAAM;AAAA,QACN,aAAa;AAAA,MACf;AAAA,MACA,UAAU;AAAA,QACR,MAAM;AAAA,QACN,aAAa;AAAA,QACb,SAAS;AAAA,QACT,MAAM,CAAC,SAAS,SAAS,QAAQ;AAAA,MACnC;AAAA,MACA,SAAS;AAAA,QACP,MAAM;AAAA,QACN,aAAa;AAAA,QACb,SAAS;AAAA,MACX;AAAA,IACF;AAAA,IACA,UAAU,CAAC,MAAM;AAAA,EACnB;AACF;AAqBA,eAAsB,wBACpB,MACA,eAC6D;AAC7D,QAAM,EAAE,MAAM,WAAW,SAAS,UAAU,QAAQ,IAAI;AAExD,SAAO,MAAM,uBAAuB,EAAE,MAAM,UAAU,QAAQ,CAAC;AAE/D,MAAI;AACF,UAAM,SAAS,MAAM,cAAc,SAAS,MAAM,UAAU,OAAO;AAEnE,WAAO,MAAM,wBAAwB,EAAE,MAAM,MAAM,OAAO,KAAK,CAAC;AAEhE,UAAM,SAAyB;AAAA,MAC7B,SAAS,OAAO;AAAA,MAChB,MAAM,OAAO;AAAA,MACb,UAAU,OAAO;AAAA,IACnB;AAEA,WAAO;AAAA,MACL,SAAS;AAAA,QACP;AAAA,UACE,MAAM;AAAA,UACN,MAAM,KAAK,UAAU,QAAQ,MAAM,CAAC;AAAA,QACtC;AAAA,MACF;AAAA,IACF;AAAA,EACF,SAAS,OAAO;AACd,UAAM,eAAe,iBAAiB,QAAQ,MAAM,UAAU,OAAO,KAAK;AAC1E,WAAO,MAAM,mBAAmB,EAAE,MAAM,OAAO,aAAa,CAAC;AAC7D,WAAO;AAAA,MACL,SAAS;AAAA,QACP;AAAA,UACE,MAAM;AAAA,UACN,MAAM,KAAK,UAAU;AAAA,YACnB,OAAO;AAAA,YACP,SAAS;AAAA,YACT,MAAM;AAAA,YACN;AAAA,UACF,GAAG,MAAM,CAAC;AAAA,QACZ;AAAA,MACF;AAAA,IACF;AAAA,EACF;AACF;;;ACzFO,IAAM,yBAA+B;AAAA,EAC1C,MAAM;AAAA,EACN,aAAa;AAAA,EACb,aAAa;AAAA,IACX,MAAM;AAAA,IACN,YAAY;AAAA,MACV,MAAM;AAAA,QACJ,MAAM;AAAA,QACN,aAAa;AAAA,MACf;AAAA,MACA,SAAS;AAAA,QACP,MAAM;AAAA,QACN,aAAa;AAAA,MACf;AAAA,MACA,UAAU;AAAA,QACR,MAAM;AAAA,QACN,aAAa;AAAA,QACb,SAAS;AAAA,MACX;AAAA,MACA,YAAY;AAAA,QACV,MAAM;AAAA,QACN,aAAa;AAAA,QACb,SAAS;AAAA,MACX;AAAA,MACA,QAAQ;AAAA,QACN,MAAM;AAAA,QACN,aAAa;AAAA,QACb,SAAS;AAAA,MACX;AAAA,IACF;AAAA,IACA,UAAU,CAAC,QAAQ,SAAS;AAAA,EAC9B;AACF;AAuBA,eAAsB,yBACpB,MACA,eAC6D;AAC7D,QAAM,EAAE,MAAM,SAAS,WAAW,SAAS,aAAa,OAAO,SAAS,MAAM,IAAI;AAElF,SAAO,MAAM,uBAAuB,EAAE,MAAM,aAAa,QAAQ,QAAQ,UAAU,YAAY,OAAO,CAAC;AAEvG,MAAI;AACF,UAAM,SAAS,MAAM,cAAc,UAAU,MAAM,SAAS;AAAA,MAC1D;AAAA,MACA;AAAA,MACA;AAAA,IACF,CAAC;AAED,WAAO,MAAM,yBAAyB,EAAE,MAAM,cAAc,OAAO,cAAc,YAAY,OAAO,WAAW,CAAC;AAEhH,UAAM,SAA0B;AAAA,MAC9B,SAAS,OAAO;AAAA,MAChB,cAAc,OAAO;AAAA,MACrB,YAAY,OAAO;AAAA,IACrB;AAEA,WAAO;AAAA,MACL,SAAS;AAAA,QACP;AAAA,UACE,MAAM;AAAA,UACN,MAAM,KAAK,UAAU,QAAQ,MAAM,CAAC;AAAA,QACtC;AAAA,MACF;AAAA,IACF;AAAA,EACF,SAAS,OAAO;AACd,UAAM,eAAe,iBAAiB,QAAQ,MAAM,UAAU,OAAO,KAAK;AAC1E,WAAO,MAAM,oBAAoB,EAAE,MAAM,OAAO,aAAa,CAAC;AAC9D,WAAO;AAAA,MACL,SAAS;AAAA,QACP;AAAA,UACE,MAAM;AAAA,UACN,MAAM,KAAK,UAAU;AAAA,YACnB,SAAS;AAAA,YACT,cAAc;AAAA,YACd,OAAO;AAAA,UACT,GAAG,MAAM,CAAC;AAAA,QACZ;AAAA,MACF;AAAA,IACF;AAAA,EACF;AACF;;;AC1GA,SAAS,cAA2B;AAQ7B,IAAM,uBAAN,MAA2B;AAAA,EACxB;AAAA,EACA;AAAA,EACA,YAAqB;AAAA,EAE7B,YAAYC,SAAmB;AAC7B,SAAK,SAASA;AACd,SAAK,SAAS,IAAI,OAAO;AAAA,EAC3B;AAAA;AAAA;AAAA;AAAA,EAKA,MAAM,UAAyB;AAC7B,QAAI,KAAK,WAAW;AAClB,aAAO,MAAM,oCAAoC;AACjD;AAAA,IACF;AAEA,WAAO,KAAK,4BAA4B;AAAA,MACtC,MAAM,KAAK,OAAO;AAAA,MAClB,MAAM,KAAK,OAAO,QAAQ;AAAA,MAC1B,UAAU,KAAK,OAAO;AAAA,IACxB,CAAC;AAED,WAAO,IAAI,QAAQ,CAAC,SAAS,WAAW;AACtC,WAAK,OACF,GAAG,SAAS,MAAM;AACjB,aAAK,YAAY;AACjB,eAAO,KAAK,8BAA8B;AAAA,UACxC,MAAM,KAAK,OAAO;AAAA,UAClB,UAAU,KAAK,OAAO;AAAA,QACxB,CAAC;AACD,gBAAQ;AAAA,MACV,CAAC,EACA,GAAG,SAAS,CAAC,QAAQ;AACpB,eAAO,MAAM,yBAAyB;AAAA,UACpC,MAAM,KAAK,OAAO;AAAA,UAClB,OAAO,IAAI;AAAA,QACb,CAAC;AACD,eAAO,GAAG;AAAA,MACZ,CAAC,EACA,QAAQ;AAAA,QACP,MAAM,KAAK,OAAO;AAAA,QAClB,MAAM,KAAK,OAAO,QAAQ;AAAA,QAC1B,UAAU,KAAK,OAAO;AAAA,QACtB,YAAY,KAAK,OAAO;AAAA,MAC1B,CAAC;AAAA,IACL,CAAC;AAAA,EACH;AAAA;AAAA;AAAA;AAAA,EAKA,MAAM,KAAK,SAAkC;AAC3C,QAAI,CAAC,KAAK,WAAW;AACnB,YAAM,KAAK,QAAQ;AAAA,IACrB;AAEA,WAAO,IAAI,QAAQ,CAAC,SAAS,WAAW;AACtC,WAAK,OAAO,KAAK,SAAS,CAAC,KAAK,WAAW;AACzC,YAAI,KAAK;AACP,iBAAO,GAAG;AACV;AAAA,QACF;AAEA,YAAI,SAAS;AACb,YAAI,SAAS;AAEb,eACG,GAAG,SAAS,CAAC,SAAiB;AAC7B,cAAI,SAAS,GAAG;AACd,mBAAO,IAAI,MAAM,4BAA4B,IAAI,KAAK,MAAM,EAAE,CAAC;AAAA,UACjE,OAAO;AACL,oBAAQ,MAAM;AAAA,UAChB;AAAA,QACF,CAAC,EACA,GAAG,QAAQ,CAAC,SAAiB;AAC5B,oBAAU,KAAK,SAAS;AAAA,QAC1B,CAAC,EACA,OAAO,GAAG,QAAQ,CAAC,SAAiB;AACnC,oBAAU,KAAK,SAAS;AAAA,QAC1B,CAAC;AAAA,MACL,CAAC;AAAA,IACH,CAAC;AAAA,EACH;AAAA;AAAA;AAAA;AAAA,EAKA,MAAM,gBACJ,SACA,iBAAyB,IACyD;AAClF,QAAI,CAAC,KAAK,WAAW;AACnB,YAAM,KAAK,QAAQ;AAAA,IACrB;AAEA,UAAM,YAAY,KAAK,IAAI;AAC3B,WAAO,WAAW,SAAS,QAAW,cAAc;AAEpD,UAAM,cAAc,IAAI,QAA8D,CAAC,SAAS,WAAW;AACzG,WAAK,OAAO,KAAK,SAAS,CAAC,KAAK,WAAW;AACzC,YAAI,KAAK;AACP,iBAAO,GAAG;AACV;AAAA,QACF;AAEA,YAAI,SAAS;AACb,YAAI,SAAS;AAEb,eACG,GAAG,SAAS,CAAC,SAAiB;AAC7B,gBAAM,WAAW,KAAK,IAAI,IAAI;AAC9B,iBAAO,iBAAiB,MAAM,UAAU,OAAO,QAAQ,OAAO,MAAM;AACpE,kBAAQ,EAAE,QAAQ,QAAQ,UAAU,KAAK,CAAC;AAAA,QAC5C,CAAC,EACA,GAAG,QAAQ,CAAC,SAAiB;AAC5B,oBAAU,KAAK,SAAS;AAAA,QAC1B,CAAC,EACA,OAAO,GAAG,QAAQ,CAAC,SAAiB;AACnC,oBAAU,KAAK,SAAS;AAAA,QAC1B,CAAC;AAAA,MACL,CAAC;AAAA,IACH,CAAC;AAED,UAAM,iBAAiB,IAAI,QAAe,CAAC,GAAG,WAAW;AACvD,iBAAW,MAAM;AACf,eAAO,IAAI,MAAM,6BAA6B,CAAC;AAAA,MACjD,GAAG,iBAAiB,GAAI;AAAA,IAC1B,CAAC;AAED,QAAI;AACF,YAAM,SAAS,MAAM,QAAQ,KAAK,CAAC,aAAa,cAAc,CAAC;AAC/D,aAAO,EAAE,GAAG,QAAQ,UAAU,MAAM;AAAA,IACtC,SAAS,OAAO;AACd,UAAI,iBAAiB,SAAS,MAAM,YAAY,+BAA+B;AAC7E,eAAO,KAAK,yBAAyB,EAAE,SAAS,SAAS,eAAe,CAAC;AACzE,eAAO;AAAA,UACL,QAAQ;AAAA,UACR,QAAQ;AAAA,UACR,UAAU;AAAA,UACV,UAAU;AAAA,QACZ;AAAA,MACF;AACA,aAAO,MAAM,gCAAgC;AAAA,QAC3C;AAAA,QACA,OAAO,iBAAiB,QAAQ,MAAM,UAAU,OAAO,KAAK;AAAA,MAC9D,CAAC;AACD,YAAM;AAAA,IACR;AAAA,EACF;AAAA;AAAA;AAAA;AAAA,EAKA,MAAM,UAAgC;AACpC,QAAI,CAAC,KAAK,WAAW;AACnB,YAAM,KAAK,QAAQ;AAAA,IACrB;AAEA,WAAO,IAAI,QAAQ,CAAC,SAAS,WAAW;AACtC,WAAK,OAAO,KAAK,CAAC,KAAK,SAAS;AAC9B,YAAI,KAAK;AACP,iBAAO,GAAG;AAAA,QACZ,OAAO;AACL,kBAAQ,IAAI;AAAA,QACd;AAAA,MACF,CAAC;AAAA,IACH,CAAC;AAAA,EACH;AAAA;AAAA;AAAA;AAAA,EAKA,MAAM,UAAU,MAAsE;AACpF,UAAM,OAAO,MAAM,KAAK,QAAQ;AAEhC,WAAO,IAAI,QAAQ,CAAC,SAAS,WAAW;AACtC,WAAK,QAAQ,MAAM,CAAC,KAAK,SAAS;AAChC,YAAI,KAAK;AACP,iBAAO,GAAG;AACV;AAAA,QACF;AAEA,cAAM,QAAQ,KAAK,IAAI,CAAC,UAAU;AAAA,UAChC,MAAM,KAAK;AAAA,UACX,aAAa,KAAK,MAAM,YAAY;AAAA,QACtC,EAAE;AAEF,gBAAQ,KAAK;AAAA,MACf,CAAC;AAAA,IACH,CAAC;AAAA,EACH;AAAA;AAAA;AAAA;AAAA,EAKA,MAAM,SACJ,MACA,WAAmB,SACnB,UAAkB,SAC4C;AAC9D,UAAM,YAAY,KAAK,IAAI;AAC3B,WAAO,cAAc,QAAQ,MAAM,EAAE,UAAU,QAAQ,CAAC;AAExD,UAAM,OAAO,MAAM,KAAK,QAAQ;AAEhC,WAAO,IAAI,QAAQ,CAAC,SAAS,WAAW;AAEtC,WAAK,KAAK,MAAM,CAAC,KAAK,UAAU;AAC9B,YAAI,KAAK;AACP,iBAAO,MAAM,oBAAoB,EAAE,MAAM,OAAO,IAAI,QAAQ,CAAC;AAC7D,iBAAO,IAAI,MAAM,mCAAmC,IAAI,EAAE,CAAC;AAC3D;AAAA,QACF;AAEA,eAAO,MAAM,uBAAuB,EAAE,MAAM,MAAM,MAAM,KAAK,CAAC;AAE9D,YAAI,MAAM,OAAO,SAAS;AACxB,iBAAO,KAAK,kBAAkB,EAAE,MAAM,MAAM,MAAM,MAAM,QAAQ,CAAC;AACjE,iBAAO,IAAI,MAAM,mBAAmB,MAAM,IAAI,gBAAgB,OAAO,SAAS,CAAC;AAC/E;AAAA,QACF;AAGA,aAAK,SAAS,MAAM,EAAE,SAAqC,GAAG,CAACC,MAAK,SAAS;AAC3E,cAAIA,MAAK;AACP,mBAAO,MAAM,oBAAoB,EAAE,MAAM,OAAOA,KAAI,QAAQ,CAAC;AAC7D,mBAAO,IAAI,MAAM,wBAAwBA,KAAI,OAAO,EAAE,CAAC;AACvD;AAAA,UACF;AAEA,gBAAM,WAAW,KAAK,IAAI,IAAI;AAC9B,iBAAO,MAAM,uBAAuB,EAAE,MAAM,MAAM,MAAM,MAAM,UAAU,GAAG,QAAQ,KAAK,CAAC;AAEzF,kBAAQ;AAAA,YACN,SAAS,KAAK,SAAS;AAAA,YACvB,MAAM,MAAM;AAAA,YACZ;AAAA,UACF,CAAC;AAAA,QACH,CAAC;AAAA,MACH,CAAC;AAAA,IACH,CAAC;AAAA,EACH;AAAA;AAAA;AAAA;AAAA,EAKA,MAAM,UACJ,MACA,SACA,UAII,CAAC,GACqE;AAC1E,UAAM,EAAE,WAAW,SAAS,aAAa,OAAO,SAAS,MAAM,IAAI;AACnE,UAAM,YAAY,KAAK,IAAI;AAE3B,WAAO,cAAc,SAAS,MAAM;AAAA,MAClC,aAAa,QAAQ;AAAA,MACrB;AAAA,MACA;AAAA,MACA;AAAA,IACF,CAAC;AAED,UAAM,OAAO,MAAM,KAAK,QAAQ;AAEhC,WAAO,IAAI,QAAQ,CAAC,SAAS,WAAW;AACtC,YAAM,iBAAiB,MAAM;AAE3B,YAAI,QAAQ;AACV,gBAAM,aAAa,GAAG,IAAI;AAC1B,eAAK,OAAO,MAAM,YAAY,CAAC,QAAQ;AACrC,gBAAI,OAAO,IAAI,YAAY,gBAAgB;AAEzC,qBAAO,IAAI,MAAM,4BAA4B,IAAI,OAAO,EAAE,CAAC;AAC3D;AAAA,YACF;AAGA,yBAAa,UAAU;AAAA,UACzB,CAAC;AAAA,QACH,OAAO;AACL,uBAAa;AAAA,QACf;AAAA,MACF;AAEA,YAAM,eAAe,CAAC,eAAwB;AAC5C,cAAM,SAAS,OAAO,KAAK,SAAS,QAA0B;AAC9D,cAAM,WAAW,GAAG,IAAI;AAGxB,aAAK,UAAU,UAAU,QAAQ,CAAC,QAAQ;AACxC,cAAI,KAAK;AACP,mBAAO,IAAI,MAAM,yBAAyB,IAAI,OAAO,EAAE,CAAC;AACxD;AAAA,UACF;AAGA,eAAK,OAAO,UAAU,MAAM,CAACA,SAAQ;AACnC,gBAAIA,MAAK;AACP,qBAAO,MAAM,sBAAsB,EAAE,UAAU,MAAM,OAAOA,KAAI,QAAQ,CAAC;AACzE,qBAAO,IAAI,MAAM,+BAA+BA,KAAI,OAAO,EAAE,CAAC;AAC9D;AAAA,YACF;AAEA,kBAAM,WAAW,KAAK,IAAI,IAAI;AAC9B,mBAAO,MAAM,wBAAwB;AAAA,cACnC;AAAA,cACA,cAAc,OAAO;AAAA,cACrB,UAAU,GAAG,QAAQ;AAAA,cACrB;AAAA,YACF,CAAC;AAED,oBAAQ;AAAA,cACN,SAAS;AAAA,cACT,cAAc,OAAO;AAAA,cACrB;AAAA,YACF,CAAC;AAAA,UACH,CAAC;AAAA,QACH,CAAC;AAAA,MACH;AAGA,UAAI,YAAY;AACd,cAAM,UAAU,KAAK,UAAU,GAAG,KAAK,YAAY,GAAG,CAAC;AACvD,aAAK,KAAK,YAAY,OAAO,EAAE,EAAE,KAAK,MAAM;AAC1C,yBAAe;AAAA,QACjB,CAAC,EAAE,MAAM,MAAM;AAAA,MACjB,OAAO;AACL,uBAAe;AAAA,MACjB;AAAA,IACF,CAAC;AAAA,EACH;AAAA;AAAA;AAAA;AAAA,EAKA,aAAmB;AACjB,QAAI,KAAK,WAAW;AAClB,aAAO,KAAK,iCAAiC;AAAA,QAC3C,MAAM,KAAK,OAAO;AAAA,QAClB,UAAU,KAAK,OAAO;AAAA,MACxB,CAAC;AACD,WAAK,OAAO,IAAI;AAChB,WAAK,YAAY;AAAA,IACnB;AAAA,EACF;AAAA;AAAA;AAAA;AAAA,EAKA,cAAuB;AACrB,WAAO,KAAK;AAAA,EACd;AACF;;;AP9VA,eAAe,OAAO;AAEpB,QAAM,aAAa,kBAAkB;AAGrC,QAAM,gBAAgB,IAAI,qBAAqB;AAAA,IAC7C,MAAM,OAAO,IAAI;AAAA,IACjB,MAAM,OAAO,IAAI;AAAA,IACjB,UAAU,OAAO,IAAI;AAAA,IACrB;AAAA,EACF,CAAC;AAGD,QAAM,cAAc,QAAQ;AAC5B,UAAQ,MAAM,4BAA4B,OAAO,IAAI,QAAQ,IAAI,OAAO,IAAI,IAAI,EAAE;AAElF,QAAM,SAAS,IAAI;AAAA,IACjB;AAAA,MACE,MAAM;AAAA,MACN,SAAS;AAAA,IACX;AAAA,IACA;AAAA,MACE,cAAc;AAAA,QACZ,OAAO,CAAC;AAAA,MACV;AAAA,IACF;AAAA,EACF;AAGA,SAAO,kBAAkB,wBAAwB,YAAY;AAC3D,WAAO,MAAM,0BAA0B;AACvC,UAAM,QAAQ,CAAC,wBAAwB,6BAA6B,uBAAuB,sBAAsB;AACjH,WAAO,MAAM,aAAa,MAAM,MAAM,UAAU,EAAE,OAAO,MAAM,IAAI,OAAK,EAAE,IAAI,EAAE,CAAC;AACjF,WAAO,EAAE,MAAM;AAAA,EACjB,CAAC;AAED,SAAO,kBAAkB,uBAAuB,OAAO,YAAY;AACjE,UAAM,YAAY,KAAK,IAAI;AAC3B,WAAO,YAAY,QAAQ,OAAO,MAAM,QAAQ,OAAO,SAAS;AAEhE,QAAI;AACF,UAAI;AAEJ,UAAI,QAAQ,OAAO,SAAS,yBAAyB;AACnD,iBAAS,MAAM,yBAAyB,QAAQ,OAAO,WAAW,aAAa;AAAA,MACjF,WAAW,QAAQ,OAAO,SAAS,8BAA8B;AAC/D,iBAAS,MAAM,8BAA8B,QAAQ,OAAO,WAAW,aAAa;AAAA,MACtF,WAAW,QAAQ,OAAO,SAAS,wBAAwB;AACzD,iBAAS,MAAM,wBAAwB,QAAQ,OAAO,WAAW,aAAa;AAAA,MAChF,WAAW,QAAQ,OAAO,SAAS,yBAAyB;AAC1D,iBAAS,MAAM,yBAAyB,QAAQ,OAAO,WAAW,aAAa;AAAA,MACjF,OAAO;AACL,cAAM,IAAI,MAAM,iBAAiB,QAAQ,OAAO,IAAI,EAAE;AAAA,MACxD;AAEA,YAAM,WAAW,KAAK,IAAI,IAAI;AAC9B,YAAM,aAAa,KAAK,UAAU,MAAM,EAAE;AAC1C,aAAO,cAAc,QAAQ,OAAO,MAAM,UAAU,UAAU;AAE9D,aAAO;AAAA,IACT,SAAS,OAAO;AACd,aAAO,WAAW,QAAQ,OAAO,MAAM,OAAgB,QAAQ,OAAO,SAAS;AAC/E,YAAM;AAAA,IACR;AAAA,EACF,CAAC;AAGD,QAAM,UAAU,MAAM;AACpB,WAAO,KAAK,sBAAsB;AAClC,kBAAc,WAAW;AACzB,YAAQ,KAAK,CAAC;AAAA,EAChB;AAEA,UAAQ,GAAG,UAAU,OAAO;AAC5B,UAAQ,GAAG,WAAW,OAAO;AAG7B,QAAM,YAAY,IAAI,qBAAqB;AAC3C,QAAM,OAAO,QAAQ,SAAS;AAC9B,SAAO,KAAK,iCAAiC;AAC/C;AAEA,KAAK,EAAE,MAAM,CAAC,UAAU;AACtB,SAAO,MAAM,yBAAyB,EAAE,OAAO,MAAM,SAAS,OAAO,MAAM,MAAM,CAAC;AAClF,UAAQ,KAAK,CAAC;AAChB,CAAC;",
|
|
6
6
|
"names": ["path", "config", "dotenv", "config", "err"]
|
|
7
7
|
}
|
|
@@ -0,0 +1,43 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Logger utility for ACP MCP Server
|
|
3
|
+
* Provides structured logging with configurable log levels
|
|
4
|
+
*/
|
|
5
|
+
export type LogLevel = 'error' | 'warn' | 'info' | 'debug' | 'trace';
|
|
6
|
+
declare class Logger {
|
|
7
|
+
private level;
|
|
8
|
+
private enabled;
|
|
9
|
+
constructor();
|
|
10
|
+
private shouldLog;
|
|
11
|
+
private formatMessage;
|
|
12
|
+
error(message: string, data?: any): void;
|
|
13
|
+
warn(message: string, data?: any): void;
|
|
14
|
+
info(message: string, data?: any): void;
|
|
15
|
+
debug(message: string, data?: any): void;
|
|
16
|
+
trace(message: string, data?: any): void;
|
|
17
|
+
/**
|
|
18
|
+
* Log tool invocation with parameters
|
|
19
|
+
*/
|
|
20
|
+
toolInvoked(toolName: string, params: any, userId?: string): void;
|
|
21
|
+
/**
|
|
22
|
+
* Log tool completion with result summary
|
|
23
|
+
*/
|
|
24
|
+
toolCompleted(toolName: string, duration: number, resultSize?: number): void;
|
|
25
|
+
/**
|
|
26
|
+
* Log tool failure with error details
|
|
27
|
+
*/
|
|
28
|
+
toolFailed(toolName: string, error: Error, params?: any): void;
|
|
29
|
+
/**
|
|
30
|
+
* Log SSH command execution
|
|
31
|
+
*/
|
|
32
|
+
sshCommand(command: string, cwd?: string, timeout?: number): void;
|
|
33
|
+
/**
|
|
34
|
+
* Log SSH command result
|
|
35
|
+
*/
|
|
36
|
+
sshCommandResult(exitCode: number, duration: number, stdoutSize: number, stderrSize: number): void;
|
|
37
|
+
/**
|
|
38
|
+
* Log file operation
|
|
39
|
+
*/
|
|
40
|
+
fileOperation(operation: string, path: string, details?: any): void;
|
|
41
|
+
}
|
|
42
|
+
export declare const logger: Logger;
|
|
43
|
+
export {};
|
package/package.json
CHANGED
package/src/server-factory.ts
CHANGED
|
@@ -10,6 +10,7 @@ import { acpRemoteReadFileTool, handleAcpRemoteReadFile } from './tools/acp-remo
|
|
|
10
10
|
import { acpRemoteWriteFileTool, handleAcpRemoteWriteFile } from './tools/acp-remote-write-file.js';
|
|
11
11
|
import { ServerConfig } from './types/ssh-config.js';
|
|
12
12
|
import { SSHConnectionManager } from './utils/ssh-connection.js';
|
|
13
|
+
import { logger } from './utils/logger.js';
|
|
13
14
|
|
|
14
15
|
/**
|
|
15
16
|
* Create an MCP server instance for a specific user with SSH configuration
|
|
@@ -19,11 +20,20 @@ import { SSHConnectionManager } from './utils/ssh-connection.js';
|
|
|
19
20
|
* @returns Configured MCP Server instance
|
|
20
21
|
*/
|
|
21
22
|
export async function createServer(serverConfig: ServerConfig): Promise<Server> {
|
|
23
|
+
logger.info('Creating server instance', { userId: serverConfig.userId });
|
|
24
|
+
logger.debug('SSH configuration', {
|
|
25
|
+
host: serverConfig.ssh.host,
|
|
26
|
+
port: serverConfig.ssh.port,
|
|
27
|
+
username: serverConfig.ssh.username,
|
|
28
|
+
});
|
|
29
|
+
|
|
22
30
|
// Create SSH connection manager
|
|
23
31
|
const sshConnection = new SSHConnectionManager(serverConfig.ssh);
|
|
24
32
|
|
|
25
33
|
// Connect to remote server
|
|
26
34
|
await sshConnection.connect();
|
|
35
|
+
|
|
36
|
+
logger.info('Server created successfully', { userId: serverConfig.userId });
|
|
27
37
|
|
|
28
38
|
const server = new Server(
|
|
29
39
|
{
|
|
@@ -38,34 +48,53 @@ export async function createServer(serverConfig: ServerConfig): Promise<Server>
|
|
|
38
48
|
);
|
|
39
49
|
|
|
40
50
|
// Register tools with SSH connection context
|
|
41
|
-
server.setRequestHandler(ListToolsRequestSchema, async () =>
|
|
42
|
-
|
|
43
|
-
|
|
51
|
+
server.setRequestHandler(ListToolsRequestSchema, async () => {
|
|
52
|
+
logger.debug('Tool discovery requested', { userId: serverConfig.userId });
|
|
53
|
+
const tools = [acpRemoteListFilesTool, acpRemoteExecuteCommandTool, acpRemoteReadFileTool, acpRemoteWriteFileTool];
|
|
54
|
+
logger.debug(`Returning ${tools.length} tools`, { tools: tools.map(t => t.name), userId: serverConfig.userId });
|
|
55
|
+
return { tools };
|
|
56
|
+
});
|
|
44
57
|
|
|
45
58
|
server.setRequestHandler(CallToolRequestSchema, async (request) => {
|
|
46
|
-
|
|
47
|
-
|
|
48
|
-
|
|
49
|
-
|
|
50
|
-
|
|
51
|
-
|
|
52
|
-
|
|
53
|
-
|
|
54
|
-
|
|
55
|
-
|
|
56
|
-
|
|
57
|
-
|
|
59
|
+
const startTime = Date.now();
|
|
60
|
+
logger.toolInvoked(request.params.name, request.params.arguments, serverConfig.userId);
|
|
61
|
+
|
|
62
|
+
try {
|
|
63
|
+
let result;
|
|
64
|
+
|
|
65
|
+
if (request.params.name === 'acp_remote_list_files') {
|
|
66
|
+
// Pass SSH connection to handler for remote operations
|
|
67
|
+
result = await handleAcpRemoteListFiles(request.params.arguments, sshConnection);
|
|
68
|
+
} else if (request.params.name === 'acp_remote_execute_command') {
|
|
69
|
+
result = await handleAcpRemoteExecuteCommand(request.params.arguments, sshConnection);
|
|
70
|
+
} else if (request.params.name === 'acp_remote_read_file') {
|
|
71
|
+
result = await handleAcpRemoteReadFile(request.params.arguments, sshConnection);
|
|
72
|
+
} else if (request.params.name === 'acp_remote_write_file') {
|
|
73
|
+
result = await handleAcpRemoteWriteFile(request.params.arguments, sshConnection);
|
|
74
|
+
} else {
|
|
75
|
+
throw new Error(`Unknown tool: ${request.params.name}`);
|
|
76
|
+
}
|
|
77
|
+
|
|
78
|
+
const duration = Date.now() - startTime;
|
|
79
|
+
const resultSize = JSON.stringify(result).length;
|
|
80
|
+
logger.toolCompleted(request.params.name, duration, resultSize);
|
|
81
|
+
|
|
82
|
+
return result;
|
|
83
|
+
} catch (error) {
|
|
84
|
+
logger.toolFailed(request.params.name, error as Error, request.params.arguments);
|
|
85
|
+
throw error;
|
|
58
86
|
}
|
|
59
|
-
throw new Error(`Unknown tool: ${request.params.name}`);
|
|
60
87
|
});
|
|
61
88
|
|
|
62
89
|
// Handle server shutdown to cleanup SSH connection
|
|
63
90
|
process.on('SIGINT', () => {
|
|
91
|
+
logger.info('Received SIGINT, shutting down', { userId: serverConfig.userId });
|
|
64
92
|
sshConnection.disconnect();
|
|
65
93
|
process.exit(0);
|
|
66
94
|
});
|
|
67
95
|
|
|
68
96
|
process.on('SIGTERM', () => {
|
|
97
|
+
logger.info('Received SIGTERM, shutting down', { userId: serverConfig.userId });
|
|
69
98
|
sshConnection.disconnect();
|
|
70
99
|
process.exit(0);
|
|
71
100
|
});
|
package/src/server.ts
CHANGED
|
@@ -12,6 +12,7 @@ import { acpRemoteExecuteCommandTool, handleAcpRemoteExecuteCommand } from './to
|
|
|
12
12
|
import { acpRemoteReadFileTool, handleAcpRemoteReadFile } from './tools/acp-remote-read-file.js';
|
|
13
13
|
import { acpRemoteWriteFileTool, handleAcpRemoteWriteFile } from './tools/acp-remote-write-file.js';
|
|
14
14
|
import { SSHConnectionManager } from './utils/ssh-connection.js';
|
|
15
|
+
import { logger } from './utils/logger.js';
|
|
15
16
|
|
|
16
17
|
async function main() {
|
|
17
18
|
// Load SSH private key
|
|
@@ -42,29 +43,46 @@ async function main() {
|
|
|
42
43
|
);
|
|
43
44
|
|
|
44
45
|
// Register tools
|
|
45
|
-
server.setRequestHandler(ListToolsRequestSchema, async () =>
|
|
46
|
-
|
|
47
|
-
|
|
46
|
+
server.setRequestHandler(ListToolsRequestSchema, async () => {
|
|
47
|
+
logger.debug('Tool discovery requested');
|
|
48
|
+
const tools = [acpRemoteListFilesTool, acpRemoteExecuteCommandTool, acpRemoteReadFileTool, acpRemoteWriteFileTool];
|
|
49
|
+
logger.debug(`Returning ${tools.length} tools`, { tools: tools.map(t => t.name) });
|
|
50
|
+
return { tools };
|
|
51
|
+
});
|
|
48
52
|
|
|
49
53
|
server.setRequestHandler(CallToolRequestSchema, async (request) => {
|
|
50
|
-
|
|
51
|
-
|
|
52
|
-
|
|
53
|
-
|
|
54
|
-
|
|
55
|
-
|
|
56
|
-
|
|
57
|
-
|
|
58
|
-
|
|
59
|
-
|
|
60
|
-
|
|
54
|
+
const startTime = Date.now();
|
|
55
|
+
logger.toolInvoked(request.params.name, request.params.arguments);
|
|
56
|
+
|
|
57
|
+
try {
|
|
58
|
+
let result;
|
|
59
|
+
|
|
60
|
+
if (request.params.name === 'acp_remote_list_files') {
|
|
61
|
+
result = await handleAcpRemoteListFiles(request.params.arguments, sshConnection);
|
|
62
|
+
} else if (request.params.name === 'acp_remote_execute_command') {
|
|
63
|
+
result = await handleAcpRemoteExecuteCommand(request.params.arguments, sshConnection);
|
|
64
|
+
} else if (request.params.name === 'acp_remote_read_file') {
|
|
65
|
+
result = await handleAcpRemoteReadFile(request.params.arguments, sshConnection);
|
|
66
|
+
} else if (request.params.name === 'acp_remote_write_file') {
|
|
67
|
+
result = await handleAcpRemoteWriteFile(request.params.arguments, sshConnection);
|
|
68
|
+
} else {
|
|
69
|
+
throw new Error(`Unknown tool: ${request.params.name}`);
|
|
70
|
+
}
|
|
71
|
+
|
|
72
|
+
const duration = Date.now() - startTime;
|
|
73
|
+
const resultSize = JSON.stringify(result).length;
|
|
74
|
+
logger.toolCompleted(request.params.name, duration, resultSize);
|
|
75
|
+
|
|
76
|
+
return result;
|
|
77
|
+
} catch (error) {
|
|
78
|
+
logger.toolFailed(request.params.name, error as Error, request.params.arguments);
|
|
79
|
+
throw error;
|
|
61
80
|
}
|
|
62
|
-
throw new Error(`Unknown tool: ${request.params.name}`);
|
|
63
81
|
});
|
|
64
82
|
|
|
65
83
|
// Handle shutdown to cleanup SSH connection
|
|
66
84
|
const cleanup = () => {
|
|
67
|
-
|
|
85
|
+
logger.info('Shutting down server');
|
|
68
86
|
sshConnection.disconnect();
|
|
69
87
|
process.exit(0);
|
|
70
88
|
};
|
|
@@ -75,10 +93,10 @@ async function main() {
|
|
|
75
93
|
// Start server
|
|
76
94
|
const transport = new StdioServerTransport();
|
|
77
95
|
await server.connect(transport);
|
|
78
|
-
|
|
96
|
+
logger.info('ACP MCP Server running on stdio');
|
|
79
97
|
}
|
|
80
98
|
|
|
81
99
|
main().catch((error) => {
|
|
82
|
-
|
|
100
|
+
logger.error('Server startup failed', { error: error.message, stack: error.stack });
|
|
83
101
|
process.exit(1);
|
|
84
102
|
});
|
|
@@ -1,5 +1,6 @@
|
|
|
1
1
|
import { Tool } from '@modelcontextprotocol/sdk/types.js';
|
|
2
2
|
import { SSHConnectionManager } from '../utils/ssh-connection.js';
|
|
3
|
+
import { logger } from '../utils/logger.js';
|
|
3
4
|
|
|
4
5
|
export const acpRemoteExecuteCommandTool: Tool = {
|
|
5
6
|
name: 'acp_remote_execute_command',
|
|
@@ -51,12 +52,21 @@ export async function handleAcpRemoteExecuteCommand(
|
|
|
51
52
|
): Promise<{ content: Array<{ type: string; text: string }> }> {
|
|
52
53
|
const { command, cwd, timeout = 30 } = args as ExecuteCommandArgs;
|
|
53
54
|
|
|
55
|
+
logger.debug('Executing remote command', { command, cwd, timeout });
|
|
56
|
+
|
|
54
57
|
try {
|
|
55
58
|
// Build command with working directory if specified
|
|
56
59
|
const fullCommand = cwd ? `cd ${cwd} && ${command}` : command;
|
|
57
60
|
|
|
58
61
|
const result = await sshConnection.execWithTimeout(fullCommand, timeout);
|
|
59
62
|
|
|
63
|
+
logger.debug('Command execution result', {
|
|
64
|
+
exitCode: result.exitCode,
|
|
65
|
+
timedOut: result.timedOut,
|
|
66
|
+
stdoutLength: result.stdout.length,
|
|
67
|
+
stderrLength: result.stderr.length,
|
|
68
|
+
});
|
|
69
|
+
|
|
60
70
|
// Format output as JSON for structured response
|
|
61
71
|
const output: ExecuteCommandResult = {
|
|
62
72
|
stdout: result.stdout,
|
|
@@ -75,6 +85,7 @@ export async function handleAcpRemoteExecuteCommand(
|
|
|
75
85
|
};
|
|
76
86
|
} catch (error) {
|
|
77
87
|
const errorMessage = error instanceof Error ? error.message : String(error);
|
|
88
|
+
logger.error('Command execution error', { command, error: errorMessage });
|
|
78
89
|
return {
|
|
79
90
|
content: [
|
|
80
91
|
{
|
|
@@ -65,6 +65,7 @@ export async function handleAcpRemoteListFiles(
|
|
|
65
65
|
|
|
66
66
|
/**
|
|
67
67
|
* Recursively list files in a remote directory via SSH
|
|
68
|
+
* Returns absolute paths for compatibility with acp_remote_read_file
|
|
68
69
|
*/
|
|
69
70
|
async function listRemoteFiles(
|
|
70
71
|
ssh: SSHConnectionManager,
|
|
@@ -75,15 +76,17 @@ async function listRemoteFiles(
|
|
|
75
76
|
const files: string[] = [];
|
|
76
77
|
|
|
77
78
|
for (const entry of entries) {
|
|
79
|
+
// Construct absolute path by combining dirPath with entry name
|
|
80
|
+
const fullPath = `${dirPath}/${entry.name}`.replace(/\/+/g, '/');
|
|
81
|
+
|
|
78
82
|
if (entry.isDirectory) {
|
|
79
|
-
files.push(`${
|
|
83
|
+
files.push(`${fullPath}/`); // Return absolute path with trailing slash
|
|
80
84
|
if (recursive) {
|
|
81
|
-
const fullPath = `${dirPath}/${entry.name}`.replace(/\/+/g, '/');
|
|
82
85
|
const subFiles = await listRemoteFiles(ssh, fullPath, recursive);
|
|
83
|
-
files.push(...subFiles
|
|
86
|
+
files.push(...subFiles); // Sub-files already have absolute paths
|
|
84
87
|
}
|
|
85
88
|
} else {
|
|
86
|
-
files.push(
|
|
89
|
+
files.push(fullPath); // Return absolute path
|
|
87
90
|
}
|
|
88
91
|
}
|
|
89
92
|
|