@raspect/workflow-sdk 0.2.0 → 0.3.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/dist/worker.d.ts +8 -2
- package/dist/worker.js +138 -30
- package/package.json +1 -1
package/dist/worker.d.ts
CHANGED
|
@@ -4,7 +4,7 @@ import { type WorkerInterface } from './workflow';
|
|
|
4
4
|
* Worker that executes workflows and activities.
|
|
5
5
|
*
|
|
6
6
|
* Usage:
|
|
7
|
-
* import { Worker, workflow, activity } from '@
|
|
7
|
+
* import { Worker, workflow, activity } from '@raspect/workflow-sdk';
|
|
8
8
|
*
|
|
9
9
|
* const sayHello = activity.defn(async (input: { name: string }) => {
|
|
10
10
|
* return { message: `Hello, ${input.name}!` };
|
|
@@ -20,7 +20,7 @@ import { type WorkerInterface } from './workflow';
|
|
|
20
20
|
*
|
|
21
21
|
* const worker = new Worker({
|
|
22
22
|
* serverUrl: 'https://workflow.example.com',
|
|
23
|
-
* apiKey: '
|
|
23
|
+
* apiKey: 'wk_your_worker_api_key', // Worker API key from server UI
|
|
24
24
|
* });
|
|
25
25
|
* worker.register({ workflows: [HelloWorldWorkflow], activities: [sayHello] });
|
|
26
26
|
* await worker.run();
|
|
@@ -39,6 +39,8 @@ export declare class Worker implements WorkerInterface {
|
|
|
39
39
|
private heartbeatIntervalMs;
|
|
40
40
|
private resultTopic;
|
|
41
41
|
private maxConcurrentWorkflows;
|
|
42
|
+
private needsReconnect;
|
|
43
|
+
private workerTopic;
|
|
42
44
|
constructor(config: WorkerConfig);
|
|
43
45
|
/**
|
|
44
46
|
* Register workflows and activities.
|
|
@@ -51,10 +53,13 @@ export declare class Worker implements WorkerInterface {
|
|
|
51
53
|
* Start the worker (blocking).
|
|
52
54
|
*/
|
|
53
55
|
run(): Promise<void>;
|
|
56
|
+
private connectAndRun;
|
|
54
57
|
/**
|
|
55
58
|
* Stop the worker gracefully.
|
|
56
59
|
*/
|
|
57
60
|
stop(): Promise<void>;
|
|
61
|
+
private formatConnectionError;
|
|
62
|
+
private connectToServerOnce;
|
|
58
63
|
private connectToServer;
|
|
59
64
|
private createKafkaClient;
|
|
60
65
|
private startHeartbeat;
|
|
@@ -67,5 +72,6 @@ export declare class Worker implements WorkerInterface {
|
|
|
67
72
|
reportLocalActivityCompletion(workflowId: string, activityName: string, activityIndex: number, output: unknown, durationMs: number): Promise<void>;
|
|
68
73
|
reportLocalActivityFailure(workflowId: string, activityName: string, activityIndex: number, error: string, errorType: string, durationMs: number): Promise<void>;
|
|
69
74
|
executeRemoteActivity(workflowId: string, activityName: string, activityIndex: number, inputData: unknown, timeoutMs: number): Promise<unknown>;
|
|
75
|
+
private cleanupKafka;
|
|
70
76
|
private cleanup;
|
|
71
77
|
}
|
package/dist/worker.js
CHANGED
|
@@ -44,7 +44,7 @@ const activity_1 = require("./activity");
|
|
|
44
44
|
* Worker that executes workflows and activities.
|
|
45
45
|
*
|
|
46
46
|
* Usage:
|
|
47
|
-
* import { Worker, workflow, activity } from '@
|
|
47
|
+
* import { Worker, workflow, activity } from '@raspect/workflow-sdk';
|
|
48
48
|
*
|
|
49
49
|
* const sayHello = activity.defn(async (input: { name: string }) => {
|
|
50
50
|
* return { message: `Hello, ${input.name}!` };
|
|
@@ -60,7 +60,7 @@ const activity_1 = require("./activity");
|
|
|
60
60
|
*
|
|
61
61
|
* const worker = new Worker({
|
|
62
62
|
* serverUrl: 'https://workflow.example.com',
|
|
63
|
-
* apiKey: '
|
|
63
|
+
* apiKey: 'wk_your_worker_api_key', // Worker API key from server UI
|
|
64
64
|
* });
|
|
65
65
|
* worker.register({ workflows: [HelloWorldWorkflow], activities: [sayHello] });
|
|
66
66
|
* await worker.run();
|
|
@@ -79,6 +79,8 @@ class Worker {
|
|
|
79
79
|
heartbeatIntervalMs = 30000;
|
|
80
80
|
resultTopic = null;
|
|
81
81
|
maxConcurrentWorkflows;
|
|
82
|
+
needsReconnect = false;
|
|
83
|
+
workerTopic = null;
|
|
82
84
|
constructor(config) {
|
|
83
85
|
this.config = {
|
|
84
86
|
...config,
|
|
@@ -93,12 +95,10 @@ class Worker {
|
|
|
93
95
|
for (const wfCls of options.workflows || []) {
|
|
94
96
|
const wfName = (0, workflow_1.getWorkflowName)(wfCls);
|
|
95
97
|
this.workflows.set(wfName, wfCls);
|
|
96
|
-
console.log(`Registered workflow: ${wfName}`);
|
|
97
98
|
}
|
|
98
99
|
for (const activityFn of options.activities || []) {
|
|
99
100
|
const activityName = (0, activity_1.getActivityName)(activityFn);
|
|
100
101
|
this.activities.set(activityName, activityFn);
|
|
101
|
-
console.log(`Registered activity: ${activityName}`);
|
|
102
102
|
}
|
|
103
103
|
}
|
|
104
104
|
/**
|
|
@@ -109,11 +109,36 @@ class Worker {
|
|
|
109
109
|
// Setup signal handlers
|
|
110
110
|
process.on('SIGINT', () => this.stop());
|
|
111
111
|
process.on('SIGTERM', () => this.stop());
|
|
112
|
+
// Main loop with automatic reconnection
|
|
113
|
+
while (this.running) {
|
|
114
|
+
try {
|
|
115
|
+
await this.connectAndRun();
|
|
116
|
+
}
|
|
117
|
+
catch (error) {
|
|
118
|
+
if (!this.running)
|
|
119
|
+
break;
|
|
120
|
+
const err = error;
|
|
121
|
+
// Check if it's an auth error that shouldn't be retried
|
|
122
|
+
if (err.message.includes('Invalid or expired API key') ||
|
|
123
|
+
err.message.includes('does not have permission')) {
|
|
124
|
+
console.error(`Worker stopped: ${err.message}`);
|
|
125
|
+
break;
|
|
126
|
+
}
|
|
127
|
+
console.error(`Worker error: ${err.message}. Will attempt to reconnect...`);
|
|
128
|
+
await this.cleanupKafka();
|
|
129
|
+
// Small delay before reconnection attempt
|
|
130
|
+
await new Promise(resolve => setTimeout(resolve, 5000));
|
|
131
|
+
}
|
|
132
|
+
}
|
|
133
|
+
}
|
|
134
|
+
async connectAndRun() {
|
|
135
|
+
this.needsReconnect = false;
|
|
112
136
|
// Connect to server and get Kafka config
|
|
113
137
|
const serverKafkaConfig = await this.connectToServer();
|
|
114
138
|
const workerTopic = serverKafkaConfig.worker_topic;
|
|
115
139
|
const resultTopic = serverKafkaConfig.result_topic;
|
|
116
140
|
this.resultTopic = resultTopic;
|
|
141
|
+
this.workerTopic = workerTopic;
|
|
117
142
|
// Create Kafka client
|
|
118
143
|
this.kafka = this.createKafkaClient(serverKafkaConfig);
|
|
119
144
|
// Create consumer
|
|
@@ -125,25 +150,21 @@ class Worker {
|
|
|
125
150
|
await this.consumer.connect();
|
|
126
151
|
await this.producer.connect();
|
|
127
152
|
await this.consumer.subscribe({ topic: workerTopic, fromBeginning: false });
|
|
153
|
+
const workflowsStr = this.workflows.size > 0 ? Array.from(this.workflows.keys()).join(', ') : '(none)';
|
|
154
|
+
const activitiesStr = this.activities.size > 0 ? Array.from(this.activities.keys()).join(', ') : '(none)';
|
|
128
155
|
console.log(`Worker ${this.workerId} started, consuming from ${workerTopic}`);
|
|
129
156
|
console.log(`Max concurrent workflows: ${this.maxConcurrentWorkflows}`);
|
|
130
|
-
console.log(`Registered workflows: ${
|
|
131
|
-
console.log(`Registered activities: ${
|
|
157
|
+
console.log(`Registered workflows: ${workflowsStr}`);
|
|
158
|
+
console.log(`Registered activities: ${activitiesStr}`);
|
|
132
159
|
// Start heartbeat
|
|
133
160
|
this.startHeartbeat();
|
|
134
|
-
// Start consuming messages
|
|
135
|
-
// partitionsConsumedConcurrently limits how many messages can be processed at once
|
|
136
|
-
// Commit immediately after fetch - workflows can run for hours, so we can't wait
|
|
137
|
-
// If worker crashes mid-workflow, use server's /resume API to recover
|
|
161
|
+
// Start consuming messages
|
|
138
162
|
await this.consumer.run({
|
|
139
163
|
autoCommit: false,
|
|
140
164
|
partitionsConsumedConcurrently: this.maxConcurrentWorkflows,
|
|
141
165
|
eachMessage: async ({ topic, partition, message }) => {
|
|
142
|
-
if (!this.running)
|
|
166
|
+
if (!this.running || this.needsReconnect)
|
|
143
167
|
return;
|
|
144
|
-
// Commit immediately after receiving - the message is now "in progress"
|
|
145
|
-
// This prevents Kafka rebalance during long-running workflows (hours)
|
|
146
|
-
// Pending messages in queue stay safe, in-progress workflows tracked in MongoDB
|
|
147
168
|
await this.consumer?.commitOffsets([{
|
|
148
169
|
topic,
|
|
149
170
|
partition,
|
|
@@ -154,23 +175,32 @@ class Worker {
|
|
|
154
175
|
await this.processMessage(data, resultTopic);
|
|
155
176
|
}
|
|
156
177
|
catch (error) {
|
|
157
|
-
|
|
158
|
-
|
|
178
|
+
const errMsg = error.message?.toLowerCase() || '';
|
|
179
|
+
if (errMsg.includes('disconnected') || errMsg.includes('connection')) {
|
|
180
|
+
console.warn('Kafka connection lost, triggering reconnection...');
|
|
181
|
+
this.needsReconnect = true;
|
|
182
|
+
}
|
|
183
|
+
else {
|
|
184
|
+
console.error('Error processing message:', error.message);
|
|
185
|
+
}
|
|
159
186
|
}
|
|
160
187
|
},
|
|
161
188
|
});
|
|
162
|
-
// Keep running until stopped
|
|
189
|
+
// Keep running until stopped or needs reconnect
|
|
163
190
|
await new Promise((resolve) => {
|
|
164
191
|
const checkInterval = setInterval(() => {
|
|
165
|
-
if (!this.running) {
|
|
192
|
+
if (!this.running || this.needsReconnect) {
|
|
166
193
|
clearInterval(checkInterval);
|
|
167
194
|
resolve();
|
|
168
195
|
}
|
|
169
196
|
}, 100);
|
|
170
197
|
});
|
|
198
|
+
if (this.needsReconnect) {
|
|
199
|
+
console.log('Reconnecting to server and Kafka...');
|
|
200
|
+
}
|
|
171
201
|
}
|
|
172
202
|
finally {
|
|
173
|
-
await this.
|
|
203
|
+
await this.cleanupKafka();
|
|
174
204
|
}
|
|
175
205
|
}
|
|
176
206
|
/**
|
|
@@ -180,7 +210,24 @@ class Worker {
|
|
|
180
210
|
console.log('Stopping worker...');
|
|
181
211
|
this.running = false;
|
|
182
212
|
}
|
|
183
|
-
|
|
213
|
+
formatConnectionError(status, errorText) {
|
|
214
|
+
if (status === 401) {
|
|
215
|
+
return 'Invalid or expired API key. Please check your worker API key.';
|
|
216
|
+
}
|
|
217
|
+
else if (status === 403) {
|
|
218
|
+
return 'API key does not have permission to connect. Please check worker permissions.';
|
|
219
|
+
}
|
|
220
|
+
else if (status === 404) {
|
|
221
|
+
return 'Server endpoint not found. Please check the server URL.';
|
|
222
|
+
}
|
|
223
|
+
else if (status >= 500) {
|
|
224
|
+
return `Server error (HTTP ${status}). The server may be temporarily unavailable.`;
|
|
225
|
+
}
|
|
226
|
+
else {
|
|
227
|
+
return `HTTP ${status} - ${errorText}`;
|
|
228
|
+
}
|
|
229
|
+
}
|
|
230
|
+
async connectToServerOnce() {
|
|
184
231
|
const hostname = os.hostname();
|
|
185
232
|
const connectPayload = {
|
|
186
233
|
workflows: Array.from(this.workflows.keys()),
|
|
@@ -197,7 +244,8 @@ class Worker {
|
|
|
197
244
|
});
|
|
198
245
|
if (!response.ok) {
|
|
199
246
|
const error = await response.text();
|
|
200
|
-
|
|
247
|
+
const errorMsg = this.formatConnectionError(response.status, error);
|
|
248
|
+
throw new Error(`Failed to connect to ${this.config.serverUrl}: ${errorMsg}`);
|
|
201
249
|
}
|
|
202
250
|
const data = await response.json();
|
|
203
251
|
this.workerId = data.worker_id;
|
|
@@ -206,6 +254,29 @@ class Worker {
|
|
|
206
254
|
console.log(`Connected to server as worker_id=${this.workerId}, instance_id=${this.instanceId}`);
|
|
207
255
|
return data.kafka;
|
|
208
256
|
}
|
|
257
|
+
async connectToServer() {
|
|
258
|
+
let retryInterval = 1000; // Start at 1 second
|
|
259
|
+
const maxRetryInterval = 600000; // Max 10 minutes
|
|
260
|
+
const backoffMultiplier = 2.0;
|
|
261
|
+
while (this.running) {
|
|
262
|
+
try {
|
|
263
|
+
return await this.connectToServerOnce();
|
|
264
|
+
}
|
|
265
|
+
catch (error) {
|
|
266
|
+
const err = error;
|
|
267
|
+
// Don't retry on auth errors
|
|
268
|
+
if (err.message.includes('Invalid or expired API key') ||
|
|
269
|
+
err.message.includes('does not have permission')) {
|
|
270
|
+
console.error(`Worker failed to start: ${err.message}`);
|
|
271
|
+
throw error;
|
|
272
|
+
}
|
|
273
|
+
console.warn(`Connection failed: ${err.message}. Retrying in ${Math.round(retryInterval / 1000)}s...`);
|
|
274
|
+
await new Promise(resolve => setTimeout(resolve, retryInterval));
|
|
275
|
+
retryInterval = Math.min(retryInterval * backoffMultiplier, maxRetryInterval);
|
|
276
|
+
}
|
|
277
|
+
}
|
|
278
|
+
throw new Error('Worker stopped before connection could be established');
|
|
279
|
+
}
|
|
209
280
|
createKafkaClient(kafkaConfig) {
|
|
210
281
|
const auth = kafkaConfig.auth;
|
|
211
282
|
// Server returns brokers as comma-separated string, convert to array
|
|
@@ -248,6 +319,8 @@ class Worker {
|
|
|
248
319
|
return new kafkajs_1.Kafka(config);
|
|
249
320
|
}
|
|
250
321
|
startHeartbeat() {
|
|
322
|
+
let consecutiveFailures = 0;
|
|
323
|
+
const maxFailuresBeforeReconnect = 3;
|
|
251
324
|
this.heartbeatInterval = setInterval(async () => {
|
|
252
325
|
if (!this.running)
|
|
253
326
|
return;
|
|
@@ -260,12 +333,31 @@ class Worker {
|
|
|
260
333
|
},
|
|
261
334
|
body: JSON.stringify({ instance_id: this.instanceId }),
|
|
262
335
|
});
|
|
263
|
-
if (
|
|
264
|
-
|
|
336
|
+
if (response.ok) {
|
|
337
|
+
consecutiveFailures = 0;
|
|
338
|
+
}
|
|
339
|
+
else if (response.status === 401) {
|
|
340
|
+
console.error('Heartbeat failed: Invalid or expired API key');
|
|
341
|
+
consecutiveFailures++;
|
|
342
|
+
}
|
|
343
|
+
else {
|
|
344
|
+
console.warn(`Heartbeat failed: HTTP ${response.status}`);
|
|
345
|
+
consecutiveFailures++;
|
|
265
346
|
}
|
|
266
347
|
}
|
|
267
348
|
catch (error) {
|
|
268
|
-
console.warn('Heartbeat error:', error);
|
|
349
|
+
console.warn('Heartbeat error:', error.message);
|
|
350
|
+
consecutiveFailures++;
|
|
351
|
+
}
|
|
352
|
+
// Trigger reconnection if too many consecutive failures
|
|
353
|
+
if (consecutiveFailures >= maxFailuresBeforeReconnect) {
|
|
354
|
+
console.warn(`Heartbeat failed ${consecutiveFailures} times consecutively. ` +
|
|
355
|
+
'Connection may be lost, triggering reconnection...');
|
|
356
|
+
this.needsReconnect = true;
|
|
357
|
+
if (this.heartbeatInterval) {
|
|
358
|
+
clearInterval(this.heartbeatInterval);
|
|
359
|
+
this.heartbeatInterval = null;
|
|
360
|
+
}
|
|
269
361
|
}
|
|
270
362
|
}, this.heartbeatIntervalMs);
|
|
271
363
|
}
|
|
@@ -592,22 +684,38 @@ class Worker {
|
|
|
592
684
|
pollInterval = Math.min(pollInterval * backoffMultiplier, maxPollInterval);
|
|
593
685
|
}
|
|
594
686
|
}
|
|
595
|
-
async
|
|
687
|
+
async cleanupKafka() {
|
|
596
688
|
// Stop heartbeat
|
|
597
689
|
if (this.heartbeatInterval) {
|
|
598
690
|
clearInterval(this.heartbeatInterval);
|
|
691
|
+
this.heartbeatInterval = null;
|
|
599
692
|
}
|
|
600
|
-
// Disconnect from server
|
|
601
|
-
await this.disconnectFromServer();
|
|
602
693
|
// Stop Kafka clients
|
|
603
694
|
if (this.consumer) {
|
|
604
|
-
|
|
695
|
+
try {
|
|
696
|
+
await this.consumer.disconnect();
|
|
697
|
+
}
|
|
698
|
+
catch (e) {
|
|
699
|
+
// Ignore disconnect errors
|
|
700
|
+
}
|
|
701
|
+
this.consumer = null;
|
|
605
702
|
}
|
|
606
703
|
if (this.producer) {
|
|
607
|
-
|
|
704
|
+
try {
|
|
705
|
+
await this.producer.disconnect();
|
|
706
|
+
}
|
|
707
|
+
catch (e) {
|
|
708
|
+
// Ignore disconnect errors
|
|
709
|
+
}
|
|
710
|
+
this.producer = null;
|
|
608
711
|
}
|
|
712
|
+
}
|
|
713
|
+
async cleanup() {
|
|
714
|
+
await this.cleanupKafka();
|
|
715
|
+
// Disconnect from server
|
|
716
|
+
await this.disconnectFromServer();
|
|
609
717
|
console.log(`Worker ${this.workerId} stopped`);
|
|
610
718
|
}
|
|
611
719
|
}
|
|
612
720
|
exports.Worker = Worker;
|
|
613
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"worker.js","sourceRoot":"","sources":["../src/worker.ts"],"names":[],"mappings":"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","sourcesContent":["// Worker - Node.js implementation for executing workflows and activities\n\nimport { Kafka, Consumer, Producer, SASLOptions, logLevel } from 'kafkajs';\nimport { generateAuthTokenFromCredentialsProvider } from 'aws-msk-iam-sasl-signer-js';\nimport * as os from 'os';\nimport type {\n  WorkerConfig,\n  ActivityHandler,\n  DecoratedActivity,\n  WorkflowClass,\n  WorkflowTask,\n  ActivityTask,\n  WorkflowResult,\n  ActivityResult,\n  WorkflowError,\n  ActivityError,\n  KafkaConfig,\n  ConnectResponse,\n} from './types';\nimport { WorkflowContext, runWithContext, getWorkflowName, findRunMethod, type WorkerInterface } from './workflow';\nimport { getActivityName } from './activity';\n\n/**\n * Worker that executes workflows and activities.\n * \n * Usage:\n *   import { Worker, workflow, activity } from '@inspectica/workflow-sdk';\n *   \n *   const sayHello = activity.defn(async (input: { name: string }) => {\n *     return { message: `Hello, ${input.name}!` };\n *   });\n *   \n *   @workflow.defn\n *   class HelloWorldWorkflow {\n *     @workflow.run\n *     async run(input: { name: string }) {\n *       return await workflow.executeActivity(sayHello, input);\n *     }\n *   }\n *   \n *   const worker = new Worker({\n *     serverUrl: 'https://workflow.example.com',\n *     apiKey: 'wk_abc123...',\n *   });\n *   worker.register({ workflows: [HelloWorldWorkflow], activities: [sayHello] });\n *   await worker.run();\n */\nexport class Worker implements WorkerInterface {\n  private config: WorkerConfig;\n  private workflows: Map<string, WorkflowClass> = new Map();\n  private activities: Map<string, ActivityHandler> = new Map();\n  \n  private workerId: string | null = null;\n  private instanceId: string | null = null;\n  private running = false;\n  private kafka: Kafka | null = null;\n  private consumer: Consumer | null = null;\n  private producer: Producer | null = null;\n  private heartbeatInterval: NodeJS.Timeout | null = null;\n  private heartbeatIntervalMs = 30000;\n  private resultTopic: string | null = null;\n  private maxConcurrentWorkflows: number;\n\n  constructor(config: WorkerConfig) {\n    this.config = {\n      ...config,\n      serverUrl: config.serverUrl.replace(/\\/$/, ''),\n    };\n    this.maxConcurrentWorkflows = config.maxConcurrentWorkflows || 10;\n  }\n\n  /**\n   * Register workflows and activities.\n   */\n  register(options: {\n    workflows?: WorkflowClass[];\n    activities?: DecoratedActivity[];\n  }): void {\n    for (const wfCls of options.workflows || []) {\n      const wfName = getWorkflowName(wfCls);\n      this.workflows.set(wfName, wfCls);\n      console.log(`Registered workflow: ${wfName}`);\n    }\n\n    for (const activityFn of options.activities || []) {\n      const activityName = getActivityName(activityFn);\n      this.activities.set(activityName, activityFn);\n      console.log(`Registered activity: ${activityName}`);\n    }\n  }\n\n  /**\n   * Start the worker (blocking).\n   */\n  async run(): Promise<void> {\n    this.running = true;\n\n    // Setup signal handlers\n    process.on('SIGINT', () => this.stop());\n    process.on('SIGTERM', () => this.stop());\n\n    // Connect to server and get Kafka config\n    const serverKafkaConfig = await this.connectToServer();\n    const workerTopic = serverKafkaConfig.worker_topic;\n    const resultTopic = serverKafkaConfig.result_topic;\n    this.resultTopic = resultTopic;\n\n    // Create Kafka client\n    this.kafka = this.createKafkaClient(serverKafkaConfig);\n    \n    // Create consumer\n    const consumerGroupId = this.config.groupId || `workflow-worker-${this.workerId}`;\n    this.consumer = this.kafka.consumer({ groupId: consumerGroupId });\n    \n    // Create producer\n    this.producer = this.kafka.producer();\n\n    try {\n      await this.consumer.connect();\n      await this.producer.connect();\n      await this.consumer.subscribe({ topic: workerTopic, fromBeginning: false });\n\n      console.log(`Worker ${this.workerId} started, consuming from ${workerTopic}`);\n      console.log(`Max concurrent workflows: ${this.maxConcurrentWorkflows}`);\n      console.log(`Registered workflows: ${Array.from(this.workflows.keys()).join(', ')}`);\n      console.log(`Registered activities: ${Array.from(this.activities.keys()).join(', ')}`);\n\n      // Start heartbeat\n      this.startHeartbeat();\n\n      // Start consuming messages with worker-level concurrency control\n      // partitionsConsumedConcurrently limits how many messages can be processed at once\n      // Commit immediately after fetch - workflows can run for hours, so we can't wait\n      // If worker crashes mid-workflow, use server's /resume API to recover\n      await this.consumer.run({\n        autoCommit: false,\n        partitionsConsumedConcurrently: this.maxConcurrentWorkflows,\n        eachMessage: async ({ topic, partition, message }) => {\n          if (!this.running) return;\n          \n          // Commit immediately after receiving - the message is now \"in progress\"\n          // This prevents Kafka rebalance during long-running workflows (hours)\n          // Pending messages in queue stay safe, in-progress workflows tracked in MongoDB\n          await this.consumer?.commitOffsets([{\n            topic,\n            partition,\n            offset: (BigInt(message.offset) + 1n).toString(),\n          }]);\n          \n          try {\n            const data = JSON.parse(message.value?.toString() || '{}');\n            await this.processMessage(data, resultTopic);\n          } catch (error) {\n            console.error('Error processing message:', error);\n            // Workflow failed - state is in MongoDB, can be resumed via server API\n          }\n        },\n      });\n\n      // Keep running until stopped\n      await new Promise<void>((resolve) => {\n        const checkInterval = setInterval(() => {\n          if (!this.running) {\n            clearInterval(checkInterval);\n            resolve();\n          }\n        }, 100);\n      });\n\n    } finally {\n      await this.cleanup();\n    }\n  }\n\n  /**\n   * Stop the worker gracefully.\n   */\n  async stop(): Promise<void> {\n    console.log('Stopping worker...');\n    this.running = false;\n  }\n\n  private async connectToServer(): Promise<KafkaConfig> {\n    const hostname = os.hostname();\n    \n    const connectPayload = {\n      workflows: Array.from(this.workflows.keys()),\n      activities: Array.from(this.activities.keys()),\n      hostname,\n    };\n\n    const response = await fetch(`${this.config.serverUrl}/workflow/api/v1/workers/connect`, {\n      method: 'POST',\n      headers: {\n        'Content-Type': 'application/json',\n        'Authorization': `Bearer ${this.config.apiKey}`,\n      },\n      body: JSON.stringify(connectPayload),\n    });\n\n    if (!response.ok) {\n      const error = await response.text();\n      throw new Error(`Failed to connect to server: ${response.status} - ${error}`);\n    }\n\n    const data = await response.json() as ConnectResponse;\n    \n    this.workerId = data.worker_id;\n    this.instanceId = data.instance_id;\n    this.heartbeatIntervalMs = data.heartbeat_interval_ms || 30000;\n\n    console.log(`Connected to server as worker_id=${this.workerId}, instance_id=${this.instanceId}`);\n\n    return data.kafka;\n  }\n\n  private createKafkaClient(kafkaConfig: KafkaConfig): Kafka {\n    const auth = kafkaConfig.auth;\n    \n    // Server returns brokers as comma-separated string, convert to array\n    const brokers = typeof kafkaConfig.brokers === 'string'\n      ? (kafkaConfig.brokers as string).split(',').map(b => b.trim()).filter(b => b)\n      : kafkaConfig.brokers;\n    \n    const config: any = {\n      clientId: `workflow-worker-${this.workerId}`,\n      brokers,\n      logLevel: logLevel.WARN,\n    };\n\n    if (auth?.type === 'msk_iam') {\n      config.ssl = true;\n      config.sasl = {\n        mechanism: 'oauthbearer',\n        oauthBearerProvider: async () => {\n          const token = await generateAuthTokenFromCredentialsProvider({\n            region: auth.region!,\n            awsCredentialsProvider: async () => ({\n              accessKeyId: auth.access_key_id!,\n              secretAccessKey: auth.secret_access_key!,\n            }),\n          });\n          return { value: token.token };\n        },\n      } as SASLOptions;\n    } else if (auth?.type === 'sasl') {\n      if (auth.security_protocol?.includes('SSL')) {\n        config.ssl = true;\n      }\n      if (auth.sasl_mechanism && auth.sasl_username && auth.sasl_password) {\n        config.sasl = {\n          mechanism: auth.sasl_mechanism.toLowerCase().replace('-', ''),\n          username: auth.sasl_username,\n          password: auth.sasl_password,\n        } as SASLOptions;\n      }\n    }\n\n    return new Kafka(config);\n  }\n\n  private startHeartbeat(): void {\n    this.heartbeatInterval = setInterval(async () => {\n      if (!this.running) return;\n      \n      try {\n        const response = await fetch(`${this.config.serverUrl}/workflow/api/v1/workers/heartbeat`, {\n          method: 'POST',\n          headers: {\n            'Content-Type': 'application/json',\n            'Authorization': `Bearer ${this.config.apiKey}`,\n          },\n          body: JSON.stringify({ instance_id: this.instanceId }),\n        });\n\n        if (!response.ok) {\n          console.warn(`Heartbeat failed: ${response.status}`);\n        }\n      } catch (error) {\n        console.warn('Heartbeat error:', error);\n      }\n    }, this.heartbeatIntervalMs);\n  }\n\n  private async disconnectFromServer(): Promise<void> {\n    if (!this.instanceId) return;\n\n    try {\n      const response = await fetch(`${this.config.serverUrl}/workflow/api/v1/workers/disconnect`, {\n        method: 'POST',\n        headers: {\n          'Content-Type': 'application/json',\n          'Authorization': `Bearer ${this.config.apiKey}`,\n        },\n        body: JSON.stringify({ instance_id: this.instanceId }),\n      });\n\n      if (response.ok) {\n        console.log('Disconnected from server');\n      }\n    } catch (error) {\n      console.warn('Failed to disconnect from server:', error);\n    }\n  }\n\n  private async processMessage(data: any, resultTopic: string): Promise<void> {\n    // Check if this is an activity result (response from another worker via server)\n    if (data.status && data.completed_at && !data.task_type) {\n      console.debug('Received activity result on worker topic (ignored - using polling)');\n      return;\n    }\n\n    const taskType = data.task_type || 'activity';\n\n    if (taskType === 'workflow') {\n      // Concurrency is controlled by partitionsConsumedConcurrently in consumer.run()\n      await this.executeWorkflow(data as WorkflowTask, resultTopic);\n    } else {\n      // Activity tasks can be processed inline\n      await this.processActivityTask(data, resultTopic);\n    }\n  }\n\n  private async executeWorkflow(task: WorkflowTask, resultTopic: string): Promise<void> {\n    console.log(`Received workflow task ${task.task_id}: ${task.workflow_name}`);\n\n    // Find the workflow class\n    const workflowCls = this.workflows.get(task.workflow_name);\n    if (!workflowCls) {\n      console.warn(`Unknown workflow: ${task.workflow_name}`);\n      await this.sendWorkflowResult(task, resultTopic, 'failed', undefined, {\n        type: 'UnknownWorkflow',\n        message: `Workflow '${task.workflow_name}' not registered`,\n      });\n      return;\n    }\n\n    // Find the run method\n    const runMethod = findRunMethod(workflowCls);\n    if (!runMethod) {\n      await this.sendWorkflowResult(task, resultTopic, 'failed', undefined, {\n        type: 'NoRunMethod',\n        message: `Workflow '${task.workflow_name}' has no run method`,\n      });\n      return;\n    }\n\n    // Execute the workflow\n    const startedAt = new Date();\n    try {\n      // Build completed activities map for replay\n      const completedActivitiesMap = new Map<number, unknown>();\n      if (task.completed_activities) {\n        for (const ca of task.completed_activities) {\n          completedActivitiesMap.set(ca.index, ca.output);\n        }\n        console.log(\n          `Resuming workflow ${task.workflow_id} with ${completedActivitiesMap.size} completed activities`\n        );\n      }\n\n      // Create workflow context\n      const ctx = new WorkflowContext(\n        this.activities,\n        task.workflow_id,\n        this,\n        completedActivitiesMap.size > 0 ? completedActivitiesMap : null,\n        task.metadata\n      );\n\n      // Execute within context\n      const output = await runWithContext(ctx, async () => {\n        return await Promise.race([\n          runMethod(task.input),\n          new Promise<never>((_, reject) =>\n            setTimeout(\n              () => reject(new Error(`Workflow timed out after ${task.timeout_ms}ms`)),\n              task.timeout_ms\n            )\n          ),\n        ]);\n      });\n\n      const completedAt = new Date();\n      const durationMs = completedAt.getTime() - startedAt.getTime();\n\n      console.log(`Workflow ${task.task_id} completed in ${durationMs}ms`);\n\n      await this.sendWorkflowResult(\n        task,\n        resultTopic,\n        'completed',\n        output,\n        undefined,\n        startedAt,\n        completedAt,\n        durationMs\n      );\n    } catch (error) {\n      const completedAt = new Date();\n      const durationMs = completedAt.getTime() - startedAt.getTime();\n      const err = error as Error;\n\n      console.error(`Workflow ${task.task_id} failed:`, err.message);\n\n      await this.sendWorkflowResult(\n        task,\n        resultTopic,\n        'failed',\n        undefined,\n        {\n          type: err.name || 'Error',\n          message: err.message,\n          stack_trace: err.stack,\n        },\n        startedAt,\n        completedAt,\n        durationMs\n      );\n    }\n  }\n\n  private async processActivityTask(data: ActivityTask, resultTopic: string): Promise<void> {\n    const task = data;\n    console.log(\n      `Received activity task ${task.task_id}: ${task.activity_name} (workflow=${task.workflow_id})`\n    );\n\n    // Find the activity handler\n    const activityFn = this.activities.get(task.activity_name);\n    if (!activityFn) {\n      console.warn(`Unknown activity: ${task.activity_name}`);\n      await this.sendActivityResult(task, resultTopic, 'failed', undefined, {\n        type: 'UnknownActivity',\n        message: `Activity '${task.activity_name}' not registered`,\n      });\n      return;\n    }\n\n    // Execute the activity\n    const startedAt = new Date();\n    try {\n      const output = await Promise.race([\n        Promise.resolve(activityFn(task.input)),\n        new Promise<never>((_, reject) =>\n          setTimeout(\n            () => reject(new Error(`Activity timed out after ${task.timeout_ms}ms`)),\n            task.timeout_ms\n          )\n        ),\n      ]);\n\n      const completedAt = new Date();\n      const durationMs = completedAt.getTime() - startedAt.getTime();\n\n      console.log(`Activity ${task.task_id} completed in ${durationMs}ms`);\n\n      await this.sendActivityResult(\n        task,\n        resultTopic,\n        'completed',\n        output,\n        undefined,\n        startedAt,\n        completedAt,\n        durationMs\n      );\n    } catch (error) {\n      const completedAt = new Date();\n      const durationMs = completedAt.getTime() - startedAt.getTime();\n      const err = error as Error;\n\n      console.error(`Activity ${task.task_id} failed:`, err.message);\n\n      await this.sendActivityResult(\n        task,\n        resultTopic,\n        'failed',\n        undefined,\n        {\n          type: err.name || 'Error',\n          message: err.message,\n          stack_trace: err.stack,\n        },\n        startedAt,\n        completedAt,\n        durationMs\n      );\n    }\n  }\n\n  private async sendWorkflowResult(\n    task: WorkflowTask,\n    resultTopic: string,\n    status: 'completed' | 'failed',\n    output?: unknown,\n    error?: WorkflowError,\n    startedAt?: Date,\n    completedAt?: Date,\n    durationMs = 0\n  ): Promise<void> {\n    const now = new Date();\n    const result: WorkflowResult = {\n      task_id: task.task_id,\n      workflow_id: task.workflow_id,\n      workflow_name: task.workflow_name,\n      status,\n      output,\n      error,\n      started_at: (startedAt || now).toISOString(),\n      completed_at: (completedAt || now).toISOString(),\n      duration_ms: durationMs,\n      worker_id: this.workerId!,\n    };\n\n    if (this.producer) {\n      await this.producer.send({\n        topic: resultTopic,\n        messages: [\n          {\n            key: task.workflow_id,\n            value: JSON.stringify(result),\n          },\n        ],\n      });\n      console.debug(`Sent workflow result for task ${task.task_id}`);\n    }\n  }\n\n  private async sendActivityResult(\n    task: ActivityTask,\n    resultTopic: string,\n    status: 'completed' | 'failed',\n    output?: unknown,\n    error?: ActivityError,\n    startedAt?: Date,\n    completedAt?: Date,\n    durationMs = 0\n  ): Promise<void> {\n    const now = new Date();\n    const result: ActivityResult = {\n      task_id: task.task_id,\n      workflow_id: task.workflow_id,\n      activity_name: task.activity_name,\n      activity_index: task.activity_index,\n      fan_out_index: task.fan_out_index,\n      status,\n      output,\n      error,\n      started_at: (startedAt || now).toISOString(),\n      completed_at: (completedAt || now).toISOString(),\n      duration_ms: durationMs,\n      worker_id: this.workerId!,\n      requester_worker_id: task.requester_worker_id,\n      correlation_id: task.correlation_id,\n    };\n\n    if (this.producer) {\n      await this.producer.send({\n        topic: resultTopic,\n        messages: [\n          {\n            key: task.workflow_id,\n            value: JSON.stringify(result),\n          },\n        ],\n      });\n      console.debug(`Sent activity result for task ${task.task_id}`);\n    }\n  }\n\n  // WorkerInterface implementation for WorkflowContext\n\n  async reportLocalActivityCompletion(\n    workflowId: string,\n    activityName: string,\n    activityIndex: number,\n    output: unknown,\n    durationMs: number\n  ): Promise<void> {\n    if (!this.producer || !this.resultTopic) {\n      console.debug('Cannot report local activity - producer not initialized');\n      return;\n    }\n\n    const completedAt = new Date();\n    const startedAt = new Date(completedAt.getTime() - durationMs);\n\n    const result = {\n      task_id: `local-${workflowId}-${activityIndex}`,\n      workflow_id: workflowId,\n      activity_name: activityName,\n      activity_index: activityIndex,\n      status: 'completed',\n      output,\n      started_at: startedAt.toISOString(),\n      completed_at: completedAt.toISOString(),\n      duration_ms: durationMs,\n      worker_id: this.workerId,\n    };\n\n    try {\n      await this.producer.send({\n        topic: this.resultTopic,\n        messages: [{ key: workflowId, value: JSON.stringify(result) }],\n      });\n      console.debug(\n        `Reported local activity completion: workflow_id=${workflowId}, activity=${activityName}, index=${activityIndex}`\n      );\n    } catch (error) {\n      console.warn('Failed to report local activity completion:', error);\n    }\n  }\n\n  async reportLocalActivityFailure(\n    workflowId: string,\n    activityName: string,\n    activityIndex: number,\n    error: string,\n    errorType: string,\n    durationMs: number\n  ): Promise<void> {\n    if (!this.producer || !this.resultTopic) {\n      console.debug('Cannot report local activity failure - producer not initialized');\n      return;\n    }\n\n    const completedAt = new Date();\n    const startedAt = new Date(completedAt.getTime() - durationMs);\n\n    const result = {\n      task_id: `local-${workflowId}-${activityIndex}`,\n      workflow_id: workflowId,\n      activity_name: activityName,\n      activity_index: activityIndex,\n      status: 'failed',\n      output: null,\n      error: { type: errorType, message: error },\n      started_at: startedAt.toISOString(),\n      completed_at: completedAt.toISOString(),\n      duration_ms: durationMs,\n      worker_id: this.workerId,\n    };\n\n    try {\n      await this.producer.send({\n        topic: this.resultTopic,\n        messages: [{ key: workflowId, value: JSON.stringify(result) }],\n      });\n      console.debug(\n        `Reported local activity failure: workflow_id=${workflowId}, activity=${activityName}, index=${activityIndex}`\n      );\n    } catch (error) {\n      console.warn('Failed to report local activity failure:', error);\n    }\n  }\n\n  async executeRemoteActivity(\n    workflowId: string,\n    activityName: string,\n    activityIndex: number,\n    inputData: unknown,\n    timeoutMs: number\n  ): Promise<unknown> {\n    if (!this.producer || !this.resultTopic) {\n      throw new Error('Kafka producer not initialized');\n    }\n\n    // Send ActivityRequest to server via Kafka\n    const requestPayload = {\n      request_type: 'activity_request',\n      request_id: crypto.randomUUID(),\n      workflow_id: workflowId,\n      activity_name: activityName,\n      activity_index: activityIndex,\n      input: inputData,\n      timeout_ms: timeoutMs,\n      requester_worker_id: this.workerId,\n      correlation_id: crypto.randomUUID(),\n      created_at: new Date().toISOString(),\n    };\n\n    await this.producer.send({\n      topic: this.resultTopic,\n      messages: [{ key: workflowId, value: JSON.stringify(requestPayload) }],\n    });\n\n    console.log(\n      `Activity request sent via Kafka: activity=${activityName}, workflow_id=${workflowId}, activity_index=${activityIndex}`\n    );\n\n    // Poll server for result with exponential backoff\n    let pollInterval = 2000; // Start at 2 seconds\n    const maxPollInterval = 900000; // Max 15 minutes\n    const backoffMultiplier = 1.5;\n    const startTime = Date.now();\n    const timeoutSeconds = timeoutMs;\n\n    const pollUrl = `${this.config.serverUrl}/workflow/api/v1/workflows/${workflowId}/activities/${activityIndex}/result`;\n    const headers = { Authorization: `Bearer ${this.config.apiKey}` };\n\n    while (true) {\n      const elapsed = Date.now() - startTime;\n      if (elapsed >= timeoutSeconds) {\n        throw new Error(`Activity ${activityName} timed out after ${timeoutMs}ms`);\n      }\n\n      try {\n        const response = await fetch(pollUrl, { headers });\n\n        if (response.status === 200) {\n          const result = await response.json() as { status: string; output?: unknown; error?: string };\n          if (result.status === 'completed') {\n            console.log(`Activity ${activityName} completed for workflow ${workflowId}`);\n            return result.output;\n          } else if (result.status === 'failed') {\n            const errorMsg = result.error || 'Activity failed';\n            console.error(`Activity ${activityName} failed: ${errorMsg}`);\n            throw new Error(errorMsg);\n          }\n        } else if (response.status === 202) {\n          // Still pending/running, continue polling\n          console.debug(`Activity ${activityName} still running, polling again in ${pollInterval}ms`);\n        } else if (response.status === 404) {\n          // Activity not found yet\n          console.debug(`Activity ${activityName} not found yet, polling again in ${pollInterval}ms`);\n        } else {\n          console.warn(`Unexpected response ${response.status} when polling activity result`);\n        }\n      } catch (error) {\n        if ((error as Error).message.includes('timed out') || (error as Error).message.includes('failed')) {\n          throw error;\n        }\n        console.warn('Error polling activity result:', error);\n      }\n\n      // Wait with exponential backoff\n      await new Promise((resolve) => setTimeout(resolve, pollInterval));\n      pollInterval = Math.min(pollInterval * backoffMultiplier, maxPollInterval);\n    }\n  }\n\n  private async cleanup(): Promise<void> {\n    // Stop heartbeat\n    if (this.heartbeatInterval) {\n      clearInterval(this.heartbeatInterval);\n    }\n\n    // Disconnect from server\n    await this.disconnectFromServer();\n\n    // Stop Kafka clients\n    if (this.consumer) {\n      await this.consumer.disconnect();\n    }\n    if (this.producer) {\n      await this.producer.disconnect();\n    }\n\n    console.log(`Worker ${this.workerId} stopped`);\n  }\n}\n"]}
|
|
721
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"worker.js","sourceRoot":"","sources":["../src/worker.ts"],"names":[],"mappings":"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","sourcesContent":["// Worker - Node.js implementation for executing workflows and activities\n\nimport { Kafka, Consumer, Producer, SASLOptions, logLevel } from 'kafkajs';\nimport { generateAuthTokenFromCredentialsProvider } from 'aws-msk-iam-sasl-signer-js';\nimport * as os from 'os';\nimport type {\n  WorkerConfig,\n  ActivityHandler,\n  DecoratedActivity,\n  WorkflowClass,\n  WorkflowTask,\n  ActivityTask,\n  WorkflowResult,\n  ActivityResult,\n  WorkflowError,\n  ActivityError,\n  KafkaConfig,\n  ConnectResponse,\n} from './types';\nimport { WorkflowContext, runWithContext, getWorkflowName, findRunMethod, type WorkerInterface } from './workflow';\nimport { getActivityName } from './activity';\n\n/**\n * Worker that executes workflows and activities.\n * \n * Usage:\n *   import { Worker, workflow, activity } from '@raspect/workflow-sdk';\n *   \n *   const sayHello = activity.defn(async (input: { name: string }) => {\n *     return { message: `Hello, ${input.name}!` };\n *   });\n *   \n *   @workflow.defn\n *   class HelloWorldWorkflow {\n *     @workflow.run\n *     async run(input: { name: string }) {\n *       return await workflow.executeActivity(sayHello, input);\n *     }\n *   }\n *   \n *   const worker = new Worker({\n *     serverUrl: 'https://workflow.example.com',\n *     apiKey: 'wk_your_worker_api_key',  // Worker API key from server UI\n *   });\n *   worker.register({ workflows: [HelloWorldWorkflow], activities: [sayHello] });\n *   await worker.run();\n */\nexport class Worker implements WorkerInterface {\n  private config: WorkerConfig;\n  private workflows: Map<string, WorkflowClass> = new Map();\n  private activities: Map<string, ActivityHandler> = new Map();\n  \n  private workerId: string | null = null;\n  private instanceId: string | null = null;\n  private running = false;\n  private kafka: Kafka | null = null;\n  private consumer: Consumer | null = null;\n  private producer: Producer | null = null;\n  private heartbeatInterval: NodeJS.Timeout | null = null;\n  private heartbeatIntervalMs = 30000;\n  private resultTopic: string | null = null;\n  private maxConcurrentWorkflows: number;\n  private needsReconnect = false;\n  private workerTopic: string | null = null;\n\n  constructor(config: WorkerConfig) {\n    this.config = {\n      ...config,\n      serverUrl: config.serverUrl.replace(/\\/$/, ''),\n    };\n    this.maxConcurrentWorkflows = config.maxConcurrentWorkflows || 10;\n  }\n\n  /**\n   * Register workflows and activities.\n   */\n  register(options: {\n    workflows?: WorkflowClass[];\n    activities?: DecoratedActivity[];\n  }): void {\n    for (const wfCls of options.workflows || []) {\n      const wfName = getWorkflowName(wfCls);\n      this.workflows.set(wfName, wfCls);\n    }\n\n    for (const activityFn of options.activities || []) {\n      const activityName = getActivityName(activityFn);\n      this.activities.set(activityName, activityFn);\n    }\n  }\n\n  /**\n   * Start the worker (blocking).\n   */\n  async run(): Promise<void> {\n    this.running = true;\n\n    // Setup signal handlers\n    process.on('SIGINT', () => this.stop());\n    process.on('SIGTERM', () => this.stop());\n\n    // Main loop with automatic reconnection\n    while (this.running) {\n      try {\n        await this.connectAndRun();\n      } catch (error) {\n        if (!this.running) break;\n        const err = error as Error;\n        // Check if it's an auth error that shouldn't be retried\n        if (err.message.includes('Invalid or expired API key') || \n            err.message.includes('does not have permission')) {\n          console.error(`Worker stopped: ${err.message}`);\n          break;\n        }\n        console.error(`Worker error: ${err.message}. Will attempt to reconnect...`);\n        await this.cleanupKafka();\n        // Small delay before reconnection attempt\n        await new Promise(resolve => setTimeout(resolve, 5000));\n      }\n    }\n  }\n\n  private async connectAndRun(): Promise<void> {\n    this.needsReconnect = false;\n\n    // Connect to server and get Kafka config\n    const serverKafkaConfig = await this.connectToServer();\n    const workerTopic = serverKafkaConfig.worker_topic;\n    const resultTopic = serverKafkaConfig.result_topic;\n    this.resultTopic = resultTopic;\n    this.workerTopic = workerTopic;\n\n    // Create Kafka client\n    this.kafka = this.createKafkaClient(serverKafkaConfig);\n    \n    // Create consumer\n    const consumerGroupId = this.config.groupId || `workflow-worker-${this.workerId}`;\n    this.consumer = this.kafka.consumer({ groupId: consumerGroupId });\n    \n    // Create producer\n    this.producer = this.kafka.producer();\n\n    try {\n      await this.consumer.connect();\n      await this.producer.connect();\n      await this.consumer.subscribe({ topic: workerTopic, fromBeginning: false });\n\n      const workflowsStr = this.workflows.size > 0 ? Array.from(this.workflows.keys()).join(', ') : '(none)';\n      const activitiesStr = this.activities.size > 0 ? Array.from(this.activities.keys()).join(', ') : '(none)';\n      console.log(`Worker ${this.workerId} started, consuming from ${workerTopic}`);\n      console.log(`Max concurrent workflows: ${this.maxConcurrentWorkflows}`);\n      console.log(`Registered workflows: ${workflowsStr}`);\n      console.log(`Registered activities: ${activitiesStr}`);\n\n      // Start heartbeat\n      this.startHeartbeat();\n\n      // Start consuming messages\n      await this.consumer.run({\n        autoCommit: false,\n        partitionsConsumedConcurrently: this.maxConcurrentWorkflows,\n        eachMessage: async ({ topic, partition, message }) => {\n          if (!this.running || this.needsReconnect) return;\n          \n          await this.consumer?.commitOffsets([{\n            topic,\n            partition,\n            offset: (BigInt(message.offset) + 1n).toString(),\n          }]);\n          \n          try {\n            const data = JSON.parse(message.value?.toString() || '{}');\n            await this.processMessage(data, resultTopic);\n          } catch (error) {\n            const errMsg = (error as Error).message?.toLowerCase() || '';\n            if (errMsg.includes('disconnected') || errMsg.includes('connection')) {\n              console.warn('Kafka connection lost, triggering reconnection...');\n              this.needsReconnect = true;\n            } else {\n              console.error('Error processing message:', (error as Error).message);\n            }\n          }\n        },\n      });\n\n      // Keep running until stopped or needs reconnect\n      await new Promise<void>((resolve) => {\n        const checkInterval = setInterval(() => {\n          if (!this.running || this.needsReconnect) {\n            clearInterval(checkInterval);\n            resolve();\n          }\n        }, 100);\n      });\n\n      if (this.needsReconnect) {\n        console.log('Reconnecting to server and Kafka...');\n      }\n    } finally {\n      await this.cleanupKafka();\n    }\n  }\n\n  /**\n   * Stop the worker gracefully.\n   */\n  async stop(): Promise<void> {\n    console.log('Stopping worker...');\n    this.running = false;\n  }\n\n  private formatConnectionError(status: number, errorText: string): string {\n    if (status === 401) {\n      return 'Invalid or expired API key. Please check your worker API key.';\n    } else if (status === 403) {\n      return 'API key does not have permission to connect. Please check worker permissions.';\n    } else if (status === 404) {\n      return 'Server endpoint not found. Please check the server URL.';\n    } else if (status >= 500) {\n      return `Server error (HTTP ${status}). The server may be temporarily unavailable.`;\n    } else {\n      return `HTTP ${status} - ${errorText}`;\n    }\n  }\n\n  private async connectToServerOnce(): Promise<KafkaConfig> {\n    const hostname = os.hostname();\n    \n    const connectPayload = {\n      workflows: Array.from(this.workflows.keys()),\n      activities: Array.from(this.activities.keys()),\n      hostname,\n    };\n\n    const response = await fetch(`${this.config.serverUrl}/workflow/api/v1/workers/connect`, {\n      method: 'POST',\n      headers: {\n        'Content-Type': 'application/json',\n        'Authorization': `Bearer ${this.config.apiKey}`,\n      },\n      body: JSON.stringify(connectPayload),\n    });\n\n    if (!response.ok) {\n      const error = await response.text();\n      const errorMsg = this.formatConnectionError(response.status, error);\n      throw new Error(`Failed to connect to ${this.config.serverUrl}: ${errorMsg}`);\n    }\n\n    const data = await response.json() as ConnectResponse;\n    \n    this.workerId = data.worker_id;\n    this.instanceId = data.instance_id;\n    this.heartbeatIntervalMs = data.heartbeat_interval_ms || 30000;\n\n    console.log(`Connected to server as worker_id=${this.workerId}, instance_id=${this.instanceId}`);\n\n    return data.kafka;\n  }\n\n  private async connectToServer(): Promise<KafkaConfig> {\n    let retryInterval = 1000; // Start at 1 second\n    const maxRetryInterval = 600000; // Max 10 minutes\n    const backoffMultiplier = 2.0;\n\n    while (this.running) {\n      try {\n        return await this.connectToServerOnce();\n      } catch (error) {\n        const err = error as Error;\n        // Don't retry on auth errors\n        if (err.message.includes('Invalid or expired API key') || \n            err.message.includes('does not have permission')) {\n          console.error(`Worker failed to start: ${err.message}`);\n          throw error;\n        }\n\n        console.warn(`Connection failed: ${err.message}. Retrying in ${Math.round(retryInterval / 1000)}s...`);\n        await new Promise(resolve => setTimeout(resolve, retryInterval));\n        retryInterval = Math.min(retryInterval * backoffMultiplier, maxRetryInterval);\n      }\n    }\n\n    throw new Error('Worker stopped before connection could be established');\n  }\n\n  private createKafkaClient(kafkaConfig: KafkaConfig): Kafka {\n    const auth = kafkaConfig.auth;\n    \n    // Server returns brokers as comma-separated string, convert to array\n    const brokers = typeof kafkaConfig.brokers === 'string'\n      ? (kafkaConfig.brokers as string).split(',').map(b => b.trim()).filter(b => b)\n      : kafkaConfig.brokers;\n    \n    const config: any = {\n      clientId: `workflow-worker-${this.workerId}`,\n      brokers,\n      logLevel: logLevel.WARN,\n    };\n\n    if (auth?.type === 'msk_iam') {\n      config.ssl = true;\n      config.sasl = {\n        mechanism: 'oauthbearer',\n        oauthBearerProvider: async () => {\n          const token = await generateAuthTokenFromCredentialsProvider({\n            region: auth.region!,\n            awsCredentialsProvider: async () => ({\n              accessKeyId: auth.access_key_id!,\n              secretAccessKey: auth.secret_access_key!,\n            }),\n          });\n          return { value: token.token };\n        },\n      } as SASLOptions;\n    } else if (auth?.type === 'sasl') {\n      if (auth.security_protocol?.includes('SSL')) {\n        config.ssl = true;\n      }\n      if (auth.sasl_mechanism && auth.sasl_username && auth.sasl_password) {\n        config.sasl = {\n          mechanism: auth.sasl_mechanism.toLowerCase().replace('-', ''),\n          username: auth.sasl_username,\n          password: auth.sasl_password,\n        } as SASLOptions;\n      }\n    }\n\n    return new Kafka(config);\n  }\n\n  private startHeartbeat(): void {\n    let consecutiveFailures = 0;\n    const maxFailuresBeforeReconnect = 3;\n\n    this.heartbeatInterval = setInterval(async () => {\n      if (!this.running) return;\n      \n      try {\n        const response = await fetch(`${this.config.serverUrl}/workflow/api/v1/workers/heartbeat`, {\n          method: 'POST',\n          headers: {\n            'Content-Type': 'application/json',\n            'Authorization': `Bearer ${this.config.apiKey}`,\n          },\n          body: JSON.stringify({ instance_id: this.instanceId }),\n        });\n\n        if (response.ok) {\n          consecutiveFailures = 0;\n        } else if (response.status === 401) {\n          console.error('Heartbeat failed: Invalid or expired API key');\n          consecutiveFailures++;\n        } else {\n          console.warn(`Heartbeat failed: HTTP ${response.status}`);\n          consecutiveFailures++;\n        }\n      } catch (error) {\n        console.warn('Heartbeat error:', (error as Error).message);\n        consecutiveFailures++;\n      }\n\n      // Trigger reconnection if too many consecutive failures\n      if (consecutiveFailures >= maxFailuresBeforeReconnect) {\n        console.warn(\n          `Heartbeat failed ${consecutiveFailures} times consecutively. ` +\n          'Connection may be lost, triggering reconnection...'\n        );\n        this.needsReconnect = true;\n        if (this.heartbeatInterval) {\n          clearInterval(this.heartbeatInterval);\n          this.heartbeatInterval = null;\n        }\n      }\n    }, this.heartbeatIntervalMs);\n  }\n\n  private async disconnectFromServer(): Promise<void> {\n    if (!this.instanceId) return;\n\n    try {\n      const response = await fetch(`${this.config.serverUrl}/workflow/api/v1/workers/disconnect`, {\n        method: 'POST',\n        headers: {\n          'Content-Type': 'application/json',\n          'Authorization': `Bearer ${this.config.apiKey}`,\n        },\n        body: JSON.stringify({ instance_id: this.instanceId }),\n      });\n\n      if (response.ok) {\n        console.log('Disconnected from server');\n      }\n    } catch (error) {\n      console.warn('Failed to disconnect from server:', error);\n    }\n  }\n\n  private async processMessage(data: any, resultTopic: string): Promise<void> {\n    // Check if this is an activity result (response from another worker via server)\n    if (data.status && data.completed_at && !data.task_type) {\n      console.debug('Received activity result on worker topic (ignored - using polling)');\n      return;\n    }\n\n    const taskType = data.task_type || 'activity';\n\n    if (taskType === 'workflow') {\n      // Concurrency is controlled by partitionsConsumedConcurrently in consumer.run()\n      await this.executeWorkflow(data as WorkflowTask, resultTopic);\n    } else {\n      // Activity tasks can be processed inline\n      await this.processActivityTask(data, resultTopic);\n    }\n  }\n\n  private async executeWorkflow(task: WorkflowTask, resultTopic: string): Promise<void> {\n    console.log(`Received workflow task ${task.task_id}: ${task.workflow_name}`);\n\n    // Find the workflow class\n    const workflowCls = this.workflows.get(task.workflow_name);\n    if (!workflowCls) {\n      console.warn(`Unknown workflow: ${task.workflow_name}`);\n      await this.sendWorkflowResult(task, resultTopic, 'failed', undefined, {\n        type: 'UnknownWorkflow',\n        message: `Workflow '${task.workflow_name}' not registered`,\n      });\n      return;\n    }\n\n    // Find the run method\n    const runMethod = findRunMethod(workflowCls);\n    if (!runMethod) {\n      await this.sendWorkflowResult(task, resultTopic, 'failed', undefined, {\n        type: 'NoRunMethod',\n        message: `Workflow '${task.workflow_name}' has no run method`,\n      });\n      return;\n    }\n\n    // Execute the workflow\n    const startedAt = new Date();\n    try {\n      // Build completed activities map for replay\n      const completedActivitiesMap = new Map<number, unknown>();\n      if (task.completed_activities) {\n        for (const ca of task.completed_activities) {\n          completedActivitiesMap.set(ca.index, ca.output);\n        }\n        console.log(\n          `Resuming workflow ${task.workflow_id} with ${completedActivitiesMap.size} completed activities`\n        );\n      }\n\n      // Create workflow context\n      const ctx = new WorkflowContext(\n        this.activities,\n        task.workflow_id,\n        this,\n        completedActivitiesMap.size > 0 ? completedActivitiesMap : null,\n        task.metadata\n      );\n\n      // Execute within context\n      const output = await runWithContext(ctx, async () => {\n        return await Promise.race([\n          runMethod(task.input),\n          new Promise<never>((_, reject) =>\n            setTimeout(\n              () => reject(new Error(`Workflow timed out after ${task.timeout_ms}ms`)),\n              task.timeout_ms\n            )\n          ),\n        ]);\n      });\n\n      const completedAt = new Date();\n      const durationMs = completedAt.getTime() - startedAt.getTime();\n\n      console.log(`Workflow ${task.task_id} completed in ${durationMs}ms`);\n\n      await this.sendWorkflowResult(\n        task,\n        resultTopic,\n        'completed',\n        output,\n        undefined,\n        startedAt,\n        completedAt,\n        durationMs\n      );\n    } catch (error) {\n      const completedAt = new Date();\n      const durationMs = completedAt.getTime() - startedAt.getTime();\n      const err = error as Error;\n\n      console.error(`Workflow ${task.task_id} failed:`, err.message);\n\n      await this.sendWorkflowResult(\n        task,\n        resultTopic,\n        'failed',\n        undefined,\n        {\n          type: err.name || 'Error',\n          message: err.message,\n          stack_trace: err.stack,\n        },\n        startedAt,\n        completedAt,\n        durationMs\n      );\n    }\n  }\n\n  private async processActivityTask(data: ActivityTask, resultTopic: string): Promise<void> {\n    const task = data;\n    console.log(\n      `Received activity task ${task.task_id}: ${task.activity_name} (workflow=${task.workflow_id})`\n    );\n\n    // Find the activity handler\n    const activityFn = this.activities.get(task.activity_name);\n    if (!activityFn) {\n      console.warn(`Unknown activity: ${task.activity_name}`);\n      await this.sendActivityResult(task, resultTopic, 'failed', undefined, {\n        type: 'UnknownActivity',\n        message: `Activity '${task.activity_name}' not registered`,\n      });\n      return;\n    }\n\n    // Execute the activity\n    const startedAt = new Date();\n    try {\n      const output = await Promise.race([\n        Promise.resolve(activityFn(task.input)),\n        new Promise<never>((_, reject) =>\n          setTimeout(\n            () => reject(new Error(`Activity timed out after ${task.timeout_ms}ms`)),\n            task.timeout_ms\n          )\n        ),\n      ]);\n\n      const completedAt = new Date();\n      const durationMs = completedAt.getTime() - startedAt.getTime();\n\n      console.log(`Activity ${task.task_id} completed in ${durationMs}ms`);\n\n      await this.sendActivityResult(\n        task,\n        resultTopic,\n        'completed',\n        output,\n        undefined,\n        startedAt,\n        completedAt,\n        durationMs\n      );\n    } catch (error) {\n      const completedAt = new Date();\n      const durationMs = completedAt.getTime() - startedAt.getTime();\n      const err = error as Error;\n\n      console.error(`Activity ${task.task_id} failed:`, err.message);\n\n      await this.sendActivityResult(\n        task,\n        resultTopic,\n        'failed',\n        undefined,\n        {\n          type: err.name || 'Error',\n          message: err.message,\n          stack_trace: err.stack,\n        },\n        startedAt,\n        completedAt,\n        durationMs\n      );\n    }\n  }\n\n  private async sendWorkflowResult(\n    task: WorkflowTask,\n    resultTopic: string,\n    status: 'completed' | 'failed',\n    output?: unknown,\n    error?: WorkflowError,\n    startedAt?: Date,\n    completedAt?: Date,\n    durationMs = 0\n  ): Promise<void> {\n    const now = new Date();\n    const result: WorkflowResult = {\n      task_id: task.task_id,\n      workflow_id: task.workflow_id,\n      workflow_name: task.workflow_name,\n      status,\n      output,\n      error,\n      started_at: (startedAt || now).toISOString(),\n      completed_at: (completedAt || now).toISOString(),\n      duration_ms: durationMs,\n      worker_id: this.workerId!,\n    };\n\n    if (this.producer) {\n      await this.producer.send({\n        topic: resultTopic,\n        messages: [\n          {\n            key: task.workflow_id,\n            value: JSON.stringify(result),\n          },\n        ],\n      });\n      console.debug(`Sent workflow result for task ${task.task_id}`);\n    }\n  }\n\n  private async sendActivityResult(\n    task: ActivityTask,\n    resultTopic: string,\n    status: 'completed' | 'failed',\n    output?: unknown,\n    error?: ActivityError,\n    startedAt?: Date,\n    completedAt?: Date,\n    durationMs = 0\n  ): Promise<void> {\n    const now = new Date();\n    const result: ActivityResult = {\n      task_id: task.task_id,\n      workflow_id: task.workflow_id,\n      activity_name: task.activity_name,\n      activity_index: task.activity_index,\n      fan_out_index: task.fan_out_index,\n      status,\n      output,\n      error,\n      started_at: (startedAt || now).toISOString(),\n      completed_at: (completedAt || now).toISOString(),\n      duration_ms: durationMs,\n      worker_id: this.workerId!,\n      requester_worker_id: task.requester_worker_id,\n      correlation_id: task.correlation_id,\n    };\n\n    if (this.producer) {\n      await this.producer.send({\n        topic: resultTopic,\n        messages: [\n          {\n            key: task.workflow_id,\n            value: JSON.stringify(result),\n          },\n        ],\n      });\n      console.debug(`Sent activity result for task ${task.task_id}`);\n    }\n  }\n\n  // WorkerInterface implementation for WorkflowContext\n\n  async reportLocalActivityCompletion(\n    workflowId: string,\n    activityName: string,\n    activityIndex: number,\n    output: unknown,\n    durationMs: number\n  ): Promise<void> {\n    if (!this.producer || !this.resultTopic) {\n      console.debug('Cannot report local activity - producer not initialized');\n      return;\n    }\n\n    const completedAt = new Date();\n    const startedAt = new Date(completedAt.getTime() - durationMs);\n\n    const result = {\n      task_id: `local-${workflowId}-${activityIndex}`,\n      workflow_id: workflowId,\n      activity_name: activityName,\n      activity_index: activityIndex,\n      status: 'completed',\n      output,\n      started_at: startedAt.toISOString(),\n      completed_at: completedAt.toISOString(),\n      duration_ms: durationMs,\n      worker_id: this.workerId,\n    };\n\n    try {\n      await this.producer.send({\n        topic: this.resultTopic,\n        messages: [{ key: workflowId, value: JSON.stringify(result) }],\n      });\n      console.debug(\n        `Reported local activity completion: workflow_id=${workflowId}, activity=${activityName}, index=${activityIndex}`\n      );\n    } catch (error) {\n      console.warn('Failed to report local activity completion:', error);\n    }\n  }\n\n  async reportLocalActivityFailure(\n    workflowId: string,\n    activityName: string,\n    activityIndex: number,\n    error: string,\n    errorType: string,\n    durationMs: number\n  ): Promise<void> {\n    if (!this.producer || !this.resultTopic) {\n      console.debug('Cannot report local activity failure - producer not initialized');\n      return;\n    }\n\n    const completedAt = new Date();\n    const startedAt = new Date(completedAt.getTime() - durationMs);\n\n    const result = {\n      task_id: `local-${workflowId}-${activityIndex}`,\n      workflow_id: workflowId,\n      activity_name: activityName,\n      activity_index: activityIndex,\n      status: 'failed',\n      output: null,\n      error: { type: errorType, message: error },\n      started_at: startedAt.toISOString(),\n      completed_at: completedAt.toISOString(),\n      duration_ms: durationMs,\n      worker_id: this.workerId,\n    };\n\n    try {\n      await this.producer.send({\n        topic: this.resultTopic,\n        messages: [{ key: workflowId, value: JSON.stringify(result) }],\n      });\n      console.debug(\n        `Reported local activity failure: workflow_id=${workflowId}, activity=${activityName}, index=${activityIndex}`\n      );\n    } catch (error) {\n      console.warn('Failed to report local activity failure:', error);\n    }\n  }\n\n  async executeRemoteActivity(\n    workflowId: string,\n    activityName: string,\n    activityIndex: number,\n    inputData: unknown,\n    timeoutMs: number\n  ): Promise<unknown> {\n    if (!this.producer || !this.resultTopic) {\n      throw new Error('Kafka producer not initialized');\n    }\n\n    // Send ActivityRequest to server via Kafka\n    const requestPayload = {\n      request_type: 'activity_request',\n      request_id: crypto.randomUUID(),\n      workflow_id: workflowId,\n      activity_name: activityName,\n      activity_index: activityIndex,\n      input: inputData,\n      timeout_ms: timeoutMs,\n      requester_worker_id: this.workerId,\n      correlation_id: crypto.randomUUID(),\n      created_at: new Date().toISOString(),\n    };\n\n    await this.producer.send({\n      topic: this.resultTopic,\n      messages: [{ key: workflowId, value: JSON.stringify(requestPayload) }],\n    });\n\n    console.log(\n      `Activity request sent via Kafka: activity=${activityName}, workflow_id=${workflowId}, activity_index=${activityIndex}`\n    );\n\n    // Poll server for result with exponential backoff\n    let pollInterval = 2000; // Start at 2 seconds\n    const maxPollInterval = 900000; // Max 15 minutes\n    const backoffMultiplier = 1.5;\n    const startTime = Date.now();\n    const timeoutSeconds = timeoutMs;\n\n    const pollUrl = `${this.config.serverUrl}/workflow/api/v1/workflows/${workflowId}/activities/${activityIndex}/result`;\n    const headers = { Authorization: `Bearer ${this.config.apiKey}` };\n\n    while (true) {\n      const elapsed = Date.now() - startTime;\n      if (elapsed >= timeoutSeconds) {\n        throw new Error(`Activity ${activityName} timed out after ${timeoutMs}ms`);\n      }\n\n      try {\n        const response = await fetch(pollUrl, { headers });\n\n        if (response.status === 200) {\n          const result = await response.json() as { status: string; output?: unknown; error?: string };\n          if (result.status === 'completed') {\n            console.log(`Activity ${activityName} completed for workflow ${workflowId}`);\n            return result.output;\n          } else if (result.status === 'failed') {\n            const errorMsg = result.error || 'Activity failed';\n            console.error(`Activity ${activityName} failed: ${errorMsg}`);\n            throw new Error(errorMsg);\n          }\n        } else if (response.status === 202) {\n          // Still pending/running, continue polling\n          console.debug(`Activity ${activityName} still running, polling again in ${pollInterval}ms`);\n        } else if (response.status === 404) {\n          // Activity not found yet\n          console.debug(`Activity ${activityName} not found yet, polling again in ${pollInterval}ms`);\n        } else {\n          console.warn(`Unexpected response ${response.status} when polling activity result`);\n        }\n      } catch (error) {\n        if ((error as Error).message.includes('timed out') || (error as Error).message.includes('failed')) {\n          throw error;\n        }\n        console.warn('Error polling activity result:', error);\n      }\n\n      // Wait with exponential backoff\n      await new Promise((resolve) => setTimeout(resolve, pollInterval));\n      pollInterval = Math.min(pollInterval * backoffMultiplier, maxPollInterval);\n    }\n  }\n\n  private async cleanupKafka(): Promise<void> {\n    // Stop heartbeat\n    if (this.heartbeatInterval) {\n      clearInterval(this.heartbeatInterval);\n      this.heartbeatInterval = null;\n    }\n\n    // Stop Kafka clients\n    if (this.consumer) {\n      try {\n        await this.consumer.disconnect();\n      } catch (e) {\n        // Ignore disconnect errors\n      }\n      this.consumer = null;\n    }\n    if (this.producer) {\n      try {\n        await this.producer.disconnect();\n      } catch (e) {\n        // Ignore disconnect errors\n      }\n      this.producer = null;\n    }\n  }\n\n  private async cleanup(): Promise<void> {\n    await this.cleanupKafka();\n\n    // Disconnect from server\n    await this.disconnectFromServer();\n\n    console.log(`Worker ${this.workerId} stopped`);\n  }\n}\n"]}
|