@whitesev/utils 2.9.12 → 2.9.13

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 (73) hide show
  1. package/README.md +176 -176
  2. package/dist/index.amd.js +128 -85
  3. package/dist/index.amd.js.map +1 -1
  4. package/dist/index.amd.min.js +1 -1
  5. package/dist/index.amd.min.js.map +1 -1
  6. package/dist/index.cjs.js +128 -85
  7. package/dist/index.cjs.js.map +1 -1
  8. package/dist/index.cjs.min.js +1 -1
  9. package/dist/index.cjs.min.js.map +1 -1
  10. package/dist/index.esm.js +128 -85
  11. package/dist/index.esm.js.map +1 -1
  12. package/dist/index.esm.min.js +1 -1
  13. package/dist/index.esm.min.js.map +1 -1
  14. package/dist/index.iife.js +128 -85
  15. package/dist/index.iife.js.map +1 -1
  16. package/dist/index.iife.min.js +1 -1
  17. package/dist/index.iife.min.js.map +1 -1
  18. package/dist/index.system.js +128 -85
  19. package/dist/index.system.js.map +1 -1
  20. package/dist/index.system.min.js +1 -1
  21. package/dist/index.system.min.js.map +1 -1
  22. package/dist/index.umd.js +128 -85
  23. package/dist/index.umd.js.map +1 -1
  24. package/dist/index.umd.min.js +1 -1
  25. package/dist/index.umd.min.js.map +1 -1
  26. package/dist/types/src/Dictionary.d.ts +2 -0
  27. package/dist/types/src/Utils.d.ts +5 -2
  28. package/dist/types/src/types/Httpx.d.ts +1344 -1344
  29. package/dist/types/src/types/Log.d.ts +19 -19
  30. package/dist/types/src/types/Progress.d.ts +20 -20
  31. package/dist/types/src/types/React.d.ts +119 -119
  32. package/dist/types/src/types/TryCatch.d.ts +9 -9
  33. package/dist/types/src/types/UtilsGMCookie.d.ts +93 -93
  34. package/dist/types/src/types/UtilsGMMenu.d.ts +77 -77
  35. package/dist/types/src/types/Vue2.d.ts +166 -166
  36. package/dist/types/src/types/WindowApi.d.ts +14 -14
  37. package/dist/types/src/types/ajaxHooker.d.ts +155 -155
  38. package/dist/types/src/types/env.d.ts +7 -7
  39. package/dist/types/src/types/global.d.ts +31 -31
  40. package/package.json +26 -24
  41. package/src/ColorConversion.ts +118 -118
  42. package/src/CommonUtil.ts +301 -301
  43. package/src/DOMUtils.ts +251 -251
  44. package/src/Dictionary.ts +205 -199
  45. package/src/GBKEncoder.ts +108 -108
  46. package/src/Hooks.ts +73 -73
  47. package/src/Httpx.ts +1457 -1457
  48. package/src/LockFunction.ts +65 -62
  49. package/src/Log.ts +233 -233
  50. package/src/ModuleRaid.js +378 -360
  51. package/src/Progress.ts +108 -108
  52. package/src/TryCatch.ts +86 -86
  53. package/src/Utils.ts +3860 -3852
  54. package/src/UtilsCommon.ts +14 -14
  55. package/src/UtilsGMCookie.ts +273 -273
  56. package/src/UtilsGMMenu.ts +460 -460
  57. package/src/Vue.ts +233 -233
  58. package/src/WindowApi.ts +59 -59
  59. package/src/ajaxHooker/ajaxHooker.js +606 -538
  60. package/src/ajaxHooker/ajaxHooker1.2.4.js +440 -438
  61. package/src/indexedDB.ts +497 -497
  62. package/src/types/Httpx.d.ts +1344 -1344
  63. package/src/types/Log.d.ts +19 -19
  64. package/src/types/Progress.d.ts +20 -20
  65. package/src/types/React.d.ts +119 -119
  66. package/src/types/TryCatch.d.ts +9 -9
  67. package/src/types/UtilsGMCookie.d.ts +93 -93
  68. package/src/types/UtilsGMMenu.d.ts +77 -77
  69. package/src/types/Vue2.d.ts +166 -166
  70. package/src/types/WindowApi.d.ts +14 -14
  71. package/src/types/ajaxHooker.d.ts +155 -155
  72. package/src/types/env.d.ts +7 -7
  73. package/src/types/global.d.ts +31 -31
@@ -6,442 +6,444 @@
6
6
  // ==/UserScript==
7
7
 
8
8
  export const AjaxHooker1_2_4 = function () {
9
- return (function () {
10
- const win = window.unsafeWindow || document.defaultView || window;
11
- const hookFns = [];
12
- const realXhr = win.XMLHttpRequest;
13
- const resProto = win.Response.prototype;
14
- const toString = Object.prototype.toString;
15
- const realFetch = win.fetch;
16
- const xhrResponses = ["response", "responseText", "responseXML"];
17
- const fetchResponses = ["arrayBuffer", "blob", "formData", "json", "text"];
18
- const xhrAsyncEvents = ["readystatechange", "load", "loadend"];
19
- let filter;
20
- function emptyFn() {}
21
- function errorFn(err) {
22
- console.error(err);
23
- }
24
- function defineProp(obj, prop, getter, setter) {
25
- Object.defineProperty(obj, prop, {
26
- configurable: true,
27
- enumerable: true,
28
- get: getter,
29
- set: setter,
30
- });
31
- }
32
- function readonly(obj, prop, value = obj[prop]) {
33
- defineProp(obj, prop, () => value, emptyFn);
34
- }
35
- function writable(obj, prop, value = obj[prop]) {
36
- Object.defineProperty(obj, prop, {
37
- configurable: true,
38
- enumerable: true,
39
- writable: true,
40
- value: value,
41
- });
42
- }
43
- function toFilterObj(obj) {
44
- return {
45
- type: obj.type,
46
- url: obj.url,
47
- method: obj.method && obj.method.toUpperCase(),
48
- };
49
- }
50
- function shouldFilter(type, url, method) {
51
- return (
52
- filter &&
53
- !filter.find(
54
- (obj) =>
55
- (!obj.type || obj.type === type) &&
56
- (!obj.url ||
57
- (toString.call(obj.url) === "[object String]"
58
- ? url.includes(obj.url)
59
- : obj.url.test(url))) &&
60
- (!obj.method || obj.method === method.toUpperCase())
61
- )
62
- );
63
- }
64
- function lookupGetter(obj, prop) {
65
- let getter;
66
- let proto = obj;
67
- while (proto) {
68
- const descriptor = Object.getOwnPropertyDescriptor(proto, prop);
69
- getter = descriptor && descriptor.get;
70
- if (getter) break;
71
- proto = Object.getPrototypeOf(proto);
72
- }
73
- return getter ? getter.bind(obj) : emptyFn;
74
- }
75
- function waitForHookFns(request) {
76
- return Promise.all(
77
- hookFns.map((fn) => Promise.resolve(fn(request)).then(emptyFn, errorFn))
78
- );
79
- }
80
- function waitForRequestKeys(request, requestClone) {
81
- return Promise.all(
82
- ["url", "method", "abort", "headers", "data"].map((key) => {
83
- return Promise.resolve(request[key]).then(
84
- (val) => (request[key] = val),
85
- () => (request[key] = requestClone[key])
86
- );
87
- })
88
- );
89
- }
90
- function fakeEventSIP() {
91
- this.ajaxHooker_stopped = true;
92
- }
93
- function xhrDelegateEvent(e) {
94
- const xhr = e.target;
95
- e.stopImmediatePropagation = fakeEventSIP;
96
- xhr.__ajaxHooker.hookedEvents[e.type].forEach(
97
- (fn) => !e.ajaxHooker_stopped && fn.call(xhr, e)
98
- );
99
- const onEvent = xhr.__ajaxHooker.hookedEvents["on" + e.type];
100
- typeof onEvent === "function" && onEvent.call(xhr, e);
101
- }
102
- function xhrReadyStateChange(e) {
103
- if (e.target.readyState === 4) {
104
- e.target.dispatchEvent(
105
- new CustomEvent("ajaxHooker_responseReady", { detail: e })
106
- );
107
- } else {
108
- e.target.__ajaxHooker.delegateEvent(e);
109
- }
110
- }
111
- function xhrLoadAndLoadend(e) {
112
- e.target.__ajaxHooker.delegateEvent(e);
113
- }
114
- function fakeXhrOpen(method, url, ...args) {
115
- const ah = this.__ajaxHooker;
116
- ah.url = url.toString();
117
- ah.method = method.toUpperCase();
118
- ah.openArgs = args;
119
- ah.headers = {};
120
- return ah.originalMethods.open(method, url, ...args);
121
- }
122
- function fakeXhr() {
123
- const xhr = new realXhr();
124
- let ah = xhr.__ajaxHooker;
125
- if (!ah) {
126
- ah = xhr.__ajaxHooker = {
127
- headers: {},
128
- hookedEvents: {
129
- readystatechange: new Set(),
130
- load: new Set(),
131
- loadend: new Set(),
132
- },
133
- delegateEvent: xhrDelegateEvent,
134
- originalGetters: {},
135
- originalMethods: {},
136
- };
137
- xhr.addEventListener("readystatechange", xhrReadyStateChange);
138
- xhr.addEventListener("load", xhrLoadAndLoadend);
139
- xhr.addEventListener("loadend", xhrLoadAndLoadend);
140
- for (const key of xhrResponses) {
141
- ah.originalGetters[key] = lookupGetter(xhr, key);
142
- }
143
- for (const method of [
144
- "open",
145
- "setRequestHeader",
146
- "addEventListener",
147
- "removeEventListener",
148
- ]) {
149
- ah.originalMethods[method] = xhr[method].bind(xhr);
150
- }
151
- xhr.open = fakeXhrOpen;
152
- xhr.setRequestHeader = (header, value) => {
153
- ah.originalMethods.setRequestHeader(header, value);
154
- if (xhr.readyState === 1) {
155
- if (ah.headers[header]) {
156
- ah.headers[header] += ", " + value;
157
- } else {
158
- ah.headers[header] = value;
159
- }
160
- }
161
- };
162
- xhr.addEventListener = function (...args) {
163
- if (xhrAsyncEvents.includes(args[0])) {
164
- ah.hookedEvents[args[0]].add(args[1]);
165
- } else {
166
- ah.originalMethods.addEventListener(...args);
167
- }
168
- };
169
- xhr.removeEventListener = function (...args) {
170
- if (xhrAsyncEvents.includes(args[0])) {
171
- ah.hookedEvents[args[0]].delete(args[1]);
172
- } else {
173
- ah.originalMethods.removeEventListener(...args);
174
- }
175
- };
176
- xhrAsyncEvents.forEach((evt) => {
177
- const onEvt = "on" + evt;
178
- defineProp(
179
- xhr,
180
- onEvt,
181
- () => {
182
- return ah.hookedEvents[onEvt] || null;
183
- },
184
- (val) => {
185
- ah.hookedEvents[onEvt] = typeof val === "function" ? val : null;
186
- }
187
- );
188
- });
189
- }
190
- const realSend = xhr.send.bind(xhr);
191
- xhr.send = function (data) {
192
- if (xhr.readyState !== 1) return realSend(data);
193
- ah.delegateEvent = xhrDelegateEvent;
194
- xhrResponses.forEach((prop) => {
195
- delete xhr[prop]; // delete descriptor
196
- });
197
- if (shouldFilter("xhr", ah.url, ah.method)) {
198
- xhr.addEventListener("ajaxHooker_responseReady", (e) => {
199
- ah.delegateEvent(e.detail);
200
- });
201
- return realSend(data);
202
- }
203
- try {
204
- const request = {
205
- type: "xhr",
206
- url: ah.url,
207
- method: ah.method,
208
- abort: false,
209
- headers: ah.headers,
210
- data: data,
211
- response: null,
212
- };
213
- const requestClone = { ...request };
214
- waitForHookFns(request).then(() => {
215
- waitForRequestKeys(request, requestClone).then(() => {
216
- if (request.abort) return;
217
- ah.originalMethods.open(
218
- request.method,
219
- request.url,
220
- ...ah.openArgs
221
- );
222
- for (const header in request.headers) {
223
- ah.originalMethods.setRequestHeader(
224
- header,
225
- request.headers[header]
226
- );
227
- }
228
- data = request.data;
229
- xhr.addEventListener("ajaxHooker_responseReady", (e) => {
230
- try {
231
- if (typeof request.response === "function") {
232
- const arg = {
233
- finalUrl: xhr.responseURL,
234
- status: xhr.status,
235
- responseHeaders: {},
236
- };
237
- for (const line of xhr
238
- .getAllResponseHeaders()
239
- .trim()
240
- .split(/[\r\n]+/)) {
241
- const parts = line.split(/:\s*/);
242
- if (parts.length === 2) {
243
- const lheader = parts[0].toLowerCase();
244
- if (arg.responseHeaders[lheader]) {
245
- arg.responseHeaders[lheader] += ", " + parts[1];
246
- } else {
247
- arg.responseHeaders[lheader] = parts[1];
248
- }
249
- }
250
- }
251
- xhrResponses.forEach((prop) => {
252
- defineProp(
253
- arg,
254
- prop,
255
- () => {
256
- return (arg[prop] = ah.originalGetters[prop]());
257
- },
258
- (val) => {
259
- delete arg[prop];
260
- arg[prop] = val;
261
- }
262
- );
263
- defineProp(xhr, prop, () => {
264
- const val = ah.originalGetters[prop]();
265
- xhr.dispatchEvent(
266
- new CustomEvent("ajaxHooker_readResponse", {
267
- detail: { prop, val },
268
- })
269
- );
270
- return val;
271
- });
272
- });
273
- xhr.addEventListener("ajaxHooker_readResponse", (e) => {
274
- arg[e.detail.prop] = e.detail.val;
275
- });
276
- const resPromise = Promise.resolve(
277
- request.response(arg)
278
- ).then(() => {
279
- const task = [];
280
- xhrResponses.forEach((prop) => {
281
- const descriptor = Object.getOwnPropertyDescriptor(
282
- arg,
283
- prop
284
- );
285
- if (descriptor && "value" in descriptor) {
286
- task.push(
287
- Promise.resolve(descriptor.value).then((val) => {
288
- arg[prop] = val;
289
- defineProp(xhr, prop, () => {
290
- xhr.dispatchEvent(
291
- new CustomEvent("ajaxHooker_readResponse", {
292
- detail: { prop, val },
293
- })
294
- );
295
- return val;
296
- });
297
- }, emptyFn)
298
- );
299
- }
300
- });
301
- return Promise.all(task);
302
- }, errorFn);
303
- const eventsClone = {};
304
- xhrAsyncEvents.forEach((type) => {
305
- eventsClone[type] = new Set([...ah.hookedEvents[type]]);
306
- eventsClone["on" + type] = ah.hookedEvents["on" + type];
307
- });
308
- ah.delegateEvent = (event) =>
309
- resPromise.then(() => {
310
- event.stopImmediatePropagation = fakeEventSIP;
311
- eventsClone[event.type].forEach(
312
- (fn) =>
313
- !event.ajaxHooker_stopped && fn.call(xhr, event)
314
- );
315
- const onEvent = eventsClone["on" + event.type];
316
- typeof onEvent === "function" &&
317
- onEvent.call(xhr, event);
318
- });
319
- }
320
- } catch (err) {
321
- console.error(err);
322
- }
323
- ah.delegateEvent(e.detail);
324
- });
325
- realSend(data);
326
- });
327
- });
328
- } catch (err) {
329
- console.error(err);
330
- realSend(data);
331
- }
332
- };
333
- return xhr;
334
- }
335
- function hookFetchResponse(response, arg, callback) {
336
- fetchResponses.forEach((prop) => {
337
- response[prop] = () =>
338
- new Promise((resolve, reject) => {
339
- resProto[prop].call(response).then((res) => {
340
- if (prop in arg) {
341
- resolve(arg[prop]);
342
- } else {
343
- try {
344
- arg[prop] = res;
345
- Promise.resolve(callback(arg)).then(() => {
346
- if (prop in arg) {
347
- Promise.resolve(arg[prop]).then(
348
- (val) => resolve((arg[prop] = val)),
349
- () => resolve(res)
350
- );
351
- } else {
352
- resolve(res);
353
- }
354
- }, errorFn);
355
- } catch (err) {
356
- console.error(err);
357
- resolve(res);
358
- }
359
- }
360
- }, reject);
361
- });
362
- });
363
- }
364
- function fakeFetch(url, init) {
365
- if (url && typeof url.toString === "function") {
366
- url = url.toString();
367
- init = init || {};
368
- init.method = init.method || "GET";
369
- init.headers = init.headers || {};
370
- if (shouldFilter("fetch", url, init.method))
371
- return realFetch.call(win, url, init);
372
- const request = {
373
- type: "fetch",
374
- url: url,
375
- method: init.method.toUpperCase(),
376
- abort: false,
377
- headers: {},
378
- data: init.body,
379
- response: null,
380
- };
381
- if (toString.call(init.headers) === "[object Headers]") {
382
- for (const [key, val] of init.headers) {
383
- request.headers[key] = val;
384
- }
385
- } else {
386
- request.headers = { ...init.headers };
387
- }
388
- const requestClone = { ...request };
389
- return new Promise((resolve, reject) => {
390
- try {
391
- waitForHookFns(request).then(() => {
392
- waitForRequestKeys(request, requestClone).then(() => {
393
- if (request.abort) return reject("aborted");
394
- url = request.url;
395
- init.method = request.method;
396
- init.headers = request.headers;
397
- init.body = request.data;
398
- realFetch.call(win, url, init).then((response) => {
399
- if (typeof request.response === "function") {
400
- const arg = {
401
- finalUrl: response.url,
402
- status: response.status,
403
- responseHeaders: {},
404
- };
405
- for (const [key, val] of response.headers) {
406
- arg.responseHeaders[key] = val;
407
- }
408
- hookFetchResponse(response, arg, request.response);
409
- response.clone = () => {
410
- const resClone = resProto.clone.call(response);
411
- hookFetchResponse(resClone, arg, request.response);
412
- return resClone;
413
- };
414
- }
415
- resolve(response);
416
- }, reject);
417
- });
418
- });
419
- } catch (err) {
420
- console.error(err);
421
- return realFetch.call(win, url, init);
422
- }
423
- });
424
- } else {
425
- return realFetch.call(win, url, init);
426
- }
427
- }
428
- win.XMLHttpRequest = fakeXhr;
429
- Object.keys(realXhr).forEach((key) => (fakeXhr[key] = realXhr[key]));
430
- fakeXhr.prototype = realXhr.prototype;
431
- win.fetch = fakeFetch;
432
- return {
433
- hook: (fn) => hookFns.push(fn),
434
- filter: (arr) => {
435
- filter = Array.isArray(arr) && arr.map(toFilterObj);
436
- },
437
- protect: () => {
438
- readonly(win, "XMLHttpRequest", fakeXhr);
439
- readonly(win, "fetch", fakeFetch);
440
- },
441
- unhook: () => {
442
- writable(win, "XMLHttpRequest", realXhr);
443
- writable(win, "fetch", realFetch);
444
- },
445
- };
446
- })();
9
+ return (function () {
10
+ const win = window.unsafeWindow || document.defaultView || window;
11
+ const hookFns = [];
12
+ const realXhr = win.XMLHttpRequest;
13
+ const resProto = win.Response.prototype;
14
+ const toString = Object.prototype.toString;
15
+ const realFetch = win.fetch;
16
+ const xhrResponses = ["response", "responseText", "responseXML"];
17
+ const fetchResponses = ["arrayBuffer", "blob", "formData", "json", "text"];
18
+ const xhrAsyncEvents = ["readystatechange", "load", "loadend"];
19
+ let filter;
20
+ function emptyFn() {}
21
+ function errorFn(err) {
22
+ console.error(err);
23
+ }
24
+ function defineProp(obj, prop, getter, setter) {
25
+ Object.defineProperty(obj, prop, {
26
+ configurable: true,
27
+ enumerable: true,
28
+ get: getter,
29
+ set: setter,
30
+ });
31
+ }
32
+ function readonly(obj, prop, value = obj[prop]) {
33
+ defineProp(obj, prop, () => value, emptyFn);
34
+ }
35
+ function writable(obj, prop, value = obj[prop]) {
36
+ Object.defineProperty(obj, prop, {
37
+ configurable: true,
38
+ enumerable: true,
39
+ writable: true,
40
+ value: value,
41
+ });
42
+ }
43
+ function toFilterObj(obj) {
44
+ return {
45
+ type: obj.type,
46
+ url: obj.url,
47
+ method: obj.method && obj.method.toUpperCase(),
48
+ };
49
+ }
50
+ function shouldFilter(type, url, method) {
51
+ return (
52
+ filter &&
53
+ !filter.find(
54
+ (obj) =>
55
+ (!obj.type || obj.type === type) &&
56
+ (!obj.url ||
57
+ (toString.call(obj.url) === "[object String]"
58
+ ? url.includes(obj.url)
59
+ : obj.url.test(url))) &&
60
+ (!obj.method || obj.method === method.toUpperCase()),
61
+ )
62
+ );
63
+ }
64
+ function lookupGetter(obj, prop) {
65
+ let getter;
66
+ let proto = obj;
67
+ while (proto) {
68
+ const descriptor = Object.getOwnPropertyDescriptor(proto, prop);
69
+ getter = descriptor && descriptor.get;
70
+ if (getter) break;
71
+ proto = Object.getPrototypeOf(proto);
72
+ }
73
+ return getter ? getter.bind(obj) : emptyFn;
74
+ }
75
+ function waitForHookFns(request) {
76
+ return Promise.all(
77
+ hookFns.map((fn) =>
78
+ Promise.resolve(fn(request)).then(emptyFn, errorFn),
79
+ ),
80
+ );
81
+ }
82
+ function waitForRequestKeys(request, requestClone) {
83
+ return Promise.all(
84
+ ["url", "method", "abort", "headers", "data"].map((key) => {
85
+ return Promise.resolve(request[key]).then(
86
+ (val) => (request[key] = val),
87
+ () => (request[key] = requestClone[key]),
88
+ );
89
+ }),
90
+ );
91
+ }
92
+ function fakeEventSIP() {
93
+ this.ajaxHooker_stopped = true;
94
+ }
95
+ function xhrDelegateEvent(e) {
96
+ const xhr = e.target;
97
+ e.stopImmediatePropagation = fakeEventSIP;
98
+ xhr.__ajaxHooker.hookedEvents[e.type].forEach(
99
+ (fn) => !e.ajaxHooker_stopped && fn.call(xhr, e),
100
+ );
101
+ const onEvent = xhr.__ajaxHooker.hookedEvents["on" + e.type];
102
+ typeof onEvent === "function" && onEvent.call(xhr, e);
103
+ }
104
+ function xhrReadyStateChange(e) {
105
+ if (e.target.readyState === 4) {
106
+ e.target.dispatchEvent(
107
+ new CustomEvent("ajaxHooker_responseReady", { detail: e }),
108
+ );
109
+ } else {
110
+ e.target.__ajaxHooker.delegateEvent(e);
111
+ }
112
+ }
113
+ function xhrLoadAndLoadend(e) {
114
+ e.target.__ajaxHooker.delegateEvent(e);
115
+ }
116
+ function fakeXhrOpen(method, url, ...args) {
117
+ const ah = this.__ajaxHooker;
118
+ ah.url = url.toString();
119
+ ah.method = method.toUpperCase();
120
+ ah.openArgs = args;
121
+ ah.headers = {};
122
+ return ah.originalMethods.open(method, url, ...args);
123
+ }
124
+ function fakeXhr() {
125
+ const xhr = new realXhr();
126
+ let ah = xhr.__ajaxHooker;
127
+ if (!ah) {
128
+ ah = xhr.__ajaxHooker = {
129
+ headers: {},
130
+ hookedEvents: {
131
+ readystatechange: new Set(),
132
+ load: new Set(),
133
+ loadend: new Set(),
134
+ },
135
+ delegateEvent: xhrDelegateEvent,
136
+ originalGetters: {},
137
+ originalMethods: {},
138
+ };
139
+ xhr.addEventListener("readystatechange", xhrReadyStateChange);
140
+ xhr.addEventListener("load", xhrLoadAndLoadend);
141
+ xhr.addEventListener("loadend", xhrLoadAndLoadend);
142
+ for (const key of xhrResponses) {
143
+ ah.originalGetters[key] = lookupGetter(xhr, key);
144
+ }
145
+ for (const method of [
146
+ "open",
147
+ "setRequestHeader",
148
+ "addEventListener",
149
+ "removeEventListener",
150
+ ]) {
151
+ ah.originalMethods[method] = xhr[method].bind(xhr);
152
+ }
153
+ xhr.open = fakeXhrOpen;
154
+ xhr.setRequestHeader = (header, value) => {
155
+ ah.originalMethods.setRequestHeader(header, value);
156
+ if (xhr.readyState === 1) {
157
+ if (ah.headers[header]) {
158
+ ah.headers[header] += ", " + value;
159
+ } else {
160
+ ah.headers[header] = value;
161
+ }
162
+ }
163
+ };
164
+ xhr.addEventListener = function (...args) {
165
+ if (xhrAsyncEvents.includes(args[0])) {
166
+ ah.hookedEvents[args[0]].add(args[1]);
167
+ } else {
168
+ ah.originalMethods.addEventListener(...args);
169
+ }
170
+ };
171
+ xhr.removeEventListener = function (...args) {
172
+ if (xhrAsyncEvents.includes(args[0])) {
173
+ ah.hookedEvents[args[0]].delete(args[1]);
174
+ } else {
175
+ ah.originalMethods.removeEventListener(...args);
176
+ }
177
+ };
178
+ xhrAsyncEvents.forEach((evt) => {
179
+ const onEvt = "on" + evt;
180
+ defineProp(
181
+ xhr,
182
+ onEvt,
183
+ () => {
184
+ return ah.hookedEvents[onEvt] || null;
185
+ },
186
+ (val) => {
187
+ ah.hookedEvents[onEvt] = typeof val === "function" ? val : null;
188
+ },
189
+ );
190
+ });
191
+ }
192
+ const realSend = xhr.send.bind(xhr);
193
+ xhr.send = function (data) {
194
+ if (xhr.readyState !== 1) return realSend(data);
195
+ ah.delegateEvent = xhrDelegateEvent;
196
+ xhrResponses.forEach((prop) => {
197
+ delete xhr[prop]; // delete descriptor
198
+ });
199
+ if (shouldFilter("xhr", ah.url, ah.method)) {
200
+ xhr.addEventListener("ajaxHooker_responseReady", (e) => {
201
+ ah.delegateEvent(e.detail);
202
+ });
203
+ return realSend(data);
204
+ }
205
+ try {
206
+ const request = {
207
+ type: "xhr",
208
+ url: ah.url,
209
+ method: ah.method,
210
+ abort: false,
211
+ headers: ah.headers,
212
+ data: data,
213
+ response: null,
214
+ };
215
+ const requestClone = { ...request };
216
+ waitForHookFns(request).then(() => {
217
+ waitForRequestKeys(request, requestClone).then(() => {
218
+ if (request.abort) return;
219
+ ah.originalMethods.open(
220
+ request.method,
221
+ request.url,
222
+ ...ah.openArgs,
223
+ );
224
+ for (const header in request.headers) {
225
+ ah.originalMethods.setRequestHeader(
226
+ header,
227
+ request.headers[header],
228
+ );
229
+ }
230
+ data = request.data;
231
+ xhr.addEventListener("ajaxHooker_responseReady", (e) => {
232
+ try {
233
+ if (typeof request.response === "function") {
234
+ const arg = {
235
+ finalUrl: xhr.responseURL,
236
+ status: xhr.status,
237
+ responseHeaders: {},
238
+ };
239
+ for (const line of xhr
240
+ .getAllResponseHeaders()
241
+ .trim()
242
+ .split(/[\r\n]+/)) {
243
+ const parts = line.split(/:\s*/);
244
+ if (parts.length === 2) {
245
+ const lheader = parts[0].toLowerCase();
246
+ if (arg.responseHeaders[lheader]) {
247
+ arg.responseHeaders[lheader] += ", " + parts[1];
248
+ } else {
249
+ arg.responseHeaders[lheader] = parts[1];
250
+ }
251
+ }
252
+ }
253
+ xhrResponses.forEach((prop) => {
254
+ defineProp(
255
+ arg,
256
+ prop,
257
+ () => {
258
+ return (arg[prop] = ah.originalGetters[prop]());
259
+ },
260
+ (val) => {
261
+ delete arg[prop];
262
+ arg[prop] = val;
263
+ },
264
+ );
265
+ defineProp(xhr, prop, () => {
266
+ const val = ah.originalGetters[prop]();
267
+ xhr.dispatchEvent(
268
+ new CustomEvent("ajaxHooker_readResponse", {
269
+ detail: { prop, val },
270
+ }),
271
+ );
272
+ return val;
273
+ });
274
+ });
275
+ xhr.addEventListener("ajaxHooker_readResponse", (e) => {
276
+ arg[e.detail.prop] = e.detail.val;
277
+ });
278
+ const resPromise = Promise.resolve(
279
+ request.response(arg),
280
+ ).then(() => {
281
+ const task = [];
282
+ xhrResponses.forEach((prop) => {
283
+ const descriptor = Object.getOwnPropertyDescriptor(
284
+ arg,
285
+ prop,
286
+ );
287
+ if (descriptor && "value" in descriptor) {
288
+ task.push(
289
+ Promise.resolve(descriptor.value).then((val) => {
290
+ arg[prop] = val;
291
+ defineProp(xhr, prop, () => {
292
+ xhr.dispatchEvent(
293
+ new CustomEvent("ajaxHooker_readResponse", {
294
+ detail: { prop, val },
295
+ }),
296
+ );
297
+ return val;
298
+ });
299
+ }, emptyFn),
300
+ );
301
+ }
302
+ });
303
+ return Promise.all(task);
304
+ }, errorFn);
305
+ const eventsClone = {};
306
+ xhrAsyncEvents.forEach((type) => {
307
+ eventsClone[type] = new Set([...ah.hookedEvents[type]]);
308
+ eventsClone["on" + type] = ah.hookedEvents["on" + type];
309
+ });
310
+ ah.delegateEvent = (event) =>
311
+ resPromise.then(() => {
312
+ event.stopImmediatePropagation = fakeEventSIP;
313
+ eventsClone[event.type].forEach(
314
+ (fn) =>
315
+ !event.ajaxHooker_stopped && fn.call(xhr, event),
316
+ );
317
+ const onEvent = eventsClone["on" + event.type];
318
+ typeof onEvent === "function" &&
319
+ onEvent.call(xhr, event);
320
+ });
321
+ }
322
+ } catch (err) {
323
+ console.error(err);
324
+ }
325
+ ah.delegateEvent(e.detail);
326
+ });
327
+ realSend(data);
328
+ });
329
+ });
330
+ } catch (err) {
331
+ console.error(err);
332
+ realSend(data);
333
+ }
334
+ };
335
+ return xhr;
336
+ }
337
+ function hookFetchResponse(response, arg, callback) {
338
+ fetchResponses.forEach((prop) => {
339
+ response[prop] = () =>
340
+ new Promise((resolve, reject) => {
341
+ resProto[prop].call(response).then((res) => {
342
+ if (prop in arg) {
343
+ resolve(arg[prop]);
344
+ } else {
345
+ try {
346
+ arg[prop] = res;
347
+ Promise.resolve(callback(arg)).then(() => {
348
+ if (prop in arg) {
349
+ Promise.resolve(arg[prop]).then(
350
+ (val) => resolve((arg[prop] = val)),
351
+ () => resolve(res),
352
+ );
353
+ } else {
354
+ resolve(res);
355
+ }
356
+ }, errorFn);
357
+ } catch (err) {
358
+ console.error(err);
359
+ resolve(res);
360
+ }
361
+ }
362
+ }, reject);
363
+ });
364
+ });
365
+ }
366
+ function fakeFetch(url, init) {
367
+ if (url && typeof url.toString === "function") {
368
+ url = url.toString();
369
+ init = init || {};
370
+ init.method = init.method || "GET";
371
+ init.headers = init.headers || {};
372
+ if (shouldFilter("fetch", url, init.method))
373
+ return realFetch.call(win, url, init);
374
+ const request = {
375
+ type: "fetch",
376
+ url: url,
377
+ method: init.method.toUpperCase(),
378
+ abort: false,
379
+ headers: {},
380
+ data: init.body,
381
+ response: null,
382
+ };
383
+ if (toString.call(init.headers) === "[object Headers]") {
384
+ for (const [key, val] of init.headers) {
385
+ request.headers[key] = val;
386
+ }
387
+ } else {
388
+ request.headers = { ...init.headers };
389
+ }
390
+ const requestClone = { ...request };
391
+ return new Promise((resolve, reject) => {
392
+ try {
393
+ waitForHookFns(request).then(() => {
394
+ waitForRequestKeys(request, requestClone).then(() => {
395
+ if (request.abort) return reject("aborted");
396
+ url = request.url;
397
+ init.method = request.method;
398
+ init.headers = request.headers;
399
+ init.body = request.data;
400
+ realFetch.call(win, url, init).then((response) => {
401
+ if (typeof request.response === "function") {
402
+ const arg = {
403
+ finalUrl: response.url,
404
+ status: response.status,
405
+ responseHeaders: {},
406
+ };
407
+ for (const [key, val] of response.headers) {
408
+ arg.responseHeaders[key] = val;
409
+ }
410
+ hookFetchResponse(response, arg, request.response);
411
+ response.clone = () => {
412
+ const resClone = resProto.clone.call(response);
413
+ hookFetchResponse(resClone, arg, request.response);
414
+ return resClone;
415
+ };
416
+ }
417
+ resolve(response);
418
+ }, reject);
419
+ });
420
+ });
421
+ } catch (err) {
422
+ console.error(err);
423
+ return realFetch.call(win, url, init);
424
+ }
425
+ });
426
+ } else {
427
+ return realFetch.call(win, url, init);
428
+ }
429
+ }
430
+ win.XMLHttpRequest = fakeXhr;
431
+ Object.keys(realXhr).forEach((key) => (fakeXhr[key] = realXhr[key]));
432
+ fakeXhr.prototype = realXhr.prototype;
433
+ win.fetch = fakeFetch;
434
+ return {
435
+ hook: (fn) => hookFns.push(fn),
436
+ filter: (arr) => {
437
+ filter = Array.isArray(arr) && arr.map(toFilterObj);
438
+ },
439
+ protect: () => {
440
+ readonly(win, "XMLHttpRequest", fakeXhr);
441
+ readonly(win, "fetch", fakeFetch);
442
+ },
443
+ unhook: () => {
444
+ writable(win, "XMLHttpRequest", realXhr);
445
+ writable(win, "fetch", realFetch);
446
+ },
447
+ };
448
+ })();
447
449
  };