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.
Files changed (358) hide show
  1. package/README.md +52 -20
  2. package/bin/forge.js +2 -1058
  3. package/bin/host-commands.d.ts +2 -0
  4. package/bin/host-commands.d.ts.map +1 -0
  5. package/bin/host-commands.js +7 -8
  6. package/bin/platform-commands.d.ts +2 -0
  7. package/bin/platform-commands.d.ts.map +1 -0
  8. package/bin/platform-commands.js +118 -36
  9. package/dist/cli/base-command.d.ts +12 -0
  10. package/dist/cli/base-command.d.ts.map +1 -0
  11. package/dist/cli/base-command.js +25 -0
  12. package/dist/cli/base-command.js.map +1 -0
  13. package/dist/cli/commands/build.d.ts +10 -0
  14. package/dist/cli/commands/build.d.ts.map +1 -0
  15. package/dist/cli/commands/build.js +110 -0
  16. package/dist/cli/commands/build.js.map +1 -0
  17. package/dist/cli/commands/deploy.d.ts +12 -0
  18. package/dist/cli/commands/deploy.d.ts.map +1 -0
  19. package/dist/cli/commands/deploy.js +143 -0
  20. package/dist/cli/commands/deploy.js.map +1 -0
  21. package/dist/cli/commands/dev.d.ts +10 -0
  22. package/dist/cli/commands/dev.d.ts.map +1 -0
  23. package/dist/cli/commands/dev.js +138 -0
  24. package/dist/cli/commands/dev.js.map +1 -0
  25. package/dist/cli/commands/generate.d.ts +10 -0
  26. package/dist/cli/commands/generate.d.ts.map +1 -0
  27. package/dist/cli/commands/generate.js +76 -0
  28. package/dist/cli/commands/generate.js.map +1 -0
  29. package/dist/cli/commands/host.d.ts +8 -0
  30. package/dist/cli/commands/host.d.ts.map +1 -0
  31. package/dist/cli/commands/host.js +20 -0
  32. package/dist/cli/commands/host.js.map +1 -0
  33. package/dist/cli/commands/init.d.ts +16 -0
  34. package/dist/cli/commands/init.d.ts.map +1 -0
  35. package/dist/cli/commands/init.js +246 -0
  36. package/dist/cli/commands/init.js.map +1 -0
  37. package/dist/cli/commands/platform.d.ts +8 -0
  38. package/dist/cli/commands/platform.d.ts.map +1 -0
  39. package/dist/cli/commands/platform.js +20 -0
  40. package/dist/cli/commands/platform.js.map +1 -0
  41. package/dist/cli/commands/restart.d.ts +8 -0
  42. package/dist/cli/commands/restart.d.ts.map +1 -0
  43. package/dist/cli/commands/restart.js +13 -0
  44. package/dist/cli/commands/restart.js.map +1 -0
  45. package/dist/cli/commands/scaffold/frontend.d.ts +10 -0
  46. package/dist/cli/commands/scaffold/frontend.d.ts.map +1 -0
  47. package/dist/cli/commands/scaffold/frontend.js +130 -0
  48. package/dist/cli/commands/scaffold/frontend.js.map +1 -0
  49. package/dist/cli/commands/scaffold/react.d.ts +7 -0
  50. package/dist/cli/commands/scaffold/react.d.ts.map +1 -0
  51. package/dist/cli/commands/scaffold/react.js +12 -0
  52. package/dist/cli/commands/scaffold/react.js.map +1 -0
  53. package/dist/cli/commands/scale.d.ts +8 -0
  54. package/dist/cli/commands/scale.d.ts.map +1 -0
  55. package/dist/cli/commands/scale.js +13 -0
  56. package/dist/cli/commands/scale.js.map +1 -0
  57. package/dist/cli/commands/start.d.ts +10 -0
  58. package/dist/cli/commands/start.d.ts.map +1 -0
  59. package/dist/cli/commands/start.js +71 -0
  60. package/dist/cli/commands/start.js.map +1 -0
  61. package/dist/cli/commands/status.d.ts +11 -0
  62. package/dist/cli/commands/status.d.ts.map +1 -0
  63. package/dist/cli/commands/status.js +60 -0
  64. package/dist/cli/commands/status.js.map +1 -0
  65. package/dist/cli/commands/stop.d.ts +10 -0
  66. package/dist/cli/commands/stop.d.ts.map +1 -0
  67. package/dist/cli/commands/stop.js +89 -0
  68. package/dist/cli/commands/stop.js.map +1 -0
  69. package/dist/cli/util/config-discovery.d.ts +8 -0
  70. package/dist/cli/util/config-discovery.d.ts.map +1 -0
  71. package/dist/cli/util/config-discovery.js +70 -0
  72. package/dist/cli/util/config-discovery.js.map +1 -0
  73. package/dist/cli/util/config-patcher.d.ts +17 -0
  74. package/dist/cli/util/config-patcher.d.ts.map +1 -0
  75. package/dist/cli/util/config-patcher.js +439 -0
  76. package/dist/cli/util/config-patcher.js.map +1 -0
  77. package/dist/cli/util/frontend-dev.d.ts +8 -0
  78. package/dist/cli/util/frontend-dev.d.ts.map +1 -0
  79. package/dist/cli/util/frontend-dev.js +117 -0
  80. package/dist/cli/util/frontend-dev.js.map +1 -0
  81. package/dist/cli/util/process.d.ts +5 -0
  82. package/dist/cli/util/process.d.ts.map +1 -0
  83. package/dist/cli/util/process.js +17 -0
  84. package/dist/cli/util/process.js.map +1 -0
  85. package/dist/cli/util/templates.d.ts +10 -0
  86. package/dist/cli/util/templates.d.ts.map +1 -0
  87. package/dist/cli/util/templates.js +157 -0
  88. package/dist/cli/util/templates.js.map +1 -0
  89. package/dist/core/AlertSink.d.ts +83 -0
  90. package/dist/core/AlertSink.d.ts.map +1 -0
  91. package/dist/core/AlertSink.js +126 -0
  92. package/dist/core/AlertSink.js.map +1 -0
  93. package/dist/core/DirectMessageBus.d.ts +88 -0
  94. package/dist/core/DirectMessageBus.d.ts.map +1 -0
  95. package/dist/core/DirectMessageBus.js +352 -0
  96. package/dist/core/DirectMessageBus.js.map +1 -0
  97. package/dist/core/EndpointResolver.d.ts +111 -0
  98. package/dist/core/EndpointResolver.d.ts.map +1 -0
  99. package/dist/core/EndpointResolver.js +336 -0
  100. package/dist/core/EndpointResolver.js.map +1 -0
  101. package/dist/core/ForgeContext.d.ts +221 -0
  102. package/dist/core/ForgeContext.d.ts.map +1 -0
  103. package/dist/core/ForgeContext.js +1169 -0
  104. package/dist/core/ForgeContext.js.map +1 -0
  105. package/dist/core/ForgeEndpoints.d.ts +71 -0
  106. package/dist/core/ForgeEndpoints.d.ts.map +1 -0
  107. package/dist/core/ForgeEndpoints.js +442 -0
  108. package/dist/core/ForgeEndpoints.js.map +1 -0
  109. package/dist/core/ForgeHost.d.ts +82 -0
  110. package/dist/core/ForgeHost.d.ts.map +1 -0
  111. package/dist/core/ForgeHost.js +107 -0
  112. package/dist/core/ForgeHost.js.map +1 -0
  113. package/dist/core/ForgePlatform.d.ts +96 -0
  114. package/dist/core/ForgePlatform.d.ts.map +1 -0
  115. package/dist/core/ForgePlatform.js +136 -0
  116. package/dist/core/ForgePlatform.js.map +1 -0
  117. package/dist/core/ForgeWebSocket.d.ts +56 -0
  118. package/dist/core/ForgeWebSocket.d.ts.map +1 -0
  119. package/dist/core/ForgeWebSocket.js +415 -0
  120. package/dist/core/ForgeWebSocket.js.map +1 -0
  121. package/dist/core/Ingress.d.ts +329 -0
  122. package/dist/core/Ingress.d.ts.map +1 -0
  123. package/dist/core/Ingress.js +694 -0
  124. package/dist/core/Ingress.js.map +1 -0
  125. package/dist/core/Interceptors.d.ts +134 -0
  126. package/dist/core/Interceptors.d.ts.map +1 -0
  127. package/dist/core/Interceptors.js +416 -0
  128. package/dist/core/Interceptors.js.map +1 -0
  129. package/dist/core/Logger.d.ts +20 -0
  130. package/dist/core/Logger.d.ts.map +1 -0
  131. package/dist/core/Logger.js +77 -0
  132. package/dist/core/Logger.js.map +1 -0
  133. package/dist/core/MessageBus.d.ts +15 -0
  134. package/dist/core/MessageBus.d.ts.map +1 -0
  135. package/dist/core/MessageBus.js +18 -0
  136. package/dist/core/MessageBus.js.map +1 -0
  137. package/dist/core/Prometheus.d.ts +80 -0
  138. package/dist/core/Prometheus.d.ts.map +1 -0
  139. package/dist/core/Prometheus.js +332 -0
  140. package/dist/core/Prometheus.js.map +1 -0
  141. package/dist/core/RequestContext.d.ts +214 -0
  142. package/dist/core/RequestContext.d.ts.map +1 -0
  143. package/dist/core/RequestContext.js +556 -0
  144. package/dist/core/RequestContext.js.map +1 -0
  145. package/dist/core/Router.d.ts +45 -0
  146. package/dist/core/Router.d.ts.map +1 -0
  147. package/dist/core/Router.js +285 -0
  148. package/dist/core/Router.js.map +1 -0
  149. package/dist/core/RoutingStrategy.d.ts +116 -0
  150. package/dist/core/RoutingStrategy.d.ts.map +1 -0
  151. package/dist/core/RoutingStrategy.js +306 -0
  152. package/dist/core/RoutingStrategy.js.map +1 -0
  153. package/dist/core/RpcConfig.d.ts +72 -0
  154. package/dist/core/RpcConfig.d.ts.map +1 -0
  155. package/dist/core/RpcConfig.js +127 -0
  156. package/dist/core/RpcConfig.js.map +1 -0
  157. package/dist/core/SignatureCache.d.ts +81 -0
  158. package/dist/core/SignatureCache.d.ts.map +1 -0
  159. package/dist/core/SignatureCache.js +172 -0
  160. package/dist/core/SignatureCache.js.map +1 -0
  161. package/dist/core/StaticFileServer.d.ts +34 -0
  162. package/dist/core/StaticFileServer.d.ts.map +1 -0
  163. package/dist/core/StaticFileServer.js +497 -0
  164. package/dist/core/StaticFileServer.js.map +1 -0
  165. package/dist/core/Supervisor.d.ts +198 -0
  166. package/dist/core/Supervisor.d.ts.map +1 -0
  167. package/dist/core/Supervisor.js +1418 -0
  168. package/dist/core/Supervisor.js.map +1 -0
  169. package/dist/core/ThreadAllocator.d.ts +52 -0
  170. package/dist/core/ThreadAllocator.d.ts.map +1 -0
  171. package/dist/core/ThreadAllocator.js +174 -0
  172. package/dist/core/ThreadAllocator.js.map +1 -0
  173. package/dist/core/WorkerChannelManager.d.ts +130 -0
  174. package/dist/core/WorkerChannelManager.d.ts.map +1 -0
  175. package/dist/core/WorkerChannelManager.js +956 -0
  176. package/dist/core/WorkerChannelManager.js.map +1 -0
  177. package/dist/core/config-enums.d.ts +41 -0
  178. package/dist/core/config-enums.d.ts.map +1 -0
  179. package/dist/core/config-enums.js +59 -0
  180. package/dist/core/config-enums.js.map +1 -0
  181. package/dist/core/config.d.ts +159 -0
  182. package/dist/core/config.d.ts.map +1 -0
  183. package/dist/core/config.js +694 -0
  184. package/dist/core/config.js.map +1 -0
  185. package/dist/core/host-config.d.ts +146 -0
  186. package/dist/core/host-config.d.ts.map +1 -0
  187. package/dist/core/host-config.js +312 -0
  188. package/dist/core/host-config.js.map +1 -0
  189. package/dist/core/ipc-errors.d.ts +27 -0
  190. package/dist/core/ipc-errors.d.ts.map +1 -0
  191. package/dist/core/ipc-errors.js +36 -0
  192. package/dist/core/ipc-errors.js.map +1 -0
  193. package/dist/core/network-utils.d.ts +35 -0
  194. package/dist/core/network-utils.d.ts.map +1 -0
  195. package/dist/core/network-utils.js +145 -0
  196. package/dist/core/network-utils.js.map +1 -0
  197. package/dist/core/platform-config.d.ts +142 -0
  198. package/dist/core/platform-config.d.ts.map +1 -0
  199. package/dist/core/platform-config.js +299 -0
  200. package/dist/core/platform-config.js.map +1 -0
  201. package/dist/decorators/ServiceProxy.d.ts +175 -0
  202. package/dist/decorators/ServiceProxy.d.ts.map +1 -0
  203. package/dist/decorators/ServiceProxy.js +969 -0
  204. package/dist/decorators/ServiceProxy.js.map +1 -0
  205. package/dist/decorators/index.d.ts +146 -0
  206. package/dist/decorators/index.d.ts.map +1 -0
  207. package/dist/decorators/index.js +545 -0
  208. package/dist/decorators/index.js.map +1 -0
  209. package/dist/deploy/NginxGenerator.d.ts +165 -0
  210. package/dist/deploy/NginxGenerator.d.ts.map +1 -0
  211. package/dist/deploy/NginxGenerator.js +781 -0
  212. package/dist/deploy/NginxGenerator.js.map +1 -0
  213. package/dist/deploy/PlatformManifestGenerator.d.ts +43 -0
  214. package/dist/deploy/PlatformManifestGenerator.d.ts.map +1 -0
  215. package/dist/deploy/PlatformManifestGenerator.js +80 -0
  216. package/dist/deploy/PlatformManifestGenerator.js.map +1 -0
  217. package/dist/deploy/RouteManifestGenerator.d.ts +42 -0
  218. package/dist/deploy/RouteManifestGenerator.d.ts.map +1 -0
  219. package/dist/deploy/RouteManifestGenerator.js +105 -0
  220. package/dist/deploy/RouteManifestGenerator.js.map +1 -0
  221. package/dist/deploy/index.d.ts +210 -0
  222. package/dist/deploy/index.d.ts.map +1 -0
  223. package/dist/deploy/index.js +918 -0
  224. package/dist/deploy/index.js.map +1 -0
  225. package/dist/frontend/FrontendDevLifecycle.d.ts +26 -0
  226. package/dist/frontend/FrontendDevLifecycle.d.ts.map +1 -0
  227. package/dist/frontend/FrontendDevLifecycle.js +60 -0
  228. package/dist/frontend/FrontendDevLifecycle.js.map +1 -0
  229. package/dist/frontend/FrontendPluginOrchestrator.d.ts +64 -0
  230. package/dist/frontend/FrontendPluginOrchestrator.d.ts.map +1 -0
  231. package/dist/frontend/FrontendPluginOrchestrator.js +167 -0
  232. package/dist/frontend/FrontendPluginOrchestrator.js.map +1 -0
  233. package/dist/frontend/SiteResolver.d.ts +33 -0
  234. package/dist/frontend/SiteResolver.d.ts.map +1 -0
  235. package/dist/frontend/SiteResolver.js +53 -0
  236. package/dist/frontend/SiteResolver.js.map +1 -0
  237. package/dist/frontend/StaticMountRegistry.d.ts +36 -0
  238. package/dist/frontend/StaticMountRegistry.d.ts.map +1 -0
  239. package/dist/frontend/StaticMountRegistry.js +94 -0
  240. package/dist/frontend/StaticMountRegistry.js.map +1 -0
  241. package/dist/frontend/index.d.ts +7 -0
  242. package/dist/frontend/index.d.ts.map +1 -0
  243. package/{src → dist}/frontend/index.js +4 -2
  244. package/dist/frontend/index.js.map +1 -0
  245. package/dist/frontend/pathUtils.d.ts +8 -0
  246. package/dist/frontend/pathUtils.d.ts.map +1 -0
  247. package/dist/frontend/pathUtils.js +17 -0
  248. package/dist/frontend/pathUtils.js.map +1 -0
  249. package/dist/frontend/plugins/index.d.ts +2 -0
  250. package/dist/frontend/plugins/index.d.ts.map +1 -0
  251. package/{src → dist}/frontend/plugins/index.js +1 -1
  252. package/dist/frontend/plugins/index.js.map +1 -0
  253. package/dist/frontend/plugins/viteFrontend.d.ts +51 -0
  254. package/dist/frontend/plugins/viteFrontend.d.ts.map +1 -0
  255. package/dist/frontend/plugins/viteFrontend.js +134 -0
  256. package/dist/frontend/plugins/viteFrontend.js.map +1 -0
  257. package/dist/frontend/types.d.ts +25 -0
  258. package/dist/frontend/types.d.ts.map +1 -0
  259. package/dist/frontend/types.js +2 -0
  260. package/dist/frontend/types.js.map +1 -0
  261. package/dist/index.d.ts +17 -0
  262. package/dist/index.d.ts.map +1 -0
  263. package/dist/index.js +32 -0
  264. package/dist/index.js.map +1 -0
  265. package/dist/internals.d.ts +21 -0
  266. package/dist/internals.d.ts.map +1 -0
  267. package/{src → dist}/internals.js +12 -14
  268. package/dist/internals.js.map +1 -0
  269. package/dist/plugins/PluginManager.d.ts +209 -0
  270. package/dist/plugins/PluginManager.d.ts.map +1 -0
  271. package/dist/plugins/PluginManager.js +365 -0
  272. package/dist/plugins/PluginManager.js.map +1 -0
  273. package/dist/plugins/ScopedPostgres.d.ts +78 -0
  274. package/dist/plugins/ScopedPostgres.d.ts.map +1 -0
  275. package/dist/plugins/ScopedPostgres.js +190 -0
  276. package/dist/plugins/ScopedPostgres.js.map +1 -0
  277. package/dist/plugins/ScopedRedis.d.ts +88 -0
  278. package/dist/plugins/ScopedRedis.d.ts.map +1 -0
  279. package/dist/plugins/ScopedRedis.js +169 -0
  280. package/dist/plugins/ScopedRedis.js.map +1 -0
  281. package/dist/plugins/index.d.ts +289 -0
  282. package/dist/plugins/index.d.ts.map +1 -0
  283. package/dist/plugins/index.js +1942 -0
  284. package/dist/plugins/index.js.map +1 -0
  285. package/dist/plugins/types.d.ts +59 -0
  286. package/dist/plugins/types.d.ts.map +1 -0
  287. package/dist/plugins/types.js +2 -0
  288. package/dist/plugins/types.js.map +1 -0
  289. package/dist/registry/ServiceRegistry.d.ts +305 -0
  290. package/dist/registry/ServiceRegistry.d.ts.map +1 -0
  291. package/dist/registry/ServiceRegistry.js +735 -0
  292. package/dist/registry/ServiceRegistry.js.map +1 -0
  293. package/dist/scaling/ScaleAdvisor.d.ts +214 -0
  294. package/dist/scaling/ScaleAdvisor.d.ts.map +1 -0
  295. package/dist/scaling/ScaleAdvisor.js +526 -0
  296. package/dist/scaling/ScaleAdvisor.js.map +1 -0
  297. package/dist/services/Service.d.ts +164 -0
  298. package/dist/services/Service.d.ts.map +1 -0
  299. package/dist/services/Service.js +106 -0
  300. package/dist/services/Service.js.map +1 -0
  301. package/dist/services/worker-bootstrap.d.ts +15 -0
  302. package/dist/services/worker-bootstrap.d.ts.map +1 -0
  303. package/dist/services/worker-bootstrap.js +744 -0
  304. package/dist/services/worker-bootstrap.js.map +1 -0
  305. package/dist/templates/auth-service.d.ts +42 -0
  306. package/dist/templates/auth-service.d.ts.map +1 -0
  307. package/dist/templates/auth-service.js +54 -0
  308. package/dist/templates/auth-service.js.map +1 -0
  309. package/dist/templates/identity-service.d.ts +50 -0
  310. package/dist/templates/identity-service.d.ts.map +1 -0
  311. package/dist/templates/identity-service.js +62 -0
  312. package/dist/templates/identity-service.js.map +1 -0
  313. package/dist/types/contract.d.ts +120 -0
  314. package/dist/types/contract.d.ts.map +1 -0
  315. package/dist/types/contract.js +69 -0
  316. package/dist/types/contract.js.map +1 -0
  317. package/package.json +78 -20
  318. package/src/core/DirectMessageBus.js +0 -364
  319. package/src/core/EndpointResolver.js +0 -259
  320. package/src/core/ForgeContext.js +0 -2236
  321. package/src/core/ForgeHost.js +0 -122
  322. package/src/core/ForgePlatform.js +0 -145
  323. package/src/core/Ingress.js +0 -768
  324. package/src/core/Interceptors.js +0 -420
  325. package/src/core/MessageBus.js +0 -321
  326. package/src/core/Prometheus.js +0 -305
  327. package/src/core/RequestContext.js +0 -413
  328. package/src/core/RoutingStrategy.js +0 -330
  329. package/src/core/Supervisor.js +0 -1349
  330. package/src/core/ThreadAllocator.js +0 -196
  331. package/src/core/WorkerChannelManager.js +0 -879
  332. package/src/core/config.js +0 -637
  333. package/src/core/host-config.js +0 -311
  334. package/src/core/network-utils.js +0 -166
  335. package/src/core/platform-config.js +0 -308
  336. package/src/decorators/ServiceProxy.js +0 -904
  337. package/src/decorators/index.js +0 -571
  338. package/src/deploy/NginxGenerator.js +0 -865
  339. package/src/deploy/PlatformManifestGenerator.js +0 -96
  340. package/src/deploy/RouteManifestGenerator.js +0 -112
  341. package/src/deploy/index.js +0 -984
  342. package/src/frontend/FrontendDevLifecycle.js +0 -65
  343. package/src/frontend/FrontendPluginOrchestrator.js +0 -187
  344. package/src/frontend/SiteResolver.js +0 -63
  345. package/src/frontend/StaticMountRegistry.js +0 -90
  346. package/src/frontend/plugins/viteFrontend.js +0 -79
  347. package/src/frontend/types.js +0 -35
  348. package/src/index.js +0 -58
  349. package/src/plugins/PluginManager.js +0 -537
  350. package/src/plugins/ScopedPostgres.js +0 -192
  351. package/src/plugins/ScopedRedis.js +0 -142
  352. package/src/plugins/index.js +0 -1756
  353. package/src/registry/ServiceRegistry.js +0 -797
  354. package/src/scaling/ScaleAdvisor.js +0 -442
  355. package/src/services/Service.js +0 -195
  356. package/src/services/worker-bootstrap.js +0 -679
  357. package/src/templates/auth-service.js +0 -65
  358. 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":"AAAA;;;;;;;;;;;;;;;;;;;;;;;GAuBG;AAEH,OAAO,EAAE,qBAAqB,EAAE,MAAM,iBAAiB,CAAC;AAKxD,UAAU,SAAS;IACjB,IAAI,CAAC,EAAE,MAAM,CAAC;IACd,UAAU,CAAC,EAAE,MAAM,CAAC;CACrB;AAED,KAAK,UAAU,GAAG,MAAM,GAAG,MAAM,GAAG,OAAO,GAAG,IAAI,GAAG,SAAS,CAAC;AAE/D,UAAU,MAAM;IACd,CAAC,GAAG,EAAE,MAAM,GAAG,UAAU,CAAC;CAC3B;AAED,UAAU,cAAc;IACtB,KAAK,EAAE,MAAM,CAAC;IACd,GAAG,EAAE,MAAM,CAAC;IACZ,OAAO,EAAE,MAAM,EAAE,CAAC;IAClB,YAAY,EAAE,MAAM,EAAE,CAAC;CACxB;AAED,6EAA6E;AAC7E,wBAAgB,kBAAkB,CAAC,GAAG,EAAE,SAAS,GAAG,MAAM,CAOzD;AAgBD,qBAAa,iBAAiB;IAC5B,WAAW,EAAE,MAAM,CAAC;IACpB,QAAQ,EAAE,MAAM,CAAC;IACjB,SAAS,EAAE,MAAM,CAAC;IAClB,WAAW,EAAE,MAAM,EAAE,CAAC;IACtB,UAAU,EAAE,MAAM,EAAE,CAAC;IACrB,SAAS,EAAE,GAAG,CAAC,MAAM,EAAE,MAAM,CAAC,CAAC;IAC/B,OAAO,EAAE,GAAG,CAAC,MAAM,EAAE,MAAM,CAAC,CAAC;IAC7B,WAAW,EAAE,GAAG,CAAC,MAAM,EAAE,cAAc,CAAC,CAAC;IACzC,iBAAiB,EAAE,UAAU,CAAC,OAAO,qBAAqB,CAAC,GAAG,IAAI,CAAC;gBAEvD,WAAW,EAAE,MAAM,EAAE,QAAQ,EAAE,MAAM;IAsBjD,gBAAgB,IAAI,IAAI;IAIxB,cAAc,CAAC,YAAY,EAAE,MAAM,EAAE,MAAM,GAAE,MAAW,GAAG,IAAI;IAO/D,OAAO,CAAC,YAAY,EAAE,MAAM,EAAE,MAAM,GAAE,MAAW,GAAG,IAAI;IAiBxD,gBAAgB,IAAI,IAAI;IAIxB,iBAAiB,IAAI,IAAI;IAIzB,SAAS,CAAC,SAAS,GAAE,MAAkB,GAAG,IAAI;IAO9C,iBAAiB,CAAC,MAAM,EAAE,MAAM,EAAE,KAAK,EAAE,MAAM,EAAE,QAAQ,EAAE,MAAM,GAAG,IAAI;IAWxE,mBAAmB,CAAC,MAAM,EAAE,MAAM,EAAE,KAAK,EAAE,MAAM,GAAG,IAAI;IAWxD,OAAO,CAAC,IAAI,EAAE,MAAM,EAAE,KAAK,GAAE,MAAU,EAAE,MAAM,GAAE,MAAW,GAAG,IAAI;IAInE,KAAK,CAAC,IAAI,EAAE,MAAM,EAAE,KAAK,EAAE,MAAM,EAAE,MAAM,GAAE,MAAW,GAAG,IAAI;IAI7D,SAAS,CAAC,IAAI,EAAE,MAAM,EAAE,KAAK,EAAE,MAAM,EAAE,MAAM,GAAE,MAAW,EAAE,OAAO,CAAC,EAAE,MAAM,EAAE,GAAG,IAAI;IAMrF;;OAEG;IACH,MAAM,IAAI,MAAM;IA4GhB,SAAS,CAAC,MAAM,EAAE,MAAM,GAAG,MAAM;IAoCjC,WAAW,CAAC,IAAI,EAAE,MAAM,EAAE,MAAM,GAAE,MAAW,EAAE,KAAK,GAAE,MAAU,GAAG,IAAI;IAYvE,SAAS,CAAC,IAAI,EAAE,MAAM,EAAE,KAAK,EAAE,MAAM,EAAE,MAAM,GAAE,MAAW,GAAG,IAAI;IAKjE,SAAS,CAAC,IAAI,EAAE,MAAM,EAAE,MAAM,GAAE,MAAW,GAAG,IAAI;IAKlD,SAAS,CAAC,IAAI,EAAE,MAAM,EAAE,MAAM,GAAE,MAAW,GAAG,IAAI;IAUlD,iBAAiB,CAAC,IAAI,EAAE,MAAM,EAAE,KAAK,EAAE,MAAM,EAAE,MAAM,EAAE,MAAM,YAAK,EAAE,OAAO,EAAE,MAAM,EAAE,GAAG,IAAI;IAqB5F;;OAEG;IACH,OAAO,IAAI,IAAI;CAMhB"}
@@ -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":"AAAA;;;;;;;;;;;;;;;;;;;;;;;GAuBG;AAEH,OAAO,EAAE,qBAAqB,EAAE,MAAM,iBAAiB,CAAC;AAExD,MAAM,mBAAmB,GAAgB,IAAI,GAAG,CAAC,CAAC,QAAQ,EAAE,QAAQ,EAAE,eAAe,EAAE,SAAS,EAAE,aAAa,CAAC,CAAC,CAAC;AAClH,MAAM,kBAAkB,GAAgB,IAAI,GAAG,CAAC,CAAC,QAAQ,EAAE,QAAQ,EAAE,QAAQ,EAAE,SAAS,CAAC,CAAC,CAAC;AAoB3F,6EAA6E;AAC7E,MAAM,UAAU,kBAAkB,CAAC,GAAc;IAC/C,IAAI,GAAG,CAAC,IAAI,KAAK,WAAW,IAAI,GAAG,CAAC,IAAI,KAAK,SAAS;QAAE,OAAO,SAAS,CAAC;IACzE,IAAI,GAAG,CAAC,IAAI,KAAK,cAAc,IAAI,GAAG,CAAC,IAAI,KAAK,YAAY;QAAE,OAAO,YAAY,CAAC;IAClF,IAAI,GAAG,CAAC,UAAU,KAAK,SAAS,IAAI,GAAG,CAAC,UAAU,IAAI,GAAG;QAAE,OAAO,cAAc,CAAC;IACjF,IAAI,GAAG,CAAC,UAAU,KAAK,SAAS,IAAI,GAAG,CAAC,UAAU,IAAI,GAAG;QAAE,OAAO,cAAc,CAAC;IACjF,IAAI,GAAG,CAAC,IAAI,KAAK,cAAc;QAAE,OAAO,cAAc,CAAC;IACvD,OAAO,SAAS,CAAC;AACnB,CAAC;AAED,8CAA8C;AAC9C,MAAM,cAAc,GAAwB,IAAI,GAAG,EAAE,CAAC;AACtD,MAAM,mBAAmB,GAAG,KAAK,CAAC;AAClC,MAAM,iBAAiB,GAAG,KAAK,CAAC;AAChC,IAAI,gBAAgB,GAAG,KAAK,CAAC;AAE7B,SAAS,YAAY,CAAC,MAAc,EAAE,OAAoB;IACxD,MAAM,QAAQ,GAAW,EAAE,CAAC;IAC5B,KAAK,MAAM,GAAG,IAAI,MAAM,CAAC,IAAI,CAAC,MAAM,CAAC,EAAE,CAAC;QACtC,IAAI,OAAO,CAAC,GAAG,CAAC,GAAG,CAAC;YAAE,QAAQ,CAAC,GAAG,CAAC,GAAG,MAAM,CAAC,GAAG,CAAC,CAAC;IACpD,CAAC;IACD,OAAO,QAAQ,CAAC;AAClB,CAAC;AAED,MAAM,OAAO,iBAAiB;IAC5B,WAAW,CAAS;IACpB,QAAQ,CAAS;IACjB,SAAS,CAAS;IAClB,WAAW,CAAW;IACtB,UAAU,CAAW;IACrB,SAAS,CAAsB;IAC/B,OAAO,CAAsB;IAC7B,WAAW,CAA8B;IACzC,iBAAiB,CAAkD;IAEnE,YAAY,WAAmB,EAAE,QAAgB;QAC/C,IAAI,CAAC,WAAW,GAAG,WAAW,CAAC;QAC/B,IAAI,CAAC,QAAQ,GAAG,QAAQ,CAAC;QACzB,IAAI,CAAC,SAAS,GAAG,IAAI,CAAC,GAAG,EAAE,CAAC;QAE5B,8BAA8B;QAC9B,IAAI,CAAC,WAAW,GAAG,CAAC,KAAK,EAAE,IAAI,EAAE,KAAK,EAAE,IAAI,EAAE,GAAG,EAAE,IAAI,EAAE,GAAG,EAAE,CAAC,EAAE,GAAG,EAAE,CAAC,EAAE,EAAE,CAAC,CAAC;QAC7E,IAAI,CAAC,UAAU,GAAG,CAAC,KAAK,EAAE,KAAK,EAAE,IAAI,EAAE,KAAK,EAAE,IAAI,EAAE,GAAG,EAAE,GAAG,EAAE,CAAC,EAAE,CAAC,CAAC,CAAC;QAEpE,IAAI,CAAC,SAAS,GAAG,IAAI,GAAG,EAAE,CAAC;QAC3B,IAAI,CAAC,OAAO,GAAG,IAAI,GAAG,EAAE,CAAC;QACzB,IAAI,CAAC,WAAW,GAAG,IAAI,GAAG,EAAE,CAAC;QAE7B,mCAAmC;QACnC,yEAAyE;QACzE,iEAAiE;QACjE,IAAI,CAAC,iBAAiB,GAAG,qBAAqB,CAAC,EAAE,UAAU,EAAE,EAAE,EAAE,CAAC,CAAC;QACnE,IAAI,CAAC,iBAAiB,CAAC,MAAM,EAAE,CAAC;IAClC,CAAC;IAED,6BAA6B;IAE7B,gBAAgB;QACd,IAAI,CAAC,SAAS,CAAC,4BAA4B,EAAE,EAAE,OAAO,EAAE,IAAI,CAAC,WAAW,EAAE,CAAC,CAAC;IAC9E,CAAC;IAED,cAAc,CAAC,YAAoB,EAAE,SAAiB,EAAE;QACtD,MAAM,QAAQ,GAAG,YAAY,CAAC,EAAE,OAAO,EAAE,IAAI,CAAC,WAAW,EAAE,GAAG,MAAM,EAAE,EAAE,mBAAmB,CAAC,CAAC;QAC7F,IAAI,CAAC,SAAS,CAAC,4BAA4B,EAAE,EAAE,OAAO,EAAE,IAAI,CAAC,WAAW,EAAE,CAAC,CAAC;QAC5E,IAAI,CAAC,iBAAiB,CAAC,qCAAqC,EAAE,YAAY,EAAE,QAAQ,EAAE,IAAI,CAAC,WAAW,CAAC,CAAC;QACxG,IAAI,CAAC,WAAW,CAAC,2BAA2B,EAAE,QAAQ,CAAC,CAAC;IAC1D,CAAC;IAED,OAAO,CAAC,YAAoB,EAAE,SAAiB,EAAE;QAC/C,MAAM,QAAQ,GAAG,YAAY,CAAC,EAAE,OAAO,EAAE,IAAI,CAAC,WAAW,EAAE,GAAG,MAAM,EAAE,EAAE,kBAAkB,CAAC,CAAC;QAC5F,IAAI,CAAC,iBAAiB,CACpB,4BAA4B,EAC5B,YAAY,EACZ;YACE,MAAM,EAAE,IAAI,CAAC,WAAW;YACxB,GAAG,QAAQ;SACZ,EACD,IAAI,CAAC,UAAU,CAChB,CAAC;QACF,IAAI,CAAC,WAAW,CAAC,iBAAiB,EAAE;YAClC,MAAM,EAAE,IAAI,CAAC,WAAW;YACxB,GAAG,QAAQ;SACZ,CAAC,CAAC;IACL,CAAC;IAED,gBAAgB;QACd,IAAI,CAAC,SAAS,CAAC,6BAA6B,EAAE,EAAE,OAAO,EAAE,IAAI,CAAC,WAAW,EAAE,CAAC,CAAC;IAC/E,CAAC;IAED,iBAAiB;QACf,IAAI,CAAC,SAAS,CAAC,6BAA6B,EAAE,EAAE,OAAO,EAAE,IAAI,CAAC,WAAW,EAAE,CAAC,CAAC;IAC/E,CAAC;IAED,SAAS,CAAC,YAAoB,SAAS;QACrC,IAAI,CAAC,WAAW,CAAC,yBAAyB,EAAE;YAC1C,OAAO,EAAE,IAAI,CAAC,WAAW;YACzB,SAAS;SACV,CAAC,CAAC;IACL,CAAC;IAED,iBAAiB,CAAC,MAAc,EAAE,KAAa,EAAE,QAAgB;QAC/D,IAAI,CAAC,SAAS,CAAC,yBAAyB,EAAE,KAAK,EAAE;YAC/C,MAAM,EAAE,IAAI,CAAC,WAAW;YACxB,MAAM;SACP,CAAC,CAAC;QACH,IAAI,CAAC,SAAS,CAAC,6BAA6B,EAAE,QAAQ,EAAE;YACtD,MAAM,EAAE,IAAI,CAAC,WAAW;YACxB,MAAM;SACP,CAAC,CAAC;IACL,CAAC;IAED,mBAAmB,CAAC,MAAc,EAAE,KAAa;QAC/C,gCAAgC;QAChC,MAAM,GAAG,GAAG,KAAK,KAAK,MAAM,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,KAAK,KAAK,WAAW,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;QACjE,IAAI,CAAC,SAAS,CAAC,6BAA6B,EAAE,GAAG,EAAE;YACjD,MAAM,EAAE,IAAI,CAAC,WAAW;YACxB,MAAM;SACP,CAAC,CAAC;IACL,CAAC;IAED,6CAA6C;IAE7C,OAAO,CAAC,IAAY,EAAE,QAAgB,CAAC,EAAE,SAAiB,EAAE;QAC1D,IAAI,CAAC,WAAW,CAAC,IAAI,EAAE,EAAE,OAAO,EAAE,IAAI,CAAC,WAAW,EAAE,GAAG,MAAM,EAAE,EAAE,KAAK,CAAC,CAAC;IAC1E,CAAC;IAED,KAAK,CAAC,IAAY,EAAE,KAAa,EAAE,SAAiB,EAAE;QACpD,IAAI,CAAC,SAAS,CAAC,IAAI,EAAE,KAAK,EAAE,EAAE,OAAO,EAAE,IAAI,CAAC,WAAW,EAAE,GAAG,MAAM,EAAE,CAAC,CAAC;IACxE,CAAC;IAED,SAAS,CAAC,IAAY,EAAE,KAAa,EAAE,SAAiB,EAAE,EAAE,OAAkB;QAC5E,IAAI,CAAC,iBAAiB,CAAC,IAAI,EAAE,KAAK,EAAE,EAAE,OAAO,EAAE,IAAI,CAAC,WAAW,EAAE,GAAG,MAAM,EAAE,EAAE,OAAO,IAAI,IAAI,CAAC,WAAW,CAAC,CAAC;IAC7G,CAAC;IAED,mBAAmB;IAEnB;;OAEG;IACH,MAAM;QACJ,MAAM,KAAK,GAAa,EAAE,CAAC;QAC3B,MAAM,GAAG,GAAG,IAAI,CAAC,GAAG,EAAE,CAAC;QAEvB,SAAS;QACT,KAAK,CAAC,IAAI,CAAC,4CAA4C,CAAC,CAAC;QACzD,KAAK,CAAC,IAAI,CAAC,mCAAmC,CAAC,CAAC;QAChD,KAAK,CAAC,IAAI,CACR,iCAAiC,IAAI,CAAC,WAAW,aAAa,IAAI,CAAC,QAAQ,MAAM,CAAC,CAAC,GAAG,GAAG,IAAI,CAAC,SAAS,CAAC,GAAG,IAAI,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC,EAAE,CAC9H,CAAC;QACF,KAAK,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;QAEf,WAAW;QACX,MAAM,YAAY,GAAG,IAAI,GAAG,CAAC,CAAC,GAAG,IAAI,CAAC,SAAS,CAAC,IAAI,EAAE,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;QACrF,KAAK,MAAM,IAAI,IAAI,YAAY,EAAE,CAAC;YAChC,KAAK,CAAC,IAAI,CAAC,UAAU,IAAI,IAAI,IAAI,EAAE,CAAC,CAAC;YACrC,KAAK,CAAC,IAAI,CAAC,UAAU,IAAI,UAAU,CAAC,CAAC;YACrC,KAAK,MAAM,CAAC,GAAG,EAAE,GAAG,CAAC,IAAI,IAAI,CAAC,SAAS,EAAE,CAAC;gBACxC,IAAI,GAAG,KAAK,IAAI,IAAI,GAAG,CAAC,UAAU,CAAC,GAAG,IAAI,GAAG,CAAC,EAAE,CAAC;oBAC/C,KAAK,CAAC,IAAI,CAAC,GAAG,GAAG,IAAI,GAAG,EAAE,CAAC,CAAC;gBAC9B,CAAC;YACH,CAAC;YACD,KAAK,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;QACjB,CAAC;QAED,SAAS;QACT,MAAM,UAAU,GAAG,IAAI,GAAG,CAAC,CAAC,GAAG,IAAI,CAAC,OAAO,CAAC,IAAI,EAAE,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;QACjF,KAAK,MAAM,IAAI,IAAI,UAAU,EAAE,CAAC;YAC9B,KAAK,CAAC,IAAI,CAAC,UAAU,IAAI,IAAI,IAAI,EAAE,CAAC,CAAC;YACrC,KAAK,CAAC,IAAI,CAAC,UAAU,IAAI,QAAQ,CAAC,CAAC;YACnC,KAAK,MAAM,CAAC,GAAG,EAAE,GAAG,CAAC,IAAI,IAAI,CAAC,OAAO,EAAE,CAAC;gBACtC,IAAI,GAAG,KAAK,IAAI,IAAI,GAAG,CAAC,UAAU,CAAC,GAAG,IAAI,GAAG,CAAC,EAAE,CAAC;oBAC/C,KAAK,CAAC,IAAI,CAAC,GAAG,GAAG,IAAI,GAAG,EAAE,CAAC,CAAC;gBAC9B,CAAC;YACH,CAAC;YACD,KAAK,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;QACjB,CAAC;QAED,aAAa;QACb,MAAM,SAAS,GAAG,IAAI,GAAG,CAAC,CAAC,GAAG,IAAI,CAAC,WAAW,CAAC,IAAI,EAAE,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;QACpF,KAAK,MAAM,IAAI,IAAI,SAAS,EAAE,CAAC;YAC7B,KAAK,CAAC,IAAI,CAAC,UAAU,IAAI,IAAI,IAAI,EAAE,CAAC,CAAC;YACrC,KAAK,CAAC,IAAI,CAAC,UAAU,IAAI,YAAY,CAAC,CAAC;YACvC,KAAK,MAAM,CAAC,GAAG,EAAE,IAAI,CAAC,IAAI,IAAI,CAAC,WAAW,EAAE,CAAC;gBAC3C,IAAI,GAAG,KAAK,IAAI,IAAI,CAAC,GAAG,CAAC,UAAU,CAAC,GAAG,IAAI,GAAG,CAAC;oBAAE,SAAS;gBAC1D,MAAM,SAAS,GAAG,GAAG,CAAC,QAAQ,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC,KAAK,CAAC,GAAG,CAAC,OAAO,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC,EAAE,CAAC;gBACvE,MAAM,MAAM,GAAG,IAAI,CAAC;gBAEpB,eAAe;gBACf,MAAM,OAAO,GAAG,IAAI,CAAC,YAAY,CAAC;gBAClC,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,OAAO,CAAC,MAAM,EAAE,CAAC,EAAE,EAAE,CAAC;oBACxC,MAAM,EAAE,GAAG,OAAO,CAAC,CAAC,CAAC,CAAC;oBACtB,MAAM,WAAW,GAAG,SAAS,CAAC,CAAC,CAAC,SAAS,CAAC,OAAO,CAAC,GAAG,EAAE,QAAQ,EAAE,IAAI,CAAC,CAAC,CAAC,CAAC,QAAQ,EAAE,IAAI,CAAC;oBACxF,KAAK,CAAC,IAAI,CAAC,GAAG,MAAM,UAAU,WAAW,IAAI,IAAI,CAAC,OAAO,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC;gBAClE,CAAC;gBACD,MAAM,QAAQ,GAAG,SAAS,CAAC,CAAC,CAAC,SAAS,CAAC,OAAO,CAAC,GAAG,EAAE,aAAa,CAAC,CAAC,CAAC,CAAC,aAAa,CAAC;gBACnF,KAAK,CAAC,IAAI,CAAC,GAAG,MAAM,UAAU,QAAQ,IAAI,IAAI,CAAC,KAAK,EAAE,CAAC,CAAC;gBACxD,KAAK,CAAC,IAAI,CAAC,GAAG,MAAM,OAAO,SAAS,IAAI,IAAI,CAAC,GAAG,CAAC,OAAO,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC;gBAC/D,KAAK,CAAC,IAAI,CAAC,GAAG,MAAM,SAAS,SAAS,IAAI,IAAI,CAAC,KAAK,EAAE,CAAC,CAAC;YAC1D,CAAC;YACD,KAAK,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;QACjB,CAAC;QAED,kBAAkB;QAClB,MAAM,GAAG,GAAG,OAAO,CAAC,WAAW,EAAE,CAAC;QAClC,KAAK,CAAC,IAAI,CAAC,oEAAoE,CAAC,CAAC;QACjF,KAAK,CAAC,IAAI,CAAC,4CAA4C,CAAC,CAAC;QACzD,KAAK,CAAC,IAAI,CAAC,0CAA0C,IAAI,CAAC,WAAW,MAAM,GAAG,CAAC,GAAG,EAAE,CAAC,CAAC;QACtF,KAAK,CAAC,IAAI,CAAC,2DAA2D,CAAC,CAAC;QACxE,KAAK,CAAC,IAAI,CAAC,sCAAsC,CAAC,CAAC;QACnD,KAAK,CAAC,IAAI,CAAC,oCAAoC,IAAI,CAAC,WAAW,MAAM,GAAG,CAAC,QAAQ,EAAE,CAAC,CAAC;QACrF,KAAK,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;QAEf,gCAAgC;QAChC,IAAI,IAAI,CAAC,iBAAiB,EAAE,CAAC;YAC3B,MAAM,OAAO,GAAG,IAAI,CAAC,iBAAiB,CAAC;YACvC,MAAM,UAAU,GAAG,YAAY,IAAI,CAAC,WAAW,aAAa,IAAI,CAAC,QAAQ,GAAG,CAAC;YAE7E,KAAK,CAAC,IAAI,CAAC,+DAA+D,CAAC,CAAC;YAC5E,KAAK,CAAC,IAAI,CAAC,2CAA2C,CAAC,CAAC;YACxD,6EAA6E;YAC7E,KAAK,CAAC,IAAI,CAAC,gCAAgC,UAAU,oBAAoB,CAAC,OAAO,CAAC,UAAU,CAAC,EAAE,CAAC,GAAG,GAAG,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC;YACtH,KAAK,CAAC,IAAI,CAAC,gCAAgC,UAAU,qBAAqB,CAAC,OAAO,CAAC,UAAU,CAAC,EAAE,CAAC,GAAG,GAAG,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC;YACvH,KAAK,CAAC,IAAI,CAAC,gCAAgC,UAAU,qBAAqB,CAAC,OAAO,CAAC,UAAU,CAAC,EAAE,CAAC,GAAG,GAAG,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC;YAEvH,KAAK,CAAC,IAAI,CAAC,2EAA2E,CAAC,CAAC;YACxF,KAAK,CAAC,IAAI,CAAC,+CAA+C,CAAC,CAAC;YAC5D,KAAK,CAAC,IAAI,CAAC,oCAAoC,UAAU,KAAK,CAAC,OAAO,CAAC,GAAG,GAAG,GAAG,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC;YAEhG,KAAK,CAAC,IAAI,CAAC,2EAA2E,CAAC,CAAC;YACxF,KAAK,CAAC,IAAI,CAAC,+CAA+C,CAAC,CAAC;YAC5D,KAAK,CAAC,IAAI,CAAC,oCAAoC,UAAU,KAAK,CAAC,OAAO,CAAC,GAAG,GAAG,GAAG,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC;YAEhG,KAAK,CAAC,IAAI,CAAC,yEAAyE,CAAC,CAAC;YACtF,KAAK,CAAC,IAAI,CAAC,gDAAgD,CAAC,CAAC;YAC7D,KAAK,CAAC,IAAI,CAAC,qCAAqC,UAAU,KAAK,CAAC,OAAO,CAAC,IAAI,GAAG,GAAG,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC;YAElG,KAAK,CAAC,IAAI,CAAC,yFAAyF,CAAC,CAAC;YACtG,KAAK,CAAC,IAAI,CAAC,kDAAkD,CAAC,CAAC;YAC/D,KAAK,CAAC,IAAI,CAAC,uCAAuC,UAAU,KAAK,CAAC,OAAO,CAAC,MAAM,GAAG,GAAG,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC;YACtG,KAAK,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;QACjB,CAAC;QAED,OAAO,GAAG,KAAK,CAAC,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC;IACjC,CAAC;IAED,iBAAiB;IAEjB,SAAS,CAAC,MAAc;QACtB,MAAM,GAAG,GAAG,IAAI,CAAC,SAAS,CAAC,MAAM,CAAC,CAAC;QACnC,IAAI,cAAc,CAAC,GAAG,CAAC,GAAG,CAAC;YAAE,OAAO,cAAc,CAAC,GAAG,CAAC,GAAG,CAAE,CAAC;QAE7D,MAAM,UAAU,GAAG,0BAA0B,CAAC;QAC9C,MAAM,KAAK,GAAa,EAAE,CAAC;QAC3B,KAAK,MAAM,CAAC,IAAI,MAAM,CAAC,IAAI,CAAC,MAAM,CAAC,EAAE,CAAC;YACpC,MAAM,CAAC,GAAG,MAAM,CAAC,CAAC,CAAC,CAAC;YACpB,IAAI,CAAC,KAAK,SAAS,IAAI,CAAC,KAAK,IAAI;gBAAE,SAAS;YAC5C,0BAA0B;YAC1B,MAAM,OAAO,GAAG,UAAU,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,OAAO,CAAC,gBAAgB,EAAE,GAAG,CAAC,CAAC,OAAO,CAAC,aAAa,EAAE,GAAG,CAAC,CAAC;YACtG,mEAAmE;YACnE,MAAM,OAAO,GAAG,MAAM,CAAC,CAAC,CAAC;iBACtB,KAAK,CAAC,CAAC,EAAE,GAAG,CAAC;iBACb,OAAO,CAAC,KAAK,EAAE,MAAM,CAAC;iBACtB,OAAO,CAAC,IAAI,EAAE,KAAK,CAAC;iBACpB,OAAO,CAAC,KAAK,EAAE,KAAK,CAAC;iBACrB,OAAO,CAAC,KAAK,EAAE,KAAK,CAAC;iBACrB,OAAO,CAAC,KAAK,EAAE,KAAK,CAAC,CAAC;YACzB,KAAK,CAAC,IAAI,CAAC,GAAG,OAAO,KAAK,OAAO,GAAG,CAAC,CAAC;QACxC,CAAC;QACD,MAAM,MAAM,GAAG,KAAK,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC,CAAC,IAAI,KAAK,CAAC,IAAI,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC;QAE9D,2EAA2E;QAC3E,IAAI,cAAc,CAAC,IAAI,IAAI,mBAAmB,EAAE,CAAC;YAC/C,MAAM,UAAU,GAAG,IAAI,CAAC,IAAI,CAAC,mBAAmB,GAAG,GAAG,CAAC,CAAC;YACxD,MAAM,IAAI,GAAG,cAAc,CAAC,IAAI,EAAE,CAAC;YACnC,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,UAAU,EAAE,CAAC,EAAE,EAAE,CAAC;gBACpC,MAAM,CAAC,GAAG,IAAI,CAAC,IAAI,EAAE,CAAC,KAAK,CAAC;gBAC5B,IAAI,CAAC,KAAK,SAAS;oBAAE,cAAc,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC;YAChD,CAAC;QACH,CAAC;QACD,cAAc,CAAC,GAAG,CAAC,GAAG,EAAE,MAAM,CAAC,CAAC;QAChC,OAAO,MAAM,CAAC;IAChB,CAAC;IAED,WAAW,CAAC,IAAY,EAAE,SAAiB,EAAE,EAAE,QAAgB,CAAC;QAC9D,MAAM,GAAG,GAAG,GAAG,IAAI,GAAG,IAAI,CAAC,SAAS,CAAC,MAAM,CAAC,EAAE,CAAC;QAC/C,IAAI,CAAC,IAAI,CAAC,SAAS,CAAC,GAAG,CAAC,GAAG,CAAC,IAAI,IAAI,CAAC,SAAS,CAAC,IAAI,IAAI,iBAAiB,EAAE,CAAC;YACzE,IAAI,CAAC,gBAAgB,EAAE,CAAC;gBACtB,gBAAgB,GAAG,IAAI,CAAC;gBACxB,OAAO,CAAC,IAAI,CAAC,2CAA2C,iBAAiB,iCAAiC,CAAC,CAAC;YAC9G,CAAC;YACD,OAAO;QACT,CAAC;QACD,IAAI,CAAC,SAAS,CAAC,GAAG,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,SAAS,CAAC,GAAG,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC,GAAG,KAAK,CAAC,CAAC;IAClE,CAAC;IAED,SAAS,CAAC,IAAY,EAAE,KAAa,EAAE,SAAiB,EAAE;QACxD,MAAM,GAAG,GAAG,GAAG,IAAI,GAAG,IAAI,CAAC,SAAS,CAAC,MAAM,CAAC,EAAE,CAAC;QAC/C,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC,GAAG,EAAE,KAAK,CAAC,CAAC;IAC/B,CAAC;IAED,SAAS,CAAC,IAAY,EAAE,SAAiB,EAAE;QACzC,MAAM,GAAG,GAAG,GAAG,IAAI,GAAG,IAAI,CAAC,SAAS,CAAC,MAAM,CAAC,EAAE,CAAC;QAC/C,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC;IAC1D,CAAC;IAED,SAAS,CAAC,IAAY,EAAE,SAAiB,EAAE;QACzC,MAAM,GAAG,GAAG,GAAG,IAAI,GAAG,IAAI,CAAC,SAAS,CAAC,MAAM,CAAC,EAAE,CAAC;QAC/C,MAAM,OAAO,GAAG,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC;QAC3C,MAAM,IAAI,GAAG,OAAO,GAAG,CAAC,CAAC;QACzB,IAAI,IAAI,GAAG,CAAC,IAAI,OAAO,CAAC,GAAG,CAAC,QAAQ,KAAK,YAAY,EAAE,CAAC;YACtD,OAAO,CAAC,IAAI,CAAC,4BAA4B,IAAI,uBAAuB,OAAO,OAAO,IAAI,kBAAkB,CAAC,CAAC;QAC5G,CAAC;QACD,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC,GAAG,EAAE,IAAI,CAAC,GAAG,CAAC,CAAC,EAAE,IAAI,CAAC,CAAC,CAAC;IAC3C,CAAC;IAED,iBAAiB,CAAC,IAAY,EAAE,KAAa,EAAE,SAAiB,EAAE,EAAE,OAAiB;QACnF,MAAM,GAAG,GAAG,GAAG,IAAI,GAAG,IAAI,CAAC,SAAS,CAAC,MAAM,CAAC,EAAE,CAAC;QAC/C,IAAI,IAAI,GAAG,IAAI,CAAC,WAAW,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC;QACrC,IAAI,CAAC,IAAI,EAAE,CAAC;YACV,IAAI,GAAG,EAAE,KAAK,EAAE,CAAC,EAAE,GAAG,EAAE,CAAC,EAAE,OAAO,EAAE,IAAI,KAAK,CAAC,OAAO,CAAC,MAAM,CAAC,CAAC,IAAI,CAAC,CAAC,CAAa,EAAE,YAAY,EAAE,OAAO,EAAE,CAAC;YAC3G,IAAI,CAAC,WAAW,CAAC,GAAG,CAAC,GAAG,EAAE,IAAI,CAAC,CAAC;QAClC,CAAC;QACD,IAAI,CAAC,KAAK,EAAE,CAAC;QACb,IAAI,CAAC,GAAG,IAAI,KAAK,CAAC;QAClB,iFAAiF;QACjF,wEAAwE;QACxE,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,OAAO,CAAC,MAAM,EAAE,CAAC,EAAE,EAAE,CAAC;YACxC,IAAI,KAAK,IAAI,OAAO,CAAC,CAAC,CAAC,EAAE,CAAC;gBACxB,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,OAAO,CAAC,MAAM,EAAE,CAAC,EAAE,EAAE,CAAC;oBACxC,IAAI,CAAC,OAAO,CAAC,CAAC,CAAC,EAAE,CAAC;gBACpB,CAAC;gBACD,MAAM;YACR,CAAC;QACH,CAAC;IACH,CAAC;IAED;;OAEG;IACH,OAAO;QACL,IAAI,IAAI,CAAC,iBAAiB,EAAE,CAAC;YAC3B,IAAI,CAAC,iBAAiB,CAAC,OAAO,EAAE,CAAC;YACjC,IAAI,CAAC,iBAAiB,GAAG,IAAI,CAAC;QAChC,CAAC;IACH,CAAC;CACF"}