@push.rocks/smartproxy 21.1.2 → 21.1.5
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/changelog.md +23 -0
- package/dist_ts/00_commitinfo_data.js +2 -2
- package/dist_ts/core/models/socket-augmentation.d.ts +3 -0
- package/dist_ts/core/models/socket-augmentation.js +1 -1
- package/dist_ts/core/utils/socket-tracker.d.ts +16 -0
- package/dist_ts/core/utils/socket-tracker.js +49 -0
- package/dist_ts/detection/detectors/http-detector.js +19 -7
- package/dist_ts/detection/detectors/tls-detector.d.ts +1 -9
- package/dist_ts/detection/detectors/tls-detector.js +6 -29
- package/dist_ts/detection/protocol-detector.d.ts +7 -2
- package/dist_ts/detection/protocol-detector.js +80 -8
- package/dist_ts/proxies/http-proxy/http-proxy.d.ts +5 -1
- package/dist_ts/proxies/http-proxy/http-proxy.js +63 -39
- package/dist_ts/proxies/smart-proxy/certificate-manager.d.ts +5 -0
- package/dist_ts/proxies/smart-proxy/certificate-manager.js +20 -6
- package/dist_ts/proxies/smart-proxy/connection-manager.js +12 -1
- package/dist_ts/proxies/smart-proxy/index.d.ts +1 -0
- package/dist_ts/proxies/smart-proxy/index.js +2 -1
- package/dist_ts/proxies/smart-proxy/metrics-collector.d.ts +4 -0
- package/dist_ts/proxies/smart-proxy/metrics-collector.js +52 -7
- package/dist_ts/proxies/smart-proxy/route-connection-handler.js +7 -7
- package/dist_ts/proxies/smart-proxy/route-orchestrator.d.ts +56 -0
- package/dist_ts/proxies/smart-proxy/route-orchestrator.js +204 -0
- package/dist_ts/proxies/smart-proxy/smart-proxy.d.ts +1 -11
- package/dist_ts/proxies/smart-proxy/smart-proxy.js +48 -237
- package/dist_ts/proxies/smart-proxy/utils/route-helpers.js +42 -7
- package/dist_ts/proxies/smart-proxy/utils/route-validator.d.ts +58 -0
- package/dist_ts/proxies/smart-proxy/utils/route-validator.js +405 -0
- package/package.json +3 -2
- package/readme.md +321 -828
- package/ts/00_commitinfo_data.ts +1 -1
- package/ts/core/models/socket-augmentation.ts +5 -0
- package/ts/core/utils/socket-tracker.ts +63 -0
- package/ts/detection/detectors/http-detector.ts +20 -7
- package/ts/detection/protocol-detector.ts +57 -6
- package/ts/proxies/http-proxy/http-proxy.ts +73 -48
- package/ts/proxies/smart-proxy/certificate-manager.ts +21 -5
- package/ts/proxies/smart-proxy/index.ts +1 -0
- package/ts/proxies/smart-proxy/metrics-collector.ts +58 -6
- package/ts/proxies/smart-proxy/route-orchestrator.ts +297 -0
- package/ts/proxies/smart-proxy/smart-proxy.ts +66 -270
- package/ts/proxies/smart-proxy/utils/route-helpers.ts +45 -6
- package/ts/proxies/smart-proxy/utils/route-validator.ts +453 -0
package/changelog.md
CHANGED
|
@@ -1,5 +1,28 @@
|
|
|
1
1
|
# Changelog
|
|
2
2
|
|
|
3
|
+
## 2025-08-19 - 21.1.5 - fix(core)
|
|
4
|
+
Prepare patch release: documentation, tests and stability fixes (metrics, ACME, connection cleanup)
|
|
5
|
+
|
|
6
|
+
- Byte counting and throughput: per-route and per-IP throughput trackers with per-second sampling; removed double-counting and improved sampling buffers for accurate rates
|
|
7
|
+
- HttpProxy and forwarding: Ensure metricsCollector.recordBytes() is called in forwarding paths so throughput is recorded reliably
|
|
8
|
+
- ACME / Certificate Manager: support for custom certProvisionFunction with configurable fallback to ACME (http01) and improved challenge route lifecycle
|
|
9
|
+
- Connection lifecycle and cleanup: improved lifecycle component timer/listener cleanup, better cleanup queue batching and zombie/half-zombie detection
|
|
10
|
+
- Various utilities and stability improvements: enhanced IP utils, path/domain matching improvements, safer socket handling and more robust fragment/ClientHello handling
|
|
11
|
+
- Tests and docs: many test files and readme.hints.md updated with byte-counting audit, connection cleanup and ACME guidance
|
|
12
|
+
|
|
13
|
+
## 2025-08-14 - 21.1.4 - fix(security)
|
|
14
|
+
Critical security and stability fixes
|
|
15
|
+
|
|
16
|
+
- Fixed critical socket.emit override vulnerability that was breaking TLS connections
|
|
17
|
+
- Implemented comprehensive socket cleanup with new socket tracker utility
|
|
18
|
+
- Improved code organization by extracting RouteOrchestrator from SmartProxy
|
|
19
|
+
- Fixed IPv6 loopback detection for proper IPv6 support
|
|
20
|
+
- Added memory bounds to prevent unbounded collection growth
|
|
21
|
+
- Fixed certificate manager race conditions with proper synchronization
|
|
22
|
+
- Unreferenced long-lived timers to prevent process hanging
|
|
23
|
+
- Enhanced route validation for socket-handler actions
|
|
24
|
+
- Fixed header parsing when extractFullHeaders option is enabled
|
|
25
|
+
|
|
3
26
|
## 2025-07-22 - 21.1.1 - fix(detection)
|
|
4
27
|
Fix SNI detection in TLS detector
|
|
5
28
|
|
|
@@ -3,7 +3,7 @@
|
|
|
3
3
|
*/
|
|
4
4
|
export const commitinfo = {
|
|
5
5
|
name: '@push.rocks/smartproxy',
|
|
6
|
-
version: '
|
|
6
|
+
version: '21.1.5',
|
|
7
7
|
description: 'A powerful proxy package with unified route-based configuration for high traffic management. Features include SSL/TLS support, flexible routing patterns, WebSocket handling, advanced security options, and automatic ACME certificate management.'
|
|
8
8
|
};
|
|
9
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
9
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiMDBfY29tbWl0aW5mb19kYXRhLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vdHMvMDBfY29tbWl0aW5mb19kYXRhLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBOztHQUVHO0FBQ0gsTUFBTSxDQUFDLE1BQU0sVUFBVSxHQUFHO0lBQ3hCLElBQUksRUFBRSx3QkFBd0I7SUFDOUIsT0FBTyxFQUFFLFFBQVE7SUFDakIsV0FBVyxFQUFFLHFQQUFxUDtDQUNuUSxDQUFBIn0=
|
|
@@ -6,6 +6,9 @@ declare module 'net' {
|
|
|
6
6
|
getTLSVersion?(): string;
|
|
7
7
|
getPeerCertificate?(detailed?: boolean): any;
|
|
8
8
|
getSession?(): Buffer;
|
|
9
|
+
_connectionId?: string;
|
|
10
|
+
_remoteIP?: string;
|
|
11
|
+
_realRemoteIP?: string;
|
|
9
12
|
}
|
|
10
13
|
}
|
|
11
14
|
export declare function isTLSSocket(socket: plugins.net.Socket): boolean;
|
|
@@ -15,4 +15,4 @@ export function getTLSVersion(socket) {
|
|
|
15
15
|
}
|
|
16
16
|
return null;
|
|
17
17
|
}
|
|
18
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
18
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic29ja2V0LWF1Z21lbnRhdGlvbi5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3RzL2NvcmUvbW9kZWxzL3NvY2tldC1hdWdtZW50YXRpb24udHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxLQUFLLE9BQU8sTUFBTSxrQkFBa0IsQ0FBQztBQXNCNUMsaUVBQWlFO0FBQ2pFLE1BQU0sVUFBVSxXQUFXLENBQUMsTUFBMEI7SUFDcEQsT0FBTyxXQUFXLElBQUksTUFBTSxJQUFJLENBQUMsQ0FBQyxNQUFNLENBQUMsU0FBUyxDQUFDO0FBQ3JELENBQUM7QUFFRCwwREFBMEQ7QUFDMUQsTUFBTSxVQUFVLGFBQWEsQ0FBQyxNQUEwQjtJQUN0RCxJQUFJLE1BQU0sQ0FBQyxhQUFhLEVBQUUsQ0FBQztRQUN6QixJQUFJLENBQUM7WUFDSCxPQUFPLE1BQU0sQ0FBQyxhQUFhLEVBQUUsQ0FBQztRQUNoQyxDQUFDO1FBQUMsT0FBTyxDQUFDLEVBQUUsQ0FBQztZQUNYLE9BQU8sSUFBSSxDQUFDO1FBQ2QsQ0FBQztJQUNILENBQUM7SUFDRCxPQUFPLElBQUksQ0FBQztBQUNkLENBQUMifQ==
|
|
@@ -0,0 +1,16 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Socket Tracker Utility
|
|
3
|
+
* Provides standardized socket cleanup with proper listener and timer management
|
|
4
|
+
*/
|
|
5
|
+
import type { Socket } from 'net';
|
|
6
|
+
export type SocketTracked = {
|
|
7
|
+
cleanup: () => void;
|
|
8
|
+
addListener: <E extends string>(event: E, listener: (...args: any[]) => void) => void;
|
|
9
|
+
addTimer: (t: NodeJS.Timeout | null | undefined) => void;
|
|
10
|
+
safeDestroy: (reason?: Error) => void;
|
|
11
|
+
};
|
|
12
|
+
/**
|
|
13
|
+
* Create a socket tracker to manage listeners and timers
|
|
14
|
+
* Ensures proper cleanup and prevents memory leaks
|
|
15
|
+
*/
|
|
16
|
+
export declare function createSocketTracker(socket: Socket): SocketTracked;
|
|
@@ -0,0 +1,49 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Socket Tracker Utility
|
|
3
|
+
* Provides standardized socket cleanup with proper listener and timer management
|
|
4
|
+
*/
|
|
5
|
+
/**
|
|
6
|
+
* Create a socket tracker to manage listeners and timers
|
|
7
|
+
* Ensures proper cleanup and prevents memory leaks
|
|
8
|
+
*/
|
|
9
|
+
export function createSocketTracker(socket) {
|
|
10
|
+
const listeners = [];
|
|
11
|
+
const timers = [];
|
|
12
|
+
let cleaned = false;
|
|
13
|
+
const addListener = (event, listener) => {
|
|
14
|
+
socket.on(event, listener);
|
|
15
|
+
listeners.push({ event, listener });
|
|
16
|
+
};
|
|
17
|
+
const addTimer = (t) => {
|
|
18
|
+
if (!t)
|
|
19
|
+
return;
|
|
20
|
+
timers.push(t);
|
|
21
|
+
// Unref timer so it doesn't keep process alive
|
|
22
|
+
if (typeof t.unref === 'function') {
|
|
23
|
+
t.unref();
|
|
24
|
+
}
|
|
25
|
+
};
|
|
26
|
+
const cleanup = () => {
|
|
27
|
+
if (cleaned)
|
|
28
|
+
return;
|
|
29
|
+
cleaned = true;
|
|
30
|
+
// Clear all tracked timers
|
|
31
|
+
for (const t of timers) {
|
|
32
|
+
clearTimeout(t);
|
|
33
|
+
}
|
|
34
|
+
timers.length = 0;
|
|
35
|
+
// Remove all tracked listeners
|
|
36
|
+
for (const { event, listener } of listeners) {
|
|
37
|
+
socket.off(event, listener);
|
|
38
|
+
}
|
|
39
|
+
listeners.length = 0;
|
|
40
|
+
};
|
|
41
|
+
const safeDestroy = (reason) => {
|
|
42
|
+
cleanup();
|
|
43
|
+
if (!socket.destroyed) {
|
|
44
|
+
socket.destroy(reason);
|
|
45
|
+
}
|
|
46
|
+
};
|
|
47
|
+
return { cleanup, addListener, addTimer, safeDestroy };
|
|
48
|
+
}
|
|
49
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic29ja2V0LXRyYWNrZXIuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi90cy9jb3JlL3V0aWxzL3NvY2tldC10cmFja2VyLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBOzs7R0FHRztBQVdIOzs7R0FHRztBQUNILE1BQU0sVUFBVSxtQkFBbUIsQ0FBQyxNQUFjO0lBQ2hELE1BQU0sU0FBUyxHQUFpRSxFQUFFLENBQUM7SUFDbkYsTUFBTSxNQUFNLEdBQXFCLEVBQUUsQ0FBQztJQUNwQyxJQUFJLE9BQU8sR0FBRyxLQUFLLENBQUM7SUFFcEIsTUFBTSxXQUFXLEdBQUcsQ0FBQyxLQUFhLEVBQUUsUUFBa0MsRUFBRSxFQUFFO1FBQ3hFLE1BQU0sQ0FBQyxFQUFFLENBQUMsS0FBSyxFQUFFLFFBQVEsQ0FBQyxDQUFDO1FBQzNCLFNBQVMsQ0FBQyxJQUFJLENBQUMsRUFBRSxLQUFLLEVBQUUsUUFBUSxFQUFFLENBQUMsQ0FBQztJQUN0QyxDQUFDLENBQUM7SUFFRixNQUFNLFFBQVEsR0FBRyxDQUFDLENBQW9DLEVBQUUsRUFBRTtRQUN4RCxJQUFJLENBQUMsQ0FBQztZQUFFLE9BQU87UUFDZixNQUFNLENBQUMsSUFBSSxDQUFDLENBQUMsQ0FBQyxDQUFDO1FBQ2YsK0NBQStDO1FBQy9DLElBQUksT0FBTyxDQUFDLENBQUMsS0FBSyxLQUFLLFVBQVUsRUFBRSxDQUFDO1lBQ2xDLENBQUMsQ0FBQyxLQUFLLEVBQUUsQ0FBQztRQUNaLENBQUM7SUFDSCxDQUFDLENBQUM7SUFFRixNQUFNLE9BQU8sR0FBRyxHQUFHLEVBQUU7UUFDbkIsSUFBSSxPQUFPO1lBQUUsT0FBTztRQUNwQixPQUFPLEdBQUcsSUFBSSxDQUFDO1FBRWYsMkJBQTJCO1FBQzNCLEtBQUssTUFBTSxDQUFDLElBQUksTUFBTSxFQUFFLENBQUM7WUFDdkIsWUFBWSxDQUFDLENBQUMsQ0FBQyxDQUFDO1FBQ2xCLENBQUM7UUFDRCxNQUFNLENBQUMsTUFBTSxHQUFHLENBQUMsQ0FBQztRQUVsQiwrQkFBK0I7UUFDL0IsS0FBSyxNQUFNLEVBQUUsS0FBSyxFQUFFLFFBQVEsRUFBRSxJQUFJLFNBQVMsRUFBRSxDQUFDO1lBQzVDLE1BQU0sQ0FBQyxHQUFHLENBQUMsS0FBSyxFQUFFLFFBQVEsQ0FBQyxDQUFDO1FBQzlCLENBQUM7UUFDRCxTQUFTLENBQUMsTUFBTSxHQUFHLENBQUMsQ0FBQztJQUN2QixDQUFDLENBQUM7SUFFRixNQUFNLFdBQVcsR0FBRyxDQUFDLE1BQWMsRUFBRSxFQUFFO1FBQ3JDLE9BQU8sRUFBRSxDQUFDO1FBQ1YsSUFBSSxDQUFDLE1BQU0sQ0FBQyxTQUFTLEVBQUUsQ0FBQztZQUN0QixNQUFNLENBQUMsT0FBTyxDQUFDLE1BQU0sQ0FBQyxDQUFDO1FBQ3pCLENBQUM7SUFDSCxDQUFDLENBQUM7SUFFRixPQUFPLEVBQUUsT0FBTyxFQUFFLFdBQVcsRUFBRSxRQUFRLEVBQUUsV0FBVyxFQUFFLENBQUM7QUFDekQsQ0FBQyJ9
|
|
@@ -6,6 +6,7 @@
|
|
|
6
6
|
import { QuickProtocolDetector } from './quick-detector.js';
|
|
7
7
|
import { RoutingExtractor } from './routing-extractor.js';
|
|
8
8
|
import { DetectionFragmentManager } from '../utils/fragment-manager.js';
|
|
9
|
+
import { HttpParser } from '../../protocols/http/parser.js';
|
|
9
10
|
/**
|
|
10
11
|
* Simplified HTTP detector
|
|
11
12
|
*/
|
|
@@ -36,10 +37,23 @@ export class HttpDetector {
|
|
|
36
37
|
if (quickResult.protocol !== 'http' || quickResult.confidence < 50) {
|
|
37
38
|
return null;
|
|
38
39
|
}
|
|
40
|
+
// Check if we have complete headers first
|
|
41
|
+
const headersEnd = buffer.indexOf('\r\n\r\n');
|
|
42
|
+
const isComplete = headersEnd !== -1;
|
|
39
43
|
// Extract routing information
|
|
40
44
|
const routing = RoutingExtractor.extract(buffer, 'http');
|
|
41
|
-
//
|
|
42
|
-
|
|
45
|
+
// Extract headers if requested and we have complete headers
|
|
46
|
+
let headers;
|
|
47
|
+
if (options?.extractFullHeaders && isComplete) {
|
|
48
|
+
const headerSection = buffer.slice(0, headersEnd).toString();
|
|
49
|
+
const lines = headerSection.split('\r\n');
|
|
50
|
+
if (lines.length > 1) {
|
|
51
|
+
// Skip the request line and parse headers
|
|
52
|
+
headers = HttpParser.parseHeaders(lines.slice(1));
|
|
53
|
+
}
|
|
54
|
+
}
|
|
55
|
+
// If we don't need full headers and we have complete headers, we can return early
|
|
56
|
+
if (quickResult.confidence >= 95 && !options?.extractFullHeaders && isComplete) {
|
|
43
57
|
return {
|
|
44
58
|
protocol: 'http',
|
|
45
59
|
connectionInfo: {
|
|
@@ -51,16 +65,14 @@ export class HttpDetector {
|
|
|
51
65
|
isComplete: true
|
|
52
66
|
};
|
|
53
67
|
}
|
|
54
|
-
// Check if we have complete headers
|
|
55
|
-
const headersEnd = buffer.indexOf('\r\n\r\n');
|
|
56
|
-
const isComplete = headersEnd !== -1;
|
|
57
68
|
return {
|
|
58
69
|
protocol: 'http',
|
|
59
70
|
connectionInfo: {
|
|
60
71
|
protocol: 'http',
|
|
61
72
|
domain: routing?.domain,
|
|
62
73
|
path: routing?.path,
|
|
63
|
-
method: quickResult.metadata?.method
|
|
74
|
+
method: quickResult.metadata?.method,
|
|
75
|
+
headers: headers
|
|
64
76
|
},
|
|
65
77
|
isComplete,
|
|
66
78
|
bytesNeeded: isComplete ? undefined : buffer.length + 512 // Need more for headers
|
|
@@ -86,4 +98,4 @@ export class HttpDetector {
|
|
|
86
98
|
return detectResult;
|
|
87
99
|
}
|
|
88
100
|
}
|
|
89
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
101
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaHR0cC1kZXRlY3Rvci5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3RzL2RldGVjdGlvbi9kZXRlY3RvcnMvaHR0cC1kZXRlY3Rvci50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQTs7OztHQUlHO0FBTUgsT0FBTyxFQUFFLHFCQUFxQixFQUFFLE1BQU0scUJBQXFCLENBQUM7QUFDNUQsT0FBTyxFQUFFLGdCQUFnQixFQUFFLE1BQU0sd0JBQXdCLENBQUM7QUFDMUQsT0FBTyxFQUFFLHdCQUF3QixFQUFFLE1BQU0sOEJBQThCLENBQUM7QUFDeEUsT0FBTyxFQUFFLFVBQVUsRUFBRSxNQUFNLGdDQUFnQyxDQUFDO0FBRTVEOztHQUVHO0FBQ0gsTUFBTSxPQUFPLFlBQVk7SUFJdkIsWUFBWSxlQUEwQztRQUg5QyxrQkFBYSxHQUFHLElBQUkscUJBQXFCLEVBQUUsQ0FBQztRQUlsRCxJQUFJLENBQUMsZUFBZSxHQUFHLGVBQWUsSUFBSSxJQUFJLHdCQUF3QixFQUFFLENBQUM7SUFDM0UsQ0FBQztJQUVEOztPQUVHO0lBQ0gsU0FBUyxDQUFDLE1BQWM7UUFDdEIsTUFBTSxNQUFNLEdBQUcsSUFBSSxDQUFDLGFBQWEsQ0FBQyxXQUFXLENBQUMsTUFBTSxDQUFDLENBQUM7UUFDdEQsT0FBTyxNQUFNLENBQUMsUUFBUSxLQUFLLE1BQU0sSUFBSSxNQUFNLENBQUMsVUFBVSxHQUFHLEVBQUUsQ0FBQztJQUM5RCxDQUFDO0lBRUQ7O09BRUc7SUFDSCxlQUFlO1FBQ2IsT0FBTyxDQUFDLENBQUMsQ0FBQyxpQkFBaUI7SUFDN0IsQ0FBQztJQUVEOztPQUVHO0lBQ0gsTUFBTSxDQUFDLE1BQWMsRUFBRSxPQUEyQjtRQUNoRCx3QkFBd0I7UUFDeEIsTUFBTSxXQUFXLEdBQUcsSUFBSSxDQUFDLGFBQWEsQ0FBQyxXQUFXLENBQUMsTUFBTSxDQUFDLENBQUM7UUFFM0QsSUFBSSxXQUFXLENBQUMsUUFBUSxLQUFLLE1BQU0sSUFBSSxXQUFXLENB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
|
|
@@ -11,14 +11,6 @@ export declare class TlsDetector implements IProtocolDetector {
|
|
|
11
11
|
* Minimum bytes needed to identify TLS (record header)
|
|
12
12
|
*/
|
|
13
13
|
private static readonly MIN_TLS_HEADER_SIZE;
|
|
14
|
-
/**
|
|
15
|
-
* Fragment tracking for incomplete handshakes
|
|
16
|
-
*/
|
|
17
|
-
private static fragmentedBuffers;
|
|
18
|
-
/**
|
|
19
|
-
* Create connection ID from context
|
|
20
|
-
*/
|
|
21
|
-
private createConnectionId;
|
|
22
14
|
/**
|
|
23
15
|
* Detect TLS protocol from buffer
|
|
24
16
|
*/
|
|
@@ -46,7 +38,7 @@ export declare class TlsDetector implements IProtocolDetector {
|
|
|
46
38
|
/**
|
|
47
39
|
* Detect with context for fragmented data
|
|
48
40
|
*/
|
|
49
|
-
detectWithContext(buffer: Buffer,
|
|
41
|
+
detectWithContext(buffer: Buffer, _context: {
|
|
50
42
|
sourceIp?: string;
|
|
51
43
|
sourcePort?: number;
|
|
52
44
|
destIp?: string;
|
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
/**
|
|
2
2
|
* TLS protocol detector
|
|
3
3
|
*/
|
|
4
|
-
import { readUInt16BE
|
|
4
|
+
import { readUInt16BE } from '../utils/buffer-utils.js';
|
|
5
5
|
import { tlsVersionToString } from '../utils/parser-utils.js';
|
|
6
6
|
// Import from protocols
|
|
7
7
|
import { TlsRecordType, TlsHandshakeType, TlsExtensionType } from '../../protocols/tls/index.js';
|
|
@@ -16,16 +16,6 @@ export class TlsDetector {
|
|
|
16
16
|
* Minimum bytes needed to identify TLS (record header)
|
|
17
17
|
*/
|
|
18
18
|
static { this.MIN_TLS_HEADER_SIZE = 5; }
|
|
19
|
-
/**
|
|
20
|
-
* Fragment tracking for incomplete handshakes
|
|
21
|
-
*/
|
|
22
|
-
static { this.fragmentedBuffers = new Map(); }
|
|
23
|
-
/**
|
|
24
|
-
* Create connection ID from context
|
|
25
|
-
*/
|
|
26
|
-
createConnectionId(context) {
|
|
27
|
-
return `${context.sourceIp || 'unknown'}:${context.sourcePort || 0}->${context.destIp || 'unknown'}:${context.destPort || 0}`;
|
|
28
|
-
}
|
|
29
19
|
/**
|
|
30
20
|
* Detect TLS protocol from buffer
|
|
31
21
|
*/
|
|
@@ -184,23 +174,10 @@ export class TlsDetector {
|
|
|
184
174
|
/**
|
|
185
175
|
* Detect with context for fragmented data
|
|
186
176
|
*/
|
|
187
|
-
detectWithContext(buffer,
|
|
188
|
-
|
|
189
|
-
//
|
|
190
|
-
|
|
191
|
-
if (!accumulator) {
|
|
192
|
-
accumulator = new BufferAccumulator();
|
|
193
|
-
TlsDetector.fragmentedBuffers.set(connectionId, accumulator);
|
|
194
|
-
}
|
|
195
|
-
// Add new data
|
|
196
|
-
accumulator.append(buffer);
|
|
197
|
-
// Try detection on accumulated data
|
|
198
|
-
const result = this.detect(accumulator.getBuffer(), options);
|
|
199
|
-
// If detection is complete or we have too much data, clean up
|
|
200
|
-
if (result?.isComplete || accumulator.length() > 65536) {
|
|
201
|
-
TlsDetector.fragmentedBuffers.delete(connectionId);
|
|
202
|
-
}
|
|
203
|
-
return result;
|
|
177
|
+
detectWithContext(buffer, _context, options) {
|
|
178
|
+
// This method is deprecated - TLS detection should use the fragment manager
|
|
179
|
+
// from the parent detector system, not maintain its own fragments
|
|
180
|
+
return this.detect(buffer, options);
|
|
204
181
|
}
|
|
205
182
|
}
|
|
206
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
183
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidGxzLWRldGVjdG9yLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vdHMvZGV0ZWN0aW9uL2RldGVjdG9ycy90bHMtZGV0ZWN0b3IudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUE7O0dBRUc7QUFLSCxPQUFPLEVBQUUsWUFBWSxFQUFFLE1BQU0sMEJBQTBCLENBQUM7QUFDeEQsT0FBTyxFQUFFLGtCQUFrQixFQUFFLE1BQU0sMEJBQTBCLENBQUM7QUFFOUQsd0JBQXdCO0FBQ3hCLE9BQU8sRUFBRSxhQUFhLEVBQUUsZ0JBQWdCLEVBQUUsZ0JBQWdCLEVBQUUsTUFBTSw4QkFBOEIsQ0FBQztBQUVqRyx5REFBeUQ7QUFDekQsT0FBTyxFQUFFLGFBQWEsRUFBRSxNQUFNLDJDQUEyQyxDQUFDO0FBQzFFLE9BQU8sRUFBRSxpQkFBaUIsRUFBRSxNQUFNLGdEQUFnRCxDQUFDO0FBRW5GOztHQUVHO0FBQ0gsTUFBTSxPQUFPLFdBQVc7SUFDdEI7O09BRUc7YUFDcUIsd0JBQW1CLEdBQUcsQ0FBQyxDQUFDO0lBR2hEOztPQUVHO0lBQ0gsTUFBTSxDQUFDLE1BQWMsRUFBRSxPQUEyQjtRQUNoRCwrQkFBK0I7UUFDL0IsSUFBSSxNQUFNLENBQUMsTUFBTSxHQUFHLFdBQVcsQ0FBQyxtQkFBbUIsRUFBRSxDQUFDO1lBQ3BELE9BQU8sSUFBSSxDQUFDO1FBQ2QsQ0FBQztRQUVELGdDQUFnQztRQUNoQyxJQUFJLENBQUMsSUFBSSxDQUFDLFdBQVcsQ0FBQyxNQUFNLENBQUMsRUFBRSxDQUFDO1lBQzlCLE9BQU8sSUFBSSxDQUFDO1FBQ2QsQ0FBQztRQUVELHlCQUF5QjtRQUN6QixNQUFNLFVBQVUsR0FBRyxNQUFNLENBQUMsQ0FBQyxDQUFDLENBQUM7UUFDN0IsTUFBTSxRQUFRLEdBQUcsTUFBTSxDQUFDLENBQUMsQ0FBQyxDQUFDO1FBQzNCLE1BQU0sUUFBUSxHQUFHLE1BQU0sQ0FBQyxDQUFDLENBQUMsQ0FBQztRQUMzQixNQUFNLFlBQVks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
|
|
@@ -13,6 +13,7 @@ export declare class ProtocolDetector {
|
|
|
13
13
|
private fragmentManager;
|
|
14
14
|
private tlsDetector;
|
|
15
15
|
private httpDetector;
|
|
16
|
+
private connectionProtocols;
|
|
16
17
|
constructor();
|
|
17
18
|
private static getInstance;
|
|
18
19
|
/**
|
|
@@ -43,9 +44,13 @@ export declare class ProtocolDetector {
|
|
|
43
44
|
/**
|
|
44
45
|
* Clean up old connection tracking entries
|
|
45
46
|
*
|
|
46
|
-
* @param
|
|
47
|
+
* @param _maxAge Maximum age in milliseconds (default: 30 seconds)
|
|
47
48
|
*/
|
|
48
|
-
static cleanupConnections(
|
|
49
|
+
static cleanupConnections(_maxAge?: number): void;
|
|
50
|
+
/**
|
|
51
|
+
* Clean up fragments for a specific connection
|
|
52
|
+
*/
|
|
53
|
+
static cleanupConnection(context: IConnectionContext): void;
|
|
49
54
|
/**
|
|
50
55
|
* Extract domain from connection info
|
|
51
56
|
*/
|
|
@@ -11,6 +11,7 @@ import { DetectionFragmentManager } from './utils/fragment-manager.js';
|
|
|
11
11
|
*/
|
|
12
12
|
export class ProtocolDetector {
|
|
13
13
|
constructor() {
|
|
14
|
+
this.connectionProtocols = new Map();
|
|
14
15
|
this.fragmentManager = new DetectionFragmentManager();
|
|
15
16
|
this.tlsDetector = new TlsDetector();
|
|
16
17
|
this.httpDetector = new HttpDetector(this.fragmentManager);
|
|
@@ -88,19 +89,77 @@ export class ProtocolDetector {
|
|
|
88
89
|
isComplete: true
|
|
89
90
|
};
|
|
90
91
|
}
|
|
91
|
-
|
|
92
|
-
if
|
|
93
|
-
|
|
92
|
+
const connectionId = DetectionFragmentManager.createConnectionId(context);
|
|
93
|
+
// Check if we already know the protocol for this connection
|
|
94
|
+
const knownProtocol = this.connectionProtocols.get(connectionId);
|
|
95
|
+
if (knownProtocol === 'http') {
|
|
96
|
+
const result = this.httpDetector.detectWithContext(buffer, context, options);
|
|
94
97
|
if (result) {
|
|
98
|
+
if (result.isComplete) {
|
|
99
|
+
this.connectionProtocols.delete(connectionId);
|
|
100
|
+
}
|
|
95
101
|
return result;
|
|
96
102
|
}
|
|
97
103
|
}
|
|
98
|
-
if (
|
|
99
|
-
|
|
104
|
+
else if (knownProtocol === 'tls') {
|
|
105
|
+
// Handle TLS with fragment accumulation
|
|
106
|
+
const handler = this.fragmentManager.getHandler('tls');
|
|
107
|
+
const fragmentResult = handler.addFragment(connectionId, buffer);
|
|
108
|
+
if (fragmentResult.error) {
|
|
109
|
+
handler.complete(connectionId);
|
|
110
|
+
this.connectionProtocols.delete(connectionId);
|
|
111
|
+
return {
|
|
112
|
+
protocol: 'unknown',
|
|
113
|
+
connectionInfo: { protocol: 'unknown' },
|
|
114
|
+
isComplete: true
|
|
115
|
+
};
|
|
116
|
+
}
|
|
117
|
+
const result = this.tlsDetector.detect(fragmentResult.buffer, options);
|
|
100
118
|
if (result) {
|
|
119
|
+
if (result.isComplete) {
|
|
120
|
+
handler.complete(connectionId);
|
|
121
|
+
this.connectionProtocols.delete(connectionId);
|
|
122
|
+
}
|
|
101
123
|
return result;
|
|
102
124
|
}
|
|
103
125
|
}
|
|
126
|
+
// If we don't know the protocol yet, try to detect it
|
|
127
|
+
if (!knownProtocol) {
|
|
128
|
+
// First peek to determine protocol type
|
|
129
|
+
if (this.tlsDetector.canHandle(buffer)) {
|
|
130
|
+
this.connectionProtocols.set(connectionId, 'tls');
|
|
131
|
+
// Handle TLS with fragment accumulation
|
|
132
|
+
const handler = this.fragmentManager.getHandler('tls');
|
|
133
|
+
const fragmentResult = handler.addFragment(connectionId, buffer);
|
|
134
|
+
if (fragmentResult.error) {
|
|
135
|
+
handler.complete(connectionId);
|
|
136
|
+
this.connectionProtocols.delete(connectionId);
|
|
137
|
+
return {
|
|
138
|
+
protocol: 'unknown',
|
|
139
|
+
connectionInfo: { protocol: 'unknown' },
|
|
140
|
+
isComplete: true
|
|
141
|
+
};
|
|
142
|
+
}
|
|
143
|
+
const result = this.tlsDetector.detect(fragmentResult.buffer, options);
|
|
144
|
+
if (result) {
|
|
145
|
+
if (result.isComplete) {
|
|
146
|
+
handler.complete(connectionId);
|
|
147
|
+
this.connectionProtocols.delete(connectionId);
|
|
148
|
+
}
|
|
149
|
+
return result;
|
|
150
|
+
}
|
|
151
|
+
}
|
|
152
|
+
if (this.httpDetector.canHandle(buffer)) {
|
|
153
|
+
this.connectionProtocols.set(connectionId, 'http');
|
|
154
|
+
const result = this.httpDetector.detectWithContext(buffer, context, options);
|
|
155
|
+
if (result) {
|
|
156
|
+
if (result.isComplete) {
|
|
157
|
+
this.connectionProtocols.delete(connectionId);
|
|
158
|
+
}
|
|
159
|
+
return result;
|
|
160
|
+
}
|
|
161
|
+
}
|
|
162
|
+
}
|
|
104
163
|
// Can't determine protocol
|
|
105
164
|
return {
|
|
106
165
|
protocol: 'unknown',
|
|
@@ -127,16 +186,29 @@ export class ProtocolDetector {
|
|
|
127
186
|
}
|
|
128
187
|
destroyInstance() {
|
|
129
188
|
this.fragmentManager.destroy();
|
|
189
|
+
this.connectionProtocols.clear();
|
|
130
190
|
}
|
|
131
191
|
/**
|
|
132
192
|
* Clean up old connection tracking entries
|
|
133
193
|
*
|
|
134
|
-
* @param
|
|
194
|
+
* @param _maxAge Maximum age in milliseconds (default: 30 seconds)
|
|
135
195
|
*/
|
|
136
|
-
static cleanupConnections(
|
|
196
|
+
static cleanupConnections(_maxAge = 30000) {
|
|
137
197
|
// Cleanup is now handled internally by the fragment manager
|
|
138
198
|
this.getInstance().fragmentManager.cleanup();
|
|
139
199
|
}
|
|
200
|
+
/**
|
|
201
|
+
* Clean up fragments for a specific connection
|
|
202
|
+
*/
|
|
203
|
+
static cleanupConnection(context) {
|
|
204
|
+
const instance = this.getInstance();
|
|
205
|
+
const connectionId = DetectionFragmentManager.createConnectionId(context);
|
|
206
|
+
// Clean up both TLS and HTTP fragments for this connection
|
|
207
|
+
instance.fragmentManager.getHandler('tls').complete(connectionId);
|
|
208
|
+
instance.fragmentManager.getHandler('http').complete(connectionId);
|
|
209
|
+
// Remove from connection protocols tracking
|
|
210
|
+
instance.connectionProtocols.delete(connectionId);
|
|
211
|
+
}
|
|
140
212
|
/**
|
|
141
213
|
* Extract domain from connection info
|
|
142
214
|
*/
|
|
@@ -170,4 +242,4 @@ export class ProtocolDetector {
|
|
|
170
242
|
};
|
|
171
243
|
}
|
|
172
244
|
}
|
|
173
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
245
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"protocol-detector.js","sourceRoot":"","sources":["../../ts/detection/protocol-detector.ts"],"names":[],"mappings":"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"}
|
|
@@ -11,7 +11,7 @@ export declare class HttpProxy implements IMetricsTracker {
|
|
|
11
11
|
toJSON(): any;
|
|
12
12
|
options: IHttpProxyOptions;
|
|
13
13
|
routes: IRouteConfig[];
|
|
14
|
-
httpsServer:
|
|
14
|
+
httpsServer: plugins.http2.Http2SecureServer;
|
|
15
15
|
private certificateManager;
|
|
16
16
|
private connectionPool;
|
|
17
17
|
private requestHandler;
|
|
@@ -64,6 +64,10 @@ export declare class HttpProxy implements IMetricsTracker {
|
|
|
64
64
|
* Starts the proxy server
|
|
65
65
|
*/
|
|
66
66
|
start(): Promise<void>;
|
|
67
|
+
/**
|
|
68
|
+
* Check if an address is a loopback address (IPv4 or IPv6)
|
|
69
|
+
*/
|
|
70
|
+
private isLoopback;
|
|
67
71
|
/**
|
|
68
72
|
* Sets up tracking of TCP connections
|
|
69
73
|
*/
|