Выполнение JavaScript
Введение
Скрипты Playwright выполняются в вашей среде Playwright. Скрипты страницы выполняются в среде браузерной страницы. Эти среды не пересекаются, они работают в разных виртуальных машинах в разных процессах и даже потенциально на разных компьютерах.
API page.evaluate() может выполнять JavaScript-функцию в контексте веб-страницы и возвращать результаты в среду Playwright. Глобальные объекты браузера, такие как window
и document
, могут использоваться в evaluate
.
const href = await page.evaluate(() => document.location.href);
Если результат является Promise или если функция асинхронная, evaluate автоматически дождется его разрешения:
const status = await page.evaluate(async () => {
const response = await fetch(location.href);
return response.status;
});
Разные среды
Выполняемые скрипты работают в среде браузера, в то время как ваш тест выполняется в тестовой среде. Это означает, что вы не можете использовать переменные из вашего теста на странице и наоборот. Вместо этого вы должны передавать их явно в качестве аргумента.
Следующий фрагмент НЕПРАВИЛЬНЫЙ, потому что он использует переменную напрямую:
const data = 'some data';
const result = await page.evaluate(() => {
// НЕПРАВИЛЬНО: на веб-странице нет "data".
window.myApp.use(data);
});
Следующий фрагмент ПРАВИЛЬНЫЙ, потому что он передает значение явно в качестве аргумента:
const data = 'some data';
// Передаем |data| как параметр.
const result = await page.evaluate(data => {
window.myApp.use(data);
}, data);
Аргумент для выполнения
Методы выполнения Playwright, такие как page.evaluate(), принимают один необязательный аргумент. Этот аргумент может быть смесью значений Serializable и экземпляров JSHandle. Обработчики автоматически преобразуются в представляемое ими значение.
// Примитивное значение.
await page.evaluate(num => num, 42);
// Массив.
await page.evaluate(array => array.length, [1, 2, 3]);
// Объект.
await page.evaluate(object => object.foo, { foo: 'bar' });
// Один обработчик.
const button = await page.evaluateHandle('window.button');
await page.evaluate(button => button.textContent, button);
// Альтернативная нотация с использованием JSHandle.evaluate.
await button.evaluate((button, from) => button.textContent.substring(from), 5);
// Объект с несколькими обработчиками.
const button1 = await page.evaluateHandle('window.button1');
const button2 = await page.evaluateHandle('window.button2');
await page.evaluate(
o => o.button1.textContent + o.button2.textContent,
{ button1, button2 });
// Деструктуризация объекта работает. Обратите внимание, что имена свойств должны совпадать
// между деструктурированным объектом и аргументом.
// Также обратите внимание на обязательные скобки.
await page.evaluate(
({ button1, button2 }) => button1.textContent + button2.textContent,
{ button1, button2 });
// Массив также работает. Можно использовать произвольные имена для деструктуризации.
// Обратите внимание на обязательные скобки.
await page.evaluate(
([b1, b2]) => b1.textContent + b2.textContent,
[button1, button2]);
// Любая смесь сериализуемых и обработчиков работает.
await page.evaluate(
x => x.button1.textContent + x.list[0].textContent + String(x.foo),
{ button1, list: [button2], foo: null });
Скрипты инициализации
Иногда удобно выполнить что-то на странице до того, как она начнет загружаться. Например, вы можете захотеть настроить некоторые моки или тестовые данные.
В этом случае используйте page.addInitScript() или browserContext.addInitScript(). В примере ниже мы заменим Math.random()
на постоянное значение.
Сначала создайте файл preload.js
, содержащий мок.
// preload.js
Math.random = () => 42;
Затем добавьте скрипт инициализации на страницу.
import { test, expect } from '@playwright/test';
import path from 'path';
test.beforeEach(async ({ page }) => {
// Добавьте скрипт для каждого теста в хуке beforeEach.
// Убедитесь, что путь к скрипту указан правильно.
await page.addInitScript({ path: path.resolve(__dirname, '../mocks/preload.js') });
});
В качестве альтернативы, вы можете передать функцию вместо создания файла скрипта предварительной загрузки. Это более удобно для коротких или одноразовых скриптов. Вы также можете передать аргумент таким образом.
import { test, expect } from '@playwright/test';
// Добавьте скрипт для каждого теста в хуке beforeEach.
test.beforeEach(async ({ page }) => {
const value = 42;
await page.addInitScript(value => {
Math.random = () => value;
}, value);
});