wu-framework 1.1.15 → 1.1.17

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 (88) hide show
  1. package/README.md +52 -20
  2. package/dist/wu-framework.cjs.js +1 -1
  3. package/dist/wu-framework.cjs.js.map +1 -1
  4. package/dist/wu-framework.dev.js +15511 -15146
  5. package/dist/wu-framework.dev.js.map +1 -1
  6. package/dist/wu-framework.esm.js +1 -1
  7. package/dist/wu-framework.esm.js.map +1 -1
  8. package/dist/wu-framework.umd.js +1 -1
  9. package/dist/wu-framework.umd.js.map +1 -1
  10. package/package.json +166 -161
  11. package/src/adapters/angular/ai.js +30 -30
  12. package/src/adapters/angular/index.d.ts +154 -154
  13. package/src/adapters/angular/index.js +932 -932
  14. package/src/adapters/angular.d.ts +3 -3
  15. package/src/adapters/angular.js +3 -3
  16. package/src/adapters/index.js +168 -168
  17. package/src/adapters/lit/ai.js +20 -20
  18. package/src/adapters/lit/index.d.ts +120 -120
  19. package/src/adapters/lit/index.js +721 -721
  20. package/src/adapters/lit.d.ts +3 -3
  21. package/src/adapters/lit.js +3 -3
  22. package/src/adapters/preact/ai.js +33 -33
  23. package/src/adapters/preact/index.d.ts +108 -108
  24. package/src/adapters/preact/index.js +661 -661
  25. package/src/adapters/preact.d.ts +3 -3
  26. package/src/adapters/preact.js +3 -3
  27. package/src/adapters/react/index.js +48 -54
  28. package/src/adapters/react.d.ts +3 -3
  29. package/src/adapters/react.js +3 -3
  30. package/src/adapters/shared.js +64 -64
  31. package/src/adapters/solid/ai.js +32 -32
  32. package/src/adapters/solid/index.d.ts +101 -101
  33. package/src/adapters/solid/index.js +586 -586
  34. package/src/adapters/solid.d.ts +3 -3
  35. package/src/adapters/solid.js +3 -3
  36. package/src/adapters/svelte/ai.js +31 -31
  37. package/src/adapters/svelte/index.d.ts +166 -166
  38. package/src/adapters/svelte/index.js +798 -798
  39. package/src/adapters/svelte.d.ts +3 -3
  40. package/src/adapters/svelte.js +3 -3
  41. package/src/adapters/vanilla/ai.js +30 -30
  42. package/src/adapters/vanilla/index.d.ts +179 -179
  43. package/src/adapters/vanilla/index.js +785 -785
  44. package/src/adapters/vanilla.d.ts +3 -3
  45. package/src/adapters/vanilla.js +3 -3
  46. package/src/adapters/vue/ai.js +52 -52
  47. package/src/adapters/vue/index.d.ts +299 -299
  48. package/src/adapters/vue/index.js +610 -610
  49. package/src/adapters/vue.d.ts +3 -3
  50. package/src/adapters/vue.js +3 -3
  51. package/src/ai/wu-ai-actions.js +261 -261
  52. package/src/ai/wu-ai-agent.js +546 -546
  53. package/src/ai/wu-ai-browser-primitives.js +354 -354
  54. package/src/ai/wu-ai-browser.js +380 -380
  55. package/src/ai/wu-ai-context.js +332 -332
  56. package/src/ai/wu-ai-conversation.js +613 -613
  57. package/src/ai/wu-ai-orchestrate.js +1021 -1021
  58. package/src/ai/wu-ai-permissions.js +381 -381
  59. package/src/ai/wu-ai-provider.js +700 -700
  60. package/src/ai/wu-ai-schema.js +225 -225
  61. package/src/ai/wu-ai-triggers.js +396 -396
  62. package/src/ai/wu-ai.js +804 -804
  63. package/src/core/wu-app.js +236 -236
  64. package/src/core/wu-cache.js +498 -477
  65. package/src/core/wu-core.js +1412 -1398
  66. package/src/core/wu-error-boundary.js +396 -382
  67. package/src/core/wu-event-bus.js +390 -348
  68. package/src/core/wu-hooks.js +350 -350
  69. package/src/core/wu-html-parser.js +199 -190
  70. package/src/core/wu-iframe-sandbox.js +328 -328
  71. package/src/core/wu-loader.js +385 -273
  72. package/src/core/wu-logger.js +142 -134
  73. package/src/core/wu-manifest.js +532 -509
  74. package/src/core/wu-mcp-bridge.js +432 -432
  75. package/src/core/wu-overrides.js +510 -510
  76. package/src/core/wu-performance.js +228 -228
  77. package/src/core/wu-plugin.js +401 -348
  78. package/src/core/wu-prefetch.js +414 -414
  79. package/src/core/wu-proxy-sandbox.js +477 -476
  80. package/src/core/wu-sandbox.js +779 -779
  81. package/src/core/wu-script-executor.js +161 -113
  82. package/src/core/wu-snapshot-sandbox.js +227 -227
  83. package/src/core/wu-store.js +13 -3
  84. package/src/core/wu-strategies.js +256 -256
  85. package/src/core/wu-style-bridge.js +477 -477
  86. package/src/index.d.ts +317 -0
  87. package/src/index.js +234 -224
  88. 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
+ }