@fgv/ts-web-extras 5.0.2 → 5.1.0-0

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 (208) hide show
  1. package/.rush/temp/81e0881271ff236956b2f52e8ca99da6574c6e1e.tar.log +223 -0
  2. package/.rush/temp/chunked-rush-logs/ts-web-extras.build.chunks.jsonl +35 -25
  3. package/.rush/temp/operation/build/all.log +35 -25
  4. package/.rush/temp/operation/build/log-chunks.jsonl +35 -25
  5. package/.rush/temp/operation/build/state.json +1 -1
  6. package/.rush/temp/shrinkwrap-deps.json +175 -163
  7. package/config/jest.config.json +4 -1
  8. package/config/typedoc.json +6 -0
  9. package/dist/index.js +2 -2
  10. package/dist/index.js.map +1 -1
  11. package/dist/packlets/crypto-utils/browserCryptoProvider.js +254 -0
  12. package/dist/packlets/crypto-utils/browserCryptoProvider.js.map +1 -0
  13. package/dist/packlets/crypto-utils/browserHashProvider.js.map +1 -0
  14. package/dist/packlets/{crypto → crypto-utils}/index.js +1 -0
  15. package/dist/packlets/crypto-utils/index.js.map +1 -0
  16. package/dist/packlets/file-api-types/index.js +27 -3
  17. package/dist/packlets/file-api-types/index.js.map +1 -1
  18. package/dist/packlets/file-tree/directoryHandleStore.js +124 -0
  19. package/dist/packlets/file-tree/directoryHandleStore.js.map +1 -0
  20. package/dist/packlets/file-tree/fileApiTreeAccessors.js +76 -0
  21. package/dist/packlets/file-tree/fileApiTreeAccessors.js.map +1 -1
  22. package/dist/packlets/file-tree/fileSystemAccessTreeAccessors.js +345 -0
  23. package/dist/packlets/file-tree/fileSystemAccessTreeAccessors.js.map +1 -0
  24. package/dist/packlets/file-tree/index.js +3 -0
  25. package/dist/packlets/file-tree/index.js.map +1 -1
  26. package/dist/packlets/file-tree/localStorageTreeAccessors.js +308 -0
  27. package/dist/packlets/file-tree/localStorageTreeAccessors.js.map +1 -0
  28. package/dist/test/mocks/idb-keyval.js +6 -0
  29. package/dist/test/mocks/idb-keyval.js.map +1 -0
  30. package/dist/test/unit/browserHashProvider.test.js +1 -1
  31. package/dist/test/unit/browserHashProvider.test.js.map +1 -1
  32. package/dist/test/unit/directoryHandleStore.test.js +190 -0
  33. package/dist/test/unit/directoryHandleStore.test.js.map +1 -0
  34. package/dist/test/unit/fileApiTypes.test.js +30 -0
  35. package/dist/test/unit/fileApiTypes.test.js.map +1 -1
  36. package/dist/test/unit/fileSystemAccessTreeAccessors.test.js +517 -0
  37. package/dist/test/unit/fileSystemAccessTreeAccessors.test.js.map +1 -0
  38. package/dist/test/unit/localStorageTreeAccessors.test.js +595 -0
  39. package/dist/test/unit/localStorageTreeAccessors.test.js.map +1 -0
  40. package/dist/test/utils/fileSystemAccessMocks.js +271 -0
  41. package/dist/test/utils/fileSystemAccessMocks.js.map +1 -0
  42. package/dist/ts-web-extras.d.ts +460 -1
  43. package/dist/tsdoc-metadata.json +1 -1
  44. package/docs/ts-web-extras.cryptoutils.browsercryptoprovider._constructor_.md +50 -0
  45. package/docs/ts-web-extras.cryptoutils.browsercryptoprovider.decrypt.md +104 -0
  46. package/docs/ts-web-extras.cryptoutils.browsercryptoprovider.derivekey.md +88 -0
  47. package/docs/ts-web-extras.cryptoutils.browsercryptoprovider.encrypt.md +72 -0
  48. package/docs/ts-web-extras.cryptoutils.browsercryptoprovider.frombase64.md +56 -0
  49. package/docs/ts-web-extras.cryptoutils.browsercryptoprovider.generatekey.md +19 -0
  50. package/docs/ts-web-extras.cryptoutils.browsercryptoprovider.generaterandombytes.md +56 -0
  51. package/docs/ts-web-extras.cryptoutils.browsercryptoprovider.md +169 -0
  52. package/docs/ts-web-extras.cryptoutils.browsercryptoprovider.tobase64.md +56 -0
  53. package/docs/{ts-web-extras.browserhashprovider.hashparts.md → ts-web-extras.cryptoutils.browserhashprovider.hashparts.md} +2 -2
  54. package/docs/{ts-web-extras.browserhashprovider.hashstring.md → ts-web-extras.cryptoutils.browserhashprovider.hashstring.md} +2 -2
  55. package/docs/{ts-web-extras.browserhashprovider.md → ts-web-extras.cryptoutils.browserhashprovider.md} +4 -4
  56. package/docs/ts-web-extras.cryptoutils.createbrowsercryptoprovider.md +19 -0
  57. package/docs/ts-web-extras.cryptoutils.md +71 -0
  58. package/docs/ts-web-extras.fileapitreeaccessors.createfromlocalstorage.md +74 -0
  59. package/docs/ts-web-extras.fileapitreeaccessors.createpersistent.md +76 -0
  60. package/docs/ts-web-extras.fileapitreeaccessors.md +32 -0
  61. package/docs/ts-web-extras.filesystemaccesstreeaccessors._constructor_.md +114 -0
  62. package/docs/ts-web-extras.filesystemaccesstreeaccessors.fileismutable.md +52 -0
  63. package/docs/ts-web-extras.filesystemaccesstreeaccessors.fromdirectoryhandle.md +72 -0
  64. package/docs/ts-web-extras.filesystemaccesstreeaccessors.getdirtypaths.md +17 -0
  65. package/docs/ts-web-extras.filesystemaccesstreeaccessors.isdirty.md +17 -0
  66. package/docs/ts-web-extras.filesystemaccesstreeaccessors.md +159 -0
  67. package/docs/ts-web-extras.filesystemaccesstreeaccessors.savefilecontents.md +66 -0
  68. package/docs/ts-web-extras.filesystemaccesstreeaccessors.synctodisk.md +17 -0
  69. package/docs/ts-web-extras.ifilesystemaccesstreeparams.autosync.md +13 -0
  70. package/docs/ts-web-extras.ifilesystemaccesstreeparams.md +78 -0
  71. package/docs/ts-web-extras.ifilesystemaccesstreeparams.requirewritepermission.md +13 -0
  72. package/docs/ts-web-extras.ilocalstoragetreeparams.autosync.md +13 -0
  73. package/docs/ts-web-extras.ilocalstoragetreeparams.md +97 -0
  74. package/docs/ts-web-extras.ilocalstoragetreeparams.pathtokeymap.md +13 -0
  75. package/docs/ts-web-extras.ilocalstoragetreeparams.storage.md +13 -0
  76. package/docs/ts-web-extras.localstoragetreeaccessors.fileismutable.md +56 -0
  77. package/docs/ts-web-extras.localstoragetreeaccessors.fromstorage.md +56 -0
  78. package/docs/ts-web-extras.localstoragetreeaccessors.getdirtypaths.md +19 -0
  79. package/docs/ts-web-extras.localstoragetreeaccessors.isdirty.md +19 -0
  80. package/docs/ts-web-extras.localstoragetreeaccessors.md +131 -0
  81. package/docs/ts-web-extras.localstoragetreeaccessors.savefilecontents.md +72 -0
  82. package/docs/ts-web-extras.localstoragetreeaccessors.synctodisk.md +19 -0
  83. package/docs/ts-web-extras.md +50 -4
  84. package/etc/ts-web-extras.api.md +91 -1
  85. package/lib/index.d.ts +2 -1
  86. package/lib/index.d.ts.map +1 -1
  87. package/lib/index.js +25 -2
  88. package/lib/index.js.map +1 -1
  89. package/lib/packlets/crypto-utils/browserCryptoProvider.d.ts +77 -0
  90. package/lib/packlets/crypto-utils/browserCryptoProvider.d.ts.map +1 -0
  91. package/lib/packlets/crypto-utils/browserCryptoProvider.js +259 -0
  92. package/lib/packlets/crypto-utils/browserCryptoProvider.js.map +1 -0
  93. package/lib/packlets/crypto-utils/browserHashProvider.d.ts.map +1 -0
  94. package/lib/packlets/crypto-utils/browserHashProvider.js.map +1 -0
  95. package/lib/packlets/{crypto → crypto-utils}/index.d.ts +1 -0
  96. package/lib/packlets/crypto-utils/index.d.ts.map +1 -0
  97. package/lib/packlets/{crypto → crypto-utils}/index.js +1 -0
  98. package/lib/packlets/crypto-utils/index.js.map +1 -0
  99. package/lib/packlets/file-api-types/index.d.ts.map +1 -1
  100. package/lib/packlets/file-api-types/index.js +27 -3
  101. package/lib/packlets/file-api-types/index.js.map +1 -1
  102. package/lib/packlets/file-tree/directoryHandleStore.d.ts +59 -0
  103. package/lib/packlets/file-tree/directoryHandleStore.d.ts.map +1 -0
  104. package/lib/packlets/file-tree/directoryHandleStore.js +128 -0
  105. package/lib/packlets/file-tree/directoryHandleStore.js.map +1 -0
  106. package/lib/packlets/file-tree/fileApiTreeAccessors.d.ts +57 -0
  107. package/lib/packlets/file-tree/fileApiTreeAccessors.d.ts.map +1 -1
  108. package/lib/packlets/file-tree/fileApiTreeAccessors.js +76 -0
  109. package/lib/packlets/file-tree/fileApiTreeAccessors.js.map +1 -1
  110. package/lib/packlets/file-tree/fileSystemAccessTreeAccessors.d.ts +136 -0
  111. package/lib/packlets/file-tree/fileSystemAccessTreeAccessors.d.ts.map +1 -0
  112. package/lib/packlets/file-tree/fileSystemAccessTreeAccessors.js +349 -0
  113. package/lib/packlets/file-tree/fileSystemAccessTreeAccessors.js.map +1 -0
  114. package/lib/packlets/file-tree/index.d.ts +3 -0
  115. package/lib/packlets/file-tree/index.d.ts.map +1 -1
  116. package/lib/packlets/file-tree/index.js +3 -0
  117. package/lib/packlets/file-tree/index.js.map +1 -1
  118. package/lib/packlets/file-tree/localStorageTreeAccessors.d.ts +129 -0
  119. package/lib/packlets/file-tree/localStorageTreeAccessors.d.ts.map +1 -0
  120. package/lib/packlets/file-tree/localStorageTreeAccessors.js +312 -0
  121. package/lib/packlets/file-tree/localStorageTreeAccessors.js.map +1 -0
  122. package/lib/test/mocks/idb-keyval.d.ts +6 -0
  123. package/lib/test/mocks/idb-keyval.d.ts.map +1 -0
  124. package/lib/test/mocks/idb-keyval.js +9 -0
  125. package/lib/test/mocks/idb-keyval.js.map +1 -0
  126. package/lib/test/unit/browserHashProvider.test.js +21 -21
  127. package/lib/test/unit/browserHashProvider.test.js.map +1 -1
  128. package/lib/test/unit/directoryHandleStore.test.d.ts +2 -0
  129. package/lib/test/unit/directoryHandleStore.test.d.ts.map +1 -0
  130. package/lib/test/unit/directoryHandleStore.test.js +192 -0
  131. package/lib/test/unit/directoryHandleStore.test.js.map +1 -0
  132. package/lib/test/unit/fileApiTypes.test.js +30 -0
  133. package/lib/test/unit/fileApiTypes.test.js.map +1 -1
  134. package/lib/test/unit/fileSystemAccessTreeAccessors.test.d.ts +2 -0
  135. package/lib/test/unit/fileSystemAccessTreeAccessors.test.d.ts.map +1 -0
  136. package/lib/test/unit/fileSystemAccessTreeAccessors.test.js +519 -0
  137. package/lib/test/unit/fileSystemAccessTreeAccessors.test.js.map +1 -0
  138. package/lib/test/unit/localStorageTreeAccessors.test.d.ts +2 -0
  139. package/lib/test/unit/localStorageTreeAccessors.test.d.ts.map +1 -0
  140. package/lib/test/unit/localStorageTreeAccessors.test.js +597 -0
  141. package/lib/test/unit/localStorageTreeAccessors.test.js.map +1 -0
  142. package/lib/test/utils/fileSystemAccessMocks.d.ts +53 -0
  143. package/lib/test/utils/fileSystemAccessMocks.d.ts.map +1 -0
  144. package/lib/test/utils/fileSystemAccessMocks.js +277 -0
  145. package/lib/test/utils/fileSystemAccessMocks.js.map +1 -0
  146. package/package.json +27 -20
  147. package/rush-logs/ts-web-extras.build.cache.log +3 -1
  148. package/rush-logs/ts-web-extras.build.log +35 -25
  149. package/src/index.ts +2 -2
  150. package/src/packlets/crypto-utils/browserCryptoProvider.ts +311 -0
  151. package/src/packlets/{crypto → crypto-utils}/index.ts +1 -0
  152. package/src/packlets/file-api-types/index.ts +24 -3
  153. package/src/packlets/file-tree/directoryHandleStore.ts +136 -0
  154. package/src/packlets/file-tree/fileApiTreeAccessors.ts +90 -0
  155. package/src/packlets/file-tree/fileSystemAccessTreeAccessors.ts +427 -0
  156. package/src/packlets/file-tree/index.ts +3 -0
  157. package/src/packlets/file-tree/localStorageTreeAccessors.ts +377 -0
  158. package/src/test/mocks/idb-keyval.ts +5 -0
  159. package/src/test/unit/browserHashProvider.test.ts +1 -1
  160. package/src/test/unit/directoryHandleStore.test.ts +251 -0
  161. package/src/test/unit/fileApiTypes.test.ts +36 -0
  162. package/src/test/unit/fileSystemAccessTreeAccessors.test.ts +732 -0
  163. package/src/test/unit/localStorageTreeAccessors.test.ts +746 -0
  164. package/src/test/utils/fileSystemAccessMocks.ts +353 -0
  165. package/temp/build/typescript/ts_8nwakTlr.json +1 -0
  166. package/temp/coverage/crypto-utils/browserCryptoProvider.ts.html +1018 -0
  167. package/temp/coverage/{crypto → crypto-utils}/browserHashProvider.ts.html +3 -3
  168. package/temp/coverage/{lcov-report/crypto → crypto-utils}/index.html +21 -6
  169. package/temp/coverage/file-tree/directoryHandleStore.ts.html +493 -0
  170. package/temp/coverage/file-tree/fileApiTreeAccessors.ts.html +276 -6
  171. package/temp/coverage/file-tree/fileSystemAccessTreeAccessors.ts.html +1366 -0
  172. package/temp/coverage/file-tree/index.html +55 -10
  173. package/temp/coverage/file-tree/localStorageTreeAccessors.ts.html +1216 -0
  174. package/temp/coverage/helpers/fileTreeHelpers.ts.html +1 -1
  175. package/temp/coverage/helpers/index.html +1 -1
  176. package/temp/coverage/index.html +15 -15
  177. package/temp/coverage/lcov-report/crypto-utils/browserCryptoProvider.ts.html +1018 -0
  178. package/temp/coverage/lcov-report/{crypto → crypto-utils}/browserHashProvider.ts.html +3 -3
  179. package/temp/coverage/{crypto → lcov-report/crypto-utils}/index.html +21 -6
  180. package/temp/coverage/lcov-report/file-tree/directoryHandleStore.ts.html +493 -0
  181. package/temp/coverage/lcov-report/file-tree/fileApiTreeAccessors.ts.html +276 -6
  182. package/temp/coverage/lcov-report/file-tree/fileSystemAccessTreeAccessors.ts.html +1366 -0
  183. package/temp/coverage/lcov-report/file-tree/index.html +55 -10
  184. package/temp/coverage/lcov-report/file-tree/localStorageTreeAccessors.ts.html +1216 -0
  185. package/temp/coverage/lcov-report/helpers/fileTreeHelpers.ts.html +1 -1
  186. package/temp/coverage/lcov-report/helpers/index.html +1 -1
  187. package/temp/coverage/lcov-report/index.html +15 -15
  188. package/temp/coverage/lcov-report/url-utils/index.html +1 -1
  189. package/temp/coverage/lcov-report/url-utils/urlParams.ts.html +1 -1
  190. package/temp/coverage/lcov.info +2128 -451
  191. package/temp/coverage/url-utils/index.html +1 -1
  192. package/temp/coverage/url-utils/urlParams.ts.html +1 -1
  193. package/temp/test/jest/haste-map-7492f1b44480e0cdd1f220078fb3afd8-c8dd6c3430605adeb2f1cadf4f75e791-8c9336785555d572065b28c111982ba4 +0 -0
  194. package/temp/test/jest/perf-cache-7492f1b44480e0cdd1f220078fb3afd8-da39a3ee5e6b4b0d3255bfef95601890 +1 -1
  195. package/temp/ts-web-extras.api.json +3236 -385
  196. package/temp/ts-web-extras.api.md +91 -1
  197. package/dist/packlets/crypto/browserHashProvider.js.map +0 -1
  198. package/dist/packlets/crypto/index.js.map +0 -1
  199. package/lib/packlets/crypto/browserHashProvider.d.ts.map +0 -1
  200. package/lib/packlets/crypto/browserHashProvider.js.map +0 -1
  201. package/lib/packlets/crypto/index.d.ts.map +0 -1
  202. package/lib/packlets/crypto/index.js.map +0 -1
  203. package/temp/build/typescript/ts_vnCx6LlY.json +0 -1
  204. /package/dist/packlets/{crypto → crypto-utils}/browserHashProvider.js +0 -0
  205. /package/lib/packlets/{crypto → crypto-utils}/browserHashProvider.d.ts +0 -0
  206. /package/lib/packlets/{crypto → crypto-utils}/browserHashProvider.js +0 -0
  207. /package/src/packlets/{crypto → crypto-utils}/browserHashProvider.ts +0 -0
  208. /package/temp/test/jest/jest-transform-cache-7492f1b44480e0cdd1f220078fb3afd8-79ef2876fae7ca75eedb2aa53dc48338/{0e/package_0eb6535f5987849d93ea51ef33a14cf6 → 8d/package_8dcbedef69e4299f0f51fcda8f4f1c8e} +0 -0
@@ -22,40 +22,40 @@
22
22
  */
23
23
  Object.defineProperty(exports, "__esModule", { value: true });
24
24
  require("@fgv/ts-utils-jest");
25
- const crypto_1 = require("../../packlets/crypto");
25
+ const crypto_utils_1 = require("../../packlets/crypto-utils");
26
26
  describe('BrowserHashProvider', () => {
27
27
  describe('hashString', () => {
28
28
  test('successfully hashes a string with SHA-256', async () => {
29
- const result = await crypto_1.BrowserHashProvider.hashString('test data', 'SHA-256');
29
+ const result = await crypto_utils_1.BrowserHashProvider.hashString('test data', 'SHA-256');
30
30
  expect(result).toSucceedAndSatisfy((hash) => {
31
31
  expect(typeof hash).toBe('string');
32
32
  expect(hash).toMatch(/^[a-f0-9]{64}$/); // SHA-256 produces 64 hex characters
33
33
  });
34
34
  });
35
35
  test('successfully hashes a string with SHA-1', async () => {
36
- const result = await crypto_1.BrowserHashProvider.hashString('test data', 'SHA-1');
36
+ const result = await crypto_utils_1.BrowserHashProvider.hashString('test data', 'SHA-1');
37
37
  expect(result).toSucceedAndSatisfy((hash) => {
38
38
  expect(typeof hash).toBe('string');
39
39
  expect(hash).toMatch(/^[a-f0-9]{40}$/); // SHA-1 produces 40 hex characters
40
40
  });
41
41
  });
42
42
  test('successfully hashes a string with SHA-512', async () => {
43
- const result = await crypto_1.BrowserHashProvider.hashString('test data', 'SHA-512');
43
+ const result = await crypto_utils_1.BrowserHashProvider.hashString('test data', 'SHA-512');
44
44
  expect(result).toSucceedAndSatisfy((hash) => {
45
45
  expect(typeof hash).toBe('string');
46
46
  expect(hash).toMatch(/^[a-f0-9]{128}$/); // SHA-512 produces 128 hex characters
47
47
  });
48
48
  });
49
49
  test('uses SHA-256 as default algorithm', async () => {
50
- const result = await crypto_1.BrowserHashProvider.hashString('test data');
50
+ const result = await crypto_utils_1.BrowserHashProvider.hashString('test data');
51
51
  expect(result).toSucceedAndSatisfy((hash) => {
52
52
  expect(hash).toMatch(/^[a-f0-9]{64}$/); // SHA-256 produces 64 hex characters
53
53
  });
54
54
  });
55
55
  test('produces consistent hash for same input', async () => {
56
56
  const input = 'consistent input';
57
- const result1 = await crypto_1.BrowserHashProvider.hashString(input);
58
- const result2 = await crypto_1.BrowserHashProvider.hashString(input);
57
+ const result1 = await crypto_utils_1.BrowserHashProvider.hashString(input);
58
+ const result2 = await crypto_utils_1.BrowserHashProvider.hashString(input);
59
59
  expect(result1).toSucceed();
60
60
  expect(result2).toSucceed();
61
61
  if (result1.isSuccess() && result2.isSuccess()) {
@@ -63,8 +63,8 @@ describe('BrowserHashProvider', () => {
63
63
  }
64
64
  });
65
65
  test('produces different hashes for different inputs', async () => {
66
- const result1 = await crypto_1.BrowserHashProvider.hashString('input1');
67
- const result2 = await crypto_1.BrowserHashProvider.hashString('input2');
66
+ const result1 = await crypto_utils_1.BrowserHashProvider.hashString('input1');
67
+ const result2 = await crypto_utils_1.BrowserHashProvider.hashString('input2');
68
68
  expect(result1).toSucceed();
69
69
  expect(result2).toSucceed();
70
70
  if (result1.isSuccess() && result2.isSuccess()) {
@@ -72,11 +72,11 @@ describe('BrowserHashProvider', () => {
72
72
  }
73
73
  });
74
74
  test('handles empty string input', async () => {
75
- const result = await crypto_1.BrowserHashProvider.hashString('');
75
+ const result = await crypto_utils_1.BrowserHashProvider.hashString('');
76
76
  expect(result).toSucceed();
77
77
  });
78
78
  test('handles unicode input', async () => {
79
- const result = await crypto_1.BrowserHashProvider.hashString('🚀 Unicode test 测试');
79
+ const result = await crypto_utils_1.BrowserHashProvider.hashString('🚀 Unicode test 测试');
80
80
  expect(result).toSucceedAndSatisfy((hash) => {
81
81
  expect(hash).toMatch(/^[a-f0-9]{64}$/);
82
82
  });
@@ -85,27 +85,27 @@ describe('BrowserHashProvider', () => {
85
85
  const originalCrypto = global.crypto;
86
86
  // @ts-ignore - Intentionally removing crypto for test
87
87
  delete global.crypto;
88
- const result = await crypto_1.BrowserHashProvider.hashString('test');
88
+ const result = await crypto_utils_1.BrowserHashProvider.hashString('test');
89
89
  expect(result).toFailWith(/Hash computation failed/);
90
90
  global.crypto = originalCrypto;
91
91
  });
92
92
  test('handles invalid algorithm gracefully', async () => {
93
- const result = await crypto_1.BrowserHashProvider.hashString('test', 'INVALID-ALG');
93
+ const result = await crypto_utils_1.BrowserHashProvider.hashString('test', 'INVALID-ALG');
94
94
  expect(result).toFailWith(/Hash computation failed/);
95
95
  });
96
96
  });
97
97
  describe('hashParts', () => {
98
98
  test('hashes multiple parts with default separator', async () => {
99
99
  const parts = ['part1', 'part2', 'part3'];
100
- const result = await crypto_1.BrowserHashProvider.hashParts(parts);
100
+ const result = await crypto_utils_1.BrowserHashProvider.hashParts(parts);
101
101
  expect(result).toSucceedAndSatisfy((hash) => {
102
102
  expect(hash).toMatch(/^[a-f0-9]{64}$/);
103
103
  });
104
104
  });
105
105
  test('hashes multiple parts with custom separator', async () => {
106
106
  const parts = ['part1', 'part2', 'part3'];
107
- const result1 = await crypto_1.BrowserHashProvider.hashParts(parts, 'SHA-256', '|');
108
- const result2 = await crypto_1.BrowserHashProvider.hashParts(parts, 'SHA-256', ',');
107
+ const result1 = await crypto_utils_1.BrowserHashProvider.hashParts(parts, 'SHA-256', '|');
108
+ const result2 = await crypto_utils_1.BrowserHashProvider.hashParts(parts, 'SHA-256', ',');
109
109
  expect(result1).toSucceed();
110
110
  expect(result2).toSucceed();
111
111
  if (result1.isSuccess() && result2.isSuccess()) {
@@ -116,8 +116,8 @@ describe('BrowserHashProvider', () => {
116
116
  const parts = ['a', 'b', 'c'];
117
117
  const separator = '-';
118
118
  const joined = parts.join(separator);
119
- const partsResult = await crypto_1.BrowserHashProvider.hashParts(parts, 'SHA-256', separator);
120
- const stringResult = await crypto_1.BrowserHashProvider.hashString(joined);
119
+ const partsResult = await crypto_utils_1.BrowserHashProvider.hashParts(parts, 'SHA-256', separator);
120
+ const stringResult = await crypto_utils_1.BrowserHashProvider.hashString(joined);
121
121
  expect(partsResult).toSucceed();
122
122
  expect(stringResult).toSucceed();
123
123
  if (partsResult.isSuccess() && stringResult.isSuccess()) {
@@ -125,12 +125,12 @@ describe('BrowserHashProvider', () => {
125
125
  }
126
126
  });
127
127
  test('handles empty array', async () => {
128
- const result = await crypto_1.BrowserHashProvider.hashParts([]);
128
+ const result = await crypto_utils_1.BrowserHashProvider.hashParts([]);
129
129
  expect(result).toSucceed();
130
130
  });
131
131
  test('handles single part', async () => {
132
- const result = await crypto_1.BrowserHashProvider.hashParts(['single']);
133
- const stringResult = await crypto_1.BrowserHashProvider.hashString('single');
132
+ const result = await crypto_utils_1.BrowserHashProvider.hashParts(['single']);
133
+ const stringResult = await crypto_utils_1.BrowserHashProvider.hashString('single');
134
134
  expect(result).toSucceed();
135
135
  expect(stringResult).toSucceed();
136
136
  if (result.isSuccess() && stringResult.isSuccess()) {
@@ -1 +1 @@
1
- {"version":3,"file":"browserHashProvider.test.js","sourceRoot":"","sources":["../../../src/test/unit/browserHashProvider.test.ts"],"names":[],"mappings":"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","sourcesContent":["/*\n * Copyright (c) 2025 Erik Fortune\n *\n * Permission is hereby granted, free of charge, to any person obtaining a copy\n * of this software and associated documentation files (the \"Software\"), to deal\n * in the Software without restriction, including without limitation the rights\n * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell\n * copies of the Software, and to permit persons to whom the Software is\n * furnished to do so, subject to the following conditions:\n *\n * The above copyright notice and this permission notice shall be included in all\n * copies or substantial portions of the Software.\n *\n * THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\n * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\n * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE\n * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\n * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,\n * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE\n * SOFTWARE.\n */\n\nimport '@fgv/ts-utils-jest';\nimport { BrowserHashProvider } from '../../packlets/crypto';\n\ndescribe('BrowserHashProvider', () => {\n describe('hashString', () => {\n test('successfully hashes a string with SHA-256', async () => {\n const result = await BrowserHashProvider.hashString('test data', 'SHA-256');\n expect(result).toSucceedAndSatisfy((hash) => {\n expect(typeof hash).toBe('string');\n expect(hash).toMatch(/^[a-f0-9]{64}$/); // SHA-256 produces 64 hex characters\n });\n });\n\n test('successfully hashes a string with SHA-1', async () => {\n const result = await BrowserHashProvider.hashString('test data', 'SHA-1');\n expect(result).toSucceedAndSatisfy((hash) => {\n expect(typeof hash).toBe('string');\n expect(hash).toMatch(/^[a-f0-9]{40}$/); // SHA-1 produces 40 hex characters\n });\n });\n\n test('successfully hashes a string with SHA-512', async () => {\n const result = await BrowserHashProvider.hashString('test data', 'SHA-512');\n expect(result).toSucceedAndSatisfy((hash) => {\n expect(typeof hash).toBe('string');\n expect(hash).toMatch(/^[a-f0-9]{128}$/); // SHA-512 produces 128 hex characters\n });\n });\n\n test('uses SHA-256 as default algorithm', async () => {\n const result = await BrowserHashProvider.hashString('test data');\n expect(result).toSucceedAndSatisfy((hash) => {\n expect(hash).toMatch(/^[a-f0-9]{64}$/); // SHA-256 produces 64 hex characters\n });\n });\n\n test('produces consistent hash for same input', async () => {\n const input = 'consistent input';\n const result1 = await BrowserHashProvider.hashString(input);\n const result2 = await BrowserHashProvider.hashString(input);\n expect(result1).toSucceed();\n expect(result2).toSucceed();\n if (result1.isSuccess() && result2.isSuccess()) {\n expect(result1.value).toBe(result2.value);\n }\n });\n\n test('produces different hashes for different inputs', async () => {\n const result1 = await BrowserHashProvider.hashString('input1');\n const result2 = await BrowserHashProvider.hashString('input2');\n expect(result1).toSucceed();\n expect(result2).toSucceed();\n if (result1.isSuccess() && result2.isSuccess()) {\n expect(result1.value).not.toBe(result2.value);\n }\n });\n\n test('handles empty string input', async () => {\n const result = await BrowserHashProvider.hashString('');\n expect(result).toSucceed();\n });\n\n test('handles unicode input', async () => {\n const result = await BrowserHashProvider.hashString('🚀 Unicode test 测试');\n expect(result).toSucceedAndSatisfy((hash) => {\n expect(hash).toMatch(/^[a-f0-9]{64}$/);\n });\n });\n\n test('fails when crypto.subtle is unavailable', async () => {\n const originalCrypto = global.crypto;\n // @ts-ignore - Intentionally removing crypto for test\n delete global.crypto;\n const result = await BrowserHashProvider.hashString('test');\n expect(result).toFailWith(/Hash computation failed/);\n global.crypto = originalCrypto;\n });\n\n test('handles invalid algorithm gracefully', async () => {\n const result = await BrowserHashProvider.hashString('test', 'INVALID-ALG');\n expect(result).toFailWith(/Hash computation failed/);\n });\n });\n\n describe('hashParts', () => {\n test('hashes multiple parts with default separator', async () => {\n const parts = ['part1', 'part2', 'part3'];\n const result = await BrowserHashProvider.hashParts(parts);\n expect(result).toSucceedAndSatisfy((hash) => {\n expect(hash).toMatch(/^[a-f0-9]{64}$/);\n });\n });\n\n test('hashes multiple parts with custom separator', async () => {\n const parts = ['part1', 'part2', 'part3'];\n const result1 = await BrowserHashProvider.hashParts(parts, 'SHA-256', '|');\n const result2 = await BrowserHashProvider.hashParts(parts, 'SHA-256', ',');\n expect(result1).toSucceed();\n expect(result2).toSucceed();\n if (result1.isSuccess() && result2.isSuccess()) {\n expect(result1.value).not.toBe(result2.value);\n }\n });\n\n test('produces same hash as hashString for joined parts', async () => {\n const parts = ['a', 'b', 'c'];\n const separator = '-';\n const joined = parts.join(separator);\n const partsResult = await BrowserHashProvider.hashParts(parts, 'SHA-256', separator);\n const stringResult = await BrowserHashProvider.hashString(joined);\n expect(partsResult).toSucceed();\n expect(stringResult).toSucceed();\n if (partsResult.isSuccess() && stringResult.isSuccess()) {\n expect(partsResult.value).toBe(stringResult.value);\n }\n });\n\n test('handles empty array', async () => {\n const result = await BrowserHashProvider.hashParts([]);\n expect(result).toSucceed();\n });\n\n test('handles single part', async () => {\n const result = await BrowserHashProvider.hashParts(['single']);\n const stringResult = await BrowserHashProvider.hashString('single');\n expect(result).toSucceed();\n expect(stringResult).toSucceed();\n if (result.isSuccess() && stringResult.isSuccess()) {\n expect(result.value).toBe(stringResult.value);\n }\n });\n });\n});\n"]}
1
+ {"version":3,"file":"browserHashProvider.test.js","sourceRoot":"","sources":["../../../src/test/unit/browserHashProvider.test.ts"],"names":[],"mappings":"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","sourcesContent":["/*\n * Copyright (c) 2025 Erik Fortune\n *\n * Permission is hereby granted, free of charge, to any person obtaining a copy\n * of this software and associated documentation files (the \"Software\"), to deal\n * in the Software without restriction, including without limitation the rights\n * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell\n * copies of the Software, and to permit persons to whom the Software is\n * furnished to do so, subject to the following conditions:\n *\n * The above copyright notice and this permission notice shall be included in all\n * copies or substantial portions of the Software.\n *\n * THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\n * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\n * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE\n * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\n * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,\n * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE\n * SOFTWARE.\n */\n\nimport '@fgv/ts-utils-jest';\nimport { BrowserHashProvider } from '../../packlets/crypto-utils';\n\ndescribe('BrowserHashProvider', () => {\n describe('hashString', () => {\n test('successfully hashes a string with SHA-256', async () => {\n const result = await BrowserHashProvider.hashString('test data', 'SHA-256');\n expect(result).toSucceedAndSatisfy((hash) => {\n expect(typeof hash).toBe('string');\n expect(hash).toMatch(/^[a-f0-9]{64}$/); // SHA-256 produces 64 hex characters\n });\n });\n\n test('successfully hashes a string with SHA-1', async () => {\n const result = await BrowserHashProvider.hashString('test data', 'SHA-1');\n expect(result).toSucceedAndSatisfy((hash) => {\n expect(typeof hash).toBe('string');\n expect(hash).toMatch(/^[a-f0-9]{40}$/); // SHA-1 produces 40 hex characters\n });\n });\n\n test('successfully hashes a string with SHA-512', async () => {\n const result = await BrowserHashProvider.hashString('test data', 'SHA-512');\n expect(result).toSucceedAndSatisfy((hash) => {\n expect(typeof hash).toBe('string');\n expect(hash).toMatch(/^[a-f0-9]{128}$/); // SHA-512 produces 128 hex characters\n });\n });\n\n test('uses SHA-256 as default algorithm', async () => {\n const result = await BrowserHashProvider.hashString('test data');\n expect(result).toSucceedAndSatisfy((hash) => {\n expect(hash).toMatch(/^[a-f0-9]{64}$/); // SHA-256 produces 64 hex characters\n });\n });\n\n test('produces consistent hash for same input', async () => {\n const input = 'consistent input';\n const result1 = await BrowserHashProvider.hashString(input);\n const result2 = await BrowserHashProvider.hashString(input);\n expect(result1).toSucceed();\n expect(result2).toSucceed();\n if (result1.isSuccess() && result2.isSuccess()) {\n expect(result1.value).toBe(result2.value);\n }\n });\n\n test('produces different hashes for different inputs', async () => {\n const result1 = await BrowserHashProvider.hashString('input1');\n const result2 = await BrowserHashProvider.hashString('input2');\n expect(result1).toSucceed();\n expect(result2).toSucceed();\n if (result1.isSuccess() && result2.isSuccess()) {\n expect(result1.value).not.toBe(result2.value);\n }\n });\n\n test('handles empty string input', async () => {\n const result = await BrowserHashProvider.hashString('');\n expect(result).toSucceed();\n });\n\n test('handles unicode input', async () => {\n const result = await BrowserHashProvider.hashString('🚀 Unicode test 测试');\n expect(result).toSucceedAndSatisfy((hash) => {\n expect(hash).toMatch(/^[a-f0-9]{64}$/);\n });\n });\n\n test('fails when crypto.subtle is unavailable', async () => {\n const originalCrypto = global.crypto;\n // @ts-ignore - Intentionally removing crypto for test\n delete global.crypto;\n const result = await BrowserHashProvider.hashString('test');\n expect(result).toFailWith(/Hash computation failed/);\n global.crypto = originalCrypto;\n });\n\n test('handles invalid algorithm gracefully', async () => {\n const result = await BrowserHashProvider.hashString('test', 'INVALID-ALG');\n expect(result).toFailWith(/Hash computation failed/);\n });\n });\n\n describe('hashParts', () => {\n test('hashes multiple parts with default separator', async () => {\n const parts = ['part1', 'part2', 'part3'];\n const result = await BrowserHashProvider.hashParts(parts);\n expect(result).toSucceedAndSatisfy((hash) => {\n expect(hash).toMatch(/^[a-f0-9]{64}$/);\n });\n });\n\n test('hashes multiple parts with custom separator', async () => {\n const parts = ['part1', 'part2', 'part3'];\n const result1 = await BrowserHashProvider.hashParts(parts, 'SHA-256', '|');\n const result2 = await BrowserHashProvider.hashParts(parts, 'SHA-256', ',');\n expect(result1).toSucceed();\n expect(result2).toSucceed();\n if (result1.isSuccess() && result2.isSuccess()) {\n expect(result1.value).not.toBe(result2.value);\n }\n });\n\n test('produces same hash as hashString for joined parts', async () => {\n const parts = ['a', 'b', 'c'];\n const separator = '-';\n const joined = parts.join(separator);\n const partsResult = await BrowserHashProvider.hashParts(parts, 'SHA-256', separator);\n const stringResult = await BrowserHashProvider.hashString(joined);\n expect(partsResult).toSucceed();\n expect(stringResult).toSucceed();\n if (partsResult.isSuccess() && stringResult.isSuccess()) {\n expect(partsResult.value).toBe(stringResult.value);\n }\n });\n\n test('handles empty array', async () => {\n const result = await BrowserHashProvider.hashParts([]);\n expect(result).toSucceed();\n });\n\n test('handles single part', async () => {\n const result = await BrowserHashProvider.hashParts(['single']);\n const stringResult = await BrowserHashProvider.hashString('single');\n expect(result).toSucceed();\n expect(stringResult).toSucceed();\n if (result.isSuccess() && stringResult.isSuccess()) {\n expect(result.value).toBe(stringResult.value);\n }\n });\n });\n});\n"]}
@@ -0,0 +1,2 @@
1
+ import '@fgv/ts-utils-jest';
2
+ //# sourceMappingURL=directoryHandleStore.test.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"directoryHandleStore.test.d.ts","sourceRoot":"","sources":["../../../src/test/unit/directoryHandleStore.test.ts"],"names":[],"mappings":"AAsBA,OAAO,oBAAoB,CAAC"}
@@ -0,0 +1,192 @@
1
+ "use strict";
2
+ /*
3
+ * Copyright (c) 2026 Erik Fortune
4
+ *
5
+ * Permission is hereby granted, free of charge, to any person obtaining a copy
6
+ * of this software and associated documentation files (the "Software"), to deal
7
+ * in the Software without restriction, including without limitation the rights
8
+ * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
9
+ * copies of the Software, and to permit persons to whom the Software is
10
+ * furnished to do so, subject to the following conditions:
11
+ *
12
+ * The above copyright notice and this permission notice shall be included in all
13
+ * copies or substantial portions of the Software.
14
+ *
15
+ * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
16
+ * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
17
+ * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
18
+ * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
19
+ * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
20
+ * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
21
+ * SOFTWARE.
22
+ */
23
+ Object.defineProperty(exports, "__esModule", { value: true });
24
+ require("@fgv/ts-utils-jest");
25
+ const directoryHandleStore_1 = require("../../packlets/file-tree/directoryHandleStore");
26
+ const idb_keyval_1 = require("idb-keyval");
27
+ const mockGet = jest.mocked(idb_keyval_1.get);
28
+ const mockSet = jest.mocked(idb_keyval_1.set);
29
+ const mockDel = jest.mocked(idb_keyval_1.del);
30
+ const mockKeys = jest.mocked(idb_keyval_1.keys);
31
+ const mockCreateStore = jest.mocked(idb_keyval_1.createStore);
32
+ function makeMockHandle(name) {
33
+ return {
34
+ kind: 'directory',
35
+ name,
36
+ isSameEntry: jest.fn(),
37
+ queryPermission: jest.fn(),
38
+ requestPermission: jest.fn(),
39
+ getDirectoryHandle: jest.fn(),
40
+ getFileHandle: jest.fn(),
41
+ removeEntry: jest.fn(),
42
+ resolve: jest.fn(),
43
+ keys: jest.fn(),
44
+ values: jest.fn(),
45
+ entries: jest.fn(),
46
+ [Symbol.asyncIterator]: jest.fn()
47
+ };
48
+ }
49
+ describe('DirectoryHandleStore', () => {
50
+ beforeEach(() => {
51
+ jest.clearAllMocks();
52
+ });
53
+ describe('constants', () => {
54
+ test('DEFAULT_DIRECTORY_HANDLE_DB has expected value', () => {
55
+ expect(directoryHandleStore_1.DEFAULT_DIRECTORY_HANDLE_DB).toBe('chocolate-lab-storage');
56
+ });
57
+ test('DEFAULT_DIRECTORY_HANDLE_STORE has expected value', () => {
58
+ expect(directoryHandleStore_1.DEFAULT_DIRECTORY_HANDLE_STORE).toBe('directory-handles');
59
+ });
60
+ });
61
+ describe('constructor', () => {
62
+ test('uses default db and store names', () => {
63
+ new directoryHandleStore_1.DirectoryHandleStore();
64
+ expect(mockCreateStore).toHaveBeenCalledWith(directoryHandleStore_1.DEFAULT_DIRECTORY_HANDLE_DB, directoryHandleStore_1.DEFAULT_DIRECTORY_HANDLE_STORE);
65
+ });
66
+ test('uses custom db and store names', () => {
67
+ new directoryHandleStore_1.DirectoryHandleStore('custom-db', 'custom-store');
68
+ expect(mockCreateStore).toHaveBeenCalledWith('custom-db', 'custom-store');
69
+ });
70
+ });
71
+ describe('save', () => {
72
+ test('saves a handle successfully', async () => {
73
+ mockSet.mockResolvedValue(undefined);
74
+ const store = new directoryHandleStore_1.DirectoryHandleStore();
75
+ const handle = makeMockHandle('my-dir');
76
+ const result = await store.save('my-dir', handle);
77
+ expect(result).toSucceed();
78
+ expect(mockSet).toHaveBeenCalledWith('my-dir', handle, expect.anything());
79
+ });
80
+ test('returns failure when set throws', async () => {
81
+ mockSet.mockRejectedValue(new Error('IndexedDB unavailable'));
82
+ const store = new directoryHandleStore_1.DirectoryHandleStore();
83
+ const handle = makeMockHandle('my-dir');
84
+ const result = await store.save('my-dir', handle);
85
+ expect(result).toFailWith(/IndexedDB unavailable/);
86
+ });
87
+ });
88
+ describe('load', () => {
89
+ test('returns the handle when found', async () => {
90
+ const handle = makeMockHandle('my-dir');
91
+ mockGet.mockResolvedValue(handle);
92
+ const store = new directoryHandleStore_1.DirectoryHandleStore();
93
+ const result = await store.load('my-dir');
94
+ expect(result).toSucceedWith(handle);
95
+ expect(mockGet).toHaveBeenCalledWith('my-dir', expect.anything());
96
+ });
97
+ test('returns undefined when not found', async () => {
98
+ mockGet.mockResolvedValue(undefined);
99
+ const store = new directoryHandleStore_1.DirectoryHandleStore();
100
+ const result = await store.load('missing');
101
+ expect(result).toSucceedWith(undefined);
102
+ });
103
+ test('returns failure when get throws', async () => {
104
+ mockGet.mockRejectedValue(new Error('read error'));
105
+ const store = new directoryHandleStore_1.DirectoryHandleStore();
106
+ const result = await store.load('my-dir');
107
+ expect(result).toFailWith(/read error/);
108
+ });
109
+ });
110
+ describe('remove', () => {
111
+ test('removes a handle successfully', async () => {
112
+ mockDel.mockResolvedValue(undefined);
113
+ const store = new directoryHandleStore_1.DirectoryHandleStore();
114
+ const result = await store.remove('my-dir');
115
+ expect(result).toSucceed();
116
+ expect(mockDel).toHaveBeenCalledWith('my-dir', expect.anything());
117
+ });
118
+ test('returns failure when del throws', async () => {
119
+ mockDel.mockRejectedValue(new Error('delete error'));
120
+ const store = new directoryHandleStore_1.DirectoryHandleStore();
121
+ const result = await store.remove('my-dir');
122
+ expect(result).toFailWith(/delete error/);
123
+ });
124
+ });
125
+ describe('getAllLabels', () => {
126
+ test('returns all keys', async () => {
127
+ mockKeys.mockResolvedValue(['dir-a', 'dir-b']);
128
+ const store = new directoryHandleStore_1.DirectoryHandleStore();
129
+ const result = await store.getAllLabels();
130
+ expect(result).toSucceedWith(['dir-a', 'dir-b']);
131
+ expect(mockKeys).toHaveBeenCalledWith(expect.anything());
132
+ });
133
+ test('returns empty array when no keys', async () => {
134
+ mockKeys.mockResolvedValue([]);
135
+ const store = new directoryHandleStore_1.DirectoryHandleStore();
136
+ const result = await store.getAllLabels();
137
+ expect(result).toSucceedWith([]);
138
+ });
139
+ test('returns failure when keys throws', async () => {
140
+ mockKeys.mockRejectedValue(new Error('keys error'));
141
+ const store = new directoryHandleStore_1.DirectoryHandleStore();
142
+ const result = await store.getAllLabels();
143
+ expect(result).toFailWith(/keys error/);
144
+ });
145
+ });
146
+ describe('getAll', () => {
147
+ test('returns all label/handle pairs', async () => {
148
+ const handleA = makeMockHandle('dir-a');
149
+ const handleB = makeMockHandle('dir-b');
150
+ mockKeys.mockResolvedValue(['dir-a', 'dir-b']);
151
+ mockGet.mockResolvedValueOnce(handleA).mockResolvedValueOnce(handleB);
152
+ const store = new directoryHandleStore_1.DirectoryHandleStore();
153
+ const result = await store.getAll();
154
+ expect(result).toSucceedAndSatisfy((entries) => {
155
+ expect(entries).toHaveLength(2);
156
+ expect(entries[0]).toEqual({ label: 'dir-a', handle: handleA });
157
+ expect(entries[1]).toEqual({ label: 'dir-b', handle: handleB });
158
+ });
159
+ });
160
+ test('returns empty array when no handles stored', async () => {
161
+ mockKeys.mockResolvedValue([]);
162
+ const store = new directoryHandleStore_1.DirectoryHandleStore();
163
+ const result = await store.getAll();
164
+ expect(result).toSucceedWith([]);
165
+ });
166
+ test('skips entries where handle is undefined', async () => {
167
+ const handleA = makeMockHandle('dir-a');
168
+ mockKeys.mockResolvedValue(['dir-a', 'dir-b']);
169
+ mockGet.mockResolvedValueOnce(handleA).mockResolvedValueOnce(undefined);
170
+ const store = new directoryHandleStore_1.DirectoryHandleStore();
171
+ const result = await store.getAll();
172
+ expect(result).toSucceedAndSatisfy((entries) => {
173
+ expect(entries).toHaveLength(1);
174
+ expect(entries[0]).toEqual({ label: 'dir-a', handle: handleA });
175
+ });
176
+ });
177
+ test('returns failure when getAllLabels fails', async () => {
178
+ mockKeys.mockRejectedValue(new Error('keys error'));
179
+ const store = new directoryHandleStore_1.DirectoryHandleStore();
180
+ const result = await store.getAll();
181
+ expect(result).toFailWith(/keys error/);
182
+ });
183
+ test('returns failure when load fails for a key', async () => {
184
+ mockKeys.mockResolvedValue(['dir-a']);
185
+ mockGet.mockRejectedValue(new Error('load error'));
186
+ const store = new directoryHandleStore_1.DirectoryHandleStore();
187
+ const result = await store.getAll();
188
+ expect(result).toFailWith(/load error/);
189
+ });
190
+ });
191
+ });
192
+ //# sourceMappingURL=directoryHandleStore.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"directoryHandleStore.test.js","sourceRoot":"","sources":["../../../src/test/unit/directoryHandleStore.test.ts"],"names":[],"mappings":"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","sourcesContent":["/*\n * Copyright (c) 2026 Erik Fortune\n *\n * Permission is hereby granted, free of charge, to any person obtaining a copy\n * of this software and associated documentation files (the \"Software\"), to deal\n * in the Software without restriction, including without limitation the rights\n * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell\n * copies of the Software, and to permit persons to whom the Software is\n * furnished to do so, subject to the following conditions:\n *\n * The above copyright notice and this permission notice shall be included in all\n * copies or substantial portions of the Software.\n *\n * THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\n * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\n * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE\n * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\n * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,\n * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE\n * SOFTWARE.\n */\n\nimport '@fgv/ts-utils-jest';\nimport {\n DirectoryHandleStore,\n DEFAULT_DIRECTORY_HANDLE_DB,\n DEFAULT_DIRECTORY_HANDLE_STORE\n} from '../../packlets/file-tree/directoryHandleStore';\nimport type { FileSystemDirectoryHandle } from '../../packlets/file-api-types';\n\nimport { get, set, del, keys, createStore } from 'idb-keyval';\n\nconst mockGet = jest.mocked(get);\nconst mockSet = jest.mocked(set);\nconst mockDel = jest.mocked(del);\nconst mockKeys = jest.mocked(keys);\nconst mockCreateStore = jest.mocked(createStore);\n\nfunction makeMockHandle(name: string): FileSystemDirectoryHandle {\n return {\n kind: 'directory',\n name,\n isSameEntry: jest.fn(),\n queryPermission: jest.fn(),\n requestPermission: jest.fn(),\n getDirectoryHandle: jest.fn(),\n getFileHandle: jest.fn(),\n removeEntry: jest.fn(),\n resolve: jest.fn(),\n keys: jest.fn(),\n values: jest.fn(),\n entries: jest.fn(),\n [Symbol.asyncIterator]: jest.fn()\n } as FileSystemDirectoryHandle;\n}\n\ndescribe('DirectoryHandleStore', () => {\n beforeEach(() => {\n jest.clearAllMocks();\n });\n\n describe('constants', () => {\n test('DEFAULT_DIRECTORY_HANDLE_DB has expected value', () => {\n expect(DEFAULT_DIRECTORY_HANDLE_DB).toBe('chocolate-lab-storage');\n });\n\n test('DEFAULT_DIRECTORY_HANDLE_STORE has expected value', () => {\n expect(DEFAULT_DIRECTORY_HANDLE_STORE).toBe('directory-handles');\n });\n });\n\n describe('constructor', () => {\n test('uses default db and store names', () => {\n new DirectoryHandleStore();\n expect(mockCreateStore).toHaveBeenCalledWith(\n DEFAULT_DIRECTORY_HANDLE_DB,\n DEFAULT_DIRECTORY_HANDLE_STORE\n );\n });\n\n test('uses custom db and store names', () => {\n new DirectoryHandleStore('custom-db', 'custom-store');\n expect(mockCreateStore).toHaveBeenCalledWith('custom-db', 'custom-store');\n });\n });\n\n describe('save', () => {\n test('saves a handle successfully', async () => {\n mockSet.mockResolvedValue(undefined);\n const store = new DirectoryHandleStore();\n const handle = makeMockHandle('my-dir');\n\n const result = await store.save('my-dir', handle);\n\n expect(result).toSucceed();\n expect(mockSet).toHaveBeenCalledWith('my-dir', handle, expect.anything());\n });\n\n test('returns failure when set throws', async () => {\n mockSet.mockRejectedValue(new Error('IndexedDB unavailable'));\n const store = new DirectoryHandleStore();\n const handle = makeMockHandle('my-dir');\n\n const result = await store.save('my-dir', handle);\n\n expect(result).toFailWith(/IndexedDB unavailable/);\n });\n });\n\n describe('load', () => {\n test('returns the handle when found', async () => {\n const handle = makeMockHandle('my-dir');\n mockGet.mockResolvedValue(handle);\n const store = new DirectoryHandleStore();\n\n const result = await store.load('my-dir');\n\n expect(result).toSucceedWith(handle);\n expect(mockGet).toHaveBeenCalledWith('my-dir', expect.anything());\n });\n\n test('returns undefined when not found', async () => {\n mockGet.mockResolvedValue(undefined);\n const store = new DirectoryHandleStore();\n\n const result = await store.load('missing');\n\n expect(result).toSucceedWith(undefined);\n });\n\n test('returns failure when get throws', async () => {\n mockGet.mockRejectedValue(new Error('read error'));\n const store = new DirectoryHandleStore();\n\n const result = await store.load('my-dir');\n\n expect(result).toFailWith(/read error/);\n });\n });\n\n describe('remove', () => {\n test('removes a handle successfully', async () => {\n mockDel.mockResolvedValue(undefined);\n const store = new DirectoryHandleStore();\n\n const result = await store.remove('my-dir');\n\n expect(result).toSucceed();\n expect(mockDel).toHaveBeenCalledWith('my-dir', expect.anything());\n });\n\n test('returns failure when del throws', async () => {\n mockDel.mockRejectedValue(new Error('delete error'));\n const store = new DirectoryHandleStore();\n\n const result = await store.remove('my-dir');\n\n expect(result).toFailWith(/delete error/);\n });\n });\n\n describe('getAllLabels', () => {\n test('returns all keys', async () => {\n mockKeys.mockResolvedValue(['dir-a', 'dir-b']);\n const store = new DirectoryHandleStore();\n\n const result = await store.getAllLabels();\n\n expect(result).toSucceedWith(['dir-a', 'dir-b']);\n expect(mockKeys).toHaveBeenCalledWith(expect.anything());\n });\n\n test('returns empty array when no keys', async () => {\n mockKeys.mockResolvedValue([]);\n const store = new DirectoryHandleStore();\n\n const result = await store.getAllLabels();\n\n expect(result).toSucceedWith([]);\n });\n\n test('returns failure when keys throws', async () => {\n mockKeys.mockRejectedValue(new Error('keys error'));\n const store = new DirectoryHandleStore();\n\n const result = await store.getAllLabels();\n\n expect(result).toFailWith(/keys error/);\n });\n });\n\n describe('getAll', () => {\n test('returns all label/handle pairs', async () => {\n const handleA = makeMockHandle('dir-a');\n const handleB = makeMockHandle('dir-b');\n mockKeys.mockResolvedValue(['dir-a', 'dir-b']);\n mockGet.mockResolvedValueOnce(handleA).mockResolvedValueOnce(handleB);\n const store = new DirectoryHandleStore();\n\n const result = await store.getAll();\n\n expect(result).toSucceedAndSatisfy((entries) => {\n expect(entries).toHaveLength(2);\n expect(entries[0]).toEqual({ label: 'dir-a', handle: handleA });\n expect(entries[1]).toEqual({ label: 'dir-b', handle: handleB });\n });\n });\n\n test('returns empty array when no handles stored', async () => {\n mockKeys.mockResolvedValue([]);\n const store = new DirectoryHandleStore();\n\n const result = await store.getAll();\n\n expect(result).toSucceedWith([]);\n });\n\n test('skips entries where handle is undefined', async () => {\n const handleA = makeMockHandle('dir-a');\n mockKeys.mockResolvedValue(['dir-a', 'dir-b']);\n mockGet.mockResolvedValueOnce(handleA).mockResolvedValueOnce(undefined);\n const store = new DirectoryHandleStore();\n\n const result = await store.getAll();\n\n expect(result).toSucceedAndSatisfy((entries) => {\n expect(entries).toHaveLength(1);\n expect(entries[0]).toEqual({ label: 'dir-a', handle: handleA });\n });\n });\n\n test('returns failure when getAllLabels fails', async () => {\n mockKeys.mockRejectedValue(new Error('keys error'));\n const store = new DirectoryHandleStore();\n\n const result = await store.getAll();\n\n expect(result).toFailWith(/keys error/);\n });\n\n test('returns failure when load fails for a key', async () => {\n mockKeys.mockResolvedValue(['dir-a']);\n mockGet.mockRejectedValue(new Error('load error'));\n const store = new DirectoryHandleStore();\n\n const result = await store.getAll();\n\n expect(result).toFailWith(/load error/);\n });\n });\n});\n"]}
@@ -219,6 +219,16 @@ describe('File API Types', () => {
219
219
  };
220
220
  await expect((0, file_api_types_1.safeShowOpenFilePicker)(mockWindow)).rejects.toThrow('User cancelled file picker');
221
221
  });
222
+ test('returns null when user cancels (AbortError)', async () => {
223
+ const abortError = new DOMException('The user aborted a request.', 'AbortError');
224
+ const mockWindow = {
225
+ showOpenFilePicker: jest.fn().mockRejectedValue(abortError),
226
+ showSaveFilePicker: jest.fn(),
227
+ showDirectoryPicker: jest.fn()
228
+ };
229
+ const result = await (0, file_api_types_1.safeShowOpenFilePicker)(mockWindow);
230
+ expect(result).toBeNull();
231
+ });
222
232
  });
223
233
  describe('safeShowSaveFilePicker', () => {
224
234
  test('calls showSaveFilePicker when File System Access API is supported', async () => {
@@ -280,6 +290,16 @@ describe('File API Types', () => {
280
290
  };
281
291
  await expect((0, file_api_types_1.safeShowSaveFilePicker)(mockWindow)).rejects.toThrow('Save operation failed');
282
292
  });
293
+ test('returns null when user cancels (AbortError)', async () => {
294
+ const abortError = new DOMException('The user aborted a request.', 'AbortError');
295
+ const mockWindow = {
296
+ showOpenFilePicker: jest.fn(),
297
+ showSaveFilePicker: jest.fn().mockRejectedValue(abortError),
298
+ showDirectoryPicker: jest.fn()
299
+ };
300
+ const result = await (0, file_api_types_1.safeShowSaveFilePicker)(mockWindow);
301
+ expect(result).toBeNull();
302
+ });
283
303
  });
284
304
  describe('safeShowDirectoryPicker', () => {
285
305
  test('calls showDirectoryPicker when File System Access API is supported', async () => {
@@ -347,6 +367,16 @@ describe('File API Types', () => {
347
367
  };
348
368
  await expect((0, file_api_types_1.safeShowDirectoryPicker)(mockWindow)).rejects.toThrow('Directory access denied');
349
369
  });
370
+ test('returns null when user cancels (AbortError)', async () => {
371
+ const abortError = new DOMException('The user aborted a request.', 'AbortError');
372
+ const mockWindow = {
373
+ showOpenFilePicker: jest.fn(),
374
+ showSaveFilePicker: jest.fn(),
375
+ showDirectoryPicker: jest.fn().mockRejectedValue(abortError)
376
+ };
377
+ const result = await (0, file_api_types_1.safeShowDirectoryPicker)(mockWindow);
378
+ expect(result).toBeNull();
379
+ });
350
380
  });
351
381
  describe('integration tests', () => {
352
382
  test('type guards work with handles returned from safe picker functions', async () => {