@sonamu-kit/tasks 0.0.1

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 (280) hide show
  1. package/.swcrc +17 -0
  2. package/README.md +7 -0
  3. package/dist/backend.d.ts +107 -0
  4. package/dist/backend.d.ts.map +1 -0
  5. package/dist/backend.js +3 -0
  6. package/dist/backend.js.map +1 -0
  7. package/dist/chaos.test.d.ts +2 -0
  8. package/dist/chaos.test.d.ts.map +1 -0
  9. package/dist/chaos.test.js +92 -0
  10. package/dist/chaos.test.js.map +1 -0
  11. package/dist/client.d.ts +178 -0
  12. package/dist/client.d.ts.map +1 -0
  13. package/dist/client.js +223 -0
  14. package/dist/client.js.map +1 -0
  15. package/dist/client.test.d.ts +2 -0
  16. package/dist/client.test.d.ts.map +1 -0
  17. package/dist/client.test.js +339 -0
  18. package/dist/client.test.js.map +1 -0
  19. package/dist/config.d.ts +22 -0
  20. package/dist/config.d.ts.map +1 -0
  21. package/dist/config.js +23 -0
  22. package/dist/config.js.map +1 -0
  23. package/dist/config.test.d.ts +2 -0
  24. package/dist/config.test.d.ts.map +1 -0
  25. package/dist/config.test.js +24 -0
  26. package/dist/config.test.js.map +1 -0
  27. package/dist/core/duration.d.ts +22 -0
  28. package/dist/core/duration.d.ts.map +1 -0
  29. package/dist/core/duration.js +64 -0
  30. package/dist/core/duration.js.map +1 -0
  31. package/dist/core/duration.test.d.ts +2 -0
  32. package/dist/core/duration.test.d.ts.map +1 -0
  33. package/dist/core/duration.test.js +265 -0
  34. package/dist/core/duration.test.js.map +1 -0
  35. package/dist/core/error.d.ts +15 -0
  36. package/dist/core/error.d.ts.map +1 -0
  37. package/dist/core/error.js +25 -0
  38. package/dist/core/error.js.map +1 -0
  39. package/dist/core/error.test.d.ts +2 -0
  40. package/dist/core/error.test.d.ts.map +1 -0
  41. package/dist/core/error.test.js +63 -0
  42. package/dist/core/error.test.js.map +1 -0
  43. package/dist/core/json.d.ts +5 -0
  44. package/dist/core/json.d.ts.map +1 -0
  45. package/dist/core/json.js +3 -0
  46. package/dist/core/json.js.map +1 -0
  47. package/dist/core/result.d.ts +22 -0
  48. package/dist/core/result.d.ts.map +1 -0
  49. package/dist/core/result.js +22 -0
  50. package/dist/core/result.js.map +1 -0
  51. package/dist/core/result.test.d.ts +2 -0
  52. package/dist/core/result.test.d.ts.map +1 -0
  53. package/dist/core/result.test.js +19 -0
  54. package/dist/core/result.test.js.map +1 -0
  55. package/dist/core/retry.d.ts +21 -0
  56. package/dist/core/retry.d.ts.map +1 -0
  57. package/dist/core/retry.js +25 -0
  58. package/dist/core/retry.js.map +1 -0
  59. package/dist/core/retry.test.d.ts +2 -0
  60. package/dist/core/retry.test.d.ts.map +1 -0
  61. package/dist/core/retry.test.js +37 -0
  62. package/dist/core/retry.test.js.map +1 -0
  63. package/dist/core/schema.d.ts +57 -0
  64. package/dist/core/schema.d.ts.map +1 -0
  65. package/dist/core/schema.js +4 -0
  66. package/dist/core/schema.js.map +1 -0
  67. package/dist/core/step.d.ts +96 -0
  68. package/dist/core/step.d.ts.map +1 -0
  69. package/dist/core/step.js +78 -0
  70. package/dist/core/step.js.map +1 -0
  71. package/dist/core/step.test.d.ts +2 -0
  72. package/dist/core/step.test.d.ts.map +1 -0
  73. package/dist/core/step.test.js +356 -0
  74. package/dist/core/step.test.js.map +1 -0
  75. package/dist/core/workflow.d.ts +78 -0
  76. package/dist/core/workflow.d.ts.map +1 -0
  77. package/dist/core/workflow.js +46 -0
  78. package/dist/core/workflow.js.map +1 -0
  79. package/dist/core/workflow.test.d.ts +2 -0
  80. package/dist/core/workflow.test.d.ts.map +1 -0
  81. package/dist/core/workflow.test.js +172 -0
  82. package/dist/core/workflow.test.js.map +1 -0
  83. package/dist/database/backend.d.ts +60 -0
  84. package/dist/database/backend.d.ts.map +1 -0
  85. package/dist/database/backend.js +387 -0
  86. package/dist/database/backend.js.map +1 -0
  87. package/dist/database/backend.test.d.ts +2 -0
  88. package/dist/database/backend.test.d.ts.map +1 -0
  89. package/dist/database/backend.test.js +17 -0
  90. package/dist/database/backend.test.js.map +1 -0
  91. package/dist/database/backend.testsuite.d.ts +20 -0
  92. package/dist/database/backend.testsuite.d.ts.map +1 -0
  93. package/dist/database/backend.testsuite.js +1174 -0
  94. package/dist/database/backend.testsuite.js.map +1 -0
  95. package/dist/database/base.d.ts +12 -0
  96. package/dist/database/base.d.ts.map +1 -0
  97. package/dist/database/base.js +19 -0
  98. package/dist/database/base.js.map +1 -0
  99. package/dist/database/migrations/20251212000000_0_init.js +9 -0
  100. package/dist/database/migrations/20251212000000_0_init.js.map +1 -0
  101. package/dist/database/migrations/20251212000000_1_tables.js +88 -0
  102. package/dist/database/migrations/20251212000000_1_tables.js.map +1 -0
  103. package/dist/database/migrations/20251212000000_2_fk.js +48 -0
  104. package/dist/database/migrations/20251212000000_2_fk.js.map +1 -0
  105. package/dist/database/migrations/20251212000000_3_indexes.js +107 -0
  106. package/dist/database/migrations/20251212000000_3_indexes.js.map +1 -0
  107. package/dist/database/pubsub.d.ts +17 -0
  108. package/dist/database/pubsub.d.ts.map +1 -0
  109. package/dist/database/pubsub.js +70 -0
  110. package/dist/database/pubsub.js.map +1 -0
  111. package/dist/database/pubsub.test.d.ts +2 -0
  112. package/dist/database/pubsub.test.d.ts.map +1 -0
  113. package/dist/database/pubsub.test.js +86 -0
  114. package/dist/database/pubsub.test.js.map +1 -0
  115. package/dist/errors.d.ts +8 -0
  116. package/dist/errors.d.ts.map +1 -0
  117. package/dist/errors.js +21 -0
  118. package/dist/errors.js.map +1 -0
  119. package/dist/execution.d.ts +82 -0
  120. package/dist/execution.d.ts.map +1 -0
  121. package/dist/execution.js +182 -0
  122. package/dist/execution.js.map +1 -0
  123. package/dist/execution.test.d.ts +2 -0
  124. package/dist/execution.test.d.ts.map +1 -0
  125. package/dist/execution.test.js +556 -0
  126. package/dist/execution.test.js.map +1 -0
  127. package/dist/index.d.ts +8 -0
  128. package/dist/index.d.ts.map +1 -0
  129. package/dist/index.js +6 -0
  130. package/dist/index.js.map +1 -0
  131. package/dist/internal.d.ts +12 -0
  132. package/dist/internal.d.ts.map +1 -0
  133. package/dist/internal.js +5 -0
  134. package/dist/internal.js.map +1 -0
  135. package/dist/practices/01-remote-workflow.d.ts +2 -0
  136. package/dist/practices/01-remote-workflow.d.ts.map +1 -0
  137. package/dist/practices/01-remote-workflow.js +69 -0
  138. package/dist/practices/01-remote-workflow.js.map +1 -0
  139. package/dist/practices/01-remote.d.ts +2 -0
  140. package/dist/practices/01-remote.d.ts.map +1 -0
  141. package/dist/practices/01-remote.js +87 -0
  142. package/dist/practices/01-remote.js.map +1 -0
  143. package/dist/practices/02-local.d.ts +2 -0
  144. package/dist/practices/02-local.d.ts.map +1 -0
  145. package/dist/practices/02-local.js +84 -0
  146. package/dist/practices/02-local.js.map +1 -0
  147. package/dist/practices/03-local-retry.d.ts +2 -0
  148. package/dist/practices/03-local-retry.d.ts.map +1 -0
  149. package/dist/practices/03-local-retry.js +85 -0
  150. package/dist/practices/03-local-retry.js.map +1 -0
  151. package/dist/practices/04-scheduler-dispose.d.ts +2 -0
  152. package/dist/practices/04-scheduler-dispose.d.ts.map +1 -0
  153. package/dist/practices/04-scheduler-dispose.js +65 -0
  154. package/dist/practices/04-scheduler-dispose.js.map +1 -0
  155. package/dist/practices/05-router.d.ts +2 -0
  156. package/dist/practices/05-router.d.ts.map +1 -0
  157. package/dist/practices/05-router.js +80 -0
  158. package/dist/practices/05-router.js.map +1 -0
  159. package/dist/registry.d.ts +33 -0
  160. package/dist/registry.d.ts.map +1 -0
  161. package/dist/registry.js +54 -0
  162. package/dist/registry.js.map +1 -0
  163. package/dist/registry.test.d.ts +2 -0
  164. package/dist/registry.test.d.ts.map +1 -0
  165. package/dist/registry.test.js +95 -0
  166. package/dist/registry.test.js.map +1 -0
  167. package/dist/scheduler.d.ts +22 -0
  168. package/dist/scheduler.d.ts.map +1 -0
  169. package/dist/scheduler.js +117 -0
  170. package/dist/scheduler.js.map +1 -0
  171. package/dist/tasks/index.d.ts +4 -0
  172. package/dist/tasks/index.d.ts.map +1 -0
  173. package/dist/tasks/index.js +5 -0
  174. package/dist/tasks/index.js.map +1 -0
  175. package/dist/tasks/local-task.d.ts +6 -0
  176. package/dist/tasks/local-task.d.ts.map +1 -0
  177. package/dist/tasks/local-task.js +95 -0
  178. package/dist/tasks/local-task.js.map +1 -0
  179. package/dist/tasks/remote-task.d.ts +11 -0
  180. package/dist/tasks/remote-task.d.ts.map +1 -0
  181. package/dist/tasks/remote-task.js +213 -0
  182. package/dist/tasks/remote-task.js.map +1 -0
  183. package/dist/tasks/shared.d.ts +8 -0
  184. package/dist/tasks/shared.d.ts.map +1 -0
  185. package/dist/tasks/shared.js +41 -0
  186. package/dist/tasks/shared.js.map +1 -0
  187. package/dist/testing/connection.d.ts +7 -0
  188. package/dist/testing/connection.d.ts.map +1 -0
  189. package/dist/testing/connection.js +38 -0
  190. package/dist/testing/connection.js.map +1 -0
  191. package/dist/types/config.d.ts +44 -0
  192. package/dist/types/config.d.ts.map +1 -0
  193. package/dist/types/config.js +3 -0
  194. package/dist/types/config.js.map +1 -0
  195. package/dist/types/context.d.ts +18 -0
  196. package/dist/types/context.d.ts.map +1 -0
  197. package/dist/types/context.js +4 -0
  198. package/dist/types/context.js.map +1 -0
  199. package/dist/types/events.d.ts +43 -0
  200. package/dist/types/events.d.ts.map +1 -0
  201. package/dist/types/events.js +3 -0
  202. package/dist/types/events.js.map +1 -0
  203. package/dist/types/index.d.ts +6 -0
  204. package/dist/types/index.d.ts.map +1 -0
  205. package/dist/types/index.js +3 -0
  206. package/dist/types/index.js.map +1 -0
  207. package/dist/types/task-items.d.ts +12 -0
  208. package/dist/types/task-items.d.ts.map +1 -0
  209. package/dist/types/task-items.js +3 -0
  210. package/dist/types/task-items.js.map +1 -0
  211. package/dist/types/utils.d.ts +4 -0
  212. package/dist/types/utils.d.ts.map +1 -0
  213. package/dist/types/utils.js +8 -0
  214. package/dist/types/utils.js.map +1 -0
  215. package/dist/worker.d.ts +61 -0
  216. package/dist/worker.d.ts.map +1 -0
  217. package/dist/worker.js +206 -0
  218. package/dist/worker.js.map +1 -0
  219. package/dist/worker.test.d.ts +2 -0
  220. package/dist/worker.test.d.ts.map +1 -0
  221. package/dist/worker.test.js +1163 -0
  222. package/dist/worker.test.js.map +1 -0
  223. package/dist/workflow.d.ts +44 -0
  224. package/dist/workflow.d.ts.map +1 -0
  225. package/dist/workflow.js +21 -0
  226. package/dist/workflow.js.map +1 -0
  227. package/dist/workflow.test.d.ts +2 -0
  228. package/dist/workflow.test.d.ts.map +1 -0
  229. package/dist/workflow.test.js +73 -0
  230. package/dist/workflow.test.js.map +1 -0
  231. package/nodemon.json +6 -0
  232. package/package.json +63 -0
  233. package/scripts/migrate.ts +11 -0
  234. package/src/backend.ts +133 -0
  235. package/src/chaos.test.ts +108 -0
  236. package/src/client.test.ts +297 -0
  237. package/src/client.ts +331 -0
  238. package/src/config.test.ts +23 -0
  239. package/src/config.ts +35 -0
  240. package/src/core/duration.test.ts +326 -0
  241. package/src/core/duration.ts +86 -0
  242. package/src/core/error.test.ts +77 -0
  243. package/src/core/error.ts +30 -0
  244. package/src/core/json.ts +2 -0
  245. package/src/core/result.test.ts +13 -0
  246. package/src/core/result.ts +29 -0
  247. package/src/core/retry.test.ts +41 -0
  248. package/src/core/retry.ts +29 -0
  249. package/src/core/schema.ts +74 -0
  250. package/src/core/step.test.ts +362 -0
  251. package/src/core/step.ts +152 -0
  252. package/src/core/workflow.test.ts +184 -0
  253. package/src/core/workflow.ts +127 -0
  254. package/src/database/backend.test.ts +16 -0
  255. package/src/database/backend.testsuite.ts +1376 -0
  256. package/src/database/backend.ts +655 -0
  257. package/src/database/base.ts +23 -0
  258. package/src/database/migrations/20251212000000_0_init.ts +10 -0
  259. package/src/database/migrations/20251212000000_1_tables.ts +54 -0
  260. package/src/database/migrations/20251212000000_2_fk.ts +46 -0
  261. package/src/database/migrations/20251212000000_3_indexes.ts +82 -0
  262. package/src/database/pubsub.test.ts +92 -0
  263. package/src/database/pubsub.ts +92 -0
  264. package/src/execution.test.ts +508 -0
  265. package/src/execution.ts +291 -0
  266. package/src/index.ts +7 -0
  267. package/src/internal.ts +11 -0
  268. package/src/practices/01-remote-workflow.ts +61 -0
  269. package/src/registry.test.ts +122 -0
  270. package/src/registry.ts +65 -0
  271. package/src/testing/connection.ts +44 -0
  272. package/src/worker.test.ts +1138 -0
  273. package/src/worker.ts +281 -0
  274. package/src/workflow.test.ts +68 -0
  275. package/src/workflow.ts +84 -0
  276. package/table_ddl.sql +60 -0
  277. package/templates/openworkflow.config.ts +22 -0
  278. package/tsconfig.json +40 -0
  279. package/tsconfig.test.json +4 -0
  280. package/vite.config.ts +13 -0
@@ -0,0 +1,326 @@
1
+ import { describe, expect, test } from "vitest";
2
+ import type { DurationString } from "./duration";
3
+ import { parseDuration } from "./duration";
4
+ import { err, ok } from "./result";
5
+
6
+ describe("parseDuration", () => {
7
+ describe("milliseconds", () => {
8
+ test("parses integer milliseconds", () => {
9
+ expect(parseDuration("100ms")).toEqual(ok(100));
10
+ expect(parseDuration("1ms")).toEqual(ok(1));
11
+ expect(parseDuration("5000ms")).toEqual(ok(5000));
12
+ });
13
+
14
+ test("parses decimal milliseconds", () => {
15
+ expect(parseDuration("1.5ms")).toEqual(ok(1.5));
16
+ expect(parseDuration("10.25ms")).toEqual(ok(10.25));
17
+ });
18
+
19
+ test("parses milliseconds with long format", () => {
20
+ expect(parseDuration("53 milliseconds")).toEqual(ok(53));
21
+ expect(parseDuration("17 msecs")).toEqual(ok(17));
22
+ expect(parseDuration("100 millisecond")).toEqual(ok(100));
23
+ });
24
+
25
+ test("parses numbers without unit as milliseconds", () => {
26
+ expect(parseDuration("100")).toEqual(ok(100));
27
+ expect(parseDuration("1000")).toEqual(ok(1000));
28
+ });
29
+
30
+ test("parses negative milliseconds", () => {
31
+ expect(parseDuration("-100ms")).toEqual(ok(-100));
32
+ expect(parseDuration("-100 milliseconds")).toEqual(ok(-100));
33
+ });
34
+ });
35
+
36
+ describe("seconds", () => {
37
+ test("parses integer seconds", () => {
38
+ expect(parseDuration("1s")).toEqual(ok(1000));
39
+ expect(parseDuration("5s")).toEqual(ok(5000));
40
+ expect(parseDuration("60s")).toEqual(ok(60_000));
41
+ });
42
+
43
+ test("parses decimal seconds", () => {
44
+ expect(parseDuration("1.5s")).toEqual(ok(1500));
45
+ expect(parseDuration("0.1s")).toEqual(ok(100));
46
+ expect(parseDuration("2.5s")).toEqual(ok(2500));
47
+ expect(parseDuration("0.001s")).toEqual(ok(1));
48
+ });
49
+
50
+ test("parses seconds with long format", () => {
51
+ expect(parseDuration("1 sec")).toEqual(ok(1000));
52
+ expect(parseDuration("5 seconds")).toEqual(ok(5000));
53
+ expect(parseDuration("10 secs")).toEqual(ok(10_000));
54
+ });
55
+
56
+ test("parses seconds with leading decimal", () => {
57
+ expect(parseDuration(".5s")).toEqual(ok(500));
58
+ expect(parseDuration(".5ms")).toEqual(ok(0.5));
59
+ });
60
+
61
+ test("parses negative seconds", () => {
62
+ expect(parseDuration("-5s")).toEqual(ok(-5000));
63
+ expect(parseDuration("-.5s")).toEqual(ok(-500));
64
+ });
65
+ });
66
+
67
+ describe("minutes", () => {
68
+ test("parses integer minutes", () => {
69
+ expect(parseDuration("1m")).toEqual(ok(60 * 1000));
70
+ expect(parseDuration("5m")).toEqual(ok(5 * 60 * 1000));
71
+ expect(parseDuration("30m")).toEqual(ok(30 * 60 * 1000));
72
+ });
73
+
74
+ test("parses decimal minutes", () => {
75
+ expect(parseDuration("1.5m")).toEqual(ok(1.5 * 60 * 1000));
76
+ expect(parseDuration("0.5m")).toEqual(ok(30 * 1000));
77
+ });
78
+
79
+ test("parses minutes with long format", () => {
80
+ expect(parseDuration("1 min")).toEqual(ok(60_000));
81
+ expect(parseDuration("5 minutes")).toEqual(ok(5 * 60 * 1000));
82
+ expect(parseDuration("10 mins")).toEqual(ok(10 * 60 * 1000));
83
+ });
84
+ });
85
+
86
+ describe("hours", () => {
87
+ test("parses integer hours", () => {
88
+ expect(parseDuration("1h")).toEqual(ok(60 * 60 * 1000));
89
+ expect(parseDuration("2h")).toEqual(ok(2 * 60 * 60 * 1000));
90
+ expect(parseDuration("24h")).toEqual(ok(24 * 60 * 60 * 1000));
91
+ });
92
+
93
+ test("parses decimal hours", () => {
94
+ expect(parseDuration("1.5h")).toEqual(ok(1.5 * 60 * 60 * 1000));
95
+ expect(parseDuration("0.25h")).toEqual(ok(15 * 60 * 1000));
96
+ });
97
+
98
+ test("parses hours with long format", () => {
99
+ expect(parseDuration("1 hr")).toEqual(ok(3_600_000));
100
+ expect(parseDuration("2 hours")).toEqual(ok(2 * 60 * 60 * 1000));
101
+ expect(parseDuration("3 hrs")).toEqual(ok(3 * 60 * 60 * 1000));
102
+ expect(parseDuration("1.5 hours")).toEqual(ok(5_400_000));
103
+ });
104
+
105
+ test("parses negative hours", () => {
106
+ expect(parseDuration("-1.5h")).toEqual(ok(-5_400_000));
107
+ expect(parseDuration("-10.5h")).toEqual(ok(-37_800_000));
108
+ expect(parseDuration("-.5h")).toEqual(ok(-1_800_000));
109
+ expect(parseDuration("-1.5 hours")).toEqual(ok(-5_400_000));
110
+ expect(parseDuration("-.5 hr")).toEqual(ok(-1_800_000));
111
+ });
112
+ });
113
+
114
+ describe("days", () => {
115
+ test("parses integer days", () => {
116
+ expect(parseDuration("1d")).toEqual(ok(24 * 60 * 60 * 1000));
117
+ expect(parseDuration("7d")).toEqual(ok(7 * 24 * 60 * 60 * 1000));
118
+ expect(parseDuration("30d")).toEqual(ok(30 * 24 * 60 * 60 * 1000));
119
+ });
120
+
121
+ test("parses decimal days", () => {
122
+ expect(parseDuration("1.5d")).toEqual(ok(1.5 * 24 * 60 * 60 * 1000));
123
+ expect(parseDuration("0.5d")).toEqual(ok(12 * 60 * 60 * 1000));
124
+ });
125
+
126
+ test("parses days with long format", () => {
127
+ expect(parseDuration("2 days")).toEqual(ok(172_800_000));
128
+ expect(parseDuration("1 day")).toEqual(ok(24 * 60 * 60 * 1000));
129
+ });
130
+ });
131
+
132
+ describe("weeks", () => {
133
+ test("parses integer weeks", () => {
134
+ expect(parseDuration("1w")).toEqual(ok(7 * 24 * 60 * 60 * 1000));
135
+ expect(parseDuration("2w")).toEqual(ok(2 * 7 * 24 * 60 * 60 * 1000));
136
+ expect(parseDuration("3w")).toEqual(ok(1_814_400_000));
137
+ });
138
+
139
+ test("parses decimal weeks", () => {
140
+ expect(parseDuration("1.5w")).toEqual(ok(1.5 * 7 * 24 * 60 * 60 * 1000));
141
+ expect(parseDuration("0.5w")).toEqual(ok(3.5 * 24 * 60 * 60 * 1000));
142
+ });
143
+
144
+ test("parses weeks with long format", () => {
145
+ expect(parseDuration("1 week")).toEqual(ok(604_800_000));
146
+ expect(parseDuration("2 weeks")).toEqual(ok(2 * 7 * 24 * 60 * 60 * 1000));
147
+ });
148
+ });
149
+
150
+ describe("months", () => {
151
+ test("parses integer months", () => {
152
+ expect(parseDuration("1mo")).toEqual(ok(2_629_800_000));
153
+ expect(parseDuration("2mo")).toEqual(ok(2 * 2_629_800_000));
154
+ expect(parseDuration("6mo")).toEqual(ok(6 * 2_629_800_000));
155
+ });
156
+
157
+ test("parses decimal months", () => {
158
+ expect(parseDuration("1.5mo")).toEqual(ok(1.5 * 2_629_800_000));
159
+ expect(parseDuration("0.5mo")).toEqual(ok(0.5 * 2_629_800_000));
160
+ });
161
+
162
+ test("parses months with long format", () => {
163
+ expect(parseDuration("1 month")).toEqual(ok(2_629_800_000));
164
+ expect(parseDuration("2 months")).toEqual(ok(2 * 2_629_800_000));
165
+ });
166
+ });
167
+
168
+ describe("years", () => {
169
+ test("parses integer years", () => {
170
+ expect(parseDuration("1y")).toEqual(ok(31_557_600_000));
171
+ expect(parseDuration("2y")).toEqual(ok(2 * 31_557_600_000));
172
+ expect(parseDuration("5y")).toEqual(ok(5 * 31_557_600_000));
173
+ });
174
+
175
+ test("parses decimal years", () => {
176
+ expect(parseDuration("1.5y")).toEqual(ok(1.5 * 31_557_600_000));
177
+ expect(parseDuration("0.5y")).toEqual(ok(0.5 * 31_557_600_000));
178
+ });
179
+
180
+ test("parses years with long format", () => {
181
+ expect(parseDuration("1 year")).toEqual(ok(31_557_600_000));
182
+ expect(parseDuration("2 years")).toEqual(ok(2 * 31_557_600_000));
183
+ expect(parseDuration("1 yr")).toEqual(ok(31_557_600_000));
184
+ expect(parseDuration("2 yrs")).toEqual(ok(2 * 31_557_600_000));
185
+ });
186
+ });
187
+
188
+ describe("case insensitivity", () => {
189
+ test("parses case-insensitive units", () => {
190
+ expect(parseDuration("5S")).toEqual(ok(5000));
191
+ expect(parseDuration("5M")).toEqual(ok(5 * 60 * 1000));
192
+ expect(parseDuration("5H")).toEqual(ok(5 * 60 * 60 * 1000));
193
+ expect(parseDuration("5D")).toEqual(ok(5 * 24 * 60 * 60 * 1000));
194
+ expect(parseDuration("5W")).toEqual(ok(5 * 7 * 24 * 60 * 60 * 1000));
195
+ });
196
+
197
+ test("parses case-insensitive long format", () => {
198
+ // @ts-expect-error - mixed-case (not in type but accepted at runtime)
199
+ expect(parseDuration("53 YeArS")).toEqual(ok(1_672_552_800_000));
200
+ // @ts-expect-error - mixed-case (not in type but accepted at runtime)
201
+ expect(parseDuration("53 WeEkS")).toEqual(ok(32_054_400_000));
202
+ // @ts-expect-error - mixed-case (not in type but accepted at runtime)
203
+ expect(parseDuration("53 DaYS")).toEqual(ok(4_579_200_000));
204
+ // @ts-expect-error - mixed-case (not in type but accepted at runtime)
205
+ expect(parseDuration("53 HoUrs")).toEqual(ok(190_800_000));
206
+ // @ts-expect-error - mixed-case (not in type but accepted at runtime)
207
+ expect(parseDuration("53 MiLliSeCondS")).toEqual(ok(53));
208
+ });
209
+ });
210
+
211
+ describe("whitespace handling", () => {
212
+ test("parses with single space", () => {
213
+ expect(parseDuration("1 s")).toEqual(ok(1000));
214
+ expect(parseDuration("5 m")).toEqual(ok(5 * 60 * 1000));
215
+ expect(parseDuration("2 h")).toEqual(ok(2 * 60 * 60 * 1000));
216
+ });
217
+
218
+ test("parses with multiple spaces", () => {
219
+ expect(parseDuration("1 s")).toEqual(ok(1000));
220
+ expect(parseDuration("5 m")).toEqual(ok(5 * 60 * 1000));
221
+ });
222
+ });
223
+
224
+ describe("edge cases", () => {
225
+ test("parses zero values", () => {
226
+ expect(parseDuration("0ms")).toEqual(ok(0));
227
+ expect(parseDuration("0s")).toEqual(ok(0));
228
+ expect(parseDuration("0m")).toEqual(ok(0));
229
+ expect(parseDuration("0h")).toEqual(ok(0));
230
+ expect(parseDuration("0d")).toEqual(ok(0));
231
+ expect(parseDuration("0")).toEqual(ok(0));
232
+ });
233
+
234
+ test("parses very small decimals", () => {
235
+ expect(parseDuration("0.001s")).toEqual(ok(1));
236
+ expect(parseDuration("0.1ms")).toEqual(ok(0.1));
237
+ });
238
+
239
+ test("parses large numbers", () => {
240
+ expect(parseDuration("999999ms")).toEqual(ok(999_999));
241
+ expect(parseDuration("1000s")).toEqual(ok(1_000_000));
242
+ });
243
+ });
244
+
245
+ describe("error cases", () => {
246
+ test("returns error on invalid format", () => {
247
+ // @ts-expect-error - invalid format
248
+ expect(parseDuration("invalid")).toEqual(
249
+ err(new Error('Invalid duration format: "invalid"')),
250
+ );
251
+ // @ts-expect-error - invalid format
252
+ expect(parseDuration("10-.5")).toEqual(err(new Error('Invalid duration format: "10-.5"')));
253
+ // @ts-expect-error - invalid format
254
+ expect(parseDuration("foo")).toEqual(err(new Error('Invalid duration format: "foo"')));
255
+ });
256
+
257
+ test("returns error on empty string", () => {
258
+ // @ts-expect-error - empty string
259
+ expect(parseDuration("")).toEqual(err(new Error('Invalid duration format: ""')));
260
+ });
261
+
262
+ test("returns error on missing number", () => {
263
+ // @ts-expect-error - unit without number
264
+ expect(parseDuration("ms")).toEqual(err(new Error('Invalid duration format: "ms"')));
265
+ // @ts-expect-error - unit without number
266
+ expect(parseDuration("s")).toEqual(err(new Error('Invalid duration format: "s"')));
267
+ // @ts-expect-error - unit without number
268
+ expect(parseDuration("m")).toEqual(err(new Error('Invalid duration format: "m"')));
269
+ // @ts-expect-error - unit without number
270
+ expect(parseDuration("h")).toEqual(err(new Error('Invalid duration format: "h"')));
271
+ });
272
+
273
+ test("returns error on unknown unit", () => {
274
+ // @ts-expect-error - unknown unit
275
+ expect(parseDuration("100x")).toEqual(err(new Error('Invalid duration format: "100x"')));
276
+ // @ts-expect-error - unknown unit
277
+ expect(parseDuration("5z")).toEqual(err(new Error('Invalid duration format: "5z"')));
278
+ });
279
+
280
+ test("returns error on multiple units", () => {
281
+ // @ts-expect-error - multiple units
282
+ expect(parseDuration("1h30m")).toEqual(err(new Error('Invalid duration format: "1h30m"')));
283
+ // @ts-expect-error - multiple units
284
+ expect(parseDuration("5s100ms")).toEqual(
285
+ err(new Error('Invalid duration format: "5s100ms"')),
286
+ );
287
+ });
288
+
289
+ test("returns error on leading/trailing spaces", () => {
290
+ expect(parseDuration(" 5s")).toEqual(err(new Error('Invalid duration format: " 5s"')));
291
+ // @ts-expect-error - trailing space
292
+ expect(parseDuration("5s ")).toEqual(err(new Error('Invalid duration format: "5s "')));
293
+ });
294
+
295
+ test("returns error on special characters", () => {
296
+ // @ts-expect-error - special characters
297
+ expect(parseDuration("5s!")).toEqual(err(new Error('Invalid duration format: "5s!"')));
298
+ // @ts-expect-error - special characters
299
+ expect(parseDuration("@5s")).toEqual(err(new Error('Invalid duration format: "@5s"')));
300
+ });
301
+
302
+ test("returns error on non-string types", () => {
303
+ expect(parseDuration(undefined as unknown as DurationString)).toEqual(
304
+ err(new TypeError("Invalid duration format: expected a string but received undefined")),
305
+ );
306
+ expect(parseDuration(null as unknown as DurationString)).toEqual(
307
+ err(new TypeError("Invalid duration format: expected a string but received object")),
308
+ );
309
+ expect(parseDuration([] as unknown as DurationString)).toEqual(
310
+ err(new TypeError("Invalid duration format: expected a string but received object")),
311
+ );
312
+ expect(parseDuration({} as unknown as DurationString)).toEqual(
313
+ err(new TypeError("Invalid duration format: expected a string but received object")),
314
+ );
315
+ expect(parseDuration(Number.NaN as unknown as DurationString)).toEqual(
316
+ err(new TypeError("Invalid duration format: expected a string but received number")),
317
+ );
318
+ expect(parseDuration(Number.POSITIVE_INFINITY as unknown as DurationString)).toEqual(
319
+ err(new TypeError("Invalid duration format: expected a string but received number")),
320
+ );
321
+ expect(parseDuration(Number.NEGATIVE_INFINITY as unknown as DurationString)).toEqual(
322
+ err(new TypeError("Invalid duration format: expected a string but received number")),
323
+ );
324
+ });
325
+ });
326
+ });
@@ -0,0 +1,86 @@
1
+ import type { Result } from "./result";
2
+ import { err, ok } from "./result";
3
+
4
+ type Years = "years" | "year" | "yrs" | "yr" | "y";
5
+ type Months = "months" | "month" | "mo";
6
+ type Weeks = "weeks" | "week" | "w";
7
+ type Days = "days" | "day" | "d";
8
+ type Hours = "hours" | "hour" | "hrs" | "hr" | "h";
9
+ type Minutes = "minutes" | "minute" | "mins" | "min" | "m";
10
+ type Seconds = "seconds" | "second" | "secs" | "sec" | "s";
11
+ type Milliseconds = "milliseconds" | "millisecond" | "msecs" | "msec" | "ms";
12
+ type Unit = Years | Months | Weeks | Days | Hours | Minutes | Seconds | Milliseconds;
13
+ type UnitAnyCase = Capitalize<Unit> | Uppercase<Unit> | Lowercase<Unit>;
14
+ export type DurationString = `${number}` | `${number}${UnitAnyCase}` | `${number} ${UnitAnyCase}`;
15
+
16
+ /**
17
+ * Parse a duration string into milliseconds. Exmaples:
18
+ * - short units: "1ms", "5s", "30m", "2h", "7d", "3w", "1y"
19
+ * - long units: "1 millisecond", "5 seconds", "30 minutes", "2 hours", "7 days", "3 weeks", "1 year"
20
+ * @param str - Duration string
21
+ * @returns Milliseconds
22
+ */
23
+ export function parseDuration(str: DurationString): Result<number> {
24
+ if (typeof str !== "string") {
25
+ return err(
26
+ new TypeError(`Invalid duration format: expected a string but received ${typeof str}`),
27
+ );
28
+ }
29
+
30
+ if (str.length === 0) {
31
+ return err(new Error('Invalid duration format: ""'));
32
+ }
33
+
34
+ const match = /^(-?\.?\d+(?:\.\d+)?)\s*([a-z]+)?$/i.exec(str);
35
+
36
+ if (!match?.[1]) {
37
+ return err(new Error(`Invalid duration format: "${str}"`));
38
+ }
39
+
40
+ const numValue = Number.parseFloat(match[1]);
41
+ const unit = match[2]?.toLowerCase() ?? "ms"; // default to ms if not provided
42
+
43
+ const multipliers: Record<string, number> = {
44
+ millisecond: 1,
45
+ milliseconds: 1,
46
+ msec: 1,
47
+ msecs: 1,
48
+ ms: 1,
49
+ second: 1000,
50
+ seconds: 1000,
51
+ sec: 1000,
52
+ secs: 1000,
53
+ s: 1000,
54
+ minute: 60 * 1000,
55
+ minutes: 60 * 1000,
56
+ min: 60 * 1000,
57
+ mins: 60 * 1000,
58
+ m: 60 * 1000,
59
+ hour: 60 * 60 * 1000,
60
+ hours: 60 * 60 * 1000,
61
+ hr: 60 * 60 * 1000,
62
+ hrs: 60 * 60 * 1000,
63
+ h: 60 * 60 * 1000,
64
+ day: 24 * 60 * 60 * 1000,
65
+ days: 24 * 60 * 60 * 1000,
66
+ d: 24 * 60 * 60 * 1000,
67
+ week: 7 * 24 * 60 * 60 * 1000,
68
+ weeks: 7 * 24 * 60 * 60 * 1000,
69
+ w: 7 * 24 * 60 * 60 * 1000,
70
+ month: 2_629_800_000,
71
+ months: 2_629_800_000,
72
+ mo: 2_629_800_000,
73
+ year: 31_557_600_000,
74
+ years: 31_557_600_000,
75
+ yr: 31_557_600_000,
76
+ yrs: 31_557_600_000,
77
+ y: 31_557_600_000,
78
+ };
79
+
80
+ const multiplier = multipliers[unit];
81
+ if (multiplier === undefined) {
82
+ return err(new Error(`Invalid duration format: "${str}"`));
83
+ }
84
+
85
+ return ok(numValue * multiplier);
86
+ }
@@ -0,0 +1,77 @@
1
+ import { describe, expect, test } from "vitest";
2
+ import { serializeError } from "./error";
3
+
4
+ describe("serializeError", () => {
5
+ test("serializes Error instance with name, message, and stack", () => {
6
+ const error = new Error("Something went wrong");
7
+ const result = serializeError(error);
8
+
9
+ expect(result.name).toBe("Error");
10
+ expect(result.message).toBe("Something went wrong");
11
+ expect(result.stack).toBeDefined();
12
+ expect(typeof result.stack).toBe("string");
13
+ });
14
+
15
+ test("serializes TypeError with correct name", () => {
16
+ const error = new TypeError("Invalid type");
17
+ const result = serializeError(error);
18
+
19
+ expect(result.name).toBe("TypeError");
20
+ expect(result.message).toBe("Invalid type");
21
+ });
22
+
23
+ test("serializes custom Error subclass", () => {
24
+ class CustomError extends Error {
25
+ constructor(message: string) {
26
+ super(message);
27
+ this.name = "CustomError";
28
+ }
29
+ }
30
+ const error = new CustomError("Custom error message");
31
+ const result = serializeError(error);
32
+
33
+ expect(result.name).toBe("CustomError");
34
+ expect(result.message).toBe("Custom error message");
35
+ });
36
+
37
+ test("serializes Error without stack as undefined", () => {
38
+ const error = new Error("No stack");
39
+ error.stack = undefined;
40
+ const result = serializeError(error);
41
+
42
+ expect(result.stack).toBeUndefined();
43
+ });
44
+
45
+ test("serializes string to message", () => {
46
+ const result = serializeError("string error");
47
+
48
+ expect(result.message).toBe("string error");
49
+ expect(result.name).toBeUndefined();
50
+ expect(result.stack).toBeUndefined();
51
+ });
52
+
53
+ test("serializes number to message", () => {
54
+ const result = serializeError(42);
55
+
56
+ expect(result.message).toBe("42");
57
+ });
58
+
59
+ test("serializes null to message", () => {
60
+ const result = serializeError(null);
61
+
62
+ expect(result.message).toBe("null");
63
+ });
64
+
65
+ test("serializes undefined to message", () => {
66
+ // eslint-disable-next-line unicorn/no-useless-undefined
67
+ const result = serializeError(undefined);
68
+
69
+ expect(result.message).toBe("undefined");
70
+ });
71
+
72
+ test("serializes object to message using String()", () => {
73
+ const result = serializeError({ foo: "bar" });
74
+
75
+ expect(result.message).toBe("[object Object]");
76
+ });
77
+ });
@@ -0,0 +1,30 @@
1
+ import type { JsonValue } from "./json";
2
+
3
+ export type SerializedError = {
4
+ name?: string;
5
+ message: string;
6
+ stack?: string;
7
+ } & {
8
+ [key: string]: JsonValue;
9
+ };
10
+
11
+ /**
12
+ * Serialize an error to a JSON-compatible format.
13
+ * @param error - The error to serialize (can be Error instance or any value)
14
+ * @returns A JSON-serializable error object
15
+ */
16
+ export function serializeError(error: unknown): SerializedError {
17
+ if (error instanceof Error) {
18
+ const { name, message, stack } = error;
19
+
20
+ if (stack) {
21
+ return { name, message, stack };
22
+ }
23
+
24
+ return { name, message };
25
+ }
26
+
27
+ return {
28
+ message: String(error),
29
+ };
30
+ }
@@ -0,0 +1,2 @@
1
+ export type JsonPrimitive = string | number | boolean | null;
2
+ export type JsonValue = JsonPrimitive | JsonValue[] | { [key: string]: JsonValue };
@@ -0,0 +1,13 @@
1
+ import { describe, expect, test } from "vitest";
2
+ import { err, ok } from "./result";
3
+
4
+ describe("Result helpers", () => {
5
+ test("ok creates success result", () => {
6
+ expect(ok(123)).toEqual({ ok: true, value: 123 });
7
+ });
8
+
9
+ test("err creates error result", () => {
10
+ const error = new Error("oops");
11
+ expect(err(error)).toEqual({ ok: false, error });
12
+ });
13
+ });
@@ -0,0 +1,29 @@
1
+ export type Result<T> = Ok<T> | Err;
2
+
3
+ export interface Ok<T> {
4
+ ok: true;
5
+ value: T;
6
+ }
7
+
8
+ export interface Err {
9
+ ok: false;
10
+ error: Error;
11
+ }
12
+
13
+ /**
14
+ * Create an Ok result.
15
+ * @param value - Result value
16
+ * @returns Ok result
17
+ */
18
+ export function ok<T>(value: T): Ok<T> {
19
+ return { ok: true, value };
20
+ }
21
+
22
+ /**
23
+ * Create an Err result.
24
+ * @param error - Result error
25
+ * @returns Err result
26
+ */
27
+ export function err(error: Readonly<Error>): Err {
28
+ return { ok: false, error };
29
+ }
@@ -0,0 +1,41 @@
1
+ import { describe, expect, test } from "vitest";
2
+ import { calculateRetryDelayMs, DEFAULT_RETRY_POLICY, shouldRetry } from "./retry";
3
+
4
+ describe("calculateRetryDelayMs", () => {
5
+ test("calculates exponential backoff correctly", () => {
6
+ expect(calculateRetryDelayMs(1)).toBe(1000);
7
+ expect(calculateRetryDelayMs(2)).toBe(2000);
8
+ expect(calculateRetryDelayMs(3)).toBe(4000);
9
+ expect(calculateRetryDelayMs(4)).toBe(8000);
10
+ expect(calculateRetryDelayMs(5)).toBe(16_000);
11
+ expect(calculateRetryDelayMs(6)).toBe(32_000);
12
+ expect(calculateRetryDelayMs(7)).toBe(64_000);
13
+ });
14
+
15
+ test("caps delay at maximum interval", () => {
16
+ const { maximumIntervalMs } = DEFAULT_RETRY_POLICY;
17
+
18
+ // attempt 8: 1s * 2^7 = 128s = 128000ms, but capped at 100000ms (max)
19
+ expect(calculateRetryDelayMs(8)).toBe(maximumIntervalMs);
20
+
21
+ // attempts 10 & 100: should still be capped
22
+ expect(calculateRetryDelayMs(10)).toBe(maximumIntervalMs);
23
+ expect(calculateRetryDelayMs(100)).toBe(maximumIntervalMs);
24
+ });
25
+
26
+ test("handles edge cases", () => {
27
+ // attempt 0: 1s * 2^-1 = 0.5s = 500ms
28
+ expect(calculateRetryDelayMs(0)).toBe(500);
29
+ expect(calculateRetryDelayMs(Infinity)).toBe(100_000);
30
+ });
31
+ });
32
+
33
+ describe("shouldRetry", () => {
34
+ test("always returns true with default policy (infinite retries)", () => {
35
+ const retryPolicy = DEFAULT_RETRY_POLICY;
36
+ expect(shouldRetry(retryPolicy, 1)).toBe(true);
37
+ expect(shouldRetry(retryPolicy, 10)).toBe(true);
38
+ expect(shouldRetry(retryPolicy, 100)).toBe(true);
39
+ expect(shouldRetry(retryPolicy, 1000)).toBe(true);
40
+ });
41
+ });
@@ -0,0 +1,29 @@
1
+ export const DEFAULT_RETRY_POLICY = {
2
+ initialIntervalMs: 1000, // 1s
3
+ backoffCoefficient: 2,
4
+ maximumIntervalMs: 100 * 1000, // 100s
5
+ maximumAttempts: Infinity, // unlimited
6
+ } as const;
7
+
8
+ export type RetryPolicy = typeof DEFAULT_RETRY_POLICY;
9
+
10
+ /**
11
+ * Calculate the next retry delay using exponential backoff.
12
+ * @param attemptNumber - Attempt number (1-based)
13
+ * @returns Delay in milliseconds
14
+ */
15
+ export function calculateRetryDelayMs(attemptNumber: number): number {
16
+ const { initialIntervalMs, backoffCoefficient, maximumIntervalMs } = DEFAULT_RETRY_POLICY;
17
+ const backoffMs = initialIntervalMs * backoffCoefficient ** (attemptNumber - 1);
18
+ return Math.min(backoffMs, maximumIntervalMs);
19
+ }
20
+
21
+ /**
22
+ * Check if an operation should be retried based on the retry policy.
23
+ * @param retryPolicy - Retry policy
24
+ * @param attemptNumber - Attempt number (1-based)
25
+ * @returns True if another attempt should be made
26
+ */
27
+ export function shouldRetry(retryPolicy: RetryPolicy, attemptNumber: number): boolean {
28
+ return attemptNumber < retryPolicy.maximumAttempts;
29
+ }