@devramps/sdk-typescript 0.1.3 → 0.1.5
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/base/base-step.d.ts +9 -12
- package/dist/base/base-step.js +4 -1
- package/dist/base/polling-step.d.ts +21 -4
- package/dist/base/polling-step.js +22 -5
- package/dist/base/simple-step.d.ts +16 -1
- package/dist/base/simple-step.js +17 -2
- package/dist/decorators/step.d.ts +9 -1
- package/dist/decorators/step.js +19 -6
- package/dist/index.d.ts +1 -1
- package/dist/index.js +1 -1
- package/dist/registry/step-registry.d.ts +6 -3
- package/dist/registry/step-registry.js +19 -19
- package/package.json +4 -1
- package/scripts/create-step-registry.js +300 -0
package/dist/base/base-step.d.ts
CHANGED
|
@@ -1,18 +1,21 @@
|
|
|
1
|
-
import type { ZodType
|
|
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
4
|
import type { ZodStandardJSONSchemaPayload } from "zod/v4/core";
|
|
5
5
|
export type StepKind = "simple" | "polling";
|
|
6
|
+
export interface StepData<S extends ZodType = ZodType> {
|
|
7
|
+
schema: S;
|
|
8
|
+
stepType: string;
|
|
9
|
+
stepKind: StepKind;
|
|
10
|
+
requiresApproval: boolean;
|
|
11
|
+
}
|
|
6
12
|
export interface StepMetadata<S extends ZodType = ZodType> {
|
|
7
13
|
name?: string;
|
|
8
14
|
shortDescription?: string;
|
|
9
15
|
longDescription?: string;
|
|
10
16
|
yamlExample?: string;
|
|
11
|
-
schema: S;
|
|
12
|
-
stepKind: StepKind;
|
|
13
|
-
requiresApproval: boolean;
|
|
14
17
|
stepType: string;
|
|
15
|
-
|
|
18
|
+
paramsJsonSchema: ZodStandardJSONSchemaPayload<S>;
|
|
16
19
|
documentationUrl?: string;
|
|
17
20
|
}
|
|
18
21
|
/**
|
|
@@ -31,6 +34,7 @@ export declare abstract class BaseStep<TParams = unknown> {
|
|
|
31
34
|
* Implemented by the @Step decorator.
|
|
32
35
|
*/
|
|
33
36
|
getMetadata(): StepMetadata;
|
|
37
|
+
getData(): StepData;
|
|
34
38
|
/**
|
|
35
39
|
* Optional prepare method for approval flow.
|
|
36
40
|
* Override this to require approval before execution.
|
|
@@ -38,10 +42,3 @@ export declare abstract class BaseStep<TParams = unknown> {
|
|
|
38
42
|
*/
|
|
39
43
|
prepare(_params: TParams): Promise<PrepareOutput>;
|
|
40
44
|
}
|
|
41
|
-
/**
|
|
42
|
-
* Type representing a class constructor that produces a BaseStep instance
|
|
43
|
-
* with the getMetadata method available.
|
|
44
|
-
*/
|
|
45
|
-
export type StepClass<S extends ZodType = ZodType> = new (...args: any[]) => BaseStep<z.infer<S>> & {
|
|
46
|
-
getMetadata(): StepMetadata<S>;
|
|
47
|
-
};
|
package/dist/base/base-step.js
CHANGED
|
@@ -23,6 +23,9 @@ class BaseStep {
|
|
|
23
23
|
getMetadata() {
|
|
24
24
|
throw new Error("getMetadata not implemented. Did you forget to add the @Step decorator?");
|
|
25
25
|
}
|
|
26
|
+
getData() {
|
|
27
|
+
throw new Error("getData not implemented. Did you forget to add the @Step decorator?");
|
|
28
|
+
}
|
|
26
29
|
/**
|
|
27
30
|
* Optional prepare method for approval flow.
|
|
28
31
|
* Override this to require approval before execution.
|
|
@@ -35,4 +38,4 @@ class BaseStep {
|
|
|
35
38
|
}
|
|
36
39
|
}
|
|
37
40
|
exports.BaseStep = BaseStep;
|
|
38
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
41
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYmFzZS1zdGVwLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vc3JjL2Jhc2UvYmFzZS1zdGVwLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7OztBQUVBLHdEQUFvRDtBQUVwRCx1REFBb0Q7QUFzQnBEOzs7R0FHRztBQUNILE1BQXNCLFFBQVE7SUFDbEIsTUFBTSxHQUFlLElBQUksd0JBQVUsRUFBRSxDQUFDO0lBRWhEOzs7T0FHRztJQUNILFVBQVUsQ0FBQyxNQUFrQjtRQUMzQixJQUFJLENBQUMsTUFBTSxHQUFHLE1BQU0sQ0FBQztJQUN2QixDQUFDO0lBRUQ7OztPQUdHO0lBQ0gsV0FBVztRQUNULE1BQU0sSUFBSSxLQUFLLENBQ2IseUVBQXlFLENBQzFFLENBQUM7SUFDSixDQUFDO0lBRUQsT0FBTztRQUNMLE1BQU0sSUFBSSxLQUFLLENBQ2IscUVBQXFFLENBQ3RFLENBQUM7SUFDSixDQUFDO0lBRUQ7Ozs7T0FJRztJQUNILEtBQUssQ0FBQyxPQUFPLENBQUMsT0FBZ0I7UUFDNUIsOERBQThEO1FBQzlELGlFQUFpRTtRQUNqRSxPQUFPLHlCQUFXLENBQUMsTUFBTSxDQUN2QixzQ0FBc0MsRUFDdEMseUJBQXlCLENBQzFCLENBQUM7SUFDSixDQUFDO0NBQ0Y7QUF4Q0QsNEJBd0NDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHR5cGUgeyBab2RUeXBlIH0gZnJvbSBcInpvZFwiO1xuaW1wb3J0IHR5cGUgeyBTdGVwTG9nZ2VyIH0gZnJvbSBcIi4uL2xvZ2dpbmcvc3RlcC1sb2dnZXJcIjtcbmltcG9ydCB7IE5vT3BMb2dnZXIgfSBmcm9tIFwiLi4vbG9nZ2luZy9zdGVwLWxvZ2dlclwiO1xuaW1wb3J0IHR5cGUgeyBQcmVwYXJlT3V0cHV0IH0gZnJvbSBcIi4uL291dHB1dC9zdGVwLW91dHB1dFwiO1xuaW1wb3J0IHsgU3RlcE91dHB1dHMgfSBmcm9tIFwiLi4vb3V0cHV0L3N0ZXAtb3V0cHV0XCI7XG5pbXBvcnQgdHlwZSB7IFpvZFN0YW5kYXJkSlNPTlNjaGVtYVBheWxvYWQgfSBmcm9tIFwiem9kL3Y0L2NvcmVcIjtcblxuZXhwb3J0IHR5cGUgU3RlcEtpbmQgPSBcInNpbXBsZVwiIHwgXCJwb2xsaW5nXCI7XG5cbmV4cG9ydCBpbnRlcmZhY2UgU3RlcERhdGE8UyBleHRlbmRzIFpvZFR5cGUgPSBab2RUeXBlPiB7XG4gIHNjaGVtYTogUztcbiAgc3RlcFR5cGU6IHN0cmluZztcbiAgc3RlcEtpbmQ6IFN0ZXBLaW5kO1xuICByZXF1aXJlc0FwcHJvdmFsOiBib29sZWFuO1xufVxuXG5leHBvcnQgaW50ZXJmYWNlIFN0ZXBNZXRhZGF0YTxTIGV4dGVuZHMgWm9kVHlwZSA9IFpvZFR5cGU+IHtcbiAgbmFtZT86IHN0cmluZztcbiAgc2hvcnREZXNjcmlwdGlvbj86IHN0cmluZztcbiAgbG9uZ0Rlc2NyaXB0aW9uPzogc3RyaW5nO1xuICB5YW1sRXhhbXBsZT86IHN0cmluZztcbiAgc3RlcFR5cGU6IHN0cmluZztcbiAgcGFyYW1zSnNvblNjaGVtYTogWm9kU3RhbmRhcmRKU09OU2NoZW1hUGF5bG9hZDxTPjtcbiAgZG9jdW1lbnRhdGlvblVybD86IHN0cmluZztcbn1cblxuLyoqXG4gKiBCYXNlIGNsYXNzIGZvciBhbGwgc3RlcHMuXG4gKiBQcm92aWRlcyBsb2dnaW5nIGFuZCBjb21tb24gZnVuY3Rpb25hbGl0eS5cbiAqL1xuZXhwb3J0IGFic3RyYWN0IGNsYXNzIEJhc2VTdGVwPFRQYXJhbXMgPSB1bmtub3duPiB7XG4gIHByb3RlY3RlZCBsb2dnZXI6IFN0ZXBMb2dnZXIgPSBuZXcgTm9PcExvZ2dlcigpO1xuXG4gIC8qKlxuICAgKiBDYWxsZWQgYnkgdGhlIHJlZ2lzdHJ5IHRvIGluamVjdCB0aGUgbG9nZ2VyIGJlZm9yZSBleGVjdXRpb24uXG4gICAqIEBpbnRlcm5hbFxuICAgKi9cbiAgX3NldExvZ2dlcihsb2dnZXI6IFN0ZXBMb2dnZXIpOiB2b2lkIHtcbiAgICB0aGlzLmxvZ2dlciA9IGxvZ2dlcjtcbiAgfVxuXG4gIC8qKlxuICAgKiBSZXR1cm5zIG1ldGFkYXRhIGFib3V0IHRoaXMgc3RlcC5cbiAgICogSW1wbGVtZW50ZWQgYnkgdGhlIEBTdGVwIGRlY29yYXRvci5cbiAgICovXG4gIGdldE1ldGFkYXRhKCk6IFN0ZXBNZXRhZGF0YSB7XG4gICAgdGhyb3cgbmV3IEVycm9yKFxuICAgICAgXCJnZXRNZXRhZGF0YSBub3QgaW1wbGVtZW50ZWQuIERpZCB5b3UgZm9yZ2V0IHRvIGFkZCB0aGUgQFN0ZXAgZGVjb3JhdG9yP1wiXG4gICAgKTtcbiAgfVxuXG4gIGdldERhdGEoKTogU3RlcERhdGEge1xuICAgIHRocm93IG5ldyBFcnJvcihcbiAgICAgIFwiZ2V0RGF0YSBub3QgaW1wbGVtZW50ZWQuIERpZCB5b3UgZm9yZ2V0IHRvIGFkZCB0aGUgQFN0ZXAgZGVjb3JhdG9yP1wiXG4gICAgKTtcbiAgfVxuXG4gIC8qKlxuICAgKiBPcHRpb25hbCBwcmVwYXJlIG1ldGhvZCBmb3IgYXBwcm92YWwgZmxvdy5cbiAgICogT3ZlcnJpZGUgdGhpcyB0byByZXF1aXJlIGFwcHJvdmFsIGJlZm9yZSBleGVjdXRpb24uXG4gICAqIElmIG5vdCBvdmVycmlkZGVuLCB0aGUgc3RlcCB3aWxsIG5vdCByZXF1aXJlIGFwcHJvdmFsLlxuICAgKi9cbiAgYXN5bmMgcHJlcGFyZShfcGFyYW1zOiBUUGFyYW1zKTogUHJvbWlzZTxQcmVwYXJlT3V0cHV0PiB7XG4gICAgLy8gRGVmYXVsdDogbm8gYXBwcm92YWwgcmVxdWlyZWQgLSB0aGlzIHNob3VsZCBuZXZlciBiZSBjYWxsZWRcbiAgICAvLyBUaGUgcmVnaXN0cnkgY2hlY2tzIGlmIHByZXBhcmUgaXMgb3ZlcnJpZGRlbiBiZWZvcmUgY2FsbGluZyBpdFxuICAgIHJldHVybiBTdGVwT3V0cHV0cy5mYWlsZWQoXG4gICAgICBcInByZXBhcmUoKSBjYWxsZWQgYnV0IG5vdCBpbXBsZW1lbnRlZFwiLFxuICAgICAgXCJQUkVQQVJFX05PVF9JTVBMRU1FTlRFRFwiXG4gICAgKTtcbiAgfVxufVxuIl19
|
|
@@ -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
|
|
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
|
|
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
|
|
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
|
-
*
|
|
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
|
|
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
|
|
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
|
|
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
|
-
*
|
|
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,
|
|
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
|
-
*
|
|
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
|
* }
|
package/dist/base/simple-step.js
CHANGED
|
@@ -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
|
-
*
|
|
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,
|
|
67
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic2ltcGxlLXN0ZXAuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvYmFzZS9zaW1wbGUtc3RlcC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7QUFDQSwyQ0FBdUM7QUFHdkM7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7OztHQWdERztBQUNILE1BQXNCLFVBQW9CLFNBQVEsb0JBQWlCO0lBUWpFOzs7T0FHRztJQUNILEtBQUssQ0FBQyxPQUFPLENBQUMsTUFBZSxFQUFFLFFBQTBCO1FBQ3ZELE9BQU8sSUFBSSxDQUFDLEdBQUcsQ0FBQyxNQUFNLEVBQUUsUUFBUSxDQUFDLENBQUM7SUFDcEMsQ0FBQztJQUVELFdBQVc7UUFDVCxNQUFNLElBQUksS0FBSyxDQUNiLHlFQUF5RSxDQUMxRSxDQUFDO0lBQ0osQ0FBQztDQUNGO0FBckJELGdDQXFCQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB0eXBlIHsgU3RlcE1ldGFkYXRhIH0gZnJvbSBcIi4vYmFzZS1zdGVwXCI7XG5pbXBvcnQgeyBCYXNlU3RlcCB9IGZyb20gXCIuL2Jhc2Utc3RlcFwiO1xuaW1wb3J0IHR5cGUgeyBBcHByb3ZhbENvbnRleHQsIFJ1bk91dHB1dCwgU3RlcE91dHB1dCB9IGZyb20gXCIuLi9vdXRwdXQvc3RlcC1vdXRwdXRcIjtcblxuLyoqXG4gKiBBIHNpbXBsZSBzdGVwIHRoYXQgcnVucyBvbmNlIHdpdGggdGhlIHByb3ZpZGVkIHBhcmFtcy5cbiAqXG4gKiBVc2VycyBleHRlbmQgdGhpcyBjbGFzcyBhbmQgaW1wbGVtZW50IHRoZSBgcnVuYCBtZXRob2QuXG4gKiBPcHRpb25hbGx5IG92ZXJyaWRlIGBwcmVwYXJlYCB0byByZXF1aXJlIGFwcHJvdmFsIGJlZm9yZSBleGVjdXRpb24uXG4gKlxuICogU3RlcHMgY2FuIGFjY2VwdCBjb25zdHJ1Y3RvciBwYXJhbWV0ZXJzIGZvciBkZXBlbmRlbmN5IGluamVjdGlvbi5cbiAqXG4gKiBAZXhhbXBsZVxuICogYGBgdHlwZXNjcmlwdFxuICogQFN0ZXAoeyBuYW1lOiBcIkRlcGxveVwiLCB0eXBlOiBcImRlcGxveVwiLCBzY2hlbWE6IGRlcGxveVNjaGVtYSB9KVxuICogY2xhc3MgRGVwbG95U3RlcCBleHRlbmRzIFNpbXBsZVN0ZXA8RGVwbG95UGFyYW1zPiB7XG4gKiAgIGNvbnN0cnVjdG9yKHByaXZhdGUgZGVwbG95U2VydmljZTogRGVwbG95U2VydmljZSkge1xuICogICAgIHN1cGVyKCk7XG4gKiAgIH1cbiAqXG4gKiAgIGFzeW5jIHJ1bihwYXJhbXM6IERlcGxveVBhcmFtcyk6IFByb21pc2U8UnVuT3V0cHV0PiB7XG4gKiAgICAgdGhpcy5sb2dnZXIuaW5mbyhcIkRlcGxveWluZ1wiLCB7IHRhcmdldDogcGFyYW1zLnRhcmdldCB9KTtcbiAqICAgICBjb25zdCByZXN1bHQgPSBhd2FpdCB0aGlzLmRlcGxveVNlcnZpY2UuZGVwbG95KHBhcmFtcy50YXJnZXQpO1xuICogICAgIHJldHVybiBTdGVwT3V0cHV0cy5zdWNjZXNzKHsgZGVwbG95bWVudElkOiByZXN1bHQuaWQgfSk7XG4gKiAgIH1cbiAqIH1cbiAqXG4gKiAvLyBSZWdpc3RlciB3aXRoIGRlcGVuZGVuY3kgaW5qZWN0ZWRcbiAqIFN0ZXBSZWdpc3RyeS5ydW4oW25ldyBEZXBsb3lTdGVwKGRlcGxveVNlcnZpY2UpXSk7XG4gKiBgYGBcbiAqXG4gKiBAZXhhbXBsZSBXaXRoIGFwcHJvdmFsXG4gKiBgYGB0eXBlc2NyaXB0XG4gKiBAU3RlcCh7IG5hbWU6IFwiRGVsZXRlIFVzZXJcIiwgdHlwZTogXCJkZWxldGUtdXNlclwiLCBzY2hlbWE6IGRlbGV0ZVVzZXJTY2hlbWEgfSlcbiAqIGNsYXNzIERlbGV0ZVVzZXJTdGVwIGV4dGVuZHMgU2ltcGxlU3RlcDxEZWxldGVVc2VyUGFyYW1zPiB7XG4gKiAgIGNvbnN0cnVjdG9yKHByaXZhdGUgdXNlclNlcnZpY2U6IFVzZXJTZXJ2aWNlKSB7XG4gKiAgICAgc3VwZXIoKTtcbiAqICAgfVxuICpcbiAqICAgYXN5bmMgcHJlcGFyZShwYXJhbXM6IERlbGV0ZVVzZXJQYXJhbXMpOiBQcm9taXNlPFByZXBhcmVPdXRwdXQ+IHtcbiAqICAgICByZXR1cm4gU3RlcE91dHB1dHMuYXBwcm92YWxSZXF1aXJlZCh7XG4gKiAgICAgICBtZXNzYWdlOiBgRGVsZXRlIHVzZXIgJHtwYXJhbXMudXNlcklkfT9gLFxuICogICAgIH0pO1xuICogICB9XG4gKlxuICogICBhc3luYyBydW4ocGFyYW1zOiBEZWxldGVVc2VyUGFyYW1zLCBhcHByb3ZhbDogQXBwcm92YWxDb250ZXh0KTogUHJvbWlzZTxSdW5PdXRwdXQ+IHtcbiAqICAgICB0aGlzLmxvZ2dlci5pbmZvKFwiRGVsZXRpbmcgdXNlclwiLCB7IGFwcHJvdmVkQnk6IGFwcHJvdmFsLmFwcHJvdmVySWQgfSk7XG4gKiAgICAgYXdhaXQgdGhpcy51c2VyU2VydmljZS5kZWxldGUocGFyYW1zLnVzZXJJZCk7XG4gKiAgICAgcmV0dXJuIFN0ZXBPdXRwdXRzLnN1Y2Nlc3MoKTtcbiAqICAgfVxuICogfVxuICogYGBgXG4gKi9cbmV4cG9ydCBhYnN0cmFjdCBjbGFzcyBTaW1wbGVTdGVwPFRQYXJhbXM+IGV4dGVuZHMgQmFzZVN0ZXA8VFBhcmFtcz4ge1xuICAvKipcbiAgICogRXhlY3V0ZSB0aGUgc3RlcCBsb2dpYy5cbiAgICogQHBhcmFtIHBhcmFtcyAtIFRoZSB2YWxpZGF0ZWQgaW5wdXQgcGFyYW1ldGVyc1xuICAgKiBAcGFyYW0gYXBwcm92YWwgLSBBcHByb3ZhbCBjb250ZXh0IGlmIHRoaXMgc3RlcCByZXF1aXJlcyBhcHByb3ZhbCAoaGFzIHByZXBhcmUgbWV0aG9kKVxuICAgKi9cbiAgYWJzdHJhY3QgcnVuKHBhcmFtczogVFBhcmFtcywgYXBwcm92YWw/OiBBcHByb3ZhbENvbnRleHQpOiBQcm9taXNlPFJ1bk91dHB1dD47XG5cbiAgLyoqXG4gICAqIENhbGxlZCBieSB0aGUgcmVnaXN0cnkgdG8gZXhlY3V0ZSB0aGlzIHN0ZXAuXG4gICAqIEBpbnRlcm5hbFxuICAgKi9cbiAgYXN5bmMgZXhlY3V0ZShwYXJhbXM6IFRQYXJhbXMsIGFwcHJvdmFsPzogQXBwcm92YWxDb250ZXh0KTogUHJvbWlzZTxTdGVwT3V0cHV0PiB7XG4gICAgcmV0dXJuIHRoaXMucnVuKHBhcmFtcywgYXBwcm92YWwpO1xuICB9XG5cbiAgZ2V0TWV0YWRhdGEoKTogU3RlcE1ldGFkYXRhIHtcbiAgICB0aHJvdyBuZXcgRXJyb3IoXG4gICAgICBcImdldE1ldGFkYXRhIG5vdCBpbXBsZW1lbnRlZC4gRGlkIHlvdSBmb3JnZXQgdG8gYWRkIHRoZSBAU3RlcCBkZWNvcmF0b3I/XCJcbiAgICApO1xuICB9XG59XG4iXX0=
|
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
import { type ZodType, z } from "zod";
|
|
2
2
|
import { BaseStep } from "../base/base-step";
|
|
3
|
-
export interface StepConfig<S extends ZodType> {
|
|
3
|
+
export interface StepConfig<S extends ZodType = ZodType> {
|
|
4
4
|
name?: string;
|
|
5
5
|
type: string;
|
|
6
6
|
shortDescription?: string;
|
|
@@ -18,10 +18,18 @@ export interface StepConfig<S extends ZodType> {
|
|
|
18
18
|
* ```typescript
|
|
19
19
|
* @Step({ name: "My Step", type: "my-step", schema: mySchema })
|
|
20
20
|
* class MyStep extends SimpleStep<MyParams> {
|
|
21
|
+
* constructor(private apiClient: ApiClient) {
|
|
22
|
+
* super();
|
|
23
|
+
* }
|
|
24
|
+
*
|
|
21
25
|
* async run(params: MyParams): Promise<RunOutput> {
|
|
26
|
+
* await this.apiClient.doSomething();
|
|
22
27
|
* return StepOutputs.success();
|
|
23
28
|
* }
|
|
24
29
|
* }
|
|
30
|
+
*
|
|
31
|
+
* // Register with dependencies injected
|
|
32
|
+
* StepRegistry.run([new MyStep(apiClient)]);
|
|
25
33
|
* ```
|
|
26
34
|
*/
|
|
27
35
|
export declare function Step<S extends ZodType>(config: StepConfig<S>): <T extends new (...args: any[]) => BaseStep<z.infer<S>>>(Base: T) => T;
|
package/dist/decorators/step.js
CHANGED
|
@@ -13,16 +13,34 @@ const base_step_1 = require("../base/base-step");
|
|
|
13
13
|
* ```typescript
|
|
14
14
|
* @Step({ name: "My Step", type: "my-step", schema: mySchema })
|
|
15
15
|
* class MyStep extends SimpleStep<MyParams> {
|
|
16
|
+
* constructor(private apiClient: ApiClient) {
|
|
17
|
+
* super();
|
|
18
|
+
* }
|
|
19
|
+
*
|
|
16
20
|
* async run(params: MyParams): Promise<RunOutput> {
|
|
21
|
+
* await this.apiClient.doSomething();
|
|
17
22
|
* return StepOutputs.success();
|
|
18
23
|
* }
|
|
19
24
|
* }
|
|
25
|
+
*
|
|
26
|
+
* // Register with dependencies injected
|
|
27
|
+
* StepRegistry.run([new MyStep(apiClient)]);
|
|
20
28
|
* ```
|
|
21
29
|
*/
|
|
22
30
|
function Step(config) {
|
|
23
31
|
return function (Base) {
|
|
24
32
|
const Enhanced = class extends Base {
|
|
25
33
|
getMetadata() {
|
|
34
|
+
return {
|
|
35
|
+
name: config.name,
|
|
36
|
+
stepType: config.type,
|
|
37
|
+
shortDescription: config.shortDescription,
|
|
38
|
+
longDescription: config.longDescription,
|
|
39
|
+
yamlExample: config.yamlExample,
|
|
40
|
+
paramsJsonSchema: zod_1.z.toJSONSchema(config.schema),
|
|
41
|
+
};
|
|
42
|
+
}
|
|
43
|
+
getData() {
|
|
26
44
|
// Detect step kind by checking for PollingStep methods
|
|
27
45
|
const hasPollingMethods = "trigger" in this &&
|
|
28
46
|
typeof this.trigger === "function" &&
|
|
@@ -32,13 +50,8 @@ function Step(config) {
|
|
|
32
50
|
// Check if prepare is overridden (not the default BaseStep.prepare)
|
|
33
51
|
const requiresApproval = this.prepare !== base_step_1.BaseStep.prototype.prepare;
|
|
34
52
|
return {
|
|
35
|
-
name: config.name,
|
|
36
53
|
stepType: config.type,
|
|
37
|
-
shortDescription: config.shortDescription,
|
|
38
|
-
longDescription: config.longDescription,
|
|
39
|
-
yamlExample: config.yamlExample,
|
|
40
54
|
schema: config.schema,
|
|
41
|
-
jsonSchema: zod_1.z.toJSONSchema(config.schema),
|
|
42
55
|
stepKind,
|
|
43
56
|
requiresApproval,
|
|
44
57
|
};
|
|
@@ -47,4 +60,4 @@ function Step(config) {
|
|
|
47
60
|
return Enhanced;
|
|
48
61
|
};
|
|
49
62
|
}
|
|
50
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
63
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic3RlcC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9kZWNvcmF0b3JzL3N0ZXAudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7QUFxQ0Esb0JBd0NDO0FBN0VELDZCQUFzQztBQUV0QyxpREFBNkM7QUFXN0M7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7O0dBdUJHO0FBQ0gsU0FBZ0IsSUFBSSxDQUFvQixNQUFxQjtJQUMzRCxPQUFPLFVBQ0wsSUFBTztRQUVQLE1BQU0sUUFBUSxHQUFHLEtBQU0sU0FBUSxJQUFJO1lBQ3hCLFdBQVc7Z0JBQ2xCLE9BQU87b0JBQ0wsSUFBSSxFQUFFLE1BQU0sQ0FBQyxJQUFJO29CQUNqQixRQUFRLEVBQUUsTUFBTSxDQUFDLElBQUk7b0JBQ3JCLGdCQUFnQixFQUFFLE1BQU0sQ0FBQyxnQkFBZ0I7b0JBQ3pDLGVBQWUsRUFBRSxNQUFNLENBQUMsZUFBZTtvQkFDdkMsV0FBVyxFQUFFLE1BQU0sQ0FBQyxXQUFXO29CQUMvQixnQkFBZ0IsRUFBRSxPQUFDLENBQUMsWUFBWSxDQUFDLE1BQU0sQ0FBQyxNQUFNLENBQUM7aUJBQ2hELENBQUM7WUFDSixDQUFDO1lBRVEsT0FBTztnQkFDZCx1REFBdUQ7Z0JBQ3ZELE1BQU0saUJBQWlCLEdBQ3JCLFNBQVMsSUFBSSxJQUFJO29CQUNqQixPQUFRLElBQVksQ0FBQyxPQUFPLEtBQUssVUFBVTtvQkFDM0MsTUFBTSxJQUFJLElBQUk7b0JBQ2QsT0FBUSxJQUFZLENBQUMsSUFBSSxLQUFLLFVBQVUsQ0FBQztnQkFFM0MsTUFBTSxRQUFRLEdBQWEsaUJBQWlCLENBQUMsQ0FBQyxDQUFDLFNBQVMsQ0FBQyxDQUFDLENBQUMsUUFBUSxDQUFDO2dCQUVwRSxvRUFBb0U7Z0JBQ3BFLE1BQU0sZ0JBQWdCLEdBQUcsSUFBSSxDQUFDLE9BQU8sS0FBSyxvQkFBUSxDQUFDLFNBQVMsQ0FBQyxPQUFPLENBQUM7Z0JBRXJFLE9BQU87b0JBQ0wsUUFBUSxFQUFFLE1BQU0sQ0FBQyxJQUFJO29CQUNyQixNQUFNLEVBQUUsTUFBTSxDQUFDLE1BQU07b0JBQ3JCLFFBQVE7b0JBQ1IsZ0JBQWdCO2lCQUNqQixDQUFDO1lBQ0osQ0FBQztTQUNGLENBQUM7UUFFRixPQUFPLFFBQWEsQ0FBQztJQUN2QixDQUFDLENBQUM7QUFDSixDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgdHlwZSBab2RUeXBlLCB6IH0gZnJvbSBcInpvZFwiO1xuaW1wb3J0IHR5cGUgeyBTdGVwRGF0YSwgU3RlcEtpbmQsIFN0ZXBNZXRhZGF0YSB9IGZyb20gXCIuLi9iYXNlL2Jhc2Utc3RlcFwiO1xuaW1wb3J0IHsgQmFzZVN0ZXAgfSBmcm9tIFwiLi4vYmFzZS9iYXNlLXN0ZXBcIjtcblxuZXhwb3J0IGludGVyZmFjZSBTdGVwQ29uZmlnPFMgZXh0ZW5kcyBab2RUeXBlID0gWm9kVHlwZT4ge1xuICBuYW1lPzogc3RyaW5nO1xuICB0eXBlOiBzdHJpbmc7XG4gIHNob3J0RGVzY3JpcHRpb24/OiBzdHJpbmc7XG4gIGxvbmdEZXNjcmlwdGlvbj86IHN0cmluZztcbiAgeWFtbEV4YW1wbGU/OiBzdHJpbmc7XG4gIHNjaGVtYTogUztcbn1cblxuLyoqXG4gKiBEZWNvcmF0b3IgdGhhdCBhZGRzIG1ldGFkYXRhIHRvIGEgc3RlcCBjbGFzcy5cbiAqIFRoaXMgaXMgcmVxdWlyZWQgZm9yIGFsbCBzdGVwcyB0byBiZSByZWdpc3RlcmVkIHdpdGggU3RlcFJlZ2lzdHJ5LlxuICpcbiAqIFRoZSBkZWNvcmF0ZWQgY2xhc3MgbXVzdCBleHRlbmQgU2ltcGxlU3RlcCBvciBQb2xsaW5nU3RlcC5cbiAqXG4gKiBAZXhhbXBsZVxuICogYGBgdHlwZXNjcmlwdFxuICogQFN0ZXAoeyBuYW1lOiBcIk15IFN0ZXBcIiwgdHlwZTogXCJteS1zdGVwXCIsIHNjaGVtYTogbXlTY2hlbWEgfSlcbiAqIGNsYXNzIE15U3RlcCBleHRlbmRzIFNpbXBsZVN0ZXA8TXlQYXJhbXM+IHtcbiAqICAgY29uc3RydWN0b3IocHJpdmF0ZSBhcGlDbGllbnQ6IEFwaUNsaWVudCkge1xuICogICAgIHN1cGVyKCk7XG4gKiAgIH1cbiAqXG4gKiAgIGFzeW5jIHJ1bihwYXJhbXM6IE15UGFyYW1zKTogUHJvbWlzZTxSdW5PdXRwdXQ+IHtcbiAqICAgICBhd2FpdCB0aGlzLmFwaUNsaWVudC5kb1NvbWV0aGluZygpO1xuICogICAgIHJldHVybiBTdGVwT3V0cHV0cy5zdWNjZXNzKCk7XG4gKiAgIH1cbiAqIH1cbiAqXG4gKiAvLyBSZWdpc3RlciB3aXRoIGRlcGVuZGVuY2llcyBpbmplY3RlZFxuICogU3RlcFJlZ2lzdHJ5LnJ1bihbbmV3IE15U3RlcChhcGlDbGllbnQpXSk7XG4gKiBgYGBcbiAqL1xuZXhwb3J0IGZ1bmN0aW9uIFN0ZXA8UyBleHRlbmRzIFpvZFR5cGU+KGNvbmZpZzogU3RlcENvbmZpZzxTPikge1xuICByZXR1cm4gZnVuY3Rpb24gPFQgZXh0ZW5kcyBuZXcgKC4uLmFyZ3M6IGFueVtdKSA9PiBCYXNlU3RlcDx6LmluZmVyPFM+Pj4oXG4gICAgQmFzZTogVFxuICApOiBUIHtcbiAgICBjb25zdCBFbmhhbmNlZCA9IGNsYXNzIGV4dGVuZHMgQmFzZSB7XG4gICAgICBvdmVycmlkZSBnZXRNZXRhZGF0YSgpOiBTdGVwTWV0YWRhdGE8Uz4ge1xuICAgICAgICByZXR1cm4ge1xuICAgICAgICAgIG5hbWU6IGNvbmZpZy5uYW1lLFxuICAgICAgICAgIHN0ZXBUeXBlOiBjb25maWcudHlwZSxcbiAgICAgICAgICBzaG9ydERlc2NyaXB0aW9uOiBjb25maWcuc2hvcnREZXNjcmlwdGlvbixcbiAgICAgICAgICBsb25nRGVzY3JpcHRpb246IGNvbmZpZy5sb25nRGVzY3JpcHRpb24sXG4gICAgICAgICAgeWFtbEV4YW1wbGU6IGNvbmZpZy55YW1sRXhhbXBsZSxcbiAgICAgICAgICBwYXJhbXNKc29uU2NoZW1hOiB6LnRvSlNPTlNjaGVtYShjb25maWcuc2NoZW1hKSxcbiAgICAgICAgfTtcbiAgICAgIH1cblxuICAgICAgb3ZlcnJpZGUgZ2V0RGF0YSgpOiBTdGVwRGF0YTxTPiB7XG4gICAgICAgIC8vIERldGVjdCBzdGVwIGtpbmQgYnkgY2hlY2tpbmcgZm9yIFBvbGxpbmdTdGVwIG1ldGhvZHNcbiAgICAgICAgY29uc3QgaGFzUG9sbGluZ01ldGhvZHMgPVxuICAgICAgICAgIFwidHJpZ2dlclwiIGluIHRoaXMgJiZcbiAgICAgICAgICB0eXBlb2YgKHRoaXMgYXMgYW55KS50cmlnZ2VyID09PSBcImZ1bmN0aW9uXCIgJiZcbiAgICAgICAgICBcInBvbGxcIiBpbiB0aGlzICYmXG4gICAgICAgICAgdHlwZW9mICh0aGlzIGFzIGFueSkucG9sbCA9PT0gXCJmdW5jdGlvblwiO1xuXG4gICAgICAgIGNvbnN0IHN0ZXBLaW5kOiBTdGVwS2luZCA9IGhhc1BvbGxpbmdNZXRob2RzID8gXCJwb2xsaW5nXCIgOiBcInNpbXBsZVwiO1xuXG4gICAgICAgIC8vIENoZWNrIGlmIHByZXBhcmUgaXMgb3ZlcnJpZGRlbiAobm90IHRoZSBkZWZhdWx0IEJhc2VTdGVwLnByZXBhcmUpXG4gICAgICAgIGNvbnN0IHJlcXVpcmVzQXBwcm92YWwgPSB0aGlzLnByZXBhcmUgIT09IEJhc2VTdGVwLnByb3RvdHlwZS5wcmVwYXJlO1xuXG4gICAgICAgIHJldHVybiB7XG4gICAgICAgICAgc3RlcFR5cGU6IGNvbmZpZy50eXBlLFxuICAgICAgICAgIHNjaGVtYTogY29uZmlnLnNjaGVtYSxcbiAgICAgICAgICBzdGVwS2luZCxcbiAgICAgICAgICByZXF1aXJlc0FwcHJvdmFsLFxuICAgICAgICB9O1xuICAgICAgfVxuICAgIH07XG5cbiAgICByZXR1cm4gRW5oYW5jZWQgYXMgVDtcbiAgfTtcbn1cbiJdfQ==
|
package/dist/index.d.ts
CHANGED
|
@@ -1,4 +1,4 @@
|
|
|
1
|
-
export { BaseStep,
|
|
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,
|
|
35
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi9zcmMvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IjtBQUFBLGdGQUFnRjtBQUNoRix3QkFBd0I7QUFDeEIsZ0ZBQWdGOzs7QUFFaEYsa0NBQWtDO0FBQ2xDLDhDQUFvRTtBQUEzRCxxR0FBQSxRQUFRLE9BQUE7QUFDakIsa0RBQWdEO0FBQXZDLHlHQUFBLFVBQVUsT0FBQTtBQUNuQixvREFBa0Q7QUFBekMsMkdBQUEsV0FBVyxPQUFBO0FBRXBCLHlDQUF5QztBQUN6QywwQ0FBcUQ7QUFBNUMsNEZBQUEsSUFBSSxPQUFBO0FBRWIsb0NBQW9DO0FBQ3BDLG9EQXlCOEI7QUF0QjVCLG9IQUFBLHFCQUFxQixPQUFBO0FBY3JCLGdDQUFnQztBQUNoQyxrSEFBQSxtQkFBbUIsT0FBQTtBQUNuQiw4R0FBQSxlQUFlLE9BQUE7QUFDZixrSEFBQSxtQkFBbUIsT0FBQTtBQUNuQiwrR0FBQSxnQkFBZ0IsT0FBQTtBQUNoQiwrR0FBQSxnQkFBZ0IsT0FBQTtBQUNoQixtQkFBbUI7QUFDbkIsMEdBQUEsV0FBVyxPQUFBO0FBR2IsNkJBQTZCO0FBQzdCLDBEQUFvRztBQUEzRiw2R0FBQSxZQUFZLE9BQUE7QUFBcUIsd0hBQUEsdUJBQXVCLE9BQUE7QUFFakUsVUFBVTtBQUNWLHFEQUF5RjtBQUFoRix5R0FBQSxVQUFVLE9BQUEiLCJzb3VyY2VzQ29udGVudCI6WyIvLyA9PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PVxuLy8gU3RlcCBTREsgLSBQdWJsaWMgQVBJXG4vLyA9PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PVxuXG4vLyBCYXNlIGNsYXNzZXMgZm9yIGNyZWF0aW5nIHN0ZXBzXG5leHBvcnQgeyBCYXNlU3RlcCwgU3RlcE1ldGFkYXRhLCBTdGVwS2luZCB9IGZyb20gXCIuL2Jhc2UvYmFzZS1zdGVwXCI7XG5leHBvcnQgeyBTaW1wbGVTdGVwIH0gZnJvbSBcIi4vYmFzZS9zaW1wbGUtc3RlcFwiO1xuZXhwb3J0IHsgUG9sbGluZ1N0ZXAgfSBmcm9tIFwiLi9iYXNlL3BvbGxpbmctc3RlcFwiO1xuXG4vLyBEZWNvcmF0b3IgZm9yIGFkZGluZyBtZXRhZGF0YSB0byBzdGVwc1xuZXhwb3J0IHsgU3RlcCwgU3RlcENvbmZpZyB9IGZyb20gXCIuL2RlY29yYXRvcnMvc3RlcFwiO1xuXG4vLyBPdXRwdXQgdHlwZXMgYW5kIGhlbHBlciBmdW5jdGlvbnNcbmV4cG9ydCB7XG4gIC8vIEFwcHJvdmFsIGNvbnRleHRcbiAgQXBwcm92YWxDb250ZXh0LFxuICBBcHByb3ZhbENvbnRleHRTY2hlbWEsXG4gIC8vIFBoYXNlLXNwZWNpZmljIG91dHB1dHNcbiAgUHJlcGFyZU91dHB1dCxcbiAgUnVuT3V0cHV0LFxuICBUcmlnZ2VyT3V0cHV0LFxuICBUcmlnZ2VyZWRPdXRwdXQsXG4gIFBvbGxPdXRwdXQsXG4gIFBvbGxBZ2Fpbk91dHB1dCxcbiAgLy8gQ29tbW9uIG91dHB1dHNcbiAgU3VjY2Vzc091dHB1dCxcbiAgRmFpbGVkT3V0cHV0LFxuICBBcHByb3ZhbFJlcXVpcmVkT3V0cHV0LFxuICAvLyBDb21iaW5lZCBvdXRwdXQgdHlwZVxuICBTdGVwT3V0cHV0LFxuICAvLyBPdXRwdXQgc2NoZW1hcyBmb3IgdmFsaWRhdGlvblxuICBQcmVwYXJlT3V0cHV0U2NoZW1hLFxuICBSdW5PdXRwdXRTY2hlbWEsXG4gIFRyaWdnZXJPdXRwdXRTY2hlbWEsXG4gIFBvbGxPdXRwdXRTY2hlbWEsXG4gIFN0ZXBPdXRwdXRTY2hlbWEsXG4gIC8vIEhlbHBlciBmdW5jdGlvbnNcbiAgU3RlcE91dHB1dHMsXG59IGZyb20gXCIuL291dHB1dC9zdGVwLW91dHB1dFwiO1xuXG4vLyBSZWdpc3RyeSBmb3IgcnVubmluZyBzdGVwc1xuZXhwb3J0IHsgU3RlcFJlZ2lzdHJ5LCBTdGVwUmVnaXN0cnlJbnB1dCwgU3RlcFJlZ2lzdHJ5SW5wdXRTY2hlbWEgfSBmcm9tIFwiLi9yZWdpc3RyeS9zdGVwLXJlZ2lzdHJ5XCI7XG5cbi8vIExvZ2dpbmdcbmV4cG9ydCB7IFN0ZXBMb2dnZXIsIFN0ZXBMb2dnZXJDb25maWcsIExvZ0VudHJ5LCBMb2dMZXZlbCB9IGZyb20gXCIuL2xvZ2dpbmcvc3RlcC1sb2dnZXJcIjtcbiJdfQ==
|
|
@@ -1,5 +1,5 @@
|
|
|
1
1
|
import z from "zod";
|
|
2
|
-
import type {
|
|
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([
|
|
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:
|
|
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
|
|
83
|
-
const instance = new StepCls();
|
|
82
|
+
for (const instance of steps) {
|
|
84
83
|
const metadata = instance.getMetadata();
|
|
85
|
-
this.steps.set(metadata.stepType,
|
|
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([
|
|
93
|
+
* StepRegistry.run([
|
|
94
|
+
* new MyStep1(dependency1, dependency2),
|
|
95
|
+
* new MyStep2(config)
|
|
96
|
+
* ]);
|
|
95
97
|
* ```
|
|
96
98
|
*
|
|
97
99
|
* CLI arguments:
|
|
@@ -130,37 +132,35 @@ class StepRegistry {
|
|
|
130
132
|
}
|
|
131
133
|
synthesizeMetadata() {
|
|
132
134
|
const metadata = [];
|
|
133
|
-
for (const
|
|
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
|
|
141
|
-
if (!
|
|
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
|
|
147
|
-
const metadata = instance.getMetadata();
|
|
147
|
+
const data = instance.getData();
|
|
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:
|
|
152
|
+
stepType: data.stepType,
|
|
153
153
|
});
|
|
154
154
|
instance._setLogger(logger);
|
|
155
155
|
// Validate params against the step's schema
|
|
156
|
-
const parseResult =
|
|
156
|
+
const parseResult = data.schema.safeParse(params);
|
|
157
157
|
if (!parseResult.success) {
|
|
158
158
|
this.writeOutput(step_output_1.StepOutputs.failed(`Invalid params: ${parseResult.error.message}`, "INVALID_PARAMS"));
|
|
159
159
|
return;
|
|
160
160
|
}
|
|
161
161
|
const validatedParams = parseResult.data;
|
|
162
162
|
// Route to appropriate phase based on step kind and input context
|
|
163
|
-
const output = await this.routeExecution(instance,
|
|
163
|
+
const output = await this.routeExecution(instance, data, validatedParams, approvalContext, pollingState);
|
|
164
164
|
this.writeOutput(output);
|
|
165
165
|
}
|
|
166
166
|
catch (error) {
|
|
@@ -168,15 +168,15 @@ class StepRegistry {
|
|
|
168
168
|
this.writeOutput(step_output_1.StepOutputs.failed(errorMessage, "EXECUTION_ERROR"));
|
|
169
169
|
}
|
|
170
170
|
}
|
|
171
|
-
async routeExecution(instance,
|
|
172
|
-
const requiresApproval =
|
|
173
|
-
if (
|
|
171
|
+
async routeExecution(instance, data, params, approvalContext, pollingState) {
|
|
172
|
+
const requiresApproval = data.requiresApproval;
|
|
173
|
+
if (data.stepKind === "simple") {
|
|
174
174
|
return this.routeSimpleStep(instance, params, requiresApproval, approvalContext);
|
|
175
175
|
}
|
|
176
|
-
if (
|
|
176
|
+
if (data.stepKind === "polling") {
|
|
177
177
|
return this.routePollingStep(instance, params, requiresApproval, approvalContext, pollingState);
|
|
178
178
|
}
|
|
179
|
-
return step_output_1.StepOutputs.failed(`Unknown step kind: ${
|
|
179
|
+
return step_output_1.StepOutputs.failed(`Unknown step kind: ${data.stepKind}`, "UNKNOWN_STEP_KIND");
|
|
180
180
|
}
|
|
181
181
|
/**
|
|
182
182
|
* Route SimpleStep execution based on approval state.
|
|
@@ -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 { 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, 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.stepType, 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(\n          `No step registered with type: ${type}`,\n          \"STEP_NOT_FOUND\"\n        )\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.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"]}
|
|
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, StepData, 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 data = instance.getData();\n\n      // Inject logger\n      const logger = new StepLogger({\n        logDir: this.logDir,\n        executionId: this.executionId,\n        stepType: data.stepType,\n      });\n      instance._setLogger(logger);\n\n      // Validate params against the step's schema\n      const parseResult = data.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        data,\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    data: StepData,\n    params: unknown,\n    approvalContext?: ApprovalContext,\n    pollingState?: Record<string, unknown>\n  ): Promise<StepOutput> {\n    const requiresApproval = data.requiresApproval;\n\n    if (data.stepKind === \"simple\") {\n      return this.routeSimpleStep(\n        instance as SimpleStep<unknown>,\n        params,\n        requiresApproval,\n        approvalContext\n      );\n    }\n\n    if (data.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: ${data.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.
|
|
3
|
+
"version": "0.1.5",
|
|
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,300 @@
|
|
|
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[3] || "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(
|
|
71
|
+
path.join(srcDir, "deployment-service.ts"),
|
|
72
|
+
deploymentServiceContent
|
|
73
|
+
);
|
|
74
|
+
|
|
75
|
+
// Create my-first-step.ts
|
|
76
|
+
console.log("Creating my-first-step.ts...");
|
|
77
|
+
const myFirstStepContent = `import {
|
|
78
|
+
RunOutput,
|
|
79
|
+
SimpleStep,
|
|
80
|
+
Step,
|
|
81
|
+
StepOutputs,
|
|
82
|
+
} from "@devramps/sdk-typescript";
|
|
83
|
+
import z from "zod";
|
|
84
|
+
import { DeploymentService } from "../deployment-service";
|
|
85
|
+
|
|
86
|
+
const DeploySchema = z.object({
|
|
87
|
+
target: z.string().describe("A Target"),
|
|
88
|
+
version: z.string().optional(),
|
|
89
|
+
});
|
|
90
|
+
|
|
91
|
+
type DeployParams = z.infer<typeof DeploySchema>;
|
|
92
|
+
|
|
93
|
+
@Step({
|
|
94
|
+
name: "Custom Deployment Action",
|
|
95
|
+
type: "CUSTOM:SCRIPT:DEPLOY",
|
|
96
|
+
schema: DeploySchema,
|
|
97
|
+
})
|
|
98
|
+
export class MyFirstStep extends SimpleStep<DeployParams> {
|
|
99
|
+
constructor(private deploymentService: DeploymentService) {
|
|
100
|
+
super();
|
|
101
|
+
}
|
|
102
|
+
|
|
103
|
+
async run(params: DeployParams): Promise<RunOutput> {
|
|
104
|
+
this.logger.info(\`Deploying with params: \${JSON.stringify(params)}\`);
|
|
105
|
+
|
|
106
|
+
const result = await this.deploymentService.deploy(
|
|
107
|
+
params.target,
|
|
108
|
+
params.version
|
|
109
|
+
);
|
|
110
|
+
|
|
111
|
+
this.logger.info(\`Deployment completed: \${result.deploymentId}\`);
|
|
112
|
+
|
|
113
|
+
return StepOutputs.success({
|
|
114
|
+
deploymentId: result.deploymentId,
|
|
115
|
+
target: params.target,
|
|
116
|
+
version: params.version,
|
|
117
|
+
});
|
|
118
|
+
}
|
|
119
|
+
}
|
|
120
|
+
`;
|
|
121
|
+
fs.writeFileSync(path.join(stepsDir, "my-first-step.ts"), myFirstStepContent);
|
|
122
|
+
|
|
123
|
+
// Create src/index.ts
|
|
124
|
+
console.log("Creating src/index.ts...");
|
|
125
|
+
const indexContent = `import { StepRegistry } from "@devramps/sdk-typescript";
|
|
126
|
+
import { MyFirstStep } from "./steps/my-first-step";
|
|
127
|
+
import { RealDeploymentService } from "./deployment-service";
|
|
128
|
+
|
|
129
|
+
// Instantiate dependencies
|
|
130
|
+
const deploymentService = new RealDeploymentService();
|
|
131
|
+
|
|
132
|
+
// Instantiate steps with their dependencies
|
|
133
|
+
const myFirstStep = new MyFirstStep(deploymentService);
|
|
134
|
+
|
|
135
|
+
// Register step instances
|
|
136
|
+
StepRegistry.run([myFirstStep]);
|
|
137
|
+
`;
|
|
138
|
+
fs.writeFileSync(path.join(srcDir, "index.ts"), indexContent);
|
|
139
|
+
|
|
140
|
+
// Create tsconfig.json
|
|
141
|
+
console.log("Creating tsconfig.json...");
|
|
142
|
+
const tsconfigContent = `{
|
|
143
|
+
"compilerOptions": {
|
|
144
|
+
"target": "ES2022",
|
|
145
|
+
"module": "commonjs",
|
|
146
|
+
"lib": ["ES2022"],
|
|
147
|
+
"types": ["node", "jest"],
|
|
148
|
+
"declaration": true,
|
|
149
|
+
"strict": true,
|
|
150
|
+
"noImplicitAny": true,
|
|
151
|
+
"strictNullChecks": true,
|
|
152
|
+
"noImplicitThis": true,
|
|
153
|
+
"alwaysStrict": true,
|
|
154
|
+
"noUnusedLocals": false,
|
|
155
|
+
"noUnusedParameters": false,
|
|
156
|
+
"noImplicitReturns": true,
|
|
157
|
+
"noFallthroughCasesInSwitch": false,
|
|
158
|
+
"inlineSourceMap": true,
|
|
159
|
+
"inlineSources": true,
|
|
160
|
+
"experimentalDecorators": true,
|
|
161
|
+
"emitDecoratorMetadata": true,
|
|
162
|
+
"strictPropertyInitialization": false,
|
|
163
|
+
"esModuleInterop": true,
|
|
164
|
+
"skipLibCheck": true,
|
|
165
|
+
"outDir": "./dist",
|
|
166
|
+
"rootDir": "./src"
|
|
167
|
+
},
|
|
168
|
+
"include": ["src/**/*"],
|
|
169
|
+
"exclude": ["node_modules", "dist"]
|
|
170
|
+
}
|
|
171
|
+
`;
|
|
172
|
+
fs.writeFileSync(path.join(baseDir, "tsconfig.json"), tsconfigContent);
|
|
173
|
+
|
|
174
|
+
// Create jest.config.js
|
|
175
|
+
console.log("Creating jest.config.js...");
|
|
176
|
+
const jestConfigContent = `module.exports = {
|
|
177
|
+
preset: 'ts-jest',
|
|
178
|
+
testEnvironment: 'node',
|
|
179
|
+
roots: ['<rootDir>/tst'],
|
|
180
|
+
testMatch: ['**/*.test.ts'],
|
|
181
|
+
moduleFileExtensions: ['ts', 'tsx', 'js', 'jsx', 'json', 'node'],
|
|
182
|
+
collectCoverageFrom: [
|
|
183
|
+
'src/**/*.ts',
|
|
184
|
+
'!src/**/*.d.ts',
|
|
185
|
+
],
|
|
186
|
+
};
|
|
187
|
+
`;
|
|
188
|
+
fs.writeFileSync(path.join(baseDir, "jest.config.js"), jestConfigContent);
|
|
189
|
+
|
|
190
|
+
// Create test file for my-first-step
|
|
191
|
+
console.log("Creating tst/steps/my-first-step.test.ts...");
|
|
192
|
+
const testContent = `import { MyFirstStep } from "../../src/steps/my-first-step";
|
|
193
|
+
import { DeploymentService } from "../../src/deployment-service";
|
|
194
|
+
import { StepOutputs } from "@devramps/sdk-typescript";
|
|
195
|
+
|
|
196
|
+
// Mock implementation of DeploymentService for testing
|
|
197
|
+
class MockDeploymentService implements DeploymentService {
|
|
198
|
+
async deploy(target: string, version?: string): Promise<{ deploymentId: string }> {
|
|
199
|
+
return { deploymentId: \`mock-deploy-\${target}-\${version || "latest"}\` };
|
|
200
|
+
}
|
|
201
|
+
}
|
|
202
|
+
|
|
203
|
+
describe("MyFirstStep", () => {
|
|
204
|
+
let mockDeploymentService: MockDeploymentService;
|
|
205
|
+
let step: MyFirstStep;
|
|
206
|
+
|
|
207
|
+
beforeEach(() => {
|
|
208
|
+
// Create mock dependency
|
|
209
|
+
mockDeploymentService = new MockDeploymentService();
|
|
210
|
+
// Inject mock into step
|
|
211
|
+
step = new MyFirstStep(mockDeploymentService);
|
|
212
|
+
});
|
|
213
|
+
|
|
214
|
+
it("should run successfully with valid parameters", async () => {
|
|
215
|
+
const params = {
|
|
216
|
+
target: "production",
|
|
217
|
+
version: "1.0.0",
|
|
218
|
+
};
|
|
219
|
+
|
|
220
|
+
const result = await step.run(params);
|
|
221
|
+
|
|
222
|
+
expect(result.status).toBe("SUCCESS");
|
|
223
|
+
if (result.status === "SUCCESS") {
|
|
224
|
+
expect(result.data).toEqual({
|
|
225
|
+
deploymentId: "mock-deploy-production-1.0.0",
|
|
226
|
+
target: "production",
|
|
227
|
+
version: "1.0.0",
|
|
228
|
+
});
|
|
229
|
+
}
|
|
230
|
+
});
|
|
231
|
+
|
|
232
|
+
it("should run successfully without optional version parameter", async () => {
|
|
233
|
+
const params = {
|
|
234
|
+
target: "staging",
|
|
235
|
+
};
|
|
236
|
+
|
|
237
|
+
const result = await step.run(params);
|
|
238
|
+
|
|
239
|
+
expect(result.status).toBe("SUCCESS");
|
|
240
|
+
if (result.status === "SUCCESS") {
|
|
241
|
+
expect(result.data).toEqual({
|
|
242
|
+
deploymentId: "mock-deploy-staging-latest",
|
|
243
|
+
target: "staging",
|
|
244
|
+
version: undefined,
|
|
245
|
+
});
|
|
246
|
+
}
|
|
247
|
+
});
|
|
248
|
+
|
|
249
|
+
it("should call deployment service with correct parameters", async () => {
|
|
250
|
+
const deploySpy = jest.spyOn(mockDeploymentService, "deploy");
|
|
251
|
+
const params = {
|
|
252
|
+
target: "development",
|
|
253
|
+
version: "2.0.0",
|
|
254
|
+
};
|
|
255
|
+
|
|
256
|
+
await step.run(params);
|
|
257
|
+
|
|
258
|
+
expect(deploySpy).toHaveBeenCalledWith("development", "2.0.0");
|
|
259
|
+
});
|
|
260
|
+
|
|
261
|
+
it("should log deployment information", async () => {
|
|
262
|
+
// Access the logger through type assertion to test logging behavior
|
|
263
|
+
const logSpy = jest.spyOn((step as any).logger, "info");
|
|
264
|
+
const params = {
|
|
265
|
+
target: "production",
|
|
266
|
+
version: "3.0.0",
|
|
267
|
+
};
|
|
268
|
+
|
|
269
|
+
await step.run(params);
|
|
270
|
+
|
|
271
|
+
expect(logSpy).toHaveBeenCalledWith(
|
|
272
|
+
\`Deploying with params: \${JSON.stringify(params)}\`
|
|
273
|
+
);
|
|
274
|
+
expect(logSpy).toHaveBeenCalledWith(
|
|
275
|
+
"Deployment completed: mock-deploy-production-3.0.0"
|
|
276
|
+
);
|
|
277
|
+
});
|
|
278
|
+
});
|
|
279
|
+
`;
|
|
280
|
+
fs.writeFileSync(path.join(tstStepsDir, "my-first-step.test.ts"), testContent);
|
|
281
|
+
|
|
282
|
+
// Update package.json to add build and start scripts
|
|
283
|
+
console.log("Adding build and start scripts to package.json...");
|
|
284
|
+
const packageJsonPath = path.join(baseDir, "package.json");
|
|
285
|
+
const packageJson = JSON.parse(fs.readFileSync(packageJsonPath, "utf8"));
|
|
286
|
+
packageJson.scripts = {
|
|
287
|
+
...packageJson.scripts,
|
|
288
|
+
"build-step-registry": "tsc -p tsconfig.json",
|
|
289
|
+
"start-step-registry": "node ./dist/index.js",
|
|
290
|
+
test: "jest",
|
|
291
|
+
"test:watch": "jest --watch",
|
|
292
|
+
"test:coverage": "jest --coverage",
|
|
293
|
+
};
|
|
294
|
+
fs.writeFileSync(packageJsonPath, JSON.stringify(packageJson, null, 2));
|
|
295
|
+
|
|
296
|
+
console.log(`\nStep registry created successfully in '${folderName}'!`);
|
|
297
|
+
console.log(`\nNext steps:`);
|
|
298
|
+
console.log(` cd ${folderName}`);
|
|
299
|
+
console.log(` npm run build-step-registry`);
|
|
300
|
+
console.log(` npm run test`);
|