@scality/data-browser-library 1.0.0-preview.11

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (256) hide show
  1. package/dist/components/Editor.d.ts +12 -0
  2. package/dist/components/Editor.js +28 -0
  3. package/dist/components/__tests__/BucketCreate.test.d.ts +1 -0
  4. package/dist/components/__tests__/BucketCreate.test.js +408 -0
  5. package/dist/components/__tests__/BucketLifecycleFormPage.test.d.ts +1 -0
  6. package/dist/components/__tests__/BucketLifecycleFormPage.test.js +618 -0
  7. package/dist/components/__tests__/BucketLifecycleList.test.d.ts +1 -0
  8. package/dist/components/__tests__/BucketLifecycleList.test.js +325 -0
  9. package/dist/components/__tests__/BucketList.test.d.ts +1 -0
  10. package/dist/components/__tests__/BucketList.test.js +415 -0
  11. package/dist/components/__tests__/BucketNotificationCreatePage.test.d.ts +1 -0
  12. package/dist/components/__tests__/BucketNotificationCreatePage.test.js +316 -0
  13. package/dist/components/__tests__/BucketOverview.test.d.ts +1 -0
  14. package/dist/components/__tests__/BucketOverview.test.js +769 -0
  15. package/dist/components/__tests__/BucketPolicyPage.test.d.ts +1 -0
  16. package/dist/components/__tests__/BucketPolicyPage.test.js +268 -0
  17. package/dist/components/__tests__/BucketReplicationFormPage.test.d.ts +1 -0
  18. package/dist/components/__tests__/BucketReplicationFormPage.test.js +1757 -0
  19. package/dist/components/__tests__/BucketReplicationList.test.d.ts +1 -0
  20. package/dist/components/__tests__/BucketReplicationList.test.js +344 -0
  21. package/dist/components/__tests__/CreateFolderButton.test.d.ts +1 -0
  22. package/dist/components/__tests__/CreateFolderButton.test.js +147 -0
  23. package/dist/components/__tests__/DeleteBucketButton.test.d.ts +1 -0
  24. package/dist/components/__tests__/DeleteBucketButton.test.js +272 -0
  25. package/dist/components/__tests__/DeleteBucketConfigRuleButton.test.d.ts +1 -0
  26. package/dist/components/__tests__/DeleteBucketConfigRuleButton.test.js +196 -0
  27. package/dist/components/__tests__/DeleteObjectButton.test.d.ts +1 -0
  28. package/dist/components/__tests__/DeleteObjectButton.test.js +302 -0
  29. package/dist/components/__tests__/EmptyBucketButton.test.d.ts +1 -0
  30. package/dist/components/__tests__/EmptyBucketButton.test.js +302 -0
  31. package/dist/components/__tests__/MetadataSearch.test.d.ts +1 -0
  32. package/dist/components/__tests__/MetadataSearch.test.js +201 -0
  33. package/dist/components/__tests__/ObjectList.test.d.ts +1 -0
  34. package/dist/components/__tests__/ObjectList.test.js +283 -0
  35. package/dist/components/__tests__/UploadButton.test.d.ts +1 -0
  36. package/dist/components/__tests__/UploadButton.test.js +144 -0
  37. package/dist/components/buckets/BucketCreate.d.ts +49 -0
  38. package/dist/components/buckets/BucketCreate.js +237 -0
  39. package/dist/components/buckets/BucketDetails.d.ts +1 -0
  40. package/dist/components/buckets/BucketDetails.js +106 -0
  41. package/dist/components/buckets/BucketLifecycleFormPage.d.ts +15 -0
  42. package/dist/components/buckets/BucketLifecycleFormPage.js +1085 -0
  43. package/dist/components/buckets/BucketLifecycleList.d.ts +10 -0
  44. package/dist/components/buckets/BucketLifecycleList.js +270 -0
  45. package/dist/components/buckets/BucketList.d.ts +15 -0
  46. package/dist/components/buckets/BucketList.js +146 -0
  47. package/dist/components/buckets/BucketLocation.d.ts +3 -0
  48. package/dist/components/buckets/BucketLocation.js +16 -0
  49. package/dist/components/buckets/BucketOverview.d.ts +88 -0
  50. package/dist/components/buckets/BucketOverview.js +291 -0
  51. package/dist/components/buckets/BucketPage.d.ts +2 -0
  52. package/dist/components/buckets/BucketPage.js +47 -0
  53. package/dist/components/buckets/BucketPolicyButton.d.ts +7 -0
  54. package/dist/components/buckets/BucketPolicyButton.js +18 -0
  55. package/dist/components/buckets/BucketPolicyPage.d.ts +1 -0
  56. package/dist/components/buckets/BucketPolicyPage.js +236 -0
  57. package/dist/components/buckets/BucketReplicationFormPage.d.ts +1 -0
  58. package/dist/components/buckets/BucketReplicationFormPage.js +834 -0
  59. package/dist/components/buckets/BucketReplicationList.d.ts +11 -0
  60. package/dist/components/buckets/BucketReplicationList.js +189 -0
  61. package/dist/components/buckets/BucketVersioning.d.ts +4 -0
  62. package/dist/components/buckets/BucketVersioning.js +73 -0
  63. package/dist/components/buckets/DeleteBucketButton.d.ts +8 -0
  64. package/dist/components/buckets/DeleteBucketButton.js +78 -0
  65. package/dist/components/buckets/DeleteBucketConfigRuleButton.d.ts +18 -0
  66. package/dist/components/buckets/DeleteBucketConfigRuleButton.js +53 -0
  67. package/dist/components/buckets/EmptyBucketButton.d.ts +5 -0
  68. package/dist/components/buckets/EmptyBucketButton.js +232 -0
  69. package/dist/components/buckets/EmptyBucketSummary.d.ts +9 -0
  70. package/dist/components/buckets/EmptyBucketSummary.js +60 -0
  71. package/dist/components/buckets/EmptyBucketSummaryList.d.ts +13 -0
  72. package/dist/components/buckets/EmptyBucketSummaryList.js +140 -0
  73. package/dist/components/buckets/__tests__/BucketVersioning.test.d.ts +1 -0
  74. package/dist/components/buckets/__tests__/BucketVersioning.test.js +163 -0
  75. package/dist/components/buckets/notifications/BucketNotificationCreatePage.d.ts +1 -0
  76. package/dist/components/buckets/notifications/BucketNotificationCreatePage.js +234 -0
  77. package/dist/components/buckets/notifications/EventsSection.d.ts +1 -0
  78. package/dist/components/buckets/notifications/EventsSection.js +123 -0
  79. package/dist/components/buckets/notifications/events.d.ts +12 -0
  80. package/dist/components/buckets/notifications/events.js +27 -0
  81. package/dist/components/index.d.ts +21 -0
  82. package/dist/components/index.js +22 -0
  83. package/dist/components/layouts/ArrowNavigation.d.ts +4 -0
  84. package/dist/components/layouts/ArrowNavigation.js +16 -0
  85. package/dist/components/layouts/BrowserPageLayout.d.ts +12 -0
  86. package/dist/components/layouts/BrowserPageLayout.js +51 -0
  87. package/dist/components/objects/CreateFolderButton.d.ts +29 -0
  88. package/dist/components/objects/CreateFolderButton.js +118 -0
  89. package/dist/components/objects/DeleteObjectButton.d.ts +8 -0
  90. package/dist/components/objects/DeleteObjectButton.js +191 -0
  91. package/dist/components/objects/ObjectDetails/ObjectMetadata.d.ts +2 -0
  92. package/dist/components/objects/ObjectDetails/ObjectMetadata.js +356 -0
  93. package/dist/components/objects/ObjectDetails/ObjectSummary.d.ts +3 -0
  94. package/dist/components/objects/ObjectDetails/ObjectSummary.js +241 -0
  95. package/dist/components/objects/ObjectDetails/ObjectTags.d.ts +3 -0
  96. package/dist/components/objects/ObjectDetails/ObjectTags.js +272 -0
  97. package/dist/components/objects/ObjectDetails/index.d.ts +9 -0
  98. package/dist/components/objects/ObjectDetails/index.js +75 -0
  99. package/dist/components/objects/ObjectList.d.ts +40 -0
  100. package/dist/components/objects/ObjectList.js +453 -0
  101. package/dist/components/objects/ObjectLock/EditRetentionButton.d.ts +4 -0
  102. package/dist/components/objects/ObjectLock/EditRetentionButton.js +32 -0
  103. package/dist/components/objects/ObjectLock/ObjectLockRetentionSettings.d.ts +3 -0
  104. package/dist/components/objects/ObjectLock/ObjectLockRetentionSettings.js +211 -0
  105. package/dist/components/objects/ObjectLock/ObjectLockSettings.d.ts +9 -0
  106. package/dist/components/objects/ObjectLock/ObjectLockSettings.js +158 -0
  107. package/dist/components/objects/ObjectLock/ObjectLockSettingsUtils.d.ts +8 -0
  108. package/dist/components/objects/ObjectLock/ObjectLockSettingsUtils.js +39 -0
  109. package/dist/components/objects/ObjectLock/__tests__/EditRetentionButton.test.d.ts +1 -0
  110. package/dist/components/objects/ObjectLock/__tests__/EditRetentionButton.test.js +204 -0
  111. package/dist/components/objects/ObjectLock/__tests__/ObjectLockSettings.test.d.ts +1 -0
  112. package/dist/components/objects/ObjectLock/__tests__/ObjectLockSettings.test.js +374 -0
  113. package/dist/components/objects/ObjectPage.d.ts +1 -0
  114. package/dist/components/objects/ObjectPage.js +45 -0
  115. package/dist/components/objects/UploadButton.d.ts +34 -0
  116. package/dist/components/objects/UploadButton.js +229 -0
  117. package/dist/components/providers/DataBrowserProvider.d.ts +20 -0
  118. package/dist/components/providers/DataBrowserProvider.js +42 -0
  119. package/dist/components/search/MetadataSearch.d.ts +5 -0
  120. package/dist/components/search/MetadataSearch.js +162 -0
  121. package/dist/components/search/SearchHints.d.ts +8 -0
  122. package/dist/components/search/SearchHints.js +21 -0
  123. package/dist/components/ui/ArrayFieldActions.d.ts +36 -0
  124. package/dist/components/ui/ArrayFieldActions.js +43 -0
  125. package/dist/components/ui/ConfirmDeleteRuleModal.d.ts +16 -0
  126. package/dist/components/ui/ConfirmDeleteRuleModal.js +43 -0
  127. package/dist/components/ui/DeleteObjectModalContent.d.ts +5 -0
  128. package/dist/components/ui/DeleteObjectModalContent.js +71 -0
  129. package/dist/components/ui/FilterFormSection.d.ts +44 -0
  130. package/dist/components/ui/FilterFormSection.js +159 -0
  131. package/dist/components/ui/Search.elements.d.ts +17 -0
  132. package/dist/components/ui/Search.elements.js +59 -0
  133. package/dist/components/ui/Table.elements.d.ts +36 -0
  134. package/dist/components/ui/Table.elements.js +87 -0
  135. package/dist/config/factory.d.ts +63 -0
  136. package/dist/config/factory.js +74 -0
  137. package/dist/config/types.d.ts +46 -0
  138. package/dist/config/types.js +0 -0
  139. package/dist/hooks/__tests__/useISVBucketDetection.test.d.ts +1 -0
  140. package/dist/hooks/__tests__/useISVBucketDetection.test.js +188 -0
  141. package/dist/hooks/__tests__/useIsBucketEmpty.test.d.ts +1 -0
  142. package/dist/hooks/__tests__/useIsBucketEmpty.test.js +122 -0
  143. package/dist/hooks/bucketConfiguration.d.ts +168 -0
  144. package/dist/hooks/bucketConfiguration.js +67 -0
  145. package/dist/hooks/bucketOperations.d.ts +36 -0
  146. package/dist/hooks/bucketOperations.js +12 -0
  147. package/dist/hooks/factories/__tests__/useCreateS3FunctionMutationHook.test.d.ts +1 -0
  148. package/dist/hooks/factories/__tests__/useCreateS3FunctionMutationHook.test.js +276 -0
  149. package/dist/hooks/factories/__tests__/useCreateS3InfiniteQueryHook.test.d.ts +1 -0
  150. package/dist/hooks/factories/__tests__/useCreateS3InfiniteQueryHook.test.js +259 -0
  151. package/dist/hooks/factories/__tests__/useCreateS3LoginHook.test.d.ts +1 -0
  152. package/dist/hooks/factories/__tests__/useCreateS3LoginHook.test.js +166 -0
  153. package/dist/hooks/factories/__tests__/useCreateS3MutationHook.test.d.ts +1 -0
  154. package/dist/hooks/factories/__tests__/useCreateS3MutationHook.test.js +200 -0
  155. package/dist/hooks/factories/__tests__/useCreateS3QueryHook.test.d.ts +1 -0
  156. package/dist/hooks/factories/__tests__/useCreateS3QueryHook.test.js +179 -0
  157. package/dist/hooks/factories/index.d.ts +18 -0
  158. package/dist/hooks/factories/index.js +5 -0
  159. package/dist/hooks/factories/useCreateS3InfiniteQueryHook.d.ts +13 -0
  160. package/dist/hooks/factories/useCreateS3InfiniteQueryHook.js +76 -0
  161. package/dist/hooks/factories/useCreateS3LoginHook.d.ts +8 -0
  162. package/dist/hooks/factories/useCreateS3LoginHook.js +22 -0
  163. package/dist/hooks/factories/useCreateS3MutationHook.d.ts +5 -0
  164. package/dist/hooks/factories/useCreateS3MutationHook.js +50 -0
  165. package/dist/hooks/factories/useCreateS3QueryHook.d.ts +3 -0
  166. package/dist/hooks/factories/useCreateS3QueryHook.js +51 -0
  167. package/dist/hooks/index.d.ts +13 -0
  168. package/dist/hooks/index.js +13 -0
  169. package/dist/hooks/loginOperations.d.ts +21 -0
  170. package/dist/hooks/loginOperations.js +9 -0
  171. package/dist/hooks/objectOperations.d.ts +190 -0
  172. package/dist/hooks/objectOperations.js +67 -0
  173. package/dist/hooks/presignedOperations.d.ts +73 -0
  174. package/dist/hooks/presignedOperations.js +72 -0
  175. package/dist/hooks/useBatchObjectLegalHold.d.ts +11 -0
  176. package/dist/hooks/useBatchObjectLegalHold.js +45 -0
  177. package/dist/hooks/useDeleteBucketConfigRule.d.ts +26 -0
  178. package/dist/hooks/useDeleteBucketConfigRule.js +46 -0
  179. package/dist/hooks/useEmptyBucket.d.ts +27 -0
  180. package/dist/hooks/useEmptyBucket.js +116 -0
  181. package/dist/hooks/useISVBucketDetection.d.ts +15 -0
  182. package/dist/hooks/useISVBucketDetection.js +27 -0
  183. package/dist/hooks/useIsBucketEmpty.d.ts +7 -0
  184. package/dist/hooks/useIsBucketEmpty.js +36 -0
  185. package/dist/hooks/useLoginMutation.d.ts +21 -0
  186. package/dist/hooks/useLoginMutation.js +9 -0
  187. package/dist/hooks/useS3Client.d.ts +1 -0
  188. package/dist/hooks/useS3Client.js +13 -0
  189. package/dist/hooks/useTableRowSelection.d.ts +9 -0
  190. package/dist/hooks/useTableRowSelection.js +45 -0
  191. package/dist/index.d.ts +6 -0
  192. package/dist/index.js +6 -0
  193. package/dist/schemas/bucketPolicySchema.json +321 -0
  194. package/dist/test/msw/handlers/deleteBucket.d.ts +1 -0
  195. package/dist/test/msw/handlers/deleteBucket.js +14 -0
  196. package/dist/test/msw/handlers/getBucketAcl.d.ts +1 -0
  197. package/dist/test/msw/handlers/getBucketAcl.js +96 -0
  198. package/dist/test/msw/handlers/getBucketLocation.d.ts +1 -0
  199. package/dist/test/msw/handlers/getBucketLocation.js +23 -0
  200. package/dist/test/msw/handlers/getBucketPolicy.d.ts +11 -0
  201. package/dist/test/msw/handlers/getBucketPolicy.js +72 -0
  202. package/dist/test/msw/handlers/headObject.d.ts +1 -0
  203. package/dist/test/msw/handlers/headObject.js +17 -0
  204. package/dist/test/msw/handlers/listBuckets.d.ts +1 -0
  205. package/dist/test/msw/handlers/listBuckets.js +24 -0
  206. package/dist/test/msw/handlers/listObjectVersions.d.ts +1 -0
  207. package/dist/test/msw/handlers/listObjectVersions.js +83 -0
  208. package/dist/test/msw/handlers/listObjects.d.ts +1 -0
  209. package/dist/test/msw/handlers/listObjects.js +66 -0
  210. package/dist/test/msw/handlers/objectLegalHold.d.ts +1 -0
  211. package/dist/test/msw/handlers/objectLegalHold.js +24 -0
  212. package/dist/test/msw/handlers/objectRetention.d.ts +1 -0
  213. package/dist/test/msw/handlers/objectRetention.js +27 -0
  214. package/dist/test/msw/handlers/putBucketAcl.d.ts +1 -0
  215. package/dist/test/msw/handlers/putBucketAcl.js +18 -0
  216. package/dist/test/msw/handlers/putObject.d.ts +1 -0
  217. package/dist/test/msw/handlers/putObject.js +16 -0
  218. package/dist/test/msw/handlers.d.ts +4 -0
  219. package/dist/test/msw/handlers.js +109 -0
  220. package/dist/test/msw/index.d.ts +2 -0
  221. package/dist/test/msw/index.js +3 -0
  222. package/dist/test/msw/server.d.ts +4 -0
  223. package/dist/test/msw/server.js +20 -0
  224. package/dist/test/msw/utils.d.ts +2 -0
  225. package/dist/test/msw/utils.js +13 -0
  226. package/dist/test/setup.d.ts +1 -0
  227. package/dist/test/setup.js +90 -0
  228. package/dist/test/testUtils.d.ts +181 -0
  229. package/dist/test/testUtils.js +310 -0
  230. package/dist/test/utils/errorHandling.test.d.ts +1 -0
  231. package/dist/test/utils/errorHandling.test.js +423 -0
  232. package/dist/types/index.d.ts +51 -0
  233. package/dist/types/index.js +0 -0
  234. package/dist/utils/constants.d.ts +12 -0
  235. package/dist/utils/constants.js +9 -0
  236. package/dist/utils/deletion/index.d.ts +2 -0
  237. package/dist/utils/deletion/index.js +2 -0
  238. package/dist/utils/deletion/messages.d.ts +5 -0
  239. package/dist/utils/deletion/messages.js +29 -0
  240. package/dist/utils/deletion/types.d.ts +11 -0
  241. package/dist/utils/deletion/types.js +0 -0
  242. package/dist/utils/errorHandling.d.ts +63 -0
  243. package/dist/utils/errorHandling.js +84 -0
  244. package/dist/utils/hooks.d.ts +2 -0
  245. package/dist/utils/hooks.js +26 -0
  246. package/dist/utils/index.d.ts +4 -0
  247. package/dist/utils/index.js +4 -0
  248. package/dist/utils/proxyMiddleware.d.ts +18 -0
  249. package/dist/utils/proxyMiddleware.js +56 -0
  250. package/dist/utils/s3Client.d.ts +5 -0
  251. package/dist/utils/s3Client.js +36 -0
  252. package/dist/utils/s3RuleUtils.d.ts +53 -0
  253. package/dist/utils/s3RuleUtils.js +101 -0
  254. package/dist/utils/useFeatures.d.ts +1 -0
  255. package/dist/utils/useFeatures.js +7 -0
  256. package/package.json +84 -0
@@ -0,0 +1,118 @@
1
+ import { Fragment, jsx, jsxs } from "react/jsx-runtime";
2
+ import { useCallback, useState } from "react";
3
+ import { Icon, InfoMessage, Modal, Stack, Wrap, spacing } from "@scality/core-ui";
4
+ import { Button, Input } from "@scality/core-ui/dist/next";
5
+ import { useCreateFolder } from "../../hooks/index.js";
6
+ import styled_components from "styled-components";
7
+ const Description = styled_components.div`
8
+ margin-top: ${spacing.r16};
9
+ width: 20.5rem;
10
+ `;
11
+ const CreateFolderButton = ({ bucket, prefix = "", label = "Folder", variant = "secondary", onFolderSuccess, onFolderError })=>{
12
+ const [isModalOpen, setIsModalOpen] = useState(false);
13
+ const [folderName, setFolderName] = useState("");
14
+ const createFolderMutation = useCreateFolder();
15
+ const openModal = useCallback(()=>{
16
+ setIsModalOpen(true);
17
+ }, []);
18
+ const closeModal = useCallback(()=>{
19
+ setFolderName("");
20
+ setIsModalOpen(false);
21
+ }, []);
22
+ const handleSave = useCallback(()=>{
23
+ if (!folderName.trim()) return;
24
+ const folderKey = prefix ? `${prefix}/${folderName}/` : `${folderName}/`;
25
+ createFolderMutation.mutate({
26
+ Bucket: bucket,
27
+ Key: folderKey,
28
+ Body: ""
29
+ }, {
30
+ onSuccess: ()=>{
31
+ onFolderSuccess?.(folderName);
32
+ setFolderName("");
33
+ setIsModalOpen(false);
34
+ },
35
+ onError: (error)=>{
36
+ onFolderError?.(error, folderName);
37
+ }
38
+ });
39
+ }, [
40
+ folderName,
41
+ prefix,
42
+ bucket,
43
+ createFolderMutation,
44
+ onFolderSuccess,
45
+ onFolderError
46
+ ]);
47
+ const handleChange = useCallback((e)=>{
48
+ setFolderName(e.target.value);
49
+ }, []);
50
+ const hasInvalidFormat = folderName.startsWith("/");
51
+ const isEmpty = "" === folderName.trim();
52
+ const isLoading = createFolderMutation.isPending;
53
+ return /*#__PURE__*/ jsxs(Fragment, {
54
+ children: [
55
+ /*#__PURE__*/ jsx(Button, {
56
+ icon: /*#__PURE__*/ jsx(Icon, {
57
+ name: "Create-add"
58
+ }),
59
+ label: label,
60
+ variant: variant,
61
+ onClick: openModal
62
+ }),
63
+ /*#__PURE__*/ jsxs(Modal, {
64
+ close: closeModal,
65
+ footer: /*#__PURE__*/ jsxs(Wrap, {
66
+ children: [
67
+ /*#__PURE__*/ jsx("p", {}),
68
+ /*#__PURE__*/ jsxs(Stack, {
69
+ children: [
70
+ /*#__PURE__*/ jsx(Button, {
71
+ variant: "outline",
72
+ onClick: closeModal,
73
+ label: "Cancel",
74
+ disabled: isLoading
75
+ }),
76
+ /*#__PURE__*/ jsx(Button, {
77
+ disabled: isLoading || isEmpty || hasInvalidFormat,
78
+ variant: "secondary",
79
+ onClick: handleSave,
80
+ label: "Save",
81
+ tooltip: hasInvalidFormat ? {
82
+ overlay: "The folder name is not valid."
83
+ } : void 0
84
+ })
85
+ ]
86
+ })
87
+ ]
88
+ }),
89
+ isOpen: isModalOpen,
90
+ title: "Create a folder",
91
+ children: [
92
+ /*#__PURE__*/ jsx(Input, {
93
+ id: "folder-name",
94
+ value: folderName,
95
+ placeholder: "New folder",
96
+ error: hasInvalidFormat ? "Folder name cannot start with '/'" : void 0,
97
+ onChange: handleChange,
98
+ autoFocus: true
99
+ }),
100
+ /*#__PURE__*/ jsx(Description, {
101
+ children: /*#__PURE__*/ jsx(InfoMessage, {
102
+ title: "Creating a folder",
103
+ content: /*#__PURE__*/ jsxs(Fragment, {
104
+ children: [
105
+ "Ensure the folder name does not begin with a slash. ",
106
+ /*#__PURE__*/ jsx("br", {}),
107
+ 'When you create a folder, Data Browser creates an object with the above name appended by suffix "/" and that object is displayed as a folder in the Data Browser.'
108
+ ]
109
+ })
110
+ })
111
+ })
112
+ ]
113
+ })
114
+ ]
115
+ });
116
+ };
117
+ const objects_CreateFolderButton = CreateFolderButton;
118
+ export { CreateFolderButton, objects_CreateFolderButton as default };
@@ -0,0 +1,8 @@
1
+ import { TableItem } from "./ObjectList";
2
+ export type Objects = TableItem[];
3
+ interface DeleteObjectButtonProps {
4
+ objects: Objects;
5
+ bucketName: string;
6
+ }
7
+ export declare const DeleteObjectButton: React.FC<DeleteObjectButtonProps>;
8
+ export default DeleteObjectButton;
@@ -0,0 +1,191 @@
1
+ import { Fragment, jsx, jsxs } from "react/jsx-runtime";
2
+ import { useCallback, useEffect, useState } from "react";
3
+ import { Banner, Icon, Modal, PrettyBytes, Stack, Text, Wrap, spacing, useToast } from "@scality/core-ui";
4
+ import { Box, Button } from "@scality/core-ui/dist/next";
5
+ import { useDeleteObjects, useGetBucketVersioning } from "../../hooks/index.js";
6
+ import { getDeletionMessages } from "../../utils/deletion/index.js";
7
+ import { DeleteObjectModalContent } from "../ui/DeleteObjectModalContent.js";
8
+ const Title = ({ objects, isCurrentSelectionPermanentlyDeleted })=>{
9
+ const foldersSize = objects.filter((object)=>"folder" === object.type).length;
10
+ const objectsSize = objects.length - foldersSize;
11
+ if (0 === objects.length) return null;
12
+ return /*#__PURE__*/ jsxs(Text, {
13
+ children: [
14
+ "Do you want to",
15
+ " ",
16
+ /*#__PURE__*/ jsxs("strong", {
17
+ children: [
18
+ " ",
19
+ isCurrentSelectionPermanentlyDeleted ? "permanently" : "",
20
+ " "
21
+ ]
22
+ }),
23
+ " ",
24
+ "delete the selected",
25
+ " ",
26
+ objectsSize > 0 && /*#__PURE__*/ jsxs(Fragment, {
27
+ children: [
28
+ objectsSize,
29
+ " ",
30
+ isCurrentSelectionPermanentlyDeleted ? `object version${objectsSize > 1 ? "s" : ""}` : `object${objectsSize > 1 ? "s" : ""}`,
31
+ foldersSize > 0 && " and "
32
+ ]
33
+ }),
34
+ foldersSize > 0 && /*#__PURE__*/ jsxs(Fragment, {
35
+ children: [
36
+ foldersSize,
37
+ " folder",
38
+ foldersSize > 1 ? "s" : ""
39
+ ]
40
+ }),
41
+ "?"
42
+ ]
43
+ });
44
+ };
45
+ const DeleteObjectButton = ({ objects, bucketName })=>{
46
+ const [isModalOpen, setIsModalOpen] = useState(false);
47
+ const [selectedObjects, setSelectedObjects] = useState(objects);
48
+ const { data: versioningData } = useGetBucketVersioning({
49
+ Bucket: bucketName
50
+ });
51
+ const { mutate: deleteObjects, status: deleteObjectsStatus } = useDeleteObjects();
52
+ const { showToast } = useToast();
53
+ const isVersioningEnabled = versioningData?.Status === "Enabled";
54
+ const isCurrentSelectionPermanentlyDeleted = !isVersioningEnabled || selectedObjects.some((object)=>!!object.VersionId);
55
+ const { info: notificationText } = getDeletionMessages({
56
+ numberOfObjects: selectedObjects.length,
57
+ selectedObjectsAreSpecificVersions: isCurrentSelectionPermanentlyDeleted,
58
+ isBucketVersioned: isVersioningEnabled
59
+ });
60
+ const handleDeleteClick = useCallback(()=>{
61
+ setIsModalOpen(true);
62
+ }, []);
63
+ const cancel = useCallback(()=>{
64
+ setIsModalOpen(false);
65
+ }, []);
66
+ const deleteSelectedFiles = useCallback(()=>{
67
+ const objectsToDelete = selectedObjects.filter((object)=>"object" === object.type && object.Key).map((object)=>{
68
+ const deleteItem = {
69
+ Key: object.Key
70
+ };
71
+ if (object.VersionId && "string" == typeof object.VersionId) deleteItem.VersionId = object.VersionId;
72
+ return deleteItem;
73
+ });
74
+ const foldersToDelete = selectedObjects.filter((object)=>"folder" === object.type && object.Key).map((object)=>({
75
+ Key: object.Key
76
+ }));
77
+ deleteObjects({
78
+ Bucket: bucketName,
79
+ Delete: {
80
+ Objects: [
81
+ ...objectsToDelete,
82
+ ...foldersToDelete
83
+ ]
84
+ }
85
+ }, {
86
+ onSuccess: ()=>{
87
+ showToast({
88
+ open: true,
89
+ message: "Objects deleted successfully",
90
+ status: "success"
91
+ });
92
+ setIsModalOpen(false);
93
+ },
94
+ onError: (error)=>{
95
+ showToast({
96
+ open: true,
97
+ message: error instanceof Error ? error.message : "Objects deleted failed",
98
+ status: "error"
99
+ });
100
+ setIsModalOpen(false);
101
+ }
102
+ });
103
+ }, [
104
+ bucketName,
105
+ selectedObjects,
106
+ deleteObjects,
107
+ showToast
108
+ ]);
109
+ const totalSize = selectedObjects.reduce((acc, object)=>object.Size ? acc + object.Size : acc, 0);
110
+ useEffect(()=>{
111
+ setSelectedObjects(objects);
112
+ }, [
113
+ objects
114
+ ]);
115
+ return /*#__PURE__*/ jsxs(Fragment, {
116
+ children: [
117
+ /*#__PURE__*/ jsx(Button, {
118
+ id: "object-list-delete-button",
119
+ icon: /*#__PURE__*/ jsx(Icon, {
120
+ name: "Delete"
121
+ }),
122
+ variant: "danger",
123
+ disabled: 0 === selectedObjects.length,
124
+ onClick: handleDeleteClick,
125
+ label: "Delete"
126
+ }),
127
+ /*#__PURE__*/ jsxs(Modal, {
128
+ close: cancel,
129
+ footer: /*#__PURE__*/ jsxs(Wrap, {
130
+ children: [
131
+ /*#__PURE__*/ jsx("p", {}),
132
+ /*#__PURE__*/ jsxs(Stack, {
133
+ children: [
134
+ /*#__PURE__*/ jsx(Button, {
135
+ id: "object-delete-cancel-button",
136
+ variant: "outline",
137
+ onClick: cancel,
138
+ label: "Cancel"
139
+ }),
140
+ /*#__PURE__*/ jsx(Button, {
141
+ isLoading: "pending" === deleteObjectsStatus,
142
+ id: "object-delete-delete-button",
143
+ variant: "danger",
144
+ onClick: deleteSelectedFiles,
145
+ disabled: 0 === selectedObjects.length,
146
+ label: "Delete"
147
+ })
148
+ ]
149
+ })
150
+ ]
151
+ }),
152
+ isOpen: isModalOpen,
153
+ title: "Confirmation",
154
+ children: [
155
+ /*#__PURE__*/ jsx(Box, {
156
+ children: /*#__PURE__*/ jsx(Title, {
157
+ objects: selectedObjects,
158
+ isCurrentSelectionPermanentlyDeleted: isCurrentSelectionPermanentlyDeleted
159
+ })
160
+ }),
161
+ /*#__PURE__*/ jsx(DeleteObjectModalContent, {
162
+ objects: selectedObjects,
163
+ onRemove: (key)=>{
164
+ setSelectedObjects(selectedObjects.filter((object)=>object.Key !== key));
165
+ }
166
+ }),
167
+ /*#__PURE__*/ jsxs(Box, {
168
+ mb: spacing.r12,
169
+ children: [
170
+ "Total: ",
171
+ PrettyBytes({
172
+ bytes: totalSize
173
+ })
174
+ ]
175
+ }),
176
+ selectedObjects.length > 0 && notificationText && /*#__PURE__*/ jsx(Banner, {
177
+ variant: "base",
178
+ icon: /*#__PURE__*/ jsx(Icon, {
179
+ name: "Info-circle"
180
+ }),
181
+ children: /*#__PURE__*/ jsx("span", {
182
+ children: notificationText
183
+ })
184
+ })
185
+ ]
186
+ })
187
+ ]
188
+ });
189
+ };
190
+ const objects_DeleteObjectButton = DeleteObjectButton;
191
+ export { DeleteObjectButton, objects_DeleteObjectButton as default };
@@ -0,0 +1,2 @@
1
+ import { ObjectCommonProps } from ".";
2
+ export declare const ObjectMetadata: ({ bucketName, objectKey, versionId, }: ObjectCommonProps) => import("react/jsx-runtime").JSX.Element;
@@ -0,0 +1,356 @@
1
+ import { jsx, jsxs } from "react/jsx-runtime";
2
+ import { useEffect, useState } from "react";
3
+ import { Icon, Loader, Text, spacing, useToast } from "@scality/core-ui";
4
+ import { Box, Button, Input, Select } from "@scality/core-ui/dist/next";
5
+ import { convertSizeToRem } from "@scality/core-ui/dist/components/inputv2/inputv2";
6
+ import { Controller, useFieldArray, useForm } from "react-hook-form";
7
+ import { useQueryClient } from "@tanstack/react-query";
8
+ import { useCopyObject, useObjectMetadata } from "../../../hooks/index.js";
9
+ import { TableContainer } from "../../ui/Table.elements.js";
10
+ import { ArrayFieldActions } from "../../ui/ArrayFieldActions.js";
11
+ const METADATA_KEYS = [
12
+ {
13
+ key: "CacheControl",
14
+ label: "cache-control"
15
+ },
16
+ {
17
+ key: "ContentDisposition",
18
+ label: "content-disposition"
19
+ },
20
+ {
21
+ key: "ContentEncoding",
22
+ label: "content-encoding"
23
+ },
24
+ {
25
+ key: "ContentType",
26
+ label: "content-type"
27
+ },
28
+ {
29
+ key: "WebsiteRedirectLocation",
30
+ label: "website-redirect-location"
31
+ },
32
+ {
33
+ key: "x-amz-meta",
34
+ label: "x-amz-meta"
35
+ }
36
+ ];
37
+ const ObjectMetadata = ({ bucketName, objectKey, versionId })=>{
38
+ const getVersionParams = ()=>versionId ? {
39
+ VersionId: versionId
40
+ } : {};
41
+ const { data: metadata, status: metadataStatus } = useObjectMetadata({
42
+ Bucket: bucketName,
43
+ Key: objectKey,
44
+ ...getVersionParams()
45
+ });
46
+ const copyObjectMutation = useCopyObject();
47
+ const { showToast } = useToast();
48
+ const queryClient = useQueryClient();
49
+ const [isInitialized, setIsInitialized] = useState(false);
50
+ const { control, handleSubmit, setValue, watch, formState: { errors, isValid, isSubmitting } } = useForm({
51
+ mode: "onChange",
52
+ defaultValues: {
53
+ metadata: []
54
+ }
55
+ });
56
+ const { fields, append, remove } = useFieldArray({
57
+ control,
58
+ name: "metadata"
59
+ });
60
+ useEffect(()=>{
61
+ setIsInitialized(false);
62
+ copyObjectMutation.reset();
63
+ }, [
64
+ bucketName,
65
+ objectKey,
66
+ versionId
67
+ ]);
68
+ useEffect(()=>{
69
+ if (metadata && "success" === metadataStatus && !isInitialized) {
70
+ const rows = [];
71
+ const standardKeys = [
72
+ "CacheControl",
73
+ "ContentDisposition",
74
+ "ContentEncoding",
75
+ "ContentType",
76
+ "WebsiteRedirectLocation"
77
+ ];
78
+ standardKeys.forEach((keyType)=>{
79
+ const value = metadata[keyType];
80
+ if (value) rows.push({
81
+ keyType,
82
+ customKey: "",
83
+ value
84
+ });
85
+ });
86
+ const userMetadata = metadata.Metadata || {};
87
+ Object.entries(userMetadata).forEach(([key, value])=>{
88
+ rows.push({
89
+ keyType: "x-amz-meta",
90
+ customKey: key,
91
+ value: value
92
+ });
93
+ });
94
+ if (0 === rows.length) rows.push({
95
+ keyType: "",
96
+ customKey: "",
97
+ value: ""
98
+ });
99
+ setValue("metadata", rows);
100
+ setIsInitialized(true);
101
+ }
102
+ }, [
103
+ metadata,
104
+ metadataStatus,
105
+ setValue,
106
+ isInitialized
107
+ ]);
108
+ const getAvailableOptions = (currentIndex)=>{
109
+ const allMetadata = watch("metadata") || [];
110
+ const usedKeys = new Set(allMetadata.filter((row, idx)=>idx !== currentIndex && "x-amz-meta" !== row.keyType).map((row)=>row.keyType));
111
+ return METADATA_KEYS.filter((key)=>"x-amz-meta" === key.key || !usedKeys.has(key.key));
112
+ };
113
+ const onSubmit = async (data)=>{
114
+ if (!metadata) return;
115
+ try {
116
+ const validMetadata = data.metadata.filter((row)=>row.keyType && row.value.trim());
117
+ const encodedKey = encodeURIComponent(objectKey);
118
+ const copySource = versionId ? `${bucketName}/${encodedKey}?versionId=${versionId}` : `${bucketName}/${encodedKey}`;
119
+ const copyParams = {
120
+ Bucket: bucketName,
121
+ Key: objectKey,
122
+ CopySource: copySource,
123
+ MetadataDirective: "REPLACE",
124
+ Metadata: {}
125
+ };
126
+ validMetadata.forEach((row)=>{
127
+ if ("x-amz-meta" === row.keyType) copyParams.Metadata[row.customKey.trim()] = row.value.trim();
128
+ else if (row.keyType) copyParams[row.keyType] = row.value.trim();
129
+ });
130
+ if (!copyParams.ContentType && metadata.ContentType) copyParams.ContentType = metadata.ContentType;
131
+ await copyObjectMutation.mutateAsync(copyParams);
132
+ await queryClient.invalidateQueries({
133
+ queryKey: [
134
+ "HeadObject",
135
+ {
136
+ Bucket: bucketName,
137
+ Key: objectKey,
138
+ ...getVersionParams()
139
+ }
140
+ ]
141
+ });
142
+ showToast({
143
+ open: true,
144
+ message: "Metadata saved successfully",
145
+ status: "success"
146
+ });
147
+ } catch (error) {
148
+ const errorMessage = error instanceof Error ? error.message : "Failed to save metadata";
149
+ showToast({
150
+ open: true,
151
+ message: errorMessage,
152
+ status: "error"
153
+ });
154
+ }
155
+ };
156
+ if ("pending" === metadataStatus) return /*#__PURE__*/ jsx(TableContainer, {
157
+ children: /*#__PURE__*/ jsx(Box, {
158
+ display: "flex",
159
+ justifyContent: "center",
160
+ padding: spacing.r32,
161
+ children: /*#__PURE__*/ jsx(Loader, {})
162
+ })
163
+ });
164
+ if ("error" === metadataStatus) return /*#__PURE__*/ jsx(TableContainer, {
165
+ children: /*#__PURE__*/ jsx(Box, {
166
+ padding: spacing.r16,
167
+ children: /*#__PURE__*/ jsx(Text, {
168
+ color: "statusCritical",
169
+ children: "Error loading metadata"
170
+ })
171
+ })
172
+ });
173
+ return /*#__PURE__*/ jsx(TableContainer, {
174
+ children: /*#__PURE__*/ jsxs("form", {
175
+ onSubmit: handleSubmit(onSubmit),
176
+ children: [
177
+ /*#__PURE__*/ jsxs(Box, {
178
+ display: "flex",
179
+ justifyContent: "space-between",
180
+ alignItems: "center",
181
+ padding: spacing.r16,
182
+ borderBottom: "1px solid",
183
+ borderColor: "backgroundLevel3",
184
+ children: [
185
+ /*#__PURE__*/ jsx(Text, {
186
+ isEmphazed: true,
187
+ children: "Edit Metadata"
188
+ }),
189
+ /*#__PURE__*/ jsx(Button, {
190
+ variant: "primary",
191
+ label: "Save",
192
+ disabled: !isValid || isSubmitting,
193
+ type: "submit",
194
+ icon: /*#__PURE__*/ jsx(Icon, {
195
+ name: "Save"
196
+ })
197
+ })
198
+ ]
199
+ }),
200
+ /*#__PURE__*/ jsxs(Box, {
201
+ padding: spacing.r16,
202
+ display: "flex",
203
+ flexDirection: "column",
204
+ gap: spacing.r12,
205
+ children: [
206
+ /*#__PURE__*/ jsxs(Box, {
207
+ display: "flex",
208
+ gap: spacing.r8,
209
+ children: [
210
+ /*#__PURE__*/ jsx(Box, {
211
+ flex: "1",
212
+ children: /*#__PURE__*/ jsx(Text, {
213
+ color: "textSecondary",
214
+ isEmphazed: true,
215
+ children: "Key"
216
+ })
217
+ }),
218
+ /*#__PURE__*/ jsx(Box, {
219
+ flex: "1",
220
+ children: /*#__PURE__*/ jsx(Text, {
221
+ color: "textSecondary",
222
+ isEmphazed: true,
223
+ children: "Value"
224
+ })
225
+ }),
226
+ /*#__PURE__*/ jsx(Box, {
227
+ width: convertSizeToRem("1/3")
228
+ })
229
+ ]
230
+ }),
231
+ fields.map((field, index)=>/*#__PURE__*/ jsxs(Box, {
232
+ children: [
233
+ /*#__PURE__*/ jsxs(Box, {
234
+ display: "flex",
235
+ gap: spacing.r8,
236
+ alignItems: "center",
237
+ children: [
238
+ /*#__PURE__*/ jsx(Controller, {
239
+ control: control,
240
+ name: `metadata.${index}.keyType`,
241
+ rules: {
242
+ validate: (value, formValues)=>{
243
+ if (1 === formValues.metadata.length && !value && !formValues.metadata[0].value.trim()) return true;
244
+ if (!value) return "Please select a metadata key";
245
+ const allMetadata = formValues.metadata;
246
+ const usedKeys = allMetadata.map((row, idx)=>{
247
+ if (idx === index) return null;
248
+ if ("x-amz-meta" === row.keyType) return row.customKey ? `x-amz-meta-${row.customKey.trim()}` : null;
249
+ return row.keyType;
250
+ }).filter(Boolean);
251
+ const currentKey = "x-amz-meta" === value ? formValues.metadata[index].customKey ? `x-amz-meta-${formValues.metadata[index].customKey.trim()}` : null : value;
252
+ if (currentKey && usedKeys.includes(currentKey)) return `Duplicate key "${currentKey}"`;
253
+ return true;
254
+ }
255
+ },
256
+ render: ({ field: { onChange, value } })=>/*#__PURE__*/ jsx(Select, {
257
+ id: `key-${field.id}`,
258
+ value: value,
259
+ onChange: (newValue)=>{
260
+ onChange(newValue);
261
+ if ("x-amz-meta" !== newValue) setValue(`metadata.${index}.customKey`, "");
262
+ },
263
+ placeholder: "Select key",
264
+ children: getAvailableOptions(index).map((key)=>/*#__PURE__*/ jsx(Select.Option, {
265
+ value: key.key,
266
+ children: key.label
267
+ }, key.key))
268
+ })
269
+ }),
270
+ "x-amz-meta" === watch(`metadata.${index}.keyType`) && /*#__PURE__*/ jsx(Controller, {
271
+ control: control,
272
+ name: `metadata.${index}.customKey`,
273
+ rules: {
274
+ validate: (value, formValues)=>{
275
+ if ("x-amz-meta" === formValues.metadata[index].keyType) {
276
+ if (!value.trim()) return "Custom key required";
277
+ }
278
+ return true;
279
+ }
280
+ },
281
+ render: ({ field: { onChange, value } })=>/*#__PURE__*/ jsx(Input, {
282
+ id: `customKey-${field.id}`,
283
+ value: value,
284
+ onChange: onChange,
285
+ size: "1/3"
286
+ })
287
+ }),
288
+ /*#__PURE__*/ jsx(Text, {
289
+ color: "textSecondary",
290
+ children: ":"
291
+ }),
292
+ /*#__PURE__*/ jsx(Controller, {
293
+ control: control,
294
+ name: `metadata.${index}.value`,
295
+ rules: {
296
+ validate: (value, formValues)=>{
297
+ if (1 === formValues.metadata.length && !value.trim() && !formValues.metadata[0].keyType) return true;
298
+ if (!value.trim()) return "Value cannot be empty";
299
+ return true;
300
+ }
301
+ },
302
+ render: ({ field: { onChange, value } })=>/*#__PURE__*/ jsx(Input, {
303
+ id: `value-${field.id}`,
304
+ value: value,
305
+ onChange: onChange,
306
+ placeholder: "Value"
307
+ })
308
+ }),
309
+ /*#__PURE__*/ jsx(ArrayFieldActions, {
310
+ showAdd: index === fields.length - 1,
311
+ onRemove: ()=>{
312
+ remove(index);
313
+ if (1 === fields.length) append({
314
+ keyType: "",
315
+ customKey: "",
316
+ value: ""
317
+ });
318
+ },
319
+ onAdd: ()=>append({
320
+ keyType: "",
321
+ customKey: "",
322
+ value: ""
323
+ }),
324
+ canRemove: true,
325
+ canAdd: !!watch(`metadata.${index}.keyType`) && !!watch(`metadata.${index}.value`),
326
+ removeLabel: "Remove metadata",
327
+ addLabel: "Add metadata"
328
+ })
329
+ ]
330
+ }),
331
+ (errors.metadata?.[index]?.keyType || errors.metadata?.[index]?.customKey || errors.metadata?.[index]?.value) && /*#__PURE__*/ jsxs(Box, {
332
+ paddingTop: spacing.r4,
333
+ children: [
334
+ errors.metadata?.[index]?.keyType && /*#__PURE__*/ jsx(Text, {
335
+ color: "statusCritical",
336
+ children: errors.metadata[index]?.keyType?.message
337
+ }),
338
+ errors.metadata?.[index]?.customKey && /*#__PURE__*/ jsx(Text, {
339
+ color: "statusCritical",
340
+ children: errors.metadata[index]?.customKey?.message
341
+ }),
342
+ errors.metadata?.[index]?.value && /*#__PURE__*/ jsx(Text, {
343
+ color: "statusCritical",
344
+ children: errors.metadata[index]?.value?.message
345
+ })
346
+ ]
347
+ })
348
+ ]
349
+ }, field.id))
350
+ ]
351
+ })
352
+ ]
353
+ })
354
+ });
355
+ };
356
+ export { ObjectMetadata };