@ls-stack/utils 3.65.0 → 3.66.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 (336) hide show
  1. package/dist/{arrayUtils.d.cts → arrayUtils.d.mts} +24 -23
  2. package/dist/arrayUtils.mjs +249 -0
  3. package/dist/assertions-qMxfVhSu.mjs +207 -0
  4. package/dist/{assertions.d.ts → assertions.d.mts} +4 -3
  5. package/dist/assertions.mjs +3 -0
  6. package/dist/asyncQueue.d.mts +497 -0
  7. package/dist/asyncQueue.mjs +757 -0
  8. package/dist/{awaitDebounce.d.cts → awaitDebounce.d.mts} +11 -6
  9. package/dist/awaitDebounce.mjs +54 -0
  10. package/dist/{cache.d.ts → cache.d.mts} +76 -68
  11. package/dist/cache.mjs +355 -0
  12. package/dist/castValues-DfICShCc.mjs +19 -0
  13. package/dist/{castValues.d.cts → castValues.d.mts} +3 -2
  14. package/dist/castValues.mjs +3 -0
  15. package/dist/{concurrentCalls.d.ts → concurrentCalls.d.mts} +74 -65
  16. package/dist/concurrentCalls.mjs +295 -0
  17. package/dist/consoleFmt.d.mts +55 -0
  18. package/dist/consoleFmt.mjs +63 -0
  19. package/dist/conversions-DTmwEMIu.mjs +12 -0
  20. package/dist/conversions.d.mts +4 -0
  21. package/dist/conversions.mjs +3 -0
  22. package/dist/createThrottleController.d.mts +18 -0
  23. package/dist/createThrottleController.mjs +40 -0
  24. package/dist/debounce.d.mts +47 -0
  25. package/dist/debounce.mjs +117 -0
  26. package/dist/dedent.d.mts +74 -0
  27. package/dist/dedent.mjs +80 -0
  28. package/dist/deepEqual-C7EZEixx.mjs +78 -0
  29. package/dist/{deepEqual.d.cts → deepEqual.d.mts} +3 -2
  30. package/dist/deepEqual.mjs +3 -0
  31. package/dist/{deepReplaceValues.d.cts → deepReplaceValues.d.mts} +4 -3
  32. package/dist/deepReplaceValues.mjs +61 -0
  33. package/dist/diffParser.d.mts +63 -0
  34. package/dist/diffParser.mjs +410 -0
  35. package/dist/enhancedMap.d.mts +21 -0
  36. package/dist/enhancedMap.mjs +69 -0
  37. package/dist/exhaustiveMatch.d.mts +10 -0
  38. package/dist/exhaustiveMatch.mjs +48 -0
  39. package/dist/{filterObjectOrArrayKeys.d.cts → filterObjectOrArrayKeys.d.mts} +15 -8
  40. package/dist/filterObjectOrArrayKeys.mjs +497 -0
  41. package/dist/{getAutoIncrementId.d.cts → getAutoIncrementId.d.mts} +9 -5
  42. package/dist/getAutoIncrementId.mjs +53 -0
  43. package/dist/{getCompositeKey.d.cts → getCompositeKey.d.mts} +3 -2
  44. package/dist/getCompositeKey.mjs +50 -0
  45. package/dist/{getValueStableKey.d.cts → getValueStableKey.d.mts} +5 -3
  46. package/dist/getValueStableKey.mjs +17 -0
  47. package/dist/{hash.d.cts → hash.d.mts} +3 -2
  48. package/dist/hash.mjs +28 -0
  49. package/dist/interpolate.d.mts +17 -0
  50. package/dist/interpolate.mjs +28 -0
  51. package/dist/{iteratorUtils.d.cts → iteratorUtils.d.mts} +5 -4
  52. package/dist/iteratorUtils.mjs +39 -0
  53. package/dist/keepPrevIfUnchanged.d.mts +12 -0
  54. package/dist/keepPrevIfUnchanged.mjs +9 -0
  55. package/dist/keyedMap.d.mts +76 -0
  56. package/dist/keyedMap.mjs +139 -0
  57. package/dist/keyedSet.d.mts +77 -0
  58. package/dist/keyedSet.mjs +129 -0
  59. package/dist/{levenshtein.d.cts → levenshtein.d.mts} +3 -2
  60. package/dist/levenshtein.mjs +121 -0
  61. package/dist/main.d.mts +4 -0
  62. package/dist/main.mjs +7 -0
  63. package/dist/matchPath.d.mts +50 -0
  64. package/dist/matchPath.mjs +81 -0
  65. package/dist/mathUtils-BDP1lM_z.mjs +81 -0
  66. package/dist/{mathUtils.d.cts → mathUtils.d.mts} +3 -2
  67. package/dist/mathUtils.mjs +3 -0
  68. package/dist/{mutationUtils.d.cts → mutationUtils.d.mts} +6 -5
  69. package/dist/mutationUtils.mjs +44 -0
  70. package/dist/{objUtils.d.ts → objUtils.d.mts} +8 -6
  71. package/dist/objUtils.mjs +115 -0
  72. package/dist/parallelAsyncCalls.d.mts +83 -0
  73. package/dist/parallelAsyncCalls.mjs +121 -0
  74. package/dist/partialEqual.d.mts +139 -0
  75. package/dist/partialEqual.mjs +1055 -0
  76. package/dist/promiseUtils.d.mts +9 -0
  77. package/dist/promiseUtils.mjs +17 -0
  78. package/dist/regexUtils.d.mts +18 -0
  79. package/dist/regexUtils.mjs +34 -0
  80. package/dist/{retryOnError.d.cts → retryOnError.d.mts} +38 -37
  81. package/dist/retryOnError.mjs +91 -0
  82. package/dist/{runShellCmd.d.ts → runShellCmd.d.mts} +24 -15
  83. package/dist/runShellCmd.mjs +151 -0
  84. package/dist/{safeJson.d.cts → safeJson.d.mts} +3 -2
  85. package/dist/safeJson.mjs +30 -0
  86. package/dist/{saferTyping.d.cts → saferTyping.d.mts} +4 -3
  87. package/dist/saferTyping.mjs +45 -0
  88. package/dist/serializeXML.d.mts +23 -0
  89. package/dist/serializeXML.mjs +74 -0
  90. package/dist/{shallowEqual.d.cts → shallowEqual.d.mts} +3 -2
  91. package/dist/shallowEqual.mjs +54 -0
  92. package/dist/sleep.d.mts +4 -0
  93. package/dist/sleep.mjs +7 -0
  94. package/dist/stringUtils-DjhWOiYn.mjs +113 -0
  95. package/dist/{stringUtils.d.cts → stringUtils.d.mts} +3 -2
  96. package/dist/stringUtils.mjs +3 -0
  97. package/dist/{testUtils.d.ts → testUtils.d.mts} +83 -52
  98. package/dist/testUtils.mjs +310 -0
  99. package/dist/{throttle.d.ts → throttle.d.mts} +18 -17
  100. package/dist/throttle.mjs +102 -0
  101. package/dist/time-sr2lhQRw.mjs +67 -0
  102. package/dist/{time.d.ts → time.d.mts} +8 -7
  103. package/dist/time.mjs +3 -0
  104. package/dist/{timers.d.cts → timers.d.mts} +22 -13
  105. package/dist/timers.mjs +220 -0
  106. package/dist/{tsResult.d.cts → tsResult.d.mts} +52 -48
  107. package/dist/tsResult.mjs +142 -0
  108. package/dist/typeGuards-B1mzA-Rz.mjs +128 -0
  109. package/dist/{typeGuards.d.cts → typeGuards.d.mts} +3 -2
  110. package/dist/typeGuards.mjs +3 -0
  111. package/dist/{typeUtils.d.ts → typeUtils.d.mts} +13 -34
  112. package/dist/typeUtils.mjs +1 -0
  113. package/dist/{typedStrings.d.cts → typedStrings.d.mts} +5 -4
  114. package/dist/typedStrings.mjs +131 -0
  115. package/dist/typingFnUtils-Bb8drgKF.mjs +101 -0
  116. package/dist/{typingFnUtils.d.cts → typingFnUtils.d.mts} +13 -22
  117. package/dist/typingFnUtils.mjs +3 -0
  118. package/dist/{typingTestUtils.d.cts → typingTestUtils.d.mts} +11 -15
  119. package/dist/typingTestUtils.mjs +80 -0
  120. package/dist/typingUtils.d.mts +20 -0
  121. package/dist/typingUtils.mjs +1 -0
  122. package/dist/yamlStringify.d.mts +17 -0
  123. package/dist/yamlStringify.mjs +189 -0
  124. package/package.json +65 -242
  125. package/dist/arrayUtils.cjs +0 -229
  126. package/dist/arrayUtils.d.ts +0 -171
  127. package/dist/arrayUtils.js +0 -42
  128. package/dist/assertions.cjs +0 -107
  129. package/dist/assertions.d.cts +0 -192
  130. package/dist/assertions.js +0 -25
  131. package/dist/asyncQueue.cjs +0 -672
  132. package/dist/asyncQueue.d.cts +0 -488
  133. package/dist/asyncQueue.d.ts +0 -488
  134. package/dist/asyncQueue.js +0 -631
  135. package/dist/awaitDebounce.cjs +0 -106
  136. package/dist/awaitDebounce.d.ts +0 -41
  137. package/dist/awaitDebounce.js +0 -28
  138. package/dist/cache.cjs +0 -367
  139. package/dist/cache.d.cts +0 -228
  140. package/dist/cache.js +0 -19
  141. package/dist/castValues.cjs +0 -50
  142. package/dist/castValues.d.ts +0 -4
  143. package/dist/castValues.js +0 -8
  144. package/dist/chunk-5DZT3Z5Z.js +0 -8
  145. package/dist/chunk-6FBIEPWU.js +0 -96
  146. package/dist/chunk-6FIBVC2P.js +0 -56
  147. package/dist/chunk-7CQPOM5I.js +0 -100
  148. package/dist/chunk-B6DNOZCP.js +0 -369
  149. package/dist/chunk-BM4PYVOX.js +0 -109
  150. package/dist/chunk-C2SVCIWE.js +0 -57
  151. package/dist/chunk-CCUPDGSZ.js +0 -132
  152. package/dist/chunk-DBOWTYR4.js +0 -49
  153. package/dist/chunk-DFXNVEH6.js +0 -14
  154. package/dist/chunk-DX2524CZ.js +0 -314
  155. package/dist/chunk-GMJTLFM6.js +0 -60
  156. package/dist/chunk-IATIXMCE.js +0 -20
  157. package/dist/chunk-II4R3VVX.js +0 -25
  158. package/dist/chunk-JF2MDHOJ.js +0 -40
  159. package/dist/chunk-JQFUKJU5.js +0 -71
  160. package/dist/chunk-MI4UE2PQ.js +0 -561
  161. package/dist/chunk-PUKVXYYL.js +0 -52
  162. package/dist/chunk-QQS7I7ZL.js +0 -16
  163. package/dist/chunk-VAAMRG4K.js +0 -20
  164. package/dist/chunk-WFQJUJTC.js +0 -182
  165. package/dist/chunk-ZXIKIA5B.js +0 -178
  166. package/dist/concurrentCalls.cjs +0 -406
  167. package/dist/concurrentCalls.d.cts +0 -116
  168. package/dist/concurrentCalls.js +0 -346
  169. package/dist/consoleFmt.cjs +0 -85
  170. package/dist/consoleFmt.d.cts +0 -54
  171. package/dist/consoleFmt.d.ts +0 -54
  172. package/dist/consoleFmt.js +0 -60
  173. package/dist/conversions.cjs +0 -44
  174. package/dist/conversions.d.cts +0 -3
  175. package/dist/conversions.d.ts +0 -3
  176. package/dist/conversions.js +0 -6
  177. package/dist/createThrottleController.cjs +0 -193
  178. package/dist/createThrottleController.d.cts +0 -13
  179. package/dist/createThrottleController.d.ts +0 -13
  180. package/dist/createThrottleController.js +0 -61
  181. package/dist/debounce.cjs +0 -157
  182. package/dist/debounce.d.cts +0 -46
  183. package/dist/debounce.d.ts +0 -46
  184. package/dist/debounce.js +0 -8
  185. package/dist/dedent.cjs +0 -104
  186. package/dist/dedent.d.cts +0 -73
  187. package/dist/dedent.d.ts +0 -73
  188. package/dist/dedent.js +0 -79
  189. package/dist/deepEqual.cjs +0 -96
  190. package/dist/deepEqual.d.ts +0 -21
  191. package/dist/deepEqual.js +0 -8
  192. package/dist/deepReplaceValues.cjs +0 -87
  193. package/dist/deepReplaceValues.d.ts +0 -27
  194. package/dist/deepReplaceValues.js +0 -7
  195. package/dist/enhancedMap.cjs +0 -131
  196. package/dist/enhancedMap.d.cts +0 -20
  197. package/dist/enhancedMap.d.ts +0 -20
  198. package/dist/enhancedMap.js +0 -10
  199. package/dist/exhaustiveMatch.cjs +0 -66
  200. package/dist/exhaustiveMatch.d.cts +0 -9
  201. package/dist/exhaustiveMatch.d.ts +0 -9
  202. package/dist/exhaustiveMatch.js +0 -40
  203. package/dist/filterObjectOrArrayKeys.cjs +0 -619
  204. package/dist/filterObjectOrArrayKeys.d.ts +0 -88
  205. package/dist/filterObjectOrArrayKeys.js +0 -9
  206. package/dist/getAutoIncrementId.cjs +0 -44
  207. package/dist/getAutoIncrementId.d.ts +0 -46
  208. package/dist/getAutoIncrementId.js +0 -18
  209. package/dist/getCompositeKey.cjs +0 -86
  210. package/dist/getCompositeKey.d.ts +0 -11
  211. package/dist/getCompositeKey.js +0 -8
  212. package/dist/getValueStableKey.cjs +0 -89
  213. package/dist/getValueStableKey.d.ts +0 -15
  214. package/dist/getValueStableKey.js +0 -11
  215. package/dist/hash.cjs +0 -57
  216. package/dist/hash.d.ts +0 -7
  217. package/dist/hash.js +0 -32
  218. package/dist/interpolate.cjs +0 -88
  219. package/dist/interpolate.d.cts +0 -11
  220. package/dist/interpolate.d.ts +0 -11
  221. package/dist/interpolate.js +0 -46
  222. package/dist/iteratorUtils.cjs +0 -73
  223. package/dist/iteratorUtils.d.ts +0 -10
  224. package/dist/iteratorUtils.js +0 -44
  225. package/dist/keepPrevIfUnchanged.cjs +0 -102
  226. package/dist/keepPrevIfUnchanged.d.cts +0 -7
  227. package/dist/keepPrevIfUnchanged.d.ts +0 -7
  228. package/dist/keepPrevIfUnchanged.js +0 -7
  229. package/dist/keyedMap.cjs +0 -224
  230. package/dist/keyedMap.d.cts +0 -75
  231. package/dist/keyedMap.d.ts +0 -75
  232. package/dist/keyedMap.js +0 -145
  233. package/dist/keyedSet.cjs +0 -205
  234. package/dist/keyedSet.d.cts +0 -76
  235. package/dist/keyedSet.d.ts +0 -76
  236. package/dist/keyedSet.js +0 -126
  237. package/dist/levenshtein.cjs +0 -180
  238. package/dist/levenshtein.d.ts +0 -5
  239. package/dist/levenshtein.js +0 -153
  240. package/dist/main.cjs +0 -32
  241. package/dist/main.d.cts +0 -3
  242. package/dist/main.d.ts +0 -3
  243. package/dist/main.js +0 -7
  244. package/dist/matchPath.cjs +0 -155
  245. package/dist/matchPath.d.cts +0 -53
  246. package/dist/matchPath.d.ts +0 -53
  247. package/dist/matchPath.js +0 -108
  248. package/dist/mathUtils.cjs +0 -81
  249. package/dist/mathUtils.d.ts +0 -54
  250. package/dist/mathUtils.js +0 -22
  251. package/dist/mutationUtils.cjs +0 -153
  252. package/dist/mutationUtils.d.ts +0 -15
  253. package/dist/mutationUtils.js +0 -55
  254. package/dist/objUtils.cjs +0 -242
  255. package/dist/objUtils.d.cts +0 -28
  256. package/dist/objUtils.js +0 -38
  257. package/dist/parallelAsyncCalls.cjs +0 -162
  258. package/dist/parallelAsyncCalls.d.cts +0 -82
  259. package/dist/parallelAsyncCalls.d.ts +0 -82
  260. package/dist/parallelAsyncCalls.js +0 -126
  261. package/dist/partialEqual.cjs +0 -1196
  262. package/dist/partialEqual.d.cts +0 -141
  263. package/dist/partialEqual.d.ts +0 -141
  264. package/dist/partialEqual.js +0 -1168
  265. package/dist/promiseUtils.cjs +0 -38
  266. package/dist/promiseUtils.d.cts +0 -8
  267. package/dist/promiseUtils.d.ts +0 -8
  268. package/dist/promiseUtils.js +0 -6
  269. package/dist/regexUtils.cjs +0 -60
  270. package/dist/regexUtils.d.cts +0 -17
  271. package/dist/regexUtils.d.ts +0 -17
  272. package/dist/regexUtils.js +0 -33
  273. package/dist/retryOnError.cjs +0 -130
  274. package/dist/retryOnError.d.ts +0 -83
  275. package/dist/retryOnError.js +0 -101
  276. package/dist/runShellCmd.cjs +0 -127
  277. package/dist/runShellCmd.d.cts +0 -90
  278. package/dist/runShellCmd.js +0 -98
  279. package/dist/safeJson.cjs +0 -45
  280. package/dist/safeJson.d.ts +0 -16
  281. package/dist/safeJson.js +0 -8
  282. package/dist/saferTyping.cjs +0 -52
  283. package/dist/saferTyping.d.ts +0 -47
  284. package/dist/saferTyping.js +0 -23
  285. package/dist/serializeXML.cjs +0 -154
  286. package/dist/serializeXML.d.cts +0 -22
  287. package/dist/serializeXML.d.ts +0 -22
  288. package/dist/serializeXML.js +0 -116
  289. package/dist/shallowEqual.cjs +0 -88
  290. package/dist/shallowEqual.d.ts +0 -4
  291. package/dist/shallowEqual.js +0 -63
  292. package/dist/sleep.cjs +0 -32
  293. package/dist/sleep.d.cts +0 -3
  294. package/dist/sleep.d.ts +0 -3
  295. package/dist/sleep.js +0 -6
  296. package/dist/stringUtils.cjs +0 -155
  297. package/dist/stringUtils.d.ts +0 -55
  298. package/dist/stringUtils.js +0 -50
  299. package/dist/testUtils.cjs +0 -1490
  300. package/dist/testUtils.d.cts +0 -133
  301. package/dist/testUtils.js +0 -359
  302. package/dist/throttle.cjs +0 -282
  303. package/dist/throttle.d.cts +0 -98
  304. package/dist/throttle.js +0 -38
  305. package/dist/time.cjs +0 -152
  306. package/dist/time.d.cts +0 -25
  307. package/dist/time.js +0 -38
  308. package/dist/timers.cjs +0 -194
  309. package/dist/timers.d.ts +0 -121
  310. package/dist/timers.js +0 -156
  311. package/dist/tsResult.cjs +0 -226
  312. package/dist/tsResult.d.ts +0 -114
  313. package/dist/tsResult.js +0 -180
  314. package/dist/typeGuards.cjs +0 -70
  315. package/dist/typeGuards.d.ts +0 -111
  316. package/dist/typeGuards.js +0 -18
  317. package/dist/typeUtils.cjs +0 -18
  318. package/dist/typeUtils.d.cts +0 -61
  319. package/dist/typeUtils.js +0 -0
  320. package/dist/typedStrings.cjs +0 -90
  321. package/dist/typedStrings.d.ts +0 -163
  322. package/dist/typedStrings.js +0 -57
  323. package/dist/typingFnUtils.cjs +0 -96
  324. package/dist/typingFnUtils.d.ts +0 -100
  325. package/dist/typingFnUtils.js +0 -30
  326. package/dist/typingTestUtils.cjs +0 -52
  327. package/dist/typingTestUtils.d.ts +0 -79
  328. package/dist/typingTestUtils.js +0 -27
  329. package/dist/typingUtils.cjs +0 -18
  330. package/dist/typingUtils.d.cts +0 -35
  331. package/dist/typingUtils.d.ts +0 -35
  332. package/dist/typingUtils.js +0 -0
  333. package/dist/yamlStringify.cjs +0 -423
  334. package/dist/yamlStringify.d.cts +0 -10
  335. package/dist/yamlStringify.d.ts +0 -10
  336. package/dist/yamlStringify.js +0 -9
@@ -0,0 +1,757 @@
1
+ import { defer } from "./promiseUtils.mjs";
2
+ import { p as durationObjToMs } from "./time-sr2lhQRw.mjs";
3
+ import { evtmitter } from "evtmitter";
4
+ import { Result, isResult, resultify, unknownToError } from "t-result";
5
+
6
+ //#region src/asyncQueue.ts
7
+ /**
8
+ * @file AsyncQueue - A powerful, type-safe async task queue with advanced error
9
+ * handling
10
+ *
11
+ * Features:
12
+ *
13
+ * - Concurrency control with configurable limits
14
+ * - Error handling with stop-on-error and reject-pending options
15
+ * - Lazy start capability for batch preparation
16
+ * - Pause/resume functionality for flow control
17
+ * - Abort signal support for cancellation
18
+ * - Timeout support per task or globally
19
+ * - Event emission for progress tracking
20
+ * - Metadata support for task context
21
+ * - Reset functionality for error recovery
22
+ *
23
+ * @example
24
+ * Basic Usage
25
+ * ```typescript
26
+ * const queue = createAsyncQueue<string>({ concurrency: 3 });
27
+ *
28
+ * queue.resultifyAdd(async () => {
29
+ * const response = await fetch('/api/data');
30
+ * return response.json();
31
+ * }).then(result => {
32
+ * if (result.ok) console.log('Success:', result.value);
33
+ * else console.error('Error:', result.error);
34
+ * });
35
+ *
36
+ * await queue.onIdle(); // Wait for all tasks to complete
37
+ * ```
38
+ *
39
+ * @example
40
+ * Error Handling
41
+ * ```typescript
42
+ * const queue = createAsyncQueue<string>({
43
+ * stopOnError: true,
44
+ * rejectPendingOnError: true
45
+ * });
46
+ *
47
+ * // Process batch with automatic error recovery
48
+ * const items = ['item1', 'item2', 'bad-item', 'item3'];
49
+ * for (const item of items) {
50
+ * queue.resultifyAdd(async () => processItem(item));
51
+ * }
52
+ *
53
+ * await queue.onIdle();
54
+ *
55
+ * if (queue.isStopped) {
56
+ * console.log(`Queue stopped after processing ${queue.completed} items`);
57
+ * queue.reset(); // Resume processing remaining items
58
+ * }
59
+ * ```
60
+ */
61
+ /**
62
+ * A powerful async task queue with advanced error handling and flow control
63
+ *
64
+ * @example
65
+ * Basic Usage
66
+ * ```typescript
67
+ * const queue = createAsyncQueue<string>({ concurrency: 2 });
68
+ *
69
+ * const processedItems: string[] = [];
70
+ *
71
+ * queue.resultifyAdd(async () => {
72
+ * await delay(100);
73
+ * return 'task completed';
74
+ * }).then(result => {
75
+ * if (result.ok) processedItems.push(result.value);
76
+ * });
77
+ *
78
+ * await queue.onIdle();
79
+ * console.log('Processed:', processedItems);
80
+ * ```
81
+ *
82
+ * @example
83
+ * Error Recovery
84
+ * ```typescript
85
+ * const queue = createAsyncQueue<string>({
86
+ * stopOnError: true,
87
+ * rejectPendingOnError: false
88
+ * });
89
+ *
90
+ * // Add batch of tasks
91
+ * const items = ['item1', 'item2', 'bad-item', 'item3'];
92
+ * items.forEach(item => {
93
+ * queue.resultifyAdd(async () => {
94
+ * if (item === 'bad-item') throw new Error('Processing failed');
95
+ * return item.toUpperCase();
96
+ * });
97
+ * });
98
+ *
99
+ * await queue.onIdle();
100
+ *
101
+ * if (queue.isStopped) {
102
+ * console.log(`Stopped at ${queue.failed} failures, ${queue.size} remaining`);
103
+ * // Reset and continue with remaining tasks
104
+ * queue.reset();
105
+ * await queue.onIdle();
106
+ * }
107
+ * ```
108
+ *
109
+ * @example
110
+ * Lazy Start
111
+ * ```typescript
112
+ * const queue = createAsyncQueue<string>({ autoStart: false });
113
+ *
114
+ * // Prepare all tasks without starting
115
+ * queue.resultifyAdd(() => processTask1());
116
+ * queue.resultifyAdd(() => processTask2());
117
+ * queue.resultifyAdd(() => processTask3());
118
+ *
119
+ * // Start processing when ready
120
+ * queue.start();
121
+ * await queue.onIdle();
122
+ * ```
123
+ *
124
+ * @template T - The type of value returned by successful tasks
125
+ * @template E - The type of errors that tasks can produce (defaults to Error)
126
+ * @template I - The type of metadata associated with tasks (defaults to
127
+ * unknown)
128
+ */
129
+ var AsyncQueue = class AsyncQueue {
130
+ #queue = [];
131
+ #pending = 0;
132
+ #size = 0;
133
+ #concurrency;
134
+ #completed = 0;
135
+ #failed = 0;
136
+ #idleResolvers = [];
137
+ #sizeLessThanWaiters = [];
138
+ /**
139
+ * Event emitter for tracking task lifecycle
140
+ *
141
+ * @example
142
+ * Listening to Events
143
+ * ```typescript
144
+ * const queue = createAsyncQueue<string>();
145
+ *
146
+ * queue.events.on('start', (event) => {
147
+ * console.log('Task started:', event.payload.meta);
148
+ * });
149
+ *
150
+ * queue.events.on('complete', (event) => {
151
+ * console.log('Task completed:', event.payload.value);
152
+ * });
153
+ *
154
+ * queue.events.on('error', (event) => {
155
+ * console.error('Task failed:', event.payload.error);
156
+ * });
157
+ * ```
158
+ */
159
+ events = evtmitter();
160
+ #signal;
161
+ #taskTimeout;
162
+ #stopped = false;
163
+ #paused = false;
164
+ #started = false;
165
+ #stopOnError = false;
166
+ #rejectPendingOnError = false;
167
+ #autoStart = true;
168
+ #stoppedReason;
169
+ #rateLimit;
170
+ #taskExecutionTimes = [];
171
+ #rateLimitTimeouts = /* @__PURE__ */ new Set();
172
+ /** Array of all task failures with metadata for debugging and analysis */
173
+ failures = [];
174
+ /** Array of all task completions with metadata for debugging and analysis */
175
+ completions = [];
176
+ constructor({ concurrency = 1, signal, timeout: taskTimeout, stopOnError = false, rejectPendingOnError = false, autoStart = true, rateLimit } = {}) {
177
+ this.#concurrency = concurrency;
178
+ this.#signal = signal;
179
+ this.#taskTimeout = taskTimeout;
180
+ this.#stopOnError = stopOnError;
181
+ this.#rejectPendingOnError = rejectPendingOnError;
182
+ this.#autoStart = autoStart;
183
+ this.#started = autoStart;
184
+ this.#rateLimit = rateLimit;
185
+ this.events.on("error", (e) => {
186
+ this.failures.push(e.payload);
187
+ });
188
+ this.events.on("complete", (e) => {
189
+ this.completions.push(e.payload);
190
+ });
191
+ }
192
+ #getRateLimitIntervalMs() {
193
+ if (!this.#rateLimit) return 0;
194
+ return typeof this.#rateLimit.interval === "number" ? this.#rateLimit.interval : durationObjToMs(this.#rateLimit.interval);
195
+ }
196
+ #cleanupExpiredExecutionTimes(now) {
197
+ if (!this.#rateLimit) return;
198
+ const cutoff = now - this.#getRateLimitIntervalMs();
199
+ this.#taskExecutionTimes = this.#taskExecutionTimes.filter((time) => time > cutoff);
200
+ }
201
+ #isRateLimited() {
202
+ if (!this.#rateLimit) return false;
203
+ const now = Date.now();
204
+ this.#cleanupExpiredExecutionTimes(now);
205
+ return this.#taskExecutionTimes.length >= this.#rateLimit.maxTasks;
206
+ }
207
+ #getRateLimitDelay() {
208
+ if (!this.#rateLimit || this.#taskExecutionTimes.length === 0) return 0;
209
+ const oldestExecution = this.#taskExecutionTimes[0];
210
+ if (oldestExecution === void 0) return 0;
211
+ const timeUntilSlotOpens = oldestExecution + this.#getRateLimitIntervalMs() - Date.now();
212
+ return Math.max(0, timeUntilSlotOpens);
213
+ }
214
+ #recordTaskExecution() {
215
+ if (!this.#rateLimit) return;
216
+ const now = Date.now();
217
+ this.#taskExecutionTimes.push(now);
218
+ this.#cleanupExpiredExecutionTimes(now);
219
+ }
220
+ #enqueue(task) {
221
+ this.#queue.push(task);
222
+ this.#size++;
223
+ }
224
+ static #createTimeoutSignal(ms) {
225
+ const controller = new AbortController();
226
+ const id = setTimeout(() => {
227
+ controller.abort(new DOMException("The operation was aborted due to timeout", "TimeoutError"));
228
+ }, ms);
229
+ controller.signal.addEventListener("abort", () => {
230
+ clearTimeout(id);
231
+ }, { once: true });
232
+ return controller.signal;
233
+ }
234
+ #resolveSizeLessThanWaiters() {
235
+ if (this.#sizeLessThanWaiters.length === 0) return;
236
+ const remaining = [];
237
+ for (const waiter of this.#sizeLessThanWaiters) if (this.#size < waiter.limit) waiter.resolve();
238
+ else remaining.push(waiter);
239
+ this.#sizeLessThanWaiters = remaining;
240
+ }
241
+ /**
242
+ * Add a task that returns a Result to the queue
243
+ *
244
+ * Use this method when your task function already returns a Result type. For
245
+ * functions that throw errors or return plain values, use `resultifyAdd`
246
+ * instead.
247
+ *
248
+ * @example
249
+ * ```typescript
250
+ * const queue = createAsyncQueue<string>();
251
+ *
252
+ * const result = await queue.add(async () => {
253
+ * try {
254
+ * const data = await fetchData();
255
+ * return Result.ok(data);
256
+ * } catch (error) {
257
+ * return Result.err(error);
258
+ * }
259
+ * });
260
+ *
261
+ * if (result.ok) {
262
+ * console.log('Success:', result.value);
263
+ * } else {
264
+ * console.log('Error:', result.error);
265
+ * }
266
+ * ```;
267
+ *
268
+ * @param fn - Task function that returns a Result
269
+ * @param options - Optional configuration for this task
270
+ * @returns Promise that resolves with the task result
271
+ */
272
+ async add(fn, options) {
273
+ if (this.#signal?.aborted) return Result.err(this.#signal.reason instanceof Error ? this.#signal.reason : new DOMException("This operation was aborted", "AbortError"));
274
+ if (this.#stopped) return Result.err(this.#stoppedReason ?? /* @__PURE__ */ new Error("Queue has been stopped"));
275
+ const deferred = defer();
276
+ const taskTimeout = this.#taskTimeout ?? options?.timeout;
277
+ const task = {
278
+ run: async (ctx) => {
279
+ return fn(ctx);
280
+ },
281
+ resolve: deferred.resolve,
282
+ reject: deferred.reject,
283
+ signal: options?.signal,
284
+ meta: options?.meta,
285
+ timeout: taskTimeout
286
+ };
287
+ this.#enqueue(task);
288
+ if (this.#autoStart && this.#started) this.#processQueue();
289
+ const r = await deferred.promise;
290
+ if (options?.onComplete) r.onOk(options.onComplete);
291
+ if (options?.onError) r.onErr(options.onError);
292
+ return r;
293
+ }
294
+ /**
295
+ * Add a task that returns a plain value or throws errors to the queue
296
+ *
297
+ * This is the most commonly used method. It automatically wraps your function
298
+ * to handle errors and convert them to Result types.
299
+ *
300
+ * @example
301
+ * Basic Usage
302
+ * ```typescript
303
+ * const queue = createAsyncQueue<string>();
304
+ *
305
+ * queue.resultifyAdd(async () => {
306
+ * const response = await fetch('/api/data');
307
+ * return response.json();
308
+ * }).then(result => {
309
+ * if (result.ok) {
310
+ * console.log('Data:', result.value);
311
+ * } else {
312
+ * console.error('Failed:', result.error);
313
+ * }
314
+ * });
315
+ * ```
316
+ *
317
+ * @example
318
+ * With Callbacks
319
+ * ```typescript
320
+ * queue.resultifyAdd(
321
+ * async () => processData(),
322
+ * {
323
+ * onComplete: (data) => console.log('Processed:', data),
324
+ * onError: (error) => console.error('Failed:', error),
325
+ * timeout: 5000
326
+ * }
327
+ * );
328
+ * ```
329
+ *
330
+ * @param fn - Task function that returns a value or throws
331
+ * @param options - Optional configuration for this task
332
+ * @returns Promise that resolves with the task result wrapped in Result
333
+ */
334
+ resultifyAdd(fn, options) {
335
+ return this.add((ctx) => resultify(async () => {
336
+ return fn(ctx);
337
+ }), options);
338
+ }
339
+ async #processQueue() {
340
+ if (this.#signal?.aborted) {
341
+ this.clear();
342
+ return;
343
+ }
344
+ if (this.#stopped || this.#paused || !this.#started) return;
345
+ if (this.#pending >= this.#concurrency || this.#queue.length === 0) return;
346
+ if (this.#isRateLimited()) {
347
+ const delay = this.#getRateLimitDelay();
348
+ if (delay > 0) {
349
+ const timeoutId = setTimeout(() => {
350
+ this.#rateLimitTimeouts.delete(timeoutId);
351
+ this.#processQueue();
352
+ }, delay);
353
+ this.#rateLimitTimeouts.add(timeoutId);
354
+ return;
355
+ }
356
+ }
357
+ const task = this.#queue.shift();
358
+ if (!task) return;
359
+ this.#pending++;
360
+ this.#size--;
361
+ this.#resolveSizeLessThanWaiters();
362
+ this.#recordTaskExecution();
363
+ const signals = [];
364
+ if (task.signal) signals.push(task.signal);
365
+ if (this.#signal) signals.push(this.#signal);
366
+ if (task.timeout !== void 0) signals.push(AsyncQueue.#createTimeoutSignal(task.timeout));
367
+ const signal = signals.length > 1 ? AbortSignal.any(signals) : signals[0];
368
+ let abortListener;
369
+ try {
370
+ if (signal?.aborted) throw signal.reason instanceof Error ? signal.reason : new DOMException("This operation was aborted", "AbortError");
371
+ const signalAbortPromise = new Promise((_, reject) => {
372
+ if (signal) {
373
+ abortListener = () => {
374
+ const reason = signal.reason;
375
+ const err$1 = reason instanceof Error ? reason : new DOMException("This operation was aborted", "AbortError");
376
+ setTimeout(() => {
377
+ reject(err$1);
378
+ }, 0);
379
+ };
380
+ signal.addEventListener("abort", abortListener, { once: true });
381
+ }
382
+ });
383
+ const taskRunPromise = task.run({
384
+ signal,
385
+ meta: task.meta
386
+ });
387
+ this.events.emit("start", { meta: task.meta });
388
+ const result = await Promise.race([taskRunPromise, signalAbortPromise]);
389
+ if (isResult(result)) {
390
+ task.resolve(result);
391
+ if (!result.ok) {
392
+ this.#failed++;
393
+ this.events.emit("error", {
394
+ meta: task.meta,
395
+ error: result.error
396
+ });
397
+ this.#stopOnErrorAndRejectPending(unknownToError(result.error));
398
+ } else {
399
+ this.#completed++;
400
+ this.events.emit("complete", {
401
+ meta: task.meta,
402
+ value: result.value
403
+ });
404
+ }
405
+ } else {
406
+ const error = /* @__PURE__ */ new Error("Response not a Result");
407
+ task.resolve(Result.err(error));
408
+ this.#failed++;
409
+ this.events.emit("error", {
410
+ meta: task.meta,
411
+ error
412
+ });
413
+ this.#stopOnErrorAndRejectPending(error);
414
+ }
415
+ } catch (error) {
416
+ const processedError = unknownToError(error);
417
+ task.resolve(Result.err(processedError));
418
+ this.#failed++;
419
+ this.events.emit("error", {
420
+ meta: task.meta,
421
+ error: processedError
422
+ });
423
+ this.#stopOnErrorAndRejectPending(processedError);
424
+ } finally {
425
+ if (signal && abortListener) signal.removeEventListener("abort", abortListener);
426
+ this.#pending--;
427
+ this.#processQueue();
428
+ if (this.#pending === 0 && this.#size === 0 && this.#rateLimitTimeouts.size === 0) this.#resolveIdleWaiters();
429
+ }
430
+ }
431
+ #resolveIdleWaiters() {
432
+ while (this.#idleResolvers.length > 0) {
433
+ const resolve = this.#idleResolvers.shift();
434
+ if (resolve) resolve();
435
+ }
436
+ }
437
+ #stopOnErrorAndRejectPending(error) {
438
+ if (!this.#stopOnError) return;
439
+ this.#stopped = true;
440
+ this.#stoppedReason = error;
441
+ if (this.#rejectPendingOnError) {
442
+ while (this.#queue.length > 0) {
443
+ const task = this.#queue.shift();
444
+ if (task) task.resolve(Result.err(error));
445
+ }
446
+ this.#size = 0;
447
+ this.#resolveSizeLessThanWaiters();
448
+ }
449
+ this.#resolveIdleWaiters();
450
+ }
451
+ /**
452
+ * Wait for the queue to become idle (no pending tasks, no queued tasks, and
453
+ * no rate-limit timers)
454
+ *
455
+ * This method resolves when:
456
+ *
457
+ * - All tasks have completed (success or failure)
458
+ * - The queue is stopped due to error (stopOnError), even with remaining tasks
459
+ * - There are no queued tasks, no running tasks, and no pending rate-limit
460
+ * timers
461
+ *
462
+ * @example
463
+ * ```typescript
464
+ * const queue = createAsyncQueue<string>();
465
+ *
466
+ * // Add multiple tasks
467
+ * for (let i = 0; i < 10; i++) {
468
+ * queue.resultifyAdd(async () => `task ${i}`);
469
+ * }
470
+ *
471
+ * // Wait for all tasks to complete
472
+ * await queue.onIdle();
473
+ *
474
+ * console.log(`Completed: ${queue.completed}, Failed: ${queue.failed}`);
475
+ * ```
476
+ *
477
+ * @returns Promise that resolves when the queue is idle
478
+ */
479
+ async onIdle() {
480
+ if (this.#stopped || this.#pending === 0 && this.#size === 0 && this.#rateLimitTimeouts.size === 0) return Promise.resolve();
481
+ return new Promise((resolve) => {
482
+ this.#idleResolvers.push(resolve);
483
+ });
484
+ }
485
+ /**
486
+ * Wait until the queued task count is below a limit
487
+ *
488
+ * Resolves immediately if `size < limit` at the moment of calling. This only
489
+ * considers queued (not yet started) tasks; running tasks are tracked by
490
+ * `pending`.
491
+ *
492
+ * @param limit Threshold that `size` must be below to resolve
493
+ */
494
+ onSizeLessThan(limit) {
495
+ if (this.#size < limit) return Promise.resolve();
496
+ return new Promise((resolve) => {
497
+ this.#sizeLessThanWaiters.push({
498
+ limit,
499
+ resolve
500
+ });
501
+ });
502
+ }
503
+ /**
504
+ * Clear all queued tasks (does not affect currently running tasks)
505
+ *
506
+ * This removes all tasks waiting in the queue but allows currently executing
507
+ * tasks to complete normally.
508
+ *
509
+ * @example
510
+ * ```typescript
511
+ * const queue = createAsyncQueue({ concurrency: 1 });
512
+ *
513
+ * // Add multiple tasks
514
+ * queue.resultifyAdd(async () => longRunningTask()); // Will start immediately
515
+ * queue.resultifyAdd(async () => task2()); // Queued
516
+ * queue.resultifyAdd(async () => task3()); // Queued
517
+ *
518
+ * // Clear remaining queued tasks
519
+ * queue.clear();
520
+ *
521
+ * // Only the first task will complete
522
+ * await queue.onIdle();
523
+ * ```;
524
+ */
525
+ clear() {
526
+ this.#queue = [];
527
+ this.#size = 0;
528
+ for (const timeoutId of this.#rateLimitTimeouts) clearTimeout(timeoutId);
529
+ this.#rateLimitTimeouts.clear();
530
+ if (this.#pending === 0) this.#resolveIdleWaiters();
531
+ this.#resolveSizeLessThanWaiters();
532
+ }
533
+ /** Number of tasks that have completed successfully */
534
+ get completed() {
535
+ return this.#completed;
536
+ }
537
+ /** Number of tasks that have failed */
538
+ get failed() {
539
+ return this.#failed;
540
+ }
541
+ /** Number of tasks currently being processed */
542
+ get pending() {
543
+ return this.#pending;
544
+ }
545
+ /** Number of tasks waiting in the queue to be processed */
546
+ get size() {
547
+ return this.#size;
548
+ }
549
+ /**
550
+ * Manually start processing tasks (only needed if autoStart: false)
551
+ *
552
+ * @example
553
+ * ```typescript
554
+ * const queue = createAsyncQueue({ autoStart: false });
555
+ *
556
+ * // Add tasks without starting processing
557
+ * queue.resultifyAdd(async () => 'task1');
558
+ * queue.resultifyAdd(async () => 'task2');
559
+ *
560
+ * // Start processing when ready
561
+ * queue.start();
562
+ * await queue.onIdle();
563
+ * ```;
564
+ */
565
+ start() {
566
+ if (this.#stopped) return;
567
+ this.#started = true;
568
+ this.#processQueue();
569
+ }
570
+ /**
571
+ * Pause processing new tasks (currently running tasks continue)
572
+ *
573
+ * @example
574
+ * ```typescript
575
+ * const queue = createAsyncQueue();
576
+ *
577
+ * // Start some tasks
578
+ * queue.resultifyAdd(async () => longRunningTask1());
579
+ * queue.resultifyAdd(async () => longRunningTask2());
580
+ *
581
+ * // Pause before more tasks are picked up
582
+ * queue.pause();
583
+ *
584
+ * // Later, resume processing
585
+ * queue.resume();
586
+ * ```;
587
+ */
588
+ pause() {
589
+ this.#paused = true;
590
+ }
591
+ /** Resume processing tasks after pause */
592
+ resume() {
593
+ this.#paused = false;
594
+ if (this.#started && !this.#stopped) this.#processQueue();
595
+ }
596
+ /**
597
+ * Reset the queue after being stopped, allowing new tasks to be processed
598
+ *
599
+ * This clears the stopped state and error reason, and resumes processing any
600
+ * remaining queued tasks if autoStart was enabled.
601
+ *
602
+ * @example
603
+ * ```typescript
604
+ * const queue = createAsyncQueue({ stopOnError: true });
605
+ *
606
+ * // Add tasks that will cause the queue to stop
607
+ * queue.resultifyAdd(async () => { throw new Error('fail'); });
608
+ * queue.resultifyAdd(async () => 'remaining task');
609
+ *
610
+ * await queue.onIdle();
611
+ *
612
+ * if (queue.isStopped) {
613
+ * console.log(`Queue stopped, ${queue.size} tasks remaining`);
614
+ *
615
+ * // Reset and process remaining tasks
616
+ * queue.reset();
617
+ * await queue.onIdle();
618
+ * }
619
+ * ```
620
+ */
621
+ reset() {
622
+ this.#stopped = false;
623
+ this.#stoppedReason = void 0;
624
+ if (this.#autoStart) {
625
+ this.#started = true;
626
+ this.#processQueue();
627
+ }
628
+ }
629
+ /** Whether the queue is stopped due to an error */
630
+ get isStopped() {
631
+ return this.#stopped;
632
+ }
633
+ /** Whether the queue is currently paused */
634
+ get isPaused() {
635
+ return this.#paused;
636
+ }
637
+ /** Whether the queue has been started (relevant for autoStart: false) */
638
+ get isStarted() {
639
+ return this.#started;
640
+ }
641
+ /** The error that caused the queue to stop (if any) */
642
+ get stoppedReason() {
643
+ return this.#stoppedReason;
644
+ }
645
+ };
646
+ /**
647
+ * AsyncQueue variant that requires metadata for all tasks
648
+ *
649
+ * This class enforces that every task must include metadata, which is useful
650
+ * when you need to track or identify tasks consistently.
651
+ *
652
+ * @example
653
+ * ```typescript
654
+ * interface TaskMeta {
655
+ * id: string;
656
+ * priority: number;
657
+ * }
658
+ *
659
+ * const queue = createAsyncQueueWithMeta<string, TaskMeta>({ concurrency: 2 });
660
+ *
661
+ * queue.resultifyAdd(
662
+ * async () => processImportantTask(),
663
+ * { meta: { id: 'task-1', priority: 1 } }
664
+ * );
665
+ *
666
+ * // Listen to events with metadata
667
+ * queue.events.on('complete', (event) => {
668
+ * console.log(`Task ${event.payload.meta.id} completed`);
669
+ * });
670
+ * ```
671
+ *
672
+ * @template T - The type of value returned by successful tasks
673
+ * @template I - The type of metadata (required for all tasks)
674
+ * @template E - The type of errors that tasks can produce
675
+ */
676
+ var AsyncQueueWithMeta = class extends AsyncQueue {
677
+ constructor(options) {
678
+ super(options);
679
+ }
680
+ add(fn, options) {
681
+ return super.add(fn, options);
682
+ }
683
+ resultifyAdd(fn, options) {
684
+ return super.resultifyAdd(fn, options);
685
+ }
686
+ };
687
+ /**
688
+ * Create a new AsyncQueue instance
689
+ *
690
+ * @example
691
+ * Basic Queue
692
+ * ```typescript
693
+ * const queue = createAsyncQueue<string>({ concurrency: 3 });
694
+ * ```
695
+ *
696
+ * @example
697
+ * Error Handling Queue
698
+ * ```typescript
699
+ * const queue = createAsyncQueue<string>({
700
+ * concurrency: 2,
701
+ * stopOnError: true,
702
+ * rejectPendingOnError: true
703
+ * });
704
+ * ```
705
+ *
706
+ * @example
707
+ * Lazy Start Queue
708
+ * ```typescript
709
+ * const queue = createAsyncQueue<string>({
710
+ * autoStart: false,
711
+ * concurrency: 1
712
+ * });
713
+ * ```
714
+ *
715
+ * @template T - The type of value returned by successful tasks
716
+ * @template E - The type of errors that tasks can produce (defaults to Error)
717
+ * @param options - Configuration options for the queue
718
+ * @returns A new AsyncQueue instance
719
+ */
720
+ function createAsyncQueue(options) {
721
+ return new AsyncQueue(options);
722
+ }
723
+ /**
724
+ * Create a new AsyncQueueWithMeta instance that requires metadata for all tasks
725
+ *
726
+ * @example
727
+ * ```typescript
728
+ * interface TaskInfo {
729
+ * taskId: string;
730
+ * userId: string;
731
+ * }
732
+ *
733
+ * const queue = createAsyncQueueWithMeta<ProcessResult, TaskInfo>({
734
+ * concurrency: 5
735
+ * });
736
+ *
737
+ * queue.resultifyAdd(
738
+ * async (ctx) => {
739
+ * console.log(`Processing task ${ctx.meta.taskId} for user ${ctx.meta.userId}`);
740
+ * return await processUserTask(ctx.meta.userId);
741
+ * },
742
+ * { meta: { taskId: '123', userId: 'user456' } }
743
+ * );
744
+ * ```
745
+ *
746
+ * @template T - The type of value returned by successful tasks
747
+ * @template I - The type of metadata (required for all tasks)
748
+ * @template E - The type of errors that tasks can produce (defaults to Error)
749
+ * @param options - Configuration options for the queue
750
+ * @returns A new AsyncQueueWithMeta instance
751
+ */
752
+ function createAsyncQueueWithMeta(options) {
753
+ return new AsyncQueueWithMeta(options);
754
+ }
755
+
756
+ //#endregion
757
+ export { createAsyncQueue, createAsyncQueueWithMeta };