sea-dev 1.0.0

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 (784) hide show
  1. package/.claude/tasks/README.md +89 -0
  2. package/.cursor/rules/commits.mdc +31 -0
  3. package/.cursor/rules/general.mdc +84 -0
  4. package/.github/workflows/ci-cd.yml +141 -0
  5. package/CLAUDE.md +337 -0
  6. package/README.md +129 -0
  7. package/apps/api/.prettierignore +6 -0
  8. package/apps/api/.prettierrc.js +3 -0
  9. package/apps/api/dotenvx-safe.sh +11 -0
  10. package/apps/api/eslint.config.mjs +3 -0
  11. package/apps/api/package.json +58 -0
  12. package/apps/api/src/clients/posthog.ts +25 -0
  13. package/apps/api/src/dal/submission.ts +59 -0
  14. package/apps/api/src/errors.ts +55 -0
  15. package/apps/api/src/index.ts +21 -0
  16. package/apps/api/src/lib/channel.ts +28 -0
  17. package/apps/api/src/lib/config.ts +9 -0
  18. package/apps/api/src/lib/fmt.test.ts +9 -0
  19. package/apps/api/src/lib/fmt.ts +62 -0
  20. package/apps/api/src/lib/invariant.ts +23 -0
  21. package/apps/api/src/middleware/auth.ts +66 -0
  22. package/apps/api/src/routes/index.ts +20 -0
  23. package/apps/api/src/routes/v2/chat/handlers.ts +693 -0
  24. package/apps/api/src/routes/v2/chat/index.ts +257 -0
  25. package/apps/api/src/routes/v2/chat/schemas.ts +43 -0
  26. package/apps/api/src/routes/v2/deals/handlers.ts +64 -0
  27. package/apps/api/src/routes/v2/deals/index.ts +88 -0
  28. package/apps/api/src/routes/v2/deals/schemas.ts +38 -0
  29. package/apps/api/src/routes/v2/forms/handlers.ts +415 -0
  30. package/apps/api/src/routes/v2/forms/index.ts +382 -0
  31. package/apps/api/src/routes/v2/forms/schemas.ts +243 -0
  32. package/apps/api/src/routes/v2/index.ts +19 -0
  33. package/apps/api/src/routes/v2/pipelines/handlers.ts +261 -0
  34. package/apps/api/src/routes/v2/pipelines/index.ts +224 -0
  35. package/apps/api/src/routes/v2/pipelines/schemas.ts +173 -0
  36. package/apps/api/src/routes/v2/submissions/handlers.ts +555 -0
  37. package/apps/api/src/routes/v2/submissions/index.ts +366 -0
  38. package/apps/api/src/routes/v2/submissions/schemas.ts +233 -0
  39. package/apps/api/src/routes/v2/workflows/handlers.ts +81 -0
  40. package/apps/api/src/routes/v2/workflows/index.ts +88 -0
  41. package/apps/api/src/routes/v2/workflows/schemas.ts +40 -0
  42. package/apps/api/src/server.ts +146 -0
  43. package/apps/api/src/static/favicon.ico +0 -0
  44. package/apps/api/src/types/api.ts +14 -0
  45. package/apps/api/src/types/result.ts +3 -0
  46. package/apps/api/tsconfig.json +22 -0
  47. package/apps/api/vite.config.ts +28 -0
  48. package/apps/api/vitest.config.ts +14 -0
  49. package/apps/conversion-worker/Dockerfile +59 -0
  50. package/apps/conversion-worker/package.json +31 -0
  51. package/apps/conversion-worker/src/lib/config.ts +7 -0
  52. package/apps/conversion-worker/src/main.ts +22 -0
  53. package/apps/conversion-worker/src/workflows/convert-pptx.ts +116 -0
  54. package/apps/conversion-worker/tsconfig.json +27 -0
  55. package/apps/conversion-worker/vite.config.ts +33 -0
  56. package/apps/main/.prettierignore +6 -0
  57. package/apps/main/.prettierrc.js +3 -0
  58. package/apps/main/CLAUDE.md +245 -0
  59. package/apps/main/Procfile +1 -0
  60. package/apps/main/README.md +193 -0
  61. package/apps/main/db-tests.jsonl +116 -0
  62. package/apps/main/dotenvx-safe.sh +11 -0
  63. package/apps/main/drizzle/meta/_journal.json +1 -0
  64. package/apps/main/drizzle.config.ts +25 -0
  65. package/apps/main/eslint.config.mjs +3 -0
  66. package/apps/main/generate-routes.mjs +5 -0
  67. package/apps/main/package.json +131 -0
  68. package/apps/main/playwright.config.ts +23 -0
  69. package/apps/main/postcss.config.ts +5 -0
  70. package/apps/main/public/bg-dark.svg +10 -0
  71. package/apps/main/public/bg.svg +10 -0
  72. package/apps/main/public/favicon.ico +0 -0
  73. package/apps/main/run.sh +146 -0
  74. package/apps/main/scripts/browser.ts +14 -0
  75. package/apps/main/scripts/db-test-cov.ts +277 -0
  76. package/apps/main/scripts/login.ts +78 -0
  77. package/apps/main/scripts/repl.ts +61 -0
  78. package/apps/main/src/_foo.ts +31 -0
  79. package/apps/main/src/_tests/db.test.ts +19 -0
  80. package/apps/main/src/_tests/mock-db.ts +60 -0
  81. package/apps/main/src/client.tsx +13 -0
  82. package/apps/main/src/clients/loops.ts +13 -0
  83. package/apps/main/src/clients/polar.ts +12 -0
  84. package/apps/main/src/clients/posthog.ts +12 -0
  85. package/apps/main/src/components/chat/chat-context.tsx +99 -0
  86. package/apps/main/src/components/chat/chat-messages.tsx +184 -0
  87. package/apps/main/src/components/chat/chat-status.tsx +140 -0
  88. package/apps/main/src/components/chat/chat.tsx +458 -0
  89. package/apps/main/src/components/chat/citation-modal.tsx +54 -0
  90. package/apps/main/src/components/cta.tsx +21 -0
  91. package/apps/main/src/components/data-display/derived.tsx +40 -0
  92. package/apps/main/src/components/data-display/group-single.tsx +57 -0
  93. package/apps/main/src/components/data-display/group-table.tsx +165 -0
  94. package/apps/main/src/components/data-display/group-wrapper.tsx +54 -0
  95. package/apps/main/src/components/data-display/item.tsx +678 -0
  96. package/apps/main/src/components/error.tsx +45 -0
  97. package/apps/main/src/components/forms/error.tsx +22 -0
  98. package/apps/main/src/components/grid.tsx +7 -0
  99. package/apps/main/src/components/header/container.tsx +73 -0
  100. package/apps/main/src/components/header/header-bar.tsx +102 -0
  101. package/apps/main/src/components/modals/copy-display.tsx +37 -0
  102. package/apps/main/src/components/modals/copy-form.tsx +152 -0
  103. package/apps/main/src/components/modals/duplicate-workflow.tsx +89 -0
  104. package/apps/main/src/components/modals/field-correction.tsx +323 -0
  105. package/apps/main/src/components/modals/form-viewer.tsx +126 -0
  106. package/apps/main/src/components/modals/modals.tsx +44 -0
  107. package/apps/main/src/components/modals/new-deal.tsx +78 -0
  108. package/apps/main/src/components/modals/new-form.tsx +133 -0
  109. package/apps/main/src/components/modals/new-pipeline.tsx +70 -0
  110. package/apps/main/src/components/modals/new-submission.tsx +321 -0
  111. package/apps/main/src/components/modals/new-workflow.tsx +342 -0
  112. package/apps/main/src/components/modals/transformation-sources-modal.tsx +157 -0
  113. package/apps/main/src/components/modals/view-report.tsx +193 -0
  114. package/apps/main/src/components/not-found.tsx +14 -0
  115. package/apps/main/src/components/search/search-bar.tsx +178 -0
  116. package/apps/main/src/components/sheet-selector.tsx +135 -0
  117. package/apps/main/src/components/side-panel/doc-list.tsx +480 -0
  118. package/apps/main/src/components/sidebar/admin-sidebar.tsx +75 -0
  119. package/apps/main/src/components/sidebar/app-sidebar.tsx +417 -0
  120. package/apps/main/src/components/sidebar/model-select.tsx +134 -0
  121. package/apps/main/src/components/sidebar/settings-sidebar.tsx +132 -0
  122. package/apps/main/src/components/sidebar/sidebar-right.tsx +22 -0
  123. package/apps/main/src/components/sidebar/stop-impersonate.tsx +21 -0
  124. package/apps/main/src/components/svg/loading.tsx +33 -0
  125. package/apps/main/src/components/theme-selector.tsx +43 -0
  126. package/apps/main/src/components/unsaved-badge.tsx +19 -0
  127. package/apps/main/src/components/upload/file-upload.tsx +354 -0
  128. package/apps/main/src/fns/submission-groups.ts +28 -0
  129. package/apps/main/src/fns/submission-items.ts +11 -0
  130. package/apps/main/src/global-middleware.ts +16 -0
  131. package/apps/main/src/hooks/use-update-state.ts +18 -0
  132. package/apps/main/src/lib/auth-client.ts +16 -0
  133. package/apps/main/src/lib/auth.test.ts +359 -0
  134. package/apps/main/src/lib/auth.ts +144 -0
  135. package/apps/main/src/lib/billing.ts +23 -0
  136. package/apps/main/src/lib/config-iso.ts +76 -0
  137. package/apps/main/src/lib/config.ts +61 -0
  138. package/apps/main/src/lib/excel.ts +16 -0
  139. package/apps/main/src/lib/feedback-cache.ts +70 -0
  140. package/apps/main/src/lib/logger.ts +44 -0
  141. package/apps/main/src/lib/models.ts +22 -0
  142. package/apps/main/src/lib/not-found.ts +17 -0
  143. package/apps/main/src/lib/pdf.ts +16 -0
  144. package/apps/main/src/lib/tabularize.ts +54 -0
  145. package/apps/main/src/lib/utils.ts +10 -0
  146. package/apps/main/src/lib/zfd.ts +217 -0
  147. package/apps/main/src/middleware.ts +55 -0
  148. package/apps/main/src/routeTree.gen.ts +1255 -0
  149. package/apps/main/src/router.tsx +24 -0
  150. package/apps/main/src/routes/__root.tsx +92 -0
  151. package/apps/main/src/routes/_authed/_app/(dashboard)/index.tsx +227 -0
  152. package/apps/main/src/routes/_authed/_app/agents/$agentId/config.tsx +224 -0
  153. package/apps/main/src/routes/_authed/_app/agents/$agentId/index.tsx +206 -0
  154. package/apps/main/src/routes/_authed/_app/agents/-components/agent-actions-menu.tsx +94 -0
  155. package/apps/main/src/routes/_authed/_app/agents/-components/agent-artifacts.tsx +153 -0
  156. package/apps/main/src/routes/_authed/_app/agents/-components/agent-chat.tsx +220 -0
  157. package/apps/main/src/routes/_authed/_app/agents/-components/agent-history-menu.tsx +81 -0
  158. package/apps/main/src/routes/_authed/_app/agents/-components/agent-model-select.tsx +84 -0
  159. package/apps/main/src/routes/_authed/_app/agents/-components/agent-relevant-items.tsx +226 -0
  160. package/apps/main/src/routes/_authed/_app/agents/-components/agent-upload-button.tsx +298 -0
  161. package/apps/main/src/routes/_authed/_app/agents/-components/context-modal.tsx +187 -0
  162. package/apps/main/src/routes/_authed/_app/agents/-fns.ts +560 -0
  163. package/apps/main/src/routes/_authed/_app/agents/index.tsx +65 -0
  164. package/apps/main/src/routes/_authed/_app/deals/$dealId/$subId/-components/citation-tree.tsx +268 -0
  165. package/apps/main/src/routes/_authed/_app/deals/$dealId/$subId.tsx +655 -0
  166. package/apps/main/src/routes/_authed/_app/deals/$dealId/-components/doc-loading.tsx +37 -0
  167. package/apps/main/src/routes/_authed/_app/deals/$dealId/-components/share-link.tsx +42 -0
  168. package/apps/main/src/routes/_authed/_app/deals/$dealId/-components/submission-card.tsx +89 -0
  169. package/apps/main/src/routes/_authed/_app/deals/$dealId/-components/submission-filter.tsx +193 -0
  170. package/apps/main/src/routes/_authed/_app/deals/$dealId/-components/submissions.tsx +36 -0
  171. package/apps/main/src/routes/_authed/_app/deals/$dealId/-components/summary.tsx +82 -0
  172. package/apps/main/src/routes/_authed/_app/deals/$dealId/-components/upload-doc.tsx +120 -0
  173. package/apps/main/src/routes/_authed/_app/deals/$dealId/-fns.ts +653 -0
  174. package/apps/main/src/routes/_authed/_app/deals/$dealId/index.tsx +259 -0
  175. package/apps/main/src/routes/_authed/_app/deals/$dealId/route.tsx +29 -0
  176. package/apps/main/src/routes/_authed/_app/deals/index.tsx +104 -0
  177. package/apps/main/src/routes/_authed/_app/feedback/index.tsx +639 -0
  178. package/apps/main/src/routes/_authed/_app/feedback/insights.tsx +250 -0
  179. package/apps/main/src/routes/_authed/_app/pipelines/$pipelineId/$runId/-components/blockers-panel.tsx +260 -0
  180. package/apps/main/src/routes/_authed/_app/pipelines/$pipelineId/$runId/-components/manual-input-panel.tsx +301 -0
  181. package/apps/main/src/routes/_authed/_app/pipelines/$pipelineId/$runId/-components/submission-selector-modal.tsx +143 -0
  182. package/apps/main/src/routes/_authed/_app/pipelines/$pipelineId/$runId/-components/upload-doc.tsx +120 -0
  183. package/apps/main/src/routes/_authed/_app/pipelines/$pipelineId/$runId/index.tsx +1485 -0
  184. package/apps/main/src/routes/_authed/_app/pipelines/$pipelineId/-components/dag-view.tsx +296 -0
  185. package/apps/main/src/routes/_authed/_app/pipelines/$pipelineId/-components/step-config-modal.tsx +634 -0
  186. package/apps/main/src/routes/_authed/_app/pipelines/$pipelineId/index.tsx +911 -0
  187. package/apps/main/src/routes/_authed/_app/pipelines/-fns.ts +510 -0
  188. package/apps/main/src/routes/_authed/_app/pipelines/index.tsx +103 -0
  189. package/apps/main/src/routes/_authed/_app/reports/$reportId.tsx +397 -0
  190. package/apps/main/src/routes/_authed/_app/reports/-fns.ts +11 -0
  191. package/apps/main/src/routes/_authed/_app/reports/index.tsx +22 -0
  192. package/apps/main/src/routes/_authed/_app/route.tsx +48 -0
  193. package/apps/main/src/routes/_authed/_app/submissions/-columns.tsx +161 -0
  194. package/apps/main/src/routes/_authed/_app/submissions/-fns.ts +128 -0
  195. package/apps/main/src/routes/_authed/_app/submissions/index.tsx +190 -0
  196. package/apps/main/src/routes/_authed/_app/workflows/$wfSlug/$formId.tsx +542 -0
  197. package/apps/main/src/routes/_authed/_app/workflows/$wfSlug/-components/derived.tsx +154 -0
  198. package/apps/main/src/routes/_authed/_app/workflows/$wfSlug/-components/field.tsx +369 -0
  199. package/apps/main/src/routes/_authed/_app/workflows/$wfSlug/-components/group.tsx +475 -0
  200. package/apps/main/src/routes/_authed/_app/workflows/$wfSlug/index.tsx +263 -0
  201. package/apps/main/src/routes/_authed/_app/workflows/$wfSlug/route.tsx +33 -0
  202. package/apps/main/src/routes/_authed/_app/workflows/-components/form-card.tsx +315 -0
  203. package/apps/main/src/routes/_authed/_app/workflows/index.tsx +86 -0
  204. package/apps/main/src/routes/_authed/admin/index.tsx +12 -0
  205. package/apps/main/src/routes/_authed/admin/route.tsx +42 -0
  206. package/apps/main/src/routes/_authed/admin/users/-columns.tsx +124 -0
  207. package/apps/main/src/routes/_authed/admin/users/-fns.ts +30 -0
  208. package/apps/main/src/routes/_authed/admin/users/index.tsx +29 -0
  209. package/apps/main/src/routes/_authed/catchNotFound.tsx +114 -0
  210. package/apps/main/src/routes/_authed/redirects/forms.$id.tsx +29 -0
  211. package/apps/main/src/routes/_authed/redirects/submissions.$id.tsx +27 -0
  212. package/apps/main/src/routes/_authed/redirects/workflows.$id.tsx +27 -0
  213. package/apps/main/src/routes/_authed/route.tsx +51 -0
  214. package/apps/main/src/routes/_authed/settings/-components/new-api-key.tsx +85 -0
  215. package/apps/main/src/routes/_authed/settings/-components/new-invite.tsx +100 -0
  216. package/apps/main/src/routes/_authed/settings/analytics.tsx +1710 -0
  217. package/apps/main/src/routes/_authed/settings/billing/-components/price-table.tsx +129 -0
  218. package/apps/main/src/routes/_authed/settings/billing/-fns.ts +76 -0
  219. package/apps/main/src/routes/_authed/settings/billing/index.tsx +119 -0
  220. package/apps/main/src/routes/_authed/settings/embed.tsx +337 -0
  221. package/apps/main/src/routes/_authed/settings/index.tsx +12 -0
  222. package/apps/main/src/routes/_authed/settings/keys.tsx +157 -0
  223. package/apps/main/src/routes/_authed/settings/members.tsx +276 -0
  224. package/apps/main/src/routes/_authed/settings/route.tsx +22 -0
  225. package/apps/main/src/routes/_authed/settings/user.tsx +87 -0
  226. package/apps/main/src/routes/_authed/settings/workspace.tsx +206 -0
  227. package/apps/main/src/routes/_public/-components/sign-in-up.tsx +96 -0
  228. package/apps/main/src/routes/_public/embedded.tsx +57 -0
  229. package/apps/main/src/routes/_public/invite.$inviteId.tsx +143 -0
  230. package/apps/main/src/routes/_public/no-access.tsx +38 -0
  231. package/apps/main/src/routes/_public/no-invite.tsx +39 -0
  232. package/apps/main/src/routes/_public/otp.tsx +103 -0
  233. package/apps/main/src/routes/_public/route.tsx +15 -0
  234. package/apps/main/src/routes/_public/sign-in.tsx +111 -0
  235. package/apps/main/src/routes/_public/sign-up.tsx +114 -0
  236. package/apps/main/src/routes/api/auth/$.ts +11 -0
  237. package/apps/main/src/routes/api/billing/paid.ts +42 -0
  238. package/apps/main/src/routes/api/billing/webhooks.ts +70 -0
  239. package/apps/main/src/routes/api/chat/agent.ts +40 -0
  240. package/apps/main/src/routes/api/chat/key.ts +42 -0
  241. package/apps/main/src/routes/api/chat/member.ts +35 -0
  242. package/apps/main/src/routes/api/test/index.ts +19 -0
  243. package/apps/main/src/server.tsx +6 -0
  244. package/apps/main/src/styles/app.css +23 -0
  245. package/apps/main/src/vite-env.d.ts +7 -0
  246. package/apps/main/test.http +6 -0
  247. package/apps/main/tsconfig.json +17 -0
  248. package/apps/main/vite.config.ts +24 -0
  249. package/apps/main/vitest.config.js +17 -0
  250. package/apps/mcp/README.md +171 -0
  251. package/apps/mcp/eslint.config.mjs +3 -0
  252. package/apps/mcp/package.json +37 -0
  253. package/apps/mcp/src/index.ts +414 -0
  254. package/apps/mcp/tsconfig.json +19 -0
  255. package/apps/mcp/vite.config.ts +22 -0
  256. package/apps/posthog-proxy/index.html +9 -0
  257. package/apps/workers/.prettierignore +7 -0
  258. package/apps/workers/.prettierrc.js +3 -0
  259. package/apps/workers/dotenvx-safe.sh +11 -0
  260. package/apps/workers/eslint.config.mjs +3 -0
  261. package/apps/workers/package.json +65 -0
  262. package/apps/workers/src/lib/config.ts +7 -0
  263. package/apps/workers/src/lib/messages.ts +0 -0
  264. package/apps/workers/src/lib/posthog.ts +25 -0
  265. package/apps/workers/src/main.ts +58 -0
  266. package/apps/workers/src/workflows/extraction.ts +866 -0
  267. package/apps/workers/src/workflows/index.ts +3 -0
  268. package/apps/workers/src/workflows/pipeline-dag.ts +210 -0
  269. package/apps/workers/src/workflows/pipeline-steps.ts +1393 -0
  270. package/apps/workers/tsconfig.json +16 -0
  271. package/apps/workers/vite.config.ts +35 -0
  272. package/docs/CHANGELOG.md +84 -0
  273. package/docs/agent-templates-and-runs.md +859 -0
  274. package/docs/aws-migration-plan.md +267 -0
  275. package/docs/impl-p0-form-builder-improvements.md +683 -0
  276. package/docs/on-prem-deployment-spec.docx +0 -0
  277. package/docs/on-prem-deployment-spec.md +378 -0
  278. package/docs/prd-form-builder-strategy.md +1120 -0
  279. package/docs/widget-ng-apf-packaging-spec.md +43 -0
  280. package/infra/k8s/charts/seadotdev/Chart.yaml +6 -0
  281. package/infra/k8s/charts/seadotdev/templates/_helpers.tpl +27 -0
  282. package/infra/k8s/charts/seadotdev/templates/api-v2.yaml +105 -0
  283. package/infra/k8s/charts/seadotdev/templates/external-secrets.yaml +83 -0
  284. package/infra/k8s/charts/seadotdev/templates/ingress.yaml +54 -0
  285. package/infra/k8s/charts/seadotdev/templates/main-app.yaml +104 -0
  286. package/infra/k8s/charts/seadotdev/templates/workers.yaml +182 -0
  287. package/infra/k8s/charts/seadotdev/values.yaml +143 -0
  288. package/infra/terraform/main.tf +399 -0
  289. package/libs/ai/.prettierignore +2 -0
  290. package/libs/ai/.prettierrc.js +5 -0
  291. package/libs/ai/README.md +139 -0
  292. package/libs/ai/eslint.config.mjs +3 -0
  293. package/libs/ai/package.json +42 -0
  294. package/libs/ai/src/index.ts +5 -0
  295. package/libs/ai/src/models.ts +19 -0
  296. package/libs/ai/src/rag/index.ts +1 -0
  297. package/libs/ai/src/rag/rag.test.ts +99 -0
  298. package/libs/ai/src/rag/rag.ts +510 -0
  299. package/libs/ai/tsconfig.json +21 -0
  300. package/libs/ai/vite.config.ts +38 -0
  301. package/libs/cache/.prettierignore +2 -0
  302. package/libs/cache/eslint.config.mjs +3 -0
  303. package/libs/cache/package.json +35 -0
  304. package/libs/cache/src/feedback.ts +77 -0
  305. package/libs/cache/src/index.ts +2 -0
  306. package/libs/cache/tsconfig.json +19 -0
  307. package/libs/cache/vite.config.ts +36 -0
  308. package/libs/clients/.prettierignore +6 -0
  309. package/libs/clients/eslint.config.mjs +3 -0
  310. package/libs/clients/package.json +59 -0
  311. package/libs/clients/src/azure.ts +249 -0
  312. package/libs/clients/src/gcp.ts +220 -0
  313. package/libs/clients/src/hatchet.ts +86 -0
  314. package/libs/clients/src/index.ts +8 -0
  315. package/libs/clients/src/loops.ts +86 -0
  316. package/libs/clients/src/polar.ts +77 -0
  317. package/libs/clients/src/posthog.ts +55 -0
  318. package/libs/clients/tsconfig.json +19 -0
  319. package/libs/clients/vite.config.ts +35 -0
  320. package/libs/config/.prettierignore +6 -0
  321. package/libs/config/.prettierrc.js +12 -0
  322. package/libs/config/eslint.config.mjs +3 -0
  323. package/libs/config/package.json +50 -0
  324. package/libs/config/src/azure.ts +54 -0
  325. package/libs/config/src/db.ts +18 -0
  326. package/libs/config/src/gcp.ts +53 -0
  327. package/libs/config/src/google.ts +17 -0
  328. package/libs/config/src/hatchet.ts +20 -0
  329. package/libs/config/src/index.ts +108 -0
  330. package/libs/config/src/llm.ts +17 -0
  331. package/libs/config/src/polar.ts +24 -0
  332. package/libs/config/src/util.ts +8 -0
  333. package/libs/config/src/vercel.ts +26 -0
  334. package/libs/config/tsconfig.json +19 -0
  335. package/libs/config/vite.config.ts +34 -0
  336. package/libs/core/.prettierignore +2 -0
  337. package/libs/core/eslint.config.mjs +3 -0
  338. package/libs/core/package.json +59 -0
  339. package/libs/core/src/chat/derived.ts +97 -0
  340. package/libs/core/src/chat/feedback.ts +293 -0
  341. package/libs/core/src/chat/index.ts +6 -0
  342. package/libs/core/src/chat/model.ts +92 -0
  343. package/libs/core/src/chat/prepare-tools.ts +286 -0
  344. package/libs/core/src/chat/prompts.ts +623 -0
  345. package/libs/core/src/chat/stream.ts +311 -0
  346. package/libs/core/src/chat/summarize.ts +168 -0
  347. package/libs/core/src/chat/tools/agent.ts +403 -0
  348. package/libs/core/src/chat/tools/chart-agent.ts +526 -0
  349. package/libs/core/src/chat/tools/chart-helpers/sandbox.ts +47 -0
  350. package/libs/core/src/chat/tools/chart.ts +86 -0
  351. package/libs/core/src/chat/tools/credit-agent.ts +1383 -0
  352. package/libs/core/src/chat/tools/credit.ts +1435 -0
  353. package/libs/core/src/chat/tools/deep-dive-agent.ts +100 -0
  354. package/libs/core/src/chat/tools/deep-dive.ts +141 -0
  355. package/libs/core/src/chat/tools/form.ts +449 -0
  356. package/libs/core/src/chat/tools/helpers.ts +91 -0
  357. package/libs/core/src/chat/tools/index.ts +42 -0
  358. package/libs/core/src/chat/tools/pipeline-artifact.ts +76 -0
  359. package/libs/core/src/chat/tools/report.ts +40 -0
  360. package/libs/core/src/chat/tools/search.ts +390 -0
  361. package/libs/core/src/chat/tools/submission.ts +227 -0
  362. package/libs/core/src/chat/tools/workflow.ts +684 -0
  363. package/libs/core/src/chat/types.ts +3 -0
  364. package/libs/core/src/data-extraction/classification/azure.ts +168 -0
  365. package/libs/core/src/data-extraction/classification/index.ts +1 -0
  366. package/libs/core/src/data-extraction/dal.ts +246 -0
  367. package/libs/core/src/data-extraction/form-structure-extractor.ts +294 -0
  368. package/libs/core/src/data-extraction/index.ts +4 -0
  369. package/libs/core/src/data-extraction/layout/azure.ts +730 -0
  370. package/libs/core/src/data-extraction/layout/excel.ts +180 -0
  371. package/libs/core/src/data-extraction/layout/gcp.ts +1071 -0
  372. package/libs/core/src/data-extraction/layout/index.ts +266 -0
  373. package/libs/core/src/data-extraction/layout/plaintext.ts +45 -0
  374. package/libs/core/src/data-extraction/models.ts +38 -0
  375. package/libs/core/src/data-extraction/pdf-utils.ts +96 -0
  376. package/libs/core/src/data-extraction/structuring/bank-statement.ts +1182 -0
  377. package/libs/core/src/data-extraction/structuring/custom.ts +495 -0
  378. package/libs/core/src/data-extraction/structuring/index.ts +290 -0
  379. package/libs/core/src/data-extraction/structuring/prompts.ts +69 -0
  380. package/libs/core/src/data-extraction/type-guards.ts +110 -0
  381. package/libs/core/src/data-extraction/types.ts +84 -0
  382. package/libs/core/src/data-extraction/utils.ts +31 -0
  383. package/libs/core/src/data-extraction/validation/bank-statement.ts +127 -0
  384. package/libs/core/src/deals.ts +17 -0
  385. package/libs/core/src/documents.ts +152 -0
  386. package/libs/core/src/index.ts +5 -0
  387. package/libs/core/src/pipelines/display.ts +678 -0
  388. package/libs/core/src/pipelines/execute.ts +2342 -0
  389. package/libs/core/src/pipelines/index.ts +4 -0
  390. package/libs/core/src/pipelines/list.ts +12 -0
  391. package/libs/core/src/pipelines/runs.ts +53 -0
  392. package/libs/core/tsconfig.json +20 -0
  393. package/libs/core/vite.config.ts +56 -0
  394. package/libs/dal/.prettierignore +6 -0
  395. package/libs/dal/.prettierrc.js +12 -0
  396. package/libs/dal/eslint.config.mjs +3 -0
  397. package/libs/dal/package.json +57 -0
  398. package/libs/dal/src/_tests/db.test.ts +19 -0
  399. package/libs/dal/src/_tests/mock-db.ts +60 -0
  400. package/libs/dal/src/api-key.test.ts +397 -0
  401. package/libs/dal/src/api-key.ts +110 -0
  402. package/libs/dal/src/billing.ts +23 -0
  403. package/libs/dal/src/conversation.test.ts +655 -0
  404. package/libs/dal/src/conversation.ts +532 -0
  405. package/libs/dal/src/deal.test.ts +45 -0
  406. package/libs/dal/src/deal.ts +87 -0
  407. package/libs/dal/src/defaults-consumer-lending-uk.ts +33 -0
  408. package/libs/dal/src/defaults-consumer-lending-us.ts +33 -0
  409. package/libs/dal/src/defaults-private-credit.ts +57 -0
  410. package/libs/dal/src/defaults-private-equity.ts +51 -0
  411. package/libs/dal/src/defaults-smb-lending-us.ts +1569 -0
  412. package/libs/dal/src/defaults-sme-lending-uk-express.ts +1527 -0
  413. package/libs/dal/src/defaults-sme-lending-uk.ts +1669 -0
  414. package/libs/dal/src/defaults-types.ts +23 -0
  415. package/libs/dal/src/defaults.ts +550 -0
  416. package/libs/dal/src/document.test.ts +70 -0
  417. package/libs/dal/src/document.ts +192 -0
  418. package/libs/dal/src/feedback.ts +255 -0
  419. package/libs/dal/src/form.test.ts +637 -0
  420. package/libs/dal/src/form.ts +1165 -0
  421. package/libs/dal/src/index.ts +20 -0
  422. package/libs/dal/src/invitation.test.ts +746 -0
  423. package/libs/dal/src/invitation.ts +207 -0
  424. package/libs/dal/src/member.test.ts +185 -0
  425. package/libs/dal/src/member.ts +80 -0
  426. package/libs/dal/src/organization.ts +116 -0
  427. package/libs/dal/src/permission.ts +25 -0
  428. package/libs/dal/src/pipeline.test.ts +388 -0
  429. package/libs/dal/src/pipeline.ts +4222 -0
  430. package/libs/dal/src/report.ts +199 -0
  431. package/libs/dal/src/result.ts +16 -0
  432. package/libs/dal/src/search.ts +172 -0
  433. package/libs/dal/src/session.test.ts +110 -0
  434. package/libs/dal/src/session.ts +31 -0
  435. package/libs/dal/src/submission.test.ts +1304 -0
  436. package/libs/dal/src/submission.ts +1396 -0
  437. package/libs/dal/src/tool.ts +159 -0
  438. package/libs/dal/src/user.ts +16 -0
  439. package/libs/dal/src/workflow.test.ts +89 -0
  440. package/libs/dal/src/workflow.ts +262 -0
  441. package/libs/dal/tsconfig.build.json +4 -0
  442. package/libs/dal/tsconfig.json +22 -0
  443. package/libs/dal/vite.config.ts +34 -0
  444. package/libs/db/.prettierignore +6 -0
  445. package/libs/db/.prettierrc.js +12 -0
  446. package/libs/db/eslint.config.mjs +3 -0
  447. package/libs/db/package.json +52 -0
  448. package/libs/db/src/index.ts +24 -0
  449. package/libs/db/src/relations.ts +549 -0
  450. package/libs/db/src/schema.ts +2 -0
  451. package/libs/db/src/schemas/api.ts +35 -0
  452. package/libs/db/src/schemas/conversations.ts +175 -0
  453. package/libs/db/src/schemas/core.ts +359 -0
  454. package/libs/db/src/schemas/documents.ts +181 -0
  455. package/libs/db/src/schemas/feedback.ts +40 -0
  456. package/libs/db/src/schemas/index.ts +26 -0
  457. package/libs/db/src/schemas/organisations.ts +97 -0
  458. package/libs/db/src/schemas/pipelines.ts +440 -0
  459. package/libs/db/src/schemas/users.ts +95 -0
  460. package/libs/db/src/types.ts +190 -0
  461. package/libs/db/src/utils.ts +14 -0
  462. package/libs/db/tsconfig.json +19 -0
  463. package/libs/db/vite.config.ts +31 -0
  464. package/libs/lint/.prettierignore +6 -0
  465. package/libs/lint/eslint.config.mjs +61 -0
  466. package/libs/lint/package.json +29 -0
  467. package/libs/lint/prettier.config.js +12 -0
  468. package/libs/schemas/.prettierignore +6 -0
  469. package/libs/schemas/.prettierrc.js +12 -0
  470. package/libs/schemas/README.md +15 -0
  471. package/libs/schemas/eslint.config.mjs +3 -0
  472. package/libs/schemas/package.json +67 -0
  473. package/libs/schemas/src/core/chat.ts +67 -0
  474. package/libs/schemas/src/core/core-result.ts +15 -0
  475. package/libs/schemas/src/core/data-extraction.ts +184 -0
  476. package/libs/schemas/src/core/layout.ts +478 -0
  477. package/libs/schemas/src/core/pipeline.ts +128 -0
  478. package/libs/schemas/src/core/submission.ts +97 -0
  479. package/libs/schemas/src/db/account.ts +57 -0
  480. package/libs/schemas/src/db/apiKey.ts +57 -0
  481. package/libs/schemas/src/db/context.ts +33 -0
  482. package/libs/schemas/src/db/conversation.ts +65 -0
  483. package/libs/schemas/src/db/deal.ts +42 -0
  484. package/libs/schemas/src/db/document.ts +103 -0
  485. package/libs/schemas/src/db/documentCitation.ts +58 -0
  486. package/libs/schemas/src/db/documentExtraction.ts +69 -0
  487. package/libs/schemas/src/db/fieldCorrection.ts +85 -0
  488. package/libs/schemas/src/db/form.ts +45 -0
  489. package/libs/schemas/src/db/formField.ts +59 -0
  490. package/libs/schemas/src/db/formGroup.ts +42 -0
  491. package/libs/schemas/src/db/impersonation.ts +39 -0
  492. package/libs/schemas/src/db/index.ts +25 -0
  493. package/libs/schemas/src/db/invitation.ts +42 -0
  494. package/libs/schemas/src/db/member.ts +36 -0
  495. package/libs/schemas/src/db/message.ts +58 -0
  496. package/libs/schemas/src/db/organization.ts +62 -0
  497. package/libs/schemas/src/db/session.ts +48 -0
  498. package/libs/schemas/src/db/submission.ts +54 -0
  499. package/libs/schemas/src/db/submissionGroup.ts +36 -0
  500. package/libs/schemas/src/db/submissionItem.ts +33 -0
  501. package/libs/schemas/src/db/submissionItemVersion.ts +70 -0
  502. package/libs/schemas/src/db/user.ts +51 -0
  503. package/libs/schemas/src/db/utils.ts +3 -0
  504. package/libs/schemas/src/db/verification.ts +36 -0
  505. package/libs/schemas/src/db/workflow.ts +42 -0
  506. package/libs/schemas/src/index.ts +10 -0
  507. package/libs/schemas/tsconfig.json +21 -0
  508. package/libs/schemas/vite.config.ts +38 -0
  509. package/libs/ui/.prettierignore +6 -0
  510. package/libs/ui/.prettierrc.js +12 -0
  511. package/libs/ui/components.json +24 -0
  512. package/libs/ui/eslint.config.mjs +3 -0
  513. package/libs/ui/package.json +142 -0
  514. package/libs/ui/src/components/chart-viz/chart.tsx +255 -0
  515. package/libs/ui/src/components/chart-viz/converters.ts +474 -0
  516. package/libs/ui/src/components/chart-viz/dashboard.tsx +146 -0
  517. package/libs/ui/src/components/chart-viz/index.ts +37 -0
  518. package/libs/ui/src/components/chart-viz/markdown.tsx +344 -0
  519. package/libs/ui/src/components/chart-viz/table.tsx +446 -0
  520. package/libs/ui/src/components/chart-viz/theme-context.tsx +70 -0
  521. package/libs/ui/src/components/chart-viz/themes/dark.ts +98 -0
  522. package/libs/ui/src/components/chart-viz/themes/index.ts +69 -0
  523. package/libs/ui/src/components/chart-viz/themes/light.ts +98 -0
  524. package/libs/ui/src/components/chart-viz/themes/tailwind.ts +326 -0
  525. package/libs/ui/src/components/chart-viz/themes/types.ts +99 -0
  526. package/libs/ui/src/components/chart-viz/tool-display.tsx +150 -0
  527. package/libs/ui/src/components/chart-viz/types.ts +95 -0
  528. package/libs/ui/src/components/doc-viewers/excel/index.tsx +431 -0
  529. package/libs/ui/src/components/doc-viewers/excel/themes.ts +160 -0
  530. package/libs/ui/src/components/doc-viewers/image/index.tsx +410 -0
  531. package/libs/ui/src/components/doc-viewers/pdf/index.tsx +258 -0
  532. package/libs/ui/src/components/doc-viewers/pdf/virtualized-pdf.tsx +556 -0
  533. package/libs/ui/src/components/misc/rel-date.tsx +52 -0
  534. package/libs/ui/src/components/misc/styled-link.tsx +2 -0
  535. package/libs/ui/src/components/table/data-table.tsx +546 -0
  536. package/libs/ui/src/components/table/report-table.tsx +305 -0
  537. package/libs/ui/src/components/table/sortable-column.tsx +34 -0
  538. package/libs/ui/src/components/ui/accordion.tsx +62 -0
  539. package/libs/ui/src/components/ui/alert-dialog.tsx +142 -0
  540. package/libs/ui/src/components/ui/alert.tsx +62 -0
  541. package/libs/ui/src/components/ui/artifact.tsx +118 -0
  542. package/libs/ui/src/components/ui/attachments.tsx +388 -0
  543. package/libs/ui/src/components/ui/avatar.tsx +39 -0
  544. package/libs/ui/src/components/ui/badge.tsx +43 -0
  545. package/libs/ui/src/components/ui/breadcrumb.tsx +102 -0
  546. package/libs/ui/src/components/ui/button-group.tsx +78 -0
  547. package/libs/ui/src/components/ui/button.tsx +79 -0
  548. package/libs/ui/src/components/ui/card.tsx +32 -0
  549. package/libs/ui/src/components/ui/carousel.tsx +228 -0
  550. package/libs/ui/src/components/ui/chain-of-thought.tsx +198 -0
  551. package/libs/ui/src/components/ui/checkbox.tsx +27 -0
  552. package/libs/ui/src/components/ui/citation.tsx +34 -0
  553. package/libs/ui/src/components/ui/code-block.tsx +500 -0
  554. package/libs/ui/src/components/ui/collapsible.tsx +19 -0
  555. package/libs/ui/src/components/ui/command.tsx +161 -0
  556. package/libs/ui/src/components/ui/conversation.tsx +90 -0
  557. package/libs/ui/src/components/ui/dialog.tsx +142 -0
  558. package/libs/ui/src/components/ui/dropdown-menu.tsx +246 -0
  559. package/libs/ui/src/components/ui/highlight.tsx +3 -0
  560. package/libs/ui/src/components/ui/hover-card.tsx +36 -0
  561. package/libs/ui/src/components/ui/inline-citation.tsx +251 -0
  562. package/libs/ui/src/components/ui/input-group.tsx +156 -0
  563. package/libs/ui/src/components/ui/input-otp.tsx +78 -0
  564. package/libs/ui/src/components/ui/input.tsx +21 -0
  565. package/libs/ui/src/components/ui/label.tsx +19 -0
  566. package/libs/ui/src/components/ui/model-selector.tsx +174 -0
  567. package/libs/ui/src/components/ui/multisidebar.tsx +750 -0
  568. package/libs/ui/src/components/ui/popover.tsx +43 -0
  569. package/libs/ui/src/components/ui/progress.tsx +28 -0
  570. package/libs/ui/src/components/ui/reasoning.tsx +178 -0
  571. package/libs/ui/src/components/ui/resizable.tsx +49 -0
  572. package/libs/ui/src/components/ui/scroll-area.tsx +54 -0
  573. package/libs/ui/src/components/ui/select.tsx +171 -0
  574. package/libs/ui/src/components/ui/separator.tsx +26 -0
  575. package/libs/ui/src/components/ui/sheet.tsx +128 -0
  576. package/libs/ui/src/components/ui/shimmer.tsx +53 -0
  577. package/libs/ui/src/components/ui/skeleton.tsx +13 -0
  578. package/libs/ui/src/components/ui/sonner.tsx +23 -0
  579. package/libs/ui/src/components/ui/switch.tsx +26 -0
  580. package/libs/ui/src/components/ui/table.tsx +96 -0
  581. package/libs/ui/src/components/ui/tabs.tsx +52 -0
  582. package/libs/ui/src/components/ui/textarea.tsx +41 -0
  583. package/libs/ui/src/components/ui/tool.tsx +209 -0
  584. package/libs/ui/src/components/ui/tooltip.tsx +58 -0
  585. package/libs/ui/src/components/ui/typography.tsx +113 -0
  586. package/libs/ui/src/fonts/manrope-v15-latin-300.woff2 +0 -0
  587. package/libs/ui/src/fonts/manrope-v15-latin-400.woff2 +0 -0
  588. package/libs/ui/src/fonts/manrope-v15-latin-500.woff2 +0 -0
  589. package/libs/ui/src/fonts/manrope-v15-latin-600.woff2 +0 -0
  590. package/libs/ui/src/hooks/use-mobile.ts +19 -0
  591. package/libs/ui/src/lib/utils.ts +6 -0
  592. package/libs/ui/src/styles/fonts.css +35 -0
  593. package/libs/ui/src/styles/style.css +218 -0
  594. package/libs/ui/tsconfig.json +21 -0
  595. package/libs/ui/vite.config.ts +80 -0
  596. package/libs/ui-lit/README.md +245 -0
  597. package/libs/ui-lit/TESTING_GUIDE.md +296 -0
  598. package/libs/ui-lit/eslint.config.mjs +3 -0
  599. package/libs/ui-lit/package.json +41 -0
  600. package/libs/ui-lit/scripts/build-css.js +43 -0
  601. package/libs/ui-lit/src/components/sea-alert.ts +132 -0
  602. package/libs/ui-lit/src/components/sea-button.ts +95 -0
  603. package/libs/ui-lit/src/components/sea-card.ts +113 -0
  604. package/libs/ui-lit/src/components/sea-input.ts +184 -0
  605. package/libs/ui-lit/src/components/sea-spinner.ts +65 -0
  606. package/libs/ui-lit/src/index.ts +15 -0
  607. package/libs/ui-lit/src/lib/utils.ts +6 -0
  608. package/libs/ui-lit/src/styles/tailwind.css +76 -0
  609. package/libs/ui-lit/src/theme.css +66 -0
  610. package/libs/ui-lit/src/theme.ts +79 -0
  611. package/libs/ui-lit/src/vite-env.d.ts +6 -0
  612. package/libs/ui-lit/tailwind.config.ts +50 -0
  613. package/libs/ui-lit/test.html +289 -0
  614. package/libs/ui-lit/tsconfig.json +23 -0
  615. package/libs/ui-lit/vite.config.ts +31 -0
  616. package/libs/ui-lit/vite.css.config.ts +20 -0
  617. package/libs/util/.prettierignore +6 -0
  618. package/libs/util/.prettierrc.js +12 -0
  619. package/libs/util/eslint.config.mjs +3 -0
  620. package/libs/util/package.json +45 -0
  621. package/libs/util/src/billing.ts +10 -0
  622. package/libs/util/src/data-transform.ts +19 -0
  623. package/libs/util/src/encryption.ts +45 -0
  624. package/libs/util/src/fmt.test.ts +9 -0
  625. package/libs/util/src/fmt.ts +71 -0
  626. package/libs/util/src/fuzzy.ts +47 -0
  627. package/libs/util/src/id.ts +24 -0
  628. package/libs/util/src/invariant.ts +31 -0
  629. package/libs/util/src/sub-name.ts +7 -0
  630. package/libs/util/tsconfig.json +19 -0
  631. package/libs/util/vite.config.ts +34 -0
  632. package/package.json +28 -0
  633. package/packages/widget/.prettierignore +6 -0
  634. package/packages/widget/.prettierrc.js +12 -0
  635. package/packages/widget/README.md +95 -0
  636. package/packages/widget/eslint.config.mjs +11 -0
  637. package/packages/widget/openapi-ts.config.ts +8 -0
  638. package/packages/widget/package.json +89 -0
  639. package/packages/widget/postcss.config.mjs +10 -0
  640. package/packages/widget/src/clients/api/client/client.ts +187 -0
  641. package/packages/widget/src/clients/api/client/index.ts +22 -0
  642. package/packages/widget/src/clients/api/client/types.ts +192 -0
  643. package/packages/widget/src/clients/api/client/utils.ts +394 -0
  644. package/packages/widget/src/clients/api/client.gen.ts +18 -0
  645. package/packages/widget/src/clients/api/core/auth.ts +39 -0
  646. package/packages/widget/src/clients/api/core/bodySerializer.ts +74 -0
  647. package/packages/widget/src/clients/api/core/params.ts +132 -0
  648. package/packages/widget/src/clients/api/core/pathSerializer.ts +169 -0
  649. package/packages/widget/src/clients/api/core/types.ts +80 -0
  650. package/packages/widget/src/clients/api/index.ts +3 -0
  651. package/packages/widget/src/clients/api/sdk.gen.ts +805 -0
  652. package/packages/widget/src/clients/api/types.gen.ts +2085 -0
  653. package/packages/widget/src/components/container.tsx +42 -0
  654. package/packages/widget/src/components/data-display.tsx +384 -0
  655. package/packages/widget/src/components/data-viewer.tsx +311 -0
  656. package/packages/widget/src/components/doc-list.tsx +102 -0
  657. package/packages/widget/src/components/field-correction-modal.tsx +265 -0
  658. package/packages/widget/src/components/header.tsx +71 -0
  659. package/packages/widget/src/components/new-submission.tsx +290 -0
  660. package/packages/widget/src/components/sidebar-right.tsx +19 -0
  661. package/packages/widget/src/components/submission-card.tsx +66 -0
  662. package/packages/widget/src/components/submission-page.tsx +75 -0
  663. package/packages/widget/src/components/upload-doc.tsx +241 -0
  664. package/packages/widget/src/components/widget.tsx +101 -0
  665. package/packages/widget/src/index.tsx +167 -0
  666. package/packages/widget/src/lib/config.ts +2 -0
  667. package/packages/widget/src/lib/util.ts +40 -0
  668. package/packages/widget/src/styles/index.css +5 -0
  669. package/packages/widget/src/styles/tw-properties.css +337 -0
  670. package/packages/widget/src/vite-env.d.ts +3 -0
  671. package/packages/widget/tsconfig.app.json +35 -0
  672. package/packages/widget/tsconfig.json +4 -0
  673. package/packages/widget/tsconfig.node.json +24 -0
  674. package/packages/widget/vite.config.ts +116 -0
  675. package/packages/widget-lit/BOTTLENECKS.md +250 -0
  676. package/packages/widget-lit/IMPLEMENTATION_SUMMARY.md +295 -0
  677. package/packages/widget-lit/README.md +232 -0
  678. package/packages/widget-lit/eslint.config.mjs +3 -0
  679. package/packages/widget-lit/package.json +52 -0
  680. package/packages/widget-lit/src/api-client.ts +230 -0
  681. package/packages/widget-lit/src/api-client.ts.backup +218 -0
  682. package/packages/widget-lit/src/components/sea-chat.ts +382 -0
  683. package/packages/widget-lit/src/components/sea-submission-viewer.ts +267 -0
  684. package/packages/widget-lit/src/components/sea-widget.ts +317 -0
  685. package/packages/widget-lit/src/index.ts +48 -0
  686. package/packages/widget-lit/src/react.ts +58 -0
  687. package/packages/widget-lit/src/style.css +47 -0
  688. package/packages/widget-lit/tsconfig.json +24 -0
  689. package/packages/widget-lit/vite.config.ts +29 -0
  690. package/packages/widget-ng/DEVELOPMENT.md +74 -0
  691. package/packages/widget-ng/README.md +657 -0
  692. package/packages/widget-ng/dev.sh +14 -0
  693. package/packages/widget-ng/eslint.config.mjs +24 -0
  694. package/packages/widget-ng/ng-package.json +9 -0
  695. package/packages/widget-ng/package.json +85 -0
  696. package/packages/widget-ng/src/index.ts +45 -0
  697. package/packages/widget-ng/src/lib/components/sea-chat.component.ts +737 -0
  698. package/packages/widget-ng/src/lib/components/sea-data-viewer.component.ts +2240 -0
  699. package/packages/widget-ng/src/lib/components/sea-deal-form-modal.component.ts +702 -0
  700. package/packages/widget-ng/src/lib/components/sea-document-list.component.ts +350 -0
  701. package/packages/widget-ng/src/lib/components/sea-feedback-modal.component.ts +461 -0
  702. package/packages/widget-ng/src/lib/components/sea-file-upload.component.ts +655 -0
  703. package/packages/widget-ng/src/lib/components/sea-model-selection-modal.component.ts +367 -0
  704. package/packages/widget-ng/src/lib/components/sea-new-submission-modal.component.ts +414 -0
  705. package/packages/widget-ng/src/lib/components/sea-pdf-viewer.component.ts +869 -0
  706. package/packages/widget-ng/src/lib/components/sea-submission-card.component.ts +251 -0
  707. package/packages/widget-ng/src/lib/components/sea-widget.component.ts +684 -0
  708. package/packages/widget-ng/src/lib/models/submission.model.ts +170 -0
  709. package/packages/widget-ng/src/lib/pipes/markdown.pipe.ts +57 -0
  710. package/packages/widget-ng/src/lib/services/api-client.service.ts +715 -0
  711. package/packages/widget-ng/src/lib/services/chat.service.ts +330 -0
  712. package/packages/widget-ng/src/lib/services/config.service.ts +107 -0
  713. package/packages/widget-ng/src/web-component.ts +56 -0
  714. package/packages/widget-ng/tsconfig.json +25 -0
  715. package/packages/widget-ng/tsconfig.lib.json +9 -0
  716. package/packages/widget-ng/vite.config.elements.ts +26 -0
  717. package/packages/widget-ng/vitest.config.ts +19 -0
  718. package/packages/widget-ng/vitest.setup.ts +13 -0
  719. package/pnpm-workspace.yaml +18 -0
  720. package/render.yaml +136 -0
  721. package/scripts/README.md +57 -0
  722. package/scripts/package.json +22 -0
  723. package/scripts/python/.python-version +1 -0
  724. package/scripts/python/README.md +3 -0
  725. package/scripts/python/export-org-data.py +693 -0
  726. package/scripts/python/pyproject.toml +29 -0
  727. package/scripts/python/requirements-dev.lock +36 -0
  728. package/scripts/python/requirements.lock +36 -0
  729. package/scripts/python/src/gen.py +297 -0
  730. package/scripts/python/test.py +34 -0
  731. package/scripts/src/fix-storage-provider-mismatch.ts +239 -0
  732. package/scripts/src/sync-render-yaml.ts +290 -0
  733. package/scripts/src/test-chat-stream.ts +300 -0
  734. package/scripts/src/test-reconciliation.ts +230 -0
  735. package/scripts/tsconfig.json +15 -0
  736. package/tests/angular-test-app/.vscode/extensions.json +4 -0
  737. package/tests/angular-test-app/.vscode/launch.json +13 -0
  738. package/tests/angular-test-app/.vscode/tasks.json +24 -0
  739. package/tests/angular-test-app/README.md +59 -0
  740. package/tests/angular-test-app/angular.json +111 -0
  741. package/tests/angular-test-app/clean-start.sh +14 -0
  742. package/tests/angular-test-app/package.json +36 -0
  743. package/tests/angular-test-app/public/favicon.ico +0 -0
  744. package/tests/angular-test-app/src/app/app.component.ts +220 -0
  745. package/tests/angular-test-app/src/app/app.config.ts +5 -0
  746. package/tests/angular-test-app/src/env.d.ts +13 -0
  747. package/tests/angular-test-app/src/index.html +13 -0
  748. package/tests/angular-test-app/src/main.ts +6 -0
  749. package/tests/angular-test-app/src/styles.css +8 -0
  750. package/tests/angular-test-app/tsconfig.app.json +15 -0
  751. package/tests/angular-test-app/tsconfig.json +27 -0
  752. package/tests/crm-viewer-app/API_INTEGRATION_SUMMARY.md +295 -0
  753. package/tests/crm-viewer-app/CURRENT_ASSETS_FIELDS.md +148 -0
  754. package/tests/crm-viewer-app/FIELD_ID_MAPPING.md +206 -0
  755. package/tests/crm-viewer-app/INTEGRATION_GUIDE.md +309 -0
  756. package/tests/crm-viewer-app/README.md +174 -0
  757. package/tests/crm-viewer-app/REAL_API_INTEGRATION.md +240 -0
  758. package/tests/crm-viewer-app/UPDATED_IMPLEMENTATION.md +279 -0
  759. package/tests/crm-viewer-app/angular.json +114 -0
  760. package/tests/crm-viewer-app/package.json +35 -0
  761. package/tests/crm-viewer-app/src/app/app.component.ts +534 -0
  762. package/tests/crm-viewer-app/src/app/citation.service.ts +316 -0
  763. package/tests/crm-viewer-app/src/env.d.ts +16 -0
  764. package/tests/crm-viewer-app/src/index.html +19 -0
  765. package/tests/crm-viewer-app/src/main.ts +7 -0
  766. package/tests/crm-viewer-app/src/styles.css +409 -0
  767. package/tests/crm-viewer-app/src/template.html +2678 -0
  768. package/tests/crm-viewer-app/tsconfig.app.json +15 -0
  769. package/tests/crm-viewer-app/tsconfig.json +27 -0
  770. package/tests/e2e/package.json +17 -0
  771. package/tests/e2e/playwright.config.ts +75 -0
  772. package/tests/e2e/tests/api/health.spec.ts +10 -0
  773. package/tests/e2e/tests/app/example.spec.ts +10 -0
  774. package/tests/widget-test-app/.prettierignore +6 -0
  775. package/tests/widget-test-app/README.md +48 -0
  776. package/tests/widget-test-app/index.html +12 -0
  777. package/tests/widget-test-app/package.json +24 -0
  778. package/tests/widget-test-app/src/App.css +192 -0
  779. package/tests/widget-test-app/src/App.tsx +80 -0
  780. package/tests/widget-test-app/src/main.tsx +9 -0
  781. package/tests/widget-test-app/src/vite-env.d.ts +4 -0
  782. package/tests/widget-test-app/tsconfig.json +25 -0
  783. package/tests/widget-test-app/tsconfig.node.json +11 -0
  784. package/tests/widget-test-app/vite.config.ts +14 -0
@@ -0,0 +1,1396 @@
1
+ import { db, schema } from "@sea/db";
2
+ import type { SubmissionItemValue } from "@sea/db/schema";
3
+ import type { Member } from "@sea/db/types";
4
+ import { invariant, InvariantError } from "@sea/util/invariant";
5
+ import { and, asc, desc, eq, gt, inArray, isNull } from "drizzle-orm";
6
+ import { getForm } from "./form";
7
+
8
+ interface DenormItem {
9
+ id: string;
10
+ name: string;
11
+ type:
12
+ | "string"
13
+ | "number"
14
+ | "boolean"
15
+ | "date"
16
+ | "email"
17
+ | "string-long"
18
+ | "money"
19
+ | "url"
20
+ | "enum";
21
+ hint: string;
22
+ value: SubmissionItemValue | null;
23
+ citationId?: string | null;
24
+ }
25
+
26
+ interface DenormDerived {
27
+ id: string;
28
+ name: string;
29
+ prompt: string;
30
+ value: string | null;
31
+ }
32
+
33
+ export interface DenormGroup {
34
+ id: string;
35
+ formGroupId: string;
36
+ name: string;
37
+ multi: boolean;
38
+ hint: string;
39
+ items: DenormItem[];
40
+ derived: DenormDerived[];
41
+ }
42
+
43
+ export interface DenormTable {
44
+ formGroupId: string;
45
+ name: string;
46
+ multi: boolean;
47
+ hint: string;
48
+ columns: string[];
49
+ rows: {
50
+ items: DenormItem[];
51
+ derived: DenormDerived[];
52
+ }[];
53
+ }
54
+
55
+ export type DenormSubmission = {
56
+ id: string;
57
+ name: string | null;
58
+ summary: string | null;
59
+ formName: string;
60
+ hint: string;
61
+ groups: DenormGroup[];
62
+ tables: DenormTable[];
63
+ };
64
+
65
+ export async function getSubmission({
66
+ member,
67
+ submissionId,
68
+ typeFilter = "extraction",
69
+ }: {
70
+ member: Member;
71
+ submissionId: string;
72
+ typeFilter?: "extraction" | "transformation" | "all";
73
+ }) {
74
+ const conditions = [
75
+ eq(schema.submission.organizationId, member.organizationId),
76
+ eq(schema.submission.id, submissionId),
77
+ isNull(schema.submission.deletedAt),
78
+ ];
79
+
80
+ // Add type filter unless "all" is specified
81
+ if (typeFilter !== "all") {
82
+ conditions.push(eq(schema.submission.type, typeFilter));
83
+ }
84
+
85
+ const submission = await db.query.submission.findFirst({
86
+ where: and(...conditions),
87
+ });
88
+ invariant(submission, `no submission found for ${submissionId}`);
89
+ return submission;
90
+ }
91
+
92
+ export async function getSubmissionGroup({
93
+ member,
94
+ submissionId,
95
+ submissionGroupId,
96
+ }: {
97
+ member: Member;
98
+ submissionId: string;
99
+ submissionGroupId: string;
100
+ }) {
101
+ const submission = await db.query.submission.findFirst({
102
+ where: and(
103
+ eq(schema.submission.organizationId, member.organizationId),
104
+ eq(schema.submission.id, submissionId),
105
+ eq(schema.submission.type, "extraction"),
106
+ isNull(schema.submission.deletedAt),
107
+ ),
108
+ with: {
109
+ groups: {
110
+ where: eq(schema.submissionGroup.id, submissionGroupId),
111
+ orderBy: asc(schema.submissionGroup.order),
112
+ },
113
+ },
114
+ });
115
+ invariant(submission, `no submission found for ${submissionId}`);
116
+ const [group] = submission.groups;
117
+ invariant(group, `no submissionGroup found for ${submissionGroupId}`);
118
+
119
+ return group;
120
+ }
121
+
122
+ export async function getDenormSubmission({
123
+ member,
124
+ submissionId,
125
+ withCitations = false,
126
+ withTransformations = false,
127
+ }: {
128
+ member: Member;
129
+ submissionId: string;
130
+ withCitations?: boolean;
131
+ withTransformations?: boolean;
132
+ }): Promise<DenormSubmission> {
133
+ const typeFilter = withTransformations ? undefined : eq(schema.submission.type, "extraction");
134
+ const baseQuery = {
135
+ where: and(
136
+ eq(schema.submission.organizationId, member.organizationId),
137
+ eq(schema.submission.id, submissionId),
138
+ typeFilter,
139
+ isNull(schema.submission.deletedAt),
140
+ ),
141
+ };
142
+
143
+ const submission = withCitations
144
+ ? await db.query.submission.findFirst({
145
+ ...baseQuery,
146
+ with: {
147
+ form: true,
148
+ groups: {
149
+ orderBy: asc(schema.submissionGroup.order),
150
+ with: {
151
+ formGroup: true,
152
+ derived: {
153
+ with: {
154
+ formDerived: true,
155
+ },
156
+ },
157
+ items: {
158
+ with: {
159
+ formField: true,
160
+ versions: {
161
+ orderBy: [
162
+ desc(schema.submissionItemVersion.selected),
163
+ desc(schema.submissionItemVersion.updatedAt),
164
+ ],
165
+ with: { documentSource: true },
166
+ limit: 1,
167
+ },
168
+ },
169
+ },
170
+ },
171
+ },
172
+ },
173
+ })
174
+ : await db.query.submission.findFirst({
175
+ ...baseQuery,
176
+ with: {
177
+ form: true,
178
+ groups: {
179
+ orderBy: asc(schema.submissionGroup.order),
180
+ with: {
181
+ formGroup: true,
182
+ derived: {
183
+ with: {
184
+ formDerived: true,
185
+ },
186
+ },
187
+ items: {
188
+ with: {
189
+ formField: true,
190
+ versions: {
191
+ orderBy: [
192
+ desc(schema.submissionItemVersion.selected),
193
+ desc(schema.submissionItemVersion.updatedAt),
194
+ ],
195
+ limit: 1,
196
+ },
197
+ },
198
+ },
199
+ },
200
+ },
201
+ },
202
+ });
203
+
204
+ invariant(submission, `no submission found for ${submissionId}`);
205
+
206
+ const denorm = {
207
+ id: submission.id,
208
+ name: submission.name,
209
+ summary: submission.summary,
210
+ formName: submission.form.name,
211
+ hint: submission.form.hint,
212
+ groups: submission.groups.map((group) => ({
213
+ id: group.id,
214
+ formGroupId: group.formGroup.id,
215
+ name: group.formGroup.name,
216
+ multi: group.formGroup.multi,
217
+ hint: group.formGroup.hint,
218
+ order: group.formGroup.order,
219
+ items: group.items
220
+ .map((item) => {
221
+ const version = item.versions.at(0);
222
+ return {
223
+ id: item.id,
224
+ name: item.formField.name,
225
+ type: item.formField.type,
226
+ hint: item.formField.hint,
227
+ order: item.formField.order,
228
+ value: version?.value ?? null,
229
+ citationId:
230
+ withCitations && version && "documentSource" in version
231
+ ? (version.documentSource?.id ?? null)
232
+ : undefined,
233
+ };
234
+ })
235
+ .sort((a, b) => a.order - b.order),
236
+ derived: group.derived.map((item) => ({
237
+ id: item.id,
238
+ name: item.formDerived.name,
239
+ prompt: item.formDerived.prompt,
240
+ value: item.value,
241
+ })),
242
+ })),
243
+ };
244
+
245
+ const processedDenorm = {
246
+ ...denorm,
247
+ groups: denorm.groups.filter((g) => !g.multi).sort((a, b) => a.order - b.order),
248
+ tables: denorm.groups
249
+ .filter((g) => g.multi === true)
250
+ .sort((a, b) => a.order - b.order)
251
+ .reduce((acc, group) => {
252
+ const existing = acc.find((t) => t.formGroupId === group.formGroupId);
253
+ if (existing) {
254
+ existing.rows.push({ items: group.items, derived: group.derived });
255
+ } else {
256
+ acc.push({
257
+ ...group,
258
+ columns: [...group.items.map((i) => i.name), ...group.derived.map((d) => d.name)],
259
+ rows: [{ items: group.items, derived: group.derived }],
260
+ });
261
+ }
262
+ return acc;
263
+ }, [] as DenormTable[]),
264
+ };
265
+
266
+ return processedDenorm;
267
+ }
268
+
269
+ export async function getDenormSubmissions({
270
+ member,
271
+ submissionIds,
272
+ withCitations = false,
273
+ withTransformations = false,
274
+ }: {
275
+ member: Member;
276
+ submissionIds: string[];
277
+ withCitations?: boolean;
278
+ withTransformations?: boolean;
279
+ }): Promise<DenormSubmission[]> {
280
+ if (submissionIds.length === 0) {
281
+ return [];
282
+ }
283
+
284
+ const typeFilter = withTransformations ? undefined : eq(schema.submission.type, "extraction");
285
+ const baseQuery = {
286
+ where: and(
287
+ eq(schema.submission.organizationId, member.organizationId),
288
+ inArray(schema.submission.id, submissionIds),
289
+ typeFilter,
290
+ isNull(schema.submission.deletedAt),
291
+ ),
292
+ };
293
+
294
+ const submissions = withCitations
295
+ ? await db.query.submission.findMany({
296
+ ...baseQuery,
297
+ with: {
298
+ form: true,
299
+ groups: {
300
+ orderBy: asc(schema.submissionGroup.order),
301
+ with: {
302
+ formGroup: true,
303
+ derived: {
304
+ with: {
305
+ formDerived: true,
306
+ },
307
+ },
308
+ items: {
309
+ with: {
310
+ formField: true,
311
+ versions: {
312
+ orderBy: [
313
+ desc(schema.submissionItemVersion.selected),
314
+ desc(schema.submissionItemVersion.updatedAt),
315
+ ],
316
+ with: { documentSource: true },
317
+ limit: 1,
318
+ },
319
+ },
320
+ },
321
+ },
322
+ },
323
+ },
324
+ })
325
+ : await db.query.submission.findMany({
326
+ ...baseQuery,
327
+ with: {
328
+ form: true,
329
+ groups: {
330
+ orderBy: asc(schema.submissionGroup.order),
331
+ with: {
332
+ formGroup: true,
333
+ derived: {
334
+ with: {
335
+ formDerived: true,
336
+ },
337
+ },
338
+ items: {
339
+ with: {
340
+ formField: true,
341
+ versions: {
342
+ orderBy: [
343
+ desc(schema.submissionItemVersion.selected),
344
+ desc(schema.submissionItemVersion.updatedAt),
345
+ ],
346
+ limit: 1,
347
+ },
348
+ },
349
+ },
350
+ },
351
+ },
352
+ },
353
+ });
354
+
355
+ return submissions.map((submission) => {
356
+ const denorm = {
357
+ id: submission.id,
358
+ name: submission.name,
359
+ summary: submission.summary,
360
+ formName: submission.form.name,
361
+ hint: submission.form.hint,
362
+ groups: submission.groups.map((group) => ({
363
+ id: group.id,
364
+ formGroupId: group.formGroup.id,
365
+ name: group.formGroup.name,
366
+ multi: group.formGroup.multi,
367
+ hint: group.formGroup.hint,
368
+ order: group.formGroup.order,
369
+ items: group.items
370
+ .map((item) => {
371
+ const version = item.versions.at(0);
372
+ return {
373
+ id: item.id,
374
+ name: item.formField.name,
375
+ type: item.formField.type,
376
+ hint: item.formField.hint,
377
+ order: item.formField.order,
378
+ value: version?.value ?? null,
379
+ citationId:
380
+ withCitations && version && "documentSource" in version
381
+ ? (version.documentSource?.id ?? null)
382
+ : undefined,
383
+ };
384
+ })
385
+ .sort((a, b) => a.order - b.order),
386
+ derived: group.derived.map((item) => ({
387
+ id: item.id,
388
+ name: item.formDerived.name,
389
+ prompt: item.formDerived.prompt,
390
+ value: item.value,
391
+ })),
392
+ })),
393
+ };
394
+
395
+ const processedDenorm = {
396
+ ...denorm,
397
+ groups: denorm.groups.filter((g) => !g.multi).sort((a, b) => a.order - b.order),
398
+ tables: denorm.groups
399
+ .filter((g) => g.multi === true)
400
+ .sort((a, b) => a.order - b.order)
401
+ .reduce((acc, group) => {
402
+ const existing = acc.find((t) => t.formGroupId === group.formGroupId);
403
+ if (existing) {
404
+ existing.rows.push({ items: group.items, derived: group.derived });
405
+ } else {
406
+ acc.push({
407
+ ...group,
408
+ columns: [...group.items.map((i) => i.name), ...group.derived.map((d) => d.name)],
409
+ rows: [{ items: group.items, derived: group.derived }],
410
+ });
411
+ }
412
+ return acc;
413
+ }, [] as DenormTable[]),
414
+ };
415
+
416
+ return processedDenorm;
417
+ });
418
+ }
419
+
420
+ export async function getSubmissionWithData({
421
+ member,
422
+ submissionId,
423
+ typeFilter = "extraction",
424
+ }: {
425
+ member: Member;
426
+ submissionId: string;
427
+ typeFilter?: "extraction" | "transformation" | "all";
428
+ }) {
429
+ const conditions = [
430
+ eq(schema.submission.organizationId, member.organizationId),
431
+ eq(schema.submission.id, submissionId),
432
+ isNull(schema.submission.deletedAt),
433
+ ];
434
+
435
+ // Add type filter unless "all" is specified
436
+ if (typeFilter !== "all") {
437
+ conditions.push(eq(schema.submission.type, typeFilter));
438
+ }
439
+
440
+ const submission = await db.query.submission.findFirst({
441
+ where: and(...conditions),
442
+ with: {
443
+ extractions: {
444
+ with: { document: true, citations: true },
445
+ where: isNull(schema.documentExtraction.deletedAt),
446
+ },
447
+ deal: true,
448
+ form: {
449
+ with: {
450
+ workflow: true,
451
+ groups: {
452
+ with: {
453
+ fields: { orderBy: asc(schema.formField.order) },
454
+ derived: true,
455
+ },
456
+ orderBy: asc(schema.formGroup.order),
457
+ },
458
+ },
459
+ },
460
+ conversation: {
461
+ with: {
462
+ messages: {
463
+ orderBy: asc(schema.message.createdAt),
464
+ },
465
+ },
466
+ },
467
+ groups: {
468
+ with: {
469
+ derived: {
470
+ with: { formDerived: true },
471
+ },
472
+ items: {
473
+ with: {
474
+ formField: true,
475
+ versions: {
476
+ orderBy: [
477
+ desc(schema.submissionItemVersion.selected),
478
+ desc(schema.submissionItemVersion.updatedAt),
479
+ ],
480
+ with: { documentSource: true, messageSource: true },
481
+ limit: 1,
482
+ },
483
+ },
484
+ },
485
+ },
486
+ orderBy: [schema.submissionGroup.order],
487
+ },
488
+ },
489
+ });
490
+ return submission;
491
+ }
492
+
493
+ export async function getSubmissionWithExtractions({
494
+ member,
495
+ submissionId,
496
+ typeFilter = "extraction",
497
+ }: {
498
+ member: Member;
499
+ submissionId: string;
500
+ typeFilter?: "extraction" | "transformation" | "all";
501
+ }) {
502
+ const conditions = [
503
+ eq(schema.submission.organizationId, member.organizationId),
504
+ eq(schema.submission.id, submissionId),
505
+ isNull(schema.submission.deletedAt),
506
+ ];
507
+
508
+ // Add type filter unless "all" is specified
509
+ if (typeFilter !== "all") {
510
+ conditions.push(eq(schema.submission.type, typeFilter));
511
+ }
512
+
513
+ const submission = await db.query.submission.findFirst({
514
+ with: {
515
+ extractions: {
516
+ where: isNull(schema.documentExtraction.deletedAt),
517
+ },
518
+ },
519
+ where: and(...conditions),
520
+ });
521
+ invariant(submission);
522
+
523
+ return submission;
524
+ }
525
+
526
+ export async function getSubmissionWithFormAndExtractions({
527
+ member,
528
+ submissionId,
529
+ }: {
530
+ member: Member;
531
+ submissionId: string;
532
+ }) {
533
+ const submission = await db.query.submission.findFirst({
534
+ where: and(
535
+ eq(schema.submission.organizationId, member.organizationId),
536
+ eq(schema.submission.id, submissionId),
537
+ isNull(schema.submission.deletedAt),
538
+ ),
539
+ with: {
540
+ form: {
541
+ with: {
542
+ groups: {
543
+ with: {
544
+ fields: { orderBy: asc(schema.formField.order) },
545
+ derived: { orderBy: asc(schema.formDerived.name) },
546
+ },
547
+ orderBy: asc(schema.formGroup.order),
548
+ },
549
+ },
550
+ },
551
+ groups: {
552
+ with: {
553
+ derived: {
554
+ with: { formDerived: true },
555
+ },
556
+ items: {
557
+ with: {
558
+ formField: true,
559
+ versions: {
560
+ orderBy: [
561
+ desc(schema.submissionItemVersion.selected),
562
+ desc(schema.submissionItemVersion.updatedAt),
563
+ ],
564
+ with: {
565
+ documentSource: true,
566
+ messageSource: true,
567
+ },
568
+ limit: 1,
569
+ },
570
+ },
571
+ },
572
+ },
573
+ orderBy: asc(schema.submissionGroup.order),
574
+ },
575
+ extractions: {
576
+ with: {
577
+ document: true,
578
+ citations: true,
579
+ },
580
+ },
581
+ },
582
+ });
583
+
584
+ if (!submission) return null;
585
+
586
+ // Check if any item versions have transformation sources
587
+ // This can happen in any submission type (extraction or transformation)
588
+ const transformationItems: Array<{ itemId: string; versionId: string; sourceId: string }> = [];
589
+ for (const group of submission.groups) {
590
+ for (const item of group.items) {
591
+ const version = item.versions[0];
592
+ if (version?.sourceType === "transformation" && version.sourceId) {
593
+ transformationItems.push({
594
+ itemId: item.id,
595
+ versionId: version.id,
596
+ sourceId: version.sourceId,
597
+ });
598
+ }
599
+ }
600
+ }
601
+
602
+ const hasTransformationSources = transformationItems.length > 0;
603
+
604
+ console.log(
605
+ `[getSubmissionWithFormAndExtractions] Submission ${submissionId}: found ${transformationItems.length} items with transformation sources`,
606
+ );
607
+
608
+ if (hasTransformationSources) {
609
+ console.log(
610
+ `[getSubmissionWithFormAndExtractions] Loading transformation citations for ${transformationItems.length} items`,
611
+ );
612
+
613
+ // Collect initial transformation sourceIds from item versions
614
+ const initialTransformationIds: string[] = [];
615
+ for (const group of submission.groups) {
616
+ for (const item of group.items) {
617
+ const version = item.versions[0];
618
+ if (version?.sourceType === "transformation" && version.sourceId) {
619
+ initialTransformationIds.push(version.sourceId);
620
+ }
621
+ }
622
+ }
623
+
624
+ // Iteratively load all transformations and source versions to arbitrary depth
625
+ const allDataTransformations: Awaited<ReturnType<typeof db.query.dataTransformation.findMany>> =
626
+ [];
627
+ const allSourceVersions: Awaited<
628
+ ReturnType<
629
+ typeof db.query.submissionItemVersion.findMany<{
630
+ with: {
631
+ documentSource: true;
632
+ submissionItem: {
633
+ with: {
634
+ formField: true;
635
+ group: {
636
+ with: {
637
+ formGroup: true;
638
+ submission: {
639
+ with: {
640
+ form: true;
641
+ };
642
+ };
643
+ };
644
+ };
645
+ };
646
+ };
647
+ };
648
+ }>
649
+ >
650
+ > = [];
651
+
652
+ let transformationIdsToLoad = initialTransformationIds;
653
+ const loadedTransformationIds = new Set<string>();
654
+
655
+ while (transformationIdsToLoad.length > 0) {
656
+ // Load transformations
657
+ const transformations = await db.query.dataTransformation.findMany({
658
+ where: inArray(schema.dataTransformation.id, transformationIdsToLoad),
659
+ });
660
+
661
+ allDataTransformations.push(...transformations);
662
+ transformationIdsToLoad.forEach((id) => loadedTransformationIds.add(id));
663
+
664
+ // Get all source version IDs from these transformations
665
+ const sourceVersionIds = transformations.flatMap((dt) => dt.sourceItemVersionIds ?? []);
666
+
667
+ // Load source versions with full context
668
+ if (sourceVersionIds.length > 0) {
669
+ const sourceVersions = await db.query.submissionItemVersion.findMany({
670
+ where: inArray(schema.submissionItemVersion.id, sourceVersionIds),
671
+ with: {
672
+ documentSource: true,
673
+ submissionItem: {
674
+ with: {
675
+ formField: true,
676
+ group: {
677
+ with: {
678
+ formGroup: true,
679
+ submission: {
680
+ with: {
681
+ form: true,
682
+ },
683
+ },
684
+ },
685
+ },
686
+ },
687
+ },
688
+ },
689
+ });
690
+
691
+ allSourceVersions.push(...sourceVersions);
692
+
693
+ // Check for more nested transformations
694
+ transformationIdsToLoad = sourceVersions
695
+ .filter((v) => v.sourceType === "transformation" && v.sourceId)
696
+ .map((v) => v.sourceId!)
697
+ .filter((id) => !loadedTransformationIds.has(id));
698
+ } else {
699
+ transformationIdsToLoad = [];
700
+ }
701
+ }
702
+
703
+ // Build a map of version ID -> full version with item context
704
+ const versionIdToVersion = new Map(allSourceVersions.map((version) => [version.id, version]));
705
+
706
+ // Build a map of transformation ID -> source versions (with full item context)
707
+ const transformationToSourceVersions = new Map<string, typeof allSourceVersions>();
708
+ // Build a map of transformation ID -> description
709
+ const transformationToDescription = new Map<string, string | null>();
710
+ for (const dt of allDataTransformations) {
711
+ const versions: typeof allSourceVersions = [];
712
+ const missingVersionIds: string[] = [];
713
+ for (const versionId of dt.sourceItemVersionIds ?? []) {
714
+ const version = versionIdToVersion.get(versionId);
715
+ if (version) {
716
+ versions.push(version);
717
+ } else {
718
+ missingVersionIds.push(versionId);
719
+ }
720
+ }
721
+ transformationToSourceVersions.set(dt.id, versions);
722
+ transformationToDescription.set(dt.id, dt.description ?? null);
723
+
724
+ // Debug: Log transformation descriptions
725
+ if (dt.description) {
726
+ console.log(
727
+ `[getSubmissionWithFormAndExtractions] Transformation ${dt.id} has description: "${dt.description}"`,
728
+ );
729
+ } else {
730
+ console.log(
731
+ `[getSubmissionWithFormAndExtractions] Transformation ${dt.id} has NO description`,
732
+ );
733
+ }
734
+
735
+ if (missingVersionIds.length > 0) {
736
+ console.warn(
737
+ `[getSubmissionWithFormAndExtractions] Transformation ${dt.id} references ${missingVersionIds.length} missing version IDs: ${missingVersionIds.join(", ")}`,
738
+ );
739
+ }
740
+ if (versions.length === 0 && (dt.sourceItemVersionIds?.length ?? 0) > 0) {
741
+ console.warn(
742
+ `[getSubmissionWithFormAndExtractions] Transformation ${dt.id} has sourceItemVersionIds but NO versions were found!`,
743
+ );
744
+ }
745
+ }
746
+
747
+ // Helper function to recursively attach transformationSource to versions
748
+ const attachTransformationSource = (
749
+ version: (typeof allSourceVersions)[0],
750
+ ): (typeof allSourceVersions)[0] => {
751
+ if (version.sourceType === "transformation" && version.sourceId) {
752
+ const nestedSourceVersions = transformationToSourceVersions.get(version.sourceId) ?? [];
753
+ const transformationDescription = transformationToDescription.get(version.sourceId) ?? null;
754
+ // Recursively attach transformationSource to nested versions
755
+ const enrichedSourceVersions: typeof allSourceVersions = nestedSourceVersions.map((sv) =>
756
+ attachTransformationSource(sv),
757
+ );
758
+ return {
759
+ ...version,
760
+ transformationSource: enrichedSourceVersions,
761
+ transformationDescription,
762
+ } as typeof version & {
763
+ transformationSource: typeof enrichedSourceVersions;
764
+ transformationDescription: string | null;
765
+ };
766
+ }
767
+ return version;
768
+ };
769
+
770
+ // Transform groups and attach source item versions to transformations
771
+ const groups = submission.groups.map((group) => {
772
+ const items = group.items.map((item) => {
773
+ const version = item.versions[0];
774
+ let selectedVersion = version ?? null;
775
+
776
+ // If this version is from a transformation, attach the source item versions recursively
777
+ if (version?.sourceType === "transformation" && version.sourceId) {
778
+ const sourceVersions = transformationToSourceVersions.get(version.sourceId) ?? [];
779
+
780
+ if (sourceVersions.length === 0) {
781
+ console.warn(
782
+ `[getSubmissionWithFormAndExtractions] WARNING: Item ${item.id} (field: ${item.formField.name}) has sourceType=transformation with sourceId=${version.sourceId} but NO source versions found!`,
783
+ );
784
+ }
785
+
786
+ // Recursively attach transformationSource to nested versions
787
+ const enrichedSourceVersions = sourceVersions.map((sv) => attachTransformationSource(sv));
788
+
789
+ // Get the description for this transformation
790
+ const description = transformationToDescription.get(version.sourceId) ?? null;
791
+
792
+ selectedVersion = {
793
+ ...version,
794
+ transformationSource: enrichedSourceVersions,
795
+ transformationDescription: description,
796
+ } as typeof selectedVersion & {
797
+ transformationSource: typeof enrichedSourceVersions;
798
+ transformationDescription: string | null;
799
+ };
800
+ }
801
+
802
+ return {
803
+ ...item,
804
+ selectedVersion,
805
+ };
806
+ });
807
+ return {
808
+ ...group,
809
+ items: [...items].sort((a, b) => a.formField.order - b.formField.order),
810
+ };
811
+ });
812
+
813
+ return {
814
+ ...submission,
815
+ groups,
816
+ };
817
+ }
818
+
819
+ // No transformation sources, use the standard logic
820
+ const groups = submission.groups.map((group) => {
821
+ const items = group.items.map((item) => ({
822
+ ...item,
823
+ selectedVersion: item.versions[0] ?? null,
824
+ }));
825
+ return {
826
+ ...group,
827
+ items: [...items].sort((a, b) => a.formField.order - b.formField.order),
828
+ };
829
+ });
830
+
831
+ return {
832
+ ...submission,
833
+ groups,
834
+ };
835
+ }
836
+
837
+ export async function getSubmissions({
838
+ member,
839
+ formId,
840
+ dealId,
841
+ afterDate,
842
+ typeFilter = "extraction",
843
+ }: {
844
+ member: Member;
845
+ formId?: string;
846
+ dealId?: string;
847
+ afterDate?: string;
848
+ typeFilter?: "extraction" | "transformation" | "all";
849
+ }) {
850
+ const conditions = [
851
+ eq(schema.submission.organizationId, member.organizationId),
852
+ isNull(schema.submission.deletedAt),
853
+ ];
854
+
855
+ // Add type filter unless "all" is specified
856
+ if (typeFilter !== "all") {
857
+ conditions.push(eq(schema.submission.type, typeFilter));
858
+ }
859
+
860
+ if (formId) {
861
+ conditions.push(eq(schema.submission.formId, formId));
862
+ }
863
+ if (dealId) {
864
+ conditions.push(eq(schema.submission.dealId, dealId));
865
+ }
866
+ if (afterDate) {
867
+ conditions.push(gt(schema.submission.createdAt, new Date(afterDate)));
868
+ }
869
+ const submissions = await db.query.submission.findMany({
870
+ where: and(...conditions),
871
+ with: { form: { with: { workflow: true } } },
872
+ });
873
+ return submissions;
874
+ }
875
+
876
+ export async function syncSubmissionToForm({
877
+ member,
878
+ submissionId,
879
+ }: {
880
+ member: Member;
881
+ submissionId: string;
882
+ }) {
883
+ const submission = await getSubmission({ member, submissionId, typeFilter: "all" });
884
+ await db.transaction(async (tx) => {
885
+ // insert all missing non-multi groups, if any
886
+ const missingGroupIds = await tx
887
+ .select({ id: schema.formGroup.id })
888
+ .from(schema.formGroup)
889
+ .leftJoin(
890
+ schema.submissionGroup,
891
+ and(
892
+ eq(schema.submissionGroup.formGroupId, schema.formGroup.id),
893
+ eq(schema.submissionGroup.submissionId, submissionId),
894
+ ),
895
+ )
896
+ .where(
897
+ and(
898
+ eq(schema.formGroup.formId, submission.formId),
899
+ eq(schema.formGroup.multi, false),
900
+ isNull(schema.submissionGroup.id),
901
+ ),
902
+ );
903
+
904
+ if (missingGroupIds.length > 0) {
905
+ await tx.insert(schema.submissionGroup).values(
906
+ missingGroupIds.map(({ id }) => ({
907
+ submissionId,
908
+ formGroupId: id,
909
+ order: 0,
910
+ })),
911
+ );
912
+ }
913
+
914
+ // insert all missing submissionItems, if any
915
+ const missingItems = await tx
916
+ .select({
917
+ submissionGroupId: schema.submissionGroup.id,
918
+ formFieldId: schema.formField.id,
919
+ })
920
+ .from(schema.submissionGroup)
921
+ .innerJoin(
922
+ schema.formField,
923
+ eq(schema.formField.formGroupId, schema.submissionGroup.formGroupId),
924
+ )
925
+ .leftJoin(
926
+ schema.submissionItem,
927
+ and(
928
+ eq(schema.submissionItem.submissionGroupId, schema.submissionGroup.id),
929
+ eq(schema.submissionItem.formFieldId, schema.formField.id),
930
+ ),
931
+ )
932
+ .where(
933
+ and(
934
+ eq(schema.submissionGroup.submissionId, submissionId),
935
+ isNull(schema.submissionItem.id), // doesn't exist yet
936
+ ),
937
+ );
938
+
939
+ if (missingItems.length > 0) {
940
+ await tx.insert(schema.submissionItem).values(missingItems);
941
+ }
942
+
943
+ // insert all missing submissionDeriveds, if any
944
+ const missingDeriveds = await tx
945
+ .select({
946
+ submissionGroupId: schema.submissionGroup.id,
947
+ formDerivedId: schema.formDerived.id,
948
+ })
949
+ .from(schema.submissionGroup)
950
+ .innerJoin(
951
+ schema.formDerived,
952
+ eq(schema.formDerived.formGroupId, schema.submissionGroup.formGroupId),
953
+ )
954
+ .leftJoin(
955
+ schema.submissionDerived,
956
+ and(
957
+ eq(schema.submissionDerived.submissionGroupId, schema.submissionGroup.id),
958
+ eq(schema.submissionDerived.formDerivedId, schema.formDerived.id),
959
+ ),
960
+ )
961
+ .where(
962
+ and(
963
+ eq(schema.submissionGroup.submissionId, submissionId),
964
+ isNull(schema.submissionDerived.id),
965
+ ),
966
+ );
967
+
968
+ if (missingDeriveds.length > 0) {
969
+ await tx.insert(schema.submissionDerived).values(missingDeriveds);
970
+ }
971
+ });
972
+ }
973
+
974
+ export async function resetSubmission({
975
+ member,
976
+ submissionId,
977
+ }: {
978
+ member: Member;
979
+ submissionId: string;
980
+ }) {
981
+ try {
982
+ // Validate access
983
+ const submission = await getSubmission({ member, submissionId });
984
+
985
+ await db.transaction(async (tx) => {
986
+ // Delete all item versions that relate to the current submission
987
+ await tx
988
+ .delete(schema.submissionItemVersion)
989
+ .where(
990
+ inArray(
991
+ schema.submissionItemVersion.submissionItemId,
992
+ tx
993
+ .select({ id: schema.submissionItem.id })
994
+ .from(schema.submissionItem)
995
+ .innerJoin(
996
+ schema.submissionGroup,
997
+ eq(schema.submissionItem.submissionGroupId, schema.submissionGroup.id),
998
+ )
999
+ .where(eq(schema.submissionGroup.submissionId, submission.id)),
1000
+ ),
1001
+ );
1002
+
1003
+ // Reset derived fields
1004
+ await tx
1005
+ .update(schema.submissionDerived)
1006
+ .set({ value: null })
1007
+ .where(
1008
+ inArray(
1009
+ schema.submissionDerived.id,
1010
+ tx
1011
+ .select({ id: schema.submissionDerived.id })
1012
+ .from(schema.submissionDerived)
1013
+ .innerJoin(
1014
+ schema.submissionGroup,
1015
+ eq(schema.submissionDerived.submissionGroupId, schema.submissionGroup.id),
1016
+ )
1017
+ .where(eq(schema.submissionGroup.submissionId, submission.id)),
1018
+ ),
1019
+ );
1020
+
1021
+ // Delete all multisubmission groups for this submission with org validation
1022
+ await tx.delete(schema.submissionGroup).where(
1023
+ inArray(
1024
+ schema.submissionGroup.id,
1025
+ tx
1026
+ .select({ id: schema.submissionGroup.id })
1027
+ .from(schema.submissionGroup)
1028
+ .innerJoin(
1029
+ schema.submission,
1030
+ and(
1031
+ eq(schema.submissionGroup.submissionId, schema.submission.id),
1032
+ eq(schema.submission.id, submission.id),
1033
+ ),
1034
+ )
1035
+ .innerJoin(
1036
+ schema.formGroup,
1037
+ eq(schema.submissionGroup.formGroupId, schema.formGroup.id),
1038
+ )
1039
+ .where(eq(schema.formGroup.multi, true)),
1040
+ ),
1041
+ );
1042
+ });
1043
+ } catch (error) {
1044
+ console.error("Failed to reset submission");
1045
+ if (!(error instanceof InvariantError)) throw error;
1046
+ }
1047
+ }
1048
+
1049
+ export async function updateSubmission({
1050
+ member,
1051
+ submissionId,
1052
+ name,
1053
+ summary,
1054
+ }: {
1055
+ member: Member;
1056
+ submissionId: string;
1057
+ name?: string;
1058
+ summary?: string;
1059
+ }) {
1060
+ // Validate access
1061
+ const submission = await getSubmission({ member, submissionId });
1062
+ const updateData: { name?: string; summary?: string; nameAutoGenerated?: boolean } = {};
1063
+ if (name !== undefined) {
1064
+ updateData.name = name;
1065
+ updateData.nameAutoGenerated = false;
1066
+ }
1067
+ if (summary !== undefined) {
1068
+ updateData.summary = summary;
1069
+ }
1070
+ const [updatedSubmission] = await db
1071
+ .update(schema.submission)
1072
+ .set(updateData)
1073
+ .where(eq(schema.submission.id, submission.id))
1074
+ .returning();
1075
+ invariant(updatedSubmission, `failed to update submission ${submissionId}`);
1076
+
1077
+ return updatedSubmission;
1078
+ }
1079
+
1080
+ /**
1081
+ * Parses a documentCitation path (e.g. "Transactions[2].Credit" or "Bank Statement.BankName")
1082
+ * and sets the given value at that location in the extraction data JSON blob.
1083
+ * Returns the mutated data object, or the original if the path cannot be resolved.
1084
+ */
1085
+ function applyValueToExtractionData(
1086
+ data: Record<string, unknown>,
1087
+ path: string,
1088
+ value: SubmissionItemValue | null,
1089
+ ): Record<string, unknown> {
1090
+ // Path format: "<groupName>[<index>].<fieldName>" or "<groupName>.<fieldName>"
1091
+ const multiMatch = path.match(/^(.+)\[(\d+)\]\.(.+)$/);
1092
+ const singleMatch = !multiMatch ? path.match(/^(.+)\.(.+)$/) : null;
1093
+
1094
+ if (multiMatch) {
1095
+ const groupName = multiMatch[1]!;
1096
+ const index = parseInt(multiMatch[2]!, 10);
1097
+ const fieldName = multiMatch[3]!;
1098
+ const group = data[groupName];
1099
+ if (Array.isArray(group) && group[index] != null && typeof group[index] === "object") {
1100
+ (group[index] as Record<string, unknown>)[fieldName] = value;
1101
+ }
1102
+ } else if (singleMatch) {
1103
+ const groupName = singleMatch[1]!;
1104
+ const fieldName = singleMatch[2]!;
1105
+ const group = data[groupName];
1106
+ if (group != null && typeof group === "object" && !Array.isArray(group)) {
1107
+ (group as Record<string, unknown>)[fieldName] = value;
1108
+ }
1109
+ }
1110
+
1111
+ return data;
1112
+ }
1113
+
1114
+ /**
1115
+ * After a manual value update, find the most recent document-sourced version for the item,
1116
+ * follow its citation to the linked documentExtraction, and patch the value in the data JSON
1117
+ * using the citation path as a guide.
1118
+ */
1119
+ async function syncValueToDocumentExtraction(
1120
+ submissionItemId: string,
1121
+ value: SubmissionItemValue | null,
1122
+ ): Promise<void> {
1123
+ // Find the most recent version sourced from a document
1124
+ const documentVersion = await db.query.submissionItemVersion.findFirst({
1125
+ where: and(
1126
+ eq(schema.submissionItemVersion.submissionItemId, submissionItemId),
1127
+ eq(schema.submissionItemVersion.sourceType, "document"),
1128
+ ),
1129
+ orderBy: desc(schema.submissionItemVersion.createdAt),
1130
+ });
1131
+
1132
+ if (!documentVersion?.sourceId) return;
1133
+
1134
+ // Resolve the citation and its parent extraction
1135
+ const citation = await db.query.documentCitation.findFirst({
1136
+ where: eq(schema.documentCitation.id, documentVersion.sourceId),
1137
+ });
1138
+
1139
+ if (!citation) return;
1140
+
1141
+ const extraction = await db.query.documentExtraction.findFirst({
1142
+ where: eq(schema.documentExtraction.id, citation.documentExtractionId),
1143
+ });
1144
+
1145
+ if (!extraction?.data) return;
1146
+
1147
+ const updatedData = applyValueToExtractionData(
1148
+ extraction.data as Record<string, unknown>,
1149
+ citation.path,
1150
+ value,
1151
+ );
1152
+
1153
+ await db
1154
+ .update(schema.documentExtraction)
1155
+ .set({ data: updatedData })
1156
+ .where(eq(schema.documentExtraction.id, extraction.id));
1157
+ }
1158
+
1159
+ export async function updateSubmissionValue({
1160
+ member,
1161
+ submissionItemId,
1162
+ value,
1163
+ }: {
1164
+ member: Member;
1165
+ submissionItemId: string;
1166
+ value: SubmissionItemValue | null;
1167
+ }) {
1168
+ const submissionItem = await db.query.submissionItem.findFirst({
1169
+ with: { group: { with: { submission: true } } },
1170
+ where: eq(schema.submissionItem.id, submissionItemId),
1171
+ });
1172
+ invariant(submissionItem, `no formField found for ${submissionItemId}`);
1173
+
1174
+ // Validate access
1175
+ if (submissionItem.group?.submission.organizationId !== member.organizationId) {
1176
+ throw new InvariantError(`no submission found containing submissionItem ${submissionItemId}`);
1177
+ }
1178
+
1179
+ await db.transaction(async (tx) => {
1180
+ await tx
1181
+ .update(schema.submissionItemVersion)
1182
+ .set({ selected: false })
1183
+ .where(eq(schema.submissionItemVersion.submissionItemId, submissionItem.id));
1184
+
1185
+ await tx.insert(schema.submissionItemVersion).values({
1186
+ sourceType: "manual",
1187
+ submissionItemId: submissionItem.id,
1188
+ value,
1189
+ selected: true,
1190
+ });
1191
+ });
1192
+
1193
+ // // Best-effort: patch the linked documentExtraction data JSON to reflect the manual edit.
1194
+ // // Errors here are non-fatal — the item version has already been persisted above.
1195
+ // await syncValueToDocumentExtraction(submissionItemId, value).catch((err) => {
1196
+ // console.error(
1197
+ // `[updateSubmissionValue] failed to sync value to documentExtraction for item ${submissionItemId}:`,
1198
+ // err,
1199
+ // );
1200
+ // });
1201
+ }
1202
+
1203
+ export async function addSubmissionGroupRow({
1204
+ member,
1205
+ submissionId,
1206
+ formGroupId,
1207
+ }: {
1208
+ member: Member;
1209
+ submissionId: string;
1210
+ formGroupId: string;
1211
+ }) {
1212
+ const formGroup = await db.query.formGroup.findFirst({
1213
+ where: and(eq(schema.formGroup.id, formGroupId), eq(schema.formGroup.multi, true)),
1214
+ });
1215
+ invariant(formGroup, `no formGroup found for ${formGroupId}`);
1216
+
1217
+ // Verify the submission exists and belongs to the user
1218
+ await getForm({ member, formId: formGroup.formId });
1219
+ const submission = await getSubmission({ member, submissionId });
1220
+ invariant(submission, `no submission found for ${submissionId}`);
1221
+
1222
+ const submissionGroup = await db.transaction(async (tx) => {
1223
+ const existingRows = await tx
1224
+ .select({ order: schema.submissionGroup.order })
1225
+ .from(schema.submissionGroup)
1226
+ .where(
1227
+ and(
1228
+ eq(schema.submissionGroup.submissionId, submissionId),
1229
+ eq(schema.submissionGroup.formGroupId, formGroup.id),
1230
+ ),
1231
+ )
1232
+ .for("update");
1233
+ const order =
1234
+ existingRows.length > 0 ? Math.max(...existingRows.map((row) => row.order)) + 1 : 0;
1235
+
1236
+ const [submissionGroup] = await tx
1237
+ .insert(schema.submissionGroup)
1238
+ .values({
1239
+ submissionId,
1240
+ formGroupId: formGroup.id,
1241
+ order,
1242
+ })
1243
+ .returning();
1244
+ invariant(submissionGroup);
1245
+ return submissionGroup;
1246
+ });
1247
+ return submissionGroup;
1248
+ }
1249
+
1250
+ export async function updateSubmissionGroup({
1251
+ member,
1252
+ submissionId,
1253
+ submissionGroupId,
1254
+ isValid,
1255
+ order,
1256
+ }: {
1257
+ member: Member;
1258
+ submissionId: string;
1259
+ submissionGroupId: string;
1260
+ isValid?: boolean;
1261
+ order?: number;
1262
+ }) {
1263
+ // Verify the submission group belongs to this submission and user has access
1264
+ const [submissionGroup] = await db
1265
+ .select({ id: schema.submissionGroup.id })
1266
+ .from(schema.submissionGroup)
1267
+ .leftJoin(schema.submission, eq(schema.submission.id, schema.submissionGroup.submissionId))
1268
+ .where(
1269
+ and(
1270
+ eq(schema.submissionGroup.id, submissionGroupId),
1271
+ eq(schema.submissionGroup.submissionId, submissionId),
1272
+ eq(schema.submission.organizationId, member.organizationId),
1273
+ eq(schema.submission.type, "extraction"),
1274
+ isNull(schema.submission.deletedAt),
1275
+ ),
1276
+ );
1277
+ invariant(submissionGroup, `no submissionGroup found for ${submissionGroupId}`);
1278
+
1279
+ const updateData: { isValid?: boolean; order?: number } = {};
1280
+ if (isValid !== undefined) updateData.isValid = isValid;
1281
+ if (order !== undefined) updateData.order = order;
1282
+
1283
+ const [updated] = await db
1284
+ .update(schema.submissionGroup)
1285
+ .set(updateData)
1286
+ .where(eq(schema.submissionGroup.id, submissionGroupId))
1287
+ .returning();
1288
+
1289
+ invariant(updated, `failed to update submissionGroup ${submissionGroupId}`);
1290
+ return updated;
1291
+ }
1292
+
1293
+ export async function createSubmissionWithConversation({
1294
+ member,
1295
+ data,
1296
+ }: {
1297
+ member: Member;
1298
+ data: {
1299
+ formId: string;
1300
+ dealId: string;
1301
+ name?: string;
1302
+ status?: "draft" | "final";
1303
+ type?: "extraction" | "transformation";
1304
+ };
1305
+ }) {
1306
+ const form = await db.query.form.findFirst({
1307
+ where: and(
1308
+ eq(schema.form.id, data.formId),
1309
+ eq(schema.form.organizationId, member.organizationId),
1310
+ isNull(schema.form.deletedAt),
1311
+ ),
1312
+ });
1313
+ invariant(form);
1314
+
1315
+ const deal = await db.query.deal.findFirst({
1316
+ where: and(
1317
+ eq(schema.deal.organizationId, member.organizationId),
1318
+ eq(schema.deal.id, data.dealId),
1319
+ isNull(schema.deal.deletedAt),
1320
+ ),
1321
+ });
1322
+ invariant(deal);
1323
+
1324
+ const createdSubmission = await db.transaction(async (tx) => {
1325
+ const [conversation] = await tx
1326
+ .insert(schema.conversation)
1327
+ .values({ organizationId: member.organizationId })
1328
+ .returning();
1329
+ invariant(conversation);
1330
+
1331
+ const [createdSubmission] = await tx
1332
+ .insert(schema.submission)
1333
+ .values({
1334
+ organizationId: member.organizationId,
1335
+ formId: data.formId,
1336
+ conversationId: conversation.id,
1337
+ dealId: deal.id,
1338
+ name: data.name,
1339
+ nameAutoGenerated: !data.name,
1340
+ status: data.status,
1341
+ type: data.type,
1342
+ })
1343
+ .returning();
1344
+ invariant(createdSubmission);
1345
+ return createdSubmission;
1346
+ });
1347
+
1348
+ return createdSubmission;
1349
+ }
1350
+
1351
+ /**
1352
+ * Get submissions with their extractions
1353
+ */
1354
+ export async function getSubmissionsWithExtractions({
1355
+ member,
1356
+ submissionIds,
1357
+ }: {
1358
+ member: Member;
1359
+ submissionIds: string[];
1360
+ }): Promise<
1361
+ Array<{
1362
+ id: string;
1363
+ extractions: Array<{
1364
+ id: string;
1365
+ status: string;
1366
+ deletedAt: Date | null;
1367
+ }>;
1368
+ }>
1369
+ > {
1370
+ if (submissionIds.length === 0) {
1371
+ return [];
1372
+ }
1373
+
1374
+ const submissions = await db.query.submission.findMany({
1375
+ where: and(
1376
+ eq(schema.submission.organizationId, member.organizationId),
1377
+ inArray(schema.submission.id, submissionIds),
1378
+ isNull(schema.submission.deletedAt),
1379
+ ),
1380
+ columns: {
1381
+ id: true,
1382
+ },
1383
+ with: {
1384
+ extractions: {
1385
+ where: isNull(schema.documentExtraction.deletedAt),
1386
+ columns: {
1387
+ id: true,
1388
+ status: true,
1389
+ deletedAt: true,
1390
+ },
1391
+ },
1392
+ },
1393
+ });
1394
+
1395
+ return submissions;
1396
+ }