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
|
@@ -1,476 +1,477 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* WU-PROXY-SANDBOX: Hardened JavaScript Isolation
|
|
3
|
-
*
|
|
4
|
-
* ES6 Proxy-based sandbox with side-effect tracking:
|
|
5
|
-
* - Timer hijacking (setTimeout, setInterval, requestAnimationFrame)
|
|
6
|
-
* - Event listener tracking (window + document addEventListener)
|
|
7
|
-
* - DOM scoping (querySelector/querySelectorAll → shadow root)
|
|
8
|
-
* - Storage scoping (localStorage/sessionStorage → prefixed keys)
|
|
9
|
-
*
|
|
10
|
-
* All tracked side effects are automatically cleaned up on deactivate().
|
|
11
|
-
*/
|
|
12
|
-
|
|
13
|
-
import { logger } from './wu-logger.js';
|
|
14
|
-
|
|
15
|
-
export class WuProxySandbox {
|
|
16
|
-
constructor(appName) {
|
|
17
|
-
this.appName = appName;
|
|
18
|
-
this.
|
|
19
|
-
this.
|
|
20
|
-
this.
|
|
21
|
-
this.
|
|
22
|
-
|
|
23
|
-
|
|
24
|
-
|
|
25
|
-
this.
|
|
26
|
-
this.
|
|
27
|
-
this.
|
|
28
|
-
|
|
29
|
-
|
|
30
|
-
|
|
31
|
-
this.
|
|
32
|
-
this.
|
|
33
|
-
this.
|
|
34
|
-
this.
|
|
35
|
-
|
|
36
|
-
|
|
37
|
-
|
|
38
|
-
this.
|
|
39
|
-
|
|
40
|
-
|
|
41
|
-
|
|
42
|
-
|
|
43
|
-
*
|
|
44
|
-
*
|
|
45
|
-
* @param {
|
|
46
|
-
|
|
47
|
-
|
|
48
|
-
|
|
49
|
-
this.
|
|
50
|
-
|
|
51
|
-
|
|
52
|
-
|
|
53
|
-
|
|
54
|
-
*
|
|
55
|
-
|
|
56
|
-
|
|
57
|
-
|
|
58
|
-
|
|
59
|
-
|
|
60
|
-
|
|
61
|
-
|
|
62
|
-
|
|
63
|
-
|
|
64
|
-
|
|
65
|
-
|
|
66
|
-
|
|
67
|
-
|
|
68
|
-
|
|
69
|
-
|
|
70
|
-
|
|
71
|
-
|
|
72
|
-
|
|
73
|
-
|
|
74
|
-
|
|
75
|
-
|
|
76
|
-
|
|
77
|
-
|
|
78
|
-
|
|
79
|
-
|
|
80
|
-
|
|
81
|
-
|
|
82
|
-
|
|
83
|
-
|
|
84
|
-
self.
|
|
85
|
-
|
|
86
|
-
|
|
87
|
-
|
|
88
|
-
|
|
89
|
-
|
|
90
|
-
|
|
91
|
-
|
|
92
|
-
|
|
93
|
-
|
|
94
|
-
|
|
95
|
-
self.
|
|
96
|
-
|
|
97
|
-
|
|
98
|
-
|
|
99
|
-
|
|
100
|
-
|
|
101
|
-
|
|
102
|
-
|
|
103
|
-
|
|
104
|
-
|
|
105
|
-
|
|
106
|
-
|
|
107
|
-
|
|
108
|
-
|
|
109
|
-
|
|
110
|
-
|
|
111
|
-
|
|
112
|
-
|
|
113
|
-
|
|
114
|
-
|
|
115
|
-
|
|
116
|
-
|
|
117
|
-
|
|
118
|
-
|
|
119
|
-
|
|
120
|
-
|
|
121
|
-
|
|
122
|
-
|
|
123
|
-
|
|
124
|
-
|
|
125
|
-
|
|
126
|
-
|
|
127
|
-
|
|
128
|
-
this._timers.
|
|
129
|
-
this.
|
|
130
|
-
this.
|
|
131
|
-
|
|
132
|
-
|
|
133
|
-
|
|
134
|
-
|
|
135
|
-
|
|
136
|
-
|
|
137
|
-
|
|
138
|
-
|
|
139
|
-
|
|
140
|
-
|
|
141
|
-
this.
|
|
142
|
-
this.
|
|
143
|
-
this.
|
|
144
|
-
this.
|
|
145
|
-
this.
|
|
146
|
-
this.
|
|
147
|
-
|
|
148
|
-
|
|
149
|
-
|
|
150
|
-
|
|
151
|
-
|
|
152
|
-
|
|
153
|
-
|
|
154
|
-
|
|
155
|
-
|
|
156
|
-
|
|
157
|
-
//
|
|
158
|
-
//
|
|
159
|
-
|
|
160
|
-
|
|
161
|
-
|
|
162
|
-
*
|
|
163
|
-
*
|
|
164
|
-
*
|
|
165
|
-
*
|
|
166
|
-
*
|
|
167
|
-
|
|
168
|
-
|
|
169
|
-
|
|
170
|
-
|
|
171
|
-
|
|
172
|
-
|
|
173
|
-
|
|
174
|
-
|
|
175
|
-
|
|
176
|
-
|
|
177
|
-
|
|
178
|
-
|
|
179
|
-
|
|
180
|
-
|
|
181
|
-
|
|
182
|
-
|
|
183
|
-
|
|
184
|
-
|
|
185
|
-
|
|
186
|
-
|
|
187
|
-
|
|
188
|
-
|
|
189
|
-
|
|
190
|
-
|
|
191
|
-
|
|
192
|
-
|
|
193
|
-
|
|
194
|
-
|
|
195
|
-
|
|
196
|
-
|
|
197
|
-
|
|
198
|
-
|
|
199
|
-
|
|
200
|
-
|
|
201
|
-
|
|
202
|
-
|
|
203
|
-
|
|
204
|
-
|
|
205
|
-
|
|
206
|
-
|
|
207
|
-
|
|
208
|
-
|
|
209
|
-
|
|
210
|
-
|
|
211
|
-
|
|
212
|
-
|
|
213
|
-
|
|
214
|
-
|
|
215
|
-
|
|
216
|
-
|
|
217
|
-
|
|
218
|
-
|
|
219
|
-
|
|
220
|
-
|
|
221
|
-
|
|
222
|
-
|
|
223
|
-
|
|
224
|
-
|
|
225
|
-
|
|
226
|
-
|
|
227
|
-
|
|
228
|
-
|
|
229
|
-
|
|
230
|
-
|
|
231
|
-
|
|
232
|
-
|
|
233
|
-
|
|
234
|
-
|
|
235
|
-
*
|
|
236
|
-
|
|
237
|
-
|
|
238
|
-
|
|
239
|
-
|
|
240
|
-
|
|
241
|
-
window.
|
|
242
|
-
window.
|
|
243
|
-
window.
|
|
244
|
-
window.
|
|
245
|
-
window.
|
|
246
|
-
window.
|
|
247
|
-
window.
|
|
248
|
-
|
|
249
|
-
|
|
250
|
-
//
|
|
251
|
-
//
|
|
252
|
-
this.
|
|
253
|
-
|
|
254
|
-
|
|
255
|
-
|
|
256
|
-
|
|
257
|
-
//
|
|
258
|
-
//
|
|
259
|
-
|
|
260
|
-
|
|
261
|
-
|
|
262
|
-
*
|
|
263
|
-
|
|
264
|
-
|
|
265
|
-
|
|
266
|
-
|
|
267
|
-
|
|
268
|
-
|
|
269
|
-
|
|
270
|
-
|
|
271
|
-
|
|
272
|
-
|
|
273
|
-
|
|
274
|
-
|
|
275
|
-
|
|
276
|
-
|
|
277
|
-
|
|
278
|
-
|
|
279
|
-
|
|
280
|
-
|
|
281
|
-
|
|
282
|
-
|
|
283
|
-
|
|
284
|
-
|
|
285
|
-
|
|
286
|
-
|
|
287
|
-
|
|
288
|
-
|
|
289
|
-
|
|
290
|
-
|
|
291
|
-
|
|
292
|
-
|
|
293
|
-
|
|
294
|
-
|
|
295
|
-
|
|
296
|
-
|
|
297
|
-
|
|
298
|
-
|
|
299
|
-
|
|
300
|
-
|
|
301
|
-
|
|
302
|
-
|
|
303
|
-
|
|
304
|
-
|
|
305
|
-
|
|
306
|
-
|
|
307
|
-
|
|
308
|
-
|
|
309
|
-
|
|
310
|
-
|
|
311
|
-
|
|
312
|
-
|
|
313
|
-
|
|
314
|
-
|
|
315
|
-
|
|
316
|
-
|
|
317
|
-
|
|
318
|
-
|
|
319
|
-
|
|
320
|
-
|
|
321
|
-
|
|
322
|
-
|
|
323
|
-
|
|
324
|
-
|
|
325
|
-
|
|
326
|
-
|
|
327
|
-
|
|
328
|
-
|
|
329
|
-
|
|
330
|
-
|
|
331
|
-
|
|
332
|
-
//
|
|
333
|
-
//
|
|
334
|
-
|
|
335
|
-
|
|
336
|
-
|
|
337
|
-
|
|
338
|
-
|
|
339
|
-
|
|
340
|
-
|
|
341
|
-
|
|
342
|
-
|
|
343
|
-
|
|
344
|
-
|
|
345
|
-
|
|
346
|
-
|
|
347
|
-
|
|
348
|
-
|
|
349
|
-
|
|
350
|
-
|
|
351
|
-
|
|
352
|
-
|
|
353
|
-
|
|
354
|
-
|
|
355
|
-
|
|
356
|
-
|
|
357
|
-
|
|
358
|
-
|
|
359
|
-
|
|
360
|
-
|
|
361
|
-
|
|
362
|
-
|
|
363
|
-
|
|
364
|
-
|
|
365
|
-
|
|
366
|
-
|
|
367
|
-
|
|
368
|
-
|
|
369
|
-
|
|
370
|
-
|
|
371
|
-
|
|
372
|
-
|
|
373
|
-
|
|
374
|
-
|
|
375
|
-
|
|
376
|
-
|
|
377
|
-
|
|
378
|
-
|
|
379
|
-
|
|
380
|
-
|
|
381
|
-
|
|
382
|
-
|
|
383
|
-
|
|
384
|
-
|
|
385
|
-
|
|
386
|
-
|
|
387
|
-
//
|
|
388
|
-
//
|
|
389
|
-
|
|
390
|
-
|
|
391
|
-
|
|
392
|
-
|
|
393
|
-
|
|
394
|
-
|
|
395
|
-
|
|
396
|
-
|
|
397
|
-
|
|
398
|
-
|
|
399
|
-
|
|
400
|
-
|
|
401
|
-
|
|
402
|
-
|
|
403
|
-
|
|
404
|
-
|
|
405
|
-
|
|
406
|
-
|
|
407
|
-
|
|
408
|
-
|
|
409
|
-
|
|
410
|
-
|
|
411
|
-
|
|
412
|
-
|
|
413
|
-
|
|
414
|
-
|
|
415
|
-
|
|
416
|
-
|
|
417
|
-
|
|
418
|
-
|
|
419
|
-
|
|
420
|
-
|
|
421
|
-
|
|
422
|
-
|
|
423
|
-
|
|
424
|
-
count
|
|
425
|
-
|
|
426
|
-
|
|
427
|
-
|
|
428
|
-
|
|
429
|
-
|
|
430
|
-
|
|
431
|
-
|
|
432
|
-
|
|
433
|
-
|
|
434
|
-
|
|
435
|
-
|
|
436
|
-
|
|
437
|
-
|
|
438
|
-
|
|
439
|
-
|
|
440
|
-
|
|
441
|
-
|
|
442
|
-
//
|
|
443
|
-
//
|
|
444
|
-
|
|
445
|
-
|
|
446
|
-
|
|
447
|
-
|
|
448
|
-
|
|
449
|
-
|
|
450
|
-
|
|
451
|
-
|
|
452
|
-
|
|
453
|
-
|
|
454
|
-
|
|
455
|
-
|
|
456
|
-
|
|
457
|
-
|
|
458
|
-
|
|
459
|
-
|
|
460
|
-
|
|
461
|
-
|
|
462
|
-
|
|
463
|
-
|
|
464
|
-
|
|
465
|
-
|
|
466
|
-
|
|
467
|
-
|
|
468
|
-
|
|
469
|
-
|
|
470
|
-
|
|
471
|
-
|
|
472
|
-
|
|
473
|
-
|
|
474
|
-
|
|
475
|
-
|
|
476
|
-
}
|
|
1
|
+
/**
|
|
2
|
+
* WU-PROXY-SANDBOX: Hardened JavaScript Isolation
|
|
3
|
+
*
|
|
4
|
+
* ES6 Proxy-based sandbox with side-effect tracking:
|
|
5
|
+
* - Timer hijacking (setTimeout, setInterval, requestAnimationFrame)
|
|
6
|
+
* - Event listener tracking (window + document addEventListener)
|
|
7
|
+
* - DOM scoping (querySelector/querySelectorAll → shadow root)
|
|
8
|
+
* - Storage scoping (localStorage/sessionStorage → prefixed keys)
|
|
9
|
+
*
|
|
10
|
+
* All tracked side effects are automatically cleaned up on deactivate().
|
|
11
|
+
*/
|
|
12
|
+
|
|
13
|
+
import { logger } from './wu-logger.js';
|
|
14
|
+
|
|
15
|
+
export class WuProxySandbox {
|
|
16
|
+
constructor(appName, options = {}) {
|
|
17
|
+
this.appName = appName;
|
|
18
|
+
this.options = options;
|
|
19
|
+
this.proxy = null;
|
|
20
|
+
this.fakeWindow = Object.create(null);
|
|
21
|
+
this.active = false;
|
|
22
|
+
this.modifiedKeys = new Set();
|
|
23
|
+
|
|
24
|
+
// --- Side-effect tracking ---
|
|
25
|
+
this._timers = new Set();
|
|
26
|
+
this._intervals = new Set();
|
|
27
|
+
this._rafs = new Set();
|
|
28
|
+
this._eventListeners = []; // [{target, event, handler, options}]
|
|
29
|
+
|
|
30
|
+
// --- DOM & Storage scoping ---
|
|
31
|
+
this._container = null;
|
|
32
|
+
this._shadowRoot = null;
|
|
33
|
+
this._scopedDocument = null;
|
|
34
|
+
this._scopedLocalStorage = null;
|
|
35
|
+
this._scopedSessionStorage = null;
|
|
36
|
+
|
|
37
|
+
// --- Window patching state ---
|
|
38
|
+
this._patched = false;
|
|
39
|
+
this._originals = null;
|
|
40
|
+
}
|
|
41
|
+
|
|
42
|
+
/**
|
|
43
|
+
* Set the DOM scope for this sandbox.
|
|
44
|
+
* Must be called before activate() for DOM scoping to work.
|
|
45
|
+
* @param {HTMLElement} container - App container element
|
|
46
|
+
* @param {ShadowRoot} shadowRoot - Shadow root containing the container
|
|
47
|
+
*/
|
|
48
|
+
setContainer(container, shadowRoot) {
|
|
49
|
+
this._container = container;
|
|
50
|
+
this._shadowRoot = shadowRoot;
|
|
51
|
+
}
|
|
52
|
+
|
|
53
|
+
/**
|
|
54
|
+
* Activate the sandbox. Creates the Proxy and starts tracking.
|
|
55
|
+
* @returns {Proxy} The sandboxed window proxy
|
|
56
|
+
*/
|
|
57
|
+
activate() {
|
|
58
|
+
if (this.active) return this.proxy;
|
|
59
|
+
|
|
60
|
+
const self = this;
|
|
61
|
+
|
|
62
|
+
this.proxy = new Proxy(window, {
|
|
63
|
+
get(target, prop) {
|
|
64
|
+
// 1. App's own isolated globals
|
|
65
|
+
if (prop in self.fakeWindow) {
|
|
66
|
+
return self.fakeWindow[prop];
|
|
67
|
+
}
|
|
68
|
+
|
|
69
|
+
// 2. Intercepted APIs
|
|
70
|
+
const intercepted = self._intercept(prop, target);
|
|
71
|
+
if (intercepted !== undefined) {
|
|
72
|
+
return intercepted;
|
|
73
|
+
}
|
|
74
|
+
|
|
75
|
+
// 3. Real window value with correct binding
|
|
76
|
+
const value = target[prop];
|
|
77
|
+
if (typeof value === 'function' && !self._isConstructor(value)) {
|
|
78
|
+
return value.bind(target);
|
|
79
|
+
}
|
|
80
|
+
return value;
|
|
81
|
+
},
|
|
82
|
+
|
|
83
|
+
set(target, prop, value) {
|
|
84
|
+
self.fakeWindow[prop] = value;
|
|
85
|
+
self.modifiedKeys.add(prop);
|
|
86
|
+
return true;
|
|
87
|
+
},
|
|
88
|
+
|
|
89
|
+
has(target, prop) {
|
|
90
|
+
return prop in self.fakeWindow || prop in target;
|
|
91
|
+
},
|
|
92
|
+
|
|
93
|
+
deleteProperty(target, prop) {
|
|
94
|
+
if (prop in self.fakeWindow) {
|
|
95
|
+
delete self.fakeWindow[prop];
|
|
96
|
+
self.modifiedKeys.delete(prop);
|
|
97
|
+
return true;
|
|
98
|
+
}
|
|
99
|
+
return false;
|
|
100
|
+
}
|
|
101
|
+
});
|
|
102
|
+
|
|
103
|
+
this.active = true;
|
|
104
|
+
logger.wuDebug(`[ProxySandbox] Activated for ${this.appName}`);
|
|
105
|
+
return this.proxy;
|
|
106
|
+
}
|
|
107
|
+
|
|
108
|
+
/**
|
|
109
|
+
* Deactivate the sandbox. Cleans up ALL tracked side effects.
|
|
110
|
+
*/
|
|
111
|
+
deactivate() {
|
|
112
|
+
if (!this.active) return;
|
|
113
|
+
|
|
114
|
+
// Unpatch window if patched
|
|
115
|
+
this.unpatchWindow();
|
|
116
|
+
|
|
117
|
+
// --- Clean timers ---
|
|
118
|
+
for (const id of this._timers) {
|
|
119
|
+
try { clearTimeout(id); } catch {}
|
|
120
|
+
}
|
|
121
|
+
for (const id of this._intervals) {
|
|
122
|
+
try { clearInterval(id); } catch {}
|
|
123
|
+
}
|
|
124
|
+
for (const id of this._rafs) {
|
|
125
|
+
try { cancelAnimationFrame(id); } catch {}
|
|
126
|
+
}
|
|
127
|
+
|
|
128
|
+
const timerCount = this._timers.size + this._intervals.size + this._rafs.size;
|
|
129
|
+
this._timers.clear();
|
|
130
|
+
this._intervals.clear();
|
|
131
|
+
this._rafs.clear();
|
|
132
|
+
|
|
133
|
+
// --- Clean event listeners ---
|
|
134
|
+
const listenerCount = this._eventListeners.length;
|
|
135
|
+
for (const { target, event, handler, options } of this._eventListeners) {
|
|
136
|
+
try { target.removeEventListener(event, handler, options); } catch {}
|
|
137
|
+
}
|
|
138
|
+
this._eventListeners = [];
|
|
139
|
+
|
|
140
|
+
// --- Clean namespace ---
|
|
141
|
+
this.fakeWindow = Object.create(null);
|
|
142
|
+
this.modifiedKeys.clear();
|
|
143
|
+
this._scopedDocument = null;
|
|
144
|
+
this._scopedLocalStorage = null;
|
|
145
|
+
this._scopedSessionStorage = null;
|
|
146
|
+
this.proxy = null;
|
|
147
|
+
this.active = false;
|
|
148
|
+
|
|
149
|
+
if (timerCount > 0 || listenerCount > 0) {
|
|
150
|
+
logger.wuDebug(
|
|
151
|
+
`[ProxySandbox] ${this.appName} cleanup: ${timerCount} timers, ${listenerCount} listeners`
|
|
152
|
+
);
|
|
153
|
+
}
|
|
154
|
+
logger.wuDebug(`[ProxySandbox] Deactivated for ${this.appName}`);
|
|
155
|
+
}
|
|
156
|
+
|
|
157
|
+
// ================================================================
|
|
158
|
+
// WINDOW PATCHING - patches real window APIs during module loading
|
|
159
|
+
// ================================================================
|
|
160
|
+
|
|
161
|
+
/**
|
|
162
|
+
* Patch real window APIs to track side effects from global code.
|
|
163
|
+
* Call before loading app module, unpatch after.
|
|
164
|
+
*
|
|
165
|
+
* IMPORTANT: Uses closure over originals so patched functions remain
|
|
166
|
+
* valid even after unpatchWindow() — prevents crashes when frameworks
|
|
167
|
+
* (React 19, etc.) cache references to patched setTimeout during import.
|
|
168
|
+
*/
|
|
169
|
+
patchWindow() {
|
|
170
|
+
if (this._patched) return;
|
|
171
|
+
|
|
172
|
+
const self = this;
|
|
173
|
+
|
|
174
|
+
// Capture originals in a local closure that survives unpatch
|
|
175
|
+
const originals = {
|
|
176
|
+
setTimeout: window.setTimeout,
|
|
177
|
+
clearTimeout: window.clearTimeout,
|
|
178
|
+
setInterval: window.setInterval,
|
|
179
|
+
clearInterval: window.clearInterval,
|
|
180
|
+
requestAnimationFrame: window.requestAnimationFrame,
|
|
181
|
+
cancelAnimationFrame: window.cancelAnimationFrame,
|
|
182
|
+
addEventListener: window.addEventListener,
|
|
183
|
+
removeEventListener: window.removeEventListener
|
|
184
|
+
};
|
|
185
|
+
|
|
186
|
+
// Store reference (used by unpatchWindow to restore)
|
|
187
|
+
this._originals = originals;
|
|
188
|
+
|
|
189
|
+
// Patch timers — closure captures `originals`, not `self._originals`
|
|
190
|
+
window.setTimeout = function(fn, delay, ...args) {
|
|
191
|
+
const id = originals.setTimeout.call(window, fn, delay, ...args);
|
|
192
|
+
if (self._patched) self._timers.add(id);
|
|
193
|
+
return id;
|
|
194
|
+
};
|
|
195
|
+
window.clearTimeout = function(id) {
|
|
196
|
+
self._timers.delete(id);
|
|
197
|
+
return originals.clearTimeout.call(window, id);
|
|
198
|
+
};
|
|
199
|
+
window.setInterval = function(fn, delay, ...args) {
|
|
200
|
+
const id = originals.setInterval.call(window, fn, delay, ...args);
|
|
201
|
+
if (self._patched) self._intervals.add(id);
|
|
202
|
+
return id;
|
|
203
|
+
};
|
|
204
|
+
window.clearInterval = function(id) {
|
|
205
|
+
self._intervals.delete(id);
|
|
206
|
+
return originals.clearInterval.call(window, id);
|
|
207
|
+
};
|
|
208
|
+
window.requestAnimationFrame = function(fn) {
|
|
209
|
+
const id = originals.requestAnimationFrame.call(window, fn);
|
|
210
|
+
if (self._patched) self._rafs.add(id);
|
|
211
|
+
return id;
|
|
212
|
+
};
|
|
213
|
+
window.cancelAnimationFrame = function(id) {
|
|
214
|
+
self._rafs.delete(id);
|
|
215
|
+
return originals.cancelAnimationFrame.call(window, id);
|
|
216
|
+
};
|
|
217
|
+
|
|
218
|
+
// Patch event listeners
|
|
219
|
+
window.addEventListener = function(event, handler, options) {
|
|
220
|
+
if (self._patched) self._eventListeners.push({ target: window, event, handler, options });
|
|
221
|
+
return originals.addEventListener.call(window, event, handler, options);
|
|
222
|
+
};
|
|
223
|
+
window.removeEventListener = function(event, handler, options) {
|
|
224
|
+
self._eventListeners = self._eventListeners.filter(
|
|
225
|
+
l => !(l.target === window && l.event === event && l.handler === handler)
|
|
226
|
+
);
|
|
227
|
+
return originals.removeEventListener.call(window, event, handler, options);
|
|
228
|
+
};
|
|
229
|
+
|
|
230
|
+
this._patched = true;
|
|
231
|
+
logger.wuDebug(`[ProxySandbox] Window patched for ${this.appName}`);
|
|
232
|
+
}
|
|
233
|
+
|
|
234
|
+
/**
|
|
235
|
+
* Restore original window APIs.
|
|
236
|
+
* Safe: patched functions still work via closure even after restore.
|
|
237
|
+
*/
|
|
238
|
+
unpatchWindow() {
|
|
239
|
+
if (!this._patched || !this._originals) return;
|
|
240
|
+
|
|
241
|
+
window.setTimeout = this._originals.setTimeout;
|
|
242
|
+
window.clearTimeout = this._originals.clearTimeout;
|
|
243
|
+
window.setInterval = this._originals.setInterval;
|
|
244
|
+
window.clearInterval = this._originals.clearInterval;
|
|
245
|
+
window.requestAnimationFrame = this._originals.requestAnimationFrame;
|
|
246
|
+
window.cancelAnimationFrame = this._originals.cancelAnimationFrame;
|
|
247
|
+
window.addEventListener = this._originals.addEventListener;
|
|
248
|
+
window.removeEventListener = this._originals.removeEventListener;
|
|
249
|
+
|
|
250
|
+
// NOTE: Do NOT null _originals — patched closures may still reference
|
|
251
|
+
// the sandbox instance (e.g. React scheduler caches setTimeout).
|
|
252
|
+
// The closure uses `originals` (local const), not `this._originals`.
|
|
253
|
+
this._patched = false;
|
|
254
|
+
logger.wuDebug(`[ProxySandbox] Window unpatched for ${this.appName}`);
|
|
255
|
+
}
|
|
256
|
+
|
|
257
|
+
// ================================================================
|
|
258
|
+
// PROXY INTERCEPTS - for code running through the proxy
|
|
259
|
+
// ================================================================
|
|
260
|
+
|
|
261
|
+
/**
|
|
262
|
+
* Intercept property access on the proxy.
|
|
263
|
+
* Returns wrapped API or undefined to fall through.
|
|
264
|
+
*/
|
|
265
|
+
_intercept(prop, target) {
|
|
266
|
+
const self = this;
|
|
267
|
+
|
|
268
|
+
switch (prop) {
|
|
269
|
+
// --- Timer hijacking ---
|
|
270
|
+
case 'setTimeout':
|
|
271
|
+
return function(fn, delay, ...args) {
|
|
272
|
+
const id = target.setTimeout(fn, delay, ...args);
|
|
273
|
+
self._timers.add(id);
|
|
274
|
+
return id;
|
|
275
|
+
};
|
|
276
|
+
case 'clearTimeout':
|
|
277
|
+
return function(id) {
|
|
278
|
+
self._timers.delete(id);
|
|
279
|
+
target.clearTimeout(id);
|
|
280
|
+
};
|
|
281
|
+
case 'setInterval':
|
|
282
|
+
return function(fn, delay, ...args) {
|
|
283
|
+
const id = target.setInterval(fn, delay, ...args);
|
|
284
|
+
self._intervals.add(id);
|
|
285
|
+
return id;
|
|
286
|
+
};
|
|
287
|
+
case 'clearInterval':
|
|
288
|
+
return function(id) {
|
|
289
|
+
self._intervals.delete(id);
|
|
290
|
+
target.clearInterval(id);
|
|
291
|
+
};
|
|
292
|
+
case 'requestAnimationFrame':
|
|
293
|
+
return function(fn) {
|
|
294
|
+
const id = target.requestAnimationFrame(fn);
|
|
295
|
+
self._rafs.add(id);
|
|
296
|
+
return id;
|
|
297
|
+
};
|
|
298
|
+
case 'cancelAnimationFrame':
|
|
299
|
+
return function(id) {
|
|
300
|
+
self._rafs.delete(id);
|
|
301
|
+
target.cancelAnimationFrame(id);
|
|
302
|
+
};
|
|
303
|
+
|
|
304
|
+
// --- Event listener tracking ---
|
|
305
|
+
case 'addEventListener':
|
|
306
|
+
return function(event, handler, options) {
|
|
307
|
+
self._eventListeners.push({ target, event, handler, options });
|
|
308
|
+
target.addEventListener(event, handler, options);
|
|
309
|
+
};
|
|
310
|
+
case 'removeEventListener':
|
|
311
|
+
return function(event, handler, options) {
|
|
312
|
+
self._eventListeners = self._eventListeners.filter(
|
|
313
|
+
l => !(l.target === target && l.event === event && l.handler === handler)
|
|
314
|
+
);
|
|
315
|
+
target.removeEventListener(event, handler, options);
|
|
316
|
+
};
|
|
317
|
+
|
|
318
|
+
// --- DOM scoping ---
|
|
319
|
+
case 'document':
|
|
320
|
+
return this._getScopedDocument();
|
|
321
|
+
|
|
322
|
+
// --- Storage scoping ---
|
|
323
|
+
case 'localStorage':
|
|
324
|
+
return this._getScopedStorage('local');
|
|
325
|
+
case 'sessionStorage':
|
|
326
|
+
return this._getScopedStorage('session');
|
|
327
|
+
}
|
|
328
|
+
|
|
329
|
+
return undefined;
|
|
330
|
+
}
|
|
331
|
+
|
|
332
|
+
// ================================================================
|
|
333
|
+
// DOM SCOPING - querySelector searches inside shadow root
|
|
334
|
+
// ================================================================
|
|
335
|
+
|
|
336
|
+
_getScopedDocument() {
|
|
337
|
+
if (this._scopedDocument) return this._scopedDocument;
|
|
338
|
+
|
|
339
|
+
const root = this._shadowRoot || this._container;
|
|
340
|
+
if (!root) return document; // No container set, pass through
|
|
341
|
+
|
|
342
|
+
const self = this;
|
|
343
|
+
|
|
344
|
+
this._scopedDocument = new Proxy(document, {
|
|
345
|
+
get(target, prop) {
|
|
346
|
+
switch (prop) {
|
|
347
|
+
case 'querySelector':
|
|
348
|
+
return (selector) => root.querySelector(selector);
|
|
349
|
+
case 'querySelectorAll':
|
|
350
|
+
return (selector) => root.querySelectorAll(selector);
|
|
351
|
+
case 'getElementById':
|
|
352
|
+
return (id) => root.querySelector(`#${CSS.escape(id)}`);
|
|
353
|
+
case 'getElementsByClassName':
|
|
354
|
+
return (className) => root.querySelectorAll(`.${CSS.escape(className)}`);
|
|
355
|
+
case 'getElementsByTagName':
|
|
356
|
+
return (tag) => root.querySelectorAll(tag);
|
|
357
|
+
|
|
358
|
+
// Track document event listeners too
|
|
359
|
+
case 'addEventListener':
|
|
360
|
+
return function(event, handler, options) {
|
|
361
|
+
self._eventListeners.push({ target, event, handler, options });
|
|
362
|
+
target.addEventListener(event, handler, options);
|
|
363
|
+
};
|
|
364
|
+
case 'removeEventListener':
|
|
365
|
+
return function(event, handler, options) {
|
|
366
|
+
self._eventListeners = self._eventListeners.filter(
|
|
367
|
+
l => !(l.target === target && l.event === event && l.handler === handler)
|
|
368
|
+
);
|
|
369
|
+
target.removeEventListener(event, handler, options);
|
|
370
|
+
};
|
|
371
|
+
|
|
372
|
+
// createElement, createTextNode, etc. - pass through
|
|
373
|
+
default: {
|
|
374
|
+
const value = target[prop];
|
|
375
|
+
if (typeof value === 'function') {
|
|
376
|
+
return value.bind(target);
|
|
377
|
+
}
|
|
378
|
+
return value;
|
|
379
|
+
}
|
|
380
|
+
}
|
|
381
|
+
}
|
|
382
|
+
});
|
|
383
|
+
|
|
384
|
+
return this._scopedDocument;
|
|
385
|
+
}
|
|
386
|
+
|
|
387
|
+
// ================================================================
|
|
388
|
+
// STORAGE SCOPING - localStorage/sessionStorage with app prefix
|
|
389
|
+
// ================================================================
|
|
390
|
+
|
|
391
|
+
_getScopedStorage(type) {
|
|
392
|
+
const cacheKey = type === 'local' ? '_scopedLocalStorage' : '_scopedSessionStorage';
|
|
393
|
+
if (this[cacheKey]) return this[cacheKey];
|
|
394
|
+
|
|
395
|
+
const realStorage = type === 'local' ? window.localStorage : window.sessionStorage;
|
|
396
|
+
if (!realStorage) return realStorage;
|
|
397
|
+
|
|
398
|
+
const prefix = `wu_${this.appName}_`;
|
|
399
|
+
|
|
400
|
+
this[cacheKey] = {
|
|
401
|
+
getItem(key) {
|
|
402
|
+
return realStorage.getItem(prefix + key);
|
|
403
|
+
},
|
|
404
|
+
setItem(key, value) {
|
|
405
|
+
realStorage.setItem(prefix + key, String(value));
|
|
406
|
+
},
|
|
407
|
+
removeItem(key) {
|
|
408
|
+
realStorage.removeItem(prefix + key);
|
|
409
|
+
},
|
|
410
|
+
clear() {
|
|
411
|
+
// Only clear this app's keys
|
|
412
|
+
const toRemove = [];
|
|
413
|
+
for (let i = 0; i < realStorage.length; i++) {
|
|
414
|
+
const k = realStorage.key(i);
|
|
415
|
+
if (k && k.startsWith(prefix)) toRemove.push(k);
|
|
416
|
+
}
|
|
417
|
+
toRemove.forEach(k => realStorage.removeItem(k));
|
|
418
|
+
},
|
|
419
|
+
key(index) {
|
|
420
|
+
let count = 0;
|
|
421
|
+
for (let i = 0; i < realStorage.length; i++) {
|
|
422
|
+
const k = realStorage.key(i);
|
|
423
|
+
if (k && k.startsWith(prefix)) {
|
|
424
|
+
if (count === index) return k.slice(prefix.length);
|
|
425
|
+
count++;
|
|
426
|
+
}
|
|
427
|
+
}
|
|
428
|
+
return null;
|
|
429
|
+
},
|
|
430
|
+
get length() {
|
|
431
|
+
let count = 0;
|
|
432
|
+
for (let i = 0; i < realStorage.length; i++) {
|
|
433
|
+
if (realStorage.key(i)?.startsWith(prefix)) count++;
|
|
434
|
+
}
|
|
435
|
+
return count;
|
|
436
|
+
}
|
|
437
|
+
};
|
|
438
|
+
|
|
439
|
+
return this[cacheKey];
|
|
440
|
+
}
|
|
441
|
+
|
|
442
|
+
// ================================================================
|
|
443
|
+
// UTILITIES
|
|
444
|
+
// ================================================================
|
|
445
|
+
|
|
446
|
+
_isConstructor(fn) {
|
|
447
|
+
try {
|
|
448
|
+
return fn.prototype && fn.prototype.constructor === fn;
|
|
449
|
+
} catch {
|
|
450
|
+
return false;
|
|
451
|
+
}
|
|
452
|
+
}
|
|
453
|
+
|
|
454
|
+
getProxy() {
|
|
455
|
+
return this.active ? this.proxy : null;
|
|
456
|
+
}
|
|
457
|
+
|
|
458
|
+
isActive() {
|
|
459
|
+
return this.active;
|
|
460
|
+
}
|
|
461
|
+
|
|
462
|
+
getStats() {
|
|
463
|
+
return {
|
|
464
|
+
appName: this.appName,
|
|
465
|
+
active: this.active,
|
|
466
|
+
patched: this._patched,
|
|
467
|
+
modifiedKeys: Array.from(this.modifiedKeys),
|
|
468
|
+
isolatedPropsCount: Object.keys(this.fakeWindow).length,
|
|
469
|
+
trackedTimers: this._timers.size,
|
|
470
|
+
trackedIntervals: this._intervals.size,
|
|
471
|
+
trackedRAFs: this._rafs.size,
|
|
472
|
+
trackedEventListeners: this._eventListeners.length,
|
|
473
|
+
hasContainer: !!this._container,
|
|
474
|
+
hasShadowRoot: !!this._shadowRoot
|
|
475
|
+
};
|
|
476
|
+
}
|
|
477
|
+
}
|