Back to template gallery

Playwright + Chrome Test Runner

Example of using the Playwright Test project to run automated website tests in the cloud and display their results. Usable as an API.

Language

typescript

Tools

nodejs

playwright

Use cases

Web scraping

src/main.ts

src/runCodegen.ts

src/transform.ts

1import { Actor } from 'apify';
2import log from '@apify/log';
3import { Dictionary } from 'apify-client';
4import fs from 'fs';
5import path from 'path';
6import { execSync } from 'child_process';
7import { collectAttachmentPaths, transformToTabular } from './transform';
8
9function ensureFolder(pathname: string) {
10    if (!fs.existsSync(pathname)) {
11        fs.mkdirSync(pathname, { recursive: true });
12    }
13}
14
15function getConfigPath(){
16    return `${__dirname}/../playwright.config.ts`;
17}
18
19function getResultDir(){
20    return `${__dirname}/../playwright-report`;
21}
22
23const getConfig = (options: {screen: {width: number, height: number}, headful: boolean, timeout: number, locale: string, darkMode: boolean, ignoreHTTPSErrors: boolean, video: string}) => {
24    const {screen, headful, timeout, ignoreHTTPSErrors, darkMode, locale, video} = options;
25
26    return `
27// Watch out! This file gets regenerated on every run of the actor.
28// Any changes you make will be lost.
29
30// Tweak your configuration through the Actor's input through the Apify console or directly in the \`input.json\` file.
31import { defineConfig } from '@playwright/test';
32export default defineConfig({
33    timeout: ${timeout},
34    use: {
35        headless: ${!headful},
36        viewport: { width: ${screen.width}, height: ${screen.height} },
37        ignoreHTTPSErrors: ${ignoreHTTPSErrors},
38        colorScheme: '${darkMode ? 'dark' : 'light'}',
39        locale: '${locale}',
40        video: '${video}',
41        launchOptions: {
42            args: [
43                '--disable-gpu', // Mitigates the "crashing GPU process" issue in Docker containers
44            ]
45        },
46    },
47    reporter: [
48        ['html', { outputFolder: '${getResultDir()}', open: 'never' }],
49        ['json', { outputFile: '${getResultDir()}/test-results.json' }]
50    ],
51});`
52}
53function runTests() {
54    try {
55        execSync(`npx playwright test --config=${getConfigPath()}`, {
56            cwd: __dirname,
57            encoding: 'utf8',
58            stdio: 'inherit',
59        });
60    } catch (e) {
61        // suppress error, the report will be generated anyway
62    }
63}
64
65function updateConfig(args: {
66    screenWidth?: number,
67    screenHeight?: number,
68    headful?: boolean,
69    timeout?: number,
70    darkMode?: boolean,
71    locale?: string,
72    ignoreHTTPSErrors?: boolean,
73    video?: string,
74}) {
75    const {
76        screenWidth = 1280,
77        screenHeight =  720,
78        headful = false,
79        timeout = 60,
80        darkMode = false,
81        locale = 'en-US',
82        ignoreHTTPSErrors = true,
83        video = 'off'
84    } = args;
85
86    const config = getConfig({screen: { width: screenWidth, height: screenHeight }, headful, timeout: timeout * 1000, locale, darkMode, ignoreHTTPSErrors, video});
87    fs.writeFileSync(getConfigPath(), config, { encoding: 'utf-8' });
88}
89
90(async () => {
91    await Actor.init();
92    const input = (await Actor.getInput() ?? {}) as Dictionary;
93
94    ensureFolder(getResultDir());
95    updateConfig(input);
96
97    runTests();
98
99    const kvs = await Actor.openKeyValueStore();
100    await kvs.setValue('report', fs.readFileSync(path.join(getResultDir(), 'index.html'), { encoding: 'utf-8' }), { contentType: 'text/html' });
101    const jsonReport = JSON.parse(fs.readFileSync(path.join(getResultDir(), 'test-results.json'), { encoding: 'utf-8' }));
102    const attachmentPaths = collectAttachmentPaths(jsonReport);
103
104    const attachmentLinks = await Promise.all(attachmentPaths.map(async (x) => {
105        const attachment = fs.readFileSync(x.path);
106        await kvs.setValue(x.key, attachment, { contentType: x.type ?? 'application/octet' });
107        return {...x, url: await kvs.getPublicUrl(x.key)};
108    }));
109
110    await Actor.pushData(transformToTabular(jsonReport, attachmentLinks));
111
112    const reportURL = await kvs.getPublicUrl('report');
113    log.info('The test run has finished! The report is available in the Output tab or at the link below:');
114    console.log(reportURL);
115
116    await Actor.exit();
117})();

Playwright test template

Run your Playwright tests on the Apify platform effectively and easily. Just set up your test environment using a user-friendly UI and let the platform do the rest.

Note: This is a custom version of Playwright Test Runner Actor. Unlike the original Actor, this version reads test suite files from the tests folder and does not allow you to pass the test files via Apify input.

Features

Run your Playwright tests on the Apify platform

No more pre-commit hooks or CI/CD pipelines. Integrate your tests with the Apify Platform using a user-friendly UI and forget about the hassle of setting up your test environment.

Test configuration with comprehensive UI

Collect and analyze your test results online

After running the tests, the Apify platform stores the results in comprehensive datasets. You can view the results directly on the platform or download them to your local machine using a REST API.

Analyzing understandable test reports

No more problems with incompatible browser versions

Playwright Test toolkit automatically downloads the latest versions of Chromium, Firefox, and WebKit browsers and installs them in the Apify platform.

This way, you can test your websites using all the popular browsers without worrying about compatibility issues.

Testing with multiple browser versions at once

How to use

Just provide your test suite files in the tests folder and run the Actor. The Actor will automatically run all the tests in the tests folder and store the results in the KVS/dataset fields.

You can also customize the test run by specifying other options in the input, e.g. the screen size, headful/headless execution or the maximum run time.

Test Generator

You can also use the Playwright Codegen to compose your test suites even faster. Just run npm run codegen in your project folder and record your workflow.

The code generator will automatically create a test suite file for you and save it in the tests folder.

Resources

Already have a solution in mind?

Sign up for a free Apify account and deploy your code to the platform in just a few minutes! If you want a head start without coding it yourself, browse our Store of existing solutions.