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,227 +1,227 @@
1
- /**
2
- * WU-SNAPSHOT-SANDBOX: JavaScript Isolation via Snapshots
3
- * Fallback for browsers without Proxy support.
4
- *
5
- * Takes a snapshot of window state before mount,
6
- * restores original state on deactivate.
7
- * Also tracks timers and event listeners for cleanup.
8
- */
9
-
10
- import { logger } from './wu-logger.js';
11
-
12
- export class WuSnapshotSandbox {
13
- constructor(appName) {
14
- this.appName = appName;
15
- this.proxy = window;
16
- this.snapshot = new Map();
17
- this.modifiedKeys = new Set();
18
- this.active = false;
19
-
20
- // Side-effect tracking (same as ProxySandbox)
21
- this._timers = new Set();
22
- this._intervals = new Set();
23
- this._rafs = new Set();
24
- this._eventListeners = [];
25
-
26
- // Window patching state
27
- this._patched = false;
28
- this._originals = null;
29
- }
30
-
31
- /**
32
- * Activate sandbox - capture window snapshot and start tracking.
33
- */
34
- activate() {
35
- if (this.active) return this.proxy;
36
-
37
- this.snapshot.clear();
38
- this.modifiedKeys.clear();
39
-
40
- // Capture current window state
41
- for (const key in window) {
42
- try {
43
- this.snapshot.set(key, window[key]);
44
- } catch {
45
- // Some properties may be inaccessible
46
- }
47
- }
48
-
49
- this.active = true;
50
- logger.wuDebug(`[SnapshotSandbox] Activated for ${this.appName} (${this.snapshot.size} props)`);
51
- return this.proxy;
52
- }
53
-
54
- /**
55
- * Deactivate sandbox - restore snapshot AND clean side effects.
56
- */
57
- deactivate() {
58
- if (!this.active) return;
59
-
60
- // Unpatch window if patched
61
- this.unpatchWindow();
62
-
63
- // --- Clean tracked timers ---
64
- for (const id of this._timers) {
65
- try { clearTimeout(id); } catch {}
66
- }
67
- for (const id of this._intervals) {
68
- try { clearInterval(id); } catch {}
69
- }
70
- for (const id of this._rafs) {
71
- try { cancelAnimationFrame(id); } catch {}
72
- }
73
-
74
- const timerCount = this._timers.size + this._intervals.size + this._rafs.size;
75
- this._timers.clear();
76
- this._intervals.clear();
77
- this._rafs.clear();
78
-
79
- // --- Clean tracked event listeners ---
80
- const listenerCount = this._eventListeners.length;
81
- for (const { target, event, handler, options } of this._eventListeners) {
82
- try { target.removeEventListener(event, handler, options); } catch {}
83
- }
84
- this._eventListeners = [];
85
-
86
- // --- Restore window snapshot ---
87
- let restoredCount = 0;
88
- let deletedCount = 0;
89
-
90
- for (const key in window) {
91
- try {
92
- const currentValue = window[key];
93
- const originalValue = this.snapshot.get(key);
94
-
95
- if (currentValue !== originalValue) {
96
- if (this.snapshot.has(key)) {
97
- window[key] = originalValue;
98
- restoredCount++;
99
- } else {
100
- try {
101
- delete window[key];
102
- deletedCount++;
103
- } catch {}
104
- }
105
- }
106
- } catch {}
107
- }
108
-
109
- this.snapshot.clear();
110
- this.modifiedKeys.clear();
111
- this.active = false;
112
-
113
- if (timerCount > 0 || listenerCount > 0) {
114
- logger.wuDebug(
115
- `[SnapshotSandbox] ${this.appName} cleanup: ${timerCount} timers, ${listenerCount} listeners, ${restoredCount} restored, ${deletedCount} deleted`
116
- );
117
- }
118
- logger.wuDebug(`[SnapshotSandbox] Deactivated for ${this.appName}`);
119
- }
120
-
121
- // ================================================================
122
- // WINDOW PATCHING - same interface as ProxySandbox
123
- // ================================================================
124
-
125
- patchWindow() {
126
- if (this._patched) return;
127
-
128
- const self = this;
129
-
130
- // Capture in local closure — survives unpatch safely
131
- const originals = {
132
- setTimeout: window.setTimeout,
133
- clearTimeout: window.clearTimeout,
134
- setInterval: window.setInterval,
135
- clearInterval: window.clearInterval,
136
- requestAnimationFrame: window.requestAnimationFrame,
137
- cancelAnimationFrame: window.cancelAnimationFrame,
138
- addEventListener: window.addEventListener,
139
- removeEventListener: window.removeEventListener
140
- };
141
-
142
- this._originals = originals;
143
-
144
- window.setTimeout = function(fn, delay, ...args) {
145
- const id = originals.setTimeout.call(window, fn, delay, ...args);
146
- if (self._patched) self._timers.add(id);
147
- return id;
148
- };
149
- window.clearTimeout = function(id) {
150
- self._timers.delete(id);
151
- return originals.clearTimeout.call(window, id);
152
- };
153
- window.setInterval = function(fn, delay, ...args) {
154
- const id = originals.setInterval.call(window, fn, delay, ...args);
155
- if (self._patched) self._intervals.add(id);
156
- return id;
157
- };
158
- window.clearInterval = function(id) {
159
- self._intervals.delete(id);
160
- return originals.clearInterval.call(window, id);
161
- };
162
- window.requestAnimationFrame = function(fn) {
163
- const id = originals.requestAnimationFrame.call(window, fn);
164
- if (self._patched) self._rafs.add(id);
165
- return id;
166
- };
167
- window.cancelAnimationFrame = function(id) {
168
- self._rafs.delete(id);
169
- return originals.cancelAnimationFrame.call(window, id);
170
- };
171
-
172
- window.addEventListener = function(event, handler, options) {
173
- if (self._patched) self._eventListeners.push({ target: window, event, handler, options });
174
- return originals.addEventListener.call(window, event, handler, options);
175
- };
176
- window.removeEventListener = function(event, handler, options) {
177
- self._eventListeners = self._eventListeners.filter(
178
- l => !(l.target === window && l.event === event && l.handler === handler)
179
- );
180
- return originals.removeEventListener.call(window, event, handler, options);
181
- };
182
-
183
- this._patched = true;
184
- logger.wuDebug(`[SnapshotSandbox] Window patched for ${this.appName}`);
185
- }
186
-
187
- unpatchWindow() {
188
- if (!this._patched || !this._originals) return;
189
-
190
- window.setTimeout = this._originals.setTimeout;
191
- window.clearTimeout = this._originals.clearTimeout;
192
- window.setInterval = this._originals.setInterval;
193
- window.clearInterval = this._originals.clearInterval;
194
- window.requestAnimationFrame = this._originals.requestAnimationFrame;
195
- window.cancelAnimationFrame = this._originals.cancelAnimationFrame;
196
- window.addEventListener = this._originals.addEventListener;
197
- window.removeEventListener = this._originals.removeEventListener;
198
-
199
- this._patched = false;
200
- logger.wuDebug(`[SnapshotSandbox] Window unpatched for ${this.appName}`);
201
- }
202
-
203
- // ================================================================
204
- // UTILITIES
205
- // ================================================================
206
-
207
- getProxy() {
208
- return this.active ? this.proxy : null;
209
- }
210
-
211
- isActive() {
212
- return this.active;
213
- }
214
-
215
- getStats() {
216
- return {
217
- appName: this.appName,
218
- active: this.active,
219
- patched: this._patched,
220
- snapshotSize: this.snapshot.size,
221
- trackedTimers: this._timers.size,
222
- trackedIntervals: this._intervals.size,
223
- trackedRAFs: this._rafs.size,
224
- trackedEventListeners: this._eventListeners.length
225
- };
226
- }
227
- }
1
+ /**
2
+ * WU-SNAPSHOT-SANDBOX: JavaScript Isolation via Snapshots
3
+ * Fallback for browsers without Proxy support.
4
+ *
5
+ * Takes a snapshot of window state before mount,
6
+ * restores original state on deactivate.
7
+ * Also tracks timers and event listeners for cleanup.
8
+ */
9
+
10
+ import { logger } from './wu-logger.js';
11
+
12
+ export class WuSnapshotSandbox {
13
+ constructor(appName) {
14
+ this.appName = appName;
15
+ this.proxy = window;
16
+ this.snapshot = new Map();
17
+ this.modifiedKeys = new Set();
18
+ this.active = false;
19
+
20
+ // Side-effect tracking (same as ProxySandbox)
21
+ this._timers = new Set();
22
+ this._intervals = new Set();
23
+ this._rafs = new Set();
24
+ this._eventListeners = [];
25
+
26
+ // Window patching state
27
+ this._patched = false;
28
+ this._originals = null;
29
+ }
30
+
31
+ /**
32
+ * Activate sandbox - capture window snapshot and start tracking.
33
+ */
34
+ activate() {
35
+ if (this.active) return this.proxy;
36
+
37
+ this.snapshot.clear();
38
+ this.modifiedKeys.clear();
39
+
40
+ // Capture current window state
41
+ for (const key in window) {
42
+ try {
43
+ this.snapshot.set(key, window[key]);
44
+ } catch {
45
+ // Some properties may be inaccessible
46
+ }
47
+ }
48
+
49
+ this.active = true;
50
+ logger.wuDebug(`[SnapshotSandbox] Activated for ${this.appName} (${this.snapshot.size} props)`);
51
+ return this.proxy;
52
+ }
53
+
54
+ /**
55
+ * Deactivate sandbox - restore snapshot AND clean side effects.
56
+ */
57
+ deactivate() {
58
+ if (!this.active) return;
59
+
60
+ // Unpatch window if patched
61
+ this.unpatchWindow();
62
+
63
+ // --- Clean tracked timers ---
64
+ for (const id of this._timers) {
65
+ try { clearTimeout(id); } catch {}
66
+ }
67
+ for (const id of this._intervals) {
68
+ try { clearInterval(id); } catch {}
69
+ }
70
+ for (const id of this._rafs) {
71
+ try { cancelAnimationFrame(id); } catch {}
72
+ }
73
+
74
+ const timerCount = this._timers.size + this._intervals.size + this._rafs.size;
75
+ this._timers.clear();
76
+ this._intervals.clear();
77
+ this._rafs.clear();
78
+
79
+ // --- Clean tracked event listeners ---
80
+ const listenerCount = this._eventListeners.length;
81
+ for (const { target, event, handler, options } of this._eventListeners) {
82
+ try { target.removeEventListener(event, handler, options); } catch {}
83
+ }
84
+ this._eventListeners = [];
85
+
86
+ // --- Restore window snapshot ---
87
+ let restoredCount = 0;
88
+ let deletedCount = 0;
89
+
90
+ for (const key in window) {
91
+ try {
92
+ const currentValue = window[key];
93
+ const originalValue = this.snapshot.get(key);
94
+
95
+ if (currentValue !== originalValue) {
96
+ if (this.snapshot.has(key)) {
97
+ window[key] = originalValue;
98
+ restoredCount++;
99
+ } else {
100
+ try {
101
+ delete window[key];
102
+ deletedCount++;
103
+ } catch {}
104
+ }
105
+ }
106
+ } catch {}
107
+ }
108
+
109
+ this.snapshot.clear();
110
+ this.modifiedKeys.clear();
111
+ this.active = false;
112
+
113
+ if (timerCount > 0 || listenerCount > 0) {
114
+ logger.wuDebug(
115
+ `[SnapshotSandbox] ${this.appName} cleanup: ${timerCount} timers, ${listenerCount} listeners, ${restoredCount} restored, ${deletedCount} deleted`
116
+ );
117
+ }
118
+ logger.wuDebug(`[SnapshotSandbox] Deactivated for ${this.appName}`);
119
+ }
120
+
121
+ // ================================================================
122
+ // WINDOW PATCHING - same interface as ProxySandbox
123
+ // ================================================================
124
+
125
+ patchWindow() {
126
+ if (this._patched) return;
127
+
128
+ const self = this;
129
+
130
+ // Capture in local closure — survives unpatch safely
131
+ const originals = {
132
+ setTimeout: window.setTimeout,
133
+ clearTimeout: window.clearTimeout,
134
+ setInterval: window.setInterval,
135
+ clearInterval: window.clearInterval,
136
+ requestAnimationFrame: window.requestAnimationFrame,
137
+ cancelAnimationFrame: window.cancelAnimationFrame,
138
+ addEventListener: window.addEventListener,
139
+ removeEventListener: window.removeEventListener
140
+ };
141
+
142
+ this._originals = originals;
143
+
144
+ window.setTimeout = function(fn, delay, ...args) {
145
+ const id = originals.setTimeout.call(window, fn, delay, ...args);
146
+ if (self._patched) self._timers.add(id);
147
+ return id;
148
+ };
149
+ window.clearTimeout = function(id) {
150
+ self._timers.delete(id);
151
+ return originals.clearTimeout.call(window, id);
152
+ };
153
+ window.setInterval = function(fn, delay, ...args) {
154
+ const id = originals.setInterval.call(window, fn, delay, ...args);
155
+ if (self._patched) self._intervals.add(id);
156
+ return id;
157
+ };
158
+ window.clearInterval = function(id) {
159
+ self._intervals.delete(id);
160
+ return originals.clearInterval.call(window, id);
161
+ };
162
+ window.requestAnimationFrame = function(fn) {
163
+ const id = originals.requestAnimationFrame.call(window, fn);
164
+ if (self._patched) self._rafs.add(id);
165
+ return id;
166
+ };
167
+ window.cancelAnimationFrame = function(id) {
168
+ self._rafs.delete(id);
169
+ return originals.cancelAnimationFrame.call(window, id);
170
+ };
171
+
172
+ window.addEventListener = function(event, handler, options) {
173
+ if (self._patched) self._eventListeners.push({ target: window, event, handler, options });
174
+ return originals.addEventListener.call(window, event, handler, options);
175
+ };
176
+ window.removeEventListener = function(event, handler, options) {
177
+ self._eventListeners = self._eventListeners.filter(
178
+ l => !(l.target === window && l.event === event && l.handler === handler)
179
+ );
180
+ return originals.removeEventListener.call(window, event, handler, options);
181
+ };
182
+
183
+ this._patched = true;
184
+ logger.wuDebug(`[SnapshotSandbox] Window patched for ${this.appName}`);
185
+ }
186
+
187
+ unpatchWindow() {
188
+ if (!this._patched || !this._originals) return;
189
+
190
+ window.setTimeout = this._originals.setTimeout;
191
+ window.clearTimeout = this._originals.clearTimeout;
192
+ window.setInterval = this._originals.setInterval;
193
+ window.clearInterval = this._originals.clearInterval;
194
+ window.requestAnimationFrame = this._originals.requestAnimationFrame;
195
+ window.cancelAnimationFrame = this._originals.cancelAnimationFrame;
196
+ window.addEventListener = this._originals.addEventListener;
197
+ window.removeEventListener = this._originals.removeEventListener;
198
+
199
+ this._patched = false;
200
+ logger.wuDebug(`[SnapshotSandbox] Window unpatched for ${this.appName}`);
201
+ }
202
+
203
+ // ================================================================
204
+ // UTILITIES
205
+ // ================================================================
206
+
207
+ getProxy() {
208
+ return this.active ? this.proxy : null;
209
+ }
210
+
211
+ isActive() {
212
+ return this.active;
213
+ }
214
+
215
+ getStats() {
216
+ return {
217
+ appName: this.appName,
218
+ active: this.active,
219
+ patched: this._patched,
220
+ snapshotSize: this.snapshot.size,
221
+ trackedTimers: this._timers.size,
222
+ trackedIntervals: this._intervals.size,
223
+ trackedRAFs: this._rafs.size,
224
+ trackedEventListeners: this._eventListeners.length
225
+ };
226
+ }
227
+ }
@@ -8,6 +8,15 @@
8
8
  * - API minimalista: get(), set(), on()
9
9
  */
10
10
 
11
+ /**
12
+ * @typedef {Object} WuStoreMetrics
13
+ * @property {number} reads - Total read operations
14
+ * @property {number} writes - Total write operations
15
+ * @property {number} notifications - Total notifications sent
16
+ * @property {number} bufferUtilization - Ring buffer utilization (0-1)
17
+ * @property {number} listenerCount - Active listener count
18
+ */
19
+
11
20
  export class WuStore {
12
21
  constructor(bufferSize = 256) {
13
22
  // Ring Buffer configuration
@@ -63,8 +72,9 @@ export class WuStore {
63
72
  set(path, value) {
64
73
  this.metrics.writes++;
65
74
 
66
- // Write to ring buffer (lock-free)
67
- const sequence = this.cursor++;
75
+ // Write to ring buffer (lock-free, wraps at buffer boundary)
76
+ const sequence = this.cursor;
77
+ this.cursor = (this.cursor + 1) % (this.bufferSize * this.bufferSize);
68
78
  const index = sequence & this.mask;
69
79
 
70
80
  // Reuse buffer slot (zero allocation)
@@ -151,7 +161,7 @@ export class WuStore {
151
161
  getMetrics() {
152
162
  return {
153
163
  ...this.metrics,
154
- bufferUtilization: (this.cursor % this.bufferSize) / this.bufferSize,
164
+ bufferUtilization: Math.min(1, this.cursor / this.bufferSize),
155
165
  listenerCount: this.listeners.size + this.patternListeners.size
156
166
  };
157
167
  }