@devramps/sdk-typescript 0.1.0
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/LICENSE +21 -0
- package/README.md +461 -0
- package/dist/base/base-step.d.ts +41 -0
- package/dist/base/base-step.js +38 -0
- package/dist/base/polling-step.d.ts +85 -0
- package/dist/base/polling-step.js +80 -0
- package/dist/base/requires-approval-step.d.ts +5 -0
- package/dist/base/requires-approval-step.js +8 -0
- package/dist/base/simple-step.d.ts +51 -0
- package/dist/base/simple-step.js +52 -0
- package/dist/decorators/step.d.ts +24 -0
- package/dist/decorators/step.js +45 -0
- package/dist/index.d.ts +7 -0
- package/dist/index.js +35 -0
- package/dist/logging/step-logger.d.ts +35 -0
- package/dist/logging/step-logger.js +94 -0
- package/dist/output/step-output.d.ts +286 -0
- package/dist/output/step-output.js +108 -0
- package/dist/registry/step-registry.d.ts +93 -0
- package/dist/registry/step-registry.js +238 -0
- package/dist/types.d.ts +1 -0
- package/dist/types.js +3 -0
- package/package.json +70 -0
|
@@ -0,0 +1,238 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
|
|
3
|
+
if (k2 === undefined) k2 = k;
|
|
4
|
+
var desc = Object.getOwnPropertyDescriptor(m, k);
|
|
5
|
+
if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
|
|
6
|
+
desc = { enumerable: true, get: function() { return m[k]; } };
|
|
7
|
+
}
|
|
8
|
+
Object.defineProperty(o, k2, desc);
|
|
9
|
+
}) : (function(o, m, k, k2) {
|
|
10
|
+
if (k2 === undefined) k2 = k;
|
|
11
|
+
o[k2] = m[k];
|
|
12
|
+
}));
|
|
13
|
+
var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
|
|
14
|
+
Object.defineProperty(o, "default", { enumerable: true, value: v });
|
|
15
|
+
}) : function(o, v) {
|
|
16
|
+
o["default"] = v;
|
|
17
|
+
});
|
|
18
|
+
var __importStar = (this && this.__importStar) || (function () {
|
|
19
|
+
var ownKeys = function(o) {
|
|
20
|
+
ownKeys = Object.getOwnPropertyNames || function (o) {
|
|
21
|
+
var ar = [];
|
|
22
|
+
for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k;
|
|
23
|
+
return ar;
|
|
24
|
+
};
|
|
25
|
+
return ownKeys(o);
|
|
26
|
+
};
|
|
27
|
+
return function (mod) {
|
|
28
|
+
if (mod && mod.__esModule) return mod;
|
|
29
|
+
var result = {};
|
|
30
|
+
if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]);
|
|
31
|
+
__setModuleDefault(result, mod);
|
|
32
|
+
return result;
|
|
33
|
+
};
|
|
34
|
+
})();
|
|
35
|
+
var __importDefault = (this && this.__importDefault) || function (mod) {
|
|
36
|
+
return (mod && mod.__esModule) ? mod : { "default": mod };
|
|
37
|
+
};
|
|
38
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
39
|
+
exports.StepRegistry = exports.StepRegistryInputSchema = void 0;
|
|
40
|
+
const fs = __importStar(require("fs"));
|
|
41
|
+
const path = __importStar(require("path"));
|
|
42
|
+
const minimist_1 = __importDefault(require("minimist"));
|
|
43
|
+
const zod_1 = __importDefault(require("zod"));
|
|
44
|
+
const step_logger_1 = require("../logging/step-logger");
|
|
45
|
+
const step_output_1 = require("../output/step-output");
|
|
46
|
+
// =============================================================================
|
|
47
|
+
// Input Schemas
|
|
48
|
+
// =============================================================================
|
|
49
|
+
const StepRegistrySynthesizeSchema = zod_1.default.object({
|
|
50
|
+
job: zod_1.default.literal("SYNTHESIZE-METADATA"),
|
|
51
|
+
});
|
|
52
|
+
const StepRegistryExecuteSchema = zod_1.default.object({
|
|
53
|
+
job: zod_1.default.literal("EXECUTE"),
|
|
54
|
+
type: zod_1.default.string(),
|
|
55
|
+
params: zod_1.default.record(zod_1.default.string(), zod_1.default.any()),
|
|
56
|
+
// Optional context - presence determines which phase to run
|
|
57
|
+
approvalContext: step_output_1.ApprovalContextSchema.optional(),
|
|
58
|
+
pollingState: zod_1.default.record(zod_1.default.string(), zod_1.default.any()).optional(),
|
|
59
|
+
});
|
|
60
|
+
exports.StepRegistryInputSchema = zod_1.default.discriminatedUnion("job", [
|
|
61
|
+
StepRegistryExecuteSchema,
|
|
62
|
+
StepRegistrySynthesizeSchema,
|
|
63
|
+
]);
|
|
64
|
+
// =============================================================================
|
|
65
|
+
// Configuration
|
|
66
|
+
// =============================================================================
|
|
67
|
+
const DEFAULT_OUTPUT_PATH = "/tmp/step-output.json";
|
|
68
|
+
const DEFAULT_LOG_DIR = "/tmp/step-logs";
|
|
69
|
+
// =============================================================================
|
|
70
|
+
// StepRegistry
|
|
71
|
+
// =============================================================================
|
|
72
|
+
class StepRegistry {
|
|
73
|
+
steps;
|
|
74
|
+
outputPath;
|
|
75
|
+
logDir;
|
|
76
|
+
executionId;
|
|
77
|
+
constructor(steps, outputPath, logDir, executionId) {
|
|
78
|
+
this.steps = new Map();
|
|
79
|
+
this.outputPath = outputPath;
|
|
80
|
+
this.logDir = logDir;
|
|
81
|
+
this.executionId = executionId;
|
|
82
|
+
for (const StepCls of steps) {
|
|
83
|
+
const instance = new StepCls();
|
|
84
|
+
const metadata = instance.getMetadata();
|
|
85
|
+
this.steps.set(metadata.type, StepCls);
|
|
86
|
+
}
|
|
87
|
+
}
|
|
88
|
+
/**
|
|
89
|
+
* Main entrypoint for running steps.
|
|
90
|
+
* Parses CLI args and either synthesizes metadata or executes a step.
|
|
91
|
+
*
|
|
92
|
+
* Usage in user's entrypoint file:
|
|
93
|
+
* ```
|
|
94
|
+
* StepRegistry.run([MyStep1, MyStep2, ...]);
|
|
95
|
+
* ```
|
|
96
|
+
*
|
|
97
|
+
* CLI arguments:
|
|
98
|
+
* - --input: JSON blob with job type and parameters
|
|
99
|
+
* - --output: Path to write output JSON (default: /tmp/step-output.json)
|
|
100
|
+
* - --log-dir: Directory for log files (default: /tmp/step-logs)
|
|
101
|
+
* - --execution-id: Unique ID for this execution (required for logging)
|
|
102
|
+
*/
|
|
103
|
+
static async run(steps) {
|
|
104
|
+
const args = (0, minimist_1.default)(process.argv.slice(2));
|
|
105
|
+
const outputPath = args["output"] || DEFAULT_OUTPUT_PATH;
|
|
106
|
+
const logDir = args["log-dir"] || DEFAULT_LOG_DIR;
|
|
107
|
+
const executionId = args["execution-id"] || `exec-${Date.now()}`;
|
|
108
|
+
const registry = new StepRegistry(steps, outputPath, logDir, executionId);
|
|
109
|
+
await registry.execute(args);
|
|
110
|
+
}
|
|
111
|
+
async execute(args) {
|
|
112
|
+
try {
|
|
113
|
+
const input = exports.StepRegistryInputSchema.parse(JSON.parse(args["input"]));
|
|
114
|
+
switch (input.job) {
|
|
115
|
+
case "SYNTHESIZE-METADATA": {
|
|
116
|
+
const metadata = this.synthesizeMetadata();
|
|
117
|
+
this.writeOutput({ status: "SUCCESS", data: { metadata } });
|
|
118
|
+
break;
|
|
119
|
+
}
|
|
120
|
+
case "EXECUTE": {
|
|
121
|
+
await this.executeStep(input.type, input.params, input.approvalContext, input.pollingState);
|
|
122
|
+
break;
|
|
123
|
+
}
|
|
124
|
+
}
|
|
125
|
+
}
|
|
126
|
+
catch (error) {
|
|
127
|
+
const errorMessage = error instanceof Error ? error.message : String(error);
|
|
128
|
+
this.writeOutput(step_output_1.StepOutputs.failed(errorMessage, "REGISTRY_ERROR"));
|
|
129
|
+
}
|
|
130
|
+
}
|
|
131
|
+
synthesizeMetadata() {
|
|
132
|
+
const metadata = [];
|
|
133
|
+
for (const StepCls of this.steps.values()) {
|
|
134
|
+
const instance = new StepCls();
|
|
135
|
+
metadata.push(instance.getMetadata());
|
|
136
|
+
}
|
|
137
|
+
return metadata;
|
|
138
|
+
}
|
|
139
|
+
async executeStep(type, params, approvalContext, pollingState) {
|
|
140
|
+
const StepCls = this.steps.get(type);
|
|
141
|
+
if (!StepCls) {
|
|
142
|
+
this.writeOutput(step_output_1.StepOutputs.failed(`No step registered with type: ${type}`, "STEP_NOT_FOUND"));
|
|
143
|
+
return;
|
|
144
|
+
}
|
|
145
|
+
try {
|
|
146
|
+
const instance = new StepCls();
|
|
147
|
+
const metadata = instance.getMetadata();
|
|
148
|
+
// Inject logger
|
|
149
|
+
const logger = new step_logger_1.StepLogger({
|
|
150
|
+
logDir: this.logDir,
|
|
151
|
+
executionId: this.executionId,
|
|
152
|
+
stepType: metadata.type,
|
|
153
|
+
});
|
|
154
|
+
instance._setLogger(logger);
|
|
155
|
+
// Validate params against the step's schema
|
|
156
|
+
const parseResult = metadata.schema.safeParse(params);
|
|
157
|
+
if (!parseResult.success) {
|
|
158
|
+
this.writeOutput(step_output_1.StepOutputs.failed(`Invalid params: ${parseResult.error.message}`, "INVALID_PARAMS"));
|
|
159
|
+
return;
|
|
160
|
+
}
|
|
161
|
+
const validatedParams = parseResult.data;
|
|
162
|
+
// Route to appropriate phase based on step kind and input context
|
|
163
|
+
const output = await this.routeExecution(instance, metadata, validatedParams, approvalContext, pollingState);
|
|
164
|
+
this.writeOutput(output);
|
|
165
|
+
}
|
|
166
|
+
catch (error) {
|
|
167
|
+
const errorMessage = error instanceof Error ? error.message : String(error);
|
|
168
|
+
this.writeOutput(step_output_1.StepOutputs.failed(errorMessage, "EXECUTION_ERROR"));
|
|
169
|
+
}
|
|
170
|
+
}
|
|
171
|
+
async routeExecution(instance, metadata, params, approvalContext, pollingState) {
|
|
172
|
+
const requiresApproval = metadata.requiresApproval;
|
|
173
|
+
if (metadata.stepKind === "simple") {
|
|
174
|
+
return this.routeSimpleStep(instance, params, requiresApproval, approvalContext);
|
|
175
|
+
}
|
|
176
|
+
if (metadata.stepKind === "polling") {
|
|
177
|
+
return this.routePollingStep(instance, params, requiresApproval, approvalContext, pollingState);
|
|
178
|
+
}
|
|
179
|
+
return step_output_1.StepOutputs.failed(`Unknown step kind: ${metadata.stepKind}`, "UNKNOWN_STEP_KIND");
|
|
180
|
+
}
|
|
181
|
+
/**
|
|
182
|
+
* Route SimpleStep execution based on approval state.
|
|
183
|
+
*
|
|
184
|
+
* | requiresApproval | approvalContext provided? | Action |
|
|
185
|
+
* |------------------|---------------------------|---------------------|
|
|
186
|
+
* | No | - | Call run(params) |
|
|
187
|
+
* | Yes | No | Call prepare(params)|
|
|
188
|
+
* | Yes | Yes | Call run(params, approval) |
|
|
189
|
+
*/
|
|
190
|
+
async routeSimpleStep(instance, params, requiresApproval, approvalContext) {
|
|
191
|
+
if (!requiresApproval) {
|
|
192
|
+
// No approval needed - just run
|
|
193
|
+
return instance.execute(params, undefined);
|
|
194
|
+
}
|
|
195
|
+
if (!approvalContext) {
|
|
196
|
+
// Needs approval but don't have it yet - call prepare
|
|
197
|
+
return instance.prepare(params);
|
|
198
|
+
}
|
|
199
|
+
// Have approval - run with approval context
|
|
200
|
+
return instance.execute(params, approvalContext);
|
|
201
|
+
}
|
|
202
|
+
/**
|
|
203
|
+
* Route PollingStep execution based on approval and polling state.
|
|
204
|
+
*
|
|
205
|
+
* | requiresApproval | approvalContext? | pollingState? | Action |
|
|
206
|
+
* |------------------|------------------|---------------|-------------------------------|
|
|
207
|
+
* | No | - | No | Call trigger(params) |
|
|
208
|
+
* | No | - | Yes | Call poll(params, pollingState)|
|
|
209
|
+
* | Yes | No | No | Call prepare(params) |
|
|
210
|
+
* | Yes | Yes | No | Call trigger(params, approval)|
|
|
211
|
+
* | Yes | Yes | Yes | Call poll(params, pollingState)|
|
|
212
|
+
*/
|
|
213
|
+
async routePollingStep(instance, params, requiresApproval, approvalContext, pollingState) {
|
|
214
|
+
// If we have polling state, we're in the poll phase (regardless of approval)
|
|
215
|
+
if (pollingState) {
|
|
216
|
+
return instance.executePoll(params, pollingState);
|
|
217
|
+
}
|
|
218
|
+
if (!requiresApproval) {
|
|
219
|
+
// No approval needed - trigger directly
|
|
220
|
+
return instance.executeTrigger(params, undefined);
|
|
221
|
+
}
|
|
222
|
+
if (!approvalContext) {
|
|
223
|
+
// Needs approval but don't have it yet - call prepare
|
|
224
|
+
return instance.prepare(params);
|
|
225
|
+
}
|
|
226
|
+
// Have approval - trigger with approval context
|
|
227
|
+
return instance.executeTrigger(params, approvalContext);
|
|
228
|
+
}
|
|
229
|
+
writeOutput(output) {
|
|
230
|
+
const outputDir = path.dirname(this.outputPath);
|
|
231
|
+
if (!fs.existsSync(outputDir)) {
|
|
232
|
+
fs.mkdirSync(outputDir, { recursive: true });
|
|
233
|
+
}
|
|
234
|
+
fs.writeFileSync(this.outputPath, JSON.stringify(output, null, 2));
|
|
235
|
+
}
|
|
236
|
+
}
|
|
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"]}
|
package/dist/types.d.ts
ADDED
|
@@ -0,0 +1 @@
|
|
|
1
|
+
export type Parameters = Record<string, string>;
|
package/dist/types.js
ADDED
|
@@ -0,0 +1,3 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHlwZXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi9zcmMvdHlwZXMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IiIsInNvdXJjZXNDb250ZW50IjpbImV4cG9ydCB0eXBlIFBhcmFtZXRlcnMgPSBSZWNvcmQ8c3RyaW5nLCBzdHJpbmc+O1xuIl19
|
package/package.json
ADDED
|
@@ -0,0 +1,70 @@
|
|
|
1
|
+
{
|
|
2
|
+
"name": "@devramps/sdk-typescript",
|
|
3
|
+
"version": "0.1.0",
|
|
4
|
+
"description": "SDK for building custom deployment steps for DevRamps",
|
|
5
|
+
"main": "dist/index.js",
|
|
6
|
+
"module": "dist/index.js",
|
|
7
|
+
"types": "dist/index.d.ts",
|
|
8
|
+
"files": [
|
|
9
|
+
"dist",
|
|
10
|
+
"LICENSE",
|
|
11
|
+
"README.md"
|
|
12
|
+
],
|
|
13
|
+
"exports": {
|
|
14
|
+
".": {
|
|
15
|
+
"import": "./dist/index.js",
|
|
16
|
+
"require": "./dist/index.js",
|
|
17
|
+
"types": "./dist/index.d.ts"
|
|
18
|
+
}
|
|
19
|
+
},
|
|
20
|
+
"scripts": {
|
|
21
|
+
"build": "tsc -p tsconfig.json",
|
|
22
|
+
"clean": "rm -rf dist",
|
|
23
|
+
"lint": "eslint src/",
|
|
24
|
+
"lint:fix": "eslint src/ --fix",
|
|
25
|
+
"test": "vitest run",
|
|
26
|
+
"test:watch": "vitest",
|
|
27
|
+
"test:coverage": "vitest run --coverage",
|
|
28
|
+
"typecheck": "tsc --noEmit",
|
|
29
|
+
"prepublishOnly": "npm run clean && npm run build",
|
|
30
|
+
"prepack": "npm run build"
|
|
31
|
+
},
|
|
32
|
+
"keywords": [
|
|
33
|
+
"devramps",
|
|
34
|
+
"deployment",
|
|
35
|
+
"custom-step",
|
|
36
|
+
"ci-cd",
|
|
37
|
+
"workflow",
|
|
38
|
+
"automation"
|
|
39
|
+
],
|
|
40
|
+
"author": "DevRamps",
|
|
41
|
+
"license": "MIT",
|
|
42
|
+
"repository": {
|
|
43
|
+
"type": "git",
|
|
44
|
+
"url": "git+https://github.com/Dev-Ramps/DevRamps-Step-Sdk-TypeScript.git"
|
|
45
|
+
},
|
|
46
|
+
"bugs": {
|
|
47
|
+
"url": "https://github.com/Dev-Ramps/DevRamps-Step-Sdk-TypeScript/issues"
|
|
48
|
+
},
|
|
49
|
+
"homepage": "https://github.com/Dev-Ramps/DevRamps-Step-Sdk-TypeScript",
|
|
50
|
+
"engines": {
|
|
51
|
+
"node": ">=18.0.0"
|
|
52
|
+
},
|
|
53
|
+
"dependencies": {
|
|
54
|
+
"minimist": "^1.2.8",
|
|
55
|
+
"zod": "^3.23.8"
|
|
56
|
+
},
|
|
57
|
+
"peerDependencies": {
|
|
58
|
+
"typescript": ">=5.0.0"
|
|
59
|
+
},
|
|
60
|
+
"devDependencies": {
|
|
61
|
+
"@eslint/js": "^9.17.0",
|
|
62
|
+
"@types/minimist": "^1.2.5",
|
|
63
|
+
"@types/node": "^22.10.0",
|
|
64
|
+
"@vitest/coverage-v8": "^2.1.0",
|
|
65
|
+
"eslint": "^9.17.0",
|
|
66
|
+
"typescript": "^5.7.0",
|
|
67
|
+
"typescript-eslint": "^8.18.0",
|
|
68
|
+
"vitest": "^2.1.0"
|
|
69
|
+
}
|
|
70
|
+
}
|