@goplusvn/core 0.1.0

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 (223) hide show
  1. package/dist/audit/index.d.mts +115 -0
  2. package/dist/audit/index.d.ts +115 -0
  3. package/dist/audit/index.js +204 -0
  4. package/dist/audit/index.js.map +1 -0
  5. package/dist/audit/index.mjs +200 -0
  6. package/dist/audit/index.mjs.map +1 -0
  7. package/dist/auth/index.d.mts +86 -0
  8. package/dist/auth/index.d.ts +86 -0
  9. package/dist/auth/index.js +210 -0
  10. package/dist/auth/index.js.map +1 -0
  11. package/dist/auth/index.mjs +198 -0
  12. package/dist/auth/index.mjs.map +1 -0
  13. package/dist/button-1dWvP9Ib.d.mts +30 -0
  14. package/dist/button-1dWvP9Ib.d.ts +30 -0
  15. package/dist/calendar-2QzdEo1z.d.mts +20 -0
  16. package/dist/calendar-2QzdEo1z.d.ts +20 -0
  17. package/dist/code-generation/index.d.mts +30 -0
  18. package/dist/code-generation/index.d.ts +30 -0
  19. package/dist/code-generation/index.js +31 -0
  20. package/dist/code-generation/index.js.map +1 -0
  21. package/dist/code-generation/index.mjs +28 -0
  22. package/dist/code-generation/index.mjs.map +1 -0
  23. package/dist/configs/index.d.mts +175 -0
  24. package/dist/configs/index.d.ts +175 -0
  25. package/dist/configs/index.js +254 -0
  26. package/dist/configs/index.js.map +1 -0
  27. package/dist/configs/index.mjs +233 -0
  28. package/dist/configs/index.mjs.map +1 -0
  29. package/dist/crud/index.d.mts +646 -0
  30. package/dist/crud/index.d.ts +646 -0
  31. package/dist/crud/index.js +11772 -0
  32. package/dist/crud/index.js.map +1 -0
  33. package/dist/crud/index.mjs +11665 -0
  34. package/dist/crud/index.mjs.map +1 -0
  35. package/dist/crud/server.d.mts +20 -0
  36. package/dist/crud/server.d.ts +20 -0
  37. package/dist/crud/server.js +123 -0
  38. package/dist/crud/server.js.map +1 -0
  39. package/dist/crud/server.mjs +120 -0
  40. package/dist/crud/server.mjs.map +1 -0
  41. package/dist/data-table-skeleton-12NA8Mjx.d.mts +39 -0
  42. package/dist/data-table-skeleton-12NA8Mjx.d.ts +39 -0
  43. package/dist/dialog-bKfjZMTd.d.mts +22 -0
  44. package/dist/dialog-bKfjZMTd.d.ts +22 -0
  45. package/dist/dynamic-icon-DrGIiu2N.d.mts +10 -0
  46. package/dist/dynamic-icon-DrGIiu2N.d.ts +10 -0
  47. package/dist/home/index.d.mts +269 -0
  48. package/dist/home/index.d.ts +269 -0
  49. package/dist/home/index.js +1678 -0
  50. package/dist/home/index.js.map +1 -0
  51. package/dist/home/index.mjs +1635 -0
  52. package/dist/home/index.mjs.map +1 -0
  53. package/dist/hooks/index.d.mts +7 -0
  54. package/dist/hooks/index.d.ts +7 -0
  55. package/dist/hooks/index.js +8316 -0
  56. package/dist/hooks/index.js.map +1 -0
  57. package/dist/hooks/index.mjs +8255 -0
  58. package/dist/hooks/index.mjs.map +1 -0
  59. package/dist/index-50hpiPrV.d.ts +116 -0
  60. package/dist/index-B9zQVEVi.d.mts +116 -0
  61. package/dist/index.d.mts +5 -0
  62. package/dist/index.d.ts +5 -0
  63. package/dist/index.js +123 -0
  64. package/dist/index.js.map +1 -0
  65. package/dist/index.mjs +118 -0
  66. package/dist/index.mjs.map +1 -0
  67. package/dist/infrastructure/index.d.mts +423 -0
  68. package/dist/infrastructure/index.d.ts +423 -0
  69. package/dist/infrastructure/index.js +633 -0
  70. package/dist/infrastructure/index.js.map +1 -0
  71. package/dist/infrastructure/index.mjs +619 -0
  72. package/dist/infrastructure/index.mjs.map +1 -0
  73. package/dist/label-DWTEkNPo.d.ts +226 -0
  74. package/dist/label-LPpdcoBx.d.mts +226 -0
  75. package/dist/layout/index.d.mts +48 -0
  76. package/dist/layout/index.d.ts +48 -0
  77. package/dist/layout/index.js +117 -0
  78. package/dist/layout/index.js.map +1 -0
  79. package/dist/layout/index.mjs +90 -0
  80. package/dist/layout/index.mjs.map +1 -0
  81. package/dist/navigation/index.d.mts +16 -0
  82. package/dist/navigation/index.d.ts +16 -0
  83. package/dist/navigation/index.js +53 -0
  84. package/dist/navigation/index.js.map +1 -0
  85. package/dist/navigation/index.mjs +50 -0
  86. package/dist/navigation/index.mjs.map +1 -0
  87. package/dist/notification/index.d.mts +105 -0
  88. package/dist/notification/index.d.ts +105 -0
  89. package/dist/notification/index.js +278 -0
  90. package/dist/notification/index.js.map +1 -0
  91. package/dist/notification/index.mjs +274 -0
  92. package/dist/notification/index.mjs.map +1 -0
  93. package/dist/organization/index.d.mts +99 -0
  94. package/dist/organization/index.d.ts +99 -0
  95. package/dist/organization/index.js +360 -0
  96. package/dist/organization/index.js.map +1 -0
  97. package/dist/organization/index.mjs +352 -0
  98. package/dist/organization/index.mjs.map +1 -0
  99. package/dist/plugin/index.d.mts +83 -0
  100. package/dist/plugin/index.d.ts +83 -0
  101. package/dist/plugin/index.js +86 -0
  102. package/dist/plugin/index.js.map +1 -0
  103. package/dist/plugin/index.mjs +84 -0
  104. package/dist/plugin/index.mjs.map +1 -0
  105. package/dist/providers/index.d.mts +25 -0
  106. package/dist/providers/index.d.ts +25 -0
  107. package/dist/providers/index.js +84 -0
  108. package/dist/providers/index.js.map +1 -0
  109. package/dist/providers/index.mjs +77 -0
  110. package/dist/providers/index.mjs.map +1 -0
  111. package/dist/rbac/index.d.mts +226 -0
  112. package/dist/rbac/index.d.ts +226 -0
  113. package/dist/rbac/index.js +4784 -0
  114. package/dist/rbac/index.js.map +1 -0
  115. package/dist/rbac/index.mjs +4722 -0
  116. package/dist/rbac/index.mjs.map +1 -0
  117. package/dist/rbac/permissions.d.mts +26 -0
  118. package/dist/rbac/permissions.d.ts +26 -0
  119. package/dist/rbac/permissions.js +94 -0
  120. package/dist/rbac/permissions.js.map +1 -0
  121. package/dist/rbac/permissions.mjs +90 -0
  122. package/dist/rbac/permissions.mjs.map +1 -0
  123. package/dist/rbac/server.d.mts +1 -0
  124. package/dist/rbac/server.d.ts +1 -0
  125. package/dist/rbac/server.js +128 -0
  126. package/dist/rbac/server.js.map +1 -0
  127. package/dist/rbac/server.mjs +124 -0
  128. package/dist/rbac/server.mjs.map +1 -0
  129. package/dist/schemas/index.d.mts +1257 -0
  130. package/dist/schemas/index.d.ts +1257 -0
  131. package/dist/schemas/index.js +572 -0
  132. package/dist/schemas/index.js.map +1 -0
  133. package/dist/schemas/index.mjs +523 -0
  134. package/dist/schemas/index.mjs.map +1 -0
  135. package/dist/server-QuYCTa89.d.mts +83 -0
  136. package/dist/server-QuYCTa89.d.ts +83 -0
  137. package/dist/sonner-C74GlRDQ.d.mts +71 -0
  138. package/dist/sonner-C74GlRDQ.d.ts +71 -0
  139. package/dist/status-BOXZgIqX.d.mts +12 -0
  140. package/dist/status-BOXZgIqX.d.ts +12 -0
  141. package/dist/system/index.d.mts +77 -0
  142. package/dist/system/index.d.ts +77 -0
  143. package/dist/system/index.js +102 -0
  144. package/dist/system/index.js.map +1 -0
  145. package/dist/system/index.mjs +100 -0
  146. package/dist/system/index.mjs.map +1 -0
  147. package/dist/tabs-C6FfBwPY.d.mts +18 -0
  148. package/dist/tabs-C6FfBwPY.d.ts +18 -0
  149. package/dist/tenant-provider-B8eC_Wpb.d.mts +27 -0
  150. package/dist/tenant-provider-B8eC_Wpb.d.ts +27 -0
  151. package/dist/types/index.d.mts +469 -0
  152. package/dist/types/index.d.ts +469 -0
  153. package/dist/types/index.js +25 -0
  154. package/dist/types/index.js.map +1 -0
  155. package/dist/types/index.mjs +21 -0
  156. package/dist/types/index.mjs.map +1 -0
  157. package/dist/ui/auth.d.mts +39 -0
  158. package/dist/ui/auth.d.ts +39 -0
  159. package/dist/ui/auth.js +4941 -0
  160. package/dist/ui/auth.js.map +1 -0
  161. package/dist/ui/auth.mjs +4896 -0
  162. package/dist/ui/auth.mjs.map +1 -0
  163. package/dist/ui/crud.d.mts +2 -0
  164. package/dist/ui/crud.d.ts +2 -0
  165. package/dist/ui/crud.js +4 -0
  166. package/dist/ui/crud.js.map +1 -0
  167. package/dist/ui/crud.mjs +3 -0
  168. package/dist/ui/crud.mjs.map +1 -0
  169. package/dist/ui/data-display.d.mts +596 -0
  170. package/dist/ui/data-display.d.ts +596 -0
  171. package/dist/ui/data-display.js +5307 -0
  172. package/dist/ui/data-display.js.map +1 -0
  173. package/dist/ui/data-display.mjs +5212 -0
  174. package/dist/ui/data-display.mjs.map +1 -0
  175. package/dist/ui/feedback.d.mts +55 -0
  176. package/dist/ui/feedback.d.ts +55 -0
  177. package/dist/ui/feedback.js +2608 -0
  178. package/dist/ui/feedback.js.map +1 -0
  179. package/dist/ui/feedback.mjs +2526 -0
  180. package/dist/ui/feedback.mjs.map +1 -0
  181. package/dist/ui/forms.d.mts +309 -0
  182. package/dist/ui/forms.d.ts +309 -0
  183. package/dist/ui/forms.js +4656 -0
  184. package/dist/ui/forms.js.map +1 -0
  185. package/dist/ui/forms.mjs +4571 -0
  186. package/dist/ui/forms.mjs.map +1 -0
  187. package/dist/ui/index.d.mts +331 -0
  188. package/dist/ui/index.d.ts +331 -0
  189. package/dist/ui/index.js +16953 -0
  190. package/dist/ui/index.js.map +1 -0
  191. package/dist/ui/index.mjs +16598 -0
  192. package/dist/ui/index.mjs.map +1 -0
  193. package/dist/ui/primitives/client.d.mts +61 -0
  194. package/dist/ui/primitives/client.d.ts +61 -0
  195. package/dist/ui/primitives/client.js +3408 -0
  196. package/dist/ui/primitives/client.js.map +1 -0
  197. package/dist/ui/primitives/client.mjs +3256 -0
  198. package/dist/ui/primitives/client.mjs.map +1 -0
  199. package/dist/ui/primitives.d.mts +113 -0
  200. package/dist/ui/primitives.d.ts +113 -0
  201. package/dist/ui/primitives.js +3356 -0
  202. package/dist/ui/primitives.js.map +1 -0
  203. package/dist/ui/primitives.mjs +3227 -0
  204. package/dist/ui/primitives.mjs.map +1 -0
  205. package/dist/user/index.d.mts +228 -0
  206. package/dist/user/index.d.ts +228 -0
  207. package/dist/user/index.js +4306 -0
  208. package/dist/user/index.js.map +1 -0
  209. package/dist/user/index.mjs +4260 -0
  210. package/dist/user/index.mjs.map +1 -0
  211. package/dist/utils/index.d.mts +205 -0
  212. package/dist/utils/index.d.ts +205 -0
  213. package/dist/utils/index.js +574 -0
  214. package/dist/utils/index.js.map +1 -0
  215. package/dist/utils/index.mjs +514 -0
  216. package/dist/utils/index.mjs.map +1 -0
  217. package/dist/workflow/index.d.mts +40 -0
  218. package/dist/workflow/index.d.ts +40 -0
  219. package/dist/workflow/index.js +3710 -0
  220. package/dist/workflow/index.js.map +1 -0
  221. package/dist/workflow/index.mjs +3677 -0
  222. package/dist/workflow/index.mjs.map +1 -0
  223. package/package.json +311 -0
@@ -0,0 +1,26 @@
1
+ import { Session } from 'next-auth';
2
+
3
+ declare const CRUD_ACTIONS: {
4
+ readonly create: "create";
5
+ readonly view: "view";
6
+ readonly update: "update";
7
+ readonly delete: "delete";
8
+ readonly export: "export";
9
+ readonly import: "import";
10
+ readonly approve: "approve";
11
+ readonly reject: "reject";
12
+ };
13
+ type CrudAction = keyof typeof CRUD_ACTIONS;
14
+ declare function getActionCode(operation: CrudAction): string;
15
+ declare function getCrudPermissionsFromSession(session: Session | null, entity: string): {
16
+ create: boolean;
17
+ view: boolean;
18
+ update: boolean;
19
+ delete: boolean;
20
+ export: boolean;
21
+ import: boolean;
22
+ approve: boolean;
23
+ reject: boolean;
24
+ };
25
+
26
+ export { CRUD_ACTIONS, type CrudAction, getActionCode, getCrudPermissionsFromSession };
@@ -0,0 +1,26 @@
1
+ import { Session } from 'next-auth';
2
+
3
+ declare const CRUD_ACTIONS: {
4
+ readonly create: "create";
5
+ readonly view: "view";
6
+ readonly update: "update";
7
+ readonly delete: "delete";
8
+ readonly export: "export";
9
+ readonly import: "import";
10
+ readonly approve: "approve";
11
+ readonly reject: "reject";
12
+ };
13
+ type CrudAction = keyof typeof CRUD_ACTIONS;
14
+ declare function getActionCode(operation: CrudAction): string;
15
+ declare function getCrudPermissionsFromSession(session: Session | null, entity: string): {
16
+ create: boolean;
17
+ view: boolean;
18
+ update: boolean;
19
+ delete: boolean;
20
+ export: boolean;
21
+ import: boolean;
22
+ approve: boolean;
23
+ reject: boolean;
24
+ };
25
+
26
+ export { CRUD_ACTIONS, type CrudAction, getActionCode, getCrudPermissionsFromSession };
@@ -0,0 +1,94 @@
1
+ 'use strict';
2
+
3
+ // src/rbac/permissions.ts
4
+ var CRUD_ACTIONS = {
5
+ create: "create",
6
+ view: "view",
7
+ update: "update",
8
+ delete: "delete",
9
+ export: "export",
10
+ import: "import",
11
+ approve: "approve",
12
+ reject: "reject"
13
+ };
14
+ function getActionCode(operation) {
15
+ return CRUD_ACTIONS[operation];
16
+ }
17
+ var BYPASS_AUTH = process.env.BYPASS_AUTH === "true" || process.env.BYPASS_AUTH === "1";
18
+ var ADMIN_ROLE_CODES = ["admin", "SUPER_ADMIN"];
19
+ function getCrudPermissionsFromSession(session, entity) {
20
+ if (BYPASS_AUTH) {
21
+ return {
22
+ create: true,
23
+ view: true,
24
+ update: true,
25
+ delete: true,
26
+ export: true,
27
+ import: true,
28
+ approve: true,
29
+ reject: true
30
+ };
31
+ }
32
+ if (!session?.user) {
33
+ return {
34
+ create: false,
35
+ view: false,
36
+ update: false,
37
+ delete: false,
38
+ export: false,
39
+ import: false,
40
+ approve: false,
41
+ reject: false
42
+ };
43
+ }
44
+ const user = session.user;
45
+ if (!user.id) {
46
+ return {
47
+ create: false,
48
+ view: false,
49
+ update: false,
50
+ delete: false,
51
+ export: false,
52
+ import: false,
53
+ approve: false,
54
+ reject: false
55
+ };
56
+ }
57
+ const isAdmin = user.roles?.some((role) => ADMIN_ROLE_CODES.includes(role));
58
+ if (isAdmin) {
59
+ return {
60
+ create: true,
61
+ view: true,
62
+ update: true,
63
+ delete: true,
64
+ export: true,
65
+ import: true,
66
+ approve: true,
67
+ reject: true
68
+ };
69
+ }
70
+ const permissions = user.permissions || [];
71
+ const permissionKeys = new Set(
72
+ permissions.map((p) => `${p.resourceCode}:${p.actionCode}`)
73
+ );
74
+ const hasPermission = (action) => {
75
+ const key = `${entity}:${action}`;
76
+ return permissionKeys.has(key);
77
+ };
78
+ return {
79
+ create: hasPermission(getActionCode("create")),
80
+ view: hasPermission(getActionCode("view")),
81
+ update: hasPermission(getActionCode("update")),
82
+ delete: hasPermission(getActionCode("delete")),
83
+ export: hasPermission(getActionCode("export")),
84
+ import: hasPermission(getActionCode("import")),
85
+ approve: hasPermission(getActionCode("approve")),
86
+ reject: hasPermission(getActionCode("reject"))
87
+ };
88
+ }
89
+
90
+ exports.CRUD_ACTIONS = CRUD_ACTIONS;
91
+ exports.getActionCode = getActionCode;
92
+ exports.getCrudPermissionsFromSession = getCrudPermissionsFromSession;
93
+ //# sourceMappingURL=permissions.js.map
94
+ //# sourceMappingURL=permissions.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../../src/rbac/permissions.ts"],"names":[],"mappings":"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","file":"permissions.js","sourcesContent":["// @goerp/core/rbac/permissions\n// Server-only permission utilities - no client/UI imports\n// Use this in Server Components and API routes to avoid bundling client code\n\nimport type { Permission } from \"../types\";\nimport type { Session } from \"next-auth\";\n\n// ============================================================================\n// Action Mapping\n// ============================================================================\n\nexport const CRUD_ACTIONS = {\n create: \"create\",\n view: \"view\",\n update: \"update\",\n delete: \"delete\",\n export: \"export\",\n import: \"import\",\n approve: \"approve\",\n reject: \"reject\",\n} as const;\n\nexport type CrudAction = keyof typeof CRUD_ACTIONS;\n\nexport function getActionCode(operation: CrudAction): string {\n return CRUD_ACTIONS[operation];\n}\n\n// ============================================================================\n// Permission Helpers\n// ============================================================================\n\ninterface ExtendedUser {\n id: string;\n name?: string | null;\n email?: string | null;\n image?: string | null;\n roles?: string[];\n permissions?: Permission[];\n}\n\nconst BYPASS_AUTH =\n process.env.BYPASS_AUTH === \"true\" || process.env.BYPASS_AUTH === \"1\";\n\nconst ADMIN_ROLE_CODES = [\"admin\", \"SUPER_ADMIN\"];\n\nexport function getCrudPermissionsFromSession(\n session: Session | null,\n entity: string,\n): {\n create: boolean;\n view: boolean;\n update: boolean;\n delete: boolean;\n export: boolean;\n import: boolean;\n approve: boolean;\n reject: boolean;\n} {\n if (BYPASS_AUTH) {\n return {\n create: true,\n view: true,\n update: true,\n delete: true,\n export: true,\n import: true,\n approve: true,\n reject: true,\n };\n }\n\n if (!session?.user) {\n return {\n create: false,\n view: false,\n update: false,\n delete: false,\n export: false,\n import: false,\n approve: false,\n reject: false,\n };\n }\n\n const user = session.user as ExtendedUser;\n\n if (!user.id) {\n return {\n create: false,\n view: false,\n update: false,\n delete: false,\n export: false,\n import: false,\n approve: false,\n reject: false,\n };\n }\n\n const isAdmin = user.roles?.some((role) => ADMIN_ROLE_CODES.includes(role));\n if (isAdmin) {\n return {\n create: true,\n view: true,\n update: true,\n delete: true,\n export: true,\n import: true,\n approve: true,\n reject: true,\n };\n }\n\n const permissions = user.permissions || [];\n const permissionKeys = new Set(\n permissions.map((p) => `${p.resourceCode}:${p.actionCode}`),\n );\n\n const hasPermission = (action: string) => {\n const key = `${entity}:${action}`;\n return permissionKeys.has(key);\n };\n\n return {\n create: hasPermission(getActionCode(\"create\")),\n view: hasPermission(getActionCode(\"view\")),\n update: hasPermission(getActionCode(\"update\")),\n delete: hasPermission(getActionCode(\"delete\")),\n export: hasPermission(getActionCode(\"export\")),\n import: hasPermission(getActionCode(\"import\")),\n approve: hasPermission(getActionCode(\"approve\")),\n reject: hasPermission(getActionCode(\"reject\")),\n };\n}\n"]}
@@ -0,0 +1,90 @@
1
+ // src/rbac/permissions.ts
2
+ var CRUD_ACTIONS = {
3
+ create: "create",
4
+ view: "view",
5
+ update: "update",
6
+ delete: "delete",
7
+ export: "export",
8
+ import: "import",
9
+ approve: "approve",
10
+ reject: "reject"
11
+ };
12
+ function getActionCode(operation) {
13
+ return CRUD_ACTIONS[operation];
14
+ }
15
+ var BYPASS_AUTH = process.env.BYPASS_AUTH === "true" || process.env.BYPASS_AUTH === "1";
16
+ var ADMIN_ROLE_CODES = ["admin", "SUPER_ADMIN"];
17
+ function getCrudPermissionsFromSession(session, entity) {
18
+ if (BYPASS_AUTH) {
19
+ return {
20
+ create: true,
21
+ view: true,
22
+ update: true,
23
+ delete: true,
24
+ export: true,
25
+ import: true,
26
+ approve: true,
27
+ reject: true
28
+ };
29
+ }
30
+ if (!session?.user) {
31
+ return {
32
+ create: false,
33
+ view: false,
34
+ update: false,
35
+ delete: false,
36
+ export: false,
37
+ import: false,
38
+ approve: false,
39
+ reject: false
40
+ };
41
+ }
42
+ const user = session.user;
43
+ if (!user.id) {
44
+ return {
45
+ create: false,
46
+ view: false,
47
+ update: false,
48
+ delete: false,
49
+ export: false,
50
+ import: false,
51
+ approve: false,
52
+ reject: false
53
+ };
54
+ }
55
+ const isAdmin = user.roles?.some((role) => ADMIN_ROLE_CODES.includes(role));
56
+ if (isAdmin) {
57
+ return {
58
+ create: true,
59
+ view: true,
60
+ update: true,
61
+ delete: true,
62
+ export: true,
63
+ import: true,
64
+ approve: true,
65
+ reject: true
66
+ };
67
+ }
68
+ const permissions = user.permissions || [];
69
+ const permissionKeys = new Set(
70
+ permissions.map((p) => `${p.resourceCode}:${p.actionCode}`)
71
+ );
72
+ const hasPermission = (action) => {
73
+ const key = `${entity}:${action}`;
74
+ return permissionKeys.has(key);
75
+ };
76
+ return {
77
+ create: hasPermission(getActionCode("create")),
78
+ view: hasPermission(getActionCode("view")),
79
+ update: hasPermission(getActionCode("update")),
80
+ delete: hasPermission(getActionCode("delete")),
81
+ export: hasPermission(getActionCode("export")),
82
+ import: hasPermission(getActionCode("import")),
83
+ approve: hasPermission(getActionCode("approve")),
84
+ reject: hasPermission(getActionCode("reject"))
85
+ };
86
+ }
87
+
88
+ export { CRUD_ACTIONS, getActionCode, getCrudPermissionsFromSession };
89
+ //# sourceMappingURL=permissions.mjs.map
90
+ //# sourceMappingURL=permissions.mjs.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../../src/rbac/permissions.ts"],"names":[],"mappings":"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","file":"permissions.mjs","sourcesContent":["// @goerp/core/rbac/permissions\n// Server-only permission utilities - no client/UI imports\n// Use this in Server Components and API routes to avoid bundling client code\n\nimport type { Permission } from \"../types\";\nimport type { Session } from \"next-auth\";\n\n// ============================================================================\n// Action Mapping\n// ============================================================================\n\nexport const CRUD_ACTIONS = {\n create: \"create\",\n view: \"view\",\n update: \"update\",\n delete: \"delete\",\n export: \"export\",\n import: \"import\",\n approve: \"approve\",\n reject: \"reject\",\n} as const;\n\nexport type CrudAction = keyof typeof CRUD_ACTIONS;\n\nexport function getActionCode(operation: CrudAction): string {\n return CRUD_ACTIONS[operation];\n}\n\n// ============================================================================\n// Permission Helpers\n// ============================================================================\n\ninterface ExtendedUser {\n id: string;\n name?: string | null;\n email?: string | null;\n image?: string | null;\n roles?: string[];\n permissions?: Permission[];\n}\n\nconst BYPASS_AUTH =\n process.env.BYPASS_AUTH === \"true\" || process.env.BYPASS_AUTH === \"1\";\n\nconst ADMIN_ROLE_CODES = [\"admin\", \"SUPER_ADMIN\"];\n\nexport function getCrudPermissionsFromSession(\n session: Session | null,\n entity: string,\n): {\n create: boolean;\n view: boolean;\n update: boolean;\n delete: boolean;\n export: boolean;\n import: boolean;\n approve: boolean;\n reject: boolean;\n} {\n if (BYPASS_AUTH) {\n return {\n create: true,\n view: true,\n update: true,\n delete: true,\n export: true,\n import: true,\n approve: true,\n reject: true,\n };\n }\n\n if (!session?.user) {\n return {\n create: false,\n view: false,\n update: false,\n delete: false,\n export: false,\n import: false,\n approve: false,\n reject: false,\n };\n }\n\n const user = session.user as ExtendedUser;\n\n if (!user.id) {\n return {\n create: false,\n view: false,\n update: false,\n delete: false,\n export: false,\n import: false,\n approve: false,\n reject: false,\n };\n }\n\n const isAdmin = user.roles?.some((role) => ADMIN_ROLE_CODES.includes(role));\n if (isAdmin) {\n return {\n create: true,\n view: true,\n update: true,\n delete: true,\n export: true,\n import: true,\n approve: true,\n reject: true,\n };\n }\n\n const permissions = user.permissions || [];\n const permissionKeys = new Set(\n permissions.map((p) => `${p.resourceCode}:${p.actionCode}`),\n );\n\n const hasPermission = (action: string) => {\n const key = `${entity}:${action}`;\n return permissionKeys.has(key);\n };\n\n return {\n create: hasPermission(getActionCode(\"create\")),\n view: hasPermission(getActionCode(\"view\")),\n update: hasPermission(getActionCode(\"update\")),\n delete: hasPermission(getActionCode(\"delete\")),\n export: hasPermission(getActionCode(\"export\")),\n import: hasPermission(getActionCode(\"import\")),\n approve: hasPermission(getActionCode(\"approve\")),\n reject: hasPermission(getActionCode(\"reject\")),\n };\n}\n"]}
@@ -0,0 +1 @@
1
+ export { R as ResourcePrismaClient, a as RoleData, b as RoleFilters, c as RolePrismaClient, g as getActionsForPermissionMatrix, e as getResourcesForPermissionMatrix, f as getRolesData } from '../server-QuYCTa89.mjs';
@@ -0,0 +1 @@
1
+ export { R as ResourcePrismaClient, a as RoleData, b as RoleFilters, c as RolePrismaClient, g as getActionsForPermissionMatrix, e as getResourcesForPermissionMatrix, f as getRolesData } from '../server-QuYCTa89.js';
@@ -0,0 +1,128 @@
1
+ 'use strict';
2
+
3
+ // src/rbac/role-service.ts
4
+ async function getRolesData(db, params = {}) {
5
+ const { page = 1, pageSize = 10, search, status } = params;
6
+ const whereConditions = [];
7
+ if (search) {
8
+ whereConditions.push({
9
+ OR: [
10
+ { name: { contains: search, mode: "insensitive" } },
11
+ { code: { contains: search, mode: "insensitive" } },
12
+ { description: { contains: search, mode: "insensitive" } }
13
+ ]
14
+ });
15
+ }
16
+ if (status && status !== "all") {
17
+ whereConditions.push({ status });
18
+ }
19
+ const where = whereConditions.length > 0 ? { AND: whereConditions } : {};
20
+ const [total, items] = await Promise.all([
21
+ db.role.count({ where }),
22
+ db.role.findMany({
23
+ where,
24
+ orderBy: { createdAt: "desc" },
25
+ skip: (page - 1) * pageSize,
26
+ take: pageSize,
27
+ include: {
28
+ userRoles: {
29
+ include: {
30
+ user: {
31
+ select: {
32
+ id: true,
33
+ name: true,
34
+ email: true,
35
+ image: true,
36
+ isActive: true
37
+ }
38
+ }
39
+ }
40
+ },
41
+ rolePermissions: {
42
+ include: {
43
+ resource: {
44
+ select: {
45
+ code: true,
46
+ name: true,
47
+ icon: true
48
+ }
49
+ },
50
+ action: {
51
+ select: {
52
+ code: true,
53
+ name: true
54
+ }
55
+ }
56
+ }
57
+ }
58
+ }
59
+ })
60
+ ]);
61
+ const transformedItems = items.map((role) => ({
62
+ id: role.id,
63
+ name: role.name,
64
+ code: role.code,
65
+ description: role.description || void 0,
66
+ status: role.status,
67
+ // Transform rolePermissions to array of "action:resource" strings
68
+ permissions: role.rolePermissions.map(
69
+ (rp) => `${rp.actionCode}:${rp.resourceCode}`
70
+ ),
71
+ usersCount: role.userRoles.length,
72
+ users: role.userRoles.map((ur) => ({
73
+ id: ur.user.id,
74
+ name: ur.user.name,
75
+ email: ur.user.email,
76
+ image: ur.user.image,
77
+ isActive: ur.user.isActive
78
+ })),
79
+ createdAt: role.createdAt.toISOString(),
80
+ updatedAt: role.updatedAt.toISOString(),
81
+ createdBy: role.createdBy || void 0,
82
+ updatedBy: role.updatedBy || void 0
83
+ }));
84
+ return { total, page, pageSize, items: transformedItems };
85
+ }
86
+
87
+ // src/rbac/resource-service.ts
88
+ async function getResourcesForPermissionMatrix(db) {
89
+ return await db.resource.findMany({
90
+ where: {
91
+ status: "active"
92
+ },
93
+ orderBy: [{ group: "asc" }, { order: "asc" }, { name: "asc" }],
94
+ select: {
95
+ id: true,
96
+ code: true,
97
+ name: true,
98
+ group: true,
99
+ description: true,
100
+ icon: true,
101
+ order: true,
102
+ config: true
103
+ }
104
+ });
105
+ }
106
+ async function getActionsForPermissionMatrix(db) {
107
+ return await db.action.findMany({
108
+ where: {
109
+ status: "active"
110
+ },
111
+ orderBy: {
112
+ code: "asc"
113
+ },
114
+ select: {
115
+ id: true,
116
+ code: true,
117
+ name: true,
118
+ description: true,
119
+ isDefault: true
120
+ }
121
+ });
122
+ }
123
+
124
+ exports.getActionsForPermissionMatrix = getActionsForPermissionMatrix;
125
+ exports.getResourcesForPermissionMatrix = getResourcesForPermissionMatrix;
126
+ exports.getRolesData = getRolesData;
127
+ //# sourceMappingURL=server.js.map
128
+ //# sourceMappingURL=server.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../../src/rbac/role-service.ts","../../src/rbac/resource-service.ts"],"names":[],"mappings":"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","file":"server.js","sourcesContent":["// Note: We use Prisma types for input construction, but we don't import the client instance.\n// If Prisma types are not available in core (since core doesn't have prisma schema),\n// we might need to define loose types or rely on the injected client's type inference if strict mode allows.\n// However, since we want to be Type-safe, we assume the host app uses standard Prisma conventions.\n// For core, we can define the expected structure of the Include/Where types if needed,\n// or simply use 'any' for the complex Prisma types if we want to avoid strict coupling to a specific schema version.\n// To be safe and generic in 'core', we will define the input params using standard TypeScript types\n// and the db client interface locally.\n\n// ============================================================================\n// Types\n// ============================================================================\n\nexport type RoleFilters = {\n page?: number;\n pageSize?: number;\n search?: string;\n status?: string;\n};\n\nexport type RoleData = {\n id: string;\n name: string;\n code: string;\n description?: string;\n status: string;\n permissions: string[];\n usersCount: number;\n users: {\n id: string;\n name: string | null;\n email: string | null;\n image: string | null;\n isActive: boolean;\n }[];\n createdAt: string;\n updatedAt: string;\n createdBy?: string;\n updatedBy?: string;\n};\n\n// Interface for the DB client injected into the service\n// We define the shape of the Prisma Client we expect\nexport interface RolePrismaClient {\n role: {\n count: (args: { where: any }) => Promise<number>;\n findMany: (args: {\n where: any;\n orderBy: any;\n skip: number;\n take: number;\n include: any;\n }) => Promise<any[]>;\n };\n}\n\n// ============================================================================\n// Service Logic\n// ============================================================================\n\n/**\n * Get roles data with pagination and filtering\n * Generic core implementation using injected DB client\n *\n * @param db - The tenant-specific Prisma Client instance\n * @param params - Filter and pagination parameters\n */\nexport async function getRolesData(\n db: RolePrismaClient,\n params: RoleFilters = {},\n): Promise<{\n total: number;\n page: number;\n pageSize: number;\n items: RoleData[];\n}> {\n const { page = 1, pageSize = 10, search, status } = params;\n\n const whereConditions: any[] = [];\n\n if (search) {\n whereConditions.push({\n OR: [\n { name: { contains: search, mode: \"insensitive\" } },\n { code: { contains: search, mode: \"insensitive\" } },\n { description: { contains: search, mode: \"insensitive\" } },\n ],\n });\n }\n\n if (status && status !== \"all\") {\n whereConditions.push({ status });\n }\n\n const where: any = whereConditions.length > 0 ? { AND: whereConditions } : {};\n\n const [total, items] = await Promise.all([\n db.role.count({ where }),\n db.role.findMany({\n where,\n orderBy: { createdAt: \"desc\" },\n skip: (page - 1) * pageSize,\n take: pageSize,\n include: {\n userRoles: {\n include: {\n user: {\n select: {\n id: true,\n name: true,\n email: true,\n image: true,\n isActive: true,\n },\n },\n },\n },\n rolePermissions: {\n include: {\n resource: {\n select: {\n code: true,\n name: true,\n icon: true,\n },\n },\n action: {\n select: {\n code: true,\n name: true,\n },\n },\n },\n },\n },\n }),\n ]);\n\n // Transform raw DB items to RoleData type\n const transformedItems = items.map((role: any) => ({\n id: role.id,\n name: role.name,\n code: role.code,\n description: role.description || undefined,\n status: role.status,\n // Transform rolePermissions to array of \"action:resource\" strings\n permissions: role.rolePermissions.map(\n (rp: any) => `${rp.actionCode}:${rp.resourceCode}`,\n ),\n usersCount: role.userRoles.length,\n users: role.userRoles.map((ur: any) => ({\n id: ur.user.id,\n name: ur.user.name,\n email: ur.user.email,\n image: ur.user.image,\n isActive: ur.user.isActive,\n })),\n createdAt: role.createdAt.toISOString(),\n updatedAt: role.updatedAt.toISOString(),\n createdBy: role.createdBy || undefined,\n updatedBy: role.updatedBy || undefined,\n }));\n\n return { total, page, pageSize, items: transformedItems };\n}\n","export interface ResourcePrismaClient {\n resource: any;\n action: any;\n}\n\n/**\n * Get resources data with pagination and filtering\n * Shared function for both API route and Server Component\n *\n * @param db - The tenant-specific Prisma Client instance\n * @param params - Filter and pagination parameters\n */\nexport async function getResourcesData(\n db: ResourcePrismaClient,\n params: any = {},\n) {\n const { page = 1, pageSize = 10, search, status, group, type } = params;\n\n const where: any = {\n AND: [\n search\n ? {\n OR: [\n { code: { contains: search, mode: \"insensitive\" } },\n { name: { contains: search, mode: \"insensitive\" } },\n { description: { contains: search, mode: \"insensitive\" } },\n { path: { contains: search, mode: \"insensitive\" } },\n ],\n }\n : {},\n status && status !== \"all\"\n ? { status: { equals: status, mode: \"insensitive\" } }\n : {},\n group && group !== \"all\" ? { group } : {},\n type && type !== \"all\" ? { type } : {},\n ],\n };\n\n const [total, items] = await Promise.all([\n db.resource.count({ where }),\n db.resource.findMany({\n where,\n orderBy: [{ order: \"asc\" }, { updatedAt: \"desc\" }],\n skip: (page - 1) * pageSize,\n take: pageSize,\n }),\n ]);\n\n // Transform Prisma items to Resource type format\n const transformedItems = items.map((item: any) => ({\n id: item.id,\n code: item.code,\n name: item.name,\n group: item.group || undefined,\n description: item.description || undefined,\n status: item.status,\n parentCode: item.parentCode || undefined,\n order: item.order || undefined,\n type: item.type || undefined,\n icon: item.icon || undefined,\n path: item.path || undefined,\n config: item.config || undefined,\n createdAt: item.createdAt.toISOString(),\n updatedAt: item.updatedAt.toISOString(),\n }));\n\n return { total, page, pageSize, items: transformedItems };\n}\n\n/**\n * Get all active resources for permission matrix\n * Used in role management to build dynamic permission matrix\n */\nexport async function getResourcesForPermissionMatrix(\n db: ResourcePrismaClient,\n) {\n return await db.resource.findMany({\n where: {\n status: \"active\",\n },\n orderBy: [{ group: \"asc\" }, { order: \"asc\" }, { name: \"asc\" }],\n select: {\n id: true,\n code: true,\n name: true,\n group: true,\n description: true,\n icon: true,\n order: true,\n config: true,\n },\n });\n}\n\n/**\n * Get all active actions for permission matrix\n * Used in role management to build dynamic permission matrix\n */\nexport async function getActionsForPermissionMatrix(db: ResourcePrismaClient) {\n return await db.action.findMany({\n where: {\n status: \"active\",\n },\n orderBy: {\n code: \"asc\",\n },\n select: {\n id: true,\n code: true,\n name: true,\n description: true,\n isDefault: true,\n },\n });\n}\n"]}
@@ -0,0 +1,124 @@
1
+ // src/rbac/role-service.ts
2
+ async function getRolesData(db, params = {}) {
3
+ const { page = 1, pageSize = 10, search, status } = params;
4
+ const whereConditions = [];
5
+ if (search) {
6
+ whereConditions.push({
7
+ OR: [
8
+ { name: { contains: search, mode: "insensitive" } },
9
+ { code: { contains: search, mode: "insensitive" } },
10
+ { description: { contains: search, mode: "insensitive" } }
11
+ ]
12
+ });
13
+ }
14
+ if (status && status !== "all") {
15
+ whereConditions.push({ status });
16
+ }
17
+ const where = whereConditions.length > 0 ? { AND: whereConditions } : {};
18
+ const [total, items] = await Promise.all([
19
+ db.role.count({ where }),
20
+ db.role.findMany({
21
+ where,
22
+ orderBy: { createdAt: "desc" },
23
+ skip: (page - 1) * pageSize,
24
+ take: pageSize,
25
+ include: {
26
+ userRoles: {
27
+ include: {
28
+ user: {
29
+ select: {
30
+ id: true,
31
+ name: true,
32
+ email: true,
33
+ image: true,
34
+ isActive: true
35
+ }
36
+ }
37
+ }
38
+ },
39
+ rolePermissions: {
40
+ include: {
41
+ resource: {
42
+ select: {
43
+ code: true,
44
+ name: true,
45
+ icon: true
46
+ }
47
+ },
48
+ action: {
49
+ select: {
50
+ code: true,
51
+ name: true
52
+ }
53
+ }
54
+ }
55
+ }
56
+ }
57
+ })
58
+ ]);
59
+ const transformedItems = items.map((role) => ({
60
+ id: role.id,
61
+ name: role.name,
62
+ code: role.code,
63
+ description: role.description || void 0,
64
+ status: role.status,
65
+ // Transform rolePermissions to array of "action:resource" strings
66
+ permissions: role.rolePermissions.map(
67
+ (rp) => `${rp.actionCode}:${rp.resourceCode}`
68
+ ),
69
+ usersCount: role.userRoles.length,
70
+ users: role.userRoles.map((ur) => ({
71
+ id: ur.user.id,
72
+ name: ur.user.name,
73
+ email: ur.user.email,
74
+ image: ur.user.image,
75
+ isActive: ur.user.isActive
76
+ })),
77
+ createdAt: role.createdAt.toISOString(),
78
+ updatedAt: role.updatedAt.toISOString(),
79
+ createdBy: role.createdBy || void 0,
80
+ updatedBy: role.updatedBy || void 0
81
+ }));
82
+ return { total, page, pageSize, items: transformedItems };
83
+ }
84
+
85
+ // src/rbac/resource-service.ts
86
+ async function getResourcesForPermissionMatrix(db) {
87
+ return await db.resource.findMany({
88
+ where: {
89
+ status: "active"
90
+ },
91
+ orderBy: [{ group: "asc" }, { order: "asc" }, { name: "asc" }],
92
+ select: {
93
+ id: true,
94
+ code: true,
95
+ name: true,
96
+ group: true,
97
+ description: true,
98
+ icon: true,
99
+ order: true,
100
+ config: true
101
+ }
102
+ });
103
+ }
104
+ async function getActionsForPermissionMatrix(db) {
105
+ return await db.action.findMany({
106
+ where: {
107
+ status: "active"
108
+ },
109
+ orderBy: {
110
+ code: "asc"
111
+ },
112
+ select: {
113
+ id: true,
114
+ code: true,
115
+ name: true,
116
+ description: true,
117
+ isDefault: true
118
+ }
119
+ });
120
+ }
121
+
122
+ export { getActionsForPermissionMatrix, getResourcesForPermissionMatrix, getRolesData };
123
+ //# sourceMappingURL=server.mjs.map
124
+ //# sourceMappingURL=server.mjs.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../../src/rbac/role-service.ts","../../src/rbac/resource-service.ts"],"names":[],"mappings":"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","file":"server.mjs","sourcesContent":["// Note: We use Prisma types for input construction, but we don't import the client instance.\n// If Prisma types are not available in core (since core doesn't have prisma schema),\n// we might need to define loose types or rely on the injected client's type inference if strict mode allows.\n// However, since we want to be Type-safe, we assume the host app uses standard Prisma conventions.\n// For core, we can define the expected structure of the Include/Where types if needed,\n// or simply use 'any' for the complex Prisma types if we want to avoid strict coupling to a specific schema version.\n// To be safe and generic in 'core', we will define the input params using standard TypeScript types\n// and the db client interface locally.\n\n// ============================================================================\n// Types\n// ============================================================================\n\nexport type RoleFilters = {\n page?: number;\n pageSize?: number;\n search?: string;\n status?: string;\n};\n\nexport type RoleData = {\n id: string;\n name: string;\n code: string;\n description?: string;\n status: string;\n permissions: string[];\n usersCount: number;\n users: {\n id: string;\n name: string | null;\n email: string | null;\n image: string | null;\n isActive: boolean;\n }[];\n createdAt: string;\n updatedAt: string;\n createdBy?: string;\n updatedBy?: string;\n};\n\n// Interface for the DB client injected into the service\n// We define the shape of the Prisma Client we expect\nexport interface RolePrismaClient {\n role: {\n count: (args: { where: any }) => Promise<number>;\n findMany: (args: {\n where: any;\n orderBy: any;\n skip: number;\n take: number;\n include: any;\n }) => Promise<any[]>;\n };\n}\n\n// ============================================================================\n// Service Logic\n// ============================================================================\n\n/**\n * Get roles data with pagination and filtering\n * Generic core implementation using injected DB client\n *\n * @param db - The tenant-specific Prisma Client instance\n * @param params - Filter and pagination parameters\n */\nexport async function getRolesData(\n db: RolePrismaClient,\n params: RoleFilters = {},\n): Promise<{\n total: number;\n page: number;\n pageSize: number;\n items: RoleData[];\n}> {\n const { page = 1, pageSize = 10, search, status } = params;\n\n const whereConditions: any[] = [];\n\n if (search) {\n whereConditions.push({\n OR: [\n { name: { contains: search, mode: \"insensitive\" } },\n { code: { contains: search, mode: \"insensitive\" } },\n { description: { contains: search, mode: \"insensitive\" } },\n ],\n });\n }\n\n if (status && status !== \"all\") {\n whereConditions.push({ status });\n }\n\n const where: any = whereConditions.length > 0 ? { AND: whereConditions } : {};\n\n const [total, items] = await Promise.all([\n db.role.count({ where }),\n db.role.findMany({\n where,\n orderBy: { createdAt: \"desc\" },\n skip: (page - 1) * pageSize,\n take: pageSize,\n include: {\n userRoles: {\n include: {\n user: {\n select: {\n id: true,\n name: true,\n email: true,\n image: true,\n isActive: true,\n },\n },\n },\n },\n rolePermissions: {\n include: {\n resource: {\n select: {\n code: true,\n name: true,\n icon: true,\n },\n },\n action: {\n select: {\n code: true,\n name: true,\n },\n },\n },\n },\n },\n }),\n ]);\n\n // Transform raw DB items to RoleData type\n const transformedItems = items.map((role: any) => ({\n id: role.id,\n name: role.name,\n code: role.code,\n description: role.description || undefined,\n status: role.status,\n // Transform rolePermissions to array of \"action:resource\" strings\n permissions: role.rolePermissions.map(\n (rp: any) => `${rp.actionCode}:${rp.resourceCode}`,\n ),\n usersCount: role.userRoles.length,\n users: role.userRoles.map((ur: any) => ({\n id: ur.user.id,\n name: ur.user.name,\n email: ur.user.email,\n image: ur.user.image,\n isActive: ur.user.isActive,\n })),\n createdAt: role.createdAt.toISOString(),\n updatedAt: role.updatedAt.toISOString(),\n createdBy: role.createdBy || undefined,\n updatedBy: role.updatedBy || undefined,\n }));\n\n return { total, page, pageSize, items: transformedItems };\n}\n","export interface ResourcePrismaClient {\n resource: any;\n action: any;\n}\n\n/**\n * Get resources data with pagination and filtering\n * Shared function for both API route and Server Component\n *\n * @param db - The tenant-specific Prisma Client instance\n * @param params - Filter and pagination parameters\n */\nexport async function getResourcesData(\n db: ResourcePrismaClient,\n params: any = {},\n) {\n const { page = 1, pageSize = 10, search, status, group, type } = params;\n\n const where: any = {\n AND: [\n search\n ? {\n OR: [\n { code: { contains: search, mode: \"insensitive\" } },\n { name: { contains: search, mode: \"insensitive\" } },\n { description: { contains: search, mode: \"insensitive\" } },\n { path: { contains: search, mode: \"insensitive\" } },\n ],\n }\n : {},\n status && status !== \"all\"\n ? { status: { equals: status, mode: \"insensitive\" } }\n : {},\n group && group !== \"all\" ? { group } : {},\n type && type !== \"all\" ? { type } : {},\n ],\n };\n\n const [total, items] = await Promise.all([\n db.resource.count({ where }),\n db.resource.findMany({\n where,\n orderBy: [{ order: \"asc\" }, { updatedAt: \"desc\" }],\n skip: (page - 1) * pageSize,\n take: pageSize,\n }),\n ]);\n\n // Transform Prisma items to Resource type format\n const transformedItems = items.map((item: any) => ({\n id: item.id,\n code: item.code,\n name: item.name,\n group: item.group || undefined,\n description: item.description || undefined,\n status: item.status,\n parentCode: item.parentCode || undefined,\n order: item.order || undefined,\n type: item.type || undefined,\n icon: item.icon || undefined,\n path: item.path || undefined,\n config: item.config || undefined,\n createdAt: item.createdAt.toISOString(),\n updatedAt: item.updatedAt.toISOString(),\n }));\n\n return { total, page, pageSize, items: transformedItems };\n}\n\n/**\n * Get all active resources for permission matrix\n * Used in role management to build dynamic permission matrix\n */\nexport async function getResourcesForPermissionMatrix(\n db: ResourcePrismaClient,\n) {\n return await db.resource.findMany({\n where: {\n status: \"active\",\n },\n orderBy: [{ group: \"asc\" }, { order: \"asc\" }, { name: \"asc\" }],\n select: {\n id: true,\n code: true,\n name: true,\n group: true,\n description: true,\n icon: true,\n order: true,\n config: true,\n },\n });\n}\n\n/**\n * Get all active actions for permission matrix\n * Used in role management to build dynamic permission matrix\n */\nexport async function getActionsForPermissionMatrix(db: ResourcePrismaClient) {\n return await db.action.findMany({\n where: {\n status: \"active\",\n },\n orderBy: {\n code: \"asc\",\n },\n select: {\n id: true,\n code: true,\n name: true,\n description: true,\n isDefault: true,\n },\n });\n}\n"]}