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.
- package/README.md +319 -0
- package/dist/__tests__/SSHManager.test.d.ts +2 -0
- package/dist/__tests__/SSHManager.test.d.ts.map +1 -0
- package/dist/__tests__/SSHManager.test.js +134 -0
- package/dist/__tests__/SSHManager.test.js.map +1 -0
- package/dist/__tests__/SessionManager.test.d.ts +2 -0
- package/dist/__tests__/SessionManager.test.d.ts.map +1 -0
- package/dist/__tests__/SessionManager.test.js +141 -0
- package/dist/__tests__/SessionManager.test.js.map +1 -0
- package/dist/__tests__/StorageManager.test.d.ts +2 -0
- package/dist/__tests__/StorageManager.test.d.ts.map +1 -0
- package/dist/__tests__/StorageManager.test.js +171 -0
- package/dist/__tests__/StorageManager.test.js.map +1 -0
- package/dist/__tests__/ansi.test.d.ts +2 -0
- package/dist/__tests__/ansi.test.d.ts.map +1 -0
- package/dist/__tests__/ansi.test.js +41 -0
- package/dist/__tests__/ansi.test.js.map +1 -0
- package/dist/__tests__/security.test.d.ts +2 -0
- package/dist/__tests__/security.test.d.ts.map +1 -0
- package/dist/__tests__/security.test.js +87 -0
- package/dist/__tests__/security.test.js.map +1 -0
- package/dist/__tests__/validation.test.d.ts +2 -0
- package/dist/__tests__/validation.test.d.ts.map +1 -0
- package/dist/__tests__/validation.test.js +23 -0
- package/dist/__tests__/validation.test.js.map +1 -0
- package/dist/core/HostSecurityManager.d.ts +25 -0
- package/dist/core/HostSecurityManager.d.ts.map +1 -0
- package/dist/core/HostSecurityManager.js +76 -0
- package/dist/core/HostSecurityManager.js.map +1 -0
- package/dist/core/SSHManager.d.ts +48 -0
- package/dist/core/SSHManager.d.ts.map +1 -0
- package/dist/core/SSHManager.js +288 -0
- package/dist/core/SSHManager.js.map +1 -0
- package/dist/core/SessionManager.d.ts +15 -0
- package/dist/core/SessionManager.d.ts.map +1 -0
- package/dist/core/SessionManager.js +96 -0
- package/dist/core/SessionManager.js.map +1 -0
- package/dist/core/StorageManager.d.ts +27 -0
- package/dist/core/StorageManager.d.ts.map +1 -0
- package/dist/core/StorageManager.js +87 -0
- package/dist/core/StorageManager.js.map +1 -0
- package/dist/core/TmuxManager.d.ts +21 -0
- package/dist/core/TmuxManager.d.ts.map +1 -0
- package/dist/core/TmuxManager.js +110 -0
- package/dist/core/TmuxManager.js.map +1 -0
- package/dist/core/ToolConfigManager.d.ts +15 -0
- package/dist/core/ToolConfigManager.d.ts.map +1 -0
- package/dist/core/ToolConfigManager.js +57 -0
- package/dist/core/ToolConfigManager.js.map +1 -0
- package/dist/index.d.ts +3 -0
- package/dist/index.d.ts.map +1 -0
- package/dist/index.js +169 -0
- package/dist/index.js.map +1 -0
- package/dist/server.d.ts +44 -0
- package/dist/server.d.ts.map +1 -0
- package/dist/server.js +152 -0
- package/dist/server.js.map +1 -0
- package/dist/tools/backup.d.ts +74 -0
- package/dist/tools/backup.d.ts.map +1 -0
- package/dist/tools/backup.js +152 -0
- package/dist/tools/backup.js.map +1 -0
- package/dist/tools/connect.d.ts +46 -0
- package/dist/tools/connect.d.ts.map +1 -0
- package/dist/tools/connect.js +235 -0
- package/dist/tools/connect.js.map +1 -0
- package/dist/tools/connection_status.d.ts +39 -0
- package/dist/tools/connection_status.d.ts.map +1 -0
- package/dist/tools/connection_status.js +67 -0
- package/dist/tools/connection_status.js.map +1 -0
- package/dist/tools/db_query.d.ts +103 -0
- package/dist/tools/db_query.d.ts.map +1 -0
- package/dist/tools/db_query.js +194 -0
- package/dist/tools/db_query.js.map +1 -0
- package/dist/tools/deploy.d.ts +127 -0
- package/dist/tools/deploy.d.ts.map +1 -0
- package/dist/tools/deploy.js +201 -0
- package/dist/tools/deploy.js.map +1 -0
- package/dist/tools/disconnect.d.ts +46 -0
- package/dist/tools/disconnect.d.ts.map +1 -0
- package/dist/tools/disconnect.js +77 -0
- package/dist/tools/disconnect.js.map +1 -0
- package/dist/tools/exec.d.ts +69 -0
- package/dist/tools/exec.d.ts.map +1 -0
- package/dist/tools/exec.js +188 -0
- package/dist/tools/exec.js.map +1 -0
- package/dist/tools/group_exec.d.ts +80 -0
- package/dist/tools/group_exec.d.ts.map +1 -0
- package/dist/tools/group_exec.js +150 -0
- package/dist/tools/group_exec.js.map +1 -0
- package/dist/tools/health_check.d.ts +38 -0
- package/dist/tools/health_check.d.ts.map +1 -0
- package/dist/tools/health_check.js +161 -0
- package/dist/tools/health_check.js.map +1 -0
- package/dist/tools/host_security.d.ts +52 -0
- package/dist/tools/host_security.d.ts.map +1 -0
- package/dist/tools/host_security.js +127 -0
- package/dist/tools/host_security.js.map +1 -0
- package/dist/tools/index.d.ts +24 -0
- package/dist/tools/index.d.ts.map +1 -0
- package/dist/tools/index.js +24 -0
- package/dist/tools/index.js.map +1 -0
- package/dist/tools/interrupt.d.ts +47 -0
- package/dist/tools/interrupt.d.ts.map +1 -0
- package/dist/tools/interrupt.js +77 -0
- package/dist/tools/interrupt.js.map +1 -0
- package/dist/tools/list_hosts.d.ts +15 -0
- package/dist/tools/list_hosts.d.ts.map +1 -0
- package/dist/tools/list_hosts.js +18 -0
- package/dist/tools/list_hosts.js.map +1 -0
- package/dist/tools/list_sessions.d.ts +16 -0
- package/dist/tools/list_sessions.d.ts.map +1 -0
- package/dist/tools/list_sessions.js +20 -0
- package/dist/tools/list_sessions.js.map +1 -0
- package/dist/tools/read_output.d.ts +46 -0
- package/dist/tools/read_output.d.ts.map +1 -0
- package/dist/tools/read_output.js +73 -0
- package/dist/tools/read_output.js.map +1 -0
- package/dist/tools/reconnect_to_tmux.d.ts +53 -0
- package/dist/tools/reconnect_to_tmux.d.ts.map +1 -0
- package/dist/tools/reconnect_to_tmux.js +199 -0
- package/dist/tools/reconnect_to_tmux.js.map +1 -0
- package/dist/tools/send_input.d.ts +45 -0
- package/dist/tools/send_input.d.ts.map +1 -0
- package/dist/tools/send_input.js +83 -0
- package/dist/tools/send_input.js.map +1 -0
- package/dist/tools/sftp_download.d.ts +52 -0
- package/dist/tools/sftp_download.d.ts.map +1 -0
- package/dist/tools/sftp_download.js +90 -0
- package/dist/tools/sftp_download.js.map +1 -0
- package/dist/tools/sftp_list.d.ts +46 -0
- package/dist/tools/sftp_list.d.ts.map +1 -0
- package/dist/tools/sftp_list.js +93 -0
- package/dist/tools/sftp_list.js.map +1 -0
- package/dist/tools/sftp_upload.d.ts +52 -0
- package/dist/tools/sftp_upload.d.ts.map +1 -0
- package/dist/tools/sftp_upload.js +98 -0
- package/dist/tools/sftp_upload.js.map +1 -0
- package/dist/tools/ssh_tunnel.d.ts +116 -0
- package/dist/tools/ssh_tunnel.d.ts.map +1 -0
- package/dist/tools/ssh_tunnel.js +282 -0
- package/dist/tools/ssh_tunnel.js.map +1 -0
- package/dist/tools/sync.d.ts +71 -0
- package/dist/tools/sync.d.ts.map +1 -0
- package/dist/tools/sync.js +310 -0
- package/dist/tools/sync.js.map +1 -0
- package/dist/tools/tail_log.d.ts +61 -0
- package/dist/tools/tail_log.d.ts.map +1 -0
- package/dist/tools/tail_log.js +111 -0
- package/dist/tools/tail_log.js.map +1 -0
- package/dist/tools/tools_config.d.ts +34 -0
- package/dist/tools/tools_config.d.ts.map +1 -0
- package/dist/tools/tools_config.js +98 -0
- package/dist/tools/tools_config.js.map +1 -0
- package/dist/types/index.d.ts +21 -0
- package/dist/types/index.d.ts.map +1 -0
- package/dist/types/index.js +2 -0
- package/dist/types/index.js.map +1 -0
- package/dist/utils/ansi.d.ts +2 -0
- package/dist/utils/ansi.d.ts.map +1 -0
- package/dist/utils/ansi.js +7 -0
- package/dist/utils/ansi.js.map +1 -0
- package/dist/utils/logger.d.ts +3 -0
- package/dist/utils/logger.d.ts.map +1 -0
- package/dist/utils/logger.js +8 -0
- package/dist/utils/logger.js.map +1 -0
- package/dist/utils/security.d.ts +7 -0
- package/dist/utils/security.d.ts.map +1 -0
- package/dist/utils/security.js +58 -0
- package/dist/utils/security.js.map +1 -0
- package/dist/utils/ssh.d.ts +4 -0
- package/dist/utils/ssh.d.ts.map +1 -0
- package/dist/utils/ssh.js +29 -0
- package/dist/utils/ssh.js.map +1 -0
- package/dist/utils/sshConfig.d.ts +4 -0
- package/dist/utils/sshConfig.d.ts.map +1 -0
- package/dist/utils/sshConfig.js +85 -0
- package/dist/utils/sshConfig.js.map +1 -0
- package/dist/utils/validation.d.ts +4 -0
- package/dist/utils/validation.d.ts.map +1 -0
- package/dist/utils/validation.js +12 -0
- package/dist/utils/validation.js.map +1 -0
- package/docs/SECURITY.md +213 -0
- package/docs/TOOLS.md +425 -0
- package/keygen.bat +325 -0
- package/package.json +48 -0
- package/test_check.bat +9 -0
- package/test_delayed.bat +12 -0
- package/vitest.config.ts +14 -0
|
@@ -0,0 +1,171 @@
|
|
|
1
|
+
import { describe, it, expect, beforeEach, afterEach } from 'vitest';
|
|
2
|
+
import fs from 'fs';
|
|
3
|
+
import path from 'path';
|
|
4
|
+
import os from 'os';
|
|
5
|
+
import { StorageManager } from '../core/StorageManager.js';
|
|
6
|
+
const TEST_DIR = path.join(os.tmpdir(), 'dynamic-ssh-mcp-test-' + Date.now());
|
|
7
|
+
describe('StorageManager', () => {
|
|
8
|
+
let storage;
|
|
9
|
+
beforeEach(() => {
|
|
10
|
+
storage = new StorageManager(TEST_DIR);
|
|
11
|
+
});
|
|
12
|
+
afterEach(() => {
|
|
13
|
+
storage.shutdown();
|
|
14
|
+
if (fs.existsSync(TEST_DIR)) {
|
|
15
|
+
fs.rmSync(TEST_DIR, { recursive: true, force: true });
|
|
16
|
+
}
|
|
17
|
+
});
|
|
18
|
+
it('should create storage directory on first save', () => {
|
|
19
|
+
expect(fs.existsSync(TEST_DIR)).toBe(false);
|
|
20
|
+
storage.save({
|
|
21
|
+
id: 'sess_test123',
|
|
22
|
+
host: 'prod',
|
|
23
|
+
tmuxSession: 'mcp_prod_abc',
|
|
24
|
+
shell: 'bash',
|
|
25
|
+
connectedAt: Date.now(),
|
|
26
|
+
lastActivity: Date.now(),
|
|
27
|
+
});
|
|
28
|
+
storage.shutdown();
|
|
29
|
+
expect(fs.existsSync(TEST_DIR)).toBe(true);
|
|
30
|
+
expect(fs.existsSync(path.join(TEST_DIR, 'sessions.json'))).toBe(true);
|
|
31
|
+
});
|
|
32
|
+
it('should save and retrieve sessions', () => {
|
|
33
|
+
const session = {
|
|
34
|
+
id: 'sess_test456',
|
|
35
|
+
host: 'staging',
|
|
36
|
+
tmuxSession: 'mcp_staging_xyz',
|
|
37
|
+
shell: 'zsh',
|
|
38
|
+
connectedAt: 1000,
|
|
39
|
+
lastActivity: 2000,
|
|
40
|
+
};
|
|
41
|
+
storage.save(session);
|
|
42
|
+
storage.shutdown();
|
|
43
|
+
// Create a new StorageManager to test persistence
|
|
44
|
+
const storage2 = new StorageManager(TEST_DIR);
|
|
45
|
+
const retrieved = storage2.get('sess_test456');
|
|
46
|
+
expect(retrieved).toBeDefined();
|
|
47
|
+
expect(retrieved.host).toBe('staging');
|
|
48
|
+
expect(retrieved.tmuxSession).toBe('mcp_staging_xyz');
|
|
49
|
+
expect(retrieved.shell).toBe('zsh');
|
|
50
|
+
expect(retrieved.connectedAt).toBe(1000);
|
|
51
|
+
storage2.shutdown();
|
|
52
|
+
});
|
|
53
|
+
it('should list all sessions', () => {
|
|
54
|
+
storage.save({
|
|
55
|
+
id: 'sess_1',
|
|
56
|
+
host: 'prod',
|
|
57
|
+
tmuxSession: 'mcp_prod_1',
|
|
58
|
+
connectedAt: 1,
|
|
59
|
+
lastActivity: 1,
|
|
60
|
+
});
|
|
61
|
+
storage.save({
|
|
62
|
+
id: 'sess_2',
|
|
63
|
+
host: 'gpu',
|
|
64
|
+
tmuxSession: 'mcp_gpu_2',
|
|
65
|
+
connectedAt: 2,
|
|
66
|
+
lastActivity: 2,
|
|
67
|
+
});
|
|
68
|
+
const list = storage.list();
|
|
69
|
+
expect(list).toHaveLength(2);
|
|
70
|
+
});
|
|
71
|
+
it('should list sessions by host', () => {
|
|
72
|
+
storage.save({
|
|
73
|
+
id: 'sess_a',
|
|
74
|
+
host: 'prod',
|
|
75
|
+
tmuxSession: 'mcp_prod_a',
|
|
76
|
+
connectedAt: 1,
|
|
77
|
+
lastActivity: 1,
|
|
78
|
+
});
|
|
79
|
+
storage.save({
|
|
80
|
+
id: 'sess_b',
|
|
81
|
+
host: 'staging',
|
|
82
|
+
tmuxSession: 'mcp_staging_b',
|
|
83
|
+
connectedAt: 2,
|
|
84
|
+
lastActivity: 2,
|
|
85
|
+
});
|
|
86
|
+
storage.save({
|
|
87
|
+
id: 'sess_c',
|
|
88
|
+
host: 'prod',
|
|
89
|
+
tmuxSession: 'mcp_prod_c',
|
|
90
|
+
connectedAt: 3,
|
|
91
|
+
lastActivity: 3,
|
|
92
|
+
});
|
|
93
|
+
const prodSessions = storage.listByHost('prod');
|
|
94
|
+
expect(prodSessions).toHaveLength(2);
|
|
95
|
+
expect(prodSessions.map((s) => s.id)).toContain('sess_a');
|
|
96
|
+
expect(prodSessions.map((s) => s.id)).toContain('sess_c');
|
|
97
|
+
});
|
|
98
|
+
it('should list tmux session names', () => {
|
|
99
|
+
storage.save({
|
|
100
|
+
id: 'sess_x',
|
|
101
|
+
host: 'prod',
|
|
102
|
+
tmuxSession: 'mcp_prod_x',
|
|
103
|
+
connectedAt: 1,
|
|
104
|
+
lastActivity: 1,
|
|
105
|
+
});
|
|
106
|
+
storage.save({
|
|
107
|
+
id: 'sess_y',
|
|
108
|
+
host: 'gpu',
|
|
109
|
+
tmuxSession: 'mcp_gpu_y',
|
|
110
|
+
connectedAt: 2,
|
|
111
|
+
lastActivity: 2,
|
|
112
|
+
});
|
|
113
|
+
const names = storage.listTmuxSessions();
|
|
114
|
+
expect(names).toContain('mcp_prod_x');
|
|
115
|
+
expect(names).toContain('mcp_gpu_y');
|
|
116
|
+
});
|
|
117
|
+
it('should remove sessions', () => {
|
|
118
|
+
storage.save({
|
|
119
|
+
id: 'sess_rm',
|
|
120
|
+
host: 'prod',
|
|
121
|
+
tmuxSession: 'mcp_prod_rm',
|
|
122
|
+
connectedAt: 1,
|
|
123
|
+
lastActivity: 1,
|
|
124
|
+
});
|
|
125
|
+
expect(storage.get('sess_rm')).toBeDefined();
|
|
126
|
+
storage.remove('sess_rm');
|
|
127
|
+
expect(storage.get('sess_rm')).toBeUndefined();
|
|
128
|
+
storage.shutdown();
|
|
129
|
+
// Verify removal persisted
|
|
130
|
+
const storage2 = new StorageManager(TEST_DIR);
|
|
131
|
+
expect(storage2.get('sess_rm')).toBeUndefined();
|
|
132
|
+
storage2.shutdown();
|
|
133
|
+
});
|
|
134
|
+
it('should update existing session on save', () => {
|
|
135
|
+
storage.save({
|
|
136
|
+
id: 'sess_upd',
|
|
137
|
+
host: 'prod',
|
|
138
|
+
tmuxSession: 'mcp_prod_upd',
|
|
139
|
+
connectedAt: 1,
|
|
140
|
+
lastActivity: 1,
|
|
141
|
+
});
|
|
142
|
+
storage.save({
|
|
143
|
+
id: 'sess_upd',
|
|
144
|
+
host: 'prod',
|
|
145
|
+
tmuxSession: 'mcp_prod_upd',
|
|
146
|
+
shell: 'zsh',
|
|
147
|
+
connectedAt: 1,
|
|
148
|
+
lastActivity: 9999,
|
|
149
|
+
});
|
|
150
|
+
const updated = storage.get('sess_upd');
|
|
151
|
+
expect(updated.shell).toBe('zsh');
|
|
152
|
+
expect(updated.lastActivity).toBe(9999);
|
|
153
|
+
expect(storage.list()).toHaveLength(1);
|
|
154
|
+
});
|
|
155
|
+
it('should handle empty storage gracefully', () => {
|
|
156
|
+
const list = storage.list();
|
|
157
|
+
expect(list).toEqual([]);
|
|
158
|
+
expect(storage.get('nonexistent')).toBeUndefined();
|
|
159
|
+
expect(storage.listByHost('nonexistent')).toEqual([]);
|
|
160
|
+
expect(storage.listTmuxSessions()).toEqual([]);
|
|
161
|
+
});
|
|
162
|
+
it('should handle corrupted storage file gracefully', () => {
|
|
163
|
+
const dir = path.join(TEST_DIR);
|
|
164
|
+
fs.mkdirSync(dir, { recursive: true });
|
|
165
|
+
fs.writeFileSync(path.join(dir, 'sessions.json'), 'invalid json{{{');
|
|
166
|
+
const storage2 = new StorageManager(TEST_DIR);
|
|
167
|
+
expect(storage2.list()).toEqual([]);
|
|
168
|
+
storage2.shutdown();
|
|
169
|
+
});
|
|
170
|
+
});
|
|
171
|
+
//# sourceMappingURL=StorageManager.test.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"StorageManager.test.js","sourceRoot":"","sources":["../../src/__tests__/StorageManager.test.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,QAAQ,EAAE,EAAE,EAAE,MAAM,EAAE,UAAU,EAAE,SAAS,EAAE,MAAM,QAAQ,CAAC;AACrE,OAAO,EAAE,MAAM,IAAI,CAAC;AACpB,OAAO,IAAI,MAAM,MAAM,CAAC;AACxB,OAAO,EAAE,MAAM,IAAI,CAAC;AACpB,OAAO,EAAE,cAAc,EAAE,MAAM,2BAA2B,CAAC;AAE3D,MAAM,QAAQ,GAAG,IAAI,CAAC,IAAI,CAAC,EAAE,CAAC,MAAM,EAAE,EAAE,uBAAuB,GAAG,IAAI,CAAC,GAAG,EAAE,CAAC,CAAC;AAE9E,QAAQ,CAAC,gBAAgB,EAAE,GAAG,EAAE;IAC9B,IAAI,OAAuB,CAAC;IAE5B,UAAU,CAAC,GAAG,EAAE;QACd,OAAO,GAAG,IAAI,cAAc,CAAC,QAAQ,CAAC,CAAC;IACzC,CAAC,CAAC,CAAC;IAEH,SAAS,CAAC,GAAG,EAAE;QACb,OAAO,CAAC,QAAQ,EAAE,CAAC;QACnB,IAAI,EAAE,CAAC,UAAU,CAAC,QAAQ,CAAC,EAAE,CAAC;YAC5B,EAAE,CAAC,MAAM,CAAC,QAAQ,EAAE,EAAE,SAAS,EAAE,IAAI,EAAE,KAAK,EAAE,IAAI,EAAE,CAAC,CAAC;QACxD,CAAC;IACH,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,+CAA+C,EAAE,GAAG,EAAE;QACvD,MAAM,CAAC,EAAE,CAAC,UAAU,CAAC,QAAQ,CAAC,CAAC,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;QAC5C,OAAO,CAAC,IAAI,CAAC;YACX,EAAE,EAAE,cAAc;YAClB,IAAI,EAAE,MAAM;YACZ,WAAW,EAAE,cAAc;YAC3B,KAAK,EAAE,MAAM;YACb,WAAW,EAAE,IAAI,CAAC,GAAG,EAAE;YACvB,YAAY,EAAE,IAAI,CAAC,GAAG,EAAE;SACzB,CAAC,CAAC;QACH,OAAO,CAAC,QAAQ,EAAE,CAAC;QACnB,MAAM,CAAC,EAAE,CAAC,UAAU,CAAC,QAAQ,CAAC,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;QAC3C,MAAM,CAAC,EAAE,CAAC,UAAU,CAAC,IAAI,CAAC,IAAI,CAAC,QAAQ,EAAE,eAAe,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;IACzE,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,mCAAmC,EAAE,GAAG,EAAE;QAC3C,MAAM,OAAO,GAAG;YACd,EAAE,EAAE,cAAc;YAClB,IAAI,EAAE,SAAS;YACf,WAAW,EAAE,iBAAiB;YAC9B,KAAK,EAAE,KAAK;YACZ,WAAW,EAAE,IAAI;YACjB,YAAY,EAAE,IAAI;SACnB,CAAC;QAEF,OAAO,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;QACtB,OAAO,CAAC,QAAQ,EAAE,CAAC;QAEnB,kDAAkD;QAClD,MAAM,QAAQ,GAAG,IAAI,cAAc,CAAC,QAAQ,CAAC,CAAC;QAC9C,MAAM,SAAS,GAAG,QAAQ,CAAC,GAAG,CAAC,cAAc,CAAC,CAAC;QAC/C,MAAM,CAAC,SAAS,CAAC,CAAC,WAAW,EAAE,CAAC;QAChC,MAAM,CAAC,SAAU,CAAC,IAAI,CAAC,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC;QACxC,MAAM,CAAC,SAAU,CAAC,WAAW,CAAC,CAAC,IAAI,CAAC,iBAAiB,CAAC,CAAC;QACvD,MAAM,CAAC,SAAU,CAAC,KAAK,CAAC,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;QACrC,MAAM,CAAC,SAAU,CAAC,WAAW,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;QAC1C,QAAQ,CAAC,QAAQ,EAAE,CAAC;IACtB,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,0BAA0B,EAAE,GAAG,EAAE;QAClC,OAAO,CAAC,IAAI,CAAC;YACX,EAAE,EAAE,QAAQ;YACZ,IAAI,EAAE,MAAM;YACZ,WAAW,EAAE,YAAY;YACzB,WAAW,EAAE,CAAC;YACd,YAAY,EAAE,CAAC;SAChB,CAAC,CAAC;QACH,OAAO,CAAC,IAAI,CAAC;YACX,EAAE,EAAE,QAAQ;YACZ,IAAI,EAAE,KAAK;YACX,WAAW,EAAE,WAAW;YACxB,WAAW,EAAE,CAAC;YACd,YAAY,EAAE,CAAC;SAChB,CAAC,CAAC;QAEH,MAAM,IAAI,GAAG,OAAO,CAAC,IAAI,EAAE,CAAC;QAC5B,MAAM,CAAC,IAAI,CAAC,CAAC,YAAY,CAAC,CAAC,CAAC,CAAC;IAC/B,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,8BAA8B,EAAE,GAAG,EAAE;QACtC,OAAO,CAAC,IAAI,CAAC;YACX,EAAE,EAAE,QAAQ;YACZ,IAAI,EAAE,MAAM;YACZ,WAAW,EAAE,YAAY;YACzB,WAAW,EAAE,CAAC;YACd,YAAY,EAAE,CAAC;SAChB,CAAC,CAAC;QACH,OAAO,CAAC,IAAI,CAAC;YACX,EAAE,EAAE,QAAQ;YACZ,IAAI,EAAE,SAAS;YACf,WAAW,EAAE,eAAe;YAC5B,WAAW,EAAE,CAAC;YACd,YAAY,EAAE,CAAC;SAChB,CAAC,CAAC;QACH,OAAO,CAAC,IAAI,CAAC;YACX,EAAE,EAAE,QAAQ;YACZ,IAAI,EAAE,MAAM;YACZ,WAAW,EAAE,YAAY;YACzB,WAAW,EAAE,CAAC;YACd,YAAY,EAAE,CAAC;SAChB,CAAC,CAAC;QAEH,MAAM,YAAY,GAAG,OAAO,CAAC,UAAU,CAAC,MAAM,CAAC,CAAC;QAChD,MAAM,CAAC,YAAY,CAAC,CAAC,YAAY,CAAC,CAAC,CAAC,CAAC;QACrC,MAAM,CAAC,YAAY,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,SAAS,CAAC,QAAQ,CAAC,CAAC;QAC1D,MAAM,CAAC,YAAY,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,SAAS,CAAC,QAAQ,CAAC,CAAC;IAC5D,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,gCAAgC,EAAE,GAAG,EAAE;QACxC,OAAO,CAAC,IAAI,CAAC;YACX,EAAE,EAAE,QAAQ;YACZ,IAAI,EAAE,MAAM;YACZ,WAAW,EAAE,YAAY;YACzB,WAAW,EAAE,CAAC;YACd,YAAY,EAAE,CAAC;SAChB,CAAC,CAAC;QACH,OAAO,CAAC,IAAI,CAAC;YACX,EAAE,EAAE,QAAQ;YACZ,IAAI,EAAE,KAAK;YACX,WAAW,EAAE,WAAW;YACxB,WAAW,EAAE,CAAC;YACd,YAAY,EAAE,CAAC;SAChB,CAAC,CAAC;QAEH,MAAM,KAAK,GAAG,OAAO,CAAC,gBAAgB,EAAE,CAAC;QACzC,MAAM,CAAC,KAAK,CAAC,CAAC,SAAS,CAAC,YAAY,CAAC,CAAC;QACtC,MAAM,CAAC,KAAK,CAAC,CAAC,SAAS,CAAC,WAAW,CAAC,CAAC;IACvC,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,wBAAwB,EAAE,GAAG,EAAE;QAChC,OAAO,CAAC,IAAI,CAAC;YACX,EAAE,EAAE,SAAS;YACb,IAAI,EAAE,MAAM;YACZ,WAAW,EAAE,aAAa;YAC1B,WAAW,EAAE,CAAC;YACd,YAAY,EAAE,CAAC;SAChB,CAAC,CAAC;QACH,MAAM,CAAC,OAAO,CAAC,GAAG,CAAC,SAAS,CAAC,CAAC,CAAC,WAAW,EAAE,CAAC;QAE7C,OAAO,CAAC,MAAM,CAAC,SAAS,CAAC,CAAC;QAC1B,MAAM,CAAC,OAAO,CAAC,GAAG,CAAC,SAAS,CAAC,CAAC,CAAC,aAAa,EAAE,CAAC;QAE/C,OAAO,CAAC,QAAQ,EAAE,CAAC;QAEnB,2BAA2B;QAC3B,MAAM,QAAQ,GAAG,IAAI,cAAc,CAAC,QAAQ,CAAC,CAAC;QAC9C,MAAM,CAAC,QAAQ,CAAC,GAAG,CAAC,SAAS,CAAC,CAAC,CAAC,aAAa,EAAE,CAAC;QAChD,QAAQ,CAAC,QAAQ,EAAE,CAAC;IACtB,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,wCAAwC,EAAE,GAAG,EAAE;QAChD,OAAO,CAAC,IAAI,CAAC;YACX,EAAE,EAAE,UAAU;YACd,IAAI,EAAE,MAAM;YACZ,WAAW,EAAE,cAAc;YAC3B,WAAW,EAAE,CAAC;YACd,YAAY,EAAE,CAAC;SAChB,CAAC,CAAC;QAEH,OAAO,CAAC,IAAI,CAAC;YACX,EAAE,EAAE,UAAU;YACd,IAAI,EAAE,MAAM;YACZ,WAAW,EAAE,cAAc;YAC3B,KAAK,EAAE,KAAK;YACZ,WAAW,EAAE,CAAC;YACd,YAAY,EAAE,IAAI;SACnB,CAAC,CAAC;QAEH,MAAM,OAAO,GAAG,OAAO,CAAC,GAAG,CAAC,UAAU,CAAC,CAAC;QACxC,MAAM,CAAC,OAAQ,CAAC,KAAK,CAAC,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;QACnC,MAAM,CAAC,OAAQ,CAAC,YAAY,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;QACzC,MAAM,CAAC,OAAO,CAAC,IAAI,EAAE,CAAC,CAAC,YAAY,CAAC,CAAC,CAAC,CAAC;IACzC,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,wCAAwC,EAAE,GAAG,EAAE;QAChD,MAAM,IAAI,GAAG,OAAO,CAAC,IAAI,EAAE,CAAC;QAC5B,MAAM,CAAC,IAAI,CAAC,CAAC,OAAO,CAAC,EAAE,CAAC,CAAC;QACzB,MAAM,CAAC,OAAO,CAAC,GAAG,CAAC,aAAa,CAAC,CAAC,CAAC,aAAa,EAAE,CAAC;QACnD,MAAM,CAAC,OAAO,CAAC,UAAU,CAAC,aAAa,CAAC,CAAC,CAAC,OAAO,CAAC,EAAE,CAAC,CAAC;QACtD,MAAM,CAAC,OAAO,CAAC,gBAAgB,EAAE,CAAC,CAAC,OAAO,CAAC,EAAE,CAAC,CAAC;IACjD,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,iDAAiD,EAAE,GAAG,EAAE;QACzD,MAAM,GAAG,GAAG,IAAI,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;QAChC,EAAE,CAAC,SAAS,CAAC,GAAG,EAAE,EAAE,SAAS,EAAE,IAAI,EAAE,CAAC,CAAC;QACvC,EAAE,CAAC,aAAa,CAAC,IAAI,CAAC,IAAI,CAAC,GAAG,EAAE,eAAe,CAAC,EAAE,iBAAiB,CAAC,CAAC;QAErE,MAAM,QAAQ,GAAG,IAAI,cAAc,CAAC,QAAQ,CAAC,CAAC;QAC9C,MAAM,CAAC,QAAQ,CAAC,IAAI,EAAE,CAAC,CAAC,OAAO,CAAC,EAAE,CAAC,CAAC;QACpC,QAAQ,CAAC,QAAQ,EAAE,CAAC;IACtB,CAAC,CAAC,CAAC;AACL,CAAC,CAAC,CAAC"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"ansi.test.d.ts","sourceRoot":"","sources":["../../src/__tests__/ansi.test.ts"],"names":[],"mappings":""}
|
|
@@ -0,0 +1,41 @@
|
|
|
1
|
+
import { describe, it, expect } from 'vitest';
|
|
2
|
+
import { stripAnsi } from '../utils/ansi.js';
|
|
3
|
+
describe('ansi', () => {
|
|
4
|
+
describe('stripAnsi', () => {
|
|
5
|
+
it('should return plain text unchanged', () => {
|
|
6
|
+
expect(stripAnsi('hello world')).toBe('hello world');
|
|
7
|
+
expect(stripAnsi('no colors here')).toBe('no colors here');
|
|
8
|
+
});
|
|
9
|
+
it('should strip color codes', () => {
|
|
10
|
+
expect(stripAnsi('\u001B[31mred\u001B[0m')).toBe('red');
|
|
11
|
+
expect(stripAnsi('\u001B[32mgreen\u001B[0m')).toBe('green');
|
|
12
|
+
expect(stripAnsi('\u001B[1;31mbold red\u001B[0m')).toBe('bold red');
|
|
13
|
+
});
|
|
14
|
+
it('should strip cursor control sequences', () => {
|
|
15
|
+
expect(stripAnsi('\u001B[1Aup\u001B[1Bdown')).toBe('updown');
|
|
16
|
+
expect(stripAnsi('\u001B[2Jcleared')).toBe('cleared');
|
|
17
|
+
});
|
|
18
|
+
it('should strip complex multi-code sequences', () => {
|
|
19
|
+
const input = '\u001B[1m\u001B[33m\u001B[44mstyled\u001B[0m text';
|
|
20
|
+
expect(stripAnsi(input)).toBe('styled text');
|
|
21
|
+
});
|
|
22
|
+
it('should handle empty string', () => {
|
|
23
|
+
expect(stripAnsi('')).toBe('');
|
|
24
|
+
});
|
|
25
|
+
it('should handle string with only ANSI codes', () => {
|
|
26
|
+
expect(stripAnsi('\u001B[31m\u001B[0m')).toBe('');
|
|
27
|
+
});
|
|
28
|
+
it('should strip more escape patterns with updated regex', () => {
|
|
29
|
+
// New comprehensive regex strips ESC followed by valid CSI terminator chars
|
|
30
|
+
// 't' falls in the R-T range used as CSI terminators
|
|
31
|
+
const input = 'text \u001B[31mred\u001B[0m normal';
|
|
32
|
+
expect(stripAnsi(input)).toBe('text red normal');
|
|
33
|
+
});
|
|
34
|
+
it('should strip prompt-style ANSI sequences', () => {
|
|
35
|
+
// Typical colored prompt
|
|
36
|
+
const input = '\u001B[01;32muser@host\u001B[00m:\u001B[01;34m~$\u001B[00m ';
|
|
37
|
+
expect(stripAnsi(input)).toBe('user@host:~$ ');
|
|
38
|
+
});
|
|
39
|
+
});
|
|
40
|
+
});
|
|
41
|
+
//# sourceMappingURL=ansi.test.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"ansi.test.js","sourceRoot":"","sources":["../../src/__tests__/ansi.test.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,QAAQ,EAAE,EAAE,EAAE,MAAM,EAAE,MAAM,QAAQ,CAAC;AAC9C,OAAO,EAAE,SAAS,EAAE,MAAM,kBAAkB,CAAC;AAE7C,QAAQ,CAAC,MAAM,EAAE,GAAG,EAAE;IACpB,QAAQ,CAAC,WAAW,EAAE,GAAG,EAAE;QACzB,EAAE,CAAC,oCAAoC,EAAE,GAAG,EAAE;YAC5C,MAAM,CAAC,SAAS,CAAC,aAAa,CAAC,CAAC,CAAC,IAAI,CAAC,aAAa,CAAC,CAAC;YACrD,MAAM,CAAC,SAAS,CAAC,gBAAgB,CAAC,CAAC,CAAC,IAAI,CAAC,gBAAgB,CAAC,CAAC;QAC7D,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,0BAA0B,EAAE,GAAG,EAAE;YAClC,MAAM,CAAC,SAAS,CAAC,wBAAwB,CAAC,CAAC,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;YACxD,MAAM,CAAC,SAAS,CAAC,0BAA0B,CAAC,CAAC,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;YAC5D,MAAM,CAAC,SAAS,CAAC,+BAA+B,CAAC,CAAC,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC;QACtE,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,uCAAuC,EAAE,GAAG,EAAE;YAC/C,MAAM,CAAC,SAAS,CAAC,0BAA0B,CAAC,CAAC,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;YAC7D,MAAM,CAAC,SAAS,CAAC,kBAAkB,CAAC,CAAC,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC;QACxD,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,2CAA2C,EAAE,GAAG,EAAE;YACnD,MAAM,KAAK,GAAG,mDAAmD,CAAC;YAClE,MAAM,CAAC,SAAS,CAAC,KAAK,CAAC,CAAC,CAAC,IAAI,CAAC,aAAa,CAAC,CAAC;QAC/C,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,4BAA4B,EAAE,GAAG,EAAE;YACpC,MAAM,CAAC,SAAS,CAAC,EAAE,CAAC,CAAC,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;QACjC,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,2CAA2C,EAAE,GAAG,EAAE;YACnD,MAAM,CAAC,SAAS,CAAC,qBAAqB,CAAC,CAAC,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;QACpD,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,sDAAsD,EAAE,GAAG,EAAE;YAC9D,4EAA4E;YAC5E,qDAAqD;YACrD,MAAM,KAAK,GAAG,oCAAoC,CAAC;YACnD,MAAM,CAAC,SAAS,CAAC,KAAK,CAAC,CAAC,CAAC,IAAI,CAAC,iBAAiB,CAAC,CAAC;QACnD,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,0CAA0C,EAAE,GAAG,EAAE;YAClD,yBAAyB;YACzB,MAAM,KAAK,GAAG,6DAA6D,CAAC;YAC5E,MAAM,CAAC,SAAS,CAAC,KAAK,CAAC,CAAC,CAAC,IAAI,CAAC,eAAe,CAAC,CAAC;QACjD,CAAC,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;AACL,CAAC,CAAC,CAAC"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"security.test.d.ts","sourceRoot":"","sources":["../../src/__tests__/security.test.ts"],"names":[],"mappings":""}
|
|
@@ -0,0 +1,87 @@
|
|
|
1
|
+
import { describe, it, expect, beforeEach, afterEach } from 'vitest';
|
|
2
|
+
import { isReadOnlyMode, isHostAllowed, isCommandDenied } from '../utils/security.js';
|
|
3
|
+
describe('security', () => {
|
|
4
|
+
const OLD_ENV = { ...process.env };
|
|
5
|
+
beforeEach(() => {
|
|
6
|
+
process.env = { ...OLD_ENV };
|
|
7
|
+
});
|
|
8
|
+
afterEach(() => {
|
|
9
|
+
process.env = OLD_ENV;
|
|
10
|
+
});
|
|
11
|
+
describe('isReadOnlyMode', () => {
|
|
12
|
+
it('should return false by default', () => {
|
|
13
|
+
delete process.env.MCP_SSH_READONLY;
|
|
14
|
+
expect(isReadOnlyMode()).toBe(false);
|
|
15
|
+
});
|
|
16
|
+
it('should return true when MCP_SSH_READONLY=true', () => {
|
|
17
|
+
process.env.MCP_SSH_READONLY = 'true';
|
|
18
|
+
expect(isReadOnlyMode()).toBe(true);
|
|
19
|
+
});
|
|
20
|
+
it('should return false for any other value', () => {
|
|
21
|
+
process.env.MCP_SSH_READONLY = 'false';
|
|
22
|
+
expect(isReadOnlyMode()).toBe(false);
|
|
23
|
+
process.env.MCP_SSH_READONLY = '1';
|
|
24
|
+
expect(isReadOnlyMode()).toBe(false);
|
|
25
|
+
});
|
|
26
|
+
});
|
|
27
|
+
describe('isHostAllowed', () => {
|
|
28
|
+
it('should allow all hosts when no filter set', () => {
|
|
29
|
+
delete process.env.MCP_SSH_ALLOWED_HOSTS;
|
|
30
|
+
expect(isHostAllowed('prod')).toBe(true);
|
|
31
|
+
expect(isHostAllowed('any-host')).toBe(true);
|
|
32
|
+
});
|
|
33
|
+
it('should allow only listed hosts', () => {
|
|
34
|
+
process.env.MCP_SSH_ALLOWED_HOSTS = 'prod,staging';
|
|
35
|
+
expect(isHostAllowed('prod')).toBe(true);
|
|
36
|
+
expect(isHostAllowed('staging')).toBe(true);
|
|
37
|
+
expect(isHostAllowed('gpu')).toBe(false);
|
|
38
|
+
});
|
|
39
|
+
it('should trim whitespace from host entries', () => {
|
|
40
|
+
process.env.MCP_SSH_ALLOWED_HOSTS = ' prod , staging ';
|
|
41
|
+
expect(isHostAllowed('prod')).toBe(true);
|
|
42
|
+
expect(isHostAllowed('staging')).toBe(true);
|
|
43
|
+
});
|
|
44
|
+
it('should handle empty list as allow all', () => {
|
|
45
|
+
process.env.MCP_SSH_ALLOWED_HOSTS = '';
|
|
46
|
+
expect(isHostAllowed('prod')).toBe(true);
|
|
47
|
+
});
|
|
48
|
+
it('should handle comma-only list', () => {
|
|
49
|
+
process.env.MCP_SSH_ALLOWED_HOSTS = ',';
|
|
50
|
+
expect(isHostAllowed('prod')).toBe(true);
|
|
51
|
+
});
|
|
52
|
+
});
|
|
53
|
+
describe('isCommandDenied', () => {
|
|
54
|
+
it('should allow all commands when no denylist', () => {
|
|
55
|
+
delete process.env.MCP_SSH_DENYLIST_COMMANDS;
|
|
56
|
+
expect(isCommandDenied('rm -rf /')).toBe(false);
|
|
57
|
+
expect(isCommandDenied('ls -la')).toBe(false);
|
|
58
|
+
});
|
|
59
|
+
it('should block denied command patterns', () => {
|
|
60
|
+
process.env.MCP_SSH_DENYLIST_COMMANDS = 'rm -rf,shutdown,halt';
|
|
61
|
+
expect(isCommandDenied('rm -rf /')).toBe(true);
|
|
62
|
+
expect(isCommandDenied('shutdown now')).toBe(true);
|
|
63
|
+
expect(isCommandDenied('sudo halt')).toBe(true);
|
|
64
|
+
});
|
|
65
|
+
it('should be case-insensitive', () => {
|
|
66
|
+
process.env.MCP_SSH_DENYLIST_COMMANDS = 'RM -RF,ShutDown';
|
|
67
|
+
expect(isCommandDenied('rm -rf /')).toBe(true);
|
|
68
|
+
expect(isCommandDenied('SHUTDOWN now')).toBe(true);
|
|
69
|
+
});
|
|
70
|
+
it('should allow safe commands', () => {
|
|
71
|
+
process.env.MCP_SSH_DENYLIST_COMMANDS = 'rm -rf,shutdown';
|
|
72
|
+
expect(isCommandDenied('ls -la')).toBe(false);
|
|
73
|
+
expect(isCommandDenied('echo hello')).toBe(false);
|
|
74
|
+
expect(isCommandDenied('npm run build')).toBe(false);
|
|
75
|
+
});
|
|
76
|
+
it('should handle empty denylist', () => {
|
|
77
|
+
process.env.MCP_SSH_DENYLIST_COMMANDS = '';
|
|
78
|
+
expect(isCommandDenied('rm -rf /')).toBe(false);
|
|
79
|
+
});
|
|
80
|
+
it('should trim whitespace from entries', () => {
|
|
81
|
+
process.env.MCP_SSH_DENYLIST_COMMANDS = ' rm -rf , shutdown ';
|
|
82
|
+
expect(isCommandDenied('rm -rf /')).toBe(true);
|
|
83
|
+
expect(isCommandDenied('shutdown -h now')).toBe(true);
|
|
84
|
+
});
|
|
85
|
+
});
|
|
86
|
+
});
|
|
87
|
+
//# sourceMappingURL=security.test.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"security.test.js","sourceRoot":"","sources":["../../src/__tests__/security.test.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"validation.test.d.ts","sourceRoot":"","sources":["../../src/__tests__/validation.test.ts"],"names":[],"mappings":""}
|
|
@@ -0,0 +1,23 @@
|
|
|
1
|
+
import { describe, it, expect } from 'vitest';
|
|
2
|
+
import { sanitizeTmuxSessionName } from '../utils/validation.js';
|
|
3
|
+
describe('validation', () => {
|
|
4
|
+
describe('sanitizeTmuxSessionName', () => {
|
|
5
|
+
it('should preserve valid names unchanged', () => {
|
|
6
|
+
expect(sanitizeTmuxSessionName('prod')).toBe('prod');
|
|
7
|
+
expect(sanitizeTmuxSessionName('my_host')).toBe('my_host');
|
|
8
|
+
expect(sanitizeTmuxSessionName('server-01')).toBe('server-01');
|
|
9
|
+
expect(sanitizeTmuxSessionName('test.example')).toBe('test.example');
|
|
10
|
+
});
|
|
11
|
+
it('should replace invalid characters with underscore', () => {
|
|
12
|
+
expect(sanitizeTmuxSessionName('hello world')).toBe('hello_world');
|
|
13
|
+
expect(sanitizeTmuxSessionName('host@domain')).toBe('host_domain');
|
|
14
|
+
expect(sanitizeTmuxSessionName('path/name')).toBe('path_name');
|
|
15
|
+
});
|
|
16
|
+
it('should prefix non-alpha starters', () => {
|
|
17
|
+
expect(sanitizeTmuxSessionName('123abc')).toBe('mcp_123abc');
|
|
18
|
+
expect(sanitizeTmuxSessionName('-test')).toBe('mcp_-test');
|
|
19
|
+
expect(sanitizeTmuxSessionName('.hidden')).toBe('mcp_.hidden');
|
|
20
|
+
});
|
|
21
|
+
});
|
|
22
|
+
});
|
|
23
|
+
//# sourceMappingURL=validation.test.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"validation.test.js","sourceRoot":"","sources":["../../src/__tests__/validation.test.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,QAAQ,EAAE,EAAE,EAAE,MAAM,EAAE,MAAM,QAAQ,CAAC;AAC9C,OAAO,EAAE,uBAAuB,EAAE,MAAM,wBAAwB,CAAC;AAEjE,QAAQ,CAAC,YAAY,EAAE,GAAG,EAAE;IAC1B,QAAQ,CAAC,yBAAyB,EAAE,GAAG,EAAE;QACvC,EAAE,CAAC,uCAAuC,EAAE,GAAG,EAAE;YAC/C,MAAM,CAAC,uBAAuB,CAAC,MAAM,CAAC,CAAC,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC;YACrD,MAAM,CAAC,uBAAuB,CAAC,SAAS,CAAC,CAAC,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC;YAC3D,MAAM,CAAC,uBAAuB,CAAC,WAAW,CAAC,CAAC,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC;YAC/D,MAAM,CAAC,uBAAuB,CAAC,cAAc,CAAC,CAAC,CAAC,IAAI,CAAC,cAAc,CAAC,CAAC;QACvE,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,mDAAmD,EAAE,GAAG,EAAE;YAC3D,MAAM,CAAC,uBAAuB,CAAC,aAAa,CAAC,CAAC,CAAC,IAAI,CAAC,aAAa,CAAC,CAAC;YACnE,MAAM,CAAC,uBAAuB,CAAC,aAAa,CAAC,CAAC,CAAC,IAAI,CAAC,aAAa,CAAC,CAAC;YACnE,MAAM,CAAC,uBAAuB,CAAC,WAAW,CAAC,CAAC,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC;QACjE,CAAC,CAAC,CAAC;QAEH,EAAE,CAAC,kCAAkC,EAAE,GAAG,EAAE;YAC1C,MAAM,CAAC,uBAAuB,CAAC,QAAQ,CAAC,CAAC,CAAC,IAAI,CAAC,YAAY,CAAC,CAAC;YAC7D,MAAM,CAAC,uBAAuB,CAAC,OAAO,CAAC,CAAC,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC;YAC3D,MAAM,CAAC,uBAAuB,CAAC,SAAS,CAAC,CAAC,CAAC,IAAI,CAAC,aAAa,CAAC,CAAC;QACjE,CAAC,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;AACL,CAAC,CAAC,CAAC"}
|
|
@@ -0,0 +1,25 @@
|
|
|
1
|
+
export interface HostSecurityEntry {
|
|
2
|
+
readonly?: boolean;
|
|
3
|
+
allow_commands?: string[];
|
|
4
|
+
deny_commands?: string[];
|
|
5
|
+
}
|
|
6
|
+
export interface HostSecurityConfig {
|
|
7
|
+
[host: string]: HostSecurityEntry;
|
|
8
|
+
}
|
|
9
|
+
export declare class HostSecurityManager {
|
|
10
|
+
private config;
|
|
11
|
+
private configPath;
|
|
12
|
+
constructor(configDir?: string);
|
|
13
|
+
private load;
|
|
14
|
+
private save;
|
|
15
|
+
isReadOnly(host?: string): boolean;
|
|
16
|
+
setReadOnly(host: string, readonly: boolean): void;
|
|
17
|
+
getAllowCommands(host: string): string[];
|
|
18
|
+
setAllowCommands(host: string, commands: string[]): void;
|
|
19
|
+
getDenyCommands(host: string): string[];
|
|
20
|
+
setDenyCommands(host: string, commands: string[]): void;
|
|
21
|
+
getHostConfig(host: string): HostSecurityEntry | null;
|
|
22
|
+
getAll(): HostSecurityConfig;
|
|
23
|
+
removeHost(host: string): void;
|
|
24
|
+
}
|
|
25
|
+
//# sourceMappingURL=HostSecurityManager.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"HostSecurityManager.d.ts","sourceRoot":"","sources":["../../src/core/HostSecurityManager.ts"],"names":[],"mappings":"AAKA,MAAM,WAAW,iBAAiB;IAChC,QAAQ,CAAC,EAAE,OAAO,CAAC;IACnB,cAAc,CAAC,EAAE,MAAM,EAAE,CAAC;IAC1B,aAAa,CAAC,EAAE,MAAM,EAAE,CAAC;CAC1B;AAED,MAAM,WAAW,kBAAkB;IACjC,CAAC,IAAI,EAAE,MAAM,GAAG,iBAAiB,CAAC;CACnC;AAKD,qBAAa,mBAAmB;IAC9B,OAAO,CAAC,MAAM,CAAqB;IACnC,OAAO,CAAC,UAAU,CAAS;gBAEf,SAAS,CAAC,EAAE,MAAM;IAK9B,OAAO,CAAC,IAAI;IAWZ,OAAO,CAAC,IAAI;IAUZ,UAAU,CAAC,IAAI,CAAC,EAAE,MAAM,GAAG,OAAO;IAKlC,WAAW,CAAC,IAAI,EAAE,MAAM,EAAE,QAAQ,EAAE,OAAO,GAAG,IAAI;IAMlD,gBAAgB,CAAC,IAAI,EAAE,MAAM,GAAG,MAAM,EAAE;IAIxC,gBAAgB,CAAC,IAAI,EAAE,MAAM,EAAE,QAAQ,EAAE,MAAM,EAAE,GAAG,IAAI;IAMxD,eAAe,CAAC,IAAI,EAAE,MAAM,GAAG,MAAM,EAAE;IAIvC,eAAe,CAAC,IAAI,EAAE,MAAM,EAAE,QAAQ,EAAE,MAAM,EAAE,GAAG,IAAI;IAMvD,aAAa,CAAC,IAAI,EAAE,MAAM,GAAG,iBAAiB,GAAG,IAAI;IAIrD,MAAM,IAAI,kBAAkB;IAI5B,UAAU,CAAC,IAAI,EAAE,MAAM,GAAG,IAAI;CAI/B"}
|
|
@@ -0,0 +1,76 @@
|
|
|
1
|
+
import { logger } from '../utils/logger.js';
|
|
2
|
+
import * as fs from 'fs';
|
|
3
|
+
import * as path from 'path';
|
|
4
|
+
import * as os from 'os';
|
|
5
|
+
const DEFAULT_DIR = path.join(os.homedir(), '.dynamic-ssh-mcp');
|
|
6
|
+
const CONFIG_FILE = 'host_security.json';
|
|
7
|
+
export class HostSecurityManager {
|
|
8
|
+
config;
|
|
9
|
+
configPath;
|
|
10
|
+
constructor(configDir) {
|
|
11
|
+
this.configPath = path.join(configDir || DEFAULT_DIR, CONFIG_FILE);
|
|
12
|
+
this.config = this.load();
|
|
13
|
+
}
|
|
14
|
+
load() {
|
|
15
|
+
try {
|
|
16
|
+
if (fs.existsSync(this.configPath)) {
|
|
17
|
+
return JSON.parse(fs.readFileSync(this.configPath, 'utf-8'));
|
|
18
|
+
}
|
|
19
|
+
}
|
|
20
|
+
catch (err) {
|
|
21
|
+
logger.warn({ err }, 'Failed to load host security config');
|
|
22
|
+
}
|
|
23
|
+
return {};
|
|
24
|
+
}
|
|
25
|
+
save() {
|
|
26
|
+
try {
|
|
27
|
+
const dir = path.dirname(this.configPath);
|
|
28
|
+
if (!fs.existsSync(dir))
|
|
29
|
+
fs.mkdirSync(dir, { recursive: true });
|
|
30
|
+
fs.writeFileSync(this.configPath, JSON.stringify(this.config, null, 2));
|
|
31
|
+
}
|
|
32
|
+
catch (err) {
|
|
33
|
+
logger.error({ err }, 'Failed to save host security config');
|
|
34
|
+
}
|
|
35
|
+
}
|
|
36
|
+
isReadOnly(host) {
|
|
37
|
+
if (!host || !this.config[host])
|
|
38
|
+
return false;
|
|
39
|
+
return this.config[host].readonly === true;
|
|
40
|
+
}
|
|
41
|
+
setReadOnly(host, readonly) {
|
|
42
|
+
if (!this.config[host])
|
|
43
|
+
this.config[host] = {};
|
|
44
|
+
this.config[host].readonly = readonly;
|
|
45
|
+
this.save();
|
|
46
|
+
}
|
|
47
|
+
getAllowCommands(host) {
|
|
48
|
+
return this.config[host]?.allow_commands || [];
|
|
49
|
+
}
|
|
50
|
+
setAllowCommands(host, commands) {
|
|
51
|
+
if (!this.config[host])
|
|
52
|
+
this.config[host] = {};
|
|
53
|
+
this.config[host].allow_commands = commands;
|
|
54
|
+
this.save();
|
|
55
|
+
}
|
|
56
|
+
getDenyCommands(host) {
|
|
57
|
+
return this.config[host]?.deny_commands || [];
|
|
58
|
+
}
|
|
59
|
+
setDenyCommands(host, commands) {
|
|
60
|
+
if (!this.config[host])
|
|
61
|
+
this.config[host] = {};
|
|
62
|
+
this.config[host].deny_commands = commands;
|
|
63
|
+
this.save();
|
|
64
|
+
}
|
|
65
|
+
getHostConfig(host) {
|
|
66
|
+
return this.config[host] || null;
|
|
67
|
+
}
|
|
68
|
+
getAll() {
|
|
69
|
+
return { ...this.config };
|
|
70
|
+
}
|
|
71
|
+
removeHost(host) {
|
|
72
|
+
delete this.config[host];
|
|
73
|
+
this.save();
|
|
74
|
+
}
|
|
75
|
+
}
|
|
76
|
+
//# sourceMappingURL=HostSecurityManager.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"HostSecurityManager.js","sourceRoot":"","sources":["../../src/core/HostSecurityManager.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,48 @@
|
|
|
1
|
+
import { Client, type ConnectConfig } from 'ssh2';
|
|
2
|
+
export interface SFTPEntry {
|
|
3
|
+
filename: string;
|
|
4
|
+
longname: string;
|
|
5
|
+
attrs: {
|
|
6
|
+
mode: number;
|
|
7
|
+
uid: number;
|
|
8
|
+
gid: number;
|
|
9
|
+
size: number;
|
|
10
|
+
atime: number;
|
|
11
|
+
mtime: number;
|
|
12
|
+
isDirectory: boolean;
|
|
13
|
+
isFile: boolean;
|
|
14
|
+
isSymbolicLink: boolean;
|
|
15
|
+
};
|
|
16
|
+
}
|
|
17
|
+
export interface SFTPStat {
|
|
18
|
+
mode: number;
|
|
19
|
+
uid: number;
|
|
20
|
+
gid: number;
|
|
21
|
+
size: number;
|
|
22
|
+
atime: number;
|
|
23
|
+
mtime: number;
|
|
24
|
+
isDirectory: boolean;
|
|
25
|
+
isFile: boolean;
|
|
26
|
+
isSymbolicLink: boolean;
|
|
27
|
+
}
|
|
28
|
+
export declare class SSHManager {
|
|
29
|
+
connect(config: ConnectConfig): Promise<Client>;
|
|
30
|
+
private getSftp;
|
|
31
|
+
sftpUpload(ssh: Client, localPath: string, remotePath: string, timeoutMs?: number): Promise<void>;
|
|
32
|
+
sftpDownload(ssh: Client, remotePath: string, localPath: string, timeoutMs?: number): Promise<void>;
|
|
33
|
+
sftpList(ssh: Client, remotePath: string, timeoutMs?: number): Promise<SFTPEntry[]>;
|
|
34
|
+
sftpStat(ssh: Client, remotePath: string, timeoutMs?: number): Promise<SFTPStat>;
|
|
35
|
+
sftpExists(ssh: Client, remotePath: string, timeoutMs?: number): Promise<boolean>;
|
|
36
|
+
sftpMkdir(ssh: Client, remotePath: string, timeoutMs?: number): Promise<void>;
|
|
37
|
+
sftpUnlink(ssh: Client, remotePath: string, timeoutMs?: number): Promise<void>;
|
|
38
|
+
isAlive(ssh: Client): boolean;
|
|
39
|
+
exec(ssh: Client, command: string, timeoutMs?: number): Promise<{
|
|
40
|
+
stdout: string;
|
|
41
|
+
stderr: string;
|
|
42
|
+
code: number | null;
|
|
43
|
+
signal: string | null;
|
|
44
|
+
}>;
|
|
45
|
+
connectWithProxy(proxyConfig: ConnectConfig, targetConfig: ConnectConfig): Promise<Client>;
|
|
46
|
+
disconnect(ssh: Client): void;
|
|
47
|
+
}
|
|
48
|
+
//# sourceMappingURL=SSHManager.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"SSHManager.d.ts","sourceRoot":"","sources":["../../src/core/SSHManager.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,MAAM,EAAE,KAAK,aAAa,EAAe,MAAM,MAAM,CAAC;AAK/D,MAAM,WAAW,SAAS;IACxB,QAAQ,EAAE,MAAM,CAAC;IACjB,QAAQ,EAAE,MAAM,CAAC;IACjB,KAAK,EAAE;QACL,IAAI,EAAE,MAAM,CAAC;QACb,GAAG,EAAE,MAAM,CAAC;QACZ,GAAG,EAAE,MAAM,CAAC;QACZ,IAAI,EAAE,MAAM,CAAC;QACb,KAAK,EAAE,MAAM,CAAC;QACd,KAAK,EAAE,MAAM,CAAC;QACd,WAAW,EAAE,OAAO,CAAC;QACrB,MAAM,EAAE,OAAO,CAAC;QAChB,cAAc,EAAE,OAAO,CAAC;KACzB,CAAC;CACH;AAED,MAAM,WAAW,QAAQ;IACvB,IAAI,EAAE,MAAM,CAAC;IACb,GAAG,EAAE,MAAM,CAAC;IACZ,GAAG,EAAE,MAAM,CAAC;IACZ,IAAI,EAAE,MAAM,CAAC;IACb,KAAK,EAAE,MAAM,CAAC;IACd,KAAK,EAAE,MAAM,CAAC;IACd,WAAW,EAAE,OAAO,CAAC;IACrB,MAAM,EAAE,OAAO,CAAC;IAChB,cAAc,EAAE,OAAO,CAAC;CACzB;AAED,qBAAa,UAAU;IACf,OAAO,CAAC,MAAM,EAAE,aAAa,GAAG,OAAO,CAAC,MAAM,CAAC;IAmCrD,OAAO,CAAC,OAAO;IAcT,UAAU,CACd,GAAG,EAAE,MAAM,EACX,SAAS,EAAE,MAAM,EACjB,UAAU,EAAE,MAAM,EAClB,SAAS,SAAS,GACjB,OAAO,CAAC,IAAI,CAAC;IAqBV,YAAY,CAChB,GAAG,EAAE,MAAM,EACX,UAAU,EAAE,MAAM,EAClB,SAAS,EAAE,MAAM,EACjB,SAAS,SAAS,GACjB,OAAO,CAAC,IAAI,CAAC;IAqBV,QAAQ,CACZ,GAAG,EAAE,MAAM,EACX,UAAU,EAAE,MAAM,EAClB,SAAS,SAAQ,GAChB,OAAO,CAAC,SAAS,EAAE,CAAC;IA0CjB,QAAQ,CACZ,GAAG,EAAE,MAAM,EACX,UAAU,EAAE,MAAM,EAClB,SAAS,SAAQ,GAChB,OAAO,CAAC,QAAQ,CAAC;IA8Bd,UAAU,CACd,GAAG,EAAE,MAAM,EACX,UAAU,EAAE,MAAM,EAClB,SAAS,SAAQ,GAChB,OAAO,CAAC,OAAO,CAAC;IASb,SAAS,CACb,GAAG,EAAE,MAAM,EACX,UAAU,EAAE,MAAM,EAClB,SAAS,SAAQ,GAChB,OAAO,CAAC,IAAI,CAAC;IAqBV,UAAU,CACd,GAAG,EAAE,MAAM,EACX,UAAU,EAAE,MAAM,EAClB,SAAS,SAAQ,GAChB,OAAO,CAAC,IAAI,CAAC;IAqBhB,OAAO,CAAC,GAAG,EAAE,MAAM,GAAG,OAAO;IAQvB,IAAI,CACR,GAAG,EAAE,MAAM,EACX,OAAO,EAAE,MAAM,EACf,SAAS,SAAQ,GAChB,OAAO,CAAC;QACT,MAAM,EAAE,MAAM,CAAC;QACf,MAAM,EAAE,MAAM,CAAC;QACf,IAAI,EAAE,MAAM,GAAG,IAAI,CAAC;QACpB,MAAM,EAAE,MAAM,GAAG,IAAI,CAAC;KACvB,CAAC;IAoCI,gBAAgB,CACpB,WAAW,EAAE,aAAa,EAC1B,YAAY,EAAE,aAAa,GAC1B,OAAO,CAAC,MAAM,CAAC;IAgDlB,UAAU,CAAC,GAAG,EAAE,MAAM,GAAG,IAAI;CAU9B"}
|