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

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 (176) hide show
  1. package/dist/components/Editor.d.ts +12 -0
  2. package/dist/components/Editor.js +28 -0
  3. package/dist/components/__tests__/BucketList.test.d.ts +1 -0
  4. package/dist/components/__tests__/BucketList.test.js +225 -0
  5. package/dist/components/__tests__/BucketOverview.test.d.ts +1 -0
  6. package/dist/components/__tests__/BucketOverview.test.js +479 -0
  7. package/dist/components/__tests__/BucketPolicyPage.test.d.ts +1 -0
  8. package/dist/components/__tests__/BucketPolicyPage.test.js +213 -0
  9. package/dist/components/__tests__/CreateFolderButton.test.d.ts +1 -0
  10. package/dist/components/__tests__/CreateFolderButton.test.js +147 -0
  11. package/dist/components/__tests__/DeleteBucketButton.test.d.ts +1 -0
  12. package/dist/components/__tests__/DeleteBucketButton.test.js +272 -0
  13. package/dist/components/__tests__/DeleteObjectButton.test.d.ts +1 -0
  14. package/dist/components/__tests__/DeleteObjectButton.test.js +302 -0
  15. package/dist/components/__tests__/MetadataSearch.test.d.ts +1 -0
  16. package/dist/components/__tests__/MetadataSearch.test.js +201 -0
  17. package/dist/components/__tests__/ObjectList.test.d.ts +1 -0
  18. package/dist/components/__tests__/ObjectList.test.js +283 -0
  19. package/dist/components/__tests__/UploadButton.test.d.ts +1 -0
  20. package/dist/components/__tests__/UploadButton.test.js +144 -0
  21. package/dist/components/buckets/BucketDetails.d.ts +1 -0
  22. package/dist/components/buckets/BucketDetails.js +51 -0
  23. package/dist/components/buckets/BucketList.d.ts +12 -0
  24. package/dist/components/buckets/BucketList.js +136 -0
  25. package/dist/components/buckets/BucketLocation.d.ts +3 -0
  26. package/dist/components/buckets/BucketLocation.js +16 -0
  27. package/dist/components/buckets/BucketOverview.d.ts +14 -0
  28. package/dist/components/buckets/BucketOverview.js +209 -0
  29. package/dist/components/buckets/BucketPage.d.ts +2 -0
  30. package/dist/components/buckets/BucketPage.js +47 -0
  31. package/dist/components/buckets/BucketPolicyButton.d.ts +7 -0
  32. package/dist/components/buckets/BucketPolicyButton.js +18 -0
  33. package/dist/components/buckets/BucketPolicyPage.d.ts +1 -0
  34. package/dist/components/buckets/BucketPolicyPage.js +205 -0
  35. package/dist/components/buckets/DeleteBucketButton.d.ts +8 -0
  36. package/dist/components/buckets/DeleteBucketButton.js +78 -0
  37. package/dist/components/index.d.ts +12 -0
  38. package/dist/components/index.js +13 -0
  39. package/dist/components/layouts/BrowserPageLayout.d.ts +9 -0
  40. package/dist/components/layouts/BrowserPageLayout.js +46 -0
  41. package/dist/components/objects/CreateFolderButton.d.ts +29 -0
  42. package/dist/components/objects/CreateFolderButton.js +118 -0
  43. package/dist/components/objects/DeleteObjectButton.d.ts +8 -0
  44. package/dist/components/objects/DeleteObjectButton.js +191 -0
  45. package/dist/components/objects/ObjectDetails/ObjectMetadata.d.ts +2 -0
  46. package/dist/components/objects/ObjectDetails/ObjectMetadata.js +323 -0
  47. package/dist/components/objects/ObjectDetails/ObjectSummary.d.ts +3 -0
  48. package/dist/components/objects/ObjectDetails/ObjectSummary.js +193 -0
  49. package/dist/components/objects/ObjectDetails/ObjectTags.d.ts +3 -0
  50. package/dist/components/objects/ObjectDetails/ObjectTags.js +300 -0
  51. package/dist/components/objects/ObjectDetails/index.d.ts +9 -0
  52. package/dist/components/objects/ObjectDetails/index.js +49 -0
  53. package/dist/components/objects/ObjectList.d.ts +40 -0
  54. package/dist/components/objects/ObjectList.js +407 -0
  55. package/dist/components/objects/ObjectPage.d.ts +1 -0
  56. package/dist/components/objects/ObjectPage.js +43 -0
  57. package/dist/components/objects/UploadButton.d.ts +34 -0
  58. package/dist/components/objects/UploadButton.js +229 -0
  59. package/dist/components/providers/DataBrowserProvider.d.ts +20 -0
  60. package/dist/components/providers/DataBrowserProvider.js +42 -0
  61. package/dist/components/search/MetadataSearch.d.ts +5 -0
  62. package/dist/components/search/MetadataSearch.js +162 -0
  63. package/dist/components/search/SearchHints.d.ts +8 -0
  64. package/dist/components/search/SearchHints.js +21 -0
  65. package/dist/components/ui/DeleteObjectModalContent.d.ts +5 -0
  66. package/dist/components/ui/DeleteObjectModalContent.js +71 -0
  67. package/dist/components/ui/Search.elements.d.ts +17 -0
  68. package/dist/components/ui/Search.elements.js +59 -0
  69. package/dist/components/ui/Table.elements.d.ts +36 -0
  70. package/dist/components/ui/Table.elements.js +87 -0
  71. package/dist/config/factory.d.ts +52 -0
  72. package/dist/config/factory.js +70 -0
  73. package/dist/config/types.d.ts +46 -0
  74. package/dist/config/types.js +0 -0
  75. package/dist/hooks/__tests__/useIsBucketEmpty.test.d.ts +1 -0
  76. package/dist/hooks/__tests__/useIsBucketEmpty.test.js +122 -0
  77. package/dist/hooks/bucketConfiguration.d.ts +147 -0
  78. package/dist/hooks/bucketConfiguration.js +59 -0
  79. package/dist/hooks/bucketOperations.d.ts +36 -0
  80. package/dist/hooks/bucketOperations.js +12 -0
  81. package/dist/hooks/factories/__tests__/useCreateS3FunctionMutationHook.test.d.ts +1 -0
  82. package/dist/hooks/factories/__tests__/useCreateS3FunctionMutationHook.test.js +276 -0
  83. package/dist/hooks/factories/__tests__/useCreateS3InfiniteQueryHook.test.d.ts +1 -0
  84. package/dist/hooks/factories/__tests__/useCreateS3InfiniteQueryHook.test.js +259 -0
  85. package/dist/hooks/factories/__tests__/useCreateS3LoginHook.test.d.ts +1 -0
  86. package/dist/hooks/factories/__tests__/useCreateS3LoginHook.test.js +166 -0
  87. package/dist/hooks/factories/__tests__/useCreateS3MutationHook.test.d.ts +1 -0
  88. package/dist/hooks/factories/__tests__/useCreateS3MutationHook.test.js +200 -0
  89. package/dist/hooks/factories/__tests__/useCreateS3QueryHook.test.d.ts +1 -0
  90. package/dist/hooks/factories/__tests__/useCreateS3QueryHook.test.js +136 -0
  91. package/dist/hooks/factories/index.d.ts +18 -0
  92. package/dist/hooks/factories/index.js +5 -0
  93. package/dist/hooks/factories/useCreateS3InfiniteQueryHook.d.ts +13 -0
  94. package/dist/hooks/factories/useCreateS3InfiniteQueryHook.js +76 -0
  95. package/dist/hooks/factories/useCreateS3LoginHook.d.ts +8 -0
  96. package/dist/hooks/factories/useCreateS3LoginHook.js +22 -0
  97. package/dist/hooks/factories/useCreateS3MutationHook.d.ts +5 -0
  98. package/dist/hooks/factories/useCreateS3MutationHook.js +50 -0
  99. package/dist/hooks/factories/useCreateS3QueryHook.d.ts +3 -0
  100. package/dist/hooks/factories/useCreateS3QueryHook.js +30 -0
  101. package/dist/hooks/index.d.ts +8 -0
  102. package/dist/hooks/index.js +8 -0
  103. package/dist/hooks/loginOperations.d.ts +21 -0
  104. package/dist/hooks/loginOperations.js +9 -0
  105. package/dist/hooks/objectOperations.d.ts +190 -0
  106. package/dist/hooks/objectOperations.js +66 -0
  107. package/dist/hooks/presignedOperations.d.ts +73 -0
  108. package/dist/hooks/presignedOperations.js +72 -0
  109. package/dist/hooks/useIsBucketEmpty.d.ts +7 -0
  110. package/dist/hooks/useIsBucketEmpty.js +36 -0
  111. package/dist/hooks/useLoginMutation.d.ts +21 -0
  112. package/dist/hooks/useLoginMutation.js +9 -0
  113. package/dist/hooks/useS3Client.d.ts +1 -0
  114. package/dist/hooks/useS3Client.js +13 -0
  115. package/dist/index.d.ts +6 -0
  116. package/dist/index.js +6 -0
  117. package/dist/schemas/bucketPolicySchema.json +321 -0
  118. package/dist/test/msw/handlers/deleteBucket.d.ts +1 -0
  119. package/dist/test/msw/handlers/deleteBucket.js +14 -0
  120. package/dist/test/msw/handlers/getBucketAcl.d.ts +1 -0
  121. package/dist/test/msw/handlers/getBucketAcl.js +96 -0
  122. package/dist/test/msw/handlers/getBucketLocation.d.ts +1 -0
  123. package/dist/test/msw/handlers/getBucketLocation.js +23 -0
  124. package/dist/test/msw/handlers/getBucketPolicy.d.ts +11 -0
  125. package/dist/test/msw/handlers/getBucketPolicy.js +72 -0
  126. package/dist/test/msw/handlers/headObject.d.ts +1 -0
  127. package/dist/test/msw/handlers/headObject.js +17 -0
  128. package/dist/test/msw/handlers/listBuckets.d.ts +1 -0
  129. package/dist/test/msw/handlers/listBuckets.js +24 -0
  130. package/dist/test/msw/handlers/listObjectVersions.d.ts +1 -0
  131. package/dist/test/msw/handlers/listObjectVersions.js +83 -0
  132. package/dist/test/msw/handlers/listObjects.d.ts +1 -0
  133. package/dist/test/msw/handlers/listObjects.js +66 -0
  134. package/dist/test/msw/handlers/objectLegalHold.d.ts +1 -0
  135. package/dist/test/msw/handlers/objectLegalHold.js +24 -0
  136. package/dist/test/msw/handlers/objectRetention.d.ts +1 -0
  137. package/dist/test/msw/handlers/objectRetention.js +27 -0
  138. package/dist/test/msw/handlers/putBucketAcl.d.ts +1 -0
  139. package/dist/test/msw/handlers/putBucketAcl.js +18 -0
  140. package/dist/test/msw/handlers/putObject.d.ts +1 -0
  141. package/dist/test/msw/handlers/putObject.js +16 -0
  142. package/dist/test/msw/handlers.d.ts +4 -0
  143. package/dist/test/msw/handlers.js +109 -0
  144. package/dist/test/msw/index.d.ts +2 -0
  145. package/dist/test/msw/index.js +3 -0
  146. package/dist/test/msw/server.d.ts +4 -0
  147. package/dist/test/msw/server.js +20 -0
  148. package/dist/test/msw/utils.d.ts +2 -0
  149. package/dist/test/msw/utils.js +13 -0
  150. package/dist/test/setup.d.ts +1 -0
  151. package/dist/test/setup.js +82 -0
  152. package/dist/test/testUtils.d.ts +82 -0
  153. package/dist/test/testUtils.js +236 -0
  154. package/dist/test/utils/errorHandling.test.d.ts +1 -0
  155. package/dist/test/utils/errorHandling.test.js +385 -0
  156. package/dist/types/index.d.ts +48 -0
  157. package/dist/types/index.js +0 -0
  158. package/dist/utils/deletion/index.d.ts +2 -0
  159. package/dist/utils/deletion/index.js +2 -0
  160. package/dist/utils/deletion/messages.d.ts +5 -0
  161. package/dist/utils/deletion/messages.js +29 -0
  162. package/dist/utils/deletion/types.d.ts +11 -0
  163. package/dist/utils/deletion/types.js +0 -0
  164. package/dist/utils/errorHandling.d.ts +54 -0
  165. package/dist/utils/errorHandling.js +79 -0
  166. package/dist/utils/hooks.d.ts +2 -0
  167. package/dist/utils/hooks.js +26 -0
  168. package/dist/utils/index.d.ts +2 -0
  169. package/dist/utils/index.js +2 -0
  170. package/dist/utils/proxyMiddleware.d.ts +18 -0
  171. package/dist/utils/proxyMiddleware.js +56 -0
  172. package/dist/utils/s3Client.d.ts +5 -0
  173. package/dist/utils/s3Client.js +35 -0
  174. package/dist/utils/useFeatures.d.ts +1 -0
  175. package/dist/utils/useFeatures.js +7 -0
  176. package/package.json +79 -0
@@ -0,0 +1,283 @@
1
+ import { jsx } from "react/jsx-runtime";
2
+ import { fireEvent, render, renderHook, screen, waitFor } from "@testing-library/react";
3
+ import { MemoryRouter } from "react-router";
4
+ import { createTestWrapper, mockOffsetSize, setupMswServer } from "../../test/testUtils.js";
5
+ import { useListObjectVersions, useListObjects } from "../../hooks/index.js";
6
+ import { ObjectList } from "../objects/ObjectList.js";
7
+ import * as __WEBPACK_EXTERNAL_MODULE__utils_useFeatures_js_1facdd0d__ from "../../utils/useFeatures.js";
8
+ setupMswServer();
9
+ const renderObjectList = (props = {})=>{
10
+ const Wrapper = createTestWrapper();
11
+ return render(/*#__PURE__*/ jsx(MemoryRouter, {
12
+ children: /*#__PURE__*/ jsx(Wrapper, {
13
+ children: /*#__PURE__*/ jsx(ObjectList, {
14
+ bucketName: "test-bucket",
15
+ prefix: "",
16
+ onObjectSelect: jest.fn(),
17
+ onPrefixChange: jest.fn(),
18
+ ...props
19
+ })
20
+ })
21
+ }));
22
+ };
23
+ describe("ObjectList", ()=>{
24
+ beforeEach(()=>{
25
+ jest.clearAllMocks();
26
+ mockOffsetSize(800, 600);
27
+ jest.spyOn(__WEBPACK_EXTERNAL_MODULE__utils_useFeatures_js_1facdd0d__, "useFeatures").mockReturnValue(false);
28
+ });
29
+ afterEach(()=>{
30
+ jest.restoreAllMocks();
31
+ });
32
+ it("shows a table with proper headers", async ()=>{
33
+ renderObjectList();
34
+ await waitFor(()=>{
35
+ expect(screen.getByRole("grid")).toBeInTheDocument();
36
+ });
37
+ expect(screen.getByText("Name")).toBeInTheDocument();
38
+ expect(screen.getByText("Modified on")).toBeInTheDocument();
39
+ expect(screen.getByText("Size")).toBeInTheDocument();
40
+ expect(screen.getByText("Storage Location")).toBeInTheDocument();
41
+ });
42
+ it("renders content and handles interactions", async ()=>{
43
+ const onObjectSelect = jest.fn();
44
+ const onPrefixChange = jest.fn();
45
+ renderObjectList({
46
+ onObjectSelect,
47
+ onPrefixChange
48
+ });
49
+ await waitFor(()=>{
50
+ expect(screen.getByText("file1.txt")).toBeInTheDocument();
51
+ });
52
+ expect(screen.getByText("file1.txt")).toBeInTheDocument();
53
+ expect(screen.getByText("file2.txt")).toBeInTheDocument();
54
+ expect(screen.getByText("folder1/")).toBeInTheDocument();
55
+ fireEvent.click(screen.getByText("file1.txt"));
56
+ expect(onObjectSelect).toHaveBeenCalledWith(expect.objectContaining({
57
+ Key: "file1.txt",
58
+ type: "object"
59
+ }));
60
+ fireEvent.click(screen.getByText("folder1/"));
61
+ expect(onPrefixChange).toHaveBeenCalledWith("folder1/");
62
+ const rows = screen.getAllByRole("row");
63
+ expect(rows.length).toBe(4);
64
+ });
65
+ it("displays data formatting correctly", async ()=>{
66
+ renderObjectList();
67
+ await waitFor(()=>{
68
+ expect(screen.getByText("file1.txt")).toBeInTheDocument();
69
+ });
70
+ const gridElement = screen.getByRole("grid");
71
+ expect(gridElement).toHaveTextContent("2023");
72
+ const dateElements = screen.getAllByText(/2023/);
73
+ expect(dateElements.length).toBeGreaterThanOrEqual(2);
74
+ expect(screen.getByText("1 KB")).toBeInTheDocument();
75
+ expect(screen.getByText("512 B")).toBeInTheDocument();
76
+ const defaultElements = screen.getAllByText("default");
77
+ expect(defaultElements.length).toBeGreaterThan(0);
78
+ const dashes = screen.getAllByText("-");
79
+ expect(dashes.length).toBeGreaterThanOrEqual(2);
80
+ });
81
+ it("handles edge cases gracefully", async ()=>{
82
+ renderObjectList({
83
+ bucketName: "empty-bucket"
84
+ });
85
+ await waitFor(()=>{
86
+ expect(screen.getByRole("grid")).toBeInTheDocument();
87
+ });
88
+ expect(screen.getByText("Name")).toBeInTheDocument();
89
+ const mockOnObjectSelect = jest.fn();
90
+ const mockOnPrefixChange = jest.fn();
91
+ renderObjectList({
92
+ onObjectSelect: mockOnObjectSelect,
93
+ onPrefixChange: mockOnPrefixChange
94
+ });
95
+ await waitFor(()=>{
96
+ expect(screen.getByText("file1.txt")).toBeInTheDocument();
97
+ });
98
+ expect(()=>{
99
+ fireEvent.click(screen.getByText("file1.txt"));
100
+ fireEvent.click(screen.getByText("folder1/"));
101
+ }).not.toThrow();
102
+ expect(mockOnObjectSelect).toHaveBeenCalled();
103
+ expect(mockOnPrefixChange).toHaveBeenCalled();
104
+ });
105
+ it("handles loading states", async ()=>{
106
+ renderObjectList();
107
+ expect(screen.getByRole("grid")).toBeInTheDocument();
108
+ await waitFor(()=>{
109
+ expect(screen.getByText("file1.txt")).toBeInTheDocument();
110
+ });
111
+ expect(screen.getByText("file1.txt")).toBeInTheDocument();
112
+ expect(screen.getByText("file2.txt")).toBeInTheDocument();
113
+ expect(screen.getByText("folder1/")).toBeInTheDocument();
114
+ });
115
+ it("prevents event propagation when clicking links", async ()=>{
116
+ const onObjectSelect = jest.fn();
117
+ const onPrefixChange = jest.fn();
118
+ renderObjectList({
119
+ onObjectSelect,
120
+ onPrefixChange
121
+ });
122
+ await waitFor(()=>{
123
+ expect(screen.getByText("file1.txt")).toBeInTheDocument();
124
+ });
125
+ fireEvent.click(screen.getByText("file1.txt"));
126
+ expect(onObjectSelect).toHaveBeenCalledWith(expect.objectContaining({
127
+ Key: "file1.txt",
128
+ type: "object"
129
+ }));
130
+ expect(onPrefixChange).not.toHaveBeenCalled();
131
+ });
132
+ it("supports prefix-based navigation", async ()=>{
133
+ renderObjectList({
134
+ prefix: "folder1/"
135
+ });
136
+ await waitFor(()=>{
137
+ expect(screen.getByRole("grid")).toBeInTheDocument();
138
+ });
139
+ expect(screen.getByText("Name")).toBeInTheDocument();
140
+ });
141
+ it("displays correct icons for folders and files", async ()=>{
142
+ renderObjectList();
143
+ await waitFor(()=>{
144
+ expect(screen.getByText("file1.txt")).toBeInTheDocument();
145
+ });
146
+ expect(screen.getByText("file1.txt")).toBeInTheDocument();
147
+ expect(screen.getByText("file2.txt")).toBeInTheDocument();
148
+ expect(screen.getByText("folder1/")).toBeInTheDocument();
149
+ const nameCells = screen.getAllByRole("gridcell").filter((cell)=>cell.textContent?.includes("folder1/") || cell.textContent?.includes("file1.txt") || cell.textContent?.includes("file2.txt"));
150
+ expect(nameCells.length).toBe(3);
151
+ nameCells.forEach((cell)=>{
152
+ const iconElement = cell.querySelector("svg, img, i");
153
+ const linkElement = cell.querySelector("a");
154
+ expect(iconElement).toBeInTheDocument();
155
+ expect(linkElement).toBeInTheDocument();
156
+ });
157
+ });
158
+ it("integrates with MultiSelectableContent", async ()=>{
159
+ renderObjectList();
160
+ await waitFor(()=>{
161
+ expect(screen.getByText("file1.txt")).toBeInTheDocument();
162
+ });
163
+ expect(screen.getByRole("grid")).toBeInTheDocument();
164
+ const rows = screen.getAllByRole("row");
165
+ expect(rows.length).toBe(4);
166
+ });
167
+ it("handles infinite scroll pagination correctly", async ()=>{
168
+ renderObjectList({
169
+ bucketName: "paginated-bucket"
170
+ });
171
+ await waitFor(()=>{
172
+ expect(screen.getByText("file1.txt")).toBeInTheDocument();
173
+ });
174
+ expect(screen.getByText("file1.txt")).toBeInTheDocument();
175
+ expect(screen.getByText("file2.txt")).toBeInTheDocument();
176
+ expect(screen.getByText("folder1/")).toBeInTheDocument();
177
+ expect(screen.queryByText("file3.txt")).not.toBeInTheDocument();
178
+ const table = screen.getByRole("grid");
179
+ expect(table).toBeInTheDocument();
180
+ const tableContent = table.querySelector('[role="rowgroup"]') || table.querySelector('[data-testid*="multi-selectable"]');
181
+ expect(tableContent).toBeInTheDocument();
182
+ const rows = screen.getAllByRole("row");
183
+ expect(rows.length).toBe(4);
184
+ expect(()=>{
185
+ screen.getByText("Name");
186
+ screen.getByText("Modified on");
187
+ screen.getByText("Size");
188
+ screen.getByText("Storage Location");
189
+ }).not.toThrow();
190
+ expect(screen.getByText("1 KB")).toBeInTheDocument();
191
+ expect(screen.getByText("512 B")).toBeInTheDocument();
192
+ });
193
+ it("integrates with infinite query hook for data loading", async ()=>{
194
+ const { result } = renderHook(()=>useListObjects({
195
+ Bucket: "paginated-bucket",
196
+ MaxKeys: 20,
197
+ Delimiter: "/"
198
+ }), {
199
+ wrapper: createTestWrapper()
200
+ });
201
+ await waitFor(()=>{
202
+ expect(result.current.isSuccess).toBe(true);
203
+ });
204
+ expect(result.current.data).toBeDefined();
205
+ expect(result.current.hasNextPage).toBe(true);
206
+ expect(typeof result.current.fetchNextPage).toBe("function");
207
+ expect(result.current.isFetchingNextPage).toBe(false);
208
+ const firstPage = result.current.data?.pages[0];
209
+ expect(firstPage).toBeDefined();
210
+ expect(firstPage?.Contents).toHaveLength(2);
211
+ expect(firstPage?.CommonPrefixes).toHaveLength(1);
212
+ result.current.fetchNextPage();
213
+ await waitFor(()=>{
214
+ const updatedData = result.current.data;
215
+ expect(updatedData?.pages).toHaveLength(2);
216
+ });
217
+ const secondPage = result.current.data?.pages[1];
218
+ expect(secondPage).toBeDefined();
219
+ expect(secondPage?.Contents).toHaveLength(1);
220
+ expect(result.current.hasNextPage).toBe(false);
221
+ });
222
+ it("renders List Versions toggle", async ()=>{
223
+ renderObjectList();
224
+ await waitFor(()=>{
225
+ expect(screen.getByText("List Versions")).toBeInTheDocument();
226
+ });
227
+ const toggleLabel = screen.getByText(/list versions/i);
228
+ const toggle = toggleLabel.closest("label")?.querySelector('input[type="checkbox"]');
229
+ expect(toggle).toBeInTheDocument();
230
+ expect(toggle).not.toBeChecked();
231
+ });
232
+ it("shows Version ID column when toggle is enabled", async ()=>{
233
+ renderObjectList();
234
+ await waitFor(()=>{
235
+ expect(screen.getByText("List Versions")).toBeInTheDocument();
236
+ });
237
+ const toggleLabel = screen.getByText(/list versions/i);
238
+ const toggle = toggleLabel.closest("label")?.querySelector('input[type="checkbox"]');
239
+ fireEvent.click(toggle);
240
+ await waitFor(()=>{
241
+ expect(screen.getByText("Version ID")).toBeInTheDocument();
242
+ });
243
+ });
244
+ it("does not show Version ID column by default", async ()=>{
245
+ renderObjectList();
246
+ await waitFor(()=>{
247
+ expect(screen.getByText("Name")).toBeInTheDocument();
248
+ });
249
+ expect(screen.queryByText("Version ID")).not.toBeInTheDocument();
250
+ });
251
+ it("integrates with version listing hook", async ()=>{
252
+ const { result } = renderHook(()=>useListObjectVersions({
253
+ Bucket: "test-bucket",
254
+ MaxKeys: 20,
255
+ Delimiter: "/"
256
+ }), {
257
+ wrapper: createTestWrapper()
258
+ });
259
+ await waitFor(()=>{
260
+ expect(result.current.isSuccess).toBe(true);
261
+ });
262
+ expect(result.current.data).toBeDefined();
263
+ expect(typeof result.current.fetchNextPage).toBe("function");
264
+ const firstPage = result.current.data?.pages[0];
265
+ expect(firstPage).toBeDefined();
266
+ });
267
+ it("renders table search when metadata-search feature is disabled", async ()=>{
268
+ jest.spyOn(__WEBPACK_EXTERNAL_MODULE__utils_useFeatures_js_1facdd0d__, "useFeatures").mockReturnValue(false);
269
+ renderObjectList();
270
+ await waitFor(()=>{
271
+ expect(screen.getByRole("grid")).toBeInTheDocument();
272
+ });
273
+ expect(screen.queryByPlaceholderText(/Metadata Search/i)).not.toBeInTheDocument();
274
+ });
275
+ it("renders MetadataSearch component when metadata-search feature is enabled", async ()=>{
276
+ jest.spyOn(__WEBPACK_EXTERNAL_MODULE__utils_useFeatures_js_1facdd0d__, "useFeatures").mockReturnValue(true);
277
+ renderObjectList();
278
+ await waitFor(()=>{
279
+ expect(screen.getByRole("grid")).toBeInTheDocument();
280
+ });
281
+ expect(screen.getByPlaceholderText(/Metadata Search/i)).toBeInTheDocument();
282
+ });
283
+ });
@@ -0,0 +1 @@
1
+ export {};
@@ -0,0 +1,144 @@
1
+ import { jsx } from "react/jsx-runtime";
2
+ import { fireEvent, render, screen, waitFor } from "@testing-library/react";
3
+ import user_event from "@testing-library/user-event";
4
+ import { createTestWrapper } from "../../test/testUtils.js";
5
+ import { UploadButton } from "../objects/UploadButton.js";
6
+ describe("UploadButton - Core Functionality", ()=>{
7
+ const defaultProps = {
8
+ bucket: "test-bucket",
9
+ prefix: "test-prefix"
10
+ };
11
+ const renderUploadButton = (props = {})=>{
12
+ const Wrapper = createTestWrapper();
13
+ return render(/*#__PURE__*/ jsx(Wrapper, {
14
+ children: /*#__PURE__*/ jsx(UploadButton, {
15
+ ...defaultProps,
16
+ ...props
17
+ })
18
+ }));
19
+ };
20
+ it("renders upload button with default label", ()=>{
21
+ renderUploadButton();
22
+ expect(screen.getByRole("button", {
23
+ name: /upload/i
24
+ })).toBeInTheDocument();
25
+ });
26
+ it("renders upload button", ()=>{
27
+ renderUploadButton();
28
+ expect(screen.getByRole("button", {
29
+ name: /upload/i
30
+ })).toBeInTheDocument();
31
+ });
32
+ it("opens modal when upload button is clicked", async ()=>{
33
+ renderUploadButton();
34
+ const uploadButton = screen.getByRole("button", {
35
+ name: /upload/i
36
+ });
37
+ fireEvent.click(uploadButton);
38
+ await waitFor(()=>{
39
+ expect(screen.getByText("Upload Files")).toBeInTheDocument();
40
+ });
41
+ });
42
+ it("closes modal when cancel button is clicked", async ()=>{
43
+ renderUploadButton();
44
+ const uploadButton = screen.getByRole("button", {
45
+ name: /upload/i
46
+ });
47
+ fireEvent.click(uploadButton);
48
+ await waitFor(()=>{
49
+ expect(screen.getByText("Upload Files")).toBeInTheDocument();
50
+ });
51
+ const cancelButton = screen.getByRole("button", {
52
+ name: /cancel/i
53
+ });
54
+ fireEvent.click(cancelButton);
55
+ await waitFor(()=>{
56
+ expect(screen.queryByText("Upload Files")).not.toBeInTheDocument();
57
+ });
58
+ });
59
+ it("displays empty state initially", async ()=>{
60
+ renderUploadButton();
61
+ const uploadButton = screen.getByRole("button", {
62
+ name: /upload/i
63
+ });
64
+ fireEvent.click(uploadButton);
65
+ await waitFor(()=>{
66
+ expect(screen.getByText("Drag and drop files and folders here")).toBeInTheDocument();
67
+ expect(screen.getByRole("button", {
68
+ name: /add files/i
69
+ })).toBeInTheDocument();
70
+ });
71
+ });
72
+ it("can add files and shows upload button enabled", async ()=>{
73
+ const onUploadSuccess = jest.fn();
74
+ renderUploadButton({
75
+ onUploadSuccess
76
+ });
77
+ const uploadButton = screen.getByRole("button", {
78
+ name: /upload/i
79
+ });
80
+ fireEvent.click(uploadButton);
81
+ await waitFor(()=>{
82
+ expect(screen.getByText("Upload Files")).toBeInTheDocument();
83
+ });
84
+ const uploadButtons = screen.getAllByRole("button", {
85
+ name: "Upload"
86
+ });
87
+ const modalUploadButton = uploadButtons.find((button)=>!button.querySelector("svg"));
88
+ expect(modalUploadButton).toBeDisabled();
89
+ const fileInput = screen.getByRole("presentation").querySelector('input[type="file"]');
90
+ const testFile = new File([
91
+ "test content"
92
+ ], "test.txt", {
93
+ type: "text/plain"
94
+ });
95
+ await user_event.upload(fileInput, testFile);
96
+ await waitFor(()=>{
97
+ expect(screen.getByText("test.txt")).toBeInTheDocument();
98
+ expect(screen.getByText("12 B")).toBeInTheDocument();
99
+ });
100
+ expect(modalUploadButton).toBeEnabled();
101
+ });
102
+ it("displays correct bucket and prefix information", ()=>{
103
+ const customBucket = "custom-bucket";
104
+ const customPrefix = "custom/prefix";
105
+ renderUploadButton({
106
+ bucket: customBucket,
107
+ prefix: customPrefix
108
+ });
109
+ expect(screen.getByRole("button", {
110
+ name: /upload/i
111
+ })).toBeInTheDocument();
112
+ });
113
+ it("handles different button variants", ()=>{
114
+ renderUploadButton({
115
+ variant: "primary"
116
+ });
117
+ const button = screen.getByRole("button", {
118
+ name: /upload/i
119
+ });
120
+ expect(button).toBeInTheDocument();
121
+ });
122
+ it("resets state when modal is closed", async ()=>{
123
+ renderUploadButton();
124
+ const uploadButton = screen.getByRole("button", {
125
+ name: /upload/i
126
+ });
127
+ fireEvent.click(uploadButton);
128
+ await waitFor(()=>{
129
+ expect(screen.getByText("Upload Files")).toBeInTheDocument();
130
+ });
131
+ const cancelButton = screen.getByRole("button", {
132
+ name: /cancel/i
133
+ });
134
+ fireEvent.click(cancelButton);
135
+ await waitFor(()=>{
136
+ expect(screen.queryByText("Upload Files")).not.toBeInTheDocument();
137
+ });
138
+ fireEvent.click(uploadButton);
139
+ await waitFor(()=>{
140
+ expect(screen.getByText("Upload Files")).toBeInTheDocument();
141
+ expect(screen.getByText("Drag and drop files and folders here")).toBeInTheDocument();
142
+ });
143
+ });
144
+ });
@@ -0,0 +1 @@
1
+ export declare function BucketDetails(): import("react/jsx-runtime").JSX.Element;
@@ -0,0 +1,51 @@
1
+ import { Fragment, jsx, jsxs } from "react/jsx-runtime";
2
+ import { useQueryParams } from "../../utils/hooks.js";
3
+ import { useNavigate, useParams } from "react-router-dom";
4
+ import { Tabs } from "@scality/core-ui/dist/next";
5
+ import { BucketOverview } from "./BucketOverview.js";
6
+ import { DeleteBucketButton } from "./DeleteBucketButton.js";
7
+ function BucketDetails() {
8
+ const query = useQueryParams();
9
+ const queryObject = Object.fromEntries(query.entries());
10
+ const { bucketName } = useParams();
11
+ const navigate = useNavigate();
12
+ if (!bucketName) return /*#__PURE__*/ jsx("div", {
13
+ children: "No bucket selected"
14
+ });
15
+ return /*#__PURE__*/ jsx(Fragment, {
16
+ children: /*#__PURE__*/ jsxs(Tabs, {
17
+ children: [
18
+ /*#__PURE__*/ jsx(Tabs.Tab, {
19
+ label: "Overview",
20
+ path: "",
21
+ query: {
22
+ ...queryObject,
23
+ tab: ""
24
+ },
25
+ children: /*#__PURE__*/ jsx(BucketOverview, {
26
+ bucketName: bucketName,
27
+ renderDeleteButton: ()=>/*#__PURE__*/ jsx(DeleteBucketButton, {
28
+ bucketName: bucketName
29
+ }),
30
+ onEditPolicy: ()=>{
31
+ navigate(`/buckets/${bucketName}/policy`);
32
+ }
33
+ })
34
+ }),
35
+ /*#__PURE__*/ jsx(Tabs.Tab, {
36
+ label: "Workflow",
37
+ path: "",
38
+ query: {
39
+ ...queryObject,
40
+ tab: "workflow"
41
+ },
42
+ withoutPadding: true,
43
+ children: /*#__PURE__*/ jsx(Fragment, {
44
+ children: "Workflow"
45
+ })
46
+ })
47
+ ]
48
+ })
49
+ });
50
+ }
51
+ export { BucketDetails };
@@ -0,0 +1,12 @@
1
+ import type { Bucket } from "@aws-sdk/client-s3";
2
+ interface BucketListProps {
3
+ buckets: Bucket[];
4
+ bucketStatus?: "idle" | "loading" | "error" | "success";
5
+ selectedBucketName?: string | null;
6
+ onBucketSelect?: (bucketName: string) => void;
7
+ onCreateBucket?: () => void;
8
+ onNavigateToBucket?: (bucketName: string) => void;
9
+ renderBucketLocation?: (bucketName: string) => React.ReactNode;
10
+ }
11
+ export declare function BucketList({ buckets, bucketStatus, selectedBucketName, onBucketSelect, onCreateBucket, onNavigateToBucket, renderBucketLocation, }: BucketListProps): import("react/jsx-runtime").JSX.Element;
12
+ export {};
@@ -0,0 +1,136 @@
1
+ import { jsx, jsxs } from "react/jsx-runtime";
2
+ import { ConstrainedText, FormattedDateTime, Icon, Link, Wrap, spacing } from "@scality/core-ui";
3
+ import { Box, Button, Table } from "@scality/core-ui/dist/next";
4
+ import { useMemo } from "react";
5
+ const SEARCH_QUERY_PARAM = "search";
6
+ function BucketList({ buckets, bucketStatus, selectedBucketName, onBucketSelect, onCreateBucket, onNavigateToBucket, renderBucketLocation }) {
7
+ const columns = useMemo(()=>[
8
+ {
9
+ Header: "Bucket Name",
10
+ accessor: "Name",
11
+ id: "name",
12
+ Cell: ({ value, row })=>{
13
+ const name = value || row.original?.Name;
14
+ if (!name) return /*#__PURE__*/ jsx("span", {
15
+ children: "-"
16
+ });
17
+ return /*#__PURE__*/ jsx(ConstrainedText, {
18
+ text: /*#__PURE__*/ jsx(Link, {
19
+ onClick: (e)=>{
20
+ e.stopPropagation();
21
+ onNavigateToBucket?.(name);
22
+ },
23
+ children: name
24
+ }),
25
+ lineClamp: 2
26
+ });
27
+ },
28
+ cellStyle: {
29
+ flex: "1",
30
+ width: "unset"
31
+ }
32
+ },
33
+ {
34
+ Header: "Storage Location",
35
+ accessor: "Name",
36
+ id: "location",
37
+ Cell ({ value }) {
38
+ return renderBucketLocation ? renderBucketLocation(value) : /*#__PURE__*/ jsx("span", {
39
+ children: "-"
40
+ });
41
+ },
42
+ cellStyle: {
43
+ width: "unset",
44
+ flex: "1.2"
45
+ }
46
+ },
47
+ {
48
+ Header: "Created on",
49
+ accessor: "CreationDate",
50
+ id: "date",
51
+ cellStyle: {
52
+ flex: "1",
53
+ textAlign: "right",
54
+ paddingRight: spacing.r16,
55
+ width: "unset"
56
+ },
57
+ Cell: ({ value, row })=>{
58
+ const date = value || row.original?.CreationDate;
59
+ if (!date) return /*#__PURE__*/ jsx("span", {
60
+ children: "-"
61
+ });
62
+ return /*#__PURE__*/ jsx(FormattedDateTime, {
63
+ format: "date-time-second",
64
+ value: new Date(date)
65
+ });
66
+ }
67
+ }
68
+ ], [
69
+ onNavigateToBucket,
70
+ renderBucketLocation
71
+ ]);
72
+ const tableData = useMemo(()=>buckets.map(({ Name, CreationDate, ...bucket })=>({
73
+ Name,
74
+ CreationDate,
75
+ ...bucket
76
+ })), [
77
+ buckets
78
+ ]);
79
+ const selectedId = useMemo(()=>{
80
+ if (buckets && selectedBucketName) return buckets.findIndex((bucket)=>bucket.Name === selectedBucketName);
81
+ return null;
82
+ }, [
83
+ selectedBucketName,
84
+ buckets
85
+ ]);
86
+ return /*#__PURE__*/ jsxs(Table, {
87
+ columns: columns,
88
+ data: tableData,
89
+ status: bucketStatus,
90
+ defaultSortingKey: "CreationDate",
91
+ entityName: {
92
+ en: {
93
+ singular: "bucket",
94
+ plural: "buckets"
95
+ }
96
+ },
97
+ children: [
98
+ /*#__PURE__*/ jsxs(Wrap, {
99
+ padding: spacing.r16,
100
+ children: [
101
+ /*#__PURE__*/ jsx(Box, {
102
+ display: "flex",
103
+ justifyContent: "space-between",
104
+ alignItems: "center",
105
+ gap: spacing.r4,
106
+ children: /*#__PURE__*/ jsx(Table.SearchWithQueryParams, {
107
+ queryParams: SEARCH_QUERY_PARAM
108
+ })
109
+ }),
110
+ /*#__PURE__*/ jsx(Box, {
111
+ gap: "r16",
112
+ children: /*#__PURE__*/ jsx(Button, {
113
+ icon: /*#__PURE__*/ jsx(Icon, {
114
+ name: "Create-add"
115
+ }),
116
+ label: "Create Bucket",
117
+ variant: "primary",
118
+ onClick: onCreateBucket,
119
+ type: "submit"
120
+ })
121
+ })
122
+ ]
123
+ }),
124
+ /*#__PURE__*/ jsx(Table.SingleSelectableContent, {
125
+ rowHeight: "h40",
126
+ selectedId: selectedId?.toString(),
127
+ onRowSelected: (row)=>{
128
+ const isSelected = selectedBucketName === row.original.Name;
129
+ if (!isSelected && row.original.Name) onBucketSelect?.(row.original.Name);
130
+ },
131
+ separationLineVariant: "backgroundLevel1"
132
+ })
133
+ ]
134
+ });
135
+ }
136
+ export { BucketList };
@@ -0,0 +1,3 @@
1
+ export declare const BucketLocation: ({ bucketName }: {
2
+ bucketName: string;
3
+ }) => import("react/jsx-runtime").JSX.Element;
@@ -0,0 +1,16 @@
1
+ import { jsx } from "react/jsx-runtime";
2
+ import { useGetBucketLocation } from "../../hooks/index.js";
3
+ import { Loader } from "@scality/core-ui";
4
+ const BucketLocation = ({ bucketName })=>{
5
+ const { data: bucketLocation, status } = useGetBucketLocation({
6
+ Bucket: bucketName
7
+ });
8
+ if ("pending" === status) return /*#__PURE__*/ jsx(Loader, {});
9
+ if ("error" === status) return /*#__PURE__*/ jsx("div", {
10
+ children: "Error"
11
+ });
12
+ return /*#__PURE__*/ jsx("div", {
13
+ children: bucketLocation?.LocationConstraint || "us-east-1"
14
+ });
15
+ };
16
+ export { BucketLocation };
@@ -0,0 +1,14 @@
1
+ import React from "react";
2
+ interface BucketOverviewProps {
3
+ bucketName: string;
4
+ onEmptyBucket?: () => void;
5
+ onDeleteBucket?: () => void;
6
+ onEditPolicy?: (bucketName: string) => void;
7
+ renderBucketLocation?: (bucketName: string) => React.ReactNode;
8
+ renderDeleteButton?: (bucketName: string) => React.ReactNode;
9
+ renderEmptyButton?: (bucketName: string) => React.ReactNode;
10
+ isEmptyBucketDisabled?: boolean;
11
+ isDeleteBucketDisabled?: boolean;
12
+ }
13
+ declare const BucketOverview: React.FC<BucketOverviewProps>;
14
+ export { BucketOverview };