ssh-agent-workspace 1.0.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (188) hide show
  1. package/README.md +319 -0
  2. package/dist/__tests__/SSHManager.test.d.ts +2 -0
  3. package/dist/__tests__/SSHManager.test.d.ts.map +1 -0
  4. package/dist/__tests__/SSHManager.test.js +134 -0
  5. package/dist/__tests__/SSHManager.test.js.map +1 -0
  6. package/dist/__tests__/SessionManager.test.d.ts +2 -0
  7. package/dist/__tests__/SessionManager.test.d.ts.map +1 -0
  8. package/dist/__tests__/SessionManager.test.js +141 -0
  9. package/dist/__tests__/SessionManager.test.js.map +1 -0
  10. package/dist/__tests__/StorageManager.test.d.ts +2 -0
  11. package/dist/__tests__/StorageManager.test.d.ts.map +1 -0
  12. package/dist/__tests__/StorageManager.test.js +171 -0
  13. package/dist/__tests__/StorageManager.test.js.map +1 -0
  14. package/dist/__tests__/ansi.test.d.ts +2 -0
  15. package/dist/__tests__/ansi.test.d.ts.map +1 -0
  16. package/dist/__tests__/ansi.test.js +41 -0
  17. package/dist/__tests__/ansi.test.js.map +1 -0
  18. package/dist/__tests__/security.test.d.ts +2 -0
  19. package/dist/__tests__/security.test.d.ts.map +1 -0
  20. package/dist/__tests__/security.test.js +87 -0
  21. package/dist/__tests__/security.test.js.map +1 -0
  22. package/dist/__tests__/validation.test.d.ts +2 -0
  23. package/dist/__tests__/validation.test.d.ts.map +1 -0
  24. package/dist/__tests__/validation.test.js +23 -0
  25. package/dist/__tests__/validation.test.js.map +1 -0
  26. package/dist/core/HostSecurityManager.d.ts +25 -0
  27. package/dist/core/HostSecurityManager.d.ts.map +1 -0
  28. package/dist/core/HostSecurityManager.js +76 -0
  29. package/dist/core/HostSecurityManager.js.map +1 -0
  30. package/dist/core/SSHManager.d.ts +48 -0
  31. package/dist/core/SSHManager.d.ts.map +1 -0
  32. package/dist/core/SSHManager.js +288 -0
  33. package/dist/core/SSHManager.js.map +1 -0
  34. package/dist/core/SessionManager.d.ts +15 -0
  35. package/dist/core/SessionManager.d.ts.map +1 -0
  36. package/dist/core/SessionManager.js +96 -0
  37. package/dist/core/SessionManager.js.map +1 -0
  38. package/dist/core/StorageManager.d.ts +27 -0
  39. package/dist/core/StorageManager.d.ts.map +1 -0
  40. package/dist/core/StorageManager.js +87 -0
  41. package/dist/core/StorageManager.js.map +1 -0
  42. package/dist/core/TmuxManager.d.ts +21 -0
  43. package/dist/core/TmuxManager.d.ts.map +1 -0
  44. package/dist/core/TmuxManager.js +110 -0
  45. package/dist/core/TmuxManager.js.map +1 -0
  46. package/dist/core/ToolConfigManager.d.ts +15 -0
  47. package/dist/core/ToolConfigManager.d.ts.map +1 -0
  48. package/dist/core/ToolConfigManager.js +57 -0
  49. package/dist/core/ToolConfigManager.js.map +1 -0
  50. package/dist/index.d.ts +3 -0
  51. package/dist/index.d.ts.map +1 -0
  52. package/dist/index.js +169 -0
  53. package/dist/index.js.map +1 -0
  54. package/dist/server.d.ts +44 -0
  55. package/dist/server.d.ts.map +1 -0
  56. package/dist/server.js +152 -0
  57. package/dist/server.js.map +1 -0
  58. package/dist/tools/backup.d.ts +74 -0
  59. package/dist/tools/backup.d.ts.map +1 -0
  60. package/dist/tools/backup.js +152 -0
  61. package/dist/tools/backup.js.map +1 -0
  62. package/dist/tools/connect.d.ts +46 -0
  63. package/dist/tools/connect.d.ts.map +1 -0
  64. package/dist/tools/connect.js +235 -0
  65. package/dist/tools/connect.js.map +1 -0
  66. package/dist/tools/connection_status.d.ts +39 -0
  67. package/dist/tools/connection_status.d.ts.map +1 -0
  68. package/dist/tools/connection_status.js +67 -0
  69. package/dist/tools/connection_status.js.map +1 -0
  70. package/dist/tools/db_query.d.ts +103 -0
  71. package/dist/tools/db_query.d.ts.map +1 -0
  72. package/dist/tools/db_query.js +194 -0
  73. package/dist/tools/db_query.js.map +1 -0
  74. package/dist/tools/deploy.d.ts +127 -0
  75. package/dist/tools/deploy.d.ts.map +1 -0
  76. package/dist/tools/deploy.js +201 -0
  77. package/dist/tools/deploy.js.map +1 -0
  78. package/dist/tools/disconnect.d.ts +46 -0
  79. package/dist/tools/disconnect.d.ts.map +1 -0
  80. package/dist/tools/disconnect.js +77 -0
  81. package/dist/tools/disconnect.js.map +1 -0
  82. package/dist/tools/exec.d.ts +69 -0
  83. package/dist/tools/exec.d.ts.map +1 -0
  84. package/dist/tools/exec.js +188 -0
  85. package/dist/tools/exec.js.map +1 -0
  86. package/dist/tools/group_exec.d.ts +80 -0
  87. package/dist/tools/group_exec.d.ts.map +1 -0
  88. package/dist/tools/group_exec.js +150 -0
  89. package/dist/tools/group_exec.js.map +1 -0
  90. package/dist/tools/health_check.d.ts +38 -0
  91. package/dist/tools/health_check.d.ts.map +1 -0
  92. package/dist/tools/health_check.js +161 -0
  93. package/dist/tools/health_check.js.map +1 -0
  94. package/dist/tools/host_security.d.ts +52 -0
  95. package/dist/tools/host_security.d.ts.map +1 -0
  96. package/dist/tools/host_security.js +127 -0
  97. package/dist/tools/host_security.js.map +1 -0
  98. package/dist/tools/index.d.ts +24 -0
  99. package/dist/tools/index.d.ts.map +1 -0
  100. package/dist/tools/index.js +24 -0
  101. package/dist/tools/index.js.map +1 -0
  102. package/dist/tools/interrupt.d.ts +47 -0
  103. package/dist/tools/interrupt.d.ts.map +1 -0
  104. package/dist/tools/interrupt.js +77 -0
  105. package/dist/tools/interrupt.js.map +1 -0
  106. package/dist/tools/list_hosts.d.ts +15 -0
  107. package/dist/tools/list_hosts.d.ts.map +1 -0
  108. package/dist/tools/list_hosts.js +18 -0
  109. package/dist/tools/list_hosts.js.map +1 -0
  110. package/dist/tools/list_sessions.d.ts +16 -0
  111. package/dist/tools/list_sessions.d.ts.map +1 -0
  112. package/dist/tools/list_sessions.js +20 -0
  113. package/dist/tools/list_sessions.js.map +1 -0
  114. package/dist/tools/read_output.d.ts +46 -0
  115. package/dist/tools/read_output.d.ts.map +1 -0
  116. package/dist/tools/read_output.js +73 -0
  117. package/dist/tools/read_output.js.map +1 -0
  118. package/dist/tools/reconnect_to_tmux.d.ts +53 -0
  119. package/dist/tools/reconnect_to_tmux.d.ts.map +1 -0
  120. package/dist/tools/reconnect_to_tmux.js +199 -0
  121. package/dist/tools/reconnect_to_tmux.js.map +1 -0
  122. package/dist/tools/send_input.d.ts +45 -0
  123. package/dist/tools/send_input.d.ts.map +1 -0
  124. package/dist/tools/send_input.js +83 -0
  125. package/dist/tools/send_input.js.map +1 -0
  126. package/dist/tools/sftp_download.d.ts +52 -0
  127. package/dist/tools/sftp_download.d.ts.map +1 -0
  128. package/dist/tools/sftp_download.js +90 -0
  129. package/dist/tools/sftp_download.js.map +1 -0
  130. package/dist/tools/sftp_list.d.ts +46 -0
  131. package/dist/tools/sftp_list.d.ts.map +1 -0
  132. package/dist/tools/sftp_list.js +93 -0
  133. package/dist/tools/sftp_list.js.map +1 -0
  134. package/dist/tools/sftp_upload.d.ts +52 -0
  135. package/dist/tools/sftp_upload.d.ts.map +1 -0
  136. package/dist/tools/sftp_upload.js +98 -0
  137. package/dist/tools/sftp_upload.js.map +1 -0
  138. package/dist/tools/ssh_tunnel.d.ts +116 -0
  139. package/dist/tools/ssh_tunnel.d.ts.map +1 -0
  140. package/dist/tools/ssh_tunnel.js +282 -0
  141. package/dist/tools/ssh_tunnel.js.map +1 -0
  142. package/dist/tools/sync.d.ts +71 -0
  143. package/dist/tools/sync.d.ts.map +1 -0
  144. package/dist/tools/sync.js +310 -0
  145. package/dist/tools/sync.js.map +1 -0
  146. package/dist/tools/tail_log.d.ts +61 -0
  147. package/dist/tools/tail_log.d.ts.map +1 -0
  148. package/dist/tools/tail_log.js +111 -0
  149. package/dist/tools/tail_log.js.map +1 -0
  150. package/dist/tools/tools_config.d.ts +34 -0
  151. package/dist/tools/tools_config.d.ts.map +1 -0
  152. package/dist/tools/tools_config.js +98 -0
  153. package/dist/tools/tools_config.js.map +1 -0
  154. package/dist/types/index.d.ts +21 -0
  155. package/dist/types/index.d.ts.map +1 -0
  156. package/dist/types/index.js +2 -0
  157. package/dist/types/index.js.map +1 -0
  158. package/dist/utils/ansi.d.ts +2 -0
  159. package/dist/utils/ansi.d.ts.map +1 -0
  160. package/dist/utils/ansi.js +7 -0
  161. package/dist/utils/ansi.js.map +1 -0
  162. package/dist/utils/logger.d.ts +3 -0
  163. package/dist/utils/logger.d.ts.map +1 -0
  164. package/dist/utils/logger.js +8 -0
  165. package/dist/utils/logger.js.map +1 -0
  166. package/dist/utils/security.d.ts +7 -0
  167. package/dist/utils/security.d.ts.map +1 -0
  168. package/dist/utils/security.js +58 -0
  169. package/dist/utils/security.js.map +1 -0
  170. package/dist/utils/ssh.d.ts +4 -0
  171. package/dist/utils/ssh.d.ts.map +1 -0
  172. package/dist/utils/ssh.js +29 -0
  173. package/dist/utils/ssh.js.map +1 -0
  174. package/dist/utils/sshConfig.d.ts +4 -0
  175. package/dist/utils/sshConfig.d.ts.map +1 -0
  176. package/dist/utils/sshConfig.js +85 -0
  177. package/dist/utils/sshConfig.js.map +1 -0
  178. package/dist/utils/validation.d.ts +4 -0
  179. package/dist/utils/validation.d.ts.map +1 -0
  180. package/dist/utils/validation.js +12 -0
  181. package/dist/utils/validation.js.map +1 -0
  182. package/docs/SECURITY.md +213 -0
  183. package/docs/TOOLS.md +425 -0
  184. package/keygen.bat +325 -0
  185. package/package.json +48 -0
  186. package/test_check.bat +9 -0
  187. package/test_delayed.bat +12 -0
  188. package/vitest.config.ts +14 -0
@@ -0,0 +1,74 @@
1
+ import { z } from 'zod';
2
+ import { SessionManager } from '../core/SessionManager.js';
3
+ import { SSHManager } from '../core/SSHManager.js';
4
+ export declare const backupSchema: z.ZodObject<{
5
+ session_id: z.ZodString;
6
+ paths: z.ZodArray<z.ZodString, "many">;
7
+ local_dest: z.ZodString;
8
+ exclude: z.ZodDefault<z.ZodArray<z.ZodString, "many">>;
9
+ remove_remote_archive: z.ZodDefault<z.ZodBoolean>;
10
+ }, "strip", z.ZodTypeAny, {
11
+ session_id: string;
12
+ paths: string[];
13
+ local_dest: string;
14
+ exclude: string[];
15
+ remove_remote_archive: boolean;
16
+ }, {
17
+ session_id: string;
18
+ paths: string[];
19
+ local_dest: string;
20
+ exclude?: string[] | undefined;
21
+ remove_remote_archive?: boolean | undefined;
22
+ }>;
23
+ export declare const backupTool: {
24
+ name: string;
25
+ description: string;
26
+ inputSchema: {
27
+ type: "object";
28
+ properties: {
29
+ session_id: {
30
+ type: string;
31
+ description: string;
32
+ };
33
+ paths: {
34
+ type: string;
35
+ items: {
36
+ type: string;
37
+ };
38
+ description: string;
39
+ };
40
+ local_dest: {
41
+ type: string;
42
+ description: string;
43
+ };
44
+ exclude: {
45
+ type: string;
46
+ items: {
47
+ type: string;
48
+ };
49
+ description: string;
50
+ default: never[];
51
+ };
52
+ remove_remote_archive: {
53
+ type: string;
54
+ description: string;
55
+ default: boolean;
56
+ };
57
+ };
58
+ required: string[];
59
+ };
60
+ };
61
+ export declare function handleBackup(args: unknown, sessionManager: SessionManager, sshManager: SSHManager): Promise<{
62
+ content: {
63
+ type: "text";
64
+ text: string;
65
+ }[];
66
+ isError: boolean;
67
+ } | {
68
+ content: {
69
+ type: "text";
70
+ text: string;
71
+ }[];
72
+ isError?: undefined;
73
+ }>;
74
+ //# sourceMappingURL=backup.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"backup.d.ts","sourceRoot":"","sources":["../../src/tools/backup.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,CAAC,EAAE,MAAM,KAAK,CAAC;AAGxB,OAAO,EAAE,cAAc,EAAE,MAAM,2BAA2B,CAAC;AAC3D,OAAO,EAAE,UAAU,EAAE,MAAM,uBAAuB,CAAC;AAInD,eAAO,MAAM,YAAY;;;;;;;;;;;;;;;;;;EAMvB,CAAC;AAEH,eAAO,MAAM,UAAU;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;CAwBtB,CAAC;AAMF,wBAAsB,YAAY,CAChC,IAAI,EAAE,OAAO,EACb,cAAc,EAAE,cAAc,EAC9B,UAAU,EAAE,UAAU;;;;;;;;;;;;GA6HvB"}
@@ -0,0 +1,152 @@
1
+ import { z } from 'zod';
2
+ import fs from 'fs';
3
+ import path from 'path';
4
+ import { isReadOnlyMode } from '../utils/security.js';
5
+ import { logger } from '../utils/logger.js';
6
+ export const backupSchema = z.object({
7
+ session_id: z.string().min(1),
8
+ paths: z.array(z.string().min(1)).min(1).max(100).describe('Absolute remote paths (files or directories) to include in the backup archive'),
9
+ local_dest: z.string().min(1).describe('Absolute local path where the backup archive will be saved (must end with .tar.gz)'),
10
+ exclude: z.array(z.string()).max(20).default([]).describe('Glob patterns to exclude (e.g. "*.log", "node_modules")'),
11
+ remove_remote_archive: z.boolean().default(true).describe('Delete the temporary archive from the remote host after download'),
12
+ });
13
+ export const backupTool = {
14
+ name: 'backup',
15
+ description: 'Create a compressed tar.gz backup of remote paths and download it locally. Steps: validate paths -> create remote archive -> download -> cleanup.',
16
+ inputSchema: {
17
+ type: 'object',
18
+ properties: {
19
+ session_id: { type: 'string', description: 'Session ID returned by connect' },
20
+ paths: {
21
+ type: 'array',
22
+ items: { type: 'string' },
23
+ description: 'Absolute remote paths (files/directories) to backup',
24
+ },
25
+ local_dest: { type: 'string', description: 'Absolute local path to save the .tar.gz archive' },
26
+ exclude: {
27
+ type: 'array',
28
+ items: { type: 'string' },
29
+ description: 'Glob patterns to exclude (e.g. "*.log", "node_modules")',
30
+ default: [],
31
+ },
32
+ remove_remote_archive: { type: 'boolean', description: 'Delete remote temp archive after download (default: true)', default: true },
33
+ },
34
+ required: ['session_id', 'paths', 'local_dest'],
35
+ },
36
+ };
37
+ function q(value) {
38
+ return `'${value.replace(/'/g, "'\\''")}'`;
39
+ }
40
+ export async function handleBackup(args, sessionManager, sshManager) {
41
+ const parsed = backupSchema.safeParse(args);
42
+ if (!parsed.success) {
43
+ return { content: [{ type: 'text', text: `Invalid arguments: ${parsed.error.message}` }], isError: true };
44
+ }
45
+ const { session_id, paths, local_dest, exclude, remove_remote_archive } = parsed.data;
46
+ const session = sessionManager.get(session_id);
47
+ if (!session) {
48
+ return { content: [{ type: 'text', text: `Error: Session '${session_id}' not found` }], isError: true };
49
+ }
50
+ if (!sshManager.isAlive(session.ssh)) {
51
+ return { content: [{ type: 'text', text: `Error: SSH connection dead for session '${session_id}'` }], isError: true };
52
+ }
53
+ if (isReadOnlyMode(session.host)) {
54
+ return { content: [{ type: 'text', text: `Error: Host '${session.host}' is in read-only mode. Backup is disabled.` }], isError: true };
55
+ }
56
+ if (!local_dest.endsWith('.tar.gz')) {
57
+ return {
58
+ content: [{ type: 'text', text: 'Error: local_dest must end with .tar.gz' }],
59
+ isError: true,
60
+ };
61
+ }
62
+ for (const p of paths) {
63
+ if (p.startsWith('-') || p.includes(';') || p.includes('&&') || p.includes('|')) {
64
+ return {
65
+ content: [{ type: 'text', text: `Error: Invalid characters in path: ${p}` }],
66
+ isError: true,
67
+ };
68
+ }
69
+ }
70
+ const resolvedLocal = path.resolve(local_dest);
71
+ const localDir = path.dirname(resolvedLocal);
72
+ if (!fs.existsSync(localDir)) {
73
+ fs.mkdirSync(localDir, { recursive: true });
74
+ }
75
+ if (fs.existsSync(resolvedLocal)) {
76
+ return {
77
+ content: [{ type: 'text', text: `Error: Local file already exists: ${resolvedLocal}` }],
78
+ isError: true,
79
+ };
80
+ }
81
+ const verifyScript = paths.map((p) => `test -e ${q(p)} || echo "MISSING:${p}"`).join('; ');
82
+ const verifyResult = await sshManager.exec(session.ssh, verifyScript, 10000);
83
+ if (verifyResult.stdout.includes('MISSING:')) {
84
+ const missing = verifyResult.stdout
85
+ .split('\n')
86
+ .filter((l) => l.startsWith('MISSING:'))
87
+ .map((l) => l.replace('MISSING:', ''))
88
+ .join(', ');
89
+ return {
90
+ content: [{ type: 'text', text: `Error: Remote paths not found: ${missing}` }],
91
+ isError: true,
92
+ };
93
+ }
94
+ const timestamp = Date.now();
95
+ const archiveName = `mcp_backup_${timestamp}.tar.gz`;
96
+ const remoteArchive = `/tmp/${archiveName}`;
97
+ const excludeArgs = exclude.map((e) => `--exclude=${q(e)}`).join(' ');
98
+ const pathArgs = paths.map((p) => q(p)).join(' ');
99
+ const tarCmd = `tar czf ${q(remoteArchive)} ${excludeArgs} ${pathArgs} 2>&1`;
100
+ const tarResult = await sshManager.exec(session.ssh, tarCmd, 120000);
101
+ if (tarResult.code !== 0) {
102
+ const error = tarResult.stderr || tarResult.stdout || 'unknown error';
103
+ logger.error({ sessionId: session_id, paths, error }, 'Tar backup failed');
104
+ return {
105
+ content: [{ type: 'text', text: `Error: Failed to create remote archive: ${error}` }],
106
+ isError: true,
107
+ };
108
+ }
109
+ const statResult = await sshManager.exec(session.ssh, `stat -c%s ${q(remoteArchive)} 2>/dev/null || wc -c < ${q(remoteArchive)}`, 5000);
110
+ let archiveSize = parseInt(statResult.stdout.trim(), 10);
111
+ if (isNaN(archiveSize))
112
+ archiveSize = 0;
113
+ try {
114
+ await sshManager.sftpDownload(session.ssh, remoteArchive, resolvedLocal);
115
+ }
116
+ catch (err) {
117
+ logger.error({ sessionId: session_id, remoteArchive, localDest: resolvedLocal, error: err.message }, 'Backup download failed');
118
+ try {
119
+ await sshManager.exec(session.ssh, `rm -f ${q(remoteArchive)}`, 5000);
120
+ }
121
+ catch { }
122
+ return {
123
+ content: [{ type: 'text', text: `Error: Download failed: ${err.message}` }],
124
+ isError: true,
125
+ };
126
+ }
127
+ const localSize = fs.statSync(resolvedLocal).size;
128
+ if (remove_remote_archive) {
129
+ try {
130
+ await sshManager.exec(session.ssh, `rm -f ${q(remoteArchive)}`, 5000);
131
+ }
132
+ catch {
133
+ logger.warn({ sessionId: session_id, remoteArchive }, 'Failed to clean up remote archive');
134
+ }
135
+ }
136
+ logger.info({ sessionId: session_id, paths, archiveSize, localSize, localDest: resolvedLocal }, 'Backup complete');
137
+ return {
138
+ content: [{
139
+ type: 'text',
140
+ text: JSON.stringify({
141
+ session_id,
142
+ local_path: resolvedLocal,
143
+ remote_archive_removed: remove_remote_archive,
144
+ archive_size_bytes: localSize,
145
+ paths_backed_up: paths,
146
+ excluded_patterns: exclude,
147
+ timestamp: new Date().toISOString(),
148
+ }, null, 2),
149
+ }],
150
+ };
151
+ }
152
+ //# sourceMappingURL=backup.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"backup.js","sourceRoot":"","sources":["../../src/tools/backup.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,46 @@
1
+ import { z } from 'zod';
2
+ import { SessionManager } from '../core/SessionManager.js';
3
+ import { SSHManager } from '../core/SSHManager.js';
4
+ import { TmuxManager } from '../core/TmuxManager.js';
5
+ export declare const connectSchema: z.ZodObject<{
6
+ host: z.ZodString;
7
+ proxy_jump: z.ZodOptional<z.ZodString>;
8
+ }, "strip", z.ZodTypeAny, {
9
+ host: string;
10
+ proxy_jump?: string | undefined;
11
+ }, {
12
+ host: string;
13
+ proxy_jump?: string | undefined;
14
+ }>;
15
+ export declare const connectTool: {
16
+ name: string;
17
+ description: string;
18
+ inputSchema: {
19
+ type: "object";
20
+ properties: {
21
+ host: {
22
+ type: string;
23
+ description: string;
24
+ };
25
+ proxy_jump: {
26
+ type: string;
27
+ description: string;
28
+ };
29
+ };
30
+ required: string[];
31
+ };
32
+ };
33
+ export declare function handleConnect(args: unknown, sessionManager: SessionManager, sshManager: SSHManager, tmuxManager: TmuxManager): Promise<{
34
+ content: {
35
+ type: "text";
36
+ text: string;
37
+ }[];
38
+ isError: boolean;
39
+ } | {
40
+ content: {
41
+ type: "text";
42
+ text: string;
43
+ }[];
44
+ isError?: undefined;
45
+ }>;
46
+ //# sourceMappingURL=connect.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"connect.d.ts","sourceRoot":"","sources":["../../src/tools/connect.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,CAAC,EAAE,MAAM,KAAK,CAAC;AACxB,OAAO,EAAE,cAAc,EAAE,MAAM,2BAA2B,CAAC;AAC3D,OAAO,EAAE,UAAU,EAAE,MAAM,uBAAuB,CAAC;AACnD,OAAO,EAAE,WAAW,EAAc,MAAM,wBAAwB,CAAC;AAQjE,eAAO,MAAM,aAAa;;;;;;;;;EAGxB,CAAC;AAEH,eAAO,MAAM,WAAW;;;;;;;;;;;;;;;;;CAkBvB,CAAC;AAEF,wBAAsB,aAAa,CACjC,IAAI,EAAE,OAAO,EACb,cAAc,EAAE,cAAc,EAC9B,UAAU,EAAE,UAAU,EACtB,WAAW,EAAE,WAAW;;;;;;;;;;;;GA+NzB"}
@@ -0,0 +1,235 @@
1
+ import { z } from 'zod';
2
+ import { MCP_PROMPT } from '../core/TmuxManager.js';
3
+ import { listHostAliases, getHostConfig } from '../utils/sshConfig.js';
4
+ import { buildSshConfig } from '../utils/ssh.js';
5
+ import { sanitizeTmuxSessionName } from '../utils/validation.js';
6
+ import { isHostAllowed } from '../utils/security.js';
7
+ import { logger } from '../utils/logger.js';
8
+ import { v4 as uuidv4 } from 'uuid';
9
+ export const connectSchema = z.object({
10
+ host: z.string().min(1).describe('SSH host alias from ~/.ssh/config'),
11
+ proxy_jump: z.string().min(1).optional().describe('SSH host alias of the bastion/jump host to tunnel through'),
12
+ });
13
+ export const connectTool = {
14
+ name: 'connect',
15
+ description: 'Connect to a remote host via SSH and establish a persistent tmux-backed interactive shell session. Supports bash and zsh only. Use proxy_jump to tunnel through a bastion host.',
16
+ inputSchema: {
17
+ type: 'object',
18
+ properties: {
19
+ host: {
20
+ type: 'string',
21
+ description: 'SSH host alias from ~/.ssh/config',
22
+ },
23
+ proxy_jump: {
24
+ type: 'string',
25
+ description: 'Optional SSH host alias of a bastion/jump host. Overrides ProxyJump from ~/.ssh/config.',
26
+ },
27
+ },
28
+ required: ['host'],
29
+ },
30
+ };
31
+ export async function handleConnect(args, sessionManager, sshManager, tmuxManager) {
32
+ const parsed = connectSchema.safeParse(args);
33
+ if (!parsed.success) {
34
+ return {
35
+ content: [
36
+ {
37
+ type: 'text',
38
+ text: `Invalid arguments: ${parsed.error.message}`,
39
+ },
40
+ ],
41
+ isError: true,
42
+ };
43
+ }
44
+ const { host } = parsed.data;
45
+ // Security: check host allowlist
46
+ if (!isHostAllowed(host)) {
47
+ logger.warn({ host }, 'Host rejected by allowlist');
48
+ return {
49
+ content: [
50
+ {
51
+ type: 'text',
52
+ text: `Error: Host '${host}' is not in the allowed hosts list`,
53
+ },
54
+ ],
55
+ isError: true,
56
+ };
57
+ }
58
+ // Validate host exists in SSH config
59
+ const aliases = listHostAliases();
60
+ if (!aliases.includes(host)) {
61
+ return {
62
+ content: [
63
+ {
64
+ type: 'text',
65
+ text: `Error: Host alias '${host}' not found in SSH config.\nAvailable aliases: ${aliases.join(', ') || '(none)'}`,
66
+ },
67
+ ],
68
+ isError: true,
69
+ };
70
+ }
71
+ const hostConfig = getHostConfig(host);
72
+ if (!hostConfig) {
73
+ return {
74
+ content: [
75
+ {
76
+ type: 'text',
77
+ text: `Error: Failed to resolve SSH configuration for '${host}'`,
78
+ },
79
+ ],
80
+ isError: true,
81
+ };
82
+ }
83
+ const proxyJumpAlias = parsed.data.proxy_jump || hostConfig.proxyJump;
84
+ const sshConfig = buildSshConfig(hostConfig);
85
+ if (!sshConfig) {
86
+ return {
87
+ content: [
88
+ {
89
+ type: 'text',
90
+ text: `Error: Failed to read SSH private key for '${host}'`,
91
+ },
92
+ ],
93
+ isError: true,
94
+ };
95
+ }
96
+ let ssh;
97
+ try {
98
+ if (proxyJumpAlias) {
99
+ if (!isHostAllowed(proxyJumpAlias)) {
100
+ return {
101
+ content: [{ type: 'text', text: `Error: Proxy host '${proxyJumpAlias}' is not in the allowed hosts list` }],
102
+ isError: true,
103
+ };
104
+ }
105
+ const proxyConfig = getHostConfig(proxyJumpAlias);
106
+ if (!proxyConfig) {
107
+ return {
108
+ content: [{ type: 'text', text: `Error: Failed to resolve SSH config for proxy '${proxyJumpAlias}'` }],
109
+ isError: true,
110
+ };
111
+ }
112
+ const proxySshConfig = buildSshConfig(proxyConfig);
113
+ if (!proxySshConfig) {
114
+ return {
115
+ content: [{ type: 'text', text: `Error: Failed to read SSH key for proxy '${proxyJumpAlias}'` }],
116
+ isError: true,
117
+ };
118
+ }
119
+ ssh = await sshManager.connectWithProxy(proxySshConfig, sshConfig);
120
+ logger.info({ host, proxy: proxyJumpAlias }, 'Connected via bastion');
121
+ }
122
+ else {
123
+ ssh = await sshManager.connect(sshConfig);
124
+ }
125
+ }
126
+ catch (err) {
127
+ logger.error({ host, proxyJump: proxyJumpAlias, error: err.message }, 'SSH connection failed');
128
+ return {
129
+ content: [
130
+ {
131
+ type: 'text',
132
+ text: `Error: SSH connection failed: ${err.message}`,
133
+ },
134
+ ],
135
+ isError: true,
136
+ };
137
+ }
138
+ // Check tmux is installed
139
+ const tmuxInstalled = await tmuxManager.isInstalled(ssh);
140
+ if (!tmuxInstalled) {
141
+ sshManager.disconnect(ssh);
142
+ return {
143
+ content: [
144
+ {
145
+ type: 'text',
146
+ text: `Error: tmux is not installed on host '${host}'. Please install tmux first.`,
147
+ },
148
+ ],
149
+ isError: true,
150
+ };
151
+ }
152
+ const tmuxSessionName = `mcp_${sanitizeTmuxSessionName(host)}_${uuidv4().replace(/-/g, '').slice(0, 12)}`;
153
+ try {
154
+ await tmuxManager.createOrAttachSession(ssh, tmuxSessionName);
155
+ }
156
+ catch (err) {
157
+ sshManager.disconnect(ssh);
158
+ return {
159
+ content: [
160
+ {
161
+ type: 'text',
162
+ text: `Error: Failed to create tmux session: ${err.message}`,
163
+ },
164
+ ],
165
+ isError: true,
166
+ };
167
+ }
168
+ // Detect shell
169
+ let shell = null;
170
+ try {
171
+ shell = await tmuxManager.detectShell(ssh, tmuxSessionName);
172
+ }
173
+ catch (err) {
174
+ logger.warn({ host, error: err.message }, 'Shell detection failed, continuing anyway');
175
+ }
176
+ const supportedShells = ['bash', 'zsh'];
177
+ if (shell && !supportedShells.includes(shell)) {
178
+ try {
179
+ await tmuxManager.killSession(ssh, tmuxSessionName);
180
+ }
181
+ catch { }
182
+ sshManager.disconnect(ssh);
183
+ return {
184
+ content: [
185
+ {
186
+ type: 'text',
187
+ text: `Error: Unsupported shell '${shell}' on host '${host}'. Only bash and zsh are supported in V1.`,
188
+ },
189
+ ],
190
+ isError: true,
191
+ };
192
+ }
193
+ // Inject deterministic prompt with retry + verification
194
+ let promptInjected = false;
195
+ for (let attempt = 0; attempt < 5; attempt++) {
196
+ try {
197
+ await new Promise((r) => setTimeout(r, 300));
198
+ await tmuxManager.injectPrompt(ssh, tmuxSessionName);
199
+ await new Promise((r) => setTimeout(r, 300));
200
+ const output = await tmuxManager.capturePane(ssh, tmuxSessionName, 10);
201
+ if (output.includes(MCP_PROMPT)) {
202
+ promptInjected = true;
203
+ break;
204
+ }
205
+ }
206
+ catch {
207
+ logger.warn({ host, attempt }, 'Prompt injection retry');
208
+ }
209
+ }
210
+ if (!promptInjected) {
211
+ logger.warn({ host }, 'Failed to inject deterministic prompt after retries, continuing');
212
+ }
213
+ // Apply tmux options for AI-friendly behavior
214
+ try {
215
+ await tmuxManager.applyOptions(ssh, tmuxSessionName);
216
+ }
217
+ catch (err) {
218
+ logger.warn({ host, tmuxSessionName, error: err.message }, 'Failed to apply tmux options, continuing anyway');
219
+ }
220
+ const session = sessionManager.create(host, ssh, tmuxSessionName, shell || undefined);
221
+ return {
222
+ content: [
223
+ {
224
+ type: 'text',
225
+ text: JSON.stringify({
226
+ session_id: session.id,
227
+ host: session.host,
228
+ tmux_session: session.tmuxSession,
229
+ shell: shell || 'unknown',
230
+ }, null, 2),
231
+ },
232
+ ],
233
+ };
234
+ }
235
+ //# sourceMappingURL=connect.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"connect.js","sourceRoot":"","sources":["../../src/tools/connect.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,39 @@
1
+ import { z } from 'zod';
2
+ import { SessionManager } from '../core/SessionManager.js';
3
+ import { SSHManager } from '../core/SSHManager.js';
4
+ import { TmuxManager } from '../core/TmuxManager.js';
5
+ export declare const connectionStatusSchema: z.ZodObject<{
6
+ session_id: z.ZodString;
7
+ }, "strip", z.ZodTypeAny, {
8
+ session_id: string;
9
+ }, {
10
+ session_id: string;
11
+ }>;
12
+ export declare const connectionStatusTool: {
13
+ name: string;
14
+ description: string;
15
+ inputSchema: {
16
+ type: "object";
17
+ properties: {
18
+ session_id: {
19
+ type: string;
20
+ description: string;
21
+ };
22
+ };
23
+ required: string[];
24
+ };
25
+ };
26
+ export declare function handleConnectionStatus(args: unknown, sessionManager: SessionManager, sshManager: SSHManager, tmuxManager: TmuxManager): Promise<{
27
+ content: {
28
+ type: "text";
29
+ text: string;
30
+ }[];
31
+ isError: boolean;
32
+ } | {
33
+ content: {
34
+ type: "text";
35
+ text: string;
36
+ }[];
37
+ isError?: undefined;
38
+ }>;
39
+ //# sourceMappingURL=connection_status.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"connection_status.d.ts","sourceRoot":"","sources":["../../src/tools/connection_status.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,CAAC,EAAE,MAAM,KAAK,CAAC;AACxB,OAAO,EAAE,cAAc,EAAE,MAAM,2BAA2B,CAAC;AAC3D,OAAO,EAAE,UAAU,EAAE,MAAM,uBAAuB,CAAC;AACnD,OAAO,EAAE,WAAW,EAAE,MAAM,wBAAwB,CAAC;AAGrD,eAAO,MAAM,sBAAsB;;;;;;EAEjC,CAAC;AAEH,eAAO,MAAM,oBAAoB;;;;;;;;;;;;;CAchC,CAAC;AAEF,wBAAsB,sBAAsB,CAC1C,IAAI,EAAE,OAAO,EACb,cAAc,EAAE,cAAc,EAC9B,UAAU,EAAE,UAAU,EACtB,WAAW,EAAE,WAAW;;;;;;;;;;;;GAmDzB"}
@@ -0,0 +1,67 @@
1
+ import { z } from 'zod';
2
+ import { logger } from '../utils/logger.js';
3
+ export const connectionStatusSchema = z.object({
4
+ session_id: z.string().min(1),
5
+ });
6
+ export const connectionStatusTool = {
7
+ name: 'connection_status',
8
+ description: 'Check the health of an active session: SSH connection liveness and tmux session existence.',
9
+ inputSchema: {
10
+ type: 'object',
11
+ properties: {
12
+ session_id: {
13
+ type: 'string',
14
+ description: 'Session ID returned by connect',
15
+ },
16
+ },
17
+ required: ['session_id'],
18
+ },
19
+ };
20
+ export async function handleConnectionStatus(args, sessionManager, sshManager, tmuxManager) {
21
+ const parsed = connectionStatusSchema.safeParse(args);
22
+ if (!parsed.success) {
23
+ return {
24
+ content: [{ type: 'text', text: `Invalid arguments: ${parsed.error.message}` }],
25
+ isError: true,
26
+ };
27
+ }
28
+ const { session_id } = parsed.data;
29
+ const session = sessionManager.get(session_id);
30
+ if (!session) {
31
+ return {
32
+ content: [{ type: 'text', text: JSON.stringify({
33
+ session_id,
34
+ exists: false,
35
+ ssh_alive: false,
36
+ tmux_exists: false,
37
+ }, null, 2) }],
38
+ };
39
+ }
40
+ let sshAlive = sshManager.isAlive(session.ssh);
41
+ let tmuxExists = false;
42
+ if (sshAlive) {
43
+ try {
44
+ tmuxExists = await tmuxManager.hasSession(session.ssh, session.tmuxSession);
45
+ }
46
+ catch (err) {
47
+ logger.warn({ sessionId: session_id, error: err.message }, 'tmux hasSession check failed');
48
+ }
49
+ }
50
+ return {
51
+ content: [{
52
+ type: 'text',
53
+ text: JSON.stringify({
54
+ session_id,
55
+ host: session.host,
56
+ exists: true,
57
+ ssh_alive: sshAlive,
58
+ tmux_exists: tmuxExists,
59
+ tmux_session: session.tmuxSession,
60
+ shell: session.shell || 'unknown',
61
+ connected_at: new Date(session.connectedAt).toISOString(),
62
+ last_activity: new Date(session.lastActivity).toISOString(),
63
+ }, null, 2),
64
+ }],
65
+ };
66
+ }
67
+ //# sourceMappingURL=connection_status.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"connection_status.js","sourceRoot":"","sources":["../../src/tools/connection_status.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,CAAC,EAAE,MAAM,KAAK,CAAC;AAIxB,OAAO,EAAE,MAAM,EAAE,MAAM,oBAAoB,CAAC;AAE5C,MAAM,CAAC,MAAM,sBAAsB,GAAG,CAAC,CAAC,MAAM,CAAC;IAC7C,UAAU,EAAE,CAAC,CAAC,MAAM,EAAE,CAAC,GAAG,CAAC,CAAC,CAAC;CAC9B,CAAC,CAAC;AAEH,MAAM,CAAC,MAAM,oBAAoB,GAAG;IAClC,IAAI,EAAE,mBAAmB;IACzB,WAAW,EACT,4FAA4F;IAC9F,WAAW,EAAE;QACX,IAAI,EAAE,QAAiB;QACvB,UAAU,EAAE;YACV,UAAU,EAAE;gBACV,IAAI,EAAE,QAAQ;gBACd,WAAW,EAAE,gCAAgC;aAC9C;SACF;QACD,QAAQ,EAAE,CAAC,YAAY,CAAC;KACzB;CACF,CAAC;AAEF,MAAM,CAAC,KAAK,UAAU,sBAAsB,CAC1C,IAAa,EACb,cAA8B,EAC9B,UAAsB,EACtB,WAAwB;IAExB,MAAM,MAAM,GAAG,sBAAsB,CAAC,SAAS,CAAC,IAAI,CAAC,CAAC;IACtD,IAAI,CAAC,MAAM,CAAC,OAAO,EAAE,CAAC;QACpB,OAAO;YACL,OAAO,EAAE,CAAC,EAAE,IAAI,EAAE,MAAe,EAAE,IAAI,EAAE,sBAAsB,MAAM,CAAC,KAAK,CAAC,OAAO,EAAE,EAAE,CAAC;YACxF,OAAO,EAAE,IAAI;SACd,CAAC;IACJ,CAAC;IAED,MAAM,EAAE,UAAU,EAAE,GAAG,MAAM,CAAC,IAAI,CAAC;IACnC,MAAM,OAAO,GAAG,cAAc,CAAC,GAAG,CAAC,UAAU,CAAC,CAAC;IAE/C,IAAI,CAAC,OAAO,EAAE,CAAC;QACb,OAAO;YACL,OAAO,EAAE,CAAC,EAAE,IAAI,EAAE,MAAe,EAAE,IAAI,EAAE,IAAI,CAAC,SAAS,CAAC;wBACtD,UAAU;wBACV,MAAM,EAAE,KAAK;wBACb,SAAS,EAAE,KAAK;wBAChB,WAAW,EAAE,KAAK;qBACnB,EAAE,IAAI,EAAE,CAAC,CAAC,EAAE,CAAC;SACf,CAAC;IACJ,CAAC;IAED,IAAI,QAAQ,GAAG,UAAU,CAAC,OAAO,CAAC,OAAO,CAAC,GAAG,CAAC,CAAC;IAC/C,IAAI,UAAU,GAAG,KAAK,CAAC;IAEvB,IAAI,QAAQ,EAAE,CAAC;QACb,IAAI,CAAC;YACH,UAAU,GAAG,MAAM,WAAW,CAAC,UAAU,CAAC,OAAO,CAAC,GAAG,EAAE,OAAO,CAAC,WAAW,CAAC,CAAC;QAC9E,CAAC;QAAC,OAAO,GAAG,EAAE,CAAC;YACb,MAAM,CAAC,IAAI,CAAC,EAAE,SAAS,EAAE,UAAU,EAAE,KAAK,EAAG,GAAa,CAAC,OAAO,EAAE,EAAE,8BAA8B,CAAC,CAAC;QACxG,CAAC;IACH,CAAC;IAED,OAAO;QACL,OAAO,EAAE,CAAC;gBACR,IAAI,EAAE,MAAe;gBACrB,IAAI,EAAE,IAAI,CAAC,SAAS,CAAC;oBACnB,UAAU;oBACV,IAAI,EAAE,OAAO,CAAC,IAAI;oBAClB,MAAM,EAAE,IAAI;oBACZ,SAAS,EAAE,QAAQ;oBACnB,WAAW,EAAE,UAAU;oBACvB,YAAY,EAAE,OAAO,CAAC,WAAW;oBACjC,KAAK,EAAE,OAAO,CAAC,KAAK,IAAI,SAAS;oBACjC,YAAY,EAAE,IAAI,IAAI,CAAC,OAAO,CAAC,WAAW,CAAC,CAAC,WAAW,EAAE;oBACzD,aAAa,EAAE,IAAI,IAAI,CAAC,OAAO,CAAC,YAAY,CAAC,CAAC,WAAW,EAAE;iBAC5D,EAAE,IAAI,EAAE,CAAC,CAAC;aACZ,CAAC;KACH,CAAC;AACJ,CAAC"}