@open-mercato/core 0.4.2-canary-10c7a8bf2a → 0.4.2-canary-c84cff7ed5

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 (526) hide show
  1. package/dist/generated/entities/notification/index.js +57 -0
  2. package/dist/generated/entities/notification/index.js.map +7 -0
  3. package/dist/generated/entities.ids.generated.js +5 -1
  4. package/dist/generated/entities.ids.generated.js.map +2 -2
  5. package/dist/generated/entity-fields-registry.js +2 -0
  6. package/dist/generated/entity-fields-registry.js.map +2 -2
  7. package/dist/modules/api_docs/frontend/docs/api/page.js +3 -2
  8. package/dist/modules/api_docs/frontend/docs/api/page.js.map +2 -2
  9. package/dist/modules/api_keys/backend/api-keys/page.js +1 -1
  10. package/dist/modules/api_keys/backend/api-keys/page.js.map +2 -2
  11. package/dist/modules/attachments/components/AttachmentLibrary.js +4 -0
  12. package/dist/modules/attachments/components/AttachmentLibrary.js.map +2 -2
  13. package/dist/modules/attachments/components/AttachmentPartitionSettings.js +2 -0
  14. package/dist/modules/attachments/components/AttachmentPartitionSettings.js.map +2 -2
  15. package/dist/modules/auth/api/admin/nav.js +4 -3
  16. package/dist/modules/auth/api/admin/nav.js.map +2 -2
  17. package/dist/modules/auth/api/profile/route.js +157 -0
  18. package/dist/modules/auth/api/profile/route.js.map +7 -0
  19. package/dist/modules/auth/api/reset/confirm.js +25 -2
  20. package/dist/modules/auth/api/reset/confirm.js.map +2 -2
  21. package/dist/modules/auth/api/reset.js +23 -0
  22. package/dist/modules/auth/api/reset.js.map +2 -2
  23. package/dist/modules/auth/api/sidebar/preferences/route.js +14 -9
  24. package/dist/modules/auth/api/sidebar/preferences/route.js.map +2 -2
  25. package/dist/modules/auth/api/users/route.js +4 -2
  26. package/dist/modules/auth/api/users/route.js.map +2 -2
  27. package/dist/modules/auth/backend/auth/profile/page.js +141 -0
  28. package/dist/modules/auth/backend/auth/profile/page.js.map +7 -0
  29. package/dist/modules/auth/backend/auth/profile/page.meta.js +13 -0
  30. package/dist/modules/auth/backend/auth/profile/page.meta.js.map +7 -0
  31. package/dist/modules/auth/backend/roles/page.js +3 -3
  32. package/dist/modules/auth/backend/roles/page.js.map +2 -2
  33. package/dist/modules/auth/backend/users/[id]/edit/page.js +14 -2
  34. package/dist/modules/auth/backend/users/[id]/edit/page.js.map +2 -2
  35. package/dist/modules/auth/backend/users/create/page.js +15 -2
  36. package/dist/modules/auth/backend/users/create/page.js.map +2 -2
  37. package/dist/modules/auth/backend/users/page.js +3 -3
  38. package/dist/modules/auth/backend/users/page.js.map +2 -2
  39. package/dist/modules/auth/cli.js +13 -0
  40. package/dist/modules/auth/cli.js.map +2 -2
  41. package/dist/modules/auth/commands/users.js +59 -2
  42. package/dist/modules/auth/commands/users.js.map +2 -2
  43. package/dist/modules/auth/data/validators.js +4 -2
  44. package/dist/modules/auth/data/validators.js.map +2 -2
  45. package/dist/modules/auth/frontend/reset/[token]/page.js +20 -10
  46. package/dist/modules/auth/frontend/reset/[token]/page.js.map +2 -2
  47. package/dist/modules/auth/lib/setup-app.js +3 -0
  48. package/dist/modules/auth/lib/setup-app.js.map +2 -2
  49. package/dist/modules/auth/notifications.js +112 -0
  50. package/dist/modules/auth/notifications.js.map +7 -0
  51. package/dist/modules/auth/services/authService.js +3 -3
  52. package/dist/modules/auth/services/authService.js.map +2 -2
  53. package/dist/modules/business_rules/backend/rules/page.js +4 -0
  54. package/dist/modules/business_rules/backend/rules/page.js.map +2 -2
  55. package/dist/modules/business_rules/backend/sets/page.js +3 -0
  56. package/dist/modules/business_rules/backend/sets/page.js.map +2 -2
  57. package/dist/modules/business_rules/notifications.js +28 -0
  58. package/dist/modules/business_rules/notifications.js.map +7 -0
  59. package/dist/modules/business_rules/subscribers/rule-execution-failed-notification.js +37 -0
  60. package/dist/modules/business_rules/subscribers/rule-execution-failed-notification.js.map +7 -0
  61. package/dist/modules/catalog/analytics.js +27 -0
  62. package/dist/modules/catalog/analytics.js.map +7 -0
  63. package/dist/modules/catalog/components/PriceKindSettings.js +2 -0
  64. package/dist/modules/catalog/components/PriceKindSettings.js.map +2 -2
  65. package/dist/modules/catalog/components/categories/CategoriesDataTable.js +2 -2
  66. package/dist/modules/catalog/components/categories/CategoriesDataTable.js.map +2 -2
  67. package/dist/modules/catalog/components/products/ProductsDataTable.js +2 -0
  68. package/dist/modules/catalog/components/products/ProductsDataTable.js.map +2 -2
  69. package/dist/modules/catalog/notifications.js +28 -0
  70. package/dist/modules/catalog/notifications.js.map +7 -0
  71. package/dist/modules/catalog/subscribers/low-stock-notification.js +38 -0
  72. package/dist/modules/catalog/subscribers/low-stock-notification.js.map +7 -0
  73. package/dist/modules/configs/cli.js +6 -0
  74. package/dist/modules/configs/cli.js.map +2 -2
  75. package/dist/modules/configs/lib/upgrade-actions.js +18 -0
  76. package/dist/modules/configs/lib/upgrade-actions.js.map +2 -2
  77. package/dist/modules/currencies/backend/currencies/page.js +3 -0
  78. package/dist/modules/currencies/backend/currencies/page.js.map +2 -2
  79. package/dist/modules/currencies/backend/exchange-rates/page.js +2 -0
  80. package/dist/modules/currencies/backend/exchange-rates/page.js.map +2 -2
  81. package/dist/modules/customers/analytics.js +50 -0
  82. package/dist/modules/customers/analytics.js.map +7 -0
  83. package/dist/modules/customers/backend/customers/companies/page.js +3 -0
  84. package/dist/modules/customers/backend/customers/companies/page.js.map +2 -2
  85. package/dist/modules/customers/backend/customers/deals/page.js +3 -0
  86. package/dist/modules/customers/backend/customers/deals/page.js.map +2 -2
  87. package/dist/modules/customers/backend/customers/people/page.js +3 -0
  88. package/dist/modules/customers/backend/customers/people/page.js.map +2 -2
  89. package/dist/modules/customers/commands/deals.js +31 -0
  90. package/dist/modules/customers/commands/deals.js.map +2 -2
  91. package/dist/modules/customers/components/CustomerTodosTable.js +1 -0
  92. package/dist/modules/customers/components/CustomerTodosTable.js.map +2 -2
  93. package/dist/modules/customers/notifications.js +48 -0
  94. package/dist/modules/customers/notifications.js.map +7 -0
  95. package/dist/modules/dashboards/acl.js +2 -1
  96. package/dist/modules/dashboards/acl.js.map +2 -2
  97. package/dist/modules/dashboards/api/widgets/data/route.js +187 -0
  98. package/dist/modules/dashboards/api/widgets/data/route.js.map +7 -0
  99. package/dist/modules/dashboards/cli.js +173 -1
  100. package/dist/modules/dashboards/cli.js.map +2 -2
  101. package/dist/modules/dashboards/di.js +11 -0
  102. package/dist/modules/dashboards/di.js.map +7 -0
  103. package/dist/modules/dashboards/lib/aggregations.js +162 -0
  104. package/dist/modules/dashboards/lib/aggregations.js.map +7 -0
  105. package/dist/modules/dashboards/lib/formatters.js +34 -0
  106. package/dist/modules/dashboards/lib/formatters.js.map +7 -0
  107. package/dist/modules/dashboards/lib/role-widgets.js +58 -0
  108. package/dist/modules/dashboards/lib/role-widgets.js.map +7 -0
  109. package/dist/modules/dashboards/seed/analytics.js +383 -0
  110. package/dist/modules/dashboards/seed/analytics.js.map +7 -0
  111. package/dist/modules/dashboards/services/analyticsRegistry.js +52 -0
  112. package/dist/modules/dashboards/services/analyticsRegistry.js.map +7 -0
  113. package/dist/modules/dashboards/services/widgetDataService.js +207 -0
  114. package/dist/modules/dashboards/services/widgetDataService.js.map +7 -0
  115. package/dist/modules/dashboards/widgets/dashboard/aov-kpi/config.js +18 -0
  116. package/dist/modules/dashboards/widgets/dashboard/aov-kpi/config.js.map +7 -0
  117. package/dist/modules/dashboards/widgets/dashboard/aov-kpi/widget.client.js +128 -0
  118. package/dist/modules/dashboards/widgets/dashboard/aov-kpi/widget.client.js.map +7 -0
  119. package/dist/modules/dashboards/widgets/dashboard/aov-kpi/widget.js +25 -0
  120. package/dist/modules/dashboards/widgets/dashboard/aov-kpi/widget.js.map +7 -0
  121. package/dist/modules/dashboards/widgets/dashboard/new-customers-kpi/config.js +18 -0
  122. package/dist/modules/dashboards/widgets/dashboard/new-customers-kpi/config.js.map +7 -0
  123. package/dist/modules/dashboards/widgets/dashboard/new-customers-kpi/widget.client.js +126 -0
  124. package/dist/modules/dashboards/widgets/dashboard/new-customers-kpi/widget.client.js.map +7 -0
  125. package/dist/modules/dashboards/widgets/dashboard/new-customers-kpi/widget.js +25 -0
  126. package/dist/modules/dashboards/widgets/dashboard/new-customers-kpi/widget.js.map +7 -0
  127. package/dist/modules/dashboards/widgets/dashboard/orders-by-status/config.js +18 -0
  128. package/dist/modules/dashboards/widgets/dashboard/orders-by-status/config.js.map +7 -0
  129. package/dist/modules/dashboards/widgets/dashboard/orders-by-status/widget.client.js +151 -0
  130. package/dist/modules/dashboards/widgets/dashboard/orders-by-status/widget.client.js.map +7 -0
  131. package/dist/modules/dashboards/widgets/dashboard/orders-by-status/widget.js +25 -0
  132. package/dist/modules/dashboards/widgets/dashboard/orders-by-status/widget.js.map +7 -0
  133. package/dist/modules/dashboards/widgets/dashboard/orders-kpi/config.js +18 -0
  134. package/dist/modules/dashboards/widgets/dashboard/orders-kpi/config.js.map +7 -0
  135. package/dist/modules/dashboards/widgets/dashboard/orders-kpi/widget.client.js +126 -0
  136. package/dist/modules/dashboards/widgets/dashboard/orders-kpi/widget.client.js.map +7 -0
  137. package/dist/modules/dashboards/widgets/dashboard/orders-kpi/widget.js +25 -0
  138. package/dist/modules/dashboards/widgets/dashboard/orders-kpi/widget.js.map +7 -0
  139. package/dist/modules/dashboards/widgets/dashboard/pipeline-summary/config.js +16 -0
  140. package/dist/modules/dashboards/widgets/dashboard/pipeline-summary/config.js.map +7 -0
  141. package/dist/modules/dashboards/widgets/dashboard/pipeline-summary/widget.client.js +123 -0
  142. package/dist/modules/dashboards/widgets/dashboard/pipeline-summary/widget.client.js.map +7 -0
  143. package/dist/modules/dashboards/widgets/dashboard/pipeline-summary/widget.js +25 -0
  144. package/dist/modules/dashboards/widgets/dashboard/pipeline-summary/widget.js.map +7 -0
  145. package/dist/modules/dashboards/widgets/dashboard/revenue-kpi/config.js +18 -0
  146. package/dist/modules/dashboards/widgets/dashboard/revenue-kpi/config.js.map +7 -0
  147. package/dist/modules/dashboards/widgets/dashboard/revenue-kpi/widget.client.js +128 -0
  148. package/dist/modules/dashboards/widgets/dashboard/revenue-kpi/widget.client.js.map +7 -0
  149. package/dist/modules/dashboards/widgets/dashboard/revenue-kpi/widget.js +25 -0
  150. package/dist/modules/dashboards/widgets/dashboard/revenue-kpi/widget.js.map +7 -0
  151. package/dist/modules/dashboards/widgets/dashboard/revenue-trend/config.js +21 -0
  152. package/dist/modules/dashboards/widgets/dashboard/revenue-trend/config.js.map +7 -0
  153. package/dist/modules/dashboards/widgets/dashboard/revenue-trend/widget.client.js +211 -0
  154. package/dist/modules/dashboards/widgets/dashboard/revenue-trend/widget.client.js.map +7 -0
  155. package/dist/modules/dashboards/widgets/dashboard/revenue-trend/widget.js +25 -0
  156. package/dist/modules/dashboards/widgets/dashboard/revenue-trend/widget.js.map +7 -0
  157. package/dist/modules/dashboards/widgets/dashboard/sales-by-region/config.js +19 -0
  158. package/dist/modules/dashboards/widgets/dashboard/sales-by-region/config.js.map +7 -0
  159. package/dist/modules/dashboards/widgets/dashboard/sales-by-region/widget.client.js +131 -0
  160. package/dist/modules/dashboards/widgets/dashboard/sales-by-region/widget.client.js.map +7 -0
  161. package/dist/modules/dashboards/widgets/dashboard/sales-by-region/widget.js +25 -0
  162. package/dist/modules/dashboards/widgets/dashboard/sales-by-region/widget.js.map +7 -0
  163. package/dist/modules/dashboards/widgets/dashboard/top-customers/config.js +19 -0
  164. package/dist/modules/dashboards/widgets/dashboard/top-customers/config.js.map +7 -0
  165. package/dist/modules/dashboards/widgets/dashboard/top-customers/widget.client.js +153 -0
  166. package/dist/modules/dashboards/widgets/dashboard/top-customers/widget.client.js.map +7 -0
  167. package/dist/modules/dashboards/widgets/dashboard/top-customers/widget.js +25 -0
  168. package/dist/modules/dashboards/widgets/dashboard/top-customers/widget.js.map +7 -0
  169. package/dist/modules/dashboards/widgets/dashboard/top-products/config.js +22 -0
  170. package/dist/modules/dashboards/widgets/dashboard/top-products/config.js.map +7 -0
  171. package/dist/modules/dashboards/widgets/dashboard/top-products/widget.client.js +180 -0
  172. package/dist/modules/dashboards/widgets/dashboard/top-products/widget.client.js.map +7 -0
  173. package/dist/modules/dashboards/widgets/dashboard/top-products/widget.js +25 -0
  174. package/dist/modules/dashboards/widgets/dashboard/top-products/widget.js.map +7 -0
  175. package/dist/modules/dictionaries/components/DictionaryTable.js +2 -0
  176. package/dist/modules/dictionaries/components/DictionaryTable.js.map +2 -2
  177. package/dist/modules/directory/backend/directory/organizations/page.js +2 -2
  178. package/dist/modules/directory/backend/directory/organizations/page.js.map +2 -2
  179. package/dist/modules/directory/backend/directory/tenants/page.js +2 -2
  180. package/dist/modules/directory/backend/directory/tenants/page.js.map +2 -2
  181. package/dist/modules/entities/backend/entities/user/[entityId]/records/page.js +2 -2
  182. package/dist/modules/entities/backend/entities/user/[entityId]/records/page.js.map +2 -2
  183. package/dist/modules/entities/components/SystemEntitiesTable.js +1 -1
  184. package/dist/modules/entities/components/SystemEntitiesTable.js.map +2 -2
  185. package/dist/modules/entities/components/UserEntitiesTable.js +2 -2
  186. package/dist/modules/entities/components/UserEntitiesTable.js.map +2 -2
  187. package/dist/modules/feature_toggles/components/FeatureTogglesTable.js +3 -3
  188. package/dist/modules/feature_toggles/components/FeatureTogglesTable.js.map +2 -2
  189. package/dist/modules/feature_toggles/components/OverridesTable.js +1 -1
  190. package/dist/modules/feature_toggles/components/OverridesTable.js.map +2 -2
  191. package/dist/modules/notifications/acl.js +11 -0
  192. package/dist/modules/notifications/acl.js.map +7 -0
  193. package/dist/modules/notifications/api/[id]/action/route.js +74 -0
  194. package/dist/modules/notifications/api/[id]/action/route.js.map +7 -0
  195. package/dist/modules/notifications/api/[id]/dismiss/route.js +15 -0
  196. package/dist/modules/notifications/api/[id]/dismiss/route.js.map +7 -0
  197. package/dist/modules/notifications/api/[id]/read/route.js +15 -0
  198. package/dist/modules/notifications/api/[id]/read/route.js.map +7 -0
  199. package/dist/modules/notifications/api/[id]/restore/route.js +53 -0
  200. package/dist/modules/notifications/api/[id]/restore/route.js.map +7 -0
  201. package/dist/modules/notifications/api/batch/route.js +17 -0
  202. package/dist/modules/notifications/api/batch/route.js.map +7 -0
  203. package/dist/modules/notifications/api/feature/route.js +17 -0
  204. package/dist/modules/notifications/api/feature/route.js.map +7 -0
  205. package/dist/modules/notifications/api/mark-all-read/route.js +35 -0
  206. package/dist/modules/notifications/api/mark-all-read/route.js.map +7 -0
  207. package/dist/modules/notifications/api/openapi.js +76 -0
  208. package/dist/modules/notifications/api/openapi.js.map +7 -0
  209. package/dist/modules/notifications/api/role/route.js +17 -0
  210. package/dist/modules/notifications/api/role/route.js.map +7 -0
  211. package/dist/modules/notifications/api/route.js +85 -0
  212. package/dist/modules/notifications/api/route.js.map +7 -0
  213. package/dist/modules/notifications/api/settings/route.js +155 -0
  214. package/dist/modules/notifications/api/settings/route.js.map +7 -0
  215. package/dist/modules/notifications/api/unread-count/route.js +38 -0
  216. package/dist/modules/notifications/api/unread-count/route.js.map +7 -0
  217. package/dist/modules/notifications/backend/config/notifications/page.js +10 -0
  218. package/dist/modules/notifications/backend/config/notifications/page.js.map +7 -0
  219. package/dist/modules/notifications/backend/config/notifications/page.meta.js +24 -0
  220. package/dist/modules/notifications/backend/config/notifications/page.meta.js.map +7 -0
  221. package/dist/modules/notifications/cli.js +16 -0
  222. package/dist/modules/notifications/cli.js.map +7 -0
  223. package/dist/modules/notifications/data/entities.js +112 -0
  224. package/dist/modules/notifications/data/entities.js.map +7 -0
  225. package/dist/modules/notifications/data/validators.js +94 -0
  226. package/dist/modules/notifications/data/validators.js.map +7 -0
  227. package/dist/modules/notifications/di.js +13 -0
  228. package/dist/modules/notifications/di.js.map +7 -0
  229. package/dist/modules/notifications/emails/NotificationEmail.js +58 -0
  230. package/dist/modules/notifications/emails/NotificationEmail.js.map +7 -0
  231. package/dist/modules/notifications/frontend/NotificationInboxPageClient.js +44 -0
  232. package/dist/modules/notifications/frontend/NotificationInboxPageClient.js.map +7 -0
  233. package/dist/modules/notifications/frontend/NotificationSettingsPageClient.js +219 -0
  234. package/dist/modules/notifications/frontend/NotificationSettingsPageClient.js.map +7 -0
  235. package/dist/modules/notifications/index.js +14 -0
  236. package/dist/modules/notifications/index.js.map +7 -0
  237. package/dist/modules/notifications/lib/deliveryConfig.js +105 -0
  238. package/dist/modules/notifications/lib/deliveryConfig.js.map +7 -0
  239. package/dist/modules/notifications/lib/events.js +12 -0
  240. package/dist/modules/notifications/lib/events.js.map +7 -0
  241. package/dist/modules/notifications/lib/notificationBuilder.js +66 -0
  242. package/dist/modules/notifications/lib/notificationBuilder.js.map +7 -0
  243. package/dist/modules/notifications/lib/notificationFactory.js +54 -0
  244. package/dist/modules/notifications/lib/notificationFactory.js.map +7 -0
  245. package/dist/modules/notifications/lib/notificationMapper.js +34 -0
  246. package/dist/modules/notifications/lib/notificationMapper.js.map +7 -0
  247. package/dist/modules/notifications/lib/notificationRecipients.js +35 -0
  248. package/dist/modules/notifications/lib/notificationRecipients.js.map +7 -0
  249. package/dist/modules/notifications/lib/notificationService.js +279 -0
  250. package/dist/modules/notifications/lib/notificationService.js.map +7 -0
  251. package/dist/modules/notifications/lib/routeHelpers.js +101 -0
  252. package/dist/modules/notifications/lib/routeHelpers.js.map +7 -0
  253. package/dist/modules/notifications/lib/safeHref.js +24 -0
  254. package/dist/modules/notifications/lib/safeHref.js.map +7 -0
  255. package/dist/modules/notifications/migrations/Migration20260123000001.js +70 -0
  256. package/dist/modules/notifications/migrations/Migration20260123000001.js.map +7 -0
  257. package/dist/modules/notifications/migrations/Migration20260126150000.js +37 -0
  258. package/dist/modules/notifications/migrations/Migration20260126150000.js.map +7 -0
  259. package/dist/modules/notifications/subscribers/deliver-notification.js +139 -0
  260. package/dist/modules/notifications/subscribers/deliver-notification.js.map +7 -0
  261. package/dist/modules/notifications/workers/create-notification.worker.js +70 -0
  262. package/dist/modules/notifications/workers/create-notification.worker.js.map +7 -0
  263. package/dist/modules/planner/backend/planner/availability-rulesets/page.js +2 -2
  264. package/dist/modules/planner/backend/planner/availability-rulesets/page.js.map +2 -2
  265. package/dist/modules/query_index/components/QueryIndexesTable.js +7 -1
  266. package/dist/modules/query_index/components/QueryIndexesTable.js.map +2 -2
  267. package/dist/modules/resources/backend/resources/resource-types/page.js +2 -2
  268. package/dist/modules/resources/backend/resources/resource-types/page.js.map +2 -2
  269. package/dist/modules/resources/backend/resources/resources/page.js +2 -2
  270. package/dist/modules/resources/backend/resources/resources/page.js.map +2 -2
  271. package/dist/modules/sales/analytics.js +67 -0
  272. package/dist/modules/sales/analytics.js.map +7 -0
  273. package/dist/modules/sales/backend/sales/channels/offers/page.js +2 -0
  274. package/dist/modules/sales/backend/sales/channels/offers/page.js.map +2 -2
  275. package/dist/modules/sales/backend/sales/channels/page.js +2 -0
  276. package/dist/modules/sales/backend/sales/channels/page.js.map +2 -2
  277. package/dist/modules/sales/commands/documents.js +53 -0
  278. package/dist/modules/sales/commands/documents.js.map +2 -2
  279. package/dist/modules/sales/commands/payments.js +26 -0
  280. package/dist/modules/sales/commands/payments.js.map +2 -2
  281. package/dist/modules/sales/components/AdjustmentKindSettings.js +2 -2
  282. package/dist/modules/sales/components/AdjustmentKindSettings.js.map +2 -2
  283. package/dist/modules/sales/components/PaymentMethodsSettings.js +2 -2
  284. package/dist/modules/sales/components/PaymentMethodsSettings.js.map +2 -2
  285. package/dist/modules/sales/components/ShippingMethodsSettings.js +2 -2
  286. package/dist/modules/sales/components/ShippingMethodsSettings.js.map +2 -2
  287. package/dist/modules/sales/components/TaxRatesSettings.js +2 -2
  288. package/dist/modules/sales/components/TaxRatesSettings.js.map +2 -2
  289. package/dist/modules/sales/components/channels/SalesChannelOffersPanel.js +2 -0
  290. package/dist/modules/sales/components/channels/SalesChannelOffersPanel.js.map +2 -2
  291. package/dist/modules/sales/components/documents/AdjustmentsSection.js +2 -0
  292. package/dist/modules/sales/components/documents/AdjustmentsSection.js.map +2 -2
  293. package/dist/modules/sales/components/documents/PaymentsSection.js +2 -1
  294. package/dist/modules/sales/components/documents/PaymentsSection.js.map +2 -2
  295. package/dist/modules/sales/components/documents/SalesDocumentsTable.js +2 -0
  296. package/dist/modules/sales/components/documents/SalesDocumentsTable.js.map +2 -2
  297. package/dist/modules/sales/notifications.client.js +51 -0
  298. package/dist/modules/sales/notifications.client.js.map +7 -0
  299. package/dist/modules/sales/notifications.js +88 -0
  300. package/dist/modules/sales/notifications.js.map +7 -0
  301. package/dist/modules/sales/subscribers/quote-expiring-notification.js +38 -0
  302. package/dist/modules/sales/subscribers/quote-expiring-notification.js.map +7 -0
  303. package/dist/modules/sales/widgets/notifications/SalesOrderCreatedRenderer.js +137 -0
  304. package/dist/modules/sales/widgets/notifications/SalesOrderCreatedRenderer.js.map +7 -0
  305. package/dist/modules/sales/widgets/notifications/SalesQuoteCreatedRenderer.js +137 -0
  306. package/dist/modules/sales/widgets/notifications/SalesQuoteCreatedRenderer.js.map +7 -0
  307. package/dist/modules/sales/widgets/notifications/index.js +7 -0
  308. package/dist/modules/sales/widgets/notifications/index.js.map +7 -0
  309. package/dist/modules/sales/widgets/notifications/useSalesDocumentTotals.js +60 -0
  310. package/dist/modules/sales/widgets/notifications/useSalesDocumentTotals.js.map +7 -0
  311. package/dist/modules/staff/backend/staff/team-members/page.js +1 -1
  312. package/dist/modules/staff/backend/staff/team-members/page.js.map +2 -2
  313. package/dist/modules/staff/backend/staff/team-roles/page.js +2 -2
  314. package/dist/modules/staff/backend/staff/team-roles/page.js.map +2 -2
  315. package/dist/modules/staff/backend/staff/teams/[id]/edit/page.js +2 -2
  316. package/dist/modules/staff/backend/staff/teams/[id]/edit/page.js.map +2 -2
  317. package/dist/modules/staff/backend/staff/teams/page.js +2 -2
  318. package/dist/modules/staff/backend/staff/teams/page.js.map +2 -2
  319. package/dist/modules/staff/commands/leave-requests.js +79 -0
  320. package/dist/modules/staff/commands/leave-requests.js.map +2 -2
  321. package/dist/modules/staff/notifications.js +75 -0
  322. package/dist/modules/staff/notifications.js.map +7 -0
  323. package/dist/modules/workflows/backend/definitions/page.js +5 -0
  324. package/dist/modules/workflows/backend/definitions/page.js.map +2 -2
  325. package/dist/modules/workflows/backend/instances/page.js +3 -0
  326. package/dist/modules/workflows/backend/instances/page.js.map +2 -2
  327. package/dist/modules/workflows/backend/tasks/page.js +3 -0
  328. package/dist/modules/workflows/backend/tasks/page.js.map +2 -2
  329. package/dist/modules/workflows/notifications.js +28 -0
  330. package/dist/modules/workflows/notifications.js.map +7 -0
  331. package/dist/modules/workflows/subscribers/task-assigned-notification.js +38 -0
  332. package/dist/modules/workflows/subscribers/task-assigned-notification.js.map +7 -0
  333. package/generated/entities/notification/index.ts +27 -0
  334. package/generated/entities.ids.generated.ts +5 -1
  335. package/generated/entity-fields-registry.ts +2 -0
  336. package/package.json +2 -2
  337. package/src/modules/api_docs/frontend/docs/api/page.tsx +3 -2
  338. package/src/modules/api_keys/backend/api-keys/page.tsx +1 -1
  339. package/src/modules/attachments/components/AttachmentLibrary.tsx +4 -0
  340. package/src/modules/attachments/components/AttachmentPartitionSettings.tsx +2 -0
  341. package/src/modules/auth/api/admin/nav.ts +10 -6
  342. package/src/modules/auth/api/profile/route.ts +163 -0
  343. package/src/modules/auth/api/reset/confirm.ts +25 -2
  344. package/src/modules/auth/api/reset.ts +23 -0
  345. package/src/modules/auth/api/sidebar/preferences/route.ts +21 -12
  346. package/src/modules/auth/api/users/route.ts +5 -2
  347. package/src/modules/auth/backend/auth/profile/page.meta.ts +9 -0
  348. package/src/modules/auth/backend/auth/profile/page.tsx +174 -0
  349. package/src/modules/auth/backend/roles/page.tsx +3 -3
  350. package/src/modules/auth/backend/users/[id]/edit/page.tsx +18 -2
  351. package/src/modules/auth/backend/users/create/page.tsx +19 -2
  352. package/src/modules/auth/backend/users/page.tsx +3 -3
  353. package/src/modules/auth/cli.ts +14 -0
  354. package/src/modules/auth/commands/users.ts +73 -2
  355. package/src/modules/auth/data/validators.ts +5 -2
  356. package/src/modules/auth/frontend/reset/[token]/page.tsx +24 -11
  357. package/src/modules/auth/i18n/de.json +43 -1
  358. package/src/modules/auth/i18n/en.json +43 -1
  359. package/src/modules/auth/i18n/es.json +43 -1
  360. package/src/modules/auth/i18n/pl.json +43 -1
  361. package/src/modules/auth/lib/setup-app.ts +3 -0
  362. package/src/modules/auth/notifications.ts +109 -0
  363. package/src/modules/auth/services/authService.ts +4 -4
  364. package/src/modules/business_rules/backend/rules/page.tsx +4 -0
  365. package/src/modules/business_rules/backend/sets/page.tsx +3 -0
  366. package/src/modules/business_rules/i18n/en.json +3 -1
  367. package/src/modules/business_rules/notifications.ts +25 -0
  368. package/src/modules/business_rules/subscribers/rule-execution-failed-notification.ts +50 -0
  369. package/src/modules/catalog/analytics.ts +24 -0
  370. package/src/modules/catalog/components/PriceKindSettings.tsx +2 -0
  371. package/src/modules/catalog/components/categories/CategoriesDataTable.tsx +2 -2
  372. package/src/modules/catalog/components/products/ProductsDataTable.tsx +2 -0
  373. package/src/modules/catalog/i18n/en.json +3 -1
  374. package/src/modules/catalog/notifications.ts +25 -0
  375. package/src/modules/catalog/subscribers/low-stock-notification.ts +52 -0
  376. package/src/modules/configs/cli.ts +6 -0
  377. package/src/modules/configs/lib/upgrade-actions.ts +18 -0
  378. package/src/modules/currencies/backend/currencies/page.tsx +3 -0
  379. package/src/modules/currencies/backend/exchange-rates/page.tsx +2 -0
  380. package/src/modules/customers/analytics.ts +47 -0
  381. package/src/modules/customers/backend/customers/companies/page.tsx +3 -0
  382. package/src/modules/customers/backend/customers/deals/page.tsx +3 -0
  383. package/src/modules/customers/backend/customers/people/page.tsx +3 -0
  384. package/src/modules/customers/commands/deals.ts +39 -0
  385. package/src/modules/customers/components/CustomerTodosTable.tsx +1 -0
  386. package/src/modules/customers/i18n/en.json +5 -1
  387. package/src/modules/customers/notifications.ts +44 -0
  388. package/src/modules/dashboards/acl.ts +1 -0
  389. package/src/modules/dashboards/api/widgets/data/route.ts +221 -0
  390. package/src/modules/dashboards/cli.ts +204 -1
  391. package/src/modules/dashboards/di.ts +9 -0
  392. package/src/modules/dashboards/i18n/de.json +115 -1
  393. package/src/modules/dashboards/i18n/en.json +115 -1
  394. package/src/modules/dashboards/i18n/es.json +115 -1
  395. package/src/modules/dashboards/i18n/pl.json +115 -1
  396. package/src/modules/dashboards/lib/__tests__/aggregations.test.ts +327 -0
  397. package/src/modules/dashboards/lib/__tests__/formatters.test.ts +128 -0
  398. package/src/modules/dashboards/lib/aggregations.ts +225 -0
  399. package/src/modules/dashboards/lib/formatters.ts +36 -0
  400. package/src/modules/dashboards/lib/role-widgets.ts +80 -0
  401. package/src/modules/dashboards/seed/analytics.ts +405 -0
  402. package/src/modules/dashboards/services/analyticsRegistry.ts +79 -0
  403. package/src/modules/dashboards/services/widgetDataService.ts +329 -0
  404. package/src/modules/dashboards/widgets/dashboard/aov-kpi/config.ts +20 -0
  405. package/src/modules/dashboards/widgets/dashboard/aov-kpi/widget.client.tsx +135 -0
  406. package/src/modules/dashboards/widgets/dashboard/aov-kpi/widget.ts +24 -0
  407. package/src/modules/dashboards/widgets/dashboard/new-customers-kpi/config.ts +20 -0
  408. package/src/modules/dashboards/widgets/dashboard/new-customers-kpi/widget.client.tsx +133 -0
  409. package/src/modules/dashboards/widgets/dashboard/new-customers-kpi/widget.ts +24 -0
  410. package/src/modules/dashboards/widgets/dashboard/orders-by-status/config.ts +20 -0
  411. package/src/modules/dashboards/widgets/dashboard/orders-by-status/widget.client.tsx +154 -0
  412. package/src/modules/dashboards/widgets/dashboard/orders-by-status/widget.ts +24 -0
  413. package/src/modules/dashboards/widgets/dashboard/orders-kpi/config.ts +20 -0
  414. package/src/modules/dashboards/widgets/dashboard/orders-kpi/widget.client.tsx +133 -0
  415. package/src/modules/dashboards/widgets/dashboard/orders-kpi/widget.ts +24 -0
  416. package/src/modules/dashboards/widgets/dashboard/pipeline-summary/config.ts +17 -0
  417. package/src/modules/dashboards/widgets/dashboard/pipeline-summary/widget.client.tsx +137 -0
  418. package/src/modules/dashboards/widgets/dashboard/pipeline-summary/widget.ts +24 -0
  419. package/src/modules/dashboards/widgets/dashboard/revenue-kpi/config.ts +20 -0
  420. package/src/modules/dashboards/widgets/dashboard/revenue-kpi/widget.client.tsx +135 -0
  421. package/src/modules/dashboards/widgets/dashboard/revenue-kpi/widget.ts +24 -0
  422. package/src/modules/dashboards/widgets/dashboard/revenue-trend/config.ts +24 -0
  423. package/src/modules/dashboards/widgets/dashboard/revenue-trend/widget.client.tsx +220 -0
  424. package/src/modules/dashboards/widgets/dashboard/revenue-trend/widget.ts +24 -0
  425. package/src/modules/dashboards/widgets/dashboard/sales-by-region/config.ts +21 -0
  426. package/src/modules/dashboards/widgets/dashboard/sales-by-region/widget.client.tsx +131 -0
  427. package/src/modules/dashboards/widgets/dashboard/sales-by-region/widget.ts +24 -0
  428. package/src/modules/dashboards/widgets/dashboard/top-customers/config.ts +21 -0
  429. package/src/modules/dashboards/widgets/dashboard/top-customers/widget.client.tsx +161 -0
  430. package/src/modules/dashboards/widgets/dashboard/top-customers/widget.ts +24 -0
  431. package/src/modules/dashboards/widgets/dashboard/top-products/config.ts +27 -0
  432. package/src/modules/dashboards/widgets/dashboard/top-products/widget.client.tsx +181 -0
  433. package/src/modules/dashboards/widgets/dashboard/top-products/widget.ts +24 -0
  434. package/src/modules/dictionaries/components/DictionaryTable.tsx +2 -0
  435. package/src/modules/directory/backend/directory/organizations/page.tsx +2 -2
  436. package/src/modules/directory/backend/directory/tenants/page.tsx +2 -2
  437. package/src/modules/entities/backend/entities/user/[entityId]/records/page.tsx +2 -2
  438. package/src/modules/entities/components/SystemEntitiesTable.tsx +1 -1
  439. package/src/modules/entities/components/UserEntitiesTable.tsx +2 -2
  440. package/src/modules/feature_toggles/components/FeatureTogglesTable.tsx +3 -4
  441. package/src/modules/feature_toggles/components/OverridesTable.tsx +1 -1
  442. package/src/modules/notifications/acl.ts +7 -0
  443. package/src/modules/notifications/api/[id]/action/route.ts +75 -0
  444. package/src/modules/notifications/api/[id]/dismiss/route.ts +12 -0
  445. package/src/modules/notifications/api/[id]/read/route.ts +12 -0
  446. package/src/modules/notifications/api/[id]/restore/route.ts +53 -0
  447. package/src/modules/notifications/api/batch/route.ts +14 -0
  448. package/src/modules/notifications/api/feature/route.ts +14 -0
  449. package/src/modules/notifications/api/mark-all-read/route.ts +34 -0
  450. package/src/modules/notifications/api/openapi.ts +76 -0
  451. package/src/modules/notifications/api/role/route.ts +14 -0
  452. package/src/modules/notifications/api/route.ts +92 -0
  453. package/src/modules/notifications/api/settings/route.ts +157 -0
  454. package/src/modules/notifications/api/unread-count/route.ts +38 -0
  455. package/src/modules/notifications/backend/config/notifications/page.meta.ts +22 -0
  456. package/src/modules/notifications/backend/config/notifications/page.tsx +12 -0
  457. package/src/modules/notifications/cli.ts +18 -0
  458. package/src/modules/notifications/data/entities.ts +99 -0
  459. package/src/modules/notifications/data/validators.ts +110 -0
  460. package/src/modules/notifications/di.ts +11 -0
  461. package/src/modules/notifications/emails/NotificationEmail.tsx +98 -0
  462. package/src/modules/notifications/frontend/NotificationInboxPageClient.tsx +42 -0
  463. package/src/modules/notifications/frontend/NotificationSettingsPageClient.tsx +231 -0
  464. package/src/modules/notifications/i18n/de.json +50 -0
  465. package/src/modules/notifications/i18n/en.json +50 -0
  466. package/src/modules/notifications/i18n/es.json +50 -0
  467. package/src/modules/notifications/i18n/pl.json +50 -0
  468. package/src/modules/notifications/index.ts +12 -0
  469. package/src/modules/notifications/lib/deliveryConfig.ts +145 -0
  470. package/src/modules/notifications/lib/events.ts +48 -0
  471. package/src/modules/notifications/lib/notificationBuilder.ts +121 -0
  472. package/src/modules/notifications/lib/notificationFactory.ts +76 -0
  473. package/src/modules/notifications/lib/notificationMapper.ts +33 -0
  474. package/src/modules/notifications/lib/notificationRecipients.ts +83 -0
  475. package/src/modules/notifications/lib/notificationService.ts +414 -0
  476. package/src/modules/notifications/lib/routeHelpers.ts +151 -0
  477. package/src/modules/notifications/lib/safeHref.ts +29 -0
  478. package/src/modules/notifications/migrations/.snapshot-open-mercato.json +300 -0
  479. package/src/modules/notifications/migrations/Migration20260123000001.ts +73 -0
  480. package/src/modules/notifications/migrations/Migration20260126150000.ts +39 -0
  481. package/src/modules/notifications/subscribers/deliver-notification.ts +175 -0
  482. package/src/modules/notifications/workers/create-notification.worker.ts +122 -0
  483. package/src/modules/planner/backend/planner/availability-rulesets/page.tsx +2 -2
  484. package/src/modules/query_index/components/QueryIndexesTable.tsx +8 -2
  485. package/src/modules/resources/backend/resources/resource-types/page.tsx +2 -2
  486. package/src/modules/resources/backend/resources/resources/page.tsx +2 -2
  487. package/src/modules/sales/analytics.ts +64 -0
  488. package/src/modules/sales/backend/sales/channels/offers/page.tsx +2 -0
  489. package/src/modules/sales/backend/sales/channels/page.tsx +2 -0
  490. package/src/modules/sales/commands/documents.ts +65 -0
  491. package/src/modules/sales/commands/payments.ts +33 -0
  492. package/src/modules/sales/components/AdjustmentKindSettings.tsx +2 -2
  493. package/src/modules/sales/components/PaymentMethodsSettings.tsx +2 -2
  494. package/src/modules/sales/components/ShippingMethodsSettings.tsx +2 -2
  495. package/src/modules/sales/components/TaxRatesSettings.tsx +2 -2
  496. package/src/modules/sales/components/channels/SalesChannelOffersPanel.tsx +2 -0
  497. package/src/modules/sales/components/documents/AdjustmentsSection.tsx +2 -0
  498. package/src/modules/sales/components/documents/PaymentsSection.tsx +2 -1
  499. package/src/modules/sales/components/documents/SalesDocumentsTable.tsx +2 -0
  500. package/src/modules/sales/i18n/de.json +20 -0
  501. package/src/modules/sales/i18n/en.json +25 -1
  502. package/src/modules/sales/i18n/es.json +20 -0
  503. package/src/modules/sales/i18n/pl.json +20 -0
  504. package/src/modules/sales/notifications.client.ts +65 -0
  505. package/src/modules/sales/notifications.ts +82 -0
  506. package/src/modules/sales/subscribers/quote-expiring-notification.ts +53 -0
  507. package/src/modules/sales/widgets/notifications/SalesOrderCreatedRenderer.tsx +156 -0
  508. package/src/modules/sales/widgets/notifications/SalesQuoteCreatedRenderer.tsx +156 -0
  509. package/src/modules/sales/widgets/notifications/index.ts +2 -0
  510. package/src/modules/sales/widgets/notifications/useSalesDocumentTotals.ts +81 -0
  511. package/src/modules/staff/backend/staff/team-members/page.tsx +1 -1
  512. package/src/modules/staff/backend/staff/team-roles/page.tsx +2 -2
  513. package/src/modules/staff/backend/staff/teams/[id]/edit/page.tsx +2 -2
  514. package/src/modules/staff/backend/staff/teams/page.tsx +2 -2
  515. package/src/modules/staff/commands/leave-requests.ts +94 -0
  516. package/src/modules/staff/i18n/de.json +4 -0
  517. package/src/modules/staff/i18n/en.json +9 -1
  518. package/src/modules/staff/i18n/es.json +4 -0
  519. package/src/modules/staff/i18n/pl.json +4 -0
  520. package/src/modules/staff/notifications.ts +71 -0
  521. package/src/modules/workflows/backend/definitions/page.tsx +5 -0
  522. package/src/modules/workflows/backend/instances/page.tsx +4 -1
  523. package/src/modules/workflows/backend/tasks/page.tsx +4 -1
  524. package/src/modules/workflows/i18n/en.json +3 -1
  525. package/src/modules/workflows/notifications.ts +25 -0
  526. package/src/modules/workflows/subscribers/task-assigned-notification.ts +53 -0
@@ -2,6 +2,8 @@ import { createRequestContainer } from "@open-mercato/shared/lib/di/container";
2
2
  import { DashboardRoleWidgets } from "@open-mercato/core/modules/dashboards/data/entities";
3
3
  import { Role } from "@open-mercato/core/modules/auth/data/entities";
4
4
  import { loadAllWidgets } from "@open-mercato/core/modules/dashboards/lib/widgets";
5
+ import { appendWidgetsToRoles, resolveAnalyticsWidgetIds } from "@open-mercato/core/modules/dashboards/lib/role-widgets";
6
+ import { seedAnalyticsData } from "./seed/analytics.js";
5
7
  function parseArgs(rest) {
6
8
  const args = {};
7
9
  for (let i = 0; i < rest.length; i += 2) {
@@ -90,7 +92,177 @@ const seedDefaults = {
90
92
  });
91
93
  }
92
94
  };
93
- var cli_default = [seedDefaults];
95
+ const enableAnalyticsWidgets = {
96
+ command: "enable-analytics-widgets",
97
+ async run(rest) {
98
+ const args = parseArgs(rest);
99
+ const tenantId = args.tenant || args.tenantId || null;
100
+ const organizationId = args.organization || args.organizationId || args.org || null;
101
+ const roleCsv = args.roles || "admin,employee";
102
+ if (!tenantId) {
103
+ console.error("Usage: mercato dashboards enable-analytics-widgets --tenant <tenantId> [--org <orgId>] [--roles admin,employee]");
104
+ return;
105
+ }
106
+ const roleNames = roleCsv.split(",").map((name) => name.trim()).filter(Boolean);
107
+ if (!roleNames.length) {
108
+ console.log("No roles provided, nothing to update.");
109
+ return;
110
+ }
111
+ const { resolve } = await createRequestContainer();
112
+ const em = resolve("em");
113
+ const widgetIds = await resolveAnalyticsWidgetIds();
114
+ const updated = await appendWidgetsToRoles(em, {
115
+ tenantId,
116
+ organizationId,
117
+ roleNames,
118
+ widgetIds
119
+ });
120
+ if (!updated) {
121
+ console.log("No dashboard role widgets updated.");
122
+ }
123
+ }
124
+ };
125
+ const seedAnalytics = {
126
+ command: "seed-analytics",
127
+ async run(rest) {
128
+ const args = parseArgs(rest);
129
+ const tenantId = args.tenant || args.tenantId || null;
130
+ const organizationId = args.organization || args.organizationId || args.org || null;
131
+ const months = args.months ? parseInt(args.months, 10) : 6;
132
+ const ordersPerMonth = args.ordersPerMonth ? parseInt(args.ordersPerMonth, 10) : 50;
133
+ if (!tenantId || !organizationId) {
134
+ console.error("Usage: mercato dashboards seed-analytics --tenant <tenantId> --organization <organizationId> [--months 6] [--ordersPerMonth 50]");
135
+ return;
136
+ }
137
+ const { resolve } = await createRequestContainer();
138
+ const em = resolve("em");
139
+ console.log(`Seeding analytics data for ${months} months with ~${ordersPerMonth} orders/month...`);
140
+ try {
141
+ const result = await em.transactional(
142
+ async (tem) => seedAnalyticsData(tem, { tenantId, organizationId }, { months, ordersPerMonth })
143
+ );
144
+ if (result.orders === 0) {
145
+ console.log("Analytics data already exists. Skipping seed.");
146
+ } else {
147
+ console.log(`Seeded analytics data:`);
148
+ console.log(` - Orders: ${result.orders}`);
149
+ console.log(` - Customers: ${result.customers}`);
150
+ console.log(` - Products: ${result.products}`);
151
+ console.log(` - Deals: ${result.deals}`);
152
+ }
153
+ } catch (error) {
154
+ console.error("Failed to seed analytics data:", error);
155
+ }
156
+ }
157
+ };
158
+ const debugAnalytics = {
159
+ command: "debug-analytics",
160
+ async run(rest) {
161
+ const args = parseArgs(rest);
162
+ const tenantId = args.tenant || args.tenantId || null;
163
+ const organizationId = args.organization || args.organizationId || args.org || null;
164
+ if (!tenantId) {
165
+ console.error("Usage: mercato dashboards debug-analytics --tenant <tenantId> [--organization <organizationId>]");
166
+ return;
167
+ }
168
+ const { resolve } = await createRequestContainer();
169
+ const em = resolve("em");
170
+ const conn = em.getConnection();
171
+ console.log("Checking analytics data...\n");
172
+ const ordersResult = await conn.execute(
173
+ `SELECT COUNT(*) as total, MIN(placed_at) as earliest, MAX(placed_at) as latest
174
+ FROM sales_orders
175
+ WHERE tenant_id = ? AND order_number LIKE 'SO-ANALYTICS-%'`,
176
+ [tenantId]
177
+ );
178
+ console.log("Orders summary:", ordersResult[0]);
179
+ const recentOrders = await conn.execute(
180
+ `SELECT order_number, placed_at, status, grand_total_gross_amount::numeric as total
181
+ FROM sales_orders
182
+ WHERE tenant_id = ? AND order_number LIKE 'SO-ANALYTICS-%'
183
+ ORDER BY placed_at DESC LIMIT 5`,
184
+ [tenantId]
185
+ );
186
+ console.log("\nRecent orders:", recentOrders);
187
+ const januaryOrders = await conn.execute(
188
+ `SELECT COUNT(*) as count, SUM(grand_total_gross_amount::numeric) as total
189
+ FROM sales_orders
190
+ WHERE tenant_id = ?
191
+ AND order_number LIKE 'SO-ANALYTICS-%'
192
+ AND placed_at >= '2026-01-01'
193
+ AND placed_at <= '2026-01-31 23:59:59'`,
194
+ [tenantId]
195
+ );
196
+ console.log("\nJanuary 2026 orders:", januaryOrders[0]);
197
+ const allOrders = await conn.execute(
198
+ `SELECT COUNT(*) as count
199
+ FROM sales_orders
200
+ WHERE tenant_id = ?`,
201
+ [tenantId]
202
+ );
203
+ console.log("\nTotal orders in tenant:", allOrders[0]);
204
+ const orgCheck = await conn.execute(
205
+ `SELECT organization_id, COUNT(*) as count
206
+ FROM sales_orders
207
+ WHERE tenant_id = ? AND order_number LIKE 'SO-ANALYTICS-%'
208
+ GROUP BY organization_id`,
209
+ [tenantId]
210
+ );
211
+ console.log("\nOrders by organization:", orgCheck);
212
+ if (organizationId) {
213
+ const orgOrders = await conn.execute(
214
+ `SELECT COUNT(*) as count, SUM(grand_total_gross_amount::numeric) as total
215
+ FROM sales_orders
216
+ WHERE tenant_id = ?
217
+ AND organization_id = ?
218
+ AND placed_at >= '2026-01-01'
219
+ AND placed_at <= '2026-01-31 23:59:59'`,
220
+ [tenantId, organizationId]
221
+ );
222
+ console.log(`
223
+ January orders for org ${organizationId}:`, orgOrders[0]);
224
+ }
225
+ const nullPlacedAt = await conn.execute(
226
+ `SELECT COUNT(*) as count
227
+ FROM sales_orders
228
+ WHERE tenant_id = ? AND placed_at IS NULL`,
229
+ [tenantId]
230
+ );
231
+ console.log("\nOrders with NULL placed_at:", nullPlacedAt[0]);
232
+ const nonAnalytics = await conn.execute(
233
+ `SELECT order_number, placed_at, status, organization_id, grand_total_gross_amount::numeric as total
234
+ FROM sales_orders
235
+ WHERE tenant_id = ? AND order_number NOT LIKE 'SO-ANALYTICS-%'
236
+ ORDER BY placed_at DESC NULLS LAST LIMIT 10`,
237
+ [tenantId]
238
+ );
239
+ console.log("\nNon-analytics orders:", nonAnalytics);
240
+ console.log("\n--- Simulating widget query for this_month ---");
241
+ const widgetQuery = await conn.execute(
242
+ `SELECT COALESCE(SUM(grand_total_gross_amount::numeric), 0) AS value
243
+ FROM sales_orders
244
+ WHERE tenant_id = ?
245
+ AND organization_id = ANY(?::uuid[])
246
+ AND deleted_at IS NULL
247
+ AND placed_at >= '2026-01-01'
248
+ AND placed_at <= '2026-01-31 23:59:59'`,
249
+ [tenantId, `{${organizationId}}`]
250
+ );
251
+ console.log("Widget query result (revenue sum):", widgetQuery[0]);
252
+ const widgetCountQuery = await conn.execute(
253
+ `SELECT COUNT(*) AS value
254
+ FROM sales_orders
255
+ WHERE tenant_id = ?
256
+ AND organization_id = ANY(?::uuid[])
257
+ AND deleted_at IS NULL
258
+ AND placed_at >= '2026-01-01'
259
+ AND placed_at <= '2026-01-31 23:59:59'`,
260
+ [tenantId, `{${organizationId}}`]
261
+ );
262
+ console.log("Widget query result (order count):", widgetCountQuery[0]);
263
+ }
264
+ };
265
+ var cli_default = [seedDefaults, enableAnalyticsWidgets, seedAnalytics, debugAnalytics];
94
266
  export {
95
267
  cli_default as default,
96
268
  seedDashboardDefaultsForTenant
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "version": 3,
3
3
  "sources": ["../../../src/modules/dashboards/cli.ts"],
4
- "sourcesContent": ["import type { ModuleCli } from '@open-mercato/shared/modules/registry'\nimport { createRequestContainer } from '@open-mercato/shared/lib/di/container'\nimport type { EntityManager } from '@mikro-orm/postgresql'\nimport { DashboardRoleWidgets } from '@open-mercato/core/modules/dashboards/data/entities'\nimport { Role } from '@open-mercato/core/modules/auth/data/entities'\nimport { loadAllWidgets } from '@open-mercato/core/modules/dashboards/lib/widgets'\n\ntype Args = Record<string, string>\n\nfunction parseArgs(rest: string[]): Args {\n const args: Args = {}\n for (let i = 0; i < rest.length; i += 2) {\n const key = rest[i]?.replace(/^--/, '')\n const value = rest[i + 1]\n if (key) args[key] = value ?? ''\n }\n return args\n}\n\nexport async function seedDashboardDefaultsForTenant(\n em: EntityManager,\n {\n tenantId,\n organizationId = null,\n roleNames = ['superadmin', 'admin', 'employee'],\n widgetIds,\n logger,\n }: {\n tenantId: string\n organizationId?: string | null\n roleNames?: string[]\n widgetIds?: string[]\n logger?: (message: string) => void\n },\n): Promise<boolean> {\n if (!tenantId) throw new Error('tenantId is required')\n const log = logger ?? (() => {})\n\n const widgets = await loadAllWidgets()\n const widgetMap = new Map(widgets.map((widget) => [widget.metadata.id, widget]))\n const resolvedWidgetIds = widgetIds && widgetIds.length\n ? widgetIds.filter((id) => widgetMap.has(id))\n : widgets.filter((widget) => widget.metadata.defaultEnabled).map((widget) => widget.metadata.id)\n\n if (!resolvedWidgetIds.length) {\n log('No widgets resolved for dashboard seeding.')\n return false\n }\n\n await em.transactional(async (tem) => {\n for (const roleName of roleNames) {\n const role = await tem.findOne(Role, { name: roleName })\n if (!role) {\n log(`Skipping role \"${roleName}\" (not found)`)\n continue\n }\n const existing = await tem.findOne(DashboardRoleWidgets, {\n roleId: String(role.id),\n tenantId,\n organizationId,\n deletedAt: null,\n })\n if (existing) {\n existing.widgetIdsJson = resolvedWidgetIds\n tem.persist(existing)\n log(`Updated dashboard widgets for role \"${roleName}\"`)\n } else {\n const record = tem.create(DashboardRoleWidgets, {\n roleId: String(role.id),\n tenantId,\n organizationId,\n widgetIdsJson: resolvedWidgetIds,\n createdAt: new Date(),\n updatedAt: null,\n deletedAt: null,\n })\n tem.persist(record)\n log(`Created dashboard widgets for role \"${roleName}\"`)\n }\n }\n })\n\n return true\n}\n\nconst seedDefaults: ModuleCli = {\n command: 'seed-defaults',\n async run(rest) {\n const args = parseArgs(rest)\n const tenantId = args.tenant || args.tenantId || null\n const organizationId = args.organization || args.organizationId || null\n const roleCsv = args.roles || 'superadmin,admin,employee'\n const widgetCsv = args.widgets || ''\n if (!tenantId) {\n console.error('Usage: mercato dashboards seed-defaults --tenant <tenantId> [--roles superadmin,admin,employee] [--widgets id1,id2]')\n return\n }\n\n const roleNames = roleCsv\n .split(',')\n .map((name) => name.trim())\n .filter(Boolean)\n\n if (!roleNames.length) {\n console.log('No roles provided, nothing to seed.')\n return\n }\n\n const { resolve } = await createRequestContainer()\n const em = resolve('em') as any\n\n await seedDashboardDefaultsForTenant(em as EntityManager, {\n tenantId,\n organizationId,\n roleNames,\n widgetIds: widgetCsv ? widgetCsv.split(',').map((id) => id.trim()).filter(Boolean) : undefined,\n logger: (message) => console.log(message),\n })\n },\n}\n\nexport default [seedDefaults]\n"],
5
- "mappings": "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",
4
+ "sourcesContent": ["import type { ModuleCli } from '@open-mercato/shared/modules/registry'\nimport { createRequestContainer } from '@open-mercato/shared/lib/di/container'\nimport type { EntityManager } from '@mikro-orm/postgresql'\nimport { DashboardRoleWidgets } from '@open-mercato/core/modules/dashboards/data/entities'\nimport { Role } from '@open-mercato/core/modules/auth/data/entities'\nimport { loadAllWidgets } from '@open-mercato/core/modules/dashboards/lib/widgets'\nimport { appendWidgetsToRoles, resolveAnalyticsWidgetIds } from '@open-mercato/core/modules/dashboards/lib/role-widgets'\nimport { seedAnalyticsData } from './seed/analytics'\n\ntype Args = Record<string, string>\n\nfunction parseArgs(rest: string[]): Args {\n const args: Args = {}\n for (let i = 0; i < rest.length; i += 2) {\n const key = rest[i]?.replace(/^--/, '')\n const value = rest[i + 1]\n if (key) args[key] = value ?? ''\n }\n return args\n}\n\nexport async function seedDashboardDefaultsForTenant(\n em: EntityManager,\n {\n tenantId,\n organizationId = null,\n roleNames = ['superadmin', 'admin', 'employee'],\n widgetIds,\n logger,\n }: {\n tenantId: string\n organizationId?: string | null\n roleNames?: string[]\n widgetIds?: string[]\n logger?: (message: string) => void\n },\n): Promise<boolean> {\n if (!tenantId) throw new Error('tenantId is required')\n const log = logger ?? (() => {})\n\n const widgets = await loadAllWidgets()\n const widgetMap = new Map(widgets.map((widget) => [widget.metadata.id, widget]))\n const resolvedWidgetIds = widgetIds && widgetIds.length\n ? widgetIds.filter((id) => widgetMap.has(id))\n : widgets.filter((widget) => widget.metadata.defaultEnabled).map((widget) => widget.metadata.id)\n\n if (!resolvedWidgetIds.length) {\n log('No widgets resolved for dashboard seeding.')\n return false\n }\n\n await em.transactional(async (tem) => {\n for (const roleName of roleNames) {\n const role = await tem.findOne(Role, { name: roleName })\n if (!role) {\n log(`Skipping role \"${roleName}\" (not found)`)\n continue\n }\n const existing = await tem.findOne(DashboardRoleWidgets, {\n roleId: String(role.id),\n tenantId,\n organizationId,\n deletedAt: null,\n })\n if (existing) {\n existing.widgetIdsJson = resolvedWidgetIds\n tem.persist(existing)\n log(`Updated dashboard widgets for role \"${roleName}\"`)\n } else {\n const record = tem.create(DashboardRoleWidgets, {\n roleId: String(role.id),\n tenantId,\n organizationId,\n widgetIdsJson: resolvedWidgetIds,\n createdAt: new Date(),\n updatedAt: null,\n deletedAt: null,\n })\n tem.persist(record)\n log(`Created dashboard widgets for role \"${roleName}\"`)\n }\n }\n })\n\n return true\n}\n\nconst seedDefaults: ModuleCli = {\n command: 'seed-defaults',\n async run(rest) {\n const args = parseArgs(rest)\n const tenantId = args.tenant || args.tenantId || null\n const organizationId = args.organization || args.organizationId || null\n const roleCsv = args.roles || 'superadmin,admin,employee'\n const widgetCsv = args.widgets || ''\n if (!tenantId) {\n console.error('Usage: mercato dashboards seed-defaults --tenant <tenantId> [--roles superadmin,admin,employee] [--widgets id1,id2]')\n return\n }\n\n const roleNames = roleCsv\n .split(',')\n .map((name) => name.trim())\n .filter(Boolean)\n\n if (!roleNames.length) {\n console.log('No roles provided, nothing to seed.')\n return\n }\n\n const { resolve } = await createRequestContainer()\n const em = resolve('em') as any\n\n await seedDashboardDefaultsForTenant(em as EntityManager, {\n tenantId,\n organizationId,\n roleNames,\n widgetIds: widgetCsv ? widgetCsv.split(',').map((id) => id.trim()).filter(Boolean) : undefined,\n logger: (message) => console.log(message),\n })\n },\n}\n\nconst enableAnalyticsWidgets: ModuleCli = {\n command: 'enable-analytics-widgets',\n async run(rest) {\n const args = parseArgs(rest)\n const tenantId = args.tenant || args.tenantId || null\n const organizationId = args.organization || args.organizationId || args.org || null\n const roleCsv = args.roles || 'admin,employee'\n if (!tenantId) {\n console.error('Usage: mercato dashboards enable-analytics-widgets --tenant <tenantId> [--org <orgId>] [--roles admin,employee]')\n return\n }\n\n const roleNames = roleCsv\n .split(',')\n .map((name) => name.trim())\n .filter(Boolean)\n\n if (!roleNames.length) {\n console.log('No roles provided, nothing to update.')\n return\n }\n\n const { resolve } = await createRequestContainer()\n const em = resolve('em') as EntityManager\n const widgetIds = await resolveAnalyticsWidgetIds()\n\n const updated = await appendWidgetsToRoles(em, {\n tenantId,\n organizationId,\n roleNames,\n widgetIds,\n })\n\n if (!updated) {\n console.log('No dashboard role widgets updated.')\n }\n },\n}\n\nconst seedAnalytics: ModuleCli = {\n command: 'seed-analytics',\n async run(rest) {\n const args = parseArgs(rest)\n const tenantId = args.tenant || args.tenantId || null\n const organizationId = args.organization || args.organizationId || args.org || null\n const months = args.months ? parseInt(args.months, 10) : 6\n const ordersPerMonth = args.ordersPerMonth ? parseInt(args.ordersPerMonth, 10) : 50\n\n if (!tenantId || !organizationId) {\n console.error('Usage: mercato dashboards seed-analytics --tenant <tenantId> --organization <organizationId> [--months 6] [--ordersPerMonth 50]')\n return\n }\n\n const { resolve } = await createRequestContainer()\n const em = resolve('em') as EntityManager\n\n console.log(`Seeding analytics data for ${months} months with ~${ordersPerMonth} orders/month...`)\n\n try {\n const result = await em.transactional(async (tem) =>\n seedAnalyticsData(tem, { tenantId, organizationId }, { months, ordersPerMonth })\n )\n\n if (result.orders === 0) {\n console.log('Analytics data already exists. Skipping seed.')\n } else {\n console.log(`Seeded analytics data:`)\n console.log(` - Orders: ${result.orders}`)\n console.log(` - Customers: ${result.customers}`)\n console.log(` - Products: ${result.products}`)\n console.log(` - Deals: ${result.deals}`)\n }\n } catch (error) {\n console.error('Failed to seed analytics data:', error)\n }\n },\n}\n\nconst debugAnalytics: ModuleCli = {\n command: 'debug-analytics',\n async run(rest) {\n const args = parseArgs(rest)\n const tenantId = args.tenant || args.tenantId || null\n const organizationId = args.organization || args.organizationId || args.org || null\n\n if (!tenantId) {\n console.error('Usage: mercato dashboards debug-analytics --tenant <tenantId> [--organization <organizationId>]')\n return\n }\n\n const { resolve } = await createRequestContainer()\n const em = resolve('em') as EntityManager\n const conn = em.getConnection()\n\n console.log('Checking analytics data...\\n')\n\n const ordersResult = await conn.execute(\n `SELECT COUNT(*) as total, MIN(placed_at) as earliest, MAX(placed_at) as latest\n FROM sales_orders\n WHERE tenant_id = ? AND order_number LIKE 'SO-ANALYTICS-%'`,\n [tenantId]\n )\n console.log('Orders summary:', ordersResult[0])\n\n const recentOrders = await conn.execute(\n `SELECT order_number, placed_at, status, grand_total_gross_amount::numeric as total\n FROM sales_orders\n WHERE tenant_id = ? AND order_number LIKE 'SO-ANALYTICS-%'\n ORDER BY placed_at DESC LIMIT 5`,\n [tenantId]\n )\n console.log('\\nRecent orders:', recentOrders)\n\n const januaryOrders = await conn.execute(\n `SELECT COUNT(*) as count, SUM(grand_total_gross_amount::numeric) as total\n FROM sales_orders\n WHERE tenant_id = ?\n AND order_number LIKE 'SO-ANALYTICS-%'\n AND placed_at >= '2026-01-01'\n AND placed_at <= '2026-01-31 23:59:59'`,\n [tenantId]\n )\n console.log('\\nJanuary 2026 orders:', januaryOrders[0])\n\n const allOrders = await conn.execute(\n `SELECT COUNT(*) as count\n FROM sales_orders\n WHERE tenant_id = ?`,\n [tenantId]\n )\n console.log('\\nTotal orders in tenant:', allOrders[0])\n\n const orgCheck = await conn.execute(\n `SELECT organization_id, COUNT(*) as count\n FROM sales_orders\n WHERE tenant_id = ? AND order_number LIKE 'SO-ANALYTICS-%'\n GROUP BY organization_id`,\n [tenantId]\n )\n console.log('\\nOrders by organization:', orgCheck)\n\n if (organizationId) {\n const orgOrders = await conn.execute(\n `SELECT COUNT(*) as count, SUM(grand_total_gross_amount::numeric) as total\n FROM sales_orders\n WHERE tenant_id = ?\n AND organization_id = ?\n AND placed_at >= '2026-01-01'\n AND placed_at <= '2026-01-31 23:59:59'`,\n [tenantId, organizationId]\n )\n console.log(`\\nJanuary orders for org ${organizationId}:`, orgOrders[0])\n }\n\n // Check for NULL placed_at\n const nullPlacedAt = await conn.execute(\n `SELECT COUNT(*) as count\n FROM sales_orders\n WHERE tenant_id = ? AND placed_at IS NULL`,\n [tenantId]\n )\n console.log('\\nOrders with NULL placed_at:', nullPlacedAt[0])\n\n // Check non-analytics orders\n const nonAnalytics = await conn.execute(\n `SELECT order_number, placed_at, status, organization_id, grand_total_gross_amount::numeric as total\n FROM sales_orders\n WHERE tenant_id = ? AND order_number NOT LIKE 'SO-ANALYTICS-%'\n ORDER BY placed_at DESC NULLS LAST LIMIT 10`,\n [tenantId]\n )\n console.log('\\nNon-analytics orders:', nonAnalytics)\n\n // Simulate widget query\n console.log('\\n--- Simulating widget query for this_month ---')\n const widgetQuery = await conn.execute(\n `SELECT COALESCE(SUM(grand_total_gross_amount::numeric), 0) AS value\n FROM sales_orders\n WHERE tenant_id = ?\n AND organization_id = ANY(?::uuid[])\n AND deleted_at IS NULL\n AND placed_at >= '2026-01-01'\n AND placed_at <= '2026-01-31 23:59:59'`,\n [tenantId, `{${organizationId}}`]\n )\n console.log('Widget query result (revenue sum):', widgetQuery[0])\n\n const widgetCountQuery = await conn.execute(\n `SELECT COUNT(*) AS value\n FROM sales_orders\n WHERE tenant_id = ?\n AND organization_id = ANY(?::uuid[])\n AND deleted_at IS NULL\n AND placed_at >= '2026-01-01'\n AND placed_at <= '2026-01-31 23:59:59'`,\n [tenantId, `{${organizationId}}`]\n )\n console.log('Widget query result (order count):', widgetCountQuery[0])\n },\n}\n\nexport default [seedDefaults, enableAnalyticsWidgets, seedAnalytics, debugAnalytics]\n"],
5
+ "mappings": "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",
6
6
  "names": []
7
7
  }
@@ -0,0 +1,11 @@
1
+ import { asFunction } from "awilix";
2
+ import { createAnalyticsRegistry } from "./services/analyticsRegistry.js";
3
+ function register(container) {
4
+ container.register({
5
+ analyticsRegistry: asFunction(() => createAnalyticsRegistry()).singleton()
6
+ });
7
+ }
8
+ export {
9
+ register
10
+ };
11
+ //# sourceMappingURL=di.js.map
@@ -0,0 +1,7 @@
1
+ {
2
+ "version": 3,
3
+ "sources": ["../../../src/modules/dashboards/di.ts"],
4
+ "sourcesContent": ["import { asFunction } from 'awilix'\nimport type { AppContainer } from '@open-mercato/shared/lib/di/container'\nimport { createAnalyticsRegistry } from './services/analyticsRegistry'\n\nexport function register(container: AppContainer): void {\n container.register({\n analyticsRegistry: asFunction(() => createAnalyticsRegistry()).singleton(),\n })\n}\n"],
5
+ "mappings": "AAAA,SAAS,kBAAkB;AAE3B,SAAS,+BAA+B;AAEjC,SAAS,SAAS,WAA+B;AACtD,YAAU,SAAS;AAAA,IACjB,mBAAmB,WAAW,MAAM,wBAAwB,CAAC,EAAE,UAAU;AAAA,EAC3E,CAAC;AACH;",
6
+ "names": []
7
+ }
@@ -0,0 +1,162 @@
1
+ const VALID_GRANULARITIES = ["day", "week", "month", "quarter", "year"];
2
+ const VALID_AGGREGATES = ["count", "sum", "avg", "min", "max"];
3
+ const SAFE_IDENTIFIER_PATTERN = /^[a-zA-Z_][a-zA-Z0-9_]*$/;
4
+ function isValidGranularity(value) {
5
+ return typeof value === "string" && VALID_GRANULARITIES.includes(value);
6
+ }
7
+ function isValidAggregate(value) {
8
+ return typeof value === "string" && VALID_AGGREGATES.includes(value);
9
+ }
10
+ function isSafeIdentifier(value) {
11
+ return SAFE_IDENTIFIER_PATTERN.test(value);
12
+ }
13
+ function buildAggregateExpression(aggregate, column) {
14
+ switch (aggregate) {
15
+ case "count":
16
+ return column === "id" ? "COUNT(*)" : `COUNT(${column})`;
17
+ case "sum":
18
+ return `COALESCE(SUM(${column}::numeric), 0)`;
19
+ case "avg":
20
+ return `COALESCE(AVG(${column}::numeric), 0)`;
21
+ case "min":
22
+ return `MIN(${column}::numeric)`;
23
+ case "max":
24
+ return `MAX(${column}::numeric)`;
25
+ default:
26
+ return `COUNT(*)`;
27
+ }
28
+ }
29
+ function buildDateTruncExpression(column, granularity) {
30
+ if (!isValidGranularity(granularity)) {
31
+ throw new Error(`Invalid granularity: ${granularity}`);
32
+ }
33
+ return `DATE_TRUNC('${granularity}', ${column})`;
34
+ }
35
+ function buildJsonbFieldExpression(column, path) {
36
+ const parts = path.split(".");
37
+ for (const part of parts) {
38
+ if (!isSafeIdentifier(part)) {
39
+ throw new Error(`Invalid JSONB path part: ${part}`);
40
+ }
41
+ }
42
+ if (parts.length === 1) {
43
+ return `${column}->>'${parts[0]}'`;
44
+ }
45
+ const intermediate = parts.slice(0, -1).map((p) => `'${p}'`).join("->");
46
+ const lastPart = parts[parts.length - 1];
47
+ return `${column}->${intermediate}->>'${lastPart}'`;
48
+ }
49
+ function buildAggregationQuery(options) {
50
+ const { registry } = options;
51
+ const config = registry.getEntityTypeConfig(options.entityType);
52
+ if (!config) return null;
53
+ const metricMapping = registry.getFieldMapping(options.entityType, options.metric.field);
54
+ if (!metricMapping) return null;
55
+ const params = [];
56
+ const tableName = config.schema ? `"${config.schema}"."${config.tableName}"` : `"${config.tableName}"`;
57
+ const aggregateExpr = buildAggregateExpression(options.metric.aggregate, metricMapping.dbColumn);
58
+ let selectClause = `SELECT ${aggregateExpr} AS value`;
59
+ let groupByClause = "";
60
+ let orderByClause = "";
61
+ let limitClause = "";
62
+ if (options.groupBy) {
63
+ let groupMapping = registry.getFieldMapping(options.entityType, options.groupBy.field);
64
+ let groupExpr = null;
65
+ if (!groupMapping && options.groupBy.field.includes(".")) {
66
+ const [baseField, ...pathParts] = options.groupBy.field.split(".");
67
+ const baseMapping = registry.getFieldMapping(options.entityType, baseField);
68
+ if (baseMapping?.type === "jsonb") {
69
+ groupExpr = buildJsonbFieldExpression(baseMapping.dbColumn, pathParts.join("."));
70
+ }
71
+ } else if (groupMapping) {
72
+ if (groupMapping.type === "timestamp" && options.groupBy.granularity) {
73
+ groupExpr = buildDateTruncExpression(groupMapping.dbColumn, options.groupBy.granularity);
74
+ } else {
75
+ groupExpr = groupMapping.dbColumn;
76
+ }
77
+ }
78
+ if (groupExpr) {
79
+ selectClause = `SELECT ${groupExpr} AS group_key, ${aggregateExpr} AS value`;
80
+ groupByClause = `GROUP BY ${groupExpr}`;
81
+ orderByClause = `ORDER BY value DESC`;
82
+ if (options.groupBy.limit && options.groupBy.limit > 0) {
83
+ limitClause = `LIMIT ${Math.min(options.groupBy.limit, 100)}`;
84
+ }
85
+ }
86
+ }
87
+ const whereClauses = [];
88
+ whereClauses.push(`tenant_id = ?`);
89
+ params.push(options.scope.tenantId);
90
+ if (options.scope.organizationIds && options.scope.organizationIds.length > 0) {
91
+ whereClauses.push(`organization_id = ANY(?::uuid[])`);
92
+ params.push(`{${options.scope.organizationIds.join(",")}}`);
93
+ }
94
+ whereClauses.push(`deleted_at IS NULL`);
95
+ if (options.dateRange) {
96
+ const dateMapping = registry.getFieldMapping(options.entityType, options.dateRange.field);
97
+ if (dateMapping) {
98
+ whereClauses.push(`${dateMapping.dbColumn} >= ?`);
99
+ params.push(options.dateRange.start);
100
+ whereClauses.push(`${dateMapping.dbColumn} <= ?`);
101
+ params.push(options.dateRange.end);
102
+ }
103
+ }
104
+ if (options.filters) {
105
+ for (const filter of options.filters) {
106
+ const filterMapping = registry.getFieldMapping(options.entityType, filter.field);
107
+ if (!filterMapping) continue;
108
+ switch (filter.operator) {
109
+ case "eq":
110
+ whereClauses.push(`${filterMapping.dbColumn} = ?`);
111
+ params.push(filter.value);
112
+ break;
113
+ case "neq":
114
+ whereClauses.push(`${filterMapping.dbColumn} != ?`);
115
+ params.push(filter.value);
116
+ break;
117
+ case "gt":
118
+ whereClauses.push(`${filterMapping.dbColumn} > ?`);
119
+ params.push(filter.value);
120
+ break;
121
+ case "gte":
122
+ whereClauses.push(`${filterMapping.dbColumn} >= ?`);
123
+ params.push(filter.value);
124
+ break;
125
+ case "lt":
126
+ whereClauses.push(`${filterMapping.dbColumn} < ?`);
127
+ params.push(filter.value);
128
+ break;
129
+ case "lte":
130
+ whereClauses.push(`${filterMapping.dbColumn} <= ?`);
131
+ params.push(filter.value);
132
+ break;
133
+ case "in":
134
+ whereClauses.push(`${filterMapping.dbColumn} = ANY(?)`);
135
+ params.push(filter.value);
136
+ break;
137
+ case "not_in":
138
+ whereClauses.push(`${filterMapping.dbColumn} != ALL(?)`);
139
+ params.push(filter.value);
140
+ break;
141
+ case "is_null":
142
+ whereClauses.push(`${filterMapping.dbColumn} IS NULL`);
143
+ break;
144
+ case "is_not_null":
145
+ whereClauses.push(`${filterMapping.dbColumn} IS NOT NULL`);
146
+ break;
147
+ }
148
+ }
149
+ }
150
+ const whereClause = whereClauses.length > 0 ? `WHERE ${whereClauses.join(" AND ")}` : "";
151
+ const sql = [selectClause, `FROM ${tableName}`, whereClause, groupByClause, orderByClause, limitClause].filter(Boolean).join(" ");
152
+ return { sql, params };
153
+ }
154
+ export {
155
+ buildAggregateExpression,
156
+ buildAggregationQuery,
157
+ buildDateTruncExpression,
158
+ buildJsonbFieldExpression,
159
+ isValidAggregate,
160
+ isValidGranularity
161
+ };
162
+ //# sourceMappingURL=aggregations.js.map
@@ -0,0 +1,7 @@
1
+ {
2
+ "version": 3,
3
+ "sources": ["../../../../src/modules/dashboards/lib/aggregations.ts"],
4
+ "sourcesContent": ["import type { AnalyticsRegistry } from '../services/analyticsRegistry'\nimport type { AnalyticsEntityTypeConfig, AnalyticsFieldMapping } from '@open-mercato/shared/modules/analytics'\n\nexport type AggregateFunction = 'count' | 'sum' | 'avg' | 'min' | 'max'\nexport type DateGranularity = 'day' | 'week' | 'month' | 'quarter' | 'year'\n\nconst VALID_GRANULARITIES: readonly DateGranularity[] = ['day', 'week', 'month', 'quarter', 'year']\nconst VALID_AGGREGATES: readonly AggregateFunction[] = ['count', 'sum', 'avg', 'min', 'max']\nconst SAFE_IDENTIFIER_PATTERN = /^[a-zA-Z_][a-zA-Z0-9_]*$/\n\nexport function isValidGranularity(value: unknown): value is DateGranularity {\n return typeof value === 'string' && VALID_GRANULARITIES.includes(value as DateGranularity)\n}\n\nexport function isValidAggregate(value: unknown): value is AggregateFunction {\n return typeof value === 'string' && VALID_AGGREGATES.includes(value as AggregateFunction)\n}\n\nfunction isSafeIdentifier(value: string): boolean {\n return SAFE_IDENTIFIER_PATTERN.test(value)\n}\n\n// Re-export types from shared module for convenience\nexport type EntityTypeConfig = AnalyticsEntityTypeConfig\nexport type FieldMapping = AnalyticsFieldMapping\n\nexport function buildAggregateExpression(aggregate: AggregateFunction, column: string): string {\n switch (aggregate) {\n case 'count':\n return column === 'id' ? 'COUNT(*)' : `COUNT(${column})`\n case 'sum':\n return `COALESCE(SUM(${column}::numeric), 0)`\n case 'avg':\n return `COALESCE(AVG(${column}::numeric), 0)`\n case 'min':\n return `MIN(${column}::numeric)`\n case 'max':\n return `MAX(${column}::numeric)`\n default:\n return `COUNT(*)`\n }\n}\n\nexport function buildDateTruncExpression(column: string, granularity: DateGranularity): string {\n if (!isValidGranularity(granularity)) {\n throw new Error(`Invalid granularity: ${granularity}`)\n }\n return `DATE_TRUNC('${granularity}', ${column})`\n}\n\nexport function buildJsonbFieldExpression(column: string, path: string): string {\n const parts = path.split('.')\n for (const part of parts) {\n if (!isSafeIdentifier(part)) {\n throw new Error(`Invalid JSONB path part: ${part}`)\n }\n }\n if (parts.length === 1) {\n return `${column}->>'${parts[0]}'`\n }\n const intermediate = parts.slice(0, -1).map((p) => `'${p}'`).join('->')\n const lastPart = parts[parts.length - 1]\n return `${column}->${intermediate}->>'${lastPart}'`\n}\n\nexport type AggregationQuery = {\n sql: string\n params: unknown[]\n}\n\nexport type BuildAggregationQueryOptions = {\n entityType: string\n metric: {\n field: string\n aggregate: AggregateFunction\n }\n groupBy?: {\n field: string\n granularity?: DateGranularity\n limit?: number\n }\n dateRange?: {\n field: string\n start: Date\n end: Date\n }\n filters?: Array<{\n field: string\n operator: 'eq' | 'neq' | 'gt' | 'gte' | 'lt' | 'lte' | 'in' | 'not_in' | 'is_null' | 'is_not_null'\n value?: unknown\n }>\n scope: {\n tenantId: string\n organizationIds?: string[]\n }\n /** Analytics registry for resolving entity and field configurations */\n registry: AnalyticsRegistry\n}\n\nexport function buildAggregationQuery(options: BuildAggregationQueryOptions): AggregationQuery | null {\n const { registry } = options\n const config = registry.getEntityTypeConfig(options.entityType)\n if (!config) return null\n\n const metricMapping = registry.getFieldMapping(options.entityType, options.metric.field)\n if (!metricMapping) return null\n\n const params: unknown[] = []\n\n const tableName = config.schema ? `\"${config.schema}\".\"${config.tableName}\"` : `\"${config.tableName}\"`\n const aggregateExpr = buildAggregateExpression(options.metric.aggregate, metricMapping.dbColumn)\n\n let selectClause = `SELECT ${aggregateExpr} AS value`\n let groupByClause = ''\n let orderByClause = ''\n let limitClause = ''\n\n if (options.groupBy) {\n let groupMapping = registry.getFieldMapping(options.entityType, options.groupBy.field)\n let groupExpr: string | null = null\n\n // Handle JSONB path notation (e.g., shippingAddressSnapshot.region)\n if (!groupMapping && options.groupBy.field.includes('.')) {\n const [baseField, ...pathParts] = options.groupBy.field.split('.')\n const baseMapping = registry.getFieldMapping(options.entityType, baseField)\n if (baseMapping?.type === 'jsonb') {\n groupExpr = buildJsonbFieldExpression(baseMapping.dbColumn, pathParts.join('.'))\n }\n } else if (groupMapping) {\n if (groupMapping.type === 'timestamp' && options.groupBy.granularity) {\n groupExpr = buildDateTruncExpression(groupMapping.dbColumn, options.groupBy.granularity)\n } else {\n groupExpr = groupMapping.dbColumn\n }\n }\n\n if (groupExpr) {\n selectClause = `SELECT ${groupExpr} AS group_key, ${aggregateExpr} AS value`\n groupByClause = `GROUP BY ${groupExpr}`\n orderByClause = `ORDER BY value DESC`\n\n if (options.groupBy.limit && options.groupBy.limit > 0) {\n limitClause = `LIMIT ${Math.min(options.groupBy.limit, 100)}`\n }\n }\n }\n\n const whereClauses: string[] = []\n\n whereClauses.push(`tenant_id = ?`)\n params.push(options.scope.tenantId)\n\n if (options.scope.organizationIds && options.scope.organizationIds.length > 0) {\n whereClauses.push(`organization_id = ANY(?::uuid[])`)\n params.push(`{${options.scope.organizationIds.join(',')}}`)\n }\n\n whereClauses.push(`deleted_at IS NULL`)\n\n if (options.dateRange) {\n const dateMapping = registry.getFieldMapping(options.entityType, options.dateRange.field)\n if (dateMapping) {\n whereClauses.push(`${dateMapping.dbColumn} >= ?`)\n params.push(options.dateRange.start)\n whereClauses.push(`${dateMapping.dbColumn} <= ?`)\n params.push(options.dateRange.end)\n }\n }\n\n if (options.filters) {\n for (const filter of options.filters) {\n const filterMapping = registry.getFieldMapping(options.entityType, filter.field)\n if (!filterMapping) continue\n\n switch (filter.operator) {\n case 'eq':\n whereClauses.push(`${filterMapping.dbColumn} = ?`)\n params.push(filter.value)\n break\n case 'neq':\n whereClauses.push(`${filterMapping.dbColumn} != ?`)\n params.push(filter.value)\n break\n case 'gt':\n whereClauses.push(`${filterMapping.dbColumn} > ?`)\n params.push(filter.value)\n break\n case 'gte':\n whereClauses.push(`${filterMapping.dbColumn} >= ?`)\n params.push(filter.value)\n break\n case 'lt':\n whereClauses.push(`${filterMapping.dbColumn} < ?`)\n params.push(filter.value)\n break\n case 'lte':\n whereClauses.push(`${filterMapping.dbColumn} <= ?`)\n params.push(filter.value)\n break\n case 'in':\n whereClauses.push(`${filterMapping.dbColumn} = ANY(?)`)\n params.push(filter.value)\n break\n case 'not_in':\n whereClauses.push(`${filterMapping.dbColumn} != ALL(?)`)\n params.push(filter.value)\n break\n case 'is_null':\n whereClauses.push(`${filterMapping.dbColumn} IS NULL`)\n break\n case 'is_not_null':\n whereClauses.push(`${filterMapping.dbColumn} IS NOT NULL`)\n break\n }\n }\n }\n\n const whereClause = whereClauses.length > 0 ? `WHERE ${whereClauses.join(' AND ')}` : ''\n\n const sql = [selectClause, `FROM ${tableName}`, whereClause, groupByClause, orderByClause, limitClause]\n .filter(Boolean)\n .join(' ')\n\n return { sql, params }\n}\n"],
5
+ "mappings": "AAMA,MAAM,sBAAkD,CAAC,OAAO,QAAQ,SAAS,WAAW,MAAM;AAClG,MAAM,mBAAiD,CAAC,SAAS,OAAO,OAAO,OAAO,KAAK;AAC3F,MAAM,0BAA0B;AAEzB,SAAS,mBAAmB,OAA0C;AAC3E,SAAO,OAAO,UAAU,YAAY,oBAAoB,SAAS,KAAwB;AAC3F;AAEO,SAAS,iBAAiB,OAA4C;AAC3E,SAAO,OAAO,UAAU,YAAY,iBAAiB,SAAS,KAA0B;AAC1F;AAEA,SAAS,iBAAiB,OAAwB;AAChD,SAAO,wBAAwB,KAAK,KAAK;AAC3C;AAMO,SAAS,yBAAyB,WAA8B,QAAwB;AAC7F,UAAQ,WAAW;AAAA,IACjB,KAAK;AACH,aAAO,WAAW,OAAO,aAAa,SAAS,MAAM;AAAA,IACvD,KAAK;AACH,aAAO,gBAAgB,MAAM;AAAA,IAC/B,KAAK;AACH,aAAO,gBAAgB,MAAM;AAAA,IAC/B,KAAK;AACH,aAAO,OAAO,MAAM;AAAA,IACtB,KAAK;AACH,aAAO,OAAO,MAAM;AAAA,IACtB;AACE,aAAO;AAAA,EACX;AACF;AAEO,SAAS,yBAAyB,QAAgB,aAAsC;AAC7F,MAAI,CAAC,mBAAmB,WAAW,GAAG;AACpC,UAAM,IAAI,MAAM,wBAAwB,WAAW,EAAE;AAAA,EACvD;AACA,SAAO,eAAe,WAAW,MAAM,MAAM;AAC/C;AAEO,SAAS,0BAA0B,QAAgB,MAAsB;AAC9E,QAAM,QAAQ,KAAK,MAAM,GAAG;AAC5B,aAAW,QAAQ,OAAO;AACxB,QAAI,CAAC,iBAAiB,IAAI,GAAG;AAC3B,YAAM,IAAI,MAAM,4BAA4B,IAAI,EAAE;AAAA,IACpD;AAAA,EACF;AACA,MAAI,MAAM,WAAW,GAAG;AACtB,WAAO,GAAG,MAAM,OAAO,MAAM,CAAC,CAAC;AAAA,EACjC;AACA,QAAM,eAAe,MAAM,MAAM,GAAG,EAAE,EAAE,IAAI,CAAC,MAAM,IAAI,CAAC,GAAG,EAAE,KAAK,IAAI;AACtE,QAAM,WAAW,MAAM,MAAM,SAAS,CAAC;AACvC,SAAO,GAAG,MAAM,KAAK,YAAY,OAAO,QAAQ;AAClD;AAoCO,SAAS,sBAAsB,SAAgE;AACpG,QAAM,EAAE,SAAS,IAAI;AACrB,QAAM,SAAS,SAAS,oBAAoB,QAAQ,UAAU;AAC9D,MAAI,CAAC,OAAQ,QAAO;AAEpB,QAAM,gBAAgB,SAAS,gBAAgB,QAAQ,YAAY,QAAQ,OAAO,KAAK;AACvF,MAAI,CAAC,cAAe,QAAO;AAE3B,QAAM,SAAoB,CAAC;AAE3B,QAAM,YAAY,OAAO,SAAS,IAAI,OAAO,MAAM,MAAM,OAAO,SAAS,MAAM,IAAI,OAAO,SAAS;AACnG,QAAM,gBAAgB,yBAAyB,QAAQ,OAAO,WAAW,cAAc,QAAQ;AAE/F,MAAI,eAAe,UAAU,aAAa;AAC1C,MAAI,gBAAgB;AACpB,MAAI,gBAAgB;AACpB,MAAI,cAAc;AAElB,MAAI,QAAQ,SAAS;AACnB,QAAI,eAAe,SAAS,gBAAgB,QAAQ,YAAY,QAAQ,QAAQ,KAAK;AACrF,QAAI,YAA2B;AAG/B,QAAI,CAAC,gBAAgB,QAAQ,QAAQ,MAAM,SAAS,GAAG,GAAG;AACxD,YAAM,CAAC,WAAW,GAAG,SAAS,IAAI,QAAQ,QAAQ,MAAM,MAAM,GAAG;AACjE,YAAM,cAAc,SAAS,gBAAgB,QAAQ,YAAY,SAAS;AAC1E,UAAI,aAAa,SAAS,SAAS;AACjC,oBAAY,0BAA0B,YAAY,UAAU,UAAU,KAAK,GAAG,CAAC;AAAA,MACjF;AAAA,IACF,WAAW,cAAc;AACvB,UAAI,aAAa,SAAS,eAAe,QAAQ,QAAQ,aAAa;AACpE,oBAAY,yBAAyB,aAAa,UAAU,QAAQ,QAAQ,WAAW;AAAA,MACzF,OAAO;AACL,oBAAY,aAAa;AAAA,MAC3B;AAAA,IACF;AAEA,QAAI,WAAW;AACb,qBAAe,UAAU,SAAS,kBAAkB,aAAa;AACjE,sBAAgB,YAAY,SAAS;AACrC,sBAAgB;AAEhB,UAAI,QAAQ,QAAQ,SAAS,QAAQ,QAAQ,QAAQ,GAAG;AACtD,sBAAc,SAAS,KAAK,IAAI,QAAQ,QAAQ,OAAO,GAAG,CAAC;AAAA,MAC7D;AAAA,IACF;AAAA,EACF;AAEA,QAAM,eAAyB,CAAC;AAEhC,eAAa,KAAK,eAAe;AACjC,SAAO,KAAK,QAAQ,MAAM,QAAQ;AAElC,MAAI,QAAQ,MAAM,mBAAmB,QAAQ,MAAM,gBAAgB,SAAS,GAAG;AAC7E,iBAAa,KAAK,kCAAkC;AACpD,WAAO,KAAK,IAAI,QAAQ,MAAM,gBAAgB,KAAK,GAAG,CAAC,GAAG;AAAA,EAC5D;AAEA,eAAa,KAAK,oBAAoB;AAEtC,MAAI,QAAQ,WAAW;AACrB,UAAM,cAAc,SAAS,gBAAgB,QAAQ,YAAY,QAAQ,UAAU,KAAK;AACxF,QAAI,aAAa;AACf,mBAAa,KAAK,GAAG,YAAY,QAAQ,OAAO;AAChD,aAAO,KAAK,QAAQ,UAAU,KAAK;AACnC,mBAAa,KAAK,GAAG,YAAY,QAAQ,OAAO;AAChD,aAAO,KAAK,QAAQ,UAAU,GAAG;AAAA,IACnC;AAAA,EACF;AAEA,MAAI,QAAQ,SAAS;AACnB,eAAW,UAAU,QAAQ,SAAS;AACpC,YAAM,gBAAgB,SAAS,gBAAgB,QAAQ,YAAY,OAAO,KAAK;AAC/E,UAAI,CAAC,cAAe;AAEpB,cAAQ,OAAO,UAAU;AAAA,QACvB,KAAK;AACH,uBAAa,KAAK,GAAG,cAAc,QAAQ,MAAM;AACjD,iBAAO,KAAK,OAAO,KAAK;AACxB;AAAA,QACF,KAAK;AACH,uBAAa,KAAK,GAAG,cAAc,QAAQ,OAAO;AAClD,iBAAO,KAAK,OAAO,KAAK;AACxB;AAAA,QACF,KAAK;AACH,uBAAa,KAAK,GAAG,cAAc,QAAQ,MAAM;AACjD,iBAAO,KAAK,OAAO,KAAK;AACxB;AAAA,QACF,KAAK;AACH,uBAAa,KAAK,GAAG,cAAc,QAAQ,OAAO;AAClD,iBAAO,KAAK,OAAO,KAAK;AACxB;AAAA,QACF,KAAK;AACH,uBAAa,KAAK,GAAG,cAAc,QAAQ,MAAM;AACjD,iBAAO,KAAK,OAAO,KAAK;AACxB;AAAA,QACF,KAAK;AACH,uBAAa,KAAK,GAAG,cAAc,QAAQ,OAAO;AAClD,iBAAO,KAAK,OAAO,KAAK;AACxB;AAAA,QACF,KAAK;AACH,uBAAa,KAAK,GAAG,cAAc,QAAQ,WAAW;AACtD,iBAAO,KAAK,OAAO,KAAK;AACxB;AAAA,QACF,KAAK;AACH,uBAAa,KAAK,GAAG,cAAc,QAAQ,YAAY;AACvD,iBAAO,KAAK,OAAO,KAAK;AACxB;AAAA,QACF,KAAK;AACH,uBAAa,KAAK,GAAG,cAAc,QAAQ,UAAU;AACrD;AAAA,QACF,KAAK;AACH,uBAAa,KAAK,GAAG,cAAc,QAAQ,cAAc;AACzD;AAAA,MACJ;AAAA,IACF;AAAA,EACF;AAEA,QAAM,cAAc,aAAa,SAAS,IAAI,SAAS,aAAa,KAAK,OAAO,CAAC,KAAK;AAEtF,QAAM,MAAM,CAAC,cAAc,QAAQ,SAAS,IAAI,aAAa,eAAe,eAAe,WAAW,EACnG,OAAO,OAAO,EACd,KAAK,GAAG;AAEX,SAAO,EAAE,KAAK,OAAO;AACvB;",
6
+ "names": []
7
+ }
@@ -0,0 +1,34 @@
1
+ function formatCurrency(value, options = {}) {
2
+ const { currency = "USD", minimumFractionDigits = 0, maximumFractionDigits = 0 } = options;
3
+ return new Intl.NumberFormat(void 0, {
4
+ style: "currency",
5
+ currency,
6
+ minimumFractionDigits,
7
+ maximumFractionDigits
8
+ }).format(value);
9
+ }
10
+ function formatCurrencyWithDecimals(value, options = {}) {
11
+ return formatCurrency(value, { minimumFractionDigits: 2, maximumFractionDigits: 2, ...options });
12
+ }
13
+ function formatCurrencyCompact(value, currencySymbol = "$") {
14
+ if (Math.abs(value) >= 1e6) {
15
+ return `${currencySymbol}${(value / 1e6).toFixed(1)}M`;
16
+ }
17
+ if (Math.abs(value) >= 1e3) {
18
+ return `${currencySymbol}${(value / 1e3).toFixed(1)}K`;
19
+ }
20
+ return `${currencySymbol}${value.toFixed(0)}`;
21
+ }
22
+ function formatCurrencySafe(value, fallback = "--") {
23
+ if (value === null || value === void 0) return fallback;
24
+ const num = Number(value);
25
+ if (!Number.isFinite(num)) return fallback;
26
+ return formatCurrency(num);
27
+ }
28
+ export {
29
+ formatCurrency,
30
+ formatCurrencyCompact,
31
+ formatCurrencySafe,
32
+ formatCurrencyWithDecimals
33
+ };
34
+ //# sourceMappingURL=formatters.js.map
@@ -0,0 +1,7 @@
1
+ {
2
+ "version": 3,
3
+ "sources": ["../../../../src/modules/dashboards/lib/formatters.ts"],
4
+ "sourcesContent": ["export type FormatCurrencyOptions = {\n currency?: string\n minimumFractionDigits?: number\n maximumFractionDigits?: number\n}\n\nexport function formatCurrency(value: number, options: FormatCurrencyOptions = {}): string {\n const { currency = 'USD', minimumFractionDigits = 0, maximumFractionDigits = 0 } = options\n return new Intl.NumberFormat(undefined, {\n style: 'currency',\n currency,\n minimumFractionDigits,\n maximumFractionDigits,\n }).format(value)\n}\n\nexport function formatCurrencyWithDecimals(value: number, options: FormatCurrencyOptions = {}): string {\n return formatCurrency(value, { minimumFractionDigits: 2, maximumFractionDigits: 2, ...options })\n}\n\nexport function formatCurrencyCompact(value: number, currencySymbol = '$'): string {\n if (Math.abs(value) >= 1_000_000) {\n return `${currencySymbol}${(value / 1_000_000).toFixed(1)}M`\n }\n if (Math.abs(value) >= 1_000) {\n return `${currencySymbol}${(value / 1_000).toFixed(1)}K`\n }\n return `${currencySymbol}${value.toFixed(0)}`\n}\n\nexport function formatCurrencySafe(value: unknown, fallback = '--'): string {\n if (value === null || value === undefined) return fallback\n const num = Number(value)\n if (!Number.isFinite(num)) return fallback\n return formatCurrency(num)\n}\n"],
5
+ "mappings": "AAMO,SAAS,eAAe,OAAe,UAAiC,CAAC,GAAW;AACzF,QAAM,EAAE,WAAW,OAAO,wBAAwB,GAAG,wBAAwB,EAAE,IAAI;AACnF,SAAO,IAAI,KAAK,aAAa,QAAW;AAAA,IACtC,OAAO;AAAA,IACP;AAAA,IACA;AAAA,IACA;AAAA,EACF,CAAC,EAAE,OAAO,KAAK;AACjB;AAEO,SAAS,2BAA2B,OAAe,UAAiC,CAAC,GAAW;AACrG,SAAO,eAAe,OAAO,EAAE,uBAAuB,GAAG,uBAAuB,GAAG,GAAG,QAAQ,CAAC;AACjG;AAEO,SAAS,sBAAsB,OAAe,iBAAiB,KAAa;AACjF,MAAI,KAAK,IAAI,KAAK,KAAK,KAAW;AAChC,WAAO,GAAG,cAAc,IAAI,QAAQ,KAAW,QAAQ,CAAC,CAAC;AAAA,EAC3D;AACA,MAAI,KAAK,IAAI,KAAK,KAAK,KAAO;AAC5B,WAAO,GAAG,cAAc,IAAI,QAAQ,KAAO,QAAQ,CAAC,CAAC;AAAA,EACvD;AACA,SAAO,GAAG,cAAc,GAAG,MAAM,QAAQ,CAAC,CAAC;AAC7C;AAEO,SAAS,mBAAmB,OAAgB,WAAW,MAAc;AAC1E,MAAI,UAAU,QAAQ,UAAU,OAAW,QAAO;AAClD,QAAM,MAAM,OAAO,KAAK;AACxB,MAAI,CAAC,OAAO,SAAS,GAAG,EAAG,QAAO;AAClC,SAAO,eAAe,GAAG;AAC3B;",
6
+ "names": []
7
+ }
@@ -0,0 +1,58 @@
1
+ import { Role } from "@open-mercato/core/modules/auth/data/entities";
2
+ import { DashboardRoleWidgets } from "../data/entities.js";
3
+ import { loadAllWidgets } from "./widgets.js";
4
+ async function findRoleByName(em, roleName, tenantId) {
5
+ const tenantRole = await em.findOne(Role, { name: roleName, tenantId });
6
+ if (tenantRole) return tenantRole;
7
+ return em.findOne(Role, { name: roleName, tenantId: null });
8
+ }
9
+ async function resolveAnalyticsWidgetIds() {
10
+ const widgets = await loadAllWidgets();
11
+ return widgets.filter((widget) => widget.metadata.category === "analytics" || widget.metadata.id.startsWith("dashboards.analytics.")).map((widget) => widget.metadata.id);
12
+ }
13
+ async function appendWidgetsToRoles(em, { tenantId, organizationId = null, roleNames, widgetIds }) {
14
+ const trimmedTenantId = tenantId.trim();
15
+ const widgets = await loadAllWidgets();
16
+ const validWidgetIds = new Set(widgets.map((widget) => widget.metadata.id));
17
+ const resolvedWidgetIds = widgetIds.filter((id) => validWidgetIds.has(id));
18
+ if (!resolvedWidgetIds.length) return false;
19
+ let updated = false;
20
+ await em.transactional(async (tem) => {
21
+ for (const roleName of roleNames) {
22
+ const role = await findRoleByName(tem, roleName, trimmedTenantId);
23
+ if (!role) continue;
24
+ const record = await tem.findOne(DashboardRoleWidgets, {
25
+ roleId: String(role.id),
26
+ tenantId: trimmedTenantId,
27
+ organizationId,
28
+ deletedAt: null
29
+ });
30
+ const roleRecord = record ?? (organizationId ? await tem.findOne(DashboardRoleWidgets, {
31
+ roleId: String(role.id),
32
+ tenantId: trimmedTenantId,
33
+ organizationId: null,
34
+ deletedAt: null
35
+ }) : null);
36
+ if (!roleRecord) continue;
37
+ const current = Array.isArray(roleRecord.widgetIdsJson) ? roleRecord.widgetIdsJson : [];
38
+ const next = [...current];
39
+ const existing = new Set(current);
40
+ for (const widgetId of resolvedWidgetIds) {
41
+ if (existing.has(widgetId)) continue;
42
+ existing.add(widgetId);
43
+ next.push(widgetId);
44
+ }
45
+ if (next.length === current.length) continue;
46
+ roleRecord.widgetIdsJson = next;
47
+ roleRecord.updatedAt = /* @__PURE__ */ new Date();
48
+ tem.persist(roleRecord);
49
+ updated = true;
50
+ }
51
+ });
52
+ return updated;
53
+ }
54
+ export {
55
+ appendWidgetsToRoles,
56
+ resolveAnalyticsWidgetIds
57
+ };
58
+ //# sourceMappingURL=role-widgets.js.map
@@ -0,0 +1,7 @@
1
+ {
2
+ "version": 3,
3
+ "sources": ["../../../../src/modules/dashboards/lib/role-widgets.ts"],
4
+ "sourcesContent": ["import type { EntityManager } from '@mikro-orm/postgresql'\nimport { Role } from '@open-mercato/core/modules/auth/data/entities'\nimport { DashboardRoleWidgets } from '../data/entities'\nimport { loadAllWidgets } from './widgets'\n\ntype RoleWidgetScope = {\n tenantId: string\n organizationId?: string | null\n roleNames: string[]\n widgetIds: string[]\n}\n\nasync function findRoleByName(\n em: EntityManager,\n roleName: string,\n tenantId: string,\n): Promise<Role | null> {\n const tenantRole = await em.findOne(Role, { name: roleName, tenantId })\n if (tenantRole) return tenantRole\n return em.findOne(Role, { name: roleName, tenantId: null })\n}\n\nexport async function resolveAnalyticsWidgetIds(): Promise<string[]> {\n const widgets = await loadAllWidgets()\n return widgets\n .filter((widget) => widget.metadata.category === 'analytics' || widget.metadata.id.startsWith('dashboards.analytics.'))\n .map((widget) => widget.metadata.id)\n}\n\nexport async function appendWidgetsToRoles(\n em: EntityManager,\n { tenantId, organizationId = null, roleNames, widgetIds }: RoleWidgetScope,\n): Promise<boolean> {\n const trimmedTenantId = tenantId.trim()\n const widgets = await loadAllWidgets()\n const validWidgetIds = new Set(widgets.map((widget) => widget.metadata.id))\n const resolvedWidgetIds = widgetIds.filter((id) => validWidgetIds.has(id))\n if (!resolvedWidgetIds.length) return false\n\n let updated = false\n await em.transactional(async (tem) => {\n for (const roleName of roleNames) {\n const role = await findRoleByName(tem, roleName, trimmedTenantId)\n if (!role) continue\n\n const record = await tem.findOne(DashboardRoleWidgets, {\n roleId: String(role.id),\n tenantId: trimmedTenantId,\n organizationId,\n deletedAt: null,\n })\n const roleRecord = record ?? (organizationId\n ? await tem.findOne(DashboardRoleWidgets, {\n roleId: String(role.id),\n tenantId: trimmedTenantId,\n organizationId: null,\n deletedAt: null,\n })\n : null)\n if (!roleRecord) continue\n\n const current = Array.isArray(roleRecord.widgetIdsJson) ? roleRecord.widgetIdsJson : []\n const next = [...current]\n const existing = new Set(current)\n for (const widgetId of resolvedWidgetIds) {\n if (existing.has(widgetId)) continue\n existing.add(widgetId)\n next.push(widgetId)\n }\n\n if (next.length === current.length) continue\n roleRecord.widgetIdsJson = next\n roleRecord.updatedAt = new Date()\n tem.persist(roleRecord)\n updated = true\n }\n })\n\n return updated\n}\n"],
5
+ "mappings": "AACA,SAAS,YAAY;AACrB,SAAS,4BAA4B;AACrC,SAAS,sBAAsB;AAS/B,eAAe,eACb,IACA,UACA,UACsB;AACtB,QAAM,aAAa,MAAM,GAAG,QAAQ,MAAM,EAAE,MAAM,UAAU,SAAS,CAAC;AACtE,MAAI,WAAY,QAAO;AACvB,SAAO,GAAG,QAAQ,MAAM,EAAE,MAAM,UAAU,UAAU,KAAK,CAAC;AAC5D;AAEA,eAAsB,4BAA+C;AACnE,QAAM,UAAU,MAAM,eAAe;AACrC,SAAO,QACJ,OAAO,CAAC,WAAW,OAAO,SAAS,aAAa,eAAe,OAAO,SAAS,GAAG,WAAW,uBAAuB,CAAC,EACrH,IAAI,CAAC,WAAW,OAAO,SAAS,EAAE;AACvC;AAEA,eAAsB,qBACpB,IACA,EAAE,UAAU,iBAAiB,MAAM,WAAW,UAAU,GACtC;AAClB,QAAM,kBAAkB,SAAS,KAAK;AACtC,QAAM,UAAU,MAAM,eAAe;AACrC,QAAM,iBAAiB,IAAI,IAAI,QAAQ,IAAI,CAAC,WAAW,OAAO,SAAS,EAAE,CAAC;AAC1E,QAAM,oBAAoB,UAAU,OAAO,CAAC,OAAO,eAAe,IAAI,EAAE,CAAC;AACzE,MAAI,CAAC,kBAAkB,OAAQ,QAAO;AAEtC,MAAI,UAAU;AACd,QAAM,GAAG,cAAc,OAAO,QAAQ;AACpC,eAAW,YAAY,WAAW;AAChC,YAAM,OAAO,MAAM,eAAe,KAAK,UAAU,eAAe;AAChE,UAAI,CAAC,KAAM;AAEX,YAAM,SAAS,MAAM,IAAI,QAAQ,sBAAsB;AAAA,QACrD,QAAQ,OAAO,KAAK,EAAE;AAAA,QACtB,UAAU;AAAA,QACV;AAAA,QACA,WAAW;AAAA,MACb,CAAC;AACD,YAAM,aAAa,WAAW,iBAC1B,MAAM,IAAI,QAAQ,sBAAsB;AAAA,QACxC,QAAQ,OAAO,KAAK,EAAE;AAAA,QACtB,UAAU;AAAA,QACV,gBAAgB;AAAA,QAChB,WAAW;AAAA,MACb,CAAC,IACC;AACJ,UAAI,CAAC,WAAY;AAEjB,YAAM,UAAU,MAAM,QAAQ,WAAW,aAAa,IAAI,WAAW,gBAAgB,CAAC;AACtF,YAAM,OAAO,CAAC,GAAG,OAAO;AACxB,YAAM,WAAW,IAAI,IAAI,OAAO;AAChC,iBAAW,YAAY,mBAAmB;AACxC,YAAI,SAAS,IAAI,QAAQ,EAAG;AAC5B,iBAAS,IAAI,QAAQ;AACrB,aAAK,KAAK,QAAQ;AAAA,MACpB;AAEA,UAAI,KAAK,WAAW,QAAQ,OAAQ;AACpC,iBAAW,gBAAgB;AAC3B,iBAAW,YAAY,oBAAI,KAAK;AAChC,UAAI,QAAQ,UAAU;AACtB,gBAAU;AAAA,IACZ;AAAA,EACF,CAAC;AAED,SAAO;AACT;",
6
+ "names": []
7
+ }