ai-workflows 2.0.2 → 2.1.3

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.
Files changed (98) hide show
  1. package/.turbo/turbo-build.log +4 -5
  2. package/.turbo/turbo-test.log +169 -0
  3. package/CHANGELOG.md +29 -0
  4. package/LICENSE +21 -0
  5. package/README.md +303 -184
  6. package/dist/barrier.d.ts +153 -0
  7. package/dist/barrier.d.ts.map +1 -0
  8. package/dist/barrier.js +339 -0
  9. package/dist/barrier.js.map +1 -0
  10. package/dist/cascade-context.d.ts +149 -0
  11. package/dist/cascade-context.d.ts.map +1 -0
  12. package/dist/cascade-context.js +324 -0
  13. package/dist/cascade-context.js.map +1 -0
  14. package/dist/cascade-executor.d.ts +196 -0
  15. package/dist/cascade-executor.d.ts.map +1 -0
  16. package/dist/cascade-executor.js +384 -0
  17. package/dist/cascade-executor.js.map +1 -0
  18. package/dist/context.d.ts.map +1 -1
  19. package/dist/context.js +4 -1
  20. package/dist/context.js.map +1 -1
  21. package/dist/dependency-graph.d.ts +157 -0
  22. package/dist/dependency-graph.d.ts.map +1 -0
  23. package/dist/dependency-graph.js +382 -0
  24. package/dist/dependency-graph.js.map +1 -0
  25. package/dist/every.d.ts +31 -2
  26. package/dist/every.d.ts.map +1 -1
  27. package/dist/every.js +63 -32
  28. package/dist/every.js.map +1 -1
  29. package/dist/graph/index.d.ts +8 -0
  30. package/dist/graph/index.d.ts.map +1 -0
  31. package/dist/graph/index.js +8 -0
  32. package/dist/graph/index.js.map +1 -0
  33. package/dist/graph/topological-sort.d.ts +121 -0
  34. package/dist/graph/topological-sort.d.ts.map +1 -0
  35. package/dist/graph/topological-sort.js +292 -0
  36. package/dist/graph/topological-sort.js.map +1 -0
  37. package/dist/index.d.ts +6 -1
  38. package/dist/index.d.ts.map +1 -1
  39. package/dist/index.js +10 -0
  40. package/dist/index.js.map +1 -1
  41. package/dist/on.d.ts +35 -10
  42. package/dist/on.d.ts.map +1 -1
  43. package/dist/on.js +52 -18
  44. package/dist/on.js.map +1 -1
  45. package/dist/send.d.ts +0 -5
  46. package/dist/send.d.ts.map +1 -1
  47. package/dist/send.js +1 -14
  48. package/dist/send.js.map +1 -1
  49. package/dist/timer-registry.d.ts +52 -0
  50. package/dist/timer-registry.d.ts.map +1 -0
  51. package/dist/timer-registry.js +120 -0
  52. package/dist/timer-registry.js.map +1 -0
  53. package/dist/types.d.ts +171 -9
  54. package/dist/types.d.ts.map +1 -1
  55. package/dist/types.js +17 -1
  56. package/dist/types.js.map +1 -1
  57. package/dist/workflow.d.ts.map +1 -1
  58. package/dist/workflow.js +22 -18
  59. package/dist/workflow.js.map +1 -1
  60. package/package.json +12 -16
  61. package/src/barrier.ts +466 -0
  62. package/src/cascade-context.ts +488 -0
  63. package/src/cascade-executor.ts +587 -0
  64. package/src/context.js +83 -0
  65. package/src/context.ts +12 -7
  66. package/src/dependency-graph.ts +518 -0
  67. package/src/every.js +267 -0
  68. package/src/every.ts +104 -35
  69. package/src/graph/index.ts +19 -0
  70. package/src/graph/topological-sort.ts +414 -0
  71. package/src/index.js +71 -0
  72. package/src/index.ts +78 -0
  73. package/src/on.js +79 -0
  74. package/src/on.ts +81 -25
  75. package/src/send.js +111 -0
  76. package/src/send.ts +1 -16
  77. package/src/timer-registry.ts +145 -0
  78. package/src/types.js +4 -0
  79. package/src/types.ts +218 -11
  80. package/src/workflow.js +455 -0
  81. package/src/workflow.ts +32 -23
  82. package/test/barrier-join.test.ts +434 -0
  83. package/test/barrier-unhandled-rejections.test.ts +359 -0
  84. package/test/cascade-context.test.ts +390 -0
  85. package/test/cascade-executor.test.ts +859 -0
  86. package/test/context.test.js +116 -0
  87. package/test/dependency-graph.test.ts +512 -0
  88. package/test/every.test.js +282 -0
  89. package/test/graph/topological-sort.test.ts +586 -0
  90. package/test/on.test.js +80 -0
  91. package/test/schedule-timer-cleanup.test.ts +344 -0
  92. package/test/send-race-conditions.test.ts +410 -0
  93. package/test/send.test.js +89 -0
  94. package/test/type-safety-every.test.ts +303 -0
  95. package/test/types-event-handler.test.ts +225 -0
  96. package/test/types-proxy-autocomplete.test.ts +345 -0
  97. package/test/workflow.test.js +224 -0
  98. package/vitest.config.js +7 -0
@@ -0,0 +1,455 @@
1
+ /**
2
+ * Unified Workflow API
3
+ *
4
+ * Usage:
5
+ * Workflow($ => {
6
+ * $.on.Customer.created(async (customer, $) => {
7
+ * $.log('Customer created', customer)
8
+ * await $.send('Email.welcome', { to: customer.email })
9
+ * })
10
+ *
11
+ * $.every.Monday.at9am(async ($) => {
12
+ * $.log('Weekly standup reminder')
13
+ * })
14
+ * })
15
+ */
16
+ /**
17
+ * Well-known cron patterns for common schedules
18
+ */
19
+ const KNOWN_PATTERNS = {
20
+ second: '* * * * * *',
21
+ minute: '* * * * *',
22
+ hour: '0 * * * *',
23
+ day: '0 0 * * *',
24
+ week: '0 0 * * 0',
25
+ month: '0 0 1 * *',
26
+ year: '0 0 1 1 *',
27
+ Monday: '0 0 * * 1',
28
+ Tuesday: '0 0 * * 2',
29
+ Wednesday: '0 0 * * 3',
30
+ Thursday: '0 0 * * 4',
31
+ Friday: '0 0 * * 5',
32
+ Saturday: '0 0 * * 6',
33
+ Sunday: '0 0 * * 0',
34
+ weekday: '0 0 * * 1-5',
35
+ weekend: '0 0 * * 0,6',
36
+ midnight: '0 0 * * *',
37
+ noon: '0 12 * * *',
38
+ };
39
+ /**
40
+ * Time suffixes for day-based schedules
41
+ */
42
+ const TIME_PATTERNS = {
43
+ at6am: { hour: 6, minute: 0 },
44
+ at7am: { hour: 7, minute: 0 },
45
+ at8am: { hour: 8, minute: 0 },
46
+ at9am: { hour: 9, minute: 0 },
47
+ at10am: { hour: 10, minute: 0 },
48
+ at11am: { hour: 11, minute: 0 },
49
+ at12pm: { hour: 12, minute: 0 },
50
+ atnoon: { hour: 12, minute: 0 },
51
+ at1pm: { hour: 13, minute: 0 },
52
+ at2pm: { hour: 14, minute: 0 },
53
+ at3pm: { hour: 15, minute: 0 },
54
+ at4pm: { hour: 16, minute: 0 },
55
+ at5pm: { hour: 17, minute: 0 },
56
+ at6pm: { hour: 18, minute: 0 },
57
+ at7pm: { hour: 19, minute: 0 },
58
+ at8pm: { hour: 20, minute: 0 },
59
+ at9pm: { hour: 21, minute: 0 },
60
+ atmidnight: { hour: 0, minute: 0 },
61
+ };
62
+ /**
63
+ * Combine a day pattern with a time pattern
64
+ */
65
+ function combineWithTime(baseCron, time) {
66
+ const parts = baseCron.split(' ');
67
+ parts[0] = String(time.minute);
68
+ parts[1] = String(time.hour);
69
+ return parts.join(' ');
70
+ }
71
+ /**
72
+ * Parse event string into noun and event
73
+ */
74
+ export function parseEvent(event) {
75
+ const parts = event.split('.');
76
+ if (parts.length !== 2) {
77
+ return null;
78
+ }
79
+ const [noun, eventName] = parts;
80
+ if (!noun || !eventName) {
81
+ return null;
82
+ }
83
+ return { noun, event: eventName };
84
+ }
85
+ /**
86
+ * Create a workflow with the $ context
87
+ *
88
+ * @example
89
+ * ```ts
90
+ * const workflow = Workflow($ => {
91
+ * $.on.Customer.created(async (customer, $) => {
92
+ * $.log('New customer:', customer.name)
93
+ * await $.send('Email.welcome', { to: customer.email })
94
+ * })
95
+ *
96
+ * $.every.hour(async ($) => {
97
+ * $.log('Hourly check')
98
+ * })
99
+ *
100
+ * $.every.Monday.at9am(async ($) => {
101
+ * $.log('Weekly standup')
102
+ * })
103
+ *
104
+ * $.every('first Monday of the month', async ($) => {
105
+ * $.log('Monthly report')
106
+ * })
107
+ * })
108
+ *
109
+ * await workflow.start()
110
+ * await workflow.send('Customer.created', { name: 'John', email: 'john@example.com' })
111
+ * ```
112
+ */
113
+ export function Workflow(setup, options = {}) {
114
+ // Registries for handlers captured during setup
115
+ const eventRegistry = [];
116
+ const scheduleRegistry = [];
117
+ // State
118
+ const state = {
119
+ context: { ...options.context },
120
+ history: [],
121
+ };
122
+ // Schedule timers
123
+ let scheduleTimers = [];
124
+ /**
125
+ * Add to history
126
+ */
127
+ const addHistory = (entry) => {
128
+ state.history.push({
129
+ ...entry,
130
+ timestamp: Date.now(),
131
+ });
132
+ };
133
+ /**
134
+ * Register an event handler
135
+ */
136
+ const registerEventHandler = (noun, event, handler) => {
137
+ eventRegistry.push({
138
+ noun,
139
+ event,
140
+ handler,
141
+ source: handler.toString(),
142
+ });
143
+ };
144
+ /**
145
+ * Register a schedule handler
146
+ */
147
+ const registerScheduleHandler = (interval, handler) => {
148
+ scheduleRegistry.push({
149
+ interval,
150
+ handler,
151
+ source: handler.toString(),
152
+ });
153
+ };
154
+ /**
155
+ * Create the $.on proxy
156
+ */
157
+ const createOnProxy = () => {
158
+ return new Proxy({}, {
159
+ get(_target, noun) {
160
+ return new Proxy({}, {
161
+ get(_eventTarget, event) {
162
+ return (handler) => {
163
+ registerEventHandler(noun, event, handler);
164
+ };
165
+ }
166
+ });
167
+ }
168
+ });
169
+ };
170
+ /**
171
+ * Create the $.every proxy
172
+ */
173
+ const createEveryProxy = () => {
174
+ const handler = {
175
+ get(_target, prop) {
176
+ const pattern = KNOWN_PATTERNS[prop];
177
+ if (pattern) {
178
+ const result = (handlerFn) => {
179
+ registerScheduleHandler({ type: 'cron', expression: pattern, natural: prop }, handlerFn);
180
+ };
181
+ return new Proxy(result, {
182
+ get(_t, timeKey) {
183
+ const time = TIME_PATTERNS[timeKey];
184
+ if (time) {
185
+ const cron = combineWithTime(pattern, time);
186
+ return (handlerFn) => {
187
+ registerScheduleHandler({ type: 'cron', expression: cron, natural: `${prop}.${timeKey}` }, handlerFn);
188
+ };
189
+ }
190
+ return undefined;
191
+ },
192
+ apply(_t, _thisArg, args) {
193
+ registerScheduleHandler({ type: 'cron', expression: pattern, natural: prop }, args[0]);
194
+ }
195
+ });
196
+ }
197
+ // Plural units (seconds, minutes, hours, days, weeks)
198
+ const pluralUnits = {
199
+ seconds: 'second',
200
+ minutes: 'minute',
201
+ hours: 'hour',
202
+ days: 'day',
203
+ weeks: 'week',
204
+ };
205
+ if (pluralUnits[prop]) {
206
+ return (value) => (handlerFn) => {
207
+ registerScheduleHandler({ type: pluralUnits[prop], value, natural: `${value} ${prop}` }, handlerFn);
208
+ };
209
+ }
210
+ return undefined;
211
+ },
212
+ apply(_target, _thisArg, args) {
213
+ const [description, handler] = args;
214
+ if (typeof description === 'string' && typeof handler === 'function') {
215
+ registerScheduleHandler({ type: 'natural', description }, handler);
216
+ }
217
+ }
218
+ };
219
+ return new Proxy(function () { }, handler);
220
+ };
221
+ /**
222
+ * Deliver an event to matching handlers (fire and forget)
223
+ */
224
+ const deliverEvent = async (event, data) => {
225
+ const parsed = parseEvent(event);
226
+ if (!parsed) {
227
+ console.warn(`Invalid event format: ${event}. Expected Noun.event`);
228
+ return;
229
+ }
230
+ const matching = eventRegistry.filter(h => h.noun === parsed.noun && h.event === parsed.event);
231
+ if (matching.length === 0) {
232
+ return;
233
+ }
234
+ await Promise.all(matching.map(async ({ handler }) => {
235
+ try {
236
+ await handler(data, $);
237
+ }
238
+ catch (error) {
239
+ console.error(`Error in handler for ${event}:`, error);
240
+ }
241
+ }));
242
+ };
243
+ /**
244
+ * Execute an event and wait for result from first matching handler
245
+ */
246
+ const executeEvent = async (event, data, durable) => {
247
+ const parsed = parseEvent(event);
248
+ if (!parsed) {
249
+ throw new Error(`Invalid event format: ${event}. Expected Noun.event`);
250
+ }
251
+ const matching = eventRegistry.filter(h => h.noun === parsed.noun && h.event === parsed.event);
252
+ if (matching.length === 0) {
253
+ throw new Error(`No handler registered for ${event}`);
254
+ }
255
+ // Use first matching handler for result
256
+ const { handler } = matching[0];
257
+ if (durable && options.db) {
258
+ // Create action for durability tracking
259
+ await options.db.createAction({
260
+ actor: 'workflow',
261
+ object: event,
262
+ action: 'execute',
263
+ metadata: { data },
264
+ });
265
+ }
266
+ try {
267
+ const result = await handler(data, $);
268
+ return result;
269
+ }
270
+ catch (error) {
271
+ if (durable) {
272
+ // Could implement retry logic here
273
+ console.error(`[workflow] Durable action failed for ${event}:`, error);
274
+ }
275
+ throw error;
276
+ }
277
+ };
278
+ /**
279
+ * Create the $ context
280
+ */
281
+ const $ = {
282
+ async send(event, data) {
283
+ addHistory({ type: 'event', name: event, data });
284
+ // Record to database if connected (durable)
285
+ if (options.db) {
286
+ await options.db.recordEvent(event, data);
287
+ }
288
+ await deliverEvent(event, data);
289
+ },
290
+ async do(event, data) {
291
+ addHistory({ type: 'action', name: `do:${event}`, data });
292
+ // Record to database (durable)
293
+ if (options.db) {
294
+ await options.db.recordEvent(event, data);
295
+ }
296
+ return executeEvent(event, data, true);
297
+ },
298
+ async try(event, data) {
299
+ addHistory({ type: 'action', name: `try:${event}`, data });
300
+ // Non-durable - no database recording
301
+ return executeEvent(event, data, false);
302
+ },
303
+ on: createOnProxy(),
304
+ every: createEveryProxy(),
305
+ // Direct access to state context
306
+ state: state.context,
307
+ getState() {
308
+ // Return a deep copy to prevent mutation
309
+ return {
310
+ current: state.current,
311
+ context: { ...state.context },
312
+ history: [...state.history],
313
+ };
314
+ },
315
+ set(key, value) {
316
+ state.context[key] = value;
317
+ },
318
+ get(key) {
319
+ return state.context[key];
320
+ },
321
+ log(message, data) {
322
+ addHistory({ type: 'action', name: 'log', data: { message, data } });
323
+ console.log(`[workflow] ${message}`, data ?? '');
324
+ },
325
+ db: options.db,
326
+ };
327
+ // Run setup to capture handlers
328
+ setup($);
329
+ /**
330
+ * Start schedule handlers
331
+ */
332
+ const startSchedules = async () => {
333
+ for (const schedule of scheduleRegistry) {
334
+ const { interval, handler } = schedule;
335
+ let ms = 0;
336
+ switch (interval.type) {
337
+ case 'second':
338
+ ms = (interval.value ?? 1) * 1000;
339
+ break;
340
+ case 'minute':
341
+ ms = (interval.value ?? 1) * 60 * 1000;
342
+ break;
343
+ case 'hour':
344
+ ms = (interval.value ?? 1) * 60 * 60 * 1000;
345
+ break;
346
+ case 'day':
347
+ ms = (interval.value ?? 1) * 24 * 60 * 60 * 1000;
348
+ break;
349
+ case 'week':
350
+ ms = (interval.value ?? 1) * 7 * 24 * 60 * 60 * 1000;
351
+ break;
352
+ case 'cron':
353
+ case 'natural':
354
+ // Cron/natural need special handling
355
+ console.log(`[workflow] Cron/natural scheduling not yet implemented: ${interval.type === 'cron' ? interval.expression : interval.description}`);
356
+ continue;
357
+ }
358
+ if (ms > 0) {
359
+ const timer = setInterval(async () => {
360
+ try {
361
+ addHistory({ type: 'schedule', name: interval.natural ?? interval.type });
362
+ await handler($);
363
+ }
364
+ catch (error) {
365
+ console.error('[workflow] Schedule handler error:', error);
366
+ }
367
+ }, ms);
368
+ scheduleTimers.push(timer);
369
+ }
370
+ }
371
+ };
372
+ const instance = {
373
+ definition: {
374
+ name: 'workflow',
375
+ events: eventRegistry,
376
+ schedules: scheduleRegistry,
377
+ initialContext: options.context,
378
+ },
379
+ get state() {
380
+ return state;
381
+ },
382
+ $,
383
+ async send(event, data) {
384
+ await $.send(event, data);
385
+ },
386
+ async start() {
387
+ console.log(`[workflow] Starting with ${eventRegistry.length} event handlers and ${scheduleRegistry.length} schedules`);
388
+ await startSchedules();
389
+ },
390
+ async stop() {
391
+ console.log('[workflow] Stopping');
392
+ for (const timer of scheduleTimers) {
393
+ clearInterval(timer);
394
+ }
395
+ scheduleTimers = [];
396
+ },
397
+ };
398
+ return instance;
399
+ }
400
+ /**
401
+ * Create an isolated $ context for testing
402
+ */
403
+ export function createTestContext() {
404
+ const emittedEvents = [];
405
+ const stateContext = {};
406
+ const history = [];
407
+ const $ = {
408
+ emittedEvents,
409
+ async send(event, data) {
410
+ emittedEvents.push({ event, data });
411
+ },
412
+ async do(_event, _data) {
413
+ throw new Error('$.do not implemented in test context - register handlers via Workflow()');
414
+ },
415
+ async try(_event, _data) {
416
+ throw new Error('$.try not implemented in test context - register handlers via Workflow()');
417
+ },
418
+ on: new Proxy({}, {
419
+ get() {
420
+ return new Proxy({}, {
421
+ get() {
422
+ return () => { }; // No-op for testing
423
+ }
424
+ });
425
+ }
426
+ }),
427
+ every: new Proxy(function () { }, {
428
+ get() {
429
+ return () => () => { }; // No-op for testing
430
+ },
431
+ apply() { }
432
+ }),
433
+ state: stateContext,
434
+ getState() {
435
+ return {
436
+ context: { ...stateContext },
437
+ history: [...history],
438
+ };
439
+ },
440
+ set(key, value) {
441
+ stateContext[key] = value;
442
+ },
443
+ get(key) {
444
+ return stateContext[key];
445
+ },
446
+ log(message, data) {
447
+ console.log(`[test] ${message}`, data ?? '');
448
+ },
449
+ };
450
+ return $;
451
+ }
452
+ // Also export standalone on/every for import { on, every } usage
453
+ export { on, registerEventHandler, getEventHandlers, clearEventHandlers } from './on.js';
454
+ export { every, registerScheduleHandler, getScheduleHandlers, clearScheduleHandlers, toCron, intervalToMs, formatInterval, setCronConverter } from './every.js';
455
+ export { send } from './send.js';
package/src/workflow.ts CHANGED
@@ -27,9 +27,11 @@ import type {
27
27
  WorkflowOptions,
28
28
  OnProxy,
29
29
  EveryProxy,
30
+ EveryProxyTarget,
30
31
  ParsedEvent,
31
32
  DatabaseContext,
32
33
  } from './types.js'
34
+ import { PLURAL_UNITS, isPluralUnitKey } from './types.js'
33
35
 
34
36
  /**
35
37
  * Well-known cron patterns for common schedules
@@ -246,17 +248,12 @@ export function Workflow(
246
248
  }
247
249
 
248
250
  // Plural units (seconds, minutes, hours, days, weeks)
249
- const pluralUnits: Record<string, string> = {
250
- seconds: 'second',
251
- minutes: 'minute',
252
- hours: 'hour',
253
- days: 'day',
254
- weeks: 'week',
255
- }
256
- if (pluralUnits[prop]) {
251
+ // Using type guard and typed constant for type-safe interval creation
252
+ if (isPluralUnitKey(prop)) {
253
+ const intervalType = PLURAL_UNITS[prop]
257
254
  return (value: number) => (handlerFn: ScheduleHandler) => {
258
255
  registerScheduleHandler(
259
- { type: pluralUnits[prop] as any, value, natural: `${value} ${prop}` },
256
+ { type: intervalType, value, natural: `${value} ${prop}` },
260
257
  handlerFn
261
258
  )
262
259
  }
@@ -273,7 +270,12 @@ export function Workflow(
273
270
  }
274
271
  }
275
272
 
276
- return new Proxy(function() {} as any, handler)
273
+ // Create callable target with proper typing
274
+ // The function serves as the Proxy target - actual behavior is in the handler's apply trap
275
+ // Cast to EveryProxy is safe: Proxy handler implements all EveryProxy behaviors dynamically
276
+ // eslint-disable-next-line @typescript-eslint/no-unused-vars
277
+ const target: EveryProxyTarget = function(_description: string, _handler: ScheduleHandler) {}
278
+ return new Proxy(target, handler) as unknown as EveryProxy
277
279
  }
278
280
 
279
281
  /**
@@ -392,11 +394,11 @@ export function Workflow(
392
394
 
393
395
  getState(): WorkflowState {
394
396
  // Return a deep copy to prevent mutation
395
- return {
397
+ return structuredClone({
396
398
  current: state.current,
397
- context: { ...state.context },
398
- history: [...state.history],
399
- }
399
+ context: state.context,
400
+ history: state.history,
401
+ })
400
402
  },
401
403
 
402
404
  set<T = unknown>(key: string, value: T): void {
@@ -444,9 +446,11 @@ export function Workflow(
444
446
  break
445
447
  case 'cron':
446
448
  case 'natural':
447
- // Cron/natural need special handling
448
- console.log(`[workflow] Cron/natural scheduling not yet implemented: ${interval.type === 'cron' ? interval.expression : interval.description}`)
449
- continue
449
+ // Cron/natural need special handling - throw error to avoid silent failures
450
+ throw new Error(
451
+ `Cron scheduling not yet implemented: "${interval.type === 'cron' ? interval.expression : interval.description}". ` +
452
+ `Use interval-based patterns like $.every.seconds(30), $.every.minutes(5), or $.every.hours(1) instead.`
453
+ )
450
454
  }
451
455
 
452
456
  if (ms > 0) {
@@ -531,12 +535,17 @@ export function createTestContext(): WorkflowContext & { emittedEvents: Array<{
531
535
  }
532
536
  }),
533
537
 
534
- every: new Proxy(function() {} as any, {
535
- get() {
536
- return () => () => {} // No-op for testing
537
- },
538
- apply() {}
539
- }),
538
+ // Cast to EveryProxy is safe: Proxy handler implements all EveryProxy behaviors dynamically
539
+ every: new Proxy(
540
+ // eslint-disable-next-line @typescript-eslint/no-unused-vars
541
+ ((_description: string, _handler: ScheduleHandler) => {}) as EveryProxyTarget,
542
+ {
543
+ get() {
544
+ return () => () => {} // No-op for testing
545
+ },
546
+ apply() {}
547
+ }
548
+ ) as unknown as EveryProxy,
540
549
 
541
550
  state: stateContext,
542
551