@kb-labs/workflow-entry 2.14.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/README.md +23 -0
- package/dist/commands/health.d.ts +10 -0
- package/dist/commands/health.js +215 -0
- package/dist/commands/health.js.map +1 -0
- package/dist/commands/list.d.ts +10 -0
- package/dist/commands/list.js +271 -0
- package/dist/commands/list.js.map +1 -0
- package/dist/commands/logs.d.ts +10 -0
- package/dist/commands/logs.js +230 -0
- package/dist/commands/logs.js.map +1 -0
- package/dist/commands/metrics.d.ts +10 -0
- package/dist/commands/metrics.js +228 -0
- package/dist/commands/metrics.js.map +1 -0
- package/dist/commands/run.d.ts +11 -0
- package/dist/commands/run.js +283 -0
- package/dist/commands/run.js.map +1 -0
- package/dist/commands/status.d.ts +10 -0
- package/dist/commands/status.js +266 -0
- package/dist/commands/status.js.map +1 -0
- package/dist/commands/workflow-run.d.ts +11 -0
- package/dist/commands/workflow-run.js +284 -0
- package/dist/commands/workflow-run.js.map +1 -0
- package/dist/flags-WLdpSZl4.d.ts +77 -0
- package/dist/index.d.ts +2 -0
- package/dist/index.js +545 -0
- package/dist/index.js.map +1 -0
- package/dist/manifest.d.ts +223 -0
- package/dist/manifest.js +545 -0
- package/dist/manifest.js.map +1 -0
- package/dist/rest/cron-list-handler.d.ts +19 -0
- package/dist/rest/cron-list-handler.js +48 -0
- package/dist/rest/cron-list-handler.js.map +1 -0
- package/dist/rest/index.d.ts +6 -0
- package/dist/rest/index.js +170 -0
- package/dist/rest/index.js.map +1 -0
- package/dist/rest/job-cancel-handler.d.ts +12 -0
- package/dist/rest/job-cancel-handler.js +52 -0
- package/dist/rest/job-cancel-handler.js.map +1 -0
- package/dist/rest/job-detail-handler.d.ts +22 -0
- package/dist/rest/job-detail-handler.js +52 -0
- package/dist/rest/job-detail-handler.js.map +1 -0
- package/dist/rest/job-logs-handler.d.ts +13 -0
- package/dist/rest/job-logs-handler.js +52 -0
- package/dist/rest/job-logs-handler.js.map +1 -0
- package/dist/rest/job-steps-handler.d.ts +9 -0
- package/dist/rest/job-steps-handler.js +43 -0
- package/dist/rest/job-steps-handler.js.map +1 -0
- package/dist/rest/jobs-list-handler.d.ts +26 -0
- package/dist/rest/jobs-list-handler.js +63 -0
- package/dist/rest/jobs-list-handler.js.map +1 -0
- package/dist/rest/pending-approvals-handler.d.ts +9 -0
- package/dist/rest/pending-approvals-handler.js +46 -0
- package/dist/rest/pending-approvals-handler.js.map +1 -0
- package/dist/rest/resolve-approval-handler.d.ts +16 -0
- package/dist/rest/resolve-approval-handler.js +58 -0
- package/dist/rest/resolve-approval-handler.js.map +1 -0
- package/dist/rest/run-detail-handler.d.ts +9 -0
- package/dist/rest/run-detail-handler.js +46 -0
- package/dist/rest/run-detail-handler.js.map +1 -0
- package/dist/rest/runs-list-handler.d.ts +11 -0
- package/dist/rest/runs-list-handler.js +50 -0
- package/dist/rest/runs-list-handler.js.map +1 -0
- package/dist/rest/stats-handler.d.ts +7 -0
- package/dist/rest/stats-handler.js +39 -0
- package/dist/rest/stats-handler.js.map +1 -0
- package/dist/rest/workflow-detail-handler.d.ts +10 -0
- package/dist/rest/workflow-detail-handler.js +46 -0
- package/dist/rest/workflow-detail-handler.js.map +1 -0
- package/dist/rest/workflow-run-cancel-handler.d.ts +13 -0
- package/dist/rest/workflow-run-cancel-handler.js +47 -0
- package/dist/rest/workflow-run-cancel-handler.js.map +1 -0
- package/dist/rest/workflow-run-handler.d.ts +14 -0
- package/dist/rest/workflow-run-handler.js +53 -0
- package/dist/rest/workflow-run-handler.js.map +1 -0
- package/dist/rest/workflow-runs-handler.d.ts +13 -0
- package/dist/rest/workflow-runs-handler.js +54 -0
- package/dist/rest/workflow-runs-handler.js.map +1 -0
- package/dist/rest/workflows-list-handler.d.ts +12 -0
- package/dist/rest/workflows-list-handler.js +51 -0
- package/dist/rest/workflows-list-handler.js.map +1 -0
- package/dist/ws/logs-channel.d.ts +17 -0
- package/dist/ws/logs-channel.js +133 -0
- package/dist/ws/logs-channel.js.map +1 -0
- package/dist/ws/progress-channel.d.ts +11 -0
- package/dist/ws/progress-channel.js +46 -0
- package/dist/ws/progress-channel.js.map +1 -0
- package/package.json +67 -0
package/README.md
ADDED
|
@@ -0,0 +1,23 @@
|
|
|
1
|
+
# @kb-labs/workflow-cli
|
|
2
|
+
|
|
3
|
+
CLI commands for KB Labs Workflow Daemon - interact with workflow daemon via HTTP API.
|
|
4
|
+
|
|
5
|
+
## Installation
|
|
6
|
+
|
|
7
|
+
```bash
|
|
8
|
+
pnpm add @kb-labs/workflow-cli
|
|
9
|
+
```
|
|
10
|
+
|
|
11
|
+
## Usage
|
|
12
|
+
|
|
13
|
+
```typescript
|
|
14
|
+
import { ... } from '@kb-labs/workflow-cli';
|
|
15
|
+
```
|
|
16
|
+
|
|
17
|
+
## API
|
|
18
|
+
|
|
19
|
+
See TypeScript types for detailed API documentation.
|
|
20
|
+
|
|
21
|
+
## License
|
|
22
|
+
|
|
23
|
+
MIT
|
|
@@ -0,0 +1,10 @@
|
|
|
1
|
+
import * as _kb_labs_shared_command_kit from '@kb-labs/shared-command-kit';
|
|
2
|
+
|
|
3
|
+
/**
|
|
4
|
+
* workflow:health command - Check daemon health
|
|
5
|
+
*/
|
|
6
|
+
declare const _default: _kb_labs_shared_command_kit.CommandHandlerV3<unknown, any, {
|
|
7
|
+
exitCode: number;
|
|
8
|
+
}>;
|
|
9
|
+
|
|
10
|
+
export { _default as default };
|
|
@@ -0,0 +1,215 @@
|
|
|
1
|
+
import { defineCommand } from '@kb-labs/sdk';
|
|
2
|
+
|
|
3
|
+
// src/commands/health.ts
|
|
4
|
+
|
|
5
|
+
// src/http-client.ts
|
|
6
|
+
var DEFAULT_DAEMON_URL = "http://localhost:7778";
|
|
7
|
+
var WorkflowDaemonClient = class {
|
|
8
|
+
baseUrl;
|
|
9
|
+
constructor(options = {}) {
|
|
10
|
+
this.baseUrl = options.url ?? process.env.WORKFLOW_DAEMON_URL ?? DEFAULT_DAEMON_URL;
|
|
11
|
+
}
|
|
12
|
+
/**
|
|
13
|
+
* Validate response Content-Type and parse JSON safely
|
|
14
|
+
*/
|
|
15
|
+
async parseJsonResponse(response) {
|
|
16
|
+
const contentType = response.headers.get("content-type");
|
|
17
|
+
if (!contentType?.includes("application/json")) {
|
|
18
|
+
throw new Error(`Invalid Content-Type: expected application/json, got ${contentType}`);
|
|
19
|
+
}
|
|
20
|
+
return response.json();
|
|
21
|
+
}
|
|
22
|
+
unwrapData(payload) {
|
|
23
|
+
if (payload && typeof payload === "object" && "ok" in payload && payload.ok === true && "data" in payload) {
|
|
24
|
+
return payload.data;
|
|
25
|
+
}
|
|
26
|
+
return payload;
|
|
27
|
+
}
|
|
28
|
+
/**
|
|
29
|
+
* Validate and encode job ID to prevent path traversal attacks
|
|
30
|
+
*/
|
|
31
|
+
validateAndEncodeJobId(jobId) {
|
|
32
|
+
if (!/^[a-zA-Z0-9_-]+$/.test(jobId)) {
|
|
33
|
+
throw new Error(`Invalid job ID format: ${jobId}`);
|
|
34
|
+
}
|
|
35
|
+
return encodeURIComponent(jobId);
|
|
36
|
+
}
|
|
37
|
+
/**
|
|
38
|
+
* Health check
|
|
39
|
+
*/
|
|
40
|
+
async health() {
|
|
41
|
+
const response = await fetch(`${this.baseUrl}/health`);
|
|
42
|
+
if (!response.ok) {
|
|
43
|
+
throw new Error(`Health check failed: ${response.statusText}`);
|
|
44
|
+
}
|
|
45
|
+
return this.parseJsonResponse(response);
|
|
46
|
+
}
|
|
47
|
+
/**
|
|
48
|
+
* Get workflow metrics
|
|
49
|
+
*/
|
|
50
|
+
async getMetrics() {
|
|
51
|
+
const response = await fetch(`${this.baseUrl}/metrics`);
|
|
52
|
+
if (!response.ok) {
|
|
53
|
+
throw new Error(`Failed to get metrics: ${response.statusText}`);
|
|
54
|
+
}
|
|
55
|
+
const data = await this.parseJsonResponse(response);
|
|
56
|
+
return this.unwrapData(data);
|
|
57
|
+
}
|
|
58
|
+
/**
|
|
59
|
+
* Get job status with full details (jobs, steps, outputs)
|
|
60
|
+
*/
|
|
61
|
+
async getJobStatus(jobId) {
|
|
62
|
+
const encodedJobId = this.validateAndEncodeJobId(jobId);
|
|
63
|
+
const response = await fetch(`${this.baseUrl}/api/v1/jobs/${encodedJobId}`);
|
|
64
|
+
if (response.status === 404) {
|
|
65
|
+
throw new Error(`Job ${jobId} not found`);
|
|
66
|
+
}
|
|
67
|
+
if (!response.ok) {
|
|
68
|
+
throw new Error(`Failed to get job status: ${response.statusText}`);
|
|
69
|
+
}
|
|
70
|
+
const data = await this.parseJsonResponse(response);
|
|
71
|
+
return this.unwrapData(data);
|
|
72
|
+
}
|
|
73
|
+
/**
|
|
74
|
+
* Get job steps with outputs
|
|
75
|
+
*/
|
|
76
|
+
async getJobSteps(jobId) {
|
|
77
|
+
const encodedJobId = this.validateAndEncodeJobId(jobId);
|
|
78
|
+
const response = await fetch(`${this.baseUrl}/api/v1/jobs/${encodedJobId}/steps`);
|
|
79
|
+
if (response.status === 404) {
|
|
80
|
+
throw new Error(`Job ${jobId} not found`);
|
|
81
|
+
}
|
|
82
|
+
if (!response.ok) {
|
|
83
|
+
throw new Error(`Failed to get job steps: ${response.statusText}`);
|
|
84
|
+
}
|
|
85
|
+
const data = await this.parseJsonResponse(response);
|
|
86
|
+
return this.unwrapData(data);
|
|
87
|
+
}
|
|
88
|
+
/**
|
|
89
|
+
* Get job logs
|
|
90
|
+
*/
|
|
91
|
+
async getJobLogs(jobId) {
|
|
92
|
+
const encodedJobId = this.validateAndEncodeJobId(jobId);
|
|
93
|
+
const response = await fetch(`${this.baseUrl}/api/v1/jobs/${encodedJobId}/logs`);
|
|
94
|
+
if (response.status === 404) {
|
|
95
|
+
throw new Error(`Job ${jobId} not found`);
|
|
96
|
+
}
|
|
97
|
+
if (!response.ok) {
|
|
98
|
+
throw new Error(`Failed to get job logs: ${response.statusText}`);
|
|
99
|
+
}
|
|
100
|
+
const data = await this.parseJsonResponse(response);
|
|
101
|
+
const unwrapped = this.unwrapData(data);
|
|
102
|
+
return unwrapped.logs ?? [];
|
|
103
|
+
}
|
|
104
|
+
/**
|
|
105
|
+
* Get active executions
|
|
106
|
+
*/
|
|
107
|
+
async getExecutions() {
|
|
108
|
+
const response = await fetch(`${this.baseUrl}/api/v1/jobs`);
|
|
109
|
+
if (!response.ok) {
|
|
110
|
+
throw new Error(`Failed to get executions: ${response.statusText}`);
|
|
111
|
+
}
|
|
112
|
+
const data = await this.parseJsonResponse(response);
|
|
113
|
+
const unwrapped = this.unwrapData(data);
|
|
114
|
+
const jobs = unwrapped.jobs ?? [];
|
|
115
|
+
return jobs.filter((job) => job.status === "running" || job.status === "pending");
|
|
116
|
+
}
|
|
117
|
+
/**
|
|
118
|
+
* Get cron jobs
|
|
119
|
+
*/
|
|
120
|
+
async getCronJobs() {
|
|
121
|
+
const response = await fetch(`${this.baseUrl}/api/v1/cron`);
|
|
122
|
+
if (!response.ok) {
|
|
123
|
+
throw new Error(`Failed to get cron jobs: ${response.statusText}`);
|
|
124
|
+
}
|
|
125
|
+
const data = await this.parseJsonResponse(response);
|
|
126
|
+
return this.unwrapData(data);
|
|
127
|
+
}
|
|
128
|
+
/**
|
|
129
|
+
* Submit a job for execution
|
|
130
|
+
*/
|
|
131
|
+
async submitJob(params) {
|
|
132
|
+
const response = await fetch(`${this.baseUrl}/api/v1/jobs`, {
|
|
133
|
+
method: "POST",
|
|
134
|
+
headers: {
|
|
135
|
+
"Content-Type": "application/json"
|
|
136
|
+
},
|
|
137
|
+
body: JSON.stringify({
|
|
138
|
+
type: params.handler,
|
|
139
|
+
payload: params.input,
|
|
140
|
+
priority: params.priority
|
|
141
|
+
})
|
|
142
|
+
});
|
|
143
|
+
if (!response.ok) {
|
|
144
|
+
const error = await response.json().catch(() => ({ error: response.statusText }));
|
|
145
|
+
throw new Error(error.error || `Failed to submit job: ${response.statusText}`);
|
|
146
|
+
}
|
|
147
|
+
const payload = await this.parseJsonResponse(response);
|
|
148
|
+
const data = this.unwrapData(payload);
|
|
149
|
+
return { id: data.jobId, status: "pending" };
|
|
150
|
+
}
|
|
151
|
+
/**
|
|
152
|
+
* Run workflow by ID
|
|
153
|
+
*/
|
|
154
|
+
async runWorkflow(workflowId, request = {}) {
|
|
155
|
+
const response = await fetch(
|
|
156
|
+
`${this.baseUrl}/api/v1/workflows/${encodeURIComponent(workflowId)}/run`,
|
|
157
|
+
{
|
|
158
|
+
method: "POST",
|
|
159
|
+
headers: {
|
|
160
|
+
"Content-Type": "application/json"
|
|
161
|
+
},
|
|
162
|
+
body: JSON.stringify(request)
|
|
163
|
+
}
|
|
164
|
+
);
|
|
165
|
+
if (!response.ok) {
|
|
166
|
+
const error = await response.json().catch(() => ({ error: response.statusText }));
|
|
167
|
+
throw new Error(error.error || `Failed to run workflow: ${response.statusText}`);
|
|
168
|
+
}
|
|
169
|
+
const payload = await this.parseJsonResponse(response);
|
|
170
|
+
return this.unwrapData(payload);
|
|
171
|
+
}
|
|
172
|
+
};
|
|
173
|
+
|
|
174
|
+
// src/commands/health.ts
|
|
175
|
+
var health_default = defineCommand({
|
|
176
|
+
id: "workflow:health",
|
|
177
|
+
description: "Check workflow daemon health status",
|
|
178
|
+
handler: {
|
|
179
|
+
async execute(ctx, input) {
|
|
180
|
+
const flags = input.flags ?? input;
|
|
181
|
+
const outputJson = flags.json ?? false;
|
|
182
|
+
try {
|
|
183
|
+
const client = new WorkflowDaemonClient();
|
|
184
|
+
const health = await client.health();
|
|
185
|
+
if (outputJson) {
|
|
186
|
+
ctx.ui?.json?.({ ok: true, data: health });
|
|
187
|
+
} else {
|
|
188
|
+
ctx.ui?.success?.("Daemon is healthy", {
|
|
189
|
+
title: "Workflow Daemon",
|
|
190
|
+
sections: [
|
|
191
|
+
{
|
|
192
|
+
header: "Status",
|
|
193
|
+
items: [`Service: ${health.service}`, "Health: OK"]
|
|
194
|
+
}
|
|
195
|
+
]
|
|
196
|
+
});
|
|
197
|
+
}
|
|
198
|
+
return { exitCode: 0 };
|
|
199
|
+
} catch (error) {
|
|
200
|
+
const message = error instanceof Error ? error.message : String(error);
|
|
201
|
+
if (outputJson) {
|
|
202
|
+
ctx.ui?.json?.({ ok: false, error: message });
|
|
203
|
+
} else {
|
|
204
|
+
ctx.ui?.error?.(`Failed to check daemon health: ${message}`);
|
|
205
|
+
ctx.ui?.warn?.(`Make sure workflow daemon is running: kb-workflow`);
|
|
206
|
+
}
|
|
207
|
+
return { exitCode: 1 };
|
|
208
|
+
}
|
|
209
|
+
}
|
|
210
|
+
}
|
|
211
|
+
});
|
|
212
|
+
|
|
213
|
+
export { health_default as default };
|
|
214
|
+
//# sourceMappingURL=health.js.map
|
|
215
|
+
//# sourceMappingURL=health.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"sources":["../../src/http-client.ts","../../src/commands/health.ts"],"names":[],"mappings":"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","file":"health.js","sourcesContent":["/**\n * HTTP client for interacting with Workflow Daemon\n */\nimport type { WorkflowRunRequest } from '@kb-labs/workflow-contracts';\n\nconst DEFAULT_DAEMON_URL = 'http://localhost:7778';\n\nexport interface DaemonClientOptions {\n url?: string;\n}\n\n/**\n * Get workflow daemon URL from environment or default\n */\nexport function getWorkflowDaemonUrl(): string {\n return process.env.WORKFLOW_DAEMON_URL ?? DEFAULT_DAEMON_URL;\n}\n\nexport class WorkflowDaemonClient {\n private readonly baseUrl: string;\n\n constructor(options: DaemonClientOptions = {}) {\n this.baseUrl = options.url ?? process.env.WORKFLOW_DAEMON_URL ?? DEFAULT_DAEMON_URL;\n }\n\n /**\n * Validate response Content-Type and parse JSON safely\n */\n private async parseJsonResponse<T>(response: Response): Promise<T> {\n const contentType = response.headers.get('content-type');\n if (!contentType?.includes('application/json')) {\n throw new Error(`Invalid Content-Type: expected application/json, got ${contentType}`);\n }\n return response.json() as Promise<T>;\n }\n\n private unwrapData<T>(payload: unknown): T {\n if (\n payload\n && typeof payload === 'object'\n && 'ok' in payload\n && (payload as any).ok === true\n && 'data' in payload\n ) {\n return (payload as { data: T }).data;\n }\n return payload as T;\n }\n\n /**\n * Validate and encode job ID to prevent path traversal attacks\n */\n private validateAndEncodeJobId(jobId: string): string {\n // Validate job ID format (alphanumeric, hyphens, underscores only)\n if (!/^[a-zA-Z0-9_-]+$/.test(jobId)) {\n throw new Error(`Invalid job ID format: ${jobId}`);\n }\n // Encode for URL safety (defense in depth)\n return encodeURIComponent(jobId);\n }\n\n /**\n * Health check\n */\n async health(): Promise<{ ok: boolean; service: string }> {\n const response = await fetch(`${this.baseUrl}/health`);\n if (!response.ok) {\n throw new Error(`Health check failed: ${response.statusText}`);\n }\n return this.parseJsonResponse(response);\n }\n\n /**\n * Get workflow metrics\n */\n async getMetrics(): Promise<any> {\n const response = await fetch(`${this.baseUrl}/metrics`);\n if (!response.ok) {\n throw new Error(`Failed to get metrics: ${response.statusText}`);\n }\n const data = await this.parseJsonResponse<any>(response);\n return this.unwrapData(data);\n }\n\n /**\n * Get job status with full details (jobs, steps, outputs)\n */\n async getJobStatus(jobId: string): Promise<any> {\n const encodedJobId = this.validateAndEncodeJobId(jobId);\n const response = await fetch(`${this.baseUrl}/api/v1/jobs/${encodedJobId}`);\n if (response.status === 404) {\n throw new Error(`Job ${jobId} not found`);\n }\n if (!response.ok) {\n throw new Error(`Failed to get job status: ${response.statusText}`);\n }\n const data = await this.parseJsonResponse<any>(response);\n return this.unwrapData(data);\n }\n\n /**\n * Get job steps with outputs\n */\n async getJobSteps(jobId: string): Promise<any> {\n const encodedJobId = this.validateAndEncodeJobId(jobId);\n const response = await fetch(`${this.baseUrl}/api/v1/jobs/${encodedJobId}/steps`);\n if (response.status === 404) {\n throw new Error(`Job ${jobId} not found`);\n }\n if (!response.ok) {\n throw new Error(`Failed to get job steps: ${response.statusText}`);\n }\n const data = await this.parseJsonResponse<any>(response);\n return this.unwrapData(data);\n }\n\n /**\n * Get job logs\n */\n async getJobLogs(jobId: string): Promise<any[]> {\n const encodedJobId = this.validateAndEncodeJobId(jobId);\n const response = await fetch(`${this.baseUrl}/api/v1/jobs/${encodedJobId}/logs`);\n if (response.status === 404) {\n throw new Error(`Job ${jobId} not found`);\n }\n if (!response.ok) {\n throw new Error(`Failed to get job logs: ${response.statusText}`);\n }\n const data = await this.parseJsonResponse<any>(response);\n const unwrapped = this.unwrapData<{ logs: any[] }>(data);\n return unwrapped.logs ?? [];\n }\n\n /**\n * Get active executions\n */\n async getExecutions(): Promise<any[]> {\n const response = await fetch(`${this.baseUrl}/api/v1/jobs`);\n if (!response.ok) {\n throw new Error(`Failed to get executions: ${response.statusText}`);\n }\n const data = await this.parseJsonResponse<any>(response);\n const unwrapped = this.unwrapData<{ jobs?: any[] }>(data);\n const jobs = unwrapped.jobs ?? [];\n return jobs.filter((job: any) => job.status === 'running' || job.status === 'pending');\n }\n\n /**\n * Get cron jobs\n */\n async getCronJobs(): Promise<{\n crons: any[];\n }> {\n const response = await fetch(`${this.baseUrl}/api/v1/cron`);\n if (!response.ok) {\n throw new Error(`Failed to get cron jobs: ${response.statusText}`);\n }\n const data = await this.parseJsonResponse<any>(response);\n return this.unwrapData(data);\n }\n\n /**\n * Submit a job for execution\n */\n async submitJob(params: {\n handler: string;\n input?: unknown;\n priority?: number;\n }): Promise<{ id: string; status: string }> {\n const response = await fetch(`${this.baseUrl}/api/v1/jobs`, {\n method: 'POST',\n headers: {\n 'Content-Type': 'application/json',\n },\n body: JSON.stringify({\n type: params.handler,\n payload: params.input,\n priority: params.priority,\n }),\n });\n\n if (!response.ok) {\n const error = (await response.json().catch(() => ({ error: response.statusText }))) as {\n error?: string;\n };\n throw new Error(error.error || `Failed to submit job: ${response.statusText}`);\n }\n\n const payload = await this.parseJsonResponse<any>(response);\n const data = this.unwrapData<{ jobId: string }>(payload);\n return { id: data.jobId, status: 'pending' };\n }\n\n /**\n * Run workflow by ID\n */\n async runWorkflow(\n workflowId: string,\n request: WorkflowRunRequest = {}\n ): Promise<{ runId: string; status: string }> {\n const response = await fetch(\n `${this.baseUrl}/api/v1/workflows/${encodeURIComponent(workflowId)}/run`,\n {\n method: 'POST',\n headers: {\n 'Content-Type': 'application/json',\n },\n body: JSON.stringify(request),\n }\n );\n\n if (!response.ok) {\n const error = (await response.json().catch(() => ({ error: response.statusText }))) as {\n error?: string;\n };\n throw new Error(error.error || `Failed to run workflow: ${response.statusText}`);\n }\n\n const payload = await this.parseJsonResponse<{ ok: boolean; data?: { runId: string; status: string } }>(response);\n return this.unwrapData<{ runId: string; status: string }>(payload);\n }\n}\n","/**\n * workflow:health command - Check daemon health\n */\n\nimport { defineCommand, type PluginContextV3 } from '@kb-labs/sdk';\nimport { type HealthFlags } from '@kb-labs/workflow-contracts';\nimport { WorkflowDaemonClient } from '../http-client.js';\n\ntype HealthInput = HealthFlags & { argv?: string[] };\n\nexport default defineCommand<unknown, HealthInput, { exitCode: number }>({\n id: 'workflow:health',\n description: 'Check workflow daemon health status',\n\n handler: {\n async execute(ctx: PluginContextV3, input: HealthInput): Promise<{ exitCode: number }> {\n const flags = (input as any).flags ?? input;\n const outputJson = flags.json ?? false;\n\n try {\n const client = new WorkflowDaemonClient();\n const health = await client.health();\n\n if (outputJson) {\n ctx.ui?.json?.({ ok: true, data: health });\n } else {\n ctx.ui?.success?.('Daemon is healthy', {\n title: 'Workflow Daemon',\n sections: [\n {\n header: 'Status',\n items: [`Service: ${health.service}`, 'Health: OK'],\n },\n ],\n });\n }\n\n return { exitCode: 0 };\n } catch (error) {\n const message = error instanceof Error ? error.message : String(error);\n\n if (outputJson) {\n ctx.ui?.json?.({ ok: false, error: message });\n } else {\n ctx.ui?.error?.(`Failed to check daemon health: ${message}`);\n ctx.ui?.warn?.(`Make sure workflow daemon is running: kb-workflow`);\n }\n\n return { exitCode: 1 };\n }\n },\n },\n});\n"]}
|
|
@@ -0,0 +1,10 @@
|
|
|
1
|
+
import * as _kb_labs_shared_command_kit from '@kb-labs/shared-command-kit';
|
|
2
|
+
|
|
3
|
+
/**
|
|
4
|
+
* workflow:list command - List active executions
|
|
5
|
+
*/
|
|
6
|
+
declare const _default: _kb_labs_shared_command_kit.CommandHandlerV3<unknown, any, {
|
|
7
|
+
exitCode: number;
|
|
8
|
+
}>;
|
|
9
|
+
|
|
10
|
+
export { _default as default };
|
|
@@ -0,0 +1,271 @@
|
|
|
1
|
+
import { defineCommand } from '@kb-labs/sdk';
|
|
2
|
+
|
|
3
|
+
// src/commands/list.ts
|
|
4
|
+
|
|
5
|
+
// src/http-client.ts
|
|
6
|
+
var DEFAULT_DAEMON_URL = "http://localhost:7778";
|
|
7
|
+
var WorkflowDaemonClient = class {
|
|
8
|
+
baseUrl;
|
|
9
|
+
constructor(options = {}) {
|
|
10
|
+
this.baseUrl = options.url ?? process.env.WORKFLOW_DAEMON_URL ?? DEFAULT_DAEMON_URL;
|
|
11
|
+
}
|
|
12
|
+
/**
|
|
13
|
+
* Validate response Content-Type and parse JSON safely
|
|
14
|
+
*/
|
|
15
|
+
async parseJsonResponse(response) {
|
|
16
|
+
const contentType = response.headers.get("content-type");
|
|
17
|
+
if (!contentType?.includes("application/json")) {
|
|
18
|
+
throw new Error(`Invalid Content-Type: expected application/json, got ${contentType}`);
|
|
19
|
+
}
|
|
20
|
+
return response.json();
|
|
21
|
+
}
|
|
22
|
+
unwrapData(payload) {
|
|
23
|
+
if (payload && typeof payload === "object" && "ok" in payload && payload.ok === true && "data" in payload) {
|
|
24
|
+
return payload.data;
|
|
25
|
+
}
|
|
26
|
+
return payload;
|
|
27
|
+
}
|
|
28
|
+
/**
|
|
29
|
+
* Validate and encode job ID to prevent path traversal attacks
|
|
30
|
+
*/
|
|
31
|
+
validateAndEncodeJobId(jobId) {
|
|
32
|
+
if (!/^[a-zA-Z0-9_-]+$/.test(jobId)) {
|
|
33
|
+
throw new Error(`Invalid job ID format: ${jobId}`);
|
|
34
|
+
}
|
|
35
|
+
return encodeURIComponent(jobId);
|
|
36
|
+
}
|
|
37
|
+
/**
|
|
38
|
+
* Health check
|
|
39
|
+
*/
|
|
40
|
+
async health() {
|
|
41
|
+
const response = await fetch(`${this.baseUrl}/health`);
|
|
42
|
+
if (!response.ok) {
|
|
43
|
+
throw new Error(`Health check failed: ${response.statusText}`);
|
|
44
|
+
}
|
|
45
|
+
return this.parseJsonResponse(response);
|
|
46
|
+
}
|
|
47
|
+
/**
|
|
48
|
+
* Get workflow metrics
|
|
49
|
+
*/
|
|
50
|
+
async getMetrics() {
|
|
51
|
+
const response = await fetch(`${this.baseUrl}/metrics`);
|
|
52
|
+
if (!response.ok) {
|
|
53
|
+
throw new Error(`Failed to get metrics: ${response.statusText}`);
|
|
54
|
+
}
|
|
55
|
+
const data = await this.parseJsonResponse(response);
|
|
56
|
+
return this.unwrapData(data);
|
|
57
|
+
}
|
|
58
|
+
/**
|
|
59
|
+
* Get job status with full details (jobs, steps, outputs)
|
|
60
|
+
*/
|
|
61
|
+
async getJobStatus(jobId) {
|
|
62
|
+
const encodedJobId = this.validateAndEncodeJobId(jobId);
|
|
63
|
+
const response = await fetch(`${this.baseUrl}/api/v1/jobs/${encodedJobId}`);
|
|
64
|
+
if (response.status === 404) {
|
|
65
|
+
throw new Error(`Job ${jobId} not found`);
|
|
66
|
+
}
|
|
67
|
+
if (!response.ok) {
|
|
68
|
+
throw new Error(`Failed to get job status: ${response.statusText}`);
|
|
69
|
+
}
|
|
70
|
+
const data = await this.parseJsonResponse(response);
|
|
71
|
+
return this.unwrapData(data);
|
|
72
|
+
}
|
|
73
|
+
/**
|
|
74
|
+
* Get job steps with outputs
|
|
75
|
+
*/
|
|
76
|
+
async getJobSteps(jobId) {
|
|
77
|
+
const encodedJobId = this.validateAndEncodeJobId(jobId);
|
|
78
|
+
const response = await fetch(`${this.baseUrl}/api/v1/jobs/${encodedJobId}/steps`);
|
|
79
|
+
if (response.status === 404) {
|
|
80
|
+
throw new Error(`Job ${jobId} not found`);
|
|
81
|
+
}
|
|
82
|
+
if (!response.ok) {
|
|
83
|
+
throw new Error(`Failed to get job steps: ${response.statusText}`);
|
|
84
|
+
}
|
|
85
|
+
const data = await this.parseJsonResponse(response);
|
|
86
|
+
return this.unwrapData(data);
|
|
87
|
+
}
|
|
88
|
+
/**
|
|
89
|
+
* Get job logs
|
|
90
|
+
*/
|
|
91
|
+
async getJobLogs(jobId) {
|
|
92
|
+
const encodedJobId = this.validateAndEncodeJobId(jobId);
|
|
93
|
+
const response = await fetch(`${this.baseUrl}/api/v1/jobs/${encodedJobId}/logs`);
|
|
94
|
+
if (response.status === 404) {
|
|
95
|
+
throw new Error(`Job ${jobId} not found`);
|
|
96
|
+
}
|
|
97
|
+
if (!response.ok) {
|
|
98
|
+
throw new Error(`Failed to get job logs: ${response.statusText}`);
|
|
99
|
+
}
|
|
100
|
+
const data = await this.parseJsonResponse(response);
|
|
101
|
+
const unwrapped = this.unwrapData(data);
|
|
102
|
+
return unwrapped.logs ?? [];
|
|
103
|
+
}
|
|
104
|
+
/**
|
|
105
|
+
* Get active executions
|
|
106
|
+
*/
|
|
107
|
+
async getExecutions() {
|
|
108
|
+
const response = await fetch(`${this.baseUrl}/api/v1/jobs`);
|
|
109
|
+
if (!response.ok) {
|
|
110
|
+
throw new Error(`Failed to get executions: ${response.statusText}`);
|
|
111
|
+
}
|
|
112
|
+
const data = await this.parseJsonResponse(response);
|
|
113
|
+
const unwrapped = this.unwrapData(data);
|
|
114
|
+
const jobs = unwrapped.jobs ?? [];
|
|
115
|
+
return jobs.filter((job) => job.status === "running" || job.status === "pending");
|
|
116
|
+
}
|
|
117
|
+
/**
|
|
118
|
+
* Get cron jobs
|
|
119
|
+
*/
|
|
120
|
+
async getCronJobs() {
|
|
121
|
+
const response = await fetch(`${this.baseUrl}/api/v1/cron`);
|
|
122
|
+
if (!response.ok) {
|
|
123
|
+
throw new Error(`Failed to get cron jobs: ${response.statusText}`);
|
|
124
|
+
}
|
|
125
|
+
const data = await this.parseJsonResponse(response);
|
|
126
|
+
return this.unwrapData(data);
|
|
127
|
+
}
|
|
128
|
+
/**
|
|
129
|
+
* Submit a job for execution
|
|
130
|
+
*/
|
|
131
|
+
async submitJob(params) {
|
|
132
|
+
const response = await fetch(`${this.baseUrl}/api/v1/jobs`, {
|
|
133
|
+
method: "POST",
|
|
134
|
+
headers: {
|
|
135
|
+
"Content-Type": "application/json"
|
|
136
|
+
},
|
|
137
|
+
body: JSON.stringify({
|
|
138
|
+
type: params.handler,
|
|
139
|
+
payload: params.input,
|
|
140
|
+
priority: params.priority
|
|
141
|
+
})
|
|
142
|
+
});
|
|
143
|
+
if (!response.ok) {
|
|
144
|
+
const error = await response.json().catch(() => ({ error: response.statusText }));
|
|
145
|
+
throw new Error(error.error || `Failed to submit job: ${response.statusText}`);
|
|
146
|
+
}
|
|
147
|
+
const payload = await this.parseJsonResponse(response);
|
|
148
|
+
const data = this.unwrapData(payload);
|
|
149
|
+
return { id: data.jobId, status: "pending" };
|
|
150
|
+
}
|
|
151
|
+
/**
|
|
152
|
+
* Run workflow by ID
|
|
153
|
+
*/
|
|
154
|
+
async runWorkflow(workflowId, request = {}) {
|
|
155
|
+
const response = await fetch(
|
|
156
|
+
`${this.baseUrl}/api/v1/workflows/${encodeURIComponent(workflowId)}/run`,
|
|
157
|
+
{
|
|
158
|
+
method: "POST",
|
|
159
|
+
headers: {
|
|
160
|
+
"Content-Type": "application/json"
|
|
161
|
+
},
|
|
162
|
+
body: JSON.stringify(request)
|
|
163
|
+
}
|
|
164
|
+
);
|
|
165
|
+
if (!response.ok) {
|
|
166
|
+
const error = await response.json().catch(() => ({ error: response.statusText }));
|
|
167
|
+
throw new Error(error.error || `Failed to run workflow: ${response.statusText}`);
|
|
168
|
+
}
|
|
169
|
+
const payload = await this.parseJsonResponse(response);
|
|
170
|
+
return this.unwrapData(payload);
|
|
171
|
+
}
|
|
172
|
+
};
|
|
173
|
+
|
|
174
|
+
// src/commands/list.ts
|
|
175
|
+
var list_default = defineCommand({
|
|
176
|
+
id: "workflow:list",
|
|
177
|
+
description: "List active workflow executions",
|
|
178
|
+
handler: {
|
|
179
|
+
// eslint-disable-next-line sonarjs/cognitive-complexity -- Workflow listing with filtering (status/type), JSON/human output formats, run state formatting, and error handling
|
|
180
|
+
async execute(ctx, input) {
|
|
181
|
+
const flags = input.flags ?? input;
|
|
182
|
+
const outputJson = flags.json ?? false;
|
|
183
|
+
const statusFilter = flags.status;
|
|
184
|
+
const typeFilter = flags.type;
|
|
185
|
+
try {
|
|
186
|
+
const client = new WorkflowDaemonClient();
|
|
187
|
+
if (typeFilter === "cron") {
|
|
188
|
+
const result = await client.getCronJobs();
|
|
189
|
+
const cronJobs = result.crons ?? [];
|
|
190
|
+
if (outputJson) {
|
|
191
|
+
ctx.ui?.json?.({ ok: true, data: result });
|
|
192
|
+
} else {
|
|
193
|
+
if (cronJobs.length === 0) {
|
|
194
|
+
ctx.ui?.warn?.("No cron jobs found");
|
|
195
|
+
ctx.ui?.info?.("");
|
|
196
|
+
ctx.ui?.info?.("To add cron jobs:");
|
|
197
|
+
ctx.ui?.info?.(' 1. Plugin manifests: Add "cron" section to manifest.ts');
|
|
198
|
+
ctx.ui?.info?.(" 2. User YAML: Create .kb/jobs/*.yml files");
|
|
199
|
+
} else {
|
|
200
|
+
const jobItems = cronJobs.map((job) => {
|
|
201
|
+
const parts = [
|
|
202
|
+
`ID: ${job.id}`,
|
|
203
|
+
`Schedule: ${job.schedule} (${job.timezone || "UTC"})`,
|
|
204
|
+
`Enabled: ${job.enabled ? "Yes" : "No"}`,
|
|
205
|
+
`Type: ${job.jobType || "unknown"}`
|
|
206
|
+
];
|
|
207
|
+
return parts.join(" | ");
|
|
208
|
+
});
|
|
209
|
+
const summaryItems = [
|
|
210
|
+
`Total Jobs: ${cronJobs.length}`
|
|
211
|
+
];
|
|
212
|
+
ctx.ui?.success?.("Cron Jobs", {
|
|
213
|
+
title: "Workflow Scheduler",
|
|
214
|
+
sections: [
|
|
215
|
+
{ header: "Summary", items: summaryItems },
|
|
216
|
+
{ header: "Registered Jobs", items: jobItems }
|
|
217
|
+
]
|
|
218
|
+
});
|
|
219
|
+
}
|
|
220
|
+
}
|
|
221
|
+
return { exitCode: 0 };
|
|
222
|
+
}
|
|
223
|
+
let executions = await client.getExecutions();
|
|
224
|
+
if (statusFilter) {
|
|
225
|
+
executions = executions.filter((exec) => exec.status === statusFilter);
|
|
226
|
+
}
|
|
227
|
+
if (outputJson) {
|
|
228
|
+
ctx.ui?.json?.({ ok: true, data: { executions } });
|
|
229
|
+
} else {
|
|
230
|
+
if (executions.length === 0) {
|
|
231
|
+
ctx.ui?.warn?.("No active executions found");
|
|
232
|
+
} else {
|
|
233
|
+
const executionItems = executions.map((exec) => {
|
|
234
|
+
const parts = [
|
|
235
|
+
`ID: ${exec.id}`,
|
|
236
|
+
`Status: ${exec.status}`,
|
|
237
|
+
`Started: ${exec.startedAt || "N/A"}`
|
|
238
|
+
];
|
|
239
|
+
return parts.join(" | ");
|
|
240
|
+
});
|
|
241
|
+
const summaryItems = [
|
|
242
|
+
`Total Executions: ${executions.length}`,
|
|
243
|
+
statusFilter ? `Filter: ${statusFilter}` : void 0
|
|
244
|
+
].filter(Boolean);
|
|
245
|
+
ctx.ui?.success?.("Active Workflow Executions", {
|
|
246
|
+
title: "Workflow Engine",
|
|
247
|
+
sections: [
|
|
248
|
+
{ header: "Summary", items: summaryItems },
|
|
249
|
+
{ header: "Executions", items: executionItems }
|
|
250
|
+
]
|
|
251
|
+
});
|
|
252
|
+
}
|
|
253
|
+
}
|
|
254
|
+
return { exitCode: 0 };
|
|
255
|
+
} catch (error) {
|
|
256
|
+
const message = error instanceof Error ? error.message : String(error);
|
|
257
|
+
if (outputJson) {
|
|
258
|
+
ctx.ui?.json?.({ ok: false, error: message });
|
|
259
|
+
} else {
|
|
260
|
+
ctx.ui?.error?.(`Failed to list: ${message}`);
|
|
261
|
+
ctx.ui?.warn?.(`Make sure workflow daemon is running: kb-workflow`);
|
|
262
|
+
}
|
|
263
|
+
return { exitCode: 1 };
|
|
264
|
+
}
|
|
265
|
+
}
|
|
266
|
+
}
|
|
267
|
+
});
|
|
268
|
+
|
|
269
|
+
export { list_default as default };
|
|
270
|
+
//# sourceMappingURL=list.js.map
|
|
271
|
+
//# sourceMappingURL=list.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"sources":["../../src/http-client.ts","../../src/commands/list.ts"],"names":[],"mappings":"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","file":"list.js","sourcesContent":["/**\n * HTTP client for interacting with Workflow Daemon\n */\nimport type { WorkflowRunRequest } from '@kb-labs/workflow-contracts';\n\nconst DEFAULT_DAEMON_URL = 'http://localhost:7778';\n\nexport interface DaemonClientOptions {\n url?: string;\n}\n\n/**\n * Get workflow daemon URL from environment or default\n */\nexport function getWorkflowDaemonUrl(): string {\n return process.env.WORKFLOW_DAEMON_URL ?? DEFAULT_DAEMON_URL;\n}\n\nexport class WorkflowDaemonClient {\n private readonly baseUrl: string;\n\n constructor(options: DaemonClientOptions = {}) {\n this.baseUrl = options.url ?? process.env.WORKFLOW_DAEMON_URL ?? DEFAULT_DAEMON_URL;\n }\n\n /**\n * Validate response Content-Type and parse JSON safely\n */\n private async parseJsonResponse<T>(response: Response): Promise<T> {\n const contentType = response.headers.get('content-type');\n if (!contentType?.includes('application/json')) {\n throw new Error(`Invalid Content-Type: expected application/json, got ${contentType}`);\n }\n return response.json() as Promise<T>;\n }\n\n private unwrapData<T>(payload: unknown): T {\n if (\n payload\n && typeof payload === 'object'\n && 'ok' in payload\n && (payload as any).ok === true\n && 'data' in payload\n ) {\n return (payload as { data: T }).data;\n }\n return payload as T;\n }\n\n /**\n * Validate and encode job ID to prevent path traversal attacks\n */\n private validateAndEncodeJobId(jobId: string): string {\n // Validate job ID format (alphanumeric, hyphens, underscores only)\n if (!/^[a-zA-Z0-9_-]+$/.test(jobId)) {\n throw new Error(`Invalid job ID format: ${jobId}`);\n }\n // Encode for URL safety (defense in depth)\n return encodeURIComponent(jobId);\n }\n\n /**\n * Health check\n */\n async health(): Promise<{ ok: boolean; service: string }> {\n const response = await fetch(`${this.baseUrl}/health`);\n if (!response.ok) {\n throw new Error(`Health check failed: ${response.statusText}`);\n }\n return this.parseJsonResponse(response);\n }\n\n /**\n * Get workflow metrics\n */\n async getMetrics(): Promise<any> {\n const response = await fetch(`${this.baseUrl}/metrics`);\n if (!response.ok) {\n throw new Error(`Failed to get metrics: ${response.statusText}`);\n }\n const data = await this.parseJsonResponse<any>(response);\n return this.unwrapData(data);\n }\n\n /**\n * Get job status with full details (jobs, steps, outputs)\n */\n async getJobStatus(jobId: string): Promise<any> {\n const encodedJobId = this.validateAndEncodeJobId(jobId);\n const response = await fetch(`${this.baseUrl}/api/v1/jobs/${encodedJobId}`);\n if (response.status === 404) {\n throw new Error(`Job ${jobId} not found`);\n }\n if (!response.ok) {\n throw new Error(`Failed to get job status: ${response.statusText}`);\n }\n const data = await this.parseJsonResponse<any>(response);\n return this.unwrapData(data);\n }\n\n /**\n * Get job steps with outputs\n */\n async getJobSteps(jobId: string): Promise<any> {\n const encodedJobId = this.validateAndEncodeJobId(jobId);\n const response = await fetch(`${this.baseUrl}/api/v1/jobs/${encodedJobId}/steps`);\n if (response.status === 404) {\n throw new Error(`Job ${jobId} not found`);\n }\n if (!response.ok) {\n throw new Error(`Failed to get job steps: ${response.statusText}`);\n }\n const data = await this.parseJsonResponse<any>(response);\n return this.unwrapData(data);\n }\n\n /**\n * Get job logs\n */\n async getJobLogs(jobId: string): Promise<any[]> {\n const encodedJobId = this.validateAndEncodeJobId(jobId);\n const response = await fetch(`${this.baseUrl}/api/v1/jobs/${encodedJobId}/logs`);\n if (response.status === 404) {\n throw new Error(`Job ${jobId} not found`);\n }\n if (!response.ok) {\n throw new Error(`Failed to get job logs: ${response.statusText}`);\n }\n const data = await this.parseJsonResponse<any>(response);\n const unwrapped = this.unwrapData<{ logs: any[] }>(data);\n return unwrapped.logs ?? [];\n }\n\n /**\n * Get active executions\n */\n async getExecutions(): Promise<any[]> {\n const response = await fetch(`${this.baseUrl}/api/v1/jobs`);\n if (!response.ok) {\n throw new Error(`Failed to get executions: ${response.statusText}`);\n }\n const data = await this.parseJsonResponse<any>(response);\n const unwrapped = this.unwrapData<{ jobs?: any[] }>(data);\n const jobs = unwrapped.jobs ?? [];\n return jobs.filter((job: any) => job.status === 'running' || job.status === 'pending');\n }\n\n /**\n * Get cron jobs\n */\n async getCronJobs(): Promise<{\n crons: any[];\n }> {\n const response = await fetch(`${this.baseUrl}/api/v1/cron`);\n if (!response.ok) {\n throw new Error(`Failed to get cron jobs: ${response.statusText}`);\n }\n const data = await this.parseJsonResponse<any>(response);\n return this.unwrapData(data);\n }\n\n /**\n * Submit a job for execution\n */\n async submitJob(params: {\n handler: string;\n input?: unknown;\n priority?: number;\n }): Promise<{ id: string; status: string }> {\n const response = await fetch(`${this.baseUrl}/api/v1/jobs`, {\n method: 'POST',\n headers: {\n 'Content-Type': 'application/json',\n },\n body: JSON.stringify({\n type: params.handler,\n payload: params.input,\n priority: params.priority,\n }),\n });\n\n if (!response.ok) {\n const error = (await response.json().catch(() => ({ error: response.statusText }))) as {\n error?: string;\n };\n throw new Error(error.error || `Failed to submit job: ${response.statusText}`);\n }\n\n const payload = await this.parseJsonResponse<any>(response);\n const data = this.unwrapData<{ jobId: string }>(payload);\n return { id: data.jobId, status: 'pending' };\n }\n\n /**\n * Run workflow by ID\n */\n async runWorkflow(\n workflowId: string,\n request: WorkflowRunRequest = {}\n ): Promise<{ runId: string; status: string }> {\n const response = await fetch(\n `${this.baseUrl}/api/v1/workflows/${encodeURIComponent(workflowId)}/run`,\n {\n method: 'POST',\n headers: {\n 'Content-Type': 'application/json',\n },\n body: JSON.stringify(request),\n }\n );\n\n if (!response.ok) {\n const error = (await response.json().catch(() => ({ error: response.statusText }))) as {\n error?: string;\n };\n throw new Error(error.error || `Failed to run workflow: ${response.statusText}`);\n }\n\n const payload = await this.parseJsonResponse<{ ok: boolean; data?: { runId: string; status: string } }>(response);\n return this.unwrapData<{ runId: string; status: string }>(payload);\n }\n}\n","/**\n * workflow:list command - List active executions\n */\n\nimport { defineCommand, type PluginContextV3 } from '@kb-labs/sdk';\nimport { type ListFlags } from '@kb-labs/workflow-contracts';\nimport { WorkflowDaemonClient } from '../http-client.js';\n\ntype ListInput = ListFlags & { argv?: string[] };\n\nexport default defineCommand<unknown, ListInput, { exitCode: number }>({\n id: 'workflow:list',\n description: 'List active workflow executions',\n\n handler: {\n // eslint-disable-next-line sonarjs/cognitive-complexity -- Workflow listing with filtering (status/type), JSON/human output formats, run state formatting, and error handling\n async execute(ctx: PluginContextV3, input: ListInput): Promise<{ exitCode: number }> {\n const flags = (input as any).flags ?? input;\n const outputJson = flags.json ?? false;\n const statusFilter = flags.status;\n const typeFilter = flags.type;\n\n try {\n const client = new WorkflowDaemonClient();\n\n // Handle --type cron filter\n if (typeFilter === 'cron') {\n const result = await client.getCronJobs();\n const cronJobs = result.crons ?? [];\n\n if (outputJson) {\n ctx.ui?.json?.({ ok: true, data: result });\n } else {\n if (cronJobs.length === 0) {\n ctx.ui?.warn?.('No cron jobs found');\n ctx.ui?.info?.('');\n ctx.ui?.info?.('To add cron jobs:');\n ctx.ui?.info?.(' 1. Plugin manifests: Add \"cron\" section to manifest.ts');\n ctx.ui?.info?.(' 2. User YAML: Create .kb/jobs/*.yml files');\n } else {\n // Build job details\n const jobItems = cronJobs.map(job => {\n const parts = [\n `ID: ${job.id}`,\n `Schedule: ${job.schedule} (${job.timezone || 'UTC'})`,\n `Enabled: ${job.enabled ? 'Yes' : 'No'}`,\n `Type: ${job.jobType || 'unknown'}`,\n ];\n return parts.join(' | ');\n });\n\n const summaryItems = [\n `Total Jobs: ${cronJobs.length}`,\n ];\n\n ctx.ui?.success?.('Cron Jobs', {\n title: 'Workflow Scheduler',\n sections: [\n { header: 'Summary', items: summaryItems },\n { header: 'Registered Jobs', items: jobItems },\n ],\n });\n }\n }\n\n return { exitCode: 0 };\n }\n\n // Default: list active executions (runs)\n let executions = await client.getExecutions();\n\n // Filter by status if provided\n if (statusFilter) {\n executions = executions.filter((exec: any) => exec.status === statusFilter);\n }\n\n if (outputJson) {\n ctx.ui?.json?.({ ok: true, data: { executions } });\n } else {\n if (executions.length === 0) {\n ctx.ui?.warn?.('No active executions found');\n } else {\n const executionItems = executions.map(exec => {\n const parts = [\n `ID: ${exec.id}`,\n `Status: ${exec.status}`,\n `Started: ${exec.startedAt || 'N/A'}`,\n ];\n return parts.join(' | ');\n });\n\n const summaryItems = [\n `Total Executions: ${executions.length}`,\n statusFilter ? `Filter: ${statusFilter}` : undefined,\n ].filter(Boolean) as string[];\n\n ctx.ui?.success?.('Active Workflow Executions', {\n title: 'Workflow Engine',\n sections: [\n { header: 'Summary', items: summaryItems },\n { header: 'Executions', items: executionItems },\n ],\n });\n }\n }\n\n return { exitCode: 0 };\n } catch (error) {\n const message = error instanceof Error ? error.message : String(error);\n\n if (outputJson) {\n ctx.ui?.json?.({ ok: false, error: message });\n } else {\n ctx.ui?.error?.(`Failed to list: ${message}`);\n ctx.ui?.warn?.(`Make sure workflow daemon is running: kb-workflow`);\n }\n\n return { exitCode: 1 };\n }\n },\n },\n});\n"]}
|
|
@@ -0,0 +1,10 @@
|
|
|
1
|
+
import * as _kb_labs_shared_command_kit from '@kb-labs/shared-command-kit';
|
|
2
|
+
|
|
3
|
+
/**
|
|
4
|
+
* workflow:logs command - Get job logs
|
|
5
|
+
*/
|
|
6
|
+
declare const _default: _kb_labs_shared_command_kit.CommandHandlerV3<unknown, any, {
|
|
7
|
+
exitCode: number;
|
|
8
|
+
}>;
|
|
9
|
+
|
|
10
|
+
export { _default as default };
|