@resolveio/server-lib 22.3.124 → 22.3.126

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 (711) hide show
  1. package/.nodemon.json +5 -0
  2. package/.vscode/settings.json +21 -0
  3. package/AGENTS.md +195 -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 +3 -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-run.collection.ts +117 -0
  29. package/src/collections/ai-terminal-conversation.collection.ts +91 -0
  30. package/src/collections/ai-terminal-issue-report.collection.ts +99 -0
  31. package/src/collections/ai-terminal-message.collection.ts +77 -0
  32. package/src/collections/app-setting.collection.ts +104 -0
  33. package/src/collections/app-status.collection.ts +58 -0
  34. package/src/collections/communication-metric.collection.ts +84 -0
  35. package/src/collections/counter.collection.ts +56 -0
  36. package/src/collections/cron-job-history.collection.ts +94 -0
  37. package/src/collections/cron-job.collection.ts +92 -0
  38. package/src/collections/customer-notification.collection.ts +131 -0
  39. package/src/collections/customer-portal-password.collection.ts +76 -0
  40. package/src/collections/email-history.collection.ts +134 -0
  41. package/src/collections/email-verified.collection.ts +62 -0
  42. package/src/collections/file.collection.ts +74 -0
  43. package/src/collections/flag-update.collection.ts +57 -0
  44. package/src/collections/flag.collection.ts +57 -0
  45. package/src/collections/log-method-latency.collection.ts +77 -0
  46. package/src/collections/log-subscription.collection.ts +80 -0
  47. package/src/collections/log.collection.ts +93 -0
  48. package/src/collections/logged-in-users.collection.ts +67 -0
  49. package/src/collections/monitor-cpu.collection.ts +65 -0
  50. package/src/collections/monitor-function.collection.ts +74 -0
  51. package/src/collections/monitor-memory.collection.ts +77 -0
  52. package/src/collections/monitor-mongo.collection.ts +71 -0
  53. package/src/collections/notification.collection.ts +57 -0
  54. package/src/collections/openai-usage-ledger.collection.ts +77 -0
  55. package/src/collections/report-builder-dashboard-builder.collection.ts +109 -0
  56. package/src/collections/report-builder-library.collection.ts +89 -0
  57. package/src/collections/report-builder-report.collection.ts +184 -0
  58. package/src/collections/user-group.collection.ts +89 -0
  59. package/src/collections/user-guide.collection.ts +57 -0
  60. package/src/collections/user.collection.ts +181 -0
  61. package/src/cron/cron.ts +117 -0
  62. package/src/fixtures/cron-jobs.ts +95 -0
  63. package/src/fixtures/init.ts +35 -0
  64. package/src/http/auth.ts +818 -0
  65. package/src/http/health.ts +7 -0
  66. package/src/http/home.ts +90 -0
  67. package/src/http/slow-query-publication.ts +49 -0
  68. package/src/index.ts +1 -0
  69. package/src/managers/ai-assistant-codex-manager.manager.ts +1131 -0
  70. package/src/managers/ai-run-evidence.manager.ts +264 -0
  71. package/src/managers/communication-metric.manager.ts +82 -0
  72. package/src/managers/cron.manager.ts +333 -0
  73. package/src/managers/customer-notification-content.manager.ts +236 -0
  74. package/src/managers/diagnostic-manager-bootstrap.ts +165 -0
  75. package/src/managers/error-auto-fix.manager.ts +2767 -0
  76. package/src/managers/local-log.manager.ts +113 -0
  77. package/src/managers/method.manager.ts +1857 -0
  78. package/src/managers/mongo.manager.ts +4575 -0
  79. package/src/managers/monitor.manager.ts +507 -0
  80. package/src/managers/openai-usage-ledger.manager.ts +57 -0
  81. package/src/managers/slow-query-verifier.manager.ts +3590 -0
  82. package/src/managers/slow-query.manager.ts +519 -0
  83. package/src/managers/subscription.manager.ts +3128 -0
  84. package/src/managers/websocket.manager.ts +746 -0
  85. package/src/managers/worker-dispatcher.manager.ts +1360 -0
  86. package/src/managers/worker-server.manager.ts +536 -0
  87. package/src/methods/accounts.ts +532 -0
  88. package/src/methods/ai-terminal.ts +23497 -0
  89. package/src/methods/app-settings.ts +114 -0
  90. package/src/methods/aws.ts +649 -0
  91. package/src/methods/collections.ts +641 -0
  92. package/src/methods/counters.ts +69 -0
  93. package/src/methods/cron-jobs.ts +2614 -0
  94. package/src/methods/customer-notifications.ts +458 -0
  95. package/src/methods/diagnostics.ts +616 -0
  96. package/src/methods/flag-updates.ts +7 -0
  97. package/src/methods/flags.ts +7 -0
  98. package/src/methods/logs.ts +657 -0
  99. package/src/methods/mongo-explorer.ts +1880 -0
  100. package/src/methods/monitor.ts +540 -0
  101. package/src/methods/pdf.ts +1236 -0
  102. package/src/methods/publications.ts +129 -0
  103. package/src/methods/report-builder.ts +3300 -0
  104. package/src/methods/support.ts +335 -0
  105. package/src/models/ai-run.model.ts +27 -0
  106. package/src/models/ai-terminal-conversation.model.ts +19 -0
  107. package/src/models/ai-terminal-issue-report.model.ts +21 -0
  108. package/src/models/ai-terminal-message.model.ts +24 -0
  109. package/src/models/app-setting.model.ts +17 -0
  110. package/{models/app-status.model.d.ts → src/models/app-status.model.ts} +3 -2
  111. package/{models/billing-logged-in-users.model.d.ts → src/models/billing-logged-in-users.model.ts} +5 -4
  112. package/src/models/collection-document.model.ts +24 -0
  113. package/src/models/communication-metric.model.ts +23 -0
  114. package/{models/counter.model.d.ts → src/models/counter.model.ts} +4 -3
  115. package/src/models/cron-job-history.model.ts +16 -0
  116. package/src/models/cron-job.model.ts +15 -0
  117. package/src/models/customer-notification.model.ts +28 -0
  118. package/src/models/customer-portal-password.model.ts +12 -0
  119. package/src/models/dialog.model.ts +25 -0
  120. package/{models/email-history.model.js → src/models/email-history.model.ts} +36 -4
  121. package/{models/email-verified.model.d.ts → src/models/email-verified.model.ts} +6 -5
  122. package/{models/file.model.d.ts → src/models/file.model.ts} +8 -7
  123. package/{models/flag-update.model.d.ts → src/models/flag-update.model.ts} +4 -3
  124. package/{models/flag.model.d.ts → src/models/flag.model.ts} +4 -3
  125. package/src/models/log-method-latency.model.ts +11 -0
  126. package/{models/log-subscription.model.d.ts → src/models/log-subscription.model.ts} +11 -9
  127. package/src/models/log.model.ts +19 -0
  128. package/{models/logged-in-users.model.d.ts → src/models/logged-in-users.model.ts} +6 -5
  129. package/{models/method-response.model.d.ts → src/models/method-response.model.ts} +7 -6
  130. package/src/models/method.model.ts +25 -0
  131. package/{models/monitor-cpu.model.d.ts → src/models/monitor-cpu.model.ts} +9 -7
  132. package/src/models/monitor-function.model.ts +16 -0
  133. package/src/models/monitor-memory.model.ts +17 -0
  134. package/src/models/monitor-mongo.model.ts +15 -0
  135. package/{models/notification.model.d.ts → src/models/notification.model.ts} +6 -4
  136. package/src/models/openai-usage-ledger.model.ts +16 -0
  137. package/src/models/pagination.model.ts +35 -0
  138. package/src/models/permission.model.ts +14 -0
  139. package/src/models/report-builder-dashboard-builder.model.ts +29 -0
  140. package/src/models/report-builder-library.model.ts +20 -0
  141. package/src/models/report-builder-report.model.ts +136 -0
  142. package/src/models/report-builder.model.ts +68 -0
  143. package/src/models/select-data-label.model.ts +9 -0
  144. package/src/models/server-message.model.ts +31 -0
  145. package/src/models/slow-query-report.model.ts +23 -0
  146. package/src/models/subscription.model.ts +73 -0
  147. package/src/models/support-ticket.model.ts +104 -0
  148. package/src/models/user-group.model.ts +24 -0
  149. package/{models/user-guide.model.d.ts → src/models/user-guide.model.ts} +5 -4
  150. package/src/models/user.model.ts +96 -0
  151. package/src/private/images/ResolveIO.png +0 -0
  152. package/{public_api.d.ts → src/public_api.ts} +7 -0
  153. package/src/publications/ai-terminal.ts +73 -0
  154. package/src/publications/app-settings.ts +25 -0
  155. package/src/publications/app-status.ts +13 -0
  156. package/src/publications/cron-jobs.ts +40 -0
  157. package/src/publications/customer-notifications.ts +101 -0
  158. package/src/publications/files.ts +33 -0
  159. package/src/publications/flags-update.ts +19 -0
  160. package/src/publications/flags.ts +19 -0
  161. package/src/publications/logs.ts +163 -0
  162. package/src/publications/notifications.ts +13 -0
  163. package/src/publications/report-builder-dashboard-builders.ts +39 -0
  164. package/src/publications/report-builder-libraries.ts +41 -0
  165. package/src/publications/report-builder-reports.ts +47 -0
  166. package/src/publications/super-admin.ts +13 -0
  167. package/src/publications/user-groups.ts +12 -0
  168. package/src/publications/user-guides.ts +12 -0
  169. package/src/resolveio-server-app.ts +617 -0
  170. package/src/server-app.ts +3354 -0
  171. package/src/services/codex-client.ts +1231 -0
  172. package/src/services/openai-client.ts +265 -0
  173. package/src/types/error-report.ts +26 -0
  174. package/src/types/js-tiktoken.d.ts +11 -0
  175. package/src/types/slow-query-report.ts +28 -0
  176. package/src/util/ai-qa-policy.ts +925 -0
  177. package/src/util/ai-run-evidence-adapters.ts +521 -0
  178. package/src/util/ai-run-evidence-dashboard.ts +271 -0
  179. package/src/util/ai-run-evidence-eval.ts +885 -0
  180. package/src/util/ai-run-evidence.ts +964 -0
  181. package/src/util/ai-runner-artifacts.ts +586 -0
  182. package/src/util/ai-runner-qa-auth.ts +821 -0
  183. package/src/util/ai-runner-qa-tools.ts +3045 -0
  184. package/src/util/aicoder-runner-v6.ts +526 -0
  185. package/src/util/common.ts +649 -0
  186. package/src/util/customer-portal-password.ts +183 -0
  187. package/src/util/error-reporter.ts +332 -0
  188. package/src/util/error-tracking.ts +79 -0
  189. package/src/util/openai-usage-cost.ts +114 -0
  190. package/src/util/report-builder-unwinds.ts +180 -0
  191. package/src/util/runner-process-janitor.ts +219 -0
  192. package/src/util/schema-report-builder.ts +448 -0
  193. package/src/util/slow-query-reporter.ts +216 -0
  194. package/src/util/subscription-dependency-context.ts +1096 -0
  195. package/src/util/support-runner-v5.ts +897 -0
  196. package/src/util/tokenizer.ts +38 -0
  197. package/src/workers/codex-runner.worker.ts +142 -0
  198. package/start_server.sh +5 -0
  199. package/tests/ai-assistant-corpus-build.ts +484 -0
  200. package/tests/ai-assistant-corpus-replay-e2e.ts +774 -0
  201. package/tests/ai-assistant-data-parity-e2e.ts +1989 -0
  202. package/tests/ai-assistant-eval-triage.ts +831 -0
  203. package/tests/ai-assistant-openai-e2e.ts +1061 -0
  204. package/tests/ai-assistant-openai-git-e2e.ts +155 -0
  205. package/tests/ai-assistant-preflight-matrix.ts +215 -0
  206. package/tests/ai-assistant-routing-eval.test.ts +560 -0
  207. package/tests/ai-assistant-snf-live-eval.ts +975 -0
  208. package/tests/ai-assistant-utils.test.ts +2968 -0
  209. package/tests/ai-run-eval.test.ts +88 -0
  210. package/tests/ai-run-evidence.test.ts +232 -0
  211. package/tests/ai-runner-contract.test.ts +488 -0
  212. package/tests/aicoder-runner-v6.test.ts +92 -0
  213. package/tests/error-reporter.test.ts +145 -0
  214. package/tests/method-publication-generator.test.ts +46 -0
  215. package/tests/report-builder-linking.test.ts +79 -0
  216. package/tests/resolveio-platform-intelligence.test.ts +352 -0
  217. package/tests/server-app-cron-owner.test.ts +127 -0
  218. package/tests/subscription-connect-race.test.ts +158 -0
  219. package/tests/subscription-dependency-context.test.ts +324 -0
  220. package/tests/subscription-manager-collection-tracking.test.ts +86 -0
  221. package/tests/subscription-manager-invalidation.test.ts +86 -0
  222. package/tests/support-runner-v5.test.ts +191 -0
  223. package/tsconfig.json +34 -0
  224. package/ai/assistant-core-heuristics.d.ts +0 -11
  225. package/ai/assistant-core-heuristics.js +0 -356
  226. package/ai/assistant-core-heuristics.js.map +0 -1
  227. package/ai/resolveio-platform-intelligence-memory-corpus.d.ts +0 -3
  228. package/ai/resolveio-platform-intelligence-memory-corpus.js +0 -214
  229. package/ai/resolveio-platform-intelligence-memory-corpus.js.map +0 -1
  230. package/ai/resolveio-platform-intelligence-memory.d.ts +0 -20
  231. package/ai/resolveio-platform-intelligence-memory.js +0 -341
  232. package/ai/resolveio-platform-intelligence-memory.js.map +0 -1
  233. package/ai/resolveio-platform-intelligence-types.js +0 -4
  234. package/ai/resolveio-platform-intelligence-types.js.map +0 -1
  235. package/ai/resolveio-platform-intelligence.d.ts +0 -6
  236. package/ai/resolveio-platform-intelligence.js +0 -463
  237. package/ai/resolveio-platform-intelligence.js.map +0 -1
  238. package/client-server-app.d.ts +0 -1
  239. package/client-server-app.js +0 -68
  240. package/client-server-app.js.map +0 -1
  241. package/collections/ai-terminal-conversation.collection.d.ts +0 -2
  242. package/collections/ai-terminal-conversation.collection.js +0 -140
  243. package/collections/ai-terminal-conversation.collection.js.map +0 -1
  244. package/collections/ai-terminal-issue-report.collection.d.ts +0 -2
  245. package/collections/ai-terminal-issue-report.collection.js +0 -148
  246. package/collections/ai-terminal-issue-report.collection.js.map +0 -1
  247. package/collections/ai-terminal-message.collection.d.ts +0 -2
  248. package/collections/ai-terminal-message.collection.js +0 -121
  249. package/collections/ai-terminal-message.collection.js.map +0 -1
  250. package/collections/app-setting.collection.d.ts +0 -3
  251. package/collections/app-setting.collection.js +0 -103
  252. package/collections/app-setting.collection.js.map +0 -1
  253. package/collections/app-status.collection.d.ts +0 -3
  254. package/collections/app-status.collection.js +0 -57
  255. package/collections/app-status.collection.js.map +0 -1
  256. package/collections/communication-metric.collection.d.ts +0 -2
  257. package/collections/communication-metric.collection.js +0 -133
  258. package/collections/communication-metric.collection.js.map +0 -1
  259. package/collections/counter.collection.d.ts +0 -3
  260. package/collections/counter.collection.js +0 -56
  261. package/collections/counter.collection.js.map +0 -1
  262. package/collections/cron-job-history.collection.d.ts +0 -3
  263. package/collections/cron-job-history.collection.js +0 -137
  264. package/collections/cron-job-history.collection.js.map +0 -1
  265. package/collections/cron-job.collection.d.ts +0 -3
  266. package/collections/cron-job.collection.js +0 -92
  267. package/collections/cron-job.collection.js.map +0 -1
  268. package/collections/customer-notification.collection.d.ts +0 -3
  269. package/collections/customer-notification.collection.js +0 -130
  270. package/collections/customer-notification.collection.js.map +0 -1
  271. package/collections/customer-portal-password.collection.d.ts +0 -3
  272. package/collections/customer-portal-password.collection.js +0 -75
  273. package/collections/customer-portal-password.collection.js.map +0 -1
  274. package/collections/email-history.collection.d.ts +0 -3
  275. package/collections/email-history.collection.js +0 -134
  276. package/collections/email-history.collection.js.map +0 -1
  277. package/collections/email-verified.collection.d.ts +0 -3
  278. package/collections/email-verified.collection.js +0 -62
  279. package/collections/email-verified.collection.js.map +0 -1
  280. package/collections/file.collection.d.ts +0 -3
  281. package/collections/file.collection.js +0 -74
  282. package/collections/file.collection.js.map +0 -1
  283. package/collections/flag-update.collection.d.ts +0 -3
  284. package/collections/flag-update.collection.js +0 -57
  285. package/collections/flag-update.collection.js.map +0 -1
  286. package/collections/flag.collection.d.ts +0 -3
  287. package/collections/flag.collection.js +0 -57
  288. package/collections/flag.collection.js.map +0 -1
  289. package/collections/log-method-latency.collection.d.ts +0 -3
  290. package/collections/log-method-latency.collection.js +0 -77
  291. package/collections/log-method-latency.collection.js.map +0 -1
  292. package/collections/log-subscription.collection.d.ts +0 -3
  293. package/collections/log-subscription.collection.js +0 -80
  294. package/collections/log-subscription.collection.js.map +0 -1
  295. package/collections/log.collection.d.ts +0 -3
  296. package/collections/log.collection.js +0 -93
  297. package/collections/log.collection.js.map +0 -1
  298. package/collections/logged-in-users.collection.d.ts +0 -3
  299. package/collections/logged-in-users.collection.js +0 -67
  300. package/collections/logged-in-users.collection.js.map +0 -1
  301. package/collections/monitor-cpu.collection.d.ts +0 -3
  302. package/collections/monitor-cpu.collection.js +0 -65
  303. package/collections/monitor-cpu.collection.js.map +0 -1
  304. package/collections/monitor-function.collection.d.ts +0 -3
  305. package/collections/monitor-function.collection.js +0 -74
  306. package/collections/monitor-function.collection.js.map +0 -1
  307. package/collections/monitor-memory.collection.d.ts +0 -3
  308. package/collections/monitor-memory.collection.js +0 -77
  309. package/collections/monitor-memory.collection.js.map +0 -1
  310. package/collections/monitor-mongo.collection.d.ts +0 -3
  311. package/collections/monitor-mongo.collection.js +0 -71
  312. package/collections/monitor-mongo.collection.js.map +0 -1
  313. package/collections/notification.collection.d.ts +0 -3
  314. package/collections/notification.collection.js +0 -57
  315. package/collections/notification.collection.js.map +0 -1
  316. package/collections/openai-usage-ledger.collection.d.ts +0 -2
  317. package/collections/openai-usage-ledger.collection.js +0 -124
  318. package/collections/openai-usage-ledger.collection.js.map +0 -1
  319. package/collections/report-builder-dashboard-builder.collection.d.ts +0 -3
  320. package/collections/report-builder-dashboard-builder.collection.js +0 -109
  321. package/collections/report-builder-dashboard-builder.collection.js.map +0 -1
  322. package/collections/report-builder-library.collection.d.ts +0 -3
  323. package/collections/report-builder-library.collection.js +0 -87
  324. package/collections/report-builder-library.collection.js.map +0 -1
  325. package/collections/report-builder-report.collection.d.ts +0 -4
  326. package/collections/report-builder-report.collection.js +0 -184
  327. package/collections/report-builder-report.collection.js.map +0 -1
  328. package/collections/user-group.collection.d.ts +0 -4
  329. package/collections/user-group.collection.js +0 -89
  330. package/collections/user-group.collection.js.map +0 -1
  331. package/collections/user-guide.collection.d.ts +0 -3
  332. package/collections/user-guide.collection.js +0 -57
  333. package/collections/user-guide.collection.js.map +0 -1
  334. package/collections/user.collection.d.ts +0 -4
  335. package/collections/user.collection.js +0 -180
  336. package/collections/user.collection.js.map +0 -1
  337. package/cron/cron.d.ts +0 -14
  338. package/cron/cron.js +0 -216
  339. package/cron/cron.js.map +0 -1
  340. package/fixtures/cron-jobs.d.ts +0 -1
  341. package/fixtures/cron-jobs.js +0 -150
  342. package/fixtures/cron-jobs.js.map +0 -1
  343. package/fixtures/init.d.ts +0 -1
  344. package/fixtures/init.js +0 -91
  345. package/fixtures/init.js.map +0 -1
  346. package/http/auth.d.ts +0 -2
  347. package/http/auth.js +0 -951
  348. package/http/auth.js.map +0 -1
  349. package/http/health.d.ts +0 -1
  350. package/http/health.js +0 -11
  351. package/http/health.js.map +0 -1
  352. package/http/home.d.ts +0 -1
  353. package/http/home.js +0 -134
  354. package/http/home.js.map +0 -1
  355. package/http/slow-query-publication.d.ts +0 -2
  356. package/http/slow-query-publication.js +0 -99
  357. package/http/slow-query-publication.js.map +0 -1
  358. package/index.d.ts +0 -1
  359. package/index.js +0 -19
  360. package/index.js.map +0 -1
  361. package/managers/ai-assistant-codex-manager.manager.d.ts +0 -67
  362. package/managers/ai-assistant-codex-manager.manager.js +0 -1113
  363. package/managers/ai-assistant-codex-manager.manager.js.map +0 -1
  364. package/managers/communication-metric.manager.d.ts +0 -16
  365. package/managers/communication-metric.manager.js +0 -134
  366. package/managers/communication-metric.manager.js.map +0 -1
  367. package/managers/cron.manager.d.ts +0 -20
  368. package/managers/cron.manager.js +0 -534
  369. package/managers/cron.manager.js.map +0 -1
  370. package/managers/customer-notification-content.manager.d.ts +0 -55
  371. package/managers/customer-notification-content.manager.js +0 -158
  372. package/managers/customer-notification-content.manager.js.map +0 -1
  373. package/managers/diagnostic-manager-bootstrap.d.ts +0 -9
  374. package/managers/diagnostic-manager-bootstrap.js +0 -260
  375. package/managers/diagnostic-manager-bootstrap.js.map +0 -1
  376. package/managers/error-auto-fix.manager.d.ts +0 -149
  377. package/managers/error-auto-fix.manager.js +0 -3064
  378. package/managers/error-auto-fix.manager.js.map +0 -1
  379. package/managers/local-log.manager.d.ts +0 -18
  380. package/managers/local-log.manager.js +0 -88
  381. package/managers/local-log.manager.js.map +0 -1
  382. package/managers/method.manager.d.ts +0 -84
  383. package/managers/method.manager.js +0 -1964
  384. package/managers/method.manager.js.map +0 -1
  385. package/managers/mongo.manager.d.ts +0 -224
  386. package/managers/mongo.manager.js +0 -5000
  387. package/managers/mongo.manager.js.map +0 -1
  388. package/managers/monitor.manager.d.ts +0 -70
  389. package/managers/monitor.manager.js +0 -550
  390. package/managers/monitor.manager.js.map +0 -1
  391. package/managers/openai-usage-ledger.manager.d.ts +0 -15
  392. package/managers/openai-usage-ledger.manager.js +0 -144
  393. package/managers/openai-usage-ledger.manager.js.map +0 -1
  394. package/managers/slow-query-verifier.manager.d.ts +0 -144
  395. package/managers/slow-query-verifier.manager.js +0 -3857
  396. package/managers/slow-query-verifier.manager.js.map +0 -1
  397. package/managers/slow-query.manager.d.ts +0 -28
  398. package/managers/slow-query.manager.js +0 -468
  399. package/managers/slow-query.manager.js.map +0 -1
  400. package/managers/subscription.manager.d.ts +0 -169
  401. package/managers/subscription.manager.js +0 -3434
  402. package/managers/subscription.manager.js.map +0 -1
  403. package/managers/websocket.manager.d.ts +0 -73
  404. package/managers/websocket.manager.js +0 -673
  405. package/managers/websocket.manager.js.map +0 -1
  406. package/managers/worker-dispatcher.manager.d.ts +0 -120
  407. package/managers/worker-dispatcher.manager.js +0 -1266
  408. package/managers/worker-dispatcher.manager.js.map +0 -1
  409. package/managers/worker-server.manager.d.ts +0 -35
  410. package/managers/worker-server.manager.js +0 -582
  411. package/managers/worker-server.manager.js.map +0 -1
  412. package/methods/accounts.d.ts +0 -2
  413. package/methods/accounts.js +0 -624
  414. package/methods/accounts.js.map +0 -1
  415. package/methods/ai-terminal.d.ts +0 -337
  416. package/methods/ai-terminal.js +0 -23166
  417. package/methods/ai-terminal.js.map +0 -1
  418. package/methods/app-settings.d.ts +0 -2
  419. package/methods/app-settings.js +0 -169
  420. package/methods/app-settings.js.map +0 -1
  421. package/methods/aws.d.ts +0 -2
  422. package/methods/aws.js +0 -877
  423. package/methods/aws.js.map +0 -1
  424. package/methods/collections.d.ts +0 -2
  425. package/methods/collections.js +0 -719
  426. package/methods/collections.js.map +0 -1
  427. package/methods/counters.d.ts +0 -2
  428. package/methods/counters.js +0 -113
  429. package/methods/counters.js.map +0 -1
  430. package/methods/cron-jobs.d.ts +0 -2
  431. package/methods/cron-jobs.js +0 -2475
  432. package/methods/cron-jobs.js.map +0 -1
  433. package/methods/customer-notifications.d.ts +0 -2
  434. package/methods/customer-notifications.js +0 -528
  435. package/methods/customer-notifications.js.map +0 -1
  436. package/methods/diagnostics.d.ts +0 -2
  437. package/methods/diagnostics.js +0 -703
  438. package/methods/diagnostics.js.map +0 -1
  439. package/methods/flag-updates.d.ts +0 -2
  440. package/methods/flag-updates.js +0 -8
  441. package/methods/flag-updates.js.map +0 -1
  442. package/methods/flags.d.ts +0 -2
  443. package/methods/flags.js +0 -8
  444. package/methods/flags.js.map +0 -1
  445. package/methods/logs.d.ts +0 -2
  446. package/methods/logs.js +0 -751
  447. package/methods/logs.js.map +0 -1
  448. package/methods/mongo-explorer.d.ts +0 -2
  449. package/methods/mongo-explorer.js +0 -1808
  450. package/methods/mongo-explorer.js.map +0 -1
  451. package/methods/monitor.d.ts +0 -2
  452. package/methods/monitor.js +0 -543
  453. package/methods/monitor.js.map +0 -1
  454. package/methods/pdf.d.ts +0 -2
  455. package/methods/pdf.js +0 -1216
  456. package/methods/pdf.js.map +0 -1
  457. package/methods/publications.d.ts +0 -1
  458. package/methods/publications.js +0 -183
  459. package/methods/publications.js.map +0 -1
  460. package/methods/report-builder.d.ts +0 -2
  461. package/methods/report-builder.js +0 -3094
  462. package/methods/report-builder.js.map +0 -1
  463. package/methods/support.d.ts +0 -2
  464. package/methods/support.js +0 -430
  465. package/methods/support.js.map +0 -1
  466. package/models/ai-terminal-conversation.model.d.ts +0 -17
  467. package/models/ai-terminal-conversation.model.js +0 -4
  468. package/models/ai-terminal-conversation.model.js.map +0 -1
  469. package/models/ai-terminal-issue-report.model.d.ts +0 -19
  470. package/models/ai-terminal-issue-report.model.js +0 -4
  471. package/models/ai-terminal-issue-report.model.js.map +0 -1
  472. package/models/ai-terminal-message.model.d.ts +0 -22
  473. package/models/ai-terminal-message.model.js +0 -4
  474. package/models/ai-terminal-message.model.js.map +0 -1
  475. package/models/app-setting.model.d.ts +0 -16
  476. package/models/app-setting.model.js +0 -4
  477. package/models/app-setting.model.js.map +0 -1
  478. package/models/app-status.model.js +0 -4
  479. package/models/app-status.model.js.map +0 -1
  480. package/models/billing-logged-in-users.model.js +0 -4
  481. package/models/billing-logged-in-users.model.js.map +0 -1
  482. package/models/collection-document.model.d.ts +0 -21
  483. package/models/collection-document.model.js +0 -4
  484. package/models/collection-document.model.js.map +0 -1
  485. package/models/communication-metric.model.d.ts +0 -20
  486. package/models/communication-metric.model.js +0 -4
  487. package/models/communication-metric.model.js.map +0 -1
  488. package/models/counter.model.js +0 -4
  489. package/models/counter.model.js.map +0 -1
  490. package/models/cron-job-history.model.d.ts +0 -15
  491. package/models/cron-job-history.model.js +0 -4
  492. package/models/cron-job-history.model.js.map +0 -1
  493. package/models/cron-job.model.d.ts +0 -14
  494. package/models/cron-job.model.js +0 -4
  495. package/models/cron-job.model.js.map +0 -1
  496. package/models/customer-notification.model.d.ts +0 -26
  497. package/models/customer-notification.model.js +0 -4
  498. package/models/customer-notification.model.js.map +0 -1
  499. package/models/customer-portal-password.model.d.ts +0 -11
  500. package/models/customer-portal-password.model.js +0 -4
  501. package/models/customer-portal-password.model.js.map +0 -1
  502. package/models/dialog.model.d.ts +0 -23
  503. package/models/dialog.model.js +0 -4
  504. package/models/dialog.model.js.map +0 -1
  505. package/models/email-history.model.d.ts +0 -32
  506. package/models/email-history.model.js.map +0 -1
  507. package/models/email-verified.model.js +0 -4
  508. package/models/email-verified.model.js.map +0 -1
  509. package/models/file.model.js +0 -4
  510. package/models/file.model.js.map +0 -1
  511. package/models/flag-update.model.js +0 -4
  512. package/models/flag-update.model.js.map +0 -1
  513. package/models/flag.model.js +0 -4
  514. package/models/flag.model.js.map +0 -1
  515. package/models/log-method-latency.model.d.ts +0 -10
  516. package/models/log-method-latency.model.js +0 -4
  517. package/models/log-method-latency.model.js.map +0 -1
  518. package/models/log-subscription.model.js +0 -4
  519. package/models/log-subscription.model.js.map +0 -1
  520. package/models/log.model.d.ts +0 -17
  521. package/models/log.model.js +0 -4
  522. package/models/log.model.js.map +0 -1
  523. package/models/logged-in-users.model.js +0 -4
  524. package/models/logged-in-users.model.js.map +0 -1
  525. package/models/method-response.model.js +0 -4
  526. package/models/method-response.model.js.map +0 -1
  527. package/models/method.model.d.ts +0 -26
  528. package/models/method.model.js +0 -4
  529. package/models/method.model.js.map +0 -1
  530. package/models/monitor-cpu.model.js +0 -4
  531. package/models/monitor-cpu.model.js.map +0 -1
  532. package/models/monitor-function.model.d.ts +0 -14
  533. package/models/monitor-function.model.js +0 -4
  534. package/models/monitor-function.model.js.map +0 -1
  535. package/models/monitor-memory.model.d.ts +0 -15
  536. package/models/monitor-memory.model.js +0 -4
  537. package/models/monitor-memory.model.js.map +0 -1
  538. package/models/monitor-mongo.model.d.ts +0 -13
  539. package/models/monitor-mongo.model.js +0 -4
  540. package/models/monitor-mongo.model.js.map +0 -1
  541. package/models/notification.model.js +0 -4
  542. package/models/notification.model.js.map +0 -1
  543. package/models/openai-usage-ledger.model.d.ts +0 -15
  544. package/models/openai-usage-ledger.model.js +0 -4
  545. package/models/openai-usage-ledger.model.js.map +0 -1
  546. package/models/pagination.model.d.ts +0 -11
  547. package/models/pagination.model.js +0 -28
  548. package/models/pagination.model.js.map +0 -1
  549. package/models/permission.model.d.ts +0 -12
  550. package/models/permission.model.js +0 -4
  551. package/models/permission.model.js.map +0 -1
  552. package/models/report-builder-dashboard-builder.model.d.ts +0 -25
  553. package/models/report-builder-dashboard-builder.model.js +0 -4
  554. package/models/report-builder-dashboard-builder.model.js.map +0 -1
  555. package/models/report-builder-library.model.d.ts +0 -17
  556. package/models/report-builder-library.model.js +0 -4
  557. package/models/report-builder-library.model.js.map +0 -1
  558. package/models/report-builder-report.model.d.ts +0 -121
  559. package/models/report-builder-report.model.js +0 -4
  560. package/models/report-builder-report.model.js.map +0 -1
  561. package/models/report-builder.model.d.ts +0 -61
  562. package/models/report-builder.model.js +0 -4
  563. package/models/report-builder.model.js.map +0 -1
  564. package/models/select-data-label.model.d.ts +0 -9
  565. package/models/select-data-label.model.js +0 -4
  566. package/models/select-data-label.model.js.map +0 -1
  567. package/models/server-message.model.d.ts +0 -32
  568. package/models/server-message.model.js +0 -4
  569. package/models/server-message.model.js.map +0 -1
  570. package/models/slow-query-report.model.d.ts +0 -23
  571. package/models/slow-query-report.model.js +0 -4
  572. package/models/slow-query-report.model.js.map +0 -1
  573. package/models/subscription.model.d.ts +0 -31
  574. package/models/subscription.model.js +0 -4
  575. package/models/subscription.model.js.map +0 -1
  576. package/models/support-ticket.model.d.ts +0 -87
  577. package/models/support-ticket.model.js +0 -4
  578. package/models/support-ticket.model.js.map +0 -1
  579. package/models/user-group.model.d.ts +0 -20
  580. package/models/user-group.model.js +0 -4
  581. package/models/user-group.model.js.map +0 -1
  582. package/models/user-guide.model.js +0 -4
  583. package/models/user-guide.model.js.map +0 -1
  584. package/models/user.model.d.ts +0 -84
  585. package/models/user.model.js +0 -4
  586. package/models/user.model.js.map +0 -1
  587. package/private/images/ResolveIO.png +0 -0
  588. package/public_api.js +0 -118
  589. package/public_api.js.map +0 -1
  590. package/publications/ai-terminal.d.ts +0 -1
  591. package/publications/ai-terminal.js +0 -122
  592. package/publications/ai-terminal.js.map +0 -1
  593. package/publications/app-settings.d.ts +0 -2
  594. package/publications/app-settings.js +0 -28
  595. package/publications/app-settings.js.map +0 -1
  596. package/publications/app-status.d.ts +0 -2
  597. package/publications/app-status.js +0 -16
  598. package/publications/app-status.js.map +0 -1
  599. package/publications/cron-jobs.d.ts +0 -2
  600. package/publications/cron-jobs.js +0 -88
  601. package/publications/cron-jobs.js.map +0 -1
  602. package/publications/customer-notifications.d.ts +0 -2
  603. package/publications/customer-notifications.js +0 -161
  604. package/publications/customer-notifications.js.map +0 -1
  605. package/publications/files.d.ts +0 -2
  606. package/publications/files.js +0 -36
  607. package/publications/files.js.map +0 -1
  608. package/publications/flags-update.d.ts +0 -2
  609. package/publications/flags-update.js +0 -22
  610. package/publications/flags-update.js.map +0 -1
  611. package/publications/flags.d.ts +0 -2
  612. package/publications/flags.js +0 -22
  613. package/publications/flags.js.map +0 -1
  614. package/publications/logs.d.ts +0 -2
  615. package/publications/logs.js +0 -164
  616. package/publications/logs.js.map +0 -1
  617. package/publications/notifications.d.ts +0 -2
  618. package/publications/notifications.js +0 -16
  619. package/publications/notifications.js.map +0 -1
  620. package/publications/report-builder-dashboard-builders.d.ts +0 -2
  621. package/publications/report-builder-dashboard-builders.js +0 -42
  622. package/publications/report-builder-dashboard-builders.js.map +0 -1
  623. package/publications/report-builder-libraries.d.ts +0 -2
  624. package/publications/report-builder-libraries.js +0 -90
  625. package/publications/report-builder-libraries.js.map +0 -1
  626. package/publications/report-builder-reports.d.ts +0 -2
  627. package/publications/report-builder-reports.js +0 -50
  628. package/publications/report-builder-reports.js.map +0 -1
  629. package/publications/super-admin.d.ts +0 -2
  630. package/publications/super-admin.js +0 -16
  631. package/publications/super-admin.js.map +0 -1
  632. package/publications/user-groups.d.ts +0 -1
  633. package/publications/user-groups.js +0 -16
  634. package/publications/user-groups.js.map +0 -1
  635. package/publications/user-guides.d.ts +0 -1
  636. package/publications/user-guides.js +0 -16
  637. package/publications/user-guides.js.map +0 -1
  638. package/resolveio-server-app.d.ts +0 -70
  639. package/resolveio-server-app.js +0 -801
  640. package/resolveio-server-app.js.map +0 -1
  641. package/server-app.d.ts +0 -228
  642. package/server-app.js +0 -3566
  643. package/server-app.js.map +0 -1
  644. package/services/codex-client.d.ts +0 -128
  645. package/services/codex-client.js +0 -1629
  646. package/services/codex-client.js.map +0 -1
  647. package/services/openai-client.d.ts +0 -46
  648. package/services/openai-client.js +0 -318
  649. package/services/openai-client.js.map +0 -1
  650. package/types/error-report.d.ts +0 -25
  651. package/types/error-report.js +0 -4
  652. package/types/error-report.js.map +0 -1
  653. package/types/slow-query-report.d.ts +0 -27
  654. package/types/slow-query-report.js +0 -6
  655. package/types/slow-query-report.js.map +0 -1
  656. package/util/ai-qa-policy.d.ts +0 -124
  657. package/util/ai-qa-policy.js +0 -736
  658. package/util/ai-qa-policy.js.map +0 -1
  659. package/util/ai-runner-artifacts.d.ts +0 -82
  660. package/util/ai-runner-artifacts.js +0 -713
  661. package/util/ai-runner-artifacts.js.map +0 -1
  662. package/util/ai-runner-qa-auth.d.ts +0 -5
  663. package/util/ai-runner-qa-auth.js +0 -822
  664. package/util/ai-runner-qa-auth.js.map +0 -1
  665. package/util/ai-runner-qa-tools.d.ts +0 -25
  666. package/util/ai-runner-qa-tools.js +0 -2910
  667. package/util/ai-runner-qa-tools.js.map +0 -1
  668. package/util/aicoder-runner-v6.d.ts +0 -168
  669. package/util/aicoder-runner-v6.js +0 -347
  670. package/util/aicoder-runner-v6.js.map +0 -1
  671. package/util/common.d.ts +0 -31
  672. package/util/common.js +0 -683
  673. package/util/common.js.map +0 -1
  674. package/util/customer-portal-password.d.ts +0 -13
  675. package/util/customer-portal-password.js +0 -209
  676. package/util/customer-portal-password.js.map +0 -1
  677. package/util/error-reporter.d.ts +0 -52
  678. package/util/error-reporter.js +0 -326
  679. package/util/error-reporter.js.map +0 -1
  680. package/util/error-tracking.d.ts +0 -13
  681. package/util/error-tracking.js +0 -120
  682. package/util/error-tracking.js.map +0 -1
  683. package/util/report-builder-unwinds.d.ts +0 -15
  684. package/util/report-builder-unwinds.js +0 -156
  685. package/util/report-builder-unwinds.js.map +0 -1
  686. package/util/runner-process-janitor.d.ts +0 -27
  687. package/util/runner-process-janitor.js +0 -208
  688. package/util/runner-process-janitor.js.map +0 -1
  689. package/util/schema-report-builder.d.ts +0 -6
  690. package/util/schema-report-builder.js +0 -481
  691. package/util/schema-report-builder.js.map +0 -1
  692. package/util/slow-query-reporter.d.ts +0 -28
  693. package/util/slow-query-reporter.js +0 -226
  694. package/util/slow-query-reporter.js.map +0 -1
  695. package/util/subscription-dependency-context.d.ts +0 -34
  696. package/util/subscription-dependency-context.js +0 -1283
  697. package/util/subscription-dependency-context.js.map +0 -1
  698. package/util/support-runner-v5.d.ts +0 -250
  699. package/util/support-runner-v5.js +0 -634
  700. package/util/support-runner-v5.js.map +0 -1
  701. package/util/tokenizer.d.ts +0 -5
  702. package/util/tokenizer.js +0 -41
  703. package/util/tokenizer.js.map +0 -1
  704. package/workers/codex-runner.worker.d.ts +0 -1
  705. package/workers/codex-runner.worker.js +0 -192
  706. package/workers/codex-runner.worker.js.map +0 -1
  707. /package/{private → src/private}/email-templates/enrollment.html +0 -0
  708. /package/{private → src/private}/email-templates/forgot-password.html +0 -0
  709. /package/{private → src/private}/email-templates/support-ticket-deleted.html +0 -0
  710. /package/{private → src/private}/email-templates/support-ticket-modified.html +0 -0
  711. /package/{private → src/private}/email-templates/support-ticket.html +0 -0
@@ -0,0 +1,1061 @@
1
+ import * as fs from 'fs';
2
+ import * as path from 'path';
3
+ import { ResolveIOServer } from '../src/resolveio-server-app';
4
+
5
+ type E2EUserProfile =
6
+ | 'super_admin'
7
+ | 'internal_invoice'
8
+ | 'internal_no_invoice'
9
+ | 'internal_blend_plural'
10
+ | 'internal_misc_invoice'
11
+ | 'customer_invoice'
12
+ | 'customer_no_invoice';
13
+
14
+ type E2ECase = {
15
+ id: string;
16
+ prompt: string;
17
+ userProfile?: E2EUserProfile;
18
+ expect: {
19
+ hasTable?: boolean;
20
+ revenueMetricGrid?: boolean;
21
+ revenueCustomerGrid?: boolean;
22
+ timeGrain?: 'day' | 'week' | 'month' | 'quarter' | 'year';
23
+ dataResponse?: boolean;
24
+ requestType?: 'data' | 'navigation' | 'feature_info' | 'bug_issue' | 'mixed' | 'unknown';
25
+ dataQuestion?: boolean;
26
+ directiveType?: 'read' | 'aggregate';
27
+ fastPathType?: 'navigation' | 'change_history';
28
+ fastPathGeneric?: boolean;
29
+ allowFastPathMiss?: boolean;
30
+ mustIncludeRoute?: string;
31
+ mustNotIncludeRoute?: string;
32
+ mustIncludeAnyRoute?: string[];
33
+ mustNotIncludeAnyRoute?: string[];
34
+ mustContainAnyText?: string[];
35
+ mustNotContainAnyText?: string[];
36
+ };
37
+ };
38
+
39
+ type E2EResult = {
40
+ id: string;
41
+ attempt?: number;
42
+ prompt: string;
43
+ pass: boolean;
44
+ error?: string;
45
+ responsePreview?: string;
46
+ conversationId?: string;
47
+ classification?: {
48
+ type?: string;
49
+ dataQuestion?: boolean;
50
+ source?: string;
51
+ };
52
+ fastPath?: {
53
+ type?: string;
54
+ reason?: string;
55
+ generic?: boolean;
56
+ commitCount?: number;
57
+ };
58
+ userProfile?: E2EUserProfile;
59
+ tableHeader?: string[];
60
+ };
61
+
62
+ const E2E_PROMPTS: E2ECase[] = [
63
+ {
64
+ id: 'navigation_invoice_route_link_fast_path',
65
+ prompt: 'I just need the route link for invoice list.',
66
+ userProfile: 'super_admin',
67
+ expect: { fastPathType: 'navigation', mustIncludeRoute: '/invoice/list' }
68
+ },
69
+ {
70
+ id: 'navigation_invoice_route_link_customer_no_invoice',
71
+ prompt: 'I just need the route link for invoice list.',
72
+ userProfile: 'customer_no_invoice',
73
+ expect: { fastPathType: 'navigation', mustNotIncludeRoute: '/invoice/' }
74
+ },
75
+ {
76
+ id: 'navigation_blend_route_link_internal_blend_plural',
77
+ prompt: 'I just need the route link for chemical blend list.',
78
+ userProfile: 'internal_blend_plural',
79
+ expect: {
80
+ fastPathType: 'navigation',
81
+ mustIncludeAnyRoute: ['/chemical-blend/list', '/chemical-blends/list'],
82
+ mustNotIncludeRoute: '/invoice/'
83
+ }
84
+ },
85
+ {
86
+ id: 'navigation_blend_route_link_internal_no_blend',
87
+ prompt: 'I just need the route link for chemical blend list.',
88
+ userProfile: 'internal_no_invoice',
89
+ expect: {
90
+ fastPathType: 'navigation',
91
+ mustNotIncludeAnyRoute: ['/chemical-blend/', '/chemical-blends/']
92
+ }
93
+ },
94
+ {
95
+ id: 'navigation_order_alias_route_link_internal_no_invoice',
96
+ prompt: 'I just need the route link for /order/list.',
97
+ userProfile: 'internal_no_invoice',
98
+ expect: {
99
+ fastPathType: 'navigation',
100
+ mustIncludeAnyRoute: ['/order/list', '/work-order/list']
101
+ }
102
+ },
103
+ {
104
+ id: 'wo_last20_this_week_group_status',
105
+ prompt: 'Show me the last 20 work orders created this week, grouped by status.',
106
+ expect: { hasTable: true, dataResponse: true }
107
+ },
108
+ {
109
+ id: 'wo_last20_this_week_by_status_alias',
110
+ prompt: 'Show me the last 20 work orders created this week by status.',
111
+ expect: { dataResponse: true, directiveType: 'aggregate' }
112
+ },
113
+ {
114
+ id: 'wo_completed_per_day_last_30d',
115
+ prompt: 'For the last 30 days, how many work orders were completed per day?',
116
+ expect: { hasTable: true, dataResponse: true }
117
+ },
118
+ {
119
+ id: 'wo_top_customers_last_6m',
120
+ prompt: 'List the top 10 customers by number of work orders in the last 6 months.',
121
+ expect: { hasTable: true, dataResponse: true }
122
+ },
123
+ {
124
+ id: 'wo_this_week_customer_summary',
125
+ prompt: 'Summarize the work orders this week by customer',
126
+ expect: {
127
+ dataResponse: true,
128
+ mustNotContainAnyText: [
129
+ "i couldn't run the data query",
130
+ 'positional operator in the middle of a path'
131
+ ]
132
+ }
133
+ },
134
+ {
135
+ id: 'wo_top_customers_last_6m_internal_no_invoice',
136
+ prompt: 'List the top 10 customers by number of work orders in the last 6 months.',
137
+ userProfile: 'internal_no_invoice',
138
+ expect: { dataResponse: true, mustNotContainAnyText: ['does not have permission', 'access denied'] }
139
+ },
140
+ {
141
+ id: 'wo_open_missing_required_fields',
142
+ prompt: 'Show open work orders with missing required fields (customer, location, or scheduled date).',
143
+ expect: { hasTable: true, dataResponse: true }
144
+ },
145
+ {
146
+ id: 'blend_last10_summary',
147
+ prompt: 'For blending: summarize the last 10 blend tickets with product, total volume, and created date.',
148
+ expect: { hasTable: true, dataResponse: true }
149
+ },
150
+ {
151
+ id: 'blend_used_red_ultra_lift_last_30d',
152
+ prompt: 'Which blends used Chemical Red Ultra Lift in the last 30 days? Include lot/batch and volume.',
153
+ expect: { hasTable: true, dataResponse: true }
154
+ },
155
+ {
156
+ id: 'blend_throughput_by_day_last_2w',
157
+ prompt: 'Show blend throughput by day for the last 2 weeks.',
158
+ expect: { hasTable: true, dataResponse: true }
159
+ },
160
+ {
161
+ id: 'blend_throughput_by_day_last_2w_internal_blend_plural',
162
+ prompt: 'Show blend throughput by day for the last 2 weeks.',
163
+ userProfile: 'internal_blend_plural',
164
+ expect: { dataResponse: true, mustNotContainAnyText: ['does not have permission', 'access denied'] }
165
+ },
166
+ {
167
+ id: 'blend_throughput_by_day_last_2w_internal_no_blend',
168
+ prompt: 'Show blend throughput by day for the last 2 weeks.',
169
+ userProfile: 'internal_no_invoice',
170
+ expect: { mustContainAnyText: ['does not have permission', 'access denied', 'ask an admin'] }
171
+ },
172
+ {
173
+ id: 'invoice_top_customers_total_last_6m',
174
+ prompt: 'Show top 10 customers by invoice total in the last 6 months.',
175
+ expect: { hasTable: true, dataResponse: true }
176
+ },
177
+ {
178
+ id: 'invoice_top_customers_total_last_6m_internal_invoice',
179
+ prompt: 'Show top 10 customers by invoice total in the last 6 months.',
180
+ userProfile: 'internal_invoice',
181
+ expect: { dataResponse: true, mustNotContainAnyText: ['does not have permission', 'access denied'] }
182
+ },
183
+ {
184
+ id: 'invoice_top_customers_total_last_6m_internal_misc_invoice',
185
+ prompt: 'Show top 10 customers by invoice total in the last 6 months.',
186
+ userProfile: 'internal_misc_invoice',
187
+ expect: { dataResponse: true, mustNotContainAnyText: ['does not have permission', 'access denied'] }
188
+ },
189
+ {
190
+ id: 'invoice_top_customers_total_last_6m_internal_no_invoice_denied',
191
+ prompt: 'Show top 10 customers by invoice total in the last 6 months.',
192
+ userProfile: 'internal_no_invoice',
193
+ expect: { mustContainAnyText: ['does not have permission', 'access denied', 'ask an admin'] }
194
+ },
195
+ {
196
+ id: 'invoice_top_customers_total_last_6m_customer_invoice',
197
+ prompt: 'Show top 10 customers by invoice total in the last 6 months.',
198
+ userProfile: 'customer_invoice',
199
+ expect: { dataResponse: true, mustNotContainAnyText: ['does not have permission', 'access denied'] }
200
+ },
201
+ {
202
+ id: 'invoice_top_customers_total_last_6m_customer_no_invoice_denied',
203
+ prompt: 'Show top 10 customers by invoice total in the last 6 months.',
204
+ userProfile: 'customer_no_invoice',
205
+ expect: { mustContainAnyText: ['does not have permission', 'access denied', 'customer portal only', 'not available in this view', 'scope'] }
206
+ },
207
+ {
208
+ id: 'invoice_revenue_monthly_grid_last_6m',
209
+ prompt: 'Break down my total revenue over the last 6 month by month',
210
+ expect: { hasTable: true, dataResponse: true, revenueMetricGrid: true }
211
+ },
212
+ {
213
+ id: 'invoice_revenue_monthly_customer_grid_last_6m',
214
+ prompt: 'Break down my total revenue over the last 6 month by month by each customer',
215
+ expect: { hasTable: true, dataResponse: true, revenueCustomerGrid: true, directiveType: 'aggregate' }
216
+ },
217
+ {
218
+ id: 'invoice_revenue_monthly_customer_grid_last_6m_per_customer',
219
+ prompt: 'Break down my total revenue over the last 6 month per month per customer',
220
+ expect: { hasTable: true, dataResponse: true, revenueCustomerGrid: true, directiveType: 'aggregate' }
221
+ },
222
+ {
223
+ id: 'invoice_revenue_weekly_grid_last_12w',
224
+ prompt: 'Break down my total revenue over the last 12 weeks by week',
225
+ expect: { hasTable: true, dataResponse: true, timeGrain: 'week' }
226
+ },
227
+ {
228
+ id: 'invoice_revenue_daily_grid_last_30d',
229
+ prompt: 'Break down my total revenue over the last 30 days by day',
230
+ expect: { hasTable: true, dataResponse: true, timeGrain: 'day' }
231
+ },
232
+ {
233
+ id: 'invoice_revenue_quarterly_grid_last_8q',
234
+ prompt: 'Break down my total revenue over the last 8 quarters by quarter',
235
+ expect: { hasTable: true, dataResponse: true, timeGrain: 'quarter' }
236
+ },
237
+ {
238
+ id: 'invoice_revenue_yearly_grid_last_3y',
239
+ prompt: 'Break down my total revenue over the last 3 years by year',
240
+ expect: { hasTable: true, dataResponse: true, timeGrain: 'year' }
241
+ },
242
+ {
243
+ id: 'invoice_missing_invoiced_date_this_month',
244
+ prompt: 'List invoices missing an invoiced date but created this month.',
245
+ expect: { hasTable: true, dataResponse: true }
246
+ },
247
+ {
248
+ id: 'deliveries_per_driver_last_month',
249
+ prompt: 'Break down the number of deliveries per driver last month',
250
+ expect: { hasTable: true, dataResponse: true }
251
+ },
252
+ {
253
+ id: 'deliveries_per_driver_last_30d_top10',
254
+ prompt: 'Deliveries by driver for the last 30 days (top 10 drivers).',
255
+ expect: { hasTable: true, dataResponse: true, directiveType: 'aggregate' }
256
+ },
257
+ {
258
+ id: 'support_ticket_billable_hours_per_user_per_month_last_6m',
259
+ prompt: 'Break up the support tickets last 6 months, give me a total per user per month of the amount of hours they are billing for',
260
+ expect: { dataResponse: true, directiveType: 'aggregate' }
261
+ },
262
+ {
263
+ id: 'last20_deliveries_product_volume_destination',
264
+ prompt: 'List the last 20 deliveries with product, volume, and destination.',
265
+ expect: { hasTable: true, dataResponse: true }
266
+ },
267
+ {
268
+ id: 'jobs_highest_treatment_volume_last_90d',
269
+ prompt: 'Which jobs have the highest treatment volume in the last 90 days?',
270
+ expect: { hasTable: true, dataResponse: true }
271
+ },
272
+ {
273
+ id: 'change_history_recent_invoice_updates',
274
+ prompt: 'What changed recently in invoices? Keep it high-level and no code.',
275
+ userProfile: 'super_admin',
276
+ expect: { fastPathType: 'change_history', fastPathGeneric: false }
277
+ },
278
+ {
279
+ id: 'change_history_invoice_document_last_updated',
280
+ prompt: 'When was the last time the invoice document was updated?',
281
+ userProfile: 'super_admin',
282
+ expect: { fastPathType: 'change_history' }
283
+ },
284
+ {
285
+ id: 'change_history_dropdown_added_to_page',
286
+ prompt: 'When does this dropdown for XYZ get added to this page? What does it do?',
287
+ userProfile: 'super_admin',
288
+ expect: { fastPathType: 'change_history' }
289
+ },
290
+ {
291
+ id: 'change_history_this_page_updates',
292
+ prompt: 'What changed on this page?',
293
+ userProfile: 'super_admin',
294
+ expect: { fastPathType: 'change_history' }
295
+ },
296
+ {
297
+ id: 'change_history_invoice_page_updates',
298
+ prompt: 'What changed on the invoice page?',
299
+ userProfile: 'super_admin',
300
+ expect: { fastPathType: 'change_history' }
301
+ },
302
+ {
303
+ id: 'change_history_last_update_did_what',
304
+ prompt: 'What did the last update just do?',
305
+ userProfile: 'super_admin',
306
+ expect: { fastPathType: 'change_history' }
307
+ },
308
+ {
309
+ id: 'change_history_last_system_update_did_what',
310
+ prompt: 'What did the last system update do?',
311
+ userProfile: 'super_admin',
312
+ expect: { fastPathType: 'change_history', allowFastPathMiss: true }
313
+ },
314
+ {
315
+ id: 'change_history_that_last_system_update_did_what',
316
+ prompt: 'What did that last system update do?',
317
+ userProfile: 'super_admin',
318
+ expect: { fastPathType: 'change_history', allowFastPathMiss: true }
319
+ },
320
+ {
321
+ id: 'change_history_recent_invoice_updates_customer_invoice',
322
+ prompt: 'What changed recently in invoices? Keep it high-level and no code.',
323
+ userProfile: 'customer_invoice',
324
+ expect: { fastPathType: 'change_history', fastPathGeneric: false }
325
+ },
326
+ {
327
+ id: 'change_history_recent_invoice_updates_internal_invoice',
328
+ prompt: 'What changed recently in invoices? Keep it high-level and no code.',
329
+ userProfile: 'internal_invoice',
330
+ expect: { fastPathType: 'change_history', fastPathGeneric: false }
331
+ },
332
+ {
333
+ id: 'change_history_recent_invoice_updates_internal_misc_invoice',
334
+ prompt: 'What changed recently in invoices? Keep it high-level and no code.',
335
+ userProfile: 'internal_misc_invoice',
336
+ expect: { fastPathType: 'change_history', fastPathGeneric: false }
337
+ },
338
+ {
339
+ id: 'change_history_duration_work_orders',
340
+ prompt: 'How long has the current work order behavior been this way?',
341
+ userProfile: 'internal_no_invoice',
342
+ expect: { fastPathType: 'change_history' }
343
+ },
344
+ {
345
+ id: 'change_history_exception_list_been_around',
346
+ prompt: 'What is this new exception list? How long has it been around?',
347
+ userProfile: 'super_admin',
348
+ expect: { fastPathType: 'change_history' }
349
+ },
350
+ {
351
+ id: 'change_history_exception_list_on_this_page_duration',
352
+ prompt: 'How long has the exception list been on this page?',
353
+ userProfile: 'super_admin',
354
+ expect: { fastPathType: 'change_history' }
355
+ },
356
+ {
357
+ id: 'change_history_recent_invoice_updates_no_access',
358
+ prompt: 'What changed recently in invoices? Keep it high-level and no code.',
359
+ userProfile: 'customer_no_invoice',
360
+ expect: { fastPathType: 'change_history', fastPathGeneric: true }
361
+ },
362
+ {
363
+ id: 'change_history_recent_invoice_updates_internal_no_invoice',
364
+ prompt: 'What changed recently in invoices? Keep it high-level and no code.',
365
+ userProfile: 'internal_no_invoice',
366
+ expect: { fastPathType: 'change_history', fastPathGeneric: true }
367
+ },
368
+ {
369
+ id: 'bug_issue_missing_asset_in_dropdown',
370
+ prompt: "Why don't I see this asset in this dropdown?",
371
+ expect: { requestType: 'bug_issue', dataQuestion: false }
372
+ },
373
+ {
374
+ id: 'bug_issue_work_order_close_greyed_out',
375
+ prompt: "Why won't this work order let me close it, it's greyed out?",
376
+ expect: { requestType: 'bug_issue', dataQuestion: false }
377
+ },
378
+ {
379
+ id: 'bug_issue_submit_button_red',
380
+ prompt: 'The submit button is red and I filled out everything. What is likely wrong?',
381
+ expect: { requestType: 'bug_issue', dataQuestion: false }
382
+ },
383
+ {
384
+ id: 'last20_completed_wo_customer_product_delivered_returned',
385
+ prompt: 'Break down the last 20 completed work orders and tell me what customer, what product and how much we delivered or returned',
386
+ expect: { hasTable: true, dataResponse: true }
387
+ }
388
+ ];
389
+
390
+ function parseDotEnvFile(filePath: string): Record<string, string> {
391
+ const parsed: Record<string, string> = {};
392
+ if (!fs.existsSync(filePath)) {
393
+ return parsed;
394
+ }
395
+ const content = fs.readFileSync(filePath, 'utf8');
396
+ content.split(/\r?\n/g).forEach((lineRaw) => {
397
+ const line = lineRaw.trim();
398
+ if (!line || line.startsWith('#')) {
399
+ return;
400
+ }
401
+ const index = line.indexOf('=');
402
+ if (index <= 0) {
403
+ return;
404
+ }
405
+ const key = line.slice(0, index).trim();
406
+ let value = line.slice(index + 1).trim();
407
+ if ((value.startsWith('"') && value.endsWith('"')) || (value.startsWith('\'') && value.endsWith('\''))) {
408
+ value = value.slice(1, -1);
409
+ }
410
+ if (key) {
411
+ parsed[key] = value;
412
+ }
413
+ });
414
+ return parsed;
415
+ }
416
+
417
+ function applyEnvFromDotEnv(filePath: string): void {
418
+ const parsed = parseDotEnvFile(filePath);
419
+ Object.keys(parsed).forEach((key) => {
420
+ if (!process.env[key]) {
421
+ process.env[key] = parsed[key];
422
+ }
423
+ });
424
+ }
425
+
426
+ function findMarkdownTableHeader(content: string): string[] {
427
+ const lines = String(content || '').split('\n');
428
+ for (let index = 0; index < lines.length - 1; index += 1) {
429
+ const header = lines[index]?.trim();
430
+ const separator = lines[index + 1]?.trim();
431
+ if (!header || !separator) {
432
+ continue;
433
+ }
434
+ if (!/^\|.+\|$/.test(header)) {
435
+ continue;
436
+ }
437
+ if (!/^\|?\s*:?-{3,}:?\s*(\|\s*:?-{3,}:?\s*)+\|?\s*$/.test(separator)) {
438
+ continue;
439
+ }
440
+ return header
441
+ .split('|')
442
+ .map(value => value.trim())
443
+ .filter(Boolean);
444
+ }
445
+ return [];
446
+ }
447
+
448
+ function isMonthHeader(value: string): boolean {
449
+ return /^\d{4}-\d{2}$/.test(String(value || '').trim());
450
+ }
451
+
452
+ function isDayHeader(value: string): boolean {
453
+ return /^\d{4}-\d{2}-\d{2}$/.test(String(value || '').trim());
454
+ }
455
+
456
+ function isWeekHeader(value: string): boolean {
457
+ const normalized = String(value || '').trim();
458
+ if (!normalized) {
459
+ return false;
460
+ }
461
+ return /^\d{4}-w\d{1,2}$/i.test(normalized)
462
+ || /^w\d{1,2}\s+\d{4}$/i.test(normalized)
463
+ || /^\d{4}-\d{2}-\d{2}\s+to\s+\d{4}-\d{2}-\d{2}$/i.test(normalized);
464
+ }
465
+
466
+ function isQuarterHeader(value: string): boolean {
467
+ const normalized = String(value || '').trim();
468
+ if (!normalized) {
469
+ return false;
470
+ }
471
+ return /^\d{4}-q[1-4]$/i.test(normalized)
472
+ || /^q[1-4]\s+\d{4}$/i.test(normalized)
473
+ || /^\d{4}\s+q[1-4]$/i.test(normalized);
474
+ }
475
+
476
+ function isYearHeader(value: string): boolean {
477
+ return /^\d{4}$/.test(String(value || '').trim());
478
+ }
479
+
480
+ function assertRevenueMetricGrid(header: string[]): string | null {
481
+ if (!header.length) {
482
+ return 'Expected markdown table header for monthly revenue grid.';
483
+ }
484
+ if (String(header[0] || '').toLowerCase() !== 'metric') {
485
+ return `Expected first column to be Metric, got ${header[0] || 'empty'}.`;
486
+ }
487
+ const monthColumns = header.slice(1).filter(isMonthHeader);
488
+ if (monthColumns.length < 3) {
489
+ return `Expected at least 3 month columns, got ${monthColumns.length}.`;
490
+ }
491
+ return null;
492
+ }
493
+
494
+ function assertRevenueCustomerGrid(header: string[]): string | null {
495
+ if (!header.length) {
496
+ return 'Expected markdown table header for customer monthly revenue grid.';
497
+ }
498
+ if (!String(header[0] || '').toLowerCase().includes('customer')) {
499
+ return `Expected first column to include Customer, got ${header[0] || 'empty'}.`;
500
+ }
501
+ const monthColumns = header.slice(1).filter(isMonthHeader);
502
+ if (monthColumns.length < 3) {
503
+ return `Expected at least 3 month columns, got ${monthColumns.length}.`;
504
+ }
505
+ return null;
506
+ }
507
+
508
+ function assertTimeGrainGrid(header: string[], grain: 'day' | 'week' | 'month' | 'quarter' | 'year'): string | null {
509
+ if (!header.length) {
510
+ return `Expected markdown table header for ${grain} report grid.`;
511
+ }
512
+ const first = String(header[0] || '').toLowerCase();
513
+ const lowered = header.map(value => String(value || '').toLowerCase());
514
+ const buckets = header.slice(1);
515
+ if (grain === 'day') {
516
+ if (lowered.some(column => column.includes('day') || column === 'date') || buckets.some(isDayHeader)) {
517
+ return null;
518
+ }
519
+ return `Expected day/date header for daily report, got ${header.join(', ')}.`;
520
+ }
521
+ if (grain === 'week') {
522
+ if (lowered.some(column => column.includes('week')) || buckets.some(isWeekHeader)) {
523
+ return null;
524
+ }
525
+ return `Expected week header for weekly report, got ${header.join(', ')}.`;
526
+ }
527
+ if (grain === 'month') {
528
+ if (lowered.some(column => column.includes('month')) || buckets.some(isMonthHeader)) {
529
+ return null;
530
+ }
531
+ return `Expected month header for monthly report, got ${header.join(', ')}.`;
532
+ }
533
+ if (grain === 'quarter') {
534
+ if (lowered.some(column => column.includes('quarter')) || buckets.some(isQuarterHeader)) {
535
+ return null;
536
+ }
537
+ return `Expected quarter header for quarterly report, got ${header.join(', ')}.`;
538
+ }
539
+ if (lowered.some(column => column.includes('year')) || buckets.some(isYearHeader)) {
540
+ return null;
541
+ }
542
+ return `Expected year header for yearly report, got ${header.join(', ')}.`;
543
+ }
544
+
545
+ async function sleep(ms: number): Promise<void> {
546
+ // eslint-disable-next-line no-restricted-syntax
547
+ return await new Promise(resolve => setTimeout(resolve, ms));
548
+ }
549
+
550
+ async function findSuperAdminUserId(db: any): Promise<string> {
551
+ const user = await db.collection('users').findOne(
552
+ { 'roles.super_admin': true },
553
+ { projection: { _id: 1, fullname: 1 }, sort: { updatedAt: -1, createdAt: -1 } }
554
+ );
555
+ if (!user?._id) {
556
+ throw new Error('Could not find super admin user in resolveio.users.');
557
+ }
558
+ return String(user._id);
559
+ }
560
+
561
+ async function upsertHarnessUser(db: any, params: {
562
+ id: string;
563
+ username: string;
564
+ fullname: string;
565
+ roles: Record<string, any>;
566
+ other?: Record<string, any>;
567
+ }): Promise<string> {
568
+ const userId = String(params.id || '').trim();
569
+ if (!userId) {
570
+ throw new Error(`Harness user id is required for ${params.username}.`);
571
+ }
572
+ const now = new Date();
573
+ const users = db.collection('users');
574
+ let existing = await users.findOne(
575
+ { username: params.username },
576
+ { projection: { _id: 1 } }
577
+ );
578
+ if (existing?._id && String(existing._id) !== userId) {
579
+ await users.deleteOne({ _id: existing._id });
580
+ existing = null;
581
+ }
582
+ const setPayload = {
583
+ _id: userId,
584
+ username: params.username,
585
+ fullname: params.fullname,
586
+ email: `${params.username}@example.invalid`,
587
+ active: true,
588
+ roles: params.roles,
589
+ other: params.other || {},
590
+ salt: 'ai-e2e',
591
+ hash: 'ai-e2e',
592
+ attempts: 0,
593
+ updatedAt: now
594
+ };
595
+ if (existing?._id && String(existing._id) === userId) {
596
+ await users.updateOne(
597
+ { _id: userId },
598
+ { $set: setPayload }
599
+ );
600
+ return userId;
601
+ }
602
+ await users.updateOne(
603
+ { _id: userId },
604
+ {
605
+ $set: setPayload,
606
+ $setOnInsert: {
607
+ createdAt: now
608
+ }
609
+ },
610
+ { upsert: true }
611
+ );
612
+ return userId;
613
+ }
614
+
615
+ async function ensureHarnessUsers(db: any, superAdminId: string): Promise<Record<E2EUserProfile, string>> {
616
+ const users: Record<E2EUserProfile, string> = {
617
+ super_admin: superAdminId,
618
+ internal_invoice: '',
619
+ internal_no_invoice: '',
620
+ internal_blend_plural: '',
621
+ internal_misc_invoice: '',
622
+ customer_invoice: '',
623
+ customer_no_invoice: ''
624
+ };
625
+ users.internal_invoice = await upsertHarnessUser(db, {
626
+ id: 'ai-e2e-internal-invoice',
627
+ username: 'ai-e2e-internal-invoice',
628
+ fullname: 'AI E2E Internal Invoice',
629
+ roles: {
630
+ super_admin: false,
631
+ groups: [
632
+ {
633
+ name: 'E2E Internal Invoice',
634
+ views: ['/dashboard', '/invoice/list', '/work-order/list']
635
+ }
636
+ ],
637
+ miscs: []
638
+ },
639
+ other: {
640
+ id_client: 'E2E-CLIENT-001'
641
+ }
642
+ });
643
+ users.internal_no_invoice = await upsertHarnessUser(db, {
644
+ id: 'ai-e2e-internal-no-invoice',
645
+ username: 'ai-e2e-internal-no-invoice',
646
+ fullname: 'AI E2E Internal No Invoice',
647
+ roles: {
648
+ super_admin: false,
649
+ groups: [
650
+ {
651
+ name: 'E2E Internal Field',
652
+ views: ['/dashboard', '/work-order/list', '/chemical/list']
653
+ }
654
+ ],
655
+ miscs: []
656
+ },
657
+ other: {
658
+ id_client: 'E2E-CLIENT-001'
659
+ }
660
+ });
661
+ users.internal_blend_plural = await upsertHarnessUser(db, {
662
+ id: 'ai-e2e-internal-blend-plural',
663
+ username: 'ai-e2e-internal-blend-plural',
664
+ fullname: 'AI E2E Internal Blend Plural',
665
+ roles: {
666
+ super_admin: false,
667
+ groups: [
668
+ {
669
+ name: 'E2E Blend Ops',
670
+ views: ['/dashboard', '/chemical-blends/list', '/work-order/list']
671
+ }
672
+ ],
673
+ miscs: []
674
+ },
675
+ other: {
676
+ id_client: 'E2E-CLIENT-001'
677
+ }
678
+ });
679
+ users.internal_misc_invoice = await upsertHarnessUser(db, {
680
+ id: 'ai-e2e-internal-misc-invoice',
681
+ username: 'ai-e2e-internal-misc-invoice',
682
+ fullname: 'AI E2E Internal Misc Invoice',
683
+ roles: {
684
+ super_admin: false,
685
+ groups: [
686
+ {
687
+ name: 'E2E Operations',
688
+ views: ['/dashboard', '/work-order/list']
689
+ }
690
+ ],
691
+ miscs: ['Invoice Dashboard Access']
692
+ },
693
+ other: {
694
+ id_client: 'E2E-CLIENT-001'
695
+ }
696
+ });
697
+ users.customer_invoice = await upsertHarnessUser(db, {
698
+ id: 'ai-e2e-customer-invoice',
699
+ username: 'ai-e2e-customer-invoice',
700
+ fullname: 'AI E2E Customer Invoice',
701
+ roles: {
702
+ super_admin: false,
703
+ groups: [
704
+ {
705
+ name: 'E2E Customer Invoice',
706
+ views: ['/customer-portal/dashboard', '/customer-portal/invoice/list']
707
+ }
708
+ ],
709
+ miscs: []
710
+ },
711
+ other: {
712
+ id_customer: 'E2E-CUST-INV'
713
+ }
714
+ });
715
+ users.customer_no_invoice = await upsertHarnessUser(db, {
716
+ id: 'ai-e2e-customer-no-invoice',
717
+ username: 'ai-e2e-customer-no-invoice',
718
+ fullname: 'AI E2E Customer No Invoice',
719
+ roles: {
720
+ super_admin: false,
721
+ groups: [
722
+ {
723
+ name: 'E2E Customer No Invoice',
724
+ views: ['/customer-portal/dashboard', '/customer-portal/order/list']
725
+ }
726
+ ],
727
+ miscs: []
728
+ },
729
+ other: {
730
+ id_customer: 'E2E-CUST-NO-INV'
731
+ }
732
+ });
733
+ return users;
734
+ }
735
+
736
+ async function pollFinalAssistantMessage(db: any, idConversation: string, timeoutMs = 180000): Promise<any> {
737
+ const started = Date.now();
738
+ while (Date.now() - started < timeoutMs) {
739
+ const message = await db.collection('ai-terminal-messages').findOne(
740
+ { id_conversation: idConversation, role: 'assistant' },
741
+ { sort: { createdAt: -1 } }
742
+ );
743
+ if (message && message?.metadata?.pending !== true) {
744
+ return message;
745
+ }
746
+ await sleep(1200);
747
+ }
748
+ throw new Error(`Timed out waiting for assistant completion for conversation ${idConversation}`);
749
+ }
750
+
751
+ async function runPromptE2E(
752
+ methodManager: any,
753
+ db: any,
754
+ idUser: string,
755
+ testCase: E2ECase
756
+ ): Promise<E2EResult> {
757
+ const managerContext = Object.assign({}, methodManager, Object.getPrototypeOf(methodManager), {
758
+ id_user: idUser,
759
+ user: 'E2E Runner',
760
+ id_ws: 'ai-assistant-openai-e2e'
761
+ });
762
+ const model = String(process.env.AI_ASSISTANT_CODEX_MODEL || process.env.OPENAI_MODEL || '').trim();
763
+ const payload = {
764
+ message: testCase.prompt,
765
+ id_app: 'snf',
766
+ max_history: 0,
767
+ config: {
768
+ ...(model ? { model } : {})
769
+ }
770
+ };
771
+ const started = Date.now();
772
+ const response = await methodManager.callMethod.call(managerContext, 'aiCoderTerminalRunCodex', payload);
773
+ const idConversation = String(response?.conversation?._id || response?.conversation?.id_conversation || '');
774
+ if (!idConversation) {
775
+ throw new Error('No conversation id returned from aiCoderTerminalRunCodex.');
776
+ }
777
+ const finalMessage = await pollFinalAssistantMessage(db, idConversation);
778
+ const content = String(finalMessage?.content || '');
779
+ const loweredContent = content.toLowerCase();
780
+ const metadata = finalMessage?.metadata && typeof finalMessage.metadata === 'object' ? finalMessage.metadata : {};
781
+ const header = findMarkdownTableHeader(content);
782
+ const classificationType = String(metadata?.debug?.requestClassification?.type || '').toLowerCase();
783
+ const dataQuestion = metadata?.debug?.requestClassification?.dataQuestion === true;
784
+
785
+ let error: string | null = null;
786
+ if (testCase.expect.hasTable && !header.length) {
787
+ error = 'Expected markdown table in final response.';
788
+ }
789
+ if (!error && testCase.expect.requestType) {
790
+ if (classificationType !== testCase.expect.requestType) {
791
+ error = `Expected request type ${testCase.expect.requestType}, got ${classificationType || 'none'}.`;
792
+ }
793
+ }
794
+ if (!error && typeof testCase.expect.dataQuestion === 'boolean') {
795
+ if (dataQuestion !== testCase.expect.dataQuestion) {
796
+ error = `Expected dataQuestion=${testCase.expect.dataQuestion}, got ${dataQuestion}.`;
797
+ }
798
+ }
799
+ if (!error && testCase.expect.dataResponse) {
800
+ if (classificationType && classificationType !== 'data' && classificationType !== 'mixed') {
801
+ error = `Expected data/mixed classification, got ${classificationType}.`;
802
+ }
803
+ else if (classificationType && !dataQuestion) {
804
+ error = 'Expected dataQuestion=true in debug classification.';
805
+ }
806
+ }
807
+ if (!error && testCase.expect.directiveType) {
808
+ const directiveType = String(metadata?.debug?.directive?.type || '').toLowerCase();
809
+ if (directiveType !== testCase.expect.directiveType) {
810
+ error = `Expected directive type ${testCase.expect.directiveType}, got ${directiveType || 'none'}.`;
811
+ }
812
+ }
813
+ if (!error && testCase.expect.revenueMetricGrid) {
814
+ error = assertRevenueMetricGrid(header);
815
+ }
816
+ if (!error && testCase.expect.revenueCustomerGrid) {
817
+ error = assertRevenueCustomerGrid(header);
818
+ }
819
+ if (!error && testCase.expect.timeGrain) {
820
+ error = assertTimeGrainGrid(header, testCase.expect.timeGrain);
821
+ }
822
+ if (!error && testCase.expect.fastPathType) {
823
+ const fastPathType = String(metadata?.fast_path || '');
824
+ const allowFastPathMiss = testCase.expect.allowFastPathMiss === true;
825
+ const allowMissingFastPath = allowFastPathMiss && !fastPathType;
826
+ if (fastPathType !== testCase.expect.fastPathType && !allowMissingFastPath) {
827
+ error = `Expected fast_path=${testCase.expect.fastPathType}, got ${fastPathType || 'none'}.`;
828
+ }
829
+ }
830
+ if (!error && typeof testCase.expect.fastPathGeneric === 'boolean') {
831
+ const fastPathType = String(metadata?.fast_path || '');
832
+ if (testCase.expect.allowFastPathMiss === true && !fastPathType) {
833
+ // Optional fast-path checks are skipped when no fast path is present.
834
+ }
835
+ else {
836
+ const fastPathGeneric = metadata?.fast_path_change_generic === true;
837
+ if (fastPathGeneric !== testCase.expect.fastPathGeneric) {
838
+ error = `Expected fast_path_change_generic=${testCase.expect.fastPathGeneric}, got ${fastPathGeneric}.`;
839
+ }
840
+ }
841
+ }
842
+ if (!error && testCase.expect.mustIncludeRoute) {
843
+ if (!content.includes(testCase.expect.mustIncludeRoute)) {
844
+ error = `Expected response to include route ${testCase.expect.mustIncludeRoute}.`;
845
+ }
846
+ }
847
+ if (!error && Array.isArray(testCase.expect.mustIncludeAnyRoute) && testCase.expect.mustIncludeAnyRoute.length) {
848
+ const hasAny = testCase.expect.mustIncludeAnyRoute.some(route => content.includes(route));
849
+ if (!hasAny) {
850
+ error = `Expected response to include at least one route: ${testCase.expect.mustIncludeAnyRoute.join(', ')}.`;
851
+ }
852
+ }
853
+ if (!error && testCase.expect.mustNotIncludeRoute) {
854
+ if (content.includes(testCase.expect.mustNotIncludeRoute)) {
855
+ error = `Expected response to exclude route ${testCase.expect.mustNotIncludeRoute}.`;
856
+ }
857
+ }
858
+ if (!error && Array.isArray(testCase.expect.mustNotIncludeAnyRoute) && testCase.expect.mustNotIncludeAnyRoute.length) {
859
+ const matched = testCase.expect.mustNotIncludeAnyRoute.find(route => content.includes(route));
860
+ if (matched) {
861
+ error = `Expected response to exclude route ${matched}.`;
862
+ }
863
+ }
864
+ if (!error && Array.isArray(testCase.expect.mustContainAnyText) && testCase.expect.mustContainAnyText.length) {
865
+ const hasAny = testCase.expect.mustContainAnyText.some(token => loweredContent.includes(String(token || '').toLowerCase()));
866
+ if (!hasAny) {
867
+ error = `Expected response to include one of: ${testCase.expect.mustContainAnyText.join(', ')}.`;
868
+ }
869
+ }
870
+ if (!error && Array.isArray(testCase.expect.mustNotContainAnyText) && testCase.expect.mustNotContainAnyText.length) {
871
+ const matched = testCase.expect.mustNotContainAnyText.find(token => loweredContent.includes(String(token || '').toLowerCase()));
872
+ if (matched) {
873
+ error = `Expected response to exclude text "${matched}".`;
874
+ }
875
+ }
876
+
877
+ return {
878
+ id: testCase.id,
879
+ prompt: testCase.prompt,
880
+ pass: !error,
881
+ error: error || undefined,
882
+ conversationId: idConversation,
883
+ responsePreview: content.slice(0, 1400),
884
+ classification: {
885
+ type: metadata?.debug?.requestClassification?.type,
886
+ dataQuestion: metadata?.debug?.requestClassification?.dataQuestion,
887
+ source: metadata?.debug?.requestClassification?.source
888
+ },
889
+ fastPath: metadata?.fast_path ? {
890
+ type: metadata?.fast_path,
891
+ reason: metadata?.fast_path_reason,
892
+ generic: metadata?.fast_path_change_generic === true,
893
+ commitCount: Number(metadata?.fast_path_change_commit_count || 0)
894
+ } : undefined,
895
+ userProfile: testCase.userProfile || 'super_admin',
896
+ tableHeader: header.length ? header : undefined
897
+ };
898
+ }
899
+
900
+ async function run(): Promise<void> {
901
+ const repoRoot = process.cwd();
902
+ const snfServerDir = path.resolve(repoRoot, '../snf/snf/server');
903
+ const snfTmpDir = path.join(snfServerDir, 'tmp');
904
+ const snfSrcDir = path.join(snfServerDir, 'src');
905
+ const snfRuntimeDir = fs.existsSync(path.join(snfTmpDir, 'server-app.js')) ? snfTmpDir : snfSrcDir;
906
+ const snfEnvPath = path.join(snfServerDir, '.env');
907
+ const settingsPathInput = String(process.env.AI_ASSISTANT_E2E_SETTINGS_PATH || '').trim();
908
+ const settingsPath = settingsPathInput
909
+ ? (path.isAbsolute(settingsPathInput) ? settingsPathInput : path.resolve(snfServerDir, settingsPathInput))
910
+ : path.join(snfServerDir, 'settings.local.json');
911
+ const clientRoutesPath = path.join(snfRuntimeDir, 'client-routes');
912
+
913
+ if (!fs.existsSync(settingsPath)) {
914
+ throw new Error(`SNF settings not found at ${settingsPath}`);
915
+ }
916
+
917
+ applyEnvFromDotEnv(snfEnvPath);
918
+ process.env.AI_ASSISTANT_WORKSPACE_ROOT = process.env.AI_ASSISTANT_WORKSPACE_ROOT || path.resolve(snfServerDir, '../..');
919
+ const appHeuristicsPath = path.join(snfServerDir, 'src', 'ai', 'assistant-heuristics.json');
920
+ if (!process.env.AI_ASSISTANT_APP_HEURISTICS_FILE && fs.existsSync(appHeuristicsPath)) {
921
+ process.env.AI_ASSISTANT_APP_HEURISTICS_FILE = appHeuristicsPath;
922
+ }
923
+ if (!process.env.AI_ASSISTANT_APP_ID) {
924
+ process.env.AI_ASSISTANT_APP_ID = 'snf';
925
+ }
926
+ if (!process.env.OPENAI_API_KEY) {
927
+ throw new Error('OPENAI_API_KEY is required for true OpenAI end-to-end tests.');
928
+ }
929
+ const codexHome = process.env.CODEX_HOME || path.join(repoRoot, 'tmp', '.codex-openai-e2e');
930
+ const codexSessions = path.join(codexHome, 'sessions');
931
+ const codexZdotdir = process.env.AI_DASHBOARD_CODEX_ZDOTDIR || path.join(repoRoot, 'tmp', '.codex-zdotdir');
932
+ fs.mkdirSync(codexSessions, { recursive: true });
933
+ fs.mkdirSync(codexZdotdir, { recursive: true });
934
+ process.env.CODEX_HOME = codexHome;
935
+ process.env.AI_DASHBOARD_CODEX_ZDOTDIR = codexZdotdir;
936
+
937
+ process.env.IS_WORKERS_ENABLED = 'false';
938
+ process.env.IS_WORKER_INSTANCE = 'false';
939
+ process.env.WORKER_INDEX = '';
940
+ process.env.NODE_APP_INSTANCE = process.env.NODE_APP_INSTANCE || '7';
941
+ process.env.AI_ASSISTANT_WORKER_DEBUG = process.env.AI_ASSISTANT_WORKER_DEBUG || 'false';
942
+ process.env.RESOLVEIO_LOCAL_LOG_DIR = process.env.RESOLVEIO_LOCAL_LOG_DIR || path.join(repoRoot, 'tmp', 'logs');
943
+
944
+ const serverConfig = JSON.parse(fs.readFileSync(settingsPath, 'utf8'));
945
+ const configuredHeuristicsFile = String(serverConfig?.AI_ASSISTANT_APP_HEURISTICS_FILE || '').trim();
946
+ if (configuredHeuristicsFile && !path.isAbsolute(configuredHeuristicsFile)) {
947
+ serverConfig.AI_ASSISTANT_APP_HEURISTICS_FILE = path.resolve(snfServerDir, configuredHeuristicsFile);
948
+ }
949
+ if (!serverConfig?.AI_ASSISTANT_APP_HEURISTICS_FILE && process.env.AI_ASSISTANT_APP_HEURISTICS_FILE) {
950
+ serverConfig.AI_ASSISTANT_APP_HEURISTICS_FILE = process.env.AI_ASSISTANT_APP_HEURISTICS_FILE;
951
+ }
952
+ if (!serverConfig?.AI_ASSISTANT_APP_ID && process.env.AI_ASSISTANT_APP_ID) {
953
+ serverConfig.AI_ASSISTANT_APP_ID = process.env.AI_ASSISTANT_APP_ID;
954
+ }
955
+ const { CLIENT_ROUTES } = require(clientRoutesPath);
956
+ await ResolveIOServer.create(serverConfig, CLIENT_ROUTES(), 'SNF', snfRuntimeDir, false, false);
957
+
958
+ const mainServer = ResolveIOServer.getMainServer();
959
+ const methodManager = mainServer?.getMethodManager();
960
+ if (!methodManager) {
961
+ throw new Error('MethodManager not available after ResolveIOServer initialization.');
962
+ }
963
+ await methodManager.waitUntilReady(120000);
964
+
965
+ const db = ResolveIOServer.getMongoConnection().db(serverConfig.DATABASE || 'resolveio');
966
+ const superAdminId = await findSuperAdminUserId(db);
967
+ const harnessUsers = await ensureHarnessUsers(db, superAdminId);
968
+ const overrideUserId = String(process.env.AI_ASSISTANT_E2E_USER_ID || '').trim();
969
+ const caseFilter = String(process.env.AI_ASSISTANT_E2E_CASES || '')
970
+ .split(',')
971
+ .map(value => value.trim())
972
+ .filter(Boolean);
973
+ const repeatCountRaw = Number(process.env.AI_ASSISTANT_E2E_REPEAT || 1);
974
+ const repeatCount = Number.isFinite(repeatCountRaw)
975
+ ? Math.min(Math.max(Math.round(repeatCountRaw), 1), 500)
976
+ : 1;
977
+ const selectedPrompts = caseFilter.length
978
+ ? E2E_PROMPTS.filter(testCase => caseFilter.includes(testCase.id))
979
+ : E2E_PROMPTS;
980
+ if (!selectedPrompts.length) {
981
+ throw new Error('No matching E2E cases selected via AI_ASSISTANT_E2E_CASES.');
982
+ }
983
+
984
+ const results: E2EResult[] = [];
985
+ for (const testCase of selectedPrompts) {
986
+ for (let attempt = 1; attempt <= repeatCount; attempt += 1) {
987
+ try {
988
+ const profile = testCase.userProfile || 'super_admin';
989
+ const idUser = overrideUserId || harnessUsers[profile] || superAdminId;
990
+ const result = await runPromptE2E(methodManager, db, idUser, testCase);
991
+ const runResult = {
992
+ ...result,
993
+ attempt
994
+ };
995
+ results.push(runResult);
996
+ const status = runResult.pass ? 'PASS' : 'FAIL';
997
+ console.log(`${status} ${testCase.id}#${attempt}/${repeatCount} ${runResult.error ? `error=${runResult.error}` : ''}`.trim());
998
+ }
999
+ catch (error: any) {
1000
+ const message = String(error?.message || error || 'Unknown error');
1001
+ results.push({
1002
+ id: testCase.id,
1003
+ attempt,
1004
+ prompt: testCase.prompt,
1005
+ pass: false,
1006
+ error: message,
1007
+ userProfile: testCase.userProfile || 'super_admin'
1008
+ });
1009
+ console.log(`FAIL ${testCase.id}#${attempt}/${repeatCount} error=${message}`);
1010
+ }
1011
+ }
1012
+ }
1013
+
1014
+ const passCount = results.filter(result => result.pass).length;
1015
+ const failCount = results.length - passCount;
1016
+ const perCase = selectedPrompts.map((testCase) => {
1017
+ const caseResults = results.filter(result => result.id === testCase.id);
1018
+ const passed = caseResults.filter(result => result.pass).length;
1019
+ const failed = caseResults.length - passed;
1020
+ let maxConsecutivePass = 0;
1021
+ let currentConsecutivePass = 0;
1022
+ caseResults.forEach((result) => {
1023
+ if (result.pass) {
1024
+ currentConsecutivePass += 1;
1025
+ if (currentConsecutivePass > maxConsecutivePass) {
1026
+ maxConsecutivePass = currentConsecutivePass;
1027
+ }
1028
+ }
1029
+ else {
1030
+ currentConsecutivePass = 0;
1031
+ }
1032
+ });
1033
+ return {
1034
+ id: testCase.id,
1035
+ runs: caseResults.length,
1036
+ passed,
1037
+ failed,
1038
+ maxConsecutivePass
1039
+ };
1040
+ });
1041
+ const output = {
1042
+ now: new Date().toISOString(),
1043
+ total: results.length,
1044
+ passed: passCount,
1045
+ failed: failCount,
1046
+ repeatCount,
1047
+ perCase,
1048
+ results
1049
+ };
1050
+
1051
+ const outputPath = path.join(repoRoot, 'tmp', 'ai-assistant-openai-e2e-output.json');
1052
+ fs.writeFileSync(outputPath, JSON.stringify(output, null, 2));
1053
+ console.log(`ai assistant openai e2e: ${passCount}/${results.length} passed`);
1054
+ console.log(`wrote ${outputPath}`);
1055
+ process.exit(failCount > 0 ? 1 : 0);
1056
+ }
1057
+
1058
+ run().catch((error) => {
1059
+ console.error(error);
1060
+ process.exit(1);
1061
+ });