mirror of
https://github.com/microsoft/playwright.git
synced 2025-06-26 21:40:17 +00:00
142 lines
4.9 KiB
TypeScript
142 lines
4.9 KiB
TypeScript
/**
|
|
* 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.
|
|
*/
|
|
|
|
import type { BrowserContext } from './browserContext';
|
|
import * as clockSource from '../generated/clockSource';
|
|
|
|
export class Clock {
|
|
private _browserContext: BrowserContext;
|
|
private _scriptInstalled = false;
|
|
|
|
constructor(browserContext: BrowserContext) {
|
|
this._browserContext = browserContext;
|
|
}
|
|
|
|
markAsUninstalled() {
|
|
this._scriptInstalled = false;
|
|
}
|
|
|
|
async fastForward(ticks: number | string) {
|
|
await this._installIfNeeded();
|
|
const ticksMillis = parseTicks(ticks);
|
|
await this._browserContext.addInitScript(`globalThis.__pwClock.controller.log('fastForward', ${Date.now()}, ${ticksMillis})`);
|
|
await this._evaluateInFrames(`globalThis.__pwClock.controller.fastForward(${ticksMillis})`);
|
|
}
|
|
|
|
async install(time: number | string | undefined) {
|
|
await this._installIfNeeded();
|
|
const timeMillis = time !== undefined ? parseTime(time) : Date.now();
|
|
await this._browserContext.addInitScript(`globalThis.__pwClock.controller.log('install', ${Date.now()}, ${timeMillis})`);
|
|
await this._evaluateInFrames(`globalThis.__pwClock.controller.install(${timeMillis})`);
|
|
}
|
|
|
|
async pauseAt(ticks: number | string) {
|
|
await this._installIfNeeded();
|
|
const timeMillis = parseTime(ticks);
|
|
await this._browserContext.addInitScript(`globalThis.__pwClock.controller.log('pauseAt', ${Date.now()}, ${timeMillis})`);
|
|
await this._evaluateInFrames(`globalThis.__pwClock.controller.pauseAt(${timeMillis})`);
|
|
}
|
|
|
|
async resume() {
|
|
await this._installIfNeeded();
|
|
await this._browserContext.addInitScript(`globalThis.__pwClock.controller.log('resume', ${Date.now()})`);
|
|
await this._evaluateInFrames(`globalThis.__pwClock.controller.resume()`);
|
|
}
|
|
|
|
async setFixedTime(time: string | number) {
|
|
await this._installIfNeeded();
|
|
const timeMillis = parseTime(time);
|
|
await this._browserContext.addInitScript(`globalThis.__pwClock.controller.log('setFixedTime', ${Date.now()}, ${timeMillis})`);
|
|
await this._evaluateInFrames(`globalThis.__pwClock.controller.setFixedTime(${timeMillis})`);
|
|
}
|
|
|
|
async setSystemTime(time: string | number) {
|
|
await this._installIfNeeded();
|
|
const timeMillis = parseTime(time);
|
|
await this._browserContext.addInitScript(`globalThis.__pwClock.controller.log('setSystemTime', ${Date.now()}, ${timeMillis})`);
|
|
await this._evaluateInFrames(`globalThis.__pwClock.controller.setSystemTime(${timeMillis})`);
|
|
}
|
|
|
|
async runFor(ticks: number | string) {
|
|
await this._installIfNeeded();
|
|
const ticksMillis = parseTicks(ticks);
|
|
await this._browserContext.addInitScript(`globalThis.__pwClock.controller.log('runFor', ${Date.now()}, ${ticksMillis})`);
|
|
await this._evaluateInFrames(`globalThis.__pwClock.controller.runFor(${ticksMillis})`);
|
|
}
|
|
|
|
private async _installIfNeeded() {
|
|
if (this._scriptInstalled)
|
|
return;
|
|
this._scriptInstalled = true;
|
|
const script = `(() => {
|
|
const module = {};
|
|
${clockSource.source}
|
|
globalThis.__pwClock = (module.exports.inject())(globalThis);
|
|
})();`;
|
|
await this._browserContext.addInitScript(script);
|
|
await this._evaluateInFrames(script);
|
|
}
|
|
|
|
private async _evaluateInFrames(script: string) {
|
|
await this._browserContext.safeNonStallingEvaluateInAllFrames(script, 'main', { throwOnJSErrors: true });
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Parse strings like '01:10:00' (meaning 1 hour, 10 minutes, 0 seconds) into
|
|
* number of milliseconds. This is used to support human-readable strings passed
|
|
* to clock.tick()
|
|
*/
|
|
function parseTicks(value: number | string): number {
|
|
if (typeof value === 'number')
|
|
return value;
|
|
if (!value)
|
|
return 0;
|
|
const str = value;
|
|
|
|
const strings = str.split(':');
|
|
const l = strings.length;
|
|
let i = l;
|
|
let ms = 0;
|
|
let parsed;
|
|
|
|
if (l > 3 || !/^(\d\d:){0,2}\d\d?$/.test(str)) {
|
|
throw new Error(
|
|
`Clock only understands numbers, 'mm:ss' and 'hh:mm:ss'`,
|
|
);
|
|
}
|
|
|
|
while (i--) {
|
|
parsed = parseInt(strings[i], 10);
|
|
if (parsed >= 60)
|
|
throw new Error(`Invalid time ${str}`);
|
|
ms += parsed * Math.pow(60, l - i - 1);
|
|
}
|
|
|
|
return ms * 1000;
|
|
}
|
|
|
|
function parseTime(epoch: string | number | undefined): number {
|
|
if (!epoch)
|
|
return 0;
|
|
if (typeof epoch === 'number')
|
|
return epoch;
|
|
const parsed = new Date(epoch);
|
|
if (!isFinite(parsed.getTime()))
|
|
throw new Error(`Invalid date: ${epoch}`);
|
|
return parsed.getTime();
|
|
}
|