@open-mercato/core 0.4.2-canary-f075c3eb92 → 0.4.2-canary-8a04af8836

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 (507) 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/login.js +25 -6
  18. package/dist/modules/auth/api/login.js.map +2 -2
  19. package/dist/modules/auth/api/profile/route.js +157 -0
  20. package/dist/modules/auth/api/profile/route.js.map +7 -0
  21. package/dist/modules/auth/api/reset/confirm.js +25 -2
  22. package/dist/modules/auth/api/reset/confirm.js.map +2 -2
  23. package/dist/modules/auth/api/reset.js +23 -0
  24. package/dist/modules/auth/api/reset.js.map +2 -2
  25. package/dist/modules/auth/api/sidebar/preferences/route.js +14 -9
  26. package/dist/modules/auth/api/sidebar/preferences/route.js.map +2 -2
  27. package/dist/modules/auth/api/users/route.js +4 -2
  28. package/dist/modules/auth/api/users/route.js.map +2 -2
  29. package/dist/modules/auth/backend/auth/profile/page.js +141 -0
  30. package/dist/modules/auth/backend/auth/profile/page.js.map +7 -0
  31. package/dist/modules/auth/backend/auth/profile/page.meta.js +13 -0
  32. package/dist/modules/auth/backend/auth/profile/page.meta.js.map +7 -0
  33. package/dist/modules/auth/backend/roles/[id]/edit/page.js +4 -1
  34. package/dist/modules/auth/backend/roles/[id]/edit/page.js.map +2 -2
  35. package/dist/modules/auth/backend/roles/page.js +3 -3
  36. package/dist/modules/auth/backend/roles/page.js.map +2 -2
  37. package/dist/modules/auth/backend/users/[id]/edit/page.js +18 -3
  38. package/dist/modules/auth/backend/users/[id]/edit/page.js.map +2 -2
  39. package/dist/modules/auth/backend/users/create/page.js +15 -2
  40. package/dist/modules/auth/backend/users/create/page.js.map +2 -2
  41. package/dist/modules/auth/backend/users/page.js +3 -3
  42. package/dist/modules/auth/backend/users/page.js.map +2 -2
  43. package/dist/modules/auth/cli.js +25 -11
  44. package/dist/modules/auth/cli.js.map +2 -2
  45. package/dist/modules/auth/commands/users.js +59 -2
  46. package/dist/modules/auth/commands/users.js.map +2 -2
  47. package/dist/modules/auth/data/validators.js +6 -3
  48. package/dist/modules/auth/data/validators.js.map +2 -2
  49. package/dist/modules/auth/frontend/login.js +112 -3
  50. package/dist/modules/auth/frontend/login.js.map +2 -2
  51. package/dist/modules/auth/frontend/reset/[token]/page.js +20 -10
  52. package/dist/modules/auth/frontend/reset/[token]/page.js.map +2 -2
  53. package/dist/modules/auth/lib/setup-app.js +46 -8
  54. package/dist/modules/auth/lib/setup-app.js.map +2 -2
  55. package/dist/modules/auth/notifications.js +112 -0
  56. package/dist/modules/auth/notifications.js.map +7 -0
  57. package/dist/modules/auth/services/authService.js +24 -3
  58. package/dist/modules/auth/services/authService.js.map +2 -2
  59. package/dist/modules/business_rules/api/execute/route.js +7 -1
  60. package/dist/modules/business_rules/api/execute/route.js.map +2 -2
  61. package/dist/modules/business_rules/backend/rules/page.js +4 -0
  62. package/dist/modules/business_rules/backend/rules/page.js.map +2 -2
  63. package/dist/modules/business_rules/backend/sets/page.js +3 -0
  64. package/dist/modules/business_rules/backend/sets/page.js.map +2 -2
  65. package/dist/modules/business_rules/cli.js +2 -1
  66. package/dist/modules/business_rules/cli.js.map +2 -2
  67. package/dist/modules/business_rules/lib/rule-engine.js +33 -3
  68. package/dist/modules/business_rules/lib/rule-engine.js.map +2 -2
  69. package/dist/modules/business_rules/notifications.js +28 -0
  70. package/dist/modules/business_rules/notifications.js.map +7 -0
  71. package/dist/modules/business_rules/subscribers/rule-execution-failed-notification.js +37 -0
  72. package/dist/modules/business_rules/subscribers/rule-execution-failed-notification.js.map +7 -0
  73. package/dist/modules/catalog/components/PriceKindSettings.js +2 -0
  74. package/dist/modules/catalog/components/PriceKindSettings.js.map +2 -2
  75. package/dist/modules/catalog/components/categories/CategoriesDataTable.js +2 -2
  76. package/dist/modules/catalog/components/categories/CategoriesDataTable.js.map +2 -2
  77. package/dist/modules/catalog/components/products/ProductsDataTable.js +2 -0
  78. package/dist/modules/catalog/components/products/ProductsDataTable.js.map +2 -2
  79. package/dist/modules/catalog/notifications.js +28 -0
  80. package/dist/modules/catalog/notifications.js.map +7 -0
  81. package/dist/modules/catalog/subscribers/low-stock-notification.js +38 -0
  82. package/dist/modules/catalog/subscribers/low-stock-notification.js.map +7 -0
  83. package/dist/modules/configs/cli.js +6 -0
  84. package/dist/modules/configs/cli.js.map +2 -2
  85. package/dist/modules/configs/components/CachePanel.js +4 -4
  86. package/dist/modules/configs/components/CachePanel.js.map +2 -2
  87. package/dist/modules/configs/lib/system-status.js +48 -1
  88. package/dist/modules/configs/lib/system-status.js.map +2 -2
  89. package/dist/modules/configs/lib/upgrade-actions.js +138 -306
  90. package/dist/modules/configs/lib/upgrade-actions.js.map +2 -2
  91. package/dist/modules/currencies/backend/currencies/page.js +3 -0
  92. package/dist/modules/currencies/backend/currencies/page.js.map +2 -2
  93. package/dist/modules/currencies/backend/exchange-rates/page.js +2 -0
  94. package/dist/modules/currencies/backend/exchange-rates/page.js.map +2 -2
  95. package/dist/modules/customers/backend/customers/companies/page.js +3 -0
  96. package/dist/modules/customers/backend/customers/companies/page.js.map +2 -2
  97. package/dist/modules/customers/backend/customers/deals/page.js +3 -0
  98. package/dist/modules/customers/backend/customers/deals/page.js.map +2 -2
  99. package/dist/modules/customers/backend/customers/people/page.js +3 -0
  100. package/dist/modules/customers/backend/customers/people/page.js.map +2 -2
  101. package/dist/modules/customers/commands/deals.js +31 -0
  102. package/dist/modules/customers/commands/deals.js.map +2 -2
  103. package/dist/modules/customers/components/CustomerTodosTable.js +1 -0
  104. package/dist/modules/customers/components/CustomerTodosTable.js.map +2 -2
  105. package/dist/modules/customers/notifications.js +48 -0
  106. package/dist/modules/customers/notifications.js.map +7 -0
  107. package/dist/modules/customers/widgets/dashboard/customer-todos/widget.js +2 -1
  108. package/dist/modules/customers/widgets/dashboard/customer-todos/widget.js.map +2 -2
  109. package/dist/modules/customers/widgets/dashboard/new-customers/widget.js +2 -1
  110. package/dist/modules/customers/widgets/dashboard/new-customers/widget.js.map +2 -2
  111. package/dist/modules/customers/widgets/dashboard/new-deals/widget.js +2 -1
  112. package/dist/modules/customers/widgets/dashboard/new-deals/widget.js.map +2 -2
  113. package/dist/modules/customers/widgets/dashboard/next-interactions/widget.js +2 -1
  114. package/dist/modules/customers/widgets/dashboard/next-interactions/widget.js.map +2 -2
  115. package/dist/modules/dashboards/cli.js +44 -5
  116. package/dist/modules/dashboards/cli.js.map +2 -2
  117. package/dist/modules/dashboards/components/WidgetVisibilityEditor.js +16 -11
  118. package/dist/modules/dashboards/components/WidgetVisibilityEditor.js.map +3 -3
  119. package/dist/modules/dashboards/lib/role-widgets.js +58 -0
  120. package/dist/modules/dashboards/lib/role-widgets.js.map +7 -0
  121. package/dist/modules/dashboards/services/widgetDataService.js +139 -3
  122. package/dist/modules/dashboards/services/widgetDataService.js.map +2 -2
  123. package/dist/modules/dashboards/setup.js +15 -0
  124. package/dist/modules/dashboards/setup.js.map +2 -2
  125. package/dist/modules/dashboards/widgets/dashboard/aov-kpi/widget.js +2 -1
  126. package/dist/modules/dashboards/widgets/dashboard/aov-kpi/widget.js.map +2 -2
  127. package/dist/modules/dashboards/widgets/dashboard/new-customers-kpi/widget.js +2 -1
  128. package/dist/modules/dashboards/widgets/dashboard/new-customers-kpi/widget.js.map +2 -2
  129. package/dist/modules/dashboards/widgets/dashboard/orders-by-status/widget.js +2 -1
  130. package/dist/modules/dashboards/widgets/dashboard/orders-by-status/widget.js.map +2 -2
  131. package/dist/modules/dashboards/widgets/dashboard/orders-kpi/widget.js +2 -1
  132. package/dist/modules/dashboards/widgets/dashboard/orders-kpi/widget.js.map +2 -2
  133. package/dist/modules/dashboards/widgets/dashboard/pipeline-summary/widget.js +2 -1
  134. package/dist/modules/dashboards/widgets/dashboard/pipeline-summary/widget.js.map +2 -2
  135. package/dist/modules/dashboards/widgets/dashboard/revenue-kpi/widget.js +2 -1
  136. package/dist/modules/dashboards/widgets/dashboard/revenue-kpi/widget.js.map +2 -2
  137. package/dist/modules/dashboards/widgets/dashboard/revenue-trend/widget.js +2 -1
  138. package/dist/modules/dashboards/widgets/dashboard/revenue-trend/widget.js.map +2 -2
  139. package/dist/modules/dashboards/widgets/dashboard/sales-by-region/widget.js +2 -1
  140. package/dist/modules/dashboards/widgets/dashboard/sales-by-region/widget.js.map +2 -2
  141. package/dist/modules/dashboards/widgets/dashboard/top-customers/widget.js +2 -1
  142. package/dist/modules/dashboards/widgets/dashboard/top-customers/widget.js.map +2 -2
  143. package/dist/modules/dashboards/widgets/dashboard/top-products/widget.js +2 -1
  144. package/dist/modules/dashboards/widgets/dashboard/top-products/widget.js.map +2 -2
  145. package/dist/modules/dictionaries/components/DictionaryTable.js +2 -0
  146. package/dist/modules/dictionaries/components/DictionaryTable.js.map +2 -2
  147. package/dist/modules/directory/api/get/tenants/lookup.js +70 -0
  148. package/dist/modules/directory/api/get/tenants/lookup.js.map +7 -0
  149. package/dist/modules/directory/backend/directory/organizations/page.js +2 -2
  150. package/dist/modules/directory/backend/directory/organizations/page.js.map +2 -2
  151. package/dist/modules/directory/backend/directory/tenants/page.js +2 -2
  152. package/dist/modules/directory/backend/directory/tenants/page.js.map +2 -2
  153. package/dist/modules/entities/backend/entities/user/[entityId]/records/page.js +2 -2
  154. package/dist/modules/entities/backend/entities/user/[entityId]/records/page.js.map +2 -2
  155. package/dist/modules/entities/components/SystemEntitiesTable.js +1 -1
  156. package/dist/modules/entities/components/SystemEntitiesTable.js.map +2 -2
  157. package/dist/modules/entities/components/UserEntitiesTable.js +2 -2
  158. package/dist/modules/entities/components/UserEntitiesTable.js.map +2 -2
  159. package/dist/modules/feature_toggles/components/FeatureTogglesTable.js +3 -3
  160. package/dist/modules/feature_toggles/components/FeatureTogglesTable.js.map +2 -2
  161. package/dist/modules/feature_toggles/components/OverridesTable.js +1 -1
  162. package/dist/modules/feature_toggles/components/OverridesTable.js.map +2 -2
  163. package/dist/modules/notifications/acl.js +11 -0
  164. package/dist/modules/notifications/acl.js.map +7 -0
  165. package/dist/modules/notifications/api/[id]/action/route.js +74 -0
  166. package/dist/modules/notifications/api/[id]/action/route.js.map +7 -0
  167. package/dist/modules/notifications/api/[id]/dismiss/route.js +15 -0
  168. package/dist/modules/notifications/api/[id]/dismiss/route.js.map +7 -0
  169. package/dist/modules/notifications/api/[id]/read/route.js +15 -0
  170. package/dist/modules/notifications/api/[id]/read/route.js.map +7 -0
  171. package/dist/modules/notifications/api/[id]/restore/route.js +53 -0
  172. package/dist/modules/notifications/api/[id]/restore/route.js.map +7 -0
  173. package/dist/modules/notifications/api/batch/route.js +17 -0
  174. package/dist/modules/notifications/api/batch/route.js.map +7 -0
  175. package/dist/modules/notifications/api/feature/route.js +17 -0
  176. package/dist/modules/notifications/api/feature/route.js.map +7 -0
  177. package/dist/modules/notifications/api/mark-all-read/route.js +35 -0
  178. package/dist/modules/notifications/api/mark-all-read/route.js.map +7 -0
  179. package/dist/modules/notifications/api/openapi.js +76 -0
  180. package/dist/modules/notifications/api/openapi.js.map +7 -0
  181. package/dist/modules/notifications/api/role/route.js +17 -0
  182. package/dist/modules/notifications/api/role/route.js.map +7 -0
  183. package/dist/modules/notifications/api/route.js +85 -0
  184. package/dist/modules/notifications/api/route.js.map +7 -0
  185. package/dist/modules/notifications/api/settings/route.js +155 -0
  186. package/dist/modules/notifications/api/settings/route.js.map +7 -0
  187. package/dist/modules/notifications/api/unread-count/route.js +38 -0
  188. package/dist/modules/notifications/api/unread-count/route.js.map +7 -0
  189. package/dist/modules/notifications/backend/config/notifications/page.js +10 -0
  190. package/dist/modules/notifications/backend/config/notifications/page.js.map +7 -0
  191. package/dist/modules/notifications/backend/config/notifications/page.meta.js +24 -0
  192. package/dist/modules/notifications/backend/config/notifications/page.meta.js.map +7 -0
  193. package/dist/modules/notifications/cli.js +16 -0
  194. package/dist/modules/notifications/cli.js.map +7 -0
  195. package/dist/modules/notifications/data/entities.js +112 -0
  196. package/dist/modules/notifications/data/entities.js.map +7 -0
  197. package/dist/modules/notifications/data/validators.js +98 -0
  198. package/dist/modules/notifications/data/validators.js.map +7 -0
  199. package/dist/modules/notifications/di.js +13 -0
  200. package/dist/modules/notifications/di.js.map +7 -0
  201. package/dist/modules/notifications/emails/NotificationEmail.js +58 -0
  202. package/dist/modules/notifications/emails/NotificationEmail.js.map +7 -0
  203. package/dist/modules/notifications/frontend/NotificationInboxPageClient.js +44 -0
  204. package/dist/modules/notifications/frontend/NotificationInboxPageClient.js.map +7 -0
  205. package/dist/modules/notifications/frontend/NotificationSettingsPageClient.js +220 -0
  206. package/dist/modules/notifications/frontend/NotificationSettingsPageClient.js.map +7 -0
  207. package/dist/modules/notifications/index.js +14 -0
  208. package/dist/modules/notifications/index.js.map +7 -0
  209. package/dist/modules/notifications/lib/deliveryConfig.js +107 -0
  210. package/dist/modules/notifications/lib/deliveryConfig.js.map +7 -0
  211. package/dist/modules/notifications/lib/deliveryStrategies.js +14 -0
  212. package/dist/modules/notifications/lib/deliveryStrategies.js.map +7 -0
  213. package/dist/modules/notifications/lib/events.js +12 -0
  214. package/dist/modules/notifications/lib/events.js.map +7 -0
  215. package/dist/modules/notifications/lib/notificationBuilder.js +66 -0
  216. package/dist/modules/notifications/lib/notificationBuilder.js.map +7 -0
  217. package/dist/modules/notifications/lib/notificationFactory.js +54 -0
  218. package/dist/modules/notifications/lib/notificationFactory.js.map +7 -0
  219. package/dist/modules/notifications/lib/notificationMapper.js +34 -0
  220. package/dist/modules/notifications/lib/notificationMapper.js.map +7 -0
  221. package/dist/modules/notifications/lib/notificationRecipients.js +35 -0
  222. package/dist/modules/notifications/lib/notificationRecipients.js.map +7 -0
  223. package/dist/modules/notifications/lib/notificationService.js +279 -0
  224. package/dist/modules/notifications/lib/notificationService.js.map +7 -0
  225. package/dist/modules/notifications/lib/routeHelpers.js +101 -0
  226. package/dist/modules/notifications/lib/routeHelpers.js.map +7 -0
  227. package/dist/modules/notifications/lib/safeHref.js +24 -0
  228. package/dist/modules/notifications/lib/safeHref.js.map +7 -0
  229. package/dist/modules/notifications/migrations/Migration20260123000001.js +70 -0
  230. package/dist/modules/notifications/migrations/Migration20260123000001.js.map +7 -0
  231. package/dist/modules/notifications/migrations/Migration20260126150000.js +37 -0
  232. package/dist/modules/notifications/migrations/Migration20260126150000.js.map +7 -0
  233. package/dist/modules/notifications/migrations/Migration20260129082610.js +13 -0
  234. package/dist/modules/notifications/migrations/Migration20260129082610.js.map +7 -0
  235. package/dist/modules/notifications/subscribers/deliver-notification.js +165 -0
  236. package/dist/modules/notifications/subscribers/deliver-notification.js.map +7 -0
  237. package/dist/modules/notifications/workers/create-notification.worker.js +70 -0
  238. package/dist/modules/notifications/workers/create-notification.worker.js.map +7 -0
  239. package/dist/modules/planner/backend/planner/availability-rulesets/page.js +2 -2
  240. package/dist/modules/planner/backend/planner/availability-rulesets/page.js.map +2 -2
  241. package/dist/modules/query_index/cli.js +63 -7
  242. package/dist/modules/query_index/cli.js.map +2 -2
  243. package/dist/modules/query_index/components/QueryIndexesTable.js +7 -1
  244. package/dist/modules/query_index/components/QueryIndexesTable.js.map +2 -2
  245. package/dist/modules/resources/backend/resources/resource-types/page.js +2 -2
  246. package/dist/modules/resources/backend/resources/resource-types/page.js.map +2 -2
  247. package/dist/modules/resources/backend/resources/resources/page.js +2 -2
  248. package/dist/modules/resources/backend/resources/resources/page.js.map +2 -2
  249. package/dist/modules/sales/backend/sales/channels/offers/page.js +2 -0
  250. package/dist/modules/sales/backend/sales/channels/offers/page.js.map +2 -2
  251. package/dist/modules/sales/backend/sales/channels/page.js +2 -0
  252. package/dist/modules/sales/backend/sales/channels/page.js.map +2 -2
  253. package/dist/modules/sales/cli.js +2 -42
  254. package/dist/modules/sales/cli.js.map +2 -2
  255. package/dist/modules/sales/commands/documents.js +53 -0
  256. package/dist/modules/sales/commands/documents.js.map +2 -2
  257. package/dist/modules/sales/commands/payments.js +26 -0
  258. package/dist/modules/sales/commands/payments.js.map +2 -2
  259. package/dist/modules/sales/components/AdjustmentKindSettings.js +2 -2
  260. package/dist/modules/sales/components/AdjustmentKindSettings.js.map +2 -2
  261. package/dist/modules/sales/components/PaymentMethodsSettings.js +2 -2
  262. package/dist/modules/sales/components/PaymentMethodsSettings.js.map +2 -2
  263. package/dist/modules/sales/components/ShippingMethodsSettings.js +2 -2
  264. package/dist/modules/sales/components/ShippingMethodsSettings.js.map +2 -2
  265. package/dist/modules/sales/components/TaxRatesSettings.js +2 -2
  266. package/dist/modules/sales/components/TaxRatesSettings.js.map +2 -2
  267. package/dist/modules/sales/components/channels/SalesChannelOffersPanel.js +2 -0
  268. package/dist/modules/sales/components/channels/SalesChannelOffersPanel.js.map +2 -2
  269. package/dist/modules/sales/components/documents/AdjustmentsSection.js +2 -0
  270. package/dist/modules/sales/components/documents/AdjustmentsSection.js.map +2 -2
  271. package/dist/modules/sales/components/documents/PaymentsSection.js +2 -1
  272. package/dist/modules/sales/components/documents/PaymentsSection.js.map +2 -2
  273. package/dist/modules/sales/components/documents/SalesDocumentsTable.js +2 -0
  274. package/dist/modules/sales/components/documents/SalesDocumentsTable.js.map +2 -2
  275. package/dist/modules/sales/lib/seeds.js +48 -0
  276. package/dist/modules/sales/lib/seeds.js.map +7 -0
  277. package/dist/modules/sales/notifications.client.js +51 -0
  278. package/dist/modules/sales/notifications.client.js.map +7 -0
  279. package/dist/modules/sales/notifications.js +88 -0
  280. package/dist/modules/sales/notifications.js.map +7 -0
  281. package/dist/modules/sales/subscribers/quote-expiring-notification.js +38 -0
  282. package/dist/modules/sales/subscribers/quote-expiring-notification.js.map +7 -0
  283. package/dist/modules/sales/widgets/notifications/SalesOrderCreatedRenderer.js +137 -0
  284. package/dist/modules/sales/widgets/notifications/SalesOrderCreatedRenderer.js.map +7 -0
  285. package/dist/modules/sales/widgets/notifications/SalesQuoteCreatedRenderer.js +137 -0
  286. package/dist/modules/sales/widgets/notifications/SalesQuoteCreatedRenderer.js.map +7 -0
  287. package/dist/modules/sales/widgets/notifications/index.js +7 -0
  288. package/dist/modules/sales/widgets/notifications/index.js.map +7 -0
  289. package/dist/modules/sales/widgets/notifications/useSalesDocumentTotals.js +60 -0
  290. package/dist/modules/sales/widgets/notifications/useSalesDocumentTotals.js.map +7 -0
  291. package/dist/modules/staff/backend/staff/team-members/page.js +1 -1
  292. package/dist/modules/staff/backend/staff/team-members/page.js.map +2 -2
  293. package/dist/modules/staff/backend/staff/team-roles/page.js +2 -2
  294. package/dist/modules/staff/backend/staff/team-roles/page.js.map +2 -2
  295. package/dist/modules/staff/backend/staff/teams/[id]/edit/page.js +2 -2
  296. package/dist/modules/staff/backend/staff/teams/[id]/edit/page.js.map +2 -2
  297. package/dist/modules/staff/backend/staff/teams/page.js +2 -2
  298. package/dist/modules/staff/backend/staff/teams/page.js.map +2 -2
  299. package/dist/modules/staff/commands/leave-requests.js +79 -0
  300. package/dist/modules/staff/commands/leave-requests.js.map +2 -2
  301. package/dist/modules/staff/notifications.js +75 -0
  302. package/dist/modules/staff/notifications.js.map +7 -0
  303. package/dist/modules/workflows/backend/definitions/page.js +5 -0
  304. package/dist/modules/workflows/backend/definitions/page.js.map +2 -2
  305. package/dist/modules/workflows/backend/instances/page.js +3 -0
  306. package/dist/modules/workflows/backend/instances/page.js.map +2 -2
  307. package/dist/modules/workflows/backend/tasks/page.js +3 -0
  308. package/dist/modules/workflows/backend/tasks/page.js.map +2 -2
  309. package/dist/modules/workflows/cli.js +12 -12
  310. package/dist/modules/workflows/cli.js.map +2 -2
  311. package/dist/modules/workflows/lib/transition-handler.js +14 -6
  312. package/dist/modules/workflows/lib/transition-handler.js.map +2 -2
  313. package/dist/modules/workflows/notifications.js +28 -0
  314. package/dist/modules/workflows/notifications.js.map +7 -0
  315. package/dist/modules/workflows/subscribers/task-assigned-notification.js +38 -0
  316. package/dist/modules/workflows/subscribers/task-assigned-notification.js.map +7 -0
  317. package/generated/entities/notification/index.ts +27 -0
  318. package/generated/entities.ids.generated.ts +5 -1
  319. package/generated/entity-fields-registry.ts +2 -0
  320. package/package.json +2 -2
  321. package/src/modules/api_docs/frontend/docs/api/page.tsx +3 -2
  322. package/src/modules/api_keys/backend/api-keys/page.tsx +1 -1
  323. package/src/modules/attachments/components/AttachmentLibrary.tsx +4 -0
  324. package/src/modules/attachments/components/AttachmentPartitionSettings.tsx +2 -0
  325. package/src/modules/auth/README.md +1 -1
  326. package/src/modules/auth/__tests__/cli-setup-acl.test.ts +1 -1
  327. package/src/modules/auth/api/__tests__/login.test.ts +2 -0
  328. package/src/modules/auth/api/admin/nav.ts +10 -6
  329. package/src/modules/auth/api/login.ts +26 -7
  330. package/src/modules/auth/api/profile/route.ts +163 -0
  331. package/src/modules/auth/api/reset/confirm.ts +25 -2
  332. package/src/modules/auth/api/reset.ts +23 -0
  333. package/src/modules/auth/api/sidebar/preferences/route.ts +21 -12
  334. package/src/modules/auth/api/users/route.ts +5 -2
  335. package/src/modules/auth/backend/auth/profile/page.meta.ts +9 -0
  336. package/src/modules/auth/backend/auth/profile/page.tsx +174 -0
  337. package/src/modules/auth/backend/roles/[id]/edit/page.tsx +4 -1
  338. package/src/modules/auth/backend/roles/page.tsx +3 -3
  339. package/src/modules/auth/backend/users/[id]/edit/page.tsx +22 -3
  340. package/src/modules/auth/backend/users/create/page.tsx +19 -2
  341. package/src/modules/auth/backend/users/page.tsx +3 -3
  342. package/src/modules/auth/cli.ts +38 -11
  343. package/src/modules/auth/commands/users.ts +73 -2
  344. package/src/modules/auth/data/validators.ts +6 -2
  345. package/src/modules/auth/frontend/login.tsx +134 -5
  346. package/src/modules/auth/frontend/reset/[token]/page.tsx +24 -11
  347. package/src/modules/auth/i18n/de.json +48 -1
  348. package/src/modules/auth/i18n/en.json +48 -1
  349. package/src/modules/auth/i18n/es.json +48 -1
  350. package/src/modules/auth/i18n/pl.json +48 -1
  351. package/src/modules/auth/lib/setup-app.ts +63 -9
  352. package/src/modules/auth/notifications.ts +109 -0
  353. package/src/modules/auth/services/authService.ts +27 -4
  354. package/src/modules/business_rules/api/execute/route.ts +8 -1
  355. package/src/modules/business_rules/backend/rules/page.tsx +4 -0
  356. package/src/modules/business_rules/backend/sets/page.tsx +3 -0
  357. package/src/modules/business_rules/cli.ts +2 -1
  358. package/src/modules/business_rules/i18n/en.json +3 -1
  359. package/src/modules/business_rules/lib/__tests__/rule-engine.test.ts +51 -0
  360. package/src/modules/business_rules/lib/rule-engine.ts +57 -3
  361. package/src/modules/business_rules/notifications.ts +25 -0
  362. package/src/modules/business_rules/subscribers/rule-execution-failed-notification.ts +50 -0
  363. package/src/modules/catalog/components/PriceKindSettings.tsx +2 -0
  364. package/src/modules/catalog/components/categories/CategoriesDataTable.tsx +2 -2
  365. package/src/modules/catalog/components/products/ProductsDataTable.tsx +2 -0
  366. package/src/modules/catalog/i18n/en.json +3 -1
  367. package/src/modules/catalog/notifications.ts +25 -0
  368. package/src/modules/catalog/subscribers/low-stock-notification.ts +52 -0
  369. package/src/modules/configs/cli.ts +6 -0
  370. package/src/modules/configs/components/CachePanel.tsx +4 -4
  371. package/src/modules/configs/i18n/en.json +12 -2
  372. package/src/modules/configs/i18n/pl.json +12 -2
  373. package/src/modules/configs/lib/system-status.ts +48 -1
  374. package/src/modules/configs/lib/system-status.types.ts +1 -0
  375. package/src/modules/configs/lib/upgrade-actions.ts +157 -351
  376. package/src/modules/currencies/backend/currencies/page.tsx +3 -0
  377. package/src/modules/currencies/backend/exchange-rates/page.tsx +2 -0
  378. package/src/modules/customers/backend/customers/companies/page.tsx +3 -0
  379. package/src/modules/customers/backend/customers/deals/page.tsx +3 -0
  380. package/src/modules/customers/backend/customers/people/page.tsx +3 -0
  381. package/src/modules/customers/commands/deals.ts +39 -0
  382. package/src/modules/customers/components/CustomerTodosTable.tsx +1 -0
  383. package/src/modules/customers/i18n/en.json +5 -1
  384. package/src/modules/customers/notifications.ts +44 -0
  385. package/src/modules/customers/widgets/dashboard/customer-todos/widget.ts +2 -2
  386. package/src/modules/customers/widgets/dashboard/new-customers/widget.ts +2 -2
  387. package/src/modules/customers/widgets/dashboard/new-deals/widget.ts +2 -2
  388. package/src/modules/customers/widgets/dashboard/next-interactions/widget.ts +2 -2
  389. package/src/modules/dashboards/cli.ts +55 -5
  390. package/src/modules/dashboards/components/WidgetVisibilityEditor.tsx +22 -11
  391. package/src/modules/dashboards/lib/role-widgets.ts +80 -0
  392. package/src/modules/dashboards/services/widgetDataService.ts +164 -4
  393. package/src/modules/dashboards/setup.ts +16 -0
  394. package/src/modules/dashboards/widgets/dashboard/aov-kpi/widget.ts +2 -2
  395. package/src/modules/dashboards/widgets/dashboard/new-customers-kpi/widget.ts +2 -2
  396. package/src/modules/dashboards/widgets/dashboard/orders-by-status/widget.ts +2 -2
  397. package/src/modules/dashboards/widgets/dashboard/orders-kpi/widget.ts +2 -2
  398. package/src/modules/dashboards/widgets/dashboard/pipeline-summary/widget.ts +2 -2
  399. package/src/modules/dashboards/widgets/dashboard/revenue-kpi/widget.ts +2 -2
  400. package/src/modules/dashboards/widgets/dashboard/revenue-trend/widget.ts +2 -2
  401. package/src/modules/dashboards/widgets/dashboard/sales-by-region/widget.ts +2 -2
  402. package/src/modules/dashboards/widgets/dashboard/top-customers/widget.ts +2 -2
  403. package/src/modules/dashboards/widgets/dashboard/top-products/widget.ts +2 -2
  404. package/src/modules/dictionaries/components/DictionaryTable.tsx +2 -0
  405. package/src/modules/directory/api/get/tenants/lookup.ts +75 -0
  406. package/src/modules/directory/backend/directory/organizations/page.tsx +2 -2
  407. package/src/modules/directory/backend/directory/tenants/page.tsx +2 -2
  408. package/src/modules/entities/backend/entities/user/[entityId]/records/page.tsx +2 -2
  409. package/src/modules/entities/components/SystemEntitiesTable.tsx +1 -1
  410. package/src/modules/entities/components/UserEntitiesTable.tsx +2 -2
  411. package/src/modules/feature_toggles/components/FeatureTogglesTable.tsx +3 -4
  412. package/src/modules/feature_toggles/components/OverridesTable.tsx +1 -1
  413. package/src/modules/notifications/__tests__/deliver-notification.test.ts +195 -0
  414. package/src/modules/notifications/__tests__/deliveryStrategies.test.ts +19 -0
  415. package/src/modules/notifications/__tests__/notificationService.test.ts +208 -0
  416. package/src/modules/notifications/acl.ts +7 -0
  417. package/src/modules/notifications/api/[id]/action/route.ts +75 -0
  418. package/src/modules/notifications/api/[id]/dismiss/route.ts +12 -0
  419. package/src/modules/notifications/api/[id]/read/route.ts +12 -0
  420. package/src/modules/notifications/api/[id]/restore/route.ts +53 -0
  421. package/src/modules/notifications/api/batch/route.ts +14 -0
  422. package/src/modules/notifications/api/feature/route.ts +14 -0
  423. package/src/modules/notifications/api/mark-all-read/route.ts +34 -0
  424. package/src/modules/notifications/api/openapi.ts +76 -0
  425. package/src/modules/notifications/api/role/route.ts +14 -0
  426. package/src/modules/notifications/api/route.ts +92 -0
  427. package/src/modules/notifications/api/settings/route.ts +157 -0
  428. package/src/modules/notifications/api/unread-count/route.ts +38 -0
  429. package/src/modules/notifications/backend/config/notifications/page.meta.ts +22 -0
  430. package/src/modules/notifications/backend/config/notifications/page.tsx +12 -0
  431. package/src/modules/notifications/cli.ts +18 -0
  432. package/src/modules/notifications/data/entities.ts +99 -0
  433. package/src/modules/notifications/data/validators.ts +115 -0
  434. package/src/modules/notifications/di.ts +11 -0
  435. package/src/modules/notifications/emails/NotificationEmail.tsx +98 -0
  436. package/src/modules/notifications/frontend/NotificationInboxPageClient.tsx +42 -0
  437. package/src/modules/notifications/frontend/NotificationSettingsPageClient.tsx +233 -0
  438. package/src/modules/notifications/i18n/de.json +50 -0
  439. package/src/modules/notifications/i18n/en.json +50 -0
  440. package/src/modules/notifications/i18n/es.json +50 -0
  441. package/src/modules/notifications/i18n/pl.json +50 -0
  442. package/src/modules/notifications/index.ts +12 -0
  443. package/src/modules/notifications/lib/deliveryConfig.ts +153 -0
  444. package/src/modules/notifications/lib/deliveryStrategies.ts +50 -0
  445. package/src/modules/notifications/lib/events.ts +48 -0
  446. package/src/modules/notifications/lib/notificationBuilder.ts +121 -0
  447. package/src/modules/notifications/lib/notificationFactory.ts +76 -0
  448. package/src/modules/notifications/lib/notificationMapper.ts +33 -0
  449. package/src/modules/notifications/lib/notificationRecipients.ts +83 -0
  450. package/src/modules/notifications/lib/notificationService.ts +414 -0
  451. package/src/modules/notifications/lib/routeHelpers.ts +151 -0
  452. package/src/modules/notifications/lib/safeHref.ts +29 -0
  453. package/src/modules/notifications/migrations/.snapshot-open-mercato.json +336 -0
  454. package/src/modules/notifications/migrations/Migration20260123000001.ts +73 -0
  455. package/src/modules/notifications/migrations/Migration20260126150000.ts +39 -0
  456. package/src/modules/notifications/migrations/Migration20260129082610.ts +13 -0
  457. package/src/modules/notifications/subscribers/deliver-notification.ts +204 -0
  458. package/src/modules/notifications/workers/create-notification.worker.ts +122 -0
  459. package/src/modules/planner/backend/planner/availability-rulesets/page.tsx +2 -2
  460. package/src/modules/query_index/cli.ts +82 -13
  461. package/src/modules/query_index/components/QueryIndexesTable.tsx +8 -2
  462. package/src/modules/resources/backend/resources/resource-types/page.tsx +2 -2
  463. package/src/modules/resources/backend/resources/resources/page.tsx +2 -2
  464. package/src/modules/sales/backend/sales/channels/offers/page.tsx +2 -0
  465. package/src/modules/sales/backend/sales/channels/page.tsx +2 -0
  466. package/src/modules/sales/cli.ts +2 -43
  467. package/src/modules/sales/commands/documents.ts +65 -0
  468. package/src/modules/sales/commands/payments.ts +33 -0
  469. package/src/modules/sales/components/AdjustmentKindSettings.tsx +2 -2
  470. package/src/modules/sales/components/PaymentMethodsSettings.tsx +2 -2
  471. package/src/modules/sales/components/ShippingMethodsSettings.tsx +2 -2
  472. package/src/modules/sales/components/TaxRatesSettings.tsx +2 -2
  473. package/src/modules/sales/components/channels/SalesChannelOffersPanel.tsx +2 -0
  474. package/src/modules/sales/components/documents/AdjustmentsSection.tsx +2 -0
  475. package/src/modules/sales/components/documents/PaymentsSection.tsx +2 -1
  476. package/src/modules/sales/components/documents/SalesDocumentsTable.tsx +2 -0
  477. package/src/modules/sales/i18n/de.json +20 -0
  478. package/src/modules/sales/i18n/en.json +25 -1
  479. package/src/modules/sales/i18n/es.json +20 -0
  480. package/src/modules/sales/i18n/pl.json +20 -0
  481. package/src/modules/sales/lib/seeds.ts +53 -0
  482. package/src/modules/sales/notifications.client.ts +65 -0
  483. package/src/modules/sales/notifications.ts +82 -0
  484. package/src/modules/sales/subscribers/quote-expiring-notification.ts +53 -0
  485. package/src/modules/sales/widgets/notifications/SalesOrderCreatedRenderer.tsx +156 -0
  486. package/src/modules/sales/widgets/notifications/SalesQuoteCreatedRenderer.tsx +156 -0
  487. package/src/modules/sales/widgets/notifications/index.ts +2 -0
  488. package/src/modules/sales/widgets/notifications/useSalesDocumentTotals.ts +81 -0
  489. package/src/modules/staff/backend/staff/team-members/page.tsx +1 -1
  490. package/src/modules/staff/backend/staff/team-roles/page.tsx +2 -2
  491. package/src/modules/staff/backend/staff/teams/[id]/edit/page.tsx +2 -2
  492. package/src/modules/staff/backend/staff/teams/page.tsx +2 -2
  493. package/src/modules/staff/commands/leave-requests.ts +94 -0
  494. package/src/modules/staff/i18n/de.json +6 -0
  495. package/src/modules/staff/i18n/en.json +9 -1
  496. package/src/modules/staff/i18n/es.json +6 -0
  497. package/src/modules/staff/i18n/pl.json +6 -0
  498. package/src/modules/staff/notifications.ts +71 -0
  499. package/src/modules/workflows/backend/definitions/page.tsx +5 -0
  500. package/src/modules/workflows/backend/instances/page.tsx +4 -1
  501. package/src/modules/workflows/backend/tasks/page.tsx +4 -1
  502. package/src/modules/workflows/cli.ts +12 -12
  503. package/src/modules/workflows/i18n/en.json +3 -1
  504. package/src/modules/workflows/lib/__tests__/transition-handler.test.ts +6 -3
  505. package/src/modules/workflows/lib/transition-handler.ts +18 -6
  506. package/src/modules/workflows/notifications.ts +25 -0
  507. package/src/modules/workflows/subscribers/task-assigned-notification.ts +53 -0
@@ -0,0 +1,414 @@
1
+ import type { EntityManager } from '@mikro-orm/core'
2
+ import type { Knex } from 'knex'
3
+ import { Notification } from '../data/entities'
4
+ import type { CreateNotificationInput, CreateBatchNotificationInput, CreateRoleNotificationInput, CreateFeatureNotificationInput, ExecuteActionInput } from '../data/validators'
5
+ import type { NotificationPollData } from '@open-mercato/shared/modules/notifications/types'
6
+ import { NOTIFICATION_EVENTS } from './events'
7
+ import { buildNotificationEntity, emitNotificationCreated, emitNotificationCreatedBatch } from './notificationFactory'
8
+ import { toNotificationDto } from './notificationMapper'
9
+ import { getRecipientUserIdsForFeature, getRecipientUserIdsForRole } from './notificationRecipients'
10
+
11
+ const DEBUG = process.env.NOTIFICATIONS_DEBUG === 'true'
12
+
13
+ function debug(...args: unknown[]): void {
14
+ if (DEBUG) {
15
+ console.log('[notifications]', ...args)
16
+ }
17
+ }
18
+
19
+ function getKnex(em: EntityManager): Knex {
20
+ return (em.getConnection() as unknown as { getKnex: () => Knex }).getKnex()
21
+ }
22
+
23
+ export interface NotificationServiceContext {
24
+ tenantId: string
25
+ organizationId?: string | null
26
+ userId?: string | null
27
+ }
28
+
29
+ export interface NotificationService {
30
+ create(input: CreateNotificationInput, ctx: NotificationServiceContext): Promise<Notification>
31
+ createBatch(input: CreateBatchNotificationInput, ctx: NotificationServiceContext): Promise<Notification[]>
32
+ createForRole(input: CreateRoleNotificationInput, ctx: NotificationServiceContext): Promise<Notification[]>
33
+ createForFeature(input: CreateFeatureNotificationInput, ctx: NotificationServiceContext): Promise<Notification[]>
34
+ markAsRead(notificationId: string, ctx: NotificationServiceContext): Promise<Notification>
35
+ markAllAsRead(ctx: NotificationServiceContext): Promise<number>
36
+ dismiss(notificationId: string, ctx: NotificationServiceContext): Promise<Notification>
37
+ restoreDismissed(
38
+ notificationId: string,
39
+ status: 'read' | 'unread' | undefined,
40
+ ctx: NotificationServiceContext
41
+ ): Promise<Notification>
42
+ executeAction(
43
+ notificationId: string,
44
+ input: ExecuteActionInput,
45
+ ctx: NotificationServiceContext
46
+ ): Promise<{ notification: Notification; result: unknown }>
47
+ getUnreadCount(ctx: NotificationServiceContext): Promise<number>
48
+ getPollData(ctx: NotificationServiceContext, since?: string): Promise<NotificationPollData>
49
+ cleanupExpired(): Promise<number>
50
+ deleteBySource(
51
+ sourceEntityType: string,
52
+ sourceEntityId: string,
53
+ ctx: NotificationServiceContext
54
+ ): Promise<number>
55
+ }
56
+
57
+ export interface NotificationServiceDeps {
58
+ em: EntityManager
59
+ eventBus: { emit: (event: string, payload: unknown) => Promise<void> }
60
+ commandBus?: {
61
+ execute: (
62
+ commandId: string,
63
+ options: { input: unknown; ctx: unknown; metadata?: unknown }
64
+ ) => Promise<{ result: unknown }>
65
+ }
66
+ container?: { resolve: (name: string) => unknown }
67
+ }
68
+
69
+ export function createNotificationService(deps: NotificationServiceDeps): NotificationService {
70
+ const { em: rootEm, eventBus, commandBus, container } = deps
71
+
72
+ return {
73
+ async create(input, ctx) {
74
+ const em = rootEm.fork()
75
+ const { recipientUserId, ...content } = input
76
+ const notification = buildNotificationEntity(em, content, recipientUserId, ctx)
77
+
78
+ await em.persistAndFlush(notification)
79
+
80
+ await emitNotificationCreated(eventBus, notification, ctx)
81
+
82
+ return notification
83
+ },
84
+
85
+ async createBatch(input, ctx) {
86
+ const em = rootEm.fork()
87
+ const { recipientUserIds, ...content } = input
88
+ const notifications: Notification[] = []
89
+
90
+ for (const recipientUserId of recipientUserIds) {
91
+ const notification = buildNotificationEntity(em, content, recipientUserId, ctx)
92
+ notifications.push(notification)
93
+ }
94
+
95
+ await em.persistAndFlush(notifications)
96
+
97
+ await emitNotificationCreatedBatch(eventBus, notifications, ctx)
98
+
99
+ return notifications
100
+ },
101
+
102
+ async createForRole(input, ctx) {
103
+ const em = rootEm.fork()
104
+
105
+ const knex = getKnex(em)
106
+ const recipientUserIds = await getRecipientUserIdsForRole(knex, ctx.tenantId, input.roleId)
107
+ if (recipientUserIds.length === 0) {
108
+ return []
109
+ }
110
+
111
+ const { roleId: _roleId, ...content } = input
112
+ const notifications: Notification[] = []
113
+
114
+ for (const recipientUserId of recipientUserIds) {
115
+ const notification = buildNotificationEntity(em, content, recipientUserId, ctx)
116
+ notifications.push(notification)
117
+ }
118
+
119
+ await em.persistAndFlush(notifications)
120
+
121
+ await emitNotificationCreatedBatch(eventBus, notifications, ctx)
122
+
123
+ return notifications
124
+ },
125
+
126
+ async createForFeature(input, ctx) {
127
+ const em = rootEm.fork()
128
+ const knex = getKnex(em)
129
+ const recipientUserIds = await getRecipientUserIdsForFeature(knex, ctx.tenantId, input.requiredFeature)
130
+
131
+ if (recipientUserIds.length === 0) {
132
+ debug('No users found with feature:', input.requiredFeature, 'in tenant:', ctx.tenantId)
133
+ return []
134
+ }
135
+
136
+ debug('Creating notifications for', recipientUserIds.length, 'user(s) with feature:', input.requiredFeature)
137
+
138
+ const { requiredFeature: _requiredFeature, ...content } = input
139
+ const notifications: Notification[] = []
140
+
141
+ for (const recipientUserId of recipientUserIds) {
142
+ const notification = buildNotificationEntity(em, content, recipientUserId, ctx)
143
+ notifications.push(notification)
144
+ }
145
+
146
+ await em.persistAndFlush(notifications)
147
+
148
+ await emitNotificationCreatedBatch(eventBus, notifications, ctx)
149
+
150
+ return notifications
151
+ },
152
+
153
+ async markAsRead(notificationId, ctx) {
154
+ const em = rootEm.fork()
155
+ const notification = await em.findOneOrFail(Notification, {
156
+ id: notificationId,
157
+ recipientUserId: ctx.userId,
158
+ tenantId: ctx.tenantId,
159
+ })
160
+
161
+ if (notification.status === 'unread') {
162
+ notification.status = 'read'
163
+ notification.readAt = new Date()
164
+ await em.flush()
165
+
166
+ await eventBus.emit(NOTIFICATION_EVENTS.READ, {
167
+ notificationId: notification.id,
168
+ userId: ctx.userId,
169
+ tenantId: ctx.tenantId,
170
+ })
171
+ }
172
+
173
+ return notification
174
+ },
175
+
176
+ async markAllAsRead(ctx) {
177
+ const em = rootEm.fork()
178
+ const knex = getKnex(em)
179
+
180
+ const result = await knex('notifications')
181
+ .where({
182
+ recipient_user_id: ctx.userId,
183
+ tenant_id: ctx.tenantId,
184
+ status: 'unread',
185
+ })
186
+ .update({
187
+ status: 'read',
188
+ read_at: knex.fn.now(),
189
+ })
190
+
191
+ return result
192
+ },
193
+
194
+ async dismiss(notificationId, ctx) {
195
+ const em = rootEm.fork()
196
+ const notification = await em.findOneOrFail(Notification, {
197
+ id: notificationId,
198
+ recipientUserId: ctx.userId,
199
+ tenantId: ctx.tenantId,
200
+ })
201
+
202
+ notification.status = 'dismissed'
203
+ notification.dismissedAt = new Date()
204
+ await em.flush()
205
+
206
+ await eventBus.emit(NOTIFICATION_EVENTS.DISMISSED, {
207
+ notificationId: notification.id,
208
+ userId: ctx.userId,
209
+ tenantId: ctx.tenantId,
210
+ })
211
+
212
+ return notification
213
+ },
214
+
215
+ async restoreDismissed(notificationId, status, ctx) {
216
+ const em = rootEm.fork()
217
+ const notification = await em.findOneOrFail(Notification, {
218
+ id: notificationId,
219
+ recipientUserId: ctx.userId,
220
+ tenantId: ctx.tenantId,
221
+ })
222
+
223
+ if (notification.status !== 'dismissed') {
224
+ return notification
225
+ }
226
+
227
+ const targetStatus = status ?? 'read'
228
+ notification.status = targetStatus
229
+ notification.dismissedAt = null
230
+
231
+ if (targetStatus === 'unread') {
232
+ notification.readAt = null
233
+ } else if (!notification.readAt) {
234
+ notification.readAt = new Date()
235
+ }
236
+
237
+ await em.flush()
238
+
239
+ await eventBus.emit(NOTIFICATION_EVENTS.RESTORED, {
240
+ notificationId: notification.id,
241
+ userId: ctx.userId,
242
+ tenantId: ctx.tenantId,
243
+ status: targetStatus,
244
+ })
245
+
246
+ return notification
247
+ },
248
+
249
+ async executeAction(notificationId, input, ctx) {
250
+ const em = rootEm.fork()
251
+ const notification = await em.findOneOrFail(Notification, {
252
+ id: notificationId,
253
+ recipientUserId: ctx.userId,
254
+ tenantId: ctx.tenantId,
255
+ })
256
+
257
+ const actionData = notification.actionData
258
+ const action = actionData?.actions?.find((a) => a.id === input.actionId)
259
+
260
+ if (!action) {
261
+ throw new Error('Action not found')
262
+ }
263
+
264
+ let result: unknown = null
265
+
266
+ if (action.commandId && commandBus && container) {
267
+ const commandInput = {
268
+ id: notification.sourceEntityId,
269
+ ...input.payload,
270
+ }
271
+
272
+ // Build a CommandRuntimeContext from the notification service context
273
+ const commandCtx = {
274
+ container,
275
+ auth: {
276
+ sub: ctx.userId,
277
+ tenantId: ctx.tenantId,
278
+ orgId: ctx.organizationId,
279
+ },
280
+ organizationScope: null,
281
+ selectedOrganizationId: ctx.organizationId ?? null,
282
+ organizationIds: ctx.organizationId ? [ctx.organizationId] : null,
283
+ }
284
+
285
+ const commandResult = await commandBus.execute(action.commandId, {
286
+ input: commandInput,
287
+ ctx: commandCtx,
288
+ metadata: {
289
+ tenantId: ctx.tenantId,
290
+ organizationId: ctx.organizationId,
291
+ resourceKind: 'notifications',
292
+ },
293
+ })
294
+
295
+ result = commandResult.result
296
+ }
297
+
298
+ notification.status = 'actioned'
299
+ notification.actionedAt = new Date()
300
+ notification.actionTaken = input.actionId
301
+ notification.actionResult = result as Record<string, unknown>
302
+
303
+ if (!notification.readAt) {
304
+ notification.readAt = new Date()
305
+ }
306
+
307
+ await em.flush()
308
+
309
+ await eventBus.emit(NOTIFICATION_EVENTS.ACTIONED, {
310
+ notificationId: notification.id,
311
+ actionId: input.actionId,
312
+ userId: ctx.userId,
313
+ tenantId: ctx.tenantId,
314
+ })
315
+
316
+ return { notification, result }
317
+ },
318
+
319
+ async getUnreadCount(ctx) {
320
+ const em = rootEm.fork()
321
+ return em.count(Notification, {
322
+ recipientUserId: ctx.userId,
323
+ tenantId: ctx.tenantId,
324
+ status: 'unread',
325
+ })
326
+ },
327
+
328
+ async getPollData(ctx, since) {
329
+ const em = rootEm.fork()
330
+ const filters: Record<string, unknown> = {
331
+ recipientUserId: ctx.userId,
332
+ tenantId: ctx.tenantId,
333
+ }
334
+
335
+ if (since) {
336
+ filters.createdAt = { $gt: new Date(since) }
337
+ }
338
+
339
+ const [notifications, unreadCount] = await Promise.all([
340
+ em.find(Notification, filters, {
341
+ orderBy: { createdAt: 'desc' },
342
+ limit: 50,
343
+ }),
344
+ em.count(Notification, {
345
+ recipientUserId: ctx.userId,
346
+ tenantId: ctx.tenantId,
347
+ status: 'unread',
348
+ }),
349
+ ])
350
+
351
+ const recent = notifications.map(toNotificationDto)
352
+ const hasNew = since ? recent.length > 0 : false
353
+
354
+ return {
355
+ unreadCount,
356
+ recent,
357
+ hasNew,
358
+ lastId: recent[0]?.id,
359
+ }
360
+ },
361
+
362
+ async cleanupExpired() {
363
+ const em = rootEm.fork()
364
+ const knex = getKnex(em)
365
+
366
+ const result = await knex('notifications')
367
+ .where('expires_at', '<', knex.fn.now())
368
+ .whereNotIn('status', ['actioned', 'dismissed'])
369
+ .update({
370
+ status: 'dismissed',
371
+ dismissed_at: knex.fn.now(),
372
+ })
373
+
374
+ return result
375
+ },
376
+
377
+ async deleteBySource(sourceEntityType, sourceEntityId, ctx) {
378
+ const em = rootEm.fork()
379
+ const knex = getKnex(em)
380
+
381
+ const result = await knex('notifications')
382
+ .where({
383
+ source_entity_type: sourceEntityType,
384
+ source_entity_id: sourceEntityId,
385
+ tenant_id: ctx.tenantId,
386
+ })
387
+ .delete()
388
+
389
+ return result
390
+ },
391
+ }
392
+ }
393
+
394
+ /**
395
+ * Helper to create notification service from a DI container.
396
+ * Use this in API routes and commands to avoid DI resolution issues.
397
+ */
398
+ export function resolveNotificationService(container: {
399
+ resolve: (name: string) => unknown
400
+ }): NotificationService {
401
+ const em = container.resolve('em') as EntityManager
402
+ const eventBus = container.resolve('eventBus') as { emit: (event: string, payload: unknown) => Promise<void> }
403
+
404
+ // commandBus may not be registered in all contexts, so resolve it safely
405
+ let commandBus: NotificationServiceDeps['commandBus']
406
+ try {
407
+ commandBus = container.resolve('commandBus') as typeof commandBus
408
+ } catch {
409
+ // commandBus not available - actions with commandId won't work
410
+ commandBus = undefined
411
+ }
412
+
413
+ return createNotificationService({ em, eventBus, commandBus, container })
414
+ }
@@ -0,0 +1,151 @@
1
+ import { z } from 'zod'
2
+ import { resolveRequestContext } from '@open-mercato/shared/lib/api/context'
3
+ import { resolveNotificationService, type NotificationService } from './notificationService'
4
+
5
+ /**
6
+ * Notification scope context for service calls
7
+ */
8
+ export interface NotificationScope {
9
+ tenantId: string
10
+ organizationId: string | null
11
+ userId: string | null
12
+ }
13
+
14
+ /**
15
+ * Resolved notification context from a request
16
+ */
17
+ export interface NotificationRequestContext {
18
+ service: NotificationService
19
+ scope: NotificationScope
20
+ ctx: Awaited<ReturnType<typeof resolveRequestContext>>['ctx']
21
+ }
22
+
23
+ /**
24
+ * Resolve notification service and scope from a request.
25
+ * Centralizes the common pattern used across all notification API routes.
26
+ */
27
+ export async function resolveNotificationContext(req: Request): Promise<NotificationRequestContext> {
28
+ const { ctx } = await resolveRequestContext(req)
29
+ return {
30
+ service: resolveNotificationService(ctx.container),
31
+ scope: {
32
+ tenantId: ctx.auth?.tenantId ?? '',
33
+ organizationId: ctx.selectedOrganizationId ?? null,
34
+ userId: ctx.auth?.sub ?? null,
35
+ },
36
+ ctx,
37
+ }
38
+ }
39
+
40
+ /**
41
+ * Create a POST handler for bulk notification creation routes.
42
+ * Used by batch, role, and feature notification endpoints.
43
+ */
44
+ export function createBulkNotificationRoute<TSchema extends z.ZodTypeAny>(
45
+ schema: TSchema,
46
+ serviceMethod: 'createBatch' | 'createForRole' | 'createForFeature'
47
+ ) {
48
+ return async function POST(req: Request) {
49
+ const { service, scope } = await resolveNotificationContext(req)
50
+
51
+ const body = await req.json().catch(() => ({}))
52
+ const input = schema.parse(body)
53
+
54
+ const notifications = await service[serviceMethod](input as never, scope)
55
+
56
+ return Response.json({
57
+ ok: true,
58
+ count: notifications.length,
59
+ ids: notifications.map((n) => n.id),
60
+ }, { status: 201 })
61
+ }
62
+ }
63
+
64
+ /**
65
+ * Create OpenAPI spec for bulk notification creation routes.
66
+ */
67
+ export function createBulkNotificationOpenApi<TSchema extends z.ZodTypeAny>(
68
+ schema: TSchema,
69
+ summary: string,
70
+ description?: string
71
+ ) {
72
+ return {
73
+ POST: {
74
+ summary,
75
+ description,
76
+ tags: ['Notifications'],
77
+ requestBody: {
78
+ required: true,
79
+ content: {
80
+ 'application/json': {
81
+ schema,
82
+ },
83
+ },
84
+ },
85
+ responses: {
86
+ 201: {
87
+ description: 'Notifications created',
88
+ content: {
89
+ 'application/json': {
90
+ schema: z.object({
91
+ ok: z.boolean(),
92
+ count: z.number(),
93
+ ids: z.array(z.string().uuid()),
94
+ }),
95
+ },
96
+ },
97
+ },
98
+ },
99
+ },
100
+ }
101
+ }
102
+
103
+ /**
104
+ * Create a PUT handler for single notification action routes.
105
+ * Used by read and dismiss endpoints.
106
+ */
107
+ export function createSingleNotificationActionRoute(
108
+ serviceMethod: 'markAsRead' | 'dismiss'
109
+ ) {
110
+ return async function PUT(req: Request, { params }: { params: Promise<{ id: string }> }) {
111
+ const { id } = await params
112
+ const { service, scope } = await resolveNotificationContext(req)
113
+
114
+ await service[serviceMethod](id, scope)
115
+
116
+ return Response.json({ ok: true })
117
+ }
118
+ }
119
+
120
+ /**
121
+ * Create OpenAPI spec for single notification action routes.
122
+ */
123
+ export function createSingleNotificationActionOpenApi(
124
+ summary: string,
125
+ description: string
126
+ ) {
127
+ return {
128
+ PUT: {
129
+ summary,
130
+ tags: ['Notifications'],
131
+ parameters: [
132
+ {
133
+ name: 'id',
134
+ in: 'path',
135
+ required: true,
136
+ schema: { type: 'string', format: 'uuid' },
137
+ },
138
+ ],
139
+ responses: {
140
+ 200: {
141
+ description,
142
+ content: {
143
+ 'application/json': {
144
+ schema: z.object({ ok: z.boolean() }),
145
+ },
146
+ },
147
+ },
148
+ },
149
+ },
150
+ }
151
+ }
@@ -0,0 +1,29 @@
1
+ import type { NotificationAction } from '@open-mercato/shared/modules/notifications/types'
2
+
3
+ export function isSafeNotificationHref(href: string): boolean {
4
+ return href.startsWith('/') && !href.startsWith('//')
5
+ }
6
+
7
+ export function assertSafeNotificationHref(href: string | undefined | null): string | undefined {
8
+ if (href == null) {
9
+ return undefined
10
+ }
11
+
12
+ if (!isSafeNotificationHref(href)) {
13
+ throw new Error('Notification href must be a same-origin relative path starting with /')
14
+ }
15
+
16
+ return href
17
+ }
18
+
19
+ export function sanitizeNotificationActions(
20
+ actions: NotificationAction[] | undefined
21
+ ): NotificationAction[] | undefined {
22
+ if (!actions) {
23
+ return undefined
24
+ }
25
+
26
+ return actions.map((action) => (
27
+ action.href ? { ...action, href: assertSafeNotificationHref(action.href) } : action
28
+ ))
29
+ }