@codemation/core 0.8.1 → 0.10.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/CHANGELOG.md +386 -0
- package/dist/{EngineRuntimeRegistration.types-BP6tsaNP.d.ts → EngineRuntimeRegistration.types-D1fyApMI.d.ts} +2 -2
- package/dist/{EngineWorkflowRunnerService-DzOCa1BW.d.cts → EngineRuntimeRegistration.types-pB3FnzqR.d.cts} +17 -17
- package/dist/{InMemoryRunDataFactory-1iz7_SnO.d.cts → InMemoryRunDataFactory-Xw7v4-sj.d.cts} +31 -29
- package/dist/InMemoryRunEventBusRegistry-VM3OWnHo.cjs +47 -0
- package/dist/InMemoryRunEventBusRegistry-VM3OWnHo.cjs.map +1 -0
- package/dist/InMemoryRunEventBusRegistry-sM4z4n_i.js +41 -0
- package/dist/InMemoryRunEventBusRegistry-sM4z4n_i.js.map +1 -0
- package/dist/{RunIntentService-BqhmdoA1.d.ts → RunIntentService-BE9CAkbf.d.ts} +966 -471
- package/dist/{RunIntentService-S-1lW-gS.d.cts → RunIntentService-siBSjaaY.d.cts} +859 -493
- package/dist/bootstrap/index.cjs +5 -2
- package/dist/bootstrap/index.d.cts +212 -135
- package/dist/bootstrap/index.d.ts +4 -4
- package/dist/bootstrap/index.js +3 -3
- package/dist/{bootstrap-Bx1u4cbS.cjs → bootstrap-Cm5ruQxx.cjs} +253 -2
- package/dist/bootstrap-Cm5ruQxx.cjs.map +1 -0
- package/dist/{bootstrap-BoknFKnw.js → bootstrap-D3r505ko.js} +236 -3
- package/dist/bootstrap-D3r505ko.js.map +1 -0
- package/dist/{index-CVs9rVhl.d.ts → index-DeLl1Tne.d.ts} +632 -230
- package/dist/index.cjs +323 -176
- package/dist/index.cjs.map +1 -1
- package/dist/index.d.cts +544 -91
- package/dist/index.d.ts +3 -3
- package/dist/index.js +299 -166
- package/dist/index.js.map +1 -1
- package/dist/{runtime-DUW6tIJ1.js → runtime-BGNbRnqs.js} +934 -75
- package/dist/runtime-BGNbRnqs.js.map +1 -0
- package/dist/{runtime-Dvo2ru5A.cjs → runtime-DKXJwTNv.cjs} +1028 -73
- package/dist/runtime-DKXJwTNv.cjs.map +1 -0
- package/dist/testing.cjs +4 -4
- package/dist/testing.cjs.map +1 -1
- package/dist/testing.d.cts +2 -2
- package/dist/testing.d.ts +2 -2
- package/dist/testing.js +3 -3
- package/package.json +7 -2
- package/src/ai/AiHost.ts +42 -14
- package/src/authoring/DefinedCollectionRegistry.ts +17 -0
- package/src/authoring/defineCollection.types.ts +181 -0
- package/src/authoring/definePollingTrigger.types.ts +396 -0
- package/src/authoring/definePollingTriggerInternals.ts +74 -0
- package/src/authoring/index.ts +19 -0
- package/src/bootstrap/index.ts +9 -0
- package/src/bootstrap/runtime/EngineRuntimeRegistrar.ts +8 -0
- package/src/browser.ts +1 -0
- package/src/contracts/CodemationTelemetryAttributeNames.ts +6 -0
- package/src/contracts/NoOpNodeExecutionTelemetry.ts +2 -11
- package/src/contracts/NoOpTelemetrySpanScope.ts +46 -10
- package/src/contracts/assertionTypes.ts +63 -0
- package/src/contracts/baseTypes.ts +12 -0
- package/src/contracts/collectionTypes.ts +44 -0
- package/src/contracts/credentialTypes.ts +23 -1
- package/src/contracts/executionPersistenceContracts.ts +30 -0
- package/src/contracts/index.ts +4 -0
- package/src/contracts/runTypes.ts +37 -1
- package/src/contracts/runtimeTypes.ts +42 -0
- package/src/contracts/telemetryTypes.ts +8 -0
- package/src/contracts/testTriggerTypes.ts +66 -0
- package/src/contracts/workflowTypes.ts +36 -7
- package/src/contracts.ts +59 -0
- package/src/events/ConnectionInvocationEventPublisher.ts +46 -0
- package/src/events/index.ts +1 -0
- package/src/events/runEvents.ts +74 -0
- package/src/execution/ChildExecutionScopeFactory.ts +55 -0
- package/src/execution/DefaultExecutionContextFactory.ts +6 -0
- package/src/execution/ExecutionTelemetryCostTrackingDecoratorFactory.ts +18 -0
- package/src/execution/NodeExecutor.ts +10 -2
- package/src/execution/NodeInstanceFactory.ts +13 -1
- package/src/execution/NodeInstantiationError.ts +16 -0
- package/src/execution/NodeRunStateWriter.ts +7 -0
- package/src/execution/NodeRunStateWriterFactory.ts +7 -0
- package/src/execution/WorkflowRunExecutionContextFactory.ts +3 -0
- package/src/execution/index.ts +2 -0
- package/src/index.ts +8 -0
- package/src/orchestration/AbortControllerFactory.ts +9 -0
- package/src/orchestration/NodeExecutionRequestHandlerService.ts +1 -0
- package/src/orchestration/RunContinuationService.ts +3 -0
- package/src/orchestration/RunStartService.ts +122 -3
- package/src/orchestration/TestSuiteOrchestrator.ts +350 -0
- package/src/orchestration/TestSuiteRunIdFactory.ts +11 -0
- package/src/orchestration/TriggerRuntimeService.ts +34 -7
- package/src/orchestration/index.ts +9 -0
- package/src/runtime/EngineFactory.ts +12 -0
- package/src/triggers/polling/PollingTriggerDedupWindow.ts +23 -0
- package/src/triggers/polling/PollingTriggerLogger.ts +18 -0
- package/src/triggers/polling/PollingTriggerRuntime.ts +122 -0
- package/src/triggers/polling/index.ts +5 -0
- package/src/types/index.ts +12 -9
- package/src/workflow/definition/NodeIterationIdFactory.ts +26 -0
- package/src/workflow/dsl/NodeIdSlugifier.ts +18 -0
- package/src/workflow/dsl/WorkflowBuilder.ts +71 -3
- package/src/workflow/dsl/WorkflowDefinitionError.ts +15 -0
- package/src/workflow/index.ts +3 -0
- package/dist/InMemoryRunEventBusRegistry-B0_C4OnP.cjs +0 -262
- package/dist/InMemoryRunEventBusRegistry-B0_C4OnP.cjs.map +0 -1
- package/dist/InMemoryRunEventBusRegistry-C2U83Hmv.js +0 -238
- package/dist/InMemoryRunEventBusRegistry-C2U83Hmv.js.map +0 -1
- package/dist/bootstrap-BoknFKnw.js.map +0 -1
- package/dist/bootstrap-Bx1u4cbS.cjs.map +0 -1
- package/dist/runtime-DUW6tIJ1.js.map +0 -1
- package/dist/runtime-Dvo2ru5A.cjs.map +0 -1
|
@@ -1,4 +1,4 @@
|
|
|
1
|
-
const require_runtime = require('./runtime-
|
|
1
|
+
const require_runtime = require('./runtime-DKXJwTNv.cjs');
|
|
2
2
|
let tsyringe = require("tsyringe");
|
|
3
3
|
tsyringe = require_runtime.__toESM(tsyringe);
|
|
4
4
|
|
|
@@ -101,6 +101,236 @@ var InMemoryWorkflowExecutionRepository = class {
|
|
|
101
101
|
}
|
|
102
102
|
};
|
|
103
103
|
|
|
104
|
+
//#endregion
|
|
105
|
+
//#region src/orchestration/AbortControllerFactory.ts
|
|
106
|
+
/**
|
|
107
|
+
* Mints fresh {@link AbortController}s. Injected (rather than direct `new`) to honor the
|
|
108
|
+
* codebase's no-direct-construction rule and to give tests a seam for substituting a fake.
|
|
109
|
+
*/
|
|
110
|
+
var AbortControllerFactory = class {
|
|
111
|
+
create() {
|
|
112
|
+
return new AbortController();
|
|
113
|
+
}
|
|
114
|
+
};
|
|
115
|
+
|
|
116
|
+
//#endregion
|
|
117
|
+
//#region src/orchestration/TestSuiteOrchestrator.ts
|
|
118
|
+
const DEFAULT_CONCURRENCY = 4;
|
|
119
|
+
/**
|
|
120
|
+
* Drives a {@link TestTriggerNodeConfig.generateItems} iterable into one workflow run per item,
|
|
121
|
+
* with bounded concurrency. Pure engine logic — no persistence, no HTTP, no UI. Hosts adapt by
|
|
122
|
+
* subscribing to {@link RunEventBus} and writing rows on `testSuite*` / `testCase*` / `nodeCompleted`.
|
|
123
|
+
*
|
|
124
|
+
* Cancellation: the supplied `AbortSignal` aborts the source iterable (so credentialed pulls bail)
|
|
125
|
+
* and stops scheduling further cases. In-flight cases are awaited; engine-level cancellation of
|
|
126
|
+
* an already-dispatched run is not yet wired (Phase 2).
|
|
127
|
+
*/
|
|
128
|
+
var TestSuiteOrchestrator = class {
|
|
129
|
+
constructor(engine, testSuiteRunIdFactory, credentialResolverFactory, abortControllerFactory, eventBus, currentDate = () => /* @__PURE__ */ new Date()) {
|
|
130
|
+
this.engine = engine;
|
|
131
|
+
this.testSuiteRunIdFactory = testSuiteRunIdFactory;
|
|
132
|
+
this.credentialResolverFactory = credentialResolverFactory;
|
|
133
|
+
this.abortControllerFactory = abortControllerFactory;
|
|
134
|
+
this.eventBus = eventBus;
|
|
135
|
+
this.currentDate = currentDate;
|
|
136
|
+
}
|
|
137
|
+
async runSuite(args) {
|
|
138
|
+
const triggerNodeId = args.triggerNodeId;
|
|
139
|
+
const definition = args.workflow.nodes.find((n) => n.id === triggerNodeId);
|
|
140
|
+
if (!definition) throw new Error(`Unknown trigger nodeId: ${triggerNodeId}`);
|
|
141
|
+
if (definition.kind !== "trigger") throw new Error(`Node ${triggerNodeId} is not a trigger`);
|
|
142
|
+
const triggerConfig = definition.config;
|
|
143
|
+
if (triggerConfig.triggerKind !== "test") throw new Error(`Node ${triggerNodeId} is not a test trigger (triggerKind="${triggerConfig.triggerKind ?? "live"}")`);
|
|
144
|
+
const testTriggerConfig = triggerConfig;
|
|
145
|
+
if (typeof testTriggerConfig.generateItems !== "function") throw new Error(`Test trigger ${triggerNodeId} is missing a generateItems implementation`);
|
|
146
|
+
const testSuiteRunId = args.testSuiteRunId ?? this.testSuiteRunIdFactory.makeTestSuiteRunId();
|
|
147
|
+
const concurrency = Math.max(1, args.concurrency ?? testTriggerConfig.concurrency ?? DEFAULT_CONCURRENCY);
|
|
148
|
+
const externalSignal = args.signal;
|
|
149
|
+
const internalAbort = this.abortControllerFactory.create();
|
|
150
|
+
const onExternalAbort = () => internalAbort.abort(externalSignal?.reason);
|
|
151
|
+
if (externalSignal) if (externalSignal.aborted) internalAbort.abort(externalSignal.reason);
|
|
152
|
+
else externalSignal.addEventListener("abort", onExternalAbort, { once: true });
|
|
153
|
+
const triggerNodeName = definition.name ?? testTriggerConfig.name;
|
|
154
|
+
await this.publish({
|
|
155
|
+
kind: "testSuiteStarted",
|
|
156
|
+
testSuiteRunId,
|
|
157
|
+
workflowId: args.workflow.id,
|
|
158
|
+
triggerNodeId,
|
|
159
|
+
...triggerNodeName ? { triggerNodeName } : {},
|
|
160
|
+
concurrency,
|
|
161
|
+
at: this.now()
|
|
162
|
+
});
|
|
163
|
+
const setupContext = {
|
|
164
|
+
workflowId: args.workflow.id,
|
|
165
|
+
nodeId: triggerNodeId,
|
|
166
|
+
config: testTriggerConfig,
|
|
167
|
+
testSuiteRunId,
|
|
168
|
+
getCredential: this.credentialResolverFactory.create(args.workflow.id, triggerNodeId, testTriggerConfig),
|
|
169
|
+
signal: internalAbort.signal
|
|
170
|
+
};
|
|
171
|
+
const cases = [];
|
|
172
|
+
let nextIndex = 0;
|
|
173
|
+
let inFlight = 0;
|
|
174
|
+
let waitForSlot;
|
|
175
|
+
let releaseSlot;
|
|
176
|
+
const queue = [];
|
|
177
|
+
let generationError;
|
|
178
|
+
const acquireSlot = async () => {
|
|
179
|
+
while (inFlight >= concurrency) {
|
|
180
|
+
if (!waitForSlot) waitForSlot = new Promise((resolve) => {
|
|
181
|
+
releaseSlot = resolve;
|
|
182
|
+
});
|
|
183
|
+
await waitForSlot;
|
|
184
|
+
}
|
|
185
|
+
inFlight += 1;
|
|
186
|
+
};
|
|
187
|
+
const release = () => {
|
|
188
|
+
inFlight -= 1;
|
|
189
|
+
if (releaseSlot) {
|
|
190
|
+
const fn = releaseSlot;
|
|
191
|
+
releaseSlot = void 0;
|
|
192
|
+
waitForSlot = void 0;
|
|
193
|
+
fn();
|
|
194
|
+
}
|
|
195
|
+
};
|
|
196
|
+
try {
|
|
197
|
+
for await (const item of testTriggerConfig.generateItems(setupContext)) {
|
|
198
|
+
if (internalAbort.signal.aborted) break;
|
|
199
|
+
await acquireSlot();
|
|
200
|
+
if (internalAbort.signal.aborted) {
|
|
201
|
+
release();
|
|
202
|
+
break;
|
|
203
|
+
}
|
|
204
|
+
const testCaseIndex = nextIndex++;
|
|
205
|
+
const testCaseLabel = this.resolveCaseLabel(testTriggerConfig, item);
|
|
206
|
+
queue.push(this.runOneCase({
|
|
207
|
+
workflow: args.workflow,
|
|
208
|
+
triggerNodeId,
|
|
209
|
+
testSuiteRunId,
|
|
210
|
+
testCaseIndex,
|
|
211
|
+
testCaseLabel,
|
|
212
|
+
item
|
|
213
|
+
}).then((outcome) => {
|
|
214
|
+
cases.push(outcome);
|
|
215
|
+
}).finally(release));
|
|
216
|
+
}
|
|
217
|
+
} catch (err) {
|
|
218
|
+
generationError = err instanceof Error ? err : new Error(String(err));
|
|
219
|
+
} finally {
|
|
220
|
+
if (externalSignal) externalSignal.removeEventListener("abort", onExternalAbort);
|
|
221
|
+
}
|
|
222
|
+
await Promise.all(queue);
|
|
223
|
+
cases.sort((a, b) => a.testCaseIndex - b.testCaseIndex);
|
|
224
|
+
const totalCases = cases.length;
|
|
225
|
+
const passedCases = cases.filter((c) => c.status === "succeeded").length;
|
|
226
|
+
const failedCases = cases.filter((c) => c.status === "failed").length;
|
|
227
|
+
const status = this.deriveSuiteStatus({
|
|
228
|
+
generationError,
|
|
229
|
+
cancelled: internalAbort.signal.aborted,
|
|
230
|
+
totalCases,
|
|
231
|
+
passedCases,
|
|
232
|
+
failedCases
|
|
233
|
+
});
|
|
234
|
+
await this.publish({
|
|
235
|
+
kind: "testSuiteFinished",
|
|
236
|
+
testSuiteRunId,
|
|
237
|
+
workflowId: args.workflow.id,
|
|
238
|
+
status,
|
|
239
|
+
totalCases,
|
|
240
|
+
passedCases,
|
|
241
|
+
failedCases,
|
|
242
|
+
at: this.now()
|
|
243
|
+
});
|
|
244
|
+
if (generationError && status === "errored") throw generationError;
|
|
245
|
+
return {
|
|
246
|
+
testSuiteRunId,
|
|
247
|
+
workflowId: args.workflow.id,
|
|
248
|
+
triggerNodeId,
|
|
249
|
+
status,
|
|
250
|
+
totalCases,
|
|
251
|
+
passedCases,
|
|
252
|
+
failedCases,
|
|
253
|
+
cases
|
|
254
|
+
};
|
|
255
|
+
}
|
|
256
|
+
async runOneCase(args) {
|
|
257
|
+
const executionOptions = { testContext: {
|
|
258
|
+
testSuiteRunId: args.testSuiteRunId,
|
|
259
|
+
testCaseIndex: args.testCaseIndex,
|
|
260
|
+
...args.testCaseLabel !== void 0 ? { testCaseLabel: args.testCaseLabel } : {}
|
|
261
|
+
} };
|
|
262
|
+
const initial = await this.engine.runWorkflow(args.workflow, args.triggerNodeId, [args.item], void 0, executionOptions);
|
|
263
|
+
const runId = initial.runId;
|
|
264
|
+
await this.publish({
|
|
265
|
+
kind: "testCaseStarted",
|
|
266
|
+
testSuiteRunId: args.testSuiteRunId,
|
|
267
|
+
testCaseIndex: args.testCaseIndex,
|
|
268
|
+
runId,
|
|
269
|
+
workflowId: args.workflow.id,
|
|
270
|
+
at: this.now(),
|
|
271
|
+
...args.testCaseLabel !== void 0 ? { testCaseLabel: args.testCaseLabel } : {}
|
|
272
|
+
});
|
|
273
|
+
let terminal;
|
|
274
|
+
if (initial.status === "completed" || initial.status === "failed") terminal = initial;
|
|
275
|
+
else terminal = await this.engine.waitForCompletion(runId);
|
|
276
|
+
const status = terminal.status === "completed" ? "succeeded" : "failed";
|
|
277
|
+
await this.publish({
|
|
278
|
+
kind: "testCaseCompleted",
|
|
279
|
+
testSuiteRunId: args.testSuiteRunId,
|
|
280
|
+
testCaseIndex: args.testCaseIndex,
|
|
281
|
+
runId,
|
|
282
|
+
workflowId: args.workflow.id,
|
|
283
|
+
status,
|
|
284
|
+
at: this.now()
|
|
285
|
+
});
|
|
286
|
+
return {
|
|
287
|
+
testCaseIndex: args.testCaseIndex,
|
|
288
|
+
runId,
|
|
289
|
+
status
|
|
290
|
+
};
|
|
291
|
+
}
|
|
292
|
+
deriveSuiteStatus(args) {
|
|
293
|
+
if (args.generationError && args.totalCases === 0) return "errored";
|
|
294
|
+
if (args.cancelled) return "cancelled";
|
|
295
|
+
if (args.generationError) return "errored";
|
|
296
|
+
if (args.totalCases === 0) return "succeeded";
|
|
297
|
+
if (args.failedCases === 0) return "succeeded";
|
|
298
|
+
if (args.passedCases === 0) return "failed";
|
|
299
|
+
return "partial";
|
|
300
|
+
}
|
|
301
|
+
now() {
|
|
302
|
+
return this.currentDate().toISOString();
|
|
303
|
+
}
|
|
304
|
+
/** Defensive label resolver — author-supplied callbacks throw / return non-strings; we tolerate both. */
|
|
305
|
+
resolveCaseLabel(config, item) {
|
|
306
|
+
if (typeof config.caseLabel !== "function") return void 0;
|
|
307
|
+
try {
|
|
308
|
+
const result = config.caseLabel(item);
|
|
309
|
+
if (typeof result !== "string") return void 0;
|
|
310
|
+
const trimmed = result.trim();
|
|
311
|
+
return trimmed.length === 0 ? void 0 : trimmed;
|
|
312
|
+
} catch {
|
|
313
|
+
return;
|
|
314
|
+
}
|
|
315
|
+
}
|
|
316
|
+
async publish(event) {
|
|
317
|
+
if (!this.eventBus) return;
|
|
318
|
+
await this.eventBus.publish(event);
|
|
319
|
+
}
|
|
320
|
+
};
|
|
321
|
+
|
|
322
|
+
//#endregion
|
|
323
|
+
//#region src/orchestration/TestSuiteRunIdFactory.ts
|
|
324
|
+
/**
|
|
325
|
+
* Mints unique TestSuiteRun identifiers. Separated from {@link import("../types").RunIdFactory}
|
|
326
|
+
* so suite ids and per-case workflow run ids never alias.
|
|
327
|
+
*/
|
|
328
|
+
var TestSuiteRunIdFactory = class {
|
|
329
|
+
makeTestSuiteRunId() {
|
|
330
|
+
return `tsr_${Date.now().toString(36)}_${Math.random().toString(36).slice(2, 10)}`;
|
|
331
|
+
}
|
|
332
|
+
};
|
|
333
|
+
|
|
104
334
|
//#endregion
|
|
105
335
|
//#region src/scheduler/InlineDrivingSchedulerFactory.ts
|
|
106
336
|
var InlineDrivingSchedulerFactory = class {
|
|
@@ -130,6 +360,9 @@ var EngineRuntimeRegistrar = class {
|
|
|
130
360
|
if (!container.isRegistered(require_runtime.ItemExprResolver, true)) container.registerSingleton(require_runtime.ItemExprResolver, require_runtime.ItemExprResolver);
|
|
131
361
|
if (!container.isRegistered(require_runtime.NodeOutputNormalizer, true)) container.registerSingleton(require_runtime.NodeOutputNormalizer, require_runtime.NodeOutputNormalizer);
|
|
132
362
|
if (!container.isRegistered(require_runtime.RunnableOutputBehaviorResolver, true)) container.registerSingleton(require_runtime.RunnableOutputBehaviorResolver, require_runtime.RunnableOutputBehaviorResolver);
|
|
363
|
+
if (!container.isRegistered(require_runtime.ChildExecutionScopeFactory, true)) container.register(require_runtime.ChildExecutionScopeFactory, { useFactory: (0, tsyringe.instanceCachingFactory)((dependencyContainer) => {
|
|
364
|
+
return new require_runtime.ChildExecutionScopeFactory(dependencyContainer.resolve(require_runtime.CoreTokens.ActivationIdFactory));
|
|
365
|
+
}) });
|
|
133
366
|
container.registerSingleton(EngineExecutionLimitsPolicyFactory, EngineExecutionLimitsPolicyFactory);
|
|
134
367
|
container.registerSingleton(require_runtime.NodeInstanceFactoryFactory, require_runtime.NodeInstanceFactoryFactory);
|
|
135
368
|
container.registerSingleton(require_runtime.DefaultAsyncSleeper, require_runtime.DefaultAsyncSleeper);
|
|
@@ -219,6 +452,12 @@ var EngineRuntimeRegistrar = class {
|
|
|
219
452
|
};
|
|
220
453
|
|
|
221
454
|
//#endregion
|
|
455
|
+
Object.defineProperty(exports, 'AbortControllerFactory', {
|
|
456
|
+
enumerable: true,
|
|
457
|
+
get: function () {
|
|
458
|
+
return AbortControllerFactory;
|
|
459
|
+
}
|
|
460
|
+
});
|
|
222
461
|
Object.defineProperty(exports, 'EngineExecutionLimitsPolicyFactory', {
|
|
223
462
|
enumerable: true,
|
|
224
463
|
get: function () {
|
|
@@ -243,4 +482,16 @@ Object.defineProperty(exports, 'RunSummaryMapper', {
|
|
|
243
482
|
return RunSummaryMapper;
|
|
244
483
|
}
|
|
245
484
|
});
|
|
246
|
-
|
|
485
|
+
Object.defineProperty(exports, 'TestSuiteOrchestrator', {
|
|
486
|
+
enumerable: true,
|
|
487
|
+
get: function () {
|
|
488
|
+
return TestSuiteOrchestrator;
|
|
489
|
+
}
|
|
490
|
+
});
|
|
491
|
+
Object.defineProperty(exports, 'TestSuiteRunIdFactory', {
|
|
492
|
+
enumerable: true,
|
|
493
|
+
get: function () {
|
|
494
|
+
return TestSuiteRunIdFactory;
|
|
495
|
+
}
|
|
496
|
+
});
|
|
497
|
+
//# sourceMappingURL=bootstrap-Cm5ruQxx.cjs.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"bootstrap-Cm5ruQxx.cjs","names":["EngineExecutionLimitsPolicy","ENGINE_EXECUTION_LIMITS_DEFAULTS","RunFinishedAtFactory","out: RunPruneCandidate[]","RunFinishedAtFactory","engine: TestSuiteOrchestratorEngine","testSuiteRunIdFactory: TestSuiteRunIdFactory","credentialResolverFactory: CredentialResolverFactory","abortControllerFactory: AbortControllerFactory","eventBus: RunEventBus | undefined","currentDate: () => Date","setupContext: TestTriggerSetupContext","cases: TestSuiteCaseOutcome[]","waitForSlot: Promise<void> | undefined","releaseSlot: (() => void) | undefined","queue: Array<Promise<void>>","generationError: Error | undefined","status: TestSuiteRunStatus","executionOptions: RunExecutionOptions","terminal: Extract<RunResult, { status: \"completed\" | \"failed\" }>","status: TestCaseRunStatus","InlineDrivingScheduler","ItemExprResolver","NodeOutputNormalizer","RunnableOutputBehaviorResolver","ChildExecutionScopeFactory","CoreTokens","NodeInstanceFactoryFactory","DefaultAsyncSleeper","InProcessRetryRunnerFactory","NodeExecutorFactory","RunIntentServiceFactory","EngineWorkflowRunnerServiceFactory","WorkflowRepositoryWebhookTriggerMatcherFactory","NodeExecutor","InlineDrivingScheduler","EngineFactory","Engine","RunIntentService"],"sources":["../src/policies/executionLimits/EngineExecutionLimitsPolicyFactory.ts","../src/runStorage/RunSummaryMapper.ts","../src/runStorage/InMemoryWorkflowExecutionRepository.ts","../src/orchestration/AbortControllerFactory.ts","../src/orchestration/TestSuiteOrchestrator.ts","../src/orchestration/TestSuiteRunIdFactory.ts","../src/scheduler/InlineDrivingSchedulerFactory.ts","../src/bootstrap/runtime/EngineRuntimeRegistrar.ts"],"sourcesContent":["import {\n EngineExecutionLimitsPolicy,\n ENGINE_EXECUTION_LIMITS_DEFAULTS,\n type EngineExecutionLimitsPolicyConfig,\n} from \"./EngineExecutionLimitsPolicy\";\n\n/**\n * Builds {@link EngineExecutionLimitsPolicy} by merging {@link ENGINE_EXECUTION_LIMITS_DEFAULTS} with optional `overrides` (e.g. host `runtime.engineExecutionLimits`).\n */\nexport class EngineExecutionLimitsPolicyFactory {\n create(overrides?: Partial<EngineExecutionLimitsPolicyConfig>): EngineExecutionLimitsPolicy {\n return new EngineExecutionLimitsPolicy({ ...ENGINE_EXECUTION_LIMITS_DEFAULTS, ...overrides });\n }\n}\n","import { RunFinishedAtFactory } from \"../contracts/runFinishedAtFactory\";\nimport type { PersistedRunState, RunSummary } from \"../types\";\n\n/** Maps persisted run state to API run summaries for listings. */\nexport class RunSummaryMapper {\n static fromPersistedState(state: PersistedRunState): RunSummary {\n return {\n runId: state.runId,\n workflowId: state.workflowId,\n startedAt: state.startedAt,\n status: state.status,\n finishedAt: RunFinishedAtFactory.resolveIso(state),\n parent: state.parent,\n executionOptions: state.executionOptions,\n };\n }\n}\n","import type {\n EngineRunCounters,\n NodeId,\n NodeOutputs,\n ParentExecutionRef,\n PersistedRunSchedulingState,\n PersistedRunState,\n RunId,\n RunSummary,\n WorkflowExecutionListingRepository,\n WorkflowExecutionPruneRepository,\n WorkflowExecutionRepository,\n RunPruneCandidate,\n WorkflowId,\n} from \"../types\";\nimport { RunFinishedAtFactory } from \"../contracts/runFinishedAtFactory\";\nimport { RunSummaryMapper } from \"./RunSummaryMapper\";\n\nexport class InMemoryWorkflowExecutionRepository\n implements WorkflowExecutionRepository, WorkflowExecutionListingRepository, WorkflowExecutionPruneRepository\n{\n private readonly runs = new Map<RunId, PersistedRunState>();\n\n async createRun(args: {\n runId: RunId;\n workflowId: WorkflowId;\n startedAt: string;\n parent?: ParentExecutionRef;\n executionOptions?: PersistedRunState[\"executionOptions\"];\n control?: PersistedRunState[\"control\"];\n workflowSnapshot?: PersistedRunState[\"workflowSnapshot\"];\n mutableState?: PersistedRunState[\"mutableState\"];\n policySnapshot?: PersistedRunState[\"policySnapshot\"];\n engineCounters?: EngineRunCounters;\n }): Promise<void> {\n this.runs.set(args.runId, {\n runId: args.runId,\n workflowId: args.workflowId,\n startedAt: args.startedAt,\n revision: 0,\n parent: args.parent,\n executionOptions: args.executionOptions,\n control: args.control,\n workflowSnapshot: args.workflowSnapshot,\n mutableState: args.mutableState,\n policySnapshot: args.policySnapshot,\n engineCounters: args.engineCounters,\n status: \"running\",\n queue: [],\n outputsByNode: {} as Record<NodeId, NodeOutputs>,\n nodeSnapshotsByNodeId: {},\n connectionInvocations: [],\n });\n }\n\n async load(runId: RunId): Promise<PersistedRunState | undefined> {\n return this.runs.get(runId);\n }\n\n async loadSchedulingState(runId: RunId): Promise<PersistedRunSchedulingState | undefined> {\n const state = this.runs.get(runId);\n if (!state) {\n return undefined;\n }\n return {\n pending: state.pending ? { ...state.pending } : undefined,\n queue: state.queue.map((entry) => ({ ...entry })),\n };\n }\n\n async save(state: PersistedRunState): Promise<void> {\n this.runs.set(state.runId, { ...state, revision: (state.revision ?? 0) + 1 });\n }\n\n async deleteRun(runId: RunId): Promise<void> {\n this.runs.delete(runId);\n }\n\n async listRuns(args?: Readonly<{ workflowId?: WorkflowId; limit?: number }>): Promise<ReadonlyArray<RunSummary>> {\n const limit = args?.limit ?? 50;\n const summaries = [...this.runs.values()]\n .filter((s) => (args?.workflowId ? s.workflowId === args.workflowId : true))\n .sort((a, b) => b.startedAt.localeCompare(a.startedAt))\n .slice(0, limit)\n .map((s) => RunSummaryMapper.fromPersistedState(s));\n return summaries;\n }\n\n async listRunsOlderThan(\n args: Readonly<{ nowIso: string; defaultRetentionSeconds: number; limit?: number }>,\n ): Promise<ReadonlyArray<RunPruneCandidate>> {\n const limit = args.limit ?? 100;\n const out: RunPruneCandidate[] = [];\n for (const s of this.runs.values()) {\n if (s.status !== \"completed\" && s.status !== \"failed\") continue;\n const finishedAt = RunFinishedAtFactory.resolveIso(s);\n const retentionSeconds = s.policySnapshot?.retentionSeconds ?? args.defaultRetentionSeconds;\n const cutoffIso = new Date(new Date(args.nowIso).getTime() - retentionSeconds * 1000).toISOString();\n if (!finishedAt || finishedAt >= cutoffIso) continue;\n out.push({\n runId: s.runId,\n workflowId: s.workflowId,\n startedAt: s.startedAt,\n finishedAt,\n });\n }\n out.sort((a, b) => a.finishedAt.localeCompare(b.finishedAt));\n return out.slice(0, limit);\n }\n}\n","/**\n * Mints fresh {@link AbortController}s. Injected (rather than direct `new`) to honor the\n * codebase's no-direct-construction rule and to give tests a seam for substituting a fake.\n */\nexport class AbortControllerFactory {\n create(): AbortController {\n return new AbortController();\n }\n}\n","import type { CredentialResolverFactory } from \"../execution/CredentialResolverFactory\";\nimport type { RunEventBus, TestCaseRunStatus, TestSuiteRunStatus } from \"../events/runEvents\";\nimport type {\n Item,\n Items,\n NodeId,\n ParentExecutionRef,\n RunExecutionOptions,\n RunId,\n RunResult,\n TestSuiteRunId,\n TestTriggerNodeConfig,\n TestTriggerSetupContext,\n TriggerNodeConfig,\n WorkflowDefinition,\n WorkflowId,\n} from \"../types\";\n\nimport type { AbortControllerFactory } from \"./AbortControllerFactory\";\nimport { TestSuiteRunIdFactory } from \"./TestSuiteRunIdFactory\";\n\nconst DEFAULT_CONCURRENCY = 4;\n\n/**\n * Engine-facade subset the orchestrator needs. Kept narrow on purpose so unit tests can\n * substitute a fake without depending on the full Engine wiring.\n */\nexport interface TestSuiteOrchestratorEngine {\n runWorkflow(\n wf: WorkflowDefinition,\n startAt: NodeId,\n items: Items,\n parent?: ParentExecutionRef,\n executionOptions?: RunExecutionOptions,\n ): Promise<RunResult>;\n waitForCompletion(runId: RunId): Promise<Extract<RunResult, { status: \"completed\" | \"failed\" }>>;\n}\n\nexport interface TestSuiteCaseOutcome {\n readonly testCaseIndex: number;\n readonly runId: RunId;\n readonly status: TestCaseRunStatus;\n}\n\nexport interface TestSuiteRunResult {\n readonly testSuiteRunId: TestSuiteRunId;\n readonly workflowId: WorkflowId;\n readonly triggerNodeId: NodeId;\n readonly status: TestSuiteRunStatus;\n readonly totalCases: number;\n readonly passedCases: number;\n readonly failedCases: number;\n readonly cases: ReadonlyArray<TestSuiteCaseOutcome>;\n}\n\nexport interface RunTestSuiteArgs {\n readonly workflow: WorkflowDefinition;\n readonly triggerNodeId: NodeId;\n readonly testSuiteRunId?: TestSuiteRunId;\n readonly concurrency?: number;\n readonly signal?: AbortSignal;\n}\n\n/**\n * Drives a {@link TestTriggerNodeConfig.generateItems} iterable into one workflow run per item,\n * with bounded concurrency. Pure engine logic — no persistence, no HTTP, no UI. Hosts adapt by\n * subscribing to {@link RunEventBus} and writing rows on `testSuite*` / `testCase*` / `nodeCompleted`.\n *\n * Cancellation: the supplied `AbortSignal` aborts the source iterable (so credentialed pulls bail)\n * and stops scheduling further cases. In-flight cases are awaited; engine-level cancellation of\n * an already-dispatched run is not yet wired (Phase 2).\n */\nexport class TestSuiteOrchestrator {\n constructor(\n private readonly engine: TestSuiteOrchestratorEngine,\n private readonly testSuiteRunIdFactory: TestSuiteRunIdFactory,\n private readonly credentialResolverFactory: CredentialResolverFactory,\n private readonly abortControllerFactory: AbortControllerFactory,\n private readonly eventBus: RunEventBus | undefined,\n private readonly currentDate: () => Date = () => new Date(),\n ) {}\n\n async runSuite(args: RunTestSuiteArgs): Promise<TestSuiteRunResult> {\n const triggerNodeId = args.triggerNodeId;\n const definition = args.workflow.nodes.find((n) => n.id === triggerNodeId);\n if (!definition) {\n throw new Error(`Unknown trigger nodeId: ${triggerNodeId}`);\n }\n if (definition.kind !== \"trigger\") {\n throw new Error(`Node ${triggerNodeId} is not a trigger`);\n }\n const triggerConfig = definition.config as TriggerNodeConfig;\n if (triggerConfig.triggerKind !== \"test\") {\n throw new Error(\n `Node ${triggerNodeId} is not a test trigger (triggerKind=\"${triggerConfig.triggerKind ?? \"live\"}\")`,\n );\n }\n const testTriggerConfig = triggerConfig as TestTriggerNodeConfig<unknown>;\n if (typeof testTriggerConfig.generateItems !== \"function\") {\n throw new Error(`Test trigger ${triggerNodeId} is missing a generateItems implementation`);\n }\n\n const testSuiteRunId = args.testSuiteRunId ?? this.testSuiteRunIdFactory.makeTestSuiteRunId();\n const concurrency = Math.max(1, args.concurrency ?? testTriggerConfig.concurrency ?? DEFAULT_CONCURRENCY);\n const externalSignal = args.signal;\n const internalAbort = this.abortControllerFactory.create();\n const onExternalAbort = () => internalAbort.abort(externalSignal?.reason);\n if (externalSignal) {\n if (externalSignal.aborted) {\n internalAbort.abort(externalSignal.reason);\n } else {\n externalSignal.addEventListener(\"abort\", onExternalAbort, { once: true });\n }\n }\n\n const triggerNodeName = definition.name ?? testTriggerConfig.name;\n\n await this.publish({\n kind: \"testSuiteStarted\",\n testSuiteRunId,\n workflowId: args.workflow.id,\n triggerNodeId,\n ...(triggerNodeName ? { triggerNodeName } : {}),\n concurrency,\n at: this.now(),\n });\n\n const setupContext: TestTriggerSetupContext = {\n workflowId: args.workflow.id,\n nodeId: triggerNodeId,\n config: testTriggerConfig,\n testSuiteRunId,\n getCredential: this.credentialResolverFactory.create(args.workflow.id, triggerNodeId, testTriggerConfig),\n signal: internalAbort.signal,\n };\n\n const cases: TestSuiteCaseOutcome[] = [];\n let nextIndex = 0;\n let inFlight = 0;\n let waitForSlot: Promise<void> | undefined;\n let releaseSlot: (() => void) | undefined;\n const queue: Array<Promise<void>> = [];\n let generationError: Error | undefined;\n\n const acquireSlot = async (): Promise<void> => {\n while (inFlight >= concurrency) {\n if (!waitForSlot) {\n waitForSlot = new Promise<void>((resolve) => {\n releaseSlot = resolve;\n });\n }\n await waitForSlot;\n }\n inFlight += 1;\n };\n\n const release = (): void => {\n inFlight -= 1;\n if (releaseSlot) {\n const fn = releaseSlot;\n releaseSlot = undefined;\n waitForSlot = undefined;\n fn();\n }\n };\n\n try {\n for await (const item of testTriggerConfig.generateItems(setupContext) as AsyncIterable<Item<unknown>>) {\n if (internalAbort.signal.aborted) {\n break;\n }\n await acquireSlot();\n if (internalAbort.signal.aborted) {\n release();\n break;\n }\n const testCaseIndex = nextIndex++;\n const testCaseLabel = this.resolveCaseLabel(testTriggerConfig, item);\n queue.push(\n this.runOneCase({\n workflow: args.workflow,\n triggerNodeId,\n testSuiteRunId,\n testCaseIndex,\n testCaseLabel,\n item,\n })\n .then((outcome) => {\n cases.push(outcome);\n })\n .finally(release),\n );\n }\n } catch (err) {\n generationError = err instanceof Error ? err : new Error(String(err));\n } finally {\n if (externalSignal) {\n externalSignal.removeEventListener(\"abort\", onExternalAbort);\n }\n }\n\n await Promise.all(queue);\n\n cases.sort((a, b) => a.testCaseIndex - b.testCaseIndex);\n const totalCases = cases.length;\n const passedCases = cases.filter((c) => c.status === \"succeeded\").length;\n const failedCases = cases.filter((c) => c.status === \"failed\").length;\n const status: TestSuiteRunStatus = this.deriveSuiteStatus({\n generationError,\n cancelled: internalAbort.signal.aborted,\n totalCases,\n passedCases,\n failedCases,\n });\n\n await this.publish({\n kind: \"testSuiteFinished\",\n testSuiteRunId,\n workflowId: args.workflow.id,\n status,\n totalCases,\n passedCases,\n failedCases,\n at: this.now(),\n });\n\n if (generationError && status === \"errored\") {\n throw generationError;\n }\n\n return {\n testSuiteRunId,\n workflowId: args.workflow.id,\n triggerNodeId,\n status,\n totalCases,\n passedCases,\n failedCases,\n cases,\n };\n }\n\n private async runOneCase(args: {\n workflow: WorkflowDefinition;\n triggerNodeId: NodeId;\n testSuiteRunId: TestSuiteRunId;\n testCaseIndex: number;\n testCaseLabel: string | undefined;\n item: Item<unknown>;\n }): Promise<TestSuiteCaseOutcome> {\n const executionOptions: RunExecutionOptions = {\n testContext: {\n testSuiteRunId: args.testSuiteRunId,\n testCaseIndex: args.testCaseIndex,\n ...(args.testCaseLabel !== undefined ? { testCaseLabel: args.testCaseLabel } : {}),\n },\n };\n\n const initial = await this.engine.runWorkflow(\n args.workflow,\n args.triggerNodeId,\n [args.item],\n undefined,\n executionOptions,\n );\n\n const runId = initial.runId;\n await this.publish({\n kind: \"testCaseStarted\",\n testSuiteRunId: args.testSuiteRunId,\n testCaseIndex: args.testCaseIndex,\n runId,\n workflowId: args.workflow.id,\n at: this.now(),\n ...(args.testCaseLabel !== undefined ? { testCaseLabel: args.testCaseLabel } : {}),\n });\n\n let terminal: Extract<RunResult, { status: \"completed\" | \"failed\" }>;\n if (initial.status === \"completed\" || initial.status === \"failed\") {\n terminal = initial;\n } else {\n terminal = await this.engine.waitForCompletion(runId);\n }\n\n // RunResult.status from the engine narrows to \"completed\" | \"failed\" here; widening to\n // \"errored\" / \"cancelled\" happens outside this code path (tracker downgrade for assertion\n // failures; outer abort handling for cancelled).\n const status: TestCaseRunStatus = terminal.status === \"completed\" ? \"succeeded\" : \"failed\";\n await this.publish({\n kind: \"testCaseCompleted\",\n testSuiteRunId: args.testSuiteRunId,\n testCaseIndex: args.testCaseIndex,\n runId,\n workflowId: args.workflow.id,\n status,\n at: this.now(),\n });\n return { testCaseIndex: args.testCaseIndex, runId, status };\n }\n\n private deriveSuiteStatus(args: {\n generationError: Error | undefined;\n cancelled: boolean;\n totalCases: number;\n passedCases: number;\n failedCases: number;\n }): TestSuiteRunStatus {\n if (args.generationError && args.totalCases === 0) {\n return \"errored\";\n }\n if (args.cancelled) {\n return \"cancelled\";\n }\n if (args.generationError) {\n return \"errored\";\n }\n if (args.totalCases === 0) {\n return \"succeeded\";\n }\n if (args.failedCases === 0) {\n return \"succeeded\";\n }\n if (args.passedCases === 0) {\n return \"failed\";\n }\n return \"partial\";\n }\n\n private now(): string {\n return this.currentDate().toISOString();\n }\n\n /** Defensive label resolver — author-supplied callbacks throw / return non-strings; we tolerate both. */\n private resolveCaseLabel(config: TestTriggerNodeConfig<unknown>, item: Item<unknown>): string | undefined {\n if (typeof config.caseLabel !== \"function\") return undefined;\n try {\n const result = config.caseLabel(item);\n if (typeof result !== \"string\") return undefined;\n const trimmed = result.trim();\n return trimmed.length === 0 ? undefined : trimmed;\n } catch {\n return undefined;\n }\n }\n\n private async publish(event: Parameters<RunEventBus[\"publish\"]>[0]): Promise<void> {\n if (!this.eventBus) return;\n await this.eventBus.publish(event);\n }\n}\n","import type { TestSuiteRunId } from \"../contracts/testTriggerTypes\";\n\n/**\n * Mints unique TestSuiteRun identifiers. Separated from {@link import(\"../types\").RunIdFactory}\n * so suite ids and per-case workflow run ids never alias.\n */\nexport class TestSuiteRunIdFactory {\n makeTestSuiteRunId(): TestSuiteRunId {\n return `tsr_${Date.now().toString(36)}_${Math.random().toString(36).slice(2, 10)}`;\n }\n}\n","import { NodeExecutor } from \"../execution/NodeExecutor\";\n\nimport { InlineDrivingScheduler } from \"./InlineDrivingScheduler\";\n\nexport class InlineDrivingSchedulerFactory {\n create(nodeExecutor: NodeExecutor): InlineDrivingScheduler {\n return new InlineDrivingScheduler(nodeExecutor);\n }\n}\n","import { instanceCachingFactory, type DependencyContainer } from \"../../di\";\nimport { CoreTokens } from \"../../di\";\nimport { EngineExecutionLimitsPolicyFactory } from \"../../policies/executionLimits/EngineExecutionLimitsPolicyFactory\";\nimport {\n ChildExecutionScopeFactory,\n DefaultAsyncSleeper,\n InProcessRetryRunnerFactory,\n ItemExprResolver,\n NodeExecutor,\n NodeExecutorFactory,\n NodeInstanceFactoryFactory,\n NodeOutputNormalizer,\n RunnableOutputBehaviorResolver,\n} from \"../../execution\";\nimport {\n EngineFactory,\n EngineWorkflowRunnerServiceFactory,\n RunIntentServiceFactory,\n RunIntentService,\n WorkflowRepositoryWebhookTriggerMatcherFactory,\n} from \"../../runtime\";\nimport { InlineDrivingScheduler } from \"../../scheduler/InlineDrivingScheduler\";\nimport { InlineDrivingSchedulerFactory } from \"../../scheduler/InlineDrivingSchedulerFactory\";\nimport { Engine } from \"../../orchestration/Engine\";\nimport type { EngineRuntimeRegistrationOptions } from \"./EngineRuntimeRegistration.types\";\nimport type { WebhookTriggerMatcherProvider } from \"./EngineRuntimeRegistration.types\";\n\n/**\n * Container-first entry: call on a host/test container **after** workflow, run, node, and credential\n * ports are registered. The registrar owns the default inline scheduler, engine binding,\n * and intent-surface wiring so hosts only override the seams they actually replace.\n */\nexport class EngineRuntimeRegistrar {\n register(container: DependencyContainer, options?: EngineRuntimeRegistrationOptions): void {\n this.registerSupportFactories(container);\n this.registerExecutionLimitsPolicy(container, options);\n this.ensureWorkflowNodeInstanceFactory(container);\n this.ensureNodeExecutor(container);\n this.registerDefaultActivationScheduler(container);\n this.registerEngine(container, options);\n this.registerIntentServices(container);\n }\n\n private registerSupportFactories(container: DependencyContainer): void {\n if (!container.isRegistered(ItemExprResolver, true)) {\n container.registerSingleton(ItemExprResolver, ItemExprResolver);\n }\n if (!container.isRegistered(NodeOutputNormalizer, true)) {\n container.registerSingleton(NodeOutputNormalizer, NodeOutputNormalizer);\n }\n if (!container.isRegistered(RunnableOutputBehaviorResolver, true)) {\n container.registerSingleton(RunnableOutputBehaviorResolver, RunnableOutputBehaviorResolver);\n }\n if (!container.isRegistered(ChildExecutionScopeFactory, true)) {\n container.register(ChildExecutionScopeFactory, {\n useFactory: instanceCachingFactory((dependencyContainer) => {\n return new ChildExecutionScopeFactory(dependencyContainer.resolve(CoreTokens.ActivationIdFactory));\n }),\n });\n }\n container.registerSingleton(EngineExecutionLimitsPolicyFactory, EngineExecutionLimitsPolicyFactory);\n container.registerSingleton(NodeInstanceFactoryFactory, NodeInstanceFactoryFactory);\n container.registerSingleton(DefaultAsyncSleeper, DefaultAsyncSleeper);\n container.registerSingleton(InProcessRetryRunnerFactory, InProcessRetryRunnerFactory);\n container.registerSingleton(NodeExecutorFactory, NodeExecutorFactory);\n container.registerSingleton(InlineDrivingSchedulerFactory, InlineDrivingSchedulerFactory);\n container.registerSingleton(RunIntentServiceFactory, RunIntentServiceFactory);\n container.registerSingleton(EngineWorkflowRunnerServiceFactory, EngineWorkflowRunnerServiceFactory);\n container.registerSingleton(\n WorkflowRepositoryWebhookTriggerMatcherFactory,\n WorkflowRepositoryWebhookTriggerMatcherFactory,\n );\n }\n\n private registerExecutionLimitsPolicy(\n container: DependencyContainer,\n options: EngineRuntimeRegistrationOptions | undefined,\n ): void {\n if (container.isRegistered(CoreTokens.EngineExecutionLimitsPolicy, true)) {\n return;\n }\n container.register(CoreTokens.EngineExecutionLimitsPolicy, {\n useFactory: instanceCachingFactory((dependencyContainer) => {\n const fromResolver = options?.resolveEngineExecutionLimits?.();\n const merged = fromResolver ?? options?.engineExecutionLimits;\n return dependencyContainer.resolve(EngineExecutionLimitsPolicyFactory).create(merged);\n }),\n });\n }\n\n private ensureWorkflowNodeInstanceFactory(container: DependencyContainer): void {\n if (container.isRegistered(CoreTokens.WorkflowNodeInstanceFactory, true)) {\n return;\n }\n container.register(CoreTokens.WorkflowNodeInstanceFactory, {\n useFactory: instanceCachingFactory((dependencyContainer) => {\n return dependencyContainer\n .resolve(NodeInstanceFactoryFactory)\n .create(dependencyContainer.resolve(CoreTokens.NodeResolver));\n }),\n });\n }\n\n private ensureNodeExecutor(container: DependencyContainer): void {\n if (container.isRegistered(NodeExecutor, true)) {\n return;\n }\n container.register(NodeExecutor, {\n useFactory: instanceCachingFactory((dependencyContainer) => {\n const retryRunner = dependencyContainer\n .resolve(InProcessRetryRunnerFactory)\n .create(dependencyContainer.resolve(DefaultAsyncSleeper));\n return dependencyContainer\n .resolve(NodeExecutorFactory)\n .create(\n dependencyContainer.resolve(CoreTokens.WorkflowNodeInstanceFactory),\n retryRunner,\n dependencyContainer.resolve(RunnableOutputBehaviorResolver),\n );\n }),\n });\n }\n\n private registerDefaultActivationScheduler(container: DependencyContainer): void {\n if (container.isRegistered(CoreTokens.NodeActivationScheduler, true)) {\n return;\n }\n container.register(InlineDrivingScheduler, {\n useFactory: instanceCachingFactory((dependencyContainer) => {\n return dependencyContainer\n .resolve(InlineDrivingSchedulerFactory)\n .create(dependencyContainer.resolve(NodeExecutor));\n }),\n });\n container.register(CoreTokens.NodeActivationScheduler, {\n useFactory: instanceCachingFactory((dependencyContainer) => {\n return dependencyContainer.resolve(InlineDrivingScheduler);\n }),\n });\n }\n\n private registerEngine(container: DependencyContainer, options: EngineRuntimeRegistrationOptions | undefined): void {\n container.registerSingleton(EngineFactory, EngineFactory);\n const matcherProvider = this.resolveMatcherProvider(options);\n container.register(Engine, {\n useFactory: instanceCachingFactory((dependencyContainer) => {\n const liveWorkflowRepository = dependencyContainer.resolve(CoreTokens.LiveWorkflowRepository);\n const nodeResolver = dependencyContainer.resolve(CoreTokens.NodeResolver);\n const tokenRegistryLike = dependencyContainer.resolve(CoreTokens.PersistedWorkflowTokenRegistry);\n const workflowActivationPolicy = dependencyContainer.resolve(CoreTokens.WorkflowActivationPolicy);\n const webhookTriggerMatcher = matcherProvider.createMatcher(dependencyContainer);\n const workflowNodeInstanceFactory = dependencyContainer.resolve(CoreTokens.WorkflowNodeInstanceFactory);\n const triggerRuntimeDiagnostics = options?.triggerRuntimeDiagnosticsProvider?.create(dependencyContainer);\n return dependencyContainer.resolve(EngineFactory).create({\n credentialSessions: dependencyContainer.resolve(CoreTokens.CredentialSessionService),\n liveWorkflowRepository,\n workflowRepository: dependencyContainer.resolve(CoreTokens.WorkflowRepository),\n workflowActivationPolicy,\n nodeResolver,\n triggerSetupStateRepository: dependencyContainer.resolve(CoreTokens.TriggerSetupStateRepository),\n webhookTriggerMatcher,\n runIdFactory: dependencyContainer.resolve(CoreTokens.RunIdFactory),\n activationIdFactory: dependencyContainer.resolve(CoreTokens.ActivationIdFactory),\n workflowExecutionRepository: dependencyContainer.resolve(CoreTokens.WorkflowExecutionRepository),\n activationScheduler: dependencyContainer.resolve(CoreTokens.NodeActivationScheduler),\n runDataFactory: dependencyContainer.resolve(CoreTokens.RunDataFactory),\n executionContextFactory: dependencyContainer.resolve(CoreTokens.ExecutionContextFactory),\n nodeExecutor: dependencyContainer.resolve(NodeExecutor),\n eventBus: dependencyContainer.resolve(CoreTokens.RunEventBus),\n tokenRegistry: tokenRegistryLike,\n workflowNodeInstanceFactory,\n executionLimitsPolicy: dependencyContainer.resolve(CoreTokens.EngineExecutionLimitsPolicy),\n workflowPolicyRuntimeDefaults: options?.workflowPolicyRuntimeDefaults,\n triggerRuntimeDiagnostics,\n });\n }),\n });\n }\n\n private registerIntentServices(container: DependencyContainer): void {\n container.register(RunIntentService, {\n useFactory: instanceCachingFactory((dependencyContainer) => {\n return dependencyContainer\n .resolve(RunIntentServiceFactory)\n .create(dependencyContainer.resolve(Engine), dependencyContainer.resolve(CoreTokens.WorkflowRepository));\n }),\n });\n container.register(CoreTokens.WorkflowRunnerService, {\n useFactory: instanceCachingFactory((dependencyContainer) => {\n return dependencyContainer\n .resolve(EngineWorkflowRunnerServiceFactory)\n .create(dependencyContainer.resolve(Engine), dependencyContainer.resolve(CoreTokens.WorkflowRepository));\n }),\n });\n }\n\n private resolveMatcherProvider(options: EngineRuntimeRegistrationOptions | undefined): WebhookTriggerMatcherProvider {\n if (options?.webhookTriggerMatcherProvider) {\n return options.webhookTriggerMatcherProvider;\n }\n return {\n createMatcher: (container) =>\n container\n .resolve(WorkflowRepositoryWebhookTriggerMatcherFactory)\n .create(\n container.resolve(CoreTokens.WorkflowRepository),\n container.resolve(CoreTokens.WorkflowActivationPolicy),\n options?.webhookTriggerRoutingDiagnostics,\n ),\n };\n }\n}\n"],"mappings":"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"}
|
|
@@ -1,4 +1,4 @@
|
|
|
1
|
-
import {
|
|
1
|
+
import { D as NodeInstanceFactoryFactory, F as RunnableOutputBehaviorResolver, I as NodeOutputNormalizer, L as ItemExprResolver, M as NodeExecutorFactory, N as NodeExecutor, P as InProcessRetryRunnerFactory, Yt as instanceCachingFactory, c as EngineFactory, en as CoreTokens, i as RunIntentService, l as Engine, o as EngineWorkflowRunnerServiceFactory, pt as ChildExecutionScopeFactory, q as RunFinishedAtFactory, r as RunIntentServiceFactory, t as WorkflowRepositoryWebhookTriggerMatcherFactory, ut as DefaultAsyncSleeper, v as ENGINE_EXECUTION_LIMITS_DEFAULTS, x as InlineDrivingScheduler, y as EngineExecutionLimitsPolicy } from "./runtime-BGNbRnqs.js";
|
|
2
2
|
|
|
3
3
|
//#region src/policies/executionLimits/EngineExecutionLimitsPolicyFactory.ts
|
|
4
4
|
/**
|
|
@@ -99,6 +99,236 @@ var InMemoryWorkflowExecutionRepository = class {
|
|
|
99
99
|
}
|
|
100
100
|
};
|
|
101
101
|
|
|
102
|
+
//#endregion
|
|
103
|
+
//#region src/orchestration/AbortControllerFactory.ts
|
|
104
|
+
/**
|
|
105
|
+
* Mints fresh {@link AbortController}s. Injected (rather than direct `new`) to honor the
|
|
106
|
+
* codebase's no-direct-construction rule and to give tests a seam for substituting a fake.
|
|
107
|
+
*/
|
|
108
|
+
var AbortControllerFactory = class {
|
|
109
|
+
create() {
|
|
110
|
+
return new AbortController();
|
|
111
|
+
}
|
|
112
|
+
};
|
|
113
|
+
|
|
114
|
+
//#endregion
|
|
115
|
+
//#region src/orchestration/TestSuiteOrchestrator.ts
|
|
116
|
+
const DEFAULT_CONCURRENCY = 4;
|
|
117
|
+
/**
|
|
118
|
+
* Drives a {@link TestTriggerNodeConfig.generateItems} iterable into one workflow run per item,
|
|
119
|
+
* with bounded concurrency. Pure engine logic — no persistence, no HTTP, no UI. Hosts adapt by
|
|
120
|
+
* subscribing to {@link RunEventBus} and writing rows on `testSuite*` / `testCase*` / `nodeCompleted`.
|
|
121
|
+
*
|
|
122
|
+
* Cancellation: the supplied `AbortSignal` aborts the source iterable (so credentialed pulls bail)
|
|
123
|
+
* and stops scheduling further cases. In-flight cases are awaited; engine-level cancellation of
|
|
124
|
+
* an already-dispatched run is not yet wired (Phase 2).
|
|
125
|
+
*/
|
|
126
|
+
var TestSuiteOrchestrator = class {
|
|
127
|
+
constructor(engine, testSuiteRunIdFactory, credentialResolverFactory, abortControllerFactory, eventBus, currentDate = () => /* @__PURE__ */ new Date()) {
|
|
128
|
+
this.engine = engine;
|
|
129
|
+
this.testSuiteRunIdFactory = testSuiteRunIdFactory;
|
|
130
|
+
this.credentialResolverFactory = credentialResolverFactory;
|
|
131
|
+
this.abortControllerFactory = abortControllerFactory;
|
|
132
|
+
this.eventBus = eventBus;
|
|
133
|
+
this.currentDate = currentDate;
|
|
134
|
+
}
|
|
135
|
+
async runSuite(args) {
|
|
136
|
+
const triggerNodeId = args.triggerNodeId;
|
|
137
|
+
const definition = args.workflow.nodes.find((n) => n.id === triggerNodeId);
|
|
138
|
+
if (!definition) throw new Error(`Unknown trigger nodeId: ${triggerNodeId}`);
|
|
139
|
+
if (definition.kind !== "trigger") throw new Error(`Node ${triggerNodeId} is not a trigger`);
|
|
140
|
+
const triggerConfig = definition.config;
|
|
141
|
+
if (triggerConfig.triggerKind !== "test") throw new Error(`Node ${triggerNodeId} is not a test trigger (triggerKind="${triggerConfig.triggerKind ?? "live"}")`);
|
|
142
|
+
const testTriggerConfig = triggerConfig;
|
|
143
|
+
if (typeof testTriggerConfig.generateItems !== "function") throw new Error(`Test trigger ${triggerNodeId} is missing a generateItems implementation`);
|
|
144
|
+
const testSuiteRunId = args.testSuiteRunId ?? this.testSuiteRunIdFactory.makeTestSuiteRunId();
|
|
145
|
+
const concurrency = Math.max(1, args.concurrency ?? testTriggerConfig.concurrency ?? DEFAULT_CONCURRENCY);
|
|
146
|
+
const externalSignal = args.signal;
|
|
147
|
+
const internalAbort = this.abortControllerFactory.create();
|
|
148
|
+
const onExternalAbort = () => internalAbort.abort(externalSignal?.reason);
|
|
149
|
+
if (externalSignal) if (externalSignal.aborted) internalAbort.abort(externalSignal.reason);
|
|
150
|
+
else externalSignal.addEventListener("abort", onExternalAbort, { once: true });
|
|
151
|
+
const triggerNodeName = definition.name ?? testTriggerConfig.name;
|
|
152
|
+
await this.publish({
|
|
153
|
+
kind: "testSuiteStarted",
|
|
154
|
+
testSuiteRunId,
|
|
155
|
+
workflowId: args.workflow.id,
|
|
156
|
+
triggerNodeId,
|
|
157
|
+
...triggerNodeName ? { triggerNodeName } : {},
|
|
158
|
+
concurrency,
|
|
159
|
+
at: this.now()
|
|
160
|
+
});
|
|
161
|
+
const setupContext = {
|
|
162
|
+
workflowId: args.workflow.id,
|
|
163
|
+
nodeId: triggerNodeId,
|
|
164
|
+
config: testTriggerConfig,
|
|
165
|
+
testSuiteRunId,
|
|
166
|
+
getCredential: this.credentialResolverFactory.create(args.workflow.id, triggerNodeId, testTriggerConfig),
|
|
167
|
+
signal: internalAbort.signal
|
|
168
|
+
};
|
|
169
|
+
const cases = [];
|
|
170
|
+
let nextIndex = 0;
|
|
171
|
+
let inFlight = 0;
|
|
172
|
+
let waitForSlot;
|
|
173
|
+
let releaseSlot;
|
|
174
|
+
const queue = [];
|
|
175
|
+
let generationError;
|
|
176
|
+
const acquireSlot = async () => {
|
|
177
|
+
while (inFlight >= concurrency) {
|
|
178
|
+
if (!waitForSlot) waitForSlot = new Promise((resolve) => {
|
|
179
|
+
releaseSlot = resolve;
|
|
180
|
+
});
|
|
181
|
+
await waitForSlot;
|
|
182
|
+
}
|
|
183
|
+
inFlight += 1;
|
|
184
|
+
};
|
|
185
|
+
const release = () => {
|
|
186
|
+
inFlight -= 1;
|
|
187
|
+
if (releaseSlot) {
|
|
188
|
+
const fn = releaseSlot;
|
|
189
|
+
releaseSlot = void 0;
|
|
190
|
+
waitForSlot = void 0;
|
|
191
|
+
fn();
|
|
192
|
+
}
|
|
193
|
+
};
|
|
194
|
+
try {
|
|
195
|
+
for await (const item of testTriggerConfig.generateItems(setupContext)) {
|
|
196
|
+
if (internalAbort.signal.aborted) break;
|
|
197
|
+
await acquireSlot();
|
|
198
|
+
if (internalAbort.signal.aborted) {
|
|
199
|
+
release();
|
|
200
|
+
break;
|
|
201
|
+
}
|
|
202
|
+
const testCaseIndex = nextIndex++;
|
|
203
|
+
const testCaseLabel = this.resolveCaseLabel(testTriggerConfig, item);
|
|
204
|
+
queue.push(this.runOneCase({
|
|
205
|
+
workflow: args.workflow,
|
|
206
|
+
triggerNodeId,
|
|
207
|
+
testSuiteRunId,
|
|
208
|
+
testCaseIndex,
|
|
209
|
+
testCaseLabel,
|
|
210
|
+
item
|
|
211
|
+
}).then((outcome) => {
|
|
212
|
+
cases.push(outcome);
|
|
213
|
+
}).finally(release));
|
|
214
|
+
}
|
|
215
|
+
} catch (err) {
|
|
216
|
+
generationError = err instanceof Error ? err : new Error(String(err));
|
|
217
|
+
} finally {
|
|
218
|
+
if (externalSignal) externalSignal.removeEventListener("abort", onExternalAbort);
|
|
219
|
+
}
|
|
220
|
+
await Promise.all(queue);
|
|
221
|
+
cases.sort((a, b) => a.testCaseIndex - b.testCaseIndex);
|
|
222
|
+
const totalCases = cases.length;
|
|
223
|
+
const passedCases = cases.filter((c) => c.status === "succeeded").length;
|
|
224
|
+
const failedCases = cases.filter((c) => c.status === "failed").length;
|
|
225
|
+
const status = this.deriveSuiteStatus({
|
|
226
|
+
generationError,
|
|
227
|
+
cancelled: internalAbort.signal.aborted,
|
|
228
|
+
totalCases,
|
|
229
|
+
passedCases,
|
|
230
|
+
failedCases
|
|
231
|
+
});
|
|
232
|
+
await this.publish({
|
|
233
|
+
kind: "testSuiteFinished",
|
|
234
|
+
testSuiteRunId,
|
|
235
|
+
workflowId: args.workflow.id,
|
|
236
|
+
status,
|
|
237
|
+
totalCases,
|
|
238
|
+
passedCases,
|
|
239
|
+
failedCases,
|
|
240
|
+
at: this.now()
|
|
241
|
+
});
|
|
242
|
+
if (generationError && status === "errored") throw generationError;
|
|
243
|
+
return {
|
|
244
|
+
testSuiteRunId,
|
|
245
|
+
workflowId: args.workflow.id,
|
|
246
|
+
triggerNodeId,
|
|
247
|
+
status,
|
|
248
|
+
totalCases,
|
|
249
|
+
passedCases,
|
|
250
|
+
failedCases,
|
|
251
|
+
cases
|
|
252
|
+
};
|
|
253
|
+
}
|
|
254
|
+
async runOneCase(args) {
|
|
255
|
+
const executionOptions = { testContext: {
|
|
256
|
+
testSuiteRunId: args.testSuiteRunId,
|
|
257
|
+
testCaseIndex: args.testCaseIndex,
|
|
258
|
+
...args.testCaseLabel !== void 0 ? { testCaseLabel: args.testCaseLabel } : {}
|
|
259
|
+
} };
|
|
260
|
+
const initial = await this.engine.runWorkflow(args.workflow, args.triggerNodeId, [args.item], void 0, executionOptions);
|
|
261
|
+
const runId = initial.runId;
|
|
262
|
+
await this.publish({
|
|
263
|
+
kind: "testCaseStarted",
|
|
264
|
+
testSuiteRunId: args.testSuiteRunId,
|
|
265
|
+
testCaseIndex: args.testCaseIndex,
|
|
266
|
+
runId,
|
|
267
|
+
workflowId: args.workflow.id,
|
|
268
|
+
at: this.now(),
|
|
269
|
+
...args.testCaseLabel !== void 0 ? { testCaseLabel: args.testCaseLabel } : {}
|
|
270
|
+
});
|
|
271
|
+
let terminal;
|
|
272
|
+
if (initial.status === "completed" || initial.status === "failed") terminal = initial;
|
|
273
|
+
else terminal = await this.engine.waitForCompletion(runId);
|
|
274
|
+
const status = terminal.status === "completed" ? "succeeded" : "failed";
|
|
275
|
+
await this.publish({
|
|
276
|
+
kind: "testCaseCompleted",
|
|
277
|
+
testSuiteRunId: args.testSuiteRunId,
|
|
278
|
+
testCaseIndex: args.testCaseIndex,
|
|
279
|
+
runId,
|
|
280
|
+
workflowId: args.workflow.id,
|
|
281
|
+
status,
|
|
282
|
+
at: this.now()
|
|
283
|
+
});
|
|
284
|
+
return {
|
|
285
|
+
testCaseIndex: args.testCaseIndex,
|
|
286
|
+
runId,
|
|
287
|
+
status
|
|
288
|
+
};
|
|
289
|
+
}
|
|
290
|
+
deriveSuiteStatus(args) {
|
|
291
|
+
if (args.generationError && args.totalCases === 0) return "errored";
|
|
292
|
+
if (args.cancelled) return "cancelled";
|
|
293
|
+
if (args.generationError) return "errored";
|
|
294
|
+
if (args.totalCases === 0) return "succeeded";
|
|
295
|
+
if (args.failedCases === 0) return "succeeded";
|
|
296
|
+
if (args.passedCases === 0) return "failed";
|
|
297
|
+
return "partial";
|
|
298
|
+
}
|
|
299
|
+
now() {
|
|
300
|
+
return this.currentDate().toISOString();
|
|
301
|
+
}
|
|
302
|
+
/** Defensive label resolver — author-supplied callbacks throw / return non-strings; we tolerate both. */
|
|
303
|
+
resolveCaseLabel(config, item) {
|
|
304
|
+
if (typeof config.caseLabel !== "function") return void 0;
|
|
305
|
+
try {
|
|
306
|
+
const result = config.caseLabel(item);
|
|
307
|
+
if (typeof result !== "string") return void 0;
|
|
308
|
+
const trimmed = result.trim();
|
|
309
|
+
return trimmed.length === 0 ? void 0 : trimmed;
|
|
310
|
+
} catch {
|
|
311
|
+
return;
|
|
312
|
+
}
|
|
313
|
+
}
|
|
314
|
+
async publish(event) {
|
|
315
|
+
if (!this.eventBus) return;
|
|
316
|
+
await this.eventBus.publish(event);
|
|
317
|
+
}
|
|
318
|
+
};
|
|
319
|
+
|
|
320
|
+
//#endregion
|
|
321
|
+
//#region src/orchestration/TestSuiteRunIdFactory.ts
|
|
322
|
+
/**
|
|
323
|
+
* Mints unique TestSuiteRun identifiers. Separated from {@link import("../types").RunIdFactory}
|
|
324
|
+
* so suite ids and per-case workflow run ids never alias.
|
|
325
|
+
*/
|
|
326
|
+
var TestSuiteRunIdFactory = class {
|
|
327
|
+
makeTestSuiteRunId() {
|
|
328
|
+
return `tsr_${Date.now().toString(36)}_${Math.random().toString(36).slice(2, 10)}`;
|
|
329
|
+
}
|
|
330
|
+
};
|
|
331
|
+
|
|
102
332
|
//#endregion
|
|
103
333
|
//#region src/scheduler/InlineDrivingSchedulerFactory.ts
|
|
104
334
|
var InlineDrivingSchedulerFactory = class {
|
|
@@ -128,6 +358,9 @@ var EngineRuntimeRegistrar = class {
|
|
|
128
358
|
if (!container.isRegistered(ItemExprResolver, true)) container.registerSingleton(ItemExprResolver, ItemExprResolver);
|
|
129
359
|
if (!container.isRegistered(NodeOutputNormalizer, true)) container.registerSingleton(NodeOutputNormalizer, NodeOutputNormalizer);
|
|
130
360
|
if (!container.isRegistered(RunnableOutputBehaviorResolver, true)) container.registerSingleton(RunnableOutputBehaviorResolver, RunnableOutputBehaviorResolver);
|
|
361
|
+
if (!container.isRegistered(ChildExecutionScopeFactory, true)) container.register(ChildExecutionScopeFactory, { useFactory: instanceCachingFactory((dependencyContainer) => {
|
|
362
|
+
return new ChildExecutionScopeFactory(dependencyContainer.resolve(CoreTokens.ActivationIdFactory));
|
|
363
|
+
}) });
|
|
131
364
|
container.registerSingleton(EngineExecutionLimitsPolicyFactory, EngineExecutionLimitsPolicyFactory);
|
|
132
365
|
container.registerSingleton(NodeInstanceFactoryFactory, NodeInstanceFactoryFactory);
|
|
133
366
|
container.registerSingleton(DefaultAsyncSleeper, DefaultAsyncSleeper);
|
|
@@ -217,5 +450,5 @@ var EngineRuntimeRegistrar = class {
|
|
|
217
450
|
};
|
|
218
451
|
|
|
219
452
|
//#endregion
|
|
220
|
-
export {
|
|
221
|
-
//# sourceMappingURL=bootstrap-
|
|
453
|
+
export { InMemoryWorkflowExecutionRepository as a, AbortControllerFactory as i, TestSuiteRunIdFactory as n, RunSummaryMapper as o, TestSuiteOrchestrator as r, EngineExecutionLimitsPolicyFactory as s, EngineRuntimeRegistrar as t };
|
|
454
|
+
//# sourceMappingURL=bootstrap-D3r505ko.js.map
|