@resolveio/server-lib 22.3.56 → 22.3.57

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 +719 -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 +195 -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 +287 -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 -336
  401. package/methods/ai-terminal.js +0 -22782
  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 -696
  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 -186
  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,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
+ });