@scality/data-browser-library 1.0.0-preview.9 → 1.0.1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/components/DataBrowserUI.d.ts +12 -0
- package/dist/components/DataBrowserUI.js +99 -0
- package/dist/components/Editor.d.ts +1 -1
- package/dist/components/Editor.js +3 -3
- package/dist/components/__tests__/BucketAccessor.test.js +214 -0
- package/dist/components/__tests__/BucketCorsPage.test.d.ts +1 -0
- package/dist/components/__tests__/BucketCorsPage.test.js +263 -0
- package/dist/components/__tests__/BucketCreate.test.js +271 -105
- package/dist/components/__tests__/BucketDetails.test.d.ts +1 -0
- package/dist/components/__tests__/BucketDetails.test.js +421 -0
- package/dist/components/__tests__/BucketLifecycleFormPage.test.d.ts +13 -0
- package/dist/components/__tests__/BucketLifecycleFormPage.test.js +178 -178
- package/dist/components/__tests__/BucketLifecycleList.test.js +85 -85
- package/dist/components/__tests__/BucketList.test.js +463 -239
- package/dist/components/__tests__/BucketNotificationFormPage.test.d.ts +1 -0
- package/dist/components/__tests__/BucketNotificationFormPage.test.js +348 -0
- package/dist/components/__tests__/BucketNotificationList.test.d.ts +1 -0
- package/dist/components/__tests__/BucketNotificationList.test.js +379 -0
- package/dist/components/__tests__/BucketOverview.test.js +281 -266
- package/dist/components/__tests__/BucketPolicyPage.test.js +151 -99
- package/dist/components/__tests__/BucketReplicationFormPage.test.d.ts +15 -0
- package/dist/components/__tests__/BucketReplicationFormPage.test.js +544 -544
- package/dist/components/__tests__/BucketReplicationList.test.js +106 -106
- package/dist/components/__tests__/CreateFolderButton.test.js +56 -56
- package/dist/components/__tests__/DeleteBucketButton.test.js +64 -64
- package/dist/components/__tests__/DeleteBucketConfigRuleButton.test.js +47 -47
- package/dist/components/__tests__/DeleteObjectButton.test.js +64 -64
- package/dist/components/__tests__/EmptyBucketButton.test.js +59 -59
- package/dist/components/__tests__/MetadataSearch.test.js +65 -65
- package/dist/components/__tests__/ObjectList.test.js +741 -240
- package/dist/components/__tests__/UploadButton.test.js +45 -45
- package/dist/components/breadcrumb/Breadcrumb.d.ts +6 -0
- package/dist/components/breadcrumb/Breadcrumb.js +37 -0
- package/dist/components/breadcrumb/DataBrowserBreadcrumb.d.ts +1 -0
- package/dist/components/breadcrumb/DataBrowserBreadcrumb.js +10 -0
- package/dist/components/breadcrumb/__tests__/Breadcrumb.test.d.ts +1 -0
- package/dist/components/breadcrumb/__tests__/Breadcrumb.test.js +196 -0
- package/dist/components/breadcrumb/__tests__/DataBrowserBreadcrumb.test.d.ts +1 -0
- package/dist/components/breadcrumb/__tests__/DataBrowserBreadcrumb.test.js +153 -0
- package/dist/components/breadcrumb/__tests__/useBreadcrumbPaths.test.d.ts +1 -0
- package/dist/components/breadcrumb/__tests__/useBreadcrumbPaths.test.js +134 -0
- package/dist/components/breadcrumb/index.d.ts +8 -0
- package/dist/components/breadcrumb/index.js +4 -0
- package/dist/components/breadcrumb/useBreadcrumbPaths.d.ts +2 -0
- package/dist/components/breadcrumb/useBreadcrumbPaths.js +82 -0
- package/dist/components/buckets/BucketAccessor.d.ts +2 -0
- package/dist/components/buckets/BucketAccessor.js +125 -0
- package/dist/components/buckets/BucketConfigEditButton.d.ts +8 -0
- package/dist/components/buckets/{BucketPolicyButton.js → BucketConfigEditButton.js} +9 -5
- package/dist/components/buckets/BucketCorsPage.d.ts +1 -0
- package/dist/components/buckets/BucketCorsPage.js +234 -0
- package/dist/components/buckets/BucketCreate.d.ts +3 -2
- package/dist/components/buckets/BucketCreate.js +89 -47
- package/dist/components/buckets/BucketDetails.d.ts +42 -0
- package/dist/components/buckets/BucketDetails.js +249 -85
- package/dist/components/buckets/BucketLifecycleFormPage.js +206 -190
- package/dist/components/buckets/BucketLifecycleList.d.ts +2 -2
- package/dist/components/buckets/BucketLifecycleList.js +47 -47
- package/dist/components/buckets/BucketList.d.ts +7 -8
- package/dist/components/buckets/BucketList.js +158 -101
- package/dist/components/buckets/BucketLocation.js +4 -4
- package/dist/components/buckets/BucketOverview.d.ts +22 -2
- package/dist/components/buckets/BucketOverview.js +394 -187
- package/dist/components/buckets/BucketPage.js +43 -21
- package/dist/components/buckets/BucketPolicyPage.js +155 -127
- package/dist/components/buckets/BucketReplicationFormPage.js +134 -133
- package/dist/components/buckets/BucketReplicationList.d.ts +2 -2
- package/dist/components/buckets/BucketReplicationList.js +42 -42
- package/dist/components/buckets/BucketVersioning.d.ts +4 -0
- package/dist/components/buckets/BucketVersioning.js +76 -0
- package/dist/components/buckets/DeleteBucketButton.js +8 -8
- package/dist/components/buckets/DeleteBucketConfigRuleButton.d.ts +2 -2
- package/dist/components/buckets/DeleteBucketConfigRuleButton.js +2 -2
- package/dist/components/buckets/EmptyBucketButton.js +24 -24
- package/dist/components/buckets/EmptyBucketSummary.d.ts +2 -2
- package/dist/components/buckets/EmptyBucketSummary.js +1 -1
- package/dist/components/buckets/EmptyBucketSummaryList.d.ts +1 -1
- package/dist/components/buckets/EmptyBucketSummaryList.js +22 -22
- package/dist/components/buckets/__tests__/BucketVersioning.test.d.ts +1 -0
- package/dist/components/buckets/__tests__/BucketVersioning.test.js +163 -0
- package/dist/components/buckets/notifications/BucketNotificationFormPage.d.ts +1 -0
- package/dist/components/buckets/notifications/BucketNotificationFormPage.js +316 -0
- package/dist/components/buckets/notifications/BucketNotificationList.d.ts +10 -0
- package/dist/components/buckets/notifications/BucketNotificationList.js +267 -0
- package/dist/components/buckets/notifications/EventsSection.js +145 -29
- package/dist/components/buckets/notifications/__tests__/events.test.d.ts +1 -0
- package/dist/components/buckets/notifications/__tests__/events.test.js +56 -0
- package/dist/components/buckets/notifications/events.d.ts +71 -7
- package/dist/components/buckets/notifications/events.js +98 -16
- package/dist/components/index.d.ts +27 -20
- package/dist/components/index.js +17 -10
- package/dist/components/layouts/ArrowNavigation.d.ts +3 -0
- package/dist/components/layouts/ArrowNavigation.js +28 -0
- package/dist/components/layouts/BrowserPageLayout.d.ts +5 -1
- package/dist/components/layouts/BrowserPageLayout.js +10 -5
- package/dist/components/objects/CreateFolderButton.d.ts +2 -2
- package/dist/components/objects/CreateFolderButton.js +12 -12
- package/dist/components/objects/DeleteObjectButton.d.ts +1 -1
- package/dist/components/objects/DeleteObjectButton.js +19 -21
- package/dist/components/objects/GetPresignedUrlButton.d.ts +7 -0
- package/dist/components/objects/GetPresignedUrlButton.js +255 -0
- package/dist/components/objects/ObjectDetails/ObjectMetadata.d.ts +2 -2
- package/dist/components/objects/ObjectDetails/ObjectMetadata.js +263 -230
- package/dist/components/objects/ObjectDetails/ObjectSummary.d.ts +2 -2
- package/dist/components/objects/ObjectDetails/ObjectSummary.js +540 -138
- package/dist/components/objects/ObjectDetails/ObjectTags.d.ts +2 -2
- package/dist/components/objects/ObjectDetails/ObjectTags.js +95 -123
- package/dist/components/objects/ObjectDetails/__tests__/ObjectDetails.test.d.ts +1 -0
- package/dist/components/objects/ObjectDetails/__tests__/ObjectDetails.test.js +516 -0
- package/dist/components/objects/ObjectDetails/__tests__/ObjectSummary.test.d.ts +1 -0
- package/dist/components/objects/ObjectDetails/__tests__/ObjectSummary.test.js +1064 -0
- package/dist/components/objects/ObjectDetails/index.d.ts +18 -2
- package/dist/components/objects/ObjectDetails/index.js +152 -40
- package/dist/components/objects/ObjectList.d.ts +12 -10
- package/dist/components/objects/ObjectList.js +590 -263
- package/dist/components/objects/ObjectLock/EditRetentionButton.js +4 -4
- package/dist/components/objects/ObjectLock/ObjectLockRetentionSettings.js +15 -15
- package/dist/components/objects/ObjectLock/ObjectLockSettings.d.ts +1 -1
- package/dist/components/objects/ObjectLock/ObjectLockSettings.js +32 -31
- package/dist/components/objects/ObjectLock/ObjectLockSettingsUtils.d.ts +1 -1
- package/dist/components/objects/ObjectLock/ObjectLockSettingsUtils.js +6 -6
- package/dist/components/objects/ObjectLock/__tests__/EditRetentionButton.test.js +51 -51
- package/dist/components/objects/ObjectLock/__tests__/ObjectLockSettings.test.js +78 -78
- package/dist/components/objects/ObjectPage.js +12 -8
- package/dist/components/objects/UploadButton.d.ts +3 -3
- package/dist/components/objects/UploadButton.js +10 -10
- package/dist/components/objects/__tests__/GetPresignedUrlButton.test.d.ts +1 -0
- package/dist/components/objects/__tests__/GetPresignedUrlButton.test.js +531 -0
- package/dist/components/providers/DataBrowserProvider.d.ts +23 -12
- package/dist/components/providers/DataBrowserProvider.js +60 -38
- package/dist/components/providers/QueryProvider.d.ts +9 -0
- package/dist/components/providers/QueryProvider.js +21 -0
- package/dist/components/search/MetadataSearch.js +29 -28
- package/dist/components/search/SearchHints.js +1 -1
- package/dist/components/ui/ArrayFieldActions.js +12 -7
- package/dist/components/ui/ConfirmDeleteRuleModal.d.ts +2 -2
- package/dist/components/ui/ConfirmDeleteRuleModal.js +6 -1
- package/dist/components/ui/DeleteObjectModalContent.d.ts +1 -1
- package/dist/components/ui/DeleteObjectModalContent.js +12 -12
- package/dist/components/ui/FilterFormSection.d.ts +2 -2
- package/dist/components/ui/FilterFormSection.js +29 -29
- package/dist/components/ui/Search.elements.d.ts +2 -2
- package/dist/components/ui/Search.elements.js +7 -7
- package/dist/components/ui/Table.elements.d.ts +2 -1
- package/dist/components/ui/Table.elements.js +18 -12
- package/dist/config/__tests__/factory.test.d.ts +1 -0
- package/dist/config/__tests__/factory.test.js +311 -0
- package/dist/config/factory.d.ts +10 -56
- package/dist/config/factory.js +23 -71
- package/dist/config/types.d.ts +212 -34
- package/dist/contexts/DataBrowserUICustomizationContext.d.ts +27 -0
- package/dist/contexts/DataBrowserUICustomizationContext.js +13 -0
- package/dist/hooks/__tests__/useAccessibleBuckets.test.d.ts +1 -0
- package/dist/hooks/__tests__/useAccessibleBuckets.test.js +145 -0
- package/dist/hooks/__tests__/useISVBucketDetection.test.js +45 -45
- package/dist/hooks/__tests__/useIsBucketEmpty.test.js +27 -27
- package/dist/hooks/__tests__/useLoginMutation.test.d.ts +1 -0
- package/dist/hooks/__tests__/useLoginMutation.test.js +194 -0
- package/dist/hooks/bucketConfiguration.d.ts +8 -1
- package/dist/hooks/bucketConfiguration.js +52 -51
- package/dist/hooks/bucketOperations.d.ts +10 -1
- package/dist/hooks/bucketOperations.js +10 -9
- package/dist/hooks/factories/__tests__/useCreateS3FunctionMutationHook.test.js +80 -80
- package/dist/hooks/factories/__tests__/useCreateS3InfiniteQueryHook.test.js +80 -80
- package/dist/hooks/factories/__tests__/useCreateS3LoginHook.test.js +44 -44
- package/dist/hooks/factories/__tests__/useCreateS3MutationHook.test.js +63 -63
- package/dist/hooks/factories/__tests__/useCreateS3QueryHook.test.js +65 -65
- package/dist/hooks/factories/index.d.ts +4 -4
- package/dist/hooks/factories/index.js +2 -2
- package/dist/hooks/factories/useCreateS3InfiniteQueryHook.d.ts +2 -2
- package/dist/hooks/factories/useCreateS3InfiniteQueryHook.js +16 -13
- package/dist/hooks/factories/useCreateS3LoginHook.d.ts +2 -2
- package/dist/hooks/factories/useCreateS3LoginHook.js +1 -1
- package/dist/hooks/factories/useCreateS3MutationHook.d.ts +3 -3
- package/dist/hooks/factories/useCreateS3MutationHook.js +7 -2
- package/dist/hooks/factories/useCreateS3QueryHook.d.ts +2 -2
- package/dist/hooks/factories/useCreateS3QueryHook.js +11 -6
- package/dist/hooks/index.d.ts +19 -12
- package/dist/hooks/index.js +16 -9
- package/dist/hooks/loginOperations.d.ts +1 -1
- package/dist/hooks/loginOperations.js +1 -1
- package/dist/hooks/objectOperations.d.ts +2 -2
- package/dist/hooks/objectOperations.js +50 -49
- package/dist/hooks/presignedOperations.d.ts +4 -4
- package/dist/hooks/presignedOperations.js +5 -5
- package/dist/hooks/useAccessibleBuckets.d.ts +11 -0
- package/dist/hooks/useAccessibleBuckets.js +115 -0
- package/dist/hooks/useBatchObjectLegalHold.d.ts +11 -0
- package/dist/hooks/useBatchObjectLegalHold.js +48 -0
- package/dist/hooks/useBucketConfigEditor.d.ts +31 -0
- package/dist/hooks/useBucketConfigEditor.js +82 -0
- package/dist/hooks/useDataBrowserNavigate.d.ts +28 -0
- package/dist/hooks/useDataBrowserNavigate.js +24 -0
- package/dist/hooks/useDeleteBucketConfigRule.d.ts +2 -2
- package/dist/hooks/useDeleteBucketConfigRule.js +4 -4
- package/dist/hooks/useEmptyBucket.js +11 -11
- package/dist/hooks/useFeatures.d.ts +7 -0
- package/dist/hooks/useFeatures.js +8 -0
- package/dist/hooks/useISVBucketDetection.js +6 -6
- package/dist/hooks/useIsBucketEmpty.js +4 -4
- package/dist/hooks/useLimitedAccessFlow.d.ts +48 -0
- package/dist/hooks/useLimitedAccessFlow.js +23 -0
- package/dist/hooks/useS3Client.d.ts +6 -0
- package/dist/hooks/useS3Client.js +3 -2
- package/dist/hooks/useS3ConfigSwitch.d.ts +11 -0
- package/dist/hooks/useS3ConfigSwitch.js +37 -0
- package/dist/hooks/useSupportedNotificationEvents.d.ts +6 -0
- package/dist/hooks/useSupportedNotificationEvents.js +8 -0
- package/dist/index.d.ts +6 -6
- package/dist/index.js +2 -2
- package/dist/schemas/bucketPolicySchema.json +3 -13
- package/dist/test/msw/handlers/deleteBucket.d.ts +1 -1
- package/dist/test/msw/handlers/deleteBucket.js +20 -10
- package/dist/test/msw/handlers/getBucketAcl.d.ts +1 -1
- package/dist/test/msw/handlers/getBucketAcl.js +29 -17
- package/dist/test/msw/handlers/getBucketLocation.d.ts +1 -1
- package/dist/test/msw/handlers/getBucketLocation.js +29 -15
- package/dist/test/msw/handlers/getBucketPolicy.d.ts +1 -1
- package/dist/test/msw/handlers/getBucketPolicy.js +52 -32
- package/dist/test/msw/handlers/headObject.d.ts +1 -1
- package/dist/test/msw/handlers/headObject.js +31 -13
- package/dist/test/msw/handlers/listBuckets.d.ts +1 -1
- package/dist/test/msw/handlers/listBuckets.js +5 -3
- package/dist/test/msw/handlers/listObjectVersions.d.ts +1 -1
- package/dist/test/msw/handlers/listObjectVersions.js +38 -26
- package/dist/test/msw/handlers/listObjects.d.ts +1 -1
- package/dist/test/msw/handlers/listObjects.js +35 -23
- package/dist/test/msw/handlers/objectLegalHold.d.ts +1 -1
- package/dist/test/msw/handlers/objectLegalHold.js +32 -17
- package/dist/test/msw/handlers/objectRetention.d.ts +1 -1
- package/dist/test/msw/handlers/objectRetention.js +31 -17
- package/dist/test/msw/handlers/putBucketAcl.d.ts +1 -1
- package/dist/test/msw/handlers/putBucketAcl.js +29 -14
- package/dist/test/msw/handlers/putObject.d.ts +1 -1
- package/dist/test/msw/handlers/putObject.js +27 -12
- package/dist/test/msw/handlers.d.ts +3 -3
- package/dist/test/msw/handlers.js +77 -54
- package/dist/test/msw/index.d.ts +2 -2
- package/dist/test/msw/index.js +1 -1
- package/dist/test/msw/server.d.ts +1 -1
- package/dist/test/msw/server.js +1 -1
- package/dist/test/msw/utils.js +2 -2
- package/dist/test/setup.d.ts +1 -1
- package/dist/test/setup.js +13 -30
- package/dist/test/testUtils.d.ts +85 -33
- package/dist/test/testUtils.js +176 -111
- package/dist/test/utils/errorHandling.test.js +119 -119
- package/dist/types/index.d.ts +49 -36
- package/dist/types/monaco.d.ts +13 -0
- package/dist/types/monaco.js +0 -0
- package/dist/utils/__tests__/proxyMiddleware.test.d.ts +1 -0
- package/dist/utils/__tests__/proxyMiddleware.test.js +579 -0
- package/dist/utils/__tests__/s3Client.test.d.ts +1 -0
- package/dist/utils/__tests__/s3Client.test.js +340 -0
- package/dist/utils/__tests__/s3ConfigIdentifier.test.d.ts +1 -0
- package/dist/utils/__tests__/s3ConfigIdentifier.test.js +437 -0
- package/dist/utils/constants.d.ts +10 -0
- package/dist/utils/constants.js +19 -9
- package/dist/utils/deletion/index.d.ts +2 -2
- package/dist/utils/deletion/index.js +1 -1
- package/dist/utils/deletion/messages.d.ts +1 -1
- package/dist/utils/deletion/messages.js +4 -4
- package/dist/utils/errorHandling.d.ts +3 -3
- package/dist/utils/errorHandling.js +6 -6
- package/dist/utils/hooks.js +8 -8
- package/dist/utils/index.d.ts +5 -4
- package/dist/utils/index.js +4 -2
- package/dist/utils/proxyMiddleware.d.ts +32 -13
- package/dist/utils/proxyMiddleware.js +90 -36
- package/dist/utils/s3Client.d.ts +14 -4
- package/dist/utils/s3Client.js +5 -26
- package/dist/utils/s3ConfigIdentifier.d.ts +79 -0
- package/dist/utils/s3ConfigIdentifier.js +57 -0
- package/dist/utils/s3RuleUtils.d.ts +5 -5
- package/dist/utils/s3RuleUtils.js +17 -17
- package/package.json +10 -8
- package/dist/components/__tests__/BucketNotificationCreatePage.test.js +0 -316
- package/dist/components/buckets/BucketPolicyButton.d.ts +0 -7
- package/dist/components/buckets/notifications/BucketNotificationCreatePage.d.ts +0 -1
- package/dist/components/buckets/notifications/BucketNotificationCreatePage.js +0 -234
- package/dist/hooks/useLoginMutation.d.ts +0 -21
- package/dist/hooks/useLoginMutation.js +0 -9
- package/dist/utils/useFeatures.d.ts +0 -1
- package/dist/utils/useFeatures.js +0 -7
- /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
|
|
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
|
-
|
|
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(
|
|
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(
|
|
22
|
-
...jest.requireActual(
|
|
21
|
+
jest.mock('react-router', ()=>({
|
|
22
|
+
...jest.requireActual('react-router'),
|
|
23
23
|
useNavigate: ()=>mockNavigate
|
|
24
24
|
}));
|
|
25
|
-
jest.mock(
|
|
26
|
-
...jest.requireActual(
|
|
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 =
|
|
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(
|
|
54
|
+
describe('BucketReplicationFormPage', ()=>{
|
|
55
55
|
const mockMutate = jest.fn();
|
|
56
56
|
const fillRequiredFields = async (options)=>{
|
|
57
|
-
const { roleArn =
|
|
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(
|
|
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:
|
|
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:
|
|
83
|
+
Name: 'test-bucket'
|
|
84
84
|
},
|
|
85
85
|
{
|
|
86
|
-
Name:
|
|
86
|
+
Name: 'destination-bucket'
|
|
87
87
|
},
|
|
88
88
|
{
|
|
89
|
-
Name:
|
|
89
|
+
Name: 'backup-bucket'
|
|
90
90
|
}
|
|
91
91
|
]
|
|
92
92
|
},
|
|
93
|
-
status:
|
|
93
|
+
status: 'success'
|
|
94
94
|
});
|
|
95
95
|
});
|
|
96
|
-
describe(
|
|
97
|
-
it(
|
|
96
|
+
describe('Page Rendering', ()=>{
|
|
97
|
+
it('renders create mode with correct title', ()=>{
|
|
98
98
|
renderBucketReplicationFormPage();
|
|
99
|
-
expect(screen.getByText(
|
|
99
|
+
expect(screen.getByText('Create Replication Rule')).toBeInTheDocument();
|
|
100
100
|
});
|
|
101
|
-
it(
|
|
101
|
+
it('renders edit mode with correct title when rule exists', async ()=>{
|
|
102
102
|
const existingRule = {
|
|
103
|
-
ID:
|
|
104
|
-
Status:
|
|
103
|
+
ID: 'test-rule',
|
|
104
|
+
Status: 'Enabled',
|
|
105
105
|
Priority: 1,
|
|
106
106
|
Destination: {
|
|
107
|
-
Bucket:
|
|
107
|
+
Bucket: 'arn:aws:s3:::destination-bucket'
|
|
108
108
|
}
|
|
109
109
|
};
|
|
110
110
|
mockUseGetBucketReplication.mockReturnValue({
|
|
111
111
|
data: {
|
|
112
112
|
ReplicationConfiguration: {
|
|
113
|
-
Role:
|
|
113
|
+
Role: 'arn:aws:iam::123456789012:role/replication-role',
|
|
114
114
|
Rules: [
|
|
115
115
|
existingRule
|
|
116
116
|
]
|
|
117
117
|
}
|
|
118
118
|
},
|
|
119
|
-
status:
|
|
119
|
+
status: 'success'
|
|
120
120
|
});
|
|
121
|
-
renderBucketReplicationFormPage(
|
|
121
|
+
renderBucketReplicationFormPage('test-bucket', 'test-rule');
|
|
122
122
|
await waitFor(()=>{
|
|
123
|
-
expect(screen.getByText(
|
|
123
|
+
expect(screen.getByText('Edit Replication Rule')).toBeInTheDocument();
|
|
124
124
|
});
|
|
125
125
|
});
|
|
126
|
-
it(
|
|
126
|
+
it('shows loading state when fetching replication data', ()=>{
|
|
127
127
|
mockUseGetBucketReplication.mockReturnValue({
|
|
128
128
|
data: void 0,
|
|
129
|
-
status:
|
|
129
|
+
status: 'pending'
|
|
130
130
|
});
|
|
131
131
|
renderBucketReplicationFormPage();
|
|
132
|
-
expect(screen.getByText(
|
|
132
|
+
expect(screen.getByText('Loading...')).toBeInTheDocument();
|
|
133
133
|
});
|
|
134
|
-
it(
|
|
134
|
+
it('shows error when rule not found in edit mode', ()=>{
|
|
135
135
|
mockUseGetBucketReplication.mockReturnValue({
|
|
136
136
|
data: {
|
|
137
137
|
ReplicationConfiguration: {
|
|
138
|
-
Role:
|
|
138
|
+
Role: 'arn:aws:iam::123456789012:role/replication-role',
|
|
139
139
|
Rules: []
|
|
140
140
|
}
|
|
141
141
|
},
|
|
142
|
-
status:
|
|
142
|
+
status: 'success'
|
|
143
143
|
});
|
|
144
|
-
renderBucketReplicationFormPage(
|
|
145
|
-
expect(screen.getByText(
|
|
144
|
+
renderBucketReplicationFormPage('test-bucket', 'non-existent');
|
|
145
|
+
expect(screen.getByText('Rule not found')).toBeInTheDocument();
|
|
146
146
|
});
|
|
147
|
-
it(
|
|
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(
|
|
153
|
-
expect(screen.getByText(
|
|
152
|
+
expect(screen.getByText('Replicate encrypted objects')).toBeInTheDocument();
|
|
153
|
+
expect(screen.getByText('Delete marker replication')).toBeInTheDocument();
|
|
154
154
|
});
|
|
155
155
|
});
|
|
156
156
|
});
|
|
157
|
-
describe(
|
|
158
|
-
it(
|
|
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(
|
|
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(
|
|
169
|
+
expect(roleInput.tagName).toBe('INPUT');
|
|
170
170
|
});
|
|
171
|
-
it(
|
|
171
|
+
it('displays existing Role as read-only text when rules exist', ()=>{
|
|
172
172
|
mockUseGetBucketReplication.mockReturnValue({
|
|
173
173
|
data: {
|
|
174
174
|
ReplicationConfiguration: {
|
|
175
|
-
Role:
|
|
175
|
+
Role: 'arn:aws:iam::123456789012:role/existing-role',
|
|
176
176
|
Rules: [
|
|
177
177
|
{
|
|
178
|
-
ID:
|
|
179
|
-
Status:
|
|
178
|
+
ID: 'existing-rule',
|
|
179
|
+
Status: 'Enabled',
|
|
180
180
|
Priority: 1,
|
|
181
181
|
Destination: {
|
|
182
|
-
Bucket:
|
|
182
|
+
Bucket: 'arn:aws:s3:::bucket'
|
|
183
183
|
}
|
|
184
184
|
}
|
|
185
185
|
]
|
|
186
186
|
}
|
|
187
187
|
},
|
|
188
|
-
status:
|
|
188
|
+
status: 'success'
|
|
189
189
|
});
|
|
190
190
|
renderBucketReplicationFormPage();
|
|
191
|
-
expect(screen.getByText(
|
|
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(
|
|
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(
|
|
199
|
+
expect(ruleIdInput.tagName).toBe('INPUT');
|
|
200
200
|
});
|
|
201
|
-
it(
|
|
201
|
+
it('displays Rule ID as read-only text in edit mode', async ()=>{
|
|
202
202
|
const existingRule = {
|
|
203
|
-
ID:
|
|
204
|
-
Status:
|
|
203
|
+
ID: 'readonly-rule-id',
|
|
204
|
+
Status: 'Enabled',
|
|
205
205
|
Priority: 1,
|
|
206
206
|
Destination: {
|
|
207
|
-
Bucket:
|
|
207
|
+
Bucket: 'arn:aws:s3:::bucket'
|
|
208
208
|
}
|
|
209
209
|
};
|
|
210
210
|
mockUseGetBucketReplication.mockReturnValue({
|
|
211
211
|
data: {
|
|
212
212
|
ReplicationConfiguration: {
|
|
213
|
-
Role:
|
|
213
|
+
Role: 'arn:aws:iam::123456789012:role/role',
|
|
214
214
|
Rules: [
|
|
215
215
|
existingRule
|
|
216
216
|
]
|
|
217
217
|
}
|
|
218
218
|
},
|
|
219
|
-
status:
|
|
219
|
+
status: 'success'
|
|
220
220
|
});
|
|
221
|
-
renderBucketReplicationFormPage(
|
|
221
|
+
renderBucketReplicationFormPage('test-bucket', 'readonly-rule-id');
|
|
222
222
|
await waitFor(()=>{
|
|
223
|
-
expect(screen.getByText(
|
|
223
|
+
expect(screen.getByText('readonly-rule-id')).toBeInTheDocument();
|
|
224
224
|
});
|
|
225
|
-
expect(screen.queryByRole(
|
|
225
|
+
expect(screen.queryByRole('textbox', {
|
|
226
226
|
name: /rule id/i
|
|
227
227
|
})).not.toBeInTheDocument();
|
|
228
228
|
});
|
|
229
|
-
it(
|
|
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(
|
|
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(
|
|
241
|
+
expect(priorityInput).toHaveAttribute('placeholder', 'Example: Auto-assigned: 0');
|
|
242
242
|
});
|
|
243
|
-
it(
|
|
243
|
+
it('renders all toggle fields', ()=>{
|
|
244
244
|
renderBucketReplicationFormPage();
|
|
245
|
-
expect(screen.getByText(
|
|
246
|
-
expect(screen.getByText(
|
|
247
|
-
expect(screen.getByText(
|
|
248
|
-
expect(screen.getByText(
|
|
249
|
-
expect(screen.getByText(
|
|
250
|
-
expect(screen.getByText(
|
|
251
|
-
});
|
|
252
|
-
it(
|
|
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(
|
|
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(
|
|
261
|
-
name:
|
|
260
|
+
expect(screen.getByRole('option', {
|
|
261
|
+
name: 'Same as source'
|
|
262
262
|
})).toBeInTheDocument();
|
|
263
|
-
expect(screen.getByRole(
|
|
264
|
-
name:
|
|
263
|
+
expect(screen.getByRole('option', {
|
|
264
|
+
name: 'Glacier'
|
|
265
265
|
})).toBeInTheDocument();
|
|
266
|
-
expect(screen.getByRole(
|
|
267
|
-
name:
|
|
266
|
+
expect(screen.getByRole('option', {
|
|
267
|
+
name: 'Glacier Deep Archive'
|
|
268
268
|
})).toBeInTheDocument();
|
|
269
|
-
expect(screen.getByRole(
|
|
270
|
-
name:
|
|
269
|
+
expect(screen.getByRole('option', {
|
|
270
|
+
name: 'Standard-IA'
|
|
271
271
|
})).toBeInTheDocument();
|
|
272
|
-
expect(screen.getByRole(
|
|
273
|
-
name:
|
|
272
|
+
expect(screen.getByRole('option', {
|
|
273
|
+
name: 'One Zone-IA'
|
|
274
274
|
})).toBeInTheDocument();
|
|
275
|
-
expect(screen.getByRole(
|
|
276
|
-
name:
|
|
275
|
+
expect(screen.getByRole('option', {
|
|
276
|
+
name: 'Intelligent-Tiering'
|
|
277
277
|
})).toBeInTheDocument();
|
|
278
|
-
expect(screen.getByRole(
|
|
279
|
-
name:
|
|
278
|
+
expect(screen.getByRole('option', {
|
|
279
|
+
name: 'Glacier Instant Retrieval'
|
|
280
280
|
})).toBeInTheDocument();
|
|
281
281
|
});
|
|
282
282
|
});
|
|
283
|
-
describe(
|
|
284
|
-
it(
|
|
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,
|
|
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(
|
|
293
|
+
it('validates Role ARN is optional when existing rules present', async ()=>{
|
|
294
294
|
mockUseGetBucketReplication.mockReturnValue({
|
|
295
295
|
data: {
|
|
296
296
|
ReplicationConfiguration: {
|
|
297
|
-
Role:
|
|
297
|
+
Role: 'arn:aws:iam::123456789012:role/existing-role',
|
|
298
298
|
Rules: [
|
|
299
299
|
{
|
|
300
|
-
ID:
|
|
301
|
-
Status:
|
|
300
|
+
ID: 'existing-rule',
|
|
301
|
+
Status: 'Enabled',
|
|
302
302
|
Priority: 1,
|
|
303
303
|
Destination: {
|
|
304
|
-
Bucket:
|
|
304
|
+
Bucket: 'arn:aws:s3:::bucket'
|
|
305
305
|
}
|
|
306
306
|
}
|
|
307
307
|
]
|
|
308
308
|
}
|
|
309
309
|
},
|
|
310
|
-
status:
|
|
310
|
+
status: 'success'
|
|
311
311
|
});
|
|
312
312
|
renderBucketReplicationFormPage();
|
|
313
|
-
await user_event.type(screen.getByLabelText(/rule id/i),
|
|
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(
|
|
317
|
-
name:
|
|
316
|
+
await user_event.click(screen.getByRole('option', {
|
|
317
|
+
name: 'destination-bucket'
|
|
318
318
|
}));
|
|
319
319
|
await waitFor(()=>{
|
|
320
|
-
const createButton = screen.getByRole(
|
|
320
|
+
const createButton = screen.getByRole('button', {
|
|
321
321
|
name: /create/i
|
|
322
322
|
});
|
|
323
323
|
expect(createButton).toBeEnabled();
|
|
324
324
|
});
|
|
325
325
|
});
|
|
326
|
-
it(
|
|
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,
|
|
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(
|
|
335
|
+
it('validates Rule ID uniqueness against existing rules', async ()=>{
|
|
336
336
|
mockUseGetBucketReplication.mockReturnValue({
|
|
337
337
|
data: {
|
|
338
338
|
ReplicationConfiguration: {
|
|
339
|
-
Role:
|
|
339
|
+
Role: 'arn:aws:iam::123456789012:role/role',
|
|
340
340
|
Rules: [
|
|
341
341
|
{
|
|
342
|
-
ID:
|
|
343
|
-
Status:
|
|
342
|
+
ID: 'existing-rule',
|
|
343
|
+
Status: 'Enabled',
|
|
344
344
|
Priority: 1,
|
|
345
345
|
Destination: {
|
|
346
|
-
Bucket:
|
|
346
|
+
Bucket: 'arn:aws:s3:::bucket'
|
|
347
347
|
}
|
|
348
348
|
}
|
|
349
349
|
]
|
|
350
350
|
}
|
|
351
351
|
},
|
|
352
|
-
status:
|
|
352
|
+
status: 'success'
|
|
353
353
|
});
|
|
354
354
|
renderBucketReplicationFormPage();
|
|
355
355
|
const ruleIdInput = screen.getByLabelText(/rule id/i);
|
|
356
|
-
await user_event.type(ruleIdInput,
|
|
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(
|
|
361
|
+
it('validates Priority must be >= 0', async ()=>{
|
|
362
362
|
renderBucketReplicationFormPage();
|
|
363
|
-
await user_event.type(screen.getByLabelText(/rule id/i),
|
|
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,
|
|
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(
|
|
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),
|
|
375
|
-
await user_event.type(screen.getByLabelText(/rule id/i),
|
|
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(
|
|
377
|
+
const createButton = screen.getByRole('button', {
|
|
378
378
|
name: /create/i
|
|
379
379
|
});
|
|
380
380
|
expect(createButton).toBeDisabled();
|
|
381
381
|
});
|
|
382
382
|
});
|
|
383
|
-
it(
|
|
383
|
+
it('shows Target Account ID field when sameAccount is false', async ()=>{
|
|
384
384
|
renderBucketReplicationFormPage();
|
|
385
|
-
const sameAccountToggle = findToggleByLabel(
|
|
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(
|
|
391
|
+
it('shows Replica KMS Key ID field when encryptReplicatedObjects is true', async ()=>{
|
|
392
392
|
renderBucketReplicationFormPage();
|
|
393
|
-
const encryptToggle = findToggleByLabel(
|
|
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(
|
|
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(
|
|
405
|
-
name:
|
|
404
|
+
expect(screen.getByRole('option', {
|
|
405
|
+
name: 'Prefix filter'
|
|
406
406
|
})).toBeInTheDocument();
|
|
407
407
|
});
|
|
408
|
-
await user_event.click(screen.getByRole(
|
|
409
|
-
name:
|
|
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(
|
|
417
|
-
it(
|
|
416
|
+
describe('Automatic Field Behaviors - Encryption', ()=>{
|
|
417
|
+
it('shows encryption requirement when includeEncryptedObjects is enabled', async ()=>{
|
|
418
418
|
renderBucketReplicationFormPage();
|
|
419
|
-
const includeEncryptedToggle = findToggleByLabel(
|
|
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(
|
|
428
|
-
it(
|
|
427
|
+
describe('Automatic Field Behaviors - RTC', ()=>{
|
|
428
|
+
it('shows metrics requirement when enforceRTC is enabled', async ()=>{
|
|
429
429
|
renderBucketReplicationFormPage();
|
|
430
|
-
const enforceRTCToggle = findToggleByLabel(
|
|
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(
|
|
438
|
-
it(
|
|
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(
|
|
443
|
-
name:
|
|
442
|
+
await user_event.click(screen.getByRole('option', {
|
|
443
|
+
name: 'Tags filter'
|
|
444
444
|
}));
|
|
445
445
|
await waitFor(()=>{
|
|
446
|
-
const deleteMarkerToggle = findToggleByLabel(
|
|
446
|
+
const deleteMarkerToggle = findToggleByLabel('Delete marker replication');
|
|
447
447
|
expect(deleteMarkerToggle).toBeDisabled();
|
|
448
448
|
});
|
|
449
449
|
});
|
|
450
|
-
it(
|
|
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(
|
|
455
|
-
name:
|
|
454
|
+
await user_event.click(screen.getByRole('option', {
|
|
455
|
+
name: 'Prefix and tags filter'
|
|
456
456
|
}));
|
|
457
457
|
await waitFor(()=>{
|
|
458
|
-
const deleteMarkerToggle = findToggleByLabel(
|
|
458
|
+
const deleteMarkerToggle = findToggleByLabel('Delete marker replication');
|
|
459
459
|
expect(deleteMarkerToggle).toBeDisabled();
|
|
460
460
|
});
|
|
461
461
|
});
|
|
462
|
-
it(
|
|
462
|
+
it('auto-disables deleteMarkerReplication when switching to tag-based filters', async ()=>{
|
|
463
463
|
renderBucketReplicationFormPage();
|
|
464
|
-
const deleteMarkerToggle = findToggleByLabel(
|
|
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(
|
|
472
|
-
name:
|
|
471
|
+
await user_event.click(screen.getByRole('option', {
|
|
472
|
+
name: 'Tags filter'
|
|
473
473
|
}));
|
|
474
474
|
await waitFor(()=>{
|
|
475
|
-
const updatedToggle = findToggleByLabel(
|
|
475
|
+
const updatedToggle = findToggleByLabel('Delete marker replication');
|
|
476
476
|
expect(updatedToggle).not.toBeChecked();
|
|
477
477
|
expect(updatedToggle).toBeDisabled();
|
|
478
478
|
});
|
|
479
479
|
});
|
|
480
|
-
it(
|
|
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(
|
|
485
|
-
name:
|
|
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(
|
|
493
|
-
it(
|
|
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(
|
|
496
|
-
const sameAccountToggle = findToggleByLabel(
|
|
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(
|
|
499
|
+
expect(screen.getByText('Switch Object ownership')).toBeInTheDocument();
|
|
500
500
|
});
|
|
501
501
|
});
|
|
502
|
-
it(
|
|
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(
|
|
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(
|
|
511
|
+
it('renders target bucket as Input for cross-account replication', async ()=>{
|
|
512
512
|
renderBucketReplicationFormPage();
|
|
513
|
-
const sameAccountToggle = findToggleByLabel(
|
|
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(
|
|
517
|
+
expect(targetBucketInput.tagName).toBe('INPUT');
|
|
518
518
|
});
|
|
519
519
|
});
|
|
520
|
-
it(
|
|
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(
|
|
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(
|
|
531
|
-
it(
|
|
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:
|
|
534
|
+
ruleId: 'minimal-rule'
|
|
535
535
|
});
|
|
536
536
|
mockSuccessSubmit(mockMutate);
|
|
537
|
-
await submitForm(
|
|
537
|
+
await submitForm('create');
|
|
538
538
|
await waitFor(()=>{
|
|
539
539
|
expect(mockMutate).toHaveBeenCalledWith(expect.objectContaining({
|
|
540
|
-
Bucket:
|
|
540
|
+
Bucket: 'test-bucket',
|
|
541
541
|
ReplicationConfiguration: {
|
|
542
|
-
Role:
|
|
542
|
+
Role: 'arn:aws:iam::123456789012:role/replication-role',
|
|
543
543
|
Rules: expect.arrayContaining([
|
|
544
544
|
expect.objectContaining({
|
|
545
|
-
ID:
|
|
546
|
-
Status:
|
|
545
|
+
ID: 'minimal-rule',
|
|
546
|
+
Status: 'Enabled',
|
|
547
547
|
Priority: 0,
|
|
548
548
|
Destination: expect.objectContaining({
|
|
549
|
-
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(
|
|
557
|
+
it('submits complete replication rule with all optional fields', async ()=>{
|
|
558
558
|
renderBucketReplicationFormPage();
|
|
559
|
-
await user_event.type(screen.getByLabelText(/role arn/i),
|
|
560
|
-
await user_event.type(screen.getByLabelText(/rule id/i),
|
|
561
|
-
await user_event.type(screen.getByLabelText(/rule priority/i),
|
|
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(
|
|
566
|
-
name:
|
|
565
|
+
expect(screen.getByRole('option', {
|
|
566
|
+
name: 'Prefix filter'
|
|
567
567
|
})).toBeInTheDocument();
|
|
568
568
|
});
|
|
569
|
-
await user_event.click(screen.getByRole(
|
|
570
|
-
name:
|
|
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),
|
|
576
|
-
const sameAccountToggle = findToggleByLabel(
|
|
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),
|
|
582
|
-
await user_event.type(screen.getByLabelText(/target bucket/i),
|
|
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(
|
|
587
|
-
name:
|
|
586
|
+
expect(screen.getByRole('option', {
|
|
587
|
+
name: 'Glacier'
|
|
588
588
|
})).toBeInTheDocument();
|
|
589
589
|
});
|
|
590
|
-
await user_event.click(screen.getByRole(
|
|
591
|
-
name:
|
|
590
|
+
await user_event.click(screen.getByRole('option', {
|
|
591
|
+
name: 'Glacier'
|
|
592
592
|
}));
|
|
593
|
-
const switchOwnershipToggle = findToggleByLabel(
|
|
593
|
+
const switchOwnershipToggle = findToggleByLabel('Switch Object ownership');
|
|
594
594
|
await user_event.click(switchOwnershipToggle);
|
|
595
|
-
const includeEncryptedToggle = findToggleByLabel(
|
|
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),
|
|
601
|
-
const enforceRTCToggle = findToggleByLabel(
|
|
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(
|
|
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(
|
|
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(
|
|
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:
|
|
621
|
-
Status:
|
|
620
|
+
ID: 'complete-rule',
|
|
621
|
+
Status: 'Enabled',
|
|
622
622
|
Priority: 5,
|
|
623
623
|
Filter: {
|
|
624
|
-
Prefix:
|
|
624
|
+
Prefix: 'logs/'
|
|
625
625
|
},
|
|
626
626
|
Destination: expect.objectContaining({
|
|
627
|
-
Bucket:
|
|
628
|
-
Account:
|
|
629
|
-
StorageClass:
|
|
627
|
+
Bucket: 'arn:aws:s3:::cross-account-bucket',
|
|
628
|
+
Account: '987654321098',
|
|
629
|
+
StorageClass: 'GLACIER',
|
|
630
630
|
AccessControlTranslation: {
|
|
631
|
-
Owner:
|
|
631
|
+
Owner: 'Destination'
|
|
632
632
|
},
|
|
633
633
|
EncryptionConfiguration: {
|
|
634
|
-
ReplicaKmsKeyID:
|
|
634
|
+
ReplicaKmsKeyID: 'arn:aws:kms:us-east-1:987654321098:key/12345678-1234-1234-1234-123456789012'
|
|
635
635
|
},
|
|
636
636
|
ReplicationTime: {
|
|
637
|
-
Status:
|
|
637
|
+
Status: 'Enabled',
|
|
638
638
|
Time: {
|
|
639
639
|
Minutes: 15
|
|
640
640
|
}
|
|
641
641
|
},
|
|
642
642
|
Metrics: expect.objectContaining({
|
|
643
|
-
Status:
|
|
643
|
+
Status: 'Enabled'
|
|
644
644
|
})
|
|
645
645
|
}),
|
|
646
646
|
SourceSelectionCriteria: expect.objectContaining({
|
|
647
647
|
SseKmsEncryptedObjects: {
|
|
648
|
-
Status:
|
|
648
|
+
Status: 'Enabled'
|
|
649
649
|
},
|
|
650
650
|
ReplicaModifications: {
|
|
651
|
-
Status:
|
|
651
|
+
Status: 'Enabled'
|
|
652
652
|
}
|
|
653
653
|
})
|
|
654
654
|
});
|
|
655
655
|
});
|
|
656
656
|
});
|
|
657
|
-
it(
|
|
657
|
+
it('appends new rule to existing rules array', async ()=>{
|
|
658
658
|
mockUseGetBucketReplication.mockReturnValue({
|
|
659
659
|
data: {
|
|
660
660
|
ReplicationConfiguration: {
|
|
661
|
-
Role:
|
|
661
|
+
Role: 'arn:aws:iam::123456789012:role/role',
|
|
662
662
|
Rules: [
|
|
663
663
|
{
|
|
664
|
-
ID:
|
|
665
|
-
Status:
|
|
664
|
+
ID: 'existing-rule',
|
|
665
|
+
Status: 'Enabled',
|
|
666
666
|
Priority: 1,
|
|
667
667
|
Destination: {
|
|
668
|
-
Bucket:
|
|
668
|
+
Bucket: 'arn:aws:s3:::existing-bucket'
|
|
669
669
|
}
|
|
670
670
|
}
|
|
671
671
|
]
|
|
672
672
|
}
|
|
673
673
|
},
|
|
674
|
-
status:
|
|
674
|
+
status: 'success'
|
|
675
675
|
});
|
|
676
676
|
renderBucketReplicationFormPage();
|
|
677
|
-
await user_event.type(screen.getByLabelText(/rule id/i),
|
|
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(
|
|
681
|
-
name:
|
|
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(
|
|
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(
|
|
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(
|
|
699
|
-
expect(call.ReplicationConfiguration.Rules[1].ID).toBe(
|
|
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(
|
|
702
|
+
it('navigates to bucket replication tab on success', async ()=>{
|
|
703
703
|
renderBucketReplicationFormPage();
|
|
704
704
|
await fillRequiredFields({
|
|
705
|
-
roleArn:
|
|
706
|
-
ruleId:
|
|
705
|
+
roleArn: 'arn:aws:iam::123456789012:role/role',
|
|
706
|
+
ruleId: 'success-rule'
|
|
707
707
|
});
|
|
708
708
|
mockSuccessSubmit(mockMutate);
|
|
709
|
-
await submitForm(
|
|
709
|
+
await submitForm('create');
|
|
710
710
|
await waitFor(()=>{
|
|
711
|
-
expect(mockNavigate).toHaveBeenCalledWith(
|
|
711
|
+
expect(mockNavigate).toHaveBeenCalledWith('/buckets/test-bucket?tab=replication');
|
|
712
712
|
});
|
|
713
713
|
});
|
|
714
|
-
it(
|
|
714
|
+
it('displays success toast after rule creation', async ()=>{
|
|
715
715
|
renderBucketReplicationFormPage();
|
|
716
716
|
await fillRequiredFields({
|
|
717
|
-
roleArn:
|
|
718
|
-
ruleId:
|
|
717
|
+
roleArn: 'arn:aws:iam::123456789012:role/role',
|
|
718
|
+
ruleId: 'toast-rule'
|
|
719
719
|
});
|
|
720
720
|
mockSuccessSubmit(mockMutate);
|
|
721
|
-
await submitForm(
|
|
721
|
+
await submitForm('create');
|
|
722
722
|
await waitFor(()=>{
|
|
723
723
|
expect(mockShowToast).toHaveBeenCalledWith({
|
|
724
724
|
open: true,
|
|
725
|
-
message:
|
|
726
|
-
status:
|
|
725
|
+
message: 'Replication rule created successfully',
|
|
726
|
+
status: 'success'
|
|
727
727
|
});
|
|
728
728
|
});
|
|
729
729
|
});
|
|
730
|
-
it(
|
|
730
|
+
it('displays error toast when creation fails', async ()=>{
|
|
731
731
|
renderBucketReplicationFormPage();
|
|
732
732
|
await fillRequiredFields({
|
|
733
|
-
roleArn:
|
|
734
|
-
ruleId:
|
|
733
|
+
roleArn: 'arn:aws:iam::123456789012:role/role',
|
|
734
|
+
ruleId: 'error-rule'
|
|
735
735
|
});
|
|
736
|
-
mockErrorSubmit(mockMutate,
|
|
737
|
-
await submitForm(
|
|
736
|
+
mockErrorSubmit(mockMutate, 'Access Denied');
|
|
737
|
+
await submitForm('create');
|
|
738
738
|
await waitFor(()=>{
|
|
739
739
|
expect(mockShowToast).toHaveBeenCalledWith({
|
|
740
740
|
open: true,
|
|
741
|
-
message:
|
|
742
|
-
status:
|
|
741
|
+
message: 'Access Denied',
|
|
742
|
+
status: 'error'
|
|
743
743
|
});
|
|
744
744
|
expect(mockNavigate).not.toHaveBeenCalled();
|
|
745
745
|
});
|
|
746
746
|
});
|
|
747
|
-
it(
|
|
747
|
+
it('disables submit button when form is pristine', ()=>{
|
|
748
748
|
renderBucketReplicationFormPage();
|
|
749
|
-
const createButton = screen.getByRole(
|
|
749
|
+
const createButton = screen.getByRole('button', {
|
|
750
750
|
name: /create/i
|
|
751
751
|
});
|
|
752
752
|
expect(createButton).toBeDisabled();
|
|
753
753
|
});
|
|
754
|
-
it(
|
|
754
|
+
it('disables submit button when form is invalid', async ()=>{
|
|
755
755
|
renderBucketReplicationFormPage();
|
|
756
|
-
await user_event.type(screen.getByLabelText(/rule id/i),
|
|
756
|
+
await user_event.type(screen.getByLabelText(/rule id/i), 'test');
|
|
757
757
|
await waitFor(()=>{
|
|
758
|
-
const createButton = screen.getByRole(
|
|
758
|
+
const createButton = screen.getByRole('button', {
|
|
759
759
|
name: /create/i
|
|
760
760
|
});
|
|
761
761
|
expect(createButton).toBeDisabled();
|
|
762
762
|
});
|
|
763
763
|
});
|
|
764
|
-
it(
|
|
764
|
+
it('enables submit button when form is valid and dirty', async ()=>{
|
|
765
765
|
renderBucketReplicationFormPage();
|
|
766
766
|
await fillRequiredFields({
|
|
767
|
-
roleArn:
|
|
768
|
-
ruleId:
|
|
767
|
+
roleArn: 'arn:aws:iam::123456789012:role/role',
|
|
768
|
+
ruleId: 'valid-rule'
|
|
769
769
|
});
|
|
770
770
|
await waitFor(()=>{
|
|
771
|
-
const createButton = screen.getByRole(
|
|
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(
|
|
779
|
-
it(
|
|
778
|
+
describe('Form Submission - Edit Mode', ()=>{
|
|
779
|
+
it('updates existing rule while preserving other rules', async ()=>{
|
|
780
780
|
const existingRules = [
|
|
781
781
|
{
|
|
782
|
-
ID:
|
|
783
|
-
Status:
|
|
782
|
+
ID: 'rule-1',
|
|
783
|
+
Status: 'Enabled',
|
|
784
784
|
Priority: 1,
|
|
785
785
|
Destination: {
|
|
786
|
-
Bucket:
|
|
786
|
+
Bucket: 'arn:aws:s3:::bucket-1'
|
|
787
787
|
}
|
|
788
788
|
},
|
|
789
789
|
{
|
|
790
|
-
ID:
|
|
791
|
-
Status:
|
|
790
|
+
ID: 'rule-2',
|
|
791
|
+
Status: 'Enabled',
|
|
792
792
|
Priority: 2,
|
|
793
793
|
Destination: {
|
|
794
|
-
Bucket:
|
|
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:
|
|
801
|
+
Role: 'arn:aws:iam::123456789012:role/role',
|
|
802
802
|
Rules: existingRules
|
|
803
803
|
}
|
|
804
804
|
},
|
|
805
|
-
status:
|
|
805
|
+
status: 'success'
|
|
806
806
|
});
|
|
807
|
-
renderBucketReplicationFormPage(
|
|
807
|
+
renderBucketReplicationFormPage('test-bucket', 'rule-1');
|
|
808
808
|
await waitFor(()=>{
|
|
809
|
-
expect(screen.getByText(
|
|
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(
|
|
814
|
-
name:
|
|
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(
|
|
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(
|
|
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(
|
|
832
|
-
expect(call.ReplicationConfiguration.Rules[0].Status).toBe(
|
|
833
|
-
expect(call.ReplicationConfiguration.Rules[1].ID).toBe(
|
|
834
|
-
expect(call.ReplicationConfiguration.Rules[1].Status).toBe(
|
|
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(
|
|
837
|
+
it('navigates to bucket replication tab on success', async ()=>{
|
|
838
838
|
const existingRule = {
|
|
839
|
-
ID:
|
|
840
|
-
Status:
|
|
839
|
+
ID: 'edit-rule',
|
|
840
|
+
Status: 'Enabled',
|
|
841
841
|
Priority: 1,
|
|
842
842
|
Destination: {
|
|
843
|
-
Bucket:
|
|
843
|
+
Bucket: 'arn:aws:s3:::bucket'
|
|
844
844
|
}
|
|
845
845
|
};
|
|
846
846
|
mockUseGetBucketReplication.mockReturnValue({
|
|
847
847
|
data: {
|
|
848
848
|
ReplicationConfiguration: {
|
|
849
|
-
Role:
|
|
849
|
+
Role: 'arn:aws:iam::123456789012:role/role',
|
|
850
850
|
Rules: [
|
|
851
851
|
existingRule
|
|
852
852
|
]
|
|
853
853
|
}
|
|
854
854
|
},
|
|
855
|
-
status:
|
|
855
|
+
status: 'success'
|
|
856
856
|
});
|
|
857
|
-
renderBucketReplicationFormPage(
|
|
857
|
+
renderBucketReplicationFormPage('test-bucket', 'edit-rule');
|
|
858
858
|
await waitFor(()=>{
|
|
859
|
-
expect(screen.getByText(
|
|
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(
|
|
864
|
-
name:
|
|
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(
|
|
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(
|
|
875
|
+
fireEvent.click(screen.getByRole('button', {
|
|
876
876
|
name: /save/i
|
|
877
877
|
}));
|
|
878
878
|
await waitFor(()=>{
|
|
879
|
-
expect(mockNavigate).toHaveBeenCalledWith(
|
|
879
|
+
expect(mockNavigate).toHaveBeenCalledWith('/buckets/test-bucket?tab=replication');
|
|
880
880
|
});
|
|
881
881
|
});
|
|
882
|
-
it(
|
|
882
|
+
it('displays update success toast in edit mode', async ()=>{
|
|
883
883
|
const existingRule = {
|
|
884
|
-
ID:
|
|
885
|
-
Status:
|
|
884
|
+
ID: 'toast-rule',
|
|
885
|
+
Status: 'Enabled',
|
|
886
886
|
Priority: 1,
|
|
887
887
|
Destination: {
|
|
888
|
-
Bucket:
|
|
888
|
+
Bucket: 'arn:aws:s3:::bucket'
|
|
889
889
|
}
|
|
890
890
|
};
|
|
891
891
|
mockUseGetBucketReplication.mockReturnValue({
|
|
892
892
|
data: {
|
|
893
893
|
ReplicationConfiguration: {
|
|
894
|
-
Role:
|
|
894
|
+
Role: 'arn:aws:iam::123456789012:role/role',
|
|
895
895
|
Rules: [
|
|
896
896
|
existingRule
|
|
897
897
|
]
|
|
898
898
|
}
|
|
899
899
|
},
|
|
900
|
-
status:
|
|
900
|
+
status: 'success'
|
|
901
901
|
});
|
|
902
|
-
renderBucketReplicationFormPage(
|
|
902
|
+
renderBucketReplicationFormPage('test-bucket', 'toast-rule');
|
|
903
903
|
await waitFor(()=>{
|
|
904
|
-
expect(screen.getByText(
|
|
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(
|
|
909
|
-
name:
|
|
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(
|
|
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(
|
|
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:
|
|
927
|
-
status:
|
|
926
|
+
message: 'Replication rule updated successfully',
|
|
927
|
+
status: 'success'
|
|
928
928
|
});
|
|
929
929
|
});
|
|
930
930
|
});
|
|
931
|
-
it(
|
|
931
|
+
it('displays error toast with error message on failure', async ()=>{
|
|
932
932
|
const existingRule = {
|
|
933
|
-
ID:
|
|
934
|
-
Status:
|
|
933
|
+
ID: 'error-rule',
|
|
934
|
+
Status: 'Enabled',
|
|
935
935
|
Priority: 1,
|
|
936
936
|
Destination: {
|
|
937
|
-
Bucket:
|
|
937
|
+
Bucket: 'arn:aws:s3:::bucket'
|
|
938
938
|
}
|
|
939
939
|
};
|
|
940
940
|
mockUseGetBucketReplication.mockReturnValue({
|
|
941
941
|
data: {
|
|
942
942
|
ReplicationConfiguration: {
|
|
943
|
-
Role:
|
|
943
|
+
Role: 'arn:aws:iam::123456789012:role/role',
|
|
944
944
|
Rules: [
|
|
945
945
|
existingRule
|
|
946
946
|
]
|
|
947
947
|
}
|
|
948
948
|
},
|
|
949
|
-
status:
|
|
949
|
+
status: 'success'
|
|
950
950
|
});
|
|
951
|
-
renderBucketReplicationFormPage(
|
|
951
|
+
renderBucketReplicationFormPage('test-bucket', 'error-rule');
|
|
952
952
|
await waitFor(()=>{
|
|
953
|
-
expect(screen.getByText(
|
|
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(
|
|
958
|
-
name:
|
|
957
|
+
await user_event.click(screen.getByRole('option', {
|
|
958
|
+
name: 'Disabled'
|
|
959
959
|
}));
|
|
960
|
-
const error = new 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(
|
|
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(
|
|
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:
|
|
977
|
-
status:
|
|
976
|
+
message: 'Network Error',
|
|
977
|
+
status: 'error'
|
|
978
978
|
});
|
|
979
979
|
expect(mockNavigate).not.toHaveBeenCalled();
|
|
980
980
|
});
|
|
981
981
|
});
|
|
982
982
|
});
|
|
983
|
-
describe(
|
|
984
|
-
it(
|
|
983
|
+
describe('Rule Data Loading in Edit Mode', ()=>{
|
|
984
|
+
it('loads rule with no filter correctly', async ()=>{
|
|
985
985
|
const rule = {
|
|
986
|
-
ID:
|
|
987
|
-
Status:
|
|
986
|
+
ID: 'no-filter-rule',
|
|
987
|
+
Status: 'Enabled',
|
|
988
988
|
Priority: 1,
|
|
989
989
|
Destination: {
|
|
990
|
-
Bucket:
|
|
990
|
+
Bucket: 'arn:aws:s3:::bucket'
|
|
991
991
|
}
|
|
992
992
|
};
|
|
993
993
|
mockUseGetBucketReplication.mockReturnValue({
|
|
994
994
|
data: {
|
|
995
995
|
ReplicationConfiguration: {
|
|
996
|
-
Role:
|
|
996
|
+
Role: 'arn:aws:iam::123456789012:role/role',
|
|
997
997
|
Rules: [
|
|
998
998
|
rule
|
|
999
999
|
]
|
|
1000
1000
|
}
|
|
1001
1001
|
},
|
|
1002
|
-
status:
|
|
1002
|
+
status: 'success'
|
|
1003
1003
|
});
|
|
1004
|
-
renderBucketReplicationFormPage(
|
|
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(
|
|
1011
|
+
it('loads rule with prefix filter correctly', async ()=>{
|
|
1012
1012
|
const rule = {
|
|
1013
|
-
ID:
|
|
1014
|
-
Status:
|
|
1013
|
+
ID: 'prefix-rule',
|
|
1014
|
+
Status: 'Enabled',
|
|
1015
1015
|
Priority: 1,
|
|
1016
1016
|
Filter: {
|
|
1017
|
-
Prefix:
|
|
1017
|
+
Prefix: 'documents/'
|
|
1018
1018
|
},
|
|
1019
1019
|
Destination: {
|
|
1020
|
-
Bucket:
|
|
1020
|
+
Bucket: 'arn:aws:s3:::bucket'
|
|
1021
1021
|
}
|
|
1022
1022
|
};
|
|
1023
1023
|
mockUseGetBucketReplication.mockReturnValue({
|
|
1024
1024
|
data: {
|
|
1025
1025
|
ReplicationConfiguration: {
|
|
1026
|
-
Role:
|
|
1026
|
+
Role: 'arn:aws:iam::123456789012:role/role',
|
|
1027
1027
|
Rules: [
|
|
1028
1028
|
rule
|
|
1029
1029
|
]
|
|
1030
1030
|
}
|
|
1031
1031
|
},
|
|
1032
|
-
status:
|
|
1032
|
+
status: 'success'
|
|
1033
1033
|
});
|
|
1034
|
-
renderBucketReplicationFormPage(
|
|
1034
|
+
renderBucketReplicationFormPage('test-bucket', 'prefix-rule');
|
|
1035
1035
|
await waitFor(()=>{
|
|
1036
1036
|
const prefixInput = screen.getByLabelText(/prefix/i);
|
|
1037
|
-
expect(prefixInput).toHaveValue(
|
|
1037
|
+
expect(prefixInput).toHaveValue('documents/');
|
|
1038
1038
|
});
|
|
1039
1039
|
});
|
|
1040
|
-
it(
|
|
1040
|
+
it('loads rule with single tag filter correctly', async ()=>{
|
|
1041
1041
|
const rule = {
|
|
1042
|
-
ID:
|
|
1043
|
-
Status:
|
|
1042
|
+
ID: 'tag-rule',
|
|
1043
|
+
Status: 'Enabled',
|
|
1044
1044
|
Priority: 1,
|
|
1045
1045
|
Filter: {
|
|
1046
1046
|
Tag: {
|
|
1047
|
-
Key:
|
|
1048
|
-
Value:
|
|
1047
|
+
Key: 'env',
|
|
1048
|
+
Value: 'prod'
|
|
1049
1049
|
}
|
|
1050
1050
|
},
|
|
1051
1051
|
Destination: {
|
|
1052
|
-
Bucket:
|
|
1052
|
+
Bucket: 'arn:aws:s3:::bucket'
|
|
1053
1053
|
}
|
|
1054
1054
|
};
|
|
1055
1055
|
mockUseGetBucketReplication.mockReturnValue({
|
|
1056
1056
|
data: {
|
|
1057
1057
|
ReplicationConfiguration: {
|
|
1058
|
-
Role:
|
|
1058
|
+
Role: 'arn:aws:iam::123456789012:role/role',
|
|
1059
1059
|
Rules: [
|
|
1060
1060
|
rule
|
|
1061
1061
|
]
|
|
1062
1062
|
}
|
|
1063
1063
|
},
|
|
1064
|
-
status:
|
|
1064
|
+
status: 'success'
|
|
1065
1065
|
});
|
|
1066
|
-
renderBucketReplicationFormPage(
|
|
1066
|
+
renderBucketReplicationFormPage('test-bucket', 'tag-rule');
|
|
1067
1067
|
await waitFor(()=>{
|
|
1068
|
-
expect(screen.getByDisplayValue(
|
|
1069
|
-
expect(screen.getByDisplayValue(
|
|
1068
|
+
expect(screen.getByDisplayValue('env')).toBeInTheDocument();
|
|
1069
|
+
expect(screen.getByDisplayValue('prod')).toBeInTheDocument();
|
|
1070
1070
|
});
|
|
1071
1071
|
});
|
|
1072
|
-
it(
|
|
1072
|
+
it('loads rule with prefix and tags filter correctly', async ()=>{
|
|
1073
1073
|
const rule = {
|
|
1074
|
-
ID:
|
|
1075
|
-
Status:
|
|
1074
|
+
ID: 'and-rule',
|
|
1075
|
+
Status: 'Enabled',
|
|
1076
1076
|
Priority: 1,
|
|
1077
1077
|
Filter: {
|
|
1078
1078
|
And: {
|
|
1079
|
-
Prefix:
|
|
1079
|
+
Prefix: 'logs/',
|
|
1080
1080
|
Tags: [
|
|
1081
1081
|
{
|
|
1082
|
-
Key:
|
|
1083
|
-
Value:
|
|
1082
|
+
Key: 'type',
|
|
1083
|
+
Value: 'audit'
|
|
1084
1084
|
},
|
|
1085
1085
|
{
|
|
1086
|
-
Key:
|
|
1087
|
-
Value:
|
|
1086
|
+
Key: 'year',
|
|
1087
|
+
Value: '2024'
|
|
1088
1088
|
}
|
|
1089
1089
|
]
|
|
1090
1090
|
}
|
|
1091
1091
|
},
|
|
1092
1092
|
Destination: {
|
|
1093
|
-
Bucket:
|
|
1093
|
+
Bucket: 'arn:aws:s3:::bucket'
|
|
1094
1094
|
}
|
|
1095
1095
|
};
|
|
1096
1096
|
mockUseGetBucketReplication.mockReturnValue({
|
|
1097
1097
|
data: {
|
|
1098
1098
|
ReplicationConfiguration: {
|
|
1099
|
-
Role:
|
|
1099
|
+
Role: 'arn:aws:iam::123456789012:role/role',
|
|
1100
1100
|
Rules: [
|
|
1101
1101
|
rule
|
|
1102
1102
|
]
|
|
1103
1103
|
}
|
|
1104
1104
|
},
|
|
1105
|
-
status:
|
|
1105
|
+
status: 'success'
|
|
1106
1106
|
});
|
|
1107
|
-
renderBucketReplicationFormPage(
|
|
1107
|
+
renderBucketReplicationFormPage('test-bucket', 'and-rule');
|
|
1108
1108
|
await waitFor(()=>{
|
|
1109
1109
|
const prefixInput = screen.getByLabelText(/prefix/i);
|
|
1110
|
-
expect(prefixInput).toHaveValue(
|
|
1111
|
-
expect(screen.getByDisplayValue(
|
|
1112
|
-
expect(screen.getByDisplayValue(
|
|
1113
|
-
expect(screen.getByDisplayValue(
|
|
1114
|
-
expect(screen.getByDisplayValue(
|
|
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(
|
|
1117
|
+
it('loads rule with includeEncryptedObjects enabled', async ()=>{
|
|
1118
1118
|
const rule = {
|
|
1119
|
-
ID:
|
|
1120
|
-
Status:
|
|
1119
|
+
ID: 'encrypted-rule',
|
|
1120
|
+
Status: 'Enabled',
|
|
1121
1121
|
Priority: 1,
|
|
1122
1122
|
Destination: {
|
|
1123
|
-
Bucket:
|
|
1123
|
+
Bucket: 'arn:aws:s3:::bucket'
|
|
1124
1124
|
},
|
|
1125
1125
|
SourceSelectionCriteria: {
|
|
1126
1126
|
SseKmsEncryptedObjects: {
|
|
1127
|
-
Status:
|
|
1127
|
+
Status: 'Enabled'
|
|
1128
1128
|
}
|
|
1129
1129
|
}
|
|
1130
1130
|
};
|
|
1131
1131
|
mockUseGetBucketReplication.mockReturnValue({
|
|
1132
1132
|
data: {
|
|
1133
1133
|
ReplicationConfiguration: {
|
|
1134
|
-
Role:
|
|
1134
|
+
Role: 'arn:aws:iam::123456789012:role/role',
|
|
1135
1135
|
Rules: [
|
|
1136
1136
|
rule
|
|
1137
1137
|
]
|
|
1138
1138
|
}
|
|
1139
1139
|
},
|
|
1140
|
-
status:
|
|
1140
|
+
status: 'success'
|
|
1141
1141
|
});
|
|
1142
|
-
renderBucketReplicationFormPage(
|
|
1142
|
+
renderBucketReplicationFormPage('test-bucket', 'encrypted-rule');
|
|
1143
1143
|
await waitFor(()=>{
|
|
1144
|
-
const includeEncryptedToggle = findToggleByLabel(
|
|
1144
|
+
const includeEncryptedToggle = findToggleByLabel('Replicate encrypted objects');
|
|
1145
1145
|
expect(includeEncryptedToggle).toBeChecked();
|
|
1146
1146
|
});
|
|
1147
1147
|
});
|
|
1148
|
-
it(
|
|
1148
|
+
it('loads rule with encryptReplicatedObjects and KMS Key ID', async ()=>{
|
|
1149
1149
|
const rule = {
|
|
1150
|
-
ID:
|
|
1151
|
-
Status:
|
|
1150
|
+
ID: 'encrypt-replica-rule',
|
|
1151
|
+
Status: 'Enabled',
|
|
1152
1152
|
Priority: 1,
|
|
1153
1153
|
Destination: {
|
|
1154
|
-
Bucket:
|
|
1154
|
+
Bucket: 'arn:aws:s3:::bucket',
|
|
1155
1155
|
EncryptionConfiguration: {
|
|
1156
|
-
ReplicaKmsKeyID:
|
|
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:
|
|
1163
|
+
Role: 'arn:aws:iam::123456789012:role/role',
|
|
1164
1164
|
Rules: [
|
|
1165
1165
|
rule
|
|
1166
1166
|
]
|
|
1167
1167
|
}
|
|
1168
1168
|
},
|
|
1169
|
-
status:
|
|
1169
|
+
status: 'success'
|
|
1170
1170
|
});
|
|
1171
|
-
renderBucketReplicationFormPage(
|
|
1171
|
+
renderBucketReplicationFormPage('test-bucket', 'encrypt-replica-rule');
|
|
1172
1172
|
await waitFor(()=>{
|
|
1173
|
-
const encryptReplicatedToggle = findToggleByLabel(
|
|
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(
|
|
1176
|
+
expect(kmsKeyInput).toHaveValue('arn:aws:kms:us-east-1:123456789012:key/12345');
|
|
1177
1177
|
});
|
|
1178
1178
|
});
|
|
1179
|
-
it(
|
|
1179
|
+
it('loads rule with enforceRTC and enableRTCNotification', async ()=>{
|
|
1180
1180
|
const rule = {
|
|
1181
|
-
ID:
|
|
1182
|
-
Status:
|
|
1181
|
+
ID: 'rtc-rule',
|
|
1182
|
+
Status: 'Enabled',
|
|
1183
1183
|
Priority: 1,
|
|
1184
1184
|
Destination: {
|
|
1185
|
-
Bucket:
|
|
1185
|
+
Bucket: 'arn:aws:s3:::bucket',
|
|
1186
1186
|
ReplicationTime: {
|
|
1187
|
-
Status:
|
|
1187
|
+
Status: 'Enabled',
|
|
1188
1188
|
Time: {
|
|
1189
1189
|
Minutes: 15
|
|
1190
1190
|
}
|
|
1191
1191
|
},
|
|
1192
1192
|
Metrics: {
|
|
1193
|
-
Status:
|
|
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:
|
|
1203
|
+
Role: 'arn:aws:iam::123456789012:role/role',
|
|
1204
1204
|
Rules: [
|
|
1205
1205
|
rule
|
|
1206
1206
|
]
|
|
1207
1207
|
}
|
|
1208
1208
|
},
|
|
1209
|
-
status:
|
|
1209
|
+
status: 'success'
|
|
1210
1210
|
});
|
|
1211
|
-
renderBucketReplicationFormPage(
|
|
1211
|
+
renderBucketReplicationFormPage('test-bucket', 'rtc-rule');
|
|
1212
1212
|
await waitFor(()=>{
|
|
1213
|
-
expect(screen.getByText(
|
|
1213
|
+
expect(screen.getByText('Replication Time Control (RTC)')).toBeInTheDocument();
|
|
1214
1214
|
});
|
|
1215
|
-
const enforceRTCToggle = findToggleByLabel(
|
|
1215
|
+
const enforceRTCToggle = findToggleByLabel('Replication Time Control (RTC)');
|
|
1216
1216
|
expect(enforceRTCToggle).toBeChecked();
|
|
1217
1217
|
});
|
|
1218
|
-
it(
|
|
1218
|
+
it('loads rule with replicaModifications enabled', async ()=>{
|
|
1219
1219
|
const rule = {
|
|
1220
|
-
ID:
|
|
1221
|
-
Status:
|
|
1220
|
+
ID: 'replica-mod-rule',
|
|
1221
|
+
Status: 'Enabled',
|
|
1222
1222
|
Priority: 1,
|
|
1223
1223
|
Destination: {
|
|
1224
|
-
Bucket:
|
|
1224
|
+
Bucket: 'arn:aws:s3:::bucket'
|
|
1225
1225
|
},
|
|
1226
1226
|
SourceSelectionCriteria: {
|
|
1227
1227
|
ReplicaModifications: {
|
|
1228
|
-
Status:
|
|
1228
|
+
Status: 'Enabled'
|
|
1229
1229
|
}
|
|
1230
1230
|
}
|
|
1231
1231
|
};
|
|
1232
1232
|
mockUseGetBucketReplication.mockReturnValue({
|
|
1233
1233
|
data: {
|
|
1234
1234
|
ReplicationConfiguration: {
|
|
1235
|
-
Role:
|
|
1235
|
+
Role: 'arn:aws:iam::123456789012:role/role',
|
|
1236
1236
|
Rules: [
|
|
1237
1237
|
rule
|
|
1238
1238
|
]
|
|
1239
1239
|
}
|
|
1240
1240
|
},
|
|
1241
|
-
status:
|
|
1241
|
+
status: 'success'
|
|
1242
1242
|
});
|
|
1243
|
-
renderBucketReplicationFormPage(
|
|
1243
|
+
renderBucketReplicationFormPage('test-bucket', 'replica-mod-rule');
|
|
1244
1244
|
await waitFor(()=>{
|
|
1245
|
-
const replicaModToggle = findToggleByLabel(
|
|
1245
|
+
const replicaModToggle = findToggleByLabel('Replica modification sync');
|
|
1246
1246
|
expect(replicaModToggle).toBeChecked();
|
|
1247
1247
|
});
|
|
1248
1248
|
});
|
|
1249
|
-
it(
|
|
1249
|
+
it('loads rule with deleteMarkerReplication enabled', async ()=>{
|
|
1250
1250
|
const rule = {
|
|
1251
|
-
ID:
|
|
1252
|
-
Status:
|
|
1251
|
+
ID: 'delete-marker-rule',
|
|
1252
|
+
Status: 'Enabled',
|
|
1253
1253
|
Priority: 1,
|
|
1254
1254
|
Destination: {
|
|
1255
|
-
Bucket:
|
|
1255
|
+
Bucket: 'arn:aws:s3:::bucket'
|
|
1256
1256
|
},
|
|
1257
1257
|
DeleteMarkerReplication: {
|
|
1258
|
-
Status:
|
|
1258
|
+
Status: 'Enabled'
|
|
1259
1259
|
}
|
|
1260
1260
|
};
|
|
1261
1261
|
mockUseGetBucketReplication.mockReturnValue({
|
|
1262
1262
|
data: {
|
|
1263
1263
|
ReplicationConfiguration: {
|
|
1264
|
-
Role:
|
|
1264
|
+
Role: 'arn:aws:iam::123456789012:role/role',
|
|
1265
1265
|
Rules: [
|
|
1266
1266
|
rule
|
|
1267
1267
|
]
|
|
1268
1268
|
}
|
|
1269
1269
|
},
|
|
1270
|
-
status:
|
|
1270
|
+
status: 'success'
|
|
1271
1271
|
});
|
|
1272
|
-
renderBucketReplicationFormPage(
|
|
1272
|
+
renderBucketReplicationFormPage('test-bucket', 'delete-marker-rule');
|
|
1273
1273
|
await waitFor(()=>{
|
|
1274
|
-
const deleteMarkerToggle = findToggleByLabel(
|
|
1274
|
+
const deleteMarkerToggle = findToggleByLabel('Delete marker replication');
|
|
1275
1275
|
expect(deleteMarkerToggle).toBeChecked();
|
|
1276
1276
|
});
|
|
1277
1277
|
});
|
|
1278
|
-
it(
|
|
1278
|
+
it('loads rule with cross-account destination', async ()=>{
|
|
1279
1279
|
const rule = {
|
|
1280
|
-
ID:
|
|
1281
|
-
Status:
|
|
1280
|
+
ID: 'cross-account-rule',
|
|
1281
|
+
Status: 'Enabled',
|
|
1282
1282
|
Priority: 1,
|
|
1283
1283
|
Destination: {
|
|
1284
|
-
Bucket:
|
|
1285
|
-
Account:
|
|
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:
|
|
1291
|
+
Role: 'arn:aws:iam::123456789012:role/role',
|
|
1292
1292
|
Rules: [
|
|
1293
1293
|
rule
|
|
1294
1294
|
]
|
|
1295
1295
|
}
|
|
1296
1296
|
},
|
|
1297
|
-
status:
|
|
1297
|
+
status: 'success'
|
|
1298
1298
|
});
|
|
1299
|
-
renderBucketReplicationFormPage(
|
|
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(
|
|
1304
|
+
expect(accountIdInput).toHaveValue('987654321098');
|
|
1305
1305
|
const targetBucketInput = screen.getByLabelText(/target bucket/i);
|
|
1306
|
-
expect(targetBucketInput).toHaveValue(
|
|
1306
|
+
expect(targetBucketInput).toHaveValue('cross-account-bucket');
|
|
1307
1307
|
});
|
|
1308
1308
|
});
|
|
1309
|
-
it(
|
|
1309
|
+
it('loads rule with switchObjectOwnership enabled', async ()=>{
|
|
1310
1310
|
const rule = {
|
|
1311
|
-
ID:
|
|
1312
|
-
Status:
|
|
1311
|
+
ID: 'ownership-rule',
|
|
1312
|
+
Status: 'Enabled',
|
|
1313
1313
|
Priority: 1,
|
|
1314
1314
|
Destination: {
|
|
1315
|
-
Bucket:
|
|
1316
|
-
Account:
|
|
1315
|
+
Bucket: 'arn:aws:s3:::bucket',
|
|
1316
|
+
Account: '987654321098',
|
|
1317
1317
|
AccessControlTranslation: {
|
|
1318
|
-
Owner:
|
|
1318
|
+
Owner: 'Destination'
|
|
1319
1319
|
}
|
|
1320
1320
|
}
|
|
1321
1321
|
};
|
|
1322
1322
|
mockUseGetBucketReplication.mockReturnValue({
|
|
1323
1323
|
data: {
|
|
1324
1324
|
ReplicationConfiguration: {
|
|
1325
|
-
Role:
|
|
1325
|
+
Role: 'arn:aws:iam::123456789012:role/role',
|
|
1326
1326
|
Rules: [
|
|
1327
1327
|
rule
|
|
1328
1328
|
]
|
|
1329
1329
|
}
|
|
1330
1330
|
},
|
|
1331
|
-
status:
|
|
1331
|
+
status: 'success'
|
|
1332
1332
|
});
|
|
1333
|
-
renderBucketReplicationFormPage(
|
|
1333
|
+
renderBucketReplicationFormPage('test-bucket', 'ownership-rule');
|
|
1334
1334
|
await waitFor(()=>{
|
|
1335
|
-
const ownershipToggle = findToggleByLabel(
|
|
1335
|
+
const ownershipToggle = findToggleByLabel('Switch Object ownership');
|
|
1336
1336
|
expect(ownershipToggle).toBeChecked();
|
|
1337
1337
|
});
|
|
1338
1338
|
});
|
|
1339
|
-
it(
|
|
1339
|
+
it('loads rule with custom StorageClass', async ()=>{
|
|
1340
1340
|
const rule = {
|
|
1341
|
-
ID:
|
|
1342
|
-
Status:
|
|
1341
|
+
ID: 'storage-class-rule',
|
|
1342
|
+
Status: 'Enabled',
|
|
1343
1343
|
Priority: 1,
|
|
1344
1344
|
Destination: {
|
|
1345
|
-
Bucket:
|
|
1346
|
-
StorageClass:
|
|
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:
|
|
1352
|
+
Role: 'arn:aws:iam::123456789012:role/role',
|
|
1353
1353
|
Rules: [
|
|
1354
1354
|
rule
|
|
1355
1355
|
]
|
|
1356
1356
|
}
|
|
1357
1357
|
},
|
|
1358
|
-
status:
|
|
1358
|
+
status: 'success'
|
|
1359
1359
|
});
|
|
1360
|
-
renderBucketReplicationFormPage(
|
|
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(
|
|
1366
|
+
it('loads rule with custom Priority', async ()=>{
|
|
1367
1367
|
const rule = {
|
|
1368
|
-
ID:
|
|
1369
|
-
Status:
|
|
1368
|
+
ID: 'priority-rule',
|
|
1369
|
+
Status: 'Enabled',
|
|
1370
1370
|
Priority: 99,
|
|
1371
1371
|
Destination: {
|
|
1372
|
-
Bucket:
|
|
1372
|
+
Bucket: 'arn:aws:s3:::bucket'
|
|
1373
1373
|
}
|
|
1374
1374
|
};
|
|
1375
1375
|
mockUseGetBucketReplication.mockReturnValue({
|
|
1376
1376
|
data: {
|
|
1377
1377
|
ReplicationConfiguration: {
|
|
1378
|
-
Role:
|
|
1378
|
+
Role: 'arn:aws:iam::123456789012:role/role',
|
|
1379
1379
|
Rules: [
|
|
1380
1380
|
rule
|
|
1381
1381
|
]
|
|
1382
1382
|
}
|
|
1383
1383
|
},
|
|
1384
|
-
status:
|
|
1384
|
+
status: 'success'
|
|
1385
1385
|
});
|
|
1386
|
-
renderBucketReplicationFormPage(
|
|
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(
|
|
1394
|
-
it(
|
|
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(
|
|
1397
|
+
expect(priorityInput).toHaveAttribute('placeholder', 'Example: Auto-assigned: 0');
|
|
1398
1398
|
});
|
|
1399
|
-
it(
|
|
1399
|
+
it('auto-assigns next available priority when rules exist', ()=>{
|
|
1400
1400
|
mockUseGetBucketReplication.mockReturnValue({
|
|
1401
1401
|
data: {
|
|
1402
1402
|
ReplicationConfiguration: {
|
|
1403
|
-
Role:
|
|
1403
|
+
Role: 'arn:aws:iam::123456789012:role/role',
|
|
1404
1404
|
Rules: [
|
|
1405
1405
|
{
|
|
1406
|
-
ID:
|
|
1407
|
-
Status:
|
|
1406
|
+
ID: 'rule-1',
|
|
1407
|
+
Status: 'Enabled',
|
|
1408
1408
|
Priority: 5,
|
|
1409
1409
|
Destination: {
|
|
1410
|
-
Bucket:
|
|
1410
|
+
Bucket: 'arn:aws:s3:::bucket'
|
|
1411
1411
|
}
|
|
1412
1412
|
},
|
|
1413
1413
|
{
|
|
1414
|
-
ID:
|
|
1415
|
-
Status:
|
|
1414
|
+
ID: 'rule-2',
|
|
1415
|
+
Status: 'Enabled',
|
|
1416
1416
|
Priority: 10,
|
|
1417
1417
|
Destination: {
|
|
1418
|
-
Bucket:
|
|
1418
|
+
Bucket: 'arn:aws:s3:::bucket'
|
|
1419
1419
|
}
|
|
1420
1420
|
}
|
|
1421
1421
|
]
|
|
1422
1422
|
}
|
|
1423
1423
|
},
|
|
1424
|
-
status:
|
|
1424
|
+
status: 'success'
|
|
1425
1425
|
});
|
|
1426
1426
|
renderBucketReplicationFormPage();
|
|
1427
1427
|
const priorityInput = screen.getByLabelText(/rule priority/i);
|
|
1428
|
-
expect(priorityInput).toHaveAttribute(
|
|
1428
|
+
expect(priorityInput).toHaveAttribute('placeholder', 'Example: Auto-assigned: 11');
|
|
1429
1429
|
});
|
|
1430
|
-
it(
|
|
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,
|
|
1433
|
+
await user_event.type(priorityInput, '25');
|
|
1434
1434
|
expect(priorityInput).toHaveValue(25);
|
|
1435
1435
|
});
|
|
1436
|
-
it(
|
|
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),
|
|
1439
|
-
await user_event.type(screen.getByLabelText(/rule id/i),
|
|
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(
|
|
1443
|
-
name:
|
|
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(
|
|
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(
|
|
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(
|
|
1464
|
-
it(
|
|
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(
|
|
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(
|
|
1474
|
-
name:
|
|
1473
|
+
expect(screen.getByRole('option', {
|
|
1474
|
+
name: 'destination-bucket'
|
|
1475
1475
|
})).toBeInTheDocument();
|
|
1476
|
-
expect(screen.getByRole(
|
|
1477
|
-
name:
|
|
1476
|
+
expect(screen.getByRole('option', {
|
|
1477
|
+
name: 'backup-bucket'
|
|
1478
1478
|
})).toBeInTheDocument();
|
|
1479
|
-
expect(screen.queryByRole(
|
|
1480
|
-
name:
|
|
1479
|
+
expect(screen.queryByRole('option', {
|
|
1480
|
+
name: 'test-bucket'
|
|
1481
1481
|
})).not.toBeInTheDocument();
|
|
1482
1482
|
});
|
|
1483
|
-
it(
|
|
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(
|
|
1489
|
+
expect(targetBucketInput.tagName).toBe('INPUT');
|
|
1490
1490
|
});
|
|
1491
1491
|
});
|
|
1492
|
-
it(
|
|
1492
|
+
it('loads available buckets using useBuckets hook', ()=>{
|
|
1493
1493
|
renderBucketReplicationFormPage();
|
|
1494
1494
|
expect(mockUseBuckets).toHaveBeenCalled();
|
|
1495
1495
|
});
|
|
1496
1496
|
});
|
|
1497
|
-
describe(
|
|
1498
|
-
it(
|
|
1497
|
+
describe('Navigation & Button States', ()=>{
|
|
1498
|
+
it('navigates back to bucket replication tab on Cancel', ()=>{
|
|
1499
1499
|
renderBucketReplicationFormPage();
|
|
1500
|
-
const cancelButton = screen.getByRole(
|
|
1500
|
+
const cancelButton = screen.getByRole('button', {
|
|
1501
1501
|
name: /cancel/i
|
|
1502
1502
|
});
|
|
1503
1503
|
fireEvent.click(cancelButton);
|
|
1504
|
-
expect(mockNavigate).toHaveBeenCalledWith(
|
|
1504
|
+
expect(mockNavigate).toHaveBeenCalledWith('/buckets/test-bucket?tab=replication');
|
|
1505
1505
|
});
|
|
1506
|
-
it(
|
|
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(
|
|
1512
|
+
const cancelButton = screen.getByRole('button', {
|
|
1513
1513
|
name: /cancel/i
|
|
1514
1514
|
});
|
|
1515
1515
|
expect(cancelButton).toBeDisabled();
|
|
1516
1516
|
});
|
|
1517
|
-
it(
|
|
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),
|
|
1524
|
-
await user_event.type(screen.getByLabelText(/rule id/i),
|
|
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(
|
|
1528
|
-
name:
|
|
1527
|
+
await user_event.click(screen.getByRole('option', {
|
|
1528
|
+
name: 'destination-bucket'
|
|
1529
1529
|
}));
|
|
1530
1530
|
await waitFor(()=>{
|
|
1531
|
-
const createButton = screen.getByRole(
|
|
1531
|
+
const createButton = screen.getByRole('button', {
|
|
1532
1532
|
name: /create/i
|
|
1533
1533
|
});
|
|
1534
1534
|
expect(createButton).toBeDisabled();
|
|
1535
1535
|
});
|
|
1536
1536
|
});
|
|
1537
|
-
it(
|
|
1537
|
+
it('shows Create button with no icon in create mode', ()=>{
|
|
1538
1538
|
renderBucketReplicationFormPage();
|
|
1539
|
-
const createButton = screen.getByRole(
|
|
1539
|
+
const createButton = screen.getByRole('button', {
|
|
1540
1540
|
name: /create/i
|
|
1541
1541
|
});
|
|
1542
1542
|
expect(createButton).toBeInTheDocument();
|
|
1543
|
-
expect(createButton.querySelector(
|
|
1543
|
+
expect(createButton.querySelector('svg')).not.toBeInTheDocument();
|
|
1544
1544
|
});
|
|
1545
|
-
it(
|
|
1545
|
+
it('shows Save button with Save icon in edit mode', async ()=>{
|
|
1546
1546
|
const existingRule = {
|
|
1547
|
-
ID:
|
|
1548
|
-
Status:
|
|
1547
|
+
ID: 'edit-rule',
|
|
1548
|
+
Status: 'Enabled',
|
|
1549
1549
|
Priority: 1,
|
|
1550
1550
|
Destination: {
|
|
1551
|
-
Bucket:
|
|
1551
|
+
Bucket: 'arn:aws:s3:::bucket'
|
|
1552
1552
|
}
|
|
1553
1553
|
};
|
|
1554
1554
|
mockUseGetBucketReplication.mockReturnValue({
|
|
1555
1555
|
data: {
|
|
1556
1556
|
ReplicationConfiguration: {
|
|
1557
|
-
Role:
|
|
1557
|
+
Role: 'arn:aws:iam::123456789012:role/role',
|
|
1558
1558
|
Rules: [
|
|
1559
1559
|
existingRule
|
|
1560
1560
|
]
|
|
1561
1561
|
}
|
|
1562
1562
|
},
|
|
1563
|
-
status:
|
|
1563
|
+
status: 'success'
|
|
1564
1564
|
});
|
|
1565
|
-
renderBucketReplicationFormPage(
|
|
1565
|
+
renderBucketReplicationFormPage('test-bucket', 'edit-rule');
|
|
1566
1566
|
await waitFor(()=>{
|
|
1567
|
-
const saveButton = screen.getByRole(
|
|
1567
|
+
const saveButton = screen.getByRole('button', {
|
|
1568
1568
|
name: /save/i
|
|
1569
1569
|
});
|
|
1570
1570
|
expect(saveButton).toBeInTheDocument();
|
|
1571
|
-
expect(saveButton.querySelector(
|
|
1571
|
+
expect(saveButton.querySelector('svg')).toBeInTheDocument();
|
|
1572
1572
|
});
|
|
1573
1573
|
});
|
|
1574
1574
|
});
|
|
1575
|
-
describe(
|
|
1576
|
-
it(
|
|
1575
|
+
describe('Edge Cases', ()=>{
|
|
1576
|
+
it('handles URL-encoded rule ID in edit mode', async ()=>{
|
|
1577
1577
|
const existingRule = {
|
|
1578
|
-
ID:
|
|
1579
|
-
Status:
|
|
1578
|
+
ID: 'rule with spaces',
|
|
1579
|
+
Status: 'Enabled',
|
|
1580
1580
|
Priority: 1,
|
|
1581
1581
|
Destination: {
|
|
1582
|
-
Bucket:
|
|
1582
|
+
Bucket: 'arn:aws:s3:::bucket'
|
|
1583
1583
|
}
|
|
1584
1584
|
};
|
|
1585
1585
|
mockUseGetBucketReplication.mockReturnValue({
|
|
1586
1586
|
data: {
|
|
1587
1587
|
ReplicationConfiguration: {
|
|
1588
|
-
Role:
|
|
1588
|
+
Role: 'arn:aws:iam::123456789012:role/role',
|
|
1589
1589
|
Rules: [
|
|
1590
1590
|
existingRule
|
|
1591
1591
|
]
|
|
1592
1592
|
}
|
|
1593
1593
|
},
|
|
1594
|
-
status:
|
|
1594
|
+
status: 'success'
|
|
1595
1595
|
});
|
|
1596
|
-
renderBucketReplicationFormPage(
|
|
1596
|
+
renderBucketReplicationFormPage('test-bucket', encodeURIComponent('rule with spaces'));
|
|
1597
1597
|
await waitFor(()=>{
|
|
1598
|
-
expect(screen.getByText(
|
|
1598
|
+
expect(screen.getByText('rule with spaces')).toBeInTheDocument();
|
|
1599
1599
|
});
|
|
1600
1600
|
});
|
|
1601
|
-
it(
|
|
1601
|
+
it('handles rules without Priority field', async ()=>{
|
|
1602
1602
|
const rule = {
|
|
1603
|
-
ID:
|
|
1604
|
-
Status:
|
|
1603
|
+
ID: 'no-priority',
|
|
1604
|
+
Status: 'Enabled',
|
|
1605
1605
|
Destination: {
|
|
1606
|
-
Bucket:
|
|
1606
|
+
Bucket: 'arn:aws:s3:::bucket'
|
|
1607
1607
|
}
|
|
1608
1608
|
};
|
|
1609
1609
|
mockUseGetBucketReplication.mockReturnValue({
|
|
1610
1610
|
data: {
|
|
1611
1611
|
ReplicationConfiguration: {
|
|
1612
|
-
Role:
|
|
1612
|
+
Role: 'arn:aws:iam::123456789012:role/role',
|
|
1613
1613
|
Rules: [
|
|
1614
1614
|
rule
|
|
1615
1615
|
]
|
|
1616
1616
|
}
|
|
1617
1617
|
},
|
|
1618
|
-
status:
|
|
1618
|
+
status: 'success'
|
|
1619
1619
|
});
|
|
1620
|
-
renderBucketReplicationFormPage(
|
|
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(
|
|
1626
|
+
it('handles rules without StorageClass', async ()=>{
|
|
1627
1627
|
const rule = {
|
|
1628
|
-
ID:
|
|
1629
|
-
Status:
|
|
1628
|
+
ID: 'no-storage-class',
|
|
1629
|
+
Status: 'Enabled',
|
|
1630
1630
|
Priority: 1,
|
|
1631
1631
|
Destination: {
|
|
1632
|
-
Bucket:
|
|
1632
|
+
Bucket: 'arn:aws:s3:::bucket'
|
|
1633
1633
|
}
|
|
1634
1634
|
};
|
|
1635
1635
|
mockUseGetBucketReplication.mockReturnValue({
|
|
1636
1636
|
data: {
|
|
1637
1637
|
ReplicationConfiguration: {
|
|
1638
|
-
Role:
|
|
1638
|
+
Role: 'arn:aws:iam::123456789012:role/role',
|
|
1639
1639
|
Rules: [
|
|
1640
1640
|
rule
|
|
1641
1641
|
]
|
|
1642
1642
|
}
|
|
1643
1643
|
},
|
|
1644
|
-
status:
|
|
1644
|
+
status: 'success'
|
|
1645
1645
|
});
|
|
1646
|
-
renderBucketReplicationFormPage(
|
|
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(
|
|
1654
|
-
it(
|
|
1653
|
+
describe('Boundary Conditions', ()=>{
|
|
1654
|
+
it('accepts Rule ID with reasonable length', async ()=>{
|
|
1655
1655
|
renderBucketReplicationFormPage();
|
|
1656
|
-
const longRuleId =
|
|
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(
|
|
1663
|
+
it('handles special characters in Rule ID', async ()=>{
|
|
1664
1664
|
renderBucketReplicationFormPage();
|
|
1665
|
-
const validSpecialChars =
|
|
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(
|
|
1673
|
+
it('validates Priority with boundary values', async ()=>{
|
|
1674
1674
|
renderBucketReplicationFormPage();
|
|
1675
|
-
await user_event.type(screen.getByLabelText(/rule id/i),
|
|
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,
|
|
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,
|
|
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(
|
|
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(
|
|
1696
|
-
name:
|
|
1695
|
+
expect(screen.getByRole('option', {
|
|
1696
|
+
name: 'Prefix filter'
|
|
1697
1697
|
})).toBeInTheDocument();
|
|
1698
1698
|
});
|
|
1699
|
-
await user_event.click(screen.getByRole(
|
|
1700
|
-
name:
|
|
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 =
|
|
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(
|
|
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,
|
|
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(
|
|
1718
|
+
expect(roleArnInput).toHaveValue('arn:aws:iam::123456789012:role/replication-role');
|
|
1719
1719
|
});
|
|
1720
1720
|
});
|
|
1721
|
-
it(
|
|
1721
|
+
it('accepts valid Target Account ID', async ()=>{
|
|
1722
1722
|
renderBucketReplicationFormPage();
|
|
1723
|
-
const sameAccountToggle = findToggleByLabel(
|
|
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,
|
|
1729
|
+
await user_event.type(accountIdInput, '123456789012');
|
|
1730
1730
|
await user_event.tab();
|
|
1731
1731
|
await waitFor(()=>{
|
|
1732
|
-
expect(accountIdInput).toHaveValue(
|
|
1732
|
+
expect(accountIdInput).toHaveValue('123456789012');
|
|
1733
1733
|
});
|
|
1734
1734
|
});
|
|
1735
|
-
it(
|
|
1735
|
+
it('handles empty form submission attempt', async ()=>{
|
|
1736
1736
|
renderBucketReplicationFormPage();
|
|
1737
|
-
const createButton = screen.getByRole(
|
|
1737
|
+
const createButton = screen.getByRole('button', {
|
|
1738
1738
|
name: /create/i
|
|
1739
1739
|
});
|
|
1740
1740
|
expect(createButton).toBeDisabled();
|
|
1741
1741
|
});
|
|
1742
|
-
it(
|
|
1742
|
+
it('validates KMS Key ID format', async ()=>{
|
|
1743
1743
|
renderBucketReplicationFormPage();
|
|
1744
|
-
const encryptToggle = findToggleByLabel(
|
|
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 =
|
|
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);
|