threadforge 0.1.1 → 0.2.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.
- package/README.md +52 -20
- package/bin/forge.js +2 -1058
- package/bin/host-commands.d.ts +2 -0
- package/bin/host-commands.d.ts.map +1 -0
- package/bin/host-commands.js +7 -8
- package/bin/platform-commands.d.ts +2 -0
- package/bin/platform-commands.d.ts.map +1 -0
- package/bin/platform-commands.js +118 -36
- package/dist/cli/base-command.d.ts +12 -0
- package/dist/cli/base-command.d.ts.map +1 -0
- package/dist/cli/base-command.js +25 -0
- package/dist/cli/base-command.js.map +1 -0
- package/dist/cli/commands/build.d.ts +10 -0
- package/dist/cli/commands/build.d.ts.map +1 -0
- package/dist/cli/commands/build.js +110 -0
- package/dist/cli/commands/build.js.map +1 -0
- package/dist/cli/commands/deploy.d.ts +12 -0
- package/dist/cli/commands/deploy.d.ts.map +1 -0
- package/dist/cli/commands/deploy.js +143 -0
- package/dist/cli/commands/deploy.js.map +1 -0
- package/dist/cli/commands/dev.d.ts +10 -0
- package/dist/cli/commands/dev.d.ts.map +1 -0
- package/dist/cli/commands/dev.js +138 -0
- package/dist/cli/commands/dev.js.map +1 -0
- package/dist/cli/commands/generate.d.ts +10 -0
- package/dist/cli/commands/generate.d.ts.map +1 -0
- package/dist/cli/commands/generate.js +76 -0
- package/dist/cli/commands/generate.js.map +1 -0
- package/dist/cli/commands/host.d.ts +8 -0
- package/dist/cli/commands/host.d.ts.map +1 -0
- package/dist/cli/commands/host.js +20 -0
- package/dist/cli/commands/host.js.map +1 -0
- package/dist/cli/commands/init.d.ts +16 -0
- package/dist/cli/commands/init.d.ts.map +1 -0
- package/dist/cli/commands/init.js +246 -0
- package/dist/cli/commands/init.js.map +1 -0
- package/dist/cli/commands/platform.d.ts +8 -0
- package/dist/cli/commands/platform.d.ts.map +1 -0
- package/dist/cli/commands/platform.js +20 -0
- package/dist/cli/commands/platform.js.map +1 -0
- package/dist/cli/commands/restart.d.ts +8 -0
- package/dist/cli/commands/restart.d.ts.map +1 -0
- package/dist/cli/commands/restart.js +13 -0
- package/dist/cli/commands/restart.js.map +1 -0
- package/dist/cli/commands/scaffold/frontend.d.ts +10 -0
- package/dist/cli/commands/scaffold/frontend.d.ts.map +1 -0
- package/dist/cli/commands/scaffold/frontend.js +130 -0
- package/dist/cli/commands/scaffold/frontend.js.map +1 -0
- package/dist/cli/commands/scaffold/react.d.ts +7 -0
- package/dist/cli/commands/scaffold/react.d.ts.map +1 -0
- package/dist/cli/commands/scaffold/react.js +12 -0
- package/dist/cli/commands/scaffold/react.js.map +1 -0
- package/dist/cli/commands/scale.d.ts +8 -0
- package/dist/cli/commands/scale.d.ts.map +1 -0
- package/dist/cli/commands/scale.js +13 -0
- package/dist/cli/commands/scale.js.map +1 -0
- package/dist/cli/commands/start.d.ts +10 -0
- package/dist/cli/commands/start.d.ts.map +1 -0
- package/dist/cli/commands/start.js +71 -0
- package/dist/cli/commands/start.js.map +1 -0
- package/dist/cli/commands/status.d.ts +11 -0
- package/dist/cli/commands/status.d.ts.map +1 -0
- package/dist/cli/commands/status.js +60 -0
- package/dist/cli/commands/status.js.map +1 -0
- package/dist/cli/commands/stop.d.ts +10 -0
- package/dist/cli/commands/stop.d.ts.map +1 -0
- package/dist/cli/commands/stop.js +89 -0
- package/dist/cli/commands/stop.js.map +1 -0
- package/dist/cli/util/config-discovery.d.ts +8 -0
- package/dist/cli/util/config-discovery.d.ts.map +1 -0
- package/dist/cli/util/config-discovery.js +70 -0
- package/dist/cli/util/config-discovery.js.map +1 -0
- package/dist/cli/util/config-patcher.d.ts +17 -0
- package/dist/cli/util/config-patcher.d.ts.map +1 -0
- package/dist/cli/util/config-patcher.js +439 -0
- package/dist/cli/util/config-patcher.js.map +1 -0
- package/dist/cli/util/frontend-dev.d.ts +8 -0
- package/dist/cli/util/frontend-dev.d.ts.map +1 -0
- package/dist/cli/util/frontend-dev.js +117 -0
- package/dist/cli/util/frontend-dev.js.map +1 -0
- package/dist/cli/util/process.d.ts +5 -0
- package/dist/cli/util/process.d.ts.map +1 -0
- package/dist/cli/util/process.js +17 -0
- package/dist/cli/util/process.js.map +1 -0
- package/dist/cli/util/templates.d.ts +10 -0
- package/dist/cli/util/templates.d.ts.map +1 -0
- package/dist/cli/util/templates.js +157 -0
- package/dist/cli/util/templates.js.map +1 -0
- package/dist/core/AlertSink.d.ts +83 -0
- package/dist/core/AlertSink.d.ts.map +1 -0
- package/dist/core/AlertSink.js +126 -0
- package/dist/core/AlertSink.js.map +1 -0
- package/dist/core/DirectMessageBus.d.ts +88 -0
- package/dist/core/DirectMessageBus.d.ts.map +1 -0
- package/dist/core/DirectMessageBus.js +352 -0
- package/dist/core/DirectMessageBus.js.map +1 -0
- package/dist/core/EndpointResolver.d.ts +111 -0
- package/dist/core/EndpointResolver.d.ts.map +1 -0
- package/dist/core/EndpointResolver.js +336 -0
- package/dist/core/EndpointResolver.js.map +1 -0
- package/dist/core/ForgeContext.d.ts +221 -0
- package/dist/core/ForgeContext.d.ts.map +1 -0
- package/dist/core/ForgeContext.js +1169 -0
- package/dist/core/ForgeContext.js.map +1 -0
- package/dist/core/ForgeEndpoints.d.ts +71 -0
- package/dist/core/ForgeEndpoints.d.ts.map +1 -0
- package/dist/core/ForgeEndpoints.js +442 -0
- package/dist/core/ForgeEndpoints.js.map +1 -0
- package/dist/core/ForgeHost.d.ts +82 -0
- package/dist/core/ForgeHost.d.ts.map +1 -0
- package/dist/core/ForgeHost.js +107 -0
- package/dist/core/ForgeHost.js.map +1 -0
- package/dist/core/ForgePlatform.d.ts +96 -0
- package/dist/core/ForgePlatform.d.ts.map +1 -0
- package/dist/core/ForgePlatform.js +136 -0
- package/dist/core/ForgePlatform.js.map +1 -0
- package/dist/core/ForgeWebSocket.d.ts +56 -0
- package/dist/core/ForgeWebSocket.d.ts.map +1 -0
- package/dist/core/ForgeWebSocket.js +415 -0
- package/dist/core/ForgeWebSocket.js.map +1 -0
- package/dist/core/Ingress.d.ts +329 -0
- package/dist/core/Ingress.d.ts.map +1 -0
- package/dist/core/Ingress.js +694 -0
- package/dist/core/Ingress.js.map +1 -0
- package/dist/core/Interceptors.d.ts +134 -0
- package/dist/core/Interceptors.d.ts.map +1 -0
- package/dist/core/Interceptors.js +416 -0
- package/dist/core/Interceptors.js.map +1 -0
- package/dist/core/Logger.d.ts +20 -0
- package/dist/core/Logger.d.ts.map +1 -0
- package/dist/core/Logger.js +77 -0
- package/dist/core/Logger.js.map +1 -0
- package/dist/core/MessageBus.d.ts +15 -0
- package/dist/core/MessageBus.d.ts.map +1 -0
- package/dist/core/MessageBus.js +18 -0
- package/dist/core/MessageBus.js.map +1 -0
- package/dist/core/Prometheus.d.ts +80 -0
- package/dist/core/Prometheus.d.ts.map +1 -0
- package/dist/core/Prometheus.js +332 -0
- package/dist/core/Prometheus.js.map +1 -0
- package/dist/core/RequestContext.d.ts +214 -0
- package/dist/core/RequestContext.d.ts.map +1 -0
- package/dist/core/RequestContext.js +556 -0
- package/dist/core/RequestContext.js.map +1 -0
- package/dist/core/Router.d.ts +45 -0
- package/dist/core/Router.d.ts.map +1 -0
- package/dist/core/Router.js +285 -0
- package/dist/core/Router.js.map +1 -0
- package/dist/core/RoutingStrategy.d.ts +116 -0
- package/dist/core/RoutingStrategy.d.ts.map +1 -0
- package/dist/core/RoutingStrategy.js +306 -0
- package/dist/core/RoutingStrategy.js.map +1 -0
- package/dist/core/RpcConfig.d.ts +72 -0
- package/dist/core/RpcConfig.d.ts.map +1 -0
- package/dist/core/RpcConfig.js +127 -0
- package/dist/core/RpcConfig.js.map +1 -0
- package/dist/core/SignatureCache.d.ts +81 -0
- package/dist/core/SignatureCache.d.ts.map +1 -0
- package/dist/core/SignatureCache.js +172 -0
- package/dist/core/SignatureCache.js.map +1 -0
- package/dist/core/StaticFileServer.d.ts +34 -0
- package/dist/core/StaticFileServer.d.ts.map +1 -0
- package/dist/core/StaticFileServer.js +497 -0
- package/dist/core/StaticFileServer.js.map +1 -0
- package/dist/core/Supervisor.d.ts +198 -0
- package/dist/core/Supervisor.d.ts.map +1 -0
- package/dist/core/Supervisor.js +1418 -0
- package/dist/core/Supervisor.js.map +1 -0
- package/dist/core/ThreadAllocator.d.ts +52 -0
- package/dist/core/ThreadAllocator.d.ts.map +1 -0
- package/dist/core/ThreadAllocator.js +174 -0
- package/dist/core/ThreadAllocator.js.map +1 -0
- package/dist/core/WorkerChannelManager.d.ts +130 -0
- package/dist/core/WorkerChannelManager.d.ts.map +1 -0
- package/dist/core/WorkerChannelManager.js +956 -0
- package/dist/core/WorkerChannelManager.js.map +1 -0
- package/dist/core/config-enums.d.ts +41 -0
- package/dist/core/config-enums.d.ts.map +1 -0
- package/dist/core/config-enums.js +59 -0
- package/dist/core/config-enums.js.map +1 -0
- package/dist/core/config.d.ts +159 -0
- package/dist/core/config.d.ts.map +1 -0
- package/dist/core/config.js +694 -0
- package/dist/core/config.js.map +1 -0
- package/dist/core/host-config.d.ts +146 -0
- package/dist/core/host-config.d.ts.map +1 -0
- package/dist/core/host-config.js +312 -0
- package/dist/core/host-config.js.map +1 -0
- package/dist/core/ipc-errors.d.ts +27 -0
- package/dist/core/ipc-errors.d.ts.map +1 -0
- package/dist/core/ipc-errors.js +36 -0
- package/dist/core/ipc-errors.js.map +1 -0
- package/dist/core/network-utils.d.ts +35 -0
- package/dist/core/network-utils.d.ts.map +1 -0
- package/dist/core/network-utils.js +145 -0
- package/dist/core/network-utils.js.map +1 -0
- package/dist/core/platform-config.d.ts +142 -0
- package/dist/core/platform-config.d.ts.map +1 -0
- package/dist/core/platform-config.js +299 -0
- package/dist/core/platform-config.js.map +1 -0
- package/dist/decorators/ServiceProxy.d.ts +175 -0
- package/dist/decorators/ServiceProxy.d.ts.map +1 -0
- package/dist/decorators/ServiceProxy.js +969 -0
- package/dist/decorators/ServiceProxy.js.map +1 -0
- package/dist/decorators/index.d.ts +146 -0
- package/dist/decorators/index.d.ts.map +1 -0
- package/dist/decorators/index.js +545 -0
- package/dist/decorators/index.js.map +1 -0
- package/dist/deploy/NginxGenerator.d.ts +165 -0
- package/dist/deploy/NginxGenerator.d.ts.map +1 -0
- package/dist/deploy/NginxGenerator.js +781 -0
- package/dist/deploy/NginxGenerator.js.map +1 -0
- package/dist/deploy/PlatformManifestGenerator.d.ts +43 -0
- package/dist/deploy/PlatformManifestGenerator.d.ts.map +1 -0
- package/dist/deploy/PlatformManifestGenerator.js +80 -0
- package/dist/deploy/PlatformManifestGenerator.js.map +1 -0
- package/dist/deploy/RouteManifestGenerator.d.ts +42 -0
- package/dist/deploy/RouteManifestGenerator.d.ts.map +1 -0
- package/dist/deploy/RouteManifestGenerator.js +105 -0
- package/dist/deploy/RouteManifestGenerator.js.map +1 -0
- package/dist/deploy/index.d.ts +210 -0
- package/dist/deploy/index.d.ts.map +1 -0
- package/dist/deploy/index.js +918 -0
- package/dist/deploy/index.js.map +1 -0
- package/dist/frontend/FrontendDevLifecycle.d.ts +26 -0
- package/dist/frontend/FrontendDevLifecycle.d.ts.map +1 -0
- package/dist/frontend/FrontendDevLifecycle.js +60 -0
- package/dist/frontend/FrontendDevLifecycle.js.map +1 -0
- package/dist/frontend/FrontendPluginOrchestrator.d.ts +64 -0
- package/dist/frontend/FrontendPluginOrchestrator.d.ts.map +1 -0
- package/dist/frontend/FrontendPluginOrchestrator.js +167 -0
- package/dist/frontend/FrontendPluginOrchestrator.js.map +1 -0
- package/dist/frontend/SiteResolver.d.ts +33 -0
- package/dist/frontend/SiteResolver.d.ts.map +1 -0
- package/dist/frontend/SiteResolver.js +53 -0
- package/dist/frontend/SiteResolver.js.map +1 -0
- package/dist/frontend/StaticMountRegistry.d.ts +36 -0
- package/dist/frontend/StaticMountRegistry.d.ts.map +1 -0
- package/dist/frontend/StaticMountRegistry.js +94 -0
- package/dist/frontend/StaticMountRegistry.js.map +1 -0
- package/dist/frontend/index.d.ts +7 -0
- package/dist/frontend/index.d.ts.map +1 -0
- package/{src → dist}/frontend/index.js +4 -2
- package/dist/frontend/index.js.map +1 -0
- package/dist/frontend/pathUtils.d.ts +8 -0
- package/dist/frontend/pathUtils.d.ts.map +1 -0
- package/dist/frontend/pathUtils.js +17 -0
- package/dist/frontend/pathUtils.js.map +1 -0
- package/dist/frontend/plugins/index.d.ts +2 -0
- package/dist/frontend/plugins/index.d.ts.map +1 -0
- package/{src → dist}/frontend/plugins/index.js +1 -1
- package/dist/frontend/plugins/index.js.map +1 -0
- package/dist/frontend/plugins/viteFrontend.d.ts +51 -0
- package/dist/frontend/plugins/viteFrontend.d.ts.map +1 -0
- package/dist/frontend/plugins/viteFrontend.js +134 -0
- package/dist/frontend/plugins/viteFrontend.js.map +1 -0
- package/dist/frontend/types.d.ts +25 -0
- package/dist/frontend/types.d.ts.map +1 -0
- package/dist/frontend/types.js +2 -0
- package/dist/frontend/types.js.map +1 -0
- package/dist/index.d.ts +17 -0
- package/dist/index.d.ts.map +1 -0
- package/dist/index.js +32 -0
- package/dist/index.js.map +1 -0
- package/dist/internals.d.ts +21 -0
- package/dist/internals.d.ts.map +1 -0
- package/{src → dist}/internals.js +12 -14
- package/dist/internals.js.map +1 -0
- package/dist/plugins/PluginManager.d.ts +209 -0
- package/dist/plugins/PluginManager.d.ts.map +1 -0
- package/dist/plugins/PluginManager.js +365 -0
- package/dist/plugins/PluginManager.js.map +1 -0
- package/dist/plugins/ScopedPostgres.d.ts +78 -0
- package/dist/plugins/ScopedPostgres.d.ts.map +1 -0
- package/dist/plugins/ScopedPostgres.js +190 -0
- package/dist/plugins/ScopedPostgres.js.map +1 -0
- package/dist/plugins/ScopedRedis.d.ts +88 -0
- package/dist/plugins/ScopedRedis.d.ts.map +1 -0
- package/dist/plugins/ScopedRedis.js +169 -0
- package/dist/plugins/ScopedRedis.js.map +1 -0
- package/dist/plugins/index.d.ts +289 -0
- package/dist/plugins/index.d.ts.map +1 -0
- package/dist/plugins/index.js +1942 -0
- package/dist/plugins/index.js.map +1 -0
- package/dist/plugins/types.d.ts +59 -0
- package/dist/plugins/types.d.ts.map +1 -0
- package/dist/plugins/types.js +2 -0
- package/dist/plugins/types.js.map +1 -0
- package/dist/registry/ServiceRegistry.d.ts +305 -0
- package/dist/registry/ServiceRegistry.d.ts.map +1 -0
- package/dist/registry/ServiceRegistry.js +735 -0
- package/dist/registry/ServiceRegistry.js.map +1 -0
- package/dist/scaling/ScaleAdvisor.d.ts +214 -0
- package/dist/scaling/ScaleAdvisor.d.ts.map +1 -0
- package/dist/scaling/ScaleAdvisor.js +526 -0
- package/dist/scaling/ScaleAdvisor.js.map +1 -0
- package/dist/services/Service.d.ts +164 -0
- package/dist/services/Service.d.ts.map +1 -0
- package/dist/services/Service.js +106 -0
- package/dist/services/Service.js.map +1 -0
- package/dist/services/worker-bootstrap.d.ts +15 -0
- package/dist/services/worker-bootstrap.d.ts.map +1 -0
- package/dist/services/worker-bootstrap.js +744 -0
- package/dist/services/worker-bootstrap.js.map +1 -0
- package/dist/templates/auth-service.d.ts +42 -0
- package/dist/templates/auth-service.d.ts.map +1 -0
- package/dist/templates/auth-service.js +54 -0
- package/dist/templates/auth-service.js.map +1 -0
- package/dist/templates/identity-service.d.ts +50 -0
- package/dist/templates/identity-service.d.ts.map +1 -0
- package/dist/templates/identity-service.js +62 -0
- package/dist/templates/identity-service.js.map +1 -0
- package/dist/types/contract.d.ts +120 -0
- package/dist/types/contract.d.ts.map +1 -0
- package/dist/types/contract.js +69 -0
- package/dist/types/contract.js.map +1 -0
- package/package.json +78 -20
- package/src/core/DirectMessageBus.js +0 -364
- package/src/core/EndpointResolver.js +0 -259
- package/src/core/ForgeContext.js +0 -2236
- package/src/core/ForgeHost.js +0 -122
- package/src/core/ForgePlatform.js +0 -145
- package/src/core/Ingress.js +0 -768
- package/src/core/Interceptors.js +0 -420
- package/src/core/MessageBus.js +0 -321
- package/src/core/Prometheus.js +0 -305
- package/src/core/RequestContext.js +0 -413
- package/src/core/RoutingStrategy.js +0 -330
- package/src/core/Supervisor.js +0 -1349
- package/src/core/ThreadAllocator.js +0 -196
- package/src/core/WorkerChannelManager.js +0 -879
- package/src/core/config.js +0 -637
- package/src/core/host-config.js +0 -311
- package/src/core/network-utils.js +0 -166
- package/src/core/platform-config.js +0 -308
- package/src/decorators/ServiceProxy.js +0 -904
- package/src/decorators/index.js +0 -571
- package/src/deploy/NginxGenerator.js +0 -865
- package/src/deploy/PlatformManifestGenerator.js +0 -96
- package/src/deploy/RouteManifestGenerator.js +0 -112
- package/src/deploy/index.js +0 -984
- package/src/frontend/FrontendDevLifecycle.js +0 -65
- package/src/frontend/FrontendPluginOrchestrator.js +0 -187
- package/src/frontend/SiteResolver.js +0 -63
- package/src/frontend/StaticMountRegistry.js +0 -90
- package/src/frontend/plugins/viteFrontend.js +0 -79
- package/src/frontend/types.js +0 -35
- package/src/index.js +0 -58
- package/src/plugins/PluginManager.js +0 -537
- package/src/plugins/ScopedPostgres.js +0 -192
- package/src/plugins/ScopedRedis.js +0 -142
- package/src/plugins/index.js +0 -1756
- package/src/registry/ServiceRegistry.js +0 -797
- package/src/scaling/ScaleAdvisor.js +0 -442
- package/src/services/Service.js +0 -195
- package/src/services/worker-bootstrap.js +0 -679
- package/src/templates/auth-service.js +0 -65
- package/src/templates/identity-service.js +0 -75
|
@@ -0,0 +1,77 @@
|
|
|
1
|
+
const LOG_LEVELS = { error: 0, warn: 1, info: 2, debug: 3 };
|
|
2
|
+
const DEFAULT_REDACTED_FIELDS = new Set([
|
|
3
|
+
"password", "passwd", "secret", "token", "authorization",
|
|
4
|
+
"cookie", "ssn", "creditcard", "credit_card", "cardnumber",
|
|
5
|
+
"card_number", "api_key", "apikey", "access_token", "refresh_token",
|
|
6
|
+
"private_key", "privatekey",
|
|
7
|
+
]);
|
|
8
|
+
const REDACTED = "[REDACTED]";
|
|
9
|
+
function redact(obj, fields) {
|
|
10
|
+
const result = {};
|
|
11
|
+
for (const [key, value] of Object.entries(obj)) {
|
|
12
|
+
if (fields.has(key.toLowerCase())) {
|
|
13
|
+
result[key] = REDACTED;
|
|
14
|
+
}
|
|
15
|
+
else if (value !== null && typeof value === "object" && !Array.isArray(value)) {
|
|
16
|
+
result[key] = redact(value, fields);
|
|
17
|
+
}
|
|
18
|
+
else {
|
|
19
|
+
result[key] = value;
|
|
20
|
+
}
|
|
21
|
+
}
|
|
22
|
+
return result;
|
|
23
|
+
}
|
|
24
|
+
/**
|
|
25
|
+
* O5: Structured logger with service context and log level filtering.
|
|
26
|
+
*/
|
|
27
|
+
export class Logger {
|
|
28
|
+
serviceName;
|
|
29
|
+
workerId;
|
|
30
|
+
_level;
|
|
31
|
+
_redactedFields;
|
|
32
|
+
constructor(serviceName, workerId) {
|
|
33
|
+
this.serviceName = serviceName;
|
|
34
|
+
this.workerId = workerId;
|
|
35
|
+
// O5: Default log level is 'info'
|
|
36
|
+
this._level = LOG_LEVELS[process.env.FORGE_LOG_LEVEL?.toLowerCase() ?? ""] ?? LOG_LEVELS.info;
|
|
37
|
+
this._redactedFields = new Set(DEFAULT_REDACTED_FIELDS);
|
|
38
|
+
}
|
|
39
|
+
/**
|
|
40
|
+
* O5: Set the log level dynamically.
|
|
41
|
+
*/
|
|
42
|
+
setLogLevel(level) {
|
|
43
|
+
const numeric = LOG_LEVELS[level];
|
|
44
|
+
if (numeric === undefined)
|
|
45
|
+
return;
|
|
46
|
+
this._level = numeric;
|
|
47
|
+
}
|
|
48
|
+
_log(level, message, meta = {}) {
|
|
49
|
+
// O5: Filter messages below current log level
|
|
50
|
+
if (LOG_LEVELS[level] > this._level)
|
|
51
|
+
return;
|
|
52
|
+
const safe = redact(meta, this._redactedFields);
|
|
53
|
+
const entry = {
|
|
54
|
+
...safe,
|
|
55
|
+
timestamp: new Date().toISOString(),
|
|
56
|
+
level,
|
|
57
|
+
service: this.serviceName,
|
|
58
|
+
worker: this.workerId,
|
|
59
|
+
message,
|
|
60
|
+
};
|
|
61
|
+
const output = level === "error" ? process.stderr : process.stdout;
|
|
62
|
+
output.write(`${JSON.stringify(entry)}\n`);
|
|
63
|
+
}
|
|
64
|
+
debug(msg, meta) {
|
|
65
|
+
this._log("debug", msg, meta);
|
|
66
|
+
}
|
|
67
|
+
info(msg, meta) {
|
|
68
|
+
this._log("info", msg, meta);
|
|
69
|
+
}
|
|
70
|
+
warn(msg, meta) {
|
|
71
|
+
this._log("warn", msg, meta);
|
|
72
|
+
}
|
|
73
|
+
error(msg, meta) {
|
|
74
|
+
this._log("error", msg, meta);
|
|
75
|
+
}
|
|
76
|
+
}
|
|
77
|
+
//# sourceMappingURL=Logger.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"Logger.js","sourceRoot":"","sources":["../../src/core/Logger.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,15 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* MessageBus
|
|
3
|
+
*
|
|
4
|
+
* @deprecated Removed. Use DirectMessageBus instead.
|
|
5
|
+
*
|
|
6
|
+
* MessageBus used older cluster IPC with round-robin dispatch.
|
|
7
|
+
* DirectMessageBus provides Unix Domain Socket mesh for direct
|
|
8
|
+
* worker-to-worker communication with better performance.
|
|
9
|
+
*
|
|
10
|
+
* See the migration guide at docs/guides/ipc.md
|
|
11
|
+
*/
|
|
12
|
+
export declare class MessageBus {
|
|
13
|
+
constructor();
|
|
14
|
+
}
|
|
15
|
+
//# sourceMappingURL=MessageBus.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"MessageBus.d.ts","sourceRoot":"","sources":["../../src/core/MessageBus.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;GAUG;AACH,qBAAa,UAAU;;CAOtB"}
|
|
@@ -0,0 +1,18 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* MessageBus
|
|
3
|
+
*
|
|
4
|
+
* @deprecated Removed. Use DirectMessageBus instead.
|
|
5
|
+
*
|
|
6
|
+
* MessageBus used older cluster IPC with round-robin dispatch.
|
|
7
|
+
* DirectMessageBus provides Unix Domain Socket mesh for direct
|
|
8
|
+
* worker-to-worker communication with better performance.
|
|
9
|
+
*
|
|
10
|
+
* See the migration guide at docs/guides/ipc.md
|
|
11
|
+
*/
|
|
12
|
+
export class MessageBus {
|
|
13
|
+
constructor() {
|
|
14
|
+
throw new Error("MessageBus is deprecated and has been removed. Use DirectMessageBus instead. " +
|
|
15
|
+
"See the migration guide at docs/guides/ipc.md");
|
|
16
|
+
}
|
|
17
|
+
}
|
|
18
|
+
//# sourceMappingURL=MessageBus.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"MessageBus.js","sourceRoot":"","sources":["../../src/core/MessageBus.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;GAUG;AACH,MAAM,OAAO,UAAU;IACrB;QACE,MAAM,IAAI,KAAK,CACb,+EAA+E;YAC7E,+CAA+C,CAClD,CAAC;IACJ,CAAC;CACF"}
|
|
@@ -0,0 +1,80 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Prometheus Metrics
|
|
3
|
+
*
|
|
4
|
+
* Proper Prometheus exposition format. Auto-tracks:
|
|
5
|
+
*
|
|
6
|
+
* forge_http_request_duration_seconds — histogram, per route + status + tenant
|
|
7
|
+
* forge_http_requests_total — counter, per route + status + tenant
|
|
8
|
+
* forge_http_active_requests — gauge, current in-flight
|
|
9
|
+
* forge_rpc_duration_seconds — histogram, per target + method
|
|
10
|
+
* forge_rpc_total — counter, per target + method + success/error
|
|
11
|
+
* forge_circuit_breaker_state — gauge, 0=closed 1=open 2=half-open
|
|
12
|
+
* forge_concurrency_limit — gauge, per-service adaptive limit
|
|
13
|
+
* forge_concurrency_in_flight — gauge, per-service in-flight
|
|
14
|
+
* forge_ws_connections_active — gauge, current WebSocket connections
|
|
15
|
+
* forge_ws_messages_total — counter, WebSocket messages
|
|
16
|
+
* forge_event_loop_lag_seconds — gauge with percentiles (p50, p95, p99)
|
|
17
|
+
*
|
|
18
|
+
* GET /metrics → text/plain Prometheus format
|
|
19
|
+
*
|
|
20
|
+
* Usage:
|
|
21
|
+
* const metrics = new PrometheusMetrics('users', 0);
|
|
22
|
+
* metrics.httpRequestDuration(0.045, { method: 'GET', path: '/users/:id', status: 200, tenant_id: 't_acme' });
|
|
23
|
+
* metrics.expose() → "# HELP forge_http_request_duration_seconds ..."
|
|
24
|
+
*/
|
|
25
|
+
import { monitorEventLoopDelay } from "node:perf_hooks";
|
|
26
|
+
interface ErrorLike {
|
|
27
|
+
code?: string;
|
|
28
|
+
statusCode?: number;
|
|
29
|
+
}
|
|
30
|
+
type LabelValue = string | number | boolean | null | undefined;
|
|
31
|
+
interface Labels {
|
|
32
|
+
[key: string]: LabelValue;
|
|
33
|
+
}
|
|
34
|
+
interface HistogramEntry {
|
|
35
|
+
count: number;
|
|
36
|
+
sum: number;
|
|
37
|
+
buckets: number[];
|
|
38
|
+
bucketBounds: number[];
|
|
39
|
+
}
|
|
40
|
+
/** P2: Normalize error codes to a finite set to prevent cardinality bombs */
|
|
41
|
+
export declare function normalizeErrorCode(err: ErrorLike): string;
|
|
42
|
+
export declare class PrometheusMetrics {
|
|
43
|
+
serviceName: string;
|
|
44
|
+
workerId: number;
|
|
45
|
+
startTime: number;
|
|
46
|
+
httpBuckets: number[];
|
|
47
|
+
rpcBuckets: number[];
|
|
48
|
+
_counters: Map<string, number>;
|
|
49
|
+
_gauges: Map<string, number>;
|
|
50
|
+
_histograms: Map<string, HistogramEntry>;
|
|
51
|
+
_eventLoopMonitor: ReturnType<typeof monitorEventLoopDelay> | null;
|
|
52
|
+
constructor(serviceName: string, workerId: number);
|
|
53
|
+
httpRequestStart(): void;
|
|
54
|
+
httpRequestEnd(durationSecs: number, labels?: Labels): void;
|
|
55
|
+
rpcCall(durationSecs: number, labels?: Labels): void;
|
|
56
|
+
wsConnectionOpen(): void;
|
|
57
|
+
wsConnectionClose(): void;
|
|
58
|
+
wsMessage(direction?: string): void;
|
|
59
|
+
concurrencyUpdate(target: string, limit: number, inFlight: number): void;
|
|
60
|
+
circuitBreakerState(target: string, state: string): void;
|
|
61
|
+
counter(name: string, value?: number, labels?: Labels): void;
|
|
62
|
+
gauge(name: string, value: number, labels?: Labels): void;
|
|
63
|
+
histogram(name: string, value: number, labels?: Labels, buckets?: number[]): void;
|
|
64
|
+
/**
|
|
65
|
+
* Render all metrics in Prometheus text exposition format.
|
|
66
|
+
*/
|
|
67
|
+
expose(): string;
|
|
68
|
+
_labelStr(labels: Labels): string;
|
|
69
|
+
_counterInc(name: string, labels?: Labels, value?: number): void;
|
|
70
|
+
_gaugeSet(name: string, value: number, labels?: Labels): void;
|
|
71
|
+
_gaugeInc(name: string, labels?: Labels): void;
|
|
72
|
+
_gaugeDec(name: string, labels?: Labels): void;
|
|
73
|
+
_histogramObserve(name: string, value: number, labels: Labels | undefined, buckets: number[]): void;
|
|
74
|
+
/**
|
|
75
|
+
* OPS-8: Cleanup event loop monitor when metrics instance is no longer needed
|
|
76
|
+
*/
|
|
77
|
+
cleanup(): void;
|
|
78
|
+
}
|
|
79
|
+
export {};
|
|
80
|
+
//# sourceMappingURL=Prometheus.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"Prometheus.d.ts","sourceRoot":"","sources":["../../src/core/Prometheus.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,332 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Prometheus Metrics
|
|
3
|
+
*
|
|
4
|
+
* Proper Prometheus exposition format. Auto-tracks:
|
|
5
|
+
*
|
|
6
|
+
* forge_http_request_duration_seconds — histogram, per route + status + tenant
|
|
7
|
+
* forge_http_requests_total — counter, per route + status + tenant
|
|
8
|
+
* forge_http_active_requests — gauge, current in-flight
|
|
9
|
+
* forge_rpc_duration_seconds — histogram, per target + method
|
|
10
|
+
* forge_rpc_total — counter, per target + method + success/error
|
|
11
|
+
* forge_circuit_breaker_state — gauge, 0=closed 1=open 2=half-open
|
|
12
|
+
* forge_concurrency_limit — gauge, per-service adaptive limit
|
|
13
|
+
* forge_concurrency_in_flight — gauge, per-service in-flight
|
|
14
|
+
* forge_ws_connections_active — gauge, current WebSocket connections
|
|
15
|
+
* forge_ws_messages_total — counter, WebSocket messages
|
|
16
|
+
* forge_event_loop_lag_seconds — gauge with percentiles (p50, p95, p99)
|
|
17
|
+
*
|
|
18
|
+
* GET /metrics → text/plain Prometheus format
|
|
19
|
+
*
|
|
20
|
+
* Usage:
|
|
21
|
+
* const metrics = new PrometheusMetrics('users', 0);
|
|
22
|
+
* metrics.httpRequestDuration(0.045, { method: 'GET', path: '/users/:id', status: 200, tenant_id: 't_acme' });
|
|
23
|
+
* metrics.expose() → "# HELP forge_http_request_duration_seconds ..."
|
|
24
|
+
*/
|
|
25
|
+
import { monitorEventLoopDelay } from "node:perf_hooks";
|
|
26
|
+
const ALLOWED_HTTP_LABELS = new Set(["method", "status", "route_pattern", "service", "status_code"]);
|
|
27
|
+
const ALLOWED_RPC_LABELS = new Set(["target", "method", "result", "service"]);
|
|
28
|
+
/** P2: Normalize error codes to a finite set to prevent cardinality bombs */
|
|
29
|
+
export function normalizeErrorCode(err) {
|
|
30
|
+
if (err.code === "ETIMEDOUT" || err.code === "TIMEOUT")
|
|
31
|
+
return "timeout";
|
|
32
|
+
if (err.code === "ECONNREFUSED" || err.code === "ECONNRESET")
|
|
33
|
+
return "connection";
|
|
34
|
+
if (err.statusCode !== undefined && err.statusCode >= 500)
|
|
35
|
+
return "server_error";
|
|
36
|
+
if (err.statusCode !== undefined && err.statusCode >= 400)
|
|
37
|
+
return "client_error";
|
|
38
|
+
if (err.code === "CIRCUIT_OPEN")
|
|
39
|
+
return "circuit_open";
|
|
40
|
+
return "unknown";
|
|
41
|
+
}
|
|
42
|
+
/** P9: Cache label key string construction */
|
|
43
|
+
const _labelKeyCache = new Map();
|
|
44
|
+
const MAX_LABEL_KEY_CACHE = 10000;
|
|
45
|
+
const MAX_METRIC_SERIES = 50000;
|
|
46
|
+
let _seriesCapWarned = false;
|
|
47
|
+
function filterLabels(labels, allowed) {
|
|
48
|
+
const filtered = {};
|
|
49
|
+
for (const key of Object.keys(labels)) {
|
|
50
|
+
if (allowed.has(key))
|
|
51
|
+
filtered[key] = labels[key];
|
|
52
|
+
}
|
|
53
|
+
return filtered;
|
|
54
|
+
}
|
|
55
|
+
export class PrometheusMetrics {
|
|
56
|
+
serviceName;
|
|
57
|
+
workerId;
|
|
58
|
+
startTime;
|
|
59
|
+
httpBuckets;
|
|
60
|
+
rpcBuckets;
|
|
61
|
+
_counters;
|
|
62
|
+
_gauges;
|
|
63
|
+
_histograms;
|
|
64
|
+
_eventLoopMonitor;
|
|
65
|
+
constructor(serviceName, workerId) {
|
|
66
|
+
this.serviceName = serviceName;
|
|
67
|
+
this.workerId = workerId;
|
|
68
|
+
this.startTime = Date.now();
|
|
69
|
+
// Histogram buckets (seconds)
|
|
70
|
+
this.httpBuckets = [0.005, 0.01, 0.025, 0.05, 0.1, 0.25, 0.5, 1, 2.5, 5, 10];
|
|
71
|
+
this.rpcBuckets = [0.001, 0.005, 0.01, 0.025, 0.05, 0.1, 0.5, 1, 5];
|
|
72
|
+
this._counters = new Map();
|
|
73
|
+
this._gauges = new Map();
|
|
74
|
+
this._histograms = new Map();
|
|
75
|
+
// OPS-8: Event loop lag monitoring
|
|
76
|
+
// Resolution of 10ms provides good balance between overhead and accuracy
|
|
77
|
+
// Overhead is minimal (~0.1% CPU) as this uses native perf_hooks
|
|
78
|
+
this._eventLoopMonitor = monitorEventLoopDelay({ resolution: 10 });
|
|
79
|
+
this._eventLoopMonitor.enable();
|
|
80
|
+
}
|
|
81
|
+
// ── Auto-tracked methods ──
|
|
82
|
+
httpRequestStart() {
|
|
83
|
+
this._gaugeInc("forge_http_active_requests", { service: this.serviceName });
|
|
84
|
+
}
|
|
85
|
+
httpRequestEnd(durationSecs, labels = {}) {
|
|
86
|
+
const filtered = filterLabels({ service: this.serviceName, ...labels }, ALLOWED_HTTP_LABELS);
|
|
87
|
+
this._gaugeDec("forge_http_active_requests", { service: this.serviceName });
|
|
88
|
+
this._histogramObserve("forge_http_request_duration_seconds", durationSecs, filtered, this.httpBuckets);
|
|
89
|
+
this._counterInc("forge_http_requests_total", filtered);
|
|
90
|
+
}
|
|
91
|
+
rpcCall(durationSecs, labels = {}) {
|
|
92
|
+
const filtered = filterLabels({ service: this.serviceName, ...labels }, ALLOWED_RPC_LABELS);
|
|
93
|
+
this._histogramObserve("forge_rpc_duration_seconds", durationSecs, {
|
|
94
|
+
source: this.serviceName,
|
|
95
|
+
...filtered,
|
|
96
|
+
}, this.rpcBuckets);
|
|
97
|
+
this._counterInc("forge_rpc_total", {
|
|
98
|
+
source: this.serviceName,
|
|
99
|
+
...filtered,
|
|
100
|
+
});
|
|
101
|
+
}
|
|
102
|
+
wsConnectionOpen() {
|
|
103
|
+
this._gaugeInc("forge_ws_connections_active", { service: this.serviceName });
|
|
104
|
+
}
|
|
105
|
+
wsConnectionClose() {
|
|
106
|
+
this._gaugeDec("forge_ws_connections_active", { service: this.serviceName });
|
|
107
|
+
}
|
|
108
|
+
wsMessage(direction = "inbound") {
|
|
109
|
+
this._counterInc("forge_ws_messages_total", {
|
|
110
|
+
service: this.serviceName,
|
|
111
|
+
direction,
|
|
112
|
+
});
|
|
113
|
+
}
|
|
114
|
+
concurrencyUpdate(target, limit, inFlight) {
|
|
115
|
+
this._gaugeSet("forge_concurrency_limit", limit, {
|
|
116
|
+
source: this.serviceName,
|
|
117
|
+
target,
|
|
118
|
+
});
|
|
119
|
+
this._gaugeSet("forge_concurrency_in_flight", inFlight, {
|
|
120
|
+
source: this.serviceName,
|
|
121
|
+
target,
|
|
122
|
+
});
|
|
123
|
+
}
|
|
124
|
+
circuitBreakerState(target, state) {
|
|
125
|
+
// 0=closed, 1=open, 2=half-open
|
|
126
|
+
const val = state === "open" ? 1 : state === "half-open" ? 2 : 0;
|
|
127
|
+
this._gaugeSet("forge_circuit_breaker_state", val, {
|
|
128
|
+
source: this.serviceName,
|
|
129
|
+
target,
|
|
130
|
+
});
|
|
131
|
+
}
|
|
132
|
+
// ── Generic methods (for custom metrics) ──
|
|
133
|
+
counter(name, value = 1, labels = {}) {
|
|
134
|
+
this._counterInc(name, { service: this.serviceName, ...labels }, value);
|
|
135
|
+
}
|
|
136
|
+
gauge(name, value, labels = {}) {
|
|
137
|
+
this._gaugeSet(name, value, { service: this.serviceName, ...labels });
|
|
138
|
+
}
|
|
139
|
+
histogram(name, value, labels = {}, buckets) {
|
|
140
|
+
this._histogramObserve(name, value, { service: this.serviceName, ...labels }, buckets || this.httpBuckets);
|
|
141
|
+
}
|
|
142
|
+
// ── Exposition ──
|
|
143
|
+
/**
|
|
144
|
+
* Render all metrics in Prometheus text exposition format.
|
|
145
|
+
*/
|
|
146
|
+
expose() {
|
|
147
|
+
const lines = [];
|
|
148
|
+
const now = Date.now();
|
|
149
|
+
// Uptime
|
|
150
|
+
lines.push(`# HELP forge_uptime_seconds Service uptime`);
|
|
151
|
+
lines.push(`# TYPE forge_uptime_seconds gauge`);
|
|
152
|
+
lines.push(`forge_uptime_seconds{service="${this.serviceName}",worker="${this.workerId}"} ${((now - this.startTime) / 1000).toFixed(1)}`);
|
|
153
|
+
lines.push("");
|
|
154
|
+
// Counters
|
|
155
|
+
const counterNames = new Set([...this._counters.keys()].map((k) => k.split("{")[0]));
|
|
156
|
+
for (const name of counterNames) {
|
|
157
|
+
lines.push(`# HELP ${name} ${name}`);
|
|
158
|
+
lines.push(`# TYPE ${name} counter`);
|
|
159
|
+
for (const [key, val] of this._counters) {
|
|
160
|
+
if (key === name || key.startsWith(`${name}{`)) {
|
|
161
|
+
lines.push(`${key} ${val}`);
|
|
162
|
+
}
|
|
163
|
+
}
|
|
164
|
+
lines.push("");
|
|
165
|
+
}
|
|
166
|
+
// Gauges
|
|
167
|
+
const gaugeNames = new Set([...this._gauges.keys()].map((k) => k.split("{")[0]));
|
|
168
|
+
for (const name of gaugeNames) {
|
|
169
|
+
lines.push(`# HELP ${name} ${name}`);
|
|
170
|
+
lines.push(`# TYPE ${name} gauge`);
|
|
171
|
+
for (const [key, val] of this._gauges) {
|
|
172
|
+
if (key === name || key.startsWith(`${name}{`)) {
|
|
173
|
+
lines.push(`${key} ${val}`);
|
|
174
|
+
}
|
|
175
|
+
}
|
|
176
|
+
lines.push("");
|
|
177
|
+
}
|
|
178
|
+
// Histograms
|
|
179
|
+
const histNames = new Set([...this._histograms.keys()].map((k) => k.split("{")[0]));
|
|
180
|
+
for (const name of histNames) {
|
|
181
|
+
lines.push(`# HELP ${name} ${name}`);
|
|
182
|
+
lines.push(`# TYPE ${name} histogram`);
|
|
183
|
+
for (const [key, hist] of this._histograms) {
|
|
184
|
+
if (key !== name && !key.startsWith(`${name}{`))
|
|
185
|
+
continue;
|
|
186
|
+
const labelPart = key.includes("{") ? key.slice(key.indexOf("{")) : "";
|
|
187
|
+
const prefix = name;
|
|
188
|
+
// Bucket lines
|
|
189
|
+
const buckets = hist.bucketBounds;
|
|
190
|
+
for (let i = 0; i < buckets.length; i++) {
|
|
191
|
+
const le = buckets[i];
|
|
192
|
+
const bucketLabel = labelPart ? labelPart.replace("}", `,le="${le}"}`) : `{le="${le}"}`;
|
|
193
|
+
lines.push(`${prefix}_bucket${bucketLabel} ${hist.buckets[i]}`);
|
|
194
|
+
}
|
|
195
|
+
const infLabel = labelPart ? labelPart.replace("}", `,le="+Inf"}`) : `{le="+Inf"}`;
|
|
196
|
+
lines.push(`${prefix}_bucket${infLabel} ${hist.count}`);
|
|
197
|
+
lines.push(`${prefix}_sum${labelPart} ${hist.sum.toFixed(6)}`);
|
|
198
|
+
lines.push(`${prefix}_count${labelPart} ${hist.count}`);
|
|
199
|
+
}
|
|
200
|
+
lines.push("");
|
|
201
|
+
}
|
|
202
|
+
// Process metrics
|
|
203
|
+
const mem = process.memoryUsage();
|
|
204
|
+
lines.push(`# HELP process_resident_memory_bytes Resident memory size in bytes`);
|
|
205
|
+
lines.push(`# TYPE process_resident_memory_bytes gauge`);
|
|
206
|
+
lines.push(`process_resident_memory_bytes{service="${this.serviceName}"} ${mem.rss}`);
|
|
207
|
+
lines.push(`# HELP process_heap_used_bytes Node.js heap used in bytes`);
|
|
208
|
+
lines.push(`# TYPE process_heap_used_bytes gauge`);
|
|
209
|
+
lines.push(`process_heap_used_bytes{service="${this.serviceName}"} ${mem.heapUsed}`);
|
|
210
|
+
lines.push("");
|
|
211
|
+
// OPS-8: Event loop lag metrics
|
|
212
|
+
if (this._eventLoopMonitor) {
|
|
213
|
+
const monitor = this._eventLoopMonitor;
|
|
214
|
+
const baseLabels = `service="${this.serviceName}",worker="${this.workerId}"`;
|
|
215
|
+
lines.push(`# HELP forge_event_loop_lag_seconds Event loop lag in seconds`);
|
|
216
|
+
lines.push(`# TYPE forge_event_loop_lag_seconds gauge`);
|
|
217
|
+
// Convert nanoseconds to seconds for consistency with other duration metrics
|
|
218
|
+
lines.push(`forge_event_loop_lag_seconds{${baseLabels},quantile="0.5"} ${(monitor.percentile(50) / 1e9).toFixed(6)}`);
|
|
219
|
+
lines.push(`forge_event_loop_lag_seconds{${baseLabels},quantile="0.95"} ${(monitor.percentile(95) / 1e9).toFixed(6)}`);
|
|
220
|
+
lines.push(`forge_event_loop_lag_seconds{${baseLabels},quantile="0.99"} ${(monitor.percentile(99) / 1e9).toFixed(6)}`);
|
|
221
|
+
lines.push(`# HELP forge_event_loop_lag_min_seconds Minimum event loop lag in seconds`);
|
|
222
|
+
lines.push(`# TYPE forge_event_loop_lag_min_seconds gauge`);
|
|
223
|
+
lines.push(`forge_event_loop_lag_min_seconds{${baseLabels}} ${(monitor.min / 1e9).toFixed(6)}`);
|
|
224
|
+
lines.push(`# HELP forge_event_loop_lag_max_seconds Maximum event loop lag in seconds`);
|
|
225
|
+
lines.push(`# TYPE forge_event_loop_lag_max_seconds gauge`);
|
|
226
|
+
lines.push(`forge_event_loop_lag_max_seconds{${baseLabels}} ${(monitor.max / 1e9).toFixed(6)}`);
|
|
227
|
+
lines.push(`# HELP forge_event_loop_lag_mean_seconds Mean event loop lag in seconds`);
|
|
228
|
+
lines.push(`# TYPE forge_event_loop_lag_mean_seconds gauge`);
|
|
229
|
+
lines.push(`forge_event_loop_lag_mean_seconds{${baseLabels}} ${(monitor.mean / 1e9).toFixed(6)}`);
|
|
230
|
+
lines.push(`# HELP forge_event_loop_lag_stddev_seconds Event loop lag standard deviation in seconds`);
|
|
231
|
+
lines.push(`# TYPE forge_event_loop_lag_stddev_seconds gauge`);
|
|
232
|
+
lines.push(`forge_event_loop_lag_stddev_seconds{${baseLabels}} ${(monitor.stddev / 1e9).toFixed(6)}`);
|
|
233
|
+
lines.push("");
|
|
234
|
+
}
|
|
235
|
+
return `${lines.join("\n")}\n`;
|
|
236
|
+
}
|
|
237
|
+
// ── Internal ──
|
|
238
|
+
_labelStr(labels) {
|
|
239
|
+
const key = JSON.stringify(labels);
|
|
240
|
+
if (_labelKeyCache.has(key))
|
|
241
|
+
return _labelKeyCache.get(key);
|
|
242
|
+
const validKeyRe = /^[a-zA-Z_][a-zA-Z0-9_]*$/;
|
|
243
|
+
const parts = [];
|
|
244
|
+
for (const k of Object.keys(labels)) {
|
|
245
|
+
const v = labels[k];
|
|
246
|
+
if (v === undefined || v === null)
|
|
247
|
+
continue;
|
|
248
|
+
// H7: Sanitize label keys
|
|
249
|
+
const safeKey = validKeyRe.test(k) ? k : k.replace(/[^a-zA-Z0-9_]/g, "_").replace(/^[^a-zA-Z_]/, "_");
|
|
250
|
+
// Truncate label values to 256 chars and escape special characters
|
|
251
|
+
const safeVal = String(v)
|
|
252
|
+
.slice(0, 256)
|
|
253
|
+
.replace(/\\/g, "\\\\")
|
|
254
|
+
.replace(/"/g, '\\"')
|
|
255
|
+
.replace(/\n/g, "\\n")
|
|
256
|
+
.replace(/\r/g, "\\r")
|
|
257
|
+
.replace(/\t/g, "\\t");
|
|
258
|
+
parts.push(`${safeKey}="${safeVal}"`);
|
|
259
|
+
}
|
|
260
|
+
const result = parts.length > 0 ? `{${parts.join(",")}}` : "";
|
|
261
|
+
// P9-fix: Evict oldest 20% when cache is full (was: stop caching entirely)
|
|
262
|
+
if (_labelKeyCache.size >= MAX_LABEL_KEY_CACHE) {
|
|
263
|
+
const evictCount = Math.ceil(MAX_LABEL_KEY_CACHE * 0.2);
|
|
264
|
+
const iter = _labelKeyCache.keys();
|
|
265
|
+
for (let i = 0; i < evictCount; i++) {
|
|
266
|
+
const k = iter.next().value;
|
|
267
|
+
if (k !== undefined)
|
|
268
|
+
_labelKeyCache.delete(k);
|
|
269
|
+
}
|
|
270
|
+
}
|
|
271
|
+
_labelKeyCache.set(key, result);
|
|
272
|
+
return result;
|
|
273
|
+
}
|
|
274
|
+
_counterInc(name, labels = {}, value = 1) {
|
|
275
|
+
const key = `${name}${this._labelStr(labels)}`;
|
|
276
|
+
if (!this._counters.has(key) && this._counters.size >= MAX_METRIC_SERIES) {
|
|
277
|
+
if (!_seriesCapWarned) {
|
|
278
|
+
_seriesCapWarned = true;
|
|
279
|
+
console.warn(`[PrometheusMetrics] Counter series cap (${MAX_METRIC_SERIES}) reached — dropping new series`);
|
|
280
|
+
}
|
|
281
|
+
return;
|
|
282
|
+
}
|
|
283
|
+
this._counters.set(key, (this._counters.get(key) ?? 0) + value);
|
|
284
|
+
}
|
|
285
|
+
_gaugeSet(name, value, labels = {}) {
|
|
286
|
+
const key = `${name}${this._labelStr(labels)}`;
|
|
287
|
+
this._gauges.set(key, value);
|
|
288
|
+
}
|
|
289
|
+
_gaugeInc(name, labels = {}) {
|
|
290
|
+
const key = `${name}${this._labelStr(labels)}`;
|
|
291
|
+
this._gauges.set(key, (this._gauges.get(key) ?? 0) + 1);
|
|
292
|
+
}
|
|
293
|
+
_gaugeDec(name, labels = {}) {
|
|
294
|
+
const key = `${name}${this._labelStr(labels)}`;
|
|
295
|
+
const current = this._gauges.get(key) ?? 0;
|
|
296
|
+
const next = current - 1;
|
|
297
|
+
if (next < 0 && process.env.NODE_ENV !== "production") {
|
|
298
|
+
console.warn(`[forge] gauge underflow: ${name} would go negative (${current} -> ${next}), clamping to 0`);
|
|
299
|
+
}
|
|
300
|
+
this._gauges.set(key, Math.max(0, next));
|
|
301
|
+
}
|
|
302
|
+
_histogramObserve(name, value, labels = {}, buckets) {
|
|
303
|
+
const key = `${name}${this._labelStr(labels)}`;
|
|
304
|
+
let hist = this._histograms.get(key);
|
|
305
|
+
if (!hist) {
|
|
306
|
+
hist = { count: 0, sum: 0, buckets: new Array(buckets.length).fill(0), bucketBounds: buckets };
|
|
307
|
+
this._histograms.set(key, hist);
|
|
308
|
+
}
|
|
309
|
+
hist.count++;
|
|
310
|
+
hist.sum += value;
|
|
311
|
+
// P18: Early break — once value <= buckets[i], all subsequent buckets also match
|
|
312
|
+
// since buckets are sorted ascending. Increment from this point to end.
|
|
313
|
+
for (let i = 0; i < buckets.length; i++) {
|
|
314
|
+
if (value <= buckets[i]) {
|
|
315
|
+
for (let j = i; j < buckets.length; j++) {
|
|
316
|
+
hist.buckets[j]++;
|
|
317
|
+
}
|
|
318
|
+
break;
|
|
319
|
+
}
|
|
320
|
+
}
|
|
321
|
+
}
|
|
322
|
+
/**
|
|
323
|
+
* OPS-8: Cleanup event loop monitor when metrics instance is no longer needed
|
|
324
|
+
*/
|
|
325
|
+
cleanup() {
|
|
326
|
+
if (this._eventLoopMonitor) {
|
|
327
|
+
this._eventLoopMonitor.disable();
|
|
328
|
+
this._eventLoopMonitor = null;
|
|
329
|
+
}
|
|
330
|
+
}
|
|
331
|
+
}
|
|
332
|
+
//# sourceMappingURL=Prometheus.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"Prometheus.js","sourceRoot":"","sources":["../../src/core/Prometheus.ts"],"names":[],"mappings":"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"}
|