wu-framework 1.1.15 → 1.1.16

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 (90) hide show
  1. package/LICENSE +39 -39
  2. package/README.md +408 -408
  3. package/dist/wu-framework.cjs.js.map +1 -1
  4. package/dist/wu-framework.dev.js +15151 -15151
  5. package/dist/wu-framework.dev.js.map +1 -1
  6. package/dist/wu-framework.esm.js.map +1 -1
  7. package/dist/wu-framework.umd.js.map +1 -1
  8. package/integrations/astro/README.md +127 -127
  9. package/integrations/astro/WuApp.astro +63 -63
  10. package/integrations/astro/WuShell.astro +39 -39
  11. package/integrations/astro/index.js +68 -68
  12. package/integrations/astro/package.json +38 -38
  13. package/integrations/astro/types.d.ts +53 -53
  14. package/package.json +161 -161
  15. package/src/adapters/angular/ai.js +30 -30
  16. package/src/adapters/angular/index.d.ts +154 -154
  17. package/src/adapters/angular/index.js +932 -932
  18. package/src/adapters/angular.d.ts +3 -3
  19. package/src/adapters/angular.js +3 -3
  20. package/src/adapters/index.js +168 -168
  21. package/src/adapters/lit/ai.js +20 -20
  22. package/src/adapters/lit/index.d.ts +120 -120
  23. package/src/adapters/lit/index.js +721 -721
  24. package/src/adapters/lit.d.ts +3 -3
  25. package/src/adapters/lit.js +3 -3
  26. package/src/adapters/preact/ai.js +33 -33
  27. package/src/adapters/preact/index.d.ts +108 -108
  28. package/src/adapters/preact/index.js +661 -661
  29. package/src/adapters/preact.d.ts +3 -3
  30. package/src/adapters/preact.js +3 -3
  31. package/src/adapters/react/index.js +48 -54
  32. package/src/adapters/react.d.ts +3 -3
  33. package/src/adapters/react.js +3 -3
  34. package/src/adapters/shared.js +64 -64
  35. package/src/adapters/solid/ai.js +32 -32
  36. package/src/adapters/solid/index.d.ts +101 -101
  37. package/src/adapters/solid/index.js +586 -586
  38. package/src/adapters/solid.d.ts +3 -3
  39. package/src/adapters/solid.js +3 -3
  40. package/src/adapters/svelte/ai.js +31 -31
  41. package/src/adapters/svelte/index.d.ts +166 -166
  42. package/src/adapters/svelte/index.js +798 -798
  43. package/src/adapters/svelte.d.ts +3 -3
  44. package/src/adapters/svelte.js +3 -3
  45. package/src/adapters/vanilla/ai.js +30 -30
  46. package/src/adapters/vanilla/index.d.ts +179 -179
  47. package/src/adapters/vanilla/index.js +785 -785
  48. package/src/adapters/vanilla.d.ts +3 -3
  49. package/src/adapters/vanilla.js +3 -3
  50. package/src/adapters/vue/ai.js +52 -52
  51. package/src/adapters/vue/index.d.ts +299 -299
  52. package/src/adapters/vue/index.js +610 -610
  53. package/src/adapters/vue.d.ts +3 -3
  54. package/src/adapters/vue.js +3 -3
  55. package/src/ai/wu-ai-actions.js +261 -261
  56. package/src/ai/wu-ai-agent.js +546 -546
  57. package/src/ai/wu-ai-browser-primitives.js +354 -354
  58. package/src/ai/wu-ai-browser.js +380 -380
  59. package/src/ai/wu-ai-context.js +332 -332
  60. package/src/ai/wu-ai-conversation.js +613 -613
  61. package/src/ai/wu-ai-orchestrate.js +1021 -1021
  62. package/src/ai/wu-ai-permissions.js +381 -381
  63. package/src/ai/wu-ai-provider.js +700 -700
  64. package/src/ai/wu-ai-schema.js +225 -225
  65. package/src/ai/wu-ai-triggers.js +396 -396
  66. package/src/ai/wu-ai.js +804 -804
  67. package/src/core/wu-app.js +236 -236
  68. package/src/core/wu-cache.js +477 -477
  69. package/src/core/wu-core.js +1398 -1398
  70. package/src/core/wu-error-boundary.js +382 -382
  71. package/src/core/wu-event-bus.js +348 -348
  72. package/src/core/wu-hooks.js +350 -350
  73. package/src/core/wu-html-parser.js +190 -190
  74. package/src/core/wu-iframe-sandbox.js +328 -328
  75. package/src/core/wu-loader.js +272 -272
  76. package/src/core/wu-logger.js +134 -134
  77. package/src/core/wu-manifest.js +509 -509
  78. package/src/core/wu-mcp-bridge.js +432 -432
  79. package/src/core/wu-overrides.js +510 -510
  80. package/src/core/wu-performance.js +228 -228
  81. package/src/core/wu-plugin.js +348 -348
  82. package/src/core/wu-prefetch.js +414 -414
  83. package/src/core/wu-proxy-sandbox.js +476 -476
  84. package/src/core/wu-sandbox.js +779 -779
  85. package/src/core/wu-script-executor.js +113 -113
  86. package/src/core/wu-snapshot-sandbox.js +227 -227
  87. package/src/core/wu-strategies.js +256 -256
  88. package/src/core/wu-style-bridge.js +477 -477
  89. package/src/index.js +224 -224
  90. package/src/utils/dependency-resolver.js +327 -327
@@ -1,396 +1,396 @@
1
- /**
2
- * WU-AI-TRIGGERS: Event-to-AI bridge
3
- *
4
- * Listens to wu.eventBus events and automatically triggers LLM interactions.
5
- * This is the "reactive AI" — the app becomes intelligent by responding
6
- * to events with AI-generated actions.
7
- *
8
- * Features:
9
- * - Pattern-based event matching (wu convention: 'cart:*', 'user:login')
10
- * - Debounce per trigger (avoid flooding LLM)
11
- * - Conditional execution (only trigger if condition returns true)
12
- * - Priority batching (high triggers fire immediately, low ones batch)
13
- * - Template interpolation ({{event.data.user}} in prompts)
14
- * - Namespace isolation (each trigger uses its own conversation namespace)
15
- */
16
-
17
- import { logger } from '../core/wu-logger.js';
18
- import { interpolate, sanitizeForPrompt } from './wu-ai-schema.js';
19
-
20
- // ─── Trigger Config ──────────────────────────────────────────────
21
-
22
- const DEFAULT_TRIGGER_CONFIG = {
23
- enabled: true,
24
- maxActiveTriggers: 20,
25
- defaultDebounceMs: 1000,
26
- batchIntervalMs: 2000,
27
- };
28
-
29
- // ─── Single Trigger ──────────────────────────────────────────────
30
-
31
- class Trigger {
32
- constructor(name, config) {
33
- this.name = name;
34
- this.pattern = config.pattern; // event pattern: 'cart:updated', 'user:*'
35
- this.prompt = config.prompt; // string or function(eventData) → string
36
- this.condition = config.condition || null; // function(eventData) → boolean
37
- this.debounceMs = config.debounce ?? DEFAULT_TRIGGER_CONFIG.defaultDebounceMs;
38
- this.priority = config.priority || 'medium'; // 'high' | 'medium' | 'low'
39
- this.namespace = config.namespace || `trigger:${name}`;
40
- this.systemPrompt = config.systemPrompt || null;
41
- this.onResult = config.onResult || null; // callback(result, eventData) → void
42
- this.enabled = config.enabled !== false;
43
- this.maxTokens = config.maxTokens || undefined;
44
- this.temperature = config.temperature || undefined;
45
-
46
- // Internal state
47
- this._debounceTimer = null;
48
- this._lastFired = 0;
49
- this._fireCount = 0;
50
- this._pendingEvent = null;
51
- }
52
-
53
- /**
54
- * Check if this trigger matches an event name.
55
- */
56
- matches(eventName) {
57
- if (!this.pattern) return false;
58
- if (this.pattern === '*') return true;
59
- if (!this.pattern.includes('*')) return eventName === this.pattern;
60
-
61
- const regex = new RegExp('^' + this.pattern.replace(/\*/g, '[^:]*') + '$');
62
- return regex.test(eventName);
63
- }
64
-
65
- /**
66
- * Build the prompt for this trigger given event data.
67
- */
68
- buildPrompt(eventData) {
69
- if (typeof this.prompt === 'function') {
70
- return this.prompt(eventData);
71
- }
72
-
73
- // Template interpolation
74
- return interpolate(this.prompt, {
75
- event: eventData,
76
- data: eventData?.data,
77
- timestamp: Date.now(),
78
- });
79
- }
80
-
81
- /**
82
- * Check condition (if any).
83
- */
84
- async checkCondition(eventData) {
85
- if (!this.condition) return true;
86
- try {
87
- const result = this.condition(eventData);
88
- return result instanceof Promise ? await result : result;
89
- } catch (err) {
90
- logger.wuDebug(`[wu-ai] Trigger '${this.name}' condition error: ${err.message}`);
91
- return false;
92
- }
93
- }
94
- }
95
-
96
- // ─── Main Triggers Manager ───────────────────────────────────────
97
-
98
- export class WuAITriggers {
99
- constructor({ eventBus, conversation, permissions }) {
100
- this._eventBus = eventBus;
101
- this._conversation = conversation;
102
- this._permissions = permissions;
103
-
104
- this._config = { ...DEFAULT_TRIGGER_CONFIG };
105
- this._triggers = new Map(); // name → Trigger
106
- this._listeners = new Map(); // name → unsubscribe function
107
- this._batchQueue = []; // low-priority triggers pending batch
108
- this._batchTimer = null;
109
- this._stats = {
110
- totalFired: 0,
111
- totalSkipped: 0,
112
- totalErrors: 0,
113
- };
114
- }
115
-
116
- /**
117
- * Configure trigger system.
118
- */
119
- configure(config) {
120
- Object.assign(this._config, config);
121
- }
122
-
123
- /**
124
- * Register a trigger.
125
- *
126
- * @param {string} name - Trigger name (unique identifier)
127
- * @param {object} config
128
- * @param {string} config.pattern - Event pattern to match (e.g., 'cart:*')
129
- * @param {string|Function} config.prompt - Prompt template or function
130
- * @param {Function} [config.condition] - Optional condition function
131
- * @param {number} [config.debounce=1000] - Debounce in ms
132
- * @param {string} [config.priority='medium'] - 'high' | 'medium' | 'low'
133
- * @param {string} [config.namespace] - Conversation namespace
134
- * @param {string} [config.systemPrompt] - Override system prompt
135
- * @param {Function} [config.onResult] - Callback for results
136
- * @param {boolean} [config.enabled=true] - Whether trigger is active
137
- */
138
- register(name, config) {
139
- if (this._triggers.size >= this._config.maxActiveTriggers) {
140
- logger.wuWarn(`[wu-ai] Max triggers (${this._config.maxActiveTriggers}) reached. Cannot register '${name}'.`);
141
- return;
142
- }
143
-
144
- // Unregister existing trigger with same name
145
- if (this._triggers.has(name)) {
146
- this.unregister(name);
147
- }
148
-
149
- const trigger = new Trigger(name, config);
150
- this._triggers.set(name, trigger);
151
-
152
- // Subscribe to matching events
153
- const handler = (eventData) => this._handleEvent(name, eventData);
154
- const unsub = this._eventBus.on(trigger.pattern, handler);
155
- this._listeners.set(name, unsub);
156
-
157
- logger.wuDebug(`[wu-ai] Trigger registered: '${name}' → pattern '${trigger.pattern}' (${trigger.priority})`);
158
- }
159
-
160
- /**
161
- * Unregister a trigger.
162
- */
163
- unregister(name) {
164
- const trigger = this._triggers.get(name);
165
- if (trigger) {
166
- // Clear any pending debounce
167
- if (trigger._debounceTimer) {
168
- clearTimeout(trigger._debounceTimer);
169
- }
170
- }
171
-
172
- // Remove event listener
173
- const unsub = this._listeners.get(name);
174
- if (typeof unsub === 'function') {
175
- unsub();
176
- }
177
-
178
- this._triggers.delete(name);
179
- this._listeners.delete(name);
180
- }
181
-
182
- /**
183
- * Enable/disable a trigger.
184
- */
185
- setEnabled(name, enabled) {
186
- const trigger = this._triggers.get(name);
187
- if (trigger) trigger.enabled = enabled;
188
- }
189
-
190
- /**
191
- * Enable/disable all triggers.
192
- */
193
- setAllEnabled(enabled) {
194
- this._config.enabled = enabled;
195
- for (const trigger of this._triggers.values()) {
196
- trigger.enabled = enabled;
197
- }
198
- }
199
-
200
- /**
201
- * Fire a trigger manually (bypasses event matching).
202
- */
203
- async fire(name, eventData = {}) {
204
- const trigger = this._triggers.get(name);
205
- if (!trigger) {
206
- logger.wuWarn(`[wu-ai] Trigger '${name}' not found`);
207
- return null;
208
- }
209
- return this._executeTrigger(trigger, eventData);
210
- }
211
-
212
- /**
213
- * Get registered trigger names.
214
- */
215
- getNames() {
216
- return [...this._triggers.keys()];
217
- }
218
-
219
- /**
220
- * Get trigger info.
221
- */
222
- getTrigger(name) {
223
- const t = this._triggers.get(name);
224
- if (!t) return null;
225
- return {
226
- name: t.name,
227
- pattern: t.pattern,
228
- priority: t.priority,
229
- namespace: t.namespace,
230
- enabled: t.enabled,
231
- fireCount: t._fireCount,
232
- lastFired: t._lastFired,
233
- };
234
- }
235
-
236
- /**
237
- * Destroy all triggers and clean up.
238
- */
239
- destroy() {
240
- for (const name of [...this._triggers.keys()]) {
241
- this.unregister(name);
242
- }
243
- if (this._batchTimer) {
244
- clearTimeout(this._batchTimer);
245
- this._batchTimer = null;
246
- }
247
- this._batchQueue = [];
248
- }
249
-
250
- // ── Private: Event Handling ──
251
-
252
- async _handleEvent(triggerName, eventData) {
253
- if (!this._config.enabled) return;
254
-
255
- const trigger = this._triggers.get(triggerName);
256
- if (!trigger || !trigger.enabled) return;
257
-
258
- // Check condition
259
- const conditionMet = await trigger.checkCondition(eventData);
260
- if (!conditionMet) {
261
- this._stats.totalSkipped++;
262
- return;
263
- }
264
-
265
- // Priority routing
266
- if (trigger.priority === 'high') {
267
- // High priority: fire immediately (with debounce)
268
- this._debouncedFire(trigger, eventData);
269
- } else if (trigger.priority === 'low') {
270
- // Low priority: batch
271
- this._batchQueue.push({ trigger, eventData });
272
- this._scheduleBatch();
273
- } else {
274
- // Medium: debounce
275
- this._debouncedFire(trigger, eventData);
276
- }
277
- }
278
-
279
- _debouncedFire(trigger, eventData) {
280
- // Store pending event (latest wins for debounce)
281
- trigger._pendingEvent = eventData;
282
-
283
- if (trigger._debounceTimer) {
284
- clearTimeout(trigger._debounceTimer);
285
- }
286
-
287
- if (trigger.debounceMs <= 0) {
288
- // No debounce
289
- this._executeTrigger(trigger, eventData);
290
- return;
291
- }
292
-
293
- trigger._debounceTimer = setTimeout(() => {
294
- trigger._debounceTimer = null;
295
- const pending = trigger._pendingEvent;
296
- trigger._pendingEvent = null;
297
- if (pending) {
298
- this._executeTrigger(trigger, pending);
299
- }
300
- }, trigger.debounceMs);
301
- }
302
-
303
- _scheduleBatch() {
304
- if (this._batchTimer) return;
305
-
306
- this._batchTimer = setTimeout(() => {
307
- this._batchTimer = null;
308
- this._processBatch();
309
- }, this._config.batchIntervalMs);
310
- }
311
-
312
- async _processBatch() {
313
- const batch = [...this._batchQueue];
314
- this._batchQueue = [];
315
-
316
- // Deduplicate: keep last event per trigger
317
- const byTrigger = new Map();
318
- for (const { trigger, eventData } of batch) {
319
- byTrigger.set(trigger.name, { trigger, eventData });
320
- }
321
-
322
- for (const { trigger, eventData } of byTrigger.values()) {
323
- await this._executeTrigger(trigger, eventData);
324
- }
325
- }
326
-
327
- async _executeTrigger(trigger, eventData) {
328
- try {
329
- const prompt = trigger.buildPrompt(eventData);
330
- if (!prompt) {
331
- this._stats.totalSkipped++;
332
- return null;
333
- }
334
-
335
- logger.wuDebug(`[wu-ai] Trigger '${trigger.name}' firing with prompt: ${prompt.slice(0, 100)}...`);
336
-
337
- const result = await this._conversation.send(prompt, {
338
- namespace: trigger.namespace,
339
- systemPrompt: trigger.systemPrompt,
340
- temperature: trigger.temperature,
341
- maxTokens: trigger.maxTokens,
342
- });
343
-
344
- trigger._fireCount++;
345
- trigger._lastFired = Date.now();
346
- this._stats.totalFired++;
347
-
348
- // Emit trigger result event
349
- this._eventBus.emit('ai:trigger:result', {
350
- trigger: trigger.name,
351
- pattern: trigger.pattern,
352
- result,
353
- }, { appName: 'wu-ai' });
354
-
355
- // Call onResult callback if provided
356
- if (trigger.onResult) {
357
- try {
358
- await trigger.onResult(result, eventData);
359
- } catch (err) {
360
- logger.wuDebug(`[wu-ai] Trigger '${trigger.name}' onResult error: ${err.message}`);
361
- }
362
- }
363
-
364
- return result;
365
- } catch (err) {
366
- this._stats.totalErrors++;
367
- logger.wuWarn(`[wu-ai] Trigger '${trigger.name}' error: ${err.message}`);
368
-
369
- this._eventBus.emit('ai:trigger:error', {
370
- trigger: trigger.name,
371
- error: err.message,
372
- }, { appName: 'wu-ai' });
373
-
374
- return null;
375
- }
376
- }
377
-
378
- getStats() {
379
- const triggers = {};
380
- for (const [name, t] of this._triggers) {
381
- triggers[name] = {
382
- pattern: t.pattern,
383
- priority: t.priority,
384
- enabled: t.enabled,
385
- fireCount: t._fireCount,
386
- lastFired: t._lastFired,
387
- };
388
- }
389
- return {
390
- ...this._stats,
391
- triggerCount: this._triggers.size,
392
- batchQueueSize: this._batchQueue.length,
393
- triggers,
394
- };
395
- }
396
- }
1
+ /**
2
+ * WU-AI-TRIGGERS: Event-to-AI bridge
3
+ *
4
+ * Listens to wu.eventBus events and automatically triggers LLM interactions.
5
+ * This is the "reactive AI" — the app becomes intelligent by responding
6
+ * to events with AI-generated actions.
7
+ *
8
+ * Features:
9
+ * - Pattern-based event matching (wu convention: 'cart:*', 'user:login')
10
+ * - Debounce per trigger (avoid flooding LLM)
11
+ * - Conditional execution (only trigger if condition returns true)
12
+ * - Priority batching (high triggers fire immediately, low ones batch)
13
+ * - Template interpolation ({{event.data.user}} in prompts)
14
+ * - Namespace isolation (each trigger uses its own conversation namespace)
15
+ */
16
+
17
+ import { logger } from '../core/wu-logger.js';
18
+ import { interpolate, sanitizeForPrompt } from './wu-ai-schema.js';
19
+
20
+ // ─── Trigger Config ──────────────────────────────────────────────
21
+
22
+ const DEFAULT_TRIGGER_CONFIG = {
23
+ enabled: true,
24
+ maxActiveTriggers: 20,
25
+ defaultDebounceMs: 1000,
26
+ batchIntervalMs: 2000,
27
+ };
28
+
29
+ // ─── Single Trigger ──────────────────────────────────────────────
30
+
31
+ class Trigger {
32
+ constructor(name, config) {
33
+ this.name = name;
34
+ this.pattern = config.pattern; // event pattern: 'cart:updated', 'user:*'
35
+ this.prompt = config.prompt; // string or function(eventData) → string
36
+ this.condition = config.condition || null; // function(eventData) → boolean
37
+ this.debounceMs = config.debounce ?? DEFAULT_TRIGGER_CONFIG.defaultDebounceMs;
38
+ this.priority = config.priority || 'medium'; // 'high' | 'medium' | 'low'
39
+ this.namespace = config.namespace || `trigger:${name}`;
40
+ this.systemPrompt = config.systemPrompt || null;
41
+ this.onResult = config.onResult || null; // callback(result, eventData) → void
42
+ this.enabled = config.enabled !== false;
43
+ this.maxTokens = config.maxTokens || undefined;
44
+ this.temperature = config.temperature || undefined;
45
+
46
+ // Internal state
47
+ this._debounceTimer = null;
48
+ this._lastFired = 0;
49
+ this._fireCount = 0;
50
+ this._pendingEvent = null;
51
+ }
52
+
53
+ /**
54
+ * Check if this trigger matches an event name.
55
+ */
56
+ matches(eventName) {
57
+ if (!this.pattern) return false;
58
+ if (this.pattern === '*') return true;
59
+ if (!this.pattern.includes('*')) return eventName === this.pattern;
60
+
61
+ const regex = new RegExp('^' + this.pattern.replace(/\*/g, '[^:]*') + '$');
62
+ return regex.test(eventName);
63
+ }
64
+
65
+ /**
66
+ * Build the prompt for this trigger given event data.
67
+ */
68
+ buildPrompt(eventData) {
69
+ if (typeof this.prompt === 'function') {
70
+ return this.prompt(eventData);
71
+ }
72
+
73
+ // Template interpolation
74
+ return interpolate(this.prompt, {
75
+ event: eventData,
76
+ data: eventData?.data,
77
+ timestamp: Date.now(),
78
+ });
79
+ }
80
+
81
+ /**
82
+ * Check condition (if any).
83
+ */
84
+ async checkCondition(eventData) {
85
+ if (!this.condition) return true;
86
+ try {
87
+ const result = this.condition(eventData);
88
+ return result instanceof Promise ? await result : result;
89
+ } catch (err) {
90
+ logger.wuDebug(`[wu-ai] Trigger '${this.name}' condition error: ${err.message}`);
91
+ return false;
92
+ }
93
+ }
94
+ }
95
+
96
+ // ─── Main Triggers Manager ───────────────────────────────────────
97
+
98
+ export class WuAITriggers {
99
+ constructor({ eventBus, conversation, permissions }) {
100
+ this._eventBus = eventBus;
101
+ this._conversation = conversation;
102
+ this._permissions = permissions;
103
+
104
+ this._config = { ...DEFAULT_TRIGGER_CONFIG };
105
+ this._triggers = new Map(); // name → Trigger
106
+ this._listeners = new Map(); // name → unsubscribe function
107
+ this._batchQueue = []; // low-priority triggers pending batch
108
+ this._batchTimer = null;
109
+ this._stats = {
110
+ totalFired: 0,
111
+ totalSkipped: 0,
112
+ totalErrors: 0,
113
+ };
114
+ }
115
+
116
+ /**
117
+ * Configure trigger system.
118
+ */
119
+ configure(config) {
120
+ Object.assign(this._config, config);
121
+ }
122
+
123
+ /**
124
+ * Register a trigger.
125
+ *
126
+ * @param {string} name - Trigger name (unique identifier)
127
+ * @param {object} config
128
+ * @param {string} config.pattern - Event pattern to match (e.g., 'cart:*')
129
+ * @param {string|Function} config.prompt - Prompt template or function
130
+ * @param {Function} [config.condition] - Optional condition function
131
+ * @param {number} [config.debounce=1000] - Debounce in ms
132
+ * @param {string} [config.priority='medium'] - 'high' | 'medium' | 'low'
133
+ * @param {string} [config.namespace] - Conversation namespace
134
+ * @param {string} [config.systemPrompt] - Override system prompt
135
+ * @param {Function} [config.onResult] - Callback for results
136
+ * @param {boolean} [config.enabled=true] - Whether trigger is active
137
+ */
138
+ register(name, config) {
139
+ if (this._triggers.size >= this._config.maxActiveTriggers) {
140
+ logger.wuWarn(`[wu-ai] Max triggers (${this._config.maxActiveTriggers}) reached. Cannot register '${name}'.`);
141
+ return;
142
+ }
143
+
144
+ // Unregister existing trigger with same name
145
+ if (this._triggers.has(name)) {
146
+ this.unregister(name);
147
+ }
148
+
149
+ const trigger = new Trigger(name, config);
150
+ this._triggers.set(name, trigger);
151
+
152
+ // Subscribe to matching events
153
+ const handler = (eventData) => this._handleEvent(name, eventData);
154
+ const unsub = this._eventBus.on(trigger.pattern, handler);
155
+ this._listeners.set(name, unsub);
156
+
157
+ logger.wuDebug(`[wu-ai] Trigger registered: '${name}' → pattern '${trigger.pattern}' (${trigger.priority})`);
158
+ }
159
+
160
+ /**
161
+ * Unregister a trigger.
162
+ */
163
+ unregister(name) {
164
+ const trigger = this._triggers.get(name);
165
+ if (trigger) {
166
+ // Clear any pending debounce
167
+ if (trigger._debounceTimer) {
168
+ clearTimeout(trigger._debounceTimer);
169
+ }
170
+ }
171
+
172
+ // Remove event listener
173
+ const unsub = this._listeners.get(name);
174
+ if (typeof unsub === 'function') {
175
+ unsub();
176
+ }
177
+
178
+ this._triggers.delete(name);
179
+ this._listeners.delete(name);
180
+ }
181
+
182
+ /**
183
+ * Enable/disable a trigger.
184
+ */
185
+ setEnabled(name, enabled) {
186
+ const trigger = this._triggers.get(name);
187
+ if (trigger) trigger.enabled = enabled;
188
+ }
189
+
190
+ /**
191
+ * Enable/disable all triggers.
192
+ */
193
+ setAllEnabled(enabled) {
194
+ this._config.enabled = enabled;
195
+ for (const trigger of this._triggers.values()) {
196
+ trigger.enabled = enabled;
197
+ }
198
+ }
199
+
200
+ /**
201
+ * Fire a trigger manually (bypasses event matching).
202
+ */
203
+ async fire(name, eventData = {}) {
204
+ const trigger = this._triggers.get(name);
205
+ if (!trigger) {
206
+ logger.wuWarn(`[wu-ai] Trigger '${name}' not found`);
207
+ return null;
208
+ }
209
+ return this._executeTrigger(trigger, eventData);
210
+ }
211
+
212
+ /**
213
+ * Get registered trigger names.
214
+ */
215
+ getNames() {
216
+ return [...this._triggers.keys()];
217
+ }
218
+
219
+ /**
220
+ * Get trigger info.
221
+ */
222
+ getTrigger(name) {
223
+ const t = this._triggers.get(name);
224
+ if (!t) return null;
225
+ return {
226
+ name: t.name,
227
+ pattern: t.pattern,
228
+ priority: t.priority,
229
+ namespace: t.namespace,
230
+ enabled: t.enabled,
231
+ fireCount: t._fireCount,
232
+ lastFired: t._lastFired,
233
+ };
234
+ }
235
+
236
+ /**
237
+ * Destroy all triggers and clean up.
238
+ */
239
+ destroy() {
240
+ for (const name of [...this._triggers.keys()]) {
241
+ this.unregister(name);
242
+ }
243
+ if (this._batchTimer) {
244
+ clearTimeout(this._batchTimer);
245
+ this._batchTimer = null;
246
+ }
247
+ this._batchQueue = [];
248
+ }
249
+
250
+ // ── Private: Event Handling ──
251
+
252
+ async _handleEvent(triggerName, eventData) {
253
+ if (!this._config.enabled) return;
254
+
255
+ const trigger = this._triggers.get(triggerName);
256
+ if (!trigger || !trigger.enabled) return;
257
+
258
+ // Check condition
259
+ const conditionMet = await trigger.checkCondition(eventData);
260
+ if (!conditionMet) {
261
+ this._stats.totalSkipped++;
262
+ return;
263
+ }
264
+
265
+ // Priority routing
266
+ if (trigger.priority === 'high') {
267
+ // High priority: fire immediately (with debounce)
268
+ this._debouncedFire(trigger, eventData);
269
+ } else if (trigger.priority === 'low') {
270
+ // Low priority: batch
271
+ this._batchQueue.push({ trigger, eventData });
272
+ this._scheduleBatch();
273
+ } else {
274
+ // Medium: debounce
275
+ this._debouncedFire(trigger, eventData);
276
+ }
277
+ }
278
+
279
+ _debouncedFire(trigger, eventData) {
280
+ // Store pending event (latest wins for debounce)
281
+ trigger._pendingEvent = eventData;
282
+
283
+ if (trigger._debounceTimer) {
284
+ clearTimeout(trigger._debounceTimer);
285
+ }
286
+
287
+ if (trigger.debounceMs <= 0) {
288
+ // No debounce
289
+ this._executeTrigger(trigger, eventData);
290
+ return;
291
+ }
292
+
293
+ trigger._debounceTimer = setTimeout(() => {
294
+ trigger._debounceTimer = null;
295
+ const pending = trigger._pendingEvent;
296
+ trigger._pendingEvent = null;
297
+ if (pending) {
298
+ this._executeTrigger(trigger, pending);
299
+ }
300
+ }, trigger.debounceMs);
301
+ }
302
+
303
+ _scheduleBatch() {
304
+ if (this._batchTimer) return;
305
+
306
+ this._batchTimer = setTimeout(() => {
307
+ this._batchTimer = null;
308
+ this._processBatch();
309
+ }, this._config.batchIntervalMs);
310
+ }
311
+
312
+ async _processBatch() {
313
+ const batch = [...this._batchQueue];
314
+ this._batchQueue = [];
315
+
316
+ // Deduplicate: keep last event per trigger
317
+ const byTrigger = new Map();
318
+ for (const { trigger, eventData } of batch) {
319
+ byTrigger.set(trigger.name, { trigger, eventData });
320
+ }
321
+
322
+ for (const { trigger, eventData } of byTrigger.values()) {
323
+ await this._executeTrigger(trigger, eventData);
324
+ }
325
+ }
326
+
327
+ async _executeTrigger(trigger, eventData) {
328
+ try {
329
+ const prompt = trigger.buildPrompt(eventData);
330
+ if (!prompt) {
331
+ this._stats.totalSkipped++;
332
+ return null;
333
+ }
334
+
335
+ logger.wuDebug(`[wu-ai] Trigger '${trigger.name}' firing with prompt: ${prompt.slice(0, 100)}...`);
336
+
337
+ const result = await this._conversation.send(prompt, {
338
+ namespace: trigger.namespace,
339
+ systemPrompt: trigger.systemPrompt,
340
+ temperature: trigger.temperature,
341
+ maxTokens: trigger.maxTokens,
342
+ });
343
+
344
+ trigger._fireCount++;
345
+ trigger._lastFired = Date.now();
346
+ this._stats.totalFired++;
347
+
348
+ // Emit trigger result event
349
+ this._eventBus.emit('ai:trigger:result', {
350
+ trigger: trigger.name,
351
+ pattern: trigger.pattern,
352
+ result,
353
+ }, { appName: 'wu-ai' });
354
+
355
+ // Call onResult callback if provided
356
+ if (trigger.onResult) {
357
+ try {
358
+ await trigger.onResult(result, eventData);
359
+ } catch (err) {
360
+ logger.wuDebug(`[wu-ai] Trigger '${trigger.name}' onResult error: ${err.message}`);
361
+ }
362
+ }
363
+
364
+ return result;
365
+ } catch (err) {
366
+ this._stats.totalErrors++;
367
+ logger.wuWarn(`[wu-ai] Trigger '${trigger.name}' error: ${err.message}`);
368
+
369
+ this._eventBus.emit('ai:trigger:error', {
370
+ trigger: trigger.name,
371
+ error: err.message,
372
+ }, { appName: 'wu-ai' });
373
+
374
+ return null;
375
+ }
376
+ }
377
+
378
+ getStats() {
379
+ const triggers = {};
380
+ for (const [name, t] of this._triggers) {
381
+ triggers[name] = {
382
+ pattern: t.pattern,
383
+ priority: t.priority,
384
+ enabled: t.enabled,
385
+ fireCount: t._fireCount,
386
+ lastFired: t._lastFired,
387
+ };
388
+ }
389
+ return {
390
+ ...this._stats,
391
+ triggerCount: this._triggers.size,
392
+ batchQueueSize: this._batchQueue.length,
393
+ triggers,
394
+ };
395
+ }
396
+ }