@fgv/ts-web-extras 5.1.0-2 → 5.1.0-21

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 (263) hide show
  1. package/dist/packlets/crypto-utils/browserCryptoProvider.js +208 -18
  2. package/dist/packlets/crypto-utils/browserCryptoProvider.js.map +1 -1
  3. package/dist/packlets/file-tree/fileApiTreeAccessors.js +1 -1
  4. package/dist/packlets/file-tree/fileApiTreeAccessors.js.map +1 -1
  5. package/dist/packlets/file-tree/httpTreeAccessors.js +72 -42
  6. package/dist/packlets/file-tree/httpTreeAccessors.js.map +1 -1
  7. package/dist/ts-web-extras.d.ts +59 -7
  8. package/dist/tsdoc-metadata.json +1 -1
  9. package/lib/packlets/crypto-utils/browserCryptoProvider.d.ts +52 -5
  10. package/lib/packlets/crypto-utils/browserCryptoProvider.d.ts.map +1 -1
  11. package/lib/packlets/crypto-utils/browserCryptoProvider.js +207 -17
  12. package/lib/packlets/crypto-utils/browserCryptoProvider.js.map +1 -1
  13. package/lib/packlets/file-tree/fileApiTreeAccessors.d.ts +1 -1
  14. package/lib/packlets/file-tree/fileApiTreeAccessors.js +1 -1
  15. package/lib/packlets/file-tree/fileApiTreeAccessors.js.map +1 -1
  16. package/lib/packlets/file-tree/httpTreeAccessors.d.ts +6 -0
  17. package/lib/packlets/file-tree/httpTreeAccessors.d.ts.map +1 -1
  18. package/lib/packlets/file-tree/httpTreeAccessors.js +72 -42
  19. package/lib/packlets/file-tree/httpTreeAccessors.js.map +1 -1
  20. package/package.json +27 -26
  21. package/.rush/temp/chunked-rush-logs/ts-web-extras.build.chunks.jsonl +0 -75
  22. package/.rush/temp/chunked-rush-logs/ts-web-extras.test.chunks.jsonl +0 -75
  23. package/.rush/temp/operation/build/all.log +0 -75
  24. package/.rush/temp/operation/build/error.log +0 -18
  25. package/.rush/temp/operation/build/log-chunks.jsonl +0 -75
  26. package/.rush/temp/operation/build/state.json +0 -3
  27. package/.rush/temp/operation/test/all.log +0 -75
  28. package/.rush/temp/operation/test/error.log +0 -18
  29. package/.rush/temp/operation/test/log-chunks.jsonl +0 -75
  30. package/.rush/temp/operation/test/state.json +0 -3
  31. package/.rush/temp/shrinkwrap-deps.json +0 -635
  32. package/CHANGELOG.md +0 -23
  33. package/config/api-extractor.json +0 -343
  34. package/config/jest.config.json +0 -19
  35. package/config/rig.json +0 -16
  36. package/config/typedoc.json +0 -6
  37. package/dist/test/mocks/idb-keyval.js +0 -6
  38. package/dist/test/mocks/idb-keyval.js.map +0 -1
  39. package/dist/test/setupTests.js +0 -74
  40. package/dist/test/setupTests.js.map +0 -1
  41. package/dist/test/unit/browserHashProvider.test.js +0 -140
  42. package/dist/test/unit/browserHashProvider.test.js.map +0 -1
  43. package/dist/test/unit/directoryHandleStore.test.js +0 -190
  44. package/dist/test/unit/directoryHandleStore.test.js.map +0 -1
  45. package/dist/test/unit/fileApiTreeAccessors.test.js +0 -1188
  46. package/dist/test/unit/fileApiTreeAccessors.test.js.map +0 -1
  47. package/dist/test/unit/fileApiTypes.test.js +0 -472
  48. package/dist/test/unit/fileApiTypes.test.js.map +0 -1
  49. package/dist/test/unit/fileSystemAccessTreeAccessors.test.js +0 -622
  50. package/dist/test/unit/fileSystemAccessTreeAccessors.test.js.map +0 -1
  51. package/dist/test/unit/fileTreeHelpers.test.js +0 -590
  52. package/dist/test/unit/fileTreeHelpers.test.js.map +0 -1
  53. package/dist/test/unit/httpTreeAccessors.test.js +0 -1229
  54. package/dist/test/unit/httpTreeAccessors.test.js.map +0 -1
  55. package/dist/test/unit/localStorageTreeAccessors.test.js +0 -812
  56. package/dist/test/unit/localStorageTreeAccessors.test.js.map +0 -1
  57. package/dist/test/unit/urlParams.test.js +0 -393
  58. package/dist/test/unit/urlParams.test.js.map +0 -1
  59. package/dist/test/utils/fileSystemAccessMocks.js +0 -271
  60. package/dist/test/utils/fileSystemAccessMocks.js.map +0 -1
  61. package/dist/test/utils/testHelpers.js +0 -124
  62. package/dist/test/utils/testHelpers.js.map +0 -1
  63. package/docs/@fgv/namespaces/CryptoUtils/README.md +0 -18
  64. package/docs/@fgv/namespaces/CryptoUtils/classes/BrowserCryptoProvider.md +0 -203
  65. package/docs/@fgv/namespaces/CryptoUtils/classes/BrowserHashProvider.md +0 -63
  66. package/docs/@fgv/namespaces/CryptoUtils/functions/createBrowserCryptoProvider.md +0 -18
  67. package/docs/@fgv/namespaces/FileTreeHelpers/README.md +0 -19
  68. package/docs/@fgv/namespaces/FileTreeHelpers/functions/extractFileListMetadata.md +0 -23
  69. package/docs/@fgv/namespaces/FileTreeHelpers/functions/extractFileMetadata.md +0 -23
  70. package/docs/@fgv/namespaces/FileTreeHelpers/functions/fromDirectoryUpload.md +0 -33
  71. package/docs/@fgv/namespaces/FileTreeHelpers/functions/fromFileList.md +0 -33
  72. package/docs/@fgv/namespaces/FileTreeHelpers/functions/getOriginalFile.md +0 -25
  73. package/docs/@fgv/namespaces/FileTreeHelpers/variables/defaultFileApiTreeInitParams.md +0 -11
  74. package/docs/README.md +0 -78
  75. package/docs/classes/DirectoryHandleStore.md +0 -116
  76. package/docs/classes/FileApiTreeAccessors.md +0 -286
  77. package/docs/classes/FileSystemAccessTreeAccessors.md +0 -557
  78. package/docs/classes/HttpTreeAccessors.md +0 -508
  79. package/docs/classes/LocalStorageTreeAccessors.md +0 -520
  80. package/docs/functions/exportAsJson.md +0 -23
  81. package/docs/functions/exportUsingFileSystemAPI.md +0 -26
  82. package/docs/functions/extractDirectoryPath.md +0 -23
  83. package/docs/functions/isDirectoryHandle.md +0 -23
  84. package/docs/functions/isFileHandle.md +0 -23
  85. package/docs/functions/isFilePath.md +0 -21
  86. package/docs/functions/parseContextFilter.md +0 -22
  87. package/docs/functions/parseQualifierDefaults.md +0 -22
  88. package/docs/functions/parseResourceTypes.md +0 -22
  89. package/docs/functions/parseUrlParameters.md +0 -15
  90. package/docs/functions/safeShowDirectoryPicker.md +0 -24
  91. package/docs/functions/safeShowOpenFilePicker.md +0 -24
  92. package/docs/functions/safeShowSaveFilePicker.md +0 -24
  93. package/docs/functions/supportsFileSystemAccess.md +0 -23
  94. package/docs/interfaces/FilePickerAcceptType.md +0 -16
  95. package/docs/interfaces/FileSystemCreateWritableOptions.md +0 -15
  96. package/docs/interfaces/FileSystemDirectoryHandle.md +0 -187
  97. package/docs/interfaces/FileSystemFileHandle.md +0 -106
  98. package/docs/interfaces/FileSystemGetDirectoryOptions.md +0 -15
  99. package/docs/interfaces/FileSystemGetFileOptions.md +0 -15
  100. package/docs/interfaces/FileSystemHandle.md +0 -69
  101. package/docs/interfaces/FileSystemHandlePermissionDescriptor.md +0 -15
  102. package/docs/interfaces/FileSystemRemoveOptions.md +0 -15
  103. package/docs/interfaces/FileSystemWritableFileStream.md +0 -127
  104. package/docs/interfaces/IDirectoryHandleTreeInitializer.md +0 -17
  105. package/docs/interfaces/IFileHandleTreeInitializer.md +0 -16
  106. package/docs/interfaces/IFileListTreeInitializer.md +0 -15
  107. package/docs/interfaces/IFileMetadata.md +0 -19
  108. package/docs/interfaces/IFileSystemAccessTreeParams.md +0 -30
  109. package/docs/interfaces/IFsAccessApis.md +0 -57
  110. package/docs/interfaces/IHttpTreeParams.md +0 -32
  111. package/docs/interfaces/ILocalStorageTreeParams.md +0 -30
  112. package/docs/interfaces/IUrlConfigOptions.md +0 -27
  113. package/docs/interfaces/ShowDirectoryPickerOptions.md +0 -17
  114. package/docs/interfaces/ShowOpenFilePickerOptions.md +0 -19
  115. package/docs/interfaces/ShowSaveFilePickerOptions.md +0 -19
  116. package/docs/type-aliases/TreeInitializer.md +0 -11
  117. package/docs/type-aliases/WellKnownDirectory.md +0 -11
  118. package/docs/type-aliases/WindowWithFsAccess.md +0 -11
  119. package/docs/variables/DEFAULT_DIRECTORY_HANDLE_DB.md +0 -11
  120. package/docs/variables/DEFAULT_DIRECTORY_HANDLE_STORE.md +0 -11
  121. package/etc/ts-web-extras.api.md +0 -433
  122. package/lib/test/mocks/idb-keyval.d.ts +0 -6
  123. package/lib/test/mocks/idb-keyval.d.ts.map +0 -1
  124. package/lib/test/mocks/idb-keyval.js +0 -9
  125. package/lib/test/mocks/idb-keyval.js.map +0 -1
  126. package/lib/test/setupTests.d.ts +0 -2
  127. package/lib/test/setupTests.d.ts.map +0 -1
  128. package/lib/test/setupTests.js +0 -76
  129. package/lib/test/setupTests.js.map +0 -1
  130. package/lib/test/unit/browserHashProvider.test.d.ts +0 -2
  131. package/lib/test/unit/browserHashProvider.test.d.ts.map +0 -1
  132. package/lib/test/unit/browserHashProvider.test.js +0 -142
  133. package/lib/test/unit/browserHashProvider.test.js.map +0 -1
  134. package/lib/test/unit/directoryHandleStore.test.d.ts +0 -2
  135. package/lib/test/unit/directoryHandleStore.test.d.ts.map +0 -1
  136. package/lib/test/unit/directoryHandleStore.test.js +0 -192
  137. package/lib/test/unit/directoryHandleStore.test.js.map +0 -1
  138. package/lib/test/unit/fileApiTreeAccessors.test.d.ts +0 -2
  139. package/lib/test/unit/fileApiTreeAccessors.test.d.ts.map +0 -1
  140. package/lib/test/unit/fileApiTreeAccessors.test.js +0 -1190
  141. package/lib/test/unit/fileApiTreeAccessors.test.js.map +0 -1
  142. package/lib/test/unit/fileApiTypes.test.d.ts +0 -2
  143. package/lib/test/unit/fileApiTypes.test.d.ts.map +0 -1
  144. package/lib/test/unit/fileApiTypes.test.js +0 -474
  145. package/lib/test/unit/fileApiTypes.test.js.map +0 -1
  146. package/lib/test/unit/fileSystemAccessTreeAccessors.test.d.ts +0 -2
  147. package/lib/test/unit/fileSystemAccessTreeAccessors.test.d.ts.map +0 -1
  148. package/lib/test/unit/fileSystemAccessTreeAccessors.test.js +0 -624
  149. package/lib/test/unit/fileSystemAccessTreeAccessors.test.js.map +0 -1
  150. package/lib/test/unit/fileTreeHelpers.test.d.ts +0 -2
  151. package/lib/test/unit/fileTreeHelpers.test.d.ts.map +0 -1
  152. package/lib/test/unit/fileTreeHelpers.test.js +0 -592
  153. package/lib/test/unit/fileTreeHelpers.test.js.map +0 -1
  154. package/lib/test/unit/httpTreeAccessors.test.d.ts +0 -2
  155. package/lib/test/unit/httpTreeAccessors.test.d.ts.map +0 -1
  156. package/lib/test/unit/httpTreeAccessors.test.js +0 -1231
  157. package/lib/test/unit/httpTreeAccessors.test.js.map +0 -1
  158. package/lib/test/unit/localStorageTreeAccessors.test.d.ts +0 -2
  159. package/lib/test/unit/localStorageTreeAccessors.test.d.ts.map +0 -1
  160. package/lib/test/unit/localStorageTreeAccessors.test.js +0 -814
  161. package/lib/test/unit/localStorageTreeAccessors.test.js.map +0 -1
  162. package/lib/test/unit/urlParams.test.d.ts +0 -2
  163. package/lib/test/unit/urlParams.test.d.ts.map +0 -1
  164. package/lib/test/unit/urlParams.test.js +0 -395
  165. package/lib/test/unit/urlParams.test.js.map +0 -1
  166. package/lib/test/utils/fileSystemAccessMocks.d.ts +0 -53
  167. package/lib/test/utils/fileSystemAccessMocks.d.ts.map +0 -1
  168. package/lib/test/utils/fileSystemAccessMocks.js +0 -277
  169. package/lib/test/utils/fileSystemAccessMocks.js.map +0 -1
  170. package/lib/test/utils/testHelpers.d.ts +0 -51
  171. package/lib/test/utils/testHelpers.d.ts.map +0 -1
  172. package/lib/test/utils/testHelpers.js +0 -133
  173. package/lib/test/utils/testHelpers.js.map +0 -1
  174. package/rush-logs/ts-web-extras.build.cache.log +0 -0
  175. package/rush-logs/ts-web-extras.build.error.log +0 -18
  176. package/rush-logs/ts-web-extras.build.log +0 -75
  177. package/rush-logs/ts-web-extras.test.cache.log +0 -1
  178. package/rush-logs/ts-web-extras.test.error.log +0 -18
  179. package/rush-logs/ts-web-extras.test.log +0 -75
  180. package/src/index.browser.ts +0 -24
  181. package/src/index.ts +0 -47
  182. package/src/packlets/crypto-utils/browserCryptoProvider.ts +0 -311
  183. package/src/packlets/crypto-utils/browserHashProvider.ts +0 -73
  184. package/src/packlets/crypto-utils/index.ts +0 -29
  185. package/src/packlets/file-api-types/index.ts +0 -366
  186. package/src/packlets/file-tree/directoryHandleStore.ts +0 -136
  187. package/src/packlets/file-tree/fileApiTreeAccessors.ts +0 -528
  188. package/src/packlets/file-tree/fileSystemAccessTreeAccessors.ts +0 -519
  189. package/src/packlets/file-tree/httpTreeAccessors.ts +0 -448
  190. package/src/packlets/file-tree/index.ts +0 -32
  191. package/src/packlets/file-tree/localStorageTreeAccessors.ts +0 -430
  192. package/src/packlets/helpers/fileTreeHelpers.ts +0 -107
  193. package/src/packlets/helpers/index.ts +0 -28
  194. package/src/packlets/url-utils/index.ts +0 -28
  195. package/src/packlets/url-utils/urlParams.ts +0 -245
  196. package/src/test/mocks/idb-keyval.ts +0 -5
  197. package/src/test/setupTests.ts +0 -87
  198. package/src/test/unit/browserHashProvider.test.ts +0 -155
  199. package/src/test/unit/browserHashProvider.test.ts.bak +0 -376
  200. package/src/test/unit/directoryHandleStore.test.ts +0 -251
  201. package/src/test/unit/fileApiTreeAccessors.test.ts +0 -1387
  202. package/src/test/unit/fileApiTypes.test.ts +0 -587
  203. package/src/test/unit/fileSystemAccessTreeAccessors.test.ts +0 -885
  204. package/src/test/unit/fileTreeHelpers.test.ts +0 -694
  205. package/src/test/unit/httpTreeAccessors.test.ts +0 -1571
  206. package/src/test/unit/localStorageTreeAccessors.test.ts +0 -1014
  207. package/src/test/unit/urlParams.test.ts +0 -464
  208. package/src/test/utils/fileSystemAccessMocks.ts +0 -353
  209. package/src/test/utils/testHelpers.ts +0 -155
  210. package/temp/build/typescript/ts_8nwakTlr.json +0 -1
  211. package/temp/coverage/base.css +0 -224
  212. package/temp/coverage/block-navigation.js +0 -87
  213. package/temp/coverage/crypto/browserHashProvider.ts.html +0 -304
  214. package/temp/coverage/crypto/index.html +0 -116
  215. package/temp/coverage/crypto-utils/browserCryptoProvider.ts.html +0 -1018
  216. package/temp/coverage/crypto-utils/browserHashProvider.ts.html +0 -304
  217. package/temp/coverage/crypto-utils/index.html +0 -131
  218. package/temp/coverage/favicon.png +0 -0
  219. package/temp/coverage/file-tree/directoryHandleStore.ts.html +0 -493
  220. package/temp/coverage/file-tree/fileApiTreeAccessors.ts.html +0 -1669
  221. package/temp/coverage/file-tree/fileSystemAccessTreeAccessors.ts.html +0 -1642
  222. package/temp/coverage/file-tree/httpTreeAccessors.ts.html +0 -1429
  223. package/temp/coverage/file-tree/index.html +0 -176
  224. package/temp/coverage/file-tree/localStorageTreeAccessors.ts.html +0 -1375
  225. package/temp/coverage/helpers/fileTreeHelpers.ts.html +0 -406
  226. package/temp/coverage/helpers/index.html +0 -116
  227. package/temp/coverage/index.html +0 -161
  228. package/temp/coverage/lcov-report/base.css +0 -224
  229. package/temp/coverage/lcov-report/block-navigation.js +0 -87
  230. package/temp/coverage/lcov-report/crypto/browserHashProvider.ts.html +0 -304
  231. package/temp/coverage/lcov-report/crypto/index.html +0 -116
  232. package/temp/coverage/lcov-report/crypto-utils/browserCryptoProvider.ts.html +0 -1018
  233. package/temp/coverage/lcov-report/crypto-utils/browserHashProvider.ts.html +0 -304
  234. package/temp/coverage/lcov-report/crypto-utils/index.html +0 -131
  235. package/temp/coverage/lcov-report/favicon.png +0 -0
  236. package/temp/coverage/lcov-report/file-tree/directoryHandleStore.ts.html +0 -493
  237. package/temp/coverage/lcov-report/file-tree/fileApiTreeAccessors.ts.html +0 -1669
  238. package/temp/coverage/lcov-report/file-tree/fileSystemAccessTreeAccessors.ts.html +0 -1642
  239. package/temp/coverage/lcov-report/file-tree/httpTreeAccessors.ts.html +0 -1429
  240. package/temp/coverage/lcov-report/file-tree/index.html +0 -176
  241. package/temp/coverage/lcov-report/file-tree/localStorageTreeAccessors.ts.html +0 -1375
  242. package/temp/coverage/lcov-report/helpers/fileTreeHelpers.ts.html +0 -406
  243. package/temp/coverage/lcov-report/helpers/index.html +0 -116
  244. package/temp/coverage/lcov-report/index.html +0 -161
  245. package/temp/coverage/lcov-report/prettify.css +0 -1
  246. package/temp/coverage/lcov-report/prettify.js +0 -2
  247. package/temp/coverage/lcov-report/sort-arrow-sprite.png +0 -0
  248. package/temp/coverage/lcov-report/sorter.js +0 -210
  249. package/temp/coverage/lcov-report/url-utils/index.html +0 -116
  250. package/temp/coverage/lcov-report/url-utils/urlParams.ts.html +0 -820
  251. package/temp/coverage/lcov.info +0 -3597
  252. package/temp/coverage/prettify.css +0 -1
  253. package/temp/coverage/prettify.js +0 -2
  254. package/temp/coverage/sort-arrow-sprite.png +0 -0
  255. package/temp/coverage/sorter.js +0 -210
  256. package/temp/coverage/url-utils/index.html +0 -116
  257. package/temp/coverage/url-utils/urlParams.ts.html +0 -820
  258. package/temp/test/jest/haste-map-b931e4e63102f86c5bd4949f7dced44f-9d713eb41149188b4e5c0ae3d86d0a57-2ad8e16b24e391b8cdbe50b55c137169 +0 -0
  259. package/temp/test/jest/jest-transform-cache-b931e4e63102f86c5bd4949f7dced44f-79ef2876fae7ca75eedb2aa53dc48338/b5/package_b5f57afc9ec2c011239b1608ee5bdfa5 +0 -53
  260. package/temp/test/jest/perf-cache-b931e4e63102f86c5bd4949f7dced44f-da39a3ee5e6b4b0d3255bfef95601890 +0 -1
  261. package/temp/ts-web-extras.api.json +0 -8850
  262. package/temp/ts-web-extras.api.md +0 -433
  263. 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
- });