@rivetkit/workflow-engine 2.3.0-rc.10 → 2.3.0-rc.11
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/dist/tsup/{chunk-CIDHCIH7.js → chunk-DFNXCZ47.js} +13 -22
- package/dist/tsup/chunk-DFNXCZ47.js.map +1 -0
- package/dist/tsup/{chunk-SFJQFMCW.cjs → chunk-U2W3KHJC.cjs} +13 -22
- package/dist/tsup/chunk-U2W3KHJC.cjs.map +1 -0
- package/dist/tsup/index.cjs +2 -2
- package/dist/tsup/index.d.cts +11 -7
- package/dist/tsup/index.d.ts +11 -7
- package/dist/tsup/index.js +1 -1
- package/dist/tsup/testing.cjs +23 -23
- package/dist/tsup/testing.js +1 -1
- package/package.json +1 -1
- package/schemas/serde.ts +1 -3
- package/src/context.ts +21 -30
- package/src/index.ts +8 -8
- package/src/location.ts +1 -1
- package/src/types.ts +3 -1
- package/dist/tsup/chunk-CIDHCIH7.js.map +0 -1
- package/dist/tsup/chunk-SFJQFMCW.cjs.map +0 -1
package/dist/tsup/index.d.ts
CHANGED
|
@@ -345,6 +345,8 @@ interface StepConfig<T> {
|
|
|
345
345
|
retryBackoffMax?: number;
|
|
346
346
|
/** Timeout in ms for step execution (default: 30000). Set to 0 to disable. */
|
|
347
347
|
timeout?: number;
|
|
348
|
+
/** If true, step timeouts retry like any other error instead of failing immediately as critical. Default: false. */
|
|
349
|
+
retryOnTimeout?: boolean;
|
|
348
350
|
}
|
|
349
351
|
type TryStepCatchKind = "critical" | "timeout" | "exhausted" | "rollback";
|
|
350
352
|
interface TryStepFailure {
|
|
@@ -398,7 +400,7 @@ type LoopResult<S, T> = {
|
|
|
398
400
|
* Stateless loops (state = undefined) may return void/undefined, which is treated as
|
|
399
401
|
* `Loop.continue(undefined)`.
|
|
400
402
|
*/
|
|
401
|
-
type LoopIterationResult<S, T> = Promise<LoopResult<S, T> | (S extends undefined ?
|
|
403
|
+
type LoopIterationResult<S, T> = Promise<LoopResult<S, T> | (S extends undefined ? undefined : never)>;
|
|
402
404
|
/**
|
|
403
405
|
* Configuration for a loop.
|
|
404
406
|
*/
|
|
@@ -713,7 +715,9 @@ declare class WorkflowContextImpl implements WorkflowContextInterface {
|
|
|
713
715
|
*
|
|
714
716
|
* Note: This does NOT cancel the underlying operation. JavaScript Promises
|
|
715
717
|
* cannot be cancelled once started. When a timeout occurs:
|
|
716
|
-
* - The step is
|
|
718
|
+
* - The step is rejected with StepTimeoutError. By default this is treated
|
|
719
|
+
* as a critical failure with no retry. Set retryOnTimeout: true on the
|
|
720
|
+
* step config to retry timeouts like any other error.
|
|
717
721
|
* - The underlying async operation continues running in the background
|
|
718
722
|
* - Any side effects from the operation may still occur
|
|
719
723
|
*
|
|
@@ -779,6 +783,11 @@ declare class WorkflowContextImpl implements WorkflowContextInterface {
|
|
|
779
783
|
private executeRemoved;
|
|
780
784
|
}
|
|
781
785
|
|
|
786
|
+
/**
|
|
787
|
+
* Extract structured error information from an error.
|
|
788
|
+
*/
|
|
789
|
+
declare function extractErrorInfo(error: unknown): WorkflowError;
|
|
790
|
+
|
|
782
791
|
/**
|
|
783
792
|
* Thrown from steps to prevent retry.
|
|
784
793
|
* Use this when an error is unrecoverable and retrying would be pointless.
|
|
@@ -883,11 +892,6 @@ declare class EntryInProgressError extends Error {
|
|
|
883
892
|
constructor();
|
|
884
893
|
}
|
|
885
894
|
|
|
886
|
-
/**
|
|
887
|
-
* Extract structured error information from an error.
|
|
888
|
-
*/
|
|
889
|
-
declare function extractErrorInfo(error: unknown): WorkflowError;
|
|
890
|
-
|
|
891
895
|
/**
|
|
892
896
|
* Check if a path segment is a loop iteration marker.
|
|
893
897
|
*/
|
package/dist/tsup/index.js
CHANGED
package/dist/tsup/testing.cjs
CHANGED
|
@@ -47,7 +47,7 @@
|
|
|
47
47
|
|
|
48
48
|
|
|
49
49
|
|
|
50
|
-
var
|
|
50
|
+
var _chunkU2W3KHJCcjs = require('./chunk-U2W3KHJC.cjs');
|
|
51
51
|
|
|
52
52
|
// src/testing.ts
|
|
53
53
|
var InMemoryWorkflowMessageDriver = class {
|
|
@@ -97,7 +97,7 @@ var InMemoryWorkflowMessageDriver = class {
|
|
|
97
97
|
}
|
|
98
98
|
async waitForMessages(messageNames, abortSignal) {
|
|
99
99
|
if (abortSignal.aborted) {
|
|
100
|
-
throw new (0,
|
|
100
|
+
throw new (0, _chunkU2W3KHJCcjs.EvictedError)();
|
|
101
101
|
}
|
|
102
102
|
const nameSet = messageNames.length > 0 ? new Set(messageNames) : void 0;
|
|
103
103
|
if (this.#messages.some(
|
|
@@ -118,7 +118,7 @@ var InMemoryWorkflowMessageDriver = class {
|
|
|
118
118
|
},
|
|
119
119
|
abortSignal,
|
|
120
120
|
onAbort: () => {
|
|
121
|
-
waiter.reject(new (0,
|
|
121
|
+
waiter.reject(new (0, _chunkU2W3KHJCcjs.EvictedError)());
|
|
122
122
|
}
|
|
123
123
|
};
|
|
124
124
|
abortSignal.addEventListener("abort", waiter.onAbort, {
|
|
@@ -158,56 +158,56 @@ var InMemoryDriver = (_class = class {constructor() { _class.prototype.__init.ca
|
|
|
158
158
|
__init4() {this.workerPollInterval = 100}
|
|
159
159
|
__init5() {this.messageDriver = this.#inMemoryMessageDriver}
|
|
160
160
|
async get(key) {
|
|
161
|
-
await
|
|
162
|
-
const entry = this.kv.get(
|
|
161
|
+
await _chunkU2W3KHJCcjs.sleep.call(void 0, this.latency);
|
|
162
|
+
const entry = this.kv.get(_chunkU2W3KHJCcjs.keyToHex.call(void 0, key));
|
|
163
163
|
return _nullishCoalesce((entry == null ? void 0 : entry.value), () => ( null));
|
|
164
164
|
}
|
|
165
165
|
async set(key, value) {
|
|
166
|
-
await
|
|
167
|
-
this.kv.set(
|
|
166
|
+
await _chunkU2W3KHJCcjs.sleep.call(void 0, this.latency);
|
|
167
|
+
this.kv.set(_chunkU2W3KHJCcjs.keyToHex.call(void 0, key), { key, value });
|
|
168
168
|
}
|
|
169
169
|
async delete(key) {
|
|
170
|
-
await
|
|
171
|
-
this.kv.delete(
|
|
170
|
+
await _chunkU2W3KHJCcjs.sleep.call(void 0, this.latency);
|
|
171
|
+
this.kv.delete(_chunkU2W3KHJCcjs.keyToHex.call(void 0, key));
|
|
172
172
|
}
|
|
173
173
|
async deletePrefix(prefix) {
|
|
174
|
-
await
|
|
174
|
+
await _chunkU2W3KHJCcjs.sleep.call(void 0, this.latency);
|
|
175
175
|
for (const [hexKey, entry] of this.kv) {
|
|
176
|
-
if (
|
|
176
|
+
if (_chunkU2W3KHJCcjs.keyStartsWith.call(void 0, entry.key, prefix)) {
|
|
177
177
|
this.kv.delete(hexKey);
|
|
178
178
|
}
|
|
179
179
|
}
|
|
180
180
|
}
|
|
181
181
|
async deleteRange(start, end) {
|
|
182
|
-
await
|
|
182
|
+
await _chunkU2W3KHJCcjs.sleep.call(void 0, this.latency);
|
|
183
183
|
for (const [hexKey, entry] of this.kv) {
|
|
184
|
-
if (
|
|
184
|
+
if (_chunkU2W3KHJCcjs.compareKeys.call(void 0, entry.key, start) >= 0 && _chunkU2W3KHJCcjs.compareKeys.call(void 0, entry.key, end) < 0) {
|
|
185
185
|
this.kv.delete(hexKey);
|
|
186
186
|
}
|
|
187
187
|
}
|
|
188
188
|
}
|
|
189
189
|
async list(prefix) {
|
|
190
|
-
await
|
|
190
|
+
await _chunkU2W3KHJCcjs.sleep.call(void 0, this.latency);
|
|
191
191
|
const results = [];
|
|
192
192
|
for (const entry of this.kv.values()) {
|
|
193
|
-
if (
|
|
193
|
+
if (_chunkU2W3KHJCcjs.keyStartsWith.call(void 0, entry.key, prefix)) {
|
|
194
194
|
results.push({ key: entry.key, value: entry.value });
|
|
195
195
|
}
|
|
196
196
|
}
|
|
197
|
-
return results.sort((a, b) =>
|
|
197
|
+
return results.sort((a, b) => _chunkU2W3KHJCcjs.compareKeys.call(void 0, a.key, b.key));
|
|
198
198
|
}
|
|
199
199
|
async batch(writes) {
|
|
200
|
-
await
|
|
200
|
+
await _chunkU2W3KHJCcjs.sleep.call(void 0, this.latency);
|
|
201
201
|
for (const { key, value } of writes) {
|
|
202
|
-
this.kv.set(
|
|
202
|
+
this.kv.set(_chunkU2W3KHJCcjs.keyToHex.call(void 0, key), { key, value });
|
|
203
203
|
}
|
|
204
204
|
}
|
|
205
205
|
async setAlarm(workflowId, wakeAt) {
|
|
206
|
-
await
|
|
206
|
+
await _chunkU2W3KHJCcjs.sleep.call(void 0, this.latency);
|
|
207
207
|
this.alarms.set(workflowId, wakeAt);
|
|
208
208
|
}
|
|
209
209
|
async clearAlarm(workflowId) {
|
|
210
|
-
await
|
|
210
|
+
await _chunkU2W3KHJCcjs.sleep.call(void 0, this.latency);
|
|
211
211
|
this.alarms.delete(workflowId);
|
|
212
212
|
}
|
|
213
213
|
async waitForMessages(messageNames, abortSignal) {
|
|
@@ -218,7 +218,7 @@ var InMemoryDriver = (_class = class {constructor() { _class.prototype.__init.ca
|
|
|
218
218
|
}
|
|
219
219
|
while (true) {
|
|
220
220
|
if (abortSignal.aborted) {
|
|
221
|
-
throw new (0,
|
|
221
|
+
throw new (0, _chunkU2W3KHJCcjs.EvictedError)();
|
|
222
222
|
}
|
|
223
223
|
const messages = await this.messageDriver.receiveMessages({
|
|
224
224
|
names: messageNames.length > 0 ? messageNames : void 0,
|
|
@@ -228,7 +228,7 @@ var InMemoryDriver = (_class = class {constructor() { _class.prototype.__init.ca
|
|
|
228
228
|
if (messages.length > 0) {
|
|
229
229
|
return;
|
|
230
230
|
}
|
|
231
|
-
await
|
|
231
|
+
await _chunkU2W3KHJCcjs.sleep.call(void 0, Math.max(1, this.latency));
|
|
232
232
|
}
|
|
233
233
|
}
|
|
234
234
|
/**
|
|
@@ -324,5 +324,5 @@ var InMemoryDriver = (_class = class {constructor() { _class.prototype.__init.ca
|
|
|
324
324
|
|
|
325
325
|
|
|
326
326
|
|
|
327
|
-
exports.CancelledError =
|
|
327
|
+
exports.CancelledError = _chunkU2W3KHJCcjs.CancelledError; exports.CriticalError = _chunkU2W3KHJCcjs.CriticalError; exports.DEFAULT_LOOP_HISTORY_PRUNE_INTERVAL = _chunkU2W3KHJCcjs.DEFAULT_LOOP_HISTORY_PRUNE_INTERVAL; exports.DEFAULT_MAX_RETRIES = _chunkU2W3KHJCcjs.DEFAULT_MAX_RETRIES; exports.DEFAULT_RETRY_BACKOFF_BASE = _chunkU2W3KHJCcjs.DEFAULT_RETRY_BACKOFF_BASE; exports.DEFAULT_RETRY_BACKOFF_MAX = _chunkU2W3KHJCcjs.DEFAULT_RETRY_BACKOFF_MAX; exports.DEFAULT_STEP_TIMEOUT = _chunkU2W3KHJCcjs.DEFAULT_STEP_TIMEOUT; exports.EntryInProgressError = _chunkU2W3KHJCcjs.EntryInProgressError; exports.EvictedError = _chunkU2W3KHJCcjs.EvictedError; exports.HistoryDivergedError = _chunkU2W3KHJCcjs.HistoryDivergedError; exports.InMemoryDriver = InMemoryDriver; exports.JoinError = _chunkU2W3KHJCcjs.JoinError; exports.Loop = _chunkU2W3KHJCcjs.Loop; exports.MessageWaitError = _chunkU2W3KHJCcjs.MessageWaitError; exports.RaceError = _chunkU2W3KHJCcjs.RaceError; exports.RollbackCheckpointError = _chunkU2W3KHJCcjs.RollbackCheckpointError; exports.RollbackError = _chunkU2W3KHJCcjs.RollbackError; exports.SleepError = _chunkU2W3KHJCcjs.SleepError; exports.StepExhaustedError = _chunkU2W3KHJCcjs.StepExhaustedError; exports.StepFailedError = _chunkU2W3KHJCcjs.StepFailedError; exports.WorkflowContextImpl = _chunkU2W3KHJCcjs.WorkflowContextImpl; exports.appendLoopIteration = _chunkU2W3KHJCcjs.appendLoopIteration; exports.appendName = _chunkU2W3KHJCcjs.appendName; exports.createEntry = _chunkU2W3KHJCcjs.createEntry; exports.createHistorySnapshot = _chunkU2W3KHJCcjs.createHistorySnapshot; exports.createStorage = _chunkU2W3KHJCcjs.createStorage; exports.deleteEntriesWithPrefix = _chunkU2W3KHJCcjs.deleteEntriesWithPrefix; exports.emptyLocation = _chunkU2W3KHJCcjs.emptyLocation; exports.extractErrorInfo = _chunkU2W3KHJCcjs.extractErrorInfo; exports.flush = _chunkU2W3KHJCcjs.flush; exports.generateId = _chunkU2W3KHJCcjs.generateId; exports.getEntry = _chunkU2W3KHJCcjs.getEntry; exports.getOrCreateMetadata = _chunkU2W3KHJCcjs.getOrCreateMetadata; exports.isLocationPrefix = _chunkU2W3KHJCcjs.isLocationPrefix; exports.isLoopIterationMarker = _chunkU2W3KHJCcjs.isLoopIterationMarker; exports.loadMetadata = _chunkU2W3KHJCcjs.loadMetadata; exports.loadStorage = _chunkU2W3KHJCcjs.loadStorage; exports.locationToKey = _chunkU2W3KHJCcjs.locationToKey; exports.locationsEqual = _chunkU2W3KHJCcjs.locationsEqual; exports.parentLocation = _chunkU2W3KHJCcjs.parentLocation; exports.registerName = _chunkU2W3KHJCcjs.registerName; exports.replayWorkflowFromStep = _chunkU2W3KHJCcjs.replayWorkflowFromStep; exports.resolveName = _chunkU2W3KHJCcjs.resolveName; exports.runWorkflow = _chunkU2W3KHJCcjs.runWorkflow; exports.setEntry = _chunkU2W3KHJCcjs.setEntry;
|
|
328
328
|
//# sourceMappingURL=testing.cjs.map
|
package/dist/tsup/testing.js
CHANGED
package/package.json
CHANGED
package/schemas/serde.ts
CHANGED
|
@@ -18,7 +18,6 @@ import type {
|
|
|
18
18
|
EntryMetadata as InternalEntryMetadata,
|
|
19
19
|
EntryStatus as InternalEntryStatus,
|
|
20
20
|
Location as InternalLocation,
|
|
21
|
-
LoopIterationMarker as InternalLoopIterationMarker,
|
|
22
21
|
PathSegment as InternalPathSegment,
|
|
23
22
|
SleepState as InternalSleepState,
|
|
24
23
|
WorkflowState as InternalWorkflowState,
|
|
@@ -27,7 +26,6 @@ import {
|
|
|
27
26
|
CURRENT_VERSION,
|
|
28
27
|
ENTRY_METADATA_VERSIONED,
|
|
29
28
|
ENTRY_VERSIONED,
|
|
30
|
-
WORKFLOW_METADATA_VERSIONED,
|
|
31
29
|
} from "./versioned.js";
|
|
32
30
|
|
|
33
31
|
// === Helper: ArrayBuffer to/from utilities ===
|
|
@@ -74,7 +72,7 @@ function assertString(
|
|
|
74
72
|
/**
|
|
75
73
|
* Validate that a value is a number.
|
|
76
74
|
*/
|
|
77
|
-
function
|
|
75
|
+
function _assertNumber(
|
|
78
76
|
value: unknown,
|
|
79
77
|
context: string,
|
|
80
78
|
): asserts value is number {
|
package/src/context.ts
CHANGED
|
@@ -21,7 +21,7 @@ import {
|
|
|
21
21
|
StepExhaustedError,
|
|
22
22
|
StepFailedError,
|
|
23
23
|
} from "./errors.js";
|
|
24
|
-
import {
|
|
24
|
+
import { buildEntryMetadataKey, buildLoopIterationRange } from "./keys.js";
|
|
25
25
|
import {
|
|
26
26
|
appendLoopIteration,
|
|
27
27
|
appendName,
|
|
@@ -36,13 +36,12 @@ import {
|
|
|
36
36
|
flush,
|
|
37
37
|
getOrCreateMetadata,
|
|
38
38
|
loadMetadata,
|
|
39
|
-
setEntry,
|
|
40
39
|
type PendingDeletions,
|
|
40
|
+
setEntry,
|
|
41
41
|
} from "./storage.js";
|
|
42
42
|
import type {
|
|
43
43
|
BranchConfig,
|
|
44
44
|
BranchOutput,
|
|
45
|
-
BranchStatus,
|
|
46
45
|
Entry,
|
|
47
46
|
EntryKindType,
|
|
48
47
|
EntryMetadata,
|
|
@@ -66,11 +65,11 @@ import type {
|
|
|
66
65
|
WorkflowError,
|
|
67
66
|
WorkflowErrorEvent,
|
|
68
67
|
WorkflowErrorHandler,
|
|
68
|
+
WorkflowMessageDriver,
|
|
69
69
|
WorkflowQueue,
|
|
70
70
|
WorkflowQueueMessage,
|
|
71
71
|
WorkflowQueueNextBatchOptions,
|
|
72
72
|
WorkflowQueueNextOptions,
|
|
73
|
-
WorkflowMessageDriver,
|
|
74
73
|
} from "./types.js";
|
|
75
74
|
import { sleep } from "./utils.js";
|
|
76
75
|
|
|
@@ -505,8 +504,7 @@ export class WorkflowContextImpl implements WorkflowContextInterface {
|
|
|
505
504
|
* Throws HistoryDivergedError if duplicate detected.
|
|
506
505
|
*/
|
|
507
506
|
private checkDuplicateName(name: string): void {
|
|
508
|
-
const fullKey =
|
|
509
|
-
locationToKey(this.storage, this.currentLocation) + "/" + name;
|
|
507
|
+
const fullKey = `${locationToKey(this.storage, this.currentLocation)}/${name}`;
|
|
510
508
|
if (this.usedNamesInExecution.has(fullKey)) {
|
|
511
509
|
throw new HistoryDivergedError(
|
|
512
510
|
`Duplicate entry name "${name}" at location "${locationToKey(this.storage, this.currentLocation)}". ` +
|
|
@@ -580,7 +578,7 @@ export class WorkflowContextImpl implements WorkflowContextInterface {
|
|
|
580
578
|
const isUnderPrefix =
|
|
581
579
|
prefix === ""
|
|
582
580
|
? true // Root: all keys are children
|
|
583
|
-
: key.startsWith(prefix
|
|
581
|
+
: key.startsWith(`${prefix}/`) || key === prefix;
|
|
584
582
|
|
|
585
583
|
if (isUnderPrefix) {
|
|
586
584
|
if (!this.visitedKeys.has(key)) {
|
|
@@ -825,10 +823,7 @@ export class WorkflowContextImpl implements WorkflowContextInterface {
|
|
|
825
823
|
const maxRetries = config.maxRetries ?? DEFAULT_MAX_RETRIES;
|
|
826
824
|
|
|
827
825
|
if (metadata.attempts > maxRetries) {
|
|
828
|
-
|
|
829
|
-
// driver implementations may persist metadata without the history
|
|
830
|
-
// entry error (e.g. partial writes/crashes between attempts).
|
|
831
|
-
const lastError = stepData.error ?? metadata.error;
|
|
826
|
+
const lastError = metadata.error;
|
|
832
827
|
const exhaustedError = new StepExhaustedError(
|
|
833
828
|
config.name,
|
|
834
829
|
lastError,
|
|
@@ -941,15 +936,16 @@ export class WorkflowContextImpl implements WorkflowContextInterface {
|
|
|
941
936
|
});
|
|
942
937
|
return output;
|
|
943
938
|
} catch (error) {
|
|
944
|
-
|
|
945
|
-
|
|
946
|
-
|
|
947
|
-
|
|
948
|
-
|
|
949
|
-
|
|
939
|
+
if (entry.kind.type === "step") {
|
|
940
|
+
entry.kind.data.error = String(error);
|
|
941
|
+
}
|
|
942
|
+
entry.dirty = true;
|
|
943
|
+
|
|
944
|
+
// Timeout errors are treated as critical by default. Steps opt
|
|
945
|
+
// into retrying on timeout with retryOnTimeout: true.
|
|
946
|
+
if (error instanceof StepTimeoutError && !config.retryOnTimeout) {
|
|
950
947
|
metadata.status = "exhausted";
|
|
951
948
|
metadata.error = String(error);
|
|
952
|
-
await this.flushStorage();
|
|
953
949
|
await this.notifyStepError(config, metadata.attempts, error, {
|
|
954
950
|
willRetry: false,
|
|
955
951
|
});
|
|
@@ -967,13 +963,8 @@ export class WorkflowContextImpl implements WorkflowContextInterface {
|
|
|
967
963
|
error instanceof CriticalError ||
|
|
968
964
|
error instanceof RollbackError
|
|
969
965
|
) {
|
|
970
|
-
if (entry.kind.type === "step") {
|
|
971
|
-
entry.kind.data.error = String(error);
|
|
972
|
-
}
|
|
973
|
-
entry.dirty = true;
|
|
974
966
|
metadata.status = "exhausted";
|
|
975
967
|
metadata.error = String(error);
|
|
976
|
-
await this.flushStorage();
|
|
977
968
|
await this.notifyStepError(config, metadata.attempts, error, {
|
|
978
969
|
willRetry: false,
|
|
979
970
|
});
|
|
@@ -990,15 +981,10 @@ export class WorkflowContextImpl implements WorkflowContextInterface {
|
|
|
990
981
|
);
|
|
991
982
|
}
|
|
992
983
|
|
|
993
|
-
if (entry.kind.type === "step") {
|
|
994
|
-
entry.kind.data.error = String(error);
|
|
995
|
-
}
|
|
996
|
-
entry.dirty = true;
|
|
997
984
|
const willRetry = metadata.attempts <= maxRetries;
|
|
998
985
|
metadata.status = willRetry ? "failed" : "exhausted";
|
|
999
986
|
metadata.error = String(error);
|
|
1000
987
|
|
|
1001
|
-
await this.flushStorage();
|
|
1002
988
|
if (willRetry) {
|
|
1003
989
|
const retryDelay = calculateBackoff(
|
|
1004
990
|
metadata.attempts,
|
|
@@ -1023,7 +1009,10 @@ export class WorkflowContextImpl implements WorkflowContextInterface {
|
|
|
1023
1009
|
attachTryStepFailure(
|
|
1024
1010
|
new StepExhaustedError(config.name, String(error)),
|
|
1025
1011
|
{
|
|
1026
|
-
kind:
|
|
1012
|
+
kind:
|
|
1013
|
+
error instanceof StepTimeoutError
|
|
1014
|
+
? "timeout"
|
|
1015
|
+
: "exhausted",
|
|
1027
1016
|
stepName: config.name,
|
|
1028
1017
|
attempts: metadata.attempts,
|
|
1029
1018
|
error: extractErrorInfo(error),
|
|
@@ -1042,7 +1031,9 @@ export class WorkflowContextImpl implements WorkflowContextInterface {
|
|
|
1042
1031
|
*
|
|
1043
1032
|
* Note: This does NOT cancel the underlying operation. JavaScript Promises
|
|
1044
1033
|
* cannot be cancelled once started. When a timeout occurs:
|
|
1045
|
-
* - The step is
|
|
1034
|
+
* - The step is rejected with StepTimeoutError. By default this is treated
|
|
1035
|
+
* as a critical failure with no retry. Set retryOnTimeout: true on the
|
|
1036
|
+
* step config to retry timeouts like any other error.
|
|
1046
1037
|
* - The underlying async operation continues running in the background
|
|
1047
1038
|
* - Any side effects from the operation may still occur
|
|
1048
1039
|
*
|
package/src/index.ts
CHANGED
|
@@ -13,6 +13,7 @@ export {
|
|
|
13
13
|
} from "./context.js";
|
|
14
14
|
// Driver
|
|
15
15
|
export type { EngineDriver, KVEntry, KVWrite } from "./driver.js";
|
|
16
|
+
export { extractErrorInfo } from "./error-utils.js";
|
|
16
17
|
// Errors
|
|
17
18
|
export {
|
|
18
19
|
CancelledError,
|
|
@@ -29,7 +30,6 @@ export {
|
|
|
29
30
|
StepExhaustedError,
|
|
30
31
|
StepFailedError,
|
|
31
32
|
} from "./errors.js";
|
|
32
|
-
export { extractErrorInfo } from "./error-utils.js";
|
|
33
33
|
|
|
34
34
|
// Location utilities
|
|
35
35
|
export {
|
|
@@ -82,9 +82,6 @@ export type {
|
|
|
82
82
|
PathSegment,
|
|
83
83
|
RaceEntry,
|
|
84
84
|
RemovedEntry,
|
|
85
|
-
WorkflowEntryMetadataSnapshot,
|
|
86
|
-
WorkflowHistoryEntry,
|
|
87
|
-
WorkflowHistorySnapshot,
|
|
88
85
|
RollbackCheckpointEntry,
|
|
89
86
|
RollbackContextInterface,
|
|
90
87
|
RunWorkflowOptions,
|
|
@@ -102,20 +99,23 @@ export type {
|
|
|
102
99
|
TryStepFailure,
|
|
103
100
|
TryStepResult,
|
|
104
101
|
WorkflowContextInterface,
|
|
102
|
+
WorkflowEntryMetadataSnapshot,
|
|
105
103
|
WorkflowError,
|
|
106
104
|
WorkflowErrorEvent,
|
|
107
105
|
WorkflowErrorHandler,
|
|
108
106
|
WorkflowFunction,
|
|
109
107
|
WorkflowHandle,
|
|
110
|
-
|
|
108
|
+
WorkflowHistoryEntry,
|
|
109
|
+
WorkflowHistorySnapshot,
|
|
110
|
+
WorkflowMessageDriver,
|
|
111
111
|
WorkflowQueue,
|
|
112
112
|
WorkflowQueueMessage,
|
|
113
113
|
WorkflowQueueNextBatchOptions,
|
|
114
114
|
WorkflowQueueNextOptions,
|
|
115
|
-
WorkflowMessageDriver,
|
|
116
115
|
WorkflowResult,
|
|
117
|
-
|
|
116
|
+
WorkflowRollbackErrorEvent,
|
|
118
117
|
WorkflowRunErrorEvent,
|
|
118
|
+
WorkflowRunMode,
|
|
119
119
|
WorkflowState,
|
|
120
120
|
WorkflowStepErrorEvent,
|
|
121
121
|
} from "./types.js";
|
|
@@ -185,9 +185,9 @@ import type {
|
|
|
185
185
|
RunWorkflowOptions,
|
|
186
186
|
Storage,
|
|
187
187
|
WorkflowErrorEvent,
|
|
188
|
-
WorkflowHistorySnapshot,
|
|
189
188
|
WorkflowFunction,
|
|
190
189
|
WorkflowHandle,
|
|
190
|
+
WorkflowHistorySnapshot,
|
|
191
191
|
WorkflowMessageDriver,
|
|
192
192
|
WorkflowResult,
|
|
193
193
|
WorkflowRunMode,
|
package/src/location.ts
CHANGED
|
@@ -159,7 +159,7 @@ export function getChildEntries(
|
|
|
159
159
|
const isChild =
|
|
160
160
|
parentKey === ""
|
|
161
161
|
? true
|
|
162
|
-
: key.startsWith(parentKey
|
|
162
|
+
: key.startsWith(`${parentKey}/`) || key === parentKey;
|
|
163
163
|
|
|
164
164
|
if (isChild) {
|
|
165
165
|
// Return the actual entry's location, not the parent location
|
package/src/types.ts
CHANGED
|
@@ -399,6 +399,8 @@ export interface StepConfig<T> {
|
|
|
399
399
|
retryBackoffMax?: number;
|
|
400
400
|
/** Timeout in ms for step execution (default: 30000). Set to 0 to disable. */
|
|
401
401
|
timeout?: number;
|
|
402
|
+
/** If true, step timeouts retry like any other error instead of failing immediately as critical. Default: false. */
|
|
403
|
+
retryOnTimeout?: boolean;
|
|
402
404
|
}
|
|
403
405
|
|
|
404
406
|
export type TryStepCatchKind =
|
|
@@ -455,7 +457,7 @@ export type LoopResult<S, T> =
|
|
|
455
457
|
* `Loop.continue(undefined)`.
|
|
456
458
|
*/
|
|
457
459
|
export type LoopIterationResult<S, T> = Promise<
|
|
458
|
-
LoopResult<S, T> | (S extends undefined ?
|
|
460
|
+
LoopResult<S, T> | (S extends undefined ? undefined : never)
|
|
459
461
|
>;
|
|
460
462
|
|
|
461
463
|
/**
|