@open-mercato/core 0.4.2-canary-e6bf6a353e → 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 (397) 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 +1 -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/components/PriceKindSettings.js +2 -0
  62. package/dist/modules/catalog/components/PriceKindSettings.js.map +2 -2
  63. package/dist/modules/catalog/components/categories/CategoriesDataTable.js +2 -2
  64. package/dist/modules/catalog/components/categories/CategoriesDataTable.js.map +2 -2
  65. package/dist/modules/catalog/components/products/ProductsDataTable.js +2 -0
  66. package/dist/modules/catalog/components/products/ProductsDataTable.js.map +2 -2
  67. package/dist/modules/catalog/notifications.js +28 -0
  68. package/dist/modules/catalog/notifications.js.map +7 -0
  69. package/dist/modules/catalog/subscribers/low-stock-notification.js +38 -0
  70. package/dist/modules/catalog/subscribers/low-stock-notification.js.map +7 -0
  71. package/dist/modules/configs/cli.js +6 -0
  72. package/dist/modules/configs/cli.js.map +2 -2
  73. package/dist/modules/configs/lib/upgrade-actions.js +18 -0
  74. package/dist/modules/configs/lib/upgrade-actions.js.map +2 -2
  75. package/dist/modules/currencies/backend/currencies/page.js +3 -0
  76. package/dist/modules/currencies/backend/currencies/page.js.map +2 -2
  77. package/dist/modules/currencies/backend/exchange-rates/page.js +2 -0
  78. package/dist/modules/currencies/backend/exchange-rates/page.js.map +2 -2
  79. package/dist/modules/customers/backend/customers/companies/page.js +3 -0
  80. package/dist/modules/customers/backend/customers/companies/page.js.map +2 -2
  81. package/dist/modules/customers/backend/customers/deals/page.js +3 -0
  82. package/dist/modules/customers/backend/customers/deals/page.js.map +2 -2
  83. package/dist/modules/customers/backend/customers/people/page.js +3 -0
  84. package/dist/modules/customers/backend/customers/people/page.js.map +2 -2
  85. package/dist/modules/customers/commands/deals.js +31 -0
  86. package/dist/modules/customers/commands/deals.js.map +2 -2
  87. package/dist/modules/customers/components/CustomerTodosTable.js +1 -0
  88. package/dist/modules/customers/components/CustomerTodosTable.js.map +2 -2
  89. package/dist/modules/customers/notifications.js +48 -0
  90. package/dist/modules/customers/notifications.js.map +7 -0
  91. package/dist/modules/dashboards/cli.js +32 -1
  92. package/dist/modules/dashboards/cli.js.map +2 -2
  93. package/dist/modules/dashboards/lib/role-widgets.js +58 -0
  94. package/dist/modules/dashboards/lib/role-widgets.js.map +7 -0
  95. package/dist/modules/dictionaries/components/DictionaryTable.js +2 -0
  96. package/dist/modules/dictionaries/components/DictionaryTable.js.map +2 -2
  97. package/dist/modules/directory/backend/directory/organizations/page.js +2 -2
  98. package/dist/modules/directory/backend/directory/organizations/page.js.map +2 -2
  99. package/dist/modules/directory/backend/directory/tenants/page.js +2 -2
  100. package/dist/modules/directory/backend/directory/tenants/page.js.map +2 -2
  101. package/dist/modules/entities/backend/entities/user/[entityId]/records/page.js +2 -2
  102. package/dist/modules/entities/backend/entities/user/[entityId]/records/page.js.map +2 -2
  103. package/dist/modules/entities/components/SystemEntitiesTable.js +1 -1
  104. package/dist/modules/entities/components/SystemEntitiesTable.js.map +2 -2
  105. package/dist/modules/entities/components/UserEntitiesTable.js +2 -2
  106. package/dist/modules/entities/components/UserEntitiesTable.js.map +2 -2
  107. package/dist/modules/feature_toggles/components/FeatureTogglesTable.js +3 -3
  108. package/dist/modules/feature_toggles/components/FeatureTogglesTable.js.map +2 -2
  109. package/dist/modules/feature_toggles/components/OverridesTable.js +1 -1
  110. package/dist/modules/feature_toggles/components/OverridesTable.js.map +2 -2
  111. package/dist/modules/notifications/acl.js +11 -0
  112. package/dist/modules/notifications/acl.js.map +7 -0
  113. package/dist/modules/notifications/api/[id]/action/route.js +74 -0
  114. package/dist/modules/notifications/api/[id]/action/route.js.map +7 -0
  115. package/dist/modules/notifications/api/[id]/dismiss/route.js +15 -0
  116. package/dist/modules/notifications/api/[id]/dismiss/route.js.map +7 -0
  117. package/dist/modules/notifications/api/[id]/read/route.js +15 -0
  118. package/dist/modules/notifications/api/[id]/read/route.js.map +7 -0
  119. package/dist/modules/notifications/api/[id]/restore/route.js +53 -0
  120. package/dist/modules/notifications/api/[id]/restore/route.js.map +7 -0
  121. package/dist/modules/notifications/api/batch/route.js +17 -0
  122. package/dist/modules/notifications/api/batch/route.js.map +7 -0
  123. package/dist/modules/notifications/api/feature/route.js +17 -0
  124. package/dist/modules/notifications/api/feature/route.js.map +7 -0
  125. package/dist/modules/notifications/api/mark-all-read/route.js +35 -0
  126. package/dist/modules/notifications/api/mark-all-read/route.js.map +7 -0
  127. package/dist/modules/notifications/api/openapi.js +76 -0
  128. package/dist/modules/notifications/api/openapi.js.map +7 -0
  129. package/dist/modules/notifications/api/role/route.js +17 -0
  130. package/dist/modules/notifications/api/role/route.js.map +7 -0
  131. package/dist/modules/notifications/api/route.js +85 -0
  132. package/dist/modules/notifications/api/route.js.map +7 -0
  133. package/dist/modules/notifications/api/settings/route.js +155 -0
  134. package/dist/modules/notifications/api/settings/route.js.map +7 -0
  135. package/dist/modules/notifications/api/unread-count/route.js +38 -0
  136. package/dist/modules/notifications/api/unread-count/route.js.map +7 -0
  137. package/dist/modules/notifications/backend/config/notifications/page.js +10 -0
  138. package/dist/modules/notifications/backend/config/notifications/page.js.map +7 -0
  139. package/dist/modules/notifications/backend/config/notifications/page.meta.js +24 -0
  140. package/dist/modules/notifications/backend/config/notifications/page.meta.js.map +7 -0
  141. package/dist/modules/notifications/cli.js +16 -0
  142. package/dist/modules/notifications/cli.js.map +7 -0
  143. package/dist/modules/notifications/data/entities.js +112 -0
  144. package/dist/modules/notifications/data/entities.js.map +7 -0
  145. package/dist/modules/notifications/data/validators.js +94 -0
  146. package/dist/modules/notifications/data/validators.js.map +7 -0
  147. package/dist/modules/notifications/di.js +13 -0
  148. package/dist/modules/notifications/di.js.map +7 -0
  149. package/dist/modules/notifications/emails/NotificationEmail.js +58 -0
  150. package/dist/modules/notifications/emails/NotificationEmail.js.map +7 -0
  151. package/dist/modules/notifications/frontend/NotificationInboxPageClient.js +44 -0
  152. package/dist/modules/notifications/frontend/NotificationInboxPageClient.js.map +7 -0
  153. package/dist/modules/notifications/frontend/NotificationSettingsPageClient.js +219 -0
  154. package/dist/modules/notifications/frontend/NotificationSettingsPageClient.js.map +7 -0
  155. package/dist/modules/notifications/index.js +14 -0
  156. package/dist/modules/notifications/index.js.map +7 -0
  157. package/dist/modules/notifications/lib/deliveryConfig.js +105 -0
  158. package/dist/modules/notifications/lib/deliveryConfig.js.map +7 -0
  159. package/dist/modules/notifications/lib/events.js +12 -0
  160. package/dist/modules/notifications/lib/events.js.map +7 -0
  161. package/dist/modules/notifications/lib/notificationBuilder.js +66 -0
  162. package/dist/modules/notifications/lib/notificationBuilder.js.map +7 -0
  163. package/dist/modules/notifications/lib/notificationFactory.js +54 -0
  164. package/dist/modules/notifications/lib/notificationFactory.js.map +7 -0
  165. package/dist/modules/notifications/lib/notificationMapper.js +34 -0
  166. package/dist/modules/notifications/lib/notificationMapper.js.map +7 -0
  167. package/dist/modules/notifications/lib/notificationRecipients.js +35 -0
  168. package/dist/modules/notifications/lib/notificationRecipients.js.map +7 -0
  169. package/dist/modules/notifications/lib/notificationService.js +279 -0
  170. package/dist/modules/notifications/lib/notificationService.js.map +7 -0
  171. package/dist/modules/notifications/lib/routeHelpers.js +101 -0
  172. package/dist/modules/notifications/lib/routeHelpers.js.map +7 -0
  173. package/dist/modules/notifications/lib/safeHref.js +24 -0
  174. package/dist/modules/notifications/lib/safeHref.js.map +7 -0
  175. package/dist/modules/notifications/migrations/Migration20260123000001.js +70 -0
  176. package/dist/modules/notifications/migrations/Migration20260123000001.js.map +7 -0
  177. package/dist/modules/notifications/migrations/Migration20260126150000.js +37 -0
  178. package/dist/modules/notifications/migrations/Migration20260126150000.js.map +7 -0
  179. package/dist/modules/notifications/subscribers/deliver-notification.js +139 -0
  180. package/dist/modules/notifications/subscribers/deliver-notification.js.map +7 -0
  181. package/dist/modules/notifications/workers/create-notification.worker.js +70 -0
  182. package/dist/modules/notifications/workers/create-notification.worker.js.map +7 -0
  183. package/dist/modules/planner/backend/planner/availability-rulesets/page.js +2 -2
  184. package/dist/modules/planner/backend/planner/availability-rulesets/page.js.map +2 -2
  185. package/dist/modules/query_index/components/QueryIndexesTable.js +7 -1
  186. package/dist/modules/query_index/components/QueryIndexesTable.js.map +2 -2
  187. package/dist/modules/resources/backend/resources/resource-types/page.js +2 -2
  188. package/dist/modules/resources/backend/resources/resource-types/page.js.map +2 -2
  189. package/dist/modules/resources/backend/resources/resources/page.js +2 -2
  190. package/dist/modules/resources/backend/resources/resources/page.js.map +2 -2
  191. package/dist/modules/sales/backend/sales/channels/offers/page.js +2 -0
  192. package/dist/modules/sales/backend/sales/channels/offers/page.js.map +2 -2
  193. package/dist/modules/sales/backend/sales/channels/page.js +2 -0
  194. package/dist/modules/sales/backend/sales/channels/page.js.map +2 -2
  195. package/dist/modules/sales/commands/documents.js +53 -0
  196. package/dist/modules/sales/commands/documents.js.map +2 -2
  197. package/dist/modules/sales/commands/payments.js +26 -0
  198. package/dist/modules/sales/commands/payments.js.map +2 -2
  199. package/dist/modules/sales/components/AdjustmentKindSettings.js +2 -2
  200. package/dist/modules/sales/components/AdjustmentKindSettings.js.map +2 -2
  201. package/dist/modules/sales/components/PaymentMethodsSettings.js +2 -2
  202. package/dist/modules/sales/components/PaymentMethodsSettings.js.map +2 -2
  203. package/dist/modules/sales/components/ShippingMethodsSettings.js +2 -2
  204. package/dist/modules/sales/components/ShippingMethodsSettings.js.map +2 -2
  205. package/dist/modules/sales/components/TaxRatesSettings.js +2 -2
  206. package/dist/modules/sales/components/TaxRatesSettings.js.map +2 -2
  207. package/dist/modules/sales/components/channels/SalesChannelOffersPanel.js +2 -0
  208. package/dist/modules/sales/components/channels/SalesChannelOffersPanel.js.map +2 -2
  209. package/dist/modules/sales/components/documents/AdjustmentsSection.js +2 -0
  210. package/dist/modules/sales/components/documents/AdjustmentsSection.js.map +2 -2
  211. package/dist/modules/sales/components/documents/PaymentsSection.js +2 -1
  212. package/dist/modules/sales/components/documents/PaymentsSection.js.map +2 -2
  213. package/dist/modules/sales/components/documents/SalesDocumentsTable.js +2 -0
  214. package/dist/modules/sales/components/documents/SalesDocumentsTable.js.map +2 -2
  215. package/dist/modules/sales/notifications.client.js +51 -0
  216. package/dist/modules/sales/notifications.client.js.map +7 -0
  217. package/dist/modules/sales/notifications.js +88 -0
  218. package/dist/modules/sales/notifications.js.map +7 -0
  219. package/dist/modules/sales/subscribers/quote-expiring-notification.js +38 -0
  220. package/dist/modules/sales/subscribers/quote-expiring-notification.js.map +7 -0
  221. package/dist/modules/sales/widgets/notifications/SalesOrderCreatedRenderer.js +137 -0
  222. package/dist/modules/sales/widgets/notifications/SalesOrderCreatedRenderer.js.map +7 -0
  223. package/dist/modules/sales/widgets/notifications/SalesQuoteCreatedRenderer.js +137 -0
  224. package/dist/modules/sales/widgets/notifications/SalesQuoteCreatedRenderer.js.map +7 -0
  225. package/dist/modules/sales/widgets/notifications/index.js +7 -0
  226. package/dist/modules/sales/widgets/notifications/index.js.map +7 -0
  227. package/dist/modules/sales/widgets/notifications/useSalesDocumentTotals.js +60 -0
  228. package/dist/modules/sales/widgets/notifications/useSalesDocumentTotals.js.map +7 -0
  229. package/dist/modules/staff/backend/staff/team-members/page.js +1 -1
  230. package/dist/modules/staff/backend/staff/team-members/page.js.map +2 -2
  231. package/dist/modules/staff/backend/staff/team-roles/page.js +2 -2
  232. package/dist/modules/staff/backend/staff/team-roles/page.js.map +2 -2
  233. package/dist/modules/staff/backend/staff/teams/[id]/edit/page.js +2 -2
  234. package/dist/modules/staff/backend/staff/teams/[id]/edit/page.js.map +2 -2
  235. package/dist/modules/staff/backend/staff/teams/page.js +2 -2
  236. package/dist/modules/staff/backend/staff/teams/page.js.map +2 -2
  237. package/dist/modules/staff/commands/leave-requests.js +79 -0
  238. package/dist/modules/staff/commands/leave-requests.js.map +2 -2
  239. package/dist/modules/staff/notifications.js +75 -0
  240. package/dist/modules/staff/notifications.js.map +7 -0
  241. package/dist/modules/workflows/backend/definitions/page.js +5 -0
  242. package/dist/modules/workflows/backend/definitions/page.js.map +2 -2
  243. package/dist/modules/workflows/backend/instances/page.js +3 -0
  244. package/dist/modules/workflows/backend/instances/page.js.map +2 -2
  245. package/dist/modules/workflows/backend/tasks/page.js +3 -0
  246. package/dist/modules/workflows/backend/tasks/page.js.map +2 -2
  247. package/dist/modules/workflows/notifications.js +28 -0
  248. package/dist/modules/workflows/notifications.js.map +7 -0
  249. package/dist/modules/workflows/subscribers/task-assigned-notification.js +38 -0
  250. package/dist/modules/workflows/subscribers/task-assigned-notification.js.map +7 -0
  251. package/generated/entities/notification/index.ts +27 -0
  252. package/generated/entities.ids.generated.ts +5 -1
  253. package/generated/entity-fields-registry.ts +2 -0
  254. package/package.json +2 -2
  255. package/src/modules/api_docs/frontend/docs/api/page.tsx +3 -2
  256. package/src/modules/api_keys/backend/api-keys/page.tsx +1 -1
  257. package/src/modules/attachments/components/AttachmentLibrary.tsx +4 -0
  258. package/src/modules/attachments/components/AttachmentPartitionSettings.tsx +2 -0
  259. package/src/modules/auth/api/admin/nav.ts +10 -6
  260. package/src/modules/auth/api/profile/route.ts +163 -0
  261. package/src/modules/auth/api/reset/confirm.ts +25 -2
  262. package/src/modules/auth/api/reset.ts +23 -0
  263. package/src/modules/auth/api/sidebar/preferences/route.ts +21 -12
  264. package/src/modules/auth/api/users/route.ts +5 -2
  265. package/src/modules/auth/backend/auth/profile/page.meta.ts +9 -0
  266. package/src/modules/auth/backend/auth/profile/page.tsx +174 -0
  267. package/src/modules/auth/backend/roles/page.tsx +3 -3
  268. package/src/modules/auth/backend/users/[id]/edit/page.tsx +18 -2
  269. package/src/modules/auth/backend/users/create/page.tsx +19 -2
  270. package/src/modules/auth/backend/users/page.tsx +3 -3
  271. package/src/modules/auth/cli.ts +14 -0
  272. package/src/modules/auth/commands/users.ts +73 -2
  273. package/src/modules/auth/data/validators.ts +5 -2
  274. package/src/modules/auth/frontend/reset/[token]/page.tsx +24 -11
  275. package/src/modules/auth/i18n/de.json +43 -1
  276. package/src/modules/auth/i18n/en.json +43 -1
  277. package/src/modules/auth/i18n/es.json +43 -1
  278. package/src/modules/auth/i18n/pl.json +43 -1
  279. package/src/modules/auth/lib/setup-app.ts +1 -0
  280. package/src/modules/auth/notifications.ts +109 -0
  281. package/src/modules/auth/services/authService.ts +4 -4
  282. package/src/modules/business_rules/backend/rules/page.tsx +4 -0
  283. package/src/modules/business_rules/backend/sets/page.tsx +3 -0
  284. package/src/modules/business_rules/i18n/en.json +3 -1
  285. package/src/modules/business_rules/notifications.ts +25 -0
  286. package/src/modules/business_rules/subscribers/rule-execution-failed-notification.ts +50 -0
  287. package/src/modules/catalog/components/PriceKindSettings.tsx +2 -0
  288. package/src/modules/catalog/components/categories/CategoriesDataTable.tsx +2 -2
  289. package/src/modules/catalog/components/products/ProductsDataTable.tsx +2 -0
  290. package/src/modules/catalog/i18n/en.json +3 -1
  291. package/src/modules/catalog/notifications.ts +25 -0
  292. package/src/modules/catalog/subscribers/low-stock-notification.ts +52 -0
  293. package/src/modules/configs/cli.ts +6 -0
  294. package/src/modules/configs/lib/upgrade-actions.ts +18 -0
  295. package/src/modules/currencies/backend/currencies/page.tsx +3 -0
  296. package/src/modules/currencies/backend/exchange-rates/page.tsx +2 -0
  297. package/src/modules/customers/backend/customers/companies/page.tsx +3 -0
  298. package/src/modules/customers/backend/customers/deals/page.tsx +3 -0
  299. package/src/modules/customers/backend/customers/people/page.tsx +3 -0
  300. package/src/modules/customers/commands/deals.ts +39 -0
  301. package/src/modules/customers/components/CustomerTodosTable.tsx +1 -0
  302. package/src/modules/customers/i18n/en.json +5 -1
  303. package/src/modules/customers/notifications.ts +44 -0
  304. package/src/modules/dashboards/cli.ts +41 -1
  305. package/src/modules/dashboards/lib/role-widgets.ts +80 -0
  306. package/src/modules/dictionaries/components/DictionaryTable.tsx +2 -0
  307. package/src/modules/directory/backend/directory/organizations/page.tsx +2 -2
  308. package/src/modules/directory/backend/directory/tenants/page.tsx +2 -2
  309. package/src/modules/entities/backend/entities/user/[entityId]/records/page.tsx +2 -2
  310. package/src/modules/entities/components/SystemEntitiesTable.tsx +1 -1
  311. package/src/modules/entities/components/UserEntitiesTable.tsx +2 -2
  312. package/src/modules/feature_toggles/components/FeatureTogglesTable.tsx +3 -4
  313. package/src/modules/feature_toggles/components/OverridesTable.tsx +1 -1
  314. package/src/modules/notifications/acl.ts +7 -0
  315. package/src/modules/notifications/api/[id]/action/route.ts +75 -0
  316. package/src/modules/notifications/api/[id]/dismiss/route.ts +12 -0
  317. package/src/modules/notifications/api/[id]/read/route.ts +12 -0
  318. package/src/modules/notifications/api/[id]/restore/route.ts +53 -0
  319. package/src/modules/notifications/api/batch/route.ts +14 -0
  320. package/src/modules/notifications/api/feature/route.ts +14 -0
  321. package/src/modules/notifications/api/mark-all-read/route.ts +34 -0
  322. package/src/modules/notifications/api/openapi.ts +76 -0
  323. package/src/modules/notifications/api/role/route.ts +14 -0
  324. package/src/modules/notifications/api/route.ts +92 -0
  325. package/src/modules/notifications/api/settings/route.ts +157 -0
  326. package/src/modules/notifications/api/unread-count/route.ts +38 -0
  327. package/src/modules/notifications/backend/config/notifications/page.meta.ts +22 -0
  328. package/src/modules/notifications/backend/config/notifications/page.tsx +12 -0
  329. package/src/modules/notifications/cli.ts +18 -0
  330. package/src/modules/notifications/data/entities.ts +99 -0
  331. package/src/modules/notifications/data/validators.ts +110 -0
  332. package/src/modules/notifications/di.ts +11 -0
  333. package/src/modules/notifications/emails/NotificationEmail.tsx +98 -0
  334. package/src/modules/notifications/frontend/NotificationInboxPageClient.tsx +42 -0
  335. package/src/modules/notifications/frontend/NotificationSettingsPageClient.tsx +231 -0
  336. package/src/modules/notifications/i18n/de.json +50 -0
  337. package/src/modules/notifications/i18n/en.json +50 -0
  338. package/src/modules/notifications/i18n/es.json +50 -0
  339. package/src/modules/notifications/i18n/pl.json +50 -0
  340. package/src/modules/notifications/index.ts +12 -0
  341. package/src/modules/notifications/lib/deliveryConfig.ts +145 -0
  342. package/src/modules/notifications/lib/events.ts +48 -0
  343. package/src/modules/notifications/lib/notificationBuilder.ts +121 -0
  344. package/src/modules/notifications/lib/notificationFactory.ts +76 -0
  345. package/src/modules/notifications/lib/notificationMapper.ts +33 -0
  346. package/src/modules/notifications/lib/notificationRecipients.ts +83 -0
  347. package/src/modules/notifications/lib/notificationService.ts +414 -0
  348. package/src/modules/notifications/lib/routeHelpers.ts +151 -0
  349. package/src/modules/notifications/lib/safeHref.ts +29 -0
  350. package/src/modules/notifications/migrations/.snapshot-open-mercato.json +300 -0
  351. package/src/modules/notifications/migrations/Migration20260123000001.ts +73 -0
  352. package/src/modules/notifications/migrations/Migration20260126150000.ts +39 -0
  353. package/src/modules/notifications/subscribers/deliver-notification.ts +175 -0
  354. package/src/modules/notifications/workers/create-notification.worker.ts +122 -0
  355. package/src/modules/planner/backend/planner/availability-rulesets/page.tsx +2 -2
  356. package/src/modules/query_index/components/QueryIndexesTable.tsx +8 -2
  357. package/src/modules/resources/backend/resources/resource-types/page.tsx +2 -2
  358. package/src/modules/resources/backend/resources/resources/page.tsx +2 -2
  359. package/src/modules/sales/backend/sales/channels/offers/page.tsx +2 -0
  360. package/src/modules/sales/backend/sales/channels/page.tsx +2 -0
  361. package/src/modules/sales/commands/documents.ts +65 -0
  362. package/src/modules/sales/commands/payments.ts +33 -0
  363. package/src/modules/sales/components/AdjustmentKindSettings.tsx +2 -2
  364. package/src/modules/sales/components/PaymentMethodsSettings.tsx +2 -2
  365. package/src/modules/sales/components/ShippingMethodsSettings.tsx +2 -2
  366. package/src/modules/sales/components/TaxRatesSettings.tsx +2 -2
  367. package/src/modules/sales/components/channels/SalesChannelOffersPanel.tsx +2 -0
  368. package/src/modules/sales/components/documents/AdjustmentsSection.tsx +2 -0
  369. package/src/modules/sales/components/documents/PaymentsSection.tsx +2 -1
  370. package/src/modules/sales/components/documents/SalesDocumentsTable.tsx +2 -0
  371. package/src/modules/sales/i18n/de.json +20 -0
  372. package/src/modules/sales/i18n/en.json +25 -1
  373. package/src/modules/sales/i18n/es.json +20 -0
  374. package/src/modules/sales/i18n/pl.json +20 -0
  375. package/src/modules/sales/notifications.client.ts +65 -0
  376. package/src/modules/sales/notifications.ts +82 -0
  377. package/src/modules/sales/subscribers/quote-expiring-notification.ts +53 -0
  378. package/src/modules/sales/widgets/notifications/SalesOrderCreatedRenderer.tsx +156 -0
  379. package/src/modules/sales/widgets/notifications/SalesQuoteCreatedRenderer.tsx +156 -0
  380. package/src/modules/sales/widgets/notifications/index.ts +2 -0
  381. package/src/modules/sales/widgets/notifications/useSalesDocumentTotals.ts +81 -0
  382. package/src/modules/staff/backend/staff/team-members/page.tsx +1 -1
  383. package/src/modules/staff/backend/staff/team-roles/page.tsx +2 -2
  384. package/src/modules/staff/backend/staff/teams/[id]/edit/page.tsx +2 -2
  385. package/src/modules/staff/backend/staff/teams/page.tsx +2 -2
  386. package/src/modules/staff/commands/leave-requests.ts +94 -0
  387. package/src/modules/staff/i18n/de.json +4 -0
  388. package/src/modules/staff/i18n/en.json +9 -1
  389. package/src/modules/staff/i18n/es.json +4 -0
  390. package/src/modules/staff/i18n/pl.json +4 -0
  391. package/src/modules/staff/notifications.ts +71 -0
  392. package/src/modules/workflows/backend/definitions/page.tsx +5 -0
  393. package/src/modules/workflows/backend/instances/page.tsx +4 -1
  394. package/src/modules/workflows/backend/tasks/page.tsx +4 -1
  395. package/src/modules/workflows/i18n/en.json +3 -1
  396. package/src/modules/workflows/notifications.ts +25 -0
  397. package/src/modules/workflows/subscribers/task-assigned-notification.ts +53 -0
@@ -650,10 +650,12 @@ export default function ProductsDataTable() {
650
650
  <RowActions
651
651
  items={[
652
652
  {
653
+ id: 'edit',
653
654
  label: t('catalog.products.table.actions.edit', 'Edit'),
654
655
  href: `/backend/catalog/products/${row.id}`,
655
656
  },
656
657
  {
658
+ id: 'delete',
657
659
  label: t('catalog.products.table.actions.delete', 'Delete'),
658
660
  destructive: true,
659
661
  onSelect: () => {
@@ -681,5 +681,7 @@
681
681
  "deleteError": "Failed to delete variant."
682
682
  }
683
683
  }
684
- }
684
+ },
685
+ "catalog.notifications.product.lowStock.title": "Low Stock Alert",
686
+ "catalog.notifications.product.lowStock.body": "{productName}{sku, select, other { ({sku})}} is running low on stock ({currentStock} remaining, threshold: {threshold})"
685
687
  }
@@ -0,0 +1,25 @@
1
+ import type { NotificationTypeDefinition } from '@open-mercato/shared/modules/notifications/types'
2
+
3
+ export const notificationTypes: NotificationTypeDefinition[] = [
4
+ {
5
+ type: 'catalog.product.low_stock',
6
+ module: 'catalog',
7
+ titleKey: 'catalog.notifications.product.lowStock.title',
8
+ bodyKey: 'catalog.notifications.product.lowStock.body',
9
+ icon: 'package-x',
10
+ severity: 'warning',
11
+ actions: [
12
+ {
13
+ id: 'view',
14
+ labelKey: 'common.view',
15
+ variant: 'outline',
16
+ href: '/backend/catalog/products/{sourceEntityId}',
17
+ icon: 'external-link',
18
+ },
19
+ ],
20
+ linkHref: '/backend/catalog/products/{sourceEntityId}',
21
+ expiresAfterHours: 72, // 3 days
22
+ },
23
+ ]
24
+
25
+ export default notificationTypes
@@ -0,0 +1,52 @@
1
+ import type { EntityManager } from '@mikro-orm/postgresql'
2
+ import { resolveNotificationService } from '../../notifications/lib/notificationService'
3
+ import { buildFeatureNotificationFromType } from '../../notifications/lib/notificationBuilder'
4
+ import { notificationTypes } from '../notifications'
5
+
6
+ export const metadata = {
7
+ event: 'catalog.product.stock_low',
8
+ persistent: true,
9
+ id: 'catalog:low-stock-notification',
10
+ }
11
+
12
+ type LowStockPayload = {
13
+ productId: string
14
+ productName: string
15
+ sku?: string | null
16
+ currentStock: number
17
+ threshold: number
18
+ tenantId: string
19
+ organizationId?: string | null
20
+ }
21
+
22
+ type ResolverContext = {
23
+ resolve: <T = unknown>(name: string) => T
24
+ }
25
+
26
+ export default async function handle(payload: LowStockPayload, ctx: ResolverContext) {
27
+ try {
28
+ const notificationService = resolveNotificationService(ctx)
29
+ const typeDef = notificationTypes.find((type) => type.type === 'catalog.product.low_stock')
30
+ if (!typeDef) return
31
+
32
+ const notificationInput = buildFeatureNotificationFromType(typeDef, {
33
+ requiredFeature: 'catalog.products.manage',
34
+ bodyVariables: {
35
+ productName: payload.productName,
36
+ sku: payload.sku ?? '',
37
+ currentStock: String(payload.currentStock),
38
+ threshold: String(payload.threshold),
39
+ },
40
+ sourceEntityType: 'catalog:product',
41
+ sourceEntityId: payload.productId,
42
+ linkHref: `/backend/catalog/products/${payload.productId}`,
43
+ })
44
+
45
+ await notificationService.createForFeature(notificationInput, {
46
+ tenantId: payload.tenantId,
47
+ organizationId: payload.organizationId ?? null,
48
+ })
49
+ } catch (err) {
50
+ console.error('[catalog:low-stock-notification] Failed to create notification:', err)
51
+ }
52
+ }
@@ -2,6 +2,7 @@ import type { ModuleCli } from '@open-mercato/shared/modules/registry'
2
2
  import { createRequestContainer } from '@open-mercato/shared/lib/di/container'
3
3
  import type { ModuleConfigService } from './lib/module-config-service'
4
4
  import { parseBooleanToken } from '@open-mercato/shared/lib/boolean'
5
+ import { DEFAULT_NOTIFICATION_DELIVERY_CONFIG, NOTIFICATIONS_DELIVERY_CONFIG_KEY } from '../notifications/lib/deliveryConfig'
5
6
 
6
7
  function envDisablesAutoIndexing(): boolean {
7
8
  const raw = process.env.DISABLE_VECTOR_SEARCH_AUTOINDEXING
@@ -31,6 +32,11 @@ const restoreDefaults: ModuleCli = {
31
32
  name: 'auto_index_enabled',
32
33
  value: defaultEnabled,
33
34
  },
35
+ {
36
+ moduleId: 'notifications',
37
+ name: NOTIFICATIONS_DELIVERY_CONFIG_KEY,
38
+ value: DEFAULT_NOTIFICATION_DELIVERY_CONFIG,
39
+ },
34
40
  ],
35
41
  { force: true },
36
42
  )
@@ -11,6 +11,7 @@ import { seedExampleWorkflows } from '@open-mercato/core/modules/workflows/lib/s
11
11
  import { seedPlannerAvailabilityRuleSetDefaults, seedPlannerUnavailabilityReasons } from '@open-mercato/core/modules/planner/lib/seeds'
12
12
  import { seedResourcesAddressTypes, seedResourcesCapacityUnits, seedResourcesResourceExamples } from '@open-mercato/core/modules/resources/lib/seeds'
13
13
  import { seedStaffTeamExamples } from '@open-mercato/core/modules/staff/lib/seeds'
14
+ import { appendWidgetsToRoles, resolveAnalyticsWidgetIds } from '@open-mercato/core/modules/dashboards/lib/role-widgets'
14
15
  import { collectCrudCacheStats, purgeCrudCacheSegment } from '@open-mercato/shared/lib/crud/cache-stats'
15
16
  import { isCrudCacheEnabled, resolveCrudCache } from '@open-mercato/shared/lib/crud/cache'
16
17
  import * as semver from 'semver'
@@ -370,6 +371,23 @@ export const upgradeActions: UpgradeActionDefinition[] = [
370
371
  await reindexModules(em, ['planner', 'staff', 'resources'], { tenantId, organizationId, vectorService })
371
372
  },
372
373
  },
374
+ {
375
+ id: 'configs.upgrades.dashboards.analytics_widgets',
376
+ version: '0.4.2',
377
+ messageKey: 'upgrades.v042.message',
378
+ ctaKey: 'upgrades.v042.cta',
379
+ successKey: 'upgrades.v042.success',
380
+ loadingKey: 'upgrades.v042.loading',
381
+ async run({ em, tenantId, organizationId }) {
382
+ const widgetIds = await resolveAnalyticsWidgetIds()
383
+ await appendWidgetsToRoles(em, {
384
+ tenantId,
385
+ organizationId,
386
+ roleNames: ['admin', 'employee'],
387
+ widgetIds,
388
+ })
389
+ },
390
+ },
373
391
  ]
374
392
 
375
393
  export function actionsUpToVersion(version: string): UpgradeActionDefinition[] {
@@ -256,12 +256,14 @@ export default function CurrenciesPage() {
256
256
  <RowActions
257
257
  items={[
258
258
  {
259
+ id: 'edit',
259
260
  label: t('common.edit'),
260
261
  href: `/backend/currencies/${row.id}`,
261
262
  },
262
263
  ...(!row.isBase
263
264
  ? [
264
265
  {
266
+ id: 'set-base',
265
267
  label: t('currencies.list.actions.setBase'),
266
268
  onSelect: () => handleSetBase(row),
267
269
  },
@@ -270,6 +272,7 @@ export default function CurrenciesPage() {
270
272
  ...(!row.isBase
271
273
  ? [
272
274
  {
275
+ id: 'delete',
273
276
  label: t('common.delete'),
274
277
  destructive: true,
275
278
  onSelect: () => handleDelete(row),
@@ -279,10 +279,12 @@ export default function ExchangeRatesPage() {
279
279
  <RowActions
280
280
  items={[
281
281
  {
282
+ id: 'edit',
282
283
  label: t('common.edit'),
283
284
  href: `/backend/exchange-rates/${row.id}`,
284
285
  },
285
286
  {
287
+ id: 'delete',
286
288
  label: t('common.delete'),
287
289
  destructive: true,
288
290
  onSelect: () => handleDelete(row),
@@ -597,14 +597,17 @@ export default function CustomersCompaniesPage() {
597
597
  <RowActions
598
598
  items={[
599
599
  {
600
+ id: 'view',
600
601
  label: t('customers.companies.list.actions.view'),
601
602
  onSelect: () => { router.push(`/backend/customers/companies/${row.id}`) },
602
603
  },
603
604
  {
605
+ id: 'open-new-tab',
604
606
  label: t('customers.companies.list.actions.openInNewTab'),
605
607
  onSelect: () => window.open(`/backend/customers/companies/${row.id}`, '_blank', 'noopener'),
606
608
  },
607
609
  {
610
+ id: 'delete',
608
611
  label: t('customers.companies.list.actions.delete'),
609
612
  destructive: true,
610
613
  onSelect: () => handleDelete(row),
@@ -889,10 +889,12 @@ export default function CustomersDealsPage() {
889
889
  <RowActions
890
890
  items={[
891
891
  {
892
+ id: 'edit',
892
893
  label: t('customers.deals.list.actions.edit', 'Edit'),
893
894
  onSelect: () => { router.push(`/backend/customers/deals/${row.id}`) },
894
895
  },
895
896
  {
897
+ id: 'open-new-tab',
896
898
  label: t('customers.deals.list.actions.openInNewTab', 'Open in new tab'),
897
899
  onSelect: () => {
898
900
  if (typeof window !== 'undefined') {
@@ -901,6 +903,7 @@ export default function CustomersDealsPage() {
901
903
  },
902
904
  },
903
905
  {
906
+ id: 'delete',
904
907
  label: isDeleting
905
908
  ? t('customers.deals.list.actions.deleting', 'Deleting…')
906
909
  : t('customers.deals.list.actions.delete', 'Delete'),
@@ -614,14 +614,17 @@ export default function CustomersPeoplePage() {
614
614
  <RowActions
615
615
  items={[
616
616
  {
617
+ id: 'view',
617
618
  label: t('customers.people.list.actions.view'),
618
619
  onSelect: () => { router.push(`/backend/customers/people/${row.id}`) },
619
620
  },
620
621
  {
622
+ id: 'open-new-tab',
621
623
  label: t('customers.people.list.actions.openInNewTab'),
622
624
  onSelect: () => window.open(`/backend/customers/people/${row.id}`, '_blank', 'noopener'),
623
625
  },
624
626
  {
627
+ id: 'delete',
625
628
  label: t('customers.people.list.actions.delete'),
626
629
  destructive: true,
627
630
  onSelect: () => handleDelete(row),
@@ -35,6 +35,9 @@ import { CrudHttpError } from '@open-mercato/shared/lib/crud/errors'
35
35
  import type { CrudIndexerConfig } from '@open-mercato/shared/lib/crud/types'
36
36
  import { E } from '#generated/entities.ids.generated'
37
37
  import { findWithDecryption } from '@open-mercato/shared/lib/encryption/find'
38
+ import { resolveNotificationService } from '../../notifications/lib/notificationService'
39
+ import { buildNotificationFromType } from '../../notifications/lib/notificationBuilder'
40
+ import { notificationTypes } from '../notifications'
38
41
 
39
42
  const DEAL_ENTITY_ID = 'customers:customer_deal'
40
43
  const dealCrudIndexer: CrudIndexerConfig<CustomerDeal> = {
@@ -281,6 +284,8 @@ const updateDealCommand: CommandHandler<DealUpdateInput, { dealId: string }> = {
281
284
  ensureTenantScope(ctx, record.tenantId)
282
285
  ensureOrganizationScope(ctx, record.organizationId)
283
286
 
287
+ const previousStatus = record.status
288
+
284
289
  if (parsed.title !== undefined) record.title = parsed.title
285
290
  if (parsed.description !== undefined) record.description = parsed.description ?? null
286
291
  if (parsed.status !== undefined) record.status = parsed.status ?? record.status
@@ -319,6 +324,40 @@ const updateDealCommand: CommandHandler<DealUpdateInput, { dealId: string }> = {
319
324
  indexer: dealCrudIndexer,
320
325
  })
321
326
 
327
+ // Send notifications for deal won/lost status changes
328
+ const newStatus = record.status
329
+ const normalizedStatus = newStatus === 'win' ? 'won' : newStatus === 'loose' ? 'lost' : newStatus
330
+ if (previousStatus !== newStatus && (normalizedStatus === 'won' || normalizedStatus === 'lost') && record.ownerUserId) {
331
+ try {
332
+ const notificationService = resolveNotificationService(ctx.container)
333
+ const notificationType = normalizedStatus === 'won' ? 'customers.deal.won' : 'customers.deal.lost'
334
+ const typeDef = notificationTypes.find((type) => type.type === notificationType)
335
+ if (typeDef) {
336
+ const valueDisplay = record.valueAmount && record.valueCurrency
337
+ ? `${record.valueCurrency} ${record.valueAmount}`
338
+ : ''
339
+
340
+ const notificationInput = buildNotificationFromType(typeDef, {
341
+ recipientUserId: record.ownerUserId,
342
+ bodyVariables: {
343
+ dealTitle: record.title,
344
+ dealValue: valueDisplay,
345
+ },
346
+ sourceEntityType: 'customers:customer_deal',
347
+ sourceEntityId: record.id,
348
+ linkHref: `/backend/customers/deals/${record.id}`,
349
+ })
350
+
351
+ await notificationService.create(notificationInput, {
352
+ tenantId: record.tenantId,
353
+ organizationId: record.organizationId,
354
+ })
355
+ }
356
+ } catch {
357
+ // Notification creation is non-critical, don't fail the command
358
+ }
359
+ }
360
+
322
361
  return { dealId: record.id }
323
362
  },
324
363
  buildLog: async ({ snapshots, ctx }) => {
@@ -253,6 +253,7 @@ export function CustomerTodosTable(): React.JSX.Element {
253
253
  <RowActions
254
254
  items={[
255
255
  {
256
+ id: 'open-customer',
256
257
  label: t('customers.workPlan.customerTodos.table.actions.openCustomer'),
257
258
  href: customerLink,
258
259
  },
@@ -943,5 +943,9 @@
943
943
  "customers.workPlan.customerTodos.table.state.empty": "No customer tasks yet.",
944
944
  "customers.workPlan.customerTodos.table.error.load": "Failed to load customer tasks.",
945
945
  "customers.workPlan.customerTodos.table.export.view": "Exports the current list with filters and visible columns.",
946
- "customers.workPlan.customerTodos.table.export.full": "Exports every linked task field, including hidden attributes."
946
+ "customers.workPlan.customerTodos.table.export.full": "Exports every linked task field, including hidden attributes.",
947
+ "customers.notifications.deal.won.title": "Deal Won",
948
+ "customers.notifications.deal.won.body": "{dealTitle} has been marked as won{dealValue, select, other { ({dealValue})}}",
949
+ "customers.notifications.deal.lost.title": "Deal Lost",
950
+ "customers.notifications.deal.lost.body": "{dealTitle} has been marked as lost"
947
951
  }
@@ -0,0 +1,44 @@
1
+ import type { NotificationTypeDefinition } from '@open-mercato/shared/modules/notifications/types'
2
+
3
+ export const notificationTypes: NotificationTypeDefinition[] = [
4
+ {
5
+ type: 'customers.deal.won',
6
+ module: 'customers',
7
+ titleKey: 'customers.notifications.deal.won.title',
8
+ bodyKey: 'customers.notifications.deal.won.body',
9
+ icon: 'trophy',
10
+ severity: 'success',
11
+ actions: [
12
+ {
13
+ id: 'view',
14
+ labelKey: 'common.view',
15
+ variant: 'outline',
16
+ href: '/backend/customers/deals/{sourceEntityId}',
17
+ icon: 'external-link',
18
+ },
19
+ ],
20
+ linkHref: '/backend/customers/deals/{sourceEntityId}',
21
+ expiresAfterHours: 168, // 7 days
22
+ },
23
+ {
24
+ type: 'customers.deal.lost',
25
+ module: 'customers',
26
+ titleKey: 'customers.notifications.deal.lost.title',
27
+ bodyKey: 'customers.notifications.deal.lost.body',
28
+ icon: 'x-circle',
29
+ severity: 'warning',
30
+ actions: [
31
+ {
32
+ id: 'view',
33
+ labelKey: 'common.view',
34
+ variant: 'outline',
35
+ href: '/backend/customers/deals/{sourceEntityId}',
36
+ icon: 'external-link',
37
+ },
38
+ ],
39
+ linkHref: '/backend/customers/deals/{sourceEntityId}',
40
+ expiresAfterHours: 168, // 7 days
41
+ },
42
+ ]
43
+
44
+ export default notificationTypes
@@ -4,6 +4,7 @@ import type { EntityManager } from '@mikro-orm/postgresql'
4
4
  import { DashboardRoleWidgets } from '@open-mercato/core/modules/dashboards/data/entities'
5
5
  import { Role } from '@open-mercato/core/modules/auth/data/entities'
6
6
  import { loadAllWidgets } from '@open-mercato/core/modules/dashboards/lib/widgets'
7
+ import { appendWidgetsToRoles, resolveAnalyticsWidgetIds } from '@open-mercato/core/modules/dashboards/lib/role-widgets'
7
8
  import { seedAnalyticsData } from './seed/analytics'
8
9
 
9
10
  type Args = Record<string, string>
@@ -120,6 +121,45 @@ const seedDefaults: ModuleCli = {
120
121
  },
121
122
  }
122
123
 
124
+ const enableAnalyticsWidgets: ModuleCli = {
125
+ command: 'enable-analytics-widgets',
126
+ async run(rest) {
127
+ const args = parseArgs(rest)
128
+ const tenantId = args.tenant || args.tenantId || null
129
+ const organizationId = args.organization || args.organizationId || args.org || null
130
+ const roleCsv = args.roles || 'admin,employee'
131
+ if (!tenantId) {
132
+ console.error('Usage: mercato dashboards enable-analytics-widgets --tenant <tenantId> [--org <orgId>] [--roles admin,employee]')
133
+ return
134
+ }
135
+
136
+ const roleNames = roleCsv
137
+ .split(',')
138
+ .map((name) => name.trim())
139
+ .filter(Boolean)
140
+
141
+ if (!roleNames.length) {
142
+ console.log('No roles provided, nothing to update.')
143
+ return
144
+ }
145
+
146
+ const { resolve } = await createRequestContainer()
147
+ const em = resolve('em') as EntityManager
148
+ const widgetIds = await resolveAnalyticsWidgetIds()
149
+
150
+ const updated = await appendWidgetsToRoles(em, {
151
+ tenantId,
152
+ organizationId,
153
+ roleNames,
154
+ widgetIds,
155
+ })
156
+
157
+ if (!updated) {
158
+ console.log('No dashboard role widgets updated.')
159
+ }
160
+ },
161
+ }
162
+
123
163
  const seedAnalytics: ModuleCli = {
124
164
  command: 'seed-analytics',
125
165
  async run(rest) {
@@ -282,4 +322,4 @@ const debugAnalytics: ModuleCli = {
282
322
  },
283
323
  }
284
324
 
285
- export default [seedDefaults, seedAnalytics, debugAnalytics]
325
+ export default [seedDefaults, enableAnalyticsWidgets, seedAnalytics, debugAnalytics]
@@ -0,0 +1,80 @@
1
+ import type { EntityManager } from '@mikro-orm/postgresql'
2
+ import { Role } from '@open-mercato/core/modules/auth/data/entities'
3
+ import { DashboardRoleWidgets } from '../data/entities'
4
+ import { loadAllWidgets } from './widgets'
5
+
6
+ type RoleWidgetScope = {
7
+ tenantId: string
8
+ organizationId?: string | null
9
+ roleNames: string[]
10
+ widgetIds: string[]
11
+ }
12
+
13
+ async function findRoleByName(
14
+ em: EntityManager,
15
+ roleName: string,
16
+ tenantId: string,
17
+ ): Promise<Role | null> {
18
+ const tenantRole = await em.findOne(Role, { name: roleName, tenantId })
19
+ if (tenantRole) return tenantRole
20
+ return em.findOne(Role, { name: roleName, tenantId: null })
21
+ }
22
+
23
+ export async function resolveAnalyticsWidgetIds(): Promise<string[]> {
24
+ const widgets = await loadAllWidgets()
25
+ return widgets
26
+ .filter((widget) => widget.metadata.category === 'analytics' || widget.metadata.id.startsWith('dashboards.analytics.'))
27
+ .map((widget) => widget.metadata.id)
28
+ }
29
+
30
+ export async function appendWidgetsToRoles(
31
+ em: EntityManager,
32
+ { tenantId, organizationId = null, roleNames, widgetIds }: RoleWidgetScope,
33
+ ): Promise<boolean> {
34
+ const trimmedTenantId = tenantId.trim()
35
+ const widgets = await loadAllWidgets()
36
+ const validWidgetIds = new Set(widgets.map((widget) => widget.metadata.id))
37
+ const resolvedWidgetIds = widgetIds.filter((id) => validWidgetIds.has(id))
38
+ if (!resolvedWidgetIds.length) return false
39
+
40
+ let updated = false
41
+ await em.transactional(async (tem) => {
42
+ for (const roleName of roleNames) {
43
+ const role = await findRoleByName(tem, roleName, trimmedTenantId)
44
+ if (!role) continue
45
+
46
+ const record = await tem.findOne(DashboardRoleWidgets, {
47
+ roleId: String(role.id),
48
+ tenantId: trimmedTenantId,
49
+ organizationId,
50
+ deletedAt: null,
51
+ })
52
+ const roleRecord = record ?? (organizationId
53
+ ? await tem.findOne(DashboardRoleWidgets, {
54
+ roleId: String(role.id),
55
+ tenantId: trimmedTenantId,
56
+ organizationId: null,
57
+ deletedAt: null,
58
+ })
59
+ : null)
60
+ if (!roleRecord) continue
61
+
62
+ const current = Array.isArray(roleRecord.widgetIdsJson) ? roleRecord.widgetIdsJson : []
63
+ const next = [...current]
64
+ const existing = new Set(current)
65
+ for (const widgetId of resolvedWidgetIds) {
66
+ if (existing.has(widgetId)) continue
67
+ existing.add(widgetId)
68
+ next.push(widgetId)
69
+ }
70
+
71
+ if (next.length === current.length) continue
72
+ roleRecord.widgetIdsJson = next
73
+ roleRecord.updatedAt = new Date()
74
+ tem.persist(roleRecord)
75
+ updated = true
76
+ }
77
+ })
78
+
79
+ return updated
80
+ }
@@ -171,12 +171,14 @@ export function DictionaryTable({
171
171
  const items: RowActionItem[] = []
172
172
  if (onEdit) {
173
173
  items.push({
174
+ id: 'edit',
174
175
  label: translations.editLabel,
175
176
  onSelect: () => onEdit(entry),
176
177
  })
177
178
  }
178
179
  if (onDelete) {
179
180
  items.push({
181
+ id: 'delete',
180
182
  label: translations.deleteLabel,
181
183
  onSelect: () => onDelete(entry),
182
184
  destructive: true,
@@ -235,8 +235,8 @@ export default function DirectoryOrganizationsPage() {
235
235
  canManage ? (
236
236
  <RowActions
237
237
  items={[
238
- { label: t('directory.organizations.list.actions.edit', 'Edit'), href: `/backend/directory/organizations/${row.id}/edit` },
239
- { label: t('directory.organizations.list.actions.delete', 'Delete'), destructive: true, onSelect: () => handleDelete(row) },
238
+ { id: 'edit', label: t('directory.organizations.list.actions.edit', 'Edit'), href: `/backend/directory/organizations/${row.id}/edit` },
239
+ { id: 'delete', label: t('directory.organizations.list.actions.delete', 'Delete'), destructive: true, onSelect: () => handleDelete(row) },
240
240
  ]}
241
241
  />
242
242
  ) : null
@@ -160,8 +160,8 @@ export default function DirectoryTenantsPage() {
160
160
  canManage ? (
161
161
  <RowActions
162
162
  items={[
163
- { label: t('common.edit', 'Edit'), href: `/backend/directory/tenants/${row.id}/edit` },
164
- { label: t('common.delete', 'Delete'), destructive: true, onSelect: () => handleDelete(row) },
163
+ { id: 'edit', label: t('common.edit', 'Edit'), href: `/backend/directory/tenants/${row.id}/edit` },
164
+ { id: 'delete', label: t('common.delete', 'Delete'), destructive: true, onSelect: () => handleDelete(row) },
165
165
  ]}
166
166
  />
167
167
  ) : null
@@ -338,8 +338,8 @@ export RECORD_ID="<record uuid>"`}</code></pre>
338
338
  rowActions={(row) => (
339
339
  <RowActions
340
340
  items={[
341
- { label: 'Edit', href: `/backend/entities/user/${encodeURIComponent(entityId)}/records/${encodeURIComponent(String((row as any).id))}` },
342
- { label: 'Delete', destructive: true, onSelect: async () => {
341
+ { id: 'edit', label: 'Edit', href: `/backend/entities/user/${encodeURIComponent(entityId)}/records/${encodeURIComponent(String((row as any).id))}` },
342
+ { id: 'delete', label: 'Delete', destructive: true, onSelect: async () => {
343
343
  try {
344
344
  if (typeof window !== 'undefined') {
345
345
  const ok = window.confirm('Delete this record?')
@@ -88,7 +88,7 @@ export default function SystemEntitiesTable() {
88
88
  rowActions={(row) => (
89
89
  <RowActions
90
90
  items={[
91
- { label: 'Edit', href: `/backend/entities/system/${encodeURIComponent(row.entityId)}` },
91
+ { id: 'edit', label: 'Edit', href: `/backend/entities/system/${encodeURIComponent(row.entityId)}` },
92
92
  ]}
93
93
  />
94
94
  )}
@@ -110,8 +110,8 @@ export default function UserEntitiesTable() {
110
110
  rowActions={(row) => (
111
111
  <RowActions
112
112
  items={[
113
- { label: t('common.edit', 'Edit'), href: `/backend/entities/user/${encodeURIComponent(row.entityId)}` },
114
- { label: t('entities.user.table.actions.showRecords', 'Show records'), href: `/backend/entities/user/${encodeURIComponent(row.entityId)}/records` },
113
+ { id: 'edit', label: t('common.edit', 'Edit'), href: `/backend/entities/user/${encodeURIComponent(row.entityId)}` },
114
+ { id: 'show-records', label: t('entities.user.table.actions.showRecords', 'Show records'), href: `/backend/entities/user/${encodeURIComponent(row.entityId)}/records` },
115
115
  ]}
116
116
  />
117
117
  )}
@@ -205,12 +205,11 @@ export function FeatureTogglesTable() {
205
205
  }}
206
206
  rowActions={(row) => (
207
207
  <RowActions items={[
208
- { label: t('common.edit', 'Edit'), href: `/backend/feature-toggles/global/${row.id}/edit` },
209
- { label: t('common.view', 'Overrides'), href: `/backend/feature-toggles/global/${row.id}` },
210
- { label: t('common.delete', 'Delete'), destructive: true, onSelect: () => { void handleDelete(row) } },
208
+ { id: 'edit', label: t('common.edit', 'Edit'), href: `/backend/feature-toggles/global/${row.id}/edit` },
209
+ { id: 'view', label: t('common.view', 'Overrides'), href: `/backend/feature-toggles/global/${row.id}` },
210
+ { id: 'delete', label: t('common.delete', 'Delete'), destructive: true, onSelect: () => { void handleDelete(row) } },
211
211
  ]} />
212
212
  )}
213
213
  />
214
214
  )
215
215
  }
216
-
@@ -161,7 +161,7 @@ export default function OverridesTable() {
161
161
  }}
162
162
  rowActions={(row) => (
163
163
  <RowActions items={[
164
- { label: t('common.edit', 'Edit'), href: `/backend/feature-toggles/global/${row.toggleId}` },
164
+ { id: 'edit', label: t('common.edit', 'Edit'), href: `/backend/feature-toggles/global/${row.toggleId}` },
165
165
  ]} />
166
166
  )}
167
167
  error={error ? error.message : undefined}
@@ -0,0 +1,7 @@
1
+ export const features = [
2
+ { id: 'notifications.view', title: 'View own notifications', module: 'notifications' },
3
+ { id: 'notifications.create', title: 'Create notifications for others', module: 'notifications' },
4
+ { id: 'notifications.manage', title: 'Manage all notifications', module: 'notifications' },
5
+ ]
6
+
7
+ export default features