No results for

Powered byAlgolia

Running browser tests

Follow along to learn how to:

  1. Run a test
  2. Interact with elements on your webpage
  3. Wait for page navigation
  4. Run both browser-level and protocol-level tests in a single script

With these example snippets, you'll run the test locally with your machine's resources. The browser module is not available within k6 cloud as of yet.

Run a test

To run a simple local script:

  1. Copy the following code, paste it into your favorite editor, and save it as script.js:

    1import { chromium } from 'k6/experimental/browser';
    3export default async function () {
    4 const browser = chromium.launch({
    5 headless: false,
    6 timeout: '60s', // Or whatever time you want to define
    7 });
    8 const page = browser.newPage();
    10 try {
    11 await page.goto('');
    12 page.screenshot({ path: 'screenshot.png' });
    13 } finally {
    14 page.close();
    15 browser.close();
    16 }

    The preceding code imports the chromium BrowserType (currently the only available BrowserType implementation), and uses its launch method to start up a Chromium Browser process. Two parameters are passed to it. One is the headless parameter with the value false so you can see the browser launching, and timeout parameter with the value 60s which will be the timeout used for various actions and navigation. For a full list of parameters that you can pass, check out the documentation for BrowserType.launch().

    After it starts, you can interact with it using the browser-level APIs. This example visits a test URL and takes a screenshot of the page. Afterwards, it closes the page and the browser.


    To provide rough compatibility with the Playwright API, the browser module API is also being converted from synchronous to asynchronous. page.goto() is now asynchronous so await keyword is used to deal with the asynchronous nature of the operation.

  2. Then, run the test on your terminal with this command:

Windows: CMD
Windows: PowerShell
$ K6_BROWSER_ENABLED=true k6 run script.js

Interact with elements on your webpage

You can use page.locator() and pass in the element's selector you want to find on the page. page.locator() will create and return a Locator object, which you can later use to interact with the element.

To find out which selectors the browser module supports, check out Selecting Elements.


You can also use page.$() instead of page.locator(). You can find the differences between page.locator() and page.$ in the Locator API documentation.

1import { chromium } from 'k6/experimental/browser';
3export default async function () {
4 const browser = chromium.launch({ headless: false });
5 const page = browser.newPage();
7 try {
8 await page.goto('');
10 // Enter login credentials
11 page.locator('input[name="login"]').type('admin');
12 page.locator('input[name="password"]').type('123');
14 page.screenshot({ path: 'screenshot.png' });
15 } finally {
16 page.close();
17 browser.close();
18 }

The preceding code creates and returns a Locator object with the selectors for both login and password passed as arguments.

Within the Locator API, various methods such as type() can be used to interact with the elements. The type() method types a text to an input field.

Asynchronous operations

Since many browser operations happen asynchronously, and to follow the Playwright API more closely, we are working on migrating most of the browser module methods to be asynchronous as well.

At the moment, methods such as page.goto(), page.waitForNavigation() and return JavaScript promises, and scripts must be written to handle this properly.

To avoid timing errors or other race conditions in your script, if you have actions that load up a different page, you need to make sure that you wait for that action to finish before continuing.

1import { chromium } from 'k6/experimental/browser';
2import { check } from 'k6';
4export default async function () {
5 const browser = chromium.launch({ headless: false });
6 const page = browser.newPage();
8 try {
9 await page.goto('');
11 page.locator('input[name="login"]').type('admin');
12 page.locator('input[name="password"]').type('123');
14 const submitButton = page.locator('input[type="submit"]');
16 await Promise.all([page.waitForNavigation(),]);
18 check(page, {
19 header: page.locator('h2').textContent() == 'Welcome, admin!',
20 });
21 } finally {
22 page.close();
23 browser.close();
24 }

The preceding code uses Promise.all([]) to wait for the two promises to be resolved before continuing. Since clicking the submit button causes page navigation, page.waitForNavigation() is needed because the page won't be ready until the navigation completes. This is required because there can be a race condition if these two actions don't happen simultaneously.

Then, you can use check from the k6 API to assert the text content of a specific element. Finally, you close the page and the browser.

Run both browser-level and protocol-level tests in a single script

The real power of the browser module shines when it’s combined with the existing features of k6. A common scenario that you can try is to mix a smaller subset of browser-level tests with a larger protocol-level test which can simulate how your website responds to various performance events. This approach is what we refer to as hybrid load testing and provides advantages such as:

  • testing real user flows on the frontend while generating a higher load in the backend
  • measuring backend and frontend performance in the same test execution
  • increased collaboration between backend and frontend teams since the same tool can be used

To run a browser-level and protocol-level test concurrently, you can use scenarios.


Keep in mind that there is an additional performance overhead when it comes to spinning up a browser VU and that the resource usage will depend on the system under test.

1import { chromium } from 'k6/experimental/browser';
2import { check } from 'k6';
3import http from 'k6/http';
5export const options = {
6 scenarios: {
7 browser: {
8 executor: 'constant-vus',
9 exec: 'browser',
10 vus: 1,
11 duration: '10s',
12 },
13 news: {
14 executor: 'constant-vus',
15 exec: 'news',
16 vus: 20,
17 duration: '1m',
18 },
19 },
22export async function browser() {
23 const browser = chromium.launch({ headless: false });
24 const page = browser.newPage();
26 try {
27 await page.goto('');
29 page.locator('#checkbox1').check();
31 check(page, {
32 'checkbox is checked':
33 page.locator('#checkbox-info-display').textContent() === 'Thanks for checking the box',
34 });
35 } finally {
36 page.close();
37 browser.close();
38 }
41export function news() {
42 const res = http.get('');
44 check(res, {
45 'status is 200': (r) => r.status === 200,
46 });

The preceding code contains two scenarios. One for the browser-level test called browser and one for the protocol-level test called news. Both scenarios are using the constant-vus executor which introduces a constant number of virtual users to execute as many iterations as possible for a specified amount of time.

Since it's all in one script, this allows for greater collaboration amongst teams.