@jmruthers/pace-core 0.2.7 → 0.5.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 (541) hide show
  1. package/dist/{DataTable-EEUDXPE5.js → DataTable-GX3XERFJ.js} +8 -4
  2. package/dist/{DataTable-C1AEm9Cx.d.ts → DataTable-ltTFXHS3.d.ts} +3 -1
  3. package/dist/{chunk-VYG4AXYW.js → chunk-5EL3KHOQ.js} +2 -2
  4. package/dist/{chunk-ETEJVKYK.js → chunk-6CR3MRZN.js} +1426 -62
  5. package/dist/chunk-6CR3MRZN.js.map +1 -0
  6. package/dist/chunk-AUE24LVR.js +268 -0
  7. package/dist/chunk-AUE24LVR.js.map +1 -0
  8. package/dist/chunk-COBPIXXQ.js +379 -0
  9. package/dist/chunk-COBPIXXQ.js.map +1 -0
  10. package/dist/{chunk-EWKPTNPO.js → chunk-GSNM5D6H.js} +388 -86
  11. package/dist/chunk-GSNM5D6H.js.map +1 -0
  12. package/dist/{chunk-2V3Y6YBC.js → chunk-OEGRKULD.js} +1 -42
  13. package/dist/chunk-OEGRKULD.js.map +1 -0
  14. package/dist/chunk-OYRY44Q2.js +62 -0
  15. package/dist/chunk-OYRY44Q2.js.map +1 -0
  16. package/dist/{chunk-RRUYHORU.js → chunk-T3XIA4AJ.js} +297 -433
  17. package/dist/chunk-T3XIA4AJ.js.map +1 -0
  18. package/dist/{chunk-HEMJ4SUJ.js → chunk-TGDCLPP2.js} +11 -7
  19. package/dist/{chunk-HEMJ4SUJ.js.map → chunk-TGDCLPP2.js.map} +1 -1
  20. package/dist/{chunk-HNDFPXUU.js → chunk-U6JDHVC2.js} +6 -4
  21. package/dist/{chunk-HNDFPXUU.js.map → chunk-U6JDHVC2.js.map} +1 -1
  22. package/dist/{chunk-TIVL4UQ7.js → chunk-XJK2J4N6.js} +6 -4
  23. package/dist/{chunk-TIVL4UQ7.js.map → chunk-XJK2J4N6.js.map} +1 -1
  24. package/dist/components.d.ts +2 -2
  25. package/dist/components.js +21 -20
  26. package/dist/components.js.map +1 -1
  27. package/dist/hooks.d.ts +1 -1
  28. package/dist/hooks.js +7 -7
  29. package/dist/index.d.ts +2 -2
  30. package/dist/index.js +26 -25
  31. package/dist/index.js.map +1 -1
  32. package/dist/providers.js +8 -7
  33. package/dist/rbac/index.d.ts +806 -806
  34. package/dist/rbac/index.js +937 -1179
  35. package/dist/rbac/index.js.map +1 -1
  36. package/dist/{types-DiRQsGJs.d.ts → types-BRDU7N6w.d.ts} +12 -1
  37. package/dist/utils.d.ts +2 -2
  38. package/dist/utils.js +6 -6
  39. package/docs/api/classes/ErrorBoundary.md +1 -1
  40. package/docs/api/classes/PublicErrorBoundary.md +1 -1
  41. package/docs/api/interfaces/AggregateConfig.md +4 -4
  42. package/docs/api/interfaces/ButtonProps.md +1 -1
  43. package/docs/api/interfaces/CardProps.md +1 -1
  44. package/docs/api/interfaces/ColorPalette.md +1 -1
  45. package/docs/api/interfaces/ColorShade.md +1 -1
  46. package/docs/api/interfaces/DataTableAction.md +21 -8
  47. package/docs/api/interfaces/DataTableColumn.md +1 -1
  48. package/docs/api/interfaces/DataTableProps.md +46 -33
  49. package/docs/api/interfaces/DataTableToolbarButton.md +7 -7
  50. package/docs/api/interfaces/EmptyStateConfig.md +5 -5
  51. package/docs/api/interfaces/EventContextType.md +1 -1
  52. package/docs/api/interfaces/EventLogoProps.md +1 -1
  53. package/docs/api/interfaces/EventProviderProps.md +1 -1
  54. package/docs/api/interfaces/FileSizeLimits.md +1 -1
  55. package/docs/api/interfaces/FileUploadProps.md +1 -1
  56. package/docs/api/interfaces/FooterProps.md +1 -1
  57. package/docs/api/interfaces/InactivityWarningModalProps.md +1 -1
  58. package/docs/api/interfaces/InputProps.md +1 -1
  59. package/docs/api/interfaces/LabelProps.md +1 -1
  60. package/docs/api/interfaces/LoginFormProps.md +1 -1
  61. package/docs/api/interfaces/NavigationItem.md +1 -1
  62. package/docs/api/interfaces/NavigationMenuProps.md +1 -1
  63. package/docs/api/interfaces/Organisation.md +1 -1
  64. package/docs/api/interfaces/OrganisationContextType.md +1 -1
  65. package/docs/api/interfaces/OrganisationMembership.md +2 -2
  66. package/docs/api/interfaces/OrganisationProviderProps.md +1 -1
  67. package/docs/api/interfaces/OrganisationSecurityError.md +1 -1
  68. package/docs/api/interfaces/PaceAppLayoutProps.md +1 -1
  69. package/docs/api/interfaces/PaceLoginPageProps.md +1 -1
  70. package/docs/api/interfaces/PaletteData.md +1 -1
  71. package/docs/api/interfaces/PublicErrorBoundaryProps.md +1 -1
  72. package/docs/api/interfaces/PublicErrorBoundaryState.md +1 -1
  73. package/docs/api/interfaces/PublicLoadingSpinnerProps.md +1 -1
  74. package/docs/api/interfaces/PublicPageFooterProps.md +1 -1
  75. package/docs/api/interfaces/PublicPageHeaderProps.md +1 -1
  76. package/docs/api/interfaces/PublicPageLayoutProps.md +1 -1
  77. package/docs/api/interfaces/StorageConfig.md +1 -1
  78. package/docs/api/interfaces/StorageFileInfo.md +1 -1
  79. package/docs/api/interfaces/StorageFileMetadata.md +1 -1
  80. package/docs/api/interfaces/StorageListOptions.md +1 -1
  81. package/docs/api/interfaces/StorageListResult.md +1 -1
  82. package/docs/api/interfaces/StorageUploadOptions.md +1 -1
  83. package/docs/api/interfaces/StorageUploadResult.md +1 -1
  84. package/docs/api/interfaces/StorageUrlOptions.md +1 -1
  85. package/docs/api/interfaces/StyleImport.md +1 -1
  86. package/docs/api/interfaces/ToastActionElement.md +1 -1
  87. package/docs/api/interfaces/ToastProps.md +1 -1
  88. package/docs/api/interfaces/UnifiedAuthContextType.md +1 -1
  89. package/docs/api/interfaces/UnifiedAuthProviderProps.md +1 -1
  90. package/docs/api/interfaces/UseInactivityTrackerOptions.md +1 -1
  91. package/docs/api/interfaces/UseInactivityTrackerReturn.md +1 -1
  92. package/docs/api/interfaces/UsePublicEventLogoOptions.md +1 -1
  93. package/docs/api/interfaces/UsePublicEventLogoReturn.md +1 -1
  94. package/docs/api/interfaces/UsePublicEventOptions.md +1 -1
  95. package/docs/api/interfaces/UsePublicEventReturn.md +1 -1
  96. package/docs/api/interfaces/UsePublicRouteParamsReturn.md +1 -1
  97. package/docs/api/interfaces/UserEventAccess.md +1 -1
  98. package/docs/api/interfaces/UserMenuProps.md +1 -1
  99. package/docs/api/interfaces/UserProfile.md +1 -1
  100. package/docs/api/modules.md +3 -3
  101. package/package.json +5 -2
  102. package/src/__tests__/REBUILD_PLAN.md +223 -0
  103. package/src/__tests__/TESTING_GUIDELINES.md +341 -0
  104. package/src/__tests__/fixtures/mocks.ts +93 -0
  105. package/src/__tests__/helpers/component-test-utils.tsx +145 -0
  106. package/src/__tests__/helpers/test-utils.tsx +117 -0
  107. package/src/__tests__/integration/UserProfile.test.tsx +128 -0
  108. package/src/__tests__/setup.ts +37 -225
  109. package/src/__tests__/templates/component.test.template.tsx +97 -75
  110. package/src/__tests__/templates/hook.test.template.ts +173 -0
  111. package/src/__tests__/types/test.types.ts +106 -0
  112. package/src/components/Alert/Alert.test.tsx +496 -0
  113. package/src/components/Avatar/Avatar.test.tsx +484 -0
  114. package/src/components/Button/Button.test.tsx +662 -0
  115. package/src/components/Card/Card.test.tsx +593 -0
  116. package/src/components/Checkbox/Checkbox.test.tsx +461 -0
  117. package/src/components/DataTable/DataTable.tsx +9 -1
  118. package/src/components/DataTable/components/AccessDeniedPage.tsx +168 -0
  119. package/src/components/DataTable/components/ActionButtons.tsx +18 -1
  120. package/src/components/DataTable/components/DataTableCore.tsx +97 -11
  121. package/src/components/DataTable/components/DataTableToolbar.tsx +22 -10
  122. package/src/components/DataTable/components/UnifiedTableBody.tsx +33 -4
  123. package/src/components/DataTable/examples/HierarchicalActionsExample.tsx +1 -0
  124. package/src/components/DataTable/examples/HierarchicalExample.tsx +3 -0
  125. package/src/components/DataTable/examples/InitialPageSizeExample.tsx +3 -0
  126. package/src/components/DataTable/examples/PerformanceExample.tsx +3 -0
  127. package/src/components/DataTable/types.ts +39 -1
  128. package/src/components/Dialog/Dialog.test.tsx +1139 -0
  129. package/src/components/ErrorBoundary/ErrorBoundary.test.tsx +752 -0
  130. package/src/components/FileUpload/FileUpload.test.tsx +665 -0
  131. package/src/hooks/useCounter.test.ts +135 -0
  132. package/src/rbac/index.ts +3 -3
  133. package/dist/chunk-2V3Y6YBC.js.map +0 -1
  134. package/dist/chunk-BEZRLNK3.js +0 -1744
  135. package/dist/chunk-BEZRLNK3.js.map +0 -1
  136. package/dist/chunk-ETEJVKYK.js.map +0 -1
  137. package/dist/chunk-EWKPTNPO.js.map +0 -1
  138. package/dist/chunk-OHXGNT3K.js +0 -21
  139. package/dist/chunk-OHXGNT3K.js.map +0 -1
  140. package/dist/chunk-RRUYHORU.js.map +0 -1
  141. package/src/__tests__/README.md +0 -404
  142. package/src/__tests__/debug-provider.unit.test.tsx +0 -67
  143. package/src/__tests__/e2e/workflows.test.tsx +0 -373
  144. package/src/__tests__/hybridPermissions.unit.test.tsx +0 -474
  145. package/src/__tests__/index.integration.test.ts +0 -491
  146. package/src/__tests__/mocks/MockAuthProvider-standalone.tsx +0 -47
  147. package/src/__tests__/mocks/MockAuthProvider.tsx +0 -63
  148. package/src/__tests__/mocks/enhancedSupabaseMock.ts +0 -252
  149. package/src/__tests__/mocks/index.test.ts +0 -23
  150. package/src/__tests__/mocks/index.ts +0 -16
  151. package/src/__tests__/mocks/mockAuth.ts +0 -155
  152. package/src/__tests__/mocks/mockSupabase.ts +0 -83
  153. package/src/__tests__/mocks/mockSupabaseClient.ts +0 -63
  154. package/src/__tests__/mocks/providers.tsx +0 -22
  155. package/src/__tests__/patterns/__tests__/testPatterns.test.ts +0 -394
  156. package/src/__tests__/patterns/testPatterns.ts +0 -124
  157. package/src/__tests__/performance/componentPerformance.performance.test.ts +0 -27
  158. package/src/__tests__/performance/index.ts +0 -24
  159. package/src/__tests__/performance/performanceValidation.performance.test.ts +0 -15
  160. package/src/__tests__/security/security.unit.test.tsx +0 -7
  161. package/src/__tests__/security/securityValidation.security.test.tsx +0 -153
  162. package/src/__tests__/setupTests.d.ts +0 -1
  163. package/src/__tests__/shared/componentTestUtils.tsx +0 -475
  164. package/src/__tests__/shared/errorHandlingTestUtils.ts +0 -107
  165. package/src/__tests__/shared/index.ts +0 -81
  166. package/src/__tests__/shared/integrationTestUtils.tsx +0 -375
  167. package/src/__tests__/shared/performanceTestUtils.tsx +0 -476
  168. package/src/__tests__/shared/testUtils.optimized.tsx +0 -685
  169. package/src/__tests__/simple.test.tsx +0 -20
  170. package/src/__tests__/test-utils/dataFactories.ts +0 -60
  171. package/src/__tests__/test-utils/index.ts +0 -6
  172. package/src/__tests__/typeSafety.unit.test.ts +0 -65
  173. package/src/__tests__/unifiedAuth.unit.test.tsx +0 -151
  174. package/src/__tests__/utils/accessibilityHelpers.ts +0 -254
  175. package/src/__tests__/utils/assertions.ts +0 -50
  176. package/src/__tests__/utils/deterministicHelpers.ts +0 -31
  177. package/src/__tests__/utils/edgeCaseConfig.test.ts +0 -75
  178. package/src/__tests__/utils/edgeCaseConfig.ts +0 -98
  179. package/src/__tests__/utils/mockHelpers.ts +0 -149
  180. package/src/__tests__/utils/mockLoader.ts +0 -101
  181. package/src/__tests__/utils/performanceHelpers.ts +0 -55
  182. package/src/__tests__/utils/performanceTestHelpers.ts +0 -68
  183. package/src/__tests__/utils/testDataFactories.ts +0 -28
  184. package/src/__tests__/utils/testIsolation.ts +0 -67
  185. package/src/__tests__/utils/visualTestHelpers.ts +0 -20
  186. package/src/__tests__/visual/__snapshots__/componentSnapshots.visual.test.tsx.snap +0 -68
  187. package/src/__tests__/visual/__snapshots__/componentVisuals.visual.test.tsx.snap +0 -14
  188. package/src/__tests__/visual/__snapshots__/visualRegression.test.tsx.snap +0 -217
  189. package/src/__tests__/visual/__snapshots__/visualRegression.visual.test.tsx.snap +0 -24
  190. package/src/__tests__/visual/componentSnapshots.visual.test.tsx +0 -33
  191. package/src/__tests__/visual/componentVisuals.visual.test.tsx +0 -12
  192. package/src/__tests__/visual/visualRegression.visual.test.tsx +0 -20
  193. package/src/components/Alert/__tests__/Alert.unit.test.tsx +0 -381
  194. package/src/components/Avatar/__tests__/Avatar.unit.test.tsx +0 -232
  195. package/src/components/Button/__tests__/Button.accessibility.test.tsx +0 -131
  196. package/src/components/Button/__tests__/Button.comprehensive.test.tsx +0 -721
  197. package/src/components/Button/__tests__/Button.unit.test.tsx +0 -189
  198. package/src/components/Button/__tests__/EventSelector.integration.test.tsx +0 -285
  199. package/src/components/Card/__tests__/Card.accessibility.test.tsx +0 -394
  200. package/src/components/Card/__tests__/Card.comprehensive.test.tsx +0 -599
  201. package/src/components/Card/__tests__/Card.integration.test.tsx +0 -673
  202. package/src/components/Card/__tests__/Card.performance.test.tsx +0 -546
  203. package/src/components/Card/__tests__/Card.unit.test.tsx +0 -330
  204. package/src/components/Card/__tests__/Card.visual.test.tsx +0 -599
  205. package/src/components/Card/__tests__/README.md +0 -211
  206. package/src/components/Checkbox/__tests__/Checkbox.unit.test.tsx +0 -520
  207. package/src/components/DataTable/__tests__/DataTable.errorHandling.test.tsx +0 -251
  208. package/src/components/DataTable/__tests__/DataTable.hierarchical.test.tsx +0 -680
  209. package/src/components/DataTable/__tests__/DataTable.infinite-loop.test.tsx +0 -323
  210. package/src/components/DataTable/__tests__/DataTable.integration.test.tsx +0 -716
  211. package/src/components/DataTable/__tests__/DataTable.performance.test.tsx +0 -589
  212. package/src/components/DataTable/__tests__/DataTable.permissions.test.tsx +0 -316
  213. package/src/components/DataTable/__tests__/DataTable.regressionFixes.test.tsx +0 -546
  214. package/src/components/DataTable/__tests__/DataTable.selection.controlled.test.tsx +0 -386
  215. package/src/components/DataTable/__tests__/DataTable.selection.test.tsx +0 -338
  216. package/src/components/DataTable/__tests__/DataTable.sorting.test.tsx +0 -321
  217. package/src/components/DataTable/__tests__/DataTable.userWorkflows.test.tsx +0 -320
  218. package/src/components/DataTable/__tests__/DataTable.workflowValidation.test.tsx +0 -583
  219. package/src/components/DataTable/__tests__/DataTable.workflows.test.tsx +0 -711
  220. package/src/components/DataTable/__tests__/performance-regression.test.tsx +0 -777
  221. package/src/components/DataTable/__tests__/performance.test.tsx +0 -365
  222. package/src/components/DataTable/components/__tests__/ActionButtons.unit.test.tsx +0 -150
  223. package/src/components/DataTable/components/__tests__/BulkOperationsDropdown.test.tsx +0 -224
  224. package/src/components/DataTable/components/__tests__/ColumnVisibilityDropdown.unit.test.tsx +0 -244
  225. package/src/components/DataTable/components/__tests__/DataTable.accessibility.test.tsx +0 -629
  226. package/src/components/DataTable/components/__tests__/DataTable.integration.test.tsx +0 -470
  227. package/src/components/DataTable/components/__tests__/DataTable.performance.test.tsx +0 -160
  228. package/src/components/DataTable/components/__tests__/DataTable.real.test.tsx +0 -251
  229. package/src/components/DataTable/components/__tests__/DataTable.security.test.tsx +0 -171
  230. package/src/components/DataTable/components/__tests__/DataTable.unit.test.tsx +0 -290
  231. package/src/components/DataTable/components/__tests__/DataTableBody.unit.test.tsx +0 -147
  232. package/src/components/DataTable/components/__tests__/DataTableErrorBoundary.unit.test.tsx +0 -182
  233. package/src/components/DataTable/components/__tests__/DataTableModals.unit.test.tsx +0 -123
  234. package/src/components/DataTable/components/__tests__/EditableRow.unit.test.tsx +0 -660
  235. package/src/components/DataTable/components/__tests__/EmptyState.unit.test.tsx +0 -256
  236. package/src/components/DataTable/components/__tests__/ExpandButton.test.tsx +0 -498
  237. package/src/components/DataTable/components/__tests__/FilterRow.unit.test.tsx +0 -112
  238. package/src/components/DataTable/components/__tests__/FilteringToggle.unit.test.tsx +0 -133
  239. package/src/components/DataTable/components/__tests__/GroupHeader.unit.test.tsx +0 -172
  240. package/src/components/DataTable/components/__tests__/GroupingDropdown.unit.test.tsx +0 -222
  241. package/src/components/DataTable/components/__tests__/ImportModal.unit.test.tsx +0 -780
  242. package/src/components/DataTable/components/__tests__/LoadingState.unit.test.tsx +0 -65
  243. package/src/components/DataTable/components/__tests__/PaginationControls.unit.test.tsx +0 -634
  244. package/src/components/DataTable/components/__tests__/StateComponents.unit.test.tsx +0 -48
  245. package/src/components/DataTable/components/__tests__/UnifiedTableBody.hierarchical.test.tsx +0 -541
  246. package/src/components/DataTable/components/__tests__/ViewRowModal.unit.test.tsx +0 -228
  247. package/src/components/DataTable/components/__tests__/VirtualizedDataTable.unit.test.tsx +0 -568
  248. package/src/components/DataTable/core/__tests__/ActionManager.unit.test.ts +0 -405
  249. package/src/components/DataTable/core/__tests__/ArchitectureIntegration.unit.test.tsx +0 -445
  250. package/src/components/DataTable/core/__tests__/ColumnFactory.unit.test.ts +0 -288
  251. package/src/components/DataTable/core/__tests__/ColumnManager.unit.test.ts +0 -623
  252. package/src/components/DataTable/core/__tests__/DataManager.unit.test.ts +0 -431
  253. package/src/components/DataTable/core/__tests__/DataTableContext.unit.test.tsx +0 -433
  254. package/src/components/DataTable/core/__tests__/LocalDataAdapter.unit.test.ts +0 -422
  255. package/src/components/DataTable/core/__tests__/PluginRegistry.unit.test.tsx +0 -207
  256. package/src/components/DataTable/core/__tests__/StateManager.unit.test.ts +0 -278
  257. package/src/components/DataTable/examples/__tests__/PerformanceExample.unit.test.tsx +0 -281
  258. package/src/components/DataTable/hooks/__tests__/useColumnOrderPersistence.unit.test.ts +0 -407
  259. package/src/components/DataTable/hooks/__tests__/useColumnReordering.unit.test.ts +0 -679
  260. package/src/components/DataTable/utils/__tests__/debugTools.unit.test.ts +0 -267
  261. package/src/components/DataTable/utils/__tests__/errorHandling.unit.test.ts +0 -467
  262. package/src/components/DataTable/utils/__tests__/exportUtils.unit.test.ts +0 -380
  263. package/src/components/DataTable/utils/__tests__/flexibleImport.unit.test.ts +0 -233
  264. package/src/components/DataTable/utils/__tests__/performanceUtils.unit.test.ts +0 -414
  265. package/src/components/Dialog/__tests__/Dialog.accessibility.test.tsx +0 -521
  266. package/src/components/Dialog/__tests__/Dialog.auto-size.example.tsx +0 -157
  267. package/src/components/Dialog/__tests__/Dialog.enhanced.test.tsx +0 -538
  268. package/src/components/Dialog/__tests__/Dialog.unit.test.tsx +0 -1373
  269. package/src/components/Dialog/examples/__tests__/SmartDialogExample.unit.test.tsx +0 -151
  270. package/src/components/Dialog/utils/__tests__/safeHtml.unit.test.ts +0 -611
  271. package/src/components/ErrorBoundary/__tests__/ErrorBoundary.accessibility.test.tsx +0 -517
  272. package/src/components/ErrorBoundary/__tests__/ErrorBoundary.integration.test.tsx +0 -572
  273. package/src/components/ErrorBoundary/__tests__/ErrorBoundary.unit.test.tsx +0 -579
  274. package/src/components/EventSelector/__tests__/EventSelector.test.tsx +0 -528
  275. package/src/components/FileUpload/__tests__/FileUpload.integration.test.tsx +0 -992
  276. package/src/components/FileUpload/__tests__/FileUpload.real.test.tsx +0 -927
  277. package/src/components/FileUpload/__tests__/FileUpload.test.tsx +0 -855
  278. package/src/components/FileUpload/__tests__/FileUpload.unit.test.tsx +0 -1311
  279. package/src/components/FileUpload/__tests__/FileUpload.unmocked.test.tsx +0 -937
  280. package/src/components/Footer/__tests__/Footer.accessibility.test.tsx +0 -359
  281. package/src/components/Footer/__tests__/Footer.integration.test.tsx +0 -353
  282. package/src/components/Footer/__tests__/Footer.performance.test.tsx +0 -309
  283. package/src/components/Footer/__tests__/Footer.unit.test.tsx +0 -309
  284. package/src/components/Footer/__tests__/Footer.visual.test.tsx +0 -335
  285. package/src/components/Form/__tests__/Form.accessibility.test.tsx +0 -820
  286. package/src/components/Form/__tests__/Form.unit.test.tsx +0 -305
  287. package/src/components/Form/__tests__/FormErrorSummary.unit.test.tsx +0 -285
  288. package/src/components/Form/__tests__/FormFieldset.unit.test.tsx +0 -241
  289. package/src/components/Header/__tests__/Header.accessibility.test.tsx +0 -382
  290. package/src/components/Header/__tests__/Header.comprehensive.test.tsx +0 -509
  291. package/src/components/Header/__tests__/Header.unit.test.tsx +0 -335
  292. package/src/components/InactivityWarningModal/InactivityWarningModal.test.tsx +0 -196
  293. package/src/components/InactivityWarningModal/__tests__/InactivityWarningModal.unit.test.tsx +0 -224
  294. package/src/components/Input/__tests__/Input.accessibility.test.tsx +0 -632
  295. package/src/components/Input/__tests__/Input.unit.test.tsx +0 -1121
  296. package/src/components/Label/__tests__/Label.accessibility.test.tsx +0 -239
  297. package/src/components/Label/__tests__/Label.unit.test.tsx +0 -331
  298. package/src/components/LoadingSpinner/__tests__/LoadingSpinner.accessibility.test.tsx +0 -116
  299. package/src/components/LoadingSpinner/__tests__/LoadingSpinner.unit.test.tsx +0 -144
  300. package/src/components/LoginForm/__tests__/LoginForm.accessibility.test.tsx +0 -201
  301. package/src/components/LoginForm/__tests__/LoginForm.unit.test.tsx +0 -119
  302. package/src/components/NavigationMenu/__tests__/NavigationMenu.accessibility.test.tsx +0 -378
  303. package/src/components/NavigationMenu/__tests__/NavigationMenu.enhanced.test.tsx +0 -768
  304. package/src/components/NavigationMenu/__tests__/NavigationMenu.integration.test.tsx +0 -576
  305. package/src/components/NavigationMenu/__tests__/NavigationMenu.performance.test.tsx +0 -585
  306. package/src/components/NavigationMenu/__tests__/NavigationMenu.real.component.test.tsx +0 -783
  307. package/src/components/NavigationMenu/__tests__/NavigationMenu.security.enhanced.test.tsx +0 -810
  308. package/src/components/NavigationMenu/__tests__/NavigationMenu.security.test.tsx +0 -494
  309. package/src/components/NavigationMenu/__tests__/NavigationMenu.unit.test.tsx +0 -331
  310. package/src/components/NavigationMenu/__tests__/NavigationMenu.userWorkflows.test.tsx +0 -347
  311. package/src/components/NavigationMenu/__tests__/NavigationMenu.workflows.test.tsx +0 -584
  312. package/src/components/OrganisationSelector/__tests__/OrganisationSelector.unit.test.tsx +0 -664
  313. package/src/components/PaceAppLayout/__tests__/PaceAppLayout.accessibility.test.tsx +0 -288
  314. package/src/components/PaceAppLayout/__tests__/PaceAppLayout.integration.test.tsx +0 -893
  315. package/src/components/PaceAppLayout/__tests__/PaceAppLayout.performance.test.tsx +0 -629
  316. package/src/components/PaceAppLayout/__tests__/PaceAppLayout.security.test.tsx +0 -782
  317. package/src/components/PaceAppLayout/__tests__/PaceAppLayout.unit.test.tsx +0 -904
  318. package/src/components/PaceLoginPage/__tests__/PaceLoginPage.accessibility.test.tsx +0 -463
  319. package/src/components/PaceLoginPage/__tests__/PaceLoginPage.integration.test.tsx +0 -586
  320. package/src/components/PaceLoginPage/__tests__/PaceLoginPage.unit.test.tsx +0 -533
  321. package/src/components/PasswordReset/__tests__/PasswordChangeForm.accessibility.test.tsx +0 -408
  322. package/src/components/PasswordReset/__tests__/PasswordChangeForm.unit.test.tsx +0 -561
  323. package/src/components/PasswordReset/__tests__/PasswordReset.integration.test.tsx +0 -304
  324. package/src/components/PasswordReset/__tests__/PasswordResetForm.accessibility.test.tsx +0 -20
  325. package/src/components/PasswordReset/__tests__/PasswordResetForm.unit.test.tsx +0 -523
  326. package/src/components/PasswordReset/__tests__/__mocks__/UnifiedAuthProvider.ts +0 -29
  327. package/src/components/Print/__tests__/Print.comprehensive.test.tsx +0 -331
  328. package/src/components/PrintButton/__tests__/PrintButton.unit.test.tsx +0 -429
  329. package/src/components/PrintButton/__tests__/PrintButtonGroup.unit.test.tsx +0 -277
  330. package/src/components/PrintButton/__tests__/PrintToolbar.unit.test.tsx +0 -264
  331. package/src/components/PrintCard/__tests__/PrintCard.unit.test.tsx +0 -233
  332. package/src/components/PrintCard/__tests__/PrintCardContent.test.tsx +0 -284
  333. package/src/components/PrintCard/__tests__/PrintCardGrid.unit.test.tsx +0 -214
  334. package/src/components/PrintCard/__tests__/PrintCardImage.unit.test.tsx +0 -264
  335. package/src/components/PrintDataTable/__tests__/PrintDataTable.unit.test.tsx +0 -361
  336. package/src/components/PrintDataTable/__tests__/PrintTableGroup.unit.test.tsx +0 -314
  337. package/src/components/PrintDataTable/__tests__/PrintTableRow.unit.test.tsx +0 -362
  338. package/src/components/PrintFooter/__tests__/PrintFooter.unit.test.tsx +0 -500
  339. package/src/components/PrintFooter/__tests__/PrintFooterContent.unit.test.tsx +0 -321
  340. package/src/components/PrintFooter/__tests__/PrintFooterInfo.unit.test.tsx +0 -335
  341. package/src/components/PrintFooter/__tests__/PrintPageNumber.unit.test.tsx +0 -340
  342. package/src/components/PrintGrid/__tests__/PrintGrid.unit.test.tsx +0 -340
  343. package/src/components/PrintGrid/__tests__/PrintGridBreakpoint.unit.test.tsx +0 -261
  344. package/src/components/PrintGrid/__tests__/PrintGridContainer.unit.test.tsx +0 -338
  345. package/src/components/PrintGrid/__tests__/PrintGridItem.unit.test.tsx +0 -338
  346. package/src/components/PrintHeader/__tests__/PrintCoverHeader.unit.test.tsx +0 -309
  347. package/src/components/PrintHeader/__tests__/PrintHeader.unit.test.tsx +0 -202
  348. package/src/components/PrintLayout/__tests__/PrintLayout.unit.test.tsx +0 -238
  349. package/src/components/PrintPageBreak/__tests__/PrintPageBreak.unit.test.tsx +0 -263
  350. package/src/components/PrintPageBreak/__tests__/PrintPageBreakGroup.unit.test.tsx +0 -239
  351. package/src/components/PrintPageBreak/__tests__/PrintPageBreakIndicator.unit.test.tsx +0 -235
  352. package/src/components/PrintSection/__tests__/PrintColumn.unit.test.tsx +0 -385
  353. package/src/components/PrintSection/__tests__/PrintDivider.unit.test.tsx +0 -373
  354. package/src/components/PrintSection/__tests__/PrintSection.unit.test.tsx +0 -390
  355. package/src/components/PrintSection/__tests__/PrintSectionContent.unit.test.tsx +0 -321
  356. package/src/components/PrintSection/__tests__/PrintSectionHeader.unit.test.tsx +0 -334
  357. package/src/components/PrintText/__tests__/PrintText.unit.test.tsx +0 -351
  358. package/src/components/Progress/__tests__/Progress.accessibility.test.tsx +0 -240
  359. package/src/components/Progress/__tests__/Progress.unit.test.tsx +0 -242
  360. package/src/components/PublicLayout/__tests__/EventLogo.test.tsx +0 -761
  361. package/src/components/PublicLayout/__tests__/PublicErrorBoundary.simplified.test.tsx +0 -228
  362. package/src/components/PublicLayout/__tests__/PublicErrorBoundary.test.tsx +0 -228
  363. package/src/components/PublicLayout/__tests__/PublicLoadingSpinner.test.tsx +0 -459
  364. package/src/components/PublicLayout/__tests__/PublicPageFooter.test.tsx +0 -362
  365. package/src/components/PublicLayout/__tests__/PublicPageHeader.test.tsx +0 -522
  366. package/src/components/PublicLayout/__tests__/PublicPageLayout.test.tsx +0 -599
  367. package/src/components/PublicLayout/__tests__/PublicPageProvider.test.tsx +0 -513
  368. package/src/components/RBAC/__tests__/PagePermissionGuard.unit.test.tsx +0 -683
  369. package/src/components/RBAC/__tests__/RBAC.integration.test.tsx +0 -573
  370. package/src/components/RBAC/__tests__/RBACGuard.unit.test.tsx +0 -467
  371. package/src/components/RBAC/__tests__/RBACProvider.accessibility.test.tsx +0 -475
  372. package/src/components/RBAC/__tests__/RBACProvider.advanced.test.tsx +0 -569
  373. package/src/components/RBAC/__tests__/RBACProvider.integration.test.tsx +0 -352
  374. package/src/components/RBAC/__tests__/RBACProvider.unit.test.tsx +0 -128
  375. package/src/components/RBAC/__tests__/RoleBasedContent.unit.test.tsx +0 -657
  376. package/src/components/Select/__tests__/SearchableSelect.unit.test.tsx +0 -437
  377. package/src/components/Select/__tests__/Select.accessibility.test.tsx +0 -1202
  378. package/src/components/Select/__tests__/Select.actual.test.tsx +0 -774
  379. package/src/components/Select/__tests__/Select.comprehensive.test.tsx +0 -837
  380. package/src/components/Select/__tests__/Select.enhanced.test.tsx +0 -1101
  381. package/src/components/Select/__tests__/Select.integration.test.tsx +0 -772
  382. package/src/components/Select/__tests__/Select.performance.test.tsx +0 -695
  383. package/src/components/Select/__tests__/Select.real-world.test.tsx +0 -1046
  384. package/src/components/Select/__tests__/Select.search-algorithms.test.tsx +0 -968
  385. package/src/components/Select/__tests__/Select.unit.test.tsx +0 -647
  386. package/src/components/Select/__tests__/Select.utils.test.tsx +0 -890
  387. package/src/components/Table/__tests__/Table.accessibility.test.tsx +0 -233
  388. package/src/components/Table/__tests__/Table.unit.test.tsx +0 -235
  389. package/src/components/Toast/__tests__/Toast.accessibility.test.tsx +0 -238
  390. package/src/components/Toast/__tests__/Toast.integration.test.tsx +0 -699
  391. package/src/components/Toast/__tests__/Toast.unit.test.tsx +0 -750
  392. package/src/components/Tooltip/__tests__/Tooltip.accessibility.test.tsx +0 -121
  393. package/src/components/Tooltip/__tests__/Tooltip.unit.test.tsx +0 -185
  394. package/src/components/UserMenu/__tests__/UserMenu.accessibility.test.tsx +0 -139
  395. package/src/components/UserMenu/__tests__/UserMenu.integration.test.tsx +0 -188
  396. package/src/components/UserMenu/__tests__/UserMenu.unit.test.tsx +0 -458
  397. package/src/components/__tests__/EdgeCaseTesting.enhanced.test.tsx +0 -524
  398. package/src/components/__tests__/ErrorTesting.enhanced.test.tsx +0 -455
  399. package/src/components/__tests__/SuperAdminGuard.test.tsx +0 -456
  400. package/src/components/__tests__/SuperAdminGuard.unit.test.tsx +0 -456
  401. package/src/components/examples/__tests__/PermissionExample.unit.test.tsx +0 -360
  402. package/src/hooks/__tests__/hooks.integration.test.tsx +0 -575
  403. package/src/hooks/__tests__/useApiFetch.unit.test.ts +0 -115
  404. package/src/hooks/__tests__/useComponentPerformance.unit.test.tsx +0 -133
  405. package/src/hooks/__tests__/useDebounce.unit.test.ts +0 -82
  406. package/src/hooks/__tests__/useFocusTrap.unit.test.tsx +0 -293
  407. package/src/hooks/__tests__/useInactivityTracker.unit.test.ts +0 -385
  408. package/src/hooks/__tests__/useOrganisationPermissions.unit.test.tsx +0 -286
  409. package/src/hooks/__tests__/useOrganisationSecurity.unit.test.tsx +0 -838
  410. package/src/hooks/__tests__/usePermissionCache.unit.test.ts +0 -627
  411. package/src/hooks/__tests__/useRBAC.unit.test.ts +0 -911
  412. package/src/hooks/__tests__/useSecureDataAccess.unit.test.tsx +0 -537
  413. package/src/hooks/__tests__/useToast.unit.test.tsx +0 -62
  414. package/src/hooks/__tests__/useZodForm.unit.test.tsx +0 -37
  415. package/src/hooks/public/__tests__/usePublicEvent.test.tsx +0 -397
  416. package/src/hooks/public/__tests__/usePublicEventLogo.test.tsx +0 -690
  417. package/src/hooks/public/__tests__/usePublicRouteParams.test.tsx +0 -449
  418. package/src/providers/__tests__/EventProvider.unit.test.tsx +0 -768
  419. package/src/providers/__tests__/OrganisationProvider.basic.test.tsx +0 -116
  420. package/src/providers/__tests__/OrganisationProvider.unit.test.tsx +0 -1312
  421. package/src/providers/__tests__/UnifiedAuthProvider.inactivity.test.tsx +0 -601
  422. package/src/providers/__tests__/UnifiedAuthProvider.unit.test.tsx +0 -683
  423. package/src/providers/__tests__/index.unit.test.ts +0 -78
  424. package/src/rbac/__tests__/PagePermissionGuard.test.tsx +0 -673
  425. package/src/rbac/__tests__/README.md +0 -170
  426. package/src/rbac/__tests__/RoleBasedRouter.test.tsx +0 -709
  427. package/src/rbac/__tests__/TestContext.tsx +0 -72
  428. package/src/rbac/__tests__/__mocks__/cache.ts +0 -144
  429. package/src/rbac/__tests__/__mocks__/supabase.ts +0 -152
  430. package/src/rbac/__tests__/adapters-hooks-comprehensive.test.tsx +0 -782
  431. package/src/rbac/__tests__/adapters-hooks.test.tsx +0 -561
  432. package/src/rbac/__tests__/adapters.comprehensive.test.tsx +0 -963
  433. package/src/rbac/__tests__/adapters.test.tsx +0 -444
  434. package/src/rbac/__tests__/api.test.ts +0 -620
  435. package/src/rbac/__tests__/audit-observability-comprehensive.test.ts +0 -792
  436. package/src/rbac/__tests__/audit-observability.test.ts +0 -549
  437. package/src/rbac/__tests__/audit.test.ts +0 -616
  438. package/src/rbac/__tests__/build-contract-compliance-simple.test.ts +0 -230
  439. package/src/rbac/__tests__/cache-invalidation-comprehensive.test.ts +0 -889
  440. package/src/rbac/__tests__/cache-invalidation.test.ts +0 -457
  441. package/src/rbac/__tests__/cache.test.ts +0 -458
  442. package/src/rbac/__tests__/components-navigation-guard.enhanced.test.tsx +0 -859
  443. package/src/rbac/__tests__/components-navigation-guard.test.tsx +0 -895
  444. package/src/rbac/__tests__/components-navigation-provider.test.tsx +0 -692
  445. package/src/rbac/__tests__/components-page-permission-guard.test.tsx +0 -673
  446. package/src/rbac/__tests__/components-page-permission-provider.test.tsx +0 -614
  447. package/src/rbac/__tests__/components-permission-enforcer.enhanced.fixed.test.tsx +0 -836
  448. package/src/rbac/__tests__/components-permission-enforcer.enhanced.test.tsx +0 -837
  449. package/src/rbac/__tests__/components-permission-enforcer.test.tsx +0 -825
  450. package/src/rbac/__tests__/components-role-based-router.test.tsx +0 -709
  451. package/src/rbac/__tests__/components-secure-data-provider.test.tsx +0 -607
  452. package/src/rbac/__tests__/config.test.ts +0 -583
  453. package/src/rbac/__tests__/core-logic-unit.test.ts +0 -190
  454. package/src/rbac/__tests__/core-permission-logic-comprehensive.test.ts +0 -1467
  455. package/src/rbac/__tests__/core-permission-logic-fixed.test.ts +0 -151
  456. package/src/rbac/__tests__/core-permission-logic-simple.test.ts +0 -968
  457. package/src/rbac/__tests__/core-permission-logic.test.ts +0 -966
  458. package/src/rbac/__tests__/edge-cases-comprehensive.test.ts +0 -988
  459. package/src/rbac/__tests__/edge-cases.test.ts +0 -654
  460. package/src/rbac/__tests__/engine.test.ts +0 -361
  461. package/src/rbac/__tests__/engine.unit.test.ts +0 -361
  462. package/src/rbac/__tests__/hooks.enhanced.test.tsx +0 -979
  463. package/src/rbac/__tests__/hooks.fixed.test.tsx +0 -475
  464. package/src/rbac/__tests__/hooks.test.tsx +0 -385
  465. package/src/rbac/__tests__/index.test.ts +0 -269
  466. package/src/rbac/__tests__/integration.enhanced.test.tsx +0 -824
  467. package/src/rbac/__tests__/page-permission-guard-super-admin.test.tsx +0 -261
  468. package/src/rbac/__tests__/performance.enhanced.test.tsx +0 -724
  469. package/src/rbac/__tests__/permissions.test.ts +0 -383
  470. package/src/rbac/__tests__/requires-event.test.ts +0 -330
  471. package/src/rbac/__tests__/scope-isolation-comprehensive.test.ts +0 -1349
  472. package/src/rbac/__tests__/scope-isolation.test.ts +0 -755
  473. package/src/rbac/__tests__/secure-client-rls-comprehensive.test.ts +0 -592
  474. package/src/rbac/__tests__/secure-client-rls.test.ts +0 -377
  475. package/src/rbac/__tests__/security.test.ts +0 -296
  476. package/src/rbac/__tests__/setup.ts +0 -228
  477. package/src/rbac/__tests__/test-utils-enhanced.tsx +0 -400
  478. package/src/rbac/__tests__/types.test.ts +0 -685
  479. package/src/rbac/components/__tests__/EnhancedNavigationMenu.test.tsx +0 -631
  480. package/src/rbac/components/__tests__/NavigationProvider.test.tsx +0 -667
  481. package/src/rbac/components/__tests__/PagePermissionProvider.test.tsx +0 -647
  482. package/src/rbac/components/__tests__/SecureDataProvider.test.tsx +0 -496
  483. package/src/rbac/testing/__tests__/index.test.tsx +0 -342
  484. package/src/rbac/utils/__tests__/eventContext.test.ts +0 -428
  485. package/src/rbac/utils/__tests__/eventContext.unit.test.ts +0 -428
  486. package/src/styles/__tests__/styles.unit.test.ts +0 -164
  487. package/src/test-dom-cleanup.test.tsx +0 -38
  488. package/src/theming/__tests__/README.md +0 -335
  489. package/src/theming/__tests__/runtime.accessibility.test.ts +0 -474
  490. package/src/theming/__tests__/runtime.error.test.ts +0 -616
  491. package/src/theming/__tests__/runtime.integration.test.ts +0 -376
  492. package/src/theming/__tests__/runtime.performance.test.ts +0 -411
  493. package/src/theming/__tests__/runtime.unit.test.ts +0 -470
  494. package/src/types/__tests__/database.unit.test.ts +0 -489
  495. package/src/types/__tests__/guards.unit.test.ts +0 -146
  496. package/src/types/__tests__/index.unit.test.ts +0 -77
  497. package/src/types/__tests__/organisation.unit.test.ts +0 -713
  498. package/src/types/__tests__/rbac.unit.test.ts +0 -621
  499. package/src/types/__tests__/security.unit.test.ts +0 -347
  500. package/src/types/__tests__/supabase.unit.test.ts +0 -658
  501. package/src/types/__tests__/theme.unit.test.ts +0 -218
  502. package/src/types/__tests__/unified.unit.test.ts +0 -537
  503. package/src/types/__tests__/validation.unit.test.ts +0 -616
  504. package/src/utils/__tests__/appConfig.unit.test.ts +0 -55
  505. package/src/utils/__tests__/appNameResolver.unit.test.ts +0 -137
  506. package/src/utils/__tests__/audit.unit.test.ts +0 -69
  507. package/src/utils/__tests__/auth-utils.unit.test.ts +0 -70
  508. package/src/utils/__tests__/bundleAnalysis.unit.test.ts +0 -317
  509. package/src/utils/__tests__/cn.unit.test.ts +0 -34
  510. package/src/utils/__tests__/deviceFingerprint.unit.test.ts +0 -480
  511. package/src/utils/__tests__/dynamicUtils.unit.test.ts +0 -322
  512. package/src/utils/__tests__/formatDate.unit.test.ts +0 -109
  513. package/src/utils/__tests__/formatting.unit.test.ts +0 -66
  514. package/src/utils/__tests__/index.unit.test.ts +0 -251
  515. package/src/utils/__tests__/lazyLoad.unit.test.tsx +0 -304
  516. package/src/utils/__tests__/organisationContext.unit.test.ts +0 -192
  517. package/src/utils/__tests__/performanceBudgets.unit.test.ts +0 -259
  518. package/src/utils/__tests__/permissionTypes.unit.test.ts +0 -250
  519. package/src/utils/__tests__/permissionUtils.unit.test.ts +0 -362
  520. package/src/utils/__tests__/sanitization.unit.test.ts +0 -346
  521. package/src/utils/__tests__/schemaUtils.unit.test.ts +0 -441
  522. package/src/utils/__tests__/secureDataAccess.unit.test.ts +0 -334
  523. package/src/utils/__tests__/secureErrors.unit.test.ts +0 -377
  524. package/src/utils/__tests__/secureStorage.unit.test.ts +0 -293
  525. package/src/utils/__tests__/security.unit.test.ts +0 -127
  526. package/src/utils/__tests__/securityMonitor.unit.test.ts +0 -280
  527. package/src/utils/__tests__/sessionTracking.unit.test.ts +0 -370
  528. package/src/utils/__tests__/validation.unit.test.ts +0 -84
  529. package/src/utils/__tests__/validationUtils.unit.test.ts +0 -571
  530. package/src/utils/print/__tests__/PrintDataProcessor.unit.test.ts +0 -219
  531. package/src/utils/print/__tests__/usePrintOptimization.unit.test.tsx +0 -353
  532. package/src/utils/storage/__tests__/config.unit.test.ts +0 -206
  533. package/src/utils/storage/__tests__/helpers.unit.test.ts +0 -648
  534. package/src/utils/storage/__tests__/index.unit.test.ts +0 -167
  535. package/src/utils/storage/__tests__/types.unit.test.ts +0 -441
  536. package/src/validation/__tests__/common.unit.test.ts +0 -101
  537. package/src/validation/__tests__/csrf.unit.test.ts +0 -302
  538. package/src/validation/__tests__/passwordSchema.unit.test.ts +0 -98
  539. package/src/validation/__tests__/sqlInjectionProtection.unit.test.ts +0 -466
  540. /package/dist/{DataTable-EEUDXPE5.js.map → DataTable-GX3XERFJ.js.map} +0 -0
  541. /package/dist/{chunk-VYG4AXYW.js.map → chunk-5EL3KHOQ.js.map} +0 -0
@@ -1,1467 +0,0 @@
1
- /**
2
- * Comprehensive Core Permission Logic Tests
3
- *
4
- * Tests the fundamental RBAC permission resolution logic including:
5
- * - Scope precedence (page → eventApp → organisation → global)
6
- * - Deny overrides allow
7
- * - Super admin bypass with audit logging
8
- * - Time-bound grants (valid_from/valid_to)
9
- * - Page-level rule refinements
10
- * - Permission matching (exact and wildcard)
11
- */
12
-
13
- import { describe, it, expect, beforeEach, vi } from 'vitest';
14
- import { createRBACEngine, RBACEngine } from '../engine';
15
- import { Scope, Permission } from '../types';
16
- import { rbacCache } from '../cache';
17
-
18
- // Mock dependencies
19
- vi.mock('../cache');
20
- vi.mock('../audit');
21
-
22
- // Create a comprehensive mock for Supabase queries
23
- function createSupabaseMock() {
24
- const mockQuery = {
25
- select: vi.fn().mockReturnThis(),
26
- eq: vi.fn().mockReturnThis(),
27
- is: vi.fn().mockReturnThis(),
28
- lte: vi.fn().mockReturnThis(),
29
- gte: vi.fn().mockReturnThis(),
30
- or: vi.fn().mockReturnThis(),
31
- and: vi.fn().mockReturnThis(),
32
- order: vi.fn().mockReturnThis(),
33
- in: vi.fn().mockReturnThis(),
34
- single: vi.fn().mockResolvedValue({ data: null, error: null }),
35
- maybeSingle: vi.fn().mockResolvedValue({ data: null, error: null }),
36
- insert: vi.fn().mockResolvedValue({ data: null, error: null }),
37
- };
38
-
39
- return {
40
- from: vi.fn().mockReturnValue(mockQuery),
41
- rpc: vi.fn().mockResolvedValue({ data: null, error: null }),
42
- };
43
- }
44
-
45
- describe('Core Permission Logic - Comprehensive', () => {
46
- let engine: RBACEngine;
47
- let mockSupabase: ReturnType<typeof createSupabaseMock>;
48
-
49
- beforeEach(() => {
50
- vi.clearAllMocks();
51
- // Clear the cache to ensure fresh state
52
- rbacCache.clear();
53
- // Reset the cache mock to return null by default
54
- vi.mocked(rbacCache.get).mockReturnValue(null);
55
- mockSupabase = createSupabaseMock();
56
- engine = createRBACEngine(mockSupabase as any);
57
- });
58
-
59
- describe('Scope Precedence', () => {
60
- it('should prioritize page permissions over event-app permissions', async () => {
61
- const userId = 'user-123';
62
- const scope: Scope = {
63
- organisationId: 'org-456',
64
- eventId: 'event-789',
65
- appId: 'app-101'
66
- };
67
- const permission: Permission = 'manage:page.events'; // Updated to match page permission format
68
- const pageId = '12345678-1234-1234-1234-123456789012';
69
-
70
- // Mock all database calls with proper sequencing
71
- let callCount = 0;
72
-
73
- // Mock RPC calls for get_rbac_permissions
74
- mockSupabase.rpc.mockImplementation((fnName: string, params: any) => {
75
- if (fnName === 'get_rbac_permissions') {
76
- callCount++;
77
- // Return page-level deny for event_admin role
78
- return Promise.resolve({
79
- data: [
80
- { permission_type: 'manage', role_name: 'event_admin', has_permission: false, granted_at: '2024-01-01T00:00:00Z' }
81
- ],
82
- error: null
83
- });
84
- }
85
- return Promise.resolve({ data: null, error: null });
86
- });
87
-
88
- mockSupabase.from.mockImplementation((tableName: string) => {
89
- const mockQuery = {
90
- select: vi.fn().mockReturnThis(),
91
- eq: vi.fn().mockReturnThis(),
92
- is: vi.fn().mockReturnThis(),
93
- lte: vi.fn().mockReturnThis(),
94
- gte: vi.fn().mockReturnThis(),
95
- or: vi.fn().mockReturnThis(),
96
- and: vi.fn().mockReturnThis(),
97
- order: vi.fn().mockReturnThis(),
98
- in: vi.fn().mockReturnThis(),
99
- single: vi.fn().mockResolvedValue({ data: null, error: null }),
100
- maybeSingle: vi.fn().mockResolvedValue({ data: null, error: null }),
101
- insert: vi.fn().mockResolvedValue({ data: null, error: null }),
102
- };
103
-
104
- switch (tableName) {
105
- case 'rbac_app_pages':
106
- // Return page name for permission matching
107
- return {
108
- ...mockQuery,
109
- single: vi.fn().mockResolvedValue({
110
- data: { page_name: 'events' },
111
- error: null
112
- })
113
- };
114
- case 'rbac_global_roles':
115
- // Super admin check - no super admin
116
- callCount++;
117
- return {
118
- ...mockQuery,
119
- single: vi.fn().mockResolvedValue({
120
- data: null,
121
- error: { code: 'PGRST116' }
122
- })
123
- };
124
- case 'rbac_apps':
125
- // App configuration - default to requires_event: false
126
- callCount++;
127
- return {
128
- ...mockQuery,
129
- single: vi.fn().mockResolvedValue({
130
- data: { requires_event: false },
131
- error: null
132
- })
133
- };
134
- case 'rbac_event_app_roles':
135
- // Event-app permission that ALLOWS (should be overridden)
136
- callCount++;
137
- return {
138
- ...mockQuery,
139
- or: vi.fn().mockResolvedValue({
140
- data: [{ role: 'event_admin', status: 'active', valid_from: '2024-01-01T00:00:00Z', valid_to: null }],
141
- error: null
142
- })
143
- };
144
- case 'rbac_organisation_roles':
145
- // Organisation permission that ALLOWS (should be overridden)
146
- callCount++;
147
- return {
148
- ...mockQuery,
149
- or: vi.fn().mockResolvedValue({
150
- data: [{ role: 'org_admin', status: 'active', valid_from: '2024-01-01T00:00:00Z', valid_to: null }],
151
- error: null
152
- })
153
- };
154
- case 'rbac_audit_events':
155
- // Audit event insert
156
- callCount++;
157
- return {
158
- ...mockQuery,
159
- insert: vi.fn().mockResolvedValue({ data: null, error: null })
160
- };
161
- default:
162
- return mockQuery;
163
- }
164
- });
165
-
166
- const result = await engine.isPermitted({
167
- userId,
168
- scope,
169
- permission,
170
- pageId
171
- });
172
-
173
- expect(result).toBe(false); // Page deny should override event-app allow
174
- expect(callCount).toBe(6); // Should check: super_admin, app_config, event_app, org, global_roles, rpc (page via RPC)
175
- });
176
-
177
- it('should prioritize event-app permissions over organisation permissions', async () => {
178
- const userId = 'user-123';
179
- const scope: Scope = {
180
- organisationId: 'org-456',
181
- eventId: 'event-789',
182
- appId: 'app-101'
183
- };
184
- const permission: Permission = 'read:events';
185
-
186
- // Mock all database calls with proper sequencing
187
- let callCount = 0;
188
- mockSupabase.from.mockImplementation((tableName: string) => {
189
- const mockQuery = {
190
- select: vi.fn().mockReturnThis(),
191
- eq: vi.fn().mockReturnThis(),
192
- is: vi.fn().mockReturnThis(),
193
- lte: vi.fn().mockReturnThis(),
194
- gte: vi.fn().mockReturnThis(),
195
- or: vi.fn().mockReturnThis(),
196
- and: vi.fn().mockReturnThis(),
197
- order: vi.fn().mockReturnThis(),
198
- in: vi.fn().mockReturnThis(),
199
- single: vi.fn().mockResolvedValue({ data: null, error: null }),
200
- maybeSingle: vi.fn().mockResolvedValue({ data: null, error: null }),
201
- insert: vi.fn().mockResolvedValue({ data: null, error: null }),
202
- };
203
-
204
- switch (tableName) {
205
- case 'rbac_global_roles':
206
- // Super admin check - no super admin
207
- callCount++;
208
- return {
209
- ...mockQuery,
210
- single: vi.fn().mockResolvedValue({
211
- data: null,
212
- error: { code: 'PGRST116' }
213
- })
214
- };
215
- case 'rbac_apps':
216
- // App configuration - default to requires_event: false
217
- callCount++;
218
- return {
219
- ...mockQuery,
220
- single: vi.fn().mockResolvedValue({
221
- data: { requires_event: false },
222
- error: null
223
- })
224
- };
225
- case 'rbac_page_permissions':
226
- // No page permissions
227
- callCount++;
228
- return {
229
- ...mockQuery,
230
- eq: vi.fn().mockResolvedValue({
231
- data: [],
232
- error: null
233
- })
234
- };
235
- case 'rbac_event_app_roles':
236
- // Event-app permission that DENIES
237
- callCount++;
238
- return {
239
- ...mockQuery,
240
- or: vi.fn().mockResolvedValue({
241
- data: [{ role: 'viewer', valid_from: '2024-01-01T00:00:00Z', valid_to: null }],
242
- error: null
243
- })
244
- };
245
- case 'rbac_organisation_roles':
246
- // Organisation permission that ALLOWS (should be overridden)
247
- callCount++;
248
- return {
249
- ...mockQuery,
250
- or: vi.fn().mockResolvedValue({
251
- data: [{ role: 'org_admin', status: 'active', valid_from: '2024-01-01T00:00:00Z', valid_to: null }],
252
- error: null
253
- })
254
- };
255
- case 'rbac_audit_events':
256
- // Audit event insert
257
- callCount++;
258
- return {
259
- ...mockQuery,
260
- insert: vi.fn().mockResolvedValue({ data: null, error: null })
261
- };
262
- default:
263
- return mockQuery;
264
- }
265
- });
266
-
267
- const result = await engine.isPermitted({
268
- userId,
269
- scope,
270
- permission
271
- });
272
-
273
- expect(result).toBe(false); // Event-app deny should override org allow
274
- expect(callCount).toBe(5); // Should check all scopes (including app config)
275
- });
276
-
277
- it('should prioritize organisation permissions over global permissions', async () => {
278
- const userId = 'user-123';
279
- const scope: Scope = {
280
- organisationId: 'org-456'
281
- };
282
- const permission: Permission = 'manage:organisation.events';
283
-
284
- // Mock all database calls with proper sequencing
285
- let callCount = 0;
286
- mockSupabase.from.mockImplementation((tableName: string) => {
287
- const mockQuery = {
288
- select: vi.fn().mockReturnThis(),
289
- eq: vi.fn().mockReturnThis(),
290
- is: vi.fn().mockReturnThis(),
291
- lte: vi.fn().mockReturnThis(),
292
- gte: vi.fn().mockReturnThis(),
293
- or: vi.fn().mockReturnThis(),
294
- and: vi.fn().mockReturnThis(),
295
- order: vi.fn().mockReturnThis(),
296
- in: vi.fn().mockReturnThis(),
297
- single: vi.fn().mockResolvedValue({ data: null, error: null }),
298
- maybeSingle: vi.fn().mockResolvedValue({ data: null, error: null }),
299
- insert: vi.fn().mockResolvedValue({ data: null, error: null }),
300
- };
301
-
302
- switch (tableName) {
303
- case 'rbac_global_roles':
304
- // Super admin check - no super admin
305
- callCount++;
306
- return {
307
- ...mockQuery,
308
- single: vi.fn().mockResolvedValue({
309
- data: null,
310
- error: { code: 'PGRST116' }
311
- })
312
- };
313
- case 'rbac_apps':
314
- // App configuration - default to requires_event: false
315
- callCount++;
316
- return {
317
- ...mockQuery,
318
- single: vi.fn().mockResolvedValue({
319
- data: { requires_event: false },
320
- error: null
321
- })
322
- };
323
- case 'rbac_page_permissions':
324
- // No page permissions
325
- callCount++;
326
- return {
327
- ...mockQuery,
328
- eq: vi.fn().mockResolvedValue({
329
- data: [],
330
- error: null
331
- })
332
- };
333
- case 'rbac_event_app_roles':
334
- // No event-app permissions
335
- callCount++;
336
- return {
337
- ...mockQuery,
338
- or: vi.fn().mockResolvedValue({
339
- data: [],
340
- error: null
341
- })
342
- };
343
- case 'rbac_organisation_roles':
344
- // Organisation permission that DENIES
345
- callCount++;
346
- return {
347
- ...mockQuery,
348
- or: vi.fn().mockResolvedValue({
349
- data: [{ role: 'supporter', status: 'active', valid_from: '2024-01-01T00:00:00Z', valid_to: null }],
350
- error: null
351
- })
352
- };
353
- case 'rbac_audit_events':
354
- // Audit event insert
355
- callCount++;
356
- return {
357
- ...mockQuery,
358
- insert: vi.fn().mockResolvedValue({ data: null, error: null })
359
- };
360
- default:
361
- return mockQuery;
362
- }
363
- });
364
-
365
- const result = await engine.isPermitted({
366
- userId,
367
- scope,
368
- permission
369
- });
370
-
371
- expect(result).toBe(false); // Org deny should override global allow
372
- expect(callCount).toBe(3); // Should check super admin + organisation + audit
373
- });
374
- });
375
-
376
- describe('Deny Overrides Allow', () => {
377
- it('should deny access when any scope has explicit deny', async () => {
378
- const userId = 'user-123';
379
- const scope: Scope = {
380
- organisationId: 'org-456',
381
- eventId: 'event-789',
382
- appId: 'app-101'
383
- };
384
- const permission: Permission = 'manage:events';
385
- const pageId = '12345678-1234-1234-1234-123456789012';
386
-
387
- // Mock all database calls with proper sequencing
388
- let callCount = 0;
389
-
390
- // Mock RPC calls for get_rbac_permissions
391
- mockSupabase.rpc.mockImplementation((fnName: string, params: any) => {
392
- if (fnName === 'get_rbac_permissions') {
393
- callCount++;
394
- // Return page-level deny for supporter role
395
- return Promise.resolve({
396
- data: [
397
- { permission_type: 'manage', role_name: 'supporter', has_permission: false, granted_at: '2024-01-01T00:00:00Z' }
398
- ],
399
- error: null
400
- });
401
- }
402
- return Promise.resolve({ data: null, error: null });
403
- });
404
-
405
- mockSupabase.from.mockImplementation((tableName: string) => {
406
- const mockQuery = {
407
- select: vi.fn().mockReturnThis(),
408
- eq: vi.fn().mockReturnThis(),
409
- is: vi.fn().mockReturnThis(),
410
- lte: vi.fn().mockReturnThis(),
411
- gte: vi.fn().mockReturnThis(),
412
- or: vi.fn().mockReturnThis(),
413
- and: vi.fn().mockReturnThis(),
414
- order: vi.fn().mockReturnThis(),
415
- in: vi.fn().mockReturnThis(),
416
- single: vi.fn().mockResolvedValue({ data: null, error: null }),
417
- maybeSingle: vi.fn().mockResolvedValue({ data: null, error: null }),
418
- insert: vi.fn().mockResolvedValue({ data: null, error: null }),
419
- };
420
-
421
- switch (tableName) {
422
- case 'rbac_global_roles':
423
- // Super admin check - no super admin
424
- callCount++;
425
- return {
426
- ...mockQuery,
427
- single: vi.fn().mockResolvedValue({
428
- data: null,
429
- error: { code: 'PGRST116' }
430
- })
431
- };
432
- case 'rbac_apps':
433
- // App configuration - default to requires_event: false
434
- callCount++;
435
- return {
436
- ...mockQuery,
437
- single: vi.fn().mockResolvedValue({
438
- data: { requires_event: false },
439
- error: null
440
- })
441
- };
442
- case 'rbac_event_app_roles':
443
- // Event-app permission that ALLOWS
444
- callCount++;
445
- return {
446
- ...mockQuery,
447
- or: vi.fn().mockResolvedValue({
448
- data: [{ role: 'event_admin', status: 'active', valid_from: '2024-01-01T00:00:00Z', valid_to: null }],
449
- error: null
450
- })
451
- };
452
- case 'rbac_organisation_roles':
453
- // Organisation permission that ALLOWS
454
- callCount++;
455
- return {
456
- ...mockQuery,
457
- or: vi.fn().mockResolvedValue({
458
- data: [{ role: 'supporter', status: 'active', valid_from: '2024-01-01T00:00:00Z', valid_to: null }],
459
- error: null
460
- })
461
- };
462
- case 'rbac_audit_events':
463
- // Audit event insert
464
- callCount++;
465
- return {
466
- ...mockQuery,
467
- insert: vi.fn().mockResolvedValue({ data: null, error: null })
468
- };
469
- default:
470
- return mockQuery;
471
- }
472
- });
473
-
474
- const result = await engine.isPermitted({
475
- userId,
476
- scope,
477
- permission,
478
- pageId
479
- });
480
-
481
- expect(result).toBe(false); // Page deny should override all allows
482
- expect(callCount).toBe(6); // Should check: super_admin, app_config, event_app, org, global_roles, rpc (page via RPC)
483
- });
484
-
485
- it('should allow access when no denies exist and at least one allow exists', async () => {
486
- const userId = 'user-123';
487
- const scope: Scope = {
488
- organisationId: 'org-456',
489
- eventId: 'event-789',
490
- appId: 'app-101'
491
- };
492
- const permission: Permission = 'read:organisation.events';
493
-
494
- // Mock all database calls with proper sequencing
495
- let callCount = 0;
496
- mockSupabase.from.mockImplementation((tableName: string) => {
497
- const mockQuery = {
498
- select: vi.fn().mockReturnThis(),
499
- eq: vi.fn().mockReturnThis(),
500
- is: vi.fn().mockReturnThis(),
501
- lte: vi.fn().mockReturnThis(),
502
- gte: vi.fn().mockReturnThis(),
503
- or: vi.fn().mockReturnThis(),
504
- and: vi.fn().mockReturnThis(),
505
- order: vi.fn().mockReturnThis(),
506
- in: vi.fn().mockReturnThis(),
507
- single: vi.fn().mockResolvedValue({ data: null, error: null }),
508
- maybeSingle: vi.fn().mockResolvedValue({ data: null, error: null }),
509
- insert: vi.fn().mockResolvedValue({ data: null, error: null }),
510
- };
511
-
512
- switch (tableName) {
513
- case 'rbac_global_roles':
514
- // Super admin check - no super admin
515
- callCount++;
516
- return {
517
- ...mockQuery,
518
- single: vi.fn().mockResolvedValue({
519
- data: null,
520
- error: { code: 'PGRST116' }
521
- })
522
- };
523
- case 'rbac_apps':
524
- // App configuration - default to requires_event: false
525
- callCount++;
526
- return {
527
- ...mockQuery,
528
- single: vi.fn().mockResolvedValue({
529
- data: { requires_event: false },
530
- error: null
531
- })
532
- };
533
- case 'rbac_page_permissions':
534
- // No page permissions
535
- callCount++;
536
- return {
537
- ...mockQuery,
538
- eq: vi.fn().mockResolvedValue({
539
- data: [],
540
- error: null
541
- })
542
- };
543
- case 'rbac_event_app_roles':
544
- // No event-app permissions
545
- callCount++;
546
- return {
547
- ...mockQuery,
548
- or: vi.fn().mockResolvedValue({
549
- data: [],
550
- error: null
551
- })
552
- };
553
- case 'rbac_organisation_roles':
554
- // Organisation permission that ALLOWS
555
- callCount++;
556
- return {
557
- ...mockQuery,
558
- or: vi.fn().mockResolvedValue({
559
- data: [{ role: 'member', status: 'active', valid_from: '2024-01-01T00:00:00Z', valid_to: null }],
560
- error: null
561
- })
562
- };
563
- case 'rbac_audit_events':
564
- // Audit event insert
565
- callCount++;
566
- return {
567
- ...mockQuery,
568
- insert: vi.fn().mockResolvedValue({ data: null, error: null })
569
- };
570
- default:
571
- return mockQuery;
572
- }
573
- });
574
-
575
- const result = await engine.isPermitted({
576
- userId,
577
- scope,
578
- permission
579
- });
580
-
581
- expect(result).toBe(true); // Should allow when organisation permission exists
582
- expect(callCount).toBe(5); // Should check all scopes (including app config)
583
- });
584
- });
585
-
586
- describe('Super Admin Bypass', () => {
587
- it('should allow super admin to access everything and log bypass', async () => {
588
- const userId = 'user-123';
589
- const scope: Scope = {
590
- organisationId: 'org-456',
591
- eventId: 'event-789',
592
- appId: 'app-101'
593
- };
594
- const permission: Permission = 'manage:everything';
595
-
596
- // Mock all database calls with proper sequencing
597
- let callCount = 0;
598
- mockSupabase.from.mockImplementation((tableName: string) => {
599
- const mockQuery = {
600
- select: vi.fn().mockReturnThis(),
601
- eq: vi.fn().mockReturnThis(),
602
- is: vi.fn().mockReturnThis(),
603
- lte: vi.fn().mockReturnThis(),
604
- gte: vi.fn().mockReturnThis(),
605
- or: vi.fn().mockReturnThis(),
606
- and: vi.fn().mockReturnThis(),
607
- order: vi.fn().mockReturnThis(),
608
- in: vi.fn().mockReturnThis(),
609
- single: vi.fn().mockResolvedValue({ data: null, error: null }),
610
- maybeSingle: vi.fn().mockResolvedValue({ data: null, error: null }),
611
- insert: vi.fn().mockResolvedValue({ data: null, error: null }),
612
- };
613
-
614
- switch (tableName) {
615
- case 'rbac_global_roles':
616
- // Super admin check - super admin exists
617
- callCount++;
618
- return {
619
- ...mockQuery,
620
- single: vi.fn().mockResolvedValue({
621
- data: { id: 'role-123' },
622
- error: null
623
- })
624
- };
625
- case 'rbac_audit_events':
626
- // Audit event insert
627
- callCount++;
628
- return {
629
- ...mockQuery,
630
- insert: vi.fn().mockResolvedValue({ data: null, error: null })
631
- };
632
- default:
633
- return mockQuery;
634
- }
635
- });
636
-
637
- const result = await engine.isPermitted({
638
- userId,
639
- scope,
640
- permission
641
- });
642
-
643
- expect(result).toBe(true); // Should allow due to super admin
644
- expect(callCount).toBe(1); // Should check super admin and audit
645
-
646
- // Verify audit event was emitted with bypass: true
647
- const { emitAuditEvent } = await import('../audit');
648
- expect(emitAuditEvent).toHaveBeenCalledWith(
649
- expect.objectContaining({
650
- type: 'permission_check',
651
- userId,
652
- organisationId: scope.organisationId,
653
- eventId: scope.eventId,
654
- appId: scope.appId,
655
- permission,
656
- decision: true,
657
- source: 'api',
658
- bypass: true,
659
- duration_ms: expect.any(Number)
660
- })
661
- );
662
- });
663
-
664
- it('should not check other permissions when super admin exists', async () => {
665
- const userId = 'user-123';
666
- const scope: Scope = {
667
- organisationId: 'org-456',
668
- eventId: 'event-789',
669
- appId: 'app-101'
670
- };
671
- const permission: Permission = 'manage:everything';
672
-
673
- // Mock all database calls with proper sequencing
674
- let callCount = 0;
675
- mockSupabase.from.mockImplementation((tableName: string) => {
676
- const mockQuery = {
677
- select: vi.fn().mockReturnThis(),
678
- eq: vi.fn().mockReturnThis(),
679
- is: vi.fn().mockReturnThis(),
680
- lte: vi.fn().mockReturnThis(),
681
- gte: vi.fn().mockReturnThis(),
682
- or: vi.fn().mockReturnThis(),
683
- and: vi.fn().mockReturnThis(),
684
- order: vi.fn().mockReturnThis(),
685
- in: vi.fn().mockReturnThis(),
686
- single: vi.fn().mockResolvedValue({ data: null, error: null }),
687
- maybeSingle: vi.fn().mockResolvedValue({ data: null, error: null }),
688
- insert: vi.fn().mockResolvedValue({ data: null, error: null }),
689
- };
690
-
691
- switch (tableName) {
692
- case 'rbac_global_roles':
693
- // Super admin check - super admin exists
694
- callCount++;
695
- return {
696
- ...mockQuery,
697
- single: vi.fn().mockResolvedValue({
698
- data: { id: 'role-123' },
699
- error: null
700
- })
701
- };
702
- case 'rbac_audit_events':
703
- // Audit event insert
704
- callCount++;
705
- return {
706
- ...mockQuery,
707
- insert: vi.fn().mockResolvedValue({ data: null, error: null })
708
- };
709
- default:
710
- return mockQuery;
711
- }
712
- });
713
-
714
- const result = await engine.isPermitted({
715
- userId,
716
- scope,
717
- permission
718
- });
719
-
720
- expect(result).toBe(true); // Should allow due to super admin
721
- expect(callCount).toBe(1); // Should check super admin and audit
722
- });
723
- });
724
-
725
- describe('Time-bound Grants', () => {
726
- it('should deny access for expired grants', async () => {
727
- const userId = 'user-123';
728
- const scope: Scope = {
729
- organisationId: 'org-456'
730
- };
731
- const permission: Permission = 'manage:organisation.events';
732
-
733
- // Mock all database calls with proper sequencing
734
- let callCount = 0;
735
- mockSupabase.from.mockImplementation((tableName: string) => {
736
- const mockQuery = {
737
- select: vi.fn().mockReturnThis(),
738
- eq: vi.fn().mockReturnThis(),
739
- is: vi.fn().mockReturnThis(),
740
- lte: vi.fn().mockReturnThis(),
741
- gte: vi.fn().mockReturnThis(),
742
- or: vi.fn().mockReturnThis(),
743
- and: vi.fn().mockReturnThis(),
744
- order: vi.fn().mockReturnThis(),
745
- in: vi.fn().mockReturnThis(),
746
- single: vi.fn().mockResolvedValue({ data: null, error: null }),
747
- maybeSingle: vi.fn().mockResolvedValue({ data: null, error: null }),
748
- insert: vi.fn().mockResolvedValue({ data: null, error: null }),
749
- };
750
-
751
- switch (tableName) {
752
- case 'rbac_global_roles':
753
- // Super admin check - no super admin
754
- callCount++;
755
- return {
756
- ...mockQuery,
757
- single: vi.fn().mockResolvedValue({
758
- data: null,
759
- error: { code: 'PGRST116' }
760
- })
761
- };
762
- case 'rbac_apps':
763
- // App configuration - default to requires_event: false
764
- callCount++;
765
- return {
766
- ...mockQuery,
767
- single: vi.fn().mockResolvedValue({
768
- data: { requires_event: false },
769
- error: null
770
- })
771
- };
772
- case 'rbac_page_permissions':
773
- // No page permissions
774
- callCount++;
775
- return {
776
- ...mockQuery,
777
- eq: vi.fn().mockResolvedValue({
778
- data: [],
779
- error: null
780
- })
781
- };
782
- case 'rbac_event_app_roles':
783
- // No event-app permissions
784
- callCount++;
785
- return {
786
- ...mockQuery,
787
- or: vi.fn().mockResolvedValue({
788
- data: [],
789
- error: null
790
- })
791
- };
792
- case 'rbac_organisation_roles':
793
- // Organisation permission that is EXPIRED - should return empty data
794
- callCount++;
795
- return {
796
- ...mockQuery,
797
- or: vi.fn().mockResolvedValue({
798
- data: [], // Empty because expired grants should be filtered out
799
- error: null
800
- })
801
- };
802
- case 'rbac_audit_events':
803
- // Audit event insert
804
- callCount++;
805
- return {
806
- ...mockQuery,
807
- insert: vi.fn().mockResolvedValue({ data: null, error: null })
808
- };
809
- default:
810
- return mockQuery;
811
- }
812
- });
813
-
814
- const result = await engine.isPermitted({
815
- userId,
816
- scope,
817
- permission
818
- });
819
-
820
- expect(result).toBe(false); // Should deny for expired grants
821
- expect(callCount).toBe(3); // Should check super admin + app config + organisation + audit
822
- });
823
-
824
- it('should deny access for future-dated grants', async () => {
825
- const userId = 'user-123';
826
- const scope: Scope = {
827
- organisationId: 'org-456'
828
- };
829
- const permission: Permission = 'manage:organisation.events';
830
-
831
- // Mock all database calls with proper sequencing
832
- let callCount = 0;
833
- mockSupabase.from.mockImplementation((tableName: string) => {
834
- const mockQuery = {
835
- select: vi.fn().mockReturnThis(),
836
- eq: vi.fn().mockReturnThis(),
837
- is: vi.fn().mockReturnThis(),
838
- lte: vi.fn().mockReturnThis(),
839
- gte: vi.fn().mockReturnThis(),
840
- or: vi.fn().mockReturnThis(),
841
- and: vi.fn().mockReturnThis(),
842
- order: vi.fn().mockReturnThis(),
843
- in: vi.fn().mockReturnThis(),
844
- single: vi.fn().mockResolvedValue({ data: null, error: null }),
845
- maybeSingle: vi.fn().mockResolvedValue({ data: null, error: null }),
846
- insert: vi.fn().mockResolvedValue({ data: null, error: null }),
847
- };
848
-
849
- switch (tableName) {
850
- case 'rbac_global_roles':
851
- // Super admin check - no super admin
852
- callCount++;
853
- return {
854
- ...mockQuery,
855
- single: vi.fn().mockResolvedValue({
856
- data: null,
857
- error: { code: 'PGRST116' }
858
- })
859
- };
860
- case 'rbac_apps':
861
- // App configuration - default to requires_event: false
862
- callCount++;
863
- return {
864
- ...mockQuery,
865
- single: vi.fn().mockResolvedValue({
866
- data: { requires_event: false },
867
- error: null
868
- })
869
- };
870
- case 'rbac_page_permissions':
871
- // No page permissions
872
- callCount++;
873
- return {
874
- ...mockQuery,
875
- eq: vi.fn().mockResolvedValue({
876
- data: [],
877
- error: null
878
- })
879
- };
880
- case 'rbac_event_app_roles':
881
- // No event-app permissions
882
- callCount++;
883
- return {
884
- ...mockQuery,
885
- or: vi.fn().mockResolvedValue({
886
- data: [],
887
- error: null
888
- })
889
- };
890
- case 'rbac_organisation_roles':
891
- // Organisation permission that is FUTURE-DATED - should return empty data
892
- callCount++;
893
- return {
894
- ...mockQuery,
895
- or: vi.fn().mockResolvedValue({
896
- data: [], // Empty because future-dated grants should be filtered out
897
- error: null
898
- })
899
- };
900
- case 'rbac_audit_events':
901
- // Audit event insert
902
- callCount++;
903
- return {
904
- ...mockQuery,
905
- insert: vi.fn().mockResolvedValue({ data: null, error: null })
906
- };
907
- default:
908
- return mockQuery;
909
- }
910
- });
911
-
912
- const result = await engine.isPermitted({
913
- userId,
914
- scope,
915
- permission
916
- });
917
-
918
- expect(result).toBe(false); // Should deny for future-dated grants
919
- expect(callCount).toBe(3); // Should check super admin + app config + organisation + audit
920
- });
921
-
922
- it('should allow access for valid time-bound grants', async () => {
923
- const userId = 'user-123';
924
- const scope: Scope = {
925
- organisationId: 'org-456'
926
- };
927
- const permission: Permission = 'manage:organisation.events';
928
-
929
- // Mock all database calls with proper sequencing
930
- let callCount = 0;
931
- mockSupabase.from.mockImplementation((tableName: string) => {
932
- const mockQuery = {
933
- select: vi.fn().mockReturnThis(),
934
- eq: vi.fn().mockReturnThis(),
935
- is: vi.fn().mockReturnThis(),
936
- lte: vi.fn().mockReturnThis(),
937
- gte: vi.fn().mockReturnThis(),
938
- or: vi.fn().mockReturnThis(),
939
- and: vi.fn().mockReturnThis(),
940
- order: vi.fn().mockReturnThis(),
941
- in: vi.fn().mockReturnThis(),
942
- single: vi.fn().mockResolvedValue({ data: null, error: null }),
943
- maybeSingle: vi.fn().mockResolvedValue({ data: null, error: null }),
944
- insert: vi.fn().mockResolvedValue({ data: null, error: null }),
945
- };
946
-
947
- switch (tableName) {
948
- case 'rbac_global_roles':
949
- // Super admin check - no super admin
950
- callCount++;
951
- return {
952
- ...mockQuery,
953
- single: vi.fn().mockResolvedValue({
954
- data: null,
955
- error: { code: 'PGRST116' }
956
- })
957
- };
958
- case 'rbac_apps':
959
- // App configuration - default to requires_event: false
960
- callCount++;
961
- return {
962
- ...mockQuery,
963
- single: vi.fn().mockResolvedValue({
964
- data: { requires_event: false },
965
- error: null
966
- })
967
- };
968
- case 'rbac_page_permissions':
969
- // No page permissions
970
- callCount++;
971
- return {
972
- ...mockQuery,
973
- eq: vi.fn().mockResolvedValue({
974
- data: [],
975
- error: null
976
- })
977
- };
978
- case 'rbac_event_app_roles':
979
- // No event-app permissions
980
- callCount++;
981
- return {
982
- ...mockQuery,
983
- or: vi.fn().mockResolvedValue({
984
- data: [],
985
- error: null
986
- })
987
- };
988
- case 'rbac_organisation_roles':
989
- // Organisation permission that is VALID time-bound
990
- const pastDate = new Date(Date.now() - 24 * 60 * 60 * 1000).toISOString(); // Yesterday
991
- const futureDate = new Date(Date.now() + 24 * 60 * 60 * 1000).toISOString(); // Tomorrow
992
- callCount++;
993
- return {
994
- ...mockQuery,
995
- or: vi.fn().mockResolvedValue({
996
- data: [{
997
- role: 'org_admin',
998
- status: 'active',
999
- valid_from: pastDate,
1000
- valid_to: futureDate
1001
- }],
1002
- error: null
1003
- })
1004
- };
1005
- case 'rbac_audit_events':
1006
- // Audit event insert
1007
- callCount++;
1008
- return {
1009
- ...mockQuery,
1010
- insert: vi.fn().mockResolvedValue({ data: null, error: null })
1011
- };
1012
- default:
1013
- return mockQuery;
1014
- }
1015
- });
1016
-
1017
- const result = await engine.isPermitted({
1018
- userId,
1019
- scope,
1020
- permission
1021
- });
1022
-
1023
- expect(result).toBe(true); // Should allow for valid time-bound grants
1024
- expect(callCount).toBe(3); // Should check super admin + app config + organisation + audit
1025
- });
1026
- });
1027
-
1028
- describe('Page-level Rule Refinements', () => {
1029
- it('should apply page-level permissions to refine broader permissions', async () => {
1030
- const userId = 'user-123';
1031
- const scope: Scope = {
1032
- organisationId: 'org-456',
1033
- eventId: 'event-789',
1034
- appId: 'app-101'
1035
- };
1036
- const permission: Permission = 'manage:event.events';
1037
- const pageId = '12345678-1234-1234-1234-123456789012';
1038
-
1039
- // Mock all database calls with proper sequencing
1040
- let callCount = 0;
1041
-
1042
- // Mock RPC calls for get_rbac_permissions
1043
- mockSupabase.rpc.mockImplementation((fnName: string, params: any) => {
1044
- if (fnName === 'get_rbac_permissions') {
1045
- callCount++;
1046
- // Return page-level allow for event_admin role
1047
- return Promise.resolve({
1048
- data: [
1049
- { permission_type: 'manage', role_name: 'event_admin', has_permission: true, granted_at: '2024-01-01T00:00:00Z' }
1050
- ],
1051
- error: null
1052
- });
1053
- }
1054
- return Promise.resolve({ data: null, error: null });
1055
- });
1056
-
1057
- mockSupabase.from.mockImplementation((tableName: string) => {
1058
- const mockQuery = {
1059
- select: vi.fn().mockReturnThis(),
1060
- eq: vi.fn().mockReturnThis(),
1061
- is: vi.fn().mockReturnThis(),
1062
- lte: vi.fn().mockReturnThis(),
1063
- gte: vi.fn().mockReturnThis(),
1064
- or: vi.fn().mockReturnThis(),
1065
- and: vi.fn().mockReturnThis(),
1066
- order: vi.fn().mockReturnThis(),
1067
- in: vi.fn().mockReturnThis(),
1068
- single: vi.fn().mockResolvedValue({ data: null, error: null }),
1069
- maybeSingle: vi.fn().mockResolvedValue({ data: null, error: null }),
1070
- insert: vi.fn().mockResolvedValue({ data: null, error: null }),
1071
- };
1072
-
1073
- switch (tableName) {
1074
- case 'rbac_global_roles':
1075
- // Super admin check - no super admin
1076
- callCount++;
1077
- return {
1078
- ...mockQuery,
1079
- single: vi.fn().mockResolvedValue({
1080
- data: null,
1081
- error: { code: 'PGRST116' }
1082
- })
1083
- };
1084
- case 'rbac_apps':
1085
- // App configuration - default to requires_event: false
1086
- callCount++;
1087
- return {
1088
- ...mockQuery,
1089
- single: vi.fn().mockResolvedValue({
1090
- data: { requires_event: false },
1091
- error: null
1092
- })
1093
- };
1094
- case 'rbac_event_app_roles':
1095
- // Event-app permissions - user has event_admin role
1096
- callCount++;
1097
- return {
1098
- ...mockQuery,
1099
- or: vi.fn().mockResolvedValue({
1100
- data: [{
1101
- role: 'event_admin',
1102
- status: 'active',
1103
- valid_from: '2024-01-01T00:00:00Z',
1104
- valid_to: null
1105
- }],
1106
- error: null
1107
- })
1108
- };
1109
- case 'rbac_organisation_roles':
1110
- // Organisation permissions - empty
1111
- callCount++;
1112
- return {
1113
- ...mockQuery,
1114
- or: vi.fn().mockResolvedValue({
1115
- data: [],
1116
- error: null
1117
- })
1118
- };
1119
- case 'rbac_audit_events':
1120
- // Audit event insert
1121
- callCount++;
1122
- return {
1123
- ...mockQuery,
1124
- insert: vi.fn().mockResolvedValue({ data: null, error: null })
1125
- };
1126
- default:
1127
- return mockQuery;
1128
- }
1129
- });
1130
-
1131
- const result = await engine.isPermitted({
1132
- userId,
1133
- scope,
1134
- permission,
1135
- pageId
1136
- });
1137
-
1138
- expect(result).toBe(true); // Should allow when page permission allows
1139
- expect(callCount).toBe(6); // Should check: super_admin, app_config, event_app, org, global_roles, rpc (page via RPC)
1140
- });
1141
-
1142
- it('should deny access when page-level permission explicitly denies', async () => {
1143
- const userId = 'user-123';
1144
- const scope: Scope = {
1145
- organisationId: 'org-456',
1146
- eventId: 'event-789',
1147
- appId: 'app-101'
1148
- };
1149
- const permission: Permission = 'manage:page.events';
1150
- const pageId = '12345678-1234-1234-1234-123456789012';
1151
-
1152
- // Mock all database calls with proper sequencing
1153
- let callCount = 0;
1154
-
1155
- // Mock RPC calls for get_rbac_permissions
1156
- mockSupabase.rpc.mockImplementation((fnName: string, params: any) => {
1157
- if (fnName === 'get_rbac_permissions') {
1158
- callCount++;
1159
- // Return page-level deny for event_admin role
1160
- return Promise.resolve({
1161
- data: [
1162
- { permission_type: 'manage', role_name: 'event_admin', has_permission: false, granted_at: '2024-01-01T00:00:00Z' }
1163
- ],
1164
- error: null
1165
- });
1166
- }
1167
- return Promise.resolve({ data: null, error: null });
1168
- });
1169
-
1170
- mockSupabase.from.mockImplementation((tableName: string) => {
1171
- const mockQuery = {
1172
- select: vi.fn().mockReturnThis(),
1173
- eq: vi.fn().mockReturnThis(),
1174
- is: vi.fn().mockReturnThis(),
1175
- lte: vi.fn().mockReturnThis(),
1176
- gte: vi.fn().mockReturnThis(),
1177
- or: vi.fn().mockReturnThis(),
1178
- and: vi.fn().mockReturnThis(),
1179
- order: vi.fn().mockReturnThis(),
1180
- in: vi.fn().mockReturnThis(),
1181
- single: vi.fn().mockResolvedValue({ data: null, error: null }),
1182
- maybeSingle: vi.fn().mockResolvedValue({ data: null, error: null }),
1183
- insert: vi.fn().mockResolvedValue({ data: null, error: null }),
1184
- };
1185
-
1186
- switch (tableName) {
1187
- case 'rbac_app_pages':
1188
- // Return page name for permission matching
1189
- return {
1190
- ...mockQuery,
1191
- single: vi.fn().mockResolvedValue({
1192
- data: { page_name: 'events' },
1193
- error: null
1194
- })
1195
- };
1196
- case 'rbac_global_roles':
1197
- // Super admin check - no super admin
1198
- callCount++;
1199
- return {
1200
- ...mockQuery,
1201
- single: vi.fn().mockResolvedValue({
1202
- data: null,
1203
- error: { code: 'PGRST116' }
1204
- })
1205
- };
1206
- case 'rbac_apps':
1207
- // App configuration - default to requires_event: false
1208
- callCount++;
1209
- return {
1210
- ...mockQuery,
1211
- single: vi.fn().mockResolvedValue({
1212
- data: { requires_event: false },
1213
- error: null
1214
- })
1215
- };
1216
- case 'rbac_event_app_roles':
1217
- // Event-app permission that ALLOWS (should be overridden)
1218
- callCount++;
1219
- return {
1220
- ...mockQuery,
1221
- or: vi.fn().mockResolvedValue({
1222
- data: [{ role: 'event_admin', status: 'active', valid_from: '2024-01-01T00:00:00Z', valid_to: null }],
1223
- error: null
1224
- })
1225
- };
1226
- case 'rbac_organisation_roles':
1227
- // Organisation permission that ALLOWS (should be overridden)
1228
- callCount++;
1229
- return {
1230
- ...mockQuery,
1231
- or: vi.fn().mockResolvedValue({
1232
- data: [{ role: 'org_admin', status: 'active', valid_from: '2024-01-01T00:00:00Z', valid_to: null }],
1233
- error: null
1234
- })
1235
- };
1236
- case 'rbac_audit_events':
1237
- // Audit event insert
1238
- callCount++;
1239
- return {
1240
- ...mockQuery,
1241
- insert: vi.fn().mockResolvedValue({ data: null, error: null })
1242
- };
1243
- default:
1244
- return mockQuery;
1245
- }
1246
- });
1247
-
1248
- const result = await engine.isPermitted({
1249
- userId,
1250
- scope,
1251
- permission,
1252
- pageId
1253
- });
1254
-
1255
- expect(result).toBe(false); // Should deny when page permission denies
1256
- expect(callCount).toBe(6); // Should check: super_admin, app_config, event_app, org, global_roles, rpc (page via RPC)
1257
- });
1258
- });
1259
-
1260
- describe('Permission Matching', () => {
1261
- it('should match exact permissions', async () => {
1262
- const userId = 'user-123';
1263
- const scope: Scope = {
1264
- organisationId: 'org-456'
1265
- };
1266
- const permission: Permission = 'read:organisation.events';
1267
-
1268
- // Mock all database calls with proper sequencing
1269
- let callCount = 0;
1270
- mockSupabase.from.mockImplementation((tableName: string) => {
1271
- const mockQuery = {
1272
- select: vi.fn().mockReturnThis(),
1273
- eq: vi.fn().mockReturnThis(),
1274
- is: vi.fn().mockReturnThis(),
1275
- lte: vi.fn().mockReturnThis(),
1276
- gte: vi.fn().mockReturnThis(),
1277
- or: vi.fn().mockReturnThis(),
1278
- and: vi.fn().mockReturnThis(),
1279
- order: vi.fn().mockReturnThis(),
1280
- in: vi.fn().mockReturnThis(),
1281
- single: vi.fn().mockResolvedValue({ data: null, error: null }),
1282
- maybeSingle: vi.fn().mockResolvedValue({ data: null, error: null }),
1283
- insert: vi.fn().mockResolvedValue({ data: null, error: null }),
1284
- };
1285
-
1286
- switch (tableName) {
1287
- case 'rbac_global_roles':
1288
- // Super admin check - no super admin
1289
- callCount++;
1290
- return {
1291
- ...mockQuery,
1292
- single: vi.fn().mockResolvedValue({
1293
- data: null,
1294
- error: { code: 'PGRST116' }
1295
- })
1296
- };
1297
- case 'rbac_apps':
1298
- // App configuration - default to requires_event: false
1299
- callCount++;
1300
- return {
1301
- ...mockQuery,
1302
- single: vi.fn().mockResolvedValue({
1303
- data: { requires_event: false },
1304
- error: null
1305
- })
1306
- };
1307
- case 'rbac_page_permissions':
1308
- // No page permissions
1309
- callCount++;
1310
- return {
1311
- ...mockQuery,
1312
- eq: vi.fn().mockResolvedValue({
1313
- data: [],
1314
- error: null
1315
- })
1316
- };
1317
- case 'rbac_event_app_roles':
1318
- // No event-app permissions
1319
- callCount++;
1320
- return {
1321
- ...mockQuery,
1322
- or: vi.fn().mockResolvedValue({
1323
- data: [],
1324
- error: null
1325
- })
1326
- };
1327
- case 'rbac_organisation_roles':
1328
- // Organisation permissions - member role
1329
- callCount++;
1330
- return {
1331
- ...mockQuery,
1332
- or: vi.fn().mockResolvedValue({
1333
- data: [{
1334
- role: 'member',
1335
- status: 'active',
1336
- valid_from: '2024-01-01T00:00:00Z',
1337
- valid_to: null
1338
- }],
1339
- error: null
1340
- })
1341
- };
1342
- case 'rbac_audit_events':
1343
- // Audit event insert
1344
- callCount++;
1345
- return {
1346
- ...mockQuery,
1347
- insert: vi.fn().mockResolvedValue({ data: null, error: null })
1348
- };
1349
- default:
1350
- return mockQuery;
1351
- }
1352
- });
1353
-
1354
- const result = await engine.isPermitted({
1355
- userId,
1356
- scope,
1357
- permission
1358
- });
1359
-
1360
- expect(result).toBe(true); // Should match exact permission
1361
- expect(callCount).toBe(3); // Should check super admin + app config + organisation + audit
1362
- });
1363
-
1364
- it('should match wildcard permissions', async () => {
1365
- const userId = 'user-123';
1366
- const scope: Scope = {
1367
- organisationId: 'org-456'
1368
- };
1369
- const permission: Permission = 'manage:organisation.events.details';
1370
-
1371
- // Mock all database calls with proper sequencing
1372
- let callCount = 0;
1373
- mockSupabase.from.mockImplementation((tableName: string) => {
1374
- const mockQuery = {
1375
- select: vi.fn().mockReturnThis(),
1376
- eq: vi.fn().mockReturnThis(),
1377
- is: vi.fn().mockReturnThis(),
1378
- lte: vi.fn().mockReturnThis(),
1379
- gte: vi.fn().mockReturnThis(),
1380
- or: vi.fn().mockReturnThis(),
1381
- and: vi.fn().mockReturnThis(),
1382
- order: vi.fn().mockReturnThis(),
1383
- in: vi.fn().mockReturnThis(),
1384
- single: vi.fn().mockResolvedValue({ data: null, error: null }),
1385
- maybeSingle: vi.fn().mockResolvedValue({ data: null, error: null }),
1386
- insert: vi.fn().mockResolvedValue({ data: null, error: null }),
1387
- };
1388
-
1389
- switch (tableName) {
1390
- case 'rbac_global_roles':
1391
- // Super admin check - no super admin
1392
- callCount++;
1393
- return {
1394
- ...mockQuery,
1395
- single: vi.fn().mockResolvedValue({
1396
- data: null,
1397
- error: { code: 'PGRST116' }
1398
- })
1399
- };
1400
- case 'rbac_apps':
1401
- // App configuration - default to requires_event: false
1402
- callCount++;
1403
- return {
1404
- ...mockQuery,
1405
- single: vi.fn().mockResolvedValue({
1406
- data: { requires_event: false },
1407
- error: null
1408
- })
1409
- };
1410
- case 'rbac_page_permissions':
1411
- // No page permissions
1412
- callCount++;
1413
- return {
1414
- ...mockQuery,
1415
- eq: vi.fn().mockResolvedValue({
1416
- data: [],
1417
- error: null
1418
- })
1419
- };
1420
- case 'rbac_event_app_roles':
1421
- // No event-app permissions
1422
- callCount++;
1423
- return {
1424
- ...mockQuery,
1425
- or: vi.fn().mockResolvedValue({
1426
- data: [],
1427
- error: null
1428
- })
1429
- };
1430
- case 'rbac_organisation_roles':
1431
- // Organisation permissions - org_admin role
1432
- callCount++;
1433
- return {
1434
- ...mockQuery,
1435
- or: vi.fn().mockResolvedValue({
1436
- data: [{
1437
- role: 'org_admin',
1438
- status: 'active',
1439
- valid_from: '2024-01-01T00:00:00Z',
1440
- valid_to: null
1441
- }],
1442
- error: null
1443
- })
1444
- };
1445
- case 'rbac_audit_events':
1446
- // Audit event insert
1447
- callCount++;
1448
- return {
1449
- ...mockQuery,
1450
- insert: vi.fn().mockResolvedValue({ data: null, error: null })
1451
- };
1452
- default:
1453
- return mockQuery;
1454
- }
1455
- });
1456
-
1457
- const result = await engine.isPermitted({
1458
- userId,
1459
- scope,
1460
- permission
1461
- });
1462
-
1463
- expect(result).toBe(true); // Should match wildcard permission
1464
- expect(callCount).toBe(3); // Should check super admin + app config + organisation + audit
1465
- });
1466
- });
1467
- });