@goplusvn/core 0.1.0 → 0.1.1

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 (369) hide show
  1. package/package.json +2 -1
  2. package/src/assets/erp_wallpaper.png +0 -0
  3. package/src/assets/goeat_logo.png +0 -0
  4. package/src/audit/audit-manager.ts +139 -0
  5. package/src/audit/index.ts +11 -0
  6. package/src/audit/memory-audit-logger.ts +86 -0
  7. package/src/audit/types.ts +50 -0
  8. package/src/auth/auth-service.ts +97 -0
  9. package/src/auth/index.ts +266 -0
  10. package/src/code-generation/index.ts +69 -0
  11. package/src/configs/auth-routes.ts +17 -0
  12. package/src/configs/crud.ts +136 -0
  13. package/src/configs/data/navigations.ts +781 -0
  14. package/src/configs/data/oauth-links.ts +10 -0
  15. package/src/configs/entities/material-categories.config.ts +125 -0
  16. package/src/configs/i18n.ts +12 -0
  17. package/src/configs/index.ts +26 -0
  18. package/src/configs/status.ts +25 -0
  19. package/src/configs/themes.ts +100 -0
  20. package/src/crud/components/crud-bulk-actions.tsx +91 -0
  21. package/src/crud/components/crud-card-view.tsx +241 -0
  22. package/src/crud/components/crud-context.tsx +122 -0
  23. package/src/crud/components/crud-delete-dialog.tsx +145 -0
  24. package/src/crud/components/crud-dialog.tsx +406 -0
  25. package/src/crud/components/crud-empty-state.tsx +104 -0
  26. package/src/crud/components/crud-export-button.tsx +170 -0
  27. package/src/crud/components/crud-field-renderer.tsx +653 -0
  28. package/src/crud/components/crud-filter-chips.tsx +102 -0
  29. package/src/crud/components/crud-filters/checkbox-filter.tsx +97 -0
  30. package/src/crud/components/crud-filters/datetime-filter.tsx +83 -0
  31. package/src/crud/components/crud-filters/filter-builder.tsx +66 -0
  32. package/src/crud/components/crud-filters/index.tsx +76 -0
  33. package/src/crud/components/crud-filters/radio-filter.tsx +86 -0
  34. package/src/crud/components/crud-filters/select-filter.tsx +141 -0
  35. package/src/crud/components/crud-filters/text-filter.tsx +86 -0
  36. package/src/crud/components/crud-form.tsx +642 -0
  37. package/src/crud/components/crud-import-dialog.tsx +440 -0
  38. package/src/crud/components/crud-infinite-scroll.tsx +116 -0
  39. package/src/crud/components/crud-page.tsx +1017 -0
  40. package/src/crud/components/crud-provider.tsx +277 -0
  41. package/src/crud/components/crud-row-actions.tsx +189 -0
  42. package/src/crud/components/crud-search.tsx +82 -0
  43. package/src/crud/components/crud-sheet.tsx +336 -0
  44. package/src/crud/components/crud-table-skeleton.tsx +26 -0
  45. package/src/crud/components/crud-table-toolbar.tsx +91 -0
  46. package/src/crud/components/crud-table.tsx +352 -0
  47. package/src/crud/components/crud-virtual-table.tsx +55 -0
  48. package/src/crud/components/index.tsx +20 -0
  49. package/src/crud/crud-filters/checkbox-filter.tsx +87 -0
  50. package/src/crud/crud-filters/datetime-filter.tsx +82 -0
  51. package/src/crud/crud-filters/filter-builder.tsx +64 -0
  52. package/src/crud/crud-filters/index.tsx +78 -0
  53. package/src/crud/crud-filters/radio-filter.tsx +79 -0
  54. package/src/crud/crud-filters/select-filter.tsx +148 -0
  55. package/src/crud/crud-filters/text-filter.tsx +81 -0
  56. package/src/crud/index.ts +43 -0
  57. package/src/crud/lib/crud-service.test.ts +334 -0
  58. package/src/crud/lib/crud-service.ts +358 -0
  59. package/src/crud/lib/crud-utils.test.ts +354 -0
  60. package/src/crud/lib/crud-utils.ts +299 -0
  61. package/src/crud/lib/crud-validator.ts +247 -0
  62. package/src/crud/lib/data-loader.ts +234 -0
  63. package/src/crud/lib/field-calculator.ts +241 -0
  64. package/src/crud/lib/field-formatter.ts +240 -0
  65. package/src/crud/lib/import-export-service.test.ts +290 -0
  66. package/src/crud/lib/import-export-service.ts +352 -0
  67. package/src/crud/lib/import-server-utils.ts +109 -0
  68. package/src/crud/lib/lazy-loader.ts +241 -0
  69. package/src/crud/lib/parse-filters.ts +85 -0
  70. package/src/crud/lib/permissions.ts +52 -0
  71. package/src/crud/lib/serialize-config.ts +60 -0
  72. package/src/crud/lib/stream-loader.ts +145 -0
  73. package/src/crud/lib/translate-config.ts +335 -0
  74. package/src/crud/lib/types.ts +11 -0
  75. package/src/crud/pages/entity-crud-page.tsx +144 -0
  76. package/src/crud/server.ts +8 -0
  77. package/src/home/constants.tsx +142 -0
  78. package/src/home/feature-showcase.tsx +171 -0
  79. package/src/home/home-page.tsx +191 -0
  80. package/src/home/hooks/index.ts +1 -0
  81. package/src/home/hooks/useWidgetPreferences.ts +167 -0
  82. package/src/home/index.ts +33 -0
  83. package/src/home/quick-access-dialog.tsx +271 -0
  84. package/src/home/quick-access-menu.tsx +267 -0
  85. package/src/home/types.ts +140 -0
  86. package/src/home/welcome-card.tsx +92 -0
  87. package/src/home/widget-container.tsx +258 -0
  88. package/src/home/widgets/base-widget.tsx +200 -0
  89. package/src/home/widgets/customers-widget.tsx +74 -0
  90. package/src/home/widgets/index.ts +6 -0
  91. package/src/home/widgets/orders-widget.tsx +87 -0
  92. package/src/home/widgets/revenue-widget.tsx +71 -0
  93. package/src/home/widgets/stock-widget.tsx +109 -0
  94. package/src/hooks/index.tsx +598 -0
  95. package/src/hooks/use-tenant.test.tsx +30 -0
  96. package/src/hooks/use-tenant.ts +5 -0
  97. package/src/index.ts +17 -0
  98. package/src/infrastructure/__tests__/architecture-verification.spec.ts +103 -0
  99. package/src/infrastructure/api-service.ts +317 -0
  100. package/src/infrastructure/cache/cache-manager.ts +107 -0
  101. package/src/infrastructure/cache/cache.ts +120 -0
  102. package/src/infrastructure/cache/index.ts +8 -0
  103. package/src/infrastructure/cache/types.ts +48 -0
  104. package/src/infrastructure/cron/cron-manager.ts +239 -0
  105. package/src/infrastructure/cron/index.ts +6 -0
  106. package/src/infrastructure/cron/types.ts +41 -0
  107. package/src/infrastructure/event-bus/event-bus.ts +145 -0
  108. package/src/infrastructure/event-bus/index.ts +2 -0
  109. package/src/infrastructure/event-bus/types.ts +22 -0
  110. package/src/infrastructure/index.ts +32 -0
  111. package/src/infrastructure/lock/decorators.ts +67 -0
  112. package/src/infrastructure/lock/index.ts +2 -0
  113. package/src/infrastructure/lock/lock-manager.ts +33 -0
  114. package/src/infrastructure/logger/index.ts +2 -0
  115. package/src/infrastructure/logger/logger.ts +96 -0
  116. package/src/infrastructure/logger/types.ts +25 -0
  117. package/src/layout/index.tsx +185 -0
  118. package/src/navigation/index.ts +91 -0
  119. package/src/notification/index.ts +14 -0
  120. package/src/notification/notification-service.ts +120 -0
  121. package/src/notification/storage/in-memory.ts +56 -0
  122. package/src/notification/storage/index.ts +1 -0
  123. package/src/notification/types.ts +51 -0
  124. package/src/organization/branch-service.ts +299 -0
  125. package/src/organization/branches.config.ts +154 -0
  126. package/src/organization/index.ts +5 -0
  127. package/src/plugin/apps-registry.ts +97 -0
  128. package/src/plugin/index.ts +5 -0
  129. package/src/plugin/types.ts +41 -0
  130. package/src/providers/index.tsx +109 -0
  131. package/src/providers/tenant-provider.tsx +45 -0
  132. package/src/rbac/components/roles/role-card.tsx +158 -0
  133. package/src/rbac/components/roles/role-stats-cards.tsx +29 -0
  134. package/src/rbac/components/roles/role-toolbar.tsx +123 -0
  135. package/src/rbac/hooks/use-role-operations.ts +159 -0
  136. package/src/rbac/hooks/use-roles-data.ts +59 -0
  137. package/src/rbac/index.ts +297 -0
  138. package/src/rbac/lib/permission-helpers.ts +63 -0
  139. package/src/rbac/pages/action-list-page.tsx +25 -0
  140. package/src/rbac/pages/resource-list-page.tsx +25 -0
  141. package/src/rbac/pages/role-list-page.tsx +378 -0
  142. package/src/rbac/permission-service.ts +140 -0
  143. package/src/rbac/permissions.ts +135 -0
  144. package/src/rbac/resource-service.ts +115 -0
  145. package/src/rbac/resource-validator.ts +119 -0
  146. package/src/rbac/role-service.ts +165 -0
  147. package/src/rbac/server.ts +16 -0
  148. package/src/rbac/types.ts +38 -0
  149. package/src/schemas/action.schema.ts +66 -0
  150. package/src/schemas/branch.schema.ts +52 -0
  151. package/src/schemas/coming-soon-schema.ts +9 -0
  152. package/src/schemas/company.schema.ts +44 -0
  153. package/src/schemas/forgot-passward-schema.ts +9 -0
  154. package/src/schemas/index.ts +30 -0
  155. package/src/schemas/material-category.schema.ts +43 -0
  156. package/src/schemas/material-pricing.schema.ts +74 -0
  157. package/src/schemas/material.schema.ts +76 -0
  158. package/src/schemas/materials.ts +52 -0
  159. package/src/schemas/new-passward-schema.ts +15 -0
  160. package/src/schemas/partner-company.schema.ts +149 -0
  161. package/src/schemas/register-schema.ts +36 -0
  162. package/src/schemas/resource.schema.ts +133 -0
  163. package/src/schemas/role.schema.ts +11 -0
  164. package/src/schemas/sign-in-schema.ts +24 -0
  165. package/src/schemas/supplier-pricing.schema.ts +15 -0
  166. package/src/schemas/supplier.schema.ts +120 -0
  167. package/src/schemas/system-category-group.schema.ts +67 -0
  168. package/src/schemas/system-category.schema.ts +77 -0
  169. package/src/schemas/system-config.schema.ts +118 -0
  170. package/src/schemas/uom.schema.ts +75 -0
  171. package/src/schemas/user-supplier.schema.ts +179 -0
  172. package/src/schemas/user.schema.ts +18 -0
  173. package/src/schemas/verify-email-schema.ts +9 -0
  174. package/src/schemas/warehouse.schema.ts +49 -0
  175. package/src/system/components/categories/category-list.tsx +529 -0
  176. package/src/system/components/categories/category-manager.tsx +89 -0
  177. package/src/system/components/categories/group-sidebar.tsx +308 -0
  178. package/src/system/components/settings/setting-dialogs.tsx +197 -0
  179. package/src/system/components/settings/setting-field.tsx +291 -0
  180. package/src/system/components/settings/setting-form-dialog.tsx +308 -0
  181. package/src/system/components/settings/settings-groups.ts +80 -0
  182. package/src/system/components/settings/settings-search.tsx +71 -0
  183. package/src/system/components/settings/settings-section.tsx +74 -0
  184. package/src/system/components/settings/settings-sidebar.tsx +81 -0
  185. package/src/system/constants.ts +3 -0
  186. package/src/system/index.ts +150 -0
  187. package/src/system/job-manager.ts +176 -0
  188. package/src/system/pages/components/categories/category-list.tsx +537 -0
  189. package/src/system/pages/components/categories/category-manager.tsx +90 -0
  190. package/src/system/pages/components/categories/group-sidebar.tsx +311 -0
  191. package/src/system/pages/components/settings/sales-rules-settings.tsx +222 -0
  192. package/src/system/pages/components/settings/setting-dialogs.tsx +197 -0
  193. package/src/system/pages/components/settings/setting-field.tsx +292 -0
  194. package/src/system/pages/components/settings/setting-form-dialog.tsx +308 -0
  195. package/src/system/pages/components/settings/settings-groups.ts +87 -0
  196. package/src/system/pages/components/settings/settings-page.tsx +372 -0
  197. package/src/system/pages/components/settings/settings-search.tsx +71 -0
  198. package/src/system/pages/components/settings/settings-section.tsx +74 -0
  199. package/src/system/pages/components/settings/settings-sidebar.tsx +81 -0
  200. package/src/system/pages/components/settings/system-settings.tsx +244 -0
  201. package/src/system/pages/system-category-page.tsx +15 -0
  202. package/src/system/pages/system-settings-page.tsx +380 -0
  203. package/src/system/schemas/system-category-group.schema.ts +46 -0
  204. package/src/system/schemas/system-category.schema.ts +56 -0
  205. package/src/system/services/settings-service.ts +127 -0
  206. package/src/system/services/system-category-service.ts +63 -0
  207. package/src/system/types.ts +45 -0
  208. package/src/types/index.ts +703 -0
  209. package/src/ui/auth/auth-layout.tsx +135 -0
  210. package/src/ui/auth/forgot-password-form.tsx +98 -0
  211. package/src/ui/auth/index.tsx +7 -0
  212. package/src/ui/auth/new-password-form.tsx +107 -0
  213. package/src/ui/auth/oauth-links.tsx +30 -0
  214. package/src/ui/auth/register-form.tsx +202 -0
  215. package/src/ui/auth/sign-in-form.tsx +238 -0
  216. package/src/ui/auth/verify-email-form.tsx +104 -0
  217. package/src/ui/crud/index.tsx +10 -0
  218. package/src/ui/data-display/accordion.tsx +65 -0
  219. package/src/ui/data-display/aspect-ratio.tsx +11 -0
  220. package/src/ui/data-display/avatar.tsx +163 -0
  221. package/src/ui/data-display/bento-grid.tsx +77 -0
  222. package/src/ui/data-display/carousel.tsx +249 -0
  223. package/src/ui/data-display/chart.tsx +363 -0
  224. package/src/ui/data-display/code-block-highlight.tsx +54 -0
  225. package/src/ui/data-display/collapsible.tsx +42 -0
  226. package/src/ui/data-display/compact-stat-bar.tsx +149 -0
  227. package/src/ui/data-display/data-table/data-table-context.tsx +255 -0
  228. package/src/ui/data-display/data-table/data-table-empty-state.tsx +133 -0
  229. package/src/ui/data-display/data-table/data-table-skeleton.tsx +145 -0
  230. package/src/ui/data-display/data-table/data-table-toolbar.tsx +353 -0
  231. package/src/ui/data-display/data-table/data-table.tsx +597 -0
  232. package/src/ui/data-display/data-table/index.ts +44 -0
  233. package/src/ui/data-display/data-table-column-header.tsx +75 -0
  234. package/src/ui/data-display/data-table-pagination.tsx +130 -0
  235. package/src/ui/data-display/data-table-view-options.tsx +59 -0
  236. package/src/ui/data-display/formatted-number-input.tsx +210 -0
  237. package/src/ui/data-display/highlight.tsx +20 -0
  238. package/src/ui/data-display/hover-card.tsx +48 -0
  239. package/src/ui/data-display/index.tsx +50 -0
  240. package/src/ui/data-display/iphone-15-pro.tsx +114 -0
  241. package/src/ui/data-display/kanban/index.ts +4 -0
  242. package/src/ui/data-display/kanban/kanban-board.tsx +192 -0
  243. package/src/ui/data-display/kanban/kanban-column.tsx +74 -0
  244. package/src/ui/data-display/kanban/kanban-item.tsx +50 -0
  245. package/src/ui/data-display/kanban/kanban-types.ts +21 -0
  246. package/src/ui/data-display/kpi-card.tsx +68 -0
  247. package/src/ui/data-display/media-grid.tsx +110 -0
  248. package/src/ui/data-display/safari.tsx +175 -0
  249. package/src/ui/data-display/show-more-text.tsx +55 -0
  250. package/src/ui/data-display/tabs.tsx +68 -0
  251. package/src/ui/data-display/timeline.tsx +256 -0
  252. package/src/ui/feedback/alert.tsx +60 -0
  253. package/src/ui/feedback/context-menu.tsx +245 -0
  254. package/src/ui/feedback/drawer.tsx +132 -0
  255. package/src/ui/feedback/error-dialog.tsx +273 -0
  256. package/src/ui/feedback/index.tsx +183 -0
  257. package/src/ui/feedback/progress.tsx +32 -0
  258. package/src/ui/feedback/sheet.tsx +148 -0
  259. package/src/ui/feedback/sonner.tsx +36 -0
  260. package/src/ui/forms/command.tsx +157 -0
  261. package/src/ui/forms/date-picker.tsx +73 -0
  262. package/src/ui/forms/date-range-picker.tsx +76 -0
  263. package/src/ui/forms/date-time-picker.tsx +109 -0
  264. package/src/ui/forms/editor/editor-menu-bar.tsx +394 -0
  265. package/src/ui/forms/editor/index.tsx +130 -0
  266. package/src/ui/forms/editor/multi-select-example.tsx +1234 -0
  267. package/src/ui/forms/emoji-picker.tsx +109 -0
  268. package/src/ui/forms/file-dropzone.tsx +169 -0
  269. package/src/ui/forms/file-thumbnail.tsx +29 -0
  270. package/src/ui/forms/index.tsx +201 -0
  271. package/src/ui/forms/input-file.tsx +99 -0
  272. package/src/ui/forms/input-group.tsx +46 -0
  273. package/src/ui/forms/input-otp.tsx +81 -0
  274. package/src/ui/forms/input-phone.tsx +172 -0
  275. package/src/ui/forms/input-spin.tsx +116 -0
  276. package/src/ui/forms/input-tags.tsx +219 -0
  277. package/src/ui/forms/input-time.tsx +42 -0
  278. package/src/ui/forms/multi-select.tsx +629 -0
  279. package/src/ui/forms/multiple-date-picker.tsx +74 -0
  280. package/src/ui/forms/radio-group.tsx +42 -0
  281. package/src/ui/forms/rating.tsx +158 -0
  282. package/src/ui/forms/time-picker.tsx +57 -0
  283. package/src/ui/index.tsx +17 -0
  284. package/src/ui/layout/animated-list.tsx +77 -0
  285. package/src/ui/layout/animated-sidebar.tsx +294 -0
  286. package/src/ui/layout/command-menu.tsx +355 -0
  287. package/src/ui/layout/customizer.tsx +324 -0
  288. package/src/ui/layout/footer.tsx +43 -0
  289. package/src/ui/layout/full-screen-toggle.tsx +52 -0
  290. package/src/ui/layout/header-breadcrumb.tsx +77 -0
  291. package/src/ui/layout/horizontal-layout-header.tsx +83 -0
  292. package/src/ui/layout/horizontal-layout.tsx +50 -0
  293. package/src/ui/layout/index.tsx +25 -0
  294. package/src/ui/layout/language-dropdown.tsx +103 -0
  295. package/src/ui/layout/logo.tsx +63 -0
  296. package/src/ui/layout/main-layout.tsx +57 -0
  297. package/src/ui/layout/mode-dropdown.tsx +58 -0
  298. package/src/ui/layout/notification-dropdown.tsx +127 -0
  299. package/src/ui/layout/page-tabs.tsx +306 -0
  300. package/src/ui/layout/route-cache.tsx +214 -0
  301. package/src/ui/layout/sidebar-group-icon-menu.tsx +195 -0
  302. package/src/ui/layout/sidebar.tsx +279 -0
  303. package/src/ui/layout/tab-content-cache.tsx +201 -0
  304. package/src/ui/layout/tab-navigation-provider.tsx +536 -0
  305. package/src/ui/layout/toggle-mobile-sidebar.tsx +33 -0
  306. package/src/ui/layout/top-bar-header-menubar.tsx +412 -0
  307. package/src/ui/layout/user-dropdown.tsx +188 -0
  308. package/src/ui/layout/vertical-layout-header.tsx +65 -0
  309. package/src/ui/layout/vertical-layout.tsx +47 -0
  310. package/src/ui/management/audit-log-page.tsx +209 -0
  311. package/src/ui/management/cache-management.tsx +349 -0
  312. package/src/ui/management/index.ts +3 -0
  313. package/src/ui/management/job-management.tsx +308 -0
  314. package/src/ui/pages/not-found.tsx +30 -0
  315. package/src/ui/primitives/badge.tsx +66 -0
  316. package/src/ui/primitives/breadcrumb.tsx +103 -0
  317. package/src/ui/primitives/button.tsx +129 -0
  318. package/src/ui/primitives/calendar.tsx +74 -0
  319. package/src/ui/primitives/card.tsx +86 -0
  320. package/src/ui/primitives/checkbox.tsx +31 -0
  321. package/src/ui/primitives/client.ts +30 -0
  322. package/src/ui/primitives/combobox.tsx +290 -0
  323. package/src/ui/primitives/dialog.tsx +121 -0
  324. package/src/ui/primitives/dropdown-menu.tsx +239 -0
  325. package/src/ui/primitives/dynamic-icon.tsx +24 -0
  326. package/src/ui/primitives/index.tsx +134 -0
  327. package/src/ui/primitives/input-number.tsx +131 -0
  328. package/src/ui/primitives/input.tsx +22 -0
  329. package/src/ui/primitives/keyboard.tsx +23 -0
  330. package/src/ui/primitives/label.tsx +24 -0
  331. package/src/ui/primitives/menubar.tsx +262 -0
  332. package/src/ui/primitives/navigation-menu.tsx +157 -0
  333. package/src/ui/primitives/pagination.tsx +118 -0
  334. package/src/ui/primitives/popover.tsx +56 -0
  335. package/src/ui/primitives/prefetch-link.tsx +60 -0
  336. package/src/ui/primitives/resizable.tsx +59 -0
  337. package/src/ui/primitives/scroll-area.tsx +63 -0
  338. package/src/ui/primitives/select.tsx +172 -0
  339. package/src/ui/primitives/separator.tsx +51 -0
  340. package/src/ui/primitives/sidebar.tsx +844 -0
  341. package/src/ui/primitives/slider.tsx +27 -0
  342. package/src/ui/primitives/status-badge.tsx +47 -0
  343. package/src/ui/primitives/sticky-layout.tsx +50 -0
  344. package/src/ui/primitives/switch.tsx +29 -0
  345. package/src/ui/primitives/table.tsx +116 -0
  346. package/src/ui/primitives/tabs.tsx +55 -0
  347. package/src/ui/primitives/toggle-group.tsx +70 -0
  348. package/src/ui/primitives/toggle.tsx +47 -0
  349. package/src/ui/primitives/tooltip.tsx +59 -0
  350. package/src/user/components/dangerous-zone.tsx +34 -0
  351. package/src/user/components/delete-account-form.tsx +40 -0
  352. package/src/user/components/index.ts +4 -0
  353. package/src/user/components/profile-info-form.tsx +390 -0
  354. package/src/user/components/profile-info.tsx +32 -0
  355. package/src/user/components/unified-profile-dialog.tsx +1019 -0
  356. package/src/user/components/user-stats.tsx +27 -0
  357. package/src/user/components/user-toolbar.tsx +137 -0
  358. package/src/user/components/users-card-view.tsx +253 -0
  359. package/src/user/index.ts +11 -0
  360. package/src/user/pages/user-list-page.tsx +234 -0
  361. package/src/user/pages/users-client-page.tsx +385 -0
  362. package/src/user/profile-page.tsx +19 -0
  363. package/src/user/schemas.ts +68 -0
  364. package/src/user/types.ts +34 -0
  365. package/src/user/user-service.ts +538 -0
  366. package/src/utils/index.ts +906 -0
  367. package/src/workflow/activity-timeline.tsx +412 -0
  368. package/src/workflow/approval-workflow.tsx +31 -0
  369. package/src/workflow/index.ts +2 -0
@@ -0,0 +1,378 @@
1
+ "use client";
2
+
3
+ import { useMemo, useCallback, useState } from "react";
4
+ import { Loader2, UserCheck, UserX, KeyRound } from "lucide-react";
5
+ import {
6
+ Button,
7
+ Badge,
8
+ AlertDialog,
9
+ AlertDialogAction,
10
+ AlertDialogCancel,
11
+ AlertDialogContent,
12
+ AlertDialogDescription,
13
+ AlertDialogFooter,
14
+ AlertDialogHeader,
15
+ AlertDialogTitle,
16
+ Dialog,
17
+ DialogContent,
18
+ DialogDescription,
19
+ DialogHeader,
20
+ DialogTitle,
21
+ DialogFooter,
22
+ Avatar,
23
+ AvatarFallback,
24
+ AvatarImage
25
+ } from "../../ui";
26
+
27
+ import type { CrudPermissions } from "../../types";
28
+
29
+ import { RoleStatsCards } from "../components/roles/role-stats-cards";
30
+ import { RoleToolbar } from "../components/roles/role-toolbar";
31
+ import { RoleCard } from "../components/roles/role-card";
32
+ import { useRolesData } from "../hooks/use-roles-data";
33
+ import { useRoleOperations } from "../hooks/use-role-operations";
34
+ import {
35
+ getUsersWithPermission,
36
+ getPermissionDescription,
37
+ } from "../lib/permission-helpers";
38
+ import type { Role, RolesResponse } from "../types";
39
+
40
+ export interface RoleListPageProps {
41
+ dictionary: any;
42
+ permissions: CrudPermissions;
43
+ lang: string;
44
+ initialData: RolesResponse;
45
+ resources: Array<{
46
+ id: string;
47
+ code: string;
48
+ name: string;
49
+ group: string | null;
50
+ description: string | null;
51
+ icon: string | null;
52
+ order: number | null;
53
+ config: any;
54
+ }>;
55
+ actions: Array<{
56
+ id: string;
57
+ code: string;
58
+ name: string;
59
+ description: string | null;
60
+ isDefault?: boolean;
61
+ }>;
62
+ }
63
+
64
+ export function RoleListPage({
65
+ dictionary,
66
+ permissions,
67
+ lang,
68
+ initialData,
69
+ resources,
70
+ actions,
71
+ }: RoleListPageProps) {
72
+ const canCreate = permissions.create;
73
+ const canUpdate = permissions.update;
74
+ const canDelete = permissions.delete;
75
+ const canImport = permissions.import;
76
+ const canExport = permissions.export;
77
+
78
+ // Data fetching
79
+ const {
80
+ roles,
81
+ totalRows,
82
+ activeRolesCount,
83
+ searchTerm,
84
+ statusFilter,
85
+ setSearchTerm,
86
+ setStatusFilter,
87
+ error,
88
+ isLoading,
89
+ mutate,
90
+ } = useRolesData(initialData);
91
+
92
+ // Role operations
93
+ const {
94
+ handleDeleteRole,
95
+ handleDuplicateRole,
96
+ } = useRoleOperations(mutate);
97
+
98
+ // UI State
99
+ const [isDeleteDialogOpen, setIsDeleteDialogOpen] = useState(false);
100
+ const [roleToDelete, setRoleToDelete] = useState<Role | null>(null);
101
+ const [isPermissionDetailOpen, setIsPermissionDetailOpen] = useState(false);
102
+ const [selectedPermission, setSelectedPermission] = useState<string | null>(
103
+ null,
104
+ );
105
+
106
+ // Build dynamic permission matrix from resources and actions
107
+ const permissionMatrix = useMemo(() => {
108
+ return resources.map((resource) => {
109
+ // Default actions using database configuration
110
+ let allowedActions: string[] = [];
111
+ const defaultActions = actions.filter((a) => a.isDefault).map((a) => a.code);
112
+ // Fallback to hardcoded if no default actions found (safety check)
113
+ const effectiveDefaultActions = defaultActions.length > 0
114
+ ? defaultActions
115
+ : ["view", "create", "update", "delete"];
116
+
117
+ if (resource.config) {
118
+ try {
119
+ const config =
120
+ typeof resource.config === "string"
121
+ ? JSON.parse(resource.config)
122
+ : resource.config;
123
+
124
+ if (config.actions && Array.isArray(config.actions)) {
125
+ // Case 2: Config exists -> use configured actions (even if empty)
126
+ const allowedCodes = config.actions;
127
+ allowedActions = actions
128
+ .map((a) => a.code)
129
+ .filter((code) => allowedCodes.includes(code));
130
+ } else {
131
+ // Config exists but no actions array -> use defaults
132
+ allowedActions = actions
133
+ .map((a) => a.code)
134
+ .filter((code) => effectiveDefaultActions.includes(code));
135
+ }
136
+ } catch (e) {
137
+ // Ignore config parsing errors, fallback to defaults
138
+ console.warn("Failed to parse resource config", e);
139
+ allowedActions = actions
140
+ .map((a) => a.code)
141
+ .filter((code) => effectiveDefaultActions.includes(code));
142
+ }
143
+ } else {
144
+ // Case 1: No config -> use defaults
145
+ allowedActions = actions
146
+ .map((a) => a.code)
147
+ .filter((code) => effectiveDefaultActions.includes(code));
148
+ }
149
+
150
+ return {
151
+ resource: resource.code,
152
+ name: resource.name,
153
+ icon: resource.icon || "📄",
154
+ actions: allowedActions,
155
+ };
156
+ });
157
+ }, [resources, actions]);
158
+
159
+ const totalPermissions = useMemo(
160
+ () => permissionMatrix.flatMap((r) => r.actions).length,
161
+ [permissionMatrix],
162
+ );
163
+
164
+ const totalUsers = useMemo(
165
+ () => roles.reduce((sum, role) => sum + role.usersCount, 0),
166
+ [roles],
167
+ );
168
+
169
+ // Handlers
170
+ const openDeleteDialog = useCallback((role: Role) => {
171
+ setRoleToDelete(role);
172
+ setIsDeleteDialogOpen(true);
173
+ }, []);
174
+
175
+ const handleDelete = useCallback(async () => {
176
+ if (!roleToDelete) return;
177
+ const success = await handleDeleteRole(roleToDelete.id);
178
+ if (success) {
179
+ setIsDeleteDialogOpen(false);
180
+ setRoleToDelete(null);
181
+ }
182
+ }, [roleToDelete, handleDeleteRole]);
183
+
184
+ // Memoize users with permission calculation
185
+ const getUsersForPermission = useCallback(
186
+ (permission: string) => {
187
+ return getUsersWithPermission(permission, roles);
188
+ },
189
+ [roles],
190
+ );
191
+
192
+ if (isLoading) {
193
+ return (
194
+ <div className="flex items-center justify-center h-96">
195
+ <Loader2 className="h-8 w-8 animate-spin text-muted-foreground" />
196
+ </div>
197
+ );
198
+ }
199
+
200
+ if (error) {
201
+ return (
202
+ <div className="text-red-500">Không thể tải dữ liệu: {error.message}</div>
203
+ );
204
+ }
205
+
206
+ return (
207
+ <div className="space-y-4">
208
+ <RoleStatsCards
209
+ totalRoles={totalRows}
210
+ activeRoles={activeRolesCount}
211
+ totalPermissions={totalPermissions}
212
+ totalUsers={totalUsers}
213
+ />
214
+
215
+ <RoleToolbar
216
+ dictionary={dictionary}
217
+ totalRows={totalRows}
218
+ searchTerm={searchTerm}
219
+ statusFilter={statusFilter}
220
+ onSearchChange={setSearchTerm}
221
+ onStatusFilterChange={setStatusFilter}
222
+ lang={lang}
223
+ canCreate={canCreate}
224
+ canImport={canImport}
225
+ canExport={canExport}
226
+ />
227
+
228
+ {/* Roles Grid */}
229
+ <div className="grid grid-cols-1 sm:grid-cols-2 lg:grid-cols-3 xl:grid-cols-4 gap-4">
230
+ {roles.length === 0 && !isLoading ? (
231
+ <div className="col-span-full text-center py-8">
232
+ <p className="text-muted-foreground">
233
+ {roles.length === 0
234
+ ? "Không có dữ liệu roles"
235
+ : "Không tìm thấy roles phù hợp với bộ lọc"}
236
+ </p>
237
+ </div>
238
+ ) : null}
239
+ {roles.map((role) => (
240
+ <RoleCard
241
+ key={role.id}
242
+ role={role}
243
+ lang={lang}
244
+ canUpdate={canUpdate}
245
+ canCreate={canCreate}
246
+ canDelete={canDelete}
247
+ onDuplicate={handleDuplicateRole}
248
+ onDelete={openDeleteDialog}
249
+ />
250
+ ))}
251
+ </div>
252
+
253
+ {/* Delete Confirmation Dialog */}
254
+ <AlertDialog
255
+ open={isDeleteDialogOpen}
256
+ onOpenChange={setIsDeleteDialogOpen}
257
+ >
258
+ <AlertDialogContent>
259
+ <AlertDialogHeader>
260
+ <AlertDialogTitle>Xác nhận xóa vai trò</AlertDialogTitle>
261
+ <AlertDialogDescription>
262
+ Bạn có chắc chắn muốn xóa vai trò "{roleToDelete?.name}"? Hành
263
+ động này không thể hoàn tác.
264
+ </AlertDialogDescription>
265
+ </AlertDialogHeader>
266
+ <AlertDialogFooter>
267
+ <AlertDialogCancel onClick={() => setRoleToDelete(null)}>
268
+ Hủy
269
+ </AlertDialogCancel>
270
+ <AlertDialogAction onClick={handleDelete}>Xóa</AlertDialogAction>
271
+ </AlertDialogFooter>
272
+ </AlertDialogContent>
273
+ </AlertDialog>
274
+
275
+ {/* Permission Detail Dialog */}
276
+ <Dialog
277
+ open={isPermissionDetailOpen}
278
+ onOpenChange={setIsPermissionDetailOpen}
279
+ >
280
+ <DialogContent className="max-w-2xl">
281
+ <DialogHeader>
282
+ <DialogTitle className="flex items-center space-x-2">
283
+ <KeyRound className="h-5 w-5" />
284
+ <span>Chi tiết Quyền hạn</span>
285
+ </DialogTitle>
286
+ <DialogDescription>
287
+ Xem danh sách người dùng được gán quyền hạn này
288
+ </DialogDescription>
289
+ </DialogHeader>
290
+
291
+ {selectedPermission && (
292
+ <div className="space-y-4">
293
+ <div className="p-4 bg-muted/50 rounded-lg">
294
+ <div className="flex items-center space-x-2 mb-2">
295
+ <Badge variant="outline" className="text-sm">
296
+ {selectedPermission.split(":")[0]}
297
+ </Badge>
298
+ <span className="text-muted-foreground">:</span>
299
+ <Badge variant="outline" className="text-sm">
300
+ {selectedPermission.split(":")[1]}
301
+ </Badge>
302
+ </div>
303
+ <p className="text-sm text-muted-foreground">
304
+ {getPermissionDescription(selectedPermission)}
305
+ </p>
306
+ </div>
307
+
308
+ {/* ... rest of the dialog content ... */}
309
+
310
+ <div>
311
+ <h4 className="font-medium mb-3">
312
+ Người dùng có quyền này (
313
+ {getUsersForPermission(selectedPermission).length})
314
+ </h4>
315
+ <div className="space-y-2 max-h-60 overflow-y-auto">
316
+ {getUsersForPermission(selectedPermission).length > 0 ? (
317
+ getUsersForPermission(selectedPermission).map((user) => (
318
+ <div
319
+ key={user.id}
320
+ className="flex items-center space-x-3 p-3 bg-muted/30 rounded-lg"
321
+ >
322
+ <Avatar className="h-10 w-10">
323
+ <AvatarImage
324
+ src={user.image || undefined}
325
+ alt={user.name || ""}
326
+ />
327
+ <AvatarFallback>
328
+ {(user.name || "U")
329
+ .split(" ")
330
+ .map((n: string) => n[0])
331
+ .join("")}
332
+ </AvatarFallback>
333
+ </Avatar>
334
+ <div className="flex-1">
335
+ <p className="font-medium">{user.name}</p>
336
+ <p className="text-sm text-muted-foreground">
337
+ {user.jobTitle || "N/A"}
338
+ </p>
339
+ <p className="text-xs text-muted-foreground">
340
+ {user.email}
341
+ </p>
342
+ </div>
343
+ <div className="flex items-center space-x-2">
344
+ {user.isActive ? (
345
+ <Badge variant="default" className="text-xs">
346
+ <UserCheck className="h-3 w-3 mr-1" />
347
+ Active
348
+ </Badge>
349
+ ) : (
350
+ <Badge variant="secondary" className="text-xs">
351
+ <UserX className="h-3 w-3 mr-1" />
352
+ Inactive
353
+ </Badge>
354
+ )}
355
+ </div>
356
+ </div>
357
+ ))
358
+ ) : (
359
+ <div className="text-center p-6 text-muted-foreground">
360
+ <UserX className="h-12 w-12 mx-auto mb-3 opacity-50" />
361
+ <p>Không có người dùng nào có quyền này</p>
362
+ </div>
363
+ )}
364
+ </div>
365
+ </div>
366
+ </div>
367
+ )}
368
+
369
+ <DialogFooter>
370
+ <Button onClick={() => setIsPermissionDetailOpen(false)}>
371
+ Đóng
372
+ </Button>
373
+ </DialogFooter>
374
+ </DialogContent>
375
+ </Dialog>
376
+ </div>
377
+ );
378
+ }
@@ -0,0 +1,140 @@
1
+ import type { Permission } from "../types";
2
+ import { logger } from "../utils";
3
+ import { getFromCache, pendingQueries, setToCache } from "./index";
4
+
5
+ /**
6
+ * Interface defining the subset of Prisma Client required for RBAC operations.
7
+ * This allows the service to be agnostic of the actual PrismaClient instance.
8
+ */
9
+ export interface RBACPrismaClient {
10
+ userRole: {
11
+ findMany: (args: {
12
+ where: { userId: string };
13
+ select: {
14
+ roleCode: boolean;
15
+ role: {
16
+ select: {
17
+ rolePermissions: {
18
+ select: {
19
+ resourceCode: boolean;
20
+ actionCode: boolean;
21
+ };
22
+ };
23
+ };
24
+ };
25
+ };
26
+ }) => Promise<
27
+ Array<{
28
+ roleCode: string;
29
+ role: {
30
+ rolePermissions: Array<{
31
+ resourceCode: string;
32
+ actionCode: string;
33
+ }>;
34
+ };
35
+ }>
36
+ >;
37
+ };
38
+ }
39
+
40
+ /**
41
+ * Fetch user permissions from database based on roles.
42
+ * Deduplicates permissions by resourceCode:actionCode.
43
+ *
44
+ * @param db - The tenant-specific Prisma Client instance
45
+ * @param userId - The user ID to fetch permissions for
46
+ */
47
+ export async function getUserPermissionsFromDB(
48
+ db: RBACPrismaClient,
49
+ userId: string,
50
+ ): Promise<Permission[]> {
51
+ try {
52
+ const userRoles = await db.userRole.findMany({
53
+ where: { userId },
54
+ select: {
55
+ roleCode: true,
56
+ role: {
57
+ select: {
58
+ rolePermissions: {
59
+ select: {
60
+ resourceCode: true,
61
+ actionCode: true,
62
+ },
63
+ },
64
+ },
65
+ },
66
+ },
67
+ });
68
+
69
+ if (userRoles.length === 0) {
70
+ return [];
71
+ }
72
+
73
+ // Flatten permissions from all roles
74
+ const allPermissions = userRoles.flatMap((ur) =>
75
+ ur.role.rolePermissions.map((rp) => ({
76
+ resourceCode: rp.resourceCode,
77
+ actionCode: rp.actionCode,
78
+ })),
79
+ );
80
+
81
+ // Deduplicate by resourceCode:actionCode
82
+ const uniquePermissions = Array.from(
83
+ new Map(
84
+ allPermissions.map((p) => [
85
+ `${p.resourceCode}:${p.actionCode}`,
86
+ {
87
+ resourceCode: p.resourceCode,
88
+ actionCode: p.actionCode,
89
+ } as Permission,
90
+ ]),
91
+ ).values(),
92
+ ) as Permission[];
93
+
94
+ return uniquePermissions;
95
+ } catch (error) {
96
+ logger.error("Error fetching user permissions from DB", error);
97
+ return [];
98
+ }
99
+ }
100
+
101
+ /**
102
+ * Get permissions with caching.
103
+ *
104
+ * @param db - The tenant-specific Prisma Client instance
105
+ * @param userId - The user ID
106
+ * @param opts.forceRefresh - Bypass cache
107
+ */
108
+ export async function getCachedUserPermissions(
109
+ db: RBACPrismaClient,
110
+ userId: string,
111
+ opts?: { forceRefresh?: boolean },
112
+ ): Promise<Permission[]> {
113
+ const force = opts?.forceRefresh === true;
114
+
115
+ if (!force) {
116
+ const cached = await getFromCache(userId);
117
+ if (cached) return cached;
118
+ }
119
+
120
+ // Check if there's already a pending query for this user
121
+ const pendingQuery = pendingQueries.get(userId);
122
+ if (pendingQuery) {
123
+ return pendingQuery;
124
+ }
125
+
126
+ // Create new query promise
127
+ const queryPromise = getUserPermissionsFromDB(db, userId)
128
+ .then(async (perms) => {
129
+ await setToCache(userId, perms);
130
+ pendingQueries.delete(userId); // Remove from pending when done
131
+ return perms;
132
+ })
133
+ .catch((error) => {
134
+ pendingQueries.delete(userId); // Remove from pending on error
135
+ throw error;
136
+ });
137
+
138
+ pendingQueries.set(userId, queryPromise);
139
+ return queryPromise;
140
+ }
@@ -0,0 +1,135 @@
1
+ // @goerp/core/rbac/permissions
2
+ // Server-only permission utilities - no client/UI imports
3
+ // Use this in Server Components and API routes to avoid bundling client code
4
+
5
+ import type { Permission } from "../types";
6
+ import type { Session } from "next-auth";
7
+
8
+ // ============================================================================
9
+ // Action Mapping
10
+ // ============================================================================
11
+
12
+ export const CRUD_ACTIONS = {
13
+ create: "create",
14
+ view: "view",
15
+ update: "update",
16
+ delete: "delete",
17
+ export: "export",
18
+ import: "import",
19
+ approve: "approve",
20
+ reject: "reject",
21
+ } as const;
22
+
23
+ export type CrudAction = keyof typeof CRUD_ACTIONS;
24
+
25
+ export function getActionCode(operation: CrudAction): string {
26
+ return CRUD_ACTIONS[operation];
27
+ }
28
+
29
+ // ============================================================================
30
+ // Permission Helpers
31
+ // ============================================================================
32
+
33
+ interface ExtendedUser {
34
+ id: string;
35
+ name?: string | null;
36
+ email?: string | null;
37
+ image?: string | null;
38
+ roles?: string[];
39
+ permissions?: Permission[];
40
+ }
41
+
42
+ const BYPASS_AUTH =
43
+ process.env.BYPASS_AUTH === "true" || process.env.BYPASS_AUTH === "1";
44
+
45
+ const ADMIN_ROLE_CODES = ["admin", "SUPER_ADMIN"];
46
+
47
+ export function getCrudPermissionsFromSession(
48
+ session: Session | null,
49
+ entity: string,
50
+ ): {
51
+ create: boolean;
52
+ view: boolean;
53
+ update: boolean;
54
+ delete: boolean;
55
+ export: boolean;
56
+ import: boolean;
57
+ approve: boolean;
58
+ reject: boolean;
59
+ } {
60
+ if (BYPASS_AUTH) {
61
+ return {
62
+ create: true,
63
+ view: true,
64
+ update: true,
65
+ delete: true,
66
+ export: true,
67
+ import: true,
68
+ approve: true,
69
+ reject: true,
70
+ };
71
+ }
72
+
73
+ if (!session?.user) {
74
+ return {
75
+ create: false,
76
+ view: false,
77
+ update: false,
78
+ delete: false,
79
+ export: false,
80
+ import: false,
81
+ approve: false,
82
+ reject: false,
83
+ };
84
+ }
85
+
86
+ const user = session.user as ExtendedUser;
87
+
88
+ if (!user.id) {
89
+ return {
90
+ create: false,
91
+ view: false,
92
+ update: false,
93
+ delete: false,
94
+ export: false,
95
+ import: false,
96
+ approve: false,
97
+ reject: false,
98
+ };
99
+ }
100
+
101
+ const isAdmin = user.roles?.some((role) => ADMIN_ROLE_CODES.includes(role));
102
+ if (isAdmin) {
103
+ return {
104
+ create: true,
105
+ view: true,
106
+ update: true,
107
+ delete: true,
108
+ export: true,
109
+ import: true,
110
+ approve: true,
111
+ reject: true,
112
+ };
113
+ }
114
+
115
+ const permissions = user.permissions || [];
116
+ const permissionKeys = new Set(
117
+ permissions.map((p) => `${p.resourceCode}:${p.actionCode}`),
118
+ );
119
+
120
+ const hasPermission = (action: string) => {
121
+ const key = `${entity}:${action}`;
122
+ return permissionKeys.has(key);
123
+ };
124
+
125
+ return {
126
+ create: hasPermission(getActionCode("create")),
127
+ view: hasPermission(getActionCode("view")),
128
+ update: hasPermission(getActionCode("update")),
129
+ delete: hasPermission(getActionCode("delete")),
130
+ export: hasPermission(getActionCode("export")),
131
+ import: hasPermission(getActionCode("import")),
132
+ approve: hasPermission(getActionCode("approve")),
133
+ reject: hasPermission(getActionCode("reject")),
134
+ };
135
+ }