@open-mercato/core 0.6.5-develop.5382.1.f542de69af → 0.6.5

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 (237) hide show
  1. package/.turbo/turbo-build.log +1 -1
  2. package/dist/bootstrap.js +46 -6
  3. package/dist/bootstrap.js.map +2 -2
  4. package/dist/generated/entities/organization/index.js +2 -0
  5. package/dist/generated/entities/organization/index.js.map +2 -2
  6. package/dist/generated/entity-fields-registry.js +1 -0
  7. package/dist/generated/entity-fields-registry.js.map +2 -2
  8. package/dist/helpers/integration/crmFixtures.js +4 -0
  9. package/dist/helpers/integration/crmFixtures.js.map +2 -2
  10. package/dist/modules/attachments/api/route.js +2 -0
  11. package/dist/modules/attachments/api/route.js.map +2 -2
  12. package/dist/modules/attachments/lib/access.js +18 -0
  13. package/dist/modules/attachments/lib/access.js.map +2 -2
  14. package/dist/modules/audit_logs/data/entities.js +2 -1
  15. package/dist/modules/audit_logs/data/entities.js.map +2 -2
  16. package/dist/modules/audit_logs/migrations/Migration20260611104500.js +13 -0
  17. package/dist/modules/audit_logs/migrations/Migration20260611104500.js.map +7 -0
  18. package/dist/modules/audit_logs/services/accessLogService.js +10 -0
  19. package/dist/modules/audit_logs/services/accessLogService.js.map +2 -2
  20. package/dist/modules/auth/api/admin/nav.js +9 -0
  21. package/dist/modules/auth/api/admin/nav.js.map +2 -2
  22. package/dist/modules/auth/api/login.js +4 -13
  23. package/dist/modules/auth/api/login.js.map +2 -2
  24. package/dist/modules/auth/data/entities.js +3 -1
  25. package/dist/modules/auth/data/entities.js.map +2 -2
  26. package/dist/modules/auth/lib/backendChrome.js +35 -2
  27. package/dist/modules/auth/lib/backendChrome.js.map +2 -2
  28. package/dist/modules/auth/lib/consentIntegrity.js +3 -3
  29. package/dist/modules/auth/lib/consentIntegrity.js.map +2 -2
  30. package/dist/modules/auth/migrations/Migration20260611103000.js +15 -0
  31. package/dist/modules/auth/migrations/Migration20260611103000.js.map +7 -0
  32. package/dist/modules/auth/services/authService.js +5 -3
  33. package/dist/modules/auth/services/authService.js.map +2 -2
  34. package/dist/modules/auth/services/rbacService.js +3 -2
  35. package/dist/modules/auth/services/rbacService.js.map +2 -2
  36. package/dist/modules/customer_accounts/backend/customer_accounts/settings/domain/components/Diagnostics.js +0 -3
  37. package/dist/modules/customer_accounts/backend/customer_accounts/settings/domain/components/Diagnostics.js.map +2 -2
  38. package/dist/modules/customers/api/deals/route.js +43 -2
  39. package/dist/modules/customers/api/deals/route.js.map +2 -2
  40. package/dist/modules/customers/api/deals/summary/route.js +402 -0
  41. package/dist/modules/customers/api/deals/summary/route.js.map +7 -0
  42. package/dist/modules/customers/backend/customers/deals/[id]/hooks/useDealActivities.js +16 -5
  43. package/dist/modules/customers/backend/customers/deals/[id]/hooks/useDealActivities.js.map +2 -2
  44. package/dist/modules/customers/backend/customers/deals/[id]/hooks/useDealData.js +22 -5
  45. package/dist/modules/customers/backend/customers/deals/[id]/hooks/useDealData.js.map +2 -2
  46. package/dist/modules/customers/backend/customers/deals/[id]/page.js +12 -2
  47. package/dist/modules/customers/backend/customers/deals/[id]/page.js.map +2 -2
  48. package/dist/modules/customers/backend/customers/deals/page.js +221 -56
  49. package/dist/modules/customers/backend/customers/deals/page.js.map +3 -3
  50. package/dist/modules/customers/backend/customers/deals/pipeline/page.js +1 -1
  51. package/dist/modules/customers/backend/customers/deals/pipeline/page.js.map +2 -2
  52. package/dist/modules/customers/backend/customers/people-v2/[id]/page.js +18 -0
  53. package/dist/modules/customers/backend/customers/people-v2/[id]/page.js.map +2 -2
  54. package/dist/modules/customers/cli.js +15 -9
  55. package/dist/modules/customers/cli.js.map +2 -2
  56. package/dist/modules/customers/components/DealsKpiStrip.js +282 -0
  57. package/dist/modules/customers/components/DealsKpiStrip.js.map +7 -0
  58. package/dist/modules/customers/components/detail/ConfirmDealLostDialog.js +0 -1
  59. package/dist/modules/customers/components/detail/ConfirmDealLostDialog.js.map +2 -2
  60. package/dist/modules/customers/components/detail/DealForm.js +100 -17
  61. package/dist/modules/customers/components/detail/DealForm.js.map +2 -2
  62. package/dist/modules/customers/components/detail/PersonDetailTabs.js +11 -3
  63. package/dist/modules/customers/components/detail/PersonDetailTabs.js.map +2 -2
  64. package/dist/modules/customers/components/detail/ScheduleActivityDialog.js +1 -2
  65. package/dist/modules/customers/components/detail/ScheduleActivityDialog.js.map +2 -2
  66. package/dist/modules/customers/components/kpi/PipelineStageBar.js +63 -0
  67. package/dist/modules/customers/components/kpi/PipelineStageBar.js.map +7 -0
  68. package/dist/modules/customers/lib/dealsMetrics.js +82 -0
  69. package/dist/modules/customers/lib/dealsMetrics.js.map +7 -0
  70. package/dist/modules/directory/api/organization-branding/route.js +214 -0
  71. package/dist/modules/directory/api/organization-branding/route.js.map +7 -0
  72. package/dist/modules/directory/api/organizations/route.js +7 -0
  73. package/dist/modules/directory/api/organizations/route.js.map +3 -3
  74. package/dist/modules/directory/backend/directory/branding/page.js +214 -0
  75. package/dist/modules/directory/backend/directory/branding/page.js.map +7 -0
  76. package/dist/modules/directory/backend/directory/branding/page.meta.js +26 -0
  77. package/dist/modules/directory/backend/directory/branding/page.meta.js.map +7 -0
  78. package/dist/modules/directory/commands/organizations.js +8 -1
  79. package/dist/modules/directory/commands/organizations.js.map +2 -2
  80. package/dist/modules/directory/data/entities.js +3 -0
  81. package/dist/modules/directory/data/entities.js.map +2 -2
  82. package/dist/modules/directory/data/validators.js +9 -0
  83. package/dist/modules/directory/data/validators.js.map +2 -2
  84. package/dist/modules/directory/migrations/Migration20260607222259_directory.js +13 -0
  85. package/dist/modules/directory/migrations/Migration20260607222259_directory.js.map +7 -0
  86. package/dist/modules/directory/subscribers/invalidateOrgScopeCache.js +2 -1
  87. package/dist/modules/directory/subscribers/invalidateOrgScopeCache.js.map +2 -2
  88. package/dist/modules/directory/utils/organizationScope.js +59 -27
  89. package/dist/modules/directory/utils/organizationScope.js.map +2 -2
  90. package/dist/modules/entities/api/definitions.batch.js +2 -1
  91. package/dist/modules/entities/api/definitions.batch.js.map +2 -2
  92. package/dist/modules/entities/api/entities.js +7 -0
  93. package/dist/modules/entities/api/entities.js.map +2 -2
  94. package/dist/modules/entities/api/records.js +26 -15
  95. package/dist/modules/entities/api/records.js.map +2 -2
  96. package/dist/modules/entities/backend/entities/user/[entityId]/records/[recordId]/page.js +14 -0
  97. package/dist/modules/entities/backend/entities/user/[entityId]/records/[recordId]/page.js.map +2 -2
  98. package/dist/modules/entities/backend/entities/user/[entityId]/records/create/page.js +14 -0
  99. package/dist/modules/entities/backend/entities/user/[entityId]/records/create/page.js.map +2 -2
  100. package/dist/modules/entities/backend/entities/user/[entityId]/records/page.js +12 -0
  101. package/dist/modules/entities/backend/entities/user/[entityId]/records/page.js.map +2 -2
  102. package/dist/modules/entities/components/useRecordsEntityGuard.js +30 -0
  103. package/dist/modules/entities/components/useRecordsEntityGuard.js.map +7 -0
  104. package/dist/modules/query_index/data/entities.js +2 -1
  105. package/dist/modules/query_index/data/entities.js.map +2 -2
  106. package/dist/modules/query_index/lib/engine.js +4 -2
  107. package/dist/modules/query_index/lib/engine.js.map +2 -2
  108. package/dist/modules/query_index/migrations/Migration20260611103000_query_index.js +16 -0
  109. package/dist/modules/query_index/migrations/Migration20260611103000_query_index.js.map +7 -0
  110. package/dist/modules/sales/commands/documents.js +7 -5
  111. package/dist/modules/sales/commands/documents.js.map +2 -2
  112. package/dist/modules/sales/components/documents/SalesDocumentsTable.js +2 -1
  113. package/dist/modules/sales/components/documents/SalesDocumentsTable.js.map +2 -2
  114. package/dist/modules/sales/components/documents/salesDocumentsColumns.js +10 -0
  115. package/dist/modules/sales/components/documents/salesDocumentsColumns.js.map +7 -0
  116. package/dist/modules/staff/api/team-members.js +9 -2
  117. package/dist/modules/staff/api/team-members.js.map +2 -2
  118. package/dist/modules/staff/api/timesheets/time-entries/[id]/timer-start/route.js +24 -1
  119. package/dist/modules/staff/api/timesheets/time-entries/[id]/timer-start/route.js.map +2 -2
  120. package/dist/modules/staff/backend/staff/team-members/[id]/page.js +11 -6
  121. package/dist/modules/staff/backend/staff/team-members/[id]/page.js.map +2 -2
  122. package/dist/modules/staff/commands/team-members.js +1 -1
  123. package/dist/modules/staff/commands/team-members.js.map +2 -2
  124. package/dist/modules/staff/components/TeamMemberForm.js +1 -1
  125. package/dist/modules/staff/components/TeamMemberForm.js.map +2 -2
  126. package/dist/modules/staff/lib/scheduleSwitch.js +23 -0
  127. package/dist/modules/staff/lib/scheduleSwitch.js.map +7 -0
  128. package/dist/modules/workflows/backend/definitions/create/page.js +1 -2
  129. package/dist/modules/workflows/backend/definitions/create/page.js.map +2 -2
  130. package/dist/modules/workflows/backend/definitions/visual-editor/page.js +1 -2
  131. package/dist/modules/workflows/backend/definitions/visual-editor/page.js.map +2 -2
  132. package/dist/modules/workflows/components/DefinitionTriggersEditor.js +1 -2
  133. package/dist/modules/workflows/components/DefinitionTriggersEditor.js.map +2 -2
  134. package/dist/modules/workflows/components/NodeEditDialog.js +4 -13
  135. package/dist/modules/workflows/components/NodeEditDialog.js.map +2 -2
  136. package/dist/modules/workflows/components/NodeEditDialogCrudForm.js +4 -13
  137. package/dist/modules/workflows/components/NodeEditDialogCrudForm.js.map +2 -2
  138. package/dist/modules/workflows/components/WorkflowGraphImpl.js +1 -4
  139. package/dist/modules/workflows/components/WorkflowGraphImpl.js.map +2 -2
  140. package/dist/modules/workflows/components/fields/FormFieldArrayEditor.js +2 -5
  141. package/dist/modules/workflows/components/fields/FormFieldArrayEditor.js.map +2 -2
  142. package/generated/entities/organization/index.ts +1 -0
  143. package/generated/entity-fields-registry.ts +1 -0
  144. package/package.json +11 -12
  145. package/src/bootstrap.ts +65 -7
  146. package/src/helpers/integration/crmFixtures.ts +21 -1
  147. package/src/modules/attachments/AGENTS.md +79 -0
  148. package/src/modules/attachments/api/route.ts +2 -0
  149. package/src/modules/attachments/lib/access.ts +36 -0
  150. package/src/modules/audit_logs/data/entities.ts +1 -0
  151. package/src/modules/audit_logs/migrations/.snapshot-open-mercato.json +10 -0
  152. package/src/modules/audit_logs/migrations/Migration20260611104500.ts +13 -0
  153. package/src/modules/audit_logs/services/accessLogService.ts +15 -0
  154. package/src/modules/auth/api/admin/nav.ts +9 -0
  155. package/src/modules/auth/api/login.ts +13 -13
  156. package/src/modules/auth/data/entities.ts +2 -0
  157. package/src/modules/auth/i18n/de.json +0 -1
  158. package/src/modules/auth/i18n/en.json +0 -1
  159. package/src/modules/auth/i18n/es.json +0 -1
  160. package/src/modules/auth/i18n/pl.json +0 -1
  161. package/src/modules/auth/lib/backendChrome.tsx +37 -1
  162. package/src/modules/auth/lib/consentIntegrity.ts +6 -3
  163. package/src/modules/auth/migrations/.snapshot-open-mercato.json +20 -0
  164. package/src/modules/auth/migrations/Migration20260611103000.ts +21 -0
  165. package/src/modules/auth/services/authService.ts +24 -4
  166. package/src/modules/auth/services/rbacService.ts +11 -2
  167. package/src/modules/customer_accounts/backend/customer_accounts/settings/domain/components/Diagnostics.tsx +0 -3
  168. package/src/modules/customers/api/deals/route.ts +51 -2
  169. package/src/modules/customers/api/deals/summary/route.ts +496 -0
  170. package/src/modules/customers/backend/customers/deals/[id]/hooks/useDealActivities.ts +28 -6
  171. package/src/modules/customers/backend/customers/deals/[id]/hooks/useDealData.ts +33 -6
  172. package/src/modules/customers/backend/customers/deals/[id]/page.tsx +17 -2
  173. package/src/modules/customers/backend/customers/deals/page.tsx +254 -66
  174. package/src/modules/customers/backend/customers/deals/pipeline/page.tsx +1 -2
  175. package/src/modules/customers/backend/customers/people-v2/[id]/page.tsx +18 -0
  176. package/src/modules/customers/cli.ts +15 -15
  177. package/src/modules/customers/components/DealsKpiStrip.tsx +389 -0
  178. package/src/modules/customers/components/detail/ConfirmDealLostDialog.tsx +0 -1
  179. package/src/modules/customers/components/detail/DealForm.tsx +121 -19
  180. package/src/modules/customers/components/detail/PersonDetailTabs.tsx +12 -2
  181. package/src/modules/customers/components/detail/ScheduleActivityDialog.tsx +1 -2
  182. package/src/modules/customers/components/kpi/PipelineStageBar.tsx +77 -0
  183. package/src/modules/customers/i18n/de.json +43 -0
  184. package/src/modules/customers/i18n/en.json +43 -0
  185. package/src/modules/customers/i18n/es.json +43 -0
  186. package/src/modules/customers/i18n/pl.json +43 -0
  187. package/src/modules/customers/lib/dealsMetrics.ts +159 -0
  188. package/src/modules/directory/api/organization-branding/route.ts +238 -0
  189. package/src/modules/directory/api/organizations/route.ts +7 -0
  190. package/src/modules/directory/backend/directory/branding/page.meta.ts +24 -0
  191. package/src/modules/directory/backend/directory/branding/page.tsx +248 -0
  192. package/src/modules/directory/commands/organizations.ts +9 -1
  193. package/src/modules/directory/data/entities.ts +3 -0
  194. package/src/modules/directory/data/validators.ts +12 -0
  195. package/src/modules/directory/i18n/de.json +21 -0
  196. package/src/modules/directory/i18n/en.json +21 -0
  197. package/src/modules/directory/i18n/es.json +21 -0
  198. package/src/modules/directory/i18n/pl.json +21 -0
  199. package/src/modules/directory/migrations/.snapshot-open-mercato.json +40 -0
  200. package/src/modules/directory/migrations/Migration20260607222259_directory.ts +13 -0
  201. package/src/modules/directory/subscribers/invalidateOrgScopeCache.ts +3 -1
  202. package/src/modules/directory/utils/organizationScope.ts +85 -30
  203. package/src/modules/entities/api/definitions.batch.ts +11 -7
  204. package/src/modules/entities/api/entities.ts +11 -0
  205. package/src/modules/entities/api/records.ts +46 -25
  206. package/src/modules/entities/backend/entities/user/[entityId]/records/[recordId]/page.tsx +15 -0
  207. package/src/modules/entities/backend/entities/user/[entityId]/records/create/page.tsx +15 -0
  208. package/src/modules/entities/backend/entities/user/[entityId]/records/page.tsx +23 -0
  209. package/src/modules/entities/components/useRecordsEntityGuard.ts +41 -0
  210. package/src/modules/entities/i18n/de.json +1 -0
  211. package/src/modules/entities/i18n/en.json +1 -0
  212. package/src/modules/entities/i18n/es.json +1 -0
  213. package/src/modules/entities/i18n/pl.json +1 -0
  214. package/src/modules/query_index/data/entities.ts +1 -0
  215. package/src/modules/query_index/lib/engine.ts +11 -5
  216. package/src/modules/query_index/migrations/.snapshot-open-mercato.json +11 -0
  217. package/src/modules/query_index/migrations/Migration20260611103000_query_index.ts +29 -0
  218. package/src/modules/sales/commands/documents.ts +7 -5
  219. package/src/modules/sales/components/documents/SalesDocumentsTable.tsx +2 -1
  220. package/src/modules/sales/components/documents/salesDocumentsColumns.ts +6 -0
  221. package/src/modules/staff/api/team-members.ts +9 -2
  222. package/src/modules/staff/api/timesheets/time-entries/[id]/timer-start/route.ts +31 -1
  223. package/src/modules/staff/backend/staff/team-members/[id]/page.tsx +18 -8
  224. package/src/modules/staff/commands/team-members.ts +5 -2
  225. package/src/modules/staff/components/TeamMemberForm.tsx +4 -1
  226. package/src/modules/staff/i18n/de.json +1 -0
  227. package/src/modules/staff/i18n/en.json +1 -0
  228. package/src/modules/staff/i18n/es.json +1 -0
  229. package/src/modules/staff/i18n/pl.json +1 -0
  230. package/src/modules/staff/lib/scheduleSwitch.ts +46 -0
  231. package/src/modules/workflows/backend/definitions/create/page.tsx +1 -2
  232. package/src/modules/workflows/backend/definitions/visual-editor/page.tsx +1 -2
  233. package/src/modules/workflows/components/DefinitionTriggersEditor.tsx +1 -2
  234. package/src/modules/workflows/components/NodeEditDialog.tsx +1 -4
  235. package/src/modules/workflows/components/NodeEditDialogCrudForm.tsx +4 -7
  236. package/src/modules/workflows/components/WorkflowGraphImpl.tsx +1 -2
  237. package/src/modules/workflows/components/fields/FormFieldArrayEditor.tsx +2 -3
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "version": 3,
3
3
  "sources": ["../../../../../src/modules/customers/api/deals/route.ts"],
4
- "sourcesContent": ["import { z } from 'zod'\nimport { makeCrudRoute } from '@open-mercato/shared/lib/crud/factory'\nimport { CrudHttpError } from '@open-mercato/shared/lib/crud/errors'\nimport { CustomerDeal, CustomerDealPersonLink, CustomerDealCompanyLink } from '../../data/entities'\nimport { dealCreateSchema, dealUpdateSchema } from '../../data/validators'\nimport { E } from '#generated/entities.ids.generated'\nimport { resolveTranslations } from '@open-mercato/shared/lib/i18n/server'\nimport { parseBooleanFromUnknown } from '@open-mercato/shared/lib/boolean'\nimport {\n applyEntityIdRestriction,\n findMatchingEntityIdsWithQueryEngine,\n findMatchingEntityIdsBySearchTokensAcrossSources,\n parseScopedCommandInput,\n} from '../utils'\nimport type { EntityManager } from '@mikro-orm/postgresql'\nimport {\n createCustomersCrudOpenApi,\n createPagedListResponseSchema,\n defaultOkResponseSchema,\n} from '../openapi'\nimport { findWithDecryption } from '@open-mercato/shared/lib/encryption/find'\nimport { isTenantDataEncryptionEnabled } from '@open-mercato/shared/lib/encryption/toggles'\nimport { escapeLikePattern } from '@open-mercato/shared/lib/db/escapeLikePattern'\nimport { consumeAdvancedFilterState, mergeAdvancedFilterTree } from '@open-mercato/shared/lib/crud/advanced-filter-integration'\nimport { fetchStuckDealIds } from '../../lib/stuckDeals'\n\nconst rawBodySchema = z.object({}).passthrough()\n\nconst stringOrStringArray = z.union([z.string(), z.array(z.string())])\nconst booleanQueryParam = z.preprocess((value) => {\n const parsed = parseBooleanFromUnknown(value)\n return parsed === null ? value : parsed\n}, z.boolean()).optional()\n\nexport const dealListQuerySchema = z\n .object({\n page: z.coerce.number().min(1).default(1),\n pageSize: z.coerce.number().min(1).max(100).default(50),\n id: z.string().uuid().optional(),\n search: z.string().optional(),\n status: stringOrStringArray.optional(),\n pipelineStage: z.string().optional(),\n pipelineId: stringOrStringArray.optional(),\n pipelineStageId: z.union([z.string().uuid(), z.literal('__unassigned')]).optional(),\n ownerUserId: stringOrStringArray.optional(),\n expectedCloseAtFrom: z.string().optional(),\n expectedCloseAtTo: z.string().optional(),\n isStuck: booleanQueryParam,\n isOverdue: booleanQueryParam,\n valueCurrency: stringOrStringArray.optional(),\n sortField: z.string().optional(),\n sortDir: z.enum(['asc', 'desc']).optional(),\n /**\n * @deprecated Use `personId` instead. The `personEntityId` alias is kept for backward\n * compatibility with older clients and will be removed in a future minor release.\n */\n personEntityId: z.string().uuid().optional().describe('Deprecated; use personId'),\n /**\n * @deprecated Use `companyId` instead. The `companyEntityId` alias is kept for backward\n * compatibility with older clients and will be removed in a future minor release.\n */\n companyEntityId: z.string().uuid().optional().describe('Deprecated; use companyId'),\n personId: stringOrStringArray.optional(),\n companyId: stringOrStringArray.optional(),\n })\n .passthrough()\n\nconst routeMetadata = {\n GET: { requireAuth: true, requireFeatures: ['customers.deals.view'] },\n POST: { requireAuth: true, requireFeatures: ['customers.deals.manage'] },\n PUT: { requireAuth: true, requireFeatures: ['customers.deals.manage'] },\n DELETE: { requireAuth: true, requireFeatures: ['customers.deals.manage'] },\n}\n\nexport const metadata = routeMetadata\n\nexport type DealListQuery = z.infer<typeof dealListQuerySchema>\n\nfunction parseUuid(value: unknown): string | null {\n if (typeof value !== 'string') return null\n const trimmed = value.trim()\n if (!trimmed.length) return null\n const result = z.string().uuid().safeParse(trimmed)\n return result.success ? trimmed : null\n}\n\nfunction normalizeStringList(value: unknown): string[] {\n const set = new Set<string>()\n const visit = (entry: unknown) => {\n if (entry == null) return\n if (Array.isArray(entry)) {\n entry.forEach(visit)\n return\n }\n if (typeof entry !== 'string') return\n entry\n .split(',')\n .map((token) => token.trim())\n .filter((token) => token.length > 0)\n .forEach((token) => set.add(token))\n }\n visit(value)\n return Array.from(set)\n}\n\nfunction parseDateInput(value: unknown): Date | null {\n if (!(typeof value === 'string') || value.trim().length === 0) return null\n const date = new Date(value)\n return Number.isNaN(date.getTime()) ? null : date\n}\n\n/**\n * Pre-pagination ID narrowing for person/company filters.\n *\n * Mirrors the SQL the aggregate route uses (`EXISTS ... IN (...)`) so the list endpoint\n * and the lane-header aggregate return a consistent set of deals for the same filters.\n * Semantics: OR within a category (any selected person/company matches), AND across\n * categories (deal must match at least one person AND at least one company when both\n * filter lists are provided).\n *\n * Returns the matched deal IDs (UUIDs). An empty array means \"no deals match\" \u2014 callers\n * must intersect with `restrictedIds` (which will collapse the list to zero).\n */\nasync function fetchDealIdsMatchingAssociations(\n em: EntityManager,\n organizationId: string,\n tenantId: string,\n personIds: string[],\n companyIds: string[],\n): Promise<string[]> {\n if (!personIds.length && !companyIds.length) return []\n const where: string[] = [\n 'organization_id = ?',\n 'tenant_id = ?',\n 'deleted_at IS NULL',\n ]\n const values: Array<string | number> = [organizationId, tenantId]\n if (personIds.length) {\n const placeholders = personIds.map(() => '?').join(',')\n where.push(\n `EXISTS (SELECT 1 FROM customer_deal_people dp WHERE dp.deal_id = customer_deals.id AND dp.person_entity_id IN (${placeholders}))`,\n )\n values.push(...personIds)\n }\n if (companyIds.length) {\n const placeholders = companyIds.map(() => '?').join(',')\n where.push(\n `EXISTS (SELECT 1 FROM customer_deal_companies dc WHERE dc.deal_id = customer_deals.id AND dc.company_entity_id IN (${placeholders}))`,\n )\n values.push(...companyIds)\n }\n const rows = await em.getConnection().execute<Array<{ id: string }>>(\n `SELECT id FROM customer_deals WHERE ${where.join(' AND ')}`,\n values,\n )\n return rows.map((row) => row.id)\n}\n\nfunction normalizeCurrencyList(value: unknown): string[] {\n const set = new Set<string>()\n const visit = (entry: unknown) => {\n if (entry == null) return\n if (Array.isArray(entry)) {\n entry.forEach(visit)\n return\n }\n if (typeof entry !== 'string') return\n entry\n .split(',')\n .map((token) => token.trim().toUpperCase())\n .filter((token) => /^[A-Z]{3}$/.test(token))\n .forEach((token) => set.add(token))\n }\n visit(value)\n return Array.from(set)\n}\n\nfunction normalizeUuidList(values: Array<unknown>): string[] {\n const set = new Set<string>()\n values.forEach((candidate) => {\n if (Array.isArray(candidate)) {\n candidate.forEach((entry) => {\n const parsed = parseUuid(entry)\n if (parsed) set.add(parsed)\n })\n return\n }\n if (typeof candidate === 'string' && candidate.includes(',')) {\n candidate\n .split(',')\n .map((entry) => entry.trim())\n .forEach((entry) => {\n const parsed = parseUuid(entry)\n if (parsed) set.add(parsed)\n })\n return\n }\n const parsed = parseUuid(candidate)\n if (parsed) set.add(parsed)\n })\n return Array.from(set)\n}\n\nexport async function buildDealListFilters(query: DealListQuery, ctx?: import('@open-mercato/shared/lib/crud/factory').CrudCtx) {\n const advancedFilterTree = consumeAdvancedFilterState(query)\n const filters: Record<string, unknown> = {}\n let restrictedIds: string[] | null = null\n\n const intersectIds = (ids: string[]) => {\n if (restrictedIds === null) {\n restrictedIds = ids\n return\n }\n const lookup = new Set(ids)\n restrictedIds = restrictedIds.filter((id) => lookup.has(id))\n }\n\n if (query.id) filters.id = { $eq: query.id }\n\n if (query.search) {\n const matchingIds = ctx\n ? await findMatchingEntityIdsBySearchTokensAcrossSources({\n ctx,\n query: query.search,\n sources: [\n {\n entityType: E.customers.customer_deal,\n fields: [\n 'title',\n 'description',\n 'status',\n 'pipeline_stage',\n 'source',\n 'value_amount',\n 'value_currency',\n 'cf:competitive_risk',\n 'cf:implementation_complexity',\n ],\n },\n ],\n })\n : null\n if (matchingIds !== null && matchingIds.length > 0) {\n intersectIds(matchingIds)\n } else if (isTenantDataEncryptionEnabled()) {\n // `customers:customer_deal.title` and `.description` are encrypted at rest\n // (see `encryption.ts`). The `$ilike` fallback below would silently match\n // nothing on the ciphertext and produce empty pages without disclosing why \u2014\n // collapse the result to \"no matches\" instead so the kanban + list views\n // agree, and lane aggregates stay consistent with the list endpoint.\n intersectIds([])\n } else {\n const searchPattern = `%${escapeLikePattern(query.search)}%`\n filters.$or = [\n { title: { $ilike: searchPattern } },\n { description: { $ilike: searchPattern } },\n ]\n }\n }\n\n const statusList = query.status ? normalizeStringList(query.status) : []\n if (statusList.length > 0) {\n filters.status = statusList.length === 1 ? { $eq: statusList[0] } : { $in: statusList }\n }\n\n if (query.pipelineStage) {\n filters.pipeline_stage = { $eq: query.pipelineStage }\n }\n\n const pipelineIds = query.pipelineId ? normalizeUuidList([query.pipelineId]) : []\n if (pipelineIds.length > 0) {\n filters.pipeline_id = pipelineIds.length === 1 ? { $eq: pipelineIds[0] } : { $in: pipelineIds }\n }\n\n if (query.pipelineStageId === '__unassigned') {\n filters.pipeline_stage_id = { $eq: null }\n } else if (query.pipelineStageId) {\n filters.pipeline_stage_id = { $eq: query.pipelineStageId }\n }\n\n const ownerUserIds = query.ownerUserId ? normalizeUuidList([query.ownerUserId]) : []\n if (ownerUserIds.length > 0) {\n filters.owner_user_id =\n ownerUserIds.length === 1 ? { $eq: ownerUserIds[0] } : { $in: ownerUserIds }\n }\n\n // Currency filter (additive; sourced by the kanban Currency popover). Codes are\n // normalised to upper-case so `usd` / `USD` / mixed-case query params all hit the same\n // stored value (the deals API persists `value_currency` upper-case already).\n const currencyCodes = query.valueCurrency ? normalizeCurrencyList(query.valueCurrency) : []\n if (currencyCodes.length > 0) {\n filters.value_currency =\n currencyCodes.length === 1 ? { $eq: currencyCodes[0] } : { $in: currencyCodes }\n }\n\n const expectedCloseFrom = parseDateInput(query.expectedCloseAtFrom)\n const expectedCloseTo = parseDateInput(query.expectedCloseAtTo)\n if (expectedCloseFrom || expectedCloseTo) {\n const range: Record<string, Date> = {}\n if (expectedCloseFrom) range.$gte = expectedCloseFrom\n if (expectedCloseTo) range.$lte = expectedCloseTo\n filters.expected_close_at = range\n }\n\n if (query.isOverdue) {\n const today = new Date()\n today.setHours(0, 0, 0, 0)\n if (statusList.length === 0) {\n filters.status = { $eq: 'open' }\n }\n const existingRange =\n filters.expected_close_at && typeof filters.expected_close_at === 'object'\n ? (filters.expected_close_at as Record<string, Date>)\n : {}\n existingRange.$lt = today\n filters.expected_close_at = existingRange\n }\n\n if (query.isStuck && ctx) {\n const tenantId = ctx.auth?.tenantId\n // CrudCtx.auth carries `orgId` (not `organizationId`). The previous code referenced\n // `organizationId` which is always `undefined`, so the typeof check below silently\n // skipped the entire isStuck branch \u2014 `?isStuck=true` was a no-op on this endpoint.\n const organizationId = ctx.auth?.orgId\n if (typeof tenantId === 'string' && typeof organizationId === 'string') {\n const em = ctx.container.resolve<EntityManager>('em')\n const stuckIds = await fetchStuckDealIds(em, organizationId, tenantId)\n intersectIds(stuckIds)\n }\n }\n\n // Pre-pagination association filter. Must run on the FULL dataset (before pagination),\n // otherwise matching deals on later pages disappear and `total` would be wrong. Read the\n // raw URL too so legacy `?personEntityId=` / `?companyEntityId=` keep working alongside the\n // canonical `?personId=` / `?companyId=`.\n const url = ctx?.request ? new URL(ctx.request.url) : null\n const personCandidates: unknown[] = [query.personId, query.personEntityId]\n const companyCandidates: unknown[] = [query.companyId, query.companyEntityId]\n if (url) {\n personCandidates.push(url.searchParams.getAll('personId'))\n personCandidates.push(url.searchParams.getAll('personEntityId'))\n companyCandidates.push(url.searchParams.getAll('companyId'))\n companyCandidates.push(url.searchParams.getAll('companyEntityId'))\n }\n const selectedPersonIds = normalizeUuidList(personCandidates)\n const selectedCompanyIds = normalizeUuidList(companyCandidates)\n if ((selectedPersonIds.length > 0 || selectedCompanyIds.length > 0) && ctx) {\n const tenantId = ctx.auth?.tenantId\n // `ctx.auth` exposes `orgId` (see AuthContext in @open-mercato/shared/lib/auth/server).\n // Read it under the correct key \u2014 the previous code's `organizationId` would always be\n // `undefined`, silently disabling association filtering on the deals list endpoint.\n const organizationId = ctx.auth?.orgId\n if (typeof tenantId === 'string' && typeof organizationId === 'string') {\n const em = ctx.container.resolve<EntityManager>('em')\n const matchedIds = await fetchDealIdsMatchingAssociations(\n em,\n organizationId,\n tenantId,\n selectedPersonIds,\n selectedCompanyIds,\n )\n // intersectIds with empty array \u2192 no rows; collapses the page to zero, total stays correct.\n intersectIds(matchedIds)\n }\n }\n\n if (ctx && advancedFilterTree) {\n const advancedFilters = mergeAdvancedFilterTree({ ...filters }, advancedFilterTree)\n const matchedIds = await findMatchingEntityIdsWithQueryEngine({\n ctx,\n entityId: E.customers.customer_deal,\n filters: advancedFilters,\n })\n if (matchedIds !== null) {\n intersectIds(matchedIds)\n }\n }\n\n if (restrictedIds !== null) {\n applyEntityIdRestriction(filters, restrictedIds)\n }\n\n return filters\n}\n\nconst crud = makeCrudRoute<unknown, unknown, DealListQuery>({\n metadata: routeMetadata,\n orm: {\n entity: CustomerDeal,\n idField: 'id',\n orgField: 'organizationId',\n tenantField: 'tenantId',\n softDeleteField: 'deletedAt',\n },\n indexer: {\n entityType: E.customers.customer_deal,\n },\n enrichers: { entityId: 'customers.deal' },\n list: {\n schema: dealListQuerySchema,\n entityId: E.customers.customer_deal,\n fields: [\n 'id',\n 'title',\n 'description',\n 'status',\n 'pipeline_stage',\n 'pipeline_id',\n 'pipeline_stage_id',\n 'value_amount',\n 'value_currency',\n 'probability',\n 'expected_close_at',\n 'owner_user_id',\n 'source',\n 'closure_outcome',\n 'loss_reason_id',\n 'loss_notes',\n 'organization_id',\n 'tenant_id',\n 'created_at',\n 'updated_at',\n ],\n decorateCustomFields: {\n entityIds: E.customers.customer_deal,\n },\n sortFieldMap: {\n createdAt: 'created_at',\n updatedAt: 'updated_at',\n title: 'title',\n value: 'value_amount',\n probability: 'probability',\n expectedCloseAt: 'expected_close_at',\n },\n buildFilters: buildDealListFilters,\n },\n actions: {\n create: {\n commandId: 'customers.deals.create',\n schema: rawBodySchema,\n mapInput: async ({ raw, ctx }) => {\n const { translate } = await resolveTranslations()\n return parseScopedCommandInput(dealCreateSchema, raw ?? {}, ctx, translate)\n },\n response: ({ result }) => ({ id: result?.dealId ?? result?.id ?? null }),\n status: 201,\n },\n update: {\n commandId: 'customers.deals.update',\n schema: rawBodySchema,\n mapInput: async ({ raw, ctx }) => {\n const { translate } = await resolveTranslations()\n return parseScopedCommandInput(dealUpdateSchema, raw ?? {}, ctx, translate)\n },\n response: () => ({ ok: true }),\n },\n delete: {\n commandId: 'customers.deals.delete',\n schema: rawBodySchema,\n mapInput: async ({ parsed, ctx }) => {\n const { translate } = await resolveTranslations()\n const id =\n parsed?.body?.id ??\n parsed?.id ??\n parsed?.query?.id ??\n (ctx.request ? new URL(ctx.request.url).searchParams.get('id') : null)\n if (!id) throw new CrudHttpError(400, { error: translate('customers.errors.deal_required', 'Deal id is required') })\n return { id }\n },\n response: () => ({ ok: true }),\n },\n },\n hooks: {\n // afterList only DECORATES results with `people`/`companies` arrays \u2014 it must not filter,\n // because filtering after pagination would drop deals on later pages and rewrite `total`\n // to a misleading value. Association filtering happens pre-pagination in `buildFilters`\n // via `fetchDealIdsMatchingAssociations`.\n afterList: async (payload, ctx) => {\n const items = Array.isArray(payload.items) ? payload.items : []\n if (!items.length) return\n const scopeSource = (items[0] ?? {}) as Record<string, unknown>\n const tenantIdRaw = scopeSource.tenantId ?? scopeSource.tenant_id\n const fallbackTenantId = (typeof tenantIdRaw === 'string' && tenantIdRaw.trim().length ? tenantIdRaw : null) ?? ctx.auth?.tenantId ?? null\n const orgIdRaw = scopeSource.organizationId ?? scopeSource.organization_id\n const fallbackOrganizationId = (typeof orgIdRaw === 'string' && orgIdRaw.trim().length ? orgIdRaw : null) ?? ctx.auth?.orgId ?? null\n const ids = items\n .map((item: unknown) => {\n if (!item || typeof item !== 'object') return null\n const candidate = (item as Record<string, unknown>).id\n return typeof candidate === 'string' && candidate.trim().length ? candidate : null\n })\n .filter((value: string | null): value is string => typeof value === 'string' && value.length > 0)\n if (!ids.length) {\n payload.items = []\n payload.total = 0\n return\n }\n try {\n const em = (ctx.container.resolve('em') as EntityManager)\n const [allPersonLinks, allCompanyLinks] = await Promise.all([\n findWithDecryption(\n em,\n CustomerDealPersonLink,\n { deal: { $in: ids } },\n { populate: ['person'] },\n { tenantId: fallbackTenantId, organizationId: fallbackOrganizationId },\n ),\n findWithDecryption(\n em,\n CustomerDealCompanyLink,\n { deal: { $in: ids } },\n { populate: ['company'] },\n { tenantId: fallbackTenantId, organizationId: fallbackOrganizationId },\n ),\n ])\n\n const personAssignments = new Map<string, { id: string; label: string }[]>()\n allPersonLinks.forEach((link) => {\n const deal = link.deal\n const dealRecord = deal && typeof deal === 'object' ? deal as unknown as Record<string, unknown> : null\n const dealId = typeof deal === 'string' ? deal\n : dealRecord && typeof dealRecord.id === 'string' ? dealRecord.id\n : null\n if (!dealId) return\n const personRef = link.person\n const personRecord = personRef && typeof personRef === 'object' ? personRef as unknown as Record<string, unknown> : null\n const personId = typeof personRef === 'string' ? personRef\n : personRecord && typeof personRecord.id === 'string' ? personRecord.id\n : null\n if (!personId) return\n const label = personRecord && typeof personRecord.displayName === 'string'\n ? personRecord.displayName\n : ''\n const bucket = personAssignments.get(dealId) ?? []\n if (!bucket.some((entry) => entry.id === personId)) {\n bucket.push({ id: personId, label })\n personAssignments.set(dealId, bucket)\n }\n })\n\n const companyAssignments = new Map<string, { id: string; label: string }[]>()\n allCompanyLinks.forEach((link) => {\n const deal = link.deal\n const dealRecord = deal && typeof deal === 'object' ? deal as unknown as Record<string, unknown> : null\n const dealId = typeof deal === 'string' ? deal\n : dealRecord && typeof dealRecord.id === 'string' ? dealRecord.id\n : null\n if (!dealId) return\n const companyRef = link.company\n const companyRecord = companyRef && typeof companyRef === 'object' ? companyRef as unknown as Record<string, unknown> : null\n const companyId = typeof companyRef === 'string' ? companyRef\n : companyRecord && typeof companyRecord.id === 'string' ? companyRecord.id\n : null\n if (!companyId) return\n const label = companyRecord && typeof companyRecord.displayName === 'string'\n ? companyRecord.displayName\n : ''\n const bucket = companyAssignments.get(dealId) ?? []\n if (!bucket.some((entry) => entry.id === companyId)) {\n bucket.push({ id: companyId, label })\n companyAssignments.set(dealId, bucket)\n }\n })\n\n const enhancedItems = items\n .map((item: unknown) => {\n if (!item || typeof item !== 'object') return null\n const data = item as Record<string, unknown>\n const candidate = typeof data.id === 'string' ? data.id : null\n if (!candidate || !candidate.trim().length) return null\n const people = personAssignments.get(candidate) ?? []\n const companies = companyAssignments.get(candidate) ?? []\n const tenantIdRaw =\n typeof data.tenantId === 'string'\n ? data.tenantId\n : typeof data.tenant_id === 'string'\n ? data.tenant_id\n : null\n const organizationIdRaw =\n typeof data.organizationId === 'string'\n ? data.organizationId\n : typeof data.organization_id === 'string'\n ? data.organization_id\n : null\n const tenantId = tenantIdRaw && tenantIdRaw.trim().length ? tenantIdRaw.trim() : null\n const organizationId = organizationIdRaw && organizationIdRaw.trim().length ? organizationIdRaw.trim() : null\n return {\n ...data,\n personIds: people.map((entry) => entry.id),\n people,\n companyIds: companies.map((entry) => entry.id),\n companies,\n tenantId,\n organizationId,\n }\n })\n .filter(\n (item: Record<string, unknown> | null): item is Record<string, unknown> => item !== null,\n )\n\n payload.items = enhancedItems\n } catch (err) {\n // We swallow rather than fail the request because the kanban is still useful without\n // people/companies labels (cards just lose their company pill). Tag every item with\n // `_associations: { ok: false }` so a future surface can render a degraded-state hint\n // instead of silently showing cards without company badges.\n console.warn('[customers.deals] failed to decorate items with person/company links', err)\n payload.items = items.map((item: unknown) => {\n if (!item || typeof item !== 'object') return item\n return {\n ...(item as Record<string, unknown>),\n _associations: {\n ok: false,\n reason: err instanceof Error ? err.message : 'unknown',\n },\n }\n })\n }\n },\n },\n})\n\nconst { POST, PUT, DELETE } = crud\n\nexport { POST, PUT, DELETE }\nexport const GET = crud.GET\n\nconst dealAssociationSchema = z.object({\n id: z.string().uuid(),\n label: z.string().nullable(),\n})\n\nconst dealListItemSchema = z\n .object({\n id: z.string().uuid(),\n title: z.string().nullable(),\n description: z.string().nullable().optional(),\n status: z.string().nullable().optional(),\n pipeline_stage: z.string().nullable().optional(),\n pipeline_id: z.string().uuid().nullable().optional(),\n pipeline_stage_id: z.string().uuid().nullable().optional(),\n value_amount: z.number().nullable().optional(),\n value_currency: z.string().nullable().optional(),\n probability: z.number().nullable().optional(),\n expected_close_at: z.string().nullable().optional(),\n owner_user_id: z.string().uuid().nullable().optional(),\n source: z.string().nullable().optional(),\n organization_id: z.string().uuid().nullable().optional(),\n tenant_id: z.string().uuid().nullable().optional(),\n created_at: z.string().nullable().optional(),\n updated_at: z.string().nullable().optional(),\n personIds: z.array(z.string().uuid()).optional(),\n people: z.array(dealAssociationSchema).optional(),\n companyIds: z.array(z.string().uuid()).optional(),\n companies: z.array(dealAssociationSchema).optional(),\n organizationId: z.string().uuid().nullable().optional(),\n tenantId: z.string().uuid().nullable().optional(),\n })\n .passthrough()\n\nconst dealCreateResponseSchema = z.object({\n id: z.string().uuid().nullable(),\n})\n\nexport const openApi = createCustomersCrudOpenApi({\n resourceName: 'Deal',\n querySchema: dealListQuerySchema,\n listResponseSchema: createPagedListResponseSchema(dealListItemSchema),\n create: {\n schema: dealCreateSchema,\n responseSchema: dealCreateResponseSchema,\n description: 'Creates a sales deal, optionally associating people and companies.',\n },\n update: {\n schema: dealUpdateSchema,\n responseSchema: defaultOkResponseSchema,\n description: 'Updates pipeline position, metadata, or associations for an existing deal.',\n },\n del: {\n schema: z.object({ id: z.string().uuid() }),\n responseSchema: defaultOkResponseSchema,\n description: 'Deletes a deal by `id`. The identifier may be provided in the body or query parameters.',\n },\n})\n"],
5
- "mappings": "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",
4
+ "sourcesContent": ["import { z } from 'zod'\nimport { makeCrudRoute } from '@open-mercato/shared/lib/crud/factory'\nimport { CrudHttpError } from '@open-mercato/shared/lib/crud/errors'\nimport { CustomerDeal, CustomerDealPersonLink, CustomerDealCompanyLink } from '../../data/entities'\nimport { dealCreateSchema, dealUpdateSchema } from '../../data/validators'\nimport { E } from '#generated/entities.ids.generated'\nimport { resolveTranslations } from '@open-mercato/shared/lib/i18n/server'\nimport { parseBooleanFromUnknown } from '@open-mercato/shared/lib/boolean'\nimport {\n applyEntityIdRestriction,\n findMatchingEntityIdsWithQueryEngine,\n findMatchingEntityIdsBySearchTokensAcrossSources,\n parseScopedCommandInput,\n} from '../utils'\nimport type { EntityManager } from '@mikro-orm/postgresql'\nimport {\n createCustomersCrudOpenApi,\n createPagedListResponseSchema,\n defaultOkResponseSchema,\n} from '../openapi'\nimport { findWithDecryption } from '@open-mercato/shared/lib/encryption/find'\nimport { isTenantDataEncryptionEnabled } from '@open-mercato/shared/lib/encryption/toggles'\nimport { escapeLikePattern } from '@open-mercato/shared/lib/db/escapeLikePattern'\nimport { consumeAdvancedFilterState, mergeAdvancedFilterTree } from '@open-mercato/shared/lib/crud/advanced-filter-integration'\nimport { fetchStuckDealIds } from '../../lib/stuckDeals'\n\nconst rawBodySchema = z.object({}).passthrough()\n\nconst stringOrStringArray = z.union([z.string(), z.array(z.string())])\nconst OPEN_DEAL_STATUSES = ['open', 'in_progress'] as const\nconst booleanQueryParam = z.preprocess((value) => {\n const parsed = parseBooleanFromUnknown(value)\n return parsed === null ? value : parsed\n}, z.boolean()).optional()\n\nexport const dealListQuerySchema = z\n .object({\n page: z.coerce.number().min(1).default(1),\n pageSize: z.coerce.number().min(1).max(100).default(50),\n id: z.string().uuid().optional(),\n search: z.string().optional(),\n status: stringOrStringArray.optional(),\n pipelineStage: z.string().optional(),\n pipelineId: stringOrStringArray.optional(),\n pipelineStageId: z.union([z.string().uuid(), z.literal('__unassigned')]).optional(),\n ownerUserId: stringOrStringArray.optional(),\n expectedCloseAtFrom: z.string().optional(),\n expectedCloseAtTo: z.string().optional(),\n isStuck: booleanQueryParam,\n isOverdue: booleanQueryParam,\n needsAttention: booleanQueryParam,\n valueCurrency: stringOrStringArray.optional(),\n sortField: z.string().optional(),\n sortDir: z.enum(['asc', 'desc']).optional(),\n /**\n * @deprecated Use `personId` instead. The `personEntityId` alias is kept for backward\n * compatibility with older clients and will be removed in a future minor release.\n */\n personEntityId: z.string().uuid().optional().describe('Deprecated; use personId'),\n /**\n * @deprecated Use `companyId` instead. The `companyEntityId` alias is kept for backward\n * compatibility with older clients and will be removed in a future minor release.\n */\n companyEntityId: z.string().uuid().optional().describe('Deprecated; use companyId'),\n personId: stringOrStringArray.optional(),\n companyId: stringOrStringArray.optional(),\n })\n .passthrough()\n\nconst routeMetadata = {\n GET: { requireAuth: true, requireFeatures: ['customers.deals.view'] },\n POST: { requireAuth: true, requireFeatures: ['customers.deals.manage'] },\n PUT: { requireAuth: true, requireFeatures: ['customers.deals.manage'] },\n DELETE: { requireAuth: true, requireFeatures: ['customers.deals.manage'] },\n}\n\nexport const metadata = routeMetadata\n\nexport type DealListQuery = z.infer<typeof dealListQuerySchema>\n\nfunction parseUuid(value: unknown): string | null {\n if (typeof value !== 'string') return null\n const trimmed = value.trim()\n if (!trimmed.length) return null\n const result = z.string().uuid().safeParse(trimmed)\n return result.success ? trimmed : null\n}\n\nfunction normalizeStringList(value: unknown): string[] {\n const set = new Set<string>()\n const visit = (entry: unknown) => {\n if (entry == null) return\n if (Array.isArray(entry)) {\n entry.forEach(visit)\n return\n }\n if (typeof entry !== 'string') return\n entry\n .split(',')\n .map((token) => token.trim())\n .filter((token) => token.length > 0)\n .forEach((token) => set.add(token))\n }\n visit(value)\n return Array.from(set)\n}\n\nfunction parseDateInput(value: unknown): Date | null {\n if (!(typeof value === 'string') || value.trim().length === 0) return null\n const date = new Date(value)\n return Number.isNaN(date.getTime()) ? null : date\n}\n\n/**\n * Pre-pagination ID narrowing for person/company filters.\n *\n * Mirrors the SQL the aggregate route uses (`EXISTS ... IN (...)`) so the list endpoint\n * and the lane-header aggregate return a consistent set of deals for the same filters.\n * Semantics: OR within a category (any selected person/company matches), AND across\n * categories (deal must match at least one person AND at least one company when both\n * filter lists are provided).\n *\n * Returns the matched deal IDs (UUIDs). An empty array means \"no deals match\" \u2014 callers\n * must intersect with `restrictedIds` (which will collapse the list to zero).\n */\nasync function fetchDealIdsMatchingAssociations(\n em: EntityManager,\n organizationId: string,\n tenantId: string,\n personIds: string[],\n companyIds: string[],\n): Promise<string[]> {\n if (!personIds.length && !companyIds.length) return []\n const where: string[] = [\n 'organization_id = ?',\n 'tenant_id = ?',\n 'deleted_at IS NULL',\n ]\n const values: Array<string | number> = [organizationId, tenantId]\n if (personIds.length) {\n const placeholders = personIds.map(() => '?').join(',')\n where.push(\n `EXISTS (SELECT 1 FROM customer_deal_people dp WHERE dp.deal_id = customer_deals.id AND dp.person_entity_id IN (${placeholders}))`,\n )\n values.push(...personIds)\n }\n if (companyIds.length) {\n const placeholders = companyIds.map(() => '?').join(',')\n where.push(\n `EXISTS (SELECT 1 FROM customer_deal_companies dc WHERE dc.deal_id = customer_deals.id AND dc.company_entity_id IN (${placeholders}))`,\n )\n values.push(...companyIds)\n }\n const rows = await em.getConnection().execute<Array<{ id: string }>>(\n `SELECT id FROM customer_deals WHERE ${where.join(' AND ')}`,\n values,\n )\n return rows.map((row) => row.id)\n}\n\nasync function fetchNeedAttentionDealIds(\n em: EntityManager,\n organizationId: string,\n tenantId: string,\n): Promise<string[]> {\n const connection = em.getConnection()\n const overdueRows = await connection.execute<Array<{ id: string }>>(\n `SELECT id FROM customer_deals\n WHERE organization_id = ?\n AND tenant_id = ?\n AND deleted_at IS NULL\n AND status = 'open'\n AND expected_close_at IS NOT NULL\n AND expected_close_at < CURRENT_DATE`,\n [organizationId, tenantId],\n )\n\n const attentionIds = new Set(overdueRows.map((row) => row.id))\n const stuckIds = await fetchStuckDealIds(em, organizationId, tenantId)\n if (stuckIds.length > 0) {\n const idPlaceholders = stuckIds.map(() => '?').join(',')\n const statusPlaceholders = OPEN_DEAL_STATUSES.map(() => '?').join(',')\n const openStuckRows = await connection.execute<Array<{ id: string }>>(\n `SELECT id FROM customer_deals\n WHERE organization_id = ?\n AND tenant_id = ?\n AND deleted_at IS NULL\n AND status IN (${statusPlaceholders})\n AND id IN (${idPlaceholders})`,\n [organizationId, tenantId, ...OPEN_DEAL_STATUSES, ...stuckIds],\n )\n for (const row of openStuckRows) attentionIds.add(row.id)\n }\n\n return Array.from(attentionIds)\n}\n\nfunction normalizeCurrencyList(value: unknown): string[] {\n const set = new Set<string>()\n const visit = (entry: unknown) => {\n if (entry == null) return\n if (Array.isArray(entry)) {\n entry.forEach(visit)\n return\n }\n if (typeof entry !== 'string') return\n entry\n .split(',')\n .map((token) => token.trim().toUpperCase())\n .filter((token) => /^[A-Z]{3}$/.test(token))\n .forEach((token) => set.add(token))\n }\n visit(value)\n return Array.from(set)\n}\n\nfunction normalizeUuidList(values: Array<unknown>): string[] {\n const set = new Set<string>()\n values.forEach((candidate) => {\n if (Array.isArray(candidate)) {\n candidate.forEach((entry) => {\n const parsed = parseUuid(entry)\n if (parsed) set.add(parsed)\n })\n return\n }\n if (typeof candidate === 'string' && candidate.includes(',')) {\n candidate\n .split(',')\n .map((entry) => entry.trim())\n .forEach((entry) => {\n const parsed = parseUuid(entry)\n if (parsed) set.add(parsed)\n })\n return\n }\n const parsed = parseUuid(candidate)\n if (parsed) set.add(parsed)\n })\n return Array.from(set)\n}\n\nexport async function buildDealListFilters(query: DealListQuery, ctx?: import('@open-mercato/shared/lib/crud/factory').CrudCtx) {\n const advancedFilterTree = consumeAdvancedFilterState(query)\n const filters: Record<string, unknown> = {}\n let restrictedIds: string[] | null = null\n\n const intersectIds = (ids: string[]) => {\n if (restrictedIds === null) {\n restrictedIds = ids\n return\n }\n const lookup = new Set(ids)\n restrictedIds = restrictedIds.filter((id) => lookup.has(id))\n }\n\n if (query.id) filters.id = { $eq: query.id }\n\n if (query.search) {\n const matchingIds = ctx\n ? await findMatchingEntityIdsBySearchTokensAcrossSources({\n ctx,\n query: query.search,\n sources: [\n {\n entityType: E.customers.customer_deal,\n fields: [\n 'title',\n 'description',\n 'status',\n 'pipeline_stage',\n 'source',\n 'value_amount',\n 'value_currency',\n 'cf:competitive_risk',\n 'cf:implementation_complexity',\n ],\n },\n ],\n })\n : null\n if (matchingIds !== null && matchingIds.length > 0) {\n intersectIds(matchingIds)\n } else if (isTenantDataEncryptionEnabled()) {\n // `customers:customer_deal.title` and `.description` are encrypted at rest\n // (see `encryption.ts`). The `$ilike` fallback below would silently match\n // nothing on the ciphertext and produce empty pages without disclosing why \u2014\n // collapse the result to \"no matches\" instead so the kanban + list views\n // agree, and lane aggregates stay consistent with the list endpoint.\n intersectIds([])\n } else {\n const searchPattern = `%${escapeLikePattern(query.search)}%`\n filters.$or = [\n { title: { $ilike: searchPattern } },\n { description: { $ilike: searchPattern } },\n ]\n }\n }\n\n const statusList = query.status ? normalizeStringList(query.status) : []\n if (statusList.length > 0) {\n filters.status = statusList.length === 1 ? { $eq: statusList[0] } : { $in: statusList }\n }\n\n if (query.pipelineStage) {\n filters.pipeline_stage = { $eq: query.pipelineStage }\n }\n\n const pipelineIds = query.pipelineId ? normalizeUuidList([query.pipelineId]) : []\n if (pipelineIds.length > 0) {\n filters.pipeline_id = pipelineIds.length === 1 ? { $eq: pipelineIds[0] } : { $in: pipelineIds }\n }\n\n if (query.pipelineStageId === '__unassigned') {\n filters.pipeline_stage_id = { $eq: null }\n } else if (query.pipelineStageId) {\n filters.pipeline_stage_id = { $eq: query.pipelineStageId }\n }\n\n const ownerUserIds = query.ownerUserId ? normalizeUuidList([query.ownerUserId]) : []\n if (ownerUserIds.length > 0) {\n filters.owner_user_id =\n ownerUserIds.length === 1 ? { $eq: ownerUserIds[0] } : { $in: ownerUserIds }\n }\n\n // Currency filter (additive; sourced by the kanban Currency popover). Codes are\n // normalised to upper-case so `usd` / `USD` / mixed-case query params all hit the same\n // stored value (the deals API persists `value_currency` upper-case already).\n const currencyCodes = query.valueCurrency ? normalizeCurrencyList(query.valueCurrency) : []\n if (currencyCodes.length > 0) {\n filters.value_currency =\n currencyCodes.length === 1 ? { $eq: currencyCodes[0] } : { $in: currencyCodes }\n }\n\n const expectedCloseFrom = parseDateInput(query.expectedCloseAtFrom)\n const expectedCloseTo = parseDateInput(query.expectedCloseAtTo)\n if (expectedCloseFrom || expectedCloseTo) {\n const range: Record<string, Date> = {}\n if (expectedCloseFrom) range.$gte = expectedCloseFrom\n if (expectedCloseTo) range.$lte = expectedCloseTo\n filters.expected_close_at = range\n }\n\n if (query.isOverdue && !query.needsAttention) {\n const today = new Date()\n today.setHours(0, 0, 0, 0)\n if (statusList.length === 0) {\n filters.status = { $eq: 'open' }\n }\n const existingRange =\n filters.expected_close_at && typeof filters.expected_close_at === 'object'\n ? (filters.expected_close_at as Record<string, Date>)\n : {}\n existingRange.$lt = today\n filters.expected_close_at = existingRange\n }\n\n if (query.isStuck && !query.needsAttention && ctx) {\n const tenantId = ctx.auth?.tenantId\n // CrudCtx.auth carries `orgId` (not `organizationId`). The previous code referenced\n // `organizationId` which is always `undefined`, so the typeof check below silently\n // skipped the entire isStuck branch \u2014 `?isStuck=true` was a no-op on this endpoint.\n const organizationId = ctx.auth?.orgId\n if (typeof tenantId === 'string' && typeof organizationId === 'string') {\n const em = ctx.container.resolve<EntityManager>('em')\n const stuckIds = await fetchStuckDealIds(em, organizationId, tenantId)\n intersectIds(stuckIds)\n }\n }\n\n if (query.needsAttention && ctx) {\n const tenantId = ctx.auth?.tenantId\n const organizationId = ctx.auth?.orgId\n if (typeof tenantId === 'string' && typeof organizationId === 'string') {\n const em = ctx.container.resolve<EntityManager>('em')\n const attentionIds = await fetchNeedAttentionDealIds(em, organizationId, tenantId)\n intersectIds(attentionIds)\n }\n }\n\n // Pre-pagination association filter. Must run on the FULL dataset (before pagination),\n // otherwise matching deals on later pages disappear and `total` would be wrong. Read the\n // raw URL too so legacy `?personEntityId=` / `?companyEntityId=` keep working alongside the\n // canonical `?personId=` / `?companyId=`.\n const url = ctx?.request ? new URL(ctx.request.url) : null\n const personCandidates: unknown[] = [query.personId, query.personEntityId]\n const companyCandidates: unknown[] = [query.companyId, query.companyEntityId]\n if (url) {\n personCandidates.push(url.searchParams.getAll('personId'))\n personCandidates.push(url.searchParams.getAll('personEntityId'))\n companyCandidates.push(url.searchParams.getAll('companyId'))\n companyCandidates.push(url.searchParams.getAll('companyEntityId'))\n }\n const selectedPersonIds = normalizeUuidList(personCandidates)\n const selectedCompanyIds = normalizeUuidList(companyCandidates)\n if ((selectedPersonIds.length > 0 || selectedCompanyIds.length > 0) && ctx) {\n const tenantId = ctx.auth?.tenantId\n // `ctx.auth` exposes `orgId` (see AuthContext in @open-mercato/shared/lib/auth/server).\n // Read it under the correct key \u2014 the previous code's `organizationId` would always be\n // `undefined`, silently disabling association filtering on the deals list endpoint.\n const organizationId = ctx.auth?.orgId\n if (typeof tenantId === 'string' && typeof organizationId === 'string') {\n const em = ctx.container.resolve<EntityManager>('em')\n const matchedIds = await fetchDealIdsMatchingAssociations(\n em,\n organizationId,\n tenantId,\n selectedPersonIds,\n selectedCompanyIds,\n )\n // intersectIds with empty array \u2192 no rows; collapses the page to zero, total stays correct.\n intersectIds(matchedIds)\n }\n }\n\n if (ctx && advancedFilterTree) {\n const advancedFilters = mergeAdvancedFilterTree({ ...filters }, advancedFilterTree)\n const matchedIds = await findMatchingEntityIdsWithQueryEngine({\n ctx,\n entityId: E.customers.customer_deal,\n filters: advancedFilters,\n })\n if (matchedIds !== null) {\n intersectIds(matchedIds)\n }\n }\n\n if (restrictedIds !== null) {\n applyEntityIdRestriction(filters, restrictedIds)\n }\n\n return filters\n}\n\nconst crud = makeCrudRoute<unknown, unknown, DealListQuery>({\n metadata: routeMetadata,\n orm: {\n entity: CustomerDeal,\n idField: 'id',\n orgField: 'organizationId',\n tenantField: 'tenantId',\n softDeleteField: 'deletedAt',\n },\n indexer: {\n entityType: E.customers.customer_deal,\n },\n enrichers: { entityId: 'customers.deal' },\n list: {\n schema: dealListQuerySchema,\n entityId: E.customers.customer_deal,\n fields: [\n 'id',\n 'title',\n 'description',\n 'status',\n 'pipeline_stage',\n 'pipeline_id',\n 'pipeline_stage_id',\n 'value_amount',\n 'value_currency',\n 'probability',\n 'expected_close_at',\n 'owner_user_id',\n 'source',\n 'closure_outcome',\n 'loss_reason_id',\n 'loss_notes',\n 'organization_id',\n 'tenant_id',\n 'created_at',\n 'updated_at',\n ],\n decorateCustomFields: {\n entityIds: E.customers.customer_deal,\n },\n sortFieldMap: {\n createdAt: 'created_at',\n updatedAt: 'updated_at',\n title: 'title',\n value: 'value_amount',\n probability: 'probability',\n expectedCloseAt: 'expected_close_at',\n },\n buildFilters: buildDealListFilters,\n },\n actions: {\n create: {\n commandId: 'customers.deals.create',\n schema: rawBodySchema,\n mapInput: async ({ raw, ctx }) => {\n const { translate } = await resolveTranslations()\n return parseScopedCommandInput(dealCreateSchema, raw ?? {}, ctx, translate)\n },\n response: ({ result }) => ({ id: result?.dealId ?? result?.id ?? null }),\n status: 201,\n },\n update: {\n commandId: 'customers.deals.update',\n schema: rawBodySchema,\n mapInput: async ({ raw, ctx }) => {\n const { translate } = await resolveTranslations()\n return parseScopedCommandInput(dealUpdateSchema, raw ?? {}, ctx, translate)\n },\n response: () => ({ ok: true }),\n },\n delete: {\n commandId: 'customers.deals.delete',\n schema: rawBodySchema,\n mapInput: async ({ parsed, ctx }) => {\n const { translate } = await resolveTranslations()\n const id =\n parsed?.body?.id ??\n parsed?.id ??\n parsed?.query?.id ??\n (ctx.request ? new URL(ctx.request.url).searchParams.get('id') : null)\n if (!id) throw new CrudHttpError(400, { error: translate('customers.errors.deal_required', 'Deal id is required') })\n return { id }\n },\n response: () => ({ ok: true }),\n },\n },\n hooks: {\n // afterList only DECORATES results with `people`/`companies` arrays \u2014 it must not filter,\n // because filtering after pagination would drop deals on later pages and rewrite `total`\n // to a misleading value. Association filtering happens pre-pagination in `buildFilters`\n // via `fetchDealIdsMatchingAssociations`.\n afterList: async (payload, ctx) => {\n const items = Array.isArray(payload.items) ? payload.items : []\n if (!items.length) return\n const scopeSource = (items[0] ?? {}) as Record<string, unknown>\n const tenantIdRaw = scopeSource.tenantId ?? scopeSource.tenant_id\n const fallbackTenantId = (typeof tenantIdRaw === 'string' && tenantIdRaw.trim().length ? tenantIdRaw : null) ?? ctx.auth?.tenantId ?? null\n const orgIdRaw = scopeSource.organizationId ?? scopeSource.organization_id\n const fallbackOrganizationId = (typeof orgIdRaw === 'string' && orgIdRaw.trim().length ? orgIdRaw : null) ?? ctx.auth?.orgId ?? null\n const ids = items\n .map((item: unknown) => {\n if (!item || typeof item !== 'object') return null\n const candidate = (item as Record<string, unknown>).id\n return typeof candidate === 'string' && candidate.trim().length ? candidate : null\n })\n .filter((value: string | null): value is string => typeof value === 'string' && value.length > 0)\n if (!ids.length) {\n payload.items = []\n payload.total = 0\n return\n }\n try {\n const em = (ctx.container.resolve('em') as EntityManager)\n const [allPersonLinks, allCompanyLinks] = await Promise.all([\n findWithDecryption(\n em,\n CustomerDealPersonLink,\n { deal: { $in: ids } },\n { populate: ['person'] },\n { tenantId: fallbackTenantId, organizationId: fallbackOrganizationId },\n ),\n findWithDecryption(\n em,\n CustomerDealCompanyLink,\n { deal: { $in: ids } },\n { populate: ['company'] },\n { tenantId: fallbackTenantId, organizationId: fallbackOrganizationId },\n ),\n ])\n\n const personAssignments = new Map<string, { id: string; label: string }[]>()\n allPersonLinks.forEach((link) => {\n const deal = link.deal\n const dealRecord = deal && typeof deal === 'object' ? deal as unknown as Record<string, unknown> : null\n const dealId = typeof deal === 'string' ? deal\n : dealRecord && typeof dealRecord.id === 'string' ? dealRecord.id\n : null\n if (!dealId) return\n const personRef = link.person\n const personRecord = personRef && typeof personRef === 'object' ? personRef as unknown as Record<string, unknown> : null\n const personId = typeof personRef === 'string' ? personRef\n : personRecord && typeof personRecord.id === 'string' ? personRecord.id\n : null\n if (!personId) return\n const label = personRecord && typeof personRecord.displayName === 'string'\n ? personRecord.displayName\n : ''\n const bucket = personAssignments.get(dealId) ?? []\n if (!bucket.some((entry) => entry.id === personId)) {\n bucket.push({ id: personId, label })\n personAssignments.set(dealId, bucket)\n }\n })\n\n const companyAssignments = new Map<string, { id: string; label: string }[]>()\n allCompanyLinks.forEach((link) => {\n const deal = link.deal\n const dealRecord = deal && typeof deal === 'object' ? deal as unknown as Record<string, unknown> : null\n const dealId = typeof deal === 'string' ? deal\n : dealRecord && typeof dealRecord.id === 'string' ? dealRecord.id\n : null\n if (!dealId) return\n const companyRef = link.company\n const companyRecord = companyRef && typeof companyRef === 'object' ? companyRef as unknown as Record<string, unknown> : null\n const companyId = typeof companyRef === 'string' ? companyRef\n : companyRecord && typeof companyRecord.id === 'string' ? companyRecord.id\n : null\n if (!companyId) return\n const label = companyRecord && typeof companyRecord.displayName === 'string'\n ? companyRecord.displayName\n : ''\n const bucket = companyAssignments.get(dealId) ?? []\n if (!bucket.some((entry) => entry.id === companyId)) {\n bucket.push({ id: companyId, label })\n companyAssignments.set(dealId, bucket)\n }\n })\n\n const enhancedItems = items\n .map((item: unknown) => {\n if (!item || typeof item !== 'object') return null\n const data = item as Record<string, unknown>\n const candidate = typeof data.id === 'string' ? data.id : null\n if (!candidate || !candidate.trim().length) return null\n const people = personAssignments.get(candidate) ?? []\n const companies = companyAssignments.get(candidate) ?? []\n const tenantIdRaw =\n typeof data.tenantId === 'string'\n ? data.tenantId\n : typeof data.tenant_id === 'string'\n ? data.tenant_id\n : null\n const organizationIdRaw =\n typeof data.organizationId === 'string'\n ? data.organizationId\n : typeof data.organization_id === 'string'\n ? data.organization_id\n : null\n const tenantId = tenantIdRaw && tenantIdRaw.trim().length ? tenantIdRaw.trim() : null\n const organizationId = organizationIdRaw && organizationIdRaw.trim().length ? organizationIdRaw.trim() : null\n return {\n ...data,\n personIds: people.map((entry) => entry.id),\n people,\n companyIds: companies.map((entry) => entry.id),\n companies,\n tenantId,\n organizationId,\n }\n })\n .filter(\n (item: Record<string, unknown> | null): item is Record<string, unknown> => item !== null,\n )\n\n payload.items = enhancedItems\n } catch (err) {\n // We swallow rather than fail the request because the kanban is still useful without\n // people/companies labels (cards just lose their company pill). Tag every item with\n // `_associations: { ok: false }` so a future surface can render a degraded-state hint\n // instead of silently showing cards without company badges.\n console.warn('[customers.deals] failed to decorate items with person/company links', err)\n payload.items = items.map((item: unknown) => {\n if (!item || typeof item !== 'object') return item\n return {\n ...(item as Record<string, unknown>),\n _associations: {\n ok: false,\n reason: err instanceof Error ? err.message : 'unknown',\n },\n }\n })\n }\n },\n },\n})\n\nconst { POST, PUT, DELETE } = crud\n\nexport { POST, PUT, DELETE }\nexport const GET = crud.GET\n\nconst dealAssociationSchema = z.object({\n id: z.string().uuid(),\n label: z.string().nullable(),\n})\n\nconst dealListItemSchema = z\n .object({\n id: z.string().uuid(),\n title: z.string().nullable(),\n description: z.string().nullable().optional(),\n status: z.string().nullable().optional(),\n pipeline_stage: z.string().nullable().optional(),\n pipeline_id: z.string().uuid().nullable().optional(),\n pipeline_stage_id: z.string().uuid().nullable().optional(),\n value_amount: z.number().nullable().optional(),\n value_currency: z.string().nullable().optional(),\n probability: z.number().nullable().optional(),\n expected_close_at: z.string().nullable().optional(),\n owner_user_id: z.string().uuid().nullable().optional(),\n source: z.string().nullable().optional(),\n organization_id: z.string().uuid().nullable().optional(),\n tenant_id: z.string().uuid().nullable().optional(),\n created_at: z.string().nullable().optional(),\n updated_at: z.string().nullable().optional(),\n personIds: z.array(z.string().uuid()).optional(),\n people: z.array(dealAssociationSchema).optional(),\n companyIds: z.array(z.string().uuid()).optional(),\n companies: z.array(dealAssociationSchema).optional(),\n organizationId: z.string().uuid().nullable().optional(),\n tenantId: z.string().uuid().nullable().optional(),\n })\n .passthrough()\n\nconst dealCreateResponseSchema = z.object({\n id: z.string().uuid().nullable(),\n})\n\nexport const openApi = createCustomersCrudOpenApi({\n resourceName: 'Deal',\n querySchema: dealListQuerySchema,\n listResponseSchema: createPagedListResponseSchema(dealListItemSchema),\n create: {\n schema: dealCreateSchema,\n responseSchema: dealCreateResponseSchema,\n description: 'Creates a sales deal, optionally associating people and companies.',\n },\n update: {\n schema: dealUpdateSchema,\n responseSchema: defaultOkResponseSchema,\n description: 'Updates pipeline position, metadata, or associations for an existing deal.',\n },\n del: {\n schema: z.object({ id: z.string().uuid() }),\n responseSchema: defaultOkResponseSchema,\n description: 'Deletes a deal by `id`. The identifier may be provided in the body or query parameters.',\n },\n})\n"],
5
+ "mappings": "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",
6
6
  "names": ["parsed", "tenantIdRaw"]
7
7
  }
@@ -0,0 +1,402 @@
1
+ import { NextResponse } from "next/server";
2
+ import { z } from "zod";
3
+ import { getAuthFromRequest } from "@open-mercato/shared/lib/auth/server";
4
+ import { createRequestContainer } from "@open-mercato/shared/lib/di/container";
5
+ import { resolveOrganizationScopeForRequest } from "@open-mercato/core/modules/directory/utils/organizationScope";
6
+ import { fetchStuckDealIds } from "../../../lib/stuckDeals.js";
7
+ import {
8
+ computeDelta,
9
+ convertSumsToBase,
10
+ getPreviousQuarterWindow,
11
+ getQuarterWindow,
12
+ getTrailingMonths
13
+ } from "../../../lib/dealsMetrics.js";
14
+ const metadata = {
15
+ GET: { requireAuth: true, requireFeatures: ["customers.deals.view"] }
16
+ };
17
+ const OPEN_STATUSES = ["open", "in_progress"];
18
+ const TRAILING_MONTHS = 6;
19
+ const TOP_OWNERS = 5;
20
+ const deltaSchema = z.object({
21
+ value: z.number(),
22
+ direction: z.enum(["up", "down", "unchanged"])
23
+ });
24
+ const stageBreakdownSchema = z.object({
25
+ stage: z.string().nullable(),
26
+ count: z.number(),
27
+ value: z.number()
28
+ });
29
+ const ownerCountSchema = z.object({
30
+ id: z.string(),
31
+ count: z.number()
32
+ });
33
+ const winRatePointSchema = z.object({
34
+ period: z.string(),
35
+ rate: z.number()
36
+ });
37
+ const summaryResponseSchema = z.object({
38
+ baseCurrencyCode: z.string().nullable(),
39
+ convertedAll: z.boolean(),
40
+ missingRateCurrencies: z.array(z.string()),
41
+ pipelineValue: z.object({
42
+ value: z.number(),
43
+ delta: deltaSchema,
44
+ stages: z.array(stageBreakdownSchema)
45
+ }),
46
+ activeDeals: z.object({
47
+ value: z.number(),
48
+ delta: deltaSchema,
49
+ ownersCount: z.number(),
50
+ needAttention: z.number(),
51
+ owners: z.array(ownerCountSchema),
52
+ ownersOverflow: z.number()
53
+ }),
54
+ wonThisQuarter: z.object({
55
+ value: z.number(),
56
+ delta: deltaSchema,
57
+ dealsClosed: z.number(),
58
+ avgDeal: z.number()
59
+ }),
60
+ winRate: z.object({
61
+ value: z.number(),
62
+ deltaPp: z.number(),
63
+ direction: z.enum(["up", "down", "unchanged"]),
64
+ previousValue: z.number(),
65
+ series: z.array(winRatePointSchema)
66
+ })
67
+ });
68
+ const summaryErrorSchema = z.object({
69
+ error: z.string()
70
+ });
71
+ const openApi = {
72
+ tag: "Customers",
73
+ summary: "Deals KPI summary",
74
+ methods: {
75
+ GET: {
76
+ summary: "Pipeline KPI metrics with period-over-period deltas for the deals list",
77
+ description: "Returns the four list-level KPI cards (pipeline value, active deals, won this quarter, win rate) with quarter-over-quarter deltas, per-stage open-pipeline breakdown, top owners, and a 6-month win-rate series. Values are converted to the tenant base currency where rates are available; partial conversions are disclosed via convertedAll/missingRateCurrencies.",
78
+ responses: [
79
+ { status: 200, description: "Deals KPI summary payload", schema: summaryResponseSchema }
80
+ ],
81
+ errors: [
82
+ { status: 401, description: "Unauthorized", schema: summaryErrorSchema }
83
+ ]
84
+ }
85
+ }
86
+ };
87
+ function toNumber(value) {
88
+ const parsed = Number(value ?? 0);
89
+ return Number.isFinite(parsed) ? parsed : 0;
90
+ }
91
+ function winRate(won, lost) {
92
+ const denom = won + lost;
93
+ if (denom <= 0) return 0;
94
+ return Math.round(100 * won / denom);
95
+ }
96
+ function sumsByCurrency(entries) {
97
+ const byCurrency = /* @__PURE__ */ new Map();
98
+ for (const entry of entries) {
99
+ const currency = (entry.currency ?? "").toString().trim().toUpperCase();
100
+ if (!currency) continue;
101
+ byCurrency.set(currency, (byCurrency.get(currency) ?? 0) + entry.total);
102
+ }
103
+ return Array.from(byCurrency.entries()).map(([currency, total]) => ({ currency, total }));
104
+ }
105
+ async function GET(req) {
106
+ const auth = await getAuthFromRequest(req);
107
+ if (!auth?.tenantId || !auth.orgId) {
108
+ return NextResponse.json({ error: "Unauthorized" }, { status: 401 });
109
+ }
110
+ const container = await createRequestContainer();
111
+ const em = container.resolve("em");
112
+ const scope = await resolveOrganizationScopeForRequest({ container, auth, request: req });
113
+ const effectiveTenantId = scope.tenantId ?? auth.tenantId;
114
+ const orgFilterIds = Array.isArray(scope.filterIds) && scope.filterIds.length > 0 ? scope.filterIds.filter((id) => typeof id === "string" && id.length > 0) : auth.orgId ? [auth.orgId] : [];
115
+ if (!effectiveTenantId || orgFilterIds.length === 0) {
116
+ return NextResponse.json({ error: "Unauthorized" }, { status: 401 });
117
+ }
118
+ const today = /* @__PURE__ */ new Date();
119
+ const currentQuarter = getQuarterWindow(today);
120
+ const previousQuarter = getPreviousQuarterWindow(today);
121
+ const trailingMonths = getTrailingMonths(today, TRAILING_MONTHS);
122
+ const seriesStart = trailingMonths[0]?.start ?? currentQuarter.start;
123
+ const connection = em.getConnection();
124
+ const baseCurrency = await connection.execute(
125
+ `SELECT code FROM currencies WHERE tenant_id = ? AND organization_id = ? AND is_base = true AND deleted_at IS NULL LIMIT 1`,
126
+ [effectiveTenantId, orgFilterIds[0]]
127
+ );
128
+ const baseCurrencyCode = baseCurrency[0]?.code ?? null;
129
+ const orgPlaceholders = orgFilterIds.map(() => "?").join(",");
130
+ const scopeWhere = `tenant_id = ? AND organization_id IN (${orgPlaceholders}) AND deleted_at IS NULL`;
131
+ const scopeValues = [effectiveTenantId, ...orgFilterIds];
132
+ const openPlaceholders = OPEN_STATUSES.map(() => "?").join(",");
133
+ const openRows = await connection.execute(
134
+ `SELECT
135
+ pipeline_stage AS stage,
136
+ UPPER(COALESCE(value_currency, '')) AS currency,
137
+ COALESCE(SUM(value_amount), 0) AS total,
138
+ COUNT(*) AS count,
139
+ owner_user_id
140
+ FROM customer_deals
141
+ WHERE ${scopeWhere} AND status IN (${openPlaceholders})
142
+ GROUP BY pipeline_stage, UPPER(COALESCE(value_currency, '')), owner_user_id`,
143
+ [...scopeValues, ...OPEN_STATUSES]
144
+ );
145
+ const inflowRows = await connection.execute(
146
+ `SELECT
147
+ UPPER(COALESCE(value_currency, '')) AS currency,
148
+ COALESCE(SUM(value_amount) FILTER (WHERE created_at >= ? AND created_at < ?), 0) AS current_total,
149
+ COUNT(*) FILTER (WHERE created_at >= ? AND created_at < ?) AS current_count,
150
+ COALESCE(SUM(value_amount) FILTER (WHERE created_at >= ? AND created_at < ?), 0) AS previous_total,
151
+ COUNT(*) FILTER (WHERE created_at >= ? AND created_at < ?) AS previous_count
152
+ FROM customer_deals
153
+ WHERE ${scopeWhere} AND status IN (${openPlaceholders})
154
+ GROUP BY UPPER(COALESCE(value_currency, ''))`,
155
+ [
156
+ currentQuarter.start.toISOString(),
157
+ currentQuarter.end.toISOString(),
158
+ currentQuarter.start.toISOString(),
159
+ currentQuarter.end.toISOString(),
160
+ previousQuarter.start.toISOString(),
161
+ previousQuarter.end.toISOString(),
162
+ previousQuarter.start.toISOString(),
163
+ previousQuarter.end.toISOString(),
164
+ ...scopeValues,
165
+ ...OPEN_STATUSES
166
+ ]
167
+ );
168
+ const wonRows = await connection.execute(
169
+ `SELECT
170
+ UPPER(COALESCE(value_currency, '')) AS currency,
171
+ COALESCE(SUM(value_amount) FILTER (WHERE updated_at >= ? AND updated_at < ?), 0) AS current_total,
172
+ COUNT(*) FILTER (WHERE updated_at >= ? AND updated_at < ?) AS current_count,
173
+ COALESCE(SUM(value_amount) FILTER (WHERE updated_at >= ? AND updated_at < ?), 0) AS previous_total,
174
+ COUNT(*) FILTER (WHERE updated_at >= ? AND updated_at < ?) AS previous_count
175
+ FROM customer_deals
176
+ WHERE ${scopeWhere} AND (status = 'win' OR closure_outcome = 'won')
177
+ GROUP BY UPPER(COALESCE(value_currency, ''))`,
178
+ [
179
+ currentQuarter.start.toISOString(),
180
+ currentQuarter.end.toISOString(),
181
+ currentQuarter.start.toISOString(),
182
+ currentQuarter.end.toISOString(),
183
+ previousQuarter.start.toISOString(),
184
+ previousQuarter.end.toISOString(),
185
+ previousQuarter.start.toISOString(),
186
+ previousQuarter.end.toISOString(),
187
+ ...scopeValues
188
+ ]
189
+ );
190
+ const winLossRows = await connection.execute(
191
+ `SELECT
192
+ COUNT(*) FILTER (WHERE (status = 'win' OR closure_outcome = 'won') AND updated_at >= ? AND updated_at < ?) AS current_won,
193
+ COUNT(*) FILTER (WHERE (status = 'loose' OR closure_outcome = 'lost') AND updated_at >= ? AND updated_at < ?) AS current_lost,
194
+ COUNT(*) FILTER (WHERE (status = 'win' OR closure_outcome = 'won') AND updated_at >= ? AND updated_at < ?) AS previous_won,
195
+ COUNT(*) FILTER (WHERE (status = 'loose' OR closure_outcome = 'lost') AND updated_at >= ? AND updated_at < ?) AS previous_lost
196
+ FROM customer_deals
197
+ WHERE ${scopeWhere}`,
198
+ [
199
+ currentQuarter.start.toISOString(),
200
+ currentQuarter.end.toISOString(),
201
+ currentQuarter.start.toISOString(),
202
+ currentQuarter.end.toISOString(),
203
+ previousQuarter.start.toISOString(),
204
+ previousQuarter.end.toISOString(),
205
+ previousQuarter.start.toISOString(),
206
+ previousQuarter.end.toISOString(),
207
+ ...scopeValues
208
+ ]
209
+ );
210
+ const seriesRows = await connection.execute(
211
+ `SELECT
212
+ to_char(date_trunc('month', updated_at AT TIME ZONE 'UTC'), 'YYYY-MM') AS period,
213
+ COUNT(*) FILTER (WHERE status = 'win' OR closure_outcome = 'won') AS won,
214
+ COUNT(*) FILTER (WHERE status = 'loose' OR closure_outcome = 'lost') AS lost
215
+ FROM customer_deals
216
+ WHERE ${scopeWhere} AND updated_at >= ?
217
+ GROUP BY 1`,
218
+ [...scopeValues, seriesStart.toISOString()]
219
+ );
220
+ const overdueRows = await connection.execute(
221
+ `SELECT id FROM customer_deals
222
+ WHERE ${scopeWhere} AND status = 'open' AND expected_close_at IS NOT NULL AND expected_close_at < CURRENT_DATE`,
223
+ [...scopeValues]
224
+ );
225
+ const stuckIdLists = await Promise.all(
226
+ orgFilterIds.map((orgId) => fetchStuckDealIds(em, orgId, effectiveTenantId))
227
+ );
228
+ const stuckIdSet = /* @__PURE__ */ new Set();
229
+ for (const list of stuckIdLists) for (const id of list) stuckIdSet.add(id);
230
+ let openStuckIds = [];
231
+ if (stuckIdSet.size > 0) {
232
+ const stuckIdValues = Array.from(stuckIdSet);
233
+ const stuckPlaceholders = stuckIdValues.map(() => "?").join(",");
234
+ const openStuckRows = await connection.execute(
235
+ `SELECT id FROM customer_deals
236
+ WHERE ${scopeWhere} AND status IN (${openPlaceholders}) AND id IN (${stuckPlaceholders})`,
237
+ [...scopeValues, ...OPEN_STATUSES, ...stuckIdValues]
238
+ );
239
+ openStuckIds = openStuckRows.map((row) => row.id);
240
+ }
241
+ const attentionIds = /* @__PURE__ */ new Set();
242
+ for (const row of overdueRows) attentionIds.add(row.id);
243
+ for (const id of openStuckIds) attentionIds.add(id);
244
+ const stageMap = /* @__PURE__ */ new Map();
245
+ const openOwnerCounts = /* @__PURE__ */ new Map();
246
+ const openSums = [];
247
+ for (const row of openRows) {
248
+ const stageKey = row.stage ?? "__null__";
249
+ const total = toNumber(row.total);
250
+ const count = toNumber(row.count);
251
+ const currency = (row.currency ?? "").toString().trim().toUpperCase();
252
+ if (!stageMap.has(stageKey)) {
253
+ stageMap.set(stageKey, { stage: row.stage ?? null, count: 0, byCurrency: [] });
254
+ }
255
+ const stageAgg = stageMap.get(stageKey);
256
+ stageAgg.count += count;
257
+ if (currency) stageAgg.byCurrency.push({ currency, total });
258
+ openSums.push({ currency, total });
259
+ if (row.owner_user_id) {
260
+ openOwnerCounts.set(row.owner_user_id, (openOwnerCounts.get(row.owner_user_id) ?? 0) + count);
261
+ }
262
+ }
263
+ const distinctCurrencies = /* @__PURE__ */ new Set();
264
+ const collect = (entries) => {
265
+ for (const entry of entries) {
266
+ const currency = (entry.currency ?? "").toString().trim().toUpperCase();
267
+ if (currency && currency !== baseCurrencyCode) distinctCurrencies.add(currency);
268
+ }
269
+ };
270
+ collect(openSums);
271
+ collect(inflowRows);
272
+ collect(wonRows);
273
+ let rates = /* @__PURE__ */ new Map();
274
+ if (baseCurrencyCode && distinctCurrencies.size > 0) {
275
+ const exchange = container.resolve("exchangeRateService");
276
+ if (exchange) {
277
+ const pairs = Array.from(distinctCurrencies).map((code) => ({
278
+ fromCurrencyCode: code,
279
+ toCurrencyCode: baseCurrencyCode
280
+ }));
281
+ try {
282
+ rates = await exchange.getRates({
283
+ pairs,
284
+ date: today,
285
+ scope: { tenantId: effectiveTenantId, organizationId: orgFilterIds[0] },
286
+ options: { maxDaysBack: 60, autoFetch: false }
287
+ });
288
+ } catch (err) {
289
+ console.warn("[customers.deals.summary] exchange-rate lookup failed; falling back to per-currency totals", err);
290
+ }
291
+ }
292
+ }
293
+ const missingRateCurrencies = /* @__PURE__ */ new Set();
294
+ const trackMissing = (missing) => {
295
+ for (const code of missing) missingRateCurrencies.add(code);
296
+ };
297
+ let convertedAll = true;
298
+ const dominantCurrencyTotal = (entries) => {
299
+ const byCurrency = /* @__PURE__ */ new Map();
300
+ for (const entry of entries) {
301
+ const currency = (entry.currency ?? "").toString().trim().toUpperCase();
302
+ if (!currency) continue;
303
+ byCurrency.set(currency, (byCurrency.get(currency) ?? 0) + entry.total);
304
+ }
305
+ let best = 0;
306
+ for (const total of byCurrency.values()) {
307
+ if (Math.abs(total) > Math.abs(best)) best = total;
308
+ }
309
+ return Math.round(best);
310
+ };
311
+ const convert = (entries) => {
312
+ if (!baseCurrencyCode) {
313
+ convertedAll = false;
314
+ trackMissing(sumsByCurrency(entries).map((entry) => entry.currency));
315
+ return dominantCurrencyTotal(entries);
316
+ }
317
+ const result = convertSumsToBase(sumsByCurrency(entries), baseCurrencyCode, rates);
318
+ if (!result.convertedAll) convertedAll = false;
319
+ trackMissing(result.missingRateCurrencies);
320
+ return result.total;
321
+ };
322
+ const pipelineValueTotal = convert(openSums);
323
+ const stages = Array.from(stageMap.values()).map((stageAgg) => ({
324
+ stage: stageAgg.stage,
325
+ count: stageAgg.count,
326
+ value: convert(stageAgg.byCurrency)
327
+ }));
328
+ const inflowCurrent = convert(inflowRows.map((row) => ({ currency: row.currency, total: toNumber(row.current_total) })));
329
+ const inflowPrevious = convert(inflowRows.map((row) => ({ currency: row.currency, total: toNumber(row.previous_total) })));
330
+ const pipelineDelta = computeDelta(inflowCurrent, inflowPrevious);
331
+ const activeDealsCount = openRows.reduce((sum, row) => sum + toNumber(row.count), 0);
332
+ const ownersCount = openOwnerCounts.size;
333
+ const inflowCurrentCount = inflowRows.reduce((sum, row) => sum + toNumber(row.current_count), 0);
334
+ const inflowPreviousCount = inflowRows.reduce((sum, row) => sum + toNumber(row.previous_count), 0);
335
+ const activeDelta = computeDelta(inflowCurrentCount, inflowPreviousCount);
336
+ const sortedOwners = Array.from(openOwnerCounts.entries()).sort((a, b) => b[1] - a[1] || a[0].localeCompare(b[0]));
337
+ const owners = sortedOwners.slice(0, TOP_OWNERS).map(([id, count]) => ({ id, count }));
338
+ const ownersOverflow = Math.max(0, ownersCount - owners.length);
339
+ const wonCurrent = convert(wonRows.map((row) => ({ currency: row.currency, total: toNumber(row.current_total) })));
340
+ const wonPrevious = convert(wonRows.map((row) => ({ currency: row.currency, total: toNumber(row.previous_total) })));
341
+ const dealsClosed = wonRows.reduce((sum, row) => sum + toNumber(row.current_count), 0);
342
+ const wonDelta = computeDelta(wonCurrent, wonPrevious);
343
+ const avgDeal = dealsClosed > 0 ? Math.round(wonCurrent / dealsClosed) : 0;
344
+ const winLoss = winLossRows[0];
345
+ const currentWon = toNumber(winLoss?.current_won);
346
+ const currentLost = toNumber(winLoss?.current_lost);
347
+ const previousWon = toNumber(winLoss?.previous_won);
348
+ const previousLost = toNumber(winLoss?.previous_lost);
349
+ const winRateValue = winRate(currentWon, currentLost);
350
+ const winRatePrevious = winRate(previousWon, previousLost);
351
+ const deltaPp = winRateValue - winRatePrevious;
352
+ const winRateDirection = deltaPp > 0 ? "up" : deltaPp < 0 ? "down" : "unchanged";
353
+ const seriesByPeriod = /* @__PURE__ */ new Map();
354
+ for (const row of seriesRows) {
355
+ seriesByPeriod.set(row.period, { won: toNumber(row.won), lost: toNumber(row.lost) });
356
+ }
357
+ const series = trailingMonths.map((month) => {
358
+ const point = seriesByPeriod.get(month.label);
359
+ const won = point?.won ?? 0;
360
+ const lost = point?.lost ?? 0;
361
+ const denom = won + lost;
362
+ return { period: month.label, rate: denom > 0 ? won / denom : 0 };
363
+ });
364
+ const response = {
365
+ baseCurrencyCode,
366
+ convertedAll,
367
+ missingRateCurrencies: Array.from(missingRateCurrencies),
368
+ pipelineValue: {
369
+ value: pipelineValueTotal,
370
+ delta: pipelineDelta,
371
+ stages
372
+ },
373
+ activeDeals: {
374
+ value: activeDealsCount,
375
+ delta: activeDelta,
376
+ ownersCount,
377
+ needAttention: attentionIds.size,
378
+ owners,
379
+ ownersOverflow
380
+ },
381
+ wonThisQuarter: {
382
+ value: wonCurrent,
383
+ delta: wonDelta,
384
+ dealsClosed,
385
+ avgDeal
386
+ },
387
+ winRate: {
388
+ value: winRateValue,
389
+ deltaPp,
390
+ direction: winRateDirection,
391
+ previousValue: winRatePrevious,
392
+ series
393
+ }
394
+ };
395
+ return NextResponse.json(response);
396
+ }
397
+ export {
398
+ GET,
399
+ metadata,
400
+ openApi
401
+ };
402
+ //# sourceMappingURL=route.js.map