@resolveio/server-lib 22.3.220 → 22.3.221
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/.nodemon.json +5 -0
- package/.vscode/settings.json +21 -0
- package/AGENTS.md +195 -0
- package/README.md +22 -0
- package/build_package.sh +5 -0
- package/compileDTS.pl +64 -0
- package/docs/ai-assistant-nightly-eval.md +65 -0
- package/docs/ai-assistant-preflight-checklist.md +23 -0
- package/docs/ai-assistant-report-builder-bridge-playbook.md +115 -0
- package/eslint-plugin-custom/index.js +7 -0
- package/eslint-plugin-custom/rules/no-filter-zero-index.js +44 -0
- package/eslint.config.js +103 -0
- package/gulpfile.js +216 -0
- package/methodAndPublicationListGenerator.py +375 -0
- package/mongodbensurers.js +2 -0
- package/mongostop.js +3 -0
- package/package.json +1 -1
- package/scripts/cleanup-bypassed-callmethod-logs.js +616 -0
- package/settings.development.json +25 -0
- package/settings.development.redacted.json +25 -0
- package/src/.env +12 -0
- package/src/ai/assistant-core-heuristics.ts +379 -0
- package/src/ai/resolveio-platform-intelligence-memory-corpus.ts +185 -0
- package/src/ai/resolveio-platform-intelligence-memory.ts +325 -0
- package/{ai/resolveio-platform-intelligence-types.d.ts → src/ai/resolveio-platform-intelligence-types.ts} +20 -15
- package/src/ai/resolveio-platform-intelligence.ts +462 -0
- package/src/client-server-app.ts +12 -0
- package/src/collections/ai-run.collection.ts +117 -0
- package/src/collections/ai-terminal-conversation.collection.ts +91 -0
- package/src/collections/ai-terminal-issue-report.collection.ts +99 -0
- package/src/collections/ai-terminal-message.collection.ts +77 -0
- package/src/collections/app-setting.collection.ts +104 -0
- package/src/collections/app-status.collection.ts +58 -0
- package/src/collections/communication-metric.collection.ts +84 -0
- package/src/collections/counter.collection.ts +56 -0
- package/src/collections/cron-job-history.collection.ts +94 -0
- package/src/collections/cron-job.collection.ts +92 -0
- package/src/collections/customer-notification.collection.ts +131 -0
- package/src/collections/customer-portal-password.collection.ts +76 -0
- package/src/collections/email-history.collection.ts +134 -0
- package/src/collections/email-verified.collection.ts +62 -0
- package/src/collections/file.collection.ts +74 -0
- package/src/collections/flag-update.collection.ts +57 -0
- package/src/collections/flag.collection.ts +57 -0
- package/src/collections/log-method-latency.collection.ts +77 -0
- package/src/collections/log-subscription.collection.ts +80 -0
- package/src/collections/log.collection.ts +93 -0
- package/src/collections/logged-in-users.collection.ts +67 -0
- package/src/collections/monitor-cpu.collection.ts +65 -0
- package/src/collections/monitor-function.collection.ts +74 -0
- package/src/collections/monitor-memory.collection.ts +77 -0
- package/src/collections/monitor-mongo.collection.ts +71 -0
- package/src/collections/notification.collection.ts +57 -0
- package/src/collections/openai-usage-ledger.collection.ts +131 -0
- package/src/collections/report-builder-dashboard-builder.collection.ts +109 -0
- package/src/collections/report-builder-library.collection.ts +89 -0
- package/src/collections/report-builder-report.collection.ts +184 -0
- package/src/collections/user-group.collection.ts +89 -0
- package/src/collections/user-guide.collection.ts +57 -0
- package/src/collections/user.collection.ts +181 -0
- package/src/cron/cron.ts +117 -0
- package/src/fixtures/cron-jobs.ts +95 -0
- package/src/fixtures/init.ts +35 -0
- package/src/http/auth.ts +818 -0
- package/src/http/health.ts +7 -0
- package/src/http/home.ts +90 -0
- package/src/http/slow-query-publication.ts +49 -0
- package/src/index.ts +1 -0
- package/src/managers/ai-assistant-codex-manager.manager.ts +1131 -0
- package/src/managers/ai-run-evidence.manager.ts +264 -0
- package/src/managers/communication-metric.manager.ts +82 -0
- package/src/managers/cron.manager.ts +333 -0
- package/src/managers/customer-notification-content.manager.ts +236 -0
- package/src/managers/diagnostic-manager-bootstrap.ts +165 -0
- package/src/managers/error-auto-fix.manager.ts +2767 -0
- package/src/managers/local-log.manager.ts +113 -0
- package/src/managers/method.manager.ts +1857 -0
- package/src/managers/mongo.manager.ts +4575 -0
- package/src/managers/monitor.manager.ts +507 -0
- package/src/managers/openai-usage-ledger.manager.ts +112 -0
- package/src/managers/slow-query-verifier.manager.ts +3590 -0
- package/src/managers/slow-query.manager.ts +519 -0
- package/src/managers/subscription.manager.ts +3128 -0
- package/src/managers/websocket.manager.ts +746 -0
- package/src/managers/worker-dispatcher.manager.ts +1360 -0
- package/src/managers/worker-server.manager.ts +536 -0
- package/src/methods/accounts.ts +532 -0
- package/src/methods/ai-terminal.ts +29070 -0
- package/src/methods/app-settings.ts +114 -0
- package/src/methods/aws.ts +649 -0
- package/src/methods/collections.ts +641 -0
- package/src/methods/counters.ts +69 -0
- package/src/methods/cron-jobs.ts +2614 -0
- package/src/methods/customer-notifications.ts +458 -0
- package/src/methods/diagnostics.ts +616 -0
- package/src/methods/flag-updates.ts +7 -0
- package/src/methods/flags.ts +7 -0
- package/src/methods/logs.ts +657 -0
- package/src/methods/mongo-explorer.ts +1880 -0
- package/src/methods/monitor.ts +540 -0
- package/src/methods/pdf.ts +1236 -0
- package/src/methods/publications.ts +129 -0
- package/src/methods/report-builder.ts +3300 -0
- package/src/methods/support.ts +335 -0
- package/src/models/ai-run.model.ts +27 -0
- package/src/models/ai-terminal-conversation.model.ts +19 -0
- package/src/models/ai-terminal-issue-report.model.ts +21 -0
- package/src/models/ai-terminal-message.model.ts +24 -0
- package/src/models/app-setting.model.ts +17 -0
- package/{models/app-status.model.d.ts → src/models/app-status.model.ts} +3 -2
- package/{models/billing-logged-in-users.model.d.ts → src/models/billing-logged-in-users.model.ts} +5 -4
- package/src/models/collection-document.model.ts +24 -0
- package/src/models/communication-metric.model.ts +23 -0
- package/{models/counter.model.d.ts → src/models/counter.model.ts} +4 -3
- package/src/models/cron-job-history.model.ts +16 -0
- package/src/models/cron-job.model.ts +15 -0
- package/src/models/customer-notification.model.ts +28 -0
- package/src/models/customer-portal-password.model.ts +12 -0
- package/src/models/dialog.model.ts +25 -0
- package/{models/email-history.model.js → src/models/email-history.model.ts} +36 -4
- package/{models/email-verified.model.d.ts → src/models/email-verified.model.ts} +6 -5
- package/{models/file.model.d.ts → src/models/file.model.ts} +8 -7
- package/{models/flag-update.model.d.ts → src/models/flag-update.model.ts} +4 -3
- package/{models/flag.model.d.ts → src/models/flag.model.ts} +4 -3
- package/src/models/log-method-latency.model.ts +11 -0
- package/{models/log-subscription.model.d.ts → src/models/log-subscription.model.ts} +11 -9
- package/src/models/log.model.ts +19 -0
- package/{models/logged-in-users.model.d.ts → src/models/logged-in-users.model.ts} +6 -5
- package/{models/method-response.model.d.ts → src/models/method-response.model.ts} +7 -6
- package/src/models/method.model.ts +25 -0
- package/{models/monitor-cpu.model.d.ts → src/models/monitor-cpu.model.ts} +9 -7
- package/src/models/monitor-function.model.ts +16 -0
- package/src/models/monitor-memory.model.ts +17 -0
- package/src/models/monitor-mongo.model.ts +15 -0
- package/{models/notification.model.d.ts → src/models/notification.model.ts} +6 -4
- package/src/models/openai-usage-ledger.model.ts +56 -0
- package/src/models/pagination.model.ts +35 -0
- package/src/models/permission.model.ts +14 -0
- package/src/models/report-builder-dashboard-builder.model.ts +29 -0
- package/src/models/report-builder-library.model.ts +20 -0
- package/src/models/report-builder-report.model.ts +136 -0
- package/src/models/report-builder.model.ts +68 -0
- package/src/models/select-data-label.model.ts +9 -0
- package/src/models/server-message.model.ts +31 -0
- package/src/models/slow-query-report.model.ts +23 -0
- package/src/models/subscription.model.ts +73 -0
- package/src/models/support-ticket.model.ts +104 -0
- package/src/models/user-group.model.ts +24 -0
- package/{models/user-guide.model.d.ts → src/models/user-guide.model.ts} +5 -4
- package/src/models/user.model.ts +96 -0
- package/src/private/images/ResolveIO.png +0 -0
- package/src/publications/ai-terminal.ts +73 -0
- package/src/publications/app-settings.ts +25 -0
- package/src/publications/app-status.ts +13 -0
- package/src/publications/cron-jobs.ts +40 -0
- package/src/publications/customer-notifications.ts +101 -0
- package/src/publications/files.ts +33 -0
- package/src/publications/flags-update.ts +19 -0
- package/src/publications/flags.ts +19 -0
- package/src/publications/logs.ts +163 -0
- package/src/publications/notifications.ts +13 -0
- package/src/publications/report-builder-dashboard-builders.ts +39 -0
- package/src/publications/report-builder-libraries.ts +41 -0
- package/src/publications/report-builder-reports.ts +47 -0
- package/src/publications/super-admin.ts +13 -0
- package/src/publications/user-groups.ts +12 -0
- package/src/publications/user-guides.ts +12 -0
- package/src/resolveio-server-app.ts +617 -0
- package/src/server-app.ts +3354 -0
- package/src/services/codex-client.ts +1231 -0
- package/src/services/openai-client.ts +265 -0
- package/src/types/error-report.ts +26 -0
- package/src/types/js-tiktoken.d.ts +11 -0
- package/src/types/slow-query-report.ts +28 -0
- package/src/util/ai-qa-policy.ts +925 -0
- package/src/util/ai-run-evidence-adapters.ts +8347 -0
- package/src/util/ai-run-evidence-dashboard.ts +323 -0
- package/src/util/ai-run-evidence-eval.ts +1057 -0
- package/src/util/ai-run-evidence.ts +1430 -0
- package/src/util/ai-runner-artifacts.ts +586 -0
- package/src/util/ai-runner-manager-autopilot.ts +961 -0
- package/src/util/ai-runner-manager-policy.ts +5011 -0
- package/src/util/ai-runner-qa-auth.ts +838 -0
- package/src/util/ai-runner-qa-tools.ts +3536 -0
- package/src/util/aicoder-runner-v6.ts +3121 -0
- package/src/util/common.ts +649 -0
- package/src/util/customer-portal-password.ts +183 -0
- package/src/util/error-reporter.ts +332 -0
- package/src/util/error-tracking.ts +79 -0
- package/src/util/openai-usage-cost.ts +114 -0
- package/src/util/report-builder-unwinds.ts +180 -0
- package/src/util/runner-process-janitor.ts +219 -0
- package/src/util/schema-report-builder.ts +448 -0
- package/src/util/slow-query-reporter.ts +216 -0
- package/src/util/subscription-dependency-context.ts +1096 -0
- package/src/util/support-runner-v5.ts +10040 -0
- package/src/util/tokenizer.ts +38 -0
- package/src/workers/codex-runner.worker.ts +142 -0
- package/start_server.sh +5 -0
- package/tests/ai-assistant-corpus-build.ts +484 -0
- package/tests/ai-assistant-corpus-replay-e2e.ts +774 -0
- package/tests/ai-assistant-data-parity-e2e.ts +1989 -0
- package/tests/ai-assistant-eval-triage.ts +831 -0
- package/tests/ai-assistant-openai-e2e.ts +1061 -0
- package/tests/ai-assistant-openai-git-e2e.ts +155 -0
- package/tests/ai-assistant-preflight-matrix.ts +215 -0
- package/tests/ai-assistant-routing-eval.test.ts +585 -0
- package/tests/ai-assistant-snf-live-eval.ts +975 -0
- package/tests/ai-assistant-utils.test.ts +4834 -0
- package/tests/ai-manager-autopilot-snapshot.test.ts +193 -0
- package/tests/ai-manager-recovery-checkpoint.test.ts +1383 -0
- package/tests/ai-run-eval.test.ts +132 -0
- package/tests/ai-run-evidence.test.ts +3773 -0
- package/tests/ai-runner-contract.test.ts +515 -0
- package/tests/aicoder-runner-v6.test.ts +822 -0
- package/tests/error-reporter.test.ts +145 -0
- package/tests/method-publication-generator.test.ts +46 -0
- package/tests/report-builder-linking.test.ts +79 -0
- package/tests/resolveio-platform-intelligence.test.ts +352 -0
- package/tests/server-app-cron-owner.test.ts +127 -0
- package/tests/subscription-connect-race.test.ts +158 -0
- package/tests/subscription-dependency-context.test.ts +324 -0
- package/tests/subscription-manager-collection-tracking.test.ts +86 -0
- package/tests/subscription-manager-invalidation.test.ts +86 -0
- package/tests/support-runner-v5.test.ts +3201 -0
- package/tsconfig.json +34 -0
- package/ai/assistant-core-heuristics.d.ts +0 -11
- package/ai/assistant-core-heuristics.js +0 -356
- package/ai/assistant-core-heuristics.js.map +0 -1
- package/ai/resolveio-platform-intelligence-memory-corpus.d.ts +0 -3
- package/ai/resolveio-platform-intelligence-memory-corpus.js +0 -214
- package/ai/resolveio-platform-intelligence-memory-corpus.js.map +0 -1
- package/ai/resolveio-platform-intelligence-memory.d.ts +0 -20
- package/ai/resolveio-platform-intelligence-memory.js +0 -341
- package/ai/resolveio-platform-intelligence-memory.js.map +0 -1
- package/ai/resolveio-platform-intelligence-types.js +0 -4
- package/ai/resolveio-platform-intelligence-types.js.map +0 -1
- package/ai/resolveio-platform-intelligence.d.ts +0 -6
- package/ai/resolveio-platform-intelligence.js +0 -463
- package/ai/resolveio-platform-intelligence.js.map +0 -1
- package/client-server-app.d.ts +0 -1
- package/client-server-app.js +0 -68
- package/client-server-app.js.map +0 -1
- package/collections/ai-run.collection.d.ts +0 -3
- package/collections/ai-run.collection.js +0 -170
- package/collections/ai-run.collection.js.map +0 -1
- package/collections/ai-terminal-conversation.collection.d.ts +0 -2
- package/collections/ai-terminal-conversation.collection.js +0 -140
- package/collections/ai-terminal-conversation.collection.js.map +0 -1
- package/collections/ai-terminal-issue-report.collection.d.ts +0 -2
- package/collections/ai-terminal-issue-report.collection.js +0 -148
- package/collections/ai-terminal-issue-report.collection.js.map +0 -1
- package/collections/ai-terminal-message.collection.d.ts +0 -2
- package/collections/ai-terminal-message.collection.js +0 -121
- package/collections/ai-terminal-message.collection.js.map +0 -1
- package/collections/app-setting.collection.d.ts +0 -3
- package/collections/app-setting.collection.js +0 -103
- package/collections/app-setting.collection.js.map +0 -1
- package/collections/app-status.collection.d.ts +0 -3
- package/collections/app-status.collection.js +0 -57
- package/collections/app-status.collection.js.map +0 -1
- package/collections/communication-metric.collection.d.ts +0 -2
- package/collections/communication-metric.collection.js +0 -133
- package/collections/communication-metric.collection.js.map +0 -1
- package/collections/counter.collection.d.ts +0 -3
- package/collections/counter.collection.js +0 -56
- package/collections/counter.collection.js.map +0 -1
- package/collections/cron-job-history.collection.d.ts +0 -3
- package/collections/cron-job-history.collection.js +0 -137
- package/collections/cron-job-history.collection.js.map +0 -1
- package/collections/cron-job.collection.d.ts +0 -3
- package/collections/cron-job.collection.js +0 -92
- package/collections/cron-job.collection.js.map +0 -1
- package/collections/customer-notification.collection.d.ts +0 -3
- package/collections/customer-notification.collection.js +0 -130
- package/collections/customer-notification.collection.js.map +0 -1
- package/collections/customer-portal-password.collection.d.ts +0 -3
- package/collections/customer-portal-password.collection.js +0 -75
- package/collections/customer-portal-password.collection.js.map +0 -1
- package/collections/email-history.collection.d.ts +0 -3
- package/collections/email-history.collection.js +0 -134
- package/collections/email-history.collection.js.map +0 -1
- package/collections/email-verified.collection.d.ts +0 -3
- package/collections/email-verified.collection.js +0 -62
- package/collections/email-verified.collection.js.map +0 -1
- package/collections/file.collection.d.ts +0 -3
- package/collections/file.collection.js +0 -74
- package/collections/file.collection.js.map +0 -1
- package/collections/flag-update.collection.d.ts +0 -3
- package/collections/flag-update.collection.js +0 -57
- package/collections/flag-update.collection.js.map +0 -1
- package/collections/flag.collection.d.ts +0 -3
- package/collections/flag.collection.js +0 -57
- package/collections/flag.collection.js.map +0 -1
- package/collections/log-method-latency.collection.d.ts +0 -3
- package/collections/log-method-latency.collection.js +0 -77
- package/collections/log-method-latency.collection.js.map +0 -1
- package/collections/log-subscription.collection.d.ts +0 -3
- package/collections/log-subscription.collection.js +0 -80
- package/collections/log-subscription.collection.js.map +0 -1
- package/collections/log.collection.d.ts +0 -3
- package/collections/log.collection.js +0 -93
- package/collections/log.collection.js.map +0 -1
- package/collections/logged-in-users.collection.d.ts +0 -3
- package/collections/logged-in-users.collection.js +0 -67
- package/collections/logged-in-users.collection.js.map +0 -1
- package/collections/monitor-cpu.collection.d.ts +0 -3
- package/collections/monitor-cpu.collection.js +0 -65
- package/collections/monitor-cpu.collection.js.map +0 -1
- package/collections/monitor-function.collection.d.ts +0 -3
- package/collections/monitor-function.collection.js +0 -74
- package/collections/monitor-function.collection.js.map +0 -1
- package/collections/monitor-memory.collection.d.ts +0 -3
- package/collections/monitor-memory.collection.js +0 -77
- package/collections/monitor-memory.collection.js.map +0 -1
- package/collections/monitor-mongo.collection.d.ts +0 -3
- package/collections/monitor-mongo.collection.js +0 -71
- package/collections/monitor-mongo.collection.js.map +0 -1
- package/collections/notification.collection.d.ts +0 -3
- package/collections/notification.collection.js +0 -57
- package/collections/notification.collection.js.map +0 -1
- package/collections/openai-usage-ledger.collection.d.ts +0 -2
- package/collections/openai-usage-ledger.collection.js +0 -188
- package/collections/openai-usage-ledger.collection.js.map +0 -1
- package/collections/report-builder-dashboard-builder.collection.d.ts +0 -3
- package/collections/report-builder-dashboard-builder.collection.js +0 -109
- package/collections/report-builder-dashboard-builder.collection.js.map +0 -1
- package/collections/report-builder-library.collection.d.ts +0 -3
- package/collections/report-builder-library.collection.js +0 -87
- package/collections/report-builder-library.collection.js.map +0 -1
- package/collections/report-builder-report.collection.d.ts +0 -4
- package/collections/report-builder-report.collection.js +0 -184
- package/collections/report-builder-report.collection.js.map +0 -1
- package/collections/user-group.collection.d.ts +0 -4
- package/collections/user-group.collection.js +0 -89
- package/collections/user-group.collection.js.map +0 -1
- package/collections/user-guide.collection.d.ts +0 -3
- package/collections/user-guide.collection.js +0 -57
- package/collections/user-guide.collection.js.map +0 -1
- package/collections/user.collection.d.ts +0 -4
- package/collections/user.collection.js +0 -180
- package/collections/user.collection.js.map +0 -1
- package/cron/cron.d.ts +0 -14
- package/cron/cron.js +0 -216
- package/cron/cron.js.map +0 -1
- package/fixtures/cron-jobs.d.ts +0 -1
- package/fixtures/cron-jobs.js +0 -150
- package/fixtures/cron-jobs.js.map +0 -1
- package/fixtures/init.d.ts +0 -1
- package/fixtures/init.js +0 -91
- package/fixtures/init.js.map +0 -1
- package/http/auth.d.ts +0 -2
- package/http/auth.js +0 -951
- package/http/auth.js.map +0 -1
- package/http/health.d.ts +0 -1
- package/http/health.js +0 -11
- package/http/health.js.map +0 -1
- package/http/home.d.ts +0 -1
- package/http/home.js +0 -134
- package/http/home.js.map +0 -1
- package/http/slow-query-publication.d.ts +0 -2
- package/http/slow-query-publication.js +0 -99
- package/http/slow-query-publication.js.map +0 -1
- package/index.d.ts +0 -1
- package/index.js +0 -19
- package/index.js.map +0 -1
- package/managers/ai-assistant-codex-manager.manager.d.ts +0 -67
- package/managers/ai-assistant-codex-manager.manager.js +0 -1113
- package/managers/ai-assistant-codex-manager.manager.js.map +0 -1
- package/managers/ai-run-evidence.manager.d.ts +0 -36
- package/managers/ai-run-evidence.manager.js +0 -377
- package/managers/ai-run-evidence.manager.js.map +0 -1
- package/managers/communication-metric.manager.d.ts +0 -16
- package/managers/communication-metric.manager.js +0 -134
- package/managers/communication-metric.manager.js.map +0 -1
- package/managers/cron.manager.d.ts +0 -20
- package/managers/cron.manager.js +0 -534
- package/managers/cron.manager.js.map +0 -1
- package/managers/customer-notification-content.manager.d.ts +0 -55
- package/managers/customer-notification-content.manager.js +0 -158
- package/managers/customer-notification-content.manager.js.map +0 -1
- package/managers/diagnostic-manager-bootstrap.d.ts +0 -9
- package/managers/diagnostic-manager-bootstrap.js +0 -260
- package/managers/diagnostic-manager-bootstrap.js.map +0 -1
- package/managers/error-auto-fix.manager.d.ts +0 -149
- package/managers/error-auto-fix.manager.js +0 -3064
- package/managers/error-auto-fix.manager.js.map +0 -1
- package/managers/local-log.manager.d.ts +0 -18
- package/managers/local-log.manager.js +0 -88
- package/managers/local-log.manager.js.map +0 -1
- package/managers/method.manager.d.ts +0 -84
- package/managers/method.manager.js +0 -1964
- package/managers/method.manager.js.map +0 -1
- package/managers/mongo.manager.d.ts +0 -224
- package/managers/mongo.manager.js +0 -5000
- package/managers/mongo.manager.js.map +0 -1
- package/managers/monitor.manager.d.ts +0 -70
- package/managers/monitor.manager.js +0 -550
- package/managers/monitor.manager.js.map +0 -1
- package/managers/openai-usage-ledger.manager.d.ts +0 -30
- package/managers/openai-usage-ledger.manager.js +0 -142
- package/managers/openai-usage-ledger.manager.js.map +0 -1
- package/managers/slow-query-verifier.manager.d.ts +0 -144
- package/managers/slow-query-verifier.manager.js +0 -3857
- package/managers/slow-query-verifier.manager.js.map +0 -1
- package/managers/slow-query.manager.d.ts +0 -28
- package/managers/slow-query.manager.js +0 -468
- package/managers/slow-query.manager.js.map +0 -1
- package/managers/subscription.manager.d.ts +0 -169
- package/managers/subscription.manager.js +0 -3434
- package/managers/subscription.manager.js.map +0 -1
- package/managers/websocket.manager.d.ts +0 -73
- package/managers/websocket.manager.js +0 -673
- package/managers/websocket.manager.js.map +0 -1
- package/managers/worker-dispatcher.manager.d.ts +0 -120
- package/managers/worker-dispatcher.manager.js +0 -1266
- package/managers/worker-dispatcher.manager.js.map +0 -1
- package/managers/worker-server.manager.d.ts +0 -35
- package/managers/worker-server.manager.js +0 -582
- package/managers/worker-server.manager.js.map +0 -1
- package/methods/accounts.d.ts +0 -2
- package/methods/accounts.js +0 -624
- package/methods/accounts.js.map +0 -1
- package/methods/ai-terminal.d.ts +0 -458
- package/methods/ai-terminal.js +0 -27991
- package/methods/ai-terminal.js.map +0 -1
- package/methods/app-settings.d.ts +0 -2
- package/methods/app-settings.js +0 -169
- package/methods/app-settings.js.map +0 -1
- package/methods/aws.d.ts +0 -2
- package/methods/aws.js +0 -877
- package/methods/aws.js.map +0 -1
- package/methods/collections.d.ts +0 -2
- package/methods/collections.js +0 -719
- package/methods/collections.js.map +0 -1
- package/methods/counters.d.ts +0 -2
- package/methods/counters.js +0 -113
- package/methods/counters.js.map +0 -1
- package/methods/cron-jobs.d.ts +0 -2
- package/methods/cron-jobs.js +0 -2475
- package/methods/cron-jobs.js.map +0 -1
- package/methods/customer-notifications.d.ts +0 -2
- package/methods/customer-notifications.js +0 -528
- package/methods/customer-notifications.js.map +0 -1
- package/methods/diagnostics.d.ts +0 -2
- package/methods/diagnostics.js +0 -703
- package/methods/diagnostics.js.map +0 -1
- package/methods/flag-updates.d.ts +0 -2
- package/methods/flag-updates.js +0 -8
- package/methods/flag-updates.js.map +0 -1
- package/methods/flags.d.ts +0 -2
- package/methods/flags.js +0 -8
- package/methods/flags.js.map +0 -1
- package/methods/logs.d.ts +0 -2
- package/methods/logs.js +0 -751
- package/methods/logs.js.map +0 -1
- package/methods/mongo-explorer.d.ts +0 -2
- package/methods/mongo-explorer.js +0 -1808
- package/methods/mongo-explorer.js.map +0 -1
- package/methods/monitor.d.ts +0 -2
- package/methods/monitor.js +0 -543
- package/methods/monitor.js.map +0 -1
- package/methods/pdf.d.ts +0 -2
- package/methods/pdf.js +0 -1216
- package/methods/pdf.js.map +0 -1
- package/methods/publications.d.ts +0 -1
- package/methods/publications.js +0 -183
- package/methods/publications.js.map +0 -1
- package/methods/report-builder.d.ts +0 -2
- package/methods/report-builder.js +0 -3094
- package/methods/report-builder.js.map +0 -1
- package/methods/support.d.ts +0 -2
- package/methods/support.js +0 -430
- package/methods/support.js.map +0 -1
- package/models/ai-run.model.d.ts +0 -19
- package/models/ai-run.model.js +0 -4
- package/models/ai-run.model.js.map +0 -1
- package/models/ai-terminal-conversation.model.d.ts +0 -17
- package/models/ai-terminal-conversation.model.js +0 -4
- package/models/ai-terminal-conversation.model.js.map +0 -1
- package/models/ai-terminal-issue-report.model.d.ts +0 -19
- package/models/ai-terminal-issue-report.model.js +0 -4
- package/models/ai-terminal-issue-report.model.js.map +0 -1
- package/models/ai-terminal-message.model.d.ts +0 -22
- package/models/ai-terminal-message.model.js +0 -4
- package/models/ai-terminal-message.model.js.map +0 -1
- package/models/app-setting.model.d.ts +0 -16
- package/models/app-setting.model.js +0 -4
- package/models/app-setting.model.js.map +0 -1
- package/models/app-status.model.js +0 -4
- package/models/app-status.model.js.map +0 -1
- package/models/billing-logged-in-users.model.js +0 -4
- package/models/billing-logged-in-users.model.js.map +0 -1
- package/models/collection-document.model.d.ts +0 -21
- package/models/collection-document.model.js +0 -4
- package/models/collection-document.model.js.map +0 -1
- package/models/communication-metric.model.d.ts +0 -20
- package/models/communication-metric.model.js +0 -4
- package/models/communication-metric.model.js.map +0 -1
- package/models/counter.model.js +0 -4
- package/models/counter.model.js.map +0 -1
- package/models/cron-job-history.model.d.ts +0 -15
- package/models/cron-job-history.model.js +0 -4
- package/models/cron-job-history.model.js.map +0 -1
- package/models/cron-job.model.d.ts +0 -14
- package/models/cron-job.model.js +0 -4
- package/models/cron-job.model.js.map +0 -1
- package/models/customer-notification.model.d.ts +0 -26
- package/models/customer-notification.model.js +0 -4
- package/models/customer-notification.model.js.map +0 -1
- package/models/customer-portal-password.model.d.ts +0 -11
- package/models/customer-portal-password.model.js +0 -4
- package/models/customer-portal-password.model.js.map +0 -1
- package/models/dialog.model.d.ts +0 -23
- package/models/dialog.model.js +0 -4
- package/models/dialog.model.js.map +0 -1
- package/models/email-history.model.d.ts +0 -32
- package/models/email-history.model.js.map +0 -1
- package/models/email-verified.model.js +0 -4
- package/models/email-verified.model.js.map +0 -1
- package/models/file.model.js +0 -4
- package/models/file.model.js.map +0 -1
- package/models/flag-update.model.js +0 -4
- package/models/flag-update.model.js.map +0 -1
- package/models/flag.model.js +0 -4
- package/models/flag.model.js.map +0 -1
- package/models/log-method-latency.model.d.ts +0 -10
- package/models/log-method-latency.model.js +0 -4
- package/models/log-method-latency.model.js.map +0 -1
- package/models/log-subscription.model.js +0 -4
- package/models/log-subscription.model.js.map +0 -1
- package/models/log.model.d.ts +0 -17
- package/models/log.model.js +0 -4
- package/models/log.model.js.map +0 -1
- package/models/logged-in-users.model.js +0 -4
- package/models/logged-in-users.model.js.map +0 -1
- package/models/method-response.model.js +0 -4
- package/models/method-response.model.js.map +0 -1
- package/models/method.model.d.ts +0 -26
- package/models/method.model.js +0 -4
- package/models/method.model.js.map +0 -1
- package/models/monitor-cpu.model.js +0 -4
- package/models/monitor-cpu.model.js.map +0 -1
- package/models/monitor-function.model.d.ts +0 -14
- package/models/monitor-function.model.js +0 -4
- package/models/monitor-function.model.js.map +0 -1
- package/models/monitor-memory.model.d.ts +0 -15
- package/models/monitor-memory.model.js +0 -4
- package/models/monitor-memory.model.js.map +0 -1
- package/models/monitor-mongo.model.d.ts +0 -13
- package/models/monitor-mongo.model.js +0 -4
- package/models/monitor-mongo.model.js.map +0 -1
- package/models/notification.model.js +0 -4
- package/models/notification.model.js.map +0 -1
- package/models/openai-usage-ledger.model.d.ts +0 -30
- package/models/openai-usage-ledger.model.js +0 -4
- package/models/openai-usage-ledger.model.js.map +0 -1
- package/models/pagination.model.d.ts +0 -11
- package/models/pagination.model.js +0 -28
- package/models/pagination.model.js.map +0 -1
- package/models/permission.model.d.ts +0 -12
- package/models/permission.model.js +0 -4
- package/models/permission.model.js.map +0 -1
- package/models/report-builder-dashboard-builder.model.d.ts +0 -25
- package/models/report-builder-dashboard-builder.model.js +0 -4
- package/models/report-builder-dashboard-builder.model.js.map +0 -1
- package/models/report-builder-library.model.d.ts +0 -17
- package/models/report-builder-library.model.js +0 -4
- package/models/report-builder-library.model.js.map +0 -1
- package/models/report-builder-report.model.d.ts +0 -121
- package/models/report-builder-report.model.js +0 -4
- package/models/report-builder-report.model.js.map +0 -1
- package/models/report-builder.model.d.ts +0 -61
- package/models/report-builder.model.js +0 -4
- package/models/report-builder.model.js.map +0 -1
- package/models/select-data-label.model.d.ts +0 -9
- package/models/select-data-label.model.js +0 -4
- package/models/select-data-label.model.js.map +0 -1
- package/models/server-message.model.d.ts +0 -32
- package/models/server-message.model.js +0 -4
- package/models/server-message.model.js.map +0 -1
- package/models/slow-query-report.model.d.ts +0 -23
- package/models/slow-query-report.model.js +0 -4
- package/models/slow-query-report.model.js.map +0 -1
- package/models/subscription.model.d.ts +0 -31
- package/models/subscription.model.js +0 -4
- package/models/subscription.model.js.map +0 -1
- package/models/support-ticket.model.d.ts +0 -87
- package/models/support-ticket.model.js +0 -4
- package/models/support-ticket.model.js.map +0 -1
- package/models/user-group.model.d.ts +0 -20
- package/models/user-group.model.js +0 -4
- package/models/user-group.model.js.map +0 -1
- package/models/user-guide.model.js +0 -4
- package/models/user-guide.model.js.map +0 -1
- package/models/user.model.d.ts +0 -84
- package/models/user.model.js +0 -4
- package/models/user.model.js.map +0 -1
- package/private/images/ResolveIO.png +0 -0
- package/public_api.js +0 -127
- package/public_api.js.map +0 -1
- package/publications/ai-terminal.d.ts +0 -1
- package/publications/ai-terminal.js +0 -122
- package/publications/ai-terminal.js.map +0 -1
- package/publications/app-settings.d.ts +0 -2
- package/publications/app-settings.js +0 -28
- package/publications/app-settings.js.map +0 -1
- package/publications/app-status.d.ts +0 -2
- package/publications/app-status.js +0 -16
- package/publications/app-status.js.map +0 -1
- package/publications/cron-jobs.d.ts +0 -2
- package/publications/cron-jobs.js +0 -88
- package/publications/cron-jobs.js.map +0 -1
- package/publications/customer-notifications.d.ts +0 -2
- package/publications/customer-notifications.js +0 -161
- package/publications/customer-notifications.js.map +0 -1
- package/publications/files.d.ts +0 -2
- package/publications/files.js +0 -36
- package/publications/files.js.map +0 -1
- package/publications/flags-update.d.ts +0 -2
- package/publications/flags-update.js +0 -22
- package/publications/flags-update.js.map +0 -1
- package/publications/flags.d.ts +0 -2
- package/publications/flags.js +0 -22
- package/publications/flags.js.map +0 -1
- package/publications/logs.d.ts +0 -2
- package/publications/logs.js +0 -164
- package/publications/logs.js.map +0 -1
- package/publications/notifications.d.ts +0 -2
- package/publications/notifications.js +0 -16
- package/publications/notifications.js.map +0 -1
- package/publications/report-builder-dashboard-builders.d.ts +0 -2
- package/publications/report-builder-dashboard-builders.js +0 -42
- package/publications/report-builder-dashboard-builders.js.map +0 -1
- package/publications/report-builder-libraries.d.ts +0 -2
- package/publications/report-builder-libraries.js +0 -90
- package/publications/report-builder-libraries.js.map +0 -1
- package/publications/report-builder-reports.d.ts +0 -2
- package/publications/report-builder-reports.js +0 -50
- package/publications/report-builder-reports.js.map +0 -1
- package/publications/super-admin.d.ts +0 -2
- package/publications/super-admin.js +0 -16
- package/publications/super-admin.js.map +0 -1
- package/publications/user-groups.d.ts +0 -1
- package/publications/user-groups.js +0 -16
- package/publications/user-groups.js.map +0 -1
- package/publications/user-guides.d.ts +0 -1
- package/publications/user-guides.js +0 -16
- package/publications/user-guides.js.map +0 -1
- package/resolveio-server-app.d.ts +0 -70
- package/resolveio-server-app.js +0 -801
- package/resolveio-server-app.js.map +0 -1
- package/server-app.d.ts +0 -228
- package/server-app.js +0 -3566
- package/server-app.js.map +0 -1
- package/services/codex-client.d.ts +0 -128
- package/services/codex-client.js +0 -1629
- package/services/codex-client.js.map +0 -1
- package/services/openai-client.d.ts +0 -46
- package/services/openai-client.js +0 -318
- package/services/openai-client.js.map +0 -1
- package/types/error-report.d.ts +0 -25
- package/types/error-report.js +0 -4
- package/types/error-report.js.map +0 -1
- package/types/slow-query-report.d.ts +0 -27
- package/types/slow-query-report.js +0 -6
- package/types/slow-query-report.js.map +0 -1
- package/util/ai-qa-policy.d.ts +0 -124
- package/util/ai-qa-policy.js +0 -736
- package/util/ai-qa-policy.js.map +0 -1
- package/util/ai-run-evidence-adapters.d.ts +0 -109
- package/util/ai-run-evidence-adapters.js +0 -7234
- package/util/ai-run-evidence-adapters.js.map +0 -1
- package/util/ai-run-evidence-dashboard.d.ts +0 -88
- package/util/ai-run-evidence-dashboard.js +0 -343
- package/util/ai-run-evidence-dashboard.js.map +0 -1
- package/util/ai-run-evidence-eval.d.ts +0 -86
- package/util/ai-run-evidence-eval.js +0 -1018
- package/util/ai-run-evidence-eval.js.map +0 -1
- package/util/ai-run-evidence.d.ts +0 -244
- package/util/ai-run-evidence.js +0 -1096
- package/util/ai-run-evidence.js.map +0 -1
- package/util/ai-runner-artifacts.d.ts +0 -82
- package/util/ai-runner-artifacts.js +0 -713
- package/util/ai-runner-artifacts.js.map +0 -1
- package/util/ai-runner-manager-autopilot.d.ts +0 -210
- package/util/ai-runner-manager-autopilot.js +0 -642
- package/util/ai-runner-manager-autopilot.js.map +0 -1
- package/util/ai-runner-manager-policy.d.ts +0 -807
- package/util/ai-runner-manager-policy.js +0 -3501
- package/util/ai-runner-manager-policy.js.map +0 -1
- package/util/ai-runner-qa-auth.d.ts +0 -5
- package/util/ai-runner-qa-auth.js +0 -839
- package/util/ai-runner-qa-auth.js.map +0 -1
- package/util/ai-runner-qa-tools.d.ts +0 -26
- package/util/ai-runner-qa-tools.js +0 -3520
- package/util/ai-runner-qa-tools.js.map +0 -1
- package/util/aicoder-runner-v6.d.ts +0 -426
- package/util/aicoder-runner-v6.js +0 -2464
- package/util/aicoder-runner-v6.js.map +0 -1
- package/util/common.d.ts +0 -31
- package/util/common.js +0 -683
- package/util/common.js.map +0 -1
- package/util/customer-portal-password.d.ts +0 -13
- package/util/customer-portal-password.js +0 -209
- package/util/customer-portal-password.js.map +0 -1
- package/util/error-reporter.d.ts +0 -52
- package/util/error-reporter.js +0 -326
- package/util/error-reporter.js.map +0 -1
- package/util/error-tracking.d.ts +0 -13
- package/util/error-tracking.js +0 -120
- package/util/error-tracking.js.map +0 -1
- package/util/openai-usage-cost.d.ts +0 -6
- package/util/openai-usage-cost.js +0 -103
- package/util/openai-usage-cost.js.map +0 -1
- package/util/report-builder-unwinds.d.ts +0 -15
- package/util/report-builder-unwinds.js +0 -156
- package/util/report-builder-unwinds.js.map +0 -1
- package/util/runner-process-janitor.d.ts +0 -27
- package/util/runner-process-janitor.js +0 -208
- package/util/runner-process-janitor.js.map +0 -1
- package/util/schema-report-builder.d.ts +0 -6
- package/util/schema-report-builder.js +0 -481
- package/util/schema-report-builder.js.map +0 -1
- package/util/slow-query-reporter.d.ts +0 -28
- package/util/slow-query-reporter.js +0 -226
- package/util/slow-query-reporter.js.map +0 -1
- package/util/subscription-dependency-context.d.ts +0 -34
- package/util/subscription-dependency-context.js +0 -1283
- package/util/subscription-dependency-context.js.map +0 -1
- package/util/support-runner-v5.d.ts +0 -1426
- package/util/support-runner-v5.js +0 -7631
- package/util/support-runner-v5.js.map +0 -1
- package/util/tokenizer.d.ts +0 -5
- package/util/tokenizer.js +0 -41
- package/util/tokenizer.js.map +0 -1
- package/workers/codex-runner.worker.d.ts +0 -1
- package/workers/codex-runner.worker.js +0 -192
- package/workers/codex-runner.worker.js.map +0 -1
- /package/{private → src/private}/email-templates/enrollment.html +0 -0
- /package/{private → src/private}/email-templates/forgot-password.html +0 -0
- /package/{private → src/private}/email-templates/support-ticket-deleted.html +0 -0
- /package/{private → src/private}/email-templates/support-ticket-modified.html +0 -0
- /package/{private → src/private}/email-templates/support-ticket.html +0 -0
- /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
|
+
});
|