@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.
- package/.swcrc +17 -0
- package/README.md +7 -0
- package/dist/backend.d.ts +107 -0
- package/dist/backend.d.ts.map +1 -0
- package/dist/backend.js +3 -0
- package/dist/backend.js.map +1 -0
- package/dist/chaos.test.d.ts +2 -0
- package/dist/chaos.test.d.ts.map +1 -0
- package/dist/chaos.test.js +92 -0
- package/dist/chaos.test.js.map +1 -0
- package/dist/client.d.ts +178 -0
- package/dist/client.d.ts.map +1 -0
- package/dist/client.js +223 -0
- package/dist/client.js.map +1 -0
- package/dist/client.test.d.ts +2 -0
- package/dist/client.test.d.ts.map +1 -0
- package/dist/client.test.js +339 -0
- package/dist/client.test.js.map +1 -0
- package/dist/config.d.ts +22 -0
- package/dist/config.d.ts.map +1 -0
- package/dist/config.js +23 -0
- package/dist/config.js.map +1 -0
- package/dist/config.test.d.ts +2 -0
- package/dist/config.test.d.ts.map +1 -0
- package/dist/config.test.js +24 -0
- package/dist/config.test.js.map +1 -0
- package/dist/core/duration.d.ts +22 -0
- package/dist/core/duration.d.ts.map +1 -0
- package/dist/core/duration.js +64 -0
- package/dist/core/duration.js.map +1 -0
- package/dist/core/duration.test.d.ts +2 -0
- package/dist/core/duration.test.d.ts.map +1 -0
- package/dist/core/duration.test.js +265 -0
- package/dist/core/duration.test.js.map +1 -0
- package/dist/core/error.d.ts +15 -0
- package/dist/core/error.d.ts.map +1 -0
- package/dist/core/error.js +25 -0
- package/dist/core/error.js.map +1 -0
- package/dist/core/error.test.d.ts +2 -0
- package/dist/core/error.test.d.ts.map +1 -0
- package/dist/core/error.test.js +63 -0
- package/dist/core/error.test.js.map +1 -0
- package/dist/core/json.d.ts +5 -0
- package/dist/core/json.d.ts.map +1 -0
- package/dist/core/json.js +3 -0
- package/dist/core/json.js.map +1 -0
- package/dist/core/result.d.ts +22 -0
- package/dist/core/result.d.ts.map +1 -0
- package/dist/core/result.js +22 -0
- package/dist/core/result.js.map +1 -0
- package/dist/core/result.test.d.ts +2 -0
- package/dist/core/result.test.d.ts.map +1 -0
- package/dist/core/result.test.js +19 -0
- package/dist/core/result.test.js.map +1 -0
- package/dist/core/retry.d.ts +21 -0
- package/dist/core/retry.d.ts.map +1 -0
- package/dist/core/retry.js +25 -0
- package/dist/core/retry.js.map +1 -0
- package/dist/core/retry.test.d.ts +2 -0
- package/dist/core/retry.test.d.ts.map +1 -0
- package/dist/core/retry.test.js +37 -0
- package/dist/core/retry.test.js.map +1 -0
- package/dist/core/schema.d.ts +57 -0
- package/dist/core/schema.d.ts.map +1 -0
- package/dist/core/schema.js +4 -0
- package/dist/core/schema.js.map +1 -0
- package/dist/core/step.d.ts +96 -0
- package/dist/core/step.d.ts.map +1 -0
- package/dist/core/step.js +78 -0
- package/dist/core/step.js.map +1 -0
- package/dist/core/step.test.d.ts +2 -0
- package/dist/core/step.test.d.ts.map +1 -0
- package/dist/core/step.test.js +356 -0
- package/dist/core/step.test.js.map +1 -0
- package/dist/core/workflow.d.ts +78 -0
- package/dist/core/workflow.d.ts.map +1 -0
- package/dist/core/workflow.js +46 -0
- package/dist/core/workflow.js.map +1 -0
- package/dist/core/workflow.test.d.ts +2 -0
- package/dist/core/workflow.test.d.ts.map +1 -0
- package/dist/core/workflow.test.js +172 -0
- package/dist/core/workflow.test.js.map +1 -0
- package/dist/database/backend.d.ts +60 -0
- package/dist/database/backend.d.ts.map +1 -0
- package/dist/database/backend.js +387 -0
- package/dist/database/backend.js.map +1 -0
- package/dist/database/backend.test.d.ts +2 -0
- package/dist/database/backend.test.d.ts.map +1 -0
- package/dist/database/backend.test.js +17 -0
- package/dist/database/backend.test.js.map +1 -0
- package/dist/database/backend.testsuite.d.ts +20 -0
- package/dist/database/backend.testsuite.d.ts.map +1 -0
- package/dist/database/backend.testsuite.js +1174 -0
- package/dist/database/backend.testsuite.js.map +1 -0
- package/dist/database/base.d.ts +12 -0
- package/dist/database/base.d.ts.map +1 -0
- package/dist/database/base.js +19 -0
- package/dist/database/base.js.map +1 -0
- package/dist/database/migrations/20251212000000_0_init.js +9 -0
- package/dist/database/migrations/20251212000000_0_init.js.map +1 -0
- package/dist/database/migrations/20251212000000_1_tables.js +88 -0
- package/dist/database/migrations/20251212000000_1_tables.js.map +1 -0
- package/dist/database/migrations/20251212000000_2_fk.js +48 -0
- package/dist/database/migrations/20251212000000_2_fk.js.map +1 -0
- package/dist/database/migrations/20251212000000_3_indexes.js +107 -0
- package/dist/database/migrations/20251212000000_3_indexes.js.map +1 -0
- package/dist/database/pubsub.d.ts +17 -0
- package/dist/database/pubsub.d.ts.map +1 -0
- package/dist/database/pubsub.js +70 -0
- package/dist/database/pubsub.js.map +1 -0
- package/dist/database/pubsub.test.d.ts +2 -0
- package/dist/database/pubsub.test.d.ts.map +1 -0
- package/dist/database/pubsub.test.js +86 -0
- package/dist/database/pubsub.test.js.map +1 -0
- package/dist/errors.d.ts +8 -0
- package/dist/errors.d.ts.map +1 -0
- package/dist/errors.js +21 -0
- package/dist/errors.js.map +1 -0
- package/dist/execution.d.ts +82 -0
- package/dist/execution.d.ts.map +1 -0
- package/dist/execution.js +182 -0
- package/dist/execution.js.map +1 -0
- package/dist/execution.test.d.ts +2 -0
- package/dist/execution.test.d.ts.map +1 -0
- package/dist/execution.test.js +556 -0
- package/dist/execution.test.js.map +1 -0
- package/dist/index.d.ts +8 -0
- package/dist/index.d.ts.map +1 -0
- package/dist/index.js +6 -0
- package/dist/index.js.map +1 -0
- package/dist/internal.d.ts +12 -0
- package/dist/internal.d.ts.map +1 -0
- package/dist/internal.js +5 -0
- package/dist/internal.js.map +1 -0
- package/dist/practices/01-remote-workflow.d.ts +2 -0
- package/dist/practices/01-remote-workflow.d.ts.map +1 -0
- package/dist/practices/01-remote-workflow.js +69 -0
- package/dist/practices/01-remote-workflow.js.map +1 -0
- package/dist/practices/01-remote.d.ts +2 -0
- package/dist/practices/01-remote.d.ts.map +1 -0
- package/dist/practices/01-remote.js +87 -0
- package/dist/practices/01-remote.js.map +1 -0
- package/dist/practices/02-local.d.ts +2 -0
- package/dist/practices/02-local.d.ts.map +1 -0
- package/dist/practices/02-local.js +84 -0
- package/dist/practices/02-local.js.map +1 -0
- package/dist/practices/03-local-retry.d.ts +2 -0
- package/dist/practices/03-local-retry.d.ts.map +1 -0
- package/dist/practices/03-local-retry.js +85 -0
- package/dist/practices/03-local-retry.js.map +1 -0
- package/dist/practices/04-scheduler-dispose.d.ts +2 -0
- package/dist/practices/04-scheduler-dispose.d.ts.map +1 -0
- package/dist/practices/04-scheduler-dispose.js +65 -0
- package/dist/practices/04-scheduler-dispose.js.map +1 -0
- package/dist/practices/05-router.d.ts +2 -0
- package/dist/practices/05-router.d.ts.map +1 -0
- package/dist/practices/05-router.js +80 -0
- package/dist/practices/05-router.js.map +1 -0
- package/dist/registry.d.ts +33 -0
- package/dist/registry.d.ts.map +1 -0
- package/dist/registry.js +54 -0
- package/dist/registry.js.map +1 -0
- package/dist/registry.test.d.ts +2 -0
- package/dist/registry.test.d.ts.map +1 -0
- package/dist/registry.test.js +95 -0
- package/dist/registry.test.js.map +1 -0
- package/dist/scheduler.d.ts +22 -0
- package/dist/scheduler.d.ts.map +1 -0
- package/dist/scheduler.js +117 -0
- package/dist/scheduler.js.map +1 -0
- package/dist/tasks/index.d.ts +4 -0
- package/dist/tasks/index.d.ts.map +1 -0
- package/dist/tasks/index.js +5 -0
- package/dist/tasks/index.js.map +1 -0
- package/dist/tasks/local-task.d.ts +6 -0
- package/dist/tasks/local-task.d.ts.map +1 -0
- package/dist/tasks/local-task.js +95 -0
- package/dist/tasks/local-task.js.map +1 -0
- package/dist/tasks/remote-task.d.ts +11 -0
- package/dist/tasks/remote-task.d.ts.map +1 -0
- package/dist/tasks/remote-task.js +213 -0
- package/dist/tasks/remote-task.js.map +1 -0
- package/dist/tasks/shared.d.ts +8 -0
- package/dist/tasks/shared.d.ts.map +1 -0
- package/dist/tasks/shared.js +41 -0
- package/dist/tasks/shared.js.map +1 -0
- package/dist/testing/connection.d.ts +7 -0
- package/dist/testing/connection.d.ts.map +1 -0
- package/dist/testing/connection.js +38 -0
- package/dist/testing/connection.js.map +1 -0
- package/dist/types/config.d.ts +44 -0
- package/dist/types/config.d.ts.map +1 -0
- package/dist/types/config.js +3 -0
- package/dist/types/config.js.map +1 -0
- package/dist/types/context.d.ts +18 -0
- package/dist/types/context.d.ts.map +1 -0
- package/dist/types/context.js +4 -0
- package/dist/types/context.js.map +1 -0
- package/dist/types/events.d.ts +43 -0
- package/dist/types/events.d.ts.map +1 -0
- package/dist/types/events.js +3 -0
- package/dist/types/events.js.map +1 -0
- package/dist/types/index.d.ts +6 -0
- package/dist/types/index.d.ts.map +1 -0
- package/dist/types/index.js +3 -0
- package/dist/types/index.js.map +1 -0
- package/dist/types/task-items.d.ts +12 -0
- package/dist/types/task-items.d.ts.map +1 -0
- package/dist/types/task-items.js +3 -0
- package/dist/types/task-items.js.map +1 -0
- package/dist/types/utils.d.ts +4 -0
- package/dist/types/utils.d.ts.map +1 -0
- package/dist/types/utils.js +8 -0
- package/dist/types/utils.js.map +1 -0
- package/dist/worker.d.ts +61 -0
- package/dist/worker.d.ts.map +1 -0
- package/dist/worker.js +206 -0
- package/dist/worker.js.map +1 -0
- package/dist/worker.test.d.ts +2 -0
- package/dist/worker.test.d.ts.map +1 -0
- package/dist/worker.test.js +1163 -0
- package/dist/worker.test.js.map +1 -0
- package/dist/workflow.d.ts +44 -0
- package/dist/workflow.d.ts.map +1 -0
- package/dist/workflow.js +21 -0
- package/dist/workflow.js.map +1 -0
- package/dist/workflow.test.d.ts +2 -0
- package/dist/workflow.test.d.ts.map +1 -0
- package/dist/workflow.test.js +73 -0
- package/dist/workflow.test.js.map +1 -0
- package/nodemon.json +6 -0
- package/package.json +63 -0
- package/scripts/migrate.ts +11 -0
- package/src/backend.ts +133 -0
- package/src/chaos.test.ts +108 -0
- package/src/client.test.ts +297 -0
- package/src/client.ts +331 -0
- package/src/config.test.ts +23 -0
- package/src/config.ts +35 -0
- package/src/core/duration.test.ts +326 -0
- package/src/core/duration.ts +86 -0
- package/src/core/error.test.ts +77 -0
- package/src/core/error.ts +30 -0
- package/src/core/json.ts +2 -0
- package/src/core/result.test.ts +13 -0
- package/src/core/result.ts +29 -0
- package/src/core/retry.test.ts +41 -0
- package/src/core/retry.ts +29 -0
- package/src/core/schema.ts +74 -0
- package/src/core/step.test.ts +362 -0
- package/src/core/step.ts +152 -0
- package/src/core/workflow.test.ts +184 -0
- package/src/core/workflow.ts +127 -0
- package/src/database/backend.test.ts +16 -0
- package/src/database/backend.testsuite.ts +1376 -0
- package/src/database/backend.ts +655 -0
- package/src/database/base.ts +23 -0
- package/src/database/migrations/20251212000000_0_init.ts +10 -0
- package/src/database/migrations/20251212000000_1_tables.ts +54 -0
- package/src/database/migrations/20251212000000_2_fk.ts +46 -0
- package/src/database/migrations/20251212000000_3_indexes.ts +82 -0
- package/src/database/pubsub.test.ts +92 -0
- package/src/database/pubsub.ts +92 -0
- package/src/execution.test.ts +508 -0
- package/src/execution.ts +291 -0
- package/src/index.ts +7 -0
- package/src/internal.ts +11 -0
- package/src/practices/01-remote-workflow.ts +61 -0
- package/src/registry.test.ts +122 -0
- package/src/registry.ts +65 -0
- package/src/testing/connection.ts +44 -0
- package/src/worker.test.ts +1138 -0
- package/src/worker.ts +281 -0
- package/src/workflow.test.ts +68 -0
- package/src/workflow.ts +84 -0
- package/table_ddl.sql +60 -0
- package/templates/openworkflow.config.ts +22 -0
- package/tsconfig.json +40 -0
- package/tsconfig.test.json +4 -0
- 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
|
+
}
|
package/src/core/json.ts
ADDED
|
@@ -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
|
+
}
|