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.
- package/README.md +52 -20
- package/dist/wu-framework.cjs.js +1 -1
- package/dist/wu-framework.cjs.js.map +1 -1
- package/dist/wu-framework.dev.js +15511 -15146
- package/dist/wu-framework.dev.js.map +1 -1
- package/dist/wu-framework.esm.js +1 -1
- package/dist/wu-framework.esm.js.map +1 -1
- package/dist/wu-framework.umd.js +1 -1
- package/dist/wu-framework.umd.js.map +1 -1
- package/package.json +166 -161
- package/src/adapters/angular/ai.js +30 -30
- package/src/adapters/angular/index.d.ts +154 -154
- package/src/adapters/angular/index.js +932 -932
- package/src/adapters/angular.d.ts +3 -3
- package/src/adapters/angular.js +3 -3
- package/src/adapters/index.js +168 -168
- package/src/adapters/lit/ai.js +20 -20
- package/src/adapters/lit/index.d.ts +120 -120
- package/src/adapters/lit/index.js +721 -721
- package/src/adapters/lit.d.ts +3 -3
- package/src/adapters/lit.js +3 -3
- package/src/adapters/preact/ai.js +33 -33
- package/src/adapters/preact/index.d.ts +108 -108
- package/src/adapters/preact/index.js +661 -661
- package/src/adapters/preact.d.ts +3 -3
- package/src/adapters/preact.js +3 -3
- package/src/adapters/react/index.js +48 -54
- package/src/adapters/react.d.ts +3 -3
- package/src/adapters/react.js +3 -3
- package/src/adapters/shared.js +64 -64
- package/src/adapters/solid/ai.js +32 -32
- package/src/adapters/solid/index.d.ts +101 -101
- package/src/adapters/solid/index.js +586 -586
- package/src/adapters/solid.d.ts +3 -3
- package/src/adapters/solid.js +3 -3
- package/src/adapters/svelte/ai.js +31 -31
- package/src/adapters/svelte/index.d.ts +166 -166
- package/src/adapters/svelte/index.js +798 -798
- package/src/adapters/svelte.d.ts +3 -3
- package/src/adapters/svelte.js +3 -3
- package/src/adapters/vanilla/ai.js +30 -30
- package/src/adapters/vanilla/index.d.ts +179 -179
- package/src/adapters/vanilla/index.js +785 -785
- package/src/adapters/vanilla.d.ts +3 -3
- package/src/adapters/vanilla.js +3 -3
- package/src/adapters/vue/ai.js +52 -52
- package/src/adapters/vue/index.d.ts +299 -299
- package/src/adapters/vue/index.js +610 -610
- package/src/adapters/vue.d.ts +3 -3
- package/src/adapters/vue.js +3 -3
- package/src/ai/wu-ai-actions.js +261 -261
- package/src/ai/wu-ai-agent.js +546 -546
- package/src/ai/wu-ai-browser-primitives.js +354 -354
- package/src/ai/wu-ai-browser.js +380 -380
- package/src/ai/wu-ai-context.js +332 -332
- package/src/ai/wu-ai-conversation.js +613 -613
- package/src/ai/wu-ai-orchestrate.js +1021 -1021
- package/src/ai/wu-ai-permissions.js +381 -381
- package/src/ai/wu-ai-provider.js +700 -700
- package/src/ai/wu-ai-schema.js +225 -225
- package/src/ai/wu-ai-triggers.js +396 -396
- package/src/ai/wu-ai.js +804 -804
- package/src/core/wu-app.js +236 -236
- package/src/core/wu-cache.js +498 -477
- package/src/core/wu-core.js +1412 -1398
- package/src/core/wu-error-boundary.js +396 -382
- package/src/core/wu-event-bus.js +390 -348
- package/src/core/wu-hooks.js +350 -350
- package/src/core/wu-html-parser.js +199 -190
- package/src/core/wu-iframe-sandbox.js +328 -328
- package/src/core/wu-loader.js +385 -273
- package/src/core/wu-logger.js +142 -134
- package/src/core/wu-manifest.js +532 -509
- package/src/core/wu-mcp-bridge.js +432 -432
- package/src/core/wu-overrides.js +510 -510
- package/src/core/wu-performance.js +228 -228
- package/src/core/wu-plugin.js +401 -348
- package/src/core/wu-prefetch.js +414 -414
- package/src/core/wu-proxy-sandbox.js +477 -476
- package/src/core/wu-sandbox.js +779 -779
- package/src/core/wu-script-executor.js +161 -113
- package/src/core/wu-snapshot-sandbox.js +227 -227
- package/src/core/wu-store.js +13 -3
- package/src/core/wu-strategies.js +256 -256
- package/src/core/wu-style-bridge.js +477 -477
- package/src/index.d.ts +317 -0
- package/src/index.js +234 -224
- package/src/utils/dependency-resolver.js +327 -327
package/src/ai/wu-ai-triggers.js
CHANGED
|
@@ -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
|
+
}
|