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.
You need to install the @playwright/test
package. You can do that like so:
npm i -D @playwright/test
This adds it to devDependencies
in your package.json
, something like this:
"devDependencies": {
"@playwright/test": "^1.23.2",
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 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.mp4
Converting 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.
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();
});
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.