@resolveio/server-lib 22.3.60 → 22.3.62

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 (691) hide show
  1. package/.nodemon.json +5 -0
  2. package/.vscode/settings.json +21 -0
  3. package/AGENTS.md +189 -0
  4. package/README.md +22 -0
  5. package/build_package.sh +5 -0
  6. package/compileDTS.pl +64 -0
  7. package/docs/ai-assistant-nightly-eval.md +65 -0
  8. package/docs/ai-assistant-preflight-checklist.md +23 -0
  9. package/docs/ai-assistant-report-builder-bridge-playbook.md +115 -0
  10. package/eslint-plugin-custom/index.js +7 -0
  11. package/eslint-plugin-custom/rules/no-filter-zero-index.js +44 -0
  12. package/eslint.config.js +103 -0
  13. package/gulpfile.js +216 -0
  14. package/methodAndPublicationListGenerator.py +375 -0
  15. package/mongodbensurers.js +2 -0
  16. package/mongostop.js +3 -0
  17. package/package.json +1 -1
  18. package/scripts/cleanup-bypassed-callmethod-logs.js +616 -0
  19. package/settings.development.json +25 -0
  20. package/settings.development.redacted.json +25 -0
  21. package/src/.env +12 -0
  22. package/src/ai/assistant-core-heuristics.ts +379 -0
  23. package/src/ai/resolveio-platform-intelligence-memory-corpus.ts +185 -0
  24. package/src/ai/resolveio-platform-intelligence-memory.ts +325 -0
  25. package/{ai/resolveio-platform-intelligence-types.d.ts → src/ai/resolveio-platform-intelligence-types.ts} +20 -15
  26. package/src/ai/resolveio-platform-intelligence.ts +462 -0
  27. package/src/client-server-app.ts +12 -0
  28. package/src/collections/ai-terminal-conversation.collection.ts +91 -0
  29. package/src/collections/ai-terminal-issue-report.collection.ts +99 -0
  30. package/src/collections/ai-terminal-message.collection.ts +77 -0
  31. package/src/collections/app-setting.collection.ts +104 -0
  32. package/src/collections/app-status.collection.ts +58 -0
  33. package/src/collections/communication-metric.collection.ts +84 -0
  34. package/src/collections/counter.collection.ts +56 -0
  35. package/src/collections/cron-job-history.collection.ts +94 -0
  36. package/src/collections/cron-job.collection.ts +92 -0
  37. package/src/collections/customer-notification.collection.ts +131 -0
  38. package/src/collections/customer-portal-password.collection.ts +76 -0
  39. package/src/collections/email-history.collection.ts +127 -0
  40. package/src/collections/email-verified.collection.ts +62 -0
  41. package/src/collections/file.collection.ts +74 -0
  42. package/src/collections/flag-update.collection.ts +57 -0
  43. package/src/collections/flag.collection.ts +57 -0
  44. package/src/collections/log-method-latency.collection.ts +77 -0
  45. package/src/collections/log-subscription.collection.ts +80 -0
  46. package/src/collections/log.collection.ts +93 -0
  47. package/src/collections/logged-in-users.collection.ts +67 -0
  48. package/src/collections/monitor-cpu.collection.ts +65 -0
  49. package/src/collections/monitor-function.collection.ts +74 -0
  50. package/src/collections/monitor-memory.collection.ts +77 -0
  51. package/src/collections/monitor-mongo.collection.ts +71 -0
  52. package/src/collections/notification.collection.ts +57 -0
  53. package/src/collections/openai-usage-ledger.collection.ts +77 -0
  54. package/src/collections/report-builder-dashboard-builder.collection.ts +109 -0
  55. package/src/collections/report-builder-library.collection.ts +89 -0
  56. package/src/collections/report-builder-report.collection.ts +184 -0
  57. package/src/collections/user-group.collection.ts +89 -0
  58. package/src/collections/user-guide.collection.ts +57 -0
  59. package/src/collections/user.collection.ts +181 -0
  60. package/src/cron/cron.ts +117 -0
  61. package/src/fixtures/cron-jobs.ts +95 -0
  62. package/src/fixtures/init.ts +35 -0
  63. package/src/http/auth.ts +764 -0
  64. package/src/http/health.ts +7 -0
  65. package/src/http/home.ts +90 -0
  66. package/src/http/slow-query-publication.ts +49 -0
  67. package/src/index.ts +1 -0
  68. package/src/managers/ai-assistant-codex-manager.manager.ts +1131 -0
  69. package/src/managers/communication-metric.manager.ts +82 -0
  70. package/src/managers/cron.manager.ts +333 -0
  71. package/src/managers/customer-notification-content.manager.ts +236 -0
  72. package/src/managers/diagnostic-manager-bootstrap.ts +165 -0
  73. package/src/managers/error-auto-fix.manager.ts +2767 -0
  74. package/src/managers/local-log.manager.ts +113 -0
  75. package/src/managers/method.manager.ts +1827 -0
  76. package/src/managers/mongo.manager.ts +4575 -0
  77. package/src/managers/monitor.manager.ts +507 -0
  78. package/src/managers/openai-usage-ledger.manager.ts +116 -0
  79. package/src/managers/slow-query-verifier.manager.ts +3590 -0
  80. package/src/managers/slow-query.manager.ts +519 -0
  81. package/src/managers/subscription.manager.ts +3128 -0
  82. package/src/managers/websocket.manager.ts +746 -0
  83. package/src/managers/worker-dispatcher.manager.ts +1360 -0
  84. package/src/managers/worker-server.manager.ts +536 -0
  85. package/src/methods/accounts.ts +532 -0
  86. package/src/methods/ai-terminal.ts +23497 -0
  87. package/src/methods/app-settings.ts +114 -0
  88. package/src/methods/aws.ts +649 -0
  89. package/src/methods/collections.ts +641 -0
  90. package/src/methods/counters.ts +69 -0
  91. package/src/methods/cron-jobs.ts +2614 -0
  92. package/src/methods/customer-notifications.ts +458 -0
  93. package/src/methods/diagnostics.ts +616 -0
  94. package/src/methods/flag-updates.ts +7 -0
  95. package/src/methods/flags.ts +7 -0
  96. package/src/methods/logs.ts +657 -0
  97. package/src/methods/mongo-explorer.ts +1880 -0
  98. package/src/methods/monitor.ts +540 -0
  99. package/src/methods/pdf.ts +1236 -0
  100. package/src/methods/publications.ts +129 -0
  101. package/src/methods/report-builder.ts +3300 -0
  102. package/src/methods/support.ts +335 -0
  103. package/src/models/ai-terminal-conversation.model.ts +19 -0
  104. package/src/models/ai-terminal-issue-report.model.ts +21 -0
  105. package/src/models/ai-terminal-message.model.ts +24 -0
  106. package/src/models/app-setting.model.ts +17 -0
  107. package/{models/app-status.model.d.ts → src/models/app-status.model.ts} +3 -2
  108. package/{models/billing-logged-in-users.model.d.ts → src/models/billing-logged-in-users.model.ts} +5 -4
  109. package/src/models/collection-document.model.ts +24 -0
  110. package/src/models/communication-metric.model.ts +23 -0
  111. package/{models/counter.model.d.ts → src/models/counter.model.ts} +4 -3
  112. package/src/models/cron-job-history.model.ts +16 -0
  113. package/src/models/cron-job.model.ts +15 -0
  114. package/src/models/customer-notification.model.ts +28 -0
  115. package/src/models/customer-portal-password.model.ts +12 -0
  116. package/src/models/dialog.model.ts +25 -0
  117. package/{models/email-history.model.js → src/models/email-history.model.ts} +35 -4
  118. package/{models/email-verified.model.d.ts → src/models/email-verified.model.ts} +6 -5
  119. package/{models/file.model.d.ts → src/models/file.model.ts} +8 -7
  120. package/{models/flag-update.model.d.ts → src/models/flag-update.model.ts} +4 -3
  121. package/{models/flag.model.d.ts → src/models/flag.model.ts} +4 -3
  122. package/src/models/log-method-latency.model.ts +11 -0
  123. package/{models/log-subscription.model.d.ts → src/models/log-subscription.model.ts} +11 -9
  124. package/src/models/log.model.ts +19 -0
  125. package/{models/logged-in-users.model.d.ts → src/models/logged-in-users.model.ts} +6 -5
  126. package/{models/method-response.model.d.ts → src/models/method-response.model.ts} +7 -6
  127. package/src/models/method.model.ts +25 -0
  128. package/{models/monitor-cpu.model.d.ts → src/models/monitor-cpu.model.ts} +9 -7
  129. package/src/models/monitor-function.model.ts +16 -0
  130. package/src/models/monitor-memory.model.ts +17 -0
  131. package/src/models/monitor-mongo.model.ts +15 -0
  132. package/{models/notification.model.d.ts → src/models/notification.model.ts} +6 -4
  133. package/src/models/openai-usage-ledger.model.ts +16 -0
  134. package/src/models/pagination.model.ts +35 -0
  135. package/src/models/permission.model.ts +14 -0
  136. package/src/models/report-builder-dashboard-builder.model.ts +29 -0
  137. package/src/models/report-builder-library.model.ts +20 -0
  138. package/src/models/report-builder-report.model.ts +136 -0
  139. package/src/models/report-builder.model.ts +68 -0
  140. package/src/models/select-data-label.model.ts +9 -0
  141. package/src/models/server-message.model.ts +31 -0
  142. package/src/models/slow-query-report.model.ts +23 -0
  143. package/src/models/subscription.model.ts +73 -0
  144. package/src/models/support-ticket.model.ts +104 -0
  145. package/src/models/user-group.model.ts +24 -0
  146. package/{models/user-guide.model.d.ts → src/models/user-guide.model.ts} +5 -4
  147. package/src/models/user.model.ts +96 -0
  148. package/src/private/images/ResolveIO.png +0 -0
  149. package/src/publications/ai-terminal.ts +73 -0
  150. package/src/publications/app-settings.ts +25 -0
  151. package/src/publications/app-status.ts +13 -0
  152. package/src/publications/cron-jobs.ts +40 -0
  153. package/src/publications/customer-notifications.ts +101 -0
  154. package/src/publications/files.ts +33 -0
  155. package/src/publications/flags-update.ts +19 -0
  156. package/src/publications/flags.ts +19 -0
  157. package/src/publications/logs.ts +163 -0
  158. package/src/publications/notifications.ts +13 -0
  159. package/src/publications/report-builder-dashboard-builders.ts +39 -0
  160. package/src/publications/report-builder-libraries.ts +41 -0
  161. package/src/publications/report-builder-reports.ts +47 -0
  162. package/src/publications/super-admin.ts +13 -0
  163. package/src/publications/user-groups.ts +12 -0
  164. package/src/publications/user-guides.ts +12 -0
  165. package/src/resolveio-server-app.ts +617 -0
  166. package/src/server-app.ts +3354 -0
  167. package/src/services/codex-client.ts +1223 -0
  168. package/src/services/openai-client.ts +265 -0
  169. package/src/types/error-report.ts +26 -0
  170. package/src/types/js-tiktoken.d.ts +11 -0
  171. package/src/types/slow-query-report.ts +28 -0
  172. package/src/util/ai-qa-policy.ts +879 -0
  173. package/src/util/ai-runner-artifacts.ts +430 -0
  174. package/src/util/ai-runner-qa-auth.ts +356 -0
  175. package/src/util/ai-runner-qa-tools.ts +803 -0
  176. package/src/util/common.ts +649 -0
  177. package/src/util/customer-portal-password.ts +183 -0
  178. package/src/util/error-reporter.ts +332 -0
  179. package/src/util/error-tracking.ts +79 -0
  180. package/src/util/report-builder-unwinds.ts +180 -0
  181. package/src/util/runner-process-janitor.ts +208 -0
  182. package/src/util/schema-report-builder.ts +448 -0
  183. package/src/util/slow-query-reporter.ts +216 -0
  184. package/src/util/subscription-dependency-context.ts +1096 -0
  185. package/src/util/tokenizer.ts +38 -0
  186. package/src/workers/codex-runner.worker.ts +142 -0
  187. package/start_server.sh +5 -0
  188. package/tests/ai-assistant-corpus-build.ts +484 -0
  189. package/tests/ai-assistant-corpus-replay-e2e.ts +774 -0
  190. package/tests/ai-assistant-data-parity-e2e.ts +1989 -0
  191. package/tests/ai-assistant-eval-triage.ts +831 -0
  192. package/tests/ai-assistant-openai-e2e.ts +1061 -0
  193. package/tests/ai-assistant-openai-git-e2e.ts +155 -0
  194. package/tests/ai-assistant-preflight-matrix.ts +215 -0
  195. package/tests/ai-assistant-routing-eval.test.ts +560 -0
  196. package/tests/ai-assistant-snf-live-eval.ts +975 -0
  197. package/tests/ai-assistant-utils.test.ts +2968 -0
  198. package/tests/ai-runner-contract.test.ts +301 -0
  199. package/tests/error-reporter.test.ts +145 -0
  200. package/tests/method-publication-generator.test.ts +46 -0
  201. package/tests/report-builder-linking.test.ts +79 -0
  202. package/tests/resolveio-platform-intelligence.test.ts +352 -0
  203. package/tests/server-app-cron-owner.test.ts +127 -0
  204. package/tests/subscription-connect-race.test.ts +158 -0
  205. package/tests/subscription-dependency-context.test.ts +324 -0
  206. package/tests/subscription-manager-collection-tracking.test.ts +86 -0
  207. package/tests/subscription-manager-invalidation.test.ts +86 -0
  208. package/tsconfig.json +34 -0
  209. package/ai/assistant-core-heuristics.d.ts +0 -11
  210. package/ai/assistant-core-heuristics.js +0 -356
  211. package/ai/assistant-core-heuristics.js.map +0 -1
  212. package/ai/resolveio-platform-intelligence-memory-corpus.d.ts +0 -3
  213. package/ai/resolveio-platform-intelligence-memory-corpus.js +0 -214
  214. package/ai/resolveio-platform-intelligence-memory-corpus.js.map +0 -1
  215. package/ai/resolveio-platform-intelligence-memory.d.ts +0 -20
  216. package/ai/resolveio-platform-intelligence-memory.js +0 -341
  217. package/ai/resolveio-platform-intelligence-memory.js.map +0 -1
  218. package/ai/resolveio-platform-intelligence-types.js +0 -4
  219. package/ai/resolveio-platform-intelligence-types.js.map +0 -1
  220. package/ai/resolveio-platform-intelligence.d.ts +0 -6
  221. package/ai/resolveio-platform-intelligence.js +0 -463
  222. package/ai/resolveio-platform-intelligence.js.map +0 -1
  223. package/client-server-app.d.ts +0 -1
  224. package/client-server-app.js +0 -68
  225. package/client-server-app.js.map +0 -1
  226. package/collections/ai-terminal-conversation.collection.d.ts +0 -2
  227. package/collections/ai-terminal-conversation.collection.js +0 -140
  228. package/collections/ai-terminal-conversation.collection.js.map +0 -1
  229. package/collections/ai-terminal-issue-report.collection.d.ts +0 -2
  230. package/collections/ai-terminal-issue-report.collection.js +0 -148
  231. package/collections/ai-terminal-issue-report.collection.js.map +0 -1
  232. package/collections/ai-terminal-message.collection.d.ts +0 -2
  233. package/collections/ai-terminal-message.collection.js +0 -121
  234. package/collections/ai-terminal-message.collection.js.map +0 -1
  235. package/collections/app-setting.collection.d.ts +0 -3
  236. package/collections/app-setting.collection.js +0 -103
  237. package/collections/app-setting.collection.js.map +0 -1
  238. package/collections/app-status.collection.d.ts +0 -3
  239. package/collections/app-status.collection.js +0 -57
  240. package/collections/app-status.collection.js.map +0 -1
  241. package/collections/communication-metric.collection.d.ts +0 -2
  242. package/collections/communication-metric.collection.js +0 -133
  243. package/collections/communication-metric.collection.js.map +0 -1
  244. package/collections/counter.collection.d.ts +0 -3
  245. package/collections/counter.collection.js +0 -56
  246. package/collections/counter.collection.js.map +0 -1
  247. package/collections/cron-job-history.collection.d.ts +0 -3
  248. package/collections/cron-job-history.collection.js +0 -137
  249. package/collections/cron-job-history.collection.js.map +0 -1
  250. package/collections/cron-job.collection.d.ts +0 -3
  251. package/collections/cron-job.collection.js +0 -92
  252. package/collections/cron-job.collection.js.map +0 -1
  253. package/collections/customer-notification.collection.d.ts +0 -3
  254. package/collections/customer-notification.collection.js +0 -130
  255. package/collections/customer-notification.collection.js.map +0 -1
  256. package/collections/customer-portal-password.collection.d.ts +0 -3
  257. package/collections/customer-portal-password.collection.js +0 -75
  258. package/collections/customer-portal-password.collection.js.map +0 -1
  259. package/collections/email-history.collection.d.ts +0 -3
  260. package/collections/email-history.collection.js +0 -127
  261. package/collections/email-history.collection.js.map +0 -1
  262. package/collections/email-verified.collection.d.ts +0 -3
  263. package/collections/email-verified.collection.js +0 -62
  264. package/collections/email-verified.collection.js.map +0 -1
  265. package/collections/file.collection.d.ts +0 -3
  266. package/collections/file.collection.js +0 -74
  267. package/collections/file.collection.js.map +0 -1
  268. package/collections/flag-update.collection.d.ts +0 -3
  269. package/collections/flag-update.collection.js +0 -57
  270. package/collections/flag-update.collection.js.map +0 -1
  271. package/collections/flag.collection.d.ts +0 -3
  272. package/collections/flag.collection.js +0 -57
  273. package/collections/flag.collection.js.map +0 -1
  274. package/collections/log-method-latency.collection.d.ts +0 -3
  275. package/collections/log-method-latency.collection.js +0 -77
  276. package/collections/log-method-latency.collection.js.map +0 -1
  277. package/collections/log-subscription.collection.d.ts +0 -3
  278. package/collections/log-subscription.collection.js +0 -80
  279. package/collections/log-subscription.collection.js.map +0 -1
  280. package/collections/log.collection.d.ts +0 -3
  281. package/collections/log.collection.js +0 -93
  282. package/collections/log.collection.js.map +0 -1
  283. package/collections/logged-in-users.collection.d.ts +0 -3
  284. package/collections/logged-in-users.collection.js +0 -67
  285. package/collections/logged-in-users.collection.js.map +0 -1
  286. package/collections/monitor-cpu.collection.d.ts +0 -3
  287. package/collections/monitor-cpu.collection.js +0 -65
  288. package/collections/monitor-cpu.collection.js.map +0 -1
  289. package/collections/monitor-function.collection.d.ts +0 -3
  290. package/collections/monitor-function.collection.js +0 -74
  291. package/collections/monitor-function.collection.js.map +0 -1
  292. package/collections/monitor-memory.collection.d.ts +0 -3
  293. package/collections/monitor-memory.collection.js +0 -77
  294. package/collections/monitor-memory.collection.js.map +0 -1
  295. package/collections/monitor-mongo.collection.d.ts +0 -3
  296. package/collections/monitor-mongo.collection.js +0 -71
  297. package/collections/monitor-mongo.collection.js.map +0 -1
  298. package/collections/notification.collection.d.ts +0 -3
  299. package/collections/notification.collection.js +0 -57
  300. package/collections/notification.collection.js.map +0 -1
  301. package/collections/openai-usage-ledger.collection.d.ts +0 -2
  302. package/collections/openai-usage-ledger.collection.js +0 -124
  303. package/collections/openai-usage-ledger.collection.js.map +0 -1
  304. package/collections/report-builder-dashboard-builder.collection.d.ts +0 -3
  305. package/collections/report-builder-dashboard-builder.collection.js +0 -109
  306. package/collections/report-builder-dashboard-builder.collection.js.map +0 -1
  307. package/collections/report-builder-library.collection.d.ts +0 -3
  308. package/collections/report-builder-library.collection.js +0 -87
  309. package/collections/report-builder-library.collection.js.map +0 -1
  310. package/collections/report-builder-report.collection.d.ts +0 -4
  311. package/collections/report-builder-report.collection.js +0 -184
  312. package/collections/report-builder-report.collection.js.map +0 -1
  313. package/collections/user-group.collection.d.ts +0 -4
  314. package/collections/user-group.collection.js +0 -89
  315. package/collections/user-group.collection.js.map +0 -1
  316. package/collections/user-guide.collection.d.ts +0 -3
  317. package/collections/user-guide.collection.js +0 -57
  318. package/collections/user-guide.collection.js.map +0 -1
  319. package/collections/user.collection.d.ts +0 -4
  320. package/collections/user.collection.js +0 -180
  321. package/collections/user.collection.js.map +0 -1
  322. package/cron/cron.d.ts +0 -14
  323. package/cron/cron.js +0 -216
  324. package/cron/cron.js.map +0 -1
  325. package/fixtures/cron-jobs.d.ts +0 -1
  326. package/fixtures/cron-jobs.js +0 -150
  327. package/fixtures/cron-jobs.js.map +0 -1
  328. package/fixtures/init.d.ts +0 -1
  329. package/fixtures/init.js +0 -91
  330. package/fixtures/init.js.map +0 -1
  331. package/http/auth.d.ts +0 -2
  332. package/http/auth.js +0 -906
  333. package/http/auth.js.map +0 -1
  334. package/http/health.d.ts +0 -1
  335. package/http/health.js +0 -11
  336. package/http/health.js.map +0 -1
  337. package/http/home.d.ts +0 -1
  338. package/http/home.js +0 -134
  339. package/http/home.js.map +0 -1
  340. package/http/slow-query-publication.d.ts +0 -2
  341. package/http/slow-query-publication.js +0 -99
  342. package/http/slow-query-publication.js.map +0 -1
  343. package/index.d.ts +0 -1
  344. package/index.js +0 -19
  345. package/index.js.map +0 -1
  346. package/managers/ai-assistant-codex-manager.manager.d.ts +0 -67
  347. package/managers/ai-assistant-codex-manager.manager.js +0 -1113
  348. package/managers/ai-assistant-codex-manager.manager.js.map +0 -1
  349. package/managers/communication-metric.manager.d.ts +0 -16
  350. package/managers/communication-metric.manager.js +0 -134
  351. package/managers/communication-metric.manager.js.map +0 -1
  352. package/managers/cron.manager.d.ts +0 -20
  353. package/managers/cron.manager.js +0 -534
  354. package/managers/cron.manager.js.map +0 -1
  355. package/managers/customer-notification-content.manager.d.ts +0 -55
  356. package/managers/customer-notification-content.manager.js +0 -158
  357. package/managers/customer-notification-content.manager.js.map +0 -1
  358. package/managers/diagnostic-manager-bootstrap.d.ts +0 -9
  359. package/managers/diagnostic-manager-bootstrap.js +0 -260
  360. package/managers/diagnostic-manager-bootstrap.js.map +0 -1
  361. package/managers/error-auto-fix.manager.d.ts +0 -149
  362. package/managers/error-auto-fix.manager.js +0 -3064
  363. package/managers/error-auto-fix.manager.js.map +0 -1
  364. package/managers/local-log.manager.d.ts +0 -18
  365. package/managers/local-log.manager.js +0 -88
  366. package/managers/local-log.manager.js.map +0 -1
  367. package/managers/method.manager.d.ts +0 -83
  368. package/managers/method.manager.js +0 -1941
  369. package/managers/method.manager.js.map +0 -1
  370. package/managers/mongo.manager.d.ts +0 -224
  371. package/managers/mongo.manager.js +0 -5000
  372. package/managers/mongo.manager.js.map +0 -1
  373. package/managers/monitor.manager.d.ts +0 -70
  374. package/managers/monitor.manager.js +0 -550
  375. package/managers/monitor.manager.js.map +0 -1
  376. package/managers/openai-usage-ledger.manager.d.ts +0 -15
  377. package/managers/openai-usage-ledger.manager.js +0 -144
  378. package/managers/openai-usage-ledger.manager.js.map +0 -1
  379. package/managers/slow-query-verifier.manager.d.ts +0 -144
  380. package/managers/slow-query-verifier.manager.js +0 -3857
  381. package/managers/slow-query-verifier.manager.js.map +0 -1
  382. package/managers/slow-query.manager.d.ts +0 -28
  383. package/managers/slow-query.manager.js +0 -468
  384. package/managers/slow-query.manager.js.map +0 -1
  385. package/managers/subscription.manager.d.ts +0 -169
  386. package/managers/subscription.manager.js +0 -3434
  387. package/managers/subscription.manager.js.map +0 -1
  388. package/managers/websocket.manager.d.ts +0 -73
  389. package/managers/websocket.manager.js +0 -673
  390. package/managers/websocket.manager.js.map +0 -1
  391. package/managers/worker-dispatcher.manager.d.ts +0 -120
  392. package/managers/worker-dispatcher.manager.js +0 -1266
  393. package/managers/worker-dispatcher.manager.js.map +0 -1
  394. package/managers/worker-server.manager.d.ts +0 -35
  395. package/managers/worker-server.manager.js +0 -582
  396. package/managers/worker-server.manager.js.map +0 -1
  397. package/methods/accounts.d.ts +0 -2
  398. package/methods/accounts.js +0 -624
  399. package/methods/accounts.js.map +0 -1
  400. package/methods/ai-terminal.d.ts +0 -337
  401. package/methods/ai-terminal.js +0 -23166
  402. package/methods/ai-terminal.js.map +0 -1
  403. package/methods/app-settings.d.ts +0 -2
  404. package/methods/app-settings.js +0 -169
  405. package/methods/app-settings.js.map +0 -1
  406. package/methods/aws.d.ts +0 -2
  407. package/methods/aws.js +0 -877
  408. package/methods/aws.js.map +0 -1
  409. package/methods/collections.d.ts +0 -2
  410. package/methods/collections.js +0 -719
  411. package/methods/collections.js.map +0 -1
  412. package/methods/counters.d.ts +0 -2
  413. package/methods/counters.js +0 -113
  414. package/methods/counters.js.map +0 -1
  415. package/methods/cron-jobs.d.ts +0 -2
  416. package/methods/cron-jobs.js +0 -2475
  417. package/methods/cron-jobs.js.map +0 -1
  418. package/methods/customer-notifications.d.ts +0 -2
  419. package/methods/customer-notifications.js +0 -528
  420. package/methods/customer-notifications.js.map +0 -1
  421. package/methods/diagnostics.d.ts +0 -2
  422. package/methods/diagnostics.js +0 -703
  423. package/methods/diagnostics.js.map +0 -1
  424. package/methods/flag-updates.d.ts +0 -2
  425. package/methods/flag-updates.js +0 -8
  426. package/methods/flag-updates.js.map +0 -1
  427. package/methods/flags.d.ts +0 -2
  428. package/methods/flags.js +0 -8
  429. package/methods/flags.js.map +0 -1
  430. package/methods/logs.d.ts +0 -2
  431. package/methods/logs.js +0 -751
  432. package/methods/logs.js.map +0 -1
  433. package/methods/mongo-explorer.d.ts +0 -2
  434. package/methods/mongo-explorer.js +0 -1808
  435. package/methods/mongo-explorer.js.map +0 -1
  436. package/methods/monitor.d.ts +0 -2
  437. package/methods/monitor.js +0 -543
  438. package/methods/monitor.js.map +0 -1
  439. package/methods/pdf.d.ts +0 -2
  440. package/methods/pdf.js +0 -1216
  441. package/methods/pdf.js.map +0 -1
  442. package/methods/publications.d.ts +0 -1
  443. package/methods/publications.js +0 -183
  444. package/methods/publications.js.map +0 -1
  445. package/methods/report-builder.d.ts +0 -2
  446. package/methods/report-builder.js +0 -3094
  447. package/methods/report-builder.js.map +0 -1
  448. package/methods/support.d.ts +0 -2
  449. package/methods/support.js +0 -430
  450. package/methods/support.js.map +0 -1
  451. package/models/ai-terminal-conversation.model.d.ts +0 -17
  452. package/models/ai-terminal-conversation.model.js +0 -4
  453. package/models/ai-terminal-conversation.model.js.map +0 -1
  454. package/models/ai-terminal-issue-report.model.d.ts +0 -19
  455. package/models/ai-terminal-issue-report.model.js +0 -4
  456. package/models/ai-terminal-issue-report.model.js.map +0 -1
  457. package/models/ai-terminal-message.model.d.ts +0 -22
  458. package/models/ai-terminal-message.model.js +0 -4
  459. package/models/ai-terminal-message.model.js.map +0 -1
  460. package/models/app-setting.model.d.ts +0 -16
  461. package/models/app-setting.model.js +0 -4
  462. package/models/app-setting.model.js.map +0 -1
  463. package/models/app-status.model.js +0 -4
  464. package/models/app-status.model.js.map +0 -1
  465. package/models/billing-logged-in-users.model.js +0 -4
  466. package/models/billing-logged-in-users.model.js.map +0 -1
  467. package/models/collection-document.model.d.ts +0 -21
  468. package/models/collection-document.model.js +0 -4
  469. package/models/collection-document.model.js.map +0 -1
  470. package/models/communication-metric.model.d.ts +0 -20
  471. package/models/communication-metric.model.js +0 -4
  472. package/models/communication-metric.model.js.map +0 -1
  473. package/models/counter.model.js +0 -4
  474. package/models/counter.model.js.map +0 -1
  475. package/models/cron-job-history.model.d.ts +0 -15
  476. package/models/cron-job-history.model.js +0 -4
  477. package/models/cron-job-history.model.js.map +0 -1
  478. package/models/cron-job.model.d.ts +0 -14
  479. package/models/cron-job.model.js +0 -4
  480. package/models/cron-job.model.js.map +0 -1
  481. package/models/customer-notification.model.d.ts +0 -26
  482. package/models/customer-notification.model.js +0 -4
  483. package/models/customer-notification.model.js.map +0 -1
  484. package/models/customer-portal-password.model.d.ts +0 -11
  485. package/models/customer-portal-password.model.js +0 -4
  486. package/models/customer-portal-password.model.js.map +0 -1
  487. package/models/dialog.model.d.ts +0 -23
  488. package/models/dialog.model.js +0 -4
  489. package/models/dialog.model.js.map +0 -1
  490. package/models/email-history.model.d.ts +0 -31
  491. package/models/email-history.model.js.map +0 -1
  492. package/models/email-verified.model.js +0 -4
  493. package/models/email-verified.model.js.map +0 -1
  494. package/models/file.model.js +0 -4
  495. package/models/file.model.js.map +0 -1
  496. package/models/flag-update.model.js +0 -4
  497. package/models/flag-update.model.js.map +0 -1
  498. package/models/flag.model.js +0 -4
  499. package/models/flag.model.js.map +0 -1
  500. package/models/log-method-latency.model.d.ts +0 -10
  501. package/models/log-method-latency.model.js +0 -4
  502. package/models/log-method-latency.model.js.map +0 -1
  503. package/models/log-subscription.model.js +0 -4
  504. package/models/log-subscription.model.js.map +0 -1
  505. package/models/log.model.d.ts +0 -17
  506. package/models/log.model.js +0 -4
  507. package/models/log.model.js.map +0 -1
  508. package/models/logged-in-users.model.js +0 -4
  509. package/models/logged-in-users.model.js.map +0 -1
  510. package/models/method-response.model.js +0 -4
  511. package/models/method-response.model.js.map +0 -1
  512. package/models/method.model.d.ts +0 -26
  513. package/models/method.model.js +0 -4
  514. package/models/method.model.js.map +0 -1
  515. package/models/monitor-cpu.model.js +0 -4
  516. package/models/monitor-cpu.model.js.map +0 -1
  517. package/models/monitor-function.model.d.ts +0 -14
  518. package/models/monitor-function.model.js +0 -4
  519. package/models/monitor-function.model.js.map +0 -1
  520. package/models/monitor-memory.model.d.ts +0 -15
  521. package/models/monitor-memory.model.js +0 -4
  522. package/models/monitor-memory.model.js.map +0 -1
  523. package/models/monitor-mongo.model.d.ts +0 -13
  524. package/models/monitor-mongo.model.js +0 -4
  525. package/models/monitor-mongo.model.js.map +0 -1
  526. package/models/notification.model.js +0 -4
  527. package/models/notification.model.js.map +0 -1
  528. package/models/openai-usage-ledger.model.d.ts +0 -15
  529. package/models/openai-usage-ledger.model.js +0 -4
  530. package/models/openai-usage-ledger.model.js.map +0 -1
  531. package/models/pagination.model.d.ts +0 -11
  532. package/models/pagination.model.js +0 -28
  533. package/models/pagination.model.js.map +0 -1
  534. package/models/permission.model.d.ts +0 -12
  535. package/models/permission.model.js +0 -4
  536. package/models/permission.model.js.map +0 -1
  537. package/models/report-builder-dashboard-builder.model.d.ts +0 -25
  538. package/models/report-builder-dashboard-builder.model.js +0 -4
  539. package/models/report-builder-dashboard-builder.model.js.map +0 -1
  540. package/models/report-builder-library.model.d.ts +0 -17
  541. package/models/report-builder-library.model.js +0 -4
  542. package/models/report-builder-library.model.js.map +0 -1
  543. package/models/report-builder-report.model.d.ts +0 -121
  544. package/models/report-builder-report.model.js +0 -4
  545. package/models/report-builder-report.model.js.map +0 -1
  546. package/models/report-builder.model.d.ts +0 -61
  547. package/models/report-builder.model.js +0 -4
  548. package/models/report-builder.model.js.map +0 -1
  549. package/models/select-data-label.model.d.ts +0 -9
  550. package/models/select-data-label.model.js +0 -4
  551. package/models/select-data-label.model.js.map +0 -1
  552. package/models/server-message.model.d.ts +0 -32
  553. package/models/server-message.model.js +0 -4
  554. package/models/server-message.model.js.map +0 -1
  555. package/models/slow-query-report.model.d.ts +0 -23
  556. package/models/slow-query-report.model.js +0 -4
  557. package/models/slow-query-report.model.js.map +0 -1
  558. package/models/subscription.model.d.ts +0 -31
  559. package/models/subscription.model.js +0 -4
  560. package/models/subscription.model.js.map +0 -1
  561. package/models/support-ticket.model.d.ts +0 -87
  562. package/models/support-ticket.model.js +0 -4
  563. package/models/support-ticket.model.js.map +0 -1
  564. package/models/user-group.model.d.ts +0 -20
  565. package/models/user-group.model.js +0 -4
  566. package/models/user-group.model.js.map +0 -1
  567. package/models/user-guide.model.js +0 -4
  568. package/models/user-guide.model.js.map +0 -1
  569. package/models/user.model.d.ts +0 -84
  570. package/models/user.model.js +0 -4
  571. package/models/user.model.js.map +0 -1
  572. package/private/images/ResolveIO.png +0 -0
  573. package/public_api.js +0 -116
  574. package/public_api.js.map +0 -1
  575. package/publications/ai-terminal.d.ts +0 -1
  576. package/publications/ai-terminal.js +0 -122
  577. package/publications/ai-terminal.js.map +0 -1
  578. package/publications/app-settings.d.ts +0 -2
  579. package/publications/app-settings.js +0 -28
  580. package/publications/app-settings.js.map +0 -1
  581. package/publications/app-status.d.ts +0 -2
  582. package/publications/app-status.js +0 -16
  583. package/publications/app-status.js.map +0 -1
  584. package/publications/cron-jobs.d.ts +0 -2
  585. package/publications/cron-jobs.js +0 -88
  586. package/publications/cron-jobs.js.map +0 -1
  587. package/publications/customer-notifications.d.ts +0 -2
  588. package/publications/customer-notifications.js +0 -161
  589. package/publications/customer-notifications.js.map +0 -1
  590. package/publications/files.d.ts +0 -2
  591. package/publications/files.js +0 -36
  592. package/publications/files.js.map +0 -1
  593. package/publications/flags-update.d.ts +0 -2
  594. package/publications/flags-update.js +0 -22
  595. package/publications/flags-update.js.map +0 -1
  596. package/publications/flags.d.ts +0 -2
  597. package/publications/flags.js +0 -22
  598. package/publications/flags.js.map +0 -1
  599. package/publications/logs.d.ts +0 -2
  600. package/publications/logs.js +0 -164
  601. package/publications/logs.js.map +0 -1
  602. package/publications/notifications.d.ts +0 -2
  603. package/publications/notifications.js +0 -16
  604. package/publications/notifications.js.map +0 -1
  605. package/publications/report-builder-dashboard-builders.d.ts +0 -2
  606. package/publications/report-builder-dashboard-builders.js +0 -42
  607. package/publications/report-builder-dashboard-builders.js.map +0 -1
  608. package/publications/report-builder-libraries.d.ts +0 -2
  609. package/publications/report-builder-libraries.js +0 -90
  610. package/publications/report-builder-libraries.js.map +0 -1
  611. package/publications/report-builder-reports.d.ts +0 -2
  612. package/publications/report-builder-reports.js +0 -50
  613. package/publications/report-builder-reports.js.map +0 -1
  614. package/publications/super-admin.d.ts +0 -2
  615. package/publications/super-admin.js +0 -16
  616. package/publications/super-admin.js.map +0 -1
  617. package/publications/user-groups.d.ts +0 -1
  618. package/publications/user-groups.js +0 -16
  619. package/publications/user-groups.js.map +0 -1
  620. package/publications/user-guides.d.ts +0 -1
  621. package/publications/user-guides.js +0 -16
  622. package/publications/user-guides.js.map +0 -1
  623. package/resolveio-server-app.d.ts +0 -70
  624. package/resolveio-server-app.js +0 -801
  625. package/resolveio-server-app.js.map +0 -1
  626. package/server-app.d.ts +0 -228
  627. package/server-app.js +0 -3566
  628. package/server-app.js.map +0 -1
  629. package/services/codex-client.d.ts +0 -126
  630. package/services/codex-client.js +0 -1622
  631. package/services/codex-client.js.map +0 -1
  632. package/services/openai-client.d.ts +0 -46
  633. package/services/openai-client.js +0 -318
  634. package/services/openai-client.js.map +0 -1
  635. package/types/error-report.d.ts +0 -25
  636. package/types/error-report.js +0 -4
  637. package/types/error-report.js.map +0 -1
  638. package/types/slow-query-report.d.ts +0 -27
  639. package/types/slow-query-report.js +0 -6
  640. package/types/slow-query-report.js.map +0 -1
  641. package/util/ai-qa-policy.d.ts +0 -124
  642. package/util/ai-qa-policy.js +0 -707
  643. package/util/ai-qa-policy.js.map +0 -1
  644. package/util/ai-runner-artifacts.d.ts +0 -74
  645. package/util/ai-runner-artifacts.js +0 -531
  646. package/util/ai-runner-artifacts.js.map +0 -1
  647. package/util/ai-runner-qa-auth.d.ts +0 -5
  648. package/util/ai-runner-qa-auth.js +0 -357
  649. package/util/ai-runner-qa-auth.js.map +0 -1
  650. package/util/ai-runner-qa-tools.d.ts +0 -19
  651. package/util/ai-runner-qa-tools.js +0 -780
  652. package/util/ai-runner-qa-tools.js.map +0 -1
  653. package/util/common.d.ts +0 -31
  654. package/util/common.js +0 -683
  655. package/util/common.js.map +0 -1
  656. package/util/customer-portal-password.d.ts +0 -13
  657. package/util/customer-portal-password.js +0 -209
  658. package/util/customer-portal-password.js.map +0 -1
  659. package/util/error-reporter.d.ts +0 -52
  660. package/util/error-reporter.js +0 -326
  661. package/util/error-reporter.js.map +0 -1
  662. package/util/error-tracking.d.ts +0 -13
  663. package/util/error-tracking.js +0 -120
  664. package/util/error-tracking.js.map +0 -1
  665. package/util/report-builder-unwinds.d.ts +0 -15
  666. package/util/report-builder-unwinds.js +0 -156
  667. package/util/report-builder-unwinds.js.map +0 -1
  668. package/util/runner-process-janitor.d.ts +0 -25
  669. package/util/runner-process-janitor.js +0 -199
  670. package/util/runner-process-janitor.js.map +0 -1
  671. package/util/schema-report-builder.d.ts +0 -6
  672. package/util/schema-report-builder.js +0 -481
  673. package/util/schema-report-builder.js.map +0 -1
  674. package/util/slow-query-reporter.d.ts +0 -28
  675. package/util/slow-query-reporter.js +0 -226
  676. package/util/slow-query-reporter.js.map +0 -1
  677. package/util/subscription-dependency-context.d.ts +0 -34
  678. package/util/subscription-dependency-context.js +0 -1283
  679. package/util/subscription-dependency-context.js.map +0 -1
  680. package/util/tokenizer.d.ts +0 -5
  681. package/util/tokenizer.js +0 -41
  682. package/util/tokenizer.js.map +0 -1
  683. package/workers/codex-runner.worker.d.ts +0 -1
  684. package/workers/codex-runner.worker.js +0 -192
  685. package/workers/codex-runner.worker.js.map +0 -1
  686. /package/{private → src/private}/email-templates/enrollment.html +0 -0
  687. /package/{private → src/private}/email-templates/forgot-password.html +0 -0
  688. /package/{private → src/private}/email-templates/support-ticket-deleted.html +0 -0
  689. /package/{private → src/private}/email-templates/support-ticket-modified.html +0 -0
  690. /package/{private → src/private}/email-templates/support-ticket.html +0 -0
  691. /package/{public_api.d.ts → src/public_api.ts} +0 -0
@@ -0,0 +1,803 @@
1
+ import { buildRunnerProcessJanitorShellLibrary } from './runner-process-janitor';
2
+
3
+ export interface ResolveIORunnerQaToolBundleOptions {
4
+ mode?: 'support' | 'runner';
5
+ qaClientPort?: number | string;
6
+ defaultUsername?: string;
7
+ defaultPassword?: string;
8
+ jobId?: string;
9
+ ownerId?: string;
10
+ runnerToken?: string;
11
+ toolsBinPath?: string;
12
+ browserslistPath?: string;
13
+ mongodbBinaryCachePath?: string;
14
+ tmpRoot?: string;
15
+ homeRoot?: string;
16
+ }
17
+
18
+ function shellDoubleQuote(value: string): string {
19
+ return String(value || '').replace(/["\\$`]/g, '\\$&');
20
+ }
21
+
22
+ function normalizePort(value: number | string | undefined, fallback: string): string {
23
+ const parsed = Number.parseInt(String(value || ''), 10);
24
+ return Number.isFinite(parsed) && parsed > 0 ? String(parsed) : fallback;
25
+ }
26
+
27
+ function envVar(mode: 'support' | 'runner', suffix: string): string {
28
+ return mode === 'support' ? `RESOLVEIO_SUPPORT_QA_${suffix}` : `RESOLVEIO_RUNNER_QA_${suffix}`;
29
+ }
30
+
31
+ export function buildResolveIORunnerQaEnvScript(options: ResolveIORunnerQaToolBundleOptions = {}): string {
32
+ const mode = options.mode || 'runner';
33
+ const altMode = mode === 'support' ? 'runner' : 'support';
34
+ const tmpRoot = options.tmpRoot || (mode === 'support' ? '/tmp/resolveio-support-qa' : '/tmp/resolveio-ai-runner-qa');
35
+ const homeRoot = options.homeRoot || `${tmpRoot}/home`;
36
+ const defaultPort = normalizePort(options.qaClientPort, '4200');
37
+ const username = shellDoubleQuote(options.defaultUsername || 'admin');
38
+ const password = shellDoubleQuote(options.defaultPassword || '');
39
+ const jobId = shellDoubleQuote(options.jobId || '');
40
+ const ownerId = shellDoubleQuote(options.ownerId || '');
41
+ const runnerToken = shellDoubleQuote(options.runnerToken || '');
42
+ const toolsBinPath = options.toolsBinPath || '';
43
+ const browserslistPath = options.browserslistPath || '';
44
+ const mongodbBinaryCachePath = options.mongodbBinaryCachePath || '';
45
+ const clientPortVar = envVar(mode, 'CLIENT_PORT');
46
+ const altClientPortVar = envVar(altMode, 'CLIENT_PORT');
47
+ const clientUrlVar = envVar(mode, 'CLIENT_URL');
48
+ const altClientUrlVar = envVar(altMode, 'CLIENT_URL');
49
+ const serverUrlVar = envVar(mode, 'SERVER_URL');
50
+ const altServerUrlVar = envVar(altMode, 'SERVER_URL');
51
+ const usernameVar = envVar(mode, 'USERNAME');
52
+ const altUsernameVar = envVar(altMode, 'USERNAME');
53
+ const passwordVar = envVar(mode, 'PASSWORD');
54
+ const altPasswordVar = envVar(altMode, 'PASSWORD');
55
+ const viewportWidthVar = envVar(mode, 'VIEWPORT_WIDTH');
56
+ const altViewportWidthVar = envVar(altMode, 'VIEWPORT_WIDTH');
57
+ const viewportHeightVar = envVar(mode, 'VIEWPORT_HEIGHT');
58
+ const altViewportHeightVar = envVar(altMode, 'VIEWPORT_HEIGHT');
59
+ const timeoutVar = envVar(mode, 'ANGULAR_STARTUP_TIMEOUT_SECONDS');
60
+ const altTimeoutVar = envVar(altMode, 'ANGULAR_STARTUP_TIMEOUT_SECONDS');
61
+ const prebundleVar = envVar(mode, 'ANGULAR_PREBUNDLE');
62
+ const altPrebundleVar = envVar(altMode, 'ANGULAR_PREBUNDLE');
63
+ const reuseVar = envVar(mode, 'REUSE_RUNNING');
64
+ const altReuseVar = envVar(altMode, 'REUSE_RUNNING');
65
+ const keepaliveVar = envVar(mode, 'KEEPALIVE');
66
+ const altKeepaliveVar = envVar(altMode, 'KEEPALIVE');
67
+ const browserLoopVar = mode === 'support' ? 'RESOLVEIO_SUPPORT_QA_BROWSER' : 'RESOLVEIO_RUNNER_QA_BROWSER';
68
+ return [
69
+ `export ${mode === 'support' ? 'RESOLVEIO_SUPPORT_QA_TMP' : 'RESOLVEIO_RUNNER_QA_TMP'}="${'${' + (mode === 'support' ? 'RESOLVEIO_SUPPORT_QA_TMP' : 'RESOLVEIO_RUNNER_QA_TMP') + ':-' + tmpRoot + '}'}"`,
70
+ `export ${mode === 'support' ? 'RESOLVEIO_SUPPORT_QA_HOME' : 'RESOLVEIO_RUNNER_QA_HOME'}="${'${' + (mode === 'support' ? 'RESOLVEIO_SUPPORT_QA_HOME' : 'RESOLVEIO_RUNNER_QA_HOME') + ':-' + homeRoot + '}'}"`,
71
+ `RESOLVEIO_QA_TMP="${'${' + (mode === 'support' ? 'RESOLVEIO_SUPPORT_QA_TMP' : 'RESOLVEIO_RUNNER_QA_TMP') + '}'}"`,
72
+ `RESOLVEIO_QA_HOME="${'${' + (mode === 'support' ? 'RESOLVEIO_SUPPORT_QA_HOME' : 'RESOLVEIO_RUNNER_QA_HOME') + '}'}"`,
73
+ 'mkdir -p "$RESOLVEIO_QA_HOME/.nvm" "$RESOLVEIO_QA_TMP/npm-cache" "$RESOLVEIO_QA_TMP/mongodb-binaries" "$RESOLVEIO_QA_TMP/mongodb-memory-server-core"',
74
+ 'if [ ! -f "$RESOLVEIO_QA_HOME/.nvm/nvm.sh" ]; then',
75
+ ' cat > "$RESOLVEIO_QA_HOME/.nvm/nvm.sh" <<\'RESOLVEIO_NVM_SHIM\'',
76
+ 'nvm() {',
77
+ ' case "$1" in',
78
+ ' use|install|alias) return 0 ;;',
79
+ ' current) node -v 2>/dev/null || true; return 0 ;;',
80
+ ' *) return 0 ;;',
81
+ ' esac',
82
+ '}',
83
+ 'RESOLVEIO_NVM_SHIM',
84
+ 'fi',
85
+ 'export HOME="$RESOLVEIO_QA_HOME"',
86
+ 'export NVM_DIR="$RESOLVEIO_QA_HOME/.nvm"',
87
+ toolsBinPath ? `export PATH="${toolsBinPath}:$PATH"` : '',
88
+ 'export NODE_ENV=development',
89
+ '# Local QA must start the application HTTP server even when launched from a dedicated support-manager worker.',
90
+ 'export IS_WORKERS_ENABLED=false',
91
+ 'export IS_WORKER_INSTANCE=false',
92
+ 'export DISABLE_WORKER_SERVER_CONNECTION=false',
93
+ 'export SUPPORT_CODEX_MANAGER_PROCESS_ONLY=false',
94
+ 'export SUPPORT_AUTO_MANAGER_PROCESS_ONLY=false',
95
+ 'export AI_ASSISTANT_CODEX_MANAGER_PROCESS_ONLY=false',
96
+ browserslistPath ? `export BROWSERSLIST_CONFIG="${browserslistPath}"` : '',
97
+ `export ${clientPortVar}="${'${' + clientPortVar + ':-${' + altClientPortVar + ':-' + defaultPort + '}}'}"`,
98
+ `export ${altClientPortVar}="${'${' + altClientPortVar + ':-${' + clientPortVar + '}}'}"`,
99
+ `export ${clientUrlVar}="${'${' + clientUrlVar + ':-${' + altClientUrlVar + ':-http://localhost:${' + clientPortVar + '}}}'}"`,
100
+ `export ${altClientUrlVar}="${'${' + altClientUrlVar + ':-${' + clientUrlVar + '}}'}"`,
101
+ `export ${serverUrlVar}="${'${' + serverUrlVar + ':-${' + altServerUrlVar + ':-http://localhost:8080}}'}"`,
102
+ `export ${altServerUrlVar}="${'${' + altServerUrlVar + ':-${' + serverUrlVar + '}}'}"`,
103
+ 'export ADDITIONAL_ALLOWED_ORIGINS="${ADDITIONAL_ALLOWED_ORIGINS:-$' + clientUrlVar + '}"',
104
+ `export ${viewportWidthVar}="${'${' + viewportWidthVar + ':-${' + altViewportWidthVar + ':-1920}}'}"`,
105
+ `export ${altViewportWidthVar}="${'${' + altViewportWidthVar + ':-${' + viewportWidthVar + '}}'}"`,
106
+ `export ${viewportHeightVar}="${'${' + viewportHeightVar + ':-${' + altViewportHeightVar + ':-1080}}'}"`,
107
+ `export ${altViewportHeightVar}="${'${' + altViewportHeightVar + ':-${' + viewportHeightVar + '}}'}"`,
108
+ `export ${timeoutVar}="${'${' + timeoutVar + ':-${' + altTimeoutVar + ':-900}}'}"`,
109
+ `export ${altTimeoutVar}="${'${' + altTimeoutVar + ':-${' + timeoutVar + '}}'}"`,
110
+ `export ${prebundleVar}="${'${' + prebundleVar + ':-${' + altPrebundleVar + ':-false}}'}"`,
111
+ `export ${altPrebundleVar}="${'${' + altPrebundleVar + ':-${' + prebundleVar + '}}'}"`,
112
+ `export ${reuseVar}="${'${' + reuseVar + ':-${' + altReuseVar + ':-true}}'}"`,
113
+ `export ${altReuseVar}="${'${' + altReuseVar + ':-${' + reuseVar + '}}'}"`,
114
+ `export ${keepaliveVar}="${'${' + keepaliveVar + ':-${' + altKeepaliveVar + ':-false}}'}"`,
115
+ `export ${altKeepaliveVar}="${'${' + altKeepaliveVar + ':-${' + keepaliveVar + '}}'}"`,
116
+ `export ${usernameVar}="${'${' + usernameVar + ':-${' + altUsernameVar + ':-' + username + '}}'}"`,
117
+ `export ${altUsernameVar}="${'${' + altUsernameVar + ':-${' + usernameVar + '}}'}"`,
118
+ `export ${passwordVar}="${'${' + passwordVar + ':-${' + altPasswordVar + ':-' + password + '}}'}"`,
119
+ `export ${altPasswordVar}="${'${' + altPasswordVar + ':-${' + passwordVar + '}}'}"`,
120
+ `export ${envVar(mode, 'JOB_ID')}="${'${' + envVar(mode, 'JOB_ID') + ':-${' + envVar(altMode, 'JOB_ID') + ':-' + jobId + '}}'}"`,
121
+ `export ${envVar(altMode, 'JOB_ID')}="${'${' + envVar(altMode, 'JOB_ID') + ':-${' + envVar(mode, 'JOB_ID') + '}}'}"`,
122
+ `export ${envVar(mode, 'OWNER_ID')}="${'${' + envVar(mode, 'OWNER_ID') + ':-${' + envVar(altMode, 'OWNER_ID') + ':-' + ownerId + '}}'}"`,
123
+ `export ${envVar(altMode, 'OWNER_ID')}="${'${' + envVar(altMode, 'OWNER_ID') + ':-${' + envVar(mode, 'OWNER_ID') + '}}'}"`,
124
+ `export ${envVar(mode, 'RUNNER_TOKEN')}="${'${' + envVar(mode, 'RUNNER_TOKEN') + ':-${' + envVar(altMode, 'RUNNER_TOKEN') + ':-' + runnerToken + '}}'}"`,
125
+ `export ${envVar(altMode, 'RUNNER_TOKEN')}="${'${' + envVar(altMode, 'RUNNER_TOKEN') + ':-${' + envVar(mode, 'RUNNER_TOKEN') + '}}'}"`,
126
+ 'export PUPPETEER_CACHE_DIR="${PUPPETEER_CACHE_DIR:-/var/lib/resolveio/puppeteer}"',
127
+ 'if [ ! -d "$PUPPETEER_CACHE_DIR" ] || [ ! -w "$PUPPETEER_CACHE_DIR" ]; then',
128
+ ' export PUPPETEER_CACHE_DIR="$RESOLVEIO_QA_TMP/puppeteer"',
129
+ ' mkdir -p "$PUPPETEER_CACHE_DIR"',
130
+ 'fi',
131
+ 'export PLAYWRIGHT_BROWSERS_PATH="${PLAYWRIGHT_BROWSERS_PATH:-$RESOLVEIO_QA_HOME/.cache/ms-playwright}"',
132
+ 'if [ -n "${PUPPETEER_EXECUTABLE_PATH:-}" ] && [ -x "$PUPPETEER_EXECUTABLE_PATH" ]; then',
133
+ ' export CHROME_BIN="${CHROME_BIN:-$PUPPETEER_EXECUTABLE_PATH}"',
134
+ 'elif [ -n "${CHROME_BIN:-}" ] && [ -x "$CHROME_BIN" ]; then',
135
+ ' export PUPPETEER_EXECUTABLE_PATH="$CHROME_BIN"',
136
+ 'else',
137
+ ` for ${browserLoopVar} in "$PUPPETEER_CACHE_DIR"/chrome-headless-shell/linux-*/chrome-headless-shell-linux64/chrome-headless-shell "$PUPPETEER_CACHE_DIR"/chrome/linux-*/chrome-linux64/chrome "$PLAYWRIGHT_BROWSERS_PATH"/chromium_headless_shell-*/chrome-headless-shell-linux64/chrome-headless-shell "$PLAYWRIGHT_BROWSERS_PATH"/chromium-*/chrome-linux64/chrome /usr/bin/google-chrome-stable /usr/bin/google-chrome /usr/bin/chromium /usr/bin/chromium-browser; do`,
138
+ ` if [ -x "$${browserLoopVar}" ]; then`,
139
+ ` export PUPPETEER_EXECUTABLE_PATH="$${browserLoopVar}"`,
140
+ ` export CHROME_BIN="$${browserLoopVar}"`,
141
+ ' break',
142
+ ' fi',
143
+ ' done',
144
+ 'fi',
145
+ `unset ${browserLoopVar}`,
146
+ 'export NPM_CONFIG_CACHE="$RESOLVEIO_QA_TMP/npm-cache"',
147
+ 'export NPM_CONFIG_PREFER_OFFLINE=true',
148
+ 'export NPM_CONFIG_PRODUCTION=false',
149
+ 'export npm_config_production=false',
150
+ 'export RESOLVEIO_SUPPORT_MONGOMS_PACKAGE_ROOT="$RESOLVEIO_QA_TMP/mongodb-memory-server-core"',
151
+ 'export MONGOMS_DOWNLOAD_DIR="$RESOLVEIO_QA_TMP/mongodb-binaries"',
152
+ mongodbBinaryCachePath ? `export RESOLVEIO_SUPPORT_SHARED_MONGOMS_DOWNLOAD_DIR="${mongodbBinaryCachePath}"` : '',
153
+ 'export MONGOMS_VERSION="${MONGOMS_VERSION:-7.0.14}"',
154
+ ''
155
+ ].filter((line) => line !== '').join('\n');
156
+ }
157
+
158
+ export function buildResolveIORunnerLocalQaScript(): string {
159
+ return [
160
+ '#!/usr/bin/env bash',
161
+ 'set -u',
162
+ 'TOOLS_DIR="$(cd "$(dirname "${BASH_SOURCE[0]}")" && pwd)"',
163
+ 'source "$TOOLS_DIR/env.sh"',
164
+ 'PROJECT_ROOT="${1:-$(pwd)}"',
165
+ 'PROJECT_ROOT="$(cd "$PROJECT_ROOT" && pwd)"',
166
+ 'ARTIFACT_DIR="$PROJECT_ROOT/qa-artifacts"',
167
+ 'mkdir -p "$ARTIFACT_DIR"',
168
+ 'CLIENT_URL="${RESOLVEIO_RUNNER_QA_CLIENT_URL:-${RESOLVEIO_SUPPORT_QA_CLIENT_URL:-http://localhost:${RESOLVEIO_RUNNER_QA_CLIENT_PORT:-${RESOLVEIO_SUPPORT_QA_CLIENT_PORT:-4200}}}}"',
169
+ 'SERVER_URL="${RESOLVEIO_RUNNER_QA_SERVER_URL:-${RESOLVEIO_SUPPORT_QA_SERVER_URL:-http://localhost:8080}}"',
170
+ 'CLIENT_PORT="${RESOLVEIO_RUNNER_QA_CLIENT_PORT:-${RESOLVEIO_SUPPORT_QA_CLIENT_PORT:-4200}}"',
171
+ 'SERVER_PORT="${RESOLVEIO_RUNNER_QA_SERVER_PORT:-${RESOLVEIO_SUPPORT_QA_SERVER_PORT:-8080}}"',
172
+ 'MONGO_PORT="${RESOLVEIO_RUNNER_QA_MONGO_PORT:-${RESOLVEIO_SUPPORT_QA_MONGO_PORT:-3001}}"',
173
+ 'INSPECT_PORT="${RESOLVEIO_RUNNER_QA_INSPECT_PORT:-${RESOLVEIO_SUPPORT_QA_INSPECT_PORT:-9229}}"',
174
+ 'STARTUP_TIMEOUT="${RESOLVEIO_RUNNER_QA_ANGULAR_STARTUP_TIMEOUT_SECONDS:-${RESOLVEIO_SUPPORT_QA_ANGULAR_STARTUP_TIMEOUT_SECONDS:-900}}"',
175
+ 'ANGULAR_PREBUNDLE="${RESOLVEIO_RUNNER_QA_ANGULAR_PREBUNDLE:-${RESOLVEIO_SUPPORT_QA_ANGULAR_PREBUNDLE:-false}}"',
176
+ 'REUSE_RUNNING="${RESOLVEIO_RUNNER_QA_REUSE_RUNNING:-${RESOLVEIO_SUPPORT_QA_REUSE_RUNNING:-true}}"',
177
+ 'if [[ "$(basename "$TOOLS_DIR")" == *support* ]]; then',
178
+ ' KEEPALIVE="${RESOLVEIO_SUPPORT_QA_KEEPALIVE:-${RESOLVEIO_RUNNER_QA_KEEPALIVE:-false}}"',
179
+ 'else',
180
+ ' KEEPALIVE="${RESOLVEIO_RUNNER_QA_KEEPALIVE:-${RESOLVEIO_SUPPORT_QA_KEEPALIVE:-false}}"',
181
+ 'fi',
182
+ 'case "${KEEPALIVE,,}" in',
183
+ ' true|1|yes|on) exec >> "$ARTIFACT_DIR/runner.log" 2>&1 ;;',
184
+ 'esac',
185
+ 'SERVER_STABLE_SECONDS="${RESOLVEIO_RUNNER_QA_SERVER_STABLE_SECONDS:-${RESOLVEIO_SUPPORT_QA_SERVER_STABLE_SECONDS:-20}}"',
186
+ 'LOCK_DIR="$ARTIFACT_DIR/.qa.lock"',
187
+ 'SERVER_PID=""',
188
+ 'CLIENT_PID=""',
189
+ 'SERVER_REQUIRED=0',
190
+ 'RUNNER_REUSED_READY=0',
191
+ 'ANGULAR_PREBUNDLE_ARGS=()',
192
+ 'REPO_ROOT="$(git -C "$PROJECT_ROOT" rev-parse --show-toplevel 2>/dev/null || echo "$PROJECT_ROOT")"',
193
+ buildRunnerProcessJanitorShellLibrary({ mode: 'support' }),
194
+ 'case "${ANGULAR_PREBUNDLE,,}" in',
195
+ ' false|0|no|off) ANGULAR_PREBUNDLE_ARGS=(--prebundle=false) ;;',
196
+ 'esac',
197
+ 'kill_tree() {',
198
+ ' local pid="$1"',
199
+ ' [ -n "$pid" ] || return 0',
200
+ ' for child in $(pgrep -P "$pid" 2>/dev/null || true); do kill_tree "$child"; done',
201
+ ' kill "$pid" >/dev/null 2>&1 || true',
202
+ ' sleep 1',
203
+ ' kill -0 "$pid" >/dev/null 2>&1 && kill -9 "$pid" >/dev/null 2>&1 || true',
204
+ '}',
205
+ 'kill_port_listeners() {',
206
+ ' local port="$1"',
207
+ ' [ -n "$port" ] || return 0',
208
+ ' local pids=""',
209
+ ' if command -v lsof >/dev/null 2>&1; then pids="$pids $(janitor_bounded 3 lsof -ti tcp:"$port")"; fi',
210
+ ' if command -v fuser >/dev/null 2>&1; then pids="$pids $(janitor_bounded 3 fuser -n tcp "$port")"; fi',
211
+ ' if command -v ss >/dev/null 2>&1; then pids="$pids $(janitor_bounded 3 ss -ltnp "sport = :$port" | sed -n \'s/.*pid=\\([0-9][0-9]*\\).*/\\1/p\' || true)"; fi',
212
+ ' for pid in $pids; do',
213
+ ' [ "$pid" = "$$" ] && continue',
214
+ ' kill_tree "$pid"',
215
+ ' done',
216
+ ' sleep 1',
217
+ '}',
218
+ 'kill_env_marked_processes() {',
219
+ ' [ -d /proc ] || return 0',
220
+ ' local job_id="${RESOLVEIO_RUNNER_QA_JOB_ID:-${RESOLVEIO_SUPPORT_QA_JOB_ID:-}}"',
221
+ ' local owner_id="${RESOLVEIO_RUNNER_QA_OWNER_ID:-${RESOLVEIO_SUPPORT_QA_OWNER_ID:-}}"',
222
+ ' local token="${RESOLVEIO_RUNNER_QA_RUNNER_TOKEN:-${RESOLVEIO_SUPPORT_QA_RUNNER_TOKEN:-}}"',
223
+ ' [ -n "$job_id$owner_id$token" ] || return 0',
224
+ ' for env_file in /proc/[0-9]*/environ; do',
225
+ ' pid="${env_file#/proc/}"',
226
+ ' pid="${pid%/environ}"',
227
+ ' skip_cleanup_pid "$pid" && continue',
228
+ ' [ "$pid" = "$$" ] && continue',
229
+ ' [ -r "$env_file" ] || continue',
230
+ ' env_text="$(tr "\\0" "\\n" < "$env_file" 2>/dev/null || true)"',
231
+ ' [ -n "$env_text" ] || continue',
232
+ ' matched=0',
233
+ ' [ -n "$job_id" ] && echo "$env_text" | grep -Eq "^(RESOLVEIO_RUNNER_QA_JOB_ID|RESOLVEIO_SUPPORT_QA_JOB_ID)=$job_id$" && matched=1',
234
+ ' [ "$matched" = "0" ] && [ -n "$owner_id" ] && echo "$env_text" | grep -Eq "^(RESOLVEIO_RUNNER_QA_OWNER_ID|RESOLVEIO_SUPPORT_QA_OWNER_ID)=$owner_id$" && matched=1',
235
+ ' [ "$matched" = "0" ] && [ -n "$token" ] && echo "$env_text" | grep -Eq "^(RESOLVEIO_RUNNER_QA_RUNNER_TOKEN|RESOLVEIO_SUPPORT_QA_RUNNER_TOKEN)=$token$" && matched=1',
236
+ ' [ "$matched" = "1" ] && kill_tree "$pid"',
237
+ ' done',
238
+ '}',
239
+ 'RUNNER_ANCESTOR_PIDS=" $$ ${PPID:-} "',
240
+ 'ancestor_pid="${PPID:-}"',
241
+ 'while [ -n "$ancestor_pid" ] && [ "$ancestor_pid" != "0" ] && [ "$ancestor_pid" != "1" ]; do',
242
+ ' ancestor_pid="$(ps -o ppid= -p "$ancestor_pid" 2>/dev/null | tr -d " " || true)"',
243
+ ' [ -n "$ancestor_pid" ] && RUNNER_ANCESTOR_PIDS="$RUNNER_ANCESTOR_PIDS $ancestor_pid "',
244
+ 'done',
245
+ 'skip_cleanup_pid() {',
246
+ ' case "$RUNNER_ANCESTOR_PIDS" in *" $1 "*) return 0 ;; esac',
247
+ ' return 1',
248
+ '}',
249
+ 'cleanup_project_processes() {',
250
+ ' for pid_file in "$ARTIFACT_DIR/server.pid" "$ARTIFACT_DIR/client.pid"; do',
251
+ ' [ -f "$pid_file" ] || continue',
252
+ ' pid="$(cat "$pid_file" 2>/dev/null || true)"',
253
+ ' kill_tree "$pid"',
254
+ ' rm -f "$pid_file"',
255
+ ' done',
256
+ ' for pass in 1 2 3; do',
257
+ ' for port in "$CLIENT_PORT" "$SERVER_PORT" "$MONGO_PORT" "$INSPECT_PORT"; do kill_port_listeners "$port"; done',
258
+ ' kill_env_marked_processes',
259
+ ' for pid in $(ps -eo pid=,args= | awk -v root="$PROJECT_ROOT" \'index($0, root) && $0 !~ /awk -v root=/ && $0 !~ /ps -eo pid=,args=/ && $0 !~ /run-local-qa\\.sh|stop-local-qa\\.sh|bugfix-comparison-qa\\.sh/ && $0 ~ /(ng serve|node .*node_modules\\/\\.bin\\/ng|esbuild|npm run client|start_client\\.sh|npm run server|start_server\\.sh|nodemon|node .*tmp\\/index\\.js|mongod|mongodb-binaries\\/mongod)/ {print $1}\' 2>/dev/null || true); do',
260
+ ' skip_cleanup_pid "$pid" && continue',
261
+ ' kill_tree "$pid"',
262
+ ' done',
263
+ ' if [ -d /proc ]; then',
264
+ ' for proc_cwd in /proc/[0-9]*/cwd; do',
265
+ ' pid="${proc_cwd#/proc/}"',
266
+ ' pid="${pid%/cwd}"',
267
+ ' skip_cleanup_pid "$pid" && continue',
268
+ ' cwd="$(readlink -f "$proc_cwd" 2>/dev/null || true)"',
269
+ ' case "$cwd" in',
270
+ ' "$PROJECT_ROOT"|"$PROJECT_ROOT"/*) kill_tree "$pid" ;;',
271
+ ' esac',
272
+ ' done',
273
+ ' fi',
274
+ ' sleep 1',
275
+ ' done',
276
+ ' local wait_until=$((SECONDS + 60))',
277
+ ' while [ "$SECONDS" -lt "$wait_until" ]; do',
278
+ ' local found=0',
279
+ ' for port in "$CLIENT_PORT" "$SERVER_PORT" "$MONGO_PORT" "$INSPECT_PORT"; do',
280
+ ' if command -v lsof >/dev/null 2>&1 && [ -n "$(janitor_bounded 3 lsof -ti tcp:"$port")" ]; then found=1; fi',
281
+ ' done',
282
+ ' if [ -d /proc ]; then',
283
+ ' job_id="${RESOLVEIO_RUNNER_QA_JOB_ID:-${RESOLVEIO_SUPPORT_QA_JOB_ID:-}}"',
284
+ ' owner_id="${RESOLVEIO_RUNNER_QA_OWNER_ID:-${RESOLVEIO_SUPPORT_QA_OWNER_ID:-}}"',
285
+ ' token="${RESOLVEIO_RUNNER_QA_RUNNER_TOKEN:-${RESOLVEIO_SUPPORT_QA_RUNNER_TOKEN:-}}"',
286
+ ' for env_file in /proc/[0-9]*/environ; do',
287
+ ' pid="${env_file#/proc/}"',
288
+ ' pid="${pid%/environ}"',
289
+ ' skip_cleanup_pid "$pid" && continue',
290
+ ' [ -r "$env_file" ] || continue',
291
+ ' env_text="$(tr "\\0" "\\n" < "$env_file" 2>/dev/null || true)"',
292
+ ' [ -n "$job_id" ] && echo "$env_text" | grep -Eq "^(RESOLVEIO_RUNNER_QA_JOB_ID|RESOLVEIO_SUPPORT_QA_JOB_ID)=$job_id$" && found=1',
293
+ ' [ -n "$owner_id" ] && echo "$env_text" | grep -Eq "^(RESOLVEIO_RUNNER_QA_OWNER_ID|RESOLVEIO_SUPPORT_QA_OWNER_ID)=$owner_id$" && found=1',
294
+ ' [ -n "$token" ] && echo "$env_text" | grep -Eq "^(RESOLVEIO_RUNNER_QA_RUNNER_TOKEN|RESOLVEIO_SUPPORT_QA_RUNNER_TOKEN)=$token$" && found=1',
295
+ ' done',
296
+ ' fi',
297
+ ' for pid in $(ps -eo pid=,args= | awk -v root="$PROJECT_ROOT" \'index($0, root) && $0 !~ /awk -v root=/ && $0 !~ /ps -eo pid=,args=/ && $0 !~ /run-local-qa\\.sh|stop-local-qa\\.sh|bugfix-comparison-qa\\.sh/ && $0 ~ /(ng serve|node .*node_modules\\/\\.bin\\/ng|esbuild|npm run client|start_client\\.sh|npm run server|start_server\\.sh|nodemon|node .*tmp\\/index\\.js|mongod|mongodb-binaries\\/mongod)/ {print $1}\' 2>/dev/null || true); do',
298
+ ' skip_cleanup_pid "$pid" && continue',
299
+ ' found=1',
300
+ ' done',
301
+ ' [ "$found" = "0" ] && break',
302
+ ' sleep 2',
303
+ ' done',
304
+ '}',
305
+ 'cleanup() {',
306
+ ' [ "${RUNNER_REUSED_READY:-0}" = "1" ] && return 0',
307
+ ' janitor_stop_heartbeat',
308
+ ' rm -f "$ARTIFACT_DIR/heartbeat.pid"',
309
+ ' janitor_update_manifest_status cleanup_started',
310
+ ' kill_tree "$SERVER_PID"',
311
+ ' kill_tree "$CLIENT_PID"',
312
+ ' cleanup_project_processes',
313
+ ' janitor_update_manifest_status cleanup_complete',
314
+ ' janitor_write_cleanup_status complete 0',
315
+ ' rm -rf "$LOCK_DIR" >/dev/null 2>&1 || true',
316
+ '}',
317
+ 'trap cleanup EXIT',
318
+ 'detach_keepalive() {',
319
+ ' janitor_update_manifest_status ready_keepalive',
320
+ ' janitor_write_cleanup_status ready_keepalive 0',
321
+ ' rm -rf "$LOCK_DIR" >/dev/null 2>&1 || true',
322
+ ' trap - EXIT',
323
+ ' disown >/dev/null 2>&1 || true',
324
+ ' echo "ResolveIO AI runner QA keepalive detached at $CLIENT_URL; stop with $TOOLS_DIR/stop-local-qa.sh $PROJECT_ROOT."',
325
+ ' exit 0',
326
+ '}',
327
+ 'probe_url() { node - "$1" <<\'RESOLVEIO_PROBE\'',
328
+ 'const http = require("http");',
329
+ 'const https = require("https");',
330
+ 'const url = process.argv[2];',
331
+ 'const mod = /^https:/i.test(url) ? https : http;',
332
+ 'const req = mod.get(url, { timeout: 2500 }, (res) => { res.resume(); process.exit(res.statusCode && res.statusCode < 500 ? 0 : 1); });',
333
+ 'req.on("timeout", () => req.destroy(new Error("timeout")));',
334
+ 'req.on("error", () => process.exit(1));',
335
+ 'RESOLVEIO_PROBE',
336
+ '}',
337
+ 'truthy() { case "${1,,}" in true|1|yes|on) return 0 ;; *) return 1 ;; esac; }',
338
+ 'reuse_running_app_if_ready() {',
339
+ ' truthy "$REUSE_RUNNING" || return 1',
340
+ ' [ -d "$PROJECT_ROOT/server" ] && SERVER_REQUIRED=1 || SERVER_REQUIRED=0',
341
+ ' probe_url "$CLIENT_URL" || return 1',
342
+ ' if [ "$SERVER_REQUIRED" = "1" ] && ! probe_url "$SERVER_URL"; then return 1; fi',
343
+ ' echo "ResolveIO AI runner QA reusing already-ready local app at $CLIENT_URL"',
344
+ ' return 0',
345
+ '}',
346
+ 'log_has_fatal() {',
347
+ ' local file="$1"',
348
+ ' [ -f "$file" ] || return 1',
349
+ ' grep -Eiq "Unhandled Rejection|Cannot read properties of undefined|TypeError|ReferenceError|EADDRINUSE|app crashed|Failed to compile|Error: Cannot find module|NG[0-9]{4}|TS[0-9]{4}" "$file"',
350
+ '}',
351
+ 'prepare_angular_cache_dirs() {',
352
+ ' [ -d "$PROJECT_ROOT" ] || return 0',
353
+ ' mkdir -p "$PROJECT_ROOT/.angular/cache" 2>/dev/null || true',
354
+ ' local version=""',
355
+ ' for pkg in "$PROJECT_ROOT/node_modules/@angular/build/package.json" "$PROJECT_ROOT/node_modules/@angular/cli/package.json" "$PROJECT_ROOT/node_modules/@angular-devkit/build-angular/package.json"; do',
356
+ ' if [ -f "$pkg" ]; then',
357
+ ' version="$(node -e "try{console.log(require(process.argv[1]).version||\\\"\\\")}catch(e){}" "$pkg" 2>/dev/null | head -1)"',
358
+ ' [ -n "$version" ] && break',
359
+ ' fi',
360
+ ' done',
361
+ ' [ -n "$version" ] || return 0',
362
+ ' node - "$PROJECT_ROOT/angular.json" "$(basename "$PROJECT_ROOT")" <<\'RESOLVEIO_ANGULAR_CACHE_PROJECTS\' | while IFS= read -r project; do',
363
+ 'const fs = require("fs");',
364
+ 'const path = process.argv[2];',
365
+ 'const fallback = process.argv[3];',
366
+ 'const names = new Set([fallback].filter(Boolean));',
367
+ 'try {',
368
+ ' const parsed = JSON.parse(fs.readFileSync(path, "utf8"));',
369
+ ' if (parsed.defaultProject) names.add(String(parsed.defaultProject));',
370
+ ' if (parsed.projects && typeof parsed.projects === "object") Object.keys(parsed.projects).forEach((name) => names.add(String(name)));',
371
+ '} catch (error) {}',
372
+ 'for (const name of names) console.log(name.replace(/[^A-Za-z0-9._-]/g, "_"));',
373
+ 'RESOLVEIO_ANGULAR_CACHE_PROJECTS',
374
+ ' [ -n "$project" ] || continue',
375
+ ' mkdir -p "$PROJECT_ROOT/.angular/cache/$version/$project/vite" 2>/dev/null || true',
376
+ ' done',
377
+ '}',
378
+ 'server_has_started() {',
379
+ ' local file="$ARTIFACT_DIR/server.log"',
380
+ ' [ -f "$file" ] || return 1',
381
+ ' grep -Eiq "nodemon.*starting|node .*tmp/index\\.js|Running as Worker|Standalone Node Reaper|listening on|server listening|Server listening|app listening|App listening|Finished .default." "$file"',
382
+ '}',
383
+ 'wait_for_server_ready() {',
384
+ ' [ "$SERVER_REQUIRED" = "1" ] || return 0',
385
+ ' local end=$((SECONDS + STARTUP_TIMEOUT))',
386
+ ' while [ "$SECONDS" -lt "$end" ]; do',
387
+ ' if log_has_fatal "$ARTIFACT_DIR/server.log"; then return 4; fi',
388
+ ' if server_has_started; then',
389
+ ' local stable_until=$((SECONDS + SERVER_STABLE_SECONDS))',
390
+ ' while [ "$SECONDS" -lt "$stable_until" ]; do if log_has_fatal "$ARTIFACT_DIR/server.log"; then return 4; fi; sleep 2; done',
391
+ ' return 0',
392
+ ' fi',
393
+ ' if probe_url "$SERVER_URL"; then',
394
+ ' local stable_until=$((SECONDS + SERVER_STABLE_SECONDS))',
395
+ ' while [ "$SECONDS" -lt "$stable_until" ]; do if log_has_fatal "$ARTIFACT_DIR/server.log"; then return 4; fi; sleep 2; done',
396
+ ' return 0',
397
+ ' fi',
398
+ ' sleep 5',
399
+ ' done',
400
+ ' return 4',
401
+ '}',
402
+ 'wait_for_client() {',
403
+ ' local end=$((SECONDS + STARTUP_TIMEOUT))',
404
+ ' while [ "$SECONDS" -lt "$end" ]; do',
405
+ ' if [ -n "$CLIENT_PID" ] && ! kill -0 "$CLIENT_PID" >/dev/null 2>&1; then return 2; fi',
406
+ ' if log_has_fatal "$ARTIFACT_DIR/client.log"; then return 3; fi',
407
+ ' if log_has_fatal "$ARTIFACT_DIR/server.log"; then return 4; fi',
408
+ ' if probe_url "$CLIENT_URL"; then wait_for_server_ready || return $?; return 0; fi',
409
+ ' sleep 5',
410
+ ' done',
411
+ ' return 1',
412
+ '}',
413
+ 'if reuse_running_app_if_ready; then RUNNER_REUSED_READY=1; exit 0; fi',
414
+ 'if ! mkdir "$LOCK_DIR" 2>/dev/null; then',
415
+ ' if janitor_lock_is_live; then',
416
+ ' echo "ResolveIO AI runner QA lock is already held by a live runner for $PROJECT_ROOT; refusing duplicate startup." | tee "$ARTIFACT_DIR/runner.log"',
417
+ ' exit 6',
418
+ ' fi',
419
+ ' echo "ResolveIO AI runner QA lock is stale for $PROJECT_ROOT; cleaning scoped local QA processes before retrying." | tee "$ARTIFACT_DIR/runner.log"',
420
+ ' cleanup_project_processes',
421
+ ' rm -rf "$LOCK_DIR" >/dev/null 2>&1 || true',
422
+ ' if ! mkdir "$LOCK_DIR" 2>/dev/null; then',
423
+ ' echo "ResolveIO AI runner QA lock is still held for $PROJECT_ROOT after cleanup. Stop the existing QA runner before starting another." | tee -a "$ARTIFACT_DIR/runner.log"',
424
+ ' exit 6',
425
+ ' fi',
426
+ 'fi',
427
+ 'janitor_write_lock',
428
+ 'cleanup_project_processes',
429
+ ': > "$ARTIFACT_DIR/server.log"',
430
+ ': > "$ARTIFACT_DIR/client.log"',
431
+ ': > "$ARTIFACT_DIR/runner.log"',
432
+ 'janitor_write_manifest',
433
+ 'janitor_start_heartbeat',
434
+ 'echo "$RUNNER_JANITOR_HEARTBEAT_PID" > "$ARTIFACT_DIR/heartbeat.pid"',
435
+ 'janitor_check_resources || exit $?',
436
+ 'if [ -d "$PROJECT_ROOT/server" ]; then',
437
+ ' SERVER_REQUIRED=1',
438
+ ' if node -e "const p=require(process.argv[1]); process.exit(p.scripts&&p.scripts.server?0:1)" "$PROJECT_ROOT/server/package.json" >/dev/null 2>&1; then',
439
+ ' (cd "$PROJECT_ROOT/server" && source "$TOOLS_DIR/env.sh" && npm run server 2>&1 | tee "$ARTIFACT_DIR/server.log") &',
440
+ ' elif [ -x "$PROJECT_ROOT/server/start_server.sh" ]; then',
441
+ ' (cd "$PROJECT_ROOT/server" && source "$TOOLS_DIR/env.sh" && ./start_server.sh 2>&1 | tee "$ARTIFACT_DIR/server.log") &',
442
+ ' else',
443
+ ' echo "ResolveIO AI runner QA cannot find npm server script or start_server.sh for $PROJECT_ROOT/server" | tee "$ARTIFACT_DIR/server.log"',
444
+ ' exit 5',
445
+ ' fi',
446
+ ' SERVER_PID=$!',
447
+ ' wait_for_server_ready',
448
+ ' SERVER_RESULT=$?',
449
+ ' if [ "$SERVER_RESULT" != "0" ]; then echo "ResolveIO AI runner QA server startup fatal error. See $ARTIFACT_DIR/server.log"; exit "$SERVER_RESULT"; fi',
450
+ 'fi',
451
+ 'CLIENT_HOST="${RESOLVEIO_RUNNER_QA_CLIENT_HOST:-${RESOLVEIO_SUPPORT_QA_CLIENT_HOST:-0.0.0.0}}"',
452
+ 'if [ -x "$PROJECT_ROOT/node_modules/.bin/ng" ]; then',
453
+ ' prepare_angular_cache_dirs',
454
+ ' (cd "$PROJECT_ROOT" && source "$TOOLS_DIR/env.sh" && node --max_old_space_size=8048 ./node_modules/.bin/ng serve --watch --configuration local --host "$CLIENT_HOST" --port "$CLIENT_PORT" "${ANGULAR_PREBUNDLE_ARGS[@]}" 2>&1 | tee "$ARTIFACT_DIR/client.log") &',
455
+ 'elif [ -x "$PROJECT_ROOT/start_client.sh" ]; then',
456
+ ' (cd "$PROJECT_ROOT" && source "$TOOLS_DIR/env.sh" && ./start_client.sh 2>&1 | tee "$ARTIFACT_DIR/client.log") &',
457
+ 'elif node -e "const p=require(process.argv[1]); process.exit(p.scripts&&p.scripts.client?0:1)" "$PROJECT_ROOT/package.json" >/dev/null 2>&1; then',
458
+ ' (cd "$PROJECT_ROOT" && source "$TOOLS_DIR/env.sh" && npm run client 2>&1 | tee "$ARTIFACT_DIR/client.log") &',
459
+ 'else',
460
+ ' echo "ResolveIO AI runner QA cannot find Angular CLI, start_client.sh, or npm client script for $PROJECT_ROOT" | tee "$ARTIFACT_DIR/client.log"',
461
+ ' exit 5',
462
+ 'fi',
463
+ 'CLIENT_PID=$!',
464
+ 'wait_for_client',
465
+ 'RESULT=$?',
466
+ 'case "$RESULT" in',
467
+ ' 0)',
468
+ ' echo "ResolveIO AI runner QA local app ready at $CLIENT_URL";',
469
+ ' echo "$SERVER_PID" > "$ARTIFACT_DIR/server.pid";',
470
+ ' echo "$CLIENT_PID" > "$ARTIFACT_DIR/client.pid";',
471
+ ' if truthy "$KEEPALIVE"; then detach_keepalive; fi;',
472
+ ' exit 0',
473
+ ' ;;',
474
+ ' 2) echo "ResolveIO AI runner QA client process exited before $CLIENT_URL became ready. See $ARTIFACT_DIR/client.log"; exit 2 ;;',
475
+ ' 3) echo "ResolveIO AI runner QA client startup fatal error. See $ARTIFACT_DIR/client.log"; exit 3 ;;',
476
+ ' 4) echo "ResolveIO AI runner QA server startup fatal error. See $ARTIFACT_DIR/server.log"; exit 4 ;;',
477
+ ' *) echo "ResolveIO AI runner QA local app did not become ready at $CLIENT_URL within ${STARTUP_TIMEOUT}s. See $ARTIFACT_DIR/client.log and $ARTIFACT_DIR/server.log"; exit 1 ;;',
478
+ 'esac',
479
+ ''
480
+ ].join('\n');
481
+ }
482
+
483
+ export function buildResolveIORunnerLocalQaStopperScript(): string {
484
+ return [
485
+ '#!/usr/bin/env bash',
486
+ 'set -u',
487
+ 'TOOLS_DIR="$(cd "$(dirname "${BASH_SOURCE[0]}")" && pwd)"',
488
+ 'source "$TOOLS_DIR/env.sh"',
489
+ 'PROJECT_ROOT="${1:-$(pwd)}"',
490
+ 'PROJECT_ROOT="$(cd "$PROJECT_ROOT" && pwd)"',
491
+ 'ARTIFACT_DIR="$PROJECT_ROOT/qa-artifacts"',
492
+ 'CLIENT_PORT="${RESOLVEIO_RUNNER_QA_CLIENT_PORT:-${RESOLVEIO_SUPPORT_QA_CLIENT_PORT:-4200}}"',
493
+ 'SERVER_PORT="${RESOLVEIO_RUNNER_QA_SERVER_PORT:-${RESOLVEIO_SUPPORT_QA_SERVER_PORT:-8080}}"',
494
+ 'MONGO_PORT="${RESOLVEIO_RUNNER_QA_MONGO_PORT:-${RESOLVEIO_SUPPORT_QA_MONGO_PORT:-3001}}"',
495
+ 'INSPECT_PORT="${RESOLVEIO_RUNNER_QA_INSPECT_PORT:-${RESOLVEIO_SUPPORT_QA_INSPECT_PORT:-9229}}"',
496
+ 'REPO_ROOT="$(git -C "$PROJECT_ROOT" rev-parse --show-toplevel 2>/dev/null || echo "$PROJECT_ROOT")"',
497
+ buildRunnerProcessJanitorShellLibrary({ mode: 'support' }),
498
+ 'STOP_LOCK_DIR="$ARTIFACT_DIR/.qa.stop.lock"',
499
+ 'if ! mkdir "$STOP_LOCK_DIR" 2>/dev/null; then',
500
+ ' lock_pid="$(cat "$STOP_LOCK_DIR/pid" 2>/dev/null || true)"',
501
+ ' if [ -n "$lock_pid" ] && kill -0 "$lock_pid" >/dev/null 2>&1; then',
502
+ ' echo "ResolveIO AI runner QA cleanup already active for $PROJECT_ROOT; refusing duplicate stopper."',
503
+ ' exit 0',
504
+ ' fi',
505
+ ' rm -rf "$STOP_LOCK_DIR" >/dev/null 2>&1 || true',
506
+ ' mkdir "$STOP_LOCK_DIR" 2>/dev/null || { echo "ResolveIO AI runner QA cleanup lock still active for $PROJECT_ROOT."; exit 0; }',
507
+ 'fi',
508
+ 'echo "$$" > "$STOP_LOCK_DIR/pid" 2>/dev/null || true',
509
+ 'trap \'rm -rf "$STOP_LOCK_DIR" >/dev/null 2>&1 || true\' EXIT',
510
+ 'kill_tree() {',
511
+ ' local pid="$1"',
512
+ ' [ -n "$pid" ] || return 0',
513
+ ' kill -0 "$pid" >/dev/null 2>&1 || return 0',
514
+ ' for child in $(pgrep -P "$pid" 2>/dev/null || true); do kill_tree "$child"; done',
515
+ ' kill "$pid" >/dev/null 2>&1 || true',
516
+ ' sleep 1',
517
+ ' kill -0 "$pid" >/dev/null 2>&1 && kill -9 "$pid" >/dev/null 2>&1 || true',
518
+ '}',
519
+ 'kill_port_listeners() {',
520
+ ' local port="$1"',
521
+ ' [ -n "$port" ] || return 0',
522
+ ' local pids=""',
523
+ ' if command -v lsof >/dev/null 2>&1; then pids="$pids $(janitor_bounded 3 lsof -ti tcp:"$port")"; fi',
524
+ ' if command -v fuser >/dev/null 2>&1; then pids="$pids $(janitor_bounded 3 fuser -n tcp "$port")"; fi',
525
+ ' if command -v ss >/dev/null 2>&1; then pids="$pids $(janitor_bounded 3 ss -ltnp "sport = :$port" | sed -n \'s/.*pid=\\([0-9][0-9]*\\).*/\\1/p\' || true)"; fi',
526
+ ' for pid in $pids; do',
527
+ ' [ "$pid" = "$$" ] && continue',
528
+ ' kill_tree "$pid"',
529
+ ' done',
530
+ ' sleep 1',
531
+ '}',
532
+ 'kill_env_marked_processes() {',
533
+ ' [ -d /proc ] || return 0',
534
+ ' local job_id="${RESOLVEIO_RUNNER_QA_JOB_ID:-${RESOLVEIO_SUPPORT_QA_JOB_ID:-}}"',
535
+ ' local owner_id="${RESOLVEIO_RUNNER_QA_OWNER_ID:-${RESOLVEIO_SUPPORT_QA_OWNER_ID:-}}"',
536
+ ' local token="${RESOLVEIO_RUNNER_QA_RUNNER_TOKEN:-${RESOLVEIO_SUPPORT_QA_RUNNER_TOKEN:-}}"',
537
+ ' [ -n "$job_id$owner_id$token" ] || return 0',
538
+ ' for env_file in /proc/[0-9]*/environ; do',
539
+ ' pid="${env_file#/proc/}"',
540
+ ' pid="${pid%/environ}"',
541
+ ' skip_cleanup_pid "$pid" && continue',
542
+ ' [ "$pid" = "$$" ] && continue',
543
+ ' [ -r "$env_file" ] || continue',
544
+ ' env_text="$(tr "\\0" "\\n" < "$env_file" 2>/dev/null || true)"',
545
+ ' [ -n "$env_text" ] || continue',
546
+ ' matched=0',
547
+ ' [ -n "$job_id" ] && echo "$env_text" | grep -Eq "^(RESOLVEIO_RUNNER_QA_JOB_ID|RESOLVEIO_SUPPORT_QA_JOB_ID)=$job_id$" && matched=1',
548
+ ' [ "$matched" = "0" ] && [ -n "$owner_id" ] && echo "$env_text" | grep -Eq "^(RESOLVEIO_RUNNER_QA_OWNER_ID|RESOLVEIO_SUPPORT_QA_OWNER_ID)=$owner_id$" && matched=1',
549
+ ' [ "$matched" = "0" ] && [ -n "$token" ] && echo "$env_text" | grep -Eq "^(RESOLVEIO_RUNNER_QA_RUNNER_TOKEN|RESOLVEIO_SUPPORT_QA_RUNNER_TOKEN)=$token$" && matched=1',
550
+ ' [ "$matched" = "1" ] && kill_tree "$pid" && killed_count=$((killed_count + 1))',
551
+ ' done',
552
+ '}',
553
+ 'RUNNER_ANCESTOR_PIDS=" $$ ${PPID:-} "',
554
+ 'ancestor_pid="${PPID:-}"',
555
+ 'while [ -n "$ancestor_pid" ] && [ "$ancestor_pid" != "0" ] && [ "$ancestor_pid" != "1" ]; do',
556
+ ' ancestor_pid="$(ps -o ppid= -p "$ancestor_pid" 2>/dev/null | tr -d " " || true)"',
557
+ ' [ -n "$ancestor_pid" ] && RUNNER_ANCESTOR_PIDS="$RUNNER_ANCESTOR_PIDS $ancestor_pid "',
558
+ 'done',
559
+ 'skip_cleanup_pid() {',
560
+ ' case "$RUNNER_ANCESTOR_PIDS" in *" $1 "*) return 0 ;; esac',
561
+ ' return 1',
562
+ '}',
563
+ 'janitor_update_manifest_status cleanup_started',
564
+ 'killed_count=0',
565
+ 'for pid_file in "$ARTIFACT_DIR/heartbeat.pid" "$ARTIFACT_DIR/server.pid" "$ARTIFACT_DIR/client.pid"; do',
566
+ ' [ -f "$pid_file" ] || continue',
567
+ ' pid="$(cat "$pid_file" 2>/dev/null || true)"',
568
+ ' kill_tree "$pid"',
569
+ ' killed_count=$((killed_count + 1))',
570
+ ' rm -f "$pid_file"',
571
+ 'done',
572
+ 'for pass in 1 2 3; do',
573
+ ' for port in "$CLIENT_PORT" "$SERVER_PORT" "$MONGO_PORT" "$INSPECT_PORT"; do kill_port_listeners "$port"; done',
574
+ ' kill_env_marked_processes',
575
+ ' for pid in $(ps -eo pid=,args= | awk -v root="$PROJECT_ROOT" \'index($0, root) && $0 !~ /awk -v root=/ && $0 !~ /ps -eo pid=,args=/ && $0 !~ /run-local-qa\\.sh|stop-local-qa\\.sh|bugfix-comparison-qa\\.sh/ && $0 ~ /(ng serve|node .*node_modules\\/\\.bin\\/ng|esbuild|npm run client|start_client\\.sh|npm run server|start_server\\.sh|nodemon|node .*tmp\\/index\\.js|mongod|mongodb-binaries\\/mongod)/ {print $1}\' 2>/dev/null || true); do',
576
+ ' skip_cleanup_pid "$pid" && continue',
577
+ ' kill_tree "$pid"',
578
+ ' killed_count=$((killed_count + 1))',
579
+ ' done',
580
+ ' if [ -d /proc ]; then',
581
+ ' for proc_cwd in /proc/[0-9]*/cwd; do',
582
+ ' pid="${proc_cwd#/proc/}"',
583
+ ' pid="${pid%/cwd}"',
584
+ ' skip_cleanup_pid "$pid" && continue',
585
+ ' cwd="$(readlink -f "$proc_cwd" 2>/dev/null || true)"',
586
+ ' case "$cwd" in',
587
+ ' "$PROJECT_ROOT"|"$PROJECT_ROOT"/*) kill_tree "$pid" ;;',
588
+ ' esac',
589
+ ' done',
590
+ ' fi',
591
+ ' sleep 1',
592
+ 'done',
593
+ 'wait_until=$((SECONDS + 60))',
594
+ 'while [ "$SECONDS" -lt "$wait_until" ]; do',
595
+ ' found=0',
596
+ ' for port in "$CLIENT_PORT" "$SERVER_PORT" "$MONGO_PORT" "$INSPECT_PORT"; do',
597
+ ' if command -v lsof >/dev/null 2>&1 && [ -n "$(janitor_bounded 3 lsof -ti tcp:"$port")" ]; then found=1; fi',
598
+ ' done',
599
+ ' if [ -d /proc ]; then',
600
+ ' job_id="${RESOLVEIO_RUNNER_QA_JOB_ID:-${RESOLVEIO_SUPPORT_QA_JOB_ID:-}}"',
601
+ ' owner_id="${RESOLVEIO_RUNNER_QA_OWNER_ID:-${RESOLVEIO_SUPPORT_QA_OWNER_ID:-}}"',
602
+ ' token="${RESOLVEIO_RUNNER_QA_RUNNER_TOKEN:-${RESOLVEIO_SUPPORT_QA_RUNNER_TOKEN:-}}"',
603
+ ' for env_file in /proc/[0-9]*/environ; do',
604
+ ' pid="${env_file#/proc/}"',
605
+ ' pid="${pid%/environ}"',
606
+ ' skip_cleanup_pid "$pid" && continue',
607
+ ' [ -r "$env_file" ] || continue',
608
+ ' env_text="$(tr "\\0" "\\n" < "$env_file" 2>/dev/null || true)"',
609
+ ' [ -n "$job_id" ] && echo "$env_text" | grep -Eq "^(RESOLVEIO_RUNNER_QA_JOB_ID|RESOLVEIO_SUPPORT_QA_JOB_ID)=$job_id$" && found=1',
610
+ ' [ -n "$owner_id" ] && echo "$env_text" | grep -Eq "^(RESOLVEIO_RUNNER_QA_OWNER_ID|RESOLVEIO_SUPPORT_QA_OWNER_ID)=$owner_id$" && found=1',
611
+ ' [ -n "$token" ] && echo "$env_text" | grep -Eq "^(RESOLVEIO_RUNNER_QA_RUNNER_TOKEN|RESOLVEIO_SUPPORT_QA_RUNNER_TOKEN)=$token$" && found=1',
612
+ ' done',
613
+ ' fi',
614
+ ' for pid in $(ps -eo pid=,args= | awk -v root="$PROJECT_ROOT" \'index($0, root) && $0 !~ /awk -v root=/ && $0 !~ /ps -eo pid=,args=/ && $0 !~ /run-local-qa\\.sh|stop-local-qa\\.sh|bugfix-comparison-qa\\.sh/ && $0 ~ /(ng serve|node .*node_modules\\/\\.bin\\/ng|esbuild|npm run client|start_client\\.sh|npm run server|start_server\\.sh|nodemon|node .*tmp\\/index\\.js|mongod|mongodb-binaries\\/mongod)/ {print $1}\' 2>/dev/null || true); do',
615
+ ' skip_cleanup_pid "$pid" && continue',
616
+ ' found=1',
617
+ ' done',
618
+ ' [ "$found" = "0" ] && break',
619
+ ' sleep 2',
620
+ 'done',
621
+ 'rm -rf "$ARTIFACT_DIR/.qa.lock" >/dev/null 2>&1 || true',
622
+ 'janitor_update_manifest_status stopped',
623
+ 'janitor_write_cleanup_status stopped "$killed_count"',
624
+ 'echo "ResolveIO AI runner QA local app stopped for $PROJECT_ROOT"',
625
+ ''
626
+ ].join('\n');
627
+ }
628
+
629
+ export function buildResolveIORunnerBugfixComparisonQaScript(): string {
630
+ return [
631
+ '#!/usr/bin/env bash',
632
+ 'set -u',
633
+ 'TOOLS_DIR="$(cd "$(dirname "${BASH_SOURCE[0]}")" && pwd)"',
634
+ 'source "$TOOLS_DIR/env.sh"',
635
+ 'PROJECT_ROOT="${1:-}"',
636
+ 'if [ -z "$PROJECT_ROOT" ]; then',
637
+ ' echo "Usage: $0 <project-root> [baseline-ref] -- <qa-command...>" >&2',
638
+ ' exit 2',
639
+ 'fi',
640
+ 'shift || true',
641
+ 'DEFAULT_BASELINE_REF="${RESOLVEIO_RUNNER_QA_BASELINE_REF:-${RESOLVEIO_SUPPORT_QA_BASELINE_REF:-origin/master}}"',
642
+ 'if [ "${1:-}" = "--" ]; then',
643
+ ' BASELINE_REF="$DEFAULT_BASELINE_REF"',
644
+ 'else',
645
+ ' BASELINE_REF="${1:-$DEFAULT_BASELINE_REF}"',
646
+ ' if [ "$#" -gt 0 ]; then shift || true; fi',
647
+ 'fi',
648
+ 'if [ "${1:-}" = "--" ]; then shift; fi',
649
+ 'if [ "$#" -eq 0 ]; then',
650
+ ' echo "Usage: $0 <project-root> [baseline-ref] -- <qa-command...>" >&2',
651
+ ' exit 2',
652
+ 'fi',
653
+ 'PROJECT_ROOT="$(cd "$PROJECT_ROOT" && pwd)"',
654
+ 'REPO_ROOT="$(git -C "$PROJECT_ROOT" rev-parse --show-toplevel)"',
655
+ 'PROJECT_REL="$(git -C "$REPO_ROOT" ls-files --full-name "$PROJECT_ROOT" 2>/dev/null | head -1 | xargs dirname 2>/dev/null || true)"',
656
+ 'if [ -z "$PROJECT_REL" ] || [ "$PROJECT_REL" = "." ]; then',
657
+ ' PROJECT_REL="$(node - "$REPO_ROOT" "$PROJECT_ROOT" <<\'RESOLVEIO_REL\'',
658
+ 'const path = require("path");',
659
+ 'console.log(path.relative(process.argv[2], process.argv[3]) || ".");',
660
+ 'RESOLVEIO_REL',
661
+ ' )"',
662
+ 'fi',
663
+ 'ARTIFACT_DIR="$PROJECT_ROOT/qa-artifacts"',
664
+ 'mkdir -p "$ARTIFACT_DIR/baseline" "$ARTIFACT_DIR/candidate"',
665
+ 'RESULT_JSON="$ARTIFACT_DIR/bugfix-comparison-result.json"',
666
+ 'CANDIDATE_PATCH="$ARTIFACT_DIR/bugfix-candidate.patch"',
667
+ 'ORIGINAL_HEAD="$(git -C "$REPO_ROOT" rev-parse HEAD)"',
668
+ 'ORIGINAL_BRANCH="$(git -C "$REPO_ROOT" symbolic-ref --short HEAD 2>/dev/null || true)"',
669
+ 'QA_COMMAND=("$@")',
670
+ 'STOPPER="$TOOLS_DIR/stop-local-qa.sh"',
671
+ 'CURRENT_PHASE=""',
672
+ 'stop_local_qa() {',
673
+ ' "$STOPPER" "$PROJECT_ROOT" >/dev/null 2>&1 || true',
674
+ '}',
675
+ 'write_json() {',
676
+ ' node - "$RESULT_JSON" "$@" <<\'RESOLVEIO_WRITE_JSON\'',
677
+ 'const fs = require("fs");',
678
+ 'const [path, status, baselineExit, candidateExit, baselineRef, originalHead, projectRel, note] = process.argv.slice(2);',
679
+ 'const payload = {',
680
+ ' status,',
681
+ ' baseline_ref: baselineRef,',
682
+ ' original_head: originalHead,',
683
+ ' project: projectRel,',
684
+ ' baseline_exit_code: Number(baselineExit),',
685
+ ' candidate_exit_code: Number(candidateExit),',
686
+ ' code_switch_proven: Number(baselineExit) !== 0 && Number(candidateExit) === 0,',
687
+ ' candidate_passed: Number(candidateExit) === 0,',
688
+ ' baseline_failed: Number(baselineExit) !== 0,',
689
+ ' note,',
690
+ ' artifact_dirs: { baseline: "qa-artifacts/baseline", candidate: "qa-artifacts/candidate" },',
691
+ ' created_at: new Date().toISOString()',
692
+ '};',
693
+ 'fs.writeFileSync(path, JSON.stringify(payload, null, 2));',
694
+ 'console.log(JSON.stringify(payload, null, 2));',
695
+ 'RESOLVEIO_WRITE_JSON',
696
+ '}',
697
+ 'snapshot_phase_artifacts() {',
698
+ ' local phase="$1"',
699
+ ' mkdir -p "$ARTIFACT_DIR/$phase"',
700
+ ' find "$ARTIFACT_DIR" -maxdepth 1 -type f \\( -name "*.png" -o -name "*.jpg" -o -name "*.jpeg" -o -name "*.webp" -o -name "*.json" -o -name "*.txt" -o -name "*.log" -o -name "*.zip" \\) -print0 2>/dev/null | while IFS= read -r -d "" file; do',
701
+ ' cp "$file" "$ARTIFACT_DIR/$phase/$(basename "$file")" 2>/dev/null || true',
702
+ ' done',
703
+ '}',
704
+ 'capture_candidate_patch() {',
705
+ ' : > "$CANDIDATE_PATCH"',
706
+ ' git -C "$REPO_ROOT" diff --binary -- "$PROJECT_REL" >> "$CANDIDATE_PATCH" || true',
707
+ ' git -C "$REPO_ROOT" diff --binary --cached -- "$PROJECT_REL" >> "$CANDIDATE_PATCH" || true',
708
+ '}',
709
+ 'checkout_project_from_ref() {',
710
+ ' local ref="$1"',
711
+ ' git -C "$REPO_ROOT" checkout "$ref" -- "$PROJECT_REL"',
712
+ '}',
713
+ 'restore_candidate() {',
714
+ ' stop_local_qa',
715
+ ' checkout_project_from_ref "$ORIGINAL_HEAD" || true',
716
+ ' if [ -s "$CANDIDATE_PATCH" ]; then',
717
+ ' git -C "$REPO_ROOT" apply --whitespace=nowarn "$CANDIDATE_PATCH" || true',
718
+ ' fi',
719
+ '}',
720
+ 'trap restore_candidate EXIT',
721
+ 'run_phase() {',
722
+ ' local phase="$1"',
723
+ ' CURRENT_PHASE="$phase"',
724
+ ' stop_local_qa',
725
+ ' export RESOLVEIO_RUNNER_QA_PHASE="$phase"',
726
+ ' export RESOLVEIO_SUPPORT_QA_PHASE="$phase"',
727
+ ' export RESOLVEIO_RUNNER_QA_ARTIFACT_DIR="$ARTIFACT_DIR/$phase"',
728
+ ' export RESOLVEIO_SUPPORT_QA_ARTIFACT_DIR="$ARTIFACT_DIR/$phase"',
729
+ ' mkdir -p "$ARTIFACT_DIR/$phase"',
730
+ ' find "$ARTIFACT_DIR/$phase" -maxdepth 1 -type f \\( -name "*.png" -o -name "*.jpg" -o -name "*.jpeg" -o -name "*.webp" -o -name "*.json" -o -name "*.txt" -o -name "*.log" -o -name "*.zip" \\) -delete 2>/dev/null || true',
731
+ ' find "$ARTIFACT_DIR" -maxdepth 1 -type f \\( -name "*.png" -o -name "*.jpg" -o -name "*.jpeg" -o -name "*.webp" -o -name "*proof.json" -o -name "auth-bootstrap-result.json" \\) -delete 2>/dev/null || true',
732
+ ' set +e',
733
+ ' (cd "$REPO_ROOT" && "${QA_COMMAND[@]}") 2>&1 | tee "$ARTIFACT_DIR/$phase/qa-command.log"',
734
+ ' local rc="${PIPESTATUS[0]}"',
735
+ ' set +e',
736
+ ' snapshot_phase_artifacts "$phase"',
737
+ ' stop_local_qa',
738
+ ' return "$rc"',
739
+ '}',
740
+ 'echo "ResolveIO bugfix comparison QA preserving candidate diff for $PROJECT_REL"',
741
+ 'capture_candidate_patch',
742
+ 'stop_local_qa',
743
+ 'echo "ResolveIO bugfix comparison QA baseline phase: $BASELINE_REF"',
744
+ 'checkout_project_from_ref "$BASELINE_REF"',
745
+ 'run_phase baseline',
746
+ 'BASELINE_EXIT="$?"',
747
+ 'echo "ResolveIO bugfix comparison QA candidate phase: restored workspace candidate"',
748
+ 'restore_candidate',
749
+ 'run_phase candidate',
750
+ 'CANDIDATE_EXIT="$?"',
751
+ 'if [ "$CANDIDATE_EXIT" = "0" ] && [ "$BASELINE_EXIT" != "0" ]; then',
752
+ ' STATUS="pass"',
753
+ ' NOTE="Baseline failed and candidate passed for the same QA command."',
754
+ 'elif [ "$CANDIDATE_EXIT" = "0" ]; then',
755
+ ' STATUS="inconclusive_baseline_also_passed"',
756
+ ' NOTE="Candidate passed, but baseline also passed; the code switch did not prove the bug fix."',
757
+ 'else',
758
+ ' STATUS="fail_candidate_failed"',
759
+ ' NOTE="Candidate failed the QA command."',
760
+ 'fi',
761
+ 'write_json "$STATUS" "$BASELINE_EXIT" "$CANDIDATE_EXIT" "$BASELINE_REF" "$ORIGINAL_HEAD" "$PROJECT_REL" "$NOTE"',
762
+ '[ "$CANDIDATE_EXIT" = "0" ] || exit "$CANDIDATE_EXIT"',
763
+ '[ "$STATUS" = "pass" ] || exit 8',
764
+ ''
765
+ ].join('\n');
766
+ }
767
+
768
+ export function buildResolveIORunnerQaToolsReadme(options: ResolveIORunnerQaToolBundleOptions = {}): string {
769
+ const mode = options.mode || 'runner';
770
+ const toolsDir = mode === 'support' ? '.resolveio-support-tools' : '.resolveio-ai-runner-tools';
771
+ const port = normalizePort(options.qaClientPort, '4200');
772
+ const clientUrlVar = envVar(mode, 'CLIENT_URL');
773
+ const keepaliveVar = envVar(mode, 'KEEPALIVE');
774
+ const usernameVar = envVar(mode, 'USERNAME');
775
+ const passwordVar = envVar(mode, 'PASSWORD');
776
+ return [
777
+ `# ResolveIO ${mode === 'support' ? 'Support' : 'AI Runner'} Local QA Tools`,
778
+ '',
779
+ 'These scripts are generated by `@resolveio/server-lib` and are shared by support tickets, AICoder app-builder runs, and AI-terminal runs.',
780
+ '',
781
+ '```bash',
782
+ `source ${toolsDir}/env.sh`,
783
+ `${toolsDir}/run-local-qa.sh <project-root>`,
784
+ `node ${toolsDir}/qa-auth-bootstrap.js <project-root> /target-route`,
785
+ `${toolsDir}/bugfix-comparison-qa.sh <project-root> origin/master -- bash -lc '<same QA command>'`,
786
+ '```',
787
+ '',
788
+ `This workspace reserves Angular QA client port ${port}; use \`$${clientUrlVar}\` instead of assuming 4200 is free.`,
789
+ 'The local QA runner starts server/client, polls the reserved client URL, writes `qa-artifacts/server.log` and `qa-artifacts/client.log`, and fails fast on fatal startup/runtime errors.',
790
+ 'The shared auth bootstrap first opens the exact localhost client origin, logs out any visible stale session, clears service workers/cache/IndexedDB/local/session storage, then calls `/login` and `/accessToken`, seeds `refreshToken`, `accessToken`, `user`, and `lastURL`, and writes `qa-artifacts/auth-bootstrap-result.json` plus a ready/failure screenshot.',
791
+ `For browser clickthrough work, start the runner once with \`${keepaliveVar}=true ${toolsDir}/run-local-qa.sh <project-root>\`; it detaches after the app is ready, and later calls should reuse \`$${clientUrlVar}\` for all login/upload/screenshot retries. Do not restart Angular for auth failures.`,
792
+ 'Do not run `npm run build-dev`, `ng build`, or another Angular compile while keepalive `ng serve` is running. If a full Angular build is required after browser QA, first run the staged `stop-local-qa.sh`, then build, then restart `run-local-qa.sh` for final browser proof.',
793
+ 'Use desktop screenshots at 1920x1080 by default unless the task is explicitly mobile/responsive. Every screenshot must have a customer-facing caption.',
794
+ 'For import/export/form-submit/data workflows, prove before/action/after with representative data and a concrete row/count/value assertion.',
795
+ 'For bug fixes, use `bugfix-comparison-qa.sh` so baseline/master runs the exact same repro before the candidate/PR run. A passing candidate is not enough unless the comparison result shows baseline failed or the report explicitly explains why the baseline failure could not be reproduced.',
796
+ `Use \`$${usernameVar}\` and \`$${passwordVar}\` for the local fixture admin account unless ticket/app-specific credentials are provided.`,
797
+ 'The env file reuses `/var/lib/resolveio/puppeteer`, npm cache, worker-safe Browserslist settings, and Angular cache prep so QA should not download a browser or rebuild cold caches unnecessarily.',
798
+ mode === 'support'
799
+ ? 'Support workspaces also stage local `mongod` and `mongosh` wrappers so app server scripts can start MongoDB when the worker image does not have MongoDB installed.'
800
+ : '',
801
+ ''
802
+ ].filter((line) => line !== '').join('\n');
803
+ }