Testing Electron apps with Playwright and GitHub Actions
Yesterday I figured out (issue 133) how to use Playwright to run tests against my Electron app, and then execute those tests in CI using GitHub Actions, for my datasett-app repo for my Datasette Desktop macOS application.
Installing @playwright/test
You need to install the @playwright/test package. You can do that like so:
npm i -D @playwright/testThis adds it to devDependencies in your package.json, something like this:
"devDependencies": { "@playwright/test": "^1.23.2",Writing a test
I dropped the following into a test/spec.mjs file:
import { test, expect } from '@playwright/test';import { _electron } from 'playwright';
test('App launches and quits', async () => { const app = await _electron.launch({args: ['main.js']); const window = await app.firstWindow(); await expect(await window.title()).toContain('Loading'); await app.close();});The .mjs extension is necessary in order to use import, since it lets Node.js know that this file is a JavaScript module.
The test can be run using playwright test.
I later added it to my package.json section like this:
"scripts": { "test": "playwright test" }Now I can run the Playwright tests using npm test.
Recording video of the tests
Recording videos of the test runs turns out to be easy: change the _electron.launch() line to look like this:
const app = await _electron.launch({ args: ['main.js'], recordVideo: {dir: 'test-videos'} });This creates the videos as .webm files in the test-videos directory.
These videos can be opened in Chrome, or can be converted to mp4 using ffmpeg (available on macOS via brew install ffmpeg):
ffmpeg -i bc74c2a51bd91fe6f6cb815e6b99b6c7.webm bc74c2a51bd91fe6f6cb815e6b99b6c7.mp4Converting to .mp4 means you can drag and drop them onto a GitHub Issues thread and get an embedded video player. Here’s an example I recorded.
Custom timeouts
Playwright has a default 30s timeout on every action it takes. This turned out to be a bit too short for one of my tests, which installs a Python interpreter and a bunch of Python packages and can take 57s. Here’s how I fixed that so the test could pass:
test('App launches and quits', async () => { // This disables the global 30s timeout test.setTimeout(0); const app = await _electron.launch({ args: ['main.js'], recordVideo: {dir: 'test-videos'} }); const window = await app.firstWindow(); // This sets a timeout of 90s for the page to load and the // element with id="run-sql-link" to appear in the DOM: await window.waitForSelector('#run-sql-link', { timeout: 90000 }); await app.close();});Running it in GitHub Actions
I’m using the macos-latest image in my GitHub Actions workflow. The relevant configuration in my .github/workflows/test.yml file looks like this:
name: Test
on: push
jobs: test: runs-on: macos-latest steps: - uses: actions/checkout@v3 - name: Configure Node caching uses: actions/cache@v3 env: cache-name: cache-node-modules with: path: ~/.npm key: ${{ runner.os }}-build-${{ env.cache-name }}-${{ hashFiles('**/package-lock.json') }} restore-keys: | ${{ runner.os }}-build-${{ env.cache-name }}- ${{ runner.os }}-build- ${{ runner.os }}- - name: Install Node dependencies run: npm install - name: Run tests run: npm test timeout-minutes: 5 - name: Upload test videos uses: actions/upload-artifact@v3 with: name: test-videos path: test-videos/This workflow configures NPM caching to avoid downloading everything every time, installs the dependencies, runs the tests, and then uploads the videos at the end.
Those videos end up attached to the workflow run as an artifact that can be downloaded and viewed locally.