Important: This documentation is about an older version. It's relevant only to the release noted, many of the features and functions have been updated or replaced. Please view the current version.
Running browser tests
Follow along to learn how to:
- Run a test
- Interact with elements on your webpage
- Wait for page navigation
- Run both browser-level and protocol-level tests in a single script
Note
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:
Copy the following code, paste it into your favorite editor, and save it as
script.js
.Note that providing an
executor
and setting thebrowser
scenario option’stype
tochromium
is mandatory. Please see the options and scenarios documentation for more information.import { browser } from 'k6/experimental/browser'; export const options = { scenarios: { ui: { executor: 'shared-iterations', options: { browser: { type: 'chromium', }, }, }, }, thresholds: { checks: ['rate==1.0'], }, }; export default async function () { const page = browser.newPage(); try { await page.goto('https://test.k6.io/'); page.screenshot({ path: 'screenshots/screenshot.png' }); } finally { page.close(); } }
The preceding code imports the
browser
the browser module), and uses itsnewPage
method to open a new page.After getting the page, you can interact with it using the Page methods. This example visits a test URL and takes a screenshot of the page.
Subsequently, the page is closed. This allows for the freeing up of allocated resources and enables the accurate calculation of Web Vital metrics.
Note
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 soawait
keyword is used to deal with the asynchronous nature of the operation.Then, run the test on your terminal with this command:
$ k6 run script.js
# WARNING! # The grafana/k6:master-with-browser image launches a Chrome browser by setting the # 'no-sandbox' argument. Only use it with trustworthy websites. # # As an alternative, you can use a Docker SECCOMP profile instead, and overwrite the # Chrome arguments to not use 'no-sandbox' such as: # docker container run --rm -i -e K6_BROWSER_ARGS='' --security-opt seccomp=$(pwd)/chrome.json grafana/k6:master-with-browser run - <script.js # # You can find an example of a hardened SECCOMP profile in: # https://raw.githubusercontent.com/jfrazelle/dotfiles/master/etc/docker/seccomp/chrome.json. docker run --rm -i -v $(pwd):/home/k6/screenshots grafana/k6:master-with-browser run - <script.js
C:\k6> k6 run script.js
PS C:\k6> k6 run script.js
You can also use the browser module options to customize the launching of a browser process. For instance, you can start a headful browser using the previous test script with this command.
$ K6_BROWSER_HEADLESS=false k6 run script.js
# WARNING! # The grafana/k6:master-with-browser image launches a Chrome browser by setting the # 'no-sandbox' argument. Only use it with trustworthy websites. # # As an alternative, you can use a Docker SECCOMP profile instead, and overwrite the # Chrome arguments to not use 'no-sandbox' such as: # docker container run --rm -i -e K6_BROWSER_ARGS='' --security-opt seccomp=$(pwd)/chrome.json grafana/k6:master-with-browser run - <script.js # # You can find an example of a hardened SECCOMP profile in: # https://raw.githubusercontent.com/jfrazelle/dotfiles/master/etc/docker/seccomp/chrome.json. docker run --rm -i -v $(pwd):/home/k6/screenshots -e K6_BROWSER_HEADLESS=false grafana/k6:master-with-browser run - <script.js
C:\k6> set "K6_BROWSER_HEADLESS=false" && k6 run script.js
PS C:\k6> $env:K6_BROWSER_HEADLESS=false ; k6 run script.js
Note
When using Docker to run k6 browser tests, make sure you have pulled the correct image with Chromium built-in. See k6 Installation via Docker for more information.
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.
Note
You can also usepage.$()
instead ofpage.locator()
. You can find the differences betweenpage.locator()
andpage.$
in the Locator API documentation.
import { browser } from 'k6/experimental/browser';
export const options = {
scenarios: {
ui: {
executor: 'shared-iterations',
options: {
browser: {
type: 'chromium',
},
},
},
},
thresholds: {
checks: ['rate==1.0'],
},
};
export default async function () {
const page = browser.newPage();
try {
await page.goto('https://test.k6.io/my_messages.php');
// Enter login credentials
page.locator('input[name="login"]').type('admin');
page.locator('input[name="password"]').type('123');
page.screenshot({ path: 'screenshots/screenshot.png' });
} finally {
page.close();
}
}
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 Element.click()
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.
import { check } from 'k6';
import { browser } from 'k6/experimental/browser';
export const options = {
scenarios: {
ui: {
executor: 'shared-iterations',
options: {
browser: {
type: 'chromium',
},
},
},
},
thresholds: {
checks: ['rate==1.0'],
},
};
export default async function () {
const page = browser.newPage();
try {
await page.goto('https://test.k6.io/my_messages.php');
page.locator('input[name="login"]').type('admin');
page.locator('input[name="password"]').type('123');
const submitButton = page.locator('input[type="submit"]');
await Promise.all([page.waitForNavigation(), submitButton.click()]);
check(page, {
header: (p) => p.locator('h2').textContent() == 'Welcome, admin!',
});
} finally {
page.close();
}
}
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.
Note
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.
import { browser } from 'k6/experimental/browser';
import { check } from 'k6';
import http from 'k6/http';
export const options = {
scenarios: {
browser: {
executor: 'constant-vus',
exec: 'browserTest',
vus: 1,
duration: '10s',
options: {
browser: {
type: 'chromium',
},
},
},
news: {
executor: 'constant-vus',
exec: 'news',
vus: 20,
duration: '1m',
},
},
};
export async function browserTest() {
const page = browser.newPage();
try {
await page.goto('https://test.k6.io/browser.php');
page.locator('#checkbox1').check();
check(page, {
'checkbox is checked':
page.locator('#checkbox-info-display').textContent() === 'Thanks for checking the box',
});
} finally {
page.close();
}
}
export function news() {
const res = http.get('https://test.k6.io/news.php');
check(res, {
'status is 200': (r) => r.status === 200,
});
}
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.