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

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,376 +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 { BrowserHashProvider, BrowserHashingNormalizer } from '../../packlets/crypto';
25
-
26
- describe('BrowserHashProvider', () => {
27
- describe('Static factory methods', () => {
28
- test('createSha256Normalizer returns BrowserHashingNormalizer with SHA-256', () => {
29
- const normalizer = BrowserHashProvider.createSha256Normalizer();
30
- expect(normalizer).toBeInstanceOf(BrowserHashingNormalizer);
31
- expect(normalizer.algorithm).toBe('SHA-256');
32
- });
33
-
34
- test('createSha1Normalizer returns BrowserHashingNormalizer with SHA-1', () => {
35
- const normalizer = BrowserHashProvider.createSha1Normalizer();
36
- expect(normalizer).toBeInstanceOf(BrowserHashingNormalizer);
37
- expect(normalizer.algorithm).toBe('SHA-1');
38
- });
39
-
40
- test('createSha512Normalizer returns BrowserHashingNormalizer with SHA-512', () => {
41
- const normalizer = BrowserHashProvider.createSha512Normalizer();
42
- expect(normalizer).toBeInstanceOf(BrowserHashingNormalizer);
43
- expect(normalizer.algorithm).toBe('SHA-512');
44
- });
45
- });
46
-
47
- describe('hashString', () => {
48
- test('successfully hashes a string with SHA-256', async () => {
49
- const result = await BrowserHashProvider.hashString('test data', 'SHA-256');
50
- expect(result).toSucceedAndSatisfy(hash => {
51
- expect(typeof hash).toBe('string');
52
- expect(hash).toMatch(/^[a-f0-9]{64}$/); // SHA-256 produces 64 hex characters
53
- });
54
- });
55
-
56
- test('successfully hashes a string with SHA-1', async () => {
57
- const result = await BrowserHashProvider.hashString('test data', 'SHA-1');
58
- expect(result).toSucceedAndSatisfy(hash => {
59
- expect(typeof hash).toBe('string');
60
- expect(hash).toMatch(/^[a-f0-9]{40}$/); // SHA-1 produces 40 hex characters
61
- });
62
- });
63
-
64
- test('successfully hashes a string with SHA-512', async () => {
65
- const result = await BrowserHashProvider.hashString('test data', 'SHA-512');
66
- expect(result).toSucceedAndSatisfy(hash => {
67
- expect(typeof hash).toBe('string');
68
- expect(hash).toMatch(/^[a-f0-9]{128}$/); // SHA-512 produces 128 hex characters
69
- });
70
- });
71
-
72
- test('uses SHA-256 as default algorithm', async () => {
73
- const result = await BrowserHashProvider.hashString('test data');
74
- expect(result).toSucceedAndSatisfy(hash => {
75
- expect(hash).toMatch(/^[a-f0-9]{64}$/); // SHA-256 produces 64 hex characters
76
- });
77
- });
78
-
79
- test('produces consistent hash for same input', async () => {
80
- const input = 'consistent input';
81
- const result1 = await BrowserHashProvider.hashString(input);
82
- const result2 = await BrowserHashProvider.hashString(input);
83
-
84
- expect(result1).toSucceed();
85
- expect(result2).toSucceed();
86
- if (result1.isSuccess() && result2.isSuccess()) {
87
- expect(result1.value).toBe(result2.value);
88
- }
89
- });
90
-
91
- test('produces different hashes for different inputs', async () => {
92
- const result1 = await BrowserHashProvider.hashString('input1');
93
- const result2 = await BrowserHashProvider.hashString('input2');
94
-
95
- expect(result1).toSucceed();
96
- expect(result2).toSucceed();
97
- if (result1.isSuccess() && result2.isSuccess()) {
98
- expect(result1.value).not.toBe(result2.value);
99
- }
100
- });
101
-
102
- test('handles empty string input', async () => {
103
- const result = await BrowserHashProvider.hashString('');
104
- expect(result).toSucceed();
105
- });
106
-
107
- test('handles unicode input', async () => {
108
- const result = await BrowserHashProvider.hashString('🚀 Unicode test 测试');
109
- expect(result).toSucceedAndSatisfy(hash => {
110
- expect(hash).toMatch(/^[a-f0-9]{64}$/);
111
- });
112
- });
113
-
114
- test('fails when crypto.subtle is unavailable', async () => {
115
- const originalCrypto = global.crypto;
116
- // @ts-ignore - Intentionally removing crypto for test
117
- delete global.crypto;
118
-
119
- const result = await BrowserHashProvider.hashString('test');
120
- expect(result).toFailWith(/Hash computation failed/);
121
-
122
- global.crypto = originalCrypto;
123
- });
124
-
125
- test('handles invalid algorithm gracefully', async () => {
126
- const result = await BrowserHashProvider.hashString('test', 'INVALID-ALG');
127
- expect(result).toFailWith(/Hash computation failed/);
128
- });
129
- });
130
-
131
- describe('hashParts', () => {
132
- test('hashes multiple parts with default separator', async () => {
133
- const parts = ['part1', 'part2', 'part3'];
134
- const result = await BrowserHashProvider.hashParts(parts);
135
- expect(result).toSucceedAndSatisfy(hash => {
136
- expect(hash).toMatch(/^[a-f0-9]{64}$/);
137
- });
138
- });
139
-
140
- test('hashes multiple parts with custom separator', async () => {
141
- const parts = ['part1', 'part2', 'part3'];
142
- const result1 = await BrowserHashProvider.hashParts(parts, 'SHA-256', '|');
143
- const result2 = await BrowserHashProvider.hashParts(parts, 'SHA-256', ',');
144
-
145
- expect(result1).toSucceed();
146
- expect(result2).toSucceed();
147
- if (result1.isSuccess() && result2.isSuccess()) {
148
- expect(result1.value).not.toBe(result2.value);
149
- }
150
- });
151
-
152
- test('produces same hash as hashString for joined parts', async () => {
153
- const parts = ['a', 'b', 'c'];
154
- const separator = '-';
155
- const joined = parts.join(separator);
156
-
157
- const partsResult = await BrowserHashProvider.hashParts(parts, 'SHA-256', separator);
158
- const stringResult = await BrowserHashProvider.hashString(joined);
159
-
160
- expect(partsResult).toSucceed();
161
- expect(stringResult).toSucceed();
162
- if (partsResult.isSuccess() && stringResult.isSuccess()) {
163
- expect(partsResult.value).toBe(stringResult.value);
164
- }
165
- });
166
-
167
- test('handles empty array', async () => {
168
- const result = await BrowserHashProvider.hashParts([]);
169
- expect(result).toSucceed();
170
- });
171
-
172
- test('handles single part', async () => {
173
- const result = await BrowserHashProvider.hashParts(['single']);
174
- const stringResult = await BrowserHashProvider.hashString('single');
175
-
176
- expect(result).toSucceed();
177
- expect(stringResult).toSucceed();
178
- if (result.isSuccess() && stringResult.isSuccess()) {
179
- expect(result.value).toBe(stringResult.value);
180
- }
181
- });
182
- });
183
- });
184
-
185
- describe('BrowserHashingNormalizer', () => {
186
- describe('constructor', () => {
187
- test('creates instance with default SHA-256 algorithm', () => {
188
- const normalizer = new BrowserHashingNormalizer();
189
- expect(normalizer).toBeInstanceOf(BrowserHashingNormalizer);
190
- expect(normalizer.algorithm).toBe('SHA-256');
191
- });
192
-
193
- test('creates instance with custom algorithm', () => {
194
- const normalizer = new BrowserHashingNormalizer('SHA-512');
195
- expect(normalizer.algorithm).toBe('SHA-512');
196
- });
197
- });
198
-
199
- describe('hashAsync', () => {
200
- test('hashes array of strings', async () => {
201
- const normalizer = new BrowserHashingNormalizer();
202
- const result = await normalizer.hashAsync(['test1', 'test2']);
203
- expect(result).toSucceedAndSatisfy(hash => {
204
- expect(hash).toMatch(/^[a-f0-9]{64}$/);
205
- });
206
- });
207
-
208
- test('produces consistent results', async () => {
209
- const normalizer = new BrowserHashingNormalizer();
210
- const parts = ['consistent', 'parts'];
211
-
212
- const result1 = await normalizer.hashAsync(parts);
213
- const result2 = await normalizer.hashAsync(parts);
214
-
215
- expect(result1).toSucceed();
216
- expect(result2).toSucceed();
217
- if (result1.isSuccess() && result2.isSuccess()) {
218
- expect(result1.value).toBe(result2.value);
219
- }
220
- });
221
- });
222
-
223
- describe('normalizeAsync', () => {
224
- test('normalizes primitive values', async () => {
225
- const normalizer = new BrowserHashingNormalizer();
226
-
227
- const stringResult = await normalizer.normalizeAsync('test');
228
- expect(stringResult).toSucceed();
229
-
230
- const numberResult = await normalizer.normalizeAsync(42);
231
- expect(numberResult).toSucceed();
232
-
233
- const booleanResult = await normalizer.normalizeAsync(true);
234
- expect(booleanResult).toSucceed();
235
-
236
- const nullResult = await normalizer.normalizeAsync(null);
237
- expect(nullResult).toSucceed();
238
-
239
- const undefinedResult = await normalizer.normalizeAsync(undefined);
240
- expect(undefinedResult).toSucceed();
241
- });
242
-
243
- test('normalizes arrays', async () => {
244
- const normalizer = new BrowserHashingNormalizer();
245
- const array = ['item1', 'item2', 42, true];
246
-
247
- const result = await normalizer.normalizeAsync(array);
248
- expect(result).toSucceedAndSatisfy(hash => {
249
- expect(hash).toMatch(/^[a-f0-9]{64}$/);
250
- });
251
- });
252
-
253
- test('normalizes nested arrays', async () => {
254
- const normalizer = new BrowserHashingNormalizer();
255
- const nestedArray = ['item1', ['nested1', 'nested2'], 'item3'];
256
-
257
- const result = await normalizer.normalizeAsync(nestedArray);
258
- expect(result).toSucceed();
259
- });
260
-
261
- test('normalizes objects', async () => {
262
- const normalizer = new BrowserHashingNormalizer();
263
- const obj = {
264
- name: 'test',
265
- value: 42,
266
- nested: { key: 'value' }
267
- };
268
-
269
- const result = await normalizer.normalizeAsync(obj);
270
- expect(result).toSucceedAndSatisfy(hash => {
271
- expect(hash).toMatch(/^[a-f0-9]{64}$/);
272
- });
273
- });
274
-
275
- test('normalizes objects with consistent key ordering', async () => {
276
- const normalizer = new BrowserHashingNormalizer();
277
- const obj1 = { b: 2, a: 1, c: 3 };
278
- const obj2 = { a: 1, c: 3, b: 2 };
279
-
280
- const result1 = await normalizer.normalizeAsync(obj1);
281
- const result2 = await normalizer.normalizeAsync(obj2);
282
-
283
- expect(result1).toSucceed();
284
- expect(result2).toSucceed();
285
- if (result1.isSuccess() && result2.isSuccess()) {
286
- expect(result1.value).toBe(result2.value);
287
- }
288
- });
289
-
290
- test('normalizes Date objects', async () => {
291
- const normalizer = new BrowserHashingNormalizer();
292
- const date = new Date('2025-01-01T00:00:00Z');
293
-
294
- const result = await normalizer.normalizeAsync(date);
295
- expect(result).toSucceed();
296
- });
297
-
298
- test('normalizes RegExp objects', async () => {
299
- const normalizer = new BrowserHashingNormalizer();
300
- const regex = /test.*pattern/gi;
301
-
302
- const result = await normalizer.normalizeAsync(regex);
303
- expect(result).toSucceed();
304
- });
305
-
306
- test('normalizes complex nested structures', async () => {
307
- const normalizer = new BrowserHashingNormalizer();
308
- const complex = {
309
- id: 'test-id',
310
- data: {
311
- values: [1, 2, 3],
312
- metadata: {
313
- created: new Date('2025-01-01'),
314
- tags: ['tag1', 'tag2']
315
- }
316
- },
317
- flags: {
318
- active: true,
319
- validated: false
320
- }
321
- };
322
-
323
- const result = await normalizer.normalizeAsync(complex);
324
- expect(result).toSucceed();
325
- });
326
-
327
- test('handles bigint values', async () => {
328
- const normalizer = new BrowserHashingNormalizer();
329
- const bigIntValue = BigInt(9007199254740991);
330
-
331
- const result = await normalizer.normalizeAsync(bigIntValue);
332
- expect(result).toSucceed();
333
- });
334
-
335
- test('handles symbol values', async () => {
336
- const normalizer = new BrowserHashingNormalizer();
337
- const symbolValue = Symbol('test');
338
-
339
- const result = await normalizer.normalizeAsync(symbolValue);
340
- expect(result).toSucceed();
341
- });
342
- });
343
-
344
- describe('computeHash', () => {
345
- test('throws error for synchronous operation', () => {
346
- const normalizer = new BrowserHashingNormalizer();
347
- const result = normalizer.computeHash('test');
348
-
349
- expect(result).toFailWith(/Synchronous hashing not supported in browser/);
350
- });
351
- });
352
-
353
- describe('algorithm getter', () => {
354
- test('returns the correct algorithm', () => {
355
- const sha256 = new BrowserHashingNormalizer('SHA-256');
356
- expect(sha256.algorithm).toBe('SHA-256');
357
-
358
- const sha1 = new BrowserHashingNormalizer('SHA-1');
359
- expect(sha1.algorithm).toBe('SHA-1');
360
-
361
- const sha512 = new BrowserHashingNormalizer('SHA-512');
362
- expect(sha512.algorithm).toBe('SHA-512');
363
- });
364
- });
365
-
366
- describe('_computeHashAsync', () => {
367
- test('handles unexpected types gracefully', async () => {
368
- const normalizer = new BrowserHashingNormalizer();
369
-
370
- // Test with function (which is an unexpected type)
371
- const func = () => 'test';
372
- const result = await normalizer.normalizeAsync(func as unknown);
373
- expect(result).toFailWith(/Unexpected type/);
374
- });
375
- });
376
- });
@@ -1,251 +0,0 @@
1
- /*
2
- * Copyright (c) 2026 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 {
25
- DirectoryHandleStore,
26
- DEFAULT_DIRECTORY_HANDLE_DB,
27
- DEFAULT_DIRECTORY_HANDLE_STORE
28
- } from '../../packlets/file-tree/directoryHandleStore';
29
- import type { FileSystemDirectoryHandle } from '../../packlets/file-api-types';
30
-
31
- import { get, set, del, keys, createStore } from 'idb-keyval';
32
-
33
- const mockGet = jest.mocked(get);
34
- const mockSet = jest.mocked(set);
35
- const mockDel = jest.mocked(del);
36
- const mockKeys = jest.mocked(keys);
37
- const mockCreateStore = jest.mocked(createStore);
38
-
39
- function makeMockHandle(name: string): FileSystemDirectoryHandle {
40
- return {
41
- kind: 'directory',
42
- name,
43
- isSameEntry: jest.fn(),
44
- queryPermission: jest.fn(),
45
- requestPermission: jest.fn(),
46
- getDirectoryHandle: jest.fn(),
47
- getFileHandle: jest.fn(),
48
- removeEntry: jest.fn(),
49
- resolve: jest.fn(),
50
- keys: jest.fn(),
51
- values: jest.fn(),
52
- entries: jest.fn(),
53
- [Symbol.asyncIterator]: jest.fn()
54
- } as FileSystemDirectoryHandle;
55
- }
56
-
57
- describe('DirectoryHandleStore', () => {
58
- beforeEach(() => {
59
- jest.clearAllMocks();
60
- });
61
-
62
- describe('constants', () => {
63
- test('DEFAULT_DIRECTORY_HANDLE_DB has expected value', () => {
64
- expect(DEFAULT_DIRECTORY_HANDLE_DB).toBe('chocolate-lab-storage');
65
- });
66
-
67
- test('DEFAULT_DIRECTORY_HANDLE_STORE has expected value', () => {
68
- expect(DEFAULT_DIRECTORY_HANDLE_STORE).toBe('directory-handles');
69
- });
70
- });
71
-
72
- describe('constructor', () => {
73
- test('uses default db and store names', () => {
74
- new DirectoryHandleStore();
75
- expect(mockCreateStore).toHaveBeenCalledWith(
76
- DEFAULT_DIRECTORY_HANDLE_DB,
77
- DEFAULT_DIRECTORY_HANDLE_STORE
78
- );
79
- });
80
-
81
- test('uses custom db and store names', () => {
82
- new DirectoryHandleStore('custom-db', 'custom-store');
83
- expect(mockCreateStore).toHaveBeenCalledWith('custom-db', 'custom-store');
84
- });
85
- });
86
-
87
- describe('save', () => {
88
- test('saves a handle successfully', async () => {
89
- mockSet.mockResolvedValue(undefined);
90
- const store = new DirectoryHandleStore();
91
- const handle = makeMockHandle('my-dir');
92
-
93
- const result = await store.save('my-dir', handle);
94
-
95
- expect(result).toSucceed();
96
- expect(mockSet).toHaveBeenCalledWith('my-dir', handle, expect.anything());
97
- });
98
-
99
- test('returns failure when set throws', async () => {
100
- mockSet.mockRejectedValue(new Error('IndexedDB unavailable'));
101
- const store = new DirectoryHandleStore();
102
- const handle = makeMockHandle('my-dir');
103
-
104
- const result = await store.save('my-dir', handle);
105
-
106
- expect(result).toFailWith(/IndexedDB unavailable/);
107
- });
108
- });
109
-
110
- describe('load', () => {
111
- test('returns the handle when found', async () => {
112
- const handle = makeMockHandle('my-dir');
113
- mockGet.mockResolvedValue(handle);
114
- const store = new DirectoryHandleStore();
115
-
116
- const result = await store.load('my-dir');
117
-
118
- expect(result).toSucceedWith(handle);
119
- expect(mockGet).toHaveBeenCalledWith('my-dir', expect.anything());
120
- });
121
-
122
- test('returns undefined when not found', async () => {
123
- mockGet.mockResolvedValue(undefined);
124
- const store = new DirectoryHandleStore();
125
-
126
- const result = await store.load('missing');
127
-
128
- expect(result).toSucceedWith(undefined);
129
- });
130
-
131
- test('returns failure when get throws', async () => {
132
- mockGet.mockRejectedValue(new Error('read error'));
133
- const store = new DirectoryHandleStore();
134
-
135
- const result = await store.load('my-dir');
136
-
137
- expect(result).toFailWith(/read error/);
138
- });
139
- });
140
-
141
- describe('remove', () => {
142
- test('removes a handle successfully', async () => {
143
- mockDel.mockResolvedValue(undefined);
144
- const store = new DirectoryHandleStore();
145
-
146
- const result = await store.remove('my-dir');
147
-
148
- expect(result).toSucceed();
149
- expect(mockDel).toHaveBeenCalledWith('my-dir', expect.anything());
150
- });
151
-
152
- test('returns failure when del throws', async () => {
153
- mockDel.mockRejectedValue(new Error('delete error'));
154
- const store = new DirectoryHandleStore();
155
-
156
- const result = await store.remove('my-dir');
157
-
158
- expect(result).toFailWith(/delete error/);
159
- });
160
- });
161
-
162
- describe('getAllLabels', () => {
163
- test('returns all keys', async () => {
164
- mockKeys.mockResolvedValue(['dir-a', 'dir-b']);
165
- const store = new DirectoryHandleStore();
166
-
167
- const result = await store.getAllLabels();
168
-
169
- expect(result).toSucceedWith(['dir-a', 'dir-b']);
170
- expect(mockKeys).toHaveBeenCalledWith(expect.anything());
171
- });
172
-
173
- test('returns empty array when no keys', async () => {
174
- mockKeys.mockResolvedValue([]);
175
- const store = new DirectoryHandleStore();
176
-
177
- const result = await store.getAllLabels();
178
-
179
- expect(result).toSucceedWith([]);
180
- });
181
-
182
- test('returns failure when keys throws', async () => {
183
- mockKeys.mockRejectedValue(new Error('keys error'));
184
- const store = new DirectoryHandleStore();
185
-
186
- const result = await store.getAllLabels();
187
-
188
- expect(result).toFailWith(/keys error/);
189
- });
190
- });
191
-
192
- describe('getAll', () => {
193
- test('returns all label/handle pairs', async () => {
194
- const handleA = makeMockHandle('dir-a');
195
- const handleB = makeMockHandle('dir-b');
196
- mockKeys.mockResolvedValue(['dir-a', 'dir-b']);
197
- mockGet.mockResolvedValueOnce(handleA).mockResolvedValueOnce(handleB);
198
- const store = new DirectoryHandleStore();
199
-
200
- const result = await store.getAll();
201
-
202
- expect(result).toSucceedAndSatisfy((entries) => {
203
- expect(entries).toHaveLength(2);
204
- expect(entries[0]).toEqual({ label: 'dir-a', handle: handleA });
205
- expect(entries[1]).toEqual({ label: 'dir-b', handle: handleB });
206
- });
207
- });
208
-
209
- test('returns empty array when no handles stored', async () => {
210
- mockKeys.mockResolvedValue([]);
211
- const store = new DirectoryHandleStore();
212
-
213
- const result = await store.getAll();
214
-
215
- expect(result).toSucceedWith([]);
216
- });
217
-
218
- test('skips entries where handle is undefined', async () => {
219
- const handleA = makeMockHandle('dir-a');
220
- mockKeys.mockResolvedValue(['dir-a', 'dir-b']);
221
- mockGet.mockResolvedValueOnce(handleA).mockResolvedValueOnce(undefined);
222
- const store = new DirectoryHandleStore();
223
-
224
- const result = await store.getAll();
225
-
226
- expect(result).toSucceedAndSatisfy((entries) => {
227
- expect(entries).toHaveLength(1);
228
- expect(entries[0]).toEqual({ label: 'dir-a', handle: handleA });
229
- });
230
- });
231
-
232
- test('returns failure when getAllLabels fails', async () => {
233
- mockKeys.mockRejectedValue(new Error('keys error'));
234
- const store = new DirectoryHandleStore();
235
-
236
- const result = await store.getAll();
237
-
238
- expect(result).toFailWith(/keys error/);
239
- });
240
-
241
- test('returns failure when load fails for a key', async () => {
242
- mockKeys.mockResolvedValue(['dir-a']);
243
- mockGet.mockRejectedValue(new Error('load error'));
244
- const store = new DirectoryHandleStore();
245
-
246
- const result = await store.getAll();
247
-
248
- expect(result).toFailWith(/load error/);
249
- });
250
- });
251
- });