@undefineds.co/xpod 0.3.53 → 0.3.54
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/config/cli.json +72 -0
- package/config/components-ignore.json +21 -0
- package/config/extensions.local.initializer.json +77 -8
- package/config/resolver.json +72 -4
- package/dist/api/ApiServer.d.ts +12 -0
- package/dist/api/ApiServer.js +14 -3
- package/dist/api/ApiServer.js.map +1 -1
- package/dist/api/auth/NodeTokenAuthenticator.d.ts +0 -8
- package/dist/api/auth/NodeTokenAuthenticator.js +3 -46
- package/dist/api/auth/NodeTokenAuthenticator.js.map +1 -1
- package/dist/api/container/local.js +5 -36
- package/dist/api/container/local.js.map +1 -1
- package/dist/api/container/routes.js +6 -0
- package/dist/api/container/routes.js.map +1 -1
- package/dist/api/handlers/EdgeNodeSignalHandler.js +25 -3
- package/dist/api/handlers/EdgeNodeSignalHandler.js.map +1 -1
- package/dist/api/handlers/ReachabilityHandler.d.ts +13 -0
- package/dist/api/handlers/ReachabilityHandler.js +388 -0
- package/dist/api/handlers/ReachabilityHandler.js.map +1 -0
- package/dist/api/runs/InngestRunExecutionBackend.d.ts +2 -2
- package/dist/api/tasks/InngestTaskScheduler.d.ts +4 -4
- package/dist/components/components.jsonld +14 -2
- package/dist/components/context.jsonld +439 -3
- package/dist/edge/EdgeNodeAgent.d.ts +43 -0
- package/dist/edge/EdgeNodeAgent.js +208 -1
- package/dist/edge/EdgeNodeAgent.js.map +1 -1
- package/dist/edge/EdgeNodeAgent.jsonld +166 -0
- package/dist/edge/EdgeNodeAgentInitializer.d.ts +20 -2
- package/dist/edge/EdgeNodeAgentInitializer.js +53 -2
- package/dist/edge/EdgeNodeAgentInitializer.js.map +1 -1
- package/dist/edge/EdgeNodeAgentInitializer.jsonld +409 -0
- package/dist/edge/reachability/CanonicalFetch.d.ts +7 -0
- package/dist/edge/reachability/CanonicalFetch.js +49 -0
- package/dist/edge/reachability/CanonicalFetch.js.map +1 -0
- package/dist/edge/reachability/CanonicalFetch.jsonld +25 -0
- package/dist/edge/reachability/ManagedClientFetch.d.ts +26 -0
- package/dist/edge/reachability/ManagedClientFetch.js +155 -0
- package/dist/edge/reachability/ManagedClientFetch.js.map +1 -0
- package/dist/edge/reachability/ManagedClientFetch.jsonld +83 -0
- package/dist/edge/reachability/ManagedClientP2PSmoke.d.ts +16 -0
- package/dist/edge/reachability/ManagedClientP2PSmoke.js +31 -0
- package/dist/edge/reachability/ManagedClientP2PSmoke.js.map +1 -0
- package/dist/edge/reachability/ManagedClientP2PSmoke.jsonld +65 -0
- package/dist/edge/reachability/ManagedRouteSelector.d.ts +7 -0
- package/dist/edge/reachability/ManagedRouteSelector.js +43 -0
- package/dist/edge/reachability/ManagedRouteSelector.js.map +1 -0
- package/dist/edge/reachability/ManagedRouteSelector.jsonld +29 -0
- package/dist/edge/reachability/P2PDataPlane.d.ts +37 -0
- package/dist/edge/reachability/P2PDataPlane.js +160 -0
- package/dist/edge/reachability/P2PDataPlane.js.map +1 -0
- package/dist/edge/reachability/P2PDataPlane.jsonld +134 -0
- package/dist/edge/reachability/P2PRealnetAcceptance.d.ts +74 -0
- package/dist/edge/reachability/P2PRealnetAcceptance.js +240 -0
- package/dist/edge/reachability/P2PRealnetAcceptance.js.map +1 -0
- package/dist/edge/reachability/P2PRealnetAcceptance.jsonld +283 -0
- package/dist/edge/reachability/P2PSignalingClient.d.ts +24 -0
- package/dist/edge/reachability/P2PSignalingClient.js +138 -0
- package/dist/edge/reachability/P2PSignalingClient.js.map +1 -0
- package/dist/edge/reachability/P2PSignalingClient.jsonld +79 -0
- package/dist/edge/reachability/ReachabilitySessionService.d.ts +55 -0
- package/dist/edge/reachability/ReachabilitySessionService.js +439 -0
- package/dist/edge/reachability/ReachabilitySessionService.js.map +1 -0
- package/dist/edge/reachability/ReachabilitySessionService.jsonld +196 -0
- package/dist/edge/reachability/RouteSetBuilder.d.ts +2 -0
- package/dist/edge/reachability/RouteSetBuilder.js +205 -0
- package/dist/edge/reachability/RouteSetBuilder.js.map +1 -0
- package/dist/edge/reachability/TcpP2PDataPlaneTransport.d.ts +47 -0
- package/dist/edge/reachability/TcpP2PDataPlaneTransport.js +281 -0
- package/dist/edge/reachability/TcpP2PDataPlaneTransport.js.map +1 -0
- package/dist/edge/reachability/TcpP2PDataPlaneTransport.jsonld +183 -0
- package/dist/edge/reachability/TcpP2PSignalingSession.d.ts +149 -0
- package/dist/edge/reachability/TcpP2PSignalingSession.js +699 -0
- package/dist/edge/reachability/TcpP2PSignalingSession.js.map +1 -0
- package/dist/edge/reachability/TcpP2PSignalingSession.jsonld +474 -0
- package/dist/edge/reachability/index.d.ts +12 -0
- package/dist/edge/reachability/index.js +29 -0
- package/dist/edge/reachability/index.js.map +1 -0
- package/dist/edge/reachability/types.d.ts +114 -0
- package/dist/edge/reachability/types.js +3 -0
- package/dist/edge/reachability/types.js.map +1 -0
- package/dist/edge/reachability/types.jsonld +457 -0
- package/dist/http/EdgeNodeProxyHttpHandler.d.ts +2 -0
- package/dist/http/EdgeNodeProxyHttpHandler.js +19 -1
- package/dist/http/EdgeNodeProxyHttpHandler.js.map +1 -1
- package/dist/http/EdgeNodeProxyHttpHandler.jsonld +8 -0
- package/dist/identity/drizzle/EdgeNodeRepository.js +1 -1
- package/dist/identity/drizzle/EdgeNodeRepository.js.map +1 -1
- package/dist/index.d.ts +4 -1
- package/dist/index.js +5 -2
- package/dist/index.js.map +1 -1
- package/dist/runtime/bootstrap.js +8 -0
- package/dist/runtime/bootstrap.js.map +1 -1
- package/dist/service/EdgeNodeSignalClient.js +5 -1
- package/dist/service/EdgeNodeSignalClient.js.map +1 -1
- package/dist/storage/rdf/PostgresRdfEngine.d.ts +1 -0
- package/dist/storage/rdf/PostgresRdfEngine.js +53 -37
- package/dist/storage/rdf/PostgresRdfEngine.js.map +1 -1
- package/dist/storage/rdf/PostgresRdfEngine.jsonld +4 -0
- package/dist/test-utils/index.d.ts +2 -0
- package/dist/test-utils/index.js +3 -1
- package/dist/test-utils/index.js.map +1 -1
- package/dist/test-utils/local-managed-client-p2p-e2e-smoke.d.ts +63 -0
- package/dist/test-utils/local-managed-client-p2p-e2e-smoke.js +478 -0
- package/dist/test-utils/local-managed-client-p2p-e2e-smoke.js.map +1 -0
- package/package.json +11 -4
- package/static/app/assets/_commonjsHelpers-B-UnjaXt.js +1 -0
- package/static/app/assets/index-AaQ1qxhy.js +171 -0
- package/static/app/assets/inrupt-smoke.js +131 -0
- package/static/app/assets/main.css +1 -0
- package/static/app/assets/main.js +6 -6
- package/static/app/index.html +2 -1
- package/static/app/inrupt-smoke.html +14 -0
- package/static/app/reachability.html +221 -0
- package/static/app/reachability.svg +7 -0
- package/static/app/reachability.webmanifest +18 -0
- package/static/app/signal-pod.html +293 -0
- package/static/app/assets/index.css +0 -1
|
@@ -12,10 +12,16 @@ const FrpcProcessManager_1 = require("./frp/FrpcProcessManager");
|
|
|
12
12
|
const AcmeCertificateManager_1 = require("./acme/AcmeCertificateManager");
|
|
13
13
|
const ClusterCertificateManager_1 = require("./acme/ClusterCertificateManager");
|
|
14
14
|
const EdgeNodeCapabilityDetector_1 = require("./EdgeNodeCapabilityDetector");
|
|
15
|
+
const reachability_1 = require("./reachability");
|
|
16
|
+
const DEFAULT_P2P_ACCEPT_INTERVAL_MS = 1_000;
|
|
15
17
|
class EdgeNodeAgent {
|
|
16
18
|
constructor() {
|
|
17
19
|
this.logger = (0, global_logger_factory_1.getLoggerFor)(this);
|
|
18
20
|
this.lastNetworkDetection = 0;
|
|
21
|
+
this.p2pAcceptRunning = false;
|
|
22
|
+
this.p2pAcceptGeneration = 0;
|
|
23
|
+
this.p2pAcceptedSessionIds = new Set();
|
|
24
|
+
this.p2pSocketHandles = new Set();
|
|
19
25
|
this.networkDetectionIntervalMs = 60_000; // 每分钟重新检测一次
|
|
20
26
|
}
|
|
21
27
|
async start(options) {
|
|
@@ -51,7 +57,7 @@ class EdgeNodeAgent {
|
|
|
51
57
|
}
|
|
52
58
|
const systemMetrics = options.includeSystemMetrics ? this.collectSystemMetrics() : undefined;
|
|
53
59
|
const metadataPayload = {
|
|
54
|
-
...(options
|
|
60
|
+
...this.buildHeartbeatMetadata(options),
|
|
55
61
|
system: systemMetrics,
|
|
56
62
|
};
|
|
57
63
|
const certificatePayload = this.clusterCertificate?.getHeartbeatPayload();
|
|
@@ -77,15 +83,118 @@ class EdgeNodeAgent {
|
|
|
77
83
|
heartbeatOptions.tunnelSupplier = () => this.buildTunnelHeartbeatPayload();
|
|
78
84
|
}
|
|
79
85
|
this.heartbeat = new EdgeNodeSignalClient_1.EdgeNodeSignalClient(heartbeatOptions);
|
|
86
|
+
this.startP2PAcceptLoop(options);
|
|
80
87
|
}
|
|
81
88
|
stop() {
|
|
82
89
|
if (this.heartbeat && typeof this.heartbeat.dispose === 'function') {
|
|
83
90
|
this.heartbeat.dispose();
|
|
84
91
|
}
|
|
85
92
|
this.heartbeat = undefined;
|
|
93
|
+
if (this.p2pAcceptInterval) {
|
|
94
|
+
clearInterval(this.p2pAcceptInterval);
|
|
95
|
+
this.p2pAcceptInterval = undefined;
|
|
96
|
+
}
|
|
97
|
+
this.p2pAcceptGeneration += 1;
|
|
98
|
+
for (const handle of this.p2pSocketHandles) {
|
|
99
|
+
handle.close();
|
|
100
|
+
}
|
|
101
|
+
this.p2pSocketHandles.clear();
|
|
102
|
+
this.p2pAcceptedSessionIds.clear();
|
|
103
|
+
this.p2pAcceptRunning = false;
|
|
86
104
|
void this.frpManager?.stop();
|
|
87
105
|
this.clusterCertificate?.stop();
|
|
88
106
|
}
|
|
107
|
+
startP2PAcceptLoop(options) {
|
|
108
|
+
const p2p = options.p2p;
|
|
109
|
+
if (!p2p || this.normalizeBoolean(p2p.enabled) === false) {
|
|
110
|
+
return;
|
|
111
|
+
}
|
|
112
|
+
const signaling = p2p.signaling ?? (0, reachability_1.createP2PSignalingClient)({
|
|
113
|
+
apiBaseUrl: this.resolveP2PApiBaseUrl(options.signalEndpoint),
|
|
114
|
+
nodeId: options.nodeId,
|
|
115
|
+
token: options.nodeToken,
|
|
116
|
+
});
|
|
117
|
+
const handler = (0, reachability_1.createP2PDataPlaneHandler)({ targetBaseUrl: p2p.targetBaseUrl });
|
|
118
|
+
const intervalMs = this.normalizePositiveInteger(p2p.acceptIntervalMs) ?? DEFAULT_P2P_ACCEPT_INTERVAL_MS;
|
|
119
|
+
this.p2pAcceptGeneration += 1;
|
|
120
|
+
const generation = this.p2pAcceptGeneration;
|
|
121
|
+
const run = () => {
|
|
122
|
+
void this.acceptP2PConnectionOnce({
|
|
123
|
+
generation,
|
|
124
|
+
signaling,
|
|
125
|
+
sourceId: options.nodeId,
|
|
126
|
+
host: p2p.host,
|
|
127
|
+
address: p2p.address,
|
|
128
|
+
handler,
|
|
129
|
+
connectTimeoutMs: this.normalizePositiveInteger(p2p.connectTimeoutMs),
|
|
130
|
+
winnerSelectionWindowMs: this.normalizeNonNegativeInteger(p2p.winnerSelectionWindowMs),
|
|
131
|
+
localAddress: p2p.localAddress,
|
|
132
|
+
sleepMs: p2p.sleepMs,
|
|
133
|
+
connectSocket: p2p.connectSocket,
|
|
134
|
+
onP2PAccept: p2p.onP2PAccept,
|
|
135
|
+
});
|
|
136
|
+
};
|
|
137
|
+
run();
|
|
138
|
+
this.p2pAcceptInterval = setInterval(run, intervalMs);
|
|
139
|
+
}
|
|
140
|
+
async acceptP2PConnectionOnce(options) {
|
|
141
|
+
if (this.p2pAcceptRunning) {
|
|
142
|
+
return;
|
|
143
|
+
}
|
|
144
|
+
this.p2pAcceptRunning = true;
|
|
145
|
+
try {
|
|
146
|
+
const { generation, onP2PAccept, ...acceptOptions } = options;
|
|
147
|
+
const accepted = await (0, reachability_1.acceptSignaledRawTcpP2PConnectionOnce)({
|
|
148
|
+
...acceptOptions,
|
|
149
|
+
signaling: this.skipAcceptedP2PSessions(acceptOptions.signaling),
|
|
150
|
+
host: acceptOptions.host,
|
|
151
|
+
});
|
|
152
|
+
if (accepted) {
|
|
153
|
+
if (generation !== this.p2pAcceptGeneration) {
|
|
154
|
+
accepted.socketHandle.close();
|
|
155
|
+
return;
|
|
156
|
+
}
|
|
157
|
+
this.p2pAcceptedSessionIds.add(accepted.session.sessionId);
|
|
158
|
+
this.p2pSocketHandles.add(accepted.socketHandle);
|
|
159
|
+
onP2PAccept?.({
|
|
160
|
+
sessionId: accepted.session.sessionId,
|
|
161
|
+
nodeId: accepted.session.nodeId,
|
|
162
|
+
clientId: accepted.session.clientId,
|
|
163
|
+
localCandidateCount: accepted.localCandidates.length,
|
|
164
|
+
remoteCandidateCount: accepted.remoteCandidates.length,
|
|
165
|
+
nodeAddress: addressEvidenceFromCandidates(accepted.localCandidates),
|
|
166
|
+
clientAddress: addressEvidenceFromCandidates(accepted.remoteCandidates),
|
|
167
|
+
acceptedAt: new Date().toISOString(),
|
|
168
|
+
});
|
|
169
|
+
this.logger.info(`Accepted raw TCP P2P session ${accepted.session.sessionId}.`);
|
|
170
|
+
}
|
|
171
|
+
}
|
|
172
|
+
catch (error) {
|
|
173
|
+
this.logger.debug(`Raw TCP P2P accept attempt failed: ${error.message}`);
|
|
174
|
+
}
|
|
175
|
+
finally {
|
|
176
|
+
this.p2pAcceptRunning = false;
|
|
177
|
+
}
|
|
178
|
+
}
|
|
179
|
+
skipAcceptedP2PSessions(signaling) {
|
|
180
|
+
return {
|
|
181
|
+
createP2PSession: (request) => signaling.createP2PSession(request),
|
|
182
|
+
getP2PSession: (sessionIdOrUrl) => signaling.getP2PSession(sessionIdOrUrl),
|
|
183
|
+
addP2PCandidates: (sessionIdOrUrl, request) => signaling.addP2PCandidates(sessionIdOrUrl, request),
|
|
184
|
+
listP2PSessions: async () => {
|
|
185
|
+
const sessions = await signaling.listP2PSessions();
|
|
186
|
+
return sessions.filter((session) => !this.p2pAcceptedSessionIds.has(session.sessionId));
|
|
187
|
+
},
|
|
188
|
+
};
|
|
189
|
+
}
|
|
190
|
+
resolveP2PApiBaseUrl(signalEndpoint) {
|
|
191
|
+
try {
|
|
192
|
+
return new URL(signalEndpoint).origin;
|
|
193
|
+
}
|
|
194
|
+
catch {
|
|
195
|
+
return signalEndpoint;
|
|
196
|
+
}
|
|
197
|
+
}
|
|
89
198
|
stringifyIfContent(data) {
|
|
90
199
|
const sanitized = {};
|
|
91
200
|
for (const [key, value] of Object.entries(data)) {
|
|
@@ -110,6 +219,81 @@ class EdgeNodeAgent {
|
|
|
110
219
|
freeMem: node_os_1.default.freemem(),
|
|
111
220
|
};
|
|
112
221
|
}
|
|
222
|
+
buildHeartbeatMetadata(options) {
|
|
223
|
+
const metadata = { ...(options.metadata ?? {}) };
|
|
224
|
+
const p2pRoute = this.buildP2PHeartbeatRoute(options);
|
|
225
|
+
if (p2pRoute) {
|
|
226
|
+
metadata.routes = this.mergeHeartbeatRoutes(metadata.routes, p2pRoute);
|
|
227
|
+
}
|
|
228
|
+
return metadata;
|
|
229
|
+
}
|
|
230
|
+
buildP2PHeartbeatRoute(options) {
|
|
231
|
+
const p2p = options.p2p;
|
|
232
|
+
if (!p2p || this.normalizeBoolean(p2p.enabled) === false) {
|
|
233
|
+
return undefined;
|
|
234
|
+
}
|
|
235
|
+
const label = typeof p2p.label === 'string' && p2p.label.length > 0 ? p2p.label : undefined;
|
|
236
|
+
return {
|
|
237
|
+
id: 'p2p-raw-tcp',
|
|
238
|
+
nodeId: options.nodeId,
|
|
239
|
+
...(options.baseUrl ? { canonicalUrl: options.baseUrl } : {}),
|
|
240
|
+
kind: 'p2p',
|
|
241
|
+
targetUrl: `tcp-punch://node/${encodeURIComponent(options.nodeId)}`,
|
|
242
|
+
priority: 40,
|
|
243
|
+
requiresManagedClient: true,
|
|
244
|
+
visibility: 'authorized-client',
|
|
245
|
+
health: 'healthy',
|
|
246
|
+
metadata: {
|
|
247
|
+
protocols: {
|
|
248
|
+
'raw-tcp-hole-punch': {
|
|
249
|
+
enabled: true,
|
|
250
|
+
...(label ? { label } : {}),
|
|
251
|
+
},
|
|
252
|
+
},
|
|
253
|
+
},
|
|
254
|
+
};
|
|
255
|
+
}
|
|
256
|
+
mergeHeartbeatRoutes(existing, p2pRoute) {
|
|
257
|
+
const routes = Array.isArray(existing)
|
|
258
|
+
? existing.filter((entry) => Boolean(entry) && typeof entry === 'object')
|
|
259
|
+
: [];
|
|
260
|
+
const withoutGeneratedRoute = routes.filter((route) => route.id !== p2pRoute.id);
|
|
261
|
+
return [...withoutGeneratedRoute, p2pRoute];
|
|
262
|
+
}
|
|
263
|
+
normalizePositiveInteger(value) {
|
|
264
|
+
if (typeof value === 'number' && Number.isFinite(value) && value > 0) {
|
|
265
|
+
return Math.floor(value);
|
|
266
|
+
}
|
|
267
|
+
if (typeof value === 'string') {
|
|
268
|
+
const parsed = Number(value);
|
|
269
|
+
if (Number.isFinite(parsed) && parsed > 0) {
|
|
270
|
+
return Math.floor(parsed);
|
|
271
|
+
}
|
|
272
|
+
}
|
|
273
|
+
return undefined;
|
|
274
|
+
}
|
|
275
|
+
normalizeNonNegativeInteger(value) {
|
|
276
|
+
if (typeof value === 'number' && Number.isFinite(value) && value >= 0) {
|
|
277
|
+
return Math.floor(value);
|
|
278
|
+
}
|
|
279
|
+
if (typeof value === 'string') {
|
|
280
|
+
const parsed = Number(value);
|
|
281
|
+
if (Number.isFinite(parsed) && parsed >= 0) {
|
|
282
|
+
return Math.floor(parsed);
|
|
283
|
+
}
|
|
284
|
+
}
|
|
285
|
+
return undefined;
|
|
286
|
+
}
|
|
287
|
+
normalizeBoolean(value) {
|
|
288
|
+
if (value === undefined) {
|
|
289
|
+
return true;
|
|
290
|
+
}
|
|
291
|
+
if (typeof value === 'boolean') {
|
|
292
|
+
return value;
|
|
293
|
+
}
|
|
294
|
+
const normalized = value.trim().toLowerCase();
|
|
295
|
+
return normalized === 'true' || normalized === '1' || normalized === 'yes' || normalized === 'on';
|
|
296
|
+
}
|
|
113
297
|
handleHeartbeatResponse(data) {
|
|
114
298
|
if (!data || typeof data !== 'object') {
|
|
115
299
|
return;
|
|
@@ -227,4 +411,27 @@ class EdgeNodeAgent {
|
|
|
227
411
|
}
|
|
228
412
|
}
|
|
229
413
|
exports.EdgeNodeAgent = EdgeNodeAgent;
|
|
414
|
+
function addressEvidenceFromCandidates(candidates) {
|
|
415
|
+
const evidence = candidates.map(candidateAddressEvidence);
|
|
416
|
+
return evidence.find((entry) => entry === 'explicit-host')
|
|
417
|
+
?? evidence.find((entry) => entry === 'explicit-address')
|
|
418
|
+
?? evidence.find((entry) => entry === 'signal-observed')
|
|
419
|
+
?? evidence.find((entry) => entry === 'candidate-url')
|
|
420
|
+
?? 'port-only';
|
|
421
|
+
}
|
|
422
|
+
function candidateAddressEvidence(candidate) {
|
|
423
|
+
if (candidate.host) {
|
|
424
|
+
return 'explicit-host';
|
|
425
|
+
}
|
|
426
|
+
if (candidate.address && candidate.url) {
|
|
427
|
+
return 'explicit-address';
|
|
428
|
+
}
|
|
429
|
+
if (candidate.address) {
|
|
430
|
+
return 'signal-observed';
|
|
431
|
+
}
|
|
432
|
+
if (candidate.url) {
|
|
433
|
+
return 'candidate-url';
|
|
434
|
+
}
|
|
435
|
+
return 'port-only';
|
|
436
|
+
}
|
|
230
437
|
//# sourceMappingURL=EdgeNodeAgent.js.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"EdgeNodeAgent.js","sourceRoot":"","sources":["../../src/edge/EdgeNodeAgent.ts"],"names":[],"mappings":"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","sourcesContent":["import os from 'node:os';\nimport { spawn } from 'node:child_process';\nimport { getLoggerFor } from 'global-logger-factory';\nimport type { EdgeNodeSignalClientOptions } from '../service/EdgeNodeSignalClient';\nimport { EdgeNodeSignalClient } from '../service/EdgeNodeSignalClient';\nimport { FrpcProcessManager, type FrpcRuntimeStatus } from './frp/FrpcProcessManager';\nimport { AcmeCertificateManager } from './acme/AcmeCertificateManager';\nimport { ClusterCertificateManager } from './acme/ClusterCertificateManager';\nimport { EdgeNodeCapabilityDetector, type NetworkAddressInfo } from './EdgeNodeCapabilityDetector';\n\nexport interface EdgeNodeAgentOptions {\n signalEndpoint: string;\n nodeId: string;\n nodeToken: string;\n baseUrl?: string;\n directCandidates?: string | string[];\n pods?: string[];\n includeSystemMetrics?: boolean;\n enableNetworkDetection?: boolean;\n metadata?: Record<string, unknown>;\n intervalMs?: number;\n onHeartbeatResponse?: (data: unknown) => void;\n acme?: {\n mode?: 'local' | 'cluster';\n email?: string;\n domains?: string[];\n directoryUrl?: string;\n accountKeyPath?: string;\n certificateKeyPath: string;\n certificatePath: string;\n fullChainPath?: string;\n renewBeforeDays?: number;\n propagationDelayMs?: number;\n postDeployCommand?: string[];\n };\n frp?: {\n binaryPath: string;\n configPath: string;\n workingDirectory?: string;\n logPrefix?: string;\n autoRestart?: boolean;\n };\n}\n\nexport class EdgeNodeAgent {\n private readonly logger = getLoggerFor(this);\n private heartbeat?: EdgeNodeSignalClient;\n private frpManager?: FrpcProcessManager;\n private clusterCertificate?: ClusterCertificateManager;\n private networkDetector?: EdgeNodeCapabilityDetector;\n private cachedNetworkInfo?: NetworkAddressInfo;\n private lastNetworkDetection = 0;\n private readonly networkDetectionIntervalMs = 60_000; // 每分钟重新检测一次\n\n public async start(options: EdgeNodeAgentOptions): Promise<void> {\n if (options.acme) {\n const mode = options.acme.mode ?? 'local';\n if (mode === 'cluster') {\n await this.ensureClusterCertificate(options);\n } else {\n await this.issueCertificateLocally(options);\n }\n }\n if (options.frp) {\n this.frpManager = new FrpcProcessManager({\n binaryPath: options.frp.binaryPath,\n configPath: options.frp.configPath,\n workingDirectory: options.frp.workingDirectory,\n logPrefix: options.frp.logPrefix,\n autoRestart: options.frp.autoRestart,\n });\n }\n \n // 初始化网络检测器\n if (options.enableNetworkDetection !== false) {\n this.networkDetector = new EdgeNodeCapabilityDetector({\n dynamicDetection: {\n enableNetworkDetection: true,\n },\n });\n // 执行初始网络检测\n this.cachedNetworkInfo = await this.networkDetector.detectNetworkAddresses();\n this.lastNetworkDetection = Date.now();\n this.logger.info(`Network detection: IPv4=${this.cachedNetworkInfo.ipv4Public ?? this.cachedNetworkInfo.ipv4}, IPv6=${this.cachedNetworkInfo.ipv6Public ?? this.cachedNetworkInfo.ipv6}, hasPublicIPv6=${this.cachedNetworkInfo.hasPublicIPv6}`);\n }\n \n const systemMetrics = options.includeSystemMetrics ? this.collectSystemMetrics() : undefined;\n const metadataPayload = {\n ...(options.metadata ?? {}),\n system: systemMetrics,\n } as Record<string, unknown>;\n\n const certificatePayload = this.clusterCertificate?.getHeartbeatPayload();\n const heartbeatOptions: EdgeNodeSignalClientOptions = {\n edgeNodesEnabled: true,\n signalEndpoint: options.signalEndpoint,\n nodeId: options.nodeId,\n nodeToken: options.nodeToken,\n baseUrl: options.baseUrl,\n directCandidates: options.directCandidates,\n pods: options.pods,\n intervalMs: options.intervalMs,\n metadata: this.stringifyIfContent(metadataPayload),\n metrics: systemMetrics ? JSON.stringify(systemMetrics) : undefined,\n certificate: certificatePayload ? JSON.stringify(certificatePayload) : undefined,\n onHeartbeatResponse: (data: unknown): void => {\n this.handleHeartbeatResponse(data);\n options.onHeartbeatResponse?.(data);\n },\n networkSupplier: this.networkDetector ? () => this.getNetworkInfo() : undefined,\n };\n if (this.frpManager) {\n heartbeatOptions.tunnelSupplier = () => this.buildTunnelHeartbeatPayload();\n }\n\n this.heartbeat = new EdgeNodeSignalClient(heartbeatOptions);\n }\n\n public stop(): void {\n if (this.heartbeat && typeof (this.heartbeat as any).dispose === 'function') {\n (this.heartbeat as any).dispose();\n }\n this.heartbeat = undefined;\n void this.frpManager?.stop();\n this.clusterCertificate?.stop();\n }\n\n private stringifyIfContent(data: Record<string, unknown>): string | undefined {\n const sanitized: Record<string, unknown> = {};\n for (const [ key, value ] of Object.entries(data)) {\n if (value !== undefined) {\n sanitized[key] = value;\n }\n }\n return Object.keys(sanitized).length > 0 ? JSON.stringify(sanitized) : undefined;\n }\n\n private collectSystemMetrics(): Record<string, unknown> {\n const load = os.loadavg();\n return {\n hostname: os.hostname(),\n platform: os.platform(),\n arch: os.arch(),\n uptime: os.uptime(),\n cpuCount: os.cpus().length,\n load1: load[0],\n load5: load[1],\n load15: load[2],\n totalMem: os.totalmem(),\n freeMem: os.freemem(),\n };\n }\n\n private handleHeartbeatResponse(data: unknown): void {\n if (!data || typeof data !== 'object') {\n return;\n }\n const body = data as Record<string, any>;\n const metadata = body.metadata as Record<string, any> | undefined;\n this.clusterCertificate?.handleHeartbeatMetadata(metadata);\n const tunnel = metadata?.tunnel as Record<string, any> | undefined;\n const config = tunnel?.config as Record<string, any> | undefined;\n if (config) {\n this.logger.debug(`接收到隧道配置: ${JSON.stringify({ entrypoint: tunnel?.entrypoint, proxyName: config.proxyName })}`);\n }\n const entrypoint = typeof tunnel?.entrypoint === 'string' ? tunnel.entrypoint :\n typeof config?.publicUrl === 'string' ? config.publicUrl : undefined;\n void this.frpManager?.applyConfig(config as any, tunnel?.status as string | undefined, entrypoint);\n }\n\n private async issueCertificateLocally(options: EdgeNodeAgentOptions): Promise<void> {\n const acmeOptions = options.acme!;\n if (!acmeOptions.email || !acmeOptions.domains || acmeOptions.domains.length === 0) {\n throw new Error('本地 ACME 模式需要提供 email 与 domains。');\n }\n if (!acmeOptions.accountKeyPath) {\n throw new Error('本地 ACME 模式需要提供 accountKeyPath。');\n }\n const manager = new AcmeCertificateManager({\n signalEndpoint: options.signalEndpoint,\n nodeId: options.nodeId,\n nodeToken: options.nodeToken,\n email: acmeOptions.email!,\n domains: acmeOptions.domains!,\n directoryUrl: acmeOptions.directoryUrl,\n accountKeyPath: acmeOptions.accountKeyPath!,\n certificateKeyPath: acmeOptions.certificateKeyPath,\n certificatePath: acmeOptions.certificatePath,\n fullChainPath: acmeOptions.fullChainPath,\n renewBeforeDays: acmeOptions.renewBeforeDays,\n propagationDelayMs: acmeOptions.propagationDelayMs,\n });\n try {\n const issued = await manager.ensureCertificate();\n if (issued && acmeOptions.postDeployCommand && acmeOptions.postDeployCommand.length > 0) {\n await this.runPostDeploy(acmeOptions.postDeployCommand);\n }\n } catch (error: unknown) {\n this.logger.error(`自动签发证书失败:${(error as Error).message}`);\n throw error;\n }\n }\n\n private async ensureClusterCertificate(options: EdgeNodeAgentOptions): Promise<void> {\n const acmeOptions = options.acme!;\n if (!acmeOptions.certificateKeyPath || !acmeOptions.certificatePath) {\n throw new Error('Cluster 模式需要提供 certificateKeyPath 与 certificatePath。');\n }\n const manager = new ClusterCertificateManager({\n signalEndpoint: options.signalEndpoint,\n nodeId: options.nodeId,\n nodeToken: options.nodeToken,\n certificateKeyPath: acmeOptions.certificateKeyPath,\n certificatePath: acmeOptions.certificatePath,\n fullChainPath: acmeOptions.fullChainPath,\n renewBeforeDays: acmeOptions.renewBeforeDays,\n onCertificateInstalled: acmeOptions.postDeployCommand ? () => this.runPostDeploy(acmeOptions.postDeployCommand!) : undefined,\n });\n this.clusterCertificate = manager;\n await manager.start();\n }\n\n private async runPostDeploy(command: string[]): Promise<void> {\n const [ executable, ...args ] = command;\n if (!executable) {\n return;\n }\n await new Promise<void>((resolve, reject) => {\n const child = spawn(executable, args, { stdio: 'inherit' });\n child.on('error', reject);\n child.on('exit', (code) => {\n if (code === 0) {\n resolve();\n } else {\n reject(new Error(`命令 ${executable} 退出码 ${code}`));\n }\n });\n });\n }\n\n private buildTunnelHeartbeatPayload(): Record<string, unknown> | undefined {\n const status: FrpcRuntimeStatus | undefined = this.frpManager?.getStatus();\n if (!status) {\n return undefined;\n }\n return { client: status };\n }\n\n /**\n * 获取网络信息(带缓存,每分钟刷新一次)\n */\n private async getNetworkInfo(): Promise<{ ipv4?: string; ipv6?: string }> {\n const now = Date.now();\n \n // 如果缓存过期,重新检测\n if (!this.cachedNetworkInfo || (now - this.lastNetworkDetection) > this.networkDetectionIntervalMs) {\n if (this.networkDetector) {\n try {\n this.cachedNetworkInfo = await this.networkDetector.detectNetworkAddresses();\n this.lastNetworkDetection = now;\n } catch (error: unknown) {\n this.logger.debug(`Network detection failed: ${(error as Error).message}`);\n }\n }\n }\n \n // 优先返回公网地址\n return {\n ipv4: this.cachedNetworkInfo?.ipv4Public ?? this.cachedNetworkInfo?.ipv4,\n ipv6: this.cachedNetworkInfo?.ipv6Public ?? this.cachedNetworkInfo?.ipv6,\n };\n }\n}\n"]}
|
|
1
|
+
{"version":3,"file":"EdgeNodeAgent.js","sourceRoot":"","sources":["../../src/edge/EdgeNodeAgent.ts"],"names":[],"mappings":"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","sourcesContent":["import os from 'node:os';\nimport { spawn } from 'node:child_process';\nimport { getLoggerFor } from 'global-logger-factory';\nimport type { EdgeNodeSignalClientOptions } from '../service/EdgeNodeSignalClient';\nimport { EdgeNodeSignalClient } from '../service/EdgeNodeSignalClient';\nimport { FrpcProcessManager, type FrpcRuntimeStatus } from './frp/FrpcProcessManager';\nimport { AcmeCertificateManager } from './acme/AcmeCertificateManager';\nimport { ClusterCertificateManager } from './acme/ClusterCertificateManager';\nimport { EdgeNodeCapabilityDetector, type NetworkAddressInfo } from './EdgeNodeCapabilityDetector';\nimport {\n acceptSignaledRawTcpP2PConnectionOnce,\n createP2PDataPlaneHandler,\n createP2PSignalingClient,\n type AccessRoute,\n type P2PSignalingClient,\n type RawTcpP2PConnectSocket,\n type RawTcpP2PSleep,\n type TcpP2PDataPlaneSocketHandle,\n} from './reachability';\n\ntype EdgeNodeP2PHeartbeatRoute = Omit<AccessRoute, 'canonicalUrl'> & { canonicalUrl?: string };\n\nexport interface EdgeNodeP2PAcceptEvent {\n sessionId: string;\n nodeId: string;\n clientId: string;\n localCandidateCount: number;\n remoteCandidateCount: number;\n nodeAddress: CandidateAddressEvidence;\n clientAddress: CandidateAddressEvidence;\n acceptedAt: string;\n}\n\ntype CandidateAddressEvidence =\n | 'explicit-host'\n | 'explicit-address'\n | 'signal-observed'\n | 'candidate-url'\n | 'port-only';\n\nconst DEFAULT_P2P_ACCEPT_INTERVAL_MS = 1_000;\n\nexport interface EdgeNodeAgentOptions {\n signalEndpoint: string;\n nodeId: string;\n nodeToken: string;\n baseUrl?: string;\n directCandidates?: string | string[];\n pods?: string[];\n includeSystemMetrics?: boolean;\n enableNetworkDetection?: boolean;\n metadata?: Record<string, unknown>;\n intervalMs?: number;\n onHeartbeatResponse?: (data: unknown) => void;\n acme?: {\n mode?: 'local' | 'cluster';\n email?: string;\n domains?: string[];\n directoryUrl?: string;\n accountKeyPath?: string;\n certificateKeyPath: string;\n certificatePath: string;\n fullChainPath?: string;\n renewBeforeDays?: number;\n propagationDelayMs?: number;\n postDeployCommand?: string[];\n };\n frp?: {\n binaryPath: string;\n configPath: string;\n workingDirectory?: string;\n logPrefix?: string;\n autoRestart?: boolean;\n };\n p2p?: {\n enabled?: boolean | string;\n targetBaseUrl: string | URL;\n label?: string;\n host?: string;\n address?: string;\n signaling?: P2PSignalingClient;\n acceptIntervalMs?: number | string;\n connectTimeoutMs?: number | string;\n winnerSelectionWindowMs?: number | string;\n localAddress?: string;\n sleepMs?: RawTcpP2PSleep;\n connectSocket?: RawTcpP2PConnectSocket;\n onP2PAccept?: (event: EdgeNodeP2PAcceptEvent) => void;\n };\n}\n\nexport class EdgeNodeAgent {\n private readonly logger = getLoggerFor(this);\n private heartbeat?: EdgeNodeSignalClient;\n private frpManager?: FrpcProcessManager;\n private clusterCertificate?: ClusterCertificateManager;\n private networkDetector?: EdgeNodeCapabilityDetector;\n private cachedNetworkInfo?: NetworkAddressInfo;\n private lastNetworkDetection = 0;\n private p2pAcceptInterval?: NodeJS.Timeout;\n private p2pAcceptRunning = false;\n private p2pAcceptGeneration = 0;\n private readonly p2pAcceptedSessionIds = new Set<string>();\n private readonly p2pSocketHandles = new Set<TcpP2PDataPlaneSocketHandle>();\n private readonly networkDetectionIntervalMs = 60_000; // 每分钟重新检测一次\n\n public async start(options: EdgeNodeAgentOptions): Promise<void> {\n if (options.acme) {\n const mode = options.acme.mode ?? 'local';\n if (mode === 'cluster') {\n await this.ensureClusterCertificate(options);\n } else {\n await this.issueCertificateLocally(options);\n }\n }\n if (options.frp) {\n this.frpManager = new FrpcProcessManager({\n binaryPath: options.frp.binaryPath,\n configPath: options.frp.configPath,\n workingDirectory: options.frp.workingDirectory,\n logPrefix: options.frp.logPrefix,\n autoRestart: options.frp.autoRestart,\n });\n }\n \n // 初始化网络检测器\n if (options.enableNetworkDetection !== false) {\n this.networkDetector = new EdgeNodeCapabilityDetector({\n dynamicDetection: {\n enableNetworkDetection: true,\n },\n });\n // 执行初始网络检测\n this.cachedNetworkInfo = await this.networkDetector.detectNetworkAddresses();\n this.lastNetworkDetection = Date.now();\n this.logger.info(`Network detection: IPv4=${this.cachedNetworkInfo.ipv4Public ?? this.cachedNetworkInfo.ipv4}, IPv6=${this.cachedNetworkInfo.ipv6Public ?? this.cachedNetworkInfo.ipv6}, hasPublicIPv6=${this.cachedNetworkInfo.hasPublicIPv6}`);\n }\n \n const systemMetrics = options.includeSystemMetrics ? this.collectSystemMetrics() : undefined;\n const metadataPayload = {\n ...this.buildHeartbeatMetadata(options),\n system: systemMetrics,\n } as Record<string, unknown>;\n\n const certificatePayload = this.clusterCertificate?.getHeartbeatPayload();\n const heartbeatOptions: EdgeNodeSignalClientOptions = {\n edgeNodesEnabled: true,\n signalEndpoint: options.signalEndpoint,\n nodeId: options.nodeId,\n nodeToken: options.nodeToken,\n baseUrl: options.baseUrl,\n directCandidates: options.directCandidates,\n pods: options.pods,\n intervalMs: options.intervalMs,\n metadata: this.stringifyIfContent(metadataPayload),\n metrics: systemMetrics ? JSON.stringify(systemMetrics) : undefined,\n certificate: certificatePayload ? JSON.stringify(certificatePayload) : undefined,\n onHeartbeatResponse: (data: unknown): void => {\n this.handleHeartbeatResponse(data);\n options.onHeartbeatResponse?.(data);\n },\n networkSupplier: this.networkDetector ? () => this.getNetworkInfo() : undefined,\n };\n if (this.frpManager) {\n heartbeatOptions.tunnelSupplier = () => this.buildTunnelHeartbeatPayload();\n }\n\n this.heartbeat = new EdgeNodeSignalClient(heartbeatOptions);\n this.startP2PAcceptLoop(options);\n }\n\n public stop(): void {\n if (this.heartbeat && typeof (this.heartbeat as any).dispose === 'function') {\n (this.heartbeat as any).dispose();\n }\n this.heartbeat = undefined;\n if (this.p2pAcceptInterval) {\n clearInterval(this.p2pAcceptInterval);\n this.p2pAcceptInterval = undefined;\n }\n this.p2pAcceptGeneration += 1;\n for (const handle of this.p2pSocketHandles) {\n handle.close();\n }\n this.p2pSocketHandles.clear();\n this.p2pAcceptedSessionIds.clear();\n this.p2pAcceptRunning = false;\n void this.frpManager?.stop();\n this.clusterCertificate?.stop();\n }\n\n private startP2PAcceptLoop(options: EdgeNodeAgentOptions): void {\n const p2p = options.p2p;\n if (!p2p || this.normalizeBoolean(p2p.enabled) === false) {\n return;\n }\n const signaling = p2p.signaling ?? createP2PSignalingClient({\n apiBaseUrl: this.resolveP2PApiBaseUrl(options.signalEndpoint),\n nodeId: options.nodeId,\n token: options.nodeToken,\n });\n const handler = createP2PDataPlaneHandler({ targetBaseUrl: p2p.targetBaseUrl });\n const intervalMs = this.normalizePositiveInteger(p2p.acceptIntervalMs) ?? DEFAULT_P2P_ACCEPT_INTERVAL_MS;\n this.p2pAcceptGeneration += 1;\n const generation = this.p2pAcceptGeneration;\n const run = (): void => {\n void this.acceptP2PConnectionOnce({\n generation,\n signaling,\n sourceId: options.nodeId,\n host: p2p.host,\n address: p2p.address,\n handler,\n connectTimeoutMs: this.normalizePositiveInteger(p2p.connectTimeoutMs),\n winnerSelectionWindowMs: this.normalizeNonNegativeInteger(p2p.winnerSelectionWindowMs),\n localAddress: p2p.localAddress,\n sleepMs: p2p.sleepMs,\n connectSocket: p2p.connectSocket,\n onP2PAccept: p2p.onP2PAccept,\n });\n };\n run();\n this.p2pAcceptInterval = setInterval(run, intervalMs);\n }\n\n private async acceptP2PConnectionOnce(options: Parameters<typeof acceptSignaledRawTcpP2PConnectionOnce>[0] & {\n generation: number;\n onP2PAccept?: (event: EdgeNodeP2PAcceptEvent) => void;\n }): Promise<void> {\n if (this.p2pAcceptRunning) {\n return;\n }\n this.p2pAcceptRunning = true;\n try {\n const { generation, onP2PAccept, ...acceptOptions } = options;\n const accepted = await acceptSignaledRawTcpP2PConnectionOnce({\n ...acceptOptions,\n signaling: this.skipAcceptedP2PSessions(acceptOptions.signaling),\n host: acceptOptions.host,\n });\n if (accepted) {\n if (generation !== this.p2pAcceptGeneration) {\n accepted.socketHandle.close();\n return;\n }\n this.p2pAcceptedSessionIds.add(accepted.session.sessionId);\n this.p2pSocketHandles.add(accepted.socketHandle);\n onP2PAccept?.({\n sessionId: accepted.session.sessionId,\n nodeId: accepted.session.nodeId,\n clientId: accepted.session.clientId,\n localCandidateCount: accepted.localCandidates.length,\n remoteCandidateCount: accepted.remoteCandidates.length,\n nodeAddress: addressEvidenceFromCandidates(accepted.localCandidates),\n clientAddress: addressEvidenceFromCandidates(accepted.remoteCandidates),\n acceptedAt: new Date().toISOString(),\n });\n this.logger.info(`Accepted raw TCP P2P session ${accepted.session.sessionId}.`);\n }\n } catch (error: unknown) {\n this.logger.debug(`Raw TCP P2P accept attempt failed: ${(error as Error).message}`);\n } finally {\n this.p2pAcceptRunning = false;\n }\n }\n\n private skipAcceptedP2PSessions(signaling: P2PSignalingClient): P2PSignalingClient {\n return {\n createP2PSession: (request) => signaling.createP2PSession(request),\n getP2PSession: (sessionIdOrUrl) => signaling.getP2PSession(sessionIdOrUrl),\n addP2PCandidates: (sessionIdOrUrl, request) => signaling.addP2PCandidates(sessionIdOrUrl, request),\n listP2PSessions: async () => {\n const sessions = await signaling.listP2PSessions();\n return sessions.filter((session) => !this.p2pAcceptedSessionIds.has(session.sessionId));\n },\n };\n }\n\n private resolveP2PApiBaseUrl(signalEndpoint: string): string {\n try {\n return new URL(signalEndpoint).origin;\n } catch {\n return signalEndpoint;\n }\n }\n\n private stringifyIfContent(data: Record<string, unknown>): string | undefined {\n const sanitized: Record<string, unknown> = {};\n for (const [ key, value ] of Object.entries(data)) {\n if (value !== undefined) {\n sanitized[key] = value;\n }\n }\n return Object.keys(sanitized).length > 0 ? JSON.stringify(sanitized) : undefined;\n }\n\n private collectSystemMetrics(): Record<string, unknown> {\n const load = os.loadavg();\n return {\n hostname: os.hostname(),\n platform: os.platform(),\n arch: os.arch(),\n uptime: os.uptime(),\n cpuCount: os.cpus().length,\n load1: load[0],\n load5: load[1],\n load15: load[2],\n totalMem: os.totalmem(),\n freeMem: os.freemem(),\n };\n }\n\n private buildHeartbeatMetadata(options: EdgeNodeAgentOptions): Record<string, unknown> {\n const metadata = { ...(options.metadata ?? {}) } as Record<string, unknown>;\n const p2pRoute = this.buildP2PHeartbeatRoute(options);\n if (p2pRoute) {\n metadata.routes = this.mergeHeartbeatRoutes(metadata.routes, p2pRoute);\n }\n return metadata;\n }\n\n private buildP2PHeartbeatRoute(options: EdgeNodeAgentOptions): EdgeNodeP2PHeartbeatRoute | undefined {\n const p2p = options.p2p;\n if (!p2p || this.normalizeBoolean(p2p.enabled) === false) {\n return undefined;\n }\n const label = typeof p2p.label === 'string' && p2p.label.length > 0 ? p2p.label : undefined;\n return {\n id: 'p2p-raw-tcp',\n nodeId: options.nodeId,\n ...(options.baseUrl ? { canonicalUrl: options.baseUrl } : {}),\n kind: 'p2p',\n targetUrl: `tcp-punch://node/${encodeURIComponent(options.nodeId)}`,\n priority: 40,\n requiresManagedClient: true,\n visibility: 'authorized-client',\n health: 'healthy',\n metadata: {\n protocols: {\n 'raw-tcp-hole-punch': {\n enabled: true,\n ...(label ? { label } : {}),\n },\n },\n },\n };\n }\n\n private mergeHeartbeatRoutes(existing: unknown, p2pRoute: EdgeNodeP2PHeartbeatRoute): EdgeNodeP2PHeartbeatRoute[] {\n const routes = Array.isArray(existing)\n ? existing.filter((entry): entry is EdgeNodeP2PHeartbeatRoute => Boolean(entry) && typeof entry === 'object')\n : [];\n const withoutGeneratedRoute = routes.filter((route) => route.id !== p2pRoute.id);\n return [...withoutGeneratedRoute, p2pRoute];\n }\n\n private normalizePositiveInteger(value: number | string | undefined): number | undefined {\n if (typeof value === 'number' && Number.isFinite(value) && value > 0) {\n return Math.floor(value);\n }\n if (typeof value === 'string') {\n const parsed = Number(value);\n if (Number.isFinite(parsed) && parsed > 0) {\n return Math.floor(parsed);\n }\n }\n return undefined;\n }\n\n private normalizeNonNegativeInteger(value: number | string | undefined): number | undefined {\n if (typeof value === 'number' && Number.isFinite(value) && value >= 0) {\n return Math.floor(value);\n }\n if (typeof value === 'string') {\n const parsed = Number(value);\n if (Number.isFinite(parsed) && parsed >= 0) {\n return Math.floor(parsed);\n }\n }\n return undefined;\n }\n\n private normalizeBoolean(value: boolean | string | undefined): boolean {\n if (value === undefined) {\n return true;\n }\n if (typeof value === 'boolean') {\n return value;\n }\n const normalized = value.trim().toLowerCase();\n return normalized === 'true' || normalized === '1' || normalized === 'yes' || normalized === 'on';\n }\n\n private handleHeartbeatResponse(data: unknown): void {\n if (!data || typeof data !== 'object') {\n return;\n }\n const body = data as Record<string, any>;\n const metadata = body.metadata as Record<string, any> | undefined;\n this.clusterCertificate?.handleHeartbeatMetadata(metadata);\n const tunnel = metadata?.tunnel as Record<string, any> | undefined;\n const config = tunnel?.config as Record<string, any> | undefined;\n if (config) {\n this.logger.debug(`接收到隧道配置: ${JSON.stringify({ entrypoint: tunnel?.entrypoint, proxyName: config.proxyName })}`);\n }\n const entrypoint = typeof tunnel?.entrypoint === 'string' ? tunnel.entrypoint :\n typeof config?.publicUrl === 'string' ? config.publicUrl : undefined;\n void this.frpManager?.applyConfig(config as any, tunnel?.status as string | undefined, entrypoint);\n }\n\n private async issueCertificateLocally(options: EdgeNodeAgentOptions): Promise<void> {\n const acmeOptions = options.acme!;\n if (!acmeOptions.email || !acmeOptions.domains || acmeOptions.domains.length === 0) {\n throw new Error('本地 ACME 模式需要提供 email 与 domains。');\n }\n if (!acmeOptions.accountKeyPath) {\n throw new Error('本地 ACME 模式需要提供 accountKeyPath。');\n }\n const manager = new AcmeCertificateManager({\n signalEndpoint: options.signalEndpoint,\n nodeId: options.nodeId,\n nodeToken: options.nodeToken,\n email: acmeOptions.email!,\n domains: acmeOptions.domains!,\n directoryUrl: acmeOptions.directoryUrl,\n accountKeyPath: acmeOptions.accountKeyPath!,\n certificateKeyPath: acmeOptions.certificateKeyPath,\n certificatePath: acmeOptions.certificatePath,\n fullChainPath: acmeOptions.fullChainPath,\n renewBeforeDays: acmeOptions.renewBeforeDays,\n propagationDelayMs: acmeOptions.propagationDelayMs,\n });\n try {\n const issued = await manager.ensureCertificate();\n if (issued && acmeOptions.postDeployCommand && acmeOptions.postDeployCommand.length > 0) {\n await this.runPostDeploy(acmeOptions.postDeployCommand);\n }\n } catch (error: unknown) {\n this.logger.error(`自动签发证书失败:${(error as Error).message}`);\n throw error;\n }\n }\n\n private async ensureClusterCertificate(options: EdgeNodeAgentOptions): Promise<void> {\n const acmeOptions = options.acme!;\n if (!acmeOptions.certificateKeyPath || !acmeOptions.certificatePath) {\n throw new Error('Cluster 模式需要提供 certificateKeyPath 与 certificatePath。');\n }\n const manager = new ClusterCertificateManager({\n signalEndpoint: options.signalEndpoint,\n nodeId: options.nodeId,\n nodeToken: options.nodeToken,\n certificateKeyPath: acmeOptions.certificateKeyPath,\n certificatePath: acmeOptions.certificatePath,\n fullChainPath: acmeOptions.fullChainPath,\n renewBeforeDays: acmeOptions.renewBeforeDays,\n onCertificateInstalled: acmeOptions.postDeployCommand ? () => this.runPostDeploy(acmeOptions.postDeployCommand!) : undefined,\n });\n this.clusterCertificate = manager;\n await manager.start();\n }\n\n private async runPostDeploy(command: string[]): Promise<void> {\n const [ executable, ...args ] = command;\n if (!executable) {\n return;\n }\n await new Promise<void>((resolve, reject) => {\n const child = spawn(executable, args, { stdio: 'inherit' });\n child.on('error', reject);\n child.on('exit', (code) => {\n if (code === 0) {\n resolve();\n } else {\n reject(new Error(`命令 ${executable} 退出码 ${code}`));\n }\n });\n });\n }\n\n private buildTunnelHeartbeatPayload(): Record<string, unknown> | undefined {\n const status: FrpcRuntimeStatus | undefined = this.frpManager?.getStatus();\n if (!status) {\n return undefined;\n }\n return { client: status };\n }\n\n /**\n * 获取网络信息(带缓存,每分钟刷新一次)\n */\n private async getNetworkInfo(): Promise<{ ipv4?: string; ipv6?: string }> {\n const now = Date.now();\n \n // 如果缓存过期,重新检测\n if (!this.cachedNetworkInfo || (now - this.lastNetworkDetection) > this.networkDetectionIntervalMs) {\n if (this.networkDetector) {\n try {\n this.cachedNetworkInfo = await this.networkDetector.detectNetworkAddresses();\n this.lastNetworkDetection = now;\n } catch (error: unknown) {\n this.logger.debug(`Network detection failed: ${(error as Error).message}`);\n }\n }\n }\n \n // 优先返回公网地址\n return {\n ipv4: this.cachedNetworkInfo?.ipv4Public ?? this.cachedNetworkInfo?.ipv4,\n ipv6: this.cachedNetworkInfo?.ipv6Public ?? this.cachedNetworkInfo?.ipv6,\n };\n }\n}\n\nfunction addressEvidenceFromCandidates(candidates: Array<{ host?: string; address?: string; url?: string }>): CandidateAddressEvidence {\n const evidence = candidates.map(candidateAddressEvidence);\n return evidence.find((entry) => entry === 'explicit-host')\n ?? evidence.find((entry) => entry === 'explicit-address')\n ?? evidence.find((entry) => entry === 'signal-observed')\n ?? evidence.find((entry) => entry === 'candidate-url')\n ?? 'port-only';\n}\n\nfunction candidateAddressEvidence(candidate: { host?: string; address?: string; url?: string }): CandidateAddressEvidence {\n if (candidate.host) {\n return 'explicit-host';\n }\n if (candidate.address && candidate.url) {\n return 'explicit-address';\n }\n if (candidate.address) {\n return 'signal-observed';\n }\n if (candidate.url) {\n return 'candidate-url';\n }\n return 'port-only';\n}\n"]}
|
|
@@ -4,6 +4,112 @@
|
|
|
4
4
|
],
|
|
5
5
|
"@id": "npmd:@undefineds.co/xpod",
|
|
6
6
|
"components": [
|
|
7
|
+
{
|
|
8
|
+
"@id": "undefineds:dist/edge/EdgeNodeAgent.jsonld#EdgeNodeAgentOptions",
|
|
9
|
+
"@type": "AbstractClass",
|
|
10
|
+
"requireElement": "EdgeNodeAgentOptions",
|
|
11
|
+
"parameters": [],
|
|
12
|
+
"memberFields": [
|
|
13
|
+
{
|
|
14
|
+
"@id": "undefineds:dist/edge/EdgeNodeAgent.jsonld#EdgeNodeAgentOptions__member_signalEndpoint",
|
|
15
|
+
"memberFieldName": "signalEndpoint"
|
|
16
|
+
},
|
|
17
|
+
{
|
|
18
|
+
"@id": "undefineds:dist/edge/EdgeNodeAgent.jsonld#EdgeNodeAgentOptions__member_nodeId",
|
|
19
|
+
"memberFieldName": "nodeId"
|
|
20
|
+
},
|
|
21
|
+
{
|
|
22
|
+
"@id": "undefineds:dist/edge/EdgeNodeAgent.jsonld#EdgeNodeAgentOptions__member_nodeToken",
|
|
23
|
+
"memberFieldName": "nodeToken"
|
|
24
|
+
},
|
|
25
|
+
{
|
|
26
|
+
"@id": "undefineds:dist/edge/EdgeNodeAgent.jsonld#EdgeNodeAgentOptions__member_baseUrl",
|
|
27
|
+
"memberFieldName": "baseUrl"
|
|
28
|
+
},
|
|
29
|
+
{
|
|
30
|
+
"@id": "undefineds:dist/edge/EdgeNodeAgent.jsonld#EdgeNodeAgentOptions__member_directCandidates",
|
|
31
|
+
"memberFieldName": "directCandidates"
|
|
32
|
+
},
|
|
33
|
+
{
|
|
34
|
+
"@id": "undefineds:dist/edge/EdgeNodeAgent.jsonld#EdgeNodeAgentOptions__member_pods",
|
|
35
|
+
"memberFieldName": "pods"
|
|
36
|
+
},
|
|
37
|
+
{
|
|
38
|
+
"@id": "undefineds:dist/edge/EdgeNodeAgent.jsonld#EdgeNodeAgentOptions__member_includeSystemMetrics",
|
|
39
|
+
"memberFieldName": "includeSystemMetrics"
|
|
40
|
+
},
|
|
41
|
+
{
|
|
42
|
+
"@id": "undefineds:dist/edge/EdgeNodeAgent.jsonld#EdgeNodeAgentOptions__member_enableNetworkDetection",
|
|
43
|
+
"memberFieldName": "enableNetworkDetection"
|
|
44
|
+
},
|
|
45
|
+
{
|
|
46
|
+
"@id": "undefineds:dist/edge/EdgeNodeAgent.jsonld#EdgeNodeAgentOptions__member_metadata",
|
|
47
|
+
"memberFieldName": "metadata"
|
|
48
|
+
},
|
|
49
|
+
{
|
|
50
|
+
"@id": "undefineds:dist/edge/EdgeNodeAgent.jsonld#EdgeNodeAgentOptions__member_intervalMs",
|
|
51
|
+
"memberFieldName": "intervalMs"
|
|
52
|
+
},
|
|
53
|
+
{
|
|
54
|
+
"@id": "undefineds:dist/edge/EdgeNodeAgent.jsonld#EdgeNodeAgentOptions__member_onHeartbeatResponse",
|
|
55
|
+
"memberFieldName": "onHeartbeatResponse"
|
|
56
|
+
},
|
|
57
|
+
{
|
|
58
|
+
"@id": "undefineds:dist/edge/EdgeNodeAgent.jsonld#EdgeNodeAgentOptions__member_acme",
|
|
59
|
+
"memberFieldName": "acme"
|
|
60
|
+
},
|
|
61
|
+
{
|
|
62
|
+
"@id": "undefineds:dist/edge/EdgeNodeAgent.jsonld#EdgeNodeAgentOptions__member_frp",
|
|
63
|
+
"memberFieldName": "frp"
|
|
64
|
+
},
|
|
65
|
+
{
|
|
66
|
+
"@id": "undefineds:dist/edge/EdgeNodeAgent.jsonld#EdgeNodeAgentOptions__member_p2p",
|
|
67
|
+
"memberFieldName": "p2p"
|
|
68
|
+
}
|
|
69
|
+
],
|
|
70
|
+
"constructorArguments": []
|
|
71
|
+
},
|
|
72
|
+
{
|
|
73
|
+
"@id": "undefineds:dist/edge/EdgeNodeAgent.jsonld#EdgeNodeP2PAcceptEvent",
|
|
74
|
+
"@type": "AbstractClass",
|
|
75
|
+
"requireElement": "EdgeNodeP2PAcceptEvent",
|
|
76
|
+
"parameters": [],
|
|
77
|
+
"memberFields": [
|
|
78
|
+
{
|
|
79
|
+
"@id": "undefineds:dist/edge/EdgeNodeAgent.jsonld#EdgeNodeP2PAcceptEvent__member_sessionId",
|
|
80
|
+
"memberFieldName": "sessionId"
|
|
81
|
+
},
|
|
82
|
+
{
|
|
83
|
+
"@id": "undefineds:dist/edge/EdgeNodeAgent.jsonld#EdgeNodeP2PAcceptEvent__member_nodeId",
|
|
84
|
+
"memberFieldName": "nodeId"
|
|
85
|
+
},
|
|
86
|
+
{
|
|
87
|
+
"@id": "undefineds:dist/edge/EdgeNodeAgent.jsonld#EdgeNodeP2PAcceptEvent__member_clientId",
|
|
88
|
+
"memberFieldName": "clientId"
|
|
89
|
+
},
|
|
90
|
+
{
|
|
91
|
+
"@id": "undefineds:dist/edge/EdgeNodeAgent.jsonld#EdgeNodeP2PAcceptEvent__member_localCandidateCount",
|
|
92
|
+
"memberFieldName": "localCandidateCount"
|
|
93
|
+
},
|
|
94
|
+
{
|
|
95
|
+
"@id": "undefineds:dist/edge/EdgeNodeAgent.jsonld#EdgeNodeP2PAcceptEvent__member_remoteCandidateCount",
|
|
96
|
+
"memberFieldName": "remoteCandidateCount"
|
|
97
|
+
},
|
|
98
|
+
{
|
|
99
|
+
"@id": "undefineds:dist/edge/EdgeNodeAgent.jsonld#EdgeNodeP2PAcceptEvent__member_nodeAddress",
|
|
100
|
+
"memberFieldName": "nodeAddress"
|
|
101
|
+
},
|
|
102
|
+
{
|
|
103
|
+
"@id": "undefineds:dist/edge/EdgeNodeAgent.jsonld#EdgeNodeP2PAcceptEvent__member_clientAddress",
|
|
104
|
+
"memberFieldName": "clientAddress"
|
|
105
|
+
},
|
|
106
|
+
{
|
|
107
|
+
"@id": "undefineds:dist/edge/EdgeNodeAgent.jsonld#EdgeNodeP2PAcceptEvent__member_acceptedAt",
|
|
108
|
+
"memberFieldName": "acceptedAt"
|
|
109
|
+
}
|
|
110
|
+
],
|
|
111
|
+
"constructorArguments": []
|
|
112
|
+
},
|
|
7
113
|
{
|
|
8
114
|
"@id": "undefineds:dist/edge/EdgeNodeAgent.jsonld#EdgeNodeAgent",
|
|
9
115
|
"@type": "Class",
|
|
@@ -38,6 +144,26 @@
|
|
|
38
144
|
"@id": "undefineds:dist/edge/EdgeNodeAgent.jsonld#EdgeNodeAgent__member_lastNetworkDetection",
|
|
39
145
|
"memberFieldName": "lastNetworkDetection"
|
|
40
146
|
},
|
|
147
|
+
{
|
|
148
|
+
"@id": "undefineds:dist/edge/EdgeNodeAgent.jsonld#EdgeNodeAgent__member_p2pAcceptInterval",
|
|
149
|
+
"memberFieldName": "p2pAcceptInterval"
|
|
150
|
+
},
|
|
151
|
+
{
|
|
152
|
+
"@id": "undefineds:dist/edge/EdgeNodeAgent.jsonld#EdgeNodeAgent__member_p2pAcceptRunning",
|
|
153
|
+
"memberFieldName": "p2pAcceptRunning"
|
|
154
|
+
},
|
|
155
|
+
{
|
|
156
|
+
"@id": "undefineds:dist/edge/EdgeNodeAgent.jsonld#EdgeNodeAgent__member_p2pAcceptGeneration",
|
|
157
|
+
"memberFieldName": "p2pAcceptGeneration"
|
|
158
|
+
},
|
|
159
|
+
{
|
|
160
|
+
"@id": "undefineds:dist/edge/EdgeNodeAgent.jsonld#EdgeNodeAgent__member_p2pAcceptedSessionIds",
|
|
161
|
+
"memberFieldName": "p2pAcceptedSessionIds"
|
|
162
|
+
},
|
|
163
|
+
{
|
|
164
|
+
"@id": "undefineds:dist/edge/EdgeNodeAgent.jsonld#EdgeNodeAgent__member_p2pSocketHandles",
|
|
165
|
+
"memberFieldName": "p2pSocketHandles"
|
|
166
|
+
},
|
|
41
167
|
{
|
|
42
168
|
"@id": "undefineds:dist/edge/EdgeNodeAgent.jsonld#EdgeNodeAgent__member_networkDetectionIntervalMs",
|
|
43
169
|
"memberFieldName": "networkDetectionIntervalMs"
|
|
@@ -50,6 +176,22 @@
|
|
|
50
176
|
"@id": "undefineds:dist/edge/EdgeNodeAgent.jsonld#EdgeNodeAgent__member_stop",
|
|
51
177
|
"memberFieldName": "stop"
|
|
52
178
|
},
|
|
179
|
+
{
|
|
180
|
+
"@id": "undefineds:dist/edge/EdgeNodeAgent.jsonld#EdgeNodeAgent__member_startP2PAcceptLoop",
|
|
181
|
+
"memberFieldName": "startP2PAcceptLoop"
|
|
182
|
+
},
|
|
183
|
+
{
|
|
184
|
+
"@id": "undefineds:dist/edge/EdgeNodeAgent.jsonld#EdgeNodeAgent__member_acceptP2PConnectionOnce",
|
|
185
|
+
"memberFieldName": "acceptP2PConnectionOnce"
|
|
186
|
+
},
|
|
187
|
+
{
|
|
188
|
+
"@id": "undefineds:dist/edge/EdgeNodeAgent.jsonld#EdgeNodeAgent__member_skipAcceptedP2PSessions",
|
|
189
|
+
"memberFieldName": "skipAcceptedP2PSessions"
|
|
190
|
+
},
|
|
191
|
+
{
|
|
192
|
+
"@id": "undefineds:dist/edge/EdgeNodeAgent.jsonld#EdgeNodeAgent__member_resolveP2PApiBaseUrl",
|
|
193
|
+
"memberFieldName": "resolveP2PApiBaseUrl"
|
|
194
|
+
},
|
|
53
195
|
{
|
|
54
196
|
"@id": "undefineds:dist/edge/EdgeNodeAgent.jsonld#EdgeNodeAgent__member_stringifyIfContent",
|
|
55
197
|
"memberFieldName": "stringifyIfContent"
|
|
@@ -58,6 +200,30 @@
|
|
|
58
200
|
"@id": "undefineds:dist/edge/EdgeNodeAgent.jsonld#EdgeNodeAgent__member_collectSystemMetrics",
|
|
59
201
|
"memberFieldName": "collectSystemMetrics"
|
|
60
202
|
},
|
|
203
|
+
{
|
|
204
|
+
"@id": "undefineds:dist/edge/EdgeNodeAgent.jsonld#EdgeNodeAgent__member_buildHeartbeatMetadata",
|
|
205
|
+
"memberFieldName": "buildHeartbeatMetadata"
|
|
206
|
+
},
|
|
207
|
+
{
|
|
208
|
+
"@id": "undefineds:dist/edge/EdgeNodeAgent.jsonld#EdgeNodeAgent__member_buildP2PHeartbeatRoute",
|
|
209
|
+
"memberFieldName": "buildP2PHeartbeatRoute"
|
|
210
|
+
},
|
|
211
|
+
{
|
|
212
|
+
"@id": "undefineds:dist/edge/EdgeNodeAgent.jsonld#EdgeNodeAgent__member_mergeHeartbeatRoutes",
|
|
213
|
+
"memberFieldName": "mergeHeartbeatRoutes"
|
|
214
|
+
},
|
|
215
|
+
{
|
|
216
|
+
"@id": "undefineds:dist/edge/EdgeNodeAgent.jsonld#EdgeNodeAgent__member_normalizePositiveInteger",
|
|
217
|
+
"memberFieldName": "normalizePositiveInteger"
|
|
218
|
+
},
|
|
219
|
+
{
|
|
220
|
+
"@id": "undefineds:dist/edge/EdgeNodeAgent.jsonld#EdgeNodeAgent__member_normalizeNonNegativeInteger",
|
|
221
|
+
"memberFieldName": "normalizeNonNegativeInteger"
|
|
222
|
+
},
|
|
223
|
+
{
|
|
224
|
+
"@id": "undefineds:dist/edge/EdgeNodeAgent.jsonld#EdgeNodeAgent__member_normalizeBoolean",
|
|
225
|
+
"memberFieldName": "normalizeBoolean"
|
|
226
|
+
},
|
|
61
227
|
{
|
|
62
228
|
"@id": "undefineds:dist/edge/EdgeNodeAgent.jsonld#EdgeNodeAgent__member_handleHeartbeatResponse",
|
|
63
229
|
"memberFieldName": "handleHeartbeatResponse"
|
|
@@ -1,10 +1,25 @@
|
|
|
1
1
|
import { Initializer } from '@solid/community-server';
|
|
2
|
-
import { type EdgeNodeAgentOptions } from './EdgeNodeAgent';
|
|
3
2
|
/**
|
|
4
3
|
* 所有配置项设为可选,以便在 disabled 状态下即使缺少 CLI 变量也能成功实例化。
|
|
5
4
|
*/
|
|
6
|
-
export interface EdgeNodeAgentInitializerOptions
|
|
5
|
+
export interface EdgeNodeAgentInitializerOptions {
|
|
7
6
|
enabled?: boolean | string;
|
|
7
|
+
signalEndpoint?: string;
|
|
8
|
+
nodeId?: string;
|
|
9
|
+
nodeToken?: string;
|
|
10
|
+
baseUrl?: string;
|
|
11
|
+
directCandidates?: string | string[];
|
|
12
|
+
pods?: string[];
|
|
13
|
+
includeSystemMetrics?: boolean;
|
|
14
|
+
enableNetworkDetection?: boolean;
|
|
15
|
+
metadata?: Record<string, unknown>;
|
|
16
|
+
intervalMs?: number | string;
|
|
17
|
+
p2pEnabled?: boolean | string;
|
|
18
|
+
p2pTargetBaseUrl?: string;
|
|
19
|
+
p2pLabel?: string;
|
|
20
|
+
p2pAcceptIntervalMs?: number | string;
|
|
21
|
+
p2pConnectTimeoutMs?: number | string;
|
|
22
|
+
p2pWinnerSelectionWindowMs?: number | string;
|
|
8
23
|
}
|
|
9
24
|
/**
|
|
10
25
|
* Initializer that starts the EdgeNodeAgent.
|
|
@@ -21,5 +36,8 @@ export declare class EdgeNodeAgentInitializer extends Initializer {
|
|
|
21
36
|
* 确保启用时必需参数存在
|
|
22
37
|
*/
|
|
23
38
|
private validateOptions;
|
|
39
|
+
private buildAgentOptions;
|
|
24
40
|
private normalizeBoolean;
|
|
41
|
+
private normalizePositiveInteger;
|
|
42
|
+
private normalizeNonNegativeInteger;
|
|
25
43
|
}
|