@scality/data-browser-library 1.0.0-preview.9 → 1.0.3

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 (287) hide show
  1. package/dist/components/DataBrowserUI.d.ts +12 -0
  2. package/dist/components/DataBrowserUI.js +99 -0
  3. package/dist/components/Editor.d.ts +1 -1
  4. package/dist/components/Editor.js +3 -3
  5. package/dist/components/__tests__/BucketAccessor.test.js +214 -0
  6. package/dist/components/__tests__/BucketCorsPage.test.d.ts +1 -0
  7. package/dist/components/__tests__/BucketCorsPage.test.js +263 -0
  8. package/dist/components/__tests__/BucketCreate.test.js +271 -105
  9. package/dist/components/__tests__/BucketDetails.test.d.ts +1 -0
  10. package/dist/components/__tests__/BucketDetails.test.js +421 -0
  11. package/dist/components/__tests__/BucketLifecycleFormPage.test.d.ts +13 -0
  12. package/dist/components/__tests__/BucketLifecycleFormPage.test.js +178 -178
  13. package/dist/components/__tests__/BucketLifecycleList.test.js +85 -85
  14. package/dist/components/__tests__/BucketList.test.js +463 -239
  15. package/dist/components/__tests__/BucketNotificationFormPage.test.d.ts +1 -0
  16. package/dist/components/__tests__/BucketNotificationFormPage.test.js +348 -0
  17. package/dist/components/__tests__/BucketNotificationList.test.d.ts +1 -0
  18. package/dist/components/__tests__/BucketNotificationList.test.js +379 -0
  19. package/dist/components/__tests__/BucketOverview.test.js +281 -266
  20. package/dist/components/__tests__/BucketPolicyPage.test.js +151 -99
  21. package/dist/components/__tests__/BucketReplicationFormPage.test.d.ts +15 -0
  22. package/dist/components/__tests__/BucketReplicationFormPage.test.js +544 -544
  23. package/dist/components/__tests__/BucketReplicationList.test.js +106 -106
  24. package/dist/components/__tests__/CreateFolderButton.test.js +56 -56
  25. package/dist/components/__tests__/DeleteBucketButton.test.js +64 -64
  26. package/dist/components/__tests__/DeleteBucketConfigRuleButton.test.js +47 -47
  27. package/dist/components/__tests__/DeleteObjectButton.test.js +64 -64
  28. package/dist/components/__tests__/EmptyBucketButton.test.js +59 -59
  29. package/dist/components/__tests__/MetadataSearch.test.js +65 -65
  30. package/dist/components/__tests__/ObjectList.test.js +741 -240
  31. package/dist/components/__tests__/UploadButton.test.js +45 -45
  32. package/dist/components/breadcrumb/Breadcrumb.d.ts +6 -0
  33. package/dist/components/breadcrumb/Breadcrumb.js +37 -0
  34. package/dist/components/breadcrumb/DataBrowserBreadcrumb.d.ts +1 -0
  35. package/dist/components/breadcrumb/DataBrowserBreadcrumb.js +10 -0
  36. package/dist/components/breadcrumb/__tests__/Breadcrumb.test.d.ts +1 -0
  37. package/dist/components/breadcrumb/__tests__/Breadcrumb.test.js +196 -0
  38. package/dist/components/breadcrumb/__tests__/DataBrowserBreadcrumb.test.d.ts +1 -0
  39. package/dist/components/breadcrumb/__tests__/DataBrowserBreadcrumb.test.js +153 -0
  40. package/dist/components/breadcrumb/__tests__/useBreadcrumbPaths.test.d.ts +1 -0
  41. package/dist/components/breadcrumb/__tests__/useBreadcrumbPaths.test.js +134 -0
  42. package/dist/components/breadcrumb/index.d.ts +8 -0
  43. package/dist/components/breadcrumb/index.js +4 -0
  44. package/dist/components/breadcrumb/useBreadcrumbPaths.d.ts +2 -0
  45. package/dist/components/breadcrumb/useBreadcrumbPaths.js +82 -0
  46. package/dist/components/buckets/BucketAccessor.d.ts +2 -0
  47. package/dist/components/buckets/BucketAccessor.js +125 -0
  48. package/dist/components/buckets/BucketConfigEditButton.d.ts +8 -0
  49. package/dist/components/buckets/{BucketPolicyButton.js → BucketConfigEditButton.js} +9 -5
  50. package/dist/components/buckets/BucketCorsPage.d.ts +1 -0
  51. package/dist/components/buckets/BucketCorsPage.js +234 -0
  52. package/dist/components/buckets/BucketCreate.d.ts +3 -2
  53. package/dist/components/buckets/BucketCreate.js +93 -47
  54. package/dist/components/buckets/BucketDetails.d.ts +42 -0
  55. package/dist/components/buckets/BucketDetails.js +249 -85
  56. package/dist/components/buckets/BucketLifecycleFormPage.js +225 -191
  57. package/dist/components/buckets/BucketLifecycleList.d.ts +2 -2
  58. package/dist/components/buckets/BucketLifecycleList.js +59 -61
  59. package/dist/components/buckets/BucketList.d.ts +7 -8
  60. package/dist/components/buckets/BucketList.js +158 -101
  61. package/dist/components/buckets/BucketLocation.js +4 -4
  62. package/dist/components/buckets/BucketOverview.d.ts +22 -2
  63. package/dist/components/buckets/BucketOverview.js +394 -187
  64. package/dist/components/buckets/BucketPage.js +43 -21
  65. package/dist/components/buckets/BucketPolicyPage.js +155 -127
  66. package/dist/components/buckets/BucketReplicationFormPage.js +134 -133
  67. package/dist/components/buckets/BucketReplicationList.d.ts +2 -2
  68. package/dist/components/buckets/BucketReplicationList.js +48 -45
  69. package/dist/components/buckets/BucketVersioning.d.ts +4 -0
  70. package/dist/components/buckets/BucketVersioning.js +76 -0
  71. package/dist/components/buckets/DeleteBucketButton.js +8 -8
  72. package/dist/components/buckets/DeleteBucketConfigRuleButton.d.ts +2 -2
  73. package/dist/components/buckets/DeleteBucketConfigRuleButton.js +2 -2
  74. package/dist/components/buckets/EmptyBucketButton.js +24 -24
  75. package/dist/components/buckets/EmptyBucketSummary.d.ts +2 -2
  76. package/dist/components/buckets/EmptyBucketSummary.js +1 -1
  77. package/dist/components/buckets/EmptyBucketSummaryList.d.ts +1 -1
  78. package/dist/components/buckets/EmptyBucketSummaryList.js +22 -22
  79. package/dist/components/buckets/__tests__/BucketVersioning.test.d.ts +1 -0
  80. package/dist/components/buckets/__tests__/BucketVersioning.test.js +163 -0
  81. package/dist/components/buckets/notifications/BucketNotificationFormPage.d.ts +1 -0
  82. package/dist/components/buckets/notifications/BucketNotificationFormPage.js +316 -0
  83. package/dist/components/buckets/notifications/BucketNotificationList.d.ts +10 -0
  84. package/dist/components/buckets/notifications/BucketNotificationList.js +267 -0
  85. package/dist/components/buckets/notifications/EventsSection.js +145 -29
  86. package/dist/components/buckets/notifications/__tests__/events.test.d.ts +1 -0
  87. package/dist/components/buckets/notifications/__tests__/events.test.js +56 -0
  88. package/dist/components/buckets/notifications/events.d.ts +71 -7
  89. package/dist/components/buckets/notifications/events.js +98 -16
  90. package/dist/components/index.d.ts +27 -20
  91. package/dist/components/index.js +17 -10
  92. package/dist/components/layouts/ArrowNavigation.d.ts +3 -0
  93. package/dist/components/layouts/ArrowNavigation.js +28 -0
  94. package/dist/components/layouts/BrowserPageLayout.d.ts +5 -1
  95. package/dist/components/layouts/BrowserPageLayout.js +10 -5
  96. package/dist/components/objects/CreateFolderButton.d.ts +2 -2
  97. package/dist/components/objects/CreateFolderButton.js +12 -12
  98. package/dist/components/objects/DeleteObjectButton.d.ts +1 -1
  99. package/dist/components/objects/DeleteObjectButton.js +19 -21
  100. package/dist/components/objects/GetPresignedUrlButton.d.ts +7 -0
  101. package/dist/components/objects/GetPresignedUrlButton.js +255 -0
  102. package/dist/components/objects/ObjectDetails/ObjectMetadata.d.ts +2 -2
  103. package/dist/components/objects/ObjectDetails/ObjectMetadata.js +289 -230
  104. package/dist/components/objects/ObjectDetails/ObjectSummary.d.ts +2 -2
  105. package/dist/components/objects/ObjectDetails/ObjectSummary.js +540 -138
  106. package/dist/components/objects/ObjectDetails/ObjectTags.d.ts +2 -2
  107. package/dist/components/objects/ObjectDetails/ObjectTags.js +103 -123
  108. package/dist/components/objects/ObjectDetails/__tests__/ObjectDetails.test.d.ts +1 -0
  109. package/dist/components/objects/ObjectDetails/__tests__/ObjectDetails.test.js +516 -0
  110. package/dist/components/objects/ObjectDetails/__tests__/ObjectSummary.test.d.ts +1 -0
  111. package/dist/components/objects/ObjectDetails/__tests__/ObjectSummary.test.js +1064 -0
  112. package/dist/components/objects/ObjectDetails/formUtils.d.ts +15 -0
  113. package/dist/components/objects/ObjectDetails/formUtils.js +7 -0
  114. package/dist/components/objects/ObjectDetails/index.d.ts +18 -2
  115. package/dist/components/objects/ObjectDetails/index.js +152 -40
  116. package/dist/components/objects/ObjectList.d.ts +12 -10
  117. package/dist/components/objects/ObjectList.js +590 -263
  118. package/dist/components/objects/ObjectLock/EditRetentionButton.js +4 -4
  119. package/dist/components/objects/ObjectLock/ObjectLockRetentionSettings.js +15 -15
  120. package/dist/components/objects/ObjectLock/ObjectLockSettings.d.ts +1 -1
  121. package/dist/components/objects/ObjectLock/ObjectLockSettings.js +32 -31
  122. package/dist/components/objects/ObjectLock/ObjectLockSettingsUtils.d.ts +1 -1
  123. package/dist/components/objects/ObjectLock/ObjectLockSettingsUtils.js +6 -6
  124. package/dist/components/objects/ObjectLock/__tests__/EditRetentionButton.test.js +51 -51
  125. package/dist/components/objects/ObjectLock/__tests__/ObjectLockSettings.test.js +78 -78
  126. package/dist/components/objects/ObjectPage.js +12 -8
  127. package/dist/components/objects/UploadButton.d.ts +3 -3
  128. package/dist/components/objects/UploadButton.js +10 -10
  129. package/dist/components/objects/__tests__/GetPresignedUrlButton.test.d.ts +1 -0
  130. package/dist/components/objects/__tests__/GetPresignedUrlButton.test.js +531 -0
  131. package/dist/components/providers/DataBrowserProvider.d.ts +23 -12
  132. package/dist/components/providers/DataBrowserProvider.js +60 -38
  133. package/dist/components/providers/QueryProvider.d.ts +9 -0
  134. package/dist/components/providers/QueryProvider.js +21 -0
  135. package/dist/components/search/MetadataSearch.js +29 -28
  136. package/dist/components/search/SearchHints.js +1 -1
  137. package/dist/components/ui/ArrayFieldActions.js +12 -7
  138. package/dist/components/ui/ConfirmDeleteRuleModal.d.ts +2 -2
  139. package/dist/components/ui/ConfirmDeleteRuleModal.js +6 -1
  140. package/dist/components/ui/DeleteObjectModalContent.d.ts +1 -1
  141. package/dist/components/ui/DeleteObjectModalContent.js +12 -12
  142. package/dist/components/ui/FilterFormSection.d.ts +2 -2
  143. package/dist/components/ui/FilterFormSection.js +29 -29
  144. package/dist/components/ui/Search.elements.d.ts +2 -2
  145. package/dist/components/ui/Search.elements.js +7 -7
  146. package/dist/components/ui/Table.elements.d.ts +2 -1
  147. package/dist/components/ui/Table.elements.js +18 -12
  148. package/dist/config/__tests__/factory.test.d.ts +1 -0
  149. package/dist/config/__tests__/factory.test.js +311 -0
  150. package/dist/config/factory.d.ts +10 -56
  151. package/dist/config/factory.js +23 -71
  152. package/dist/config/types.d.ts +212 -34
  153. package/dist/contexts/DataBrowserUICustomizationContext.d.ts +27 -0
  154. package/dist/contexts/DataBrowserUICustomizationContext.js +13 -0
  155. package/dist/hooks/__tests__/useAccessibleBuckets.test.d.ts +1 -0
  156. package/dist/hooks/__tests__/useAccessibleBuckets.test.js +145 -0
  157. package/dist/hooks/__tests__/useISVBucketDetection.test.js +45 -45
  158. package/dist/hooks/__tests__/useIsBucketEmpty.test.js +27 -27
  159. package/dist/hooks/__tests__/useLoginMutation.test.d.ts +1 -0
  160. package/dist/hooks/__tests__/useLoginMutation.test.js +194 -0
  161. package/dist/hooks/bucketConfiguration.d.ts +8 -1
  162. package/dist/hooks/bucketConfiguration.js +52 -51
  163. package/dist/hooks/bucketOperations.d.ts +10 -1
  164. package/dist/hooks/bucketOperations.js +10 -9
  165. package/dist/hooks/factories/__tests__/useCreateS3FunctionMutationHook.test.js +80 -80
  166. package/dist/hooks/factories/__tests__/useCreateS3InfiniteQueryHook.test.js +80 -80
  167. package/dist/hooks/factories/__tests__/useCreateS3LoginHook.test.js +44 -44
  168. package/dist/hooks/factories/__tests__/useCreateS3MutationHook.test.js +63 -63
  169. package/dist/hooks/factories/__tests__/useCreateS3QueryHook.test.js +65 -65
  170. package/dist/hooks/factories/index.d.ts +4 -4
  171. package/dist/hooks/factories/index.js +2 -2
  172. package/dist/hooks/factories/useCreateS3InfiniteQueryHook.d.ts +2 -2
  173. package/dist/hooks/factories/useCreateS3InfiniteQueryHook.js +16 -13
  174. package/dist/hooks/factories/useCreateS3LoginHook.d.ts +2 -2
  175. package/dist/hooks/factories/useCreateS3LoginHook.js +1 -1
  176. package/dist/hooks/factories/useCreateS3MutationHook.d.ts +3 -3
  177. package/dist/hooks/factories/useCreateS3MutationHook.js +7 -2
  178. package/dist/hooks/factories/useCreateS3QueryHook.d.ts +2 -2
  179. package/dist/hooks/factories/useCreateS3QueryHook.js +11 -6
  180. package/dist/hooks/index.d.ts +19 -12
  181. package/dist/hooks/index.js +16 -9
  182. package/dist/hooks/loginOperations.d.ts +1 -1
  183. package/dist/hooks/loginOperations.js +1 -1
  184. package/dist/hooks/objectOperations.d.ts +2 -2
  185. package/dist/hooks/objectOperations.js +50 -49
  186. package/dist/hooks/presignedOperations.d.ts +4 -4
  187. package/dist/hooks/presignedOperations.js +5 -5
  188. package/dist/hooks/useAccessibleBuckets.d.ts +11 -0
  189. package/dist/hooks/useAccessibleBuckets.js +115 -0
  190. package/dist/hooks/useBatchObjectLegalHold.d.ts +11 -0
  191. package/dist/hooks/useBatchObjectLegalHold.js +48 -0
  192. package/dist/hooks/useBucketConfigEditor.d.ts +31 -0
  193. package/dist/hooks/useBucketConfigEditor.js +82 -0
  194. package/dist/hooks/useDataBrowserNavigate.d.ts +28 -0
  195. package/dist/hooks/useDataBrowserNavigate.js +24 -0
  196. package/dist/hooks/useDeleteBucketConfigRule.d.ts +2 -2
  197. package/dist/hooks/useDeleteBucketConfigRule.js +4 -4
  198. package/dist/hooks/useEmptyBucket.js +11 -11
  199. package/dist/hooks/useFeatures.d.ts +7 -0
  200. package/dist/hooks/useFeatures.js +8 -0
  201. package/dist/hooks/useISVBucketDetection.js +6 -6
  202. package/dist/hooks/useIsBucketEmpty.js +4 -4
  203. package/dist/hooks/useLimitedAccessFlow.d.ts +48 -0
  204. package/dist/hooks/useLimitedAccessFlow.js +23 -0
  205. package/dist/hooks/useS3Client.d.ts +6 -0
  206. package/dist/hooks/useS3Client.js +3 -2
  207. package/dist/hooks/useS3ConfigSwitch.d.ts +11 -0
  208. package/dist/hooks/useS3ConfigSwitch.js +37 -0
  209. package/dist/hooks/useSupportedNotificationEvents.d.ts +6 -0
  210. package/dist/hooks/useSupportedNotificationEvents.js +8 -0
  211. package/dist/index.d.ts +6 -6
  212. package/dist/index.js +2 -2
  213. package/dist/schemas/bucketPolicySchema.json +3 -13
  214. package/dist/test/msw/handlers/deleteBucket.d.ts +1 -1
  215. package/dist/test/msw/handlers/deleteBucket.js +20 -10
  216. package/dist/test/msw/handlers/getBucketAcl.d.ts +1 -1
  217. package/dist/test/msw/handlers/getBucketAcl.js +29 -17
  218. package/dist/test/msw/handlers/getBucketLocation.d.ts +1 -1
  219. package/dist/test/msw/handlers/getBucketLocation.js +29 -15
  220. package/dist/test/msw/handlers/getBucketPolicy.d.ts +1 -1
  221. package/dist/test/msw/handlers/getBucketPolicy.js +52 -32
  222. package/dist/test/msw/handlers/headObject.d.ts +1 -1
  223. package/dist/test/msw/handlers/headObject.js +31 -13
  224. package/dist/test/msw/handlers/listBuckets.d.ts +1 -1
  225. package/dist/test/msw/handlers/listBuckets.js +5 -3
  226. package/dist/test/msw/handlers/listObjectVersions.d.ts +1 -1
  227. package/dist/test/msw/handlers/listObjectVersions.js +38 -26
  228. package/dist/test/msw/handlers/listObjects.d.ts +1 -1
  229. package/dist/test/msw/handlers/listObjects.js +35 -23
  230. package/dist/test/msw/handlers/objectLegalHold.d.ts +1 -1
  231. package/dist/test/msw/handlers/objectLegalHold.js +32 -17
  232. package/dist/test/msw/handlers/objectRetention.d.ts +1 -1
  233. package/dist/test/msw/handlers/objectRetention.js +31 -17
  234. package/dist/test/msw/handlers/putBucketAcl.d.ts +1 -1
  235. package/dist/test/msw/handlers/putBucketAcl.js +29 -14
  236. package/dist/test/msw/handlers/putObject.d.ts +1 -1
  237. package/dist/test/msw/handlers/putObject.js +27 -12
  238. package/dist/test/msw/handlers.d.ts +3 -3
  239. package/dist/test/msw/handlers.js +77 -54
  240. package/dist/test/msw/index.d.ts +2 -2
  241. package/dist/test/msw/index.js +1 -1
  242. package/dist/test/msw/server.d.ts +1 -1
  243. package/dist/test/msw/server.js +1 -1
  244. package/dist/test/msw/utils.js +2 -2
  245. package/dist/test/setup.d.ts +1 -1
  246. package/dist/test/setup.js +13 -30
  247. package/dist/test/testUtils.d.ts +85 -33
  248. package/dist/test/testUtils.js +176 -111
  249. package/dist/test/utils/errorHandling.test.js +119 -119
  250. package/dist/types/index.d.ts +50 -37
  251. package/dist/types/monaco.d.ts +13 -0
  252. package/dist/types/monaco.js +0 -0
  253. package/dist/utils/__tests__/proxyMiddleware.test.d.ts +1 -0
  254. package/dist/utils/__tests__/proxyMiddleware.test.js +579 -0
  255. package/dist/utils/__tests__/s3Client.test.d.ts +1 -0
  256. package/dist/utils/__tests__/s3Client.test.js +340 -0
  257. package/dist/utils/__tests__/s3ConfigIdentifier.test.d.ts +1 -0
  258. package/dist/utils/__tests__/s3ConfigIdentifier.test.js +437 -0
  259. package/dist/utils/constants.d.ts +10 -0
  260. package/dist/utils/constants.js +19 -9
  261. package/dist/utils/deletion/index.d.ts +2 -2
  262. package/dist/utils/deletion/index.js +1 -1
  263. package/dist/utils/deletion/messages.d.ts +1 -1
  264. package/dist/utils/deletion/messages.js +4 -4
  265. package/dist/utils/errorHandling.d.ts +3 -3
  266. package/dist/utils/errorHandling.js +6 -6
  267. package/dist/utils/hooks.js +8 -8
  268. package/dist/utils/index.d.ts +5 -4
  269. package/dist/utils/index.js +4 -2
  270. package/dist/utils/proxyMiddleware.d.ts +32 -13
  271. package/dist/utils/proxyMiddleware.js +90 -36
  272. package/dist/utils/s3Client.d.ts +14 -4
  273. package/dist/utils/s3Client.js +5 -26
  274. package/dist/utils/s3ConfigIdentifier.d.ts +79 -0
  275. package/dist/utils/s3ConfigIdentifier.js +57 -0
  276. package/dist/utils/s3RuleUtils.d.ts +5 -5
  277. package/dist/utils/s3RuleUtils.js +17 -17
  278. package/package.json +10 -8
  279. package/dist/components/__tests__/BucketNotificationCreatePage.test.js +0 -316
  280. package/dist/components/buckets/BucketPolicyButton.d.ts +0 -7
  281. package/dist/components/buckets/notifications/BucketNotificationCreatePage.d.ts +0 -1
  282. package/dist/components/buckets/notifications/BucketNotificationCreatePage.js +0 -234
  283. package/dist/hooks/useLoginMutation.d.ts +0 -21
  284. package/dist/hooks/useLoginMutation.js +0 -9
  285. package/dist/utils/useFeatures.d.ts +0 -1
  286. package/dist/utils/useFeatures.js +0 -7
  287. /package/dist/components/__tests__/{BucketNotificationCreatePage.test.d.ts → BucketAccessor.test.d.ts} +0 -0
@@ -1,16 +1,16 @@
1
1
  import { jsx, jsxs } from "react/jsx-runtime";
2
2
  import { fireEvent, render, screen, waitFor } from "@testing-library/react";
3
3
  import user_event from "@testing-library/user-event";
4
- import { MemoryRouter, Route, Routes } from "react-router-dom";
5
- import { createTestWrapper, findToggleByLabel, mockErrorSubmit, mockOffsetSize, mockSuccessSubmit, submitForm } from "../../test/testUtils.js";
6
- import { BucketReplicationFormPage } from "../buckets/BucketReplicationFormPage.js";
4
+ import { MemoryRouter, Route, Routes } from "react-router";
7
5
  import { useGetBucketReplication, useSetBucketReplication } from "../../hooks/bucketConfiguration.js";
8
6
  import { useBuckets } from "../../hooks/bucketOperations.js";
9
- jest.mock("../../hooks/bucketConfiguration", ()=>({
7
+ import { createTestWrapper, findToggleByLabel, mockErrorSubmit, mockOffsetSize, mockSuccessSubmit, submitForm } from "../../test/testUtils.js";
8
+ import { BucketReplicationFormPage } from "../buckets/BucketReplicationFormPage.js";
9
+ jest.mock('../../hooks/bucketConfiguration', ()=>({
10
10
  useGetBucketReplication: jest.fn(),
11
11
  useSetBucketReplication: jest.fn()
12
12
  }));
13
- jest.mock("../../hooks/bucketOperations", ()=>({
13
+ jest.mock('../../hooks/bucketOperations', ()=>({
14
14
  useBuckets: jest.fn()
15
15
  }));
16
16
  const mockUseGetBucketReplication = jest.mocked(useGetBucketReplication);
@@ -18,17 +18,17 @@ const mockUseSetBucketReplication = jest.mocked(useSetBucketReplication);
18
18
  const mockUseBuckets = jest.mocked(useBuckets);
19
19
  const mockNavigate = jest.fn();
20
20
  const mockShowToast = jest.fn();
21
- jest.mock("react-router-dom", ()=>({
22
- ...jest.requireActual("react-router-dom"),
21
+ jest.mock('react-router', ()=>({
22
+ ...jest.requireActual('react-router'),
23
23
  useNavigate: ()=>mockNavigate
24
24
  }));
25
- jest.mock("@scality/core-ui", ()=>({
26
- ...jest.requireActual("@scality/core-ui"),
25
+ jest.mock('@scality/core-ui', ()=>({
26
+ ...jest.requireActual('@scality/core-ui'),
27
27
  useToast: ()=>({
28
28
  showToast: mockShowToast
29
29
  })
30
30
  }));
31
- const renderBucketReplicationFormPage = (bucketName = "test-bucket", ruleId)=>{
31
+ const renderBucketReplicationFormPage = (bucketName = 'test-bucket', ruleId)=>{
32
32
  const Wrapper = createTestWrapper();
33
33
  const path = ruleId ? `/buckets/${bucketName}/replication/${ruleId}/edit` : `/buckets/${bucketName}/replication/create`;
34
34
  return render(/*#__PURE__*/ jsx(MemoryRouter, {
@@ -51,15 +51,15 @@ const renderBucketReplicationFormPage = (bucketName = "test-bucket", ruleId)=>{
51
51
  })
52
52
  }));
53
53
  };
54
- describe("BucketReplicationFormPage", ()=>{
54
+ describe('BucketReplicationFormPage', ()=>{
55
55
  const mockMutate = jest.fn();
56
56
  const fillRequiredFields = async (options)=>{
57
- const { roleArn = "arn:aws:iam::123456789012:role/replication-role", ruleId, targetBucket = "destination-bucket" } = options;
57
+ const { roleArn = 'arn:aws:iam::123456789012:role/replication-role', ruleId, targetBucket = 'destination-bucket' } = options;
58
58
  await user_event.type(screen.getByLabelText(/role arn/i), roleArn);
59
59
  await user_event.type(screen.getByLabelText(/rule id/i), ruleId);
60
60
  const targetBucketSelect = screen.getByLabelText(/target bucket/i);
61
61
  await user_event.click(targetBucketSelect);
62
- await user_event.click(screen.getByRole("option", {
62
+ await user_event.click(screen.getByRole('option', {
63
63
  name: targetBucket
64
64
  }));
65
65
  };
@@ -70,7 +70,7 @@ describe("BucketReplicationFormPage", ()=>{
70
70
  mockOffsetSize(800, 600);
71
71
  mockUseGetBucketReplication.mockReturnValue({
72
72
  data: void 0,
73
- status: "success"
73
+ status: 'success'
74
74
  });
75
75
  mockUseSetBucketReplication.mockReturnValue({
76
76
  mutate: mockMutate,
@@ -80,153 +80,153 @@ describe("BucketReplicationFormPage", ()=>{
80
80
  data: {
81
81
  Buckets: [
82
82
  {
83
- Name: "test-bucket"
83
+ Name: 'test-bucket'
84
84
  },
85
85
  {
86
- Name: "destination-bucket"
86
+ Name: 'destination-bucket'
87
87
  },
88
88
  {
89
- Name: "backup-bucket"
89
+ Name: 'backup-bucket'
90
90
  }
91
91
  ]
92
92
  },
93
- status: "success"
93
+ status: 'success'
94
94
  });
95
95
  });
96
- describe("Page Rendering", ()=>{
97
- it("renders create mode with correct title", ()=>{
96
+ describe('Page Rendering', ()=>{
97
+ it('renders create mode with correct title', ()=>{
98
98
  renderBucketReplicationFormPage();
99
- expect(screen.getByText("Create Replication Rule")).toBeInTheDocument();
99
+ expect(screen.getByText('Create Replication Rule')).toBeInTheDocument();
100
100
  });
101
- it("renders edit mode with correct title when rule exists", async ()=>{
101
+ it('renders edit mode with correct title when rule exists', async ()=>{
102
102
  const existingRule = {
103
- ID: "test-rule",
104
- Status: "Enabled",
103
+ ID: 'test-rule',
104
+ Status: 'Enabled',
105
105
  Priority: 1,
106
106
  Destination: {
107
- Bucket: "arn:aws:s3:::destination-bucket"
107
+ Bucket: 'arn:aws:s3:::destination-bucket'
108
108
  }
109
109
  };
110
110
  mockUseGetBucketReplication.mockReturnValue({
111
111
  data: {
112
112
  ReplicationConfiguration: {
113
- Role: "arn:aws:iam::123456789012:role/replication-role",
113
+ Role: 'arn:aws:iam::123456789012:role/replication-role',
114
114
  Rules: [
115
115
  existingRule
116
116
  ]
117
117
  }
118
118
  },
119
- status: "success"
119
+ status: 'success'
120
120
  });
121
- renderBucketReplicationFormPage("test-bucket", "test-rule");
121
+ renderBucketReplicationFormPage('test-bucket', 'test-rule');
122
122
  await waitFor(()=>{
123
- expect(screen.getByText("Edit Replication Rule")).toBeInTheDocument();
123
+ expect(screen.getByText('Edit Replication Rule')).toBeInTheDocument();
124
124
  });
125
125
  });
126
- it("shows loading state when fetching replication data", ()=>{
126
+ it('shows loading state when fetching replication data', ()=>{
127
127
  mockUseGetBucketReplication.mockReturnValue({
128
128
  data: void 0,
129
- status: "pending"
129
+ status: 'pending'
130
130
  });
131
131
  renderBucketReplicationFormPage();
132
- expect(screen.getByText("Loading...")).toBeInTheDocument();
132
+ expect(screen.getByText('Loading...')).toBeInTheDocument();
133
133
  });
134
- it("shows error when rule not found in edit mode", ()=>{
134
+ it('shows error when rule not found in edit mode', ()=>{
135
135
  mockUseGetBucketReplication.mockReturnValue({
136
136
  data: {
137
137
  ReplicationConfiguration: {
138
- Role: "arn:aws:iam::123456789012:role/replication-role",
138
+ Role: 'arn:aws:iam::123456789012:role/replication-role',
139
139
  Rules: []
140
140
  }
141
141
  },
142
- status: "success"
142
+ status: 'success'
143
143
  });
144
- renderBucketReplicationFormPage("test-bucket", "non-existent");
145
- expect(screen.getByText("Rule not found")).toBeInTheDocument();
144
+ renderBucketReplicationFormPage('test-bucket', 'non-existent');
145
+ expect(screen.getByText('Rule not found')).toBeInTheDocument();
146
146
  });
147
- it("displays all form sections", async ()=>{
147
+ it('displays all form sections', async ()=>{
148
148
  renderBucketReplicationFormPage();
149
149
  await waitFor(()=>{
150
150
  expect(screen.getByLabelText(/role arn/i)).toBeInTheDocument();
151
151
  expect(screen.getByLabelText(/rule id/i)).toBeInTheDocument();
152
- expect(screen.getByText("Replicate encrypted objects")).toBeInTheDocument();
153
- expect(screen.getByText("Delete marker replication")).toBeInTheDocument();
152
+ expect(screen.getByText('Replicate encrypted objects')).toBeInTheDocument();
153
+ expect(screen.getByText('Delete marker replication')).toBeInTheDocument();
154
154
  });
155
155
  });
156
156
  });
157
- describe("Form Fields - Initial State", ()=>{
158
- it("renders required form fields in create mode", ()=>{
157
+ describe('Form Fields - Initial State', ()=>{
158
+ it('renders required form fields in create mode', ()=>{
159
159
  renderBucketReplicationFormPage();
160
160
  expect(screen.getByLabelText(/role arn/i)).toBeInTheDocument();
161
161
  expect(screen.getByLabelText(/rule id/i)).toBeInTheDocument();
162
162
  expect(screen.getByLabelText(/status/i)).toBeInTheDocument();
163
163
  expect(screen.getByLabelText(/target bucket/i)).toBeInTheDocument();
164
164
  });
165
- it("shows Role ARN input when no existing rules", ()=>{
165
+ it('shows Role ARN input when no existing rules', ()=>{
166
166
  renderBucketReplicationFormPage();
167
167
  const roleInput = screen.getByLabelText(/role arn/i);
168
168
  expect(roleInput).toBeInTheDocument();
169
- expect(roleInput.tagName).toBe("INPUT");
169
+ expect(roleInput.tagName).toBe('INPUT');
170
170
  });
171
- it("displays existing Role as read-only text when rules exist", ()=>{
171
+ it('displays existing Role as read-only text when rules exist', ()=>{
172
172
  mockUseGetBucketReplication.mockReturnValue({
173
173
  data: {
174
174
  ReplicationConfiguration: {
175
- Role: "arn:aws:iam::123456789012:role/existing-role",
175
+ Role: 'arn:aws:iam::123456789012:role/existing-role',
176
176
  Rules: [
177
177
  {
178
- ID: "existing-rule",
179
- Status: "Enabled",
178
+ ID: 'existing-rule',
179
+ Status: 'Enabled',
180
180
  Priority: 1,
181
181
  Destination: {
182
- Bucket: "arn:aws:s3:::bucket"
182
+ Bucket: 'arn:aws:s3:::bucket'
183
183
  }
184
184
  }
185
185
  ]
186
186
  }
187
187
  },
188
- status: "success"
188
+ status: 'success'
189
189
  });
190
190
  renderBucketReplicationFormPage();
191
- expect(screen.getByText("arn:aws:iam::123456789012:role/existing-role")).toBeInTheDocument();
191
+ expect(screen.getByText('arn:aws:iam::123456789012:role/existing-role')).toBeInTheDocument();
192
192
  expect(screen.getByText(/to change the role, edit it through bucket/i)).toBeInTheDocument();
193
193
  expect(screen.queryByPlaceholderText(/arn:aws:iam/)).not.toBeInTheDocument();
194
194
  });
195
- it("shows Rule ID input in create mode", ()=>{
195
+ it('shows Rule ID input in create mode', ()=>{
196
196
  renderBucketReplicationFormPage();
197
197
  const ruleIdInput = screen.getByLabelText(/rule id/i);
198
198
  expect(ruleIdInput).toBeInTheDocument();
199
- expect(ruleIdInput.tagName).toBe("INPUT");
199
+ expect(ruleIdInput.tagName).toBe('INPUT');
200
200
  });
201
- it("displays Rule ID as read-only text in edit mode", async ()=>{
201
+ it('displays Rule ID as read-only text in edit mode', async ()=>{
202
202
  const existingRule = {
203
- ID: "readonly-rule-id",
204
- Status: "Enabled",
203
+ ID: 'readonly-rule-id',
204
+ Status: 'Enabled',
205
205
  Priority: 1,
206
206
  Destination: {
207
- Bucket: "arn:aws:s3:::bucket"
207
+ Bucket: 'arn:aws:s3:::bucket'
208
208
  }
209
209
  };
210
210
  mockUseGetBucketReplication.mockReturnValue({
211
211
  data: {
212
212
  ReplicationConfiguration: {
213
- Role: "arn:aws:iam::123456789012:role/role",
213
+ Role: 'arn:aws:iam::123456789012:role/role',
214
214
  Rules: [
215
215
  existingRule
216
216
  ]
217
217
  }
218
218
  },
219
- status: "success"
219
+ status: 'success'
220
220
  });
221
- renderBucketReplicationFormPage("test-bucket", "readonly-rule-id");
221
+ renderBucketReplicationFormPage('test-bucket', 'readonly-rule-id');
222
222
  await waitFor(()=>{
223
- expect(screen.getByText("readonly-rule-id")).toBeInTheDocument();
223
+ expect(screen.getByText('readonly-rule-id')).toBeInTheDocument();
224
224
  });
225
- expect(screen.queryByRole("textbox", {
225
+ expect(screen.queryByRole('textbox', {
226
226
  name: /rule id/i
227
227
  })).not.toBeInTheDocument();
228
228
  });
229
- it("renders Status select field", async ()=>{
229
+ it('renders Status select field', async ()=>{
230
230
  renderBucketReplicationFormPage();
231
231
  await waitFor(()=>{
232
232
  expect(screen.getByLabelText(/rule id/i)).toBeInTheDocument();
@@ -234,189 +234,189 @@ describe("BucketReplicationFormPage", ()=>{
234
234
  const statusElement = document.querySelector('[id="status"]');
235
235
  expect(statusElement).toBeInTheDocument();
236
236
  });
237
- it("renders Priority number input with auto-assigned placeholder", ()=>{
237
+ it('renders Priority number input with auto-assigned placeholder', ()=>{
238
238
  renderBucketReplicationFormPage();
239
239
  const priorityInput = screen.getByLabelText(/rule priority/i);
240
240
  expect(priorityInput).toBeInTheDocument();
241
- expect(priorityInput).toHaveAttribute("placeholder", "Example: Auto-assigned: 0");
241
+ expect(priorityInput).toHaveAttribute('placeholder', 'Example: Auto-assigned: 0');
242
242
  });
243
- it("renders all toggle fields", ()=>{
243
+ it('renders all toggle fields', ()=>{
244
244
  renderBucketReplicationFormPage();
245
- expect(screen.getByText("Replicate encrypted objects")).toBeInTheDocument();
246
- expect(screen.getByText("Encrypt replicated objects")).toBeInTheDocument();
247
- expect(screen.getByText("Replication Time Control (RTC)")).toBeInTheDocument();
248
- expect(screen.getByText("RTC metrics and notifications")).toBeInTheDocument();
249
- expect(screen.getByText("Replica modification sync")).toBeInTheDocument();
250
- expect(screen.getByText("Delete marker replication")).toBeInTheDocument();
251
- });
252
- it("renders target bucket as Select for same account", ()=>{
245
+ expect(screen.getByText('Replicate encrypted objects')).toBeInTheDocument();
246
+ expect(screen.getByText('Encrypt replicated objects')).toBeInTheDocument();
247
+ expect(screen.getByText('Replication Time Control (RTC)')).toBeInTheDocument();
248
+ expect(screen.getByText('RTC metrics and notifications')).toBeInTheDocument();
249
+ expect(screen.getByText('Replica modification sync')).toBeInTheDocument();
250
+ expect(screen.getByText('Delete marker replication')).toBeInTheDocument();
251
+ });
252
+ it('renders target bucket as Select for same account', ()=>{
253
253
  renderBucketReplicationFormPage();
254
254
  expect(screen.getAllByText(/target bucket/i)[0]).toBeInTheDocument();
255
255
  });
256
- it("renders storage class select with all options", async ()=>{
256
+ it('renders storage class select with all options', async ()=>{
257
257
  renderBucketReplicationFormPage();
258
258
  const storageClassSelect = screen.getByLabelText(/storage class/i);
259
259
  await user_event.click(storageClassSelect);
260
- expect(screen.getByRole("option", {
261
- name: "Same as source"
260
+ expect(screen.getByRole('option', {
261
+ name: 'Same as source'
262
262
  })).toBeInTheDocument();
263
- expect(screen.getByRole("option", {
264
- name: "Glacier"
263
+ expect(screen.getByRole('option', {
264
+ name: 'Glacier'
265
265
  })).toBeInTheDocument();
266
- expect(screen.getByRole("option", {
267
- name: "Glacier Deep Archive"
266
+ expect(screen.getByRole('option', {
267
+ name: 'Glacier Deep Archive'
268
268
  })).toBeInTheDocument();
269
- expect(screen.getByRole("option", {
270
- name: "Standard-IA"
269
+ expect(screen.getByRole('option', {
270
+ name: 'Standard-IA'
271
271
  })).toBeInTheDocument();
272
- expect(screen.getByRole("option", {
273
- name: "One Zone-IA"
272
+ expect(screen.getByRole('option', {
273
+ name: 'One Zone-IA'
274
274
  })).toBeInTheDocument();
275
- expect(screen.getByRole("option", {
276
- name: "Intelligent-Tiering"
275
+ expect(screen.getByRole('option', {
276
+ name: 'Intelligent-Tiering'
277
277
  })).toBeInTheDocument();
278
- expect(screen.getByRole("option", {
279
- name: "Glacier Instant Retrieval"
278
+ expect(screen.getByRole('option', {
279
+ name: 'Glacier Instant Retrieval'
280
280
  })).toBeInTheDocument();
281
281
  });
282
282
  });
283
- describe("Form Validation", ()=>{
284
- it("validates Role ARN is required for first replication rule", async ()=>{
283
+ describe('Form Validation', ()=>{
284
+ it('validates Role ARN is required for first replication rule', async ()=>{
285
285
  renderBucketReplicationFormPage();
286
286
  const roleInput = screen.getByLabelText(/role arn/i);
287
- await user_event.type(roleInput, "test");
287
+ await user_event.type(roleInput, 'test');
288
288
  await user_event.clear(roleInput);
289
289
  await waitFor(()=>{
290
290
  expect(screen.getByText(/role arn is required for first replication rule/i)).toBeInTheDocument();
291
291
  });
292
292
  });
293
- it("validates Role ARN is optional when existing rules present", async ()=>{
293
+ it('validates Role ARN is optional when existing rules present', async ()=>{
294
294
  mockUseGetBucketReplication.mockReturnValue({
295
295
  data: {
296
296
  ReplicationConfiguration: {
297
- Role: "arn:aws:iam::123456789012:role/existing-role",
297
+ Role: 'arn:aws:iam::123456789012:role/existing-role',
298
298
  Rules: [
299
299
  {
300
- ID: "existing-rule",
301
- Status: "Enabled",
300
+ ID: 'existing-rule',
301
+ Status: 'Enabled',
302
302
  Priority: 1,
303
303
  Destination: {
304
- Bucket: "arn:aws:s3:::bucket"
304
+ Bucket: 'arn:aws:s3:::bucket'
305
305
  }
306
306
  }
307
307
  ]
308
308
  }
309
309
  },
310
- status: "success"
310
+ status: 'success'
311
311
  });
312
312
  renderBucketReplicationFormPage();
313
- await user_event.type(screen.getByLabelText(/rule id/i), "new-rule");
313
+ await user_event.type(screen.getByLabelText(/rule id/i), 'new-rule');
314
314
  const targetBucketSelect = screen.getByLabelText(/target bucket/i);
315
315
  await user_event.click(targetBucketSelect);
316
- await user_event.click(screen.getByRole("option", {
317
- name: "destination-bucket"
316
+ await user_event.click(screen.getByRole('option', {
317
+ name: 'destination-bucket'
318
318
  }));
319
319
  await waitFor(()=>{
320
- const createButton = screen.getByRole("button", {
320
+ const createButton = screen.getByRole('button', {
321
321
  name: /create/i
322
322
  });
323
323
  expect(createButton).toBeEnabled();
324
324
  });
325
325
  });
326
- it("validates Rule ID is required", async ()=>{
326
+ it('validates Rule ID is required', async ()=>{
327
327
  renderBucketReplicationFormPage();
328
328
  const ruleIdInput = screen.getByLabelText(/rule id/i);
329
- await user_event.type(ruleIdInput, "test");
329
+ await user_event.type(ruleIdInput, 'test');
330
330
  await user_event.clear(ruleIdInput);
331
331
  await waitFor(()=>{
332
332
  expect(screen.getByText(/rule id is required/i)).toBeInTheDocument();
333
333
  });
334
334
  });
335
- it("validates Rule ID uniqueness against existing rules", async ()=>{
335
+ it('validates Rule ID uniqueness against existing rules', async ()=>{
336
336
  mockUseGetBucketReplication.mockReturnValue({
337
337
  data: {
338
338
  ReplicationConfiguration: {
339
- Role: "arn:aws:iam::123456789012:role/role",
339
+ Role: 'arn:aws:iam::123456789012:role/role',
340
340
  Rules: [
341
341
  {
342
- ID: "existing-rule",
343
- Status: "Enabled",
342
+ ID: 'existing-rule',
343
+ Status: 'Enabled',
344
344
  Priority: 1,
345
345
  Destination: {
346
- Bucket: "arn:aws:s3:::bucket"
346
+ Bucket: 'arn:aws:s3:::bucket'
347
347
  }
348
348
  }
349
349
  ]
350
350
  }
351
351
  },
352
- status: "success"
352
+ status: 'success'
353
353
  });
354
354
  renderBucketReplicationFormPage();
355
355
  const ruleIdInput = screen.getByLabelText(/rule id/i);
356
- await user_event.type(ruleIdInput, "existing-rule");
356
+ await user_event.type(ruleIdInput, 'existing-rule');
357
357
  await waitFor(()=>{
358
358
  expect(screen.getByText(/a rule with this id already exists/i)).toBeInTheDocument();
359
359
  });
360
360
  });
361
- it("validates Priority must be >= 0", async ()=>{
361
+ it('validates Priority must be >= 0', async ()=>{
362
362
  renderBucketReplicationFormPage();
363
- await user_event.type(screen.getByLabelText(/rule id/i), "test-rule");
363
+ await user_event.type(screen.getByLabelText(/rule id/i), 'test-rule');
364
364
  const priorityInput = screen.getByLabelText(/rule priority/i);
365
365
  await user_event.clear(priorityInput);
366
- await user_event.type(priorityInput, "-1");
366
+ await user_event.type(priorityInput, '-1');
367
367
  await user_event.tab();
368
368
  await waitFor(()=>{
369
369
  expect(screen.getByText(/priority must be at least 0/i)).toBeInTheDocument();
370
370
  });
371
371
  });
372
- it("disables submit button when Target Bucket is not selected", async ()=>{
372
+ it('disables submit button when Target Bucket is not selected', async ()=>{
373
373
  renderBucketReplicationFormPage();
374
- await user_event.type(screen.getByLabelText(/role arn/i), "arn:aws:iam::123456789012:role/role");
375
- await user_event.type(screen.getByLabelText(/rule id/i), "test-rule");
374
+ await user_event.type(screen.getByLabelText(/role arn/i), 'arn:aws:iam::123456789012:role/role');
375
+ await user_event.type(screen.getByLabelText(/rule id/i), 'test-rule');
376
376
  await waitFor(()=>{
377
- const createButton = screen.getByRole("button", {
377
+ const createButton = screen.getByRole('button', {
378
378
  name: /create/i
379
379
  });
380
380
  expect(createButton).toBeDisabled();
381
381
  });
382
382
  });
383
- it("shows Target Account ID field when sameAccount is false", async ()=>{
383
+ it('shows Target Account ID field when sameAccount is false', async ()=>{
384
384
  renderBucketReplicationFormPage();
385
- const sameAccountToggle = findToggleByLabel("Same account destination");
385
+ const sameAccountToggle = findToggleByLabel('Same account destination');
386
386
  await user_event.click(sameAccountToggle);
387
387
  await waitFor(()=>{
388
388
  expect(screen.getByLabelText(/target account id/i)).toBeInTheDocument();
389
389
  });
390
390
  });
391
- it("shows Replica KMS Key ID field when encryptReplicatedObjects is true", async ()=>{
391
+ it('shows Replica KMS Key ID field when encryptReplicatedObjects is true', async ()=>{
392
392
  renderBucketReplicationFormPage();
393
- const encryptToggle = findToggleByLabel("Encrypt replicated objects");
393
+ const encryptToggle = findToggleByLabel('Encrypt replicated objects');
394
394
  await user_event.click(encryptToggle);
395
395
  await waitFor(()=>{
396
396
  expect(screen.getByLabelText(/replica kms key id/i)).toBeInTheDocument();
397
397
  });
398
398
  });
399
- it("shows prefix field when filterType is prefix", async ()=>{
399
+ it('shows prefix field when filterType is prefix', async ()=>{
400
400
  renderBucketReplicationFormPage();
401
401
  const filterSelect = document.querySelector('[id="filterType"]');
402
402
  await user_event.click(filterSelect);
403
403
  await waitFor(()=>{
404
- expect(screen.getByRole("option", {
405
- name: "Prefix filter"
404
+ expect(screen.getByRole('option', {
405
+ name: 'Prefix filter'
406
406
  })).toBeInTheDocument();
407
407
  });
408
- await user_event.click(screen.getByRole("option", {
409
- name: "Prefix filter"
408
+ await user_event.click(screen.getByRole('option', {
409
+ name: 'Prefix filter'
410
410
  }));
411
411
  await waitFor(()=>{
412
412
  expect(screen.getByLabelText(/prefix/i)).toBeInTheDocument();
413
413
  });
414
414
  });
415
415
  });
416
- describe("Automatic Field Behaviors - Encryption", ()=>{
417
- it("shows encryption requirement when includeEncryptedObjects is enabled", async ()=>{
416
+ describe('Automatic Field Behaviors - Encryption', ()=>{
417
+ it('shows encryption requirement when includeEncryptedObjects is enabled', async ()=>{
418
418
  renderBucketReplicationFormPage();
419
- const includeEncryptedToggle = findToggleByLabel("Replicate encrypted objects");
419
+ const includeEncryptedToggle = findToggleByLabel('Replicate encrypted objects');
420
420
  expect(includeEncryptedToggle).toBeInTheDocument();
421
421
  await user_event.click(includeEncryptedToggle);
422
422
  await waitFor(()=>{
@@ -424,129 +424,129 @@ describe("BucketReplicationFormPage", ()=>{
424
424
  });
425
425
  });
426
426
  });
427
- describe("Automatic Field Behaviors - RTC", ()=>{
428
- it("shows metrics requirement when enforceRTC is enabled", async ()=>{
427
+ describe('Automatic Field Behaviors - RTC', ()=>{
428
+ it('shows metrics requirement when enforceRTC is enabled', async ()=>{
429
429
  renderBucketReplicationFormPage();
430
- const enforceRTCToggle = findToggleByLabel("Replication Time Control (RTC)");
430
+ const enforceRTCToggle = findToggleByLabel('Replication Time Control (RTC)');
431
431
  await user_event.click(enforceRTCToggle);
432
432
  await waitFor(()=>{
433
433
  expect(screen.getByText(/metrics must be enabled when rtc is enabled/i)).toBeInTheDocument();
434
434
  });
435
435
  });
436
436
  });
437
- describe("Automatic Field Behaviors - Delete Marker", ()=>{
438
- it("disables deleteMarkerReplication when filterType is tags", async ()=>{
437
+ describe('Automatic Field Behaviors - Delete Marker', ()=>{
438
+ it('disables deleteMarkerReplication when filterType is tags', async ()=>{
439
439
  renderBucketReplicationFormPage();
440
440
  const filterSelect = screen.getByLabelText(/^filter$/i);
441
441
  await user_event.click(filterSelect);
442
- await user_event.click(screen.getByRole("option", {
443
- name: "Tags filter"
442
+ await user_event.click(screen.getByRole('option', {
443
+ name: 'Tags filter'
444
444
  }));
445
445
  await waitFor(()=>{
446
- const deleteMarkerToggle = findToggleByLabel("Delete marker replication");
446
+ const deleteMarkerToggle = findToggleByLabel('Delete marker replication');
447
447
  expect(deleteMarkerToggle).toBeDisabled();
448
448
  });
449
449
  });
450
- it("disables deleteMarkerReplication when filterType is and", async ()=>{
450
+ it('disables deleteMarkerReplication when filterType is and', async ()=>{
451
451
  renderBucketReplicationFormPage();
452
452
  const filterSelect = screen.getByLabelText(/^filter$/i);
453
453
  await user_event.click(filterSelect);
454
- await user_event.click(screen.getByRole("option", {
455
- name: "Prefix and tags filter"
454
+ await user_event.click(screen.getByRole('option', {
455
+ name: 'Prefix and tags filter'
456
456
  }));
457
457
  await waitFor(()=>{
458
- const deleteMarkerToggle = findToggleByLabel("Delete marker replication");
458
+ const deleteMarkerToggle = findToggleByLabel('Delete marker replication');
459
459
  expect(deleteMarkerToggle).toBeDisabled();
460
460
  });
461
461
  });
462
- it("auto-disables deleteMarkerReplication when switching to tag-based filters", async ()=>{
462
+ it('auto-disables deleteMarkerReplication when switching to tag-based filters', async ()=>{
463
463
  renderBucketReplicationFormPage();
464
- const deleteMarkerToggle = findToggleByLabel("Delete marker replication");
464
+ const deleteMarkerToggle = findToggleByLabel('Delete marker replication');
465
465
  await user_event.click(deleteMarkerToggle);
466
466
  await waitFor(()=>{
467
467
  expect(deleteMarkerToggle).toBeChecked();
468
468
  });
469
469
  const filterSelect = screen.getByLabelText(/^filter$/i);
470
470
  await user_event.click(filterSelect);
471
- await user_event.click(screen.getByRole("option", {
472
- name: "Tags filter"
471
+ await user_event.click(screen.getByRole('option', {
472
+ name: 'Tags filter'
473
473
  }));
474
474
  await waitFor(()=>{
475
- const updatedToggle = findToggleByLabel("Delete marker replication");
475
+ const updatedToggle = findToggleByLabel('Delete marker replication');
476
476
  expect(updatedToggle).not.toBeChecked();
477
477
  expect(updatedToggle).toBeDisabled();
478
478
  });
479
479
  });
480
- it("shows helper text explaining tag filter restriction", async ()=>{
480
+ it('shows helper text explaining tag filter restriction', async ()=>{
481
481
  renderBucketReplicationFormPage();
482
482
  const filterSelect = screen.getByLabelText(/^filter$/i);
483
483
  await user_event.click(filterSelect);
484
- await user_event.click(screen.getByRole("option", {
485
- name: "Tags filter"
484
+ await user_event.click(screen.getByRole('option', {
485
+ name: 'Tags filter'
486
486
  }));
487
487
  await waitFor(()=>{
488
488
  expect(screen.getByText(/delete marker replication is not supported for tag-based filters/i)).toBeInTheDocument();
489
489
  });
490
490
  });
491
491
  });
492
- describe("Automatic Field Behaviors - Conditional Fields", ()=>{
493
- it("shows switchObjectOwnership toggle only when sameAccount is false", async ()=>{
492
+ describe('Automatic Field Behaviors - Conditional Fields', ()=>{
493
+ it('shows switchObjectOwnership toggle only when sameAccount is false', async ()=>{
494
494
  renderBucketReplicationFormPage();
495
- expect(screen.queryByText("Switch Object ownership")).not.toBeInTheDocument();
496
- const sameAccountToggle = findToggleByLabel("Same account destination");
495
+ expect(screen.queryByText('Switch Object ownership')).not.toBeInTheDocument();
496
+ const sameAccountToggle = findToggleByLabel('Same account destination');
497
497
  await user_event.click(sameAccountToggle);
498
498
  await waitFor(()=>{
499
- expect(screen.getByText("Switch Object ownership")).toBeInTheDocument();
499
+ expect(screen.getByText('Switch Object ownership')).toBeInTheDocument();
500
500
  });
501
501
  });
502
- it("shows Target Account ID input when sameAccount is false", async ()=>{
502
+ it('shows Target Account ID input when sameAccount is false', async ()=>{
503
503
  renderBucketReplicationFormPage();
504
504
  expect(screen.queryByLabelText(/target account id/i)).not.toBeInTheDocument();
505
- const sameAccountToggle = findToggleByLabel("Same account destination");
505
+ const sameAccountToggle = findToggleByLabel('Same account destination');
506
506
  await user_event.click(sameAccountToggle);
507
507
  await waitFor(()=>{
508
508
  expect(screen.getByLabelText(/target account id/i)).toBeInTheDocument();
509
509
  });
510
510
  });
511
- it("renders target bucket as Input for cross-account replication", async ()=>{
511
+ it('renders target bucket as Input for cross-account replication', async ()=>{
512
512
  renderBucketReplicationFormPage();
513
- const sameAccountToggle = findToggleByLabel("Same account destination");
513
+ const sameAccountToggle = findToggleByLabel('Same account destination');
514
514
  await user_event.click(sameAccountToggle);
515
515
  await waitFor(()=>{
516
516
  const targetBucketInput = screen.getByLabelText(/target bucket/i);
517
- expect(targetBucketInput.tagName).toBe("INPUT");
517
+ expect(targetBucketInput.tagName).toBe('INPUT');
518
518
  });
519
519
  });
520
- it("shows Replica KMS Key ID input when encryptReplicatedObjects is true", async ()=>{
520
+ it('shows Replica KMS Key ID input when encryptReplicatedObjects is true', async ()=>{
521
521
  renderBucketReplicationFormPage();
522
522
  expect(screen.queryByLabelText(/replica kms key id/i)).not.toBeInTheDocument();
523
- const encryptToggle = findToggleByLabel("Encrypt replicated objects");
523
+ const encryptToggle = findToggleByLabel('Encrypt replicated objects');
524
524
  await user_event.click(encryptToggle);
525
525
  await waitFor(()=>{
526
526
  expect(screen.getByLabelText(/replica kms key id/i)).toBeInTheDocument();
527
527
  });
528
528
  });
529
529
  });
530
- describe("Form Submission - Create Mode", ()=>{
531
- it("submits minimal replication rule with required fields only", async ()=>{
530
+ describe('Form Submission - Create Mode', ()=>{
531
+ it('submits minimal replication rule with required fields only', async ()=>{
532
532
  renderBucketReplicationFormPage();
533
533
  await fillRequiredFields({
534
- ruleId: "minimal-rule"
534
+ ruleId: 'minimal-rule'
535
535
  });
536
536
  mockSuccessSubmit(mockMutate);
537
- await submitForm("create");
537
+ await submitForm('create');
538
538
  await waitFor(()=>{
539
539
  expect(mockMutate).toHaveBeenCalledWith(expect.objectContaining({
540
- Bucket: "test-bucket",
540
+ Bucket: 'test-bucket',
541
541
  ReplicationConfiguration: {
542
- Role: "arn:aws:iam::123456789012:role/replication-role",
542
+ Role: 'arn:aws:iam::123456789012:role/replication-role',
543
543
  Rules: expect.arrayContaining([
544
544
  expect.objectContaining({
545
- ID: "minimal-rule",
546
- Status: "Enabled",
545
+ ID: 'minimal-rule',
546
+ Status: 'Enabled',
547
547
  Priority: 0,
548
548
  Destination: expect.objectContaining({
549
- Bucket: "arn:aws:s3:::destination-bucket"
549
+ Bucket: 'arn:aws:s3:::destination-bucket'
550
550
  })
551
551
  })
552
552
  ])
@@ -554,643 +554,643 @@ describe("BucketReplicationFormPage", ()=>{
554
554
  }), expect.any(Object));
555
555
  });
556
556
  });
557
- it("submits complete replication rule with all optional fields", async ()=>{
557
+ it('submits complete replication rule with all optional fields', async ()=>{
558
558
  renderBucketReplicationFormPage();
559
- await user_event.type(screen.getByLabelText(/role arn/i), "arn:aws:iam::123456789012:role/replication-role");
560
- await user_event.type(screen.getByLabelText(/rule id/i), "complete-rule");
561
- await user_event.type(screen.getByLabelText(/rule priority/i), "5");
559
+ await user_event.type(screen.getByLabelText(/role arn/i), 'arn:aws:iam::123456789012:role/replication-role');
560
+ await user_event.type(screen.getByLabelText(/rule id/i), 'complete-rule');
561
+ await user_event.type(screen.getByLabelText(/rule priority/i), '5');
562
562
  const filterSelect = document.querySelector('[id="filterType"]');
563
563
  await user_event.click(filterSelect);
564
564
  await waitFor(()=>{
565
- expect(screen.getByRole("option", {
566
- name: "Prefix filter"
565
+ expect(screen.getByRole('option', {
566
+ name: 'Prefix filter'
567
567
  })).toBeInTheDocument();
568
568
  });
569
- await user_event.click(screen.getByRole("option", {
570
- name: "Prefix filter"
569
+ await user_event.click(screen.getByRole('option', {
570
+ name: 'Prefix filter'
571
571
  }));
572
572
  await waitFor(()=>{
573
573
  expect(screen.getByLabelText(/prefix/i)).toBeInTheDocument();
574
574
  });
575
- await user_event.type(screen.getByLabelText(/prefix/i), "logs/");
576
- const sameAccountToggle = findToggleByLabel("Same account destination");
575
+ await user_event.type(screen.getByLabelText(/prefix/i), 'logs/');
576
+ const sameAccountToggle = findToggleByLabel('Same account destination');
577
577
  await user_event.click(sameAccountToggle);
578
578
  await waitFor(()=>{
579
579
  expect(screen.getByLabelText(/target account id/i)).toBeInTheDocument();
580
580
  });
581
- await user_event.type(screen.getByLabelText(/target account id/i), "987654321098");
582
- await user_event.type(screen.getByLabelText(/target bucket/i), "cross-account-bucket");
581
+ await user_event.type(screen.getByLabelText(/target account id/i), '987654321098');
582
+ await user_event.type(screen.getByLabelText(/target bucket/i), 'cross-account-bucket');
583
583
  const storageClassSelect = document.querySelector('[id="storageClass"]');
584
584
  await user_event.click(storageClassSelect);
585
585
  await waitFor(()=>{
586
- expect(screen.getByRole("option", {
587
- name: "Glacier"
586
+ expect(screen.getByRole('option', {
587
+ name: 'Glacier'
588
588
  })).toBeInTheDocument();
589
589
  });
590
- await user_event.click(screen.getByRole("option", {
591
- name: "Glacier"
590
+ await user_event.click(screen.getByRole('option', {
591
+ name: 'Glacier'
592
592
  }));
593
- const switchOwnershipToggle = findToggleByLabel("Switch Object ownership");
593
+ const switchOwnershipToggle = findToggleByLabel('Switch Object ownership');
594
594
  await user_event.click(switchOwnershipToggle);
595
- const includeEncryptedToggle = findToggleByLabel("Replicate encrypted objects");
595
+ const includeEncryptedToggle = findToggleByLabel('Replicate encrypted objects');
596
596
  await user_event.click(includeEncryptedToggle);
597
597
  await waitFor(()=>{
598
598
  expect(screen.getByLabelText(/replica kms key id/i)).toBeInTheDocument();
599
599
  });
600
- await user_event.type(screen.getByLabelText(/replica kms key id/i), "arn:aws:kms:us-east-1:987654321098:key/12345678-1234-1234-1234-123456789012");
601
- const enforceRTCToggle = findToggleByLabel("Replication Time Control (RTC)");
600
+ await user_event.type(screen.getByLabelText(/replica kms key id/i), 'arn:aws:kms:us-east-1:987654321098:key/12345678-1234-1234-1234-123456789012');
601
+ const enforceRTCToggle = findToggleByLabel('Replication Time Control (RTC)');
602
602
  await user_event.click(enforceRTCToggle);
603
- const replicaModToggle = findToggleByLabel("Replica modification sync");
603
+ const replicaModToggle = findToggleByLabel('Replica modification sync');
604
604
  await user_event.click(replicaModToggle);
605
605
  mockMutate.mockImplementation((_, options)=>{
606
606
  options?.onSuccess?.();
607
607
  });
608
608
  await waitFor(()=>{
609
- const createButton = screen.getByRole("button", {
609
+ const createButton = screen.getByRole('button', {
610
610
  name: /create/i
611
611
  });
612
612
  expect(createButton).toBeEnabled();
613
613
  });
614
- fireEvent.click(screen.getByRole("button", {
614
+ fireEvent.click(screen.getByRole('button', {
615
615
  name: /create/i
616
616
  }));
617
617
  await waitFor(()=>{
618
618
  const call = mockMutate.mock.calls[0][0];
619
619
  expect(call.ReplicationConfiguration.Rules[0]).toMatchObject({
620
- ID: "complete-rule",
621
- Status: "Enabled",
620
+ ID: 'complete-rule',
621
+ Status: 'Enabled',
622
622
  Priority: 5,
623
623
  Filter: {
624
- Prefix: "logs/"
624
+ Prefix: 'logs/'
625
625
  },
626
626
  Destination: expect.objectContaining({
627
- Bucket: "arn:aws:s3:::cross-account-bucket",
628
- Account: "987654321098",
629
- StorageClass: "GLACIER",
627
+ Bucket: 'arn:aws:s3:::cross-account-bucket',
628
+ Account: '987654321098',
629
+ StorageClass: 'GLACIER',
630
630
  AccessControlTranslation: {
631
- Owner: "Destination"
631
+ Owner: 'Destination'
632
632
  },
633
633
  EncryptionConfiguration: {
634
- ReplicaKmsKeyID: "arn:aws:kms:us-east-1:987654321098:key/12345678-1234-1234-1234-123456789012"
634
+ ReplicaKmsKeyID: 'arn:aws:kms:us-east-1:987654321098:key/12345678-1234-1234-1234-123456789012'
635
635
  },
636
636
  ReplicationTime: {
637
- Status: "Enabled",
637
+ Status: 'Enabled',
638
638
  Time: {
639
639
  Minutes: 15
640
640
  }
641
641
  },
642
642
  Metrics: expect.objectContaining({
643
- Status: "Enabled"
643
+ Status: 'Enabled'
644
644
  })
645
645
  }),
646
646
  SourceSelectionCriteria: expect.objectContaining({
647
647
  SseKmsEncryptedObjects: {
648
- Status: "Enabled"
648
+ Status: 'Enabled'
649
649
  },
650
650
  ReplicaModifications: {
651
- Status: "Enabled"
651
+ Status: 'Enabled'
652
652
  }
653
653
  })
654
654
  });
655
655
  });
656
656
  });
657
- it("appends new rule to existing rules array", async ()=>{
657
+ it('appends new rule to existing rules array', async ()=>{
658
658
  mockUseGetBucketReplication.mockReturnValue({
659
659
  data: {
660
660
  ReplicationConfiguration: {
661
- Role: "arn:aws:iam::123456789012:role/role",
661
+ Role: 'arn:aws:iam::123456789012:role/role',
662
662
  Rules: [
663
663
  {
664
- ID: "existing-rule",
665
- Status: "Enabled",
664
+ ID: 'existing-rule',
665
+ Status: 'Enabled',
666
666
  Priority: 1,
667
667
  Destination: {
668
- Bucket: "arn:aws:s3:::existing-bucket"
668
+ Bucket: 'arn:aws:s3:::existing-bucket'
669
669
  }
670
670
  }
671
671
  ]
672
672
  }
673
673
  },
674
- status: "success"
674
+ status: 'success'
675
675
  });
676
676
  renderBucketReplicationFormPage();
677
- await user_event.type(screen.getByLabelText(/rule id/i), "new-rule");
677
+ await user_event.type(screen.getByLabelText(/rule id/i), 'new-rule');
678
678
  const targetBucketSelect = screen.getByLabelText(/target bucket/i);
679
679
  await user_event.click(targetBucketSelect);
680
- await user_event.click(screen.getByRole("option", {
681
- name: "destination-bucket"
680
+ await user_event.click(screen.getByRole('option', {
681
+ name: 'destination-bucket'
682
682
  }));
683
683
  mockMutate.mockImplementation((_, options)=>{
684
684
  options?.onSuccess?.();
685
685
  });
686
686
  await waitFor(()=>{
687
- const createButton = screen.getByRole("button", {
687
+ const createButton = screen.getByRole('button', {
688
688
  name: /create/i
689
689
  });
690
690
  expect(createButton).toBeEnabled();
691
691
  });
692
- fireEvent.click(screen.getByRole("button", {
692
+ fireEvent.click(screen.getByRole('button', {
693
693
  name: /create/i
694
694
  }));
695
695
  await waitFor(()=>{
696
696
  const call = mockMutate.mock.calls[0][0];
697
697
  expect(call.ReplicationConfiguration.Rules).toHaveLength(2);
698
- expect(call.ReplicationConfiguration.Rules[0].ID).toBe("existing-rule");
699
- expect(call.ReplicationConfiguration.Rules[1].ID).toBe("new-rule");
698
+ expect(call.ReplicationConfiguration.Rules[0].ID).toBe('existing-rule');
699
+ expect(call.ReplicationConfiguration.Rules[1].ID).toBe('new-rule');
700
700
  });
701
701
  });
702
- it("navigates to bucket replication tab on success", async ()=>{
702
+ it('navigates to bucket replication tab on success', async ()=>{
703
703
  renderBucketReplicationFormPage();
704
704
  await fillRequiredFields({
705
- roleArn: "arn:aws:iam::123456789012:role/role",
706
- ruleId: "success-rule"
705
+ roleArn: 'arn:aws:iam::123456789012:role/role',
706
+ ruleId: 'success-rule'
707
707
  });
708
708
  mockSuccessSubmit(mockMutate);
709
- await submitForm("create");
709
+ await submitForm('create');
710
710
  await waitFor(()=>{
711
- expect(mockNavigate).toHaveBeenCalledWith("/buckets/test-bucket?tab=replication");
711
+ expect(mockNavigate).toHaveBeenCalledWith('/buckets/test-bucket?tab=replication');
712
712
  });
713
713
  });
714
- it("displays success toast after rule creation", async ()=>{
714
+ it('displays success toast after rule creation', async ()=>{
715
715
  renderBucketReplicationFormPage();
716
716
  await fillRequiredFields({
717
- roleArn: "arn:aws:iam::123456789012:role/role",
718
- ruleId: "toast-rule"
717
+ roleArn: 'arn:aws:iam::123456789012:role/role',
718
+ ruleId: 'toast-rule'
719
719
  });
720
720
  mockSuccessSubmit(mockMutate);
721
- await submitForm("create");
721
+ await submitForm('create');
722
722
  await waitFor(()=>{
723
723
  expect(mockShowToast).toHaveBeenCalledWith({
724
724
  open: true,
725
- message: "Replication rule created successfully",
726
- status: "success"
725
+ message: 'Replication rule created successfully',
726
+ status: 'success'
727
727
  });
728
728
  });
729
729
  });
730
- it("displays error toast when creation fails", async ()=>{
730
+ it('displays error toast when creation fails', async ()=>{
731
731
  renderBucketReplicationFormPage();
732
732
  await fillRequiredFields({
733
- roleArn: "arn:aws:iam::123456789012:role/role",
734
- ruleId: "error-rule"
733
+ roleArn: 'arn:aws:iam::123456789012:role/role',
734
+ ruleId: 'error-rule'
735
735
  });
736
- mockErrorSubmit(mockMutate, "Access Denied");
737
- await submitForm("create");
736
+ mockErrorSubmit(mockMutate, 'Access Denied');
737
+ await submitForm('create');
738
738
  await waitFor(()=>{
739
739
  expect(mockShowToast).toHaveBeenCalledWith({
740
740
  open: true,
741
- message: "Access Denied",
742
- status: "error"
741
+ message: 'Access Denied',
742
+ status: 'error'
743
743
  });
744
744
  expect(mockNavigate).not.toHaveBeenCalled();
745
745
  });
746
746
  });
747
- it("disables submit button when form is pristine", ()=>{
747
+ it('disables submit button when form is pristine', ()=>{
748
748
  renderBucketReplicationFormPage();
749
- const createButton = screen.getByRole("button", {
749
+ const createButton = screen.getByRole('button', {
750
750
  name: /create/i
751
751
  });
752
752
  expect(createButton).toBeDisabled();
753
753
  });
754
- it("disables submit button when form is invalid", async ()=>{
754
+ it('disables submit button when form is invalid', async ()=>{
755
755
  renderBucketReplicationFormPage();
756
- await user_event.type(screen.getByLabelText(/rule id/i), "test");
756
+ await user_event.type(screen.getByLabelText(/rule id/i), 'test');
757
757
  await waitFor(()=>{
758
- const createButton = screen.getByRole("button", {
758
+ const createButton = screen.getByRole('button', {
759
759
  name: /create/i
760
760
  });
761
761
  expect(createButton).toBeDisabled();
762
762
  });
763
763
  });
764
- it("enables submit button when form is valid and dirty", async ()=>{
764
+ it('enables submit button when form is valid and dirty', async ()=>{
765
765
  renderBucketReplicationFormPage();
766
766
  await fillRequiredFields({
767
- roleArn: "arn:aws:iam::123456789012:role/role",
768
- ruleId: "valid-rule"
767
+ roleArn: 'arn:aws:iam::123456789012:role/role',
768
+ ruleId: 'valid-rule'
769
769
  });
770
770
  await waitFor(()=>{
771
- const createButton = screen.getByRole("button", {
771
+ const createButton = screen.getByRole('button', {
772
772
  name: /create/i
773
773
  });
774
774
  expect(createButton).toBeEnabled();
775
775
  });
776
776
  });
777
777
  });
778
- describe("Form Submission - Edit Mode", ()=>{
779
- it("updates existing rule while preserving other rules", async ()=>{
778
+ describe('Form Submission - Edit Mode', ()=>{
779
+ it('updates existing rule while preserving other rules', async ()=>{
780
780
  const existingRules = [
781
781
  {
782
- ID: "rule-1",
783
- Status: "Enabled",
782
+ ID: 'rule-1',
783
+ Status: 'Enabled',
784
784
  Priority: 1,
785
785
  Destination: {
786
- Bucket: "arn:aws:s3:::bucket-1"
786
+ Bucket: 'arn:aws:s3:::bucket-1'
787
787
  }
788
788
  },
789
789
  {
790
- ID: "rule-2",
791
- Status: "Enabled",
790
+ ID: 'rule-2',
791
+ Status: 'Enabled',
792
792
  Priority: 2,
793
793
  Destination: {
794
- Bucket: "arn:aws:s3:::bucket-2"
794
+ Bucket: 'arn:aws:s3:::bucket-2'
795
795
  }
796
796
  }
797
797
  ];
798
798
  mockUseGetBucketReplication.mockReturnValue({
799
799
  data: {
800
800
  ReplicationConfiguration: {
801
- Role: "arn:aws:iam::123456789012:role/role",
801
+ Role: 'arn:aws:iam::123456789012:role/role',
802
802
  Rules: existingRules
803
803
  }
804
804
  },
805
- status: "success"
805
+ status: 'success'
806
806
  });
807
- renderBucketReplicationFormPage("test-bucket", "rule-1");
807
+ renderBucketReplicationFormPage('test-bucket', 'rule-1');
808
808
  await waitFor(()=>{
809
- expect(screen.getByText("Edit Replication Rule")).toBeInTheDocument();
809
+ expect(screen.getByText('Edit Replication Rule')).toBeInTheDocument();
810
810
  });
811
811
  const statusSelect = screen.getByLabelText(/status/i);
812
812
  await user_event.click(statusSelect);
813
- await user_event.click(screen.getByRole("option", {
814
- name: "Disabled"
813
+ await user_event.click(screen.getByRole('option', {
814
+ name: 'Disabled'
815
815
  }));
816
816
  mockMutate.mockImplementation((_, options)=>{
817
817
  options?.onSuccess?.();
818
818
  });
819
819
  await waitFor(()=>{
820
- const saveButton = screen.getByRole("button", {
820
+ const saveButton = screen.getByRole('button', {
821
821
  name: /save/i
822
822
  });
823
823
  expect(saveButton).toBeEnabled();
824
824
  });
825
- fireEvent.click(screen.getByRole("button", {
825
+ fireEvent.click(screen.getByRole('button', {
826
826
  name: /save/i
827
827
  }));
828
828
  await waitFor(()=>{
829
829
  const call = mockMutate.mock.calls[0][0];
830
830
  expect(call.ReplicationConfiguration.Rules).toHaveLength(2);
831
- expect(call.ReplicationConfiguration.Rules[0].ID).toBe("rule-1");
832
- expect(call.ReplicationConfiguration.Rules[0].Status).toBe("Disabled");
833
- expect(call.ReplicationConfiguration.Rules[1].ID).toBe("rule-2");
834
- expect(call.ReplicationConfiguration.Rules[1].Status).toBe("Enabled");
831
+ expect(call.ReplicationConfiguration.Rules[0].ID).toBe('rule-1');
832
+ expect(call.ReplicationConfiguration.Rules[0].Status).toBe('Disabled');
833
+ expect(call.ReplicationConfiguration.Rules[1].ID).toBe('rule-2');
834
+ expect(call.ReplicationConfiguration.Rules[1].Status).toBe('Enabled');
835
835
  });
836
836
  });
837
- it("navigates to bucket replication tab on success", async ()=>{
837
+ it('navigates to bucket replication tab on success', async ()=>{
838
838
  const existingRule = {
839
- ID: "edit-rule",
840
- Status: "Enabled",
839
+ ID: 'edit-rule',
840
+ Status: 'Enabled',
841
841
  Priority: 1,
842
842
  Destination: {
843
- Bucket: "arn:aws:s3:::bucket"
843
+ Bucket: 'arn:aws:s3:::bucket'
844
844
  }
845
845
  };
846
846
  mockUseGetBucketReplication.mockReturnValue({
847
847
  data: {
848
848
  ReplicationConfiguration: {
849
- Role: "arn:aws:iam::123456789012:role/role",
849
+ Role: 'arn:aws:iam::123456789012:role/role',
850
850
  Rules: [
851
851
  existingRule
852
852
  ]
853
853
  }
854
854
  },
855
- status: "success"
855
+ status: 'success'
856
856
  });
857
- renderBucketReplicationFormPage("test-bucket", "edit-rule");
857
+ renderBucketReplicationFormPage('test-bucket', 'edit-rule');
858
858
  await waitFor(()=>{
859
- expect(screen.getByText("Edit Replication Rule")).toBeInTheDocument();
859
+ expect(screen.getByText('Edit Replication Rule')).toBeInTheDocument();
860
860
  });
861
861
  const statusSelect = screen.getByLabelText(/status/i);
862
862
  await user_event.click(statusSelect);
863
- await user_event.click(screen.getByRole("option", {
864
- name: "Disabled"
863
+ await user_event.click(screen.getByRole('option', {
864
+ name: 'Disabled'
865
865
  }));
866
866
  mockMutate.mockImplementation((_, options)=>{
867
867
  options?.onSuccess?.();
868
868
  });
869
869
  await waitFor(()=>{
870
- const saveButton = screen.getByRole("button", {
870
+ const saveButton = screen.getByRole('button', {
871
871
  name: /save/i
872
872
  });
873
873
  expect(saveButton).toBeEnabled();
874
874
  });
875
- fireEvent.click(screen.getByRole("button", {
875
+ fireEvent.click(screen.getByRole('button', {
876
876
  name: /save/i
877
877
  }));
878
878
  await waitFor(()=>{
879
- expect(mockNavigate).toHaveBeenCalledWith("/buckets/test-bucket?tab=replication");
879
+ expect(mockNavigate).toHaveBeenCalledWith('/buckets/test-bucket?tab=replication');
880
880
  });
881
881
  });
882
- it("displays update success toast in edit mode", async ()=>{
882
+ it('displays update success toast in edit mode', async ()=>{
883
883
  const existingRule = {
884
- ID: "toast-rule",
885
- Status: "Enabled",
884
+ ID: 'toast-rule',
885
+ Status: 'Enabled',
886
886
  Priority: 1,
887
887
  Destination: {
888
- Bucket: "arn:aws:s3:::bucket"
888
+ Bucket: 'arn:aws:s3:::bucket'
889
889
  }
890
890
  };
891
891
  mockUseGetBucketReplication.mockReturnValue({
892
892
  data: {
893
893
  ReplicationConfiguration: {
894
- Role: "arn:aws:iam::123456789012:role/role",
894
+ Role: 'arn:aws:iam::123456789012:role/role',
895
895
  Rules: [
896
896
  existingRule
897
897
  ]
898
898
  }
899
899
  },
900
- status: "success"
900
+ status: 'success'
901
901
  });
902
- renderBucketReplicationFormPage("test-bucket", "toast-rule");
902
+ renderBucketReplicationFormPage('test-bucket', 'toast-rule');
903
903
  await waitFor(()=>{
904
- expect(screen.getByText("Edit Replication Rule")).toBeInTheDocument();
904
+ expect(screen.getByText('Edit Replication Rule')).toBeInTheDocument();
905
905
  });
906
906
  const statusSelect = screen.getByLabelText(/status/i);
907
907
  await user_event.click(statusSelect);
908
- await user_event.click(screen.getByRole("option", {
909
- name: "Disabled"
908
+ await user_event.click(screen.getByRole('option', {
909
+ name: 'Disabled'
910
910
  }));
911
911
  mockMutate.mockImplementation((_, options)=>{
912
912
  options?.onSuccess?.();
913
913
  });
914
914
  await waitFor(()=>{
915
- const saveButton = screen.getByRole("button", {
915
+ const saveButton = screen.getByRole('button', {
916
916
  name: /save/i
917
917
  });
918
918
  expect(saveButton).toBeEnabled();
919
919
  });
920
- fireEvent.click(screen.getByRole("button", {
920
+ fireEvent.click(screen.getByRole('button', {
921
921
  name: /save/i
922
922
  }));
923
923
  await waitFor(()=>{
924
924
  expect(mockShowToast).toHaveBeenCalledWith({
925
925
  open: true,
926
- message: "Replication rule updated successfully",
927
- status: "success"
926
+ message: 'Replication rule updated successfully',
927
+ status: 'success'
928
928
  });
929
929
  });
930
930
  });
931
- it("displays error toast with error message on failure", async ()=>{
931
+ it('displays error toast with error message on failure', async ()=>{
932
932
  const existingRule = {
933
- ID: "error-rule",
934
- Status: "Enabled",
933
+ ID: 'error-rule',
934
+ Status: 'Enabled',
935
935
  Priority: 1,
936
936
  Destination: {
937
- Bucket: "arn:aws:s3:::bucket"
937
+ Bucket: 'arn:aws:s3:::bucket'
938
938
  }
939
939
  };
940
940
  mockUseGetBucketReplication.mockReturnValue({
941
941
  data: {
942
942
  ReplicationConfiguration: {
943
- Role: "arn:aws:iam::123456789012:role/role",
943
+ Role: 'arn:aws:iam::123456789012:role/role',
944
944
  Rules: [
945
945
  existingRule
946
946
  ]
947
947
  }
948
948
  },
949
- status: "success"
949
+ status: 'success'
950
950
  });
951
- renderBucketReplicationFormPage("test-bucket", "error-rule");
951
+ renderBucketReplicationFormPage('test-bucket', 'error-rule');
952
952
  await waitFor(()=>{
953
- expect(screen.getByText("Edit Replication Rule")).toBeInTheDocument();
953
+ expect(screen.getByText('Edit Replication Rule')).toBeInTheDocument();
954
954
  });
955
955
  const statusSelect = screen.getByLabelText(/status/i);
956
956
  await user_event.click(statusSelect);
957
- await user_event.click(screen.getByRole("option", {
958
- name: "Disabled"
957
+ await user_event.click(screen.getByRole('option', {
958
+ name: 'Disabled'
959
959
  }));
960
- const error = new Error("Network Error");
960
+ const error = new Error('Network Error');
961
961
  mockMutate.mockImplementation((_, options)=>{
962
962
  options?.onError?.(error);
963
963
  });
964
964
  await waitFor(()=>{
965
- const saveButton = screen.getByRole("button", {
965
+ const saveButton = screen.getByRole('button', {
966
966
  name: /save/i
967
967
  });
968
968
  expect(saveButton).toBeEnabled();
969
969
  });
970
- fireEvent.click(screen.getByRole("button", {
970
+ fireEvent.click(screen.getByRole('button', {
971
971
  name: /save/i
972
972
  }));
973
973
  await waitFor(()=>{
974
974
  expect(mockShowToast).toHaveBeenCalledWith({
975
975
  open: true,
976
- message: "Network Error",
977
- status: "error"
976
+ message: 'Network Error',
977
+ status: 'error'
978
978
  });
979
979
  expect(mockNavigate).not.toHaveBeenCalled();
980
980
  });
981
981
  });
982
982
  });
983
- describe("Rule Data Loading in Edit Mode", ()=>{
984
- it("loads rule with no filter correctly", async ()=>{
983
+ describe('Rule Data Loading in Edit Mode', ()=>{
984
+ it('loads rule with no filter correctly', async ()=>{
985
985
  const rule = {
986
- ID: "no-filter-rule",
987
- Status: "Enabled",
986
+ ID: 'no-filter-rule',
987
+ Status: 'Enabled',
988
988
  Priority: 1,
989
989
  Destination: {
990
- Bucket: "arn:aws:s3:::bucket"
990
+ Bucket: 'arn:aws:s3:::bucket'
991
991
  }
992
992
  };
993
993
  mockUseGetBucketReplication.mockReturnValue({
994
994
  data: {
995
995
  ReplicationConfiguration: {
996
- Role: "arn:aws:iam::123456789012:role/role",
996
+ Role: 'arn:aws:iam::123456789012:role/role',
997
997
  Rules: [
998
998
  rule
999
999
  ]
1000
1000
  }
1001
1001
  },
1002
- status: "success"
1002
+ status: 'success'
1003
1003
  });
1004
- renderBucketReplicationFormPage("test-bucket", "no-filter-rule");
1004
+ renderBucketReplicationFormPage('test-bucket', 'no-filter-rule');
1005
1005
  await waitFor(()=>{
1006
1006
  const filterSelect = screen.getByLabelText(/^filter$/i);
1007
1007
  expect(filterSelect).toBeInTheDocument();
1008
1008
  expect(screen.queryByLabelText(/prefix/i)).not.toBeInTheDocument();
1009
1009
  });
1010
1010
  });
1011
- it("loads rule with prefix filter correctly", async ()=>{
1011
+ it('loads rule with prefix filter correctly', async ()=>{
1012
1012
  const rule = {
1013
- ID: "prefix-rule",
1014
- Status: "Enabled",
1013
+ ID: 'prefix-rule',
1014
+ Status: 'Enabled',
1015
1015
  Priority: 1,
1016
1016
  Filter: {
1017
- Prefix: "documents/"
1017
+ Prefix: 'documents/'
1018
1018
  },
1019
1019
  Destination: {
1020
- Bucket: "arn:aws:s3:::bucket"
1020
+ Bucket: 'arn:aws:s3:::bucket'
1021
1021
  }
1022
1022
  };
1023
1023
  mockUseGetBucketReplication.mockReturnValue({
1024
1024
  data: {
1025
1025
  ReplicationConfiguration: {
1026
- Role: "arn:aws:iam::123456789012:role/role",
1026
+ Role: 'arn:aws:iam::123456789012:role/role',
1027
1027
  Rules: [
1028
1028
  rule
1029
1029
  ]
1030
1030
  }
1031
1031
  },
1032
- status: "success"
1032
+ status: 'success'
1033
1033
  });
1034
- renderBucketReplicationFormPage("test-bucket", "prefix-rule");
1034
+ renderBucketReplicationFormPage('test-bucket', 'prefix-rule');
1035
1035
  await waitFor(()=>{
1036
1036
  const prefixInput = screen.getByLabelText(/prefix/i);
1037
- expect(prefixInput).toHaveValue("documents/");
1037
+ expect(prefixInput).toHaveValue('documents/');
1038
1038
  });
1039
1039
  });
1040
- it("loads rule with single tag filter correctly", async ()=>{
1040
+ it('loads rule with single tag filter correctly', async ()=>{
1041
1041
  const rule = {
1042
- ID: "tag-rule",
1043
- Status: "Enabled",
1042
+ ID: 'tag-rule',
1043
+ Status: 'Enabled',
1044
1044
  Priority: 1,
1045
1045
  Filter: {
1046
1046
  Tag: {
1047
- Key: "env",
1048
- Value: "prod"
1047
+ Key: 'env',
1048
+ Value: 'prod'
1049
1049
  }
1050
1050
  },
1051
1051
  Destination: {
1052
- Bucket: "arn:aws:s3:::bucket"
1052
+ Bucket: 'arn:aws:s3:::bucket'
1053
1053
  }
1054
1054
  };
1055
1055
  mockUseGetBucketReplication.mockReturnValue({
1056
1056
  data: {
1057
1057
  ReplicationConfiguration: {
1058
- Role: "arn:aws:iam::123456789012:role/role",
1058
+ Role: 'arn:aws:iam::123456789012:role/role',
1059
1059
  Rules: [
1060
1060
  rule
1061
1061
  ]
1062
1062
  }
1063
1063
  },
1064
- status: "success"
1064
+ status: 'success'
1065
1065
  });
1066
- renderBucketReplicationFormPage("test-bucket", "tag-rule");
1066
+ renderBucketReplicationFormPage('test-bucket', 'tag-rule');
1067
1067
  await waitFor(()=>{
1068
- expect(screen.getByDisplayValue("env")).toBeInTheDocument();
1069
- expect(screen.getByDisplayValue("prod")).toBeInTheDocument();
1068
+ expect(screen.getByDisplayValue('env')).toBeInTheDocument();
1069
+ expect(screen.getByDisplayValue('prod')).toBeInTheDocument();
1070
1070
  });
1071
1071
  });
1072
- it("loads rule with prefix and tags filter correctly", async ()=>{
1072
+ it('loads rule with prefix and tags filter correctly', async ()=>{
1073
1073
  const rule = {
1074
- ID: "and-rule",
1075
- Status: "Enabled",
1074
+ ID: 'and-rule',
1075
+ Status: 'Enabled',
1076
1076
  Priority: 1,
1077
1077
  Filter: {
1078
1078
  And: {
1079
- Prefix: "logs/",
1079
+ Prefix: 'logs/',
1080
1080
  Tags: [
1081
1081
  {
1082
- Key: "type",
1083
- Value: "audit"
1082
+ Key: 'type',
1083
+ Value: 'audit'
1084
1084
  },
1085
1085
  {
1086
- Key: "year",
1087
- Value: "2024"
1086
+ Key: 'year',
1087
+ Value: '2024'
1088
1088
  }
1089
1089
  ]
1090
1090
  }
1091
1091
  },
1092
1092
  Destination: {
1093
- Bucket: "arn:aws:s3:::bucket"
1093
+ Bucket: 'arn:aws:s3:::bucket'
1094
1094
  }
1095
1095
  };
1096
1096
  mockUseGetBucketReplication.mockReturnValue({
1097
1097
  data: {
1098
1098
  ReplicationConfiguration: {
1099
- Role: "arn:aws:iam::123456789012:role/role",
1099
+ Role: 'arn:aws:iam::123456789012:role/role',
1100
1100
  Rules: [
1101
1101
  rule
1102
1102
  ]
1103
1103
  }
1104
1104
  },
1105
- status: "success"
1105
+ status: 'success'
1106
1106
  });
1107
- renderBucketReplicationFormPage("test-bucket", "and-rule");
1107
+ renderBucketReplicationFormPage('test-bucket', 'and-rule');
1108
1108
  await waitFor(()=>{
1109
1109
  const prefixInput = screen.getByLabelText(/prefix/i);
1110
- expect(prefixInput).toHaveValue("logs/");
1111
- expect(screen.getByDisplayValue("type")).toBeInTheDocument();
1112
- expect(screen.getByDisplayValue("audit")).toBeInTheDocument();
1113
- expect(screen.getByDisplayValue("year")).toBeInTheDocument();
1114
- expect(screen.getByDisplayValue("2024")).toBeInTheDocument();
1110
+ expect(prefixInput).toHaveValue('logs/');
1111
+ expect(screen.getByDisplayValue('type')).toBeInTheDocument();
1112
+ expect(screen.getByDisplayValue('audit')).toBeInTheDocument();
1113
+ expect(screen.getByDisplayValue('year')).toBeInTheDocument();
1114
+ expect(screen.getByDisplayValue('2024')).toBeInTheDocument();
1115
1115
  });
1116
1116
  });
1117
- it("loads rule with includeEncryptedObjects enabled", async ()=>{
1117
+ it('loads rule with includeEncryptedObjects enabled', async ()=>{
1118
1118
  const rule = {
1119
- ID: "encrypted-rule",
1120
- Status: "Enabled",
1119
+ ID: 'encrypted-rule',
1120
+ Status: 'Enabled',
1121
1121
  Priority: 1,
1122
1122
  Destination: {
1123
- Bucket: "arn:aws:s3:::bucket"
1123
+ Bucket: 'arn:aws:s3:::bucket'
1124
1124
  },
1125
1125
  SourceSelectionCriteria: {
1126
1126
  SseKmsEncryptedObjects: {
1127
- Status: "Enabled"
1127
+ Status: 'Enabled'
1128
1128
  }
1129
1129
  }
1130
1130
  };
1131
1131
  mockUseGetBucketReplication.mockReturnValue({
1132
1132
  data: {
1133
1133
  ReplicationConfiguration: {
1134
- Role: "arn:aws:iam::123456789012:role/role",
1134
+ Role: 'arn:aws:iam::123456789012:role/role',
1135
1135
  Rules: [
1136
1136
  rule
1137
1137
  ]
1138
1138
  }
1139
1139
  },
1140
- status: "success"
1140
+ status: 'success'
1141
1141
  });
1142
- renderBucketReplicationFormPage("test-bucket", "encrypted-rule");
1142
+ renderBucketReplicationFormPage('test-bucket', 'encrypted-rule');
1143
1143
  await waitFor(()=>{
1144
- const includeEncryptedToggle = findToggleByLabel("Replicate encrypted objects");
1144
+ const includeEncryptedToggle = findToggleByLabel('Replicate encrypted objects');
1145
1145
  expect(includeEncryptedToggle).toBeChecked();
1146
1146
  });
1147
1147
  });
1148
- it("loads rule with encryptReplicatedObjects and KMS Key ID", async ()=>{
1148
+ it('loads rule with encryptReplicatedObjects and KMS Key ID', async ()=>{
1149
1149
  const rule = {
1150
- ID: "encrypt-replica-rule",
1151
- Status: "Enabled",
1150
+ ID: 'encrypt-replica-rule',
1151
+ Status: 'Enabled',
1152
1152
  Priority: 1,
1153
1153
  Destination: {
1154
- Bucket: "arn:aws:s3:::bucket",
1154
+ Bucket: 'arn:aws:s3:::bucket',
1155
1155
  EncryptionConfiguration: {
1156
- ReplicaKmsKeyID: "arn:aws:kms:us-east-1:123456789012:key/12345"
1156
+ ReplicaKmsKeyID: 'arn:aws:kms:us-east-1:123456789012:key/12345'
1157
1157
  }
1158
1158
  }
1159
1159
  };
1160
1160
  mockUseGetBucketReplication.mockReturnValue({
1161
1161
  data: {
1162
1162
  ReplicationConfiguration: {
1163
- Role: "arn:aws:iam::123456789012:role/role",
1163
+ Role: 'arn:aws:iam::123456789012:role/role',
1164
1164
  Rules: [
1165
1165
  rule
1166
1166
  ]
1167
1167
  }
1168
1168
  },
1169
- status: "success"
1169
+ status: 'success'
1170
1170
  });
1171
- renderBucketReplicationFormPage("test-bucket", "encrypt-replica-rule");
1171
+ renderBucketReplicationFormPage('test-bucket', 'encrypt-replica-rule');
1172
1172
  await waitFor(()=>{
1173
- const encryptReplicatedToggle = findToggleByLabel("Encrypt replicated objects");
1173
+ const encryptReplicatedToggle = findToggleByLabel('Encrypt replicated objects');
1174
1174
  expect(encryptReplicatedToggle).toBeChecked();
1175
1175
  const kmsKeyInput = screen.getByLabelText(/replica kms key id/i);
1176
- expect(kmsKeyInput).toHaveValue("arn:aws:kms:us-east-1:123456789012:key/12345");
1176
+ expect(kmsKeyInput).toHaveValue('arn:aws:kms:us-east-1:123456789012:key/12345');
1177
1177
  });
1178
1178
  });
1179
- it("loads rule with enforceRTC and enableRTCNotification", async ()=>{
1179
+ it('loads rule with enforceRTC and enableRTCNotification', async ()=>{
1180
1180
  const rule = {
1181
- ID: "rtc-rule",
1182
- Status: "Enabled",
1181
+ ID: 'rtc-rule',
1182
+ Status: 'Enabled',
1183
1183
  Priority: 1,
1184
1184
  Destination: {
1185
- Bucket: "arn:aws:s3:::bucket",
1185
+ Bucket: 'arn:aws:s3:::bucket',
1186
1186
  ReplicationTime: {
1187
- Status: "Enabled",
1187
+ Status: 'Enabled',
1188
1188
  Time: {
1189
1189
  Minutes: 15
1190
1190
  }
1191
1191
  },
1192
1192
  Metrics: {
1193
- Status: "Enabled",
1193
+ Status: 'Enabled',
1194
1194
  EventThreshold: {
1195
1195
  Minutes: 15
1196
1196
  }
@@ -1200,258 +1200,258 @@ describe("BucketReplicationFormPage", ()=>{
1200
1200
  mockUseGetBucketReplication.mockReturnValue({
1201
1201
  data: {
1202
1202
  ReplicationConfiguration: {
1203
- Role: "arn:aws:iam::123456789012:role/role",
1203
+ Role: 'arn:aws:iam::123456789012:role/role',
1204
1204
  Rules: [
1205
1205
  rule
1206
1206
  ]
1207
1207
  }
1208
1208
  },
1209
- status: "success"
1209
+ status: 'success'
1210
1210
  });
1211
- renderBucketReplicationFormPage("test-bucket", "rtc-rule");
1211
+ renderBucketReplicationFormPage('test-bucket', 'rtc-rule');
1212
1212
  await waitFor(()=>{
1213
- expect(screen.getByText("Replication Time Control (RTC)")).toBeInTheDocument();
1213
+ expect(screen.getByText('Replication Time Control (RTC)')).toBeInTheDocument();
1214
1214
  });
1215
- const enforceRTCToggle = findToggleByLabel("Replication Time Control (RTC)");
1215
+ const enforceRTCToggle = findToggleByLabel('Replication Time Control (RTC)');
1216
1216
  expect(enforceRTCToggle).toBeChecked();
1217
1217
  });
1218
- it("loads rule with replicaModifications enabled", async ()=>{
1218
+ it('loads rule with replicaModifications enabled', async ()=>{
1219
1219
  const rule = {
1220
- ID: "replica-mod-rule",
1221
- Status: "Enabled",
1220
+ ID: 'replica-mod-rule',
1221
+ Status: 'Enabled',
1222
1222
  Priority: 1,
1223
1223
  Destination: {
1224
- Bucket: "arn:aws:s3:::bucket"
1224
+ Bucket: 'arn:aws:s3:::bucket'
1225
1225
  },
1226
1226
  SourceSelectionCriteria: {
1227
1227
  ReplicaModifications: {
1228
- Status: "Enabled"
1228
+ Status: 'Enabled'
1229
1229
  }
1230
1230
  }
1231
1231
  };
1232
1232
  mockUseGetBucketReplication.mockReturnValue({
1233
1233
  data: {
1234
1234
  ReplicationConfiguration: {
1235
- Role: "arn:aws:iam::123456789012:role/role",
1235
+ Role: 'arn:aws:iam::123456789012:role/role',
1236
1236
  Rules: [
1237
1237
  rule
1238
1238
  ]
1239
1239
  }
1240
1240
  },
1241
- status: "success"
1241
+ status: 'success'
1242
1242
  });
1243
- renderBucketReplicationFormPage("test-bucket", "replica-mod-rule");
1243
+ renderBucketReplicationFormPage('test-bucket', 'replica-mod-rule');
1244
1244
  await waitFor(()=>{
1245
- const replicaModToggle = findToggleByLabel("Replica modification sync");
1245
+ const replicaModToggle = findToggleByLabel('Replica modification sync');
1246
1246
  expect(replicaModToggle).toBeChecked();
1247
1247
  });
1248
1248
  });
1249
- it("loads rule with deleteMarkerReplication enabled", async ()=>{
1249
+ it('loads rule with deleteMarkerReplication enabled', async ()=>{
1250
1250
  const rule = {
1251
- ID: "delete-marker-rule",
1252
- Status: "Enabled",
1251
+ ID: 'delete-marker-rule',
1252
+ Status: 'Enabled',
1253
1253
  Priority: 1,
1254
1254
  Destination: {
1255
- Bucket: "arn:aws:s3:::bucket"
1255
+ Bucket: 'arn:aws:s3:::bucket'
1256
1256
  },
1257
1257
  DeleteMarkerReplication: {
1258
- Status: "Enabled"
1258
+ Status: 'Enabled'
1259
1259
  }
1260
1260
  };
1261
1261
  mockUseGetBucketReplication.mockReturnValue({
1262
1262
  data: {
1263
1263
  ReplicationConfiguration: {
1264
- Role: "arn:aws:iam::123456789012:role/role",
1264
+ Role: 'arn:aws:iam::123456789012:role/role',
1265
1265
  Rules: [
1266
1266
  rule
1267
1267
  ]
1268
1268
  }
1269
1269
  },
1270
- status: "success"
1270
+ status: 'success'
1271
1271
  });
1272
- renderBucketReplicationFormPage("test-bucket", "delete-marker-rule");
1272
+ renderBucketReplicationFormPage('test-bucket', 'delete-marker-rule');
1273
1273
  await waitFor(()=>{
1274
- const deleteMarkerToggle = findToggleByLabel("Delete marker replication");
1274
+ const deleteMarkerToggle = findToggleByLabel('Delete marker replication');
1275
1275
  expect(deleteMarkerToggle).toBeChecked();
1276
1276
  });
1277
1277
  });
1278
- it("loads rule with cross-account destination", async ()=>{
1278
+ it('loads rule with cross-account destination', async ()=>{
1279
1279
  const rule = {
1280
- ID: "cross-account-rule",
1281
- Status: "Enabled",
1280
+ ID: 'cross-account-rule',
1281
+ Status: 'Enabled',
1282
1282
  Priority: 1,
1283
1283
  Destination: {
1284
- Bucket: "arn:aws:s3:::cross-account-bucket",
1285
- Account: "987654321098"
1284
+ Bucket: 'arn:aws:s3:::cross-account-bucket',
1285
+ Account: '987654321098'
1286
1286
  }
1287
1287
  };
1288
1288
  mockUseGetBucketReplication.mockReturnValue({
1289
1289
  data: {
1290
1290
  ReplicationConfiguration: {
1291
- Role: "arn:aws:iam::123456789012:role/role",
1291
+ Role: 'arn:aws:iam::123456789012:role/role',
1292
1292
  Rules: [
1293
1293
  rule
1294
1294
  ]
1295
1295
  }
1296
1296
  },
1297
- status: "success"
1297
+ status: 'success'
1298
1298
  });
1299
- renderBucketReplicationFormPage("test-bucket", "cross-account-rule");
1299
+ renderBucketReplicationFormPage('test-bucket', 'cross-account-rule');
1300
1300
  await waitFor(()=>{
1301
1301
  const sameAccountToggle = screen.getByText(/not same account destination/i);
1302
1302
  expect(sameAccountToggle).toBeInTheDocument();
1303
1303
  const accountIdInput = screen.getByLabelText(/target account id/i);
1304
- expect(accountIdInput).toHaveValue("987654321098");
1304
+ expect(accountIdInput).toHaveValue('987654321098');
1305
1305
  const targetBucketInput = screen.getByLabelText(/target bucket/i);
1306
- expect(targetBucketInput).toHaveValue("cross-account-bucket");
1306
+ expect(targetBucketInput).toHaveValue('cross-account-bucket');
1307
1307
  });
1308
1308
  });
1309
- it("loads rule with switchObjectOwnership enabled", async ()=>{
1309
+ it('loads rule with switchObjectOwnership enabled', async ()=>{
1310
1310
  const rule = {
1311
- ID: "ownership-rule",
1312
- Status: "Enabled",
1311
+ ID: 'ownership-rule',
1312
+ Status: 'Enabled',
1313
1313
  Priority: 1,
1314
1314
  Destination: {
1315
- Bucket: "arn:aws:s3:::bucket",
1316
- Account: "987654321098",
1315
+ Bucket: 'arn:aws:s3:::bucket',
1316
+ Account: '987654321098',
1317
1317
  AccessControlTranslation: {
1318
- Owner: "Destination"
1318
+ Owner: 'Destination'
1319
1319
  }
1320
1320
  }
1321
1321
  };
1322
1322
  mockUseGetBucketReplication.mockReturnValue({
1323
1323
  data: {
1324
1324
  ReplicationConfiguration: {
1325
- Role: "arn:aws:iam::123456789012:role/role",
1325
+ Role: 'arn:aws:iam::123456789012:role/role',
1326
1326
  Rules: [
1327
1327
  rule
1328
1328
  ]
1329
1329
  }
1330
1330
  },
1331
- status: "success"
1331
+ status: 'success'
1332
1332
  });
1333
- renderBucketReplicationFormPage("test-bucket", "ownership-rule");
1333
+ renderBucketReplicationFormPage('test-bucket', 'ownership-rule');
1334
1334
  await waitFor(()=>{
1335
- const ownershipToggle = findToggleByLabel("Switch Object ownership");
1335
+ const ownershipToggle = findToggleByLabel('Switch Object ownership');
1336
1336
  expect(ownershipToggle).toBeChecked();
1337
1337
  });
1338
1338
  });
1339
- it("loads rule with custom StorageClass", async ()=>{
1339
+ it('loads rule with custom StorageClass', async ()=>{
1340
1340
  const rule = {
1341
- ID: "storage-class-rule",
1342
- Status: "Enabled",
1341
+ ID: 'storage-class-rule',
1342
+ Status: 'Enabled',
1343
1343
  Priority: 1,
1344
1344
  Destination: {
1345
- Bucket: "arn:aws:s3:::bucket",
1346
- StorageClass: "GLACIER"
1345
+ Bucket: 'arn:aws:s3:::bucket',
1346
+ StorageClass: 'GLACIER'
1347
1347
  }
1348
1348
  };
1349
1349
  mockUseGetBucketReplication.mockReturnValue({
1350
1350
  data: {
1351
1351
  ReplicationConfiguration: {
1352
- Role: "arn:aws:iam::123456789012:role/role",
1352
+ Role: 'arn:aws:iam::123456789012:role/role',
1353
1353
  Rules: [
1354
1354
  rule
1355
1355
  ]
1356
1356
  }
1357
1357
  },
1358
- status: "success"
1358
+ status: 'success'
1359
1359
  });
1360
- renderBucketReplicationFormPage("test-bucket", "storage-class-rule");
1360
+ renderBucketReplicationFormPage('test-bucket', 'storage-class-rule');
1361
1361
  await waitFor(()=>{
1362
1362
  const storageClassSelect = screen.getByLabelText(/storage class/i);
1363
1363
  expect(storageClassSelect).toBeInTheDocument();
1364
1364
  });
1365
1365
  });
1366
- it("loads rule with custom Priority", async ()=>{
1366
+ it('loads rule with custom Priority', async ()=>{
1367
1367
  const rule = {
1368
- ID: "priority-rule",
1369
- Status: "Enabled",
1368
+ ID: 'priority-rule',
1369
+ Status: 'Enabled',
1370
1370
  Priority: 99,
1371
1371
  Destination: {
1372
- Bucket: "arn:aws:s3:::bucket"
1372
+ Bucket: 'arn:aws:s3:::bucket'
1373
1373
  }
1374
1374
  };
1375
1375
  mockUseGetBucketReplication.mockReturnValue({
1376
1376
  data: {
1377
1377
  ReplicationConfiguration: {
1378
- Role: "arn:aws:iam::123456789012:role/role",
1378
+ Role: 'arn:aws:iam::123456789012:role/role',
1379
1379
  Rules: [
1380
1380
  rule
1381
1381
  ]
1382
1382
  }
1383
1383
  },
1384
- status: "success"
1384
+ status: 'success'
1385
1385
  });
1386
- renderBucketReplicationFormPage("test-bucket", "priority-rule");
1386
+ renderBucketReplicationFormPage('test-bucket', 'priority-rule');
1387
1387
  await waitFor(()=>{
1388
1388
  const priorityInput = screen.getByLabelText(/rule priority/i);
1389
1389
  expect(priorityInput).toHaveValue(99);
1390
1390
  });
1391
1391
  });
1392
1392
  });
1393
- describe("Priority Auto-assignment", ()=>{
1394
- it("auto-assigns priority 0 when no existing rules", ()=>{
1393
+ describe('Priority Auto-assignment', ()=>{
1394
+ it('auto-assigns priority 0 when no existing rules', ()=>{
1395
1395
  renderBucketReplicationFormPage();
1396
1396
  const priorityInput = screen.getByLabelText(/rule priority/i);
1397
- expect(priorityInput).toHaveAttribute("placeholder", "Example: Auto-assigned: 0");
1397
+ expect(priorityInput).toHaveAttribute('placeholder', 'Example: Auto-assigned: 0');
1398
1398
  });
1399
- it("auto-assigns next available priority when rules exist", ()=>{
1399
+ it('auto-assigns next available priority when rules exist', ()=>{
1400
1400
  mockUseGetBucketReplication.mockReturnValue({
1401
1401
  data: {
1402
1402
  ReplicationConfiguration: {
1403
- Role: "arn:aws:iam::123456789012:role/role",
1403
+ Role: 'arn:aws:iam::123456789012:role/role',
1404
1404
  Rules: [
1405
1405
  {
1406
- ID: "rule-1",
1407
- Status: "Enabled",
1406
+ ID: 'rule-1',
1407
+ Status: 'Enabled',
1408
1408
  Priority: 5,
1409
1409
  Destination: {
1410
- Bucket: "arn:aws:s3:::bucket"
1410
+ Bucket: 'arn:aws:s3:::bucket'
1411
1411
  }
1412
1412
  },
1413
1413
  {
1414
- ID: "rule-2",
1415
- Status: "Enabled",
1414
+ ID: 'rule-2',
1415
+ Status: 'Enabled',
1416
1416
  Priority: 10,
1417
1417
  Destination: {
1418
- Bucket: "arn:aws:s3:::bucket"
1418
+ Bucket: 'arn:aws:s3:::bucket'
1419
1419
  }
1420
1420
  }
1421
1421
  ]
1422
1422
  }
1423
1423
  },
1424
- status: "success"
1424
+ status: 'success'
1425
1425
  });
1426
1426
  renderBucketReplicationFormPage();
1427
1427
  const priorityInput = screen.getByLabelText(/rule priority/i);
1428
- expect(priorityInput).toHaveAttribute("placeholder", "Example: Auto-assigned: 11");
1428
+ expect(priorityInput).toHaveAttribute('placeholder', 'Example: Auto-assigned: 11');
1429
1429
  });
1430
- it("allows manual priority override", async ()=>{
1430
+ it('allows manual priority override', async ()=>{
1431
1431
  renderBucketReplicationFormPage();
1432
1432
  const priorityInput = screen.getByLabelText(/rule priority/i);
1433
- await user_event.type(priorityInput, "25");
1433
+ await user_event.type(priorityInput, '25');
1434
1434
  expect(priorityInput).toHaveValue(25);
1435
1435
  });
1436
- it("accepts empty priority and defaults to 0 on submit", async ()=>{
1436
+ it('accepts empty priority and defaults to 0 on submit', async ()=>{
1437
1437
  renderBucketReplicationFormPage();
1438
- await user_event.type(screen.getByLabelText(/role arn/i), "arn:aws:iam::123456789012:role/role");
1439
- await user_event.type(screen.getByLabelText(/rule id/i), "no-priority-rule");
1438
+ await user_event.type(screen.getByLabelText(/role arn/i), 'arn:aws:iam::123456789012:role/role');
1439
+ await user_event.type(screen.getByLabelText(/rule id/i), 'no-priority-rule');
1440
1440
  const targetBucketSelect = screen.getByLabelText(/target bucket/i);
1441
1441
  await user_event.click(targetBucketSelect);
1442
- await user_event.click(screen.getByRole("option", {
1443
- name: "destination-bucket"
1442
+ await user_event.click(screen.getByRole('option', {
1443
+ name: 'destination-bucket'
1444
1444
  }));
1445
1445
  mockMutate.mockImplementation((_, options)=>{
1446
1446
  options?.onSuccess?.();
1447
1447
  });
1448
1448
  await waitFor(()=>{
1449
- const createButton = screen.getByRole("button", {
1449
+ const createButton = screen.getByRole('button', {
1450
1450
  name: /create/i
1451
1451
  });
1452
1452
  expect(createButton).toBeEnabled();
1453
1453
  });
1454
- fireEvent.click(screen.getByRole("button", {
1454
+ fireEvent.click(screen.getByRole('button', {
1455
1455
  name: /create/i
1456
1456
  }));
1457
1457
  await waitFor(()=>{
@@ -1460,209 +1460,209 @@ describe("BucketReplicationFormPage", ()=>{
1460
1460
  });
1461
1461
  });
1462
1462
  });
1463
- describe("Target Bucket Selection", ()=>{
1464
- it("renders target bucket field for same-account replication", ()=>{
1463
+ describe('Target Bucket Selection', ()=>{
1464
+ it('renders target bucket field for same-account replication', ()=>{
1465
1465
  renderBucketReplicationFormPage();
1466
1466
  const targetBucketSelect = screen.getByLabelText(/target bucket/i);
1467
1467
  expect(targetBucketSelect).toBeInTheDocument();
1468
1468
  });
1469
- it("filters out current bucket from bucket list", async ()=>{
1469
+ it('filters out current bucket from bucket list', async ()=>{
1470
1470
  renderBucketReplicationFormPage();
1471
1471
  const targetBucketSelect = screen.getByLabelText(/target bucket/i);
1472
1472
  await user_event.click(targetBucketSelect);
1473
- expect(screen.getByRole("option", {
1474
- name: "destination-bucket"
1473
+ expect(screen.getByRole('option', {
1474
+ name: 'destination-bucket'
1475
1475
  })).toBeInTheDocument();
1476
- expect(screen.getByRole("option", {
1477
- name: "backup-bucket"
1476
+ expect(screen.getByRole('option', {
1477
+ name: 'backup-bucket'
1478
1478
  })).toBeInTheDocument();
1479
- expect(screen.queryByRole("option", {
1480
- name: "test-bucket"
1479
+ expect(screen.queryByRole('option', {
1480
+ name: 'test-bucket'
1481
1481
  })).not.toBeInTheDocument();
1482
1482
  });
1483
- it("shows text Input for cross-account replication", async ()=>{
1483
+ it('shows text Input for cross-account replication', async ()=>{
1484
1484
  renderBucketReplicationFormPage();
1485
1485
  const sameAccountToggle = screen.getByText(/same account destination/i);
1486
1486
  await user_event.click(sameAccountToggle);
1487
1487
  await waitFor(()=>{
1488
1488
  const targetBucketInput = screen.getByLabelText(/target bucket/i);
1489
- expect(targetBucketInput.tagName).toBe("INPUT");
1489
+ expect(targetBucketInput.tagName).toBe('INPUT');
1490
1490
  });
1491
1491
  });
1492
- it("loads available buckets using useBuckets hook", ()=>{
1492
+ it('loads available buckets using useBuckets hook', ()=>{
1493
1493
  renderBucketReplicationFormPage();
1494
1494
  expect(mockUseBuckets).toHaveBeenCalled();
1495
1495
  });
1496
1496
  });
1497
- describe("Navigation & Button States", ()=>{
1498
- it("navigates back to bucket replication tab on Cancel", ()=>{
1497
+ describe('Navigation & Button States', ()=>{
1498
+ it('navigates back to bucket replication tab on Cancel', ()=>{
1499
1499
  renderBucketReplicationFormPage();
1500
- const cancelButton = screen.getByRole("button", {
1500
+ const cancelButton = screen.getByRole('button', {
1501
1501
  name: /cancel/i
1502
1502
  });
1503
1503
  fireEvent.click(cancelButton);
1504
- expect(mockNavigate).toHaveBeenCalledWith("/buckets/test-bucket?tab=replication");
1504
+ expect(mockNavigate).toHaveBeenCalledWith('/buckets/test-bucket?tab=replication');
1505
1505
  });
1506
- it("disables Cancel button when mutation is pending", ()=>{
1506
+ it('disables Cancel button when mutation is pending', ()=>{
1507
1507
  mockUseSetBucketReplication.mockReturnValue({
1508
1508
  mutate: mockMutate,
1509
1509
  isPending: true
1510
1510
  });
1511
1511
  renderBucketReplicationFormPage();
1512
- const cancelButton = screen.getByRole("button", {
1512
+ const cancelButton = screen.getByRole('button', {
1513
1513
  name: /cancel/i
1514
1514
  });
1515
1515
  expect(cancelButton).toBeDisabled();
1516
1516
  });
1517
- it("disables Submit button when mutation is pending", async ()=>{
1517
+ it('disables Submit button when mutation is pending', async ()=>{
1518
1518
  mockUseSetBucketReplication.mockReturnValue({
1519
1519
  mutate: mockMutate,
1520
1520
  isPending: true
1521
1521
  });
1522
1522
  renderBucketReplicationFormPage();
1523
- await user_event.type(screen.getByLabelText(/role arn/i), "arn:aws:iam::123456789012:role/role");
1524
- await user_event.type(screen.getByLabelText(/rule id/i), "test-rule");
1523
+ await user_event.type(screen.getByLabelText(/role arn/i), 'arn:aws:iam::123456789012:role/role');
1524
+ await user_event.type(screen.getByLabelText(/rule id/i), 'test-rule');
1525
1525
  const targetBucketSelect = screen.getByLabelText(/target bucket/i);
1526
1526
  await user_event.click(targetBucketSelect);
1527
- await user_event.click(screen.getByRole("option", {
1528
- name: "destination-bucket"
1527
+ await user_event.click(screen.getByRole('option', {
1528
+ name: 'destination-bucket'
1529
1529
  }));
1530
1530
  await waitFor(()=>{
1531
- const createButton = screen.getByRole("button", {
1531
+ const createButton = screen.getByRole('button', {
1532
1532
  name: /create/i
1533
1533
  });
1534
1534
  expect(createButton).toBeDisabled();
1535
1535
  });
1536
1536
  });
1537
- it("shows Create button with no icon in create mode", ()=>{
1537
+ it('shows Create button with no icon in create mode', ()=>{
1538
1538
  renderBucketReplicationFormPage();
1539
- const createButton = screen.getByRole("button", {
1539
+ const createButton = screen.getByRole('button', {
1540
1540
  name: /create/i
1541
1541
  });
1542
1542
  expect(createButton).toBeInTheDocument();
1543
- expect(createButton.querySelector("svg")).not.toBeInTheDocument();
1543
+ expect(createButton.querySelector('svg')).not.toBeInTheDocument();
1544
1544
  });
1545
- it("shows Save button with Save icon in edit mode", async ()=>{
1545
+ it('shows Save button with Save icon in edit mode', async ()=>{
1546
1546
  const existingRule = {
1547
- ID: "edit-rule",
1548
- Status: "Enabled",
1547
+ ID: 'edit-rule',
1548
+ Status: 'Enabled',
1549
1549
  Priority: 1,
1550
1550
  Destination: {
1551
- Bucket: "arn:aws:s3:::bucket"
1551
+ Bucket: 'arn:aws:s3:::bucket'
1552
1552
  }
1553
1553
  };
1554
1554
  mockUseGetBucketReplication.mockReturnValue({
1555
1555
  data: {
1556
1556
  ReplicationConfiguration: {
1557
- Role: "arn:aws:iam::123456789012:role/role",
1557
+ Role: 'arn:aws:iam::123456789012:role/role',
1558
1558
  Rules: [
1559
1559
  existingRule
1560
1560
  ]
1561
1561
  }
1562
1562
  },
1563
- status: "success"
1563
+ status: 'success'
1564
1564
  });
1565
- renderBucketReplicationFormPage("test-bucket", "edit-rule");
1565
+ renderBucketReplicationFormPage('test-bucket', 'edit-rule');
1566
1566
  await waitFor(()=>{
1567
- const saveButton = screen.getByRole("button", {
1567
+ const saveButton = screen.getByRole('button', {
1568
1568
  name: /save/i
1569
1569
  });
1570
1570
  expect(saveButton).toBeInTheDocument();
1571
- expect(saveButton.querySelector("svg")).toBeInTheDocument();
1571
+ expect(saveButton.querySelector('svg')).toBeInTheDocument();
1572
1572
  });
1573
1573
  });
1574
1574
  });
1575
- describe("Edge Cases", ()=>{
1576
- it("handles URL-encoded rule ID in edit mode", async ()=>{
1575
+ describe('Edge Cases', ()=>{
1576
+ it('handles URL-encoded rule ID in edit mode', async ()=>{
1577
1577
  const existingRule = {
1578
- ID: "rule with spaces",
1579
- Status: "Enabled",
1578
+ ID: 'rule with spaces',
1579
+ Status: 'Enabled',
1580
1580
  Priority: 1,
1581
1581
  Destination: {
1582
- Bucket: "arn:aws:s3:::bucket"
1582
+ Bucket: 'arn:aws:s3:::bucket'
1583
1583
  }
1584
1584
  };
1585
1585
  mockUseGetBucketReplication.mockReturnValue({
1586
1586
  data: {
1587
1587
  ReplicationConfiguration: {
1588
- Role: "arn:aws:iam::123456789012:role/role",
1588
+ Role: 'arn:aws:iam::123456789012:role/role',
1589
1589
  Rules: [
1590
1590
  existingRule
1591
1591
  ]
1592
1592
  }
1593
1593
  },
1594
- status: "success"
1594
+ status: 'success'
1595
1595
  });
1596
- renderBucketReplicationFormPage("test-bucket", encodeURIComponent("rule with spaces"));
1596
+ renderBucketReplicationFormPage('test-bucket', encodeURIComponent('rule with spaces'));
1597
1597
  await waitFor(()=>{
1598
- expect(screen.getByText("rule with spaces")).toBeInTheDocument();
1598
+ expect(screen.getByText('rule with spaces')).toBeInTheDocument();
1599
1599
  });
1600
1600
  });
1601
- it("handles rules without Priority field", async ()=>{
1601
+ it('handles rules without Priority field', async ()=>{
1602
1602
  const rule = {
1603
- ID: "no-priority",
1604
- Status: "Enabled",
1603
+ ID: 'no-priority',
1604
+ Status: 'Enabled',
1605
1605
  Destination: {
1606
- Bucket: "arn:aws:s3:::bucket"
1606
+ Bucket: 'arn:aws:s3:::bucket'
1607
1607
  }
1608
1608
  };
1609
1609
  mockUseGetBucketReplication.mockReturnValue({
1610
1610
  data: {
1611
1611
  ReplicationConfiguration: {
1612
- Role: "arn:aws:iam::123456789012:role/role",
1612
+ Role: 'arn:aws:iam::123456789012:role/role',
1613
1613
  Rules: [
1614
1614
  rule
1615
1615
  ]
1616
1616
  }
1617
1617
  },
1618
- status: "success"
1618
+ status: 'success'
1619
1619
  });
1620
- renderBucketReplicationFormPage("test-bucket", "no-priority");
1620
+ renderBucketReplicationFormPage('test-bucket', 'no-priority');
1621
1621
  await waitFor(()=>{
1622
1622
  const priorityInput = screen.getByLabelText(/rule priority/i);
1623
1623
  expect(priorityInput).toHaveValue(null);
1624
1624
  });
1625
1625
  });
1626
- it("handles rules without StorageClass", async ()=>{
1626
+ it('handles rules without StorageClass', async ()=>{
1627
1627
  const rule = {
1628
- ID: "no-storage-class",
1629
- Status: "Enabled",
1628
+ ID: 'no-storage-class',
1629
+ Status: 'Enabled',
1630
1630
  Priority: 1,
1631
1631
  Destination: {
1632
- Bucket: "arn:aws:s3:::bucket"
1632
+ Bucket: 'arn:aws:s3:::bucket'
1633
1633
  }
1634
1634
  };
1635
1635
  mockUseGetBucketReplication.mockReturnValue({
1636
1636
  data: {
1637
1637
  ReplicationConfiguration: {
1638
- Role: "arn:aws:iam::123456789012:role/role",
1638
+ Role: 'arn:aws:iam::123456789012:role/role',
1639
1639
  Rules: [
1640
1640
  rule
1641
1641
  ]
1642
1642
  }
1643
1643
  },
1644
- status: "success"
1644
+ status: 'success'
1645
1645
  });
1646
- renderBucketReplicationFormPage("test-bucket", "no-storage-class");
1646
+ renderBucketReplicationFormPage('test-bucket', 'no-storage-class');
1647
1647
  await waitFor(()=>{
1648
1648
  const storageClassSelect = screen.getByLabelText(/storage class/i);
1649
1649
  expect(storageClassSelect).toBeInTheDocument();
1650
1650
  });
1651
1651
  });
1652
1652
  });
1653
- describe("Boundary Conditions", ()=>{
1654
- it("accepts Rule ID with reasonable length", async ()=>{
1653
+ describe('Boundary Conditions', ()=>{
1654
+ it('accepts Rule ID with reasonable length', async ()=>{
1655
1655
  renderBucketReplicationFormPage();
1656
- const longRuleId = "rule-name-with-many-segments-" + "segment-".repeat(10);
1656
+ const longRuleId = 'rule-name-with-many-segments-' + 'segment-'.repeat(10);
1657
1657
  const ruleIdInput = screen.getByLabelText(/rule id/i);
1658
1658
  await user_event.type(ruleIdInput, longRuleId);
1659
1659
  await waitFor(()=>{
1660
1660
  expect(ruleIdInput).toHaveValue(longRuleId);
1661
1661
  });
1662
1662
  });
1663
- it("handles special characters in Rule ID", async ()=>{
1663
+ it('handles special characters in Rule ID', async ()=>{
1664
1664
  renderBucketReplicationFormPage();
1665
- const validSpecialChars = "rule-name_with.special-chars_123";
1665
+ const validSpecialChars = 'rule-name_with.special-chars_123';
1666
1666
  const ruleIdInput = screen.getByLabelText(/rule id/i);
1667
1667
  await user_event.type(ruleIdInput, validSpecialChars);
1668
1668
  await user_event.tab();
@@ -1670,84 +1670,84 @@ describe("BucketReplicationFormPage", ()=>{
1670
1670
  expect(ruleIdInput).toHaveValue(validSpecialChars);
1671
1671
  });
1672
1672
  });
1673
- it("validates Priority with boundary values", async ()=>{
1673
+ it('validates Priority with boundary values', async ()=>{
1674
1674
  renderBucketReplicationFormPage();
1675
- await user_event.type(screen.getByLabelText(/rule id/i), "boundary-test");
1675
+ await user_event.type(screen.getByLabelText(/rule id/i), 'boundary-test');
1676
1676
  const priorityInput = screen.getByLabelText(/rule priority/i);
1677
1677
  await user_event.clear(priorityInput);
1678
- await user_event.type(priorityInput, "0");
1678
+ await user_event.type(priorityInput, '0');
1679
1679
  await user_event.tab();
1680
1680
  await waitFor(()=>{
1681
1681
  expect(screen.queryByText(/priority must be at least 0/i)).not.toBeInTheDocument();
1682
1682
  });
1683
1683
  await user_event.clear(priorityInput);
1684
- await user_event.type(priorityInput, "2147483647");
1684
+ await user_event.type(priorityInput, '2147483647');
1685
1685
  await user_event.tab();
1686
1686
  await waitFor(()=>{
1687
1687
  expect(priorityInput).toHaveValue(2147483647);
1688
1688
  });
1689
1689
  });
1690
- it("handles very long prefix values", async ()=>{
1690
+ it('handles very long prefix values', async ()=>{
1691
1691
  renderBucketReplicationFormPage();
1692
1692
  const filterSelect = document.querySelector('[id="filterType"]');
1693
1693
  await user_event.click(filterSelect);
1694
1694
  await waitFor(()=>{
1695
- expect(screen.getByRole("option", {
1696
- name: "Prefix filter"
1695
+ expect(screen.getByRole('option', {
1696
+ name: 'Prefix filter'
1697
1697
  })).toBeInTheDocument();
1698
1698
  });
1699
- await user_event.click(screen.getByRole("option", {
1700
- name: "Prefix filter"
1699
+ await user_event.click(screen.getByRole('option', {
1700
+ name: 'Prefix filter'
1701
1701
  }));
1702
1702
  await waitFor(()=>{
1703
1703
  expect(screen.getByLabelText(/prefix/i)).toBeInTheDocument();
1704
1704
  });
1705
- const longPrefix = "logs/" + "a".repeat(100);
1705
+ const longPrefix = 'logs/' + 'a'.repeat(100);
1706
1706
  const prefixInput = screen.getByLabelText(/prefix/i);
1707
1707
  await user_event.type(prefixInput, longPrefix);
1708
1708
  await waitFor(()=>{
1709
1709
  expect(prefixInput).toHaveValue(longPrefix);
1710
1710
  });
1711
1711
  });
1712
- it("accepts valid Role ARN format", async ()=>{
1712
+ it('accepts valid Role ARN format', async ()=>{
1713
1713
  renderBucketReplicationFormPage();
1714
1714
  const roleArnInput = screen.getByLabelText(/role arn/i);
1715
- await user_event.type(roleArnInput, "arn:aws:iam::123456789012:role/replication-role");
1715
+ await user_event.type(roleArnInput, 'arn:aws:iam::123456789012:role/replication-role');
1716
1716
  await user_event.tab();
1717
1717
  await waitFor(()=>{
1718
- expect(roleArnInput).toHaveValue("arn:aws:iam::123456789012:role/replication-role");
1718
+ expect(roleArnInput).toHaveValue('arn:aws:iam::123456789012:role/replication-role');
1719
1719
  });
1720
1720
  });
1721
- it("accepts valid Target Account ID", async ()=>{
1721
+ it('accepts valid Target Account ID', async ()=>{
1722
1722
  renderBucketReplicationFormPage();
1723
- const sameAccountToggle = findToggleByLabel("Same account destination");
1723
+ const sameAccountToggle = findToggleByLabel('Same account destination');
1724
1724
  await user_event.click(sameAccountToggle);
1725
1725
  await waitFor(()=>{
1726
1726
  expect(screen.getByLabelText(/target account id/i)).toBeInTheDocument();
1727
1727
  });
1728
1728
  const accountIdInput = screen.getByLabelText(/target account id/i);
1729
- await user_event.type(accountIdInput, "123456789012");
1729
+ await user_event.type(accountIdInput, '123456789012');
1730
1730
  await user_event.tab();
1731
1731
  await waitFor(()=>{
1732
- expect(accountIdInput).toHaveValue("123456789012");
1732
+ expect(accountIdInput).toHaveValue('123456789012');
1733
1733
  });
1734
1734
  });
1735
- it("handles empty form submission attempt", async ()=>{
1735
+ it('handles empty form submission attempt', async ()=>{
1736
1736
  renderBucketReplicationFormPage();
1737
- const createButton = screen.getByRole("button", {
1737
+ const createButton = screen.getByRole('button', {
1738
1738
  name: /create/i
1739
1739
  });
1740
1740
  expect(createButton).toBeDisabled();
1741
1741
  });
1742
- it("validates KMS Key ID format", async ()=>{
1742
+ it('validates KMS Key ID format', async ()=>{
1743
1743
  renderBucketReplicationFormPage();
1744
- const encryptToggle = findToggleByLabel("Encrypt replicated objects");
1744
+ const encryptToggle = findToggleByLabel('Encrypt replicated objects');
1745
1745
  await user_event.click(encryptToggle);
1746
1746
  await waitFor(()=>{
1747
1747
  expect(screen.getByLabelText(/replica kms key id/i)).toBeInTheDocument();
1748
1748
  });
1749
1749
  const kmsKeyInput = screen.getByLabelText(/replica kms key id/i);
1750
- const validKmsArn = "arn:aws:kms:us-east-1:123456789012:key/12345678-1234-1234-1234-123456789012";
1750
+ const validKmsArn = 'arn:aws:kms:us-east-1:123456789012:key/12345678-1234-1234-1234-123456789012';
1751
1751
  await user_event.type(kmsKeyInput, validKmsArn);
1752
1752
  await waitFor(()=>{
1753
1753
  expect(kmsKeyInput).toHaveValue(validKmsArn);