@fgv/ts-web-extras 5.1.0-3 → 5.1.0-30

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 (275) hide show
  1. package/dist/packlets/crypto-utils/browserCryptoProvider.js +330 -18
  2. package/dist/packlets/crypto-utils/browserCryptoProvider.js.map +1 -1
  3. package/dist/packlets/crypto-utils/index.js +12 -0
  4. package/dist/packlets/crypto-utils/index.js.map +1 -1
  5. package/dist/packlets/file-tree/directoryHandleStore.js.map +1 -1
  6. package/dist/packlets/file-tree/fileApiTreeAccessors.js +1 -1
  7. package/dist/packlets/file-tree/fileApiTreeAccessors.js.map +1 -1
  8. package/dist/packlets/file-tree/fileSystemAccessTreeAccessors.js +2 -2
  9. package/dist/packlets/file-tree/fileSystemAccessTreeAccessors.js.map +1 -1
  10. package/dist/packlets/file-tree/httpTreeAccessors.js +74 -44
  11. package/dist/packlets/file-tree/httpTreeAccessors.js.map +1 -1
  12. package/dist/packlets/file-tree/localStorageTreeAccessors.js.map +1 -1
  13. package/dist/packlets/helpers/fileTreeHelpers.js +1 -1
  14. package/dist/packlets/helpers/fileTreeHelpers.js.map +1 -1
  15. package/dist/ts-web-extras.d.ts +141 -10
  16. package/dist/tsdoc-metadata.json +1 -1
  17. package/eslint.config.js +32 -0
  18. package/lib/packlets/crypto-utils/browserCryptoProvider.d.ts +115 -6
  19. package/lib/packlets/crypto-utils/browserCryptoProvider.d.ts.map +1 -1
  20. package/lib/packlets/crypto-utils/browserCryptoProvider.js +329 -17
  21. package/lib/packlets/crypto-utils/browserCryptoProvider.js.map +1 -1
  22. package/lib/packlets/crypto-utils/index.d.ts +13 -0
  23. package/lib/packlets/crypto-utils/index.d.ts.map +1 -1
  24. package/lib/packlets/crypto-utils/index.js +13 -0
  25. package/lib/packlets/crypto-utils/index.js.map +1 -1
  26. package/lib/packlets/file-tree/directoryHandleStore.d.ts +2 -2
  27. package/lib/packlets/file-tree/directoryHandleStore.d.ts.map +1 -1
  28. package/lib/packlets/file-tree/directoryHandleStore.js.map +1 -1
  29. package/lib/packlets/file-tree/fileApiTreeAccessors.d.ts +2 -2
  30. package/lib/packlets/file-tree/fileApiTreeAccessors.d.ts.map +1 -1
  31. package/lib/packlets/file-tree/fileApiTreeAccessors.js +1 -1
  32. package/lib/packlets/file-tree/fileApiTreeAccessors.js.map +1 -1
  33. package/lib/packlets/file-tree/fileSystemAccessTreeAccessors.d.ts.map +1 -1
  34. package/lib/packlets/file-tree/fileSystemAccessTreeAccessors.js +2 -2
  35. package/lib/packlets/file-tree/fileSystemAccessTreeAccessors.js.map +1 -1
  36. package/lib/packlets/file-tree/httpTreeAccessors.d.ts +6 -0
  37. package/lib/packlets/file-tree/httpTreeAccessors.d.ts.map +1 -1
  38. package/lib/packlets/file-tree/httpTreeAccessors.js +74 -44
  39. package/lib/packlets/file-tree/httpTreeAccessors.js.map +1 -1
  40. package/lib/packlets/file-tree/localStorageTreeAccessors.js.map +1 -1
  41. package/lib/packlets/helpers/fileTreeHelpers.d.ts +1 -1
  42. package/lib/packlets/helpers/fileTreeHelpers.d.ts.map +1 -1
  43. package/lib/packlets/helpers/fileTreeHelpers.js +6 -6
  44. package/lib/packlets/helpers/fileTreeHelpers.js.map +1 -1
  45. package/package.json +16 -15
  46. package/.rush/temp/61c1d4a91d98f048b475a5bb3e6541c5d27819de.tar.log +0 -237
  47. package/.rush/temp/chunked-rush-logs/ts-web-extras.build.chunks.jsonl +0 -55
  48. package/.rush/temp/operation/build/all.log +0 -55
  49. package/.rush/temp/operation/build/log-chunks.jsonl +0 -55
  50. package/.rush/temp/operation/build/state.json +0 -3
  51. package/.rush/temp/shrinkwrap-deps.json +0 -635
  52. package/CHANGELOG.md +0 -23
  53. package/config/api-extractor.json +0 -343
  54. package/config/jest.config.json +0 -19
  55. package/config/rig.json +0 -16
  56. package/config/typedoc.json +0 -6
  57. package/dist/test/mocks/idb-keyval.js +0 -6
  58. package/dist/test/mocks/idb-keyval.js.map +0 -1
  59. package/dist/test/setupTests.js +0 -74
  60. package/dist/test/setupTests.js.map +0 -1
  61. package/dist/test/unit/browserHashProvider.test.js +0 -140
  62. package/dist/test/unit/browserHashProvider.test.js.map +0 -1
  63. package/dist/test/unit/directoryHandleStore.test.js +0 -190
  64. package/dist/test/unit/directoryHandleStore.test.js.map +0 -1
  65. package/dist/test/unit/fileApiTreeAccessors.test.js +0 -1188
  66. package/dist/test/unit/fileApiTreeAccessors.test.js.map +0 -1
  67. package/dist/test/unit/fileApiTypes.test.js +0 -472
  68. package/dist/test/unit/fileApiTypes.test.js.map +0 -1
  69. package/dist/test/unit/fileSystemAccessTreeAccessors.test.js +0 -622
  70. package/dist/test/unit/fileSystemAccessTreeAccessors.test.js.map +0 -1
  71. package/dist/test/unit/fileTreeHelpers.test.js +0 -590
  72. package/dist/test/unit/fileTreeHelpers.test.js.map +0 -1
  73. package/dist/test/unit/httpTreeAccessors.test.js +0 -1229
  74. package/dist/test/unit/httpTreeAccessors.test.js.map +0 -1
  75. package/dist/test/unit/localStorageTreeAccessors.test.js +0 -812
  76. package/dist/test/unit/localStorageTreeAccessors.test.js.map +0 -1
  77. package/dist/test/unit/urlParams.test.js +0 -393
  78. package/dist/test/unit/urlParams.test.js.map +0 -1
  79. package/dist/test/utils/fileSystemAccessMocks.js +0 -271
  80. package/dist/test/utils/fileSystemAccessMocks.js.map +0 -1
  81. package/dist/test/utils/testHelpers.js +0 -124
  82. package/dist/test/utils/testHelpers.js.map +0 -1
  83. package/docs/@fgv/namespaces/CryptoUtils/README.md +0 -18
  84. package/docs/@fgv/namespaces/CryptoUtils/classes/BrowserCryptoProvider.md +0 -203
  85. package/docs/@fgv/namespaces/CryptoUtils/classes/BrowserHashProvider.md +0 -63
  86. package/docs/@fgv/namespaces/CryptoUtils/functions/createBrowserCryptoProvider.md +0 -18
  87. package/docs/@fgv/namespaces/FileTreeHelpers/README.md +0 -19
  88. package/docs/@fgv/namespaces/FileTreeHelpers/functions/extractFileListMetadata.md +0 -23
  89. package/docs/@fgv/namespaces/FileTreeHelpers/functions/extractFileMetadata.md +0 -23
  90. package/docs/@fgv/namespaces/FileTreeHelpers/functions/fromDirectoryUpload.md +0 -33
  91. package/docs/@fgv/namespaces/FileTreeHelpers/functions/fromFileList.md +0 -33
  92. package/docs/@fgv/namespaces/FileTreeHelpers/functions/getOriginalFile.md +0 -25
  93. package/docs/@fgv/namespaces/FileTreeHelpers/variables/defaultFileApiTreeInitParams.md +0 -11
  94. package/docs/README.md +0 -78
  95. package/docs/classes/DirectoryHandleStore.md +0 -116
  96. package/docs/classes/FileApiTreeAccessors.md +0 -286
  97. package/docs/classes/FileSystemAccessTreeAccessors.md +0 -557
  98. package/docs/classes/HttpTreeAccessors.md +0 -508
  99. package/docs/classes/LocalStorageTreeAccessors.md +0 -520
  100. package/docs/functions/exportAsJson.md +0 -23
  101. package/docs/functions/exportUsingFileSystemAPI.md +0 -26
  102. package/docs/functions/extractDirectoryPath.md +0 -23
  103. package/docs/functions/isDirectoryHandle.md +0 -23
  104. package/docs/functions/isFileHandle.md +0 -23
  105. package/docs/functions/isFilePath.md +0 -21
  106. package/docs/functions/parseContextFilter.md +0 -22
  107. package/docs/functions/parseQualifierDefaults.md +0 -22
  108. package/docs/functions/parseResourceTypes.md +0 -22
  109. package/docs/functions/parseUrlParameters.md +0 -15
  110. package/docs/functions/safeShowDirectoryPicker.md +0 -24
  111. package/docs/functions/safeShowOpenFilePicker.md +0 -24
  112. package/docs/functions/safeShowSaveFilePicker.md +0 -24
  113. package/docs/functions/supportsFileSystemAccess.md +0 -23
  114. package/docs/interfaces/FilePickerAcceptType.md +0 -16
  115. package/docs/interfaces/FileSystemCreateWritableOptions.md +0 -15
  116. package/docs/interfaces/FileSystemDirectoryHandle.md +0 -187
  117. package/docs/interfaces/FileSystemFileHandle.md +0 -106
  118. package/docs/interfaces/FileSystemGetDirectoryOptions.md +0 -15
  119. package/docs/interfaces/FileSystemGetFileOptions.md +0 -15
  120. package/docs/interfaces/FileSystemHandle.md +0 -69
  121. package/docs/interfaces/FileSystemHandlePermissionDescriptor.md +0 -15
  122. package/docs/interfaces/FileSystemRemoveOptions.md +0 -15
  123. package/docs/interfaces/FileSystemWritableFileStream.md +0 -127
  124. package/docs/interfaces/IDirectoryHandleTreeInitializer.md +0 -17
  125. package/docs/interfaces/IFileHandleTreeInitializer.md +0 -16
  126. package/docs/interfaces/IFileListTreeInitializer.md +0 -15
  127. package/docs/interfaces/IFileMetadata.md +0 -19
  128. package/docs/interfaces/IFileSystemAccessTreeParams.md +0 -30
  129. package/docs/interfaces/IFsAccessApis.md +0 -57
  130. package/docs/interfaces/IHttpTreeParams.md +0 -32
  131. package/docs/interfaces/ILocalStorageTreeParams.md +0 -30
  132. package/docs/interfaces/IUrlConfigOptions.md +0 -27
  133. package/docs/interfaces/ShowDirectoryPickerOptions.md +0 -17
  134. package/docs/interfaces/ShowOpenFilePickerOptions.md +0 -19
  135. package/docs/interfaces/ShowSaveFilePickerOptions.md +0 -19
  136. package/docs/type-aliases/TreeInitializer.md +0 -11
  137. package/docs/type-aliases/WellKnownDirectory.md +0 -11
  138. package/docs/type-aliases/WindowWithFsAccess.md +0 -11
  139. package/docs/variables/DEFAULT_DIRECTORY_HANDLE_DB.md +0 -11
  140. package/docs/variables/DEFAULT_DIRECTORY_HANDLE_STORE.md +0 -11
  141. package/etc/ts-web-extras.api.md +0 -433
  142. package/lib/test/mocks/idb-keyval.d.ts +0 -6
  143. package/lib/test/mocks/idb-keyval.d.ts.map +0 -1
  144. package/lib/test/mocks/idb-keyval.js +0 -9
  145. package/lib/test/mocks/idb-keyval.js.map +0 -1
  146. package/lib/test/setupTests.d.ts +0 -2
  147. package/lib/test/setupTests.d.ts.map +0 -1
  148. package/lib/test/setupTests.js +0 -76
  149. package/lib/test/setupTests.js.map +0 -1
  150. package/lib/test/unit/browserHashProvider.test.d.ts +0 -2
  151. package/lib/test/unit/browserHashProvider.test.d.ts.map +0 -1
  152. package/lib/test/unit/browserHashProvider.test.js +0 -142
  153. package/lib/test/unit/browserHashProvider.test.js.map +0 -1
  154. package/lib/test/unit/directoryHandleStore.test.d.ts +0 -2
  155. package/lib/test/unit/directoryHandleStore.test.d.ts.map +0 -1
  156. package/lib/test/unit/directoryHandleStore.test.js +0 -192
  157. package/lib/test/unit/directoryHandleStore.test.js.map +0 -1
  158. package/lib/test/unit/fileApiTreeAccessors.test.d.ts +0 -2
  159. package/lib/test/unit/fileApiTreeAccessors.test.d.ts.map +0 -1
  160. package/lib/test/unit/fileApiTreeAccessors.test.js +0 -1190
  161. package/lib/test/unit/fileApiTreeAccessors.test.js.map +0 -1
  162. package/lib/test/unit/fileApiTypes.test.d.ts +0 -2
  163. package/lib/test/unit/fileApiTypes.test.d.ts.map +0 -1
  164. package/lib/test/unit/fileApiTypes.test.js +0 -474
  165. package/lib/test/unit/fileApiTypes.test.js.map +0 -1
  166. package/lib/test/unit/fileSystemAccessTreeAccessors.test.d.ts +0 -2
  167. package/lib/test/unit/fileSystemAccessTreeAccessors.test.d.ts.map +0 -1
  168. package/lib/test/unit/fileSystemAccessTreeAccessors.test.js +0 -624
  169. package/lib/test/unit/fileSystemAccessTreeAccessors.test.js.map +0 -1
  170. package/lib/test/unit/fileTreeHelpers.test.d.ts +0 -2
  171. package/lib/test/unit/fileTreeHelpers.test.d.ts.map +0 -1
  172. package/lib/test/unit/fileTreeHelpers.test.js +0 -592
  173. package/lib/test/unit/fileTreeHelpers.test.js.map +0 -1
  174. package/lib/test/unit/httpTreeAccessors.test.d.ts +0 -2
  175. package/lib/test/unit/httpTreeAccessors.test.d.ts.map +0 -1
  176. package/lib/test/unit/httpTreeAccessors.test.js +0 -1231
  177. package/lib/test/unit/httpTreeAccessors.test.js.map +0 -1
  178. package/lib/test/unit/localStorageTreeAccessors.test.d.ts +0 -2
  179. package/lib/test/unit/localStorageTreeAccessors.test.d.ts.map +0 -1
  180. package/lib/test/unit/localStorageTreeAccessors.test.js +0 -814
  181. package/lib/test/unit/localStorageTreeAccessors.test.js.map +0 -1
  182. package/lib/test/unit/urlParams.test.d.ts +0 -2
  183. package/lib/test/unit/urlParams.test.d.ts.map +0 -1
  184. package/lib/test/unit/urlParams.test.js +0 -395
  185. package/lib/test/unit/urlParams.test.js.map +0 -1
  186. package/lib/test/utils/fileSystemAccessMocks.d.ts +0 -53
  187. package/lib/test/utils/fileSystemAccessMocks.d.ts.map +0 -1
  188. package/lib/test/utils/fileSystemAccessMocks.js +0 -277
  189. package/lib/test/utils/fileSystemAccessMocks.js.map +0 -1
  190. package/lib/test/utils/testHelpers.d.ts +0 -51
  191. package/lib/test/utils/testHelpers.d.ts.map +0 -1
  192. package/lib/test/utils/testHelpers.js +0 -133
  193. package/lib/test/utils/testHelpers.js.map +0 -1
  194. package/rush-logs/ts-web-extras.build.cache.log +0 -3
  195. package/rush-logs/ts-web-extras.build.log +0 -55
  196. package/src/index.browser.ts +0 -24
  197. package/src/index.ts +0 -47
  198. package/src/packlets/crypto-utils/browserCryptoProvider.ts +0 -311
  199. package/src/packlets/crypto-utils/browserHashProvider.ts +0 -73
  200. package/src/packlets/crypto-utils/index.ts +0 -29
  201. package/src/packlets/file-api-types/index.ts +0 -366
  202. package/src/packlets/file-tree/directoryHandleStore.ts +0 -136
  203. package/src/packlets/file-tree/fileApiTreeAccessors.ts +0 -528
  204. package/src/packlets/file-tree/fileSystemAccessTreeAccessors.ts +0 -519
  205. package/src/packlets/file-tree/httpTreeAccessors.ts +0 -448
  206. package/src/packlets/file-tree/index.ts +0 -32
  207. package/src/packlets/file-tree/localStorageTreeAccessors.ts +0 -430
  208. package/src/packlets/helpers/fileTreeHelpers.ts +0 -107
  209. package/src/packlets/helpers/index.ts +0 -28
  210. package/src/packlets/url-utils/index.ts +0 -28
  211. package/src/packlets/url-utils/urlParams.ts +0 -245
  212. package/src/test/mocks/idb-keyval.ts +0 -5
  213. package/src/test/setupTests.ts +0 -87
  214. package/src/test/unit/browserHashProvider.test.ts +0 -155
  215. package/src/test/unit/browserHashProvider.test.ts.bak +0 -376
  216. package/src/test/unit/directoryHandleStore.test.ts +0 -251
  217. package/src/test/unit/fileApiTreeAccessors.test.ts +0 -1387
  218. package/src/test/unit/fileApiTypes.test.ts +0 -587
  219. package/src/test/unit/fileSystemAccessTreeAccessors.test.ts +0 -885
  220. package/src/test/unit/fileTreeHelpers.test.ts +0 -694
  221. package/src/test/unit/httpTreeAccessors.test.ts +0 -1571
  222. package/src/test/unit/localStorageTreeAccessors.test.ts +0 -1014
  223. package/src/test/unit/urlParams.test.ts +0 -464
  224. package/src/test/utils/fileSystemAccessMocks.ts +0 -353
  225. package/src/test/utils/testHelpers.ts +0 -155
  226. package/temp/build/typescript/ts_8nwakTlr.json +0 -1
  227. package/temp/coverage/base.css +0 -224
  228. package/temp/coverage/block-navigation.js +0 -87
  229. package/temp/coverage/crypto-utils/browserCryptoProvider.ts.html +0 -1018
  230. package/temp/coverage/crypto-utils/browserHashProvider.ts.html +0 -304
  231. package/temp/coverage/crypto-utils/index.html +0 -131
  232. package/temp/coverage/favicon.png +0 -0
  233. package/temp/coverage/file-tree/directoryHandleStore.ts.html +0 -493
  234. package/temp/coverage/file-tree/fileApiTreeAccessors.ts.html +0 -1669
  235. package/temp/coverage/file-tree/fileSystemAccessTreeAccessors.ts.html +0 -1642
  236. package/temp/coverage/file-tree/httpTreeAccessors.ts.html +0 -1429
  237. package/temp/coverage/file-tree/index.html +0 -176
  238. package/temp/coverage/file-tree/localStorageTreeAccessors.ts.html +0 -1375
  239. package/temp/coverage/helpers/fileTreeHelpers.ts.html +0 -406
  240. package/temp/coverage/helpers/index.html +0 -116
  241. package/temp/coverage/index.html +0 -161
  242. package/temp/coverage/lcov-report/base.css +0 -224
  243. package/temp/coverage/lcov-report/block-navigation.js +0 -87
  244. package/temp/coverage/lcov-report/crypto-utils/browserCryptoProvider.ts.html +0 -1018
  245. package/temp/coverage/lcov-report/crypto-utils/browserHashProvider.ts.html +0 -304
  246. package/temp/coverage/lcov-report/crypto-utils/index.html +0 -131
  247. package/temp/coverage/lcov-report/favicon.png +0 -0
  248. package/temp/coverage/lcov-report/file-tree/directoryHandleStore.ts.html +0 -493
  249. package/temp/coverage/lcov-report/file-tree/fileApiTreeAccessors.ts.html +0 -1669
  250. package/temp/coverage/lcov-report/file-tree/fileSystemAccessTreeAccessors.ts.html +0 -1642
  251. package/temp/coverage/lcov-report/file-tree/httpTreeAccessors.ts.html +0 -1429
  252. package/temp/coverage/lcov-report/file-tree/index.html +0 -176
  253. package/temp/coverage/lcov-report/file-tree/localStorageTreeAccessors.ts.html +0 -1375
  254. package/temp/coverage/lcov-report/helpers/fileTreeHelpers.ts.html +0 -406
  255. package/temp/coverage/lcov-report/helpers/index.html +0 -116
  256. package/temp/coverage/lcov-report/index.html +0 -161
  257. package/temp/coverage/lcov-report/prettify.css +0 -1
  258. package/temp/coverage/lcov-report/prettify.js +0 -2
  259. package/temp/coverage/lcov-report/sort-arrow-sprite.png +0 -0
  260. package/temp/coverage/lcov-report/sorter.js +0 -210
  261. package/temp/coverage/lcov-report/url-utils/index.html +0 -116
  262. package/temp/coverage/lcov-report/url-utils/urlParams.ts.html +0 -820
  263. package/temp/coverage/lcov.info +0 -3597
  264. package/temp/coverage/prettify.css +0 -1
  265. package/temp/coverage/prettify.js +0 -2
  266. package/temp/coverage/sort-arrow-sprite.png +0 -0
  267. package/temp/coverage/sorter.js +0 -210
  268. package/temp/coverage/url-utils/index.html +0 -116
  269. package/temp/coverage/url-utils/urlParams.ts.html +0 -820
  270. package/temp/test/jest/haste-map-7492f1b44480e0cdd1f220078fb3afd8-c8dd6c3430605adeb2f1cadf4f75e791-8c9336785555d572065b28c111982ba4 +0 -0
  271. package/temp/test/jest/jest-transform-cache-7492f1b44480e0cdd1f220078fb3afd8-79ef2876fae7ca75eedb2aa53dc48338/d6/package_d6e3b0fa94752e16b0b2a2777739b973 +0 -53
  272. package/temp/test/jest/perf-cache-7492f1b44480e0cdd1f220078fb3afd8-da39a3ee5e6b4b0d3255bfef95601890 +0 -1
  273. package/temp/ts-web-extras.api.json +0 -8850
  274. package/temp/ts-web-extras.api.md +0 -433
  275. package/tsconfig.json +0 -7
@@ -1,694 +0,0 @@
1
- /*
2
- * Copyright (c) 2025 Erik Fortune
3
- *
4
- * Permission is hereby granted, free of charge, to any person obtaining a copy
5
- * of this software and associated documentation files (the "Software"), to deal
6
- * in the Software without restriction, including without limitation the rights
7
- * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
8
- * copies of the Software, and to permit persons to whom the Software is
9
- * furnished to do so, subject to the following conditions:
10
- *
11
- * The above copyright notice and this permission notice shall be included in all
12
- * copies or substantial portions of the Software.
13
- *
14
- * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
15
- * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
16
- * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
17
- * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
18
- * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
19
- * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
20
- * SOFTWARE.
21
- */
22
-
23
- import '@fgv/ts-utils-jest';
24
- import { FileTreeHelpers } from '../../packlets/helpers';
25
- import { FileApiTreeAccessors } from '../../packlets/file-tree';
26
- import { createMockFile, createMockFileList, createMockDirectoryFileList } from '../utils/testHelpers';
27
-
28
- describe('FileTreeHelpers', () => {
29
- describe('fromFileList', () => {
30
- test('creates FileTree from simple FileList', async () => {
31
- const fileList = createMockFileList([
32
- { name: 'file1.txt', content: 'content1' },
33
- { name: 'file2.txt', content: 'content2' }
34
- ]);
35
-
36
- const result = await FileTreeHelpers.fromFileList(fileList);
37
- expect(result).toSucceedAndSatisfy((fileTree) => {
38
- // Verify files are accessible with proper paths
39
- expect(fileTree.getFile('/file1.txt')).toSucceed();
40
- expect(fileTree.getFile('/file2.txt')).toSucceed();
41
-
42
- // Verify content can be retrieved
43
- const content1 = fileTree.hal.getFileContents('/file1.txt');
44
- expect(content1).toSucceedWith('content1');
45
-
46
- const content2 = fileTree.hal.getFileContents('/file2.txt');
47
- expect(content2).toSucceedWith('content2');
48
- });
49
- });
50
-
51
- test('creates FileTree from FileList with prefix', async () => {
52
- const fileList = createMockFileList([{ name: 'test.txt', content: 'test content' }]);
53
-
54
- const result = await FileTreeHelpers.fromFileList(fileList, { prefix: '/upload' });
55
- expect(result).toSucceed();
56
- });
57
-
58
- test('handles empty FileList', async () => {
59
- const fileList = createMockFileList([]);
60
- const result = await FileTreeHelpers.fromFileList(fileList);
61
- expect(result).toSucceed();
62
- });
63
-
64
- test('handles various file types', async () => {
65
- const fileList = createMockFileList([
66
- { name: 'data.json', content: '{"key": "value"}', type: 'application/json' },
67
- { name: 'script.js', content: 'console.log("hello");', type: 'application/javascript' },
68
- { name: 'style.css', content: 'body { margin: 0; }', type: 'text/css' },
69
- { name: 'readme.txt', content: 'This is a readme', type: 'text/plain' }
70
- ]);
71
-
72
- const result = await FileTreeHelpers.fromFileList(fileList);
73
- expect(result).toSucceedAndSatisfy((fileTree) => {
74
- expect(fileTree.getFile('/data.json')).toSucceed();
75
- expect(fileTree.getFile('/script.js')).toSucceed();
76
- expect(fileTree.getFile('/style.css')).toSucceed();
77
- expect(fileTree.getFile('/readme.txt')).toSucceed();
78
- });
79
- });
80
-
81
- test('automatically applies MIME types as contentType', async () => {
82
- const fileList = createMockFileList([
83
- { name: 'data.json', content: '{"key": "value"}', type: 'application/json' },
84
- { name: 'script.js', content: 'console.log("hello");', type: 'application/javascript' },
85
- { name: 'style.css', content: 'body { margin: 0; }', type: 'text/css' },
86
- { name: 'readme.txt', content: 'This is a readme', type: 'text/plain' },
87
- { name: 'image.png', content: 'PNG data', type: 'image/png' },
88
- { name: 'document.pdf', content: 'PDF content', type: 'application/pdf' },
89
- { name: 'archive.zip', content: 'ZIP data', type: 'application/zip' }
90
- ]);
91
-
92
- const result = await FileTreeHelpers.fromFileList(fileList);
93
- expect(result).toSucceedAndSatisfy((fileTree) => {
94
- // Verify MIME types are automatically applied as contentType
95
- expect(fileTree.getFile('/data.json')).toSucceedAndSatisfy((file) => {
96
- expect(file.contentType).toBe('application/json');
97
- });
98
-
99
- expect(fileTree.getFile('/script.js')).toSucceedAndSatisfy((file) => {
100
- expect(file.contentType).toBe('application/javascript');
101
- });
102
-
103
- expect(fileTree.getFile('/style.css')).toSucceedAndSatisfy((file) => {
104
- expect(file.contentType).toBe('text/css');
105
- });
106
-
107
- expect(fileTree.getFile('/readme.txt')).toSucceedAndSatisfy((file) => {
108
- expect(file.contentType).toBe('text/plain');
109
- });
110
-
111
- expect(fileTree.getFile('/image.png')).toSucceedAndSatisfy((file) => {
112
- expect(file.contentType).toBe('image/png');
113
- });
114
-
115
- expect(fileTree.getFile('/document.pdf')).toSucceedAndSatisfy((file) => {
116
- expect(file.contentType).toBe('application/pdf');
117
- });
118
-
119
- expect(fileTree.getFile('/archive.zip')).toSucceedAndSatisfy((file) => {
120
- expect(file.contentType).toBe('application/zip');
121
- });
122
- });
123
- });
124
-
125
- test('handles files without MIME types', async () => {
126
- const fileList = createMockFileList([
127
- { name: 'no-type.txt', content: 'content without type' },
128
- { name: 'empty-type.dat', content: 'empty type', type: '' },
129
- { name: 'unknown.ext', content: 'unknown extension' }
130
- ]);
131
-
132
- const result = await FileTreeHelpers.fromFileList(fileList);
133
- expect(result).toSucceedAndSatisfy((fileTree) => {
134
- // Files without MIME types should have undefined contentType
135
- expect(fileTree.getFile('/no-type.txt')).toSucceedAndSatisfy((file) => {
136
- expect(file.contentType).toBe('text/plain'); // Default from createMockFile
137
- });
138
-
139
- expect(fileTree.getFile('/empty-type.dat')).toSucceedAndSatisfy((file) => {
140
- expect(file.contentType).toBe('text/plain'); // Browser defaults empty type to text/plain
141
- });
142
-
143
- expect(fileTree.getFile('/unknown.ext')).toSucceedAndSatisfy((file) => {
144
- expect(file.contentType).toBe('text/plain'); // Default from createMockFile
145
- });
146
- });
147
- });
148
-
149
- test('forces content type to string (not templated)', async () => {
150
- const fileList = createMockFileList([
151
- { name: 'test.json', content: '{"test": true}', type: 'application/json' }
152
- ]);
153
-
154
- const result = await FileTreeHelpers.fromFileList(fileList);
155
- expect(result).toSucceedAndSatisfy((fileTree) => {
156
- // Verify the return type is FileTree<string> not FileTree<T>
157
- expect(fileTree.getFile('/test.json')).toSucceedAndSatisfy((file) => {
158
- expect(typeof file.contentType).toBe('string');
159
- expect(file.contentType).toBe('application/json');
160
- });
161
- });
162
- });
163
-
164
- test('propagates file reading errors', async () => {
165
- const badFile = {
166
- name: 'bad.txt',
167
- size: 10,
168
- type: 'text/plain',
169
- lastModified: Date.now(),
170
- text: () => Promise.reject(new Error('Read failed'))
171
- } as unknown as File;
172
-
173
- const fileList = createMockFileList([{ name: 'good.txt', content: 'good content' }]);
174
-
175
- // Create a FileList with just the bad file
176
- const dt = new DataTransfer();
177
- dt.items.add(badFile);
178
- const badFileList = dt.files;
179
-
180
- const result = await FileTreeHelpers.fromFileList(badFileList);
181
- expect(result).toFailWith(/Failed to read file/);
182
- });
183
-
184
- test('maintains file accessibility through FileTree API', async () => {
185
- const fileList = createMockFileList([{ name: 'test.json', content: '{"name": "test", "value": 42}' }]);
186
-
187
- const result = await FileTreeHelpers.fromFileList(fileList);
188
- expect(result).toSucceedAndSatisfy((fileTree) => {
189
- const file = fileTree.getFile('/test.json');
190
- expect(file).toSucceedAndSatisfy((fileItem) => {
191
- // Test JSON parsing through FileTree
192
- const jsonResult = fileItem.getContents();
193
- expect(jsonResult).toSucceedAndSatisfy((parsed) => {
194
- expect(parsed).toEqual({ name: 'test', value: 42 });
195
- });
196
-
197
- // Test raw content access
198
- const rawResult = fileItem.getRawContents();
199
- expect(rawResult).toSucceedWith('{"name": "test", "value": 42}');
200
-
201
- // Test file properties
202
- expect(fileItem.name).toBe('test.json');
203
- expect(fileItem.extension).toBe('.json');
204
- expect(fileItem.baseName).toBe('test');
205
- });
206
- });
207
- });
208
- });
209
-
210
- describe('fromDirectoryUpload', () => {
211
- test('creates FileTree from directory structure', async () => {
212
- const fileList = createMockDirectoryFileList([
213
- { path: 'project/src/index.js', content: 'console.log("main");' },
214
- { path: 'project/src/utils.js', content: 'export const helper = () => {};' },
215
- { path: 'project/package.json', content: '{"name": "test-project"}' },
216
- { path: 'project/README.md', content: '# Test Project' }
217
- ]);
218
-
219
- const result = await FileTreeHelpers.fromDirectoryUpload(fileList);
220
- expect(result).toSucceedAndSatisfy((fileTree) => {
221
- // Verify all files are accessible
222
- expect(fileTree.getFile('/project/src/index.js')).toSucceed();
223
- expect(fileTree.getFile('/project/src/utils.js')).toSucceed();
224
- expect(fileTree.getFile('/project/package.json')).toSucceed();
225
- expect(fileTree.getFile('/project/README.md')).toSucceed();
226
-
227
- // Verify directory structure
228
- expect(fileTree.getDirectory('/project')).toSucceed();
229
- expect(fileTree.getDirectory('/project/src')).toSucceed();
230
-
231
- // Verify content retrieval
232
- const packageContent = fileTree.hal.getFileContents('/project/package.json');
233
- expect(packageContent).toSucceedWith('{"name": "test-project"}');
234
- });
235
- });
236
-
237
- test('creates FileTree from directory with prefix', async () => {
238
- const fileList = createMockDirectoryFileList([
239
- { path: 'app/config.json', content: '{"setting": "value"}' },
240
- { path: 'app/main.js', content: 'console.log("app");' }
241
- ]);
242
-
243
- const result = await FileTreeHelpers.fromDirectoryUpload(fileList, { prefix: '/upload' });
244
- expect(result).toSucceed();
245
- });
246
-
247
- test('preserves directory hierarchy', async () => {
248
- const fileList = createMockDirectoryFileList([
249
- { path: 'deep/nested/structure/file1.txt', content: 'content1' },
250
- { path: 'deep/nested/structure/file2.txt', content: 'content2' },
251
- { path: 'deep/other/branch/file3.txt', content: 'content3' },
252
- { path: 'deep/file4.txt', content: 'content4' }
253
- ]);
254
-
255
- const result = await FileTreeHelpers.fromDirectoryUpload(fileList);
256
- expect(result).toSucceedAndSatisfy((fileTree) => {
257
- // Test all directory levels
258
- expect(fileTree.getDirectory('/deep')).toSucceed();
259
- expect(fileTree.getDirectory('/deep/nested')).toSucceed();
260
- expect(fileTree.getDirectory('/deep/nested/structure')).toSucceed();
261
- expect(fileTree.getDirectory('/deep/other')).toSucceed();
262
- expect(fileTree.getDirectory('/deep/other/branch')).toSucceed();
263
-
264
- // Test directory traversal
265
- const deepDir = fileTree.getDirectory('/deep');
266
- expect(deepDir).toSucceedAndSatisfy((dir) => {
267
- const children = dir.getChildren();
268
- expect(children).toSucceedAndSatisfy((items) => {
269
- const names = items.map((item) => item.name).sort();
270
- expect(names).toContain('nested');
271
- expect(names).toContain('other');
272
- expect(names).toContain('file4.txt');
273
- });
274
- });
275
- });
276
- });
277
-
278
- test('handles webkitRelativePath correctly', async () => {
279
- // Create files with explicit webkitRelativePath
280
- const files = [
281
- createMockFile({
282
- name: 'index.html',
283
- content: '<html></html>',
284
- webkitRelativePath: 'website/index.html'
285
- }),
286
- createMockFile({
287
- name: 'style.css',
288
- content: 'body {}',
289
- webkitRelativePath: 'website/css/style.css'
290
- })
291
- ];
292
-
293
- // Create FileList manually to preserve webkitRelativePath
294
- const dt = new DataTransfer();
295
- files.forEach((file) => dt.items.add(file));
296
- const fileList = dt.files;
297
-
298
- const result = await FileTreeHelpers.fromDirectoryUpload(fileList);
299
- expect(result).toSucceedAndSatisfy((fileTree) => {
300
- expect(fileTree.getFile('/website/index.html')).toSucceed();
301
- expect(fileTree.getFile('/website/css/style.css')).toSucceed();
302
- expect(fileTree.getDirectory('/website')).toSucceed();
303
- expect(fileTree.getDirectory('/website/css')).toSucceed();
304
- });
305
- });
306
-
307
- test('handles single file in directory', async () => {
308
- const fileList = createMockDirectoryFileList([{ path: 'single/file.txt', content: 'lone file' }]);
309
-
310
- const result = await FileTreeHelpers.fromDirectoryUpload(fileList);
311
- expect(result).toSucceedAndSatisfy((fileTree) => {
312
- expect(fileTree.getFile('/single/file.txt')).toSucceed();
313
- expect(fileTree.getDirectory('/single')).toSucceed();
314
- });
315
- });
316
-
317
- test('handles empty directory upload', async () => {
318
- const fileList = createMockDirectoryFileList([]);
319
- const result = await FileTreeHelpers.fromDirectoryUpload(fileList);
320
- expect(result).toSucceed();
321
- });
322
-
323
- test('automatically applies MIME types as contentType in directory upload', async () => {
324
- const fileList = createMockDirectoryFileList([
325
- { path: 'project/package.json', content: '{"name": "test"}', type: 'application/json' },
326
- { path: 'project/src/index.js', content: 'console.log("main");', type: 'application/javascript' },
327
- { path: 'project/src/styles.css', content: 'body { color: red; }', type: 'text/css' },
328
- { path: 'project/README.md', content: '# Project', type: 'text/markdown' },
329
- { path: 'project/data.xml', content: '<root></root>', type: 'application/xml' },
330
- { path: 'project/assets/logo.svg', content: '<svg></svg>', type: 'image/svg+xml' },
331
- { path: 'project/docs/manual.pdf', content: 'PDF content', type: 'application/pdf' }
332
- ]);
333
-
334
- const result = await FileTreeHelpers.fromDirectoryUpload(fileList);
335
- expect(result).toSucceedAndSatisfy((fileTree) => {
336
- // Verify MIME types are automatically applied as contentType
337
- expect(fileTree.getFile('/project/package.json')).toSucceedAndSatisfy((file) => {
338
- expect(file.contentType).toBe('application/json');
339
- });
340
-
341
- expect(fileTree.getFile('/project/src/index.js')).toSucceedAndSatisfy((file) => {
342
- expect(file.contentType).toBe('application/javascript');
343
- });
344
-
345
- expect(fileTree.getFile('/project/src/styles.css')).toSucceedAndSatisfy((file) => {
346
- expect(file.contentType).toBe('text/css');
347
- });
348
-
349
- expect(fileTree.getFile('/project/README.md')).toSucceedAndSatisfy((file) => {
350
- expect(file.contentType).toBe('text/markdown');
351
- });
352
-
353
- expect(fileTree.getFile('/project/data.xml')).toSucceedAndSatisfy((file) => {
354
- expect(file.contentType).toBe('application/xml');
355
- });
356
-
357
- expect(fileTree.getFile('/project/assets/logo.svg')).toSucceedAndSatisfy((file) => {
358
- expect(file.contentType).toBe('image/svg+xml');
359
- });
360
-
361
- expect(fileTree.getFile('/project/docs/manual.pdf')).toSucceedAndSatisfy((file) => {
362
- expect(file.contentType).toBe('application/pdf');
363
- });
364
- });
365
- });
366
-
367
- test('handles mixed MIME types and missing types in directory upload', async () => {
368
- const fileList = createMockDirectoryFileList([
369
- { path: 'mixed/typed.json', content: '{}', type: 'application/json' },
370
- { path: 'mixed/untyped.txt', content: 'plain text' }, // No type specified
371
- { path: 'mixed/empty-type.dat', content: 'data', type: '' }, // Empty type
372
- { path: 'mixed/binary.bin', content: 'binary data', type: 'application/octet-stream' }
373
- ]);
374
-
375
- const result = await FileTreeHelpers.fromDirectoryUpload(fileList);
376
- expect(result).toSucceedAndSatisfy((fileTree) => {
377
- expect(fileTree.getFile('/mixed/typed.json')).toSucceedAndSatisfy((file) => {
378
- expect(file.contentType).toBe('application/json');
379
- });
380
-
381
- expect(fileTree.getFile('/mixed/untyped.txt')).toSucceedAndSatisfy((file) => {
382
- expect(file.contentType).toBe('text/plain'); // Default from createMockFile
383
- });
384
-
385
- expect(fileTree.getFile('/mixed/empty-type.dat')).toSucceedAndSatisfy((file) => {
386
- expect(file.contentType).toBe('text/plain'); // Browser defaults empty type to text/plain
387
- });
388
-
389
- expect(fileTree.getFile('/mixed/binary.bin')).toSucceedAndSatisfy((file) => {
390
- expect(file.contentType).toBe('application/octet-stream');
391
- });
392
- });
393
- });
394
-
395
- test('forces content type to string for directory upload (not templated)', async () => {
396
- const fileList = createMockDirectoryFileList([
397
- { path: 'app/config.json', content: '{"app": "test"}', type: 'application/json' }
398
- ]);
399
-
400
- const result = await FileTreeHelpers.fromDirectoryUpload(fileList);
401
- expect(result).toSucceedAndSatisfy((fileTree) => {
402
- // Verify the return type is FileTree<string> not FileTree<T>
403
- expect(fileTree.getFile('/app/config.json')).toSucceedAndSatisfy((file) => {
404
- expect(typeof file.contentType).toBe('string');
405
- expect(file.contentType).toBe('application/json');
406
- });
407
- });
408
- });
409
- });
410
-
411
- // Note: fromFileApiFiles method was deprecated and removed.
412
- // The functionality has been replaced by the create() method in FileApiTreeAccessors
413
- // which uses TreeInitializer[] instead of IFileApiFile[].
414
-
415
- describe('getOriginalFile', () => {
416
- test('retrieves original File object by path', () => {
417
- const fileList = createMockFileList([
418
- { name: 'target.txt', content: 'target content', type: 'text/plain' },
419
- { name: 'other.txt', content: 'other content', type: 'text/plain' }
420
- ]);
421
-
422
- const result = FileTreeHelpers.getOriginalFile(fileList, 'target.txt');
423
- expect(result).toSucceedAndSatisfy((file) => {
424
- expect(file.name).toBe('target.txt');
425
- expect(file.type).toBe('text/plain');
426
- expect(file.size).toBeGreaterThan(0);
427
- });
428
- });
429
-
430
- test('handles webkitRelativePath matching', () => {
431
- const fileList = createMockDirectoryFileList([
432
- { path: 'folder/subfolder/file.txt', content: 'content', type: 'text/plain' }
433
- ]);
434
-
435
- const result = FileTreeHelpers.getOriginalFile(fileList, 'folder/subfolder/file.txt');
436
- expect(result).toSucceedAndSatisfy((file) => {
437
- expect(file.name).toBe('file.txt');
438
- expect((file as any).webkitRelativePath).toBe('folder/subfolder/file.txt');
439
- });
440
- });
441
-
442
- test('fails gracefully for non-existent files', () => {
443
- const fileList = createMockFileList([{ name: 'exists.txt', content: 'content' }]);
444
-
445
- const result = FileTreeHelpers.getOriginalFile(fileList, 'missing.txt');
446
- expect(result).toFailWith(/File not found: missing\.txt/);
447
- });
448
-
449
- test('handles empty FileList', () => {
450
- const fileList = createMockFileList([]);
451
- const result = FileTreeHelpers.getOriginalFile(fileList, 'any.txt');
452
- expect(result).toFailWith(/File not found/);
453
- });
454
-
455
- test('prioritizes exact name match over webkitRelativePath', () => {
456
- // Create files with both regular names and webkitRelativePath
457
- const files = [
458
- createMockFile({ name: 'file.txt', content: 'regular file' }),
459
- createMockFile({
460
- name: 'other.txt',
461
- content: 'directory file',
462
- webkitRelativePath: 'file.txt' // Same as first file's name
463
- })
464
- ];
465
-
466
- const dt = new DataTransfer();
467
- files.forEach((file) => dt.items.add(file));
468
- const fileList = dt.files;
469
-
470
- const result = FileTreeHelpers.getOriginalFile(fileList, 'file.txt');
471
- expect(result).toSucceedAndSatisfy((file) => {
472
- expect(file.name).toBe('file.txt');
473
- expect(file.size).toBeGreaterThan(0);
474
- });
475
- });
476
- });
477
-
478
- describe('extractFileListMetadata', () => {
479
- test('extracts complete metadata from FileList', () => {
480
- const testTime = Date.now();
481
- const fileList = createMockFileList([
482
- {
483
- name: 'document.pdf',
484
- content: 'PDF content here',
485
- type: 'application/pdf',
486
- lastModified: testTime
487
- },
488
- {
489
- name: 'image.png',
490
- content: 'PNG data',
491
- type: 'image/png',
492
- lastModified: testTime + 1000
493
- }
494
- ]);
495
-
496
- const metadata = FileTreeHelpers.extractFileListMetadata(fileList);
497
- expect(metadata).toHaveLength(2);
498
-
499
- const pdfMeta = metadata.find((m) => m.name === 'document.pdf');
500
- expect(pdfMeta).toEqual({
501
- path: 'document.pdf',
502
- name: 'document.pdf',
503
- size: expect.any(Number),
504
- type: 'application/pdf',
505
- lastModified: testTime
506
- });
507
-
508
- const pngMeta = metadata.find((m) => m.name === 'image.png');
509
- expect(pngMeta).toEqual({
510
- path: 'image.png',
511
- name: 'image.png',
512
- size: expect.any(Number),
513
- type: 'image/png',
514
- lastModified: testTime + 1000
515
- });
516
- });
517
-
518
- test('includes webkitRelativePath in path field', () => {
519
- const fileList = createMockDirectoryFileList([
520
- { path: 'project/src/main.js', content: 'main code', type: 'application/javascript' },
521
- { path: 'project/assets/logo.png', content: 'logo data', type: 'image/png' }
522
- ]);
523
-
524
- const metadata = FileTreeHelpers.extractFileListMetadata(fileList);
525
- expect(metadata).toHaveLength(2);
526
-
527
- expect(metadata.find((m) => m.name === 'main.js')?.path).toBe('project/src/main.js');
528
- expect(metadata.find((m) => m.name === 'logo.png')?.path).toBe('project/assets/logo.png');
529
- });
530
-
531
- test('handles various file sizes correctly', () => {
532
- const fileList = createMockFileList([
533
- { name: 'tiny.txt', content: '' },
534
- { name: 'small.txt', content: 'small' },
535
- { name: 'large.txt', content: 'x'.repeat(10000) }
536
- ]);
537
-
538
- const metadata = FileTreeHelpers.extractFileListMetadata(fileList);
539
- const sizes = metadata.map((m) => m.size);
540
-
541
- expect(sizes[0]).toBe(0); // empty file
542
- expect(sizes[1]).toBeGreaterThan(0); // small file
543
- expect(sizes[2]).toBeGreaterThan(sizes[1]); // large file
544
- });
545
-
546
- test('returns empty array for empty FileList', () => {
547
- const fileList = createMockFileList([]);
548
- const metadata = FileTreeHelpers.extractFileListMetadata(fileList);
549
- expect(metadata).toEqual([]);
550
- });
551
-
552
- test('preserves file order', () => {
553
- const expectedOrder = ['first.txt', 'second.txt', 'third.txt'];
554
- const fileList = createMockFileList(expectedOrder.map((name) => ({ name, content: 'content' })));
555
-
556
- const metadata = FileTreeHelpers.extractFileListMetadata(fileList);
557
- const actualOrder = metadata.map((m) => m.name);
558
- expect(actualOrder).toEqual(expectedOrder);
559
- });
560
- });
561
-
562
- describe('extractFileMetadata', () => {
563
- test('extracts metadata from single file', () => {
564
- const testTime = Date.now();
565
- const file = createMockFile({
566
- name: 'single.txt',
567
- content: 'single file content',
568
- type: 'text/plain',
569
- lastModified: testTime
570
- });
571
-
572
- const metadata = FileTreeHelpers.extractFileMetadata(file);
573
- expect(metadata).toEqual({
574
- path: 'single.txt',
575
- name: 'single.txt',
576
- size: expect.any(Number),
577
- type: 'text/plain',
578
- lastModified: testTime
579
- });
580
- });
581
-
582
- test('handles webkitRelativePath for single file', () => {
583
- const file = createMockFile({
584
- name: 'nested.txt',
585
- content: 'nested content',
586
- type: 'text/plain',
587
- webkitRelativePath: 'folder/nested.txt'
588
- });
589
-
590
- const metadata = FileTreeHelpers.extractFileMetadata(file);
591
- expect(metadata.path).toBe('folder/nested.txt');
592
- expect(metadata.name).toBe('nested.txt');
593
- });
594
- });
595
-
596
- describe('Error propagation', () => {
597
- test('propagates errors from FileApiTreeAccessors.fromFileList', async () => {
598
- const badFile = {
599
- name: 'bad.txt',
600
- size: 10,
601
- type: 'text/plain',
602
- lastModified: Date.now(),
603
- text: () => Promise.reject(new Error('File system error'))
604
- } as unknown as File;
605
-
606
- // Create a FileList with just the bad file
607
- const dt = new DataTransfer();
608
- dt.items.add(badFile);
609
- const badFileList = dt.files;
610
-
611
- const result = await FileTreeHelpers.fromFileList(badFileList);
612
- expect(result).toFailWith(/Failed to read file/);
613
- });
614
-
615
- test('propagates errors from FileApiTreeAccessors.fromDirectoryUpload', async () => {
616
- const badFile = {
617
- name: 'bad.txt',
618
- size: 10,
619
- type: 'text/plain',
620
- lastModified: Date.now(),
621
- webkitRelativePath: 'folder/bad.txt',
622
- text: () => Promise.reject(new Error('Directory read error'))
623
- } as unknown as File;
624
-
625
- const dt = new DataTransfer();
626
- dt.items.add(badFile);
627
- const fileList = dt.files;
628
-
629
- const result = await FileTreeHelpers.fromDirectoryUpload(fileList);
630
- expect(result).toFailWith(/Failed to read file/);
631
- });
632
-
633
- test('propagates errors from FileApiTreeAccessors.create via fromFileList', async () => {
634
- const badFile = {
635
- name: 'bad.txt',
636
- size: 10,
637
- type: 'text/plain',
638
- lastModified: Date.now(),
639
- text: () => Promise.reject(new Error('API error'))
640
- } as unknown as File;
641
-
642
- // Create FileList with the bad file
643
- const dt = new DataTransfer();
644
- dt.items.add(badFile);
645
- const fileList = dt.files;
646
-
647
- const result = await FileTreeHelpers.fromFileList(fileList);
648
- expect(result).toFailWith(/Failed to read file/);
649
- });
650
- });
651
-
652
- describe('Integration scenarios', () => {
653
- test('handles realistic file upload scenario', async () => {
654
- const fileList = createMockFileList([
655
- { name: 'photo1.jpg', content: 'JPEG data 1', type: 'image/jpeg' },
656
- { name: 'photo2.jpg', content: 'JPEG data 2', type: 'image/jpeg' },
657
- { name: 'metadata.txt', content: 'Photo metadata', type: 'text/plain' }
658
- ]);
659
-
660
- const result = await FileTreeHelpers.fromFileList(fileList, { prefix: '/uploads' });
661
- expect(result).toSucceed();
662
- });
663
-
664
- test('handles realistic project upload scenario', async () => {
665
- const fileList = createMockDirectoryFileList([
666
- { path: 'my-app/package.json', content: '{"name": "my-app"}' },
667
- { path: 'my-app/src/index.js', content: 'console.log("Hello");' },
668
- { path: 'my-app/src/components/App.jsx', content: 'export default App;' },
669
- { path: 'my-app/public/index.html', content: '<html></html>' },
670
- { path: 'my-app/README.md', content: '# My App' }
671
- ]);
672
-
673
- const result = await FileTreeHelpers.fromDirectoryUpload(fileList);
674
- expect(result).toSucceedAndSatisfy((fileTree) => {
675
- // Verify complete project structure
676
- expect(fileTree.getDirectory('/my-app')).toSucceed();
677
- expect(fileTree.getDirectory('/my-app/src')).toSucceed();
678
- expect(fileTree.getDirectory('/my-app/src/components')).toSucceed();
679
- expect(fileTree.getDirectory('/my-app/public')).toSucceed();
680
-
681
- // Verify all files
682
- expect(fileTree.getFile('/my-app/package.json')).toSucceed();
683
- expect(fileTree.getFile('/my-app/src/index.js')).toSucceed();
684
- expect(fileTree.getFile('/my-app/src/components/App.jsx')).toSucceed();
685
- expect(fileTree.getFile('/my-app/public/index.html')).toSucceed();
686
- expect(fileTree.getFile('/my-app/README.md')).toSucceed();
687
-
688
- // Verify content access
689
- const packageContent = fileTree.hal.getFileContents('/my-app/package.json');
690
- expect(packageContent).toSucceedWith('{"name": "my-app"}');
691
- });
692
- });
693
- });
694
- });