2020-05-21 00:55:33 +02:00
|
|
|
/**
|
|
|
|
|
* Copyright (c) Microsoft Corporation.
|
|
|
|
|
*
|
|
|
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
|
* you may not use this file except in compliance with the License.
|
|
|
|
|
* You may obtain a copy of the License at
|
|
|
|
|
*
|
|
|
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
|
*
|
|
|
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
|
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
|
* See the License for the specific language governing permissions and
|
|
|
|
|
* limitations under the License.
|
|
|
|
|
*/
|
|
|
|
|
|
2022-04-08 06:48:41 +02:00
|
|
|
import { serializeAsCallArgument, parseEvaluationResultValue } from '../isomorphic/utilityScriptSerializers';
|
2020-05-28 02:19:05 +02:00
|
|
|
|
2022-03-29 08:10:17 +02:00
|
|
|
export class UtilityScript {
|
2024-05-31 23:44:26 +02:00
|
|
|
constructor(isUnderTest: boolean) {
|
|
|
|
|
if (isUnderTest)
|
|
|
|
|
this._setBuiltins();
|
|
|
|
|
}
|
|
|
|
|
|
2022-11-30 01:57:11 +01:00
|
|
|
serializeAsCallArgument = serializeAsCallArgument;
|
|
|
|
|
parseEvaluationResultValue = parseEvaluationResultValue;
|
|
|
|
|
|
2024-05-31 23:44:26 +02:00
|
|
|
evaluate(isFunction: boolean | undefined, returnByValue: boolean, expression: string, argCount: number, ...argsAndHandles: any[]) {
|
2020-07-20 04:46:19 +02:00
|
|
|
const args = argsAndHandles.slice(0, argCount);
|
|
|
|
|
const handles = argsAndHandles.slice(argCount);
|
2023-04-21 19:52:13 +02:00
|
|
|
const parameters = [];
|
|
|
|
|
for (let i = 0; i < args.length; i++)
|
|
|
|
|
parameters[i] = this.parseEvaluationResultValue(args[i], handles);
|
2022-11-30 01:57:11 +01:00
|
|
|
|
2023-02-17 20:19:53 +01:00
|
|
|
// eslint-disable-next-line no-restricted-globals
|
2022-03-29 08:10:17 +02:00
|
|
|
let result = globalThis.eval(expression);
|
2021-02-03 22:49:25 +01:00
|
|
|
if (isFunction === true) {
|
|
|
|
|
result = result(...parameters);
|
|
|
|
|
} else if (isFunction === false) {
|
|
|
|
|
result = result;
|
|
|
|
|
} else {
|
|
|
|
|
// auto detect.
|
|
|
|
|
if (typeof result === 'function')
|
|
|
|
|
result = result(...parameters);
|
|
|
|
|
}
|
2020-05-28 02:19:05 +02:00
|
|
|
return returnByValue ? this._promiseAwareJsonValueNoThrow(result) : result;
|
2020-05-22 01:00:55 +02:00
|
|
|
}
|
|
|
|
|
|
2020-05-28 02:19:05 +02:00
|
|
|
jsonValue(returnByValue: true, value: any) {
|
2020-06-26 18:50:57 +02:00
|
|
|
// Special handling of undefined to work-around multi-step returnByValue handling in WebKit.
|
|
|
|
|
if (Object.is(value, undefined))
|
|
|
|
|
return undefined;
|
2020-05-28 02:19:05 +02:00
|
|
|
return serializeAsCallArgument(value, (value: any) => ({ fallThrough: value }));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private _promiseAwareJsonValueNoThrow(value: any) {
|
|
|
|
|
const safeJson = (value: any) => {
|
|
|
|
|
try {
|
|
|
|
|
return this.jsonValue(true, value);
|
|
|
|
|
} catch (e) {
|
|
|
|
|
return undefined;
|
2020-05-22 01:00:55 +02:00
|
|
|
}
|
2020-05-28 02:19:05 +02:00
|
|
|
};
|
|
|
|
|
|
2020-06-30 01:25:52 +02:00
|
|
|
if (value && typeof value === 'object' && typeof value.then === 'function') {
|
|
|
|
|
return (async () => {
|
|
|
|
|
// By using async function we ensure that return value is a native Promise,
|
2021-02-25 00:11:34 +01:00
|
|
|
// and not some overridden Promise in the page.
|
2020-06-30 01:25:52 +02:00
|
|
|
// This makes Firefox and WebKit debugging protocols recognize it as a Promise,
|
|
|
|
|
// properly await and return the value.
|
|
|
|
|
const promiseValue = await value;
|
|
|
|
|
return safeJson(promiseValue);
|
|
|
|
|
})();
|
|
|
|
|
}
|
2020-05-28 02:19:05 +02:00
|
|
|
return safeJson(value);
|
2020-05-21 00:55:33 +02:00
|
|
|
}
|
2024-05-31 23:44:26 +02:00
|
|
|
|
|
|
|
|
private _setBuiltins() {
|
|
|
|
|
// eslint-disable-next-line no-restricted-globals
|
|
|
|
|
const window = (globalThis as any);
|
|
|
|
|
window.builtinSetTimeout = (callback: Function, timeout: number) => {
|
|
|
|
|
if (window.__pwFakeTimers?.builtin)
|
|
|
|
|
return window.__pwFakeTimers.builtin.setTimeout(callback, timeout);
|
|
|
|
|
return setTimeout(callback, timeout);
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
window.builtinClearTimeout = (id: number) => {
|
|
|
|
|
if (window.__pwFakeTimers?.builtin)
|
|
|
|
|
return window.__pwFakeTimers.builtin.clearTimeout(id);
|
|
|
|
|
return clearTimeout(id);
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
window.builtinSetInterval = (callback: Function, timeout: number) => {
|
|
|
|
|
if (window.__pwFakeTimers?.builtin)
|
|
|
|
|
return window.__pwFakeTimers.builtin.setInterval(callback, timeout);
|
|
|
|
|
return setInterval(callback, timeout);
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
window.builtinClearInterval = (id: number) => {
|
|
|
|
|
if (window.__pwFakeTimers?.builtin)
|
|
|
|
|
return window.__pwFakeTimers.builtin.clearInterval(id);
|
|
|
|
|
return clearInterval(id);
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
window.builtinRequestAnimationFrame = (callback: FrameRequestCallback) => {
|
|
|
|
|
if (window.__pwFakeTimers?.builtin)
|
|
|
|
|
return window.__pwFakeTimers.builtin.requestAnimationFrame(callback);
|
|
|
|
|
return requestAnimationFrame(callback);
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
window.builtinCancelAnimationFrame = (id: number) => {
|
|
|
|
|
if (window.__pwFakeTimers?.builtin)
|
|
|
|
|
return window.__pwFakeTimers.builtin.cancelAnimationFrame(id);
|
|
|
|
|
return cancelAnimationFrame(id);
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
window.builtinDate = window.__pwFakeTimers?.builtin.Date || Date;
|
|
|
|
|
window.builtinPerformance = window.__pwFakeTimers?.builtin.performance || performance;
|
|
|
|
|
}
|
2020-05-21 00:55:33 +02:00
|
|
|
}
|