@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 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 '@inspectica/workflow-sdk';
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: 'wk_abc123...',
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 '@inspectica/workflow-sdk';
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: 'wk_abc123...',
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: ${Array.from(this.workflows.keys()).join(', ')}`);
131
- console.log(`Registered activities: ${Array.from(this.activities.keys()).join(', ')}`);
157
+ console.log(`Registered workflows: ${workflowsStr}`);
158
+ console.log(`Registered activities: ${activitiesStr}`);
132
159
  // Start heartbeat
133
160
  this.startHeartbeat();
134
- // Start consuming messages with worker-level concurrency control
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
- console.error('Error processing message:', error);
158
- // Workflow failed - state is in MongoDB, can be resumed via server API
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.cleanup();
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
- async connectToServer() {
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
- throw new Error(`Failed to connect to server: ${response.status} - ${error}`);
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 (!response.ok) {
264
- console.warn(`Heartbeat failed: ${response.status}`);
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 cleanup() {
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
- await this.consumer.disconnect();
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
- await this.producer.disconnect();
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"]}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@raspect/workflow-sdk",
3
- "version": "0.2.0",
3
+ "version": "0.3.0",
4
4
  "description": "Node.js SDK for Inspectica Workflow Server - Activity worker implementation",
5
5
  "main": "dist/index.js",
6
6
  "types": "dist/index.d.ts",