@raspect/workflow-sdk 0.1.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -0,0 +1,74 @@
1
+ import type { ActivityHandler, WorkflowClass } from './types';
2
+ /**
3
+ * Context for workflow execution, holds activities and allows execute_activity calls.
4
+ */
5
+ export declare class WorkflowContext {
6
+ readonly activities: Map<string, ActivityHandler>;
7
+ readonly workflowId: string;
8
+ readonly worker: WorkerInterface | null;
9
+ readonly activityResults: unknown[];
10
+ readonly metadata: Record<string, unknown>;
11
+ private _completedActivities;
12
+ private _currentActivityIndex;
13
+ private _isReplaying;
14
+ constructor(activities: Map<string, ActivityHandler>, workflowId: string, worker?: WorkerInterface | null, completedActivities?: Map<number, unknown> | null, metadata?: Record<string, unknown> | null);
15
+ /**
16
+ * Execute an activity and return its result.
17
+ */
18
+ executeActivity<TInput = unknown, TOutput = unknown>(activity: ActivityHandler<TInput, TOutput> | string, inputData?: TInput, startToCloseTimeoutMs?: number): Promise<TOutput>;
19
+ }
20
+ export interface WorkerInterface {
21
+ reportLocalActivityCompletion(workflowId: string, activityName: string, activityIndex: number, output: unknown, durationMs: number): Promise<void>;
22
+ reportLocalActivityFailure(workflowId: string, activityName: string, activityIndex: number, error: string, errorType: string, durationMs: number): Promise<void>;
23
+ executeRemoteActivity(workflowId: string, activityName: string, activityIndex: number, inputData: unknown, timeoutMs: number): Promise<unknown>;
24
+ }
25
+ /**
26
+ * Get the current workflow context.
27
+ */
28
+ export declare function getCurrentContext(): WorkflowContext;
29
+ /**
30
+ * Run a function within a workflow context.
31
+ */
32
+ export declare function runWithContext<T>(ctx: WorkflowContext, fn: () => Promise<T>): Promise<T>;
33
+ /**
34
+ * Decorator to mark a class as a workflow definition.
35
+ *
36
+ * Usage:
37
+ * import { workflow } from '@inspectica/workflow-sdk';
38
+ *
39
+ * @workflow.defn
40
+ * class MyWorkflow {
41
+ * @workflow.run
42
+ * async run(input: { name: string }) {
43
+ * return await workflow.executeActivity(myActivity, input);
44
+ * }
45
+ * }
46
+ */
47
+ export declare function defn<T extends WorkflowClass>(cls: T): T;
48
+ /**
49
+ * Decorator to mark a method as the workflow entry point.
50
+ */
51
+ export declare function run(target: any, propertyKey: string, descriptor: PropertyDescriptor): PropertyDescriptor;
52
+ /**
53
+ * Execute an activity within a workflow.
54
+ *
55
+ * Usage:
56
+ * // Option 1: Local activity (function reference)
57
+ * const result = await workflow.executeActivity(myActivity, { key: "value" });
58
+ *
59
+ * // Option 2: Remote activity (string name) - for cross-worker calls
60
+ * const result = await workflow.executeActivity("say_count", { message: "hi", count: 1 });
61
+ */
62
+ export declare function executeActivity<TInput = unknown, TOutput = unknown>(activity: ActivityHandler<TInput, TOutput> | string, inputData?: TInput, startToCloseTimeoutMs?: number): Promise<TOutput>;
63
+ /**
64
+ * Get the workflow name from a class
65
+ */
66
+ export declare function getWorkflowName(cls: WorkflowClass): string;
67
+ /**
68
+ * Check if a class is a decorated workflow
69
+ */
70
+ export declare function isWorkflow(cls: unknown): boolean;
71
+ /**
72
+ * Find the run method in a workflow class
73
+ */
74
+ export declare function findRunMethod(cls: WorkflowClass): ((input: unknown) => Promise<unknown>) | null;
@@ -0,0 +1,186 @@
1
+ "use strict";
2
+ // Workflow decorators and context utilities
3
+ Object.defineProperty(exports, "__esModule", { value: true });
4
+ exports.WorkflowContext = void 0;
5
+ exports.getCurrentContext = getCurrentContext;
6
+ exports.runWithContext = runWithContext;
7
+ exports.defn = defn;
8
+ exports.run = run;
9
+ exports.executeActivity = executeActivity;
10
+ exports.getWorkflowName = getWorkflowName;
11
+ exports.isWorkflow = isWorkflow;
12
+ exports.findRunMethod = findRunMethod;
13
+ const activity_1 = require("./activity");
14
+ // Symbol for workflow metadata
15
+ const WORKFLOW_NAME = Symbol('workflowName');
16
+ const IS_WORKFLOW = Symbol('isWorkflow');
17
+ const IS_WORKFLOW_RUN = Symbol('isWorkflowRun');
18
+ // AsyncLocalStorage for workflow context (Node.js equivalent of Python's contextvars)
19
+ const async_hooks_1 = require("async_hooks");
20
+ const workflowContextStorage = new async_hooks_1.AsyncLocalStorage();
21
+ /**
22
+ * Context for workflow execution, holds activities and allows execute_activity calls.
23
+ */
24
+ class WorkflowContext {
25
+ activities;
26
+ workflowId;
27
+ worker;
28
+ activityResults = [];
29
+ metadata;
30
+ _completedActivities;
31
+ _currentActivityIndex = 0;
32
+ _isReplaying;
33
+ constructor(activities, workflowId, worker = null, completedActivities = null, metadata = null) {
34
+ this.activities = activities;
35
+ this.workflowId = workflowId;
36
+ this.worker = worker;
37
+ this._completedActivities = completedActivities || new Map();
38
+ this._currentActivityIndex = 0;
39
+ this._isReplaying = completedActivities !== null && completedActivities.size > 0;
40
+ this.metadata = metadata || {};
41
+ }
42
+ /**
43
+ * Execute an activity and return its result.
44
+ */
45
+ async executeActivity(activity, inputData, startToCloseTimeoutMs = 60000) {
46
+ // Increment activity index for this call
47
+ this._currentActivityIndex += 1;
48
+ const currentIndex = this._currentActivityIndex;
49
+ // Check if this activity was already completed (replay mode)
50
+ if (this._completedActivities.has(currentIndex)) {
51
+ const cachedResult = this._completedActivities.get(currentIndex);
52
+ console.log(`Replay: Returning cached result for activity index ${currentIndex}`);
53
+ this.activityResults.push(cachedResult);
54
+ return cachedResult;
55
+ }
56
+ // Support both function reference and string name
57
+ const activityName = (0, activity_1.getActivityName)(activity);
58
+ // Check if activity is registered locally
59
+ const activityFn = this.activities.get(activityName);
60
+ if (activityFn) {
61
+ // Execute locally (in-process)
62
+ const startTime = Date.now();
63
+ try {
64
+ const result = await Promise.race([
65
+ Promise.resolve(activityFn(inputData)),
66
+ new Promise((_, reject) => setTimeout(() => reject(new Error(`Activity timed out after ${startToCloseTimeoutMs}ms`)), startToCloseTimeoutMs)),
67
+ ]);
68
+ const durationMs = Date.now() - startTime;
69
+ // Report local activity completion to server for activity history tracking
70
+ if (this.worker) {
71
+ await this.worker.reportLocalActivityCompletion(this.workflowId, activityName, currentIndex, result, durationMs);
72
+ }
73
+ this.activityResults.push(result);
74
+ return result;
75
+ }
76
+ catch (error) {
77
+ const durationMs = Date.now() - startTime;
78
+ const err = error;
79
+ // Report local activity failure to server
80
+ if (this.worker) {
81
+ await this.worker.reportLocalActivityFailure(this.workflowId, activityName, currentIndex, err.message, err.name, durationMs);
82
+ }
83
+ throw error;
84
+ }
85
+ }
86
+ // Activity not registered locally - route via server
87
+ if (!this.worker) {
88
+ throw new Error(`Activity '${activityName}' not registered locally and no worker available for remote execution`);
89
+ }
90
+ // Execute remotely via server routing
91
+ const result = await this.worker.executeRemoteActivity(this.workflowId, activityName, currentIndex, inputData, startToCloseTimeoutMs);
92
+ this.activityResults.push(result);
93
+ return result;
94
+ }
95
+ }
96
+ exports.WorkflowContext = WorkflowContext;
97
+ /**
98
+ * Get the current workflow context.
99
+ */
100
+ function getCurrentContext() {
101
+ const ctx = workflowContextStorage.getStore();
102
+ if (!ctx) {
103
+ throw new Error('Not running inside a workflow context');
104
+ }
105
+ return ctx;
106
+ }
107
+ /**
108
+ * Run a function within a workflow context.
109
+ */
110
+ function runWithContext(ctx, fn) {
111
+ return workflowContextStorage.run(ctx, fn);
112
+ }
113
+ /**
114
+ * Decorator to mark a class as a workflow definition.
115
+ *
116
+ * Usage:
117
+ * import { workflow } from '@inspectica/workflow-sdk';
118
+ *
119
+ * @workflow.defn
120
+ * class MyWorkflow {
121
+ * @workflow.run
122
+ * async run(input: { name: string }) {
123
+ * return await workflow.executeActivity(myActivity, input);
124
+ * }
125
+ * }
126
+ */
127
+ function defn(cls) {
128
+ cls.__workflowName__ = cls.name;
129
+ cls.__isWorkflow__ = true;
130
+ cls[WORKFLOW_NAME] = cls.name;
131
+ cls[IS_WORKFLOW] = true;
132
+ return cls;
133
+ }
134
+ /**
135
+ * Decorator to mark a method as the workflow entry point.
136
+ */
137
+ function run(target, propertyKey, descriptor) {
138
+ descriptor.value.__isWorkflowRun__ = true;
139
+ descriptor.value[IS_WORKFLOW_RUN] = true;
140
+ return descriptor;
141
+ }
142
+ /**
143
+ * Execute an activity within a workflow.
144
+ *
145
+ * Usage:
146
+ * // Option 1: Local activity (function reference)
147
+ * const result = await workflow.executeActivity(myActivity, { key: "value" });
148
+ *
149
+ * // Option 2: Remote activity (string name) - for cross-worker calls
150
+ * const result = await workflow.executeActivity("say_count", { message: "hi", count: 1 });
151
+ */
152
+ async function executeActivity(activity, inputData, startToCloseTimeoutMs = 60000) {
153
+ const ctx = getCurrentContext();
154
+ return ctx.executeActivity(activity, inputData, startToCloseTimeoutMs);
155
+ }
156
+ /**
157
+ * Get the workflow name from a class
158
+ */
159
+ function getWorkflowName(cls) {
160
+ return cls.__workflowName__ || cls.name;
161
+ }
162
+ /**
163
+ * Check if a class is a decorated workflow
164
+ */
165
+ function isWorkflow(cls) {
166
+ return typeof cls === 'function' && cls.__isWorkflow__ === true;
167
+ }
168
+ /**
169
+ * Find the run method in a workflow class
170
+ */
171
+ function findRunMethod(cls) {
172
+ const instance = new cls();
173
+ const proto = Object.getPrototypeOf(instance);
174
+ for (const key of Object.getOwnPropertyNames(proto)) {
175
+ const method = instance[key];
176
+ if (typeof method === 'function' && method.__isWorkflowRun__) {
177
+ return method.bind(instance);
178
+ }
179
+ }
180
+ // Fallback: check for a method named 'run'
181
+ if (typeof instance.run === 'function') {
182
+ return instance.run.bind(instance);
183
+ }
184
+ return null;
185
+ }
186
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"workflow.js","sourceRoot":"","sources":["../src/workflow.ts"],"names":[],"mappings":"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","sourcesContent":["// Workflow decorators and context utilities\n\nimport type { ActivityHandler, WorkflowClass, WorkflowInstance } from './types';\nimport { getActivityName } from './activity';\n\n// Symbol for workflow metadata\nconst WORKFLOW_NAME = Symbol('workflowName');\nconst IS_WORKFLOW = Symbol('isWorkflow');\nconst IS_WORKFLOW_RUN = Symbol('isWorkflowRun');\n\n// AsyncLocalStorage for workflow context (Node.js equivalent of Python's contextvars)\nimport { AsyncLocalStorage } from 'async_hooks';\n\nconst workflowContextStorage = new AsyncLocalStorage<WorkflowContext>();\n\n/**\n * Context for workflow execution, holds activities and allows execute_activity calls.\n */\nexport class WorkflowContext {\n  public readonly activities: Map<string, ActivityHandler>;\n  public readonly workflowId: string;\n  public readonly worker: WorkerInterface | null;\n  public readonly activityResults: unknown[] = [];\n  public readonly metadata: Record<string, unknown>;\n\n  private _completedActivities: Map<number, unknown>;\n  private _currentActivityIndex = 0;\n  private _isReplaying: boolean;\n\n  constructor(\n    activities: Map<string, ActivityHandler>,\n    workflowId: string,\n    worker: WorkerInterface | null = null,\n    completedActivities: Map<number, unknown> | null = null,\n    metadata: Record<string, unknown> | null = null\n  ) {\n    this.activities = activities;\n    this.workflowId = workflowId;\n    this.worker = worker;\n    this._completedActivities = completedActivities || new Map();\n    this._currentActivityIndex = 0;\n    this._isReplaying = completedActivities !== null && completedActivities.size > 0;\n    this.metadata = metadata || {};\n  }\n\n  /**\n   * Execute an activity and return its result.\n   */\n  async executeActivity<TInput = unknown, TOutput = unknown>(\n    activity: ActivityHandler<TInput, TOutput> | string,\n    inputData?: TInput,\n    startToCloseTimeoutMs = 60000\n  ): Promise<TOutput> {\n    // Increment activity index for this call\n    this._currentActivityIndex += 1;\n    const currentIndex = this._currentActivityIndex;\n\n    // Check if this activity was already completed (replay mode)\n    if (this._completedActivities.has(currentIndex)) {\n      const cachedResult = this._completedActivities.get(currentIndex) as TOutput;\n      console.log(`Replay: Returning cached result for activity index ${currentIndex}`);\n      this.activityResults.push(cachedResult);\n      return cachedResult;\n    }\n\n    // Support both function reference and string name\n    const activityName = getActivityName(activity as ActivityHandler | string);\n\n    // Check if activity is registered locally\n    const activityFn = this.activities.get(activityName);\n    if (activityFn) {\n      // Execute locally (in-process)\n      const startTime = Date.now();\n\n      try {\n        const result = await Promise.race([\n          Promise.resolve(activityFn(inputData)),\n          new Promise<never>((_, reject) =>\n            setTimeout(() => reject(new Error(`Activity timed out after ${startToCloseTimeoutMs}ms`)), startToCloseTimeoutMs)\n          ),\n        ]) as TOutput;\n\n        const durationMs = Date.now() - startTime;\n\n        // Report local activity completion to server for activity history tracking\n        if (this.worker) {\n          await this.worker.reportLocalActivityCompletion(\n            this.workflowId,\n            activityName,\n            currentIndex,\n            result,\n            durationMs\n          );\n        }\n\n        this.activityResults.push(result);\n        return result;\n      } catch (error) {\n        const durationMs = Date.now() - startTime;\n        const err = error as Error;\n\n        // Report local activity failure to server\n        if (this.worker) {\n          await this.worker.reportLocalActivityFailure(\n            this.workflowId,\n            activityName,\n            currentIndex,\n            err.message,\n            err.name,\n            durationMs\n          );\n        }\n\n        throw error;\n      }\n    }\n\n    // Activity not registered locally - route via server\n    if (!this.worker) {\n      throw new Error(\n        `Activity '${activityName}' not registered locally and no worker available for remote execution`\n      );\n    }\n\n    // Execute remotely via server routing\n    const result = await this.worker.executeRemoteActivity(\n      this.workflowId,\n      activityName,\n      currentIndex,\n      inputData,\n      startToCloseTimeoutMs\n    ) as TOutput;\n\n    this.activityResults.push(result);\n    return result;\n  }\n}\n\n// Interface for worker methods needed by WorkflowContext\nexport interface WorkerInterface {\n  reportLocalActivityCompletion(\n    workflowId: string,\n    activityName: string,\n    activityIndex: number,\n    output: unknown,\n    durationMs: number\n  ): Promise<void>;\n  reportLocalActivityFailure(\n    workflowId: string,\n    activityName: string,\n    activityIndex: number,\n    error: string,\n    errorType: string,\n    durationMs: number\n  ): Promise<void>;\n  executeRemoteActivity(\n    workflowId: string,\n    activityName: string,\n    activityIndex: number,\n    inputData: unknown,\n    timeoutMs: number\n  ): Promise<unknown>;\n}\n\n/**\n * Get the current workflow context.\n */\nexport function getCurrentContext(): WorkflowContext {\n  const ctx = workflowContextStorage.getStore();\n  if (!ctx) {\n    throw new Error('Not running inside a workflow context');\n  }\n  return ctx;\n}\n\n/**\n * Run a function within a workflow context.\n */\nexport function runWithContext<T>(ctx: WorkflowContext, fn: () => Promise<T>): Promise<T> {\n  return workflowContextStorage.run(ctx, fn);\n}\n\n/**\n * Decorator to mark a class as a workflow definition.\n * \n * Usage:\n *   import { workflow } from '@inspectica/workflow-sdk';\n *   \n *   @workflow.defn\n *   class MyWorkflow {\n *     @workflow.run\n *     async run(input: { name: string }) {\n *       return await workflow.executeActivity(myActivity, input);\n *     }\n *   }\n */\nexport function defn<T extends WorkflowClass>(cls: T): T {\n  (cls as any).__workflowName__ = cls.name;\n  (cls as any).__isWorkflow__ = true;\n  (cls as any)[WORKFLOW_NAME] = cls.name;\n  (cls as any)[IS_WORKFLOW] = true;\n  return cls;\n}\n\n/**\n * Decorator to mark a method as the workflow entry point.\n */\nexport function run(\n  target: any,\n  propertyKey: string,\n  descriptor: PropertyDescriptor\n): PropertyDescriptor {\n  descriptor.value.__isWorkflowRun__ = true;\n  descriptor.value[IS_WORKFLOW_RUN] = true;\n  return descriptor;\n}\n\n/**\n * Execute an activity within a workflow.\n * \n * Usage:\n *   // Option 1: Local activity (function reference)\n *   const result = await workflow.executeActivity(myActivity, { key: \"value\" });\n *   \n *   // Option 2: Remote activity (string name) - for cross-worker calls\n *   const result = await workflow.executeActivity(\"say_count\", { message: \"hi\", count: 1 });\n */\nexport async function executeActivity<TInput = unknown, TOutput = unknown>(\n  activity: ActivityHandler<TInput, TOutput> | string,\n  inputData?: TInput,\n  startToCloseTimeoutMs = 60000\n): Promise<TOutput> {\n  const ctx = getCurrentContext();\n  return ctx.executeActivity(activity, inputData, startToCloseTimeoutMs);\n}\n\n/**\n * Get the workflow name from a class\n */\nexport function getWorkflowName(cls: WorkflowClass): string {\n  return (cls as any).__workflowName__ || cls.name;\n}\n\n/**\n * Check if a class is a decorated workflow\n */\nexport function isWorkflow(cls: unknown): boolean {\n  return typeof cls === 'function' && (cls as any).__isWorkflow__ === true;\n}\n\n/**\n * Find the run method in a workflow class\n */\nexport function findRunMethod(cls: WorkflowClass): ((input: unknown) => Promise<unknown>) | null {\n  const instance = new cls();\n  const proto = Object.getPrototypeOf(instance);\n  \n  for (const key of Object.getOwnPropertyNames(proto)) {\n    const method = (instance as any)[key];\n    if (typeof method === 'function' && method.__isWorkflowRun__) {\n      return method.bind(instance);\n    }\n  }\n  \n  // Fallback: check for a method named 'run'\n  if (typeof instance.run === 'function') {\n    return instance.run.bind(instance);\n  }\n  \n  return null;\n}\n"]}
package/package.json ADDED
@@ -0,0 +1,49 @@
1
+ {
2
+ "name": "@raspect/workflow-sdk",
3
+ "version": "0.1.0",
4
+ "description": "Node.js SDK for Inspectica Workflow Server - Activity worker implementation",
5
+ "main": "dist/index.js",
6
+ "types": "dist/index.d.ts",
7
+ "files": [
8
+ "dist"
9
+ ],
10
+ "scripts": {
11
+ "build": "tsc",
12
+ "test": "jest",
13
+ "lint": "eslint src --ext .ts",
14
+ "format": "prettier --write src/**/*.ts",
15
+ "prepublishOnly": "npm run build"
16
+ },
17
+ "keywords": [
18
+ "workflow",
19
+ "orchestration",
20
+ "kafka",
21
+ "distributed",
22
+ "activity",
23
+ "worker"
24
+ ],
25
+ "author": "Inspectica Team",
26
+ "license": "MIT",
27
+ "repository": {
28
+ "type": "git",
29
+ "url": "https://github.com/your-org/inspectica-workflow-server.git",
30
+ "directory": "packages/node-sdk"
31
+ },
32
+ "engines": {
33
+ "node": ">=18.0.0"
34
+ },
35
+ "dependencies": {
36
+ "aws-msk-iam-sasl-signer-js": "^1.0.0",
37
+ "kafkajs": "^2.2.4"
38
+ },
39
+ "devDependencies": {
40
+ "@types/jest": "^29.5.14",
41
+ "@types/node": "^22.10.0",
42
+ "eslint": "^9.17.0",
43
+ "jest": "^29.7.0",
44
+ "prettier": "^3.4.2",
45
+ "ts-jest": "^29.2.5",
46
+ "typescript": "^5.7.2",
47
+ "typescript-eslint": "^8.18.0"
48
+ }
49
+ }