@devramps/sdk-typescript 0.1.2 → 0.1.4

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.
@@ -1,13 +1,19 @@
1
- import type { ZodType, z } from "zod";
1
+ import type { ZodType } from "zod";
2
2
  import type { StepLogger } from "../logging/step-logger";
3
3
  import type { PrepareOutput } from "../output/step-output";
4
+ import type { ZodStandardJSONSchemaPayload } from "zod/v4/core";
4
5
  export type StepKind = "simple" | "polling";
5
6
  export interface StepMetadata<S extends ZodType = ZodType> {
6
- name: string;
7
- type: string;
7
+ name?: string;
8
+ shortDescription?: string;
9
+ longDescription?: string;
10
+ yamlExample?: string;
8
11
  schema: S;
9
12
  stepKind: StepKind;
10
13
  requiresApproval: boolean;
14
+ stepType: string;
15
+ jsonSchema: ZodStandardJSONSchemaPayload<S>;
16
+ documentationUrl?: string;
11
17
  }
12
18
  /**
13
19
  * Base class for all steps.
@@ -32,10 +38,3 @@ export declare abstract class BaseStep<TParams = unknown> {
32
38
  */
33
39
  prepare(_params: TParams): Promise<PrepareOutput>;
34
40
  }
35
- /**
36
- * Type representing a class constructor that produces a BaseStep instance
37
- * with the getMetadata method available.
38
- */
39
- export type StepClass<S extends ZodType = ZodType> = new (...args: any[]) => BaseStep<z.infer<S>> & {
40
- getMetadata(): StepMetadata<S>;
41
- };
@@ -35,4 +35,4 @@ class BaseStep {
35
35
  }
36
36
  }
37
37
  exports.BaseStep = BaseStep;
38
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYmFzZS1zdGVwLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vc3JjL2Jhc2UvYmFzZS1zdGVwLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7OztBQUVBLHdEQUFvRDtBQUVwRCx1REFBb0Q7QUFZcEQ7OztHQUdHO0FBQ0gsTUFBc0IsUUFBUTtJQUNsQixNQUFNLEdBQWUsSUFBSSx3QkFBVSxFQUFFLENBQUM7SUFFaEQ7OztPQUdHO0lBQ0gsVUFBVSxDQUFDLE1BQWtCO1FBQzNCLElBQUksQ0FBQyxNQUFNLEdBQUcsTUFBTSxDQUFDO0lBQ3ZCLENBQUM7SUFFRDs7O09BR0c7SUFDSCxXQUFXO1FBQ1QsTUFBTSxJQUFJLEtBQUssQ0FDYix5RUFBeUUsQ0FDMUUsQ0FBQztJQUNKLENBQUM7SUFFRDs7OztPQUlHO0lBQ0gsS0FBSyxDQUFDLE9BQU8sQ0FBQyxPQUFnQjtRQUM1Qiw4REFBOEQ7UUFDOUQsaUVBQWlFO1FBQ2pFLE9BQU8seUJBQVcsQ0FBQyxNQUFNLENBQ3ZCLHNDQUFzQyxFQUN0Qyx5QkFBeUIsQ0FDMUIsQ0FBQztJQUNKLENBQUM7Q0FDRjtBQWxDRCw0QkFrQ0MiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgdHlwZSB7IFpvZFR5cGUsIHogfSBmcm9tIFwiem9kXCI7XG5pbXBvcnQgdHlwZSB7IFN0ZXBMb2dnZXJ9IGZyb20gXCIuLi9sb2dnaW5nL3N0ZXAtbG9nZ2VyXCI7XG5pbXBvcnQgeyBOb09wTG9nZ2VyIH0gZnJvbSBcIi4uL2xvZ2dpbmcvc3RlcC1sb2dnZXJcIjtcbmltcG9ydCB0eXBlIHsgUHJlcGFyZU91dHB1dH0gZnJvbSBcIi4uL291dHB1dC9zdGVwLW91dHB1dFwiO1xuaW1wb3J0IHsgU3RlcE91dHB1dHMgfSBmcm9tIFwiLi4vb3V0cHV0L3N0ZXAtb3V0cHV0XCI7XG5cbmV4cG9ydCB0eXBlIFN0ZXBLaW5kID0gXCJzaW1wbGVcIiB8IFwicG9sbGluZ1wiO1xuXG5leHBvcnQgaW50ZXJmYWNlIFN0ZXBNZXRhZGF0YTxTIGV4dGVuZHMgWm9kVHlwZSA9IFpvZFR5cGU+IHtcbiAgbmFtZTogc3RyaW5nO1xuICB0eXBlOiBzdHJpbmc7XG4gIHNjaGVtYTogUztcbiAgc3RlcEtpbmQ6IFN0ZXBLaW5kO1xuICByZXF1aXJlc0FwcHJvdmFsOiBib29sZWFuO1xufVxuXG4vKipcbiAqIEJhc2UgY2xhc3MgZm9yIGFsbCBzdGVwcy5cbiAqIFByb3ZpZGVzIGxvZ2dpbmcgYW5kIGNvbW1vbiBmdW5jdGlvbmFsaXR5LlxuICovXG5leHBvcnQgYWJzdHJhY3QgY2xhc3MgQmFzZVN0ZXA8VFBhcmFtcyA9IHVua25vd24+IHtcbiAgcHJvdGVjdGVkIGxvZ2dlcjogU3RlcExvZ2dlciA9IG5ldyBOb09wTG9nZ2VyKCk7XG5cbiAgLyoqXG4gICAqIENhbGxlZCBieSB0aGUgcmVnaXN0cnkgdG8gaW5qZWN0IHRoZSBsb2dnZXIgYmVmb3JlIGV4ZWN1dGlvbi5cbiAgICogQGludGVybmFsXG4gICAqL1xuICBfc2V0TG9nZ2VyKGxvZ2dlcjogU3RlcExvZ2dlcik6IHZvaWQge1xuICAgIHRoaXMubG9nZ2VyID0gbG9nZ2VyO1xuICB9XG5cbiAgLyoqXG4gICAqIFJldHVybnMgbWV0YWRhdGEgYWJvdXQgdGhpcyBzdGVwLlxuICAgKiBJbXBsZW1lbnRlZCBieSB0aGUgQFN0ZXAgZGVjb3JhdG9yLlxuICAgKi9cbiAgZ2V0TWV0YWRhdGEoKTogU3RlcE1ldGFkYXRhIHtcbiAgICB0aHJvdyBuZXcgRXJyb3IoXG4gICAgICBcImdldE1ldGFkYXRhIG5vdCBpbXBsZW1lbnRlZC4gRGlkIHlvdSBmb3JnZXQgdG8gYWRkIHRoZSBAU3RlcCBkZWNvcmF0b3I/XCJcbiAgICApO1xuICB9XG5cbiAgLyoqXG4gICAqIE9wdGlvbmFsIHByZXBhcmUgbWV0aG9kIGZvciBhcHByb3ZhbCBmbG93LlxuICAgKiBPdmVycmlkZSB0aGlzIHRvIHJlcXVpcmUgYXBwcm92YWwgYmVmb3JlIGV4ZWN1dGlvbi5cbiAgICogSWYgbm90IG92ZXJyaWRkZW4sIHRoZSBzdGVwIHdpbGwgbm90IHJlcXVpcmUgYXBwcm92YWwuXG4gICAqL1xuICBhc3luYyBwcmVwYXJlKF9wYXJhbXM6IFRQYXJhbXMpOiBQcm9taXNlPFByZXBhcmVPdXRwdXQ+IHtcbiAgICAvLyBEZWZhdWx0OiBubyBhcHByb3ZhbCByZXF1aXJlZCAtIHRoaXMgc2hvdWxkIG5ldmVyIGJlIGNhbGxlZFxuICAgIC8vIFRoZSByZWdpc3RyeSBjaGVja3MgaWYgcHJlcGFyZSBpcyBvdmVycmlkZGVuIGJlZm9yZSBjYWxsaW5nIGl0XG4gICAgcmV0dXJuIFN0ZXBPdXRwdXRzLmZhaWxlZChcbiAgICAgIFwicHJlcGFyZSgpIGNhbGxlZCBidXQgbm90IGltcGxlbWVudGVkXCIsXG4gICAgICBcIlBSRVBBUkVfTk9UX0lNUExFTUVOVEVEXCJcbiAgICApO1xuICB9XG59XG5cbi8qKlxuICogVHlwZSByZXByZXNlbnRpbmcgYSBjbGFzcyBjb25zdHJ1Y3RvciB0aGF0IHByb2R1Y2VzIGEgQmFzZVN0ZXAgaW5zdGFuY2VcbiAqIHdpdGggdGhlIGdldE1ldGFkYXRhIG1ldGhvZCBhdmFpbGFibGUuXG4gKi9cbmV4cG9ydCB0eXBlIFN0ZXBDbGFzczxTIGV4dGVuZHMgWm9kVHlwZSA9IFpvZFR5cGU+ID0gbmV3IChcbiAgLi4uYXJnczogYW55W11cbikgPT4gQmFzZVN0ZXA8ei5pbmZlcjxTPj4gJiB7XG4gIGdldE1ldGFkYXRhKCk6IFN0ZXBNZXRhZGF0YTxTPjtcbn07XG4iXX0=
38
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYmFzZS1zdGVwLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vc3JjL2Jhc2UvYmFzZS1zdGVwLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7OztBQUVBLHdEQUFvRDtBQUVwRCx1REFBb0Q7QUFrQnBEOzs7R0FHRztBQUNILE1BQXNCLFFBQVE7SUFDbEIsTUFBTSxHQUFlLElBQUksd0JBQVUsRUFBRSxDQUFDO0lBRWhEOzs7T0FHRztJQUNILFVBQVUsQ0FBQyxNQUFrQjtRQUMzQixJQUFJLENBQUMsTUFBTSxHQUFHLE1BQU0sQ0FBQztJQUN2QixDQUFDO0lBRUQ7OztPQUdHO0lBQ0gsV0FBVztRQUNULE1BQU0sSUFBSSxLQUFLLENBQ2IseUVBQXlFLENBQzFFLENBQUM7SUFDSixDQUFDO0lBRUQ7Ozs7T0FJRztJQUNILEtBQUssQ0FBQyxPQUFPLENBQUMsT0FBZ0I7UUFDNUIsOERBQThEO1FBQzlELGlFQUFpRTtRQUNqRSxPQUFPLHlCQUFXLENBQUMsTUFBTSxDQUN2QixzQ0FBc0MsRUFDdEMseUJBQXlCLENBQzFCLENBQUM7SUFDSixDQUFDO0NBQ0Y7QUFsQ0QsNEJBa0NDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHR5cGUgeyBab2RUeXBlIH0gZnJvbSBcInpvZFwiO1xuaW1wb3J0IHR5cGUgeyBTdGVwTG9nZ2VyIH0gZnJvbSBcIi4uL2xvZ2dpbmcvc3RlcC1sb2dnZXJcIjtcbmltcG9ydCB7IE5vT3BMb2dnZXIgfSBmcm9tIFwiLi4vbG9nZ2luZy9zdGVwLWxvZ2dlclwiO1xuaW1wb3J0IHR5cGUgeyBQcmVwYXJlT3V0cHV0IH0gZnJvbSBcIi4uL291dHB1dC9zdGVwLW91dHB1dFwiO1xuaW1wb3J0IHsgU3RlcE91dHB1dHMgfSBmcm9tIFwiLi4vb3V0cHV0L3N0ZXAtb3V0cHV0XCI7XG5pbXBvcnQgdHlwZSB7IFpvZFN0YW5kYXJkSlNPTlNjaGVtYVBheWxvYWQgfSBmcm9tIFwiem9kL3Y0L2NvcmVcIjtcblxuZXhwb3J0IHR5cGUgU3RlcEtpbmQgPSBcInNpbXBsZVwiIHwgXCJwb2xsaW5nXCI7XG5cbmV4cG9ydCBpbnRlcmZhY2UgU3RlcE1ldGFkYXRhPFMgZXh0ZW5kcyBab2RUeXBlID0gWm9kVHlwZT4ge1xuICBuYW1lPzogc3RyaW5nO1xuICBzaG9ydERlc2NyaXB0aW9uPzogc3RyaW5nO1xuICBsb25nRGVzY3JpcHRpb24/OiBzdHJpbmc7XG4gIHlhbWxFeGFtcGxlPzogc3RyaW5nO1xuICBzY2hlbWE6IFM7XG4gIHN0ZXBLaW5kOiBTdGVwS2luZDtcbiAgcmVxdWlyZXNBcHByb3ZhbDogYm9vbGVhbjtcbiAgc3RlcFR5cGU6IHN0cmluZztcbiAganNvblNjaGVtYTogWm9kU3RhbmRhcmRKU09OU2NoZW1hUGF5bG9hZDxTPjtcbiAgZG9jdW1lbnRhdGlvblVybD86IHN0cmluZztcbn1cblxuLyoqXG4gKiBCYXNlIGNsYXNzIGZvciBhbGwgc3RlcHMuXG4gKiBQcm92aWRlcyBsb2dnaW5nIGFuZCBjb21tb24gZnVuY3Rpb25hbGl0eS5cbiAqL1xuZXhwb3J0IGFic3RyYWN0IGNsYXNzIEJhc2VTdGVwPFRQYXJhbXMgPSB1bmtub3duPiB7XG4gIHByb3RlY3RlZCBsb2dnZXI6IFN0ZXBMb2dnZXIgPSBuZXcgTm9PcExvZ2dlcigpO1xuXG4gIC8qKlxuICAgKiBDYWxsZWQgYnkgdGhlIHJlZ2lzdHJ5IHRvIGluamVjdCB0aGUgbG9nZ2VyIGJlZm9yZSBleGVjdXRpb24uXG4gICAqIEBpbnRlcm5hbFxuICAgKi9cbiAgX3NldExvZ2dlcihsb2dnZXI6IFN0ZXBMb2dnZXIpOiB2b2lkIHtcbiAgICB0aGlzLmxvZ2dlciA9IGxvZ2dlcjtcbiAgfVxuXG4gIC8qKlxuICAgKiBSZXR1cm5zIG1ldGFkYXRhIGFib3V0IHRoaXMgc3RlcC5cbiAgICogSW1wbGVtZW50ZWQgYnkgdGhlIEBTdGVwIGRlY29yYXRvci5cbiAgICovXG4gIGdldE1ldGFkYXRhKCk6IFN0ZXBNZXRhZGF0YSB7XG4gICAgdGhyb3cgbmV3IEVycm9yKFxuICAgICAgXCJnZXRNZXRhZGF0YSBub3QgaW1wbGVtZW50ZWQuIERpZCB5b3UgZm9yZ2V0IHRvIGFkZCB0aGUgQFN0ZXAgZGVjb3JhdG9yP1wiXG4gICAgKTtcbiAgfVxuXG4gIC8qKlxuICAgKiBPcHRpb25hbCBwcmVwYXJlIG1ldGhvZCBmb3IgYXBwcm92YWwgZmxvdy5cbiAgICogT3ZlcnJpZGUgdGhpcyB0byByZXF1aXJlIGFwcHJvdmFsIGJlZm9yZSBleGVjdXRpb24uXG4gICAqIElmIG5vdCBvdmVycmlkZGVuLCB0aGUgc3RlcCB3aWxsIG5vdCByZXF1aXJlIGFwcHJvdmFsLlxuICAgKi9cbiAgYXN5bmMgcHJlcGFyZShfcGFyYW1zOiBUUGFyYW1zKTogUHJvbWlzZTxQcmVwYXJlT3V0cHV0PiB7XG4gICAgLy8gRGVmYXVsdDogbm8gYXBwcm92YWwgcmVxdWlyZWQgLSB0aGlzIHNob3VsZCBuZXZlciBiZSBjYWxsZWRcbiAgICAvLyBUaGUgcmVnaXN0cnkgY2hlY2tzIGlmIHByZXBhcmUgaXMgb3ZlcnJpZGRlbiBiZWZvcmUgY2FsbGluZyBpdFxuICAgIHJldHVybiBTdGVwT3V0cHV0cy5mYWlsZWQoXG4gICAgICBcInByZXBhcmUoKSBjYWxsZWQgYnV0IG5vdCBpbXBsZW1lbnRlZFwiLFxuICAgICAgXCJQUkVQQVJFX05PVF9JTVBMRU1FTlRFRFwiXG4gICAgKTtcbiAgfVxufVxuXG4iXX0=
@@ -10,6 +10,8 @@ import type { ApprovalContext, PollOutput, StepOutput, TriggerOutput } from "../
10
10
  *
11
11
  * Optionally override `prepare` to require approval before triggering.
12
12
  *
13
+ * Steps can accept constructor parameters for dependency injection.
14
+ *
13
15
  * @typeParam TParams - The input parameters type (validated by schema)
14
16
  * @typeParam TPollingState - The polling state type passed between poll calls
15
17
  *
@@ -20,25 +22,36 @@ import type { ApprovalContext, PollOutput, StepOutput, TriggerOutput } from "../
20
22
  *
21
23
  * @Step({ name: "Long Job", type: "long-job", schema: jobSchema })
22
24
  * class LongJobStep extends PollingStep<JobParams, JobPollingState> {
25
+ * constructor(private jobService: JobService) {
26
+ * super();
27
+ * }
28
+ *
23
29
  * async trigger(params: JobParams): Promise<TriggerOutput<JobPollingState>> {
24
- * const jobId = await startJob(params);
30
+ * const jobId = await this.jobService.start(params);
25
31
  * return StepOutputs.triggered({ jobId, startedAt: Date.now() });
26
32
  * }
27
33
  *
28
34
  * async poll(params: JobParams, state: JobPollingState): Promise<PollOutput<JobPollingState>> {
29
- * const status = await checkJob(state.jobId);
35
+ * const status = await this.jobService.checkStatus(state.jobId);
30
36
  * if (status === "running") {
31
37
  * return StepOutputs.pollAgain(state, 5000);
32
38
  * }
33
39
  * return StepOutputs.success({ result: status });
34
40
  * }
35
41
  * }
42
+ *
43
+ * // Register with dependency injected
44
+ * StepRegistry.run([new LongJobStep(jobService)]);
36
45
  * ```
37
46
  *
38
47
  * @example With approval
39
48
  * ```typescript
40
49
  * @Step({ name: "Dangerous Job", type: "dangerous-job", schema: dangerousSchema })
41
50
  * class DangerousJobStep extends PollingStep<DangerousParams, DangerousPollingState> {
51
+ * constructor(private jobService: JobService) {
52
+ * super();
53
+ * }
54
+ *
42
55
  * async prepare(params: DangerousParams): Promise<PrepareOutput> {
43
56
  * return StepOutputs.approvalRequired({
44
57
  * message: `Run dangerous job on ${params.target}?`,
@@ -46,12 +59,16 @@ import type { ApprovalContext, PollOutput, StepOutput, TriggerOutput } from "../
46
59
  * }
47
60
  *
48
61
  * async trigger(params: DangerousParams, approval: ApprovalContext): Promise<TriggerOutput<DangerousPollingState>> {
49
- * const jobId = await startDangerousJob(params);
62
+ * const jobId = await this.jobService.startDangerous(params);
50
63
  * return StepOutputs.triggered({ jobId });
51
64
  * }
52
65
  *
53
66
  * async poll(params: DangerousParams, state: DangerousPollingState): Promise<PollOutput<DangerousPollingState>> {
54
- * // ... check status
67
+ * const status = await this.jobService.checkStatus(state.jobId);
68
+ * if (status === "running") {
69
+ * return StepOutputs.pollAgain(state, 5000);
70
+ * }
71
+ * return StepOutputs.success();
55
72
  * }
56
73
  * }
57
74
  * ```
@@ -11,6 +11,8 @@ const base_step_1 = require("./base-step");
11
11
  *
12
12
  * Optionally override `prepare` to require approval before triggering.
13
13
  *
14
+ * Steps can accept constructor parameters for dependency injection.
15
+ *
14
16
  * @typeParam TParams - The input parameters type (validated by schema)
15
17
  * @typeParam TPollingState - The polling state type passed between poll calls
16
18
  *
@@ -21,25 +23,36 @@ const base_step_1 = require("./base-step");
21
23
  *
22
24
  * @Step({ name: "Long Job", type: "long-job", schema: jobSchema })
23
25
  * class LongJobStep extends PollingStep<JobParams, JobPollingState> {
26
+ * constructor(private jobService: JobService) {
27
+ * super();
28
+ * }
29
+ *
24
30
  * async trigger(params: JobParams): Promise<TriggerOutput<JobPollingState>> {
25
- * const jobId = await startJob(params);
31
+ * const jobId = await this.jobService.start(params);
26
32
  * return StepOutputs.triggered({ jobId, startedAt: Date.now() });
27
33
  * }
28
34
  *
29
35
  * async poll(params: JobParams, state: JobPollingState): Promise<PollOutput<JobPollingState>> {
30
- * const status = await checkJob(state.jobId);
36
+ * const status = await this.jobService.checkStatus(state.jobId);
31
37
  * if (status === "running") {
32
38
  * return StepOutputs.pollAgain(state, 5000);
33
39
  * }
34
40
  * return StepOutputs.success({ result: status });
35
41
  * }
36
42
  * }
43
+ *
44
+ * // Register with dependency injected
45
+ * StepRegistry.run([new LongJobStep(jobService)]);
37
46
  * ```
38
47
  *
39
48
  * @example With approval
40
49
  * ```typescript
41
50
  * @Step({ name: "Dangerous Job", type: "dangerous-job", schema: dangerousSchema })
42
51
  * class DangerousJobStep extends PollingStep<DangerousParams, DangerousPollingState> {
52
+ * constructor(private jobService: JobService) {
53
+ * super();
54
+ * }
55
+ *
43
56
  * async prepare(params: DangerousParams): Promise<PrepareOutput> {
44
57
  * return StepOutputs.approvalRequired({
45
58
  * message: `Run dangerous job on ${params.target}?`,
@@ -47,12 +60,16 @@ const base_step_1 = require("./base-step");
47
60
  * }
48
61
  *
49
62
  * async trigger(params: DangerousParams, approval: ApprovalContext): Promise<TriggerOutput<DangerousPollingState>> {
50
- * const jobId = await startDangerousJob(params);
63
+ * const jobId = await this.jobService.startDangerous(params);
51
64
  * return StepOutputs.triggered({ jobId });
52
65
  * }
53
66
  *
54
67
  * async poll(params: DangerousParams, state: DangerousPollingState): Promise<PollOutput<DangerousPollingState>> {
55
- * // ... check status
68
+ * const status = await this.jobService.checkStatus(state.jobId);
69
+ * if (status === "running") {
70
+ * return StepOutputs.pollAgain(state, 5000);
71
+ * }
72
+ * return StepOutputs.success();
56
73
  * }
57
74
  * }
58
75
  * ```
@@ -77,4 +94,4 @@ class PollingStep extends base_step_1.BaseStep {
77
94
  }
78
95
  }
79
96
  exports.PollingStep = PollingStep;
80
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicG9sbGluZy1zdGVwLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vc3JjL2Jhc2UvcG9sbGluZy1zdGVwLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7OztBQUNBLDJDQUF1QztBQVF2Qzs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7O0dBc0RHO0FBQ0gsTUFBc0IsV0FHcEIsU0FBUSxvQkFBaUI7SUF1QnpCOzs7T0FHRztJQUNILEtBQUssQ0FBQyxjQUFjLENBQ2xCLE1BQWUsRUFDZixRQUEwQjtRQUUxQixPQUFPLElBQUksQ0FBQyxPQUFPLENBQUMsTUFBTSxFQUFFLFFBQVEsQ0FBQyxDQUFDO0lBQ3hDLENBQUM7SUFFRDs7O09BR0c7SUFDSCxLQUFLLENBQUMsV0FBVyxDQUNmLE1BQWUsRUFDZixZQUEyQjtRQUUzQixPQUFPLElBQUksQ0FBQyxJQUFJLENBQUMsTUFBTSxFQUFFLFlBQVksQ0FBQyxDQUFDO0lBQ3pDLENBQUM7SUFFRCxXQUFXO1FBQ1QsTUFBTSxJQUFJLEtBQUssQ0FDYix5RUFBeUUsQ0FDMUUsQ0FBQztJQUNKLENBQUM7Q0FDRjtBQXJERCxrQ0FxREMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgdHlwZSB7IFN0ZXBNZXRhZGF0YSB9IGZyb20gXCIuL2Jhc2Utc3RlcFwiO1xuaW1wb3J0IHsgQmFzZVN0ZXAgfSBmcm9tIFwiLi9iYXNlLXN0ZXBcIjtcbmltcG9ydCB0eXBlIHtcbiAgQXBwcm92YWxDb250ZXh0LFxuICBQb2xsT3V0cHV0LFxuICBTdGVwT3V0cHV0LFxuICBUcmlnZ2VyT3V0cHV0LFxufSBmcm9tIFwiLi4vb3V0cHV0L3N0ZXAtb3V0cHV0XCI7XG5cbi8qKlxuICogQSBwb2xsaW5nIHN0ZXAgZm9yIGxvbmctcnVubmluZyBvcGVyYXRpb25zIHRoYXQgbmVlZCBzdGF0dXMgY2hlY2tzLlxuICpcbiAqIFVzZXJzIGV4dGVuZCB0aGlzIGNsYXNzIGFuZCBpbXBsZW1lbnQ6XG4gKiAtIGB0cmlnZ2VyKHBhcmFtcylgIC0gU3RhcnQgdGhlIG9wZXJhdGlvbiwgcmV0dXJuIGluaXRpYWwgcG9sbGluZyBzdGF0ZVxuICogLSBgcG9sbChwYXJhbXMsIHBvbGxpbmdTdGF0ZSlgIC0gQ2hlY2sgc3RhdHVzLCByZXR1cm4gUE9MTF9BR0FJTiwgU1VDQ0VTUywgb3IgRkFJTEVEXG4gKlxuICogT3B0aW9uYWxseSBvdmVycmlkZSBgcHJlcGFyZWAgdG8gcmVxdWlyZSBhcHByb3ZhbCBiZWZvcmUgdHJpZ2dlcmluZy5cbiAqXG4gKiBAdHlwZVBhcmFtIFRQYXJhbXMgLSBUaGUgaW5wdXQgcGFyYW1ldGVycyB0eXBlICh2YWxpZGF0ZWQgYnkgc2NoZW1hKVxuICogQHR5cGVQYXJhbSBUUG9sbGluZ1N0YXRlIC0gVGhlIHBvbGxpbmcgc3RhdGUgdHlwZSBwYXNzZWQgYmV0d2VlbiBwb2xsIGNhbGxzXG4gKlxuICogQGV4YW1wbGVcbiAqIGBgYHR5cGVzY3JpcHRcbiAqIHR5cGUgSm9iUGFyYW1zID0geyB0YXJnZXQ6IHN0cmluZyB9O1xuICogdHlwZSBKb2JQb2xsaW5nU3RhdGUgPSB7IGpvYklkOiBzdHJpbmc7IHN0YXJ0ZWRBdDogbnVtYmVyIH07XG4gKlxuICogQFN0ZXAoeyBuYW1lOiBcIkxvbmcgSm9iXCIsIHR5cGU6IFwibG9uZy1qb2JcIiwgc2NoZW1hOiBqb2JTY2hlbWEgfSlcbiAqIGNsYXNzIExvbmdKb2JTdGVwIGV4dGVuZHMgUG9sbGluZ1N0ZXA8Sm9iUGFyYW1zLCBKb2JQb2xsaW5nU3RhdGU+IHtcbiAqICAgYXN5bmMgdHJpZ2dlcihwYXJhbXM6IEpvYlBhcmFtcyk6IFByb21pc2U8VHJpZ2dlck91dHB1dDxKb2JQb2xsaW5nU3RhdGU+PiB7XG4gKiAgICAgY29uc3Qgam9iSWQgPSBhd2FpdCBzdGFydEpvYihwYXJhbXMpO1xuICogICAgIHJldHVybiBTdGVwT3V0cHV0cy50cmlnZ2VyZWQoeyBqb2JJZCwgc3RhcnRlZEF0OiBEYXRlLm5vdygpIH0pO1xuICogICB9XG4gKlxuICogICBhc3luYyBwb2xsKHBhcmFtczogSm9iUGFyYW1zLCBzdGF0ZTogSm9iUG9sbGluZ1N0YXRlKTogUHJvbWlzZTxQb2xsT3V0cHV0PEpvYlBvbGxpbmdTdGF0ZT4+IHtcbiAqICAgICBjb25zdCBzdGF0dXMgPSBhd2FpdCBjaGVja0pvYihzdGF0ZS5qb2JJZCk7XG4gKiAgICAgaWYgKHN0YXR1cyA9PT0gXCJydW5uaW5nXCIpIHtcbiAqICAgICAgIHJldHVybiBTdGVwT3V0cHV0cy5wb2xsQWdhaW4oc3RhdGUsIDUwMDApO1xuICogICAgIH1cbiAqICAgICByZXR1cm4gU3RlcE91dHB1dHMuc3VjY2Vzcyh7IHJlc3VsdDogc3RhdHVzIH0pO1xuICogICB9XG4gKiB9XG4gKiBgYGBcbiAqXG4gKiBAZXhhbXBsZSBXaXRoIGFwcHJvdmFsXG4gKiBgYGB0eXBlc2NyaXB0XG4gKiBAU3RlcCh7IG5hbWU6IFwiRGFuZ2Vyb3VzIEpvYlwiLCB0eXBlOiBcImRhbmdlcm91cy1qb2JcIiwgc2NoZW1hOiBkYW5nZXJvdXNTY2hlbWEgfSlcbiAqIGNsYXNzIERhbmdlcm91c0pvYlN0ZXAgZXh0ZW5kcyBQb2xsaW5nU3RlcDxEYW5nZXJvdXNQYXJhbXMsIERhbmdlcm91c1BvbGxpbmdTdGF0ZT4ge1xuICogICBhc3luYyBwcmVwYXJlKHBhcmFtczogRGFuZ2Vyb3VzUGFyYW1zKTogUHJvbWlzZTxQcmVwYXJlT3V0cHV0PiB7XG4gKiAgICAgcmV0dXJuIFN0ZXBPdXRwdXRzLmFwcHJvdmFsUmVxdWlyZWQoe1xuICogICAgICAgbWVzc2FnZTogYFJ1biBkYW5nZXJvdXMgam9iIG9uICR7cGFyYW1zLnRhcmdldH0/YCxcbiAqICAgICB9KTtcbiAqICAgfVxuICpcbiAqICAgYXN5bmMgdHJpZ2dlcihwYXJhbXM6IERhbmdlcm91c1BhcmFtcywgYXBwcm92YWw6IEFwcHJvdmFsQ29udGV4dCk6IFByb21pc2U8VHJpZ2dlck91dHB1dDxEYW5nZXJvdXNQb2xsaW5nU3RhdGU+PiB7XG4gKiAgICAgY29uc3Qgam9iSWQgPSBhd2FpdCBzdGFydERhbmdlcm91c0pvYihwYXJhbXMpO1xuICogICAgIHJldHVybiBTdGVwT3V0cHV0cy50cmlnZ2VyZWQoeyBqb2JJZCB9KTtcbiAqICAgfVxuICpcbiAqICAgYXN5bmMgcG9sbChwYXJhbXM6IERhbmdlcm91c1BhcmFtcywgc3RhdGU6IERhbmdlcm91c1BvbGxpbmdTdGF0ZSk6IFByb21pc2U8UG9sbE91dHB1dDxEYW5nZXJvdXNQb2xsaW5nU3RhdGU+PiB7XG4gKiAgICAgLy8gLi4uIGNoZWNrIHN0YXR1c1xuICogICB9XG4gKiB9XG4gKiBgYGBcbiAqL1xuZXhwb3J0IGFic3RyYWN0IGNsYXNzIFBvbGxpbmdTdGVwPFxuICBUUGFyYW1zLFxuICBUUG9sbGluZ1N0YXRlIGV4dGVuZHMgUmVjb3JkPHN0cmluZywgdW5rbm93bj4gPSBSZWNvcmQ8c3RyaW5nLCB1bmtub3duPlxuPiBleHRlbmRzIEJhc2VTdGVwPFRQYXJhbXM+IHtcbiAgLyoqXG4gICAqIFN0YXJ0IHRoZSBsb25nLXJ1bm5pbmcgb3BlcmF0aW9uLlxuICAgKiBAcGFyYW0gcGFyYW1zIC0gVGhlIHZhbGlkYXRlZCBpbnB1dCBwYXJhbWV0ZXJzXG4gICAqIEBwYXJhbSBhcHByb3ZhbCAtIEFwcHJvdmFsIGNvbnRleHQgaWYgdGhpcyBzdGVwIHJlcXVpcmVzIGFwcHJvdmFsIChoYXMgcHJlcGFyZSBtZXRob2QpXG4gICAqIEByZXR1cm5zIFRSSUdHRVJFRCB3aXRoIGluaXRpYWwgcG9sbGluZyBzdGF0ZSwgb3IgRkFJTEVEXG4gICAqL1xuICBhYnN0cmFjdCB0cmlnZ2VyKFxuICAgIHBhcmFtczogVFBhcmFtcyxcbiAgICBhcHByb3ZhbD86IEFwcHJvdmFsQ29udGV4dFxuICApOiBQcm9taXNlPFRyaWdnZXJPdXRwdXQ8VFBvbGxpbmdTdGF0ZT4+O1xuXG4gIC8qKlxuICAgKiBDaGVjayB0aGUgc3RhdHVzIG9mIHRoZSBvcGVyYXRpb24uXG4gICAqIEBwYXJhbSBwYXJhbXMgLSBUaGUgb3JpZ2luYWwgaW5wdXQgcGFyYW1ldGVyc1xuICAgKiBAcGFyYW0gcG9sbGluZ1N0YXRlIC0gU3RhdGUgZnJvbSBwcmV2aW91cyB0cmlnZ2VyKCkgb3IgcG9sbCgpIGNhbGxcbiAgICogQHJldHVybnMgUE9MTF9BR0FJTiB3aXRoIHVwZGF0ZWQgc3RhdGUsIFNVQ0NFU1MsIG9yIEZBSUxFRFxuICAgKi9cbiAgYWJzdHJhY3QgcG9sbChcbiAgICBwYXJhbXM6IFRQYXJhbXMsXG4gICAgcG9sbGluZ1N0YXRlOiBUUG9sbGluZ1N0YXRlXG4gICk6IFByb21pc2U8UG9sbE91dHB1dDxUUG9sbGluZ1N0YXRlPj47XG5cbiAgLyoqXG4gICAqIENhbGxlZCBieSB0aGUgcmVnaXN0cnkgdG8gZXhlY3V0ZSB0aGUgdHJpZ2dlciBwaGFzZS5cbiAgICogQGludGVybmFsXG4gICAqL1xuICBhc3luYyBleGVjdXRlVHJpZ2dlcihcbiAgICBwYXJhbXM6IFRQYXJhbXMsXG4gICAgYXBwcm92YWw/OiBBcHByb3ZhbENvbnRleHRcbiAgKTogUHJvbWlzZTxTdGVwT3V0cHV0PiB7XG4gICAgcmV0dXJuIHRoaXMudHJpZ2dlcihwYXJhbXMsIGFwcHJvdmFsKTtcbiAgfVxuXG4gIC8qKlxuICAgKiBDYWxsZWQgYnkgdGhlIHJlZ2lzdHJ5IHRvIGV4ZWN1dGUgdGhlIHBvbGwgcGhhc2UuXG4gICAqIEBpbnRlcm5hbFxuICAgKi9cbiAgYXN5bmMgZXhlY3V0ZVBvbGwoXG4gICAgcGFyYW1zOiBUUGFyYW1zLFxuICAgIHBvbGxpbmdTdGF0ZTogVFBvbGxpbmdTdGF0ZVxuICApOiBQcm9taXNlPFN0ZXBPdXRwdXQ+IHtcbiAgICByZXR1cm4gdGhpcy5wb2xsKHBhcmFtcywgcG9sbGluZ1N0YXRlKTtcbiAgfVxuXG4gIGdldE1ldGFkYXRhKCk6IFN0ZXBNZXRhZGF0YSB7XG4gICAgdGhyb3cgbmV3IEVycm9yKFxuICAgICAgXCJnZXRNZXRhZGF0YSBub3QgaW1wbGVtZW50ZWQuIERpZCB5b3UgZm9yZ2V0IHRvIGFkZCB0aGUgQFN0ZXAgZGVjb3JhdG9yP1wiXG4gICAgKTtcbiAgfVxufVxuIl19
97
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicG9sbGluZy1zdGVwLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vc3JjL2Jhc2UvcG9sbGluZy1zdGVwLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7OztBQUNBLDJDQUF1QztBQVF2Qzs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7R0F1RUc7QUFDSCxNQUFzQixXQUdwQixTQUFRLG9CQUFpQjtJQXVCekI7OztPQUdHO0lBQ0gsS0FBSyxDQUFDLGNBQWMsQ0FDbEIsTUFBZSxFQUNmLFFBQTBCO1FBRTFCLE9BQU8sSUFBSSxDQUFDLE9BQU8sQ0FBQyxNQUFNLEVBQUUsUUFBUSxDQUFDLENBQUM7SUFDeEMsQ0FBQztJQUVEOzs7T0FHRztJQUNILEtBQUssQ0FBQyxXQUFXLENBQ2YsTUFBZSxFQUNmLFlBQTJCO1FBRTNCLE9BQU8sSUFBSSxDQUFDLElBQUksQ0FBQyxNQUFNLEVBQUUsWUFBWSxDQUFDLENBQUM7SUFDekMsQ0FBQztJQUVELFdBQVc7UUFDVCxNQUFNLElBQUksS0FBSyxDQUNiLHlFQUF5RSxDQUMxRSxDQUFDO0lBQ0osQ0FBQztDQUNGO0FBckRELGtDQXFEQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB0eXBlIHsgU3RlcE1ldGFkYXRhIH0gZnJvbSBcIi4vYmFzZS1zdGVwXCI7XG5pbXBvcnQgeyBCYXNlU3RlcCB9IGZyb20gXCIuL2Jhc2Utc3RlcFwiO1xuaW1wb3J0IHR5cGUge1xuICBBcHByb3ZhbENvbnRleHQsXG4gIFBvbGxPdXRwdXQsXG4gIFN0ZXBPdXRwdXQsXG4gIFRyaWdnZXJPdXRwdXQsXG59IGZyb20gXCIuLi9vdXRwdXQvc3RlcC1vdXRwdXRcIjtcblxuLyoqXG4gKiBBIHBvbGxpbmcgc3RlcCBmb3IgbG9uZy1ydW5uaW5nIG9wZXJhdGlvbnMgdGhhdCBuZWVkIHN0YXR1cyBjaGVja3MuXG4gKlxuICogVXNlcnMgZXh0ZW5kIHRoaXMgY2xhc3MgYW5kIGltcGxlbWVudDpcbiAqIC0gYHRyaWdnZXIocGFyYW1zKWAgLSBTdGFydCB0aGUgb3BlcmF0aW9uLCByZXR1cm4gaW5pdGlhbCBwb2xsaW5nIHN0YXRlXG4gKiAtIGBwb2xsKHBhcmFtcywgcG9sbGluZ1N0YXRlKWAgLSBDaGVjayBzdGF0dXMsIHJldHVybiBQT0xMX0FHQUlOLCBTVUNDRVNTLCBvciBGQUlMRURcbiAqXG4gKiBPcHRpb25hbGx5IG92ZXJyaWRlIGBwcmVwYXJlYCB0byByZXF1aXJlIGFwcHJvdmFsIGJlZm9yZSB0cmlnZ2VyaW5nLlxuICpcbiAqIFN0ZXBzIGNhbiBhY2NlcHQgY29uc3RydWN0b3IgcGFyYW1ldGVycyBmb3IgZGVwZW5kZW5jeSBpbmplY3Rpb24uXG4gKlxuICogQHR5cGVQYXJhbSBUUGFyYW1zIC0gVGhlIGlucHV0IHBhcmFtZXRlcnMgdHlwZSAodmFsaWRhdGVkIGJ5IHNjaGVtYSlcbiAqIEB0eXBlUGFyYW0gVFBvbGxpbmdTdGF0ZSAtIFRoZSBwb2xsaW5nIHN0YXRlIHR5cGUgcGFzc2VkIGJldHdlZW4gcG9sbCBjYWxsc1xuICpcbiAqIEBleGFtcGxlXG4gKiBgYGB0eXBlc2NyaXB0XG4gKiB0eXBlIEpvYlBhcmFtcyA9IHsgdGFyZ2V0OiBzdHJpbmcgfTtcbiAqIHR5cGUgSm9iUG9sbGluZ1N0YXRlID0geyBqb2JJZDogc3RyaW5nOyBzdGFydGVkQXQ6IG51bWJlciB9O1xuICpcbiAqIEBTdGVwKHsgbmFtZTogXCJMb25nIEpvYlwiLCB0eXBlOiBcImxvbmctam9iXCIsIHNjaGVtYTogam9iU2NoZW1hIH0pXG4gKiBjbGFzcyBMb25nSm9iU3RlcCBleHRlbmRzIFBvbGxpbmdTdGVwPEpvYlBhcmFtcywgSm9iUG9sbGluZ1N0YXRlPiB7XG4gKiAgIGNvbnN0cnVjdG9yKHByaXZhdGUgam9iU2VydmljZTogSm9iU2VydmljZSkge1xuICogICAgIHN1cGVyKCk7XG4gKiAgIH1cbiAqXG4gKiAgIGFzeW5jIHRyaWdnZXIocGFyYW1zOiBKb2JQYXJhbXMpOiBQcm9taXNlPFRyaWdnZXJPdXRwdXQ8Sm9iUG9sbGluZ1N0YXRlPj4ge1xuICogICAgIGNvbnN0IGpvYklkID0gYXdhaXQgdGhpcy5qb2JTZXJ2aWNlLnN0YXJ0KHBhcmFtcyk7XG4gKiAgICAgcmV0dXJuIFN0ZXBPdXRwdXRzLnRyaWdnZXJlZCh7IGpvYklkLCBzdGFydGVkQXQ6IERhdGUubm93KCkgfSk7XG4gKiAgIH1cbiAqXG4gKiAgIGFzeW5jIHBvbGwocGFyYW1zOiBKb2JQYXJhbXMsIHN0YXRlOiBKb2JQb2xsaW5nU3RhdGUpOiBQcm9taXNlPFBvbGxPdXRwdXQ8Sm9iUG9sbGluZ1N0YXRlPj4ge1xuICogICAgIGNvbnN0IHN0YXR1cyA9IGF3YWl0IHRoaXMuam9iU2VydmljZS5jaGVja1N0YXR1cyhzdGF0ZS5qb2JJZCk7XG4gKiAgICAgaWYgKHN0YXR1cyA9PT0gXCJydW5uaW5nXCIpIHtcbiAqICAgICAgIHJldHVybiBTdGVwT3V0cHV0cy5wb2xsQWdhaW4oc3RhdGUsIDUwMDApO1xuICogICAgIH1cbiAqICAgICByZXR1cm4gU3RlcE91dHB1dHMuc3VjY2Vzcyh7IHJlc3VsdDogc3RhdHVzIH0pO1xuICogICB9XG4gKiB9XG4gKlxuICogLy8gUmVnaXN0ZXIgd2l0aCBkZXBlbmRlbmN5IGluamVjdGVkXG4gKiBTdGVwUmVnaXN0cnkucnVuKFtuZXcgTG9uZ0pvYlN0ZXAoam9iU2VydmljZSldKTtcbiAqIGBgYFxuICpcbiAqIEBleGFtcGxlIFdpdGggYXBwcm92YWxcbiAqIGBgYHR5cGVzY3JpcHRcbiAqIEBTdGVwKHsgbmFtZTogXCJEYW5nZXJvdXMgSm9iXCIsIHR5cGU6IFwiZGFuZ2Vyb3VzLWpvYlwiLCBzY2hlbWE6IGRhbmdlcm91c1NjaGVtYSB9KVxuICogY2xhc3MgRGFuZ2Vyb3VzSm9iU3RlcCBleHRlbmRzIFBvbGxpbmdTdGVwPERhbmdlcm91c1BhcmFtcywgRGFuZ2Vyb3VzUG9sbGluZ1N0YXRlPiB7XG4gKiAgIGNvbnN0cnVjdG9yKHByaXZhdGUgam9iU2VydmljZTogSm9iU2VydmljZSkge1xuICogICAgIHN1cGVyKCk7XG4gKiAgIH1cbiAqXG4gKiAgIGFzeW5jIHByZXBhcmUocGFyYW1zOiBEYW5nZXJvdXNQYXJhbXMpOiBQcm9taXNlPFByZXBhcmVPdXRwdXQ+IHtcbiAqICAgICByZXR1cm4gU3RlcE91dHB1dHMuYXBwcm92YWxSZXF1aXJlZCh7XG4gKiAgICAgICBtZXNzYWdlOiBgUnVuIGRhbmdlcm91cyBqb2Igb24gJHtwYXJhbXMudGFyZ2V0fT9gLFxuICogICAgIH0pO1xuICogICB9XG4gKlxuICogICBhc3luYyB0cmlnZ2VyKHBhcmFtczogRGFuZ2Vyb3VzUGFyYW1zLCBhcHByb3ZhbDogQXBwcm92YWxDb250ZXh0KTogUHJvbWlzZTxUcmlnZ2VyT3V0cHV0PERhbmdlcm91c1BvbGxpbmdTdGF0ZT4+IHtcbiAqICAgICBjb25zdCBqb2JJZCA9IGF3YWl0IHRoaXMuam9iU2VydmljZS5zdGFydERhbmdlcm91cyhwYXJhbXMpO1xuICogICAgIHJldHVybiBTdGVwT3V0cHV0cy50cmlnZ2VyZWQoeyBqb2JJZCB9KTtcbiAqICAgfVxuICpcbiAqICAgYXN5bmMgcG9sbChwYXJhbXM6IERhbmdlcm91c1BhcmFtcywgc3RhdGU6IERhbmdlcm91c1BvbGxpbmdTdGF0ZSk6IFByb21pc2U8UG9sbE91dHB1dDxEYW5nZXJvdXNQb2xsaW5nU3RhdGU+PiB7XG4gKiAgICAgY29uc3Qgc3RhdHVzID0gYXdhaXQgdGhpcy5qb2JTZXJ2aWNlLmNoZWNrU3RhdHVzKHN0YXRlLmpvYklkKTtcbiAqICAgICBpZiAoc3RhdHVzID09PSBcInJ1bm5pbmdcIikge1xuICogICAgICAgcmV0dXJuIFN0ZXBPdXRwdXRzLnBvbGxBZ2FpbihzdGF0ZSwgNTAwMCk7XG4gKiAgICAgfVxuICogICAgIHJldHVybiBTdGVwT3V0cHV0cy5zdWNjZXNzKCk7XG4gKiAgIH1cbiAqIH1cbiAqIGBgYFxuICovXG5leHBvcnQgYWJzdHJhY3QgY2xhc3MgUG9sbGluZ1N0ZXA8XG4gIFRQYXJhbXMsXG4gIFRQb2xsaW5nU3RhdGUgZXh0ZW5kcyBSZWNvcmQ8c3RyaW5nLCB1bmtub3duPiA9IFJlY29yZDxzdHJpbmcsIHVua25vd24+XG4+IGV4dGVuZHMgQmFzZVN0ZXA8VFBhcmFtcz4ge1xuICAvKipcbiAgICogU3RhcnQgdGhlIGxvbmctcnVubmluZyBvcGVyYXRpb24uXG4gICAqIEBwYXJhbSBwYXJhbXMgLSBUaGUgdmFsaWRhdGVkIGlucHV0IHBhcmFtZXRlcnNcbiAgICogQHBhcmFtIGFwcHJvdmFsIC0gQXBwcm92YWwgY29udGV4dCBpZiB0aGlzIHN0ZXAgcmVxdWlyZXMgYXBwcm92YWwgKGhhcyBwcmVwYXJlIG1ldGhvZClcbiAgICogQHJldHVybnMgVFJJR0dFUkVEIHdpdGggaW5pdGlhbCBwb2xsaW5nIHN0YXRlLCBvciBGQUlMRURcbiAgICovXG4gIGFic3RyYWN0IHRyaWdnZXIoXG4gICAgcGFyYW1zOiBUUGFyYW1zLFxuICAgIGFwcHJvdmFsPzogQXBwcm92YWxDb250ZXh0XG4gICk6IFByb21pc2U8VHJpZ2dlck91dHB1dDxUUG9sbGluZ1N0YXRlPj47XG5cbiAgLyoqXG4gICAqIENoZWNrIHRoZSBzdGF0dXMgb2YgdGhlIG9wZXJhdGlvbi5cbiAgICogQHBhcmFtIHBhcmFtcyAtIFRoZSBvcmlnaW5hbCBpbnB1dCBwYXJhbWV0ZXJzXG4gICAqIEBwYXJhbSBwb2xsaW5nU3RhdGUgLSBTdGF0ZSBmcm9tIHByZXZpb3VzIHRyaWdnZXIoKSBvciBwb2xsKCkgY2FsbFxuICAgKiBAcmV0dXJucyBQT0xMX0FHQUlOIHdpdGggdXBkYXRlZCBzdGF0ZSwgU1VDQ0VTUywgb3IgRkFJTEVEXG4gICAqL1xuICBhYnN0cmFjdCBwb2xsKFxuICAgIHBhcmFtczogVFBhcmFtcyxcbiAgICBwb2xsaW5nU3RhdGU6IFRQb2xsaW5nU3RhdGVcbiAgKTogUHJvbWlzZTxQb2xsT3V0cHV0PFRQb2xsaW5nU3RhdGU+PjtcblxuICAvKipcbiAgICogQ2FsbGVkIGJ5IHRoZSByZWdpc3RyeSB0byBleGVjdXRlIHRoZSB0cmlnZ2VyIHBoYXNlLlxuICAgKiBAaW50ZXJuYWxcbiAgICovXG4gIGFzeW5jIGV4ZWN1dGVUcmlnZ2VyKFxuICAgIHBhcmFtczogVFBhcmFtcyxcbiAgICBhcHByb3ZhbD86IEFwcHJvdmFsQ29udGV4dFxuICApOiBQcm9taXNlPFN0ZXBPdXRwdXQ+IHtcbiAgICByZXR1cm4gdGhpcy50cmlnZ2VyKHBhcmFtcywgYXBwcm92YWwpO1xuICB9XG5cbiAgLyoqXG4gICAqIENhbGxlZCBieSB0aGUgcmVnaXN0cnkgdG8gZXhlY3V0ZSB0aGUgcG9sbCBwaGFzZS5cbiAgICogQGludGVybmFsXG4gICAqL1xuICBhc3luYyBleGVjdXRlUG9sbChcbiAgICBwYXJhbXM6IFRQYXJhbXMsXG4gICAgcG9sbGluZ1N0YXRlOiBUUG9sbGluZ1N0YXRlXG4gICk6IFByb21pc2U8U3RlcE91dHB1dD4ge1xuICAgIHJldHVybiB0aGlzLnBvbGwocGFyYW1zLCBwb2xsaW5nU3RhdGUpO1xuICB9XG5cbiAgZ2V0TWV0YWRhdGEoKTogU3RlcE1ldGFkYXRhIHtcbiAgICB0aHJvdyBuZXcgRXJyb3IoXG4gICAgICBcImdldE1ldGFkYXRhIG5vdCBpbXBsZW1lbnRlZC4gRGlkIHlvdSBmb3JnZXQgdG8gYWRkIHRoZSBAU3RlcCBkZWNvcmF0b3I/XCJcbiAgICApO1xuICB9XG59XG4iXX0=
@@ -7,21 +7,35 @@ import type { ApprovalContext, RunOutput, StepOutput } from "../output/step-outp
7
7
  * Users extend this class and implement the `run` method.
8
8
  * Optionally override `prepare` to require approval before execution.
9
9
  *
10
+ * Steps can accept constructor parameters for dependency injection.
11
+ *
10
12
  * @example
11
13
  * ```typescript
12
14
  * @Step({ name: "Deploy", type: "deploy", schema: deploySchema })
13
15
  * class DeployStep extends SimpleStep<DeployParams> {
16
+ * constructor(private deployService: DeployService) {
17
+ * super();
18
+ * }
19
+ *
14
20
  * async run(params: DeployParams): Promise<RunOutput> {
15
21
  * this.logger.info("Deploying", { target: params.target });
16
- * return StepOutputs.success({ deploymentId: "123" });
22
+ * const result = await this.deployService.deploy(params.target);
23
+ * return StepOutputs.success({ deploymentId: result.id });
17
24
  * }
18
25
  * }
26
+ *
27
+ * // Register with dependency injected
28
+ * StepRegistry.run([new DeployStep(deployService)]);
19
29
  * ```
20
30
  *
21
31
  * @example With approval
22
32
  * ```typescript
23
33
  * @Step({ name: "Delete User", type: "delete-user", schema: deleteUserSchema })
24
34
  * class DeleteUserStep extends SimpleStep<DeleteUserParams> {
35
+ * constructor(private userService: UserService) {
36
+ * super();
37
+ * }
38
+ *
25
39
  * async prepare(params: DeleteUserParams): Promise<PrepareOutput> {
26
40
  * return StepOutputs.approvalRequired({
27
41
  * message: `Delete user ${params.userId}?`,
@@ -30,6 +44,7 @@ import type { ApprovalContext, RunOutput, StepOutput } from "../output/step-outp
30
44
  *
31
45
  * async run(params: DeleteUserParams, approval: ApprovalContext): Promise<RunOutput> {
32
46
  * this.logger.info("Deleting user", { approvedBy: approval.approverId });
47
+ * await this.userService.delete(params.userId);
33
48
  * return StepOutputs.success();
34
49
  * }
35
50
  * }
@@ -8,21 +8,35 @@ const base_step_1 = require("./base-step");
8
8
  * Users extend this class and implement the `run` method.
9
9
  * Optionally override `prepare` to require approval before execution.
10
10
  *
11
+ * Steps can accept constructor parameters for dependency injection.
12
+ *
11
13
  * @example
12
14
  * ```typescript
13
15
  * @Step({ name: "Deploy", type: "deploy", schema: deploySchema })
14
16
  * class DeployStep extends SimpleStep<DeployParams> {
17
+ * constructor(private deployService: DeployService) {
18
+ * super();
19
+ * }
20
+ *
15
21
  * async run(params: DeployParams): Promise<RunOutput> {
16
22
  * this.logger.info("Deploying", { target: params.target });
17
- * return StepOutputs.success({ deploymentId: "123" });
23
+ * const result = await this.deployService.deploy(params.target);
24
+ * return StepOutputs.success({ deploymentId: result.id });
18
25
  * }
19
26
  * }
27
+ *
28
+ * // Register with dependency injected
29
+ * StepRegistry.run([new DeployStep(deployService)]);
20
30
  * ```
21
31
  *
22
32
  * @example With approval
23
33
  * ```typescript
24
34
  * @Step({ name: "Delete User", type: "delete-user", schema: deleteUserSchema })
25
35
  * class DeleteUserStep extends SimpleStep<DeleteUserParams> {
36
+ * constructor(private userService: UserService) {
37
+ * super();
38
+ * }
39
+ *
26
40
  * async prepare(params: DeleteUserParams): Promise<PrepareOutput> {
27
41
  * return StepOutputs.approvalRequired({
28
42
  * message: `Delete user ${params.userId}?`,
@@ -31,6 +45,7 @@ const base_step_1 = require("./base-step");
31
45
  *
32
46
  * async run(params: DeleteUserParams, approval: ApprovalContext): Promise<RunOutput> {
33
47
  * this.logger.info("Deleting user", { approvedBy: approval.approverId });
48
+ * await this.userService.delete(params.userId);
34
49
  * return StepOutputs.success();
35
50
  * }
36
51
  * }
@@ -49,4 +64,4 @@ class SimpleStep extends base_step_1.BaseStep {
49
64
  }
50
65
  }
51
66
  exports.SimpleStep = SimpleStep;
52
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic2ltcGxlLXN0ZXAuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvYmFzZS9zaW1wbGUtc3RlcC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7QUFDQSwyQ0FBdUM7QUFHdkM7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7OztHQWlDRztBQUNILE1BQXNCLFVBQW9CLFNBQVEsb0JBQWlCO0lBUWpFOzs7T0FHRztJQUNILEtBQUssQ0FBQyxPQUFPLENBQUMsTUFBZSxFQUFFLFFBQTBCO1FBQ3ZELE9BQU8sSUFBSSxDQUFDLEdBQUcsQ0FBQyxNQUFNLEVBQUUsUUFBUSxDQUFDLENBQUM7SUFDcEMsQ0FBQztJQUVELFdBQVc7UUFDVCxNQUFNLElBQUksS0FBSyxDQUNiLHlFQUF5RSxDQUMxRSxDQUFDO0lBQ0osQ0FBQztDQUNGO0FBckJELGdDQXFCQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB0eXBlIHsgU3RlcE1ldGFkYXRhIH0gZnJvbSBcIi4vYmFzZS1zdGVwXCI7XG5pbXBvcnQgeyBCYXNlU3RlcCB9IGZyb20gXCIuL2Jhc2Utc3RlcFwiO1xuaW1wb3J0IHR5cGUgeyBBcHByb3ZhbENvbnRleHQsIFJ1bk91dHB1dCwgU3RlcE91dHB1dCB9IGZyb20gXCIuLi9vdXRwdXQvc3RlcC1vdXRwdXRcIjtcblxuLyoqXG4gKiBBIHNpbXBsZSBzdGVwIHRoYXQgcnVucyBvbmNlIHdpdGggdGhlIHByb3ZpZGVkIHBhcmFtcy5cbiAqXG4gKiBVc2VycyBleHRlbmQgdGhpcyBjbGFzcyBhbmQgaW1wbGVtZW50IHRoZSBgcnVuYCBtZXRob2QuXG4gKiBPcHRpb25hbGx5IG92ZXJyaWRlIGBwcmVwYXJlYCB0byByZXF1aXJlIGFwcHJvdmFsIGJlZm9yZSBleGVjdXRpb24uXG4gKlxuICogQGV4YW1wbGVcbiAqIGBgYHR5cGVzY3JpcHRcbiAqIEBTdGVwKHsgbmFtZTogXCJEZXBsb3lcIiwgdHlwZTogXCJkZXBsb3lcIiwgc2NoZW1hOiBkZXBsb3lTY2hlbWEgfSlcbiAqIGNsYXNzIERlcGxveVN0ZXAgZXh0ZW5kcyBTaW1wbGVTdGVwPERlcGxveVBhcmFtcz4ge1xuICogICBhc3luYyBydW4ocGFyYW1zOiBEZXBsb3lQYXJhbXMpOiBQcm9taXNlPFJ1bk91dHB1dD4ge1xuICogICAgIHRoaXMubG9nZ2VyLmluZm8oXCJEZXBsb3lpbmdcIiwgeyB0YXJnZXQ6IHBhcmFtcy50YXJnZXQgfSk7XG4gKiAgICAgcmV0dXJuIFN0ZXBPdXRwdXRzLnN1Y2Nlc3MoeyBkZXBsb3ltZW50SWQ6IFwiMTIzXCIgfSk7XG4gKiAgIH1cbiAqIH1cbiAqIGBgYFxuICpcbiAqIEBleGFtcGxlIFdpdGggYXBwcm92YWxcbiAqIGBgYHR5cGVzY3JpcHRcbiAqIEBTdGVwKHsgbmFtZTogXCJEZWxldGUgVXNlclwiLCB0eXBlOiBcImRlbGV0ZS11c2VyXCIsIHNjaGVtYTogZGVsZXRlVXNlclNjaGVtYSB9KVxuICogY2xhc3MgRGVsZXRlVXNlclN0ZXAgZXh0ZW5kcyBTaW1wbGVTdGVwPERlbGV0ZVVzZXJQYXJhbXM+IHtcbiAqICAgYXN5bmMgcHJlcGFyZShwYXJhbXM6IERlbGV0ZVVzZXJQYXJhbXMpOiBQcm9taXNlPFByZXBhcmVPdXRwdXQ+IHtcbiAqICAgICByZXR1cm4gU3RlcE91dHB1dHMuYXBwcm92YWxSZXF1aXJlZCh7XG4gKiAgICAgICBtZXNzYWdlOiBgRGVsZXRlIHVzZXIgJHtwYXJhbXMudXNlcklkfT9gLFxuICogICAgIH0pO1xuICogICB9XG4gKlxuICogICBhc3luYyBydW4ocGFyYW1zOiBEZWxldGVVc2VyUGFyYW1zLCBhcHByb3ZhbDogQXBwcm92YWxDb250ZXh0KTogUHJvbWlzZTxSdW5PdXRwdXQ+IHtcbiAqICAgICB0aGlzLmxvZ2dlci5pbmZvKFwiRGVsZXRpbmcgdXNlclwiLCB7IGFwcHJvdmVkQnk6IGFwcHJvdmFsLmFwcHJvdmVySWQgfSk7XG4gKiAgICAgcmV0dXJuIFN0ZXBPdXRwdXRzLnN1Y2Nlc3MoKTtcbiAqICAgfVxuICogfVxuICogYGBgXG4gKi9cbmV4cG9ydCBhYnN0cmFjdCBjbGFzcyBTaW1wbGVTdGVwPFRQYXJhbXM+IGV4dGVuZHMgQmFzZVN0ZXA8VFBhcmFtcz4ge1xuICAvKipcbiAgICogRXhlY3V0ZSB0aGUgc3RlcCBsb2dpYy5cbiAgICogQHBhcmFtIHBhcmFtcyAtIFRoZSB2YWxpZGF0ZWQgaW5wdXQgcGFyYW1ldGVyc1xuICAgKiBAcGFyYW0gYXBwcm92YWwgLSBBcHByb3ZhbCBjb250ZXh0IGlmIHRoaXMgc3RlcCByZXF1aXJlcyBhcHByb3ZhbCAoaGFzIHByZXBhcmUgbWV0aG9kKVxuICAgKi9cbiAgYWJzdHJhY3QgcnVuKHBhcmFtczogVFBhcmFtcywgYXBwcm92YWw/OiBBcHByb3ZhbENvbnRleHQpOiBQcm9taXNlPFJ1bk91dHB1dD47XG5cbiAgLyoqXG4gICAqIENhbGxlZCBieSB0aGUgcmVnaXN0cnkgdG8gZXhlY3V0ZSB0aGlzIHN0ZXAuXG4gICAqIEBpbnRlcm5hbFxuICAgKi9cbiAgYXN5bmMgZXhlY3V0ZShwYXJhbXM6IFRQYXJhbXMsIGFwcHJvdmFsPzogQXBwcm92YWxDb250ZXh0KTogUHJvbWlzZTxTdGVwT3V0cHV0PiB7XG4gICAgcmV0dXJuIHRoaXMucnVuKHBhcmFtcywgYXBwcm92YWwpO1xuICB9XG5cbiAgZ2V0TWV0YWRhdGEoKTogU3RlcE1ldGFkYXRhIHtcbiAgICB0aHJvdyBuZXcgRXJyb3IoXG4gICAgICBcImdldE1ldGFkYXRhIG5vdCBpbXBsZW1lbnRlZC4gRGlkIHlvdSBmb3JnZXQgdG8gYWRkIHRoZSBAU3RlcCBkZWNvcmF0b3I/XCJcbiAgICApO1xuICB9XG59XG4iXX0=
67
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic2ltcGxlLXN0ZXAuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvYmFzZS9zaW1wbGUtc3RlcC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7QUFDQSwyQ0FBdUM7QUFHdkM7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7OztHQWdERztBQUNILE1BQXNCLFVBQW9CLFNBQVEsb0JBQWlCO0lBUWpFOzs7T0FHRztJQUNILEtBQUssQ0FBQyxPQUFPLENBQUMsTUFBZSxFQUFFLFFBQTBCO1FBQ3ZELE9BQU8sSUFBSSxDQUFDLEdBQUcsQ0FBQyxNQUFNLEVBQUUsUUFBUSxDQUFDLENBQUM7SUFDcEMsQ0FBQztJQUVELFdBQVc7UUFDVCxNQUFNLElBQUksS0FBSyxDQUNiLHlFQUF5RSxDQUMxRSxDQUFDO0lBQ0osQ0FBQztDQUNGO0FBckJELGdDQXFCQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB0eXBlIHsgU3RlcE1ldGFkYXRhIH0gZnJvbSBcIi4vYmFzZS1zdGVwXCI7XG5pbXBvcnQgeyBCYXNlU3RlcCB9IGZyb20gXCIuL2Jhc2Utc3RlcFwiO1xuaW1wb3J0IHR5cGUgeyBBcHByb3ZhbENvbnRleHQsIFJ1bk91dHB1dCwgU3RlcE91dHB1dCB9IGZyb20gXCIuLi9vdXRwdXQvc3RlcC1vdXRwdXRcIjtcblxuLyoqXG4gKiBBIHNpbXBsZSBzdGVwIHRoYXQgcnVucyBvbmNlIHdpdGggdGhlIHByb3ZpZGVkIHBhcmFtcy5cbiAqXG4gKiBVc2VycyBleHRlbmQgdGhpcyBjbGFzcyBhbmQgaW1wbGVtZW50IHRoZSBgcnVuYCBtZXRob2QuXG4gKiBPcHRpb25hbGx5IG92ZXJyaWRlIGBwcmVwYXJlYCB0byByZXF1aXJlIGFwcHJvdmFsIGJlZm9yZSBleGVjdXRpb24uXG4gKlxuICogU3RlcHMgY2FuIGFjY2VwdCBjb25zdHJ1Y3RvciBwYXJhbWV0ZXJzIGZvciBkZXBlbmRlbmN5IGluamVjdGlvbi5cbiAqXG4gKiBAZXhhbXBsZVxuICogYGBgdHlwZXNjcmlwdFxuICogQFN0ZXAoeyBuYW1lOiBcIkRlcGxveVwiLCB0eXBlOiBcImRlcGxveVwiLCBzY2hlbWE6IGRlcGxveVNjaGVtYSB9KVxuICogY2xhc3MgRGVwbG95U3RlcCBleHRlbmRzIFNpbXBsZVN0ZXA8RGVwbG95UGFyYW1zPiB7XG4gKiAgIGNvbnN0cnVjdG9yKHByaXZhdGUgZGVwbG95U2VydmljZTogRGVwbG95U2VydmljZSkge1xuICogICAgIHN1cGVyKCk7XG4gKiAgIH1cbiAqXG4gKiAgIGFzeW5jIHJ1bihwYXJhbXM6IERlcGxveVBhcmFtcyk6IFByb21pc2U8UnVuT3V0cHV0PiB7XG4gKiAgICAgdGhpcy5sb2dnZXIuaW5mbyhcIkRlcGxveWluZ1wiLCB7IHRhcmdldDogcGFyYW1zLnRhcmdldCB9KTtcbiAqICAgICBjb25zdCByZXN1bHQgPSBhd2FpdCB0aGlzLmRlcGxveVNlcnZpY2UuZGVwbG95KHBhcmFtcy50YXJnZXQpO1xuICogICAgIHJldHVybiBTdGVwT3V0cHV0cy5zdWNjZXNzKHsgZGVwbG95bWVudElkOiByZXN1bHQuaWQgfSk7XG4gKiAgIH1cbiAqIH1cbiAqXG4gKiAvLyBSZWdpc3RlciB3aXRoIGRlcGVuZGVuY3kgaW5qZWN0ZWRcbiAqIFN0ZXBSZWdpc3RyeS5ydW4oW25ldyBEZXBsb3lTdGVwKGRlcGxveVNlcnZpY2UpXSk7XG4gKiBgYGBcbiAqXG4gKiBAZXhhbXBsZSBXaXRoIGFwcHJvdmFsXG4gKiBgYGB0eXBlc2NyaXB0XG4gKiBAU3RlcCh7IG5hbWU6IFwiRGVsZXRlIFVzZXJcIiwgdHlwZTogXCJkZWxldGUtdXNlclwiLCBzY2hlbWE6IGRlbGV0ZVVzZXJTY2hlbWEgfSlcbiAqIGNsYXNzIERlbGV0ZVVzZXJTdGVwIGV4dGVuZHMgU2ltcGxlU3RlcDxEZWxldGVVc2VyUGFyYW1zPiB7XG4gKiAgIGNvbnN0cnVjdG9yKHByaXZhdGUgdXNlclNlcnZpY2U6IFVzZXJTZXJ2aWNlKSB7XG4gKiAgICAgc3VwZXIoKTtcbiAqICAgfVxuICpcbiAqICAgYXN5bmMgcHJlcGFyZShwYXJhbXM6IERlbGV0ZVVzZXJQYXJhbXMpOiBQcm9taXNlPFByZXBhcmVPdXRwdXQ+IHtcbiAqICAgICByZXR1cm4gU3RlcE91dHB1dHMuYXBwcm92YWxSZXF1aXJlZCh7XG4gKiAgICAgICBtZXNzYWdlOiBgRGVsZXRlIHVzZXIgJHtwYXJhbXMudXNlcklkfT9gLFxuICogICAgIH0pO1xuICogICB9XG4gKlxuICogICBhc3luYyBydW4ocGFyYW1zOiBEZWxldGVVc2VyUGFyYW1zLCBhcHByb3ZhbDogQXBwcm92YWxDb250ZXh0KTogUHJvbWlzZTxSdW5PdXRwdXQ+IHtcbiAqICAgICB0aGlzLmxvZ2dlci5pbmZvKFwiRGVsZXRpbmcgdXNlclwiLCB7IGFwcHJvdmVkQnk6IGFwcHJvdmFsLmFwcHJvdmVySWQgfSk7XG4gKiAgICAgYXdhaXQgdGhpcy51c2VyU2VydmljZS5kZWxldGUocGFyYW1zLnVzZXJJZCk7XG4gKiAgICAgcmV0dXJuIFN0ZXBPdXRwdXRzLnN1Y2Nlc3MoKTtcbiAqICAgfVxuICogfVxuICogYGBgXG4gKi9cbmV4cG9ydCBhYnN0cmFjdCBjbGFzcyBTaW1wbGVTdGVwPFRQYXJhbXM+IGV4dGVuZHMgQmFzZVN0ZXA8VFBhcmFtcz4ge1xuICAvKipcbiAgICogRXhlY3V0ZSB0aGUgc3RlcCBsb2dpYy5cbiAgICogQHBhcmFtIHBhcmFtcyAtIFRoZSB2YWxpZGF0ZWQgaW5wdXQgcGFyYW1ldGVyc1xuICAgKiBAcGFyYW0gYXBwcm92YWwgLSBBcHByb3ZhbCBjb250ZXh0IGlmIHRoaXMgc3RlcCByZXF1aXJlcyBhcHByb3ZhbCAoaGFzIHByZXBhcmUgbWV0aG9kKVxuICAgKi9cbiAgYWJzdHJhY3QgcnVuKHBhcmFtczogVFBhcmFtcywgYXBwcm92YWw/OiBBcHByb3ZhbENvbnRleHQpOiBQcm9taXNlPFJ1bk91dHB1dD47XG5cbiAgLyoqXG4gICAqIENhbGxlZCBieSB0aGUgcmVnaXN0cnkgdG8gZXhlY3V0ZSB0aGlzIHN0ZXAuXG4gICAqIEBpbnRlcm5hbFxuICAgKi9cbiAgYXN5bmMgZXhlY3V0ZShwYXJhbXM6IFRQYXJhbXMsIGFwcHJvdmFsPzogQXBwcm92YWxDb250ZXh0KTogUHJvbWlzZTxTdGVwT3V0cHV0PiB7XG4gICAgcmV0dXJuIHRoaXMucnVuKHBhcmFtcywgYXBwcm92YWwpO1xuICB9XG5cbiAgZ2V0TWV0YWRhdGEoKTogU3RlcE1ldGFkYXRhIHtcbiAgICB0aHJvdyBuZXcgRXJyb3IoXG4gICAgICBcImdldE1ldGFkYXRhIG5vdCBpbXBsZW1lbnRlZC4gRGlkIHlvdSBmb3JnZXQgdG8gYWRkIHRoZSBAU3RlcCBkZWNvcmF0b3I/XCJcbiAgICApO1xuICB9XG59XG4iXX0=
@@ -1,8 +1,11 @@
1
- import type { ZodType, z } from "zod";
1
+ import { type ZodType, z } from "zod";
2
2
  import { BaseStep } from "../base/base-step";
3
3
  export interface StepConfig<S extends ZodType> {
4
- name: string;
4
+ name?: string;
5
5
  type: string;
6
+ shortDescription?: string;
7
+ longDescription?: string;
8
+ yamlExample?: string;
6
9
  schema: S;
7
10
  }
8
11
  /**
@@ -15,10 +18,18 @@ export interface StepConfig<S extends ZodType> {
15
18
  * ```typescript
16
19
  * @Step({ name: "My Step", type: "my-step", schema: mySchema })
17
20
  * class MyStep extends SimpleStep<MyParams> {
21
+ * constructor(private apiClient: ApiClient) {
22
+ * super();
23
+ * }
24
+ *
18
25
  * async run(params: MyParams): Promise<RunOutput> {
26
+ * await this.apiClient.doSomething();
19
27
  * return StepOutputs.success();
20
28
  * }
21
29
  * }
30
+ *
31
+ * // Register with dependencies injected
32
+ * StepRegistry.run([new MyStep(apiClient)]);
22
33
  * ```
23
34
  */
24
35
  export declare function Step<S extends ZodType>(config: StepConfig<S>): <T extends new (...args: any[]) => BaseStep<z.infer<S>>>(Base: T) => T;
@@ -1,6 +1,7 @@
1
1
  "use strict";
2
2
  Object.defineProperty(exports, "__esModule", { value: true });
3
3
  exports.Step = Step;
4
+ const zod_1 = require("zod");
4
5
  const base_step_1 = require("../base/base-step");
5
6
  /**
6
7
  * Decorator that adds metadata to a step class.
@@ -12,10 +13,18 @@ const base_step_1 = require("../base/base-step");
12
13
  * ```typescript
13
14
  * @Step({ name: "My Step", type: "my-step", schema: mySchema })
14
15
  * class MyStep extends SimpleStep<MyParams> {
16
+ * constructor(private apiClient: ApiClient) {
17
+ * super();
18
+ * }
19
+ *
15
20
  * async run(params: MyParams): Promise<RunOutput> {
21
+ * await this.apiClient.doSomething();
16
22
  * return StepOutputs.success();
17
23
  * }
18
24
  * }
25
+ *
26
+ * // Register with dependencies injected
27
+ * StepRegistry.run([new MyStep(apiClient)]);
19
28
  * ```
20
29
  */
21
30
  function Step(config) {
@@ -31,9 +40,13 @@ function Step(config) {
31
40
  // Check if prepare is overridden (not the default BaseStep.prepare)
32
41
  const requiresApproval = this.prepare !== base_step_1.BaseStep.prototype.prepare;
33
42
  return {
34
- type: config.type,
35
43
  name: config.name,
44
+ stepType: config.type,
45
+ shortDescription: config.shortDescription,
46
+ longDescription: config.longDescription,
47
+ yamlExample: config.yamlExample,
36
48
  schema: config.schema,
49
+ jsonSchema: zod_1.z.toJSONSchema(config.schema),
37
50
  stepKind,
38
51
  requiresApproval,
39
52
  };
@@ -42,4 +55,4 @@ function Step(config) {
42
55
  return Enhanced;
43
56
  };
44
57
  }
45
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic3RlcC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9kZWNvcmF0b3JzL3N0ZXAudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7QUEwQkEsb0JBOEJDO0FBdERELGlEQUE2QztBQVE3Qzs7Ozs7Ozs7Ozs7Ozs7O0dBZUc7QUFDSCxTQUFnQixJQUFJLENBQW9CLE1BQXFCO0lBQzNELE9BQU8sVUFDTCxJQUFPO1FBRVAsTUFBTSxRQUFRLEdBQUcsS0FBTSxTQUFRLElBQUk7WUFDeEIsV0FBVztnQkFDbEIsdURBQXVEO2dCQUN2RCxNQUFNLGlCQUFpQixHQUNyQixTQUFTLElBQUksSUFBSTtvQkFDakIsT0FBUSxJQUFZLENBQUMsT0FBTyxLQUFLLFVBQVU7b0JBQzNDLE1BQU0sSUFBSSxJQUFJO29CQUNkLE9BQVEsSUFBWSxDQUFDLElBQUksS0FBSyxVQUFVLENBQUM7Z0JBRTNDLE1BQU0sUUFBUSxHQUFhLGlCQUFpQixDQUFDLENBQUMsQ0FBQyxTQUFTLENBQUMsQ0FBQyxDQUFDLFFBQVEsQ0FBQztnQkFFcEUsb0VBQW9FO2dCQUNwRSxNQUFNLGdCQUFnQixHQUFHLElBQUksQ0FBQyxPQUFPLEtBQUssb0JBQVEsQ0FBQyxTQUFTLENBQUMsT0FBTyxDQUFDO2dCQUVyRSxPQUFPO29CQUNMLElBQUksRUFBRSxNQUFNLENBQUMsSUFBSTtvQkFDakIsSUFBSSxFQUFFLE1BQU0sQ0FBQyxJQUFJO29CQUNqQixNQUFNLEVBQUUsTUFBTSxDQUFDLE1BQU07b0JBQ3JCLFFBQVE7b0JBQ1IsZ0JBQWdCO2lCQUNqQixDQUFDO1lBQ0osQ0FBQztTQUNGLENBQUM7UUFFRixPQUFPLFFBQWEsQ0FBQztJQUN2QixDQUFDLENBQUM7QUFDSixDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHR5cGUgeyBab2RUeXBlLCB6IH0gZnJvbSBcInpvZFwiO1xuaW1wb3J0IHR5cGUgeyBTdGVwS2luZCwgU3RlcE1ldGFkYXRhIH0gZnJvbSBcIi4uL2Jhc2UvYmFzZS1zdGVwXCI7XG5pbXBvcnQgeyBCYXNlU3RlcCB9IGZyb20gXCIuLi9iYXNlL2Jhc2Utc3RlcFwiO1xuXG5leHBvcnQgaW50ZXJmYWNlIFN0ZXBDb25maWc8UyBleHRlbmRzIFpvZFR5cGU+IHtcbiAgbmFtZTogc3RyaW5nO1xuICB0eXBlOiBzdHJpbmc7XG4gIHNjaGVtYTogUztcbn1cblxuLyoqXG4gKiBEZWNvcmF0b3IgdGhhdCBhZGRzIG1ldGFkYXRhIHRvIGEgc3RlcCBjbGFzcy5cbiAqIFRoaXMgaXMgcmVxdWlyZWQgZm9yIGFsbCBzdGVwcyB0byBiZSByZWdpc3RlcmVkIHdpdGggU3RlcFJlZ2lzdHJ5LlxuICpcbiAqIFRoZSBkZWNvcmF0ZWQgY2xhc3MgbXVzdCBleHRlbmQgU2ltcGxlU3RlcCBvciBQb2xsaW5nU3RlcC5cbiAqXG4gKiBAZXhhbXBsZVxuICogYGBgdHlwZXNjcmlwdFxuICogQFN0ZXAoeyBuYW1lOiBcIk15IFN0ZXBcIiwgdHlwZTogXCJteS1zdGVwXCIsIHNjaGVtYTogbXlTY2hlbWEgfSlcbiAqIGNsYXNzIE15U3RlcCBleHRlbmRzIFNpbXBsZVN0ZXA8TXlQYXJhbXM+IHtcbiAqICAgYXN5bmMgcnVuKHBhcmFtczogTXlQYXJhbXMpOiBQcm9taXNlPFJ1bk91dHB1dD4ge1xuICogICAgIHJldHVybiBTdGVwT3V0cHV0cy5zdWNjZXNzKCk7XG4gKiAgIH1cbiAqIH1cbiAqIGBgYFxuICovXG5leHBvcnQgZnVuY3Rpb24gU3RlcDxTIGV4dGVuZHMgWm9kVHlwZT4oY29uZmlnOiBTdGVwQ29uZmlnPFM+KSB7XG4gIHJldHVybiBmdW5jdGlvbiA8VCBleHRlbmRzIG5ldyAoLi4uYXJnczogYW55W10pID0+IEJhc2VTdGVwPHouaW5mZXI8Uz4+PihcbiAgICBCYXNlOiBUXG4gICk6IFQge1xuICAgIGNvbnN0IEVuaGFuY2VkID0gY2xhc3MgZXh0ZW5kcyBCYXNlIHtcbiAgICAgIG92ZXJyaWRlIGdldE1ldGFkYXRhKCk6IFN0ZXBNZXRhZGF0YTxTPiB7XG4gICAgICAgIC8vIERldGVjdCBzdGVwIGtpbmQgYnkgY2hlY2tpbmcgZm9yIFBvbGxpbmdTdGVwIG1ldGhvZHNcbiAgICAgICAgY29uc3QgaGFzUG9sbGluZ01ldGhvZHMgPVxuICAgICAgICAgIFwidHJpZ2dlclwiIGluIHRoaXMgJiZcbiAgICAgICAgICB0eXBlb2YgKHRoaXMgYXMgYW55KS50cmlnZ2VyID09PSBcImZ1bmN0aW9uXCIgJiZcbiAgICAgICAgICBcInBvbGxcIiBpbiB0aGlzICYmXG4gICAgICAgICAgdHlwZW9mICh0aGlzIGFzIGFueSkucG9sbCA9PT0gXCJmdW5jdGlvblwiO1xuXG4gICAgICAgIGNvbnN0IHN0ZXBLaW5kOiBTdGVwS2luZCA9IGhhc1BvbGxpbmdNZXRob2RzID8gXCJwb2xsaW5nXCIgOiBcInNpbXBsZVwiO1xuXG4gICAgICAgIC8vIENoZWNrIGlmIHByZXBhcmUgaXMgb3ZlcnJpZGRlbiAobm90IHRoZSBkZWZhdWx0IEJhc2VTdGVwLnByZXBhcmUpXG4gICAgICAgIGNvbnN0IHJlcXVpcmVzQXBwcm92YWwgPSB0aGlzLnByZXBhcmUgIT09IEJhc2VTdGVwLnByb3RvdHlwZS5wcmVwYXJlO1xuXG4gICAgICAgIHJldHVybiB7XG4gICAgICAgICAgdHlwZTogY29uZmlnLnR5cGUsXG4gICAgICAgICAgbmFtZTogY29uZmlnLm5hbWUsXG4gICAgICAgICAgc2NoZW1hOiBjb25maWcuc2NoZW1hLFxuICAgICAgICAgIHN0ZXBLaW5kLFxuICAgICAgICAgIHJlcXVpcmVzQXBwcm92YWwsXG4gICAgICAgIH07XG4gICAgICB9XG4gICAgfTtcblxuICAgIHJldHVybiBFbmhhbmNlZCBhcyBUO1xuICB9O1xufVxuIl19
58
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic3RlcC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9kZWNvcmF0b3JzL3N0ZXAudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7QUFxQ0Esb0JBa0NDO0FBdkVELDZCQUFzQztBQUV0QyxpREFBNkM7QUFXN0M7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7O0dBdUJHO0FBQ0gsU0FBZ0IsSUFBSSxDQUFvQixNQUFxQjtJQUMzRCxPQUFPLFVBQ0wsSUFBTztRQUVQLE1BQU0sUUFBUSxHQUFHLEtBQU0sU0FBUSxJQUFJO1lBQ3hCLFdBQVc7Z0JBQ2xCLHVEQUF1RDtnQkFDdkQsTUFBTSxpQkFBaUIsR0FDckIsU0FBUyxJQUFJLElBQUk7b0JBQ2pCLE9BQVEsSUFBWSxDQUFDLE9BQU8sS0FBSyxVQUFVO29CQUMzQyxNQUFNLElBQUksSUFBSTtvQkFDZCxPQUFRLElBQVksQ0FBQyxJQUFJLEtBQUssVUFBVSxDQUFDO2dCQUUzQyxNQUFNLFFBQVEsR0FBYSxpQkFBaUIsQ0FBQyxDQUFDLENBQUMsU0FBUyxDQUFDLENBQUMsQ0FBQyxRQUFRLENBQUM7Z0JBRXBFLG9FQUFvRTtnQkFDcEUsTUFBTSxnQkFBZ0IsR0FBRyxJQUFJLENBQUMsT0FBTyxLQUFLLG9CQUFRLENBQUMsU0FBUyxDQUFDLE9BQU8sQ0FBQztnQkFFckUsT0FBTztvQkFDTCxJQUFJLEVBQUUsTUFBTSxDQUFDLElBQUk7b0JBQ2pCLFFBQVEsRUFBRSxNQUFNLENBQUMsSUFBSTtvQkFDckIsZ0JBQWdCLEVBQUUsTUFBTSxDQUFDLGdCQUFnQjtvQkFDekMsZUFBZSxFQUFFLE1BQU0sQ0FBQyxlQUFlO29CQUN2QyxXQUFXLEVBQUUsTUFBTSxDQUFDLFdBQVc7b0JBQy9CLE1BQU0sRUFBRSxNQUFNLENBQUMsTUFBTTtvQkFDckIsVUFBVSxFQUFFLE9BQUMsQ0FBQyxZQUFZLENBQUMsTUFBTSxDQUFDLE1BQU0sQ0FBQztvQkFDekMsUUFBUTtvQkFDUixnQkFBZ0I7aUJBQ2pCLENBQUM7WUFDSixDQUFDO1NBQ0YsQ0FBQztRQUVGLE9BQU8sUUFBYSxDQUFDO0lBQ3ZCLENBQUMsQ0FBQztBQUNKLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyB0eXBlIFpvZFR5cGUsIHogfSBmcm9tIFwiem9kXCI7XG5pbXBvcnQgdHlwZSB7IFN0ZXBLaW5kLCBTdGVwTWV0YWRhdGEgfSBmcm9tIFwiLi4vYmFzZS9iYXNlLXN0ZXBcIjtcbmltcG9ydCB7IEJhc2VTdGVwIH0gZnJvbSBcIi4uL2Jhc2UvYmFzZS1zdGVwXCI7XG5cbmV4cG9ydCBpbnRlcmZhY2UgU3RlcENvbmZpZzxTIGV4dGVuZHMgWm9kVHlwZT4ge1xuICBuYW1lPzogc3RyaW5nO1xuICB0eXBlOiBzdHJpbmc7XG4gIHNob3J0RGVzY3JpcHRpb24/OiBzdHJpbmc7XG4gIGxvbmdEZXNjcmlwdGlvbj86IHN0cmluZztcbiAgeWFtbEV4YW1wbGU/OiBzdHJpbmc7XG4gIHNjaGVtYTogUztcbn1cblxuLyoqXG4gKiBEZWNvcmF0b3IgdGhhdCBhZGRzIG1ldGFkYXRhIHRvIGEgc3RlcCBjbGFzcy5cbiAqIFRoaXMgaXMgcmVxdWlyZWQgZm9yIGFsbCBzdGVwcyB0byBiZSByZWdpc3RlcmVkIHdpdGggU3RlcFJlZ2lzdHJ5LlxuICpcbiAqIFRoZSBkZWNvcmF0ZWQgY2xhc3MgbXVzdCBleHRlbmQgU2ltcGxlU3RlcCBvciBQb2xsaW5nU3RlcC5cbiAqXG4gKiBAZXhhbXBsZVxuICogYGBgdHlwZXNjcmlwdFxuICogQFN0ZXAoeyBuYW1lOiBcIk15IFN0ZXBcIiwgdHlwZTogXCJteS1zdGVwXCIsIHNjaGVtYTogbXlTY2hlbWEgfSlcbiAqIGNsYXNzIE15U3RlcCBleHRlbmRzIFNpbXBsZVN0ZXA8TXlQYXJhbXM+IHtcbiAqICAgY29uc3RydWN0b3IocHJpdmF0ZSBhcGlDbGllbnQ6IEFwaUNsaWVudCkge1xuICogICAgIHN1cGVyKCk7XG4gKiAgIH1cbiAqXG4gKiAgIGFzeW5jIHJ1bihwYXJhbXM6IE15UGFyYW1zKTogUHJvbWlzZTxSdW5PdXRwdXQ+IHtcbiAqICAgICBhd2FpdCB0aGlzLmFwaUNsaWVudC5kb1NvbWV0aGluZygpO1xuICogICAgIHJldHVybiBTdGVwT3V0cHV0cy5zdWNjZXNzKCk7XG4gKiAgIH1cbiAqIH1cbiAqXG4gKiAvLyBSZWdpc3RlciB3aXRoIGRlcGVuZGVuY2llcyBpbmplY3RlZFxuICogU3RlcFJlZ2lzdHJ5LnJ1bihbbmV3IE15U3RlcChhcGlDbGllbnQpXSk7XG4gKiBgYGBcbiAqL1xuZXhwb3J0IGZ1bmN0aW9uIFN0ZXA8UyBleHRlbmRzIFpvZFR5cGU+KGNvbmZpZzogU3RlcENvbmZpZzxTPikge1xuICByZXR1cm4gZnVuY3Rpb24gPFQgZXh0ZW5kcyBuZXcgKC4uLmFyZ3M6IGFueVtdKSA9PiBCYXNlU3RlcDx6LmluZmVyPFM+Pj4oXG4gICAgQmFzZTogVFxuICApOiBUIHtcbiAgICBjb25zdCBFbmhhbmNlZCA9IGNsYXNzIGV4dGVuZHMgQmFzZSB7XG4gICAgICBvdmVycmlkZSBnZXRNZXRhZGF0YSgpOiBTdGVwTWV0YWRhdGE8Uz4ge1xuICAgICAgICAvLyBEZXRlY3Qgc3RlcCBraW5kIGJ5IGNoZWNraW5nIGZvciBQb2xsaW5nU3RlcCBtZXRob2RzXG4gICAgICAgIGNvbnN0IGhhc1BvbGxpbmdNZXRob2RzID1cbiAgICAgICAgICBcInRyaWdnZXJcIiBpbiB0aGlzICYmXG4gICAgICAgICAgdHlwZW9mICh0aGlzIGFzIGFueSkudHJpZ2dlciA9PT0gXCJmdW5jdGlvblwiICYmXG4gICAgICAgICAgXCJwb2xsXCIgaW4gdGhpcyAmJlxuICAgICAgICAgIHR5cGVvZiAodGhpcyBhcyBhbnkpLnBvbGwgPT09IFwiZnVuY3Rpb25cIjtcblxuICAgICAgICBjb25zdCBzdGVwS2luZDogU3RlcEtpbmQgPSBoYXNQb2xsaW5nTWV0aG9kcyA/IFwicG9sbGluZ1wiIDogXCJzaW1wbGVcIjtcblxuICAgICAgICAvLyBDaGVjayBpZiBwcmVwYXJlIGlzIG92ZXJyaWRkZW4gKG5vdCB0aGUgZGVmYXVsdCBCYXNlU3RlcC5wcmVwYXJlKVxuICAgICAgICBjb25zdCByZXF1aXJlc0FwcHJvdmFsID0gdGhpcy5wcmVwYXJlICE9PSBCYXNlU3RlcC5wcm90b3R5cGUucHJlcGFyZTtcblxuICAgICAgICByZXR1cm4ge1xuICAgICAgICAgIG5hbWU6IGNvbmZpZy5uYW1lLFxuICAgICAgICAgIHN0ZXBUeXBlOiBjb25maWcudHlwZSxcbiAgICAgICAgICBzaG9ydERlc2NyaXB0aW9uOiBjb25maWcuc2hvcnREZXNjcmlwdGlvbixcbiAgICAgICAgICBsb25nRGVzY3JpcHRpb246IGNvbmZpZy5sb25nRGVzY3JpcHRpb24sXG4gICAgICAgICAgeWFtbEV4YW1wbGU6IGNvbmZpZy55YW1sRXhhbXBsZSxcbiAgICAgICAgICBzY2hlbWE6IGNvbmZpZy5zY2hlbWEsXG4gICAgICAgICAganNvblNjaGVtYTogei50b0pTT05TY2hlbWEoY29uZmlnLnNjaGVtYSksXG4gICAgICAgICAgc3RlcEtpbmQsXG4gICAgICAgICAgcmVxdWlyZXNBcHByb3ZhbCxcbiAgICAgICAgfTtcbiAgICAgIH1cbiAgICB9O1xuXG4gICAgcmV0dXJuIEVuaGFuY2VkIGFzIFQ7XG4gIH07XG59XG4iXX0=
package/dist/index.d.ts CHANGED
@@ -1,4 +1,4 @@
1
- export { BaseStep, StepClass, StepMetadata, StepKind } from "./base/base-step";
1
+ export { BaseStep, StepMetadata, StepKind } from "./base/base-step";
2
2
  export { SimpleStep } from "./base/simple-step";
3
3
  export { PollingStep } from "./base/polling-step";
4
4
  export { Step, StepConfig } from "./decorators/step";
package/dist/index.js CHANGED
@@ -32,4 +32,4 @@ Object.defineProperty(exports, "StepRegistryInputSchema", { enumerable: true, ge
32
32
  // Logging
33
33
  var step_logger_1 = require("./logging/step-logger");
34
34
  Object.defineProperty(exports, "StepLogger", { enumerable: true, get: function () { return step_logger_1.StepLogger; } });
35
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi9zcmMvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IjtBQUFBLGdGQUFnRjtBQUNoRix3QkFBd0I7QUFDeEIsZ0ZBQWdGOzs7QUFFaEYsa0NBQWtDO0FBQ2xDLDhDQUErRTtBQUF0RSxxR0FBQSxRQUFRLE9BQUE7QUFDakIsa0RBQWdEO0FBQXZDLHlHQUFBLFVBQVUsT0FBQTtBQUNuQixvREFBa0Q7QUFBekMsMkdBQUEsV0FBVyxPQUFBO0FBRXBCLHlDQUF5QztBQUN6QywwQ0FBcUQ7QUFBNUMsNEZBQUEsSUFBSSxPQUFBO0FBRWIsb0NBQW9DO0FBQ3BDLG9EQXlCOEI7QUF0QjVCLG9IQUFBLHFCQUFxQixPQUFBO0FBY3JCLGdDQUFnQztBQUNoQyxrSEFBQSxtQkFBbUIsT0FBQTtBQUNuQiw4R0FBQSxlQUFlLE9BQUE7QUFDZixrSEFBQSxtQkFBbUIsT0FBQTtBQUNuQiwrR0FBQSxnQkFBZ0IsT0FBQTtBQUNoQiwrR0FBQSxnQkFBZ0IsT0FBQTtBQUNoQixtQkFBbUI7QUFDbkIsMEdBQUEsV0FBVyxPQUFBO0FBR2IsNkJBQTZCO0FBQzdCLDBEQUFvRztBQUEzRiw2R0FBQSxZQUFZLE9BQUE7QUFBcUIsd0hBQUEsdUJBQXVCLE9BQUE7QUFFakUsVUFBVTtBQUNWLHFEQUF5RjtBQUFoRix5R0FBQSxVQUFVLE9BQUEiLCJzb3VyY2VzQ29udGVudCI6WyIvLyA9PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PVxuLy8gU3RlcCBTREsgLSBQdWJsaWMgQVBJXG4vLyA9PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PVxuXG4vLyBCYXNlIGNsYXNzZXMgZm9yIGNyZWF0aW5nIHN0ZXBzXG5leHBvcnQgeyBCYXNlU3RlcCwgU3RlcENsYXNzLCBTdGVwTWV0YWRhdGEsIFN0ZXBLaW5kIH0gZnJvbSBcIi4vYmFzZS9iYXNlLXN0ZXBcIjtcbmV4cG9ydCB7IFNpbXBsZVN0ZXAgfSBmcm9tIFwiLi9iYXNlL3NpbXBsZS1zdGVwXCI7XG5leHBvcnQgeyBQb2xsaW5nU3RlcCB9IGZyb20gXCIuL2Jhc2UvcG9sbGluZy1zdGVwXCI7XG5cbi8vIERlY29yYXRvciBmb3IgYWRkaW5nIG1ldGFkYXRhIHRvIHN0ZXBzXG5leHBvcnQgeyBTdGVwLCBTdGVwQ29uZmlnIH0gZnJvbSBcIi4vZGVjb3JhdG9ycy9zdGVwXCI7XG5cbi8vIE91dHB1dCB0eXBlcyBhbmQgaGVscGVyIGZ1bmN0aW9uc1xuZXhwb3J0IHtcbiAgLy8gQXBwcm92YWwgY29udGV4dFxuICBBcHByb3ZhbENvbnRleHQsXG4gIEFwcHJvdmFsQ29udGV4dFNjaGVtYSxcbiAgLy8gUGhhc2Utc3BlY2lmaWMgb3V0cHV0c1xuICBQcmVwYXJlT3V0cHV0LFxuICBSdW5PdXRwdXQsXG4gIFRyaWdnZXJPdXRwdXQsXG4gIFRyaWdnZXJlZE91dHB1dCxcbiAgUG9sbE91dHB1dCxcbiAgUG9sbEFnYWluT3V0cHV0LFxuICAvLyBDb21tb24gb3V0cHV0c1xuICBTdWNjZXNzT3V0cHV0LFxuICBGYWlsZWRPdXRwdXQsXG4gIEFwcHJvdmFsUmVxdWlyZWRPdXRwdXQsXG4gIC8vIENvbWJpbmVkIG91dHB1dCB0eXBlXG4gIFN0ZXBPdXRwdXQsXG4gIC8vIE91dHB1dCBzY2hlbWFzIGZvciB2YWxpZGF0aW9uXG4gIFByZXBhcmVPdXRwdXRTY2hlbWEsXG4gIFJ1bk91dHB1dFNjaGVtYSxcbiAgVHJpZ2dlck91dHB1dFNjaGVtYSxcbiAgUG9sbE91dHB1dFNjaGVtYSxcbiAgU3RlcE91dHB1dFNjaGVtYSxcbiAgLy8gSGVscGVyIGZ1bmN0aW9uc1xuICBTdGVwT3V0cHV0cyxcbn0gZnJvbSBcIi4vb3V0cHV0L3N0ZXAtb3V0cHV0XCI7XG5cbi8vIFJlZ2lzdHJ5IGZvciBydW5uaW5nIHN0ZXBzXG5leHBvcnQgeyBTdGVwUmVnaXN0cnksIFN0ZXBSZWdpc3RyeUlucHV0LCBTdGVwUmVnaXN0cnlJbnB1dFNjaGVtYSB9IGZyb20gXCIuL3JlZ2lzdHJ5L3N0ZXAtcmVnaXN0cnlcIjtcblxuLy8gTG9nZ2luZ1xuZXhwb3J0IHsgU3RlcExvZ2dlciwgU3RlcExvZ2dlckNvbmZpZywgTG9nRW50cnksIExvZ0xldmVsIH0gZnJvbSBcIi4vbG9nZ2luZy9zdGVwLWxvZ2dlclwiO1xuIl19
35
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi9zcmMvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IjtBQUFBLGdGQUFnRjtBQUNoRix3QkFBd0I7QUFDeEIsZ0ZBQWdGOzs7QUFFaEYsa0NBQWtDO0FBQ2xDLDhDQUFvRTtBQUEzRCxxR0FBQSxRQUFRLE9BQUE7QUFDakIsa0RBQWdEO0FBQXZDLHlHQUFBLFVBQVUsT0FBQTtBQUNuQixvREFBa0Q7QUFBekMsMkdBQUEsV0FBVyxPQUFBO0FBRXBCLHlDQUF5QztBQUN6QywwQ0FBcUQ7QUFBNUMsNEZBQUEsSUFBSSxPQUFBO0FBRWIsb0NBQW9DO0FBQ3BDLG9EQXlCOEI7QUF0QjVCLG9IQUFBLHFCQUFxQixPQUFBO0FBY3JCLGdDQUFnQztBQUNoQyxrSEFBQSxtQkFBbUIsT0FBQTtBQUNuQiw4R0FBQSxlQUFlLE9BQUE7QUFDZixrSEFBQSxtQkFBbUIsT0FBQTtBQUNuQiwrR0FBQSxnQkFBZ0IsT0FBQTtBQUNoQiwrR0FBQSxnQkFBZ0IsT0FBQTtBQUNoQixtQkFBbUI7QUFDbkIsMEdBQUEsV0FBVyxPQUFBO0FBR2IsNkJBQTZCO0FBQzdCLDBEQUFvRztBQUEzRiw2R0FBQSxZQUFZLE9BQUE7QUFBcUIsd0hBQUEsdUJBQXVCLE9BQUE7QUFFakUsVUFBVTtBQUNWLHFEQUF5RjtBQUFoRix5R0FBQSxVQUFVLE9BQUEiLCJzb3VyY2VzQ29udGVudCI6WyIvLyA9PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PVxuLy8gU3RlcCBTREsgLSBQdWJsaWMgQVBJXG4vLyA9PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PVxuXG4vLyBCYXNlIGNsYXNzZXMgZm9yIGNyZWF0aW5nIHN0ZXBzXG5leHBvcnQgeyBCYXNlU3RlcCwgU3RlcE1ldGFkYXRhLCBTdGVwS2luZCB9IGZyb20gXCIuL2Jhc2UvYmFzZS1zdGVwXCI7XG5leHBvcnQgeyBTaW1wbGVTdGVwIH0gZnJvbSBcIi4vYmFzZS9zaW1wbGUtc3RlcFwiO1xuZXhwb3J0IHsgUG9sbGluZ1N0ZXAgfSBmcm9tIFwiLi9iYXNlL3BvbGxpbmctc3RlcFwiO1xuXG4vLyBEZWNvcmF0b3IgZm9yIGFkZGluZyBtZXRhZGF0YSB0byBzdGVwc1xuZXhwb3J0IHsgU3RlcCwgU3RlcENvbmZpZyB9IGZyb20gXCIuL2RlY29yYXRvcnMvc3RlcFwiO1xuXG4vLyBPdXRwdXQgdHlwZXMgYW5kIGhlbHBlciBmdW5jdGlvbnNcbmV4cG9ydCB7XG4gIC8vIEFwcHJvdmFsIGNvbnRleHRcbiAgQXBwcm92YWxDb250ZXh0LFxuICBBcHByb3ZhbENvbnRleHRTY2hlbWEsXG4gIC8vIFBoYXNlLXNwZWNpZmljIG91dHB1dHNcbiAgUHJlcGFyZU91dHB1dCxcbiAgUnVuT3V0cHV0LFxuICBUcmlnZ2VyT3V0cHV0LFxuICBUcmlnZ2VyZWRPdXRwdXQsXG4gIFBvbGxPdXRwdXQsXG4gIFBvbGxBZ2Fpbk91dHB1dCxcbiAgLy8gQ29tbW9uIG91dHB1dHNcbiAgU3VjY2Vzc091dHB1dCxcbiAgRmFpbGVkT3V0cHV0LFxuICBBcHByb3ZhbFJlcXVpcmVkT3V0cHV0LFxuICAvLyBDb21iaW5lZCBvdXRwdXQgdHlwZVxuICBTdGVwT3V0cHV0LFxuICAvLyBPdXRwdXQgc2NoZW1hcyBmb3IgdmFsaWRhdGlvblxuICBQcmVwYXJlT3V0cHV0U2NoZW1hLFxuICBSdW5PdXRwdXRTY2hlbWEsXG4gIFRyaWdnZXJPdXRwdXRTY2hlbWEsXG4gIFBvbGxPdXRwdXRTY2hlbWEsXG4gIFN0ZXBPdXRwdXRTY2hlbWEsXG4gIC8vIEhlbHBlciBmdW5jdGlvbnNcbiAgU3RlcE91dHB1dHMsXG59IGZyb20gXCIuL291dHB1dC9zdGVwLW91dHB1dFwiO1xuXG4vLyBSZWdpc3RyeSBmb3IgcnVubmluZyBzdGVwc1xuZXhwb3J0IHsgU3RlcFJlZ2lzdHJ5LCBTdGVwUmVnaXN0cnlJbnB1dCwgU3RlcFJlZ2lzdHJ5SW5wdXRTY2hlbWEgfSBmcm9tIFwiLi9yZWdpc3RyeS9zdGVwLXJlZ2lzdHJ5XCI7XG5cbi8vIExvZ2dpbmdcbmV4cG9ydCB7IFN0ZXBMb2dnZXIsIFN0ZXBMb2dnZXJDb25maWcsIExvZ0VudHJ5LCBMb2dMZXZlbCB9IGZyb20gXCIuL2xvZ2dpbmcvc3RlcC1sb2dnZXJcIjtcbiJdfQ==
@@ -1,5 +1,5 @@
1
1
  import z from "zod";
2
- import type { StepClass } from "../base/base-step";
2
+ import type { BaseStep } from "../base/base-step";
3
3
  export declare const StepRegistryInputSchema: z.ZodDiscriminatedUnion<[z.ZodObject<{
4
4
  job: z.ZodLiteral<"EXECUTE">;
5
5
  type: z.ZodString;
@@ -25,7 +25,10 @@ export declare class StepRegistry {
25
25
  *
26
26
  * Usage in user's entrypoint file:
27
27
  * ```
28
- * StepRegistry.run([MyStep1, MyStep2, ...]);
28
+ * StepRegistry.run([
29
+ * new MyStep1(dependency1, dependency2),
30
+ * new MyStep2(config)
31
+ * ]);
29
32
  * ```
30
33
  *
31
34
  * CLI arguments:
@@ -34,7 +37,7 @@ export declare class StepRegistry {
34
37
  * - --log-dir: Directory for log files (default: /tmp/step-logs)
35
38
  * - --execution-id: Unique ID for this execution (required for logging)
36
39
  */
37
- static run(steps: StepClass[]): Promise<void>;
40
+ static run(steps: BaseStep<unknown>[]): Promise<void>;
38
41
  private execute;
39
42
  private synthesizeMetadata;
40
43
  private executeStep;
@@ -79,10 +79,9 @@ class StepRegistry {
79
79
  this.outputPath = outputPath;
80
80
  this.logDir = logDir;
81
81
  this.executionId = executionId;
82
- for (const StepCls of steps) {
83
- const instance = new StepCls();
82
+ for (const instance of steps) {
84
83
  const metadata = instance.getMetadata();
85
- this.steps.set(metadata.type, StepCls);
84
+ this.steps.set(metadata.stepType, instance);
86
85
  }
87
86
  }
88
87
  /**
@@ -91,7 +90,10 @@ class StepRegistry {
91
90
  *
92
91
  * Usage in user's entrypoint file:
93
92
  * ```
94
- * StepRegistry.run([MyStep1, MyStep2, ...]);
93
+ * StepRegistry.run([
94
+ * new MyStep1(dependency1, dependency2),
95
+ * new MyStep2(config)
96
+ * ]);
95
97
  * ```
96
98
  *
97
99
  * CLI arguments:
@@ -130,26 +132,24 @@ class StepRegistry {
130
132
  }
131
133
  synthesizeMetadata() {
132
134
  const metadata = [];
133
- for (const StepCls of this.steps.values()) {
134
- const instance = new StepCls();
135
+ for (const instance of this.steps.values()) {
135
136
  metadata.push(instance.getMetadata());
136
137
  }
137
138
  return metadata;
138
139
  }
139
140
  async executeStep(type, params, approvalContext, pollingState) {
140
- const StepCls = this.steps.get(type);
141
- if (!StepCls) {
141
+ const instance = this.steps.get(type);
142
+ if (!instance) {
142
143
  this.writeOutput(step_output_1.StepOutputs.failed(`No step registered with type: ${type}`, "STEP_NOT_FOUND"));
143
144
  return;
144
145
  }
145
146
  try {
146
- const instance = new StepCls();
147
147
  const metadata = instance.getMetadata();
148
148
  // Inject logger
149
149
  const logger = new step_logger_1.StepLogger({
150
150
  logDir: this.logDir,
151
151
  executionId: this.executionId,
152
- stepType: metadata.type,
152
+ stepType: metadata.stepType,
153
153
  });
154
154
  instance._setLogger(logger);
155
155
  // Validate params against the step's schema
@@ -235,4 +235,4 @@ class StepRegistry {
235
235
  }
236
236
  }
237
237
  exports.StepRegistry = StepRegistry;
238
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"step-registry.js","sourceRoot":"","sources":["../../src/registry/step-registry.ts"],"names":[],"mappings":"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","sourcesContent":["import * as fs from \"fs\";\nimport * as path from \"path\";\nimport minimist from \"minimist\";\nimport z from \"zod\";\nimport type { BaseStep, StepClass, StepMetadata } from \"../base/base-step\";\nimport type { SimpleStep } from \"../base/simple-step\";\nimport type { PollingStep } from \"../base/polling-step\";\nimport { StepLogger } from \"../logging/step-logger\";\nimport type {\n  ApprovalContext,\n  StepOutput} from \"../output/step-output\";\nimport {\n  ApprovalContextSchema,\n  StepOutputs,\n} from \"../output/step-output\";\n\n// =============================================================================\n// Input Schemas\n// =============================================================================\n\nconst StepRegistrySynthesizeSchema = z.object({\n  job: z.literal(\"SYNTHESIZE-METADATA\"),\n});\n\nconst StepRegistryExecuteSchema = z.object({\n  job: z.literal(\"EXECUTE\"),\n  type: z.string(),\n  params: z.record(z.string(), z.any()),\n  // Optional context - presence determines which phase to run\n  approvalContext: ApprovalContextSchema.optional(),\n  pollingState: z.record(z.string(), z.any()).optional(),\n});\n\nexport const StepRegistryInputSchema = z.discriminatedUnion(\"job\", [\n  StepRegistryExecuteSchema,\n  StepRegistrySynthesizeSchema,\n]);\n\nexport type StepRegistryInput = z.infer<typeof StepRegistryInputSchema>;\n\n// =============================================================================\n// Configuration\n// =============================================================================\n\nconst DEFAULT_OUTPUT_PATH = \"/tmp/step-output.json\";\nconst DEFAULT_LOG_DIR = \"/tmp/step-logs\";\n\n// =============================================================================\n// StepRegistry\n// =============================================================================\n\nexport class StepRegistry {\n  private steps: Map<string, StepClass>;\n  private outputPath: string;\n  private logDir: string;\n  private executionId: string;\n\n  private constructor(\n    steps: StepClass[],\n    outputPath: string,\n    logDir: string,\n    executionId: string\n  ) {\n    this.steps = new Map();\n    this.outputPath = outputPath;\n    this.logDir = logDir;\n    this.executionId = executionId;\n\n    for (const StepCls of steps) {\n      const instance = new StepCls();\n      const metadata = instance.getMetadata();\n      this.steps.set(metadata.type, StepCls);\n    }\n  }\n\n  /**\n   * Main entrypoint for running steps.\n   * Parses CLI args and either synthesizes metadata or executes a step.\n   *\n   * Usage in user's entrypoint file:\n   * ```\n   * StepRegistry.run([MyStep1, MyStep2, ...]);\n   * ```\n   *\n   * CLI arguments:\n   * - --input: JSON blob with job type and parameters\n   * - --output: Path to write output JSON (default: /tmp/step-output.json)\n   * - --log-dir: Directory for log files (default: /tmp/step-logs)\n   * - --execution-id: Unique ID for this execution (required for logging)\n   */\n  static async run(steps: StepClass[]): Promise<void> {\n    const args = minimist(process.argv.slice(2));\n    const outputPath = args[\"output\"] || DEFAULT_OUTPUT_PATH;\n    const logDir = args[\"log-dir\"] || DEFAULT_LOG_DIR;\n    const executionId = args[\"execution-id\"] || `exec-${Date.now()}`;\n\n    const registry = new StepRegistry(steps, outputPath, logDir, executionId);\n    await registry.execute(args);\n  }\n\n  private async execute(args: minimist.ParsedArgs): Promise<void> {\n    try {\n      const input = StepRegistryInputSchema.parse(JSON.parse(args[\"input\"]));\n\n      switch (input.job) {\n        case \"SYNTHESIZE-METADATA\": {\n          const metadata = this.synthesizeMetadata();\n          this.writeOutput({ status: \"SUCCESS\", data: { metadata } });\n          break;\n        }\n\n        case \"EXECUTE\": {\n          await this.executeStep(\n            input.type,\n            input.params,\n            input.approvalContext,\n            input.pollingState\n          );\n          break;\n        }\n      }\n    } catch (error) {\n      const errorMessage =\n        error instanceof Error ? error.message : String(error);\n      this.writeOutput(StepOutputs.failed(errorMessage, \"REGISTRY_ERROR\"));\n    }\n  }\n\n  private synthesizeMetadata(): StepMetadata[] {\n    const metadata: StepMetadata[] = [];\n\n    for (const StepCls of this.steps.values()) {\n      const instance = new StepCls();\n      metadata.push(instance.getMetadata());\n    }\n\n    return metadata;\n  }\n\n  private async executeStep(\n    type: string,\n    params: Record<string, unknown>,\n    approvalContext?: ApprovalContext,\n    pollingState?: Record<string, unknown>\n  ): Promise<void> {\n    const StepCls = this.steps.get(type);\n\n    if (!StepCls) {\n      this.writeOutput(\n        StepOutputs.failed(`No step registered with type: ${type}`, \"STEP_NOT_FOUND\")\n      );\n      return;\n    }\n\n    try {\n      const instance = new StepCls();\n      const metadata = instance.getMetadata();\n\n      // Inject logger\n      const logger = new StepLogger({\n        logDir: this.logDir,\n        executionId: this.executionId,\n        stepType: metadata.type,\n      });\n      instance._setLogger(logger);\n\n      // Validate params against the step's schema\n      const parseResult = metadata.schema.safeParse(params);\n      if (!parseResult.success) {\n        this.writeOutput(\n          StepOutputs.failed(\n            `Invalid params: ${parseResult.error.message}`,\n            \"INVALID_PARAMS\"\n          )\n        );\n        return;\n      }\n\n      const validatedParams = parseResult.data;\n\n      // Route to appropriate phase based on step kind and input context\n      const output = await this.routeExecution(\n        instance,\n        metadata,\n        validatedParams,\n        approvalContext,\n        pollingState\n      );\n\n      this.writeOutput(output);\n    } catch (error) {\n      const errorMessage =\n        error instanceof Error ? error.message : String(error);\n      this.writeOutput(StepOutputs.failed(errorMessage, \"EXECUTION_ERROR\"));\n    }\n  }\n\n  private async routeExecution(\n    instance: BaseStep<unknown>,\n    metadata: StepMetadata,\n    params: unknown,\n    approvalContext?: ApprovalContext,\n    pollingState?: Record<string, unknown>\n  ): Promise<StepOutput> {\n    const requiresApproval = metadata.requiresApproval;\n\n    if (metadata.stepKind === \"simple\") {\n      return this.routeSimpleStep(\n        instance as SimpleStep<unknown>,\n        params,\n        requiresApproval,\n        approvalContext\n      );\n    }\n\n    if (metadata.stepKind === \"polling\") {\n      return this.routePollingStep(\n        instance as PollingStep<unknown, Record<string, unknown>>,\n        params,\n        requiresApproval,\n        approvalContext,\n        pollingState\n      );\n    }\n\n    return StepOutputs.failed(\n      `Unknown step kind: ${metadata.stepKind}`,\n      \"UNKNOWN_STEP_KIND\"\n    );\n  }\n\n  /**\n   * Route SimpleStep execution based on approval state.\n   *\n   * | requiresApproval | approvalContext provided? | Action              |\n   * |------------------|---------------------------|---------------------|\n   * | No               | -                         | Call run(params)    |\n   * | Yes              | No                        | Call prepare(params)|\n   * | Yes              | Yes                       | Call run(params, approval) |\n   */\n  private async routeSimpleStep(\n    instance: SimpleStep<unknown>,\n    params: unknown,\n    requiresApproval: boolean,\n    approvalContext?: ApprovalContext\n  ): Promise<StepOutput> {\n    if (!requiresApproval) {\n      // No approval needed - just run\n      return instance.execute(params, undefined);\n    }\n\n    if (!approvalContext) {\n      // Needs approval but don't have it yet - call prepare\n      return instance.prepare(params);\n    }\n\n    // Have approval - run with approval context\n    return instance.execute(params, approvalContext);\n  }\n\n  /**\n   * Route PollingStep execution based on approval and polling state.\n   *\n   * | requiresApproval | approvalContext? | pollingState? | Action                        |\n   * |------------------|------------------|---------------|-------------------------------|\n   * | No               | -                | No            | Call trigger(params)          |\n   * | No               | -                | Yes           | Call poll(params, pollingState)|\n   * | Yes              | No               | No            | Call prepare(params)          |\n   * | Yes              | Yes              | No            | Call trigger(params, approval)|\n   * | Yes              | Yes              | Yes           | Call poll(params, pollingState)|\n   */\n  private async routePollingStep(\n    instance: PollingStep<unknown, Record<string, unknown>>,\n    params: unknown,\n    requiresApproval: boolean,\n    approvalContext?: ApprovalContext,\n    pollingState?: Record<string, unknown>\n  ): Promise<StepOutput> {\n    // If we have polling state, we're in the poll phase (regardless of approval)\n    if (pollingState) {\n      return instance.executePoll(params, pollingState);\n    }\n\n    if (!requiresApproval) {\n      // No approval needed - trigger directly\n      return instance.executeTrigger(params, undefined);\n    }\n\n    if (!approvalContext) {\n      // Needs approval but don't have it yet - call prepare\n      return instance.prepare(params);\n    }\n\n    // Have approval - trigger with approval context\n    return instance.executeTrigger(params, approvalContext);\n  }\n\n  private writeOutput(output: StepOutput): void {\n    const outputDir = path.dirname(this.outputPath);\n    if (!fs.existsSync(outputDir)) {\n      fs.mkdirSync(outputDir, { recursive: true });\n    }\n\n    fs.writeFileSync(this.outputPath, JSON.stringify(output, null, 2));\n  }\n}\n"]}
238
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"step-registry.js","sourceRoot":"","sources":["../../src/registry/step-registry.ts"],"names":[],"mappings":"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","sourcesContent":["import * as fs from \"fs\";\nimport * as path from \"path\";\nimport minimist from \"minimist\";\nimport z from \"zod\";\nimport type { BaseStep, StepMetadata } from \"../base/base-step\";\nimport type { SimpleStep } from \"../base/simple-step\";\nimport type { PollingStep } from \"../base/polling-step\";\nimport { StepLogger } from \"../logging/step-logger\";\nimport type { ApprovalContext, StepOutput } from \"../output/step-output\";\nimport { ApprovalContextSchema, StepOutputs } from \"../output/step-output\";\n\n// =============================================================================\n// Input Schemas\n// =============================================================================\n\nconst StepRegistrySynthesizeSchema = z.object({\n  job: z.literal(\"SYNTHESIZE-METADATA\"),\n});\n\nconst StepRegistryExecuteSchema = z.object({\n  job: z.literal(\"EXECUTE\"),\n  type: z.string(),\n  params: z.record(z.string(), z.any()),\n  // Optional context - presence determines which phase to run\n  approvalContext: ApprovalContextSchema.optional(),\n  pollingState: z.record(z.string(), z.any()).optional(),\n});\n\nexport const StepRegistryInputSchema = z.discriminatedUnion(\"job\", [\n  StepRegistryExecuteSchema,\n  StepRegistrySynthesizeSchema,\n]);\n\nexport type StepRegistryInput = z.infer<typeof StepRegistryInputSchema>;\n\n// =============================================================================\n// Configuration\n// =============================================================================\n\nconst DEFAULT_OUTPUT_PATH = \"/tmp/step-output.json\";\nconst DEFAULT_LOG_DIR = \"/tmp/step-logs\";\n\n// =============================================================================\n// StepRegistry\n// =============================================================================\n\nexport class StepRegistry {\n  private steps: Map<string, BaseStep<unknown>>;\n  private outputPath: string;\n  private logDir: string;\n  private executionId: string;\n\n  private constructor(\n    steps: BaseStep<unknown>[],\n    outputPath: string,\n    logDir: string,\n    executionId: string\n  ) {\n    this.steps = new Map();\n    this.outputPath = outputPath;\n    this.logDir = logDir;\n    this.executionId = executionId;\n\n    for (const instance of steps) {\n      const metadata = instance.getMetadata();\n      this.steps.set(metadata.stepType, instance);\n    }\n  }\n\n  /**\n   * Main entrypoint for running steps.\n   * Parses CLI args and either synthesizes metadata or executes a step.\n   *\n   * Usage in user's entrypoint file:\n   * ```\n   * StepRegistry.run([\n   *   new MyStep1(dependency1, dependency2),\n   *   new MyStep2(config)\n   * ]);\n   * ```\n   *\n   * CLI arguments:\n   * - --input: JSON blob with job type and parameters\n   * - --output: Path to write output JSON (default: /tmp/step-output.json)\n   * - --log-dir: Directory for log files (default: /tmp/step-logs)\n   * - --execution-id: Unique ID for this execution (required for logging)\n   */\n  static async run(steps: BaseStep<unknown>[]): Promise<void> {\n    const args = minimist(process.argv.slice(2));\n    const outputPath = args[\"output\"] || DEFAULT_OUTPUT_PATH;\n    const logDir = args[\"log-dir\"] || DEFAULT_LOG_DIR;\n    const executionId = args[\"execution-id\"] || `exec-${Date.now()}`;\n\n    const registry = new StepRegistry(steps, outputPath, logDir, executionId);\n    await registry.execute(args);\n  }\n\n  private async execute(args: minimist.ParsedArgs): Promise<void> {\n    try {\n      const input = StepRegistryInputSchema.parse(JSON.parse(args[\"input\"]));\n\n      switch (input.job) {\n        case \"SYNTHESIZE-METADATA\": {\n          const metadata = this.synthesizeMetadata();\n          this.writeOutput({ status: \"SUCCESS\", data: { metadata } });\n          break;\n        }\n\n        case \"EXECUTE\": {\n          await this.executeStep(\n            input.type,\n            input.params,\n            input.approvalContext,\n            input.pollingState\n          );\n          break;\n        }\n      }\n    } catch (error) {\n      const errorMessage =\n        error instanceof Error ? error.message : String(error);\n      this.writeOutput(StepOutputs.failed(errorMessage, \"REGISTRY_ERROR\"));\n    }\n  }\n\n  private synthesizeMetadata(): StepMetadata[] {\n    const metadata: StepMetadata[] = [];\n\n    for (const instance of this.steps.values()) {\n      metadata.push(instance.getMetadata());\n    }\n\n    return metadata;\n  }\n\n  private async executeStep(\n    type: string,\n    params: Record<string, unknown>,\n    approvalContext?: ApprovalContext,\n    pollingState?: Record<string, unknown>\n  ): Promise<void> {\n    const instance = this.steps.get(type);\n\n    if (!instance) {\n      this.writeOutput(\n        StepOutputs.failed(\n          `No step registered with type: ${type}`,\n          \"STEP_NOT_FOUND\"\n        )\n      );\n      return;\n    }\n\n    try {\n      const metadata = instance.getMetadata();\n\n      // Inject logger\n      const logger = new StepLogger({\n        logDir: this.logDir,\n        executionId: this.executionId,\n        stepType: metadata.stepType,\n      });\n      instance._setLogger(logger);\n\n      // Validate params against the step's schema\n      const parseResult = metadata.schema.safeParse(params);\n      if (!parseResult.success) {\n        this.writeOutput(\n          StepOutputs.failed(\n            `Invalid params: ${parseResult.error.message}`,\n            \"INVALID_PARAMS\"\n          )\n        );\n        return;\n      }\n\n      const validatedParams = parseResult.data;\n\n      // Route to appropriate phase based on step kind and input context\n      const output = await this.routeExecution(\n        instance,\n        metadata,\n        validatedParams,\n        approvalContext,\n        pollingState\n      );\n\n      this.writeOutput(output);\n    } catch (error) {\n      const errorMessage =\n        error instanceof Error ? error.message : String(error);\n      this.writeOutput(StepOutputs.failed(errorMessage, \"EXECUTION_ERROR\"));\n    }\n  }\n\n  private async routeExecution(\n    instance: BaseStep<unknown>,\n    metadata: StepMetadata,\n    params: unknown,\n    approvalContext?: ApprovalContext,\n    pollingState?: Record<string, unknown>\n  ): Promise<StepOutput> {\n    const requiresApproval = metadata.requiresApproval;\n\n    if (metadata.stepKind === \"simple\") {\n      return this.routeSimpleStep(\n        instance as SimpleStep<unknown>,\n        params,\n        requiresApproval,\n        approvalContext\n      );\n    }\n\n    if (metadata.stepKind === \"polling\") {\n      return this.routePollingStep(\n        instance as PollingStep<unknown, Record<string, unknown>>,\n        params,\n        requiresApproval,\n        approvalContext,\n        pollingState\n      );\n    }\n\n    return StepOutputs.failed(\n      `Unknown step kind: ${metadata.stepKind}`,\n      \"UNKNOWN_STEP_KIND\"\n    );\n  }\n\n  /**\n   * Route SimpleStep execution based on approval state.\n   *\n   * | requiresApproval | approvalContext provided? | Action              |\n   * |------------------|---------------------------|---------------------|\n   * | No               | -                         | Call run(params)    |\n   * | Yes              | No                        | Call prepare(params)|\n   * | Yes              | Yes                       | Call run(params, approval) |\n   */\n  private async routeSimpleStep(\n    instance: SimpleStep<unknown>,\n    params: unknown,\n    requiresApproval: boolean,\n    approvalContext?: ApprovalContext\n  ): Promise<StepOutput> {\n    if (!requiresApproval) {\n      // No approval needed - just run\n      return instance.execute(params, undefined);\n    }\n\n    if (!approvalContext) {\n      // Needs approval but don't have it yet - call prepare\n      return instance.prepare(params);\n    }\n\n    // Have approval - run with approval context\n    return instance.execute(params, approvalContext);\n  }\n\n  /**\n   * Route PollingStep execution based on approval and polling state.\n   *\n   * | requiresApproval | approvalContext? | pollingState? | Action                        |\n   * |------------------|------------------|---------------|-------------------------------|\n   * | No               | -                | No            | Call trigger(params)          |\n   * | No               | -                | Yes           | Call poll(params, pollingState)|\n   * | Yes              | No               | No            | Call prepare(params)          |\n   * | Yes              | Yes              | No            | Call trigger(params, approval)|\n   * | Yes              | Yes              | Yes           | Call poll(params, pollingState)|\n   */\n  private async routePollingStep(\n    instance: PollingStep<unknown, Record<string, unknown>>,\n    params: unknown,\n    requiresApproval: boolean,\n    approvalContext?: ApprovalContext,\n    pollingState?: Record<string, unknown>\n  ): Promise<StepOutput> {\n    // If we have polling state, we're in the poll phase (regardless of approval)\n    if (pollingState) {\n      return instance.executePoll(params, pollingState);\n    }\n\n    if (!requiresApproval) {\n      // No approval needed - trigger directly\n      return instance.executeTrigger(params, undefined);\n    }\n\n    if (!approvalContext) {\n      // Needs approval but don't have it yet - call prepare\n      return instance.prepare(params);\n    }\n\n    // Have approval - trigger with approval context\n    return instance.executeTrigger(params, approvalContext);\n  }\n\n  private writeOutput(output: StepOutput): void {\n    const outputDir = path.dirname(this.outputPath);\n    if (!fs.existsSync(outputDir)) {\n      fs.mkdirSync(outputDir, { recursive: true });\n    }\n\n    fs.writeFileSync(this.outputPath, JSON.stringify(output, null, 2));\n  }\n}\n"]}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@devramps/sdk-typescript",
3
- "version": "0.1.2",
3
+ "version": "0.1.4",
4
4
  "description": "SDK for building custom deployment steps for DevRamps",
5
5
  "main": "dist/index.js",
6
6
  "module": "dist/index.js",
@@ -10,6 +10,9 @@
10
10
  "LICENSE",
11
11
  "README.md"
12
12
  ],
13
+ "bin": {
14
+ "create-step-registry": "./scripts/create-step-registry.js"
15
+ },
13
16
  "exports": {
14
17
  ".": {
15
18
  "import": "./dist/index.js",
@@ -0,0 +1,297 @@
1
+ #!/usr/bin/env node
2
+
3
+ const fs = require("fs");
4
+ const path = require("path");
5
+ const { execSync } = require("child_process");
6
+
7
+ // Get optional folder name from command line args, default to "step-registry"
8
+ const folderName = process.argv[2] || "step-registry";
9
+
10
+ // Define the base directory
11
+ const baseDir = path.join(process.cwd(), folderName);
12
+
13
+ // Create the folder
14
+ console.log(`Creating ${folderName} folder...`);
15
+ if (!fs.existsSync(baseDir)) {
16
+ fs.mkdirSync(baseDir);
17
+ }
18
+
19
+ // Initialize NPM project
20
+ console.log("Initializing NPM project...");
21
+ execSync("npm init -y", { cwd: baseDir, stdio: "inherit" });
22
+
23
+ // Install dependencies
24
+ console.log("Installing dependencies...");
25
+ execSync("npm install @devramps/sdk-typescript zod", {
26
+ cwd: baseDir,
27
+ stdio: "inherit",
28
+ });
29
+ execSync("npm install --save-dev @types/node jest ts-jest @types/jest", {
30
+ cwd: baseDir,
31
+ stdio: "inherit",
32
+ });
33
+
34
+ // Create src folder structure
35
+ const srcDir = path.join(baseDir, "src");
36
+ const stepsDir = path.join(srcDir, "steps");
37
+ console.log("Creating src/steps folder structure...");
38
+ if (!fs.existsSync(srcDir)) {
39
+ fs.mkdirSync(srcDir);
40
+ }
41
+ if (!fs.existsSync(stepsDir)) {
42
+ fs.mkdirSync(stepsDir, { recursive: true });
43
+ }
44
+
45
+ // Create tst folder structure
46
+ const tstDir = path.join(baseDir, "tst");
47
+ const tstStepsDir = path.join(tstDir, "steps");
48
+ console.log("Creating tst/steps folder structure...");
49
+ if (!fs.existsSync(tstDir)) {
50
+ fs.mkdirSync(tstDir);
51
+ }
52
+ if (!fs.existsSync(tstStepsDir)) {
53
+ fs.mkdirSync(tstStepsDir, { recursive: true });
54
+ }
55
+
56
+ // Create deployment-service.ts (dependency example)
57
+ console.log("Creating deployment-service.ts...");
58
+ const deploymentServiceContent = `export interface DeploymentService {
59
+ deploy(target: string, version?: string): Promise<{ deploymentId: string }>;
60
+ }
61
+
62
+ export class RealDeploymentService implements DeploymentService {
63
+ async deploy(target: string, version?: string): Promise<{ deploymentId: string }> {
64
+ // Simulate a deployment
65
+ console.log(\`Deploying to \${target}\${version ? \` version \${version}\` : ""}\`);
66
+ return { deploymentId: \`deploy-\${Date.now()}\` };
67
+ }
68
+ }
69
+ `;
70
+ fs.writeFileSync(path.join(srcDir, "deployment-service.ts"), deploymentServiceContent);
71
+
72
+ // Create my-first-step.ts
73
+ console.log("Creating my-first-step.ts...");
74
+ const myFirstStepContent = `import {
75
+ RunOutput,
76
+ SimpleStep,
77
+ Step,
78
+ StepOutputs,
79
+ } from "@devramps/sdk-typescript";
80
+ import z from "zod";
81
+ import { DeploymentService } from "../deployment-service";
82
+
83
+ const DeploySchema = z.object({
84
+ target: z.string().describe("A Target"),
85
+ version: z.string().optional(),
86
+ });
87
+
88
+ type DeployParams = z.infer<typeof DeploySchema>;
89
+
90
+ @Step({
91
+ name: "Custom Deployment Action",
92
+ type: "CUSTOM:SCRIPT:DEPLOY",
93
+ schema: DeploySchema,
94
+ })
95
+ export class MyFirstStep extends SimpleStep<DeployParams> {
96
+ constructor(private deploymentService: DeploymentService) {
97
+ super();
98
+ }
99
+
100
+ async run(params: DeployParams): Promise<RunOutput> {
101
+ this.logger.info(\`Deploying with params: \${JSON.stringify(params)}\`);
102
+
103
+ const result = await this.deploymentService.deploy(
104
+ params.target,
105
+ params.version
106
+ );
107
+
108
+ this.logger.info(\`Deployment completed: \${result.deploymentId}\`);
109
+
110
+ return StepOutputs.success({
111
+ deploymentId: result.deploymentId,
112
+ target: params.target,
113
+ version: params.version,
114
+ });
115
+ }
116
+ }
117
+ `;
118
+ fs.writeFileSync(path.join(stepsDir, "my-first-step.ts"), myFirstStepContent);
119
+
120
+ // Create src/index.ts
121
+ console.log("Creating src/index.ts...");
122
+ const indexContent = `import { StepRegistry } from "@devramps/sdk-typescript";
123
+ import { MyFirstStep } from "./steps/my-first-step";
124
+ import { RealDeploymentService } from "./deployment-service";
125
+
126
+ // Instantiate dependencies
127
+ const deploymentService = new RealDeploymentService();
128
+
129
+ // Instantiate steps with their dependencies
130
+ const myFirstStep = new MyFirstStep(deploymentService);
131
+
132
+ // Register step instances
133
+ StepRegistry.run([myFirstStep]);
134
+ `;
135
+ fs.writeFileSync(path.join(srcDir, "index.ts"), indexContent);
136
+
137
+ // Create tsconfig.json
138
+ console.log("Creating tsconfig.json...");
139
+ const tsconfigContent = `{
140
+ "compilerOptions": {
141
+ "target": "ES2022",
142
+ "module": "commonjs",
143
+ "lib": ["ES2022"],
144
+ "types": ["node", "jest"],
145
+ "declaration": true,
146
+ "strict": true,
147
+ "noImplicitAny": true,
148
+ "strictNullChecks": true,
149
+ "noImplicitThis": true,
150
+ "alwaysStrict": true,
151
+ "noUnusedLocals": false,
152
+ "noUnusedParameters": false,
153
+ "noImplicitReturns": true,
154
+ "noFallthroughCasesInSwitch": false,
155
+ "inlineSourceMap": true,
156
+ "inlineSources": true,
157
+ "experimentalDecorators": true,
158
+ "emitDecoratorMetadata": true,
159
+ "strictPropertyInitialization": false,
160
+ "esModuleInterop": true,
161
+ "skipLibCheck": true,
162
+ "outDir": "./dist",
163
+ "rootDir": "./src"
164
+ },
165
+ "include": ["src/**/*", "tst/**/*"],
166
+ "exclude": ["node_modules", "dist"]
167
+ }
168
+ `;
169
+ fs.writeFileSync(path.join(baseDir, "tsconfig.json"), tsconfigContent);
170
+
171
+ // Create jest.config.js
172
+ console.log("Creating jest.config.js...");
173
+ const jestConfigContent = `module.exports = {
174
+ preset: 'ts-jest',
175
+ testEnvironment: 'node',
176
+ roots: ['<rootDir>/tst'],
177
+ testMatch: ['**/*.test.ts'],
178
+ moduleFileExtensions: ['ts', 'tsx', 'js', 'jsx', 'json', 'node'],
179
+ collectCoverageFrom: [
180
+ 'src/**/*.ts',
181
+ '!src/**/*.d.ts',
182
+ ],
183
+ };
184
+ `;
185
+ fs.writeFileSync(path.join(baseDir, "jest.config.js"), jestConfigContent);
186
+
187
+ // Create test file for my-first-step
188
+ console.log("Creating tst/steps/my-first-step.test.ts...");
189
+ const testContent = `import { MyFirstStep } from "../../src/steps/my-first-step";
190
+ import { DeploymentService } from "../../src/deployment-service";
191
+ import { StepOutputs } from "@devramps/sdk-typescript";
192
+
193
+ // Mock implementation of DeploymentService for testing
194
+ class MockDeploymentService implements DeploymentService {
195
+ async deploy(target: string, version?: string): Promise<{ deploymentId: string }> {
196
+ return { deploymentId: \`mock-deploy-\${target}-\${version || "latest"}\` };
197
+ }
198
+ }
199
+
200
+ describe("MyFirstStep", () => {
201
+ let mockDeploymentService: MockDeploymentService;
202
+ let step: MyFirstStep;
203
+
204
+ beforeEach(() => {
205
+ // Create mock dependency
206
+ mockDeploymentService = new MockDeploymentService();
207
+ // Inject mock into step
208
+ step = new MyFirstStep(mockDeploymentService);
209
+ });
210
+
211
+ it("should run successfully with valid parameters", async () => {
212
+ const params = {
213
+ target: "production",
214
+ version: "1.0.0",
215
+ };
216
+
217
+ const result = await step.run(params);
218
+
219
+ expect(result.status).toBe("SUCCESS");
220
+ if (result.status === "SUCCESS") {
221
+ expect(result.data).toEqual({
222
+ deploymentId: "mock-deploy-production-1.0.0",
223
+ target: "production",
224
+ version: "1.0.0",
225
+ });
226
+ }
227
+ });
228
+
229
+ it("should run successfully without optional version parameter", async () => {
230
+ const params = {
231
+ target: "staging",
232
+ };
233
+
234
+ const result = await step.run(params);
235
+
236
+ expect(result.status).toBe("SUCCESS");
237
+ if (result.status === "SUCCESS") {
238
+ expect(result.data).toEqual({
239
+ deploymentId: "mock-deploy-staging-latest",
240
+ target: "staging",
241
+ version: undefined,
242
+ });
243
+ }
244
+ });
245
+
246
+ it("should call deployment service with correct parameters", async () => {
247
+ const deploySpy = jest.spyOn(mockDeploymentService, "deploy");
248
+ const params = {
249
+ target: "development",
250
+ version: "2.0.0",
251
+ };
252
+
253
+ await step.run(params);
254
+
255
+ expect(deploySpy).toHaveBeenCalledWith("development", "2.0.0");
256
+ });
257
+
258
+ it("should log deployment information", async () => {
259
+ // Access the logger through type assertion to test logging behavior
260
+ const logSpy = jest.spyOn((step as any).logger, "info");
261
+ const params = {
262
+ target: "production",
263
+ version: "3.0.0",
264
+ };
265
+
266
+ await step.run(params);
267
+
268
+ expect(logSpy).toHaveBeenCalledWith(
269
+ \`Deploying with params: \${JSON.stringify(params)}\`
270
+ );
271
+ expect(logSpy).toHaveBeenCalledWith(
272
+ "Deployment completed: mock-deploy-production-3.0.0"
273
+ );
274
+ });
275
+ });
276
+ `;
277
+ fs.writeFileSync(path.join(tstStepsDir, "my-first-step.test.ts"), testContent);
278
+
279
+ // Update package.json to add build and start scripts
280
+ console.log('Adding build and start scripts to package.json...');
281
+ const packageJsonPath = path.join(baseDir, 'package.json');
282
+ const packageJson = JSON.parse(fs.readFileSync(packageJsonPath, 'utf8'));
283
+ packageJson.scripts = {
284
+ ...packageJson.scripts,
285
+ "build-step-registry": "tsc -p tsconfig.json",
286
+ "start-step-registry": "node ./dist/index.js",
287
+ "test": "jest",
288
+ "test:watch": "jest --watch",
289
+ "test:coverage": "jest --coverage"
290
+ };
291
+ fs.writeFileSync(packageJsonPath, JSON.stringify(packageJson, null, 2));
292
+
293
+ console.log(`\nStep registry created successfully in '${folderName}'!`);
294
+ console.log(`\nNext steps:`);
295
+ console.log(` cd ${folderName}`);
296
+ console.log(` npm run build-step-registry`);
297
+ console.log(` npm run test`);