Jest 與 puppeteer 一起使用

2021-09-04 13:41 更新

借助全局設(shè)置/拆卸異步測試環(huán)境API,Jest 可以與puppeteer順利工作。

如果測試使用?page.$eval?,?page.$$eval?或者?page.evaluate則當(dāng)前無法使用 Puppeteer 為測試文件生成代碼覆蓋率。因?yàn)閭鬟f的函數(shù)在 Jest 的作用域之外執(zhí)行。查看GitHub 上的問題 #7962以獲取解決方法。

使用 jest-puppeteer 預(yù)設(shè)

Jest Puppeteer提供了使用 Puppeteer 運(yùn)行測試所需的所有配置。首先,安裝 jest-puppeteer

  1. yarn add --dev jest-puppeteer

在 Jest 配置中指定預(yù)設(shè):

  1. {
  2. "preset": "jest-puppeteer"
  3. }

寫你的測試:

  1. describe('Google', () => {
  2. beforeAll(async () => {
  3. await page.goto('https://google.com');
  4. });
  5. it('should be titled "Google"', async () => {
  6. await expect(page.title()).resolves.toMatch('Google');
  7. });
  8. });

無需加載任何依賴項(xiàng)。Puppeteer 的?page?和?browser?類將自動公開

請參閱文檔

沒有 jest-puppeteer 預(yù)設(shè)的自定義示例

可以從頭開始連接 puppeteer?;舅枷胧牵?/p>

  1. 使用全局設(shè)置啟動并歸檔 puppeteer 的 websocket 端點(diǎn)
  2. 從每個(gè)測試環(huán)境連接到 puppeteer
  3. 使用 Global Teardown 關(guān)閉 puppeteer

這是 GlobalSetup 腳本的示例

  1. // setup.js
  2. const path = require('path');
  3. const fs = require('fs');
  4. const os = require('os');
  5. const mkdirp = require('mkdirp');
  6. const puppeteer = require('puppeteer');
  7. const DIR = path.join(os.tmpdir(), 'jest_puppeteer_global_setup');
  8. module.exports = async function () {
  9. const browser = await puppeteer.launch();
  10. // store the browser instance so we can teardown it later
  11. // this global is only available in the teardown but not in TestEnvironments
  12. global.__BROWSER_GLOBAL__ = browser;
  13. // use the file system to expose the wsEndpoint for TestEnvironments
  14. mkdirp.sync(DIR);
  15. fs.writeFileSync(path.join(DIR, 'wsEndpoint'), browser.wsEndpoint());
  16. };

然后我們需要為 puppeteer 定制一個(gè)測試環(huán)境

  1. // puppeteer_environment.js
  2. const fs = require('fs');
  3. const path = require('path');
  4. const os = require('os');
  5. const puppeteer = require('puppeteer');
  6. const NodeEnvironment = require('jest-environment-node');
  7. const DIR = path.join(os.tmpdir(), 'jest_puppeteer_global_setup');
  8. class PuppeteerEnvironment extends NodeEnvironment {
  9. constructor(config) {
  10. super(config);
  11. }
  12. async setup() {
  13. await super.setup();
  14. // get the wsEndpoint
  15. const wsEndpoint = fs.readFileSync(path.join(DIR, 'wsEndpoint'), 'utf8');
  16. if (!wsEndpoint) {
  17. throw new Error('wsEndpoint not found');
  18. }
  19. // connect to puppeteer
  20. this.global.__BROWSER__ = await puppeteer.connect({
  21. browserWSEndpoint: wsEndpoint,
  22. });
  23. }
  24. async teardown() {
  25. await super.teardown();
  26. }
  27. runScript(script) {
  28. return super.runScript(script);
  29. }
  30. }
  31. module.exports = PuppeteerEnvironment;

最后,我們可以關(guān)閉 puppeteer 實(shí)例并清理文件

  1. // teardown.js
  2. const os = require('os');
  3. const path = require('path');
  4. const rimraf = require('rimraf');
  5. const DIR = path.join(os.tmpdir(), 'jest_puppeteer_global_setup');
  6. module.exports = async function () {
  7. // close the browser instance
  8. await global.__BROWSER_GLOBAL__.close();
  9. // clean-up the wsEndpoint file
  10. rimraf.sync(DIR);
  11. };

完成所有設(shè)置后,我們現(xiàn)在可以像這樣編寫測試:

  1. // test.js
  2. const timeout = 5000;
  3. describe(
  4. '/ (Home Page)',
  5. () => {
  6. let page;
  7. beforeAll(async () => {
  8. page = await global.__BROWSER__.newPage();
  9. await page.goto('https://google.com');
  10. }, timeout);
  11. it('should load without error', async () => {
  12. const text = await page.evaluate(() => document.body.textContent);
  13. expect(text).toContain('google');
  14. });
  15. },
  16. timeout,
  17. );

最后,設(shè)置?jest.config.js?為從這些文件中讀取。(?jest-puppeteer?預(yù)設(shè)在幕后做了類似的事情。)

  1. module.exports = {
  2. globalSetup: './setup.js',
  3. globalTeardown: './teardown.js',
  4. testEnvironment: './puppeteer_environment.js',
  5. };

這是完整工作示例的代碼。


以上內(nèi)容是否對您有幫助:
在線筆記
App下載
App下載

掃描二維碼

下載編程獅App

公眾號
微信公眾號

編程獅公眾號