评估 JavaScript
介绍
¥Introduction
Playwright 脚本在你的 Playwright 环境中运行。你的页面脚本在浏览器页面环境中运行。这些环境并不交叉,它们运行在不同进程的不同虚拟机中,甚至可能运行在不同的计算机上。
¥Playwright scripts run in your Playwright environment. Your page scripts run in the browser page environment. Those environments don't intersect, they are running in different virtual machines in different processes and even potentially on different computers.
page.evaluate() API 可以在网页上下文中运行 JavaScript 函数,并将结果带回 Playwright 环境。像 window
和 document
这样的浏览器全局变量可以在 evaluate
中使用。
¥The page.evaluate() API can run a JavaScript function in the context of the web page and bring results back to the Playwright environment. Browser globals like window
and document
can be used in evaluate
.
const href = await page.evaluate(() => document.location.href);
如果结果是 Promise 或者函数是异步的,则评估将自动等待直到解决:
¥If the result is a Promise or if the function is asynchronous evaluate will automatically wait until it's resolved:
const status = await page.evaluate(async () => {
const response = await fetch(location.href);
return response.status;
});
不同的环境
¥Different environments
评估脚本在浏览器环境中运行,而测试在测试环境中运行。这意味着你不能在页面中使用测试中的变量,反之亦然。相反,你应该将它们明确作为参数传递。
¥Evaluated scripts run in the browser environment, while your test runs in a testing environments. This means you cannot use variables from your test in the page and vice versa. Instead, you should pass them explicitly as an argument.
以下代码片段是错误的,因为它直接使用变量:
¥The following snippet is WRONG because it uses the variable directly:
const data = 'some data';
const result = await page.evaluate(() => {
// WRONG: there is no "data" in the web page.
window.myApp.use(data);
});
以下代码片段是正确的,因为它将值明确作为参数传递:
¥The following snippet is CORRECT because it passes the value explicitly as an argument:
const data = 'some data';
// Pass |data| as a parameter.
const result = await page.evaluate(data => {
window.myApp.use(data);
}, data);
评价参数
¥Evaluation Argument
像 page.evaluate() 这样的 Playwright 评估方法采用单个可选参数。此参数可以是 可串行化 值和 JSHandle 实例的混合。句柄会自动转换为其代表的值。
¥Playwright evaluation methods like page.evaluate() take a single optional argument. This argument can be a mix of Serializable values and JSHandle instances. Handles are automatically converted to the value they represent.
// A primitive value.
await page.evaluate(num => num, 42);
// An array.
await page.evaluate(array => array.length, [1, 2, 3]);
// An object.
await page.evaluate(object => object.foo, { foo: 'bar' });
// A single handle.
const button = await page.evaluateHandle('window.button');
await page.evaluate(button => button.textContent, button);
// Alternative notation using JSHandle.evaluate.
await button.evaluate((button, from) => button.textContent.substring(from), 5);
// Object with multiple handles.
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 });
// Object destructuring works. Note that property names must match
// between the destructured object and the argument.
// Also note the required parenthesis.
await page.evaluate(
({ button1, button2 }) => button1.textContent + button2.textContent,
{ button1, button2 });
// Array works as well. Arbitrary names can be used for destructuring.
// Note the required parenthesis.
await page.evaluate(
([b1, b2]) => b1.textContent + b2.textContent,
[button1, button2]);
// Any mix of serializables and handles works.
await page.evaluate(
x => x.button1.textContent + x.list[0].textContent + String(x.foo),
{ button1, list: [button2], foo: null });
初始化脚本
¥Init scripts
有时在页面开始加载之前评估页面中的某些内容很方便。例如,你可能需要设置一些模拟或测试数据。
¥Sometimes it is convenient to evaluate something in the page before it starts loading. For example, you might want to setup some mocks or test data.
在这种情况下,使用 page.addInitScript() 或 browserContext.addInitScript()。在下面的示例中,我们将用常量值替换 Math.random()
。
¥In this case, use page.addInitScript() or browserContext.addInitScript(). In the example below, we will replace Math.random()
with a constant value.
首先,创建一个包含模拟的 preload.js
文件。
¥First, create a preload.js
file that contains the mock.
// preload.js
Math.random = () => 42;
接下来,将初始化脚本添加到页面。
¥Next, add init script to the page.
import { test, expect } from '@playwright/test';
import path from 'path';
test.beforeEach(async ({ page }) => {
// Add script for every test in the beforeEach hook.
// Make sure to correctly resolve the script path.
await page.addInitScript({ path: path.resolve(__dirname, '../mocks/preload.js') });
});
或者,你可以传递一个函数,而不是创建预加载脚本文件。这对于简短或一次性脚本更方便。你也可以通过这种方式传递参数。
¥Alternatively, you can pass a function instead of creating a preload script file. This is more convenient for short or one-off scripts. You can also pass an argument this way.
import { test, expect } from '@playwright/test';
// Add script for every test in the beforeEach hook.
test.beforeEach(async ({ page }) => {
const value = 42;
await page.addInitScript(value => {
Math.random = () => value;
}, value);
});