@zlash65/postgres-ssh-mcp 0.0.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.
Files changed (63) hide show
  1. package/LICENSE +21 -0
  2. package/README.md +161 -0
  3. package/dist/config.d.ts +3 -0
  4. package/dist/config.d.ts.map +1 -0
  5. package/dist/config.js +122 -0
  6. package/dist/config.js.map +1 -0
  7. package/dist/connection/host-key-verifier.d.ts +13 -0
  8. package/dist/connection/host-key-verifier.d.ts.map +1 -0
  9. package/dist/connection/host-key-verifier.js +127 -0
  10. package/dist/connection/host-key-verifier.js.map +1 -0
  11. package/dist/connection/index.d.ts +7 -0
  12. package/dist/connection/index.d.ts.map +1 -0
  13. package/dist/connection/index.js +7 -0
  14. package/dist/connection/index.js.map +1 -0
  15. package/dist/connection/postgres-pool.d.ts +23 -0
  16. package/dist/connection/postgres-pool.d.ts.map +1 -0
  17. package/dist/connection/postgres-pool.js +295 -0
  18. package/dist/connection/postgres-pool.js.map +1 -0
  19. package/dist/connection/ssh-tunnel.d.ts +34 -0
  20. package/dist/connection/ssh-tunnel.d.ts.map +1 -0
  21. package/dist/connection/ssh-tunnel.js +295 -0
  22. package/dist/connection/ssh-tunnel.js.map +1 -0
  23. package/dist/index.d.ts +3 -0
  24. package/dist/index.d.ts.map +1 -0
  25. package/dist/index.js +67 -0
  26. package/dist/index.js.map +1 -0
  27. package/dist/lib/obfuscate.d.ts +2 -0
  28. package/dist/lib/obfuscate.d.ts.map +1 -0
  29. package/dist/lib/obfuscate.js +13 -0
  30. package/dist/lib/obfuscate.js.map +1 -0
  31. package/dist/lib/sql-validator.d.ts +6 -0
  32. package/dist/lib/sql-validator.d.ts.map +1 -0
  33. package/dist/lib/sql-validator.js +684 -0
  34. package/dist/lib/sql-validator.js.map +1 -0
  35. package/dist/lib/tool-response.d.ts +5 -0
  36. package/dist/lib/tool-response.d.ts.map +1 -0
  37. package/dist/lib/tool-response.js +30 -0
  38. package/dist/lib/tool-response.js.map +1 -0
  39. package/dist/server.d.ts +8 -0
  40. package/dist/server.d.ts.map +1 -0
  41. package/dist/server.js +24 -0
  42. package/dist/server.js.map +1 -0
  43. package/dist/tools/admin.d.ts +4 -0
  44. package/dist/tools/admin.d.ts.map +1 -0
  45. package/dist/tools/admin.js +184 -0
  46. package/dist/tools/admin.js.map +1 -0
  47. package/dist/tools/index.d.ts +8 -0
  48. package/dist/tools/index.d.ts.map +1 -0
  49. package/dist/tools/index.js +8 -0
  50. package/dist/tools/index.js.map +1 -0
  51. package/dist/tools/query.d.ts +4 -0
  52. package/dist/tools/query.d.ts.map +1 -0
  53. package/dist/tools/query.js +65 -0
  54. package/dist/tools/query.js.map +1 -0
  55. package/dist/tools/schema.d.ts +4 -0
  56. package/dist/tools/schema.d.ts.map +1 -0
  57. package/dist/tools/schema.js +189 -0
  58. package/dist/tools/schema.js.map +1 -0
  59. package/dist/types.d.ts +93 -0
  60. package/dist/types.d.ts.map +1 -0
  61. package/dist/types.js +2 -0
  62. package/dist/types.js.map +1 -0
  63. package/package.json +80 -0
@@ -0,0 +1,295 @@
1
+ import { Pool } from 'pg';
2
+ import * as fs from 'fs';
3
+ import { SSHTunnelManager } from './ssh-tunnel.js';
4
+ import { validateReadOnlyStatement, stripLeadingComments, cteContainsDML, extractFinalStatementAfterCTEs, getFirstKeyword, } from '../lib/sql-validator.js';
5
+ import { obfuscateConnectionString } from '../lib/obfuscate.js';
6
+ export class ConnectionManager {
7
+ pool = null;
8
+ tunnelManager = null;
9
+ config;
10
+ isInitialized = false;
11
+ isReconnecting = false;
12
+ currentLocalPort = null;
13
+ sslEnabled = false;
14
+ constructor(config) {
15
+ this.config = config;
16
+ }
17
+ async initialize() {
18
+ if (this.isInitialized) {
19
+ return;
20
+ }
21
+ if (this.config.ssh) {
22
+ const target = {
23
+ host: this.config.database.host,
24
+ port: this.config.database.port,
25
+ };
26
+ this.tunnelManager = new SSHTunnelManager(this.config.ssh, target);
27
+ this.tunnelManager.on('disconnecting', () => {
28
+ console.error('[DB] Tunnel disconnecting, queries may fail...');
29
+ this.isReconnecting = true;
30
+ });
31
+ this.tunnelManager.on('reconnected', async (info) => {
32
+ console.error(`[DB] Tunnel reconnected: ${info.oldPort} -> ${info.newPort}`);
33
+ await this.recreatePool(info.newPort);
34
+ this.isReconnecting = false;
35
+ });
36
+ this.tunnelManager.on('failed', (err) => {
37
+ console.error(`[DB] Tunnel failed permanently: ${err.message}`);
38
+ this.isInitialized = false;
39
+ this.isReconnecting = false;
40
+ });
41
+ this.currentLocalPort = await this.tunnelManager.connect();
42
+ }
43
+ await this.createPool();
44
+ this.isInitialized = true;
45
+ console.error('[DB] Connection manager initialized');
46
+ }
47
+ resolveSSLConfig(preference, originalHost) {
48
+ if (preference.explicit === 'false') {
49
+ console.error('[DB] SSL disabled (explicit configuration)');
50
+ return false;
51
+ }
52
+ if (preference.explicit === 'true') {
53
+ console.error('[DB] SSL enabled (explicit configuration)');
54
+ return {
55
+ rejectUnauthorized: preference.rejectUnauthorized,
56
+ ca: preference.ca
57
+ ? fs.readFileSync(preference.ca, 'utf8')
58
+ : undefined,
59
+ };
60
+ }
61
+ const isLocalhost = originalHost === 'localhost' ||
62
+ originalHost === '127.0.0.1' ||
63
+ originalHost === '::1';
64
+ if (isLocalhost) {
65
+ console.error('[DB] SSL disabled (localhost database)');
66
+ return false;
67
+ }
68
+ console.error('[DB] SSL enabled by default (non-localhost database)');
69
+ return {
70
+ rejectUnauthorized: preference.rejectUnauthorized,
71
+ ca: preference.ca ? fs.readFileSync(preference.ca, 'utf8') : undefined,
72
+ };
73
+ }
74
+ async createPool() {
75
+ const host = this.tunnelManager ? '127.0.0.1' : this.config.database.host;
76
+ const port = this.currentLocalPort || this.config.database.port;
77
+ const sslConfig = this.resolveSSLConfig(this.config.sslPreference, this.config.database.host);
78
+ this.sslEnabled = sslConfig !== false;
79
+ this.pool = new Pool({
80
+ host,
81
+ port,
82
+ database: this.config.database.database,
83
+ user: this.config.database.user,
84
+ password: this.config.database.password,
85
+ max: 10,
86
+ idleTimeoutMillis: 30000,
87
+ connectionTimeoutMillis: 10000,
88
+ statement_timeout: this.config.queryTimeout,
89
+ ssl: sslConfig,
90
+ });
91
+ this.pool.on('error', (err) => {
92
+ console.error('[DB] Pool background error:', obfuscateConnectionString(err.message));
93
+ });
94
+ const client = await this.pool.connect();
95
+ try {
96
+ await client.query('SELECT 1');
97
+ }
98
+ finally {
99
+ client.release();
100
+ }
101
+ console.error(`[DB] Pool created on ${host}:${port}`);
102
+ }
103
+ async recreatePool(newLocalPort) {
104
+ const oldPool = this.pool;
105
+ this.pool = null;
106
+ this.currentLocalPort = newLocalPort;
107
+ if (oldPool) {
108
+ console.error('[DB] Draining old pool...');
109
+ try {
110
+ await Promise.race([
111
+ oldPool.end(),
112
+ new Promise((_, reject) => setTimeout(() => reject(new Error('Pool drain timeout')), 5000)),
113
+ ]);
114
+ console.error('[DB] Old pool drained');
115
+ }
116
+ catch {
117
+ console.error('[DB] Pool drain timeout, forcing close');
118
+ oldPool.end().catch(() => { });
119
+ }
120
+ }
121
+ await this.createPool();
122
+ console.error('[DB] Pool recreated successfully');
123
+ }
124
+ async executeQuery(sql, params) {
125
+ if (!this.pool) {
126
+ if (this.isReconnecting) {
127
+ throw new Error('Database connection lost, reconnecting...');
128
+ }
129
+ throw new Error('Connection not initialized');
130
+ }
131
+ if (this.config.readOnly) {
132
+ validateReadOnlyStatement(sql);
133
+ return this.executeReadOnlyQuery(sql, params || []);
134
+ }
135
+ return this.executeWriteQuery(sql, params || []);
136
+ }
137
+ async executeReadOnlyQuery(sql, params) {
138
+ const client = await this.pool.connect();
139
+ try {
140
+ await client.query('BEGIN TRANSACTION READ ONLY');
141
+ try {
142
+ let result;
143
+ if (this.shouldUseCursorLimiting(sql)) {
144
+ result = await this.executeQueryWithLimit(client, sql, params, this.config.maxRows, true);
145
+ }
146
+ else {
147
+ const pgResult = await client.query(sql, params);
148
+ result = {
149
+ rows: pgResult.rows,
150
+ rowCount: pgResult.rowCount || 0,
151
+ truncated: false,
152
+ fields: pgResult.fields?.map((f) => ({
153
+ name: f.name,
154
+ dataTypeID: f.dataTypeID,
155
+ })),
156
+ command: pgResult.command,
157
+ };
158
+ }
159
+ await client.query('ROLLBACK');
160
+ return result;
161
+ }
162
+ catch (err) {
163
+ await client.query('ROLLBACK').catch(() => { });
164
+ throw err;
165
+ }
166
+ }
167
+ finally {
168
+ client.release();
169
+ }
170
+ }
171
+ async executeWriteQuery(sql, params) {
172
+ const client = await this.pool.connect();
173
+ try {
174
+ if (this.shouldUseCursorLimiting(sql)) {
175
+ return await this.executeQueryWithLimit(client, sql, params, this.config.maxRows, false);
176
+ }
177
+ const pgResult = await client.query(sql, params);
178
+ const truncated = pgResult.rows && pgResult.rows.length > this.config.maxRows;
179
+ const rows = truncated
180
+ ? pgResult.rows.slice(0, this.config.maxRows)
181
+ : pgResult.rows || [];
182
+ return {
183
+ rows,
184
+ rowCount: pgResult.rowCount || 0,
185
+ truncated,
186
+ fields: pgResult.fields?.map((f) => ({
187
+ name: f.name,
188
+ dataTypeID: f.dataTypeID,
189
+ })),
190
+ command: pgResult.command,
191
+ };
192
+ }
193
+ finally {
194
+ client.release();
195
+ }
196
+ }
197
+ async executeQueryWithLimit(client, sql, params, maxRows, isReadOnlyMode) {
198
+ const cursorName = `mcp_cursor_${Date.now()}_${Math.random().toString(36).slice(2)}`;
199
+ const needsTransaction = !isReadOnlyMode;
200
+ try {
201
+ if (needsTransaction) {
202
+ await client.query('BEGIN');
203
+ }
204
+ await client.query(`DECLARE ${cursorName} CURSOR FOR ${sql}`, params);
205
+ const result = await client.query(`FETCH ${maxRows + 1} FROM ${cursorName}`);
206
+ const truncated = result.rows.length > maxRows;
207
+ const rows = truncated ? result.rows.slice(0, maxRows) : result.rows;
208
+ await client.query(`CLOSE ${cursorName}`);
209
+ if (needsTransaction) {
210
+ await client.query('COMMIT');
211
+ }
212
+ return {
213
+ rows,
214
+ rowCount: rows.length,
215
+ truncated,
216
+ fields: result.fields?.map((f) => ({
217
+ name: f.name,
218
+ dataTypeID: f.dataTypeID,
219
+ })),
220
+ command: result.command,
221
+ };
222
+ }
223
+ catch (err) {
224
+ await client.query(`CLOSE ${cursorName}`).catch(() => { });
225
+ if (needsTransaction) {
226
+ await client.query('ROLLBACK').catch(() => { });
227
+ }
228
+ throw err;
229
+ }
230
+ }
231
+ shouldUseCursorLimiting(sql) {
232
+ const firstKeyword = getFirstKeyword(sql);
233
+ if (!firstKeyword) {
234
+ return false;
235
+ }
236
+ if (firstKeyword === 'WITH') {
237
+ if (cteContainsDML(sql)) {
238
+ return false;
239
+ }
240
+ const finalStatement = extractFinalStatementAfterCTEs(sql);
241
+ if (finalStatement) {
242
+ const normalizedFinal = stripLeadingComments(finalStatement).trim().toUpperCase();
243
+ if (normalizedFinal.startsWith('INSERT ') ||
244
+ normalizedFinal.startsWith('UPDATE ') ||
245
+ normalizedFinal.startsWith('DELETE ') ||
246
+ normalizedFinal.startsWith('MERGE ')) {
247
+ return false;
248
+ }
249
+ }
250
+ return true;
251
+ }
252
+ if (firstKeyword === 'SELECT' ||
253
+ firstKeyword === 'TABLE' ||
254
+ firstKeyword === 'VALUES') {
255
+ return true;
256
+ }
257
+ return false;
258
+ }
259
+ getStatus() {
260
+ return {
261
+ initialized: this.isInitialized,
262
+ reconnecting: this.isReconnecting,
263
+ database: {
264
+ host: this.config.database.host,
265
+ port: this.config.database.port,
266
+ database: this.config.database.database,
267
+ user: this.config.database.user,
268
+ ssl: this.sslEnabled,
269
+ },
270
+ tunnel: this.tunnelManager?.getState(),
271
+ pool: {
272
+ totalCount: this.pool?.totalCount || 0,
273
+ idleCount: this.pool?.idleCount || 0,
274
+ waitingCount: this.pool?.waitingCount || 0,
275
+ },
276
+ mode: this.config.readOnly ? 'read-only' : 'read-write',
277
+ maxRows: this.config.maxRows,
278
+ queryTimeout: this.config.queryTimeout,
279
+ };
280
+ }
281
+ async close() {
282
+ console.error('[DB] Closing connection manager...');
283
+ if (this.pool) {
284
+ await this.pool.end();
285
+ this.pool = null;
286
+ }
287
+ if (this.tunnelManager) {
288
+ await this.tunnelManager.close();
289
+ this.tunnelManager = null;
290
+ }
291
+ this.isInitialized = false;
292
+ console.error('[DB] Connection manager closed');
293
+ }
294
+ }
295
+ //# sourceMappingURL=postgres-pool.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"postgres-pool.js","sourceRoot":"","sources":["../../src/connection/postgres-pool.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,IAAI,EAAc,MAAM,IAAI,CAAC;AACtC,OAAO,KAAK,EAAE,MAAM,IAAI,CAAC;AACzB,OAAO,EAAE,gBAAgB,EAAE,MAAM,iBAAiB,CAAC;AACnD,OAAO,EACL,yBAAyB,EACzB,oBAAoB,EACpB,cAAc,EACd,8BAA8B,EAC9B,eAAe,GAChB,MAAM,yBAAyB,CAAC;AACjC,OAAO,EAAE,yBAAyB,EAAE,MAAM,qBAAqB,CAAC;AAShE,MAAM,OAAO,iBAAiB;IACpB,IAAI,GAAgB,IAAI,CAAC;IACzB,aAAa,GAA4B,IAAI,CAAC;IAC9C,MAAM,CAAe;IACrB,aAAa,GAAG,KAAK,CAAC;IACtB,cAAc,GAAG,KAAK,CAAC;IACvB,gBAAgB,GAAkB,IAAI,CAAC;IACvC,UAAU,GAAG,KAAK,CAAC;IAE3B,YAAY,MAAoB;QAC9B,IAAI,CAAC,MAAM,GAAG,MAAM,CAAC;IACvB,CAAC;IAED,KAAK,CAAC,UAAU;QACd,IAAI,IAAI,CAAC,aAAa,EAAE,CAAC;YACvB,OAAO;QACT,CAAC;QAED,IAAI,IAAI,CAAC,MAAM,CAAC,GAAG,EAAE,CAAC;YACpB,MAAM,MAAM,GAAiB;gBAC3B,IAAI,EAAE,IAAI,CAAC,MAAM,CAAC,QAAQ,CAAC,IAAI;gBAC/B,IAAI,EAAE,IAAI,CAAC,MAAM,CAAC,QAAQ,CAAC,IAAI;aAChC,CAAC;YAEF,IAAI,CAAC,aAAa,GAAG,IAAI,gBAAgB,CAAC,IAAI,CAAC,MAAM,CAAC,GAAG,EAAE,MAAM,CAAC,CAAC;YAEnE,IAAI,CAAC,aAAa,CAAC,EAAE,CAAC,eAAe,EAAE,GAAG,EAAE;gBAC1C,OAAO,CAAC,KAAK,CAAC,gDAAgD,CAAC,CAAC;gBAChE,IAAI,CAAC,cAAc,GAAG,IAAI,CAAC;YAC7B,CAAC,CAAC,CAAC;YAEH,IAAI,CAAC,aAAa,CAAC,EAAE,CACnB,aAAa,EACb,KAAK,EAAE,IAA0C,EAAE,EAAE;gBACnD,OAAO,CAAC,KAAK,CACX,4BAA4B,IAAI,CAAC,OAAO,OAAO,IAAI,CAAC,OAAO,EAAE,CAC9D,CAAC;gBACF,MAAM,IAAI,CAAC,YAAY,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;gBACtC,IAAI,CAAC,cAAc,GAAG,KAAK,CAAC;YAC9B,CAAC,CACF,CAAC;YAEF,IAAI,CAAC,aAAa,CAAC,EAAE,CAAC,QAAQ,EAAE,CAAC,GAAU,EAAE,EAAE;gBAC7C,OAAO,CAAC,KAAK,CAAC,mCAAmC,GAAG,CAAC,OAAO,EAAE,CAAC,CAAC;gBAChE,IAAI,CAAC,aAAa,GAAG,KAAK,CAAC;gBAC3B,IAAI,CAAC,cAAc,GAAG,KAAK,CAAC;YAC9B,CAAC,CAAC,CAAC;YAEH,IAAI,CAAC,gBAAgB,GAAG,MAAM,IAAI,CAAC,aAAa,CAAC,OAAO,EAAE,CAAC;QAC7D,CAAC;QAED,MAAM,IAAI,CAAC,UAAU,EAAE,CAAC;QACxB,IAAI,CAAC,aAAa,GAAG,IAAI,CAAC;QAE1B,OAAO,CAAC,KAAK,CAAC,qCAAqC,CAAC,CAAC;IACvD,CAAC;IAEO,gBAAgB,CACtB,UAAyB,EACzB,YAAoB;QAEpB,IAAI,UAAU,CAAC,QAAQ,KAAK,OAAO,EAAE,CAAC;YACpC,OAAO,CAAC,KAAK,CAAC,4CAA4C,CAAC,CAAC;YAC5D,OAAO,KAAK,CAAC;QACf,CAAC;QAED,IAAI,UAAU,CAAC,QAAQ,KAAK,MAAM,EAAE,CAAC;YACnC,OAAO,CAAC,KAAK,CAAC,2CAA2C,CAAC,CAAC;YAC3D,OAAO;gBACL,kBAAkB,EAAE,UAAU,CAAC,kBAAkB;gBACjD,EAAE,EAAE,UAAU,CAAC,EAAE;oBACf,CAAC,CAAC,EAAE,CAAC,YAAY,CAAC,UAAU,CAAC,EAAE,EAAE,MAAM,CAAC;oBACxC,CAAC,CAAC,SAAS;aACd,CAAC;QACJ,CAAC;QAED,MAAM,WAAW,GACf,YAAY,KAAK,WAAW;YAC5B,YAAY,KAAK,WAAW;YAC5B,YAAY,KAAK,KAAK,CAAC;QAEzB,IAAI,WAAW,EAAE,CAAC;YAChB,OAAO,CAAC,KAAK,CAAC,wCAAwC,CAAC,CAAC;YACxD,OAAO,KAAK,CAAC;QACf,CAAC;QAED,OAAO,CAAC,KAAK,CAAC,sDAAsD,CAAC,CAAC;QACtE,OAAO;YACL,kBAAkB,EAAE,UAAU,CAAC,kBAAkB;YACjD,EAAE,EAAE,UAAU,CAAC,EAAE,CAAC,CAAC,CAAC,EAAE,CAAC,YAAY,CAAC,UAAU,CAAC,EAAE,EAAE,MAAM,CAAC,CAAC,CAAC,CAAC,SAAS;SACvE,CAAC;IACJ,CAAC;IAEO,KAAK,CAAC,UAAU;QACtB,MAAM,IAAI,GAAG,IAAI,CAAC,aAAa,CAAC,CAAC,CAAC,WAAW,CAAC,CAAC,CAAC,IAAI,CAAC,MAAM,CAAC,QAAQ,CAAC,IAAI,CAAC;QAC1E,MAAM,IAAI,GAAG,IAAI,CAAC,gBAAgB,IAAI,IAAI,CAAC,MAAM,CAAC,QAAQ,CAAC,IAAI,CAAC;QAEhE,MAAM,SAAS,GAAG,IAAI,CAAC,gBAAgB,CACrC,IAAI,CAAC,MAAM,CAAC,aAAa,EACzB,IAAI,CAAC,MAAM,CAAC,QAAQ,CAAC,IAAI,CAC1B,CAAC;QAEF,IAAI,CAAC,UAAU,GAAG,SAAS,KAAK,KAAK,CAAC;QAEtC,IAAI,CAAC,IAAI,GAAG,IAAI,IAAI,CAAC;YACnB,IAAI;YACJ,IAAI;YACJ,QAAQ,EAAE,IAAI,CAAC,MAAM,CAAC,QAAQ,CAAC,QAAQ;YACvC,IAAI,EAAE,IAAI,CAAC,MAAM,CAAC,QAAQ,CAAC,IAAI;YAC/B,QAAQ,EAAE,IAAI,CAAC,MAAM,CAAC,QAAQ,CAAC,QAAQ;YACvC,GAAG,EAAE,EAAE;YACP,iBAAiB,EAAE,KAAK;YACxB,uBAAuB,EAAE,KAAK;YAC9B,iBAAiB,EAAE,IAAI,CAAC,MAAM,CAAC,YAAY;YAC3C,GAAG,EAAE,SAAS;SACf,CAAC,CAAC;QAEH,IAAI,CAAC,IAAI,CAAC,EAAE,CAAC,OAAO,EAAE,CAAC,GAAG,EAAE,EAAE;YAC5B,OAAO,CAAC,KAAK,CACX,6BAA6B,EAC7B,yBAAyB,CAAC,GAAG,CAAC,OAAO,CAAC,CACvC,CAAC;QACJ,CAAC,CAAC,CAAC;QAEH,MAAM,MAAM,GAAG,MAAM,IAAI,CAAC,IAAI,CAAC,OAAO,EAAE,CAAC;QACzC,IAAI,CAAC;YACH,MAAM,MAAM,CAAC,KAAK,CAAC,UAAU,CAAC,CAAC;QACjC,CAAC;gBAAS,CAAC;YACT,MAAM,CAAC,OAAO,EAAE,CAAC;QACnB,CAAC;QAED,OAAO,CAAC,KAAK,CAAC,wBAAwB,IAAI,IAAI,IAAI,EAAE,CAAC,CAAC;IACxD,CAAC;IAEO,KAAK,CAAC,YAAY,CAAC,YAAoB;QAC7C,MAAM,OAAO,GAAG,IAAI,CAAC,IAAI,CAAC;QAC1B,IAAI,CAAC,IAAI,GAAG,IAAI,CAAC;QACjB,IAAI,CAAC,gBAAgB,GAAG,YAAY,CAAC;QAErC,IAAI,OAAO,EAAE,CAAC;YACZ,OAAO,CAAC,KAAK,CAAC,2BAA2B,CAAC,CAAC;YAC3C,IAAI,CAAC;gBACH,MAAM,OAAO,CAAC,IAAI,CAAC;oBACjB,OAAO,CAAC,GAAG,EAAE;oBACb,IAAI,OAAO,CAAC,CAAC,CAAC,EAAE,MAAM,EAAE,EAAE,CACxB,UAAU,CAAC,GAAG,EAAE,CAAC,MAAM,CAAC,IAAI,KAAK,CAAC,oBAAoB,CAAC,CAAC,EAAE,IAAI,CAAC,CAChE;iBACF,CAAC,CAAC;gBACH,OAAO,CAAC,KAAK,CAAC,uBAAuB,CAAC,CAAC;YACzC,CAAC;YAAC,MAAM,CAAC;gBACP,OAAO,CAAC,KAAK,CAAC,wCAAwC,CAAC,CAAC;gBACxD,OAAO,CAAC,GAAG,EAAE,CAAC,KAAK,CAAC,GAAG,EAAE,GAAE,CAAC,CAAC,CAAC;YAChC,CAAC;QACH,CAAC;QAED,MAAM,IAAI,CAAC,UAAU,EAAE,CAAC;QACxB,OAAO,CAAC,KAAK,CAAC,kCAAkC,CAAC,CAAC;IACpD,CAAC;IAED,KAAK,CAAC,YAAY,CAChB,GAAW,EACX,MAAkB;QAElB,IAAI,CAAC,IAAI,CAAC,IAAI,EAAE,CAAC;YACf,IAAI,IAAI,CAAC,cAAc,EAAE,CAAC;gBACxB,MAAM,IAAI,KAAK,CAAC,2CAA2C,CAAC,CAAC;YAC/D,CAAC;YACD,MAAM,IAAI,KAAK,CAAC,4BAA4B,CAAC,CAAC;QAChD,CAAC;QAED,IAAI,IAAI,CAAC,MAAM,CAAC,QAAQ,EAAE,CAAC;YACzB,yBAAyB,CAAC,GAAG,CAAC,CAAC;YAC/B,OAAO,IAAI,CAAC,oBAAoB,CAAC,GAAG,EAAE,MAAM,IAAI,EAAE,CAAC,CAAC;QACtD,CAAC;QAED,OAAO,IAAI,CAAC,iBAAiB,CAAC,GAAG,EAAE,MAAM,IAAI,EAAE,CAAC,CAAC;IACnD,CAAC;IAEO,KAAK,CAAC,oBAAoB,CAChC,GAAW,EACX,MAAiB;QAEjB,MAAM,MAAM,GAAG,MAAM,IAAI,CAAC,IAAK,CAAC,OAAO,EAAE,CAAC;QAE1C,IAAI,CAAC;YACH,MAAM,MAAM,CAAC,KAAK,CAAC,6BAA6B,CAAC,CAAC;YAElD,IAAI,CAAC;gBACH,IAAI,MAA2B,CAAC;gBAEhC,IAAI,IAAI,CAAC,uBAAuB,CAAC,GAAG,CAAC,EAAE,CAAC;oBACtC,MAAM,GAAG,MAAM,IAAI,CAAC,qBAAqB,CACvC,MAAM,EACN,GAAG,EACH,MAAM,EACN,IAAI,CAAC,MAAM,CAAC,OAAO,EACnB,IAAI,CACL,CAAC;gBACJ,CAAC;qBAAM,CAAC;oBACN,MAAM,QAAQ,GAAG,MAAM,MAAM,CAAC,KAAK,CAAC,GAAG,EAAE,MAAM,CAAC,CAAC;oBACjD,MAAM,GAAG;wBACP,IAAI,EAAE,QAAQ,CAAC,IAAI;wBACnB,QAAQ,EAAE,QAAQ,CAAC,QAAQ,IAAI,CAAC;wBAChC,SAAS,EAAE,KAAK;wBAChB,MAAM,EAAE,QAAQ,CAAC,MAAM,EAAE,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC;4BACnC,IAAI,EAAE,CAAC,CAAC,IAAI;4BACZ,UAAU,EAAE,CAAC,CAAC,UAAU;yBACzB,CAAC,CAAC;wBACH,OAAO,EAAE,QAAQ,CAAC,OAAO;qBAC1B,CAAC;gBACJ,CAAC;gBAED,MAAM,MAAM,CAAC,KAAK,CAAC,UAAU,CAAC,CAAC;gBAE/B,OAAO,MAAM,CAAC;YAChB,CAAC;YAAC,OAAO,GAAG,EAAE,CAAC;gBACb,MAAM,MAAM,CAAC,KAAK,CAAC,UAAU,CAAC,CAAC,KAAK,CAAC,GAAG,EAAE,GAAE,CAAC,CAAC,CAAC;gBAC/C,MAAM,GAAG,CAAC;YACZ,CAAC;QACH,CAAC;gBAAS,CAAC;YACT,MAAM,CAAC,OAAO,EAAE,CAAC;QACnB,CAAC;IACH,CAAC;IAEO,KAAK,CAAC,iBAAiB,CAC7B,GAAW,EACX,MAAiB;QAEjB,MAAM,MAAM,GAAG,MAAM,IAAI,CAAC,IAAK,CAAC,OAAO,EAAE,CAAC;QAE1C,IAAI,CAAC;YACH,IAAI,IAAI,CAAC,uBAAuB,CAAC,GAAG,CAAC,EAAE,CAAC;gBACtC,OAAO,MAAM,IAAI,CAAC,qBAAqB,CACrC,MAAM,EACN,GAAG,EACH,MAAM,EACN,IAAI,CAAC,MAAM,CAAC,OAAO,EACnB,KAAK,CACN,CAAC;YACJ,CAAC;YAED,MAAM,QAAQ,GAAG,MAAM,MAAM,CAAC,KAAK,CAAC,GAAG,EAAE,MAAM,CAAC,CAAC;YAEjD,MAAM,SAAS,GACb,QAAQ,CAAC,IAAI,IAAI,QAAQ,CAAC,IAAI,CAAC,MAAM,GAAG,IAAI,CAAC,MAAM,CAAC,OAAO,CAAC;YAC9D,MAAM,IAAI,GAAG,SAAS;gBACpB,CAAC,CAAC,QAAQ,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC,EAAE,IAAI,CAAC,MAAM,CAAC,OAAO,CAAC;gBAC7C,CAAC,CAAC,QAAQ,CAAC,IAAI,IAAI,EAAE,CAAC;YAExB,OAAO;gBACL,IAAI;gBACJ,QAAQ,EAAE,QAAQ,CAAC,QAAQ,IAAI,CAAC;gBAChC,SAAS;gBACT,MAAM,EAAE,QAAQ,CAAC,MAAM,EAAE,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC;oBACnC,IAAI,EAAE,CAAC,CAAC,IAAI;oBACZ,UAAU,EAAE,CAAC,CAAC,UAAU;iBACzB,CAAC,CAAC;gBACH,OAAO,EAAE,QAAQ,CAAC,OAAO;aAC1B,CAAC;QACJ,CAAC;gBAAS,CAAC;YACT,MAAM,CAAC,OAAO,EAAE,CAAC;QACnB,CAAC;IACH,CAAC;IAEO,KAAK,CAAC,qBAAqB,CACjC,MAAkB,EAClB,GAAW,EACX,MAAiB,EACjB,OAAe,EACf,cAAuB;QAEvB,MAAM,UAAU,GAAG,cAAc,IAAI,CAAC,GAAG,EAAE,IAAI,IAAI,CAAC,MAAM,EAAE,CAAC,QAAQ,CAAC,EAAE,CAAC,CAAC,KAAK,CAAC,CAAC,CAAC,EAAE,CAAC;QAErF,MAAM,gBAAgB,GAAG,CAAC,cAAc,CAAC;QAEzC,IAAI,CAAC;YACH,IAAI,gBAAgB,EAAE,CAAC;gBACrB,MAAM,MAAM,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC;YAC9B,CAAC;YAED,MAAM,MAAM,CAAC,KAAK,CAAC,WAAW,UAAU,eAAe,GAAG,EAAE,EAAE,MAAM,CAAC,CAAC;YAEtE,MAAM,MAAM,GAAG,MAAM,MAAM,CAAC,KAAK,CAC/B,SAAS,OAAO,GAAG,CAAC,SAAS,UAAU,EAAE,CAC1C,CAAC;YAEF,MAAM,SAAS,GAAG,MAAM,CAAC,IAAI,CAAC,MAAM,GAAG,OAAO,CAAC;YAC/C,MAAM,IAAI,GAAG,SAAS,CAAC,CAAC,CAAC,MAAM,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC,EAAE,OAAO,CAAC,CAAC,CAAC,CAAC,MAAM,CAAC,IAAI,CAAC;YAErE,MAAM,MAAM,CAAC,KAAK,CAAC,SAAS,UAAU,EAAE,CAAC,CAAC;YAE1C,IAAI,gBAAgB,EAAE,CAAC;gBACrB,MAAM,MAAM,CAAC,KAAK,CAAC,QAAQ,CAAC,CAAC;YAC/B,CAAC;YAED,OAAO;gBACL,IAAI;gBACJ,QAAQ,EAAE,IAAI,CAAC,MAAM;gBACrB,SAAS;gBACT,MAAM,EAAE,MAAM,CAAC,MAAM,EAAE,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC;oBACjC,IAAI,EAAE,CAAC,CAAC,IAAI;oBACZ,UAAU,EAAE,CAAC,CAAC,UAAU;iBACzB,CAAC,CAAC;gBACH,OAAO,EAAE,MAAM,CAAC,OAAO;aACxB,CAAC;QACJ,CAAC;QAAC,OAAO,GAAG,EAAE,CAAC;YACb,MAAM,MAAM,CAAC,KAAK,CAAC,SAAS,UAAU,EAAE,CAAC,CAAC,KAAK,CAAC,GAAG,EAAE,GAAE,CAAC,CAAC,CAAC;YAC1D,IAAI,gBAAgB,EAAE,CAAC;gBACrB,MAAM,MAAM,CAAC,KAAK,CAAC,UAAU,CAAC,CAAC,KAAK,CAAC,GAAG,EAAE,GAAE,CAAC,CAAC,CAAC;YACjD,CAAC;YACD,MAAM,GAAG,CAAC;QACZ,CAAC;IACH,CAAC;IAEO,uBAAuB,CAAC,GAAW;QACzC,MAAM,YAAY,GAAG,eAAe,CAAC,GAAG,CAAC,CAAC;QAC1C,IAAI,CAAC,YAAY,EAAE,CAAC;YAClB,OAAO,KAAK,CAAC;QACf,CAAC;QAED,IAAI,YAAY,KAAK,MAAM,EAAE,CAAC;YAC5B,IAAI,cAAc,CAAC,GAAG,CAAC,EAAE,CAAC;gBACxB,OAAO,KAAK,CAAC;YACf,CAAC;YAED,MAAM,cAAc,GAAG,8BAA8B,CAAC,GAAG,CAAC,CAAC;YAC3D,IAAI,cAAc,EAAE,CAAC;gBACnB,MAAM,eAAe,GAAG,oBAAoB,CAAC,cAAc,CAAC,CAAC,IAAI,EAAE,CAAC,WAAW,EAAE,CAAC;gBAClF,IACE,eAAe,CAAC,UAAU,CAAC,SAAS,CAAC;oBACrC,eAAe,CAAC,UAAU,CAAC,SAAS,CAAC;oBACrC,eAAe,CAAC,UAAU,CAAC,SAAS,CAAC;oBACrC,eAAe,CAAC,UAAU,CAAC,QAAQ,CAAC,EACpC,CAAC;oBACD,OAAO,KAAK,CAAC;gBACf,CAAC;YACH,CAAC;YAED,OAAO,IAAI,CAAC;QACd,CAAC;QAED,IACE,YAAY,KAAK,QAAQ;YACzB,YAAY,KAAK,OAAO;YACxB,YAAY,KAAK,QAAQ,EACzB,CAAC;YACD,OAAO,IAAI,CAAC;QACd,CAAC;QAED,OAAO,KAAK,CAAC;IACf,CAAC;IAED,SAAS;QACP,OAAO;YACL,WAAW,EAAE,IAAI,CAAC,aAAa;YAC/B,YAAY,EAAE,IAAI,CAAC,cAAc;YACjC,QAAQ,EAAE;gBACR,IAAI,EAAE,IAAI,CAAC,MAAM,CAAC,QAAQ,CAAC,IAAI;gBAC/B,IAAI,EAAE,IAAI,CAAC,MAAM,CAAC,QAAQ,CAAC,IAAI;gBAC/B,QAAQ,EAAE,IAAI,CAAC,MAAM,CAAC,QAAQ,CAAC,QAAQ;gBACvC,IAAI,EAAE,IAAI,CAAC,MAAM,CAAC,QAAQ,CAAC,IAAI;gBAC/B,GAAG,EAAE,IAAI,CAAC,UAAU;aACrB;YACD,MAAM,EAAE,IAAI,CAAC,aAAa,EAAE,QAAQ,EAAE;YACtC,IAAI,EAAE;gBACJ,UAAU,EAAE,IAAI,CAAC,IAAI,EAAE,UAAU,IAAI,CAAC;gBACtC,SAAS,EAAE,IAAI,CAAC,IAAI,EAAE,SAAS,IAAI,CAAC;gBACpC,YAAY,EAAE,IAAI,CAAC,IAAI,EAAE,YAAY,IAAI,CAAC;aAC3C;YACD,IAAI,EAAE,IAAI,CAAC,MAAM,CAAC,QAAQ,CAAC,CAAC,CAAC,WAAW,CAAC,CAAC,CAAC,YAAY;YACvD,OAAO,EAAE,IAAI,CAAC,MAAM,CAAC,OAAO;YAC5B,YAAY,EAAE,IAAI,CAAC,MAAM,CAAC,YAAY;SACvC,CAAC;IACJ,CAAC;IAED,KAAK,CAAC,KAAK;QACT,OAAO,CAAC,KAAK,CAAC,oCAAoC,CAAC,CAAC;QAEpD,IAAI,IAAI,CAAC,IAAI,EAAE,CAAC;YACd,MAAM,IAAI,CAAC,IAAI,CAAC,GAAG,EAAE,CAAC;YACtB,IAAI,CAAC,IAAI,GAAG,IAAI,CAAC;QACnB,CAAC;QAED,IAAI,IAAI,CAAC,aAAa,EAAE,CAAC;YACvB,MAAM,IAAI,CAAC,aAAa,CAAC,KAAK,EAAE,CAAC;YACjC,IAAI,CAAC,aAAa,GAAG,IAAI,CAAC;QAC5B,CAAC;QAED,IAAI,CAAC,aAAa,GAAG,KAAK,CAAC;QAC3B,OAAO,CAAC,KAAK,CAAC,gCAAgC,CAAC,CAAC;IAClD,CAAC;CACF"}
@@ -0,0 +1,34 @@
1
+ import { EventEmitter } from 'events';
2
+ import type { SSHTunnelConfig, TunnelTarget, TunnelState } from '../types.js';
3
+ export declare class SSHTunnelManager extends EventEmitter {
4
+ private config;
5
+ private target;
6
+ private client;
7
+ private server;
8
+ private localPort;
9
+ private status;
10
+ private reconnectAttempts;
11
+ private maxReconnectAttempts;
12
+ private startTime;
13
+ private lastError?;
14
+ private hostKeyVerifier;
15
+ private hostKeyVerified;
16
+ private hostKeyError;
17
+ private activeConnections;
18
+ private isShuttingDown;
19
+ constructor(config: SSHTunnelConfig, target: TunnelTarget);
20
+ connect(): Promise<number>;
21
+ private parseHostKey;
22
+ private validateKeyPermissions;
23
+ private setupLocalServer;
24
+ private handleIncomingConnection;
25
+ private buildConnectConfig;
26
+ private handleDisconnect;
27
+ private scheduleReconnect;
28
+ private cleanup;
29
+ private setStatus;
30
+ isConnected(): boolean;
31
+ getState(): TunnelState;
32
+ close(): Promise<void>;
33
+ }
34
+ //# sourceMappingURL=ssh-tunnel.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"ssh-tunnel.d.ts","sourceRoot":"","sources":["../../src/connection/ssh-tunnel.ts"],"names":[],"mappings":"AAGA,OAAO,EAAE,YAAY,EAAE,MAAM,QAAQ,CAAC;AAGtC,OAAO,KAAK,EACV,eAAe,EACf,YAAY,EAEZ,WAAW,EACZ,MAAM,aAAa,CAAC;AAErB,qBAAa,gBAAiB,SAAQ,YAAY;IAgB9C,OAAO,CAAC,MAAM;IACd,OAAO,CAAC,MAAM;IAhBhB,OAAO,CAAC,MAAM,CAAuB;IACrC,OAAO,CAAC,MAAM,CAA2B;IACzC,OAAO,CAAC,SAAS,CAAa;IAC9B,OAAO,CAAC,MAAM,CAAgC;IAC9C,OAAO,CAAC,iBAAiB,CAAK;IAC9B,OAAO,CAAC,oBAAoB,CAAK;IACjC,OAAO,CAAC,SAAS,CAAa;IAC9B,OAAO,CAAC,SAAS,CAAC,CAAS;IAC3B,OAAO,CAAC,eAAe,CAAgC;IACvD,OAAO,CAAC,eAAe,CAAkB;IACzC,OAAO,CAAC,YAAY,CAAuB;IAC3C,OAAO,CAAC,iBAAiB,CAA8B;IACvD,OAAO,CAAC,cAAc,CAAS;gBAGrB,MAAM,EAAE,eAAe,EACvB,MAAM,EAAE,YAAY;IAcxB,OAAO,IAAI,OAAO,CAAC,MAAM,CAAC;IAiGhC,OAAO,CAAC,YAAY;YAiBN,sBAAsB;IAqBpC,OAAO,CAAC,gBAAgB;IA4BxB,OAAO,CAAC,wBAAwB;IAmDhC,OAAO,CAAC,kBAAkB;IA2B1B,OAAO,CAAC,gBAAgB;YAqBV,iBAAiB;YAyCjB,OAAO;IAerB,OAAO,CAAC,SAAS;IAKjB,WAAW,IAAI,OAAO;IAItB,QAAQ,IAAI,WAAW;IAUjB,KAAK,IAAI,OAAO,CAAC,IAAI,CAAC;CAW7B"}
@@ -0,0 +1,295 @@
1
+ import { Client } from 'ssh2';
2
+ import * as net from 'net';
3
+ import * as fs from 'fs';
4
+ import { EventEmitter } from 'events';
5
+ import { HostKeyVerifier } from './host-key-verifier.js';
6
+ import { obfuscateConnectionString } from '../lib/obfuscate.js';
7
+ export class SSHTunnelManager extends EventEmitter {
8
+ config;
9
+ target;
10
+ client = null;
11
+ server = null;
12
+ localPort = 0;
13
+ status = 'disconnected';
14
+ reconnectAttempts = 0;
15
+ maxReconnectAttempts = 5;
16
+ startTime = 0;
17
+ lastError;
18
+ hostKeyVerifier = null;
19
+ hostKeyVerified = false;
20
+ hostKeyError = null;
21
+ activeConnections = new Set();
22
+ isShuttingDown = false;
23
+ constructor(config, target) {
24
+ super();
25
+ this.config = config;
26
+ this.target = target;
27
+ if (config.strictHostKey !== false) {
28
+ this.hostKeyVerifier = new HostKeyVerifier(config.knownHostsPath);
29
+ }
30
+ else {
31
+ console.error('[SSH] WARNING: Host key verification DISABLED (strictHostKey=false)');
32
+ console.error('[SSH] This is INSECURE - only use for development/testing');
33
+ }
34
+ }
35
+ async connect() {
36
+ if (this.status === 'connected') {
37
+ return this.localPort;
38
+ }
39
+ this.isShuttingDown = false;
40
+ this.setStatus('connecting');
41
+ if (this.config.privateKeyPath) {
42
+ await this.validateKeyPermissions(this.config.privateKeyPath);
43
+ }
44
+ return new Promise((resolve, reject) => {
45
+ this.client = new Client();
46
+ this.hostKeyVerified = false;
47
+ this.hostKeyError = null;
48
+ const connectConfig = this.buildConnectConfig();
49
+ if (this.hostKeyVerifier) {
50
+ connectConfig.hostVerifier = (key) => {
51
+ const parsed = this.parseHostKey(key);
52
+ if (!parsed) {
53
+ this.hostKeyError = 'Failed to parse server host key';
54
+ return false;
55
+ }
56
+ const result = this.hostKeyVerifier.verifyHostKey(this.config.host, this.config.port, parsed.keyType, parsed.keyData);
57
+ if (!result.verified) {
58
+ this.hostKeyError = result.reason;
59
+ console.error(`[SSH] HOST KEY VERIFICATION FAILED: ${result.reason}`);
60
+ return false;
61
+ }
62
+ this.hostKeyVerified = true;
63
+ console.error('[SSH] Host key verified successfully');
64
+ return true;
65
+ };
66
+ }
67
+ this.client.on('ready', () => {
68
+ if (this.hostKeyVerifier && !this.hostKeyVerified) {
69
+ console.error('[SSH] BUG: Reached ready without host key verification');
70
+ this.client?.end();
71
+ reject(new Error('Internal error: host key not verified'));
72
+ return;
73
+ }
74
+ console.error(`[SSH] Connected to ${this.config.host}:${this.config.port}`);
75
+ this.startTime = Date.now();
76
+ this.setupLocalServer(resolve, reject);
77
+ });
78
+ this.client.on('error', (err) => {
79
+ const message = this.hostKeyError || err.message;
80
+ const safeMessage = obfuscateConnectionString(message);
81
+ console.error('[SSH] Connection error:', safeMessage);
82
+ this.lastError = safeMessage;
83
+ if (this.status === 'connecting') {
84
+ this.setStatus('failed');
85
+ reject(new Error(`SSH connection failed: ${safeMessage}`));
86
+ }
87
+ else {
88
+ this.handleDisconnect();
89
+ }
90
+ });
91
+ this.client.on('close', () => {
92
+ console.error('[SSH] Connection closed');
93
+ this.handleDisconnect();
94
+ });
95
+ this.client.on('end', () => {
96
+ console.error('[SSH] Connection ended');
97
+ this.handleDisconnect();
98
+ });
99
+ try {
100
+ this.client.connect(connectConfig);
101
+ }
102
+ catch (err) {
103
+ const message = err instanceof Error ? err.message : String(err);
104
+ const safeMessage = obfuscateConnectionString(message);
105
+ this.lastError = safeMessage;
106
+ this.setStatus('failed');
107
+ reject(new Error(`SSH connection failed: ${safeMessage}`));
108
+ }
109
+ });
110
+ }
111
+ parseHostKey(key) {
112
+ try {
113
+ let offset = 0;
114
+ const typeLen = key.readUInt32BE(offset);
115
+ offset += 4;
116
+ const keyType = key.subarray(offset, offset + typeLen).toString('utf8');
117
+ return { keyType, keyData: key };
118
+ }
119
+ catch {
120
+ return null;
121
+ }
122
+ }
123
+ async validateKeyPermissions(keyPath) {
124
+ const expandedPath = keyPath.replace(/^~/, process.env.HOME || '');
125
+ try {
126
+ const stats = fs.statSync(expandedPath);
127
+ const mode = stats.mode & 0o777;
128
+ if (mode & 0o077) {
129
+ throw new Error(`SSH key file has insecure permissions (${mode.toString(8)}). ` +
130
+ `Required: 600 or 400. Fix with: chmod 600 ${keyPath}`);
131
+ }
132
+ }
133
+ catch (err) {
134
+ if (err instanceof Error && 'code' in err && err.code === 'ENOENT') {
135
+ throw new Error(`SSH key file not found: ${keyPath}`);
136
+ }
137
+ throw err;
138
+ }
139
+ }
140
+ setupLocalServer(resolve, reject) {
141
+ this.server = net.createServer((socket) => {
142
+ this.handleIncomingConnection(socket);
143
+ });
144
+ this.server.on('error', (err) => {
145
+ console.error('[SSH] Local server error:', err.message);
146
+ reject(err);
147
+ });
148
+ this.server.listen(0, '127.0.0.1', () => {
149
+ const addr = this.server.address();
150
+ this.localPort = addr.port;
151
+ this.reconnectAttempts = 0;
152
+ this.setStatus('connected');
153
+ console.error(`[SSH] Tunnel established: localhost:${this.localPort} -> ` +
154
+ `${this.target.host}:${this.target.port}`);
155
+ resolve(this.localPort);
156
+ });
157
+ }
158
+ handleIncomingConnection(socket) {
159
+ if (!this.client) {
160
+ socket.end();
161
+ return;
162
+ }
163
+ this.activeConnections.add(socket);
164
+ this.client.forwardOut('127.0.0.1', socket.localPort || 0, this.target.host, this.target.port, (err, stream) => {
165
+ if (err) {
166
+ console.error('[SSH] Forward error:', obfuscateConnectionString(err.message));
167
+ socket.end();
168
+ this.activeConnections.delete(socket);
169
+ return;
170
+ }
171
+ socket.pipe(stream).pipe(socket);
172
+ stream.on('close', () => {
173
+ socket.end();
174
+ this.activeConnections.delete(socket);
175
+ });
176
+ socket.on('close', () => {
177
+ stream.end();
178
+ this.activeConnections.delete(socket);
179
+ });
180
+ stream.on('error', (streamErr) => {
181
+ console.error('[SSH] Stream error:', streamErr.message);
182
+ socket.end();
183
+ this.activeConnections.delete(socket);
184
+ });
185
+ socket.on('error', (socketErr) => {
186
+ console.error('[SSH] Socket error:', socketErr.message);
187
+ stream.end();
188
+ this.activeConnections.delete(socket);
189
+ });
190
+ });
191
+ }
192
+ buildConnectConfig() {
193
+ const config = {
194
+ host: this.config.host,
195
+ port: this.config.port,
196
+ username: this.config.username,
197
+ keepaliveInterval: this.config.keepaliveInterval || 10000,
198
+ keepaliveCountMax: 3,
199
+ readyTimeout: 20000,
200
+ };
201
+ if (this.config.privateKeyPath) {
202
+ const keyPath = this.config.privateKeyPath.replace(/^~/, process.env.HOME || '');
203
+ config.privateKey = fs.readFileSync(keyPath);
204
+ if (this.config.privateKeyPassphrase) {
205
+ config.passphrase = this.config.privateKeyPassphrase;
206
+ }
207
+ }
208
+ else if (this.config.password) {
209
+ config.password = this.config.password;
210
+ }
211
+ return config;
212
+ }
213
+ handleDisconnect() {
214
+ if (this.status === 'disconnected' ||
215
+ this.status === 'failed' ||
216
+ this.isShuttingDown) {
217
+ return;
218
+ }
219
+ const oldPort = this.localPort;
220
+ for (const socket of this.activeConnections) {
221
+ socket.destroy();
222
+ }
223
+ this.activeConnections.clear();
224
+ this.setStatus('disconnected');
225
+ this.emit('disconnecting', { oldPort });
226
+ this.scheduleReconnect();
227
+ }
228
+ async scheduleReconnect() {
229
+ if (this.isShuttingDown) {
230
+ return;
231
+ }
232
+ if (this.reconnectAttempts >= this.maxReconnectAttempts) {
233
+ console.error('[SSH] Max reconnection attempts reached. Giving up.');
234
+ this.setStatus('failed');
235
+ this.emit('failed', new Error('Max reconnection attempts reached'));
236
+ return;
237
+ }
238
+ const backoff = Math.min(1000 * Math.pow(2, this.reconnectAttempts), 30000);
239
+ this.reconnectAttempts++;
240
+ this.setStatus('reconnecting');
241
+ console.error(`[SSH] Reconnecting in ${backoff}ms ` +
242
+ `(attempt ${this.reconnectAttempts}/${this.maxReconnectAttempts})`);
243
+ await new Promise((r) => setTimeout(r, backoff));
244
+ if (this.isShuttingDown) {
245
+ return;
246
+ }
247
+ await this.cleanup();
248
+ try {
249
+ const oldPort = this.localPort;
250
+ const newPort = await this.connect();
251
+ this.emit('reconnected', { oldPort, newPort });
252
+ }
253
+ catch {
254
+ console.error('[SSH] Reconnection failed');
255
+ }
256
+ }
257
+ async cleanup() {
258
+ if (this.server) {
259
+ this.server.close();
260
+ this.server = null;
261
+ }
262
+ if (this.client) {
263
+ this.client.end();
264
+ this.client = null;
265
+ }
266
+ this.localPort = 0;
267
+ this.startTime = 0;
268
+ }
269
+ setStatus(status) {
270
+ this.status = status;
271
+ this.emit('statusChange', this.getState());
272
+ }
273
+ isConnected() {
274
+ return this.status === 'connected';
275
+ }
276
+ getState() {
277
+ return {
278
+ status: this.status,
279
+ localPort: this.localPort || null,
280
+ uptime: this.startTime > 0 ? Date.now() - this.startTime : 0,
281
+ reconnectAttempts: this.reconnectAttempts,
282
+ lastError: this.lastError,
283
+ };
284
+ }
285
+ async close() {
286
+ this.isShuttingDown = true;
287
+ for (const socket of this.activeConnections) {
288
+ socket.destroy();
289
+ }
290
+ this.activeConnections.clear();
291
+ await this.cleanup();
292
+ this.setStatus('disconnected');
293
+ }
294
+ }
295
+ //# sourceMappingURL=ssh-tunnel.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"ssh-tunnel.js","sourceRoot":"","sources":["../../src/connection/ssh-tunnel.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,3 @@
1
+ #!/usr/bin/env node
2
+ export {};
3
+ //# sourceMappingURL=index.d.ts.map