Web Extension Testing
WebdriverIO is the ideal tool to automate a browser. Web Extensions are a part of the browser and can be automated in the same way. Whenever your web extension uses content scripts to run JavaScript on websites or offer a popup modal, you can run an e2e test for that using WebdriverIO.
Loading a Web Extension into the Browser
As a first step we have to load the extension under test into the browser as part of our session. This works differently for Chrome and Firefox.
These docs leave out Safari web extensions as their support for it is way behind and user demand not high. If you are building a web extension for Safari, please raise an issue and collaborate on including it here as well.
Chrome
Loading a web extension in Chrome can be done through providing a base64
encoded string of the crx
file or by providing a path to the web extension folder. The easiest is just to do the latter by defining your Chrome capabilities as following:
import path from 'node:path'
import url from 'node:url'
const __dirname = url.fileURLToPath(new URL('.', import.meta.url))
export const config = {
// ...
capabilities: [{
browserName,
'goog:chromeOptions': {
// given your wdio.conf.js is in the root directory and your compiled
// web extension files are located in the `./dist` folder
args: [`--load-extension=${path.join(__dirname, '..', '..', 'dist')}`]
}
}]
}
If you automate a different browser than Chrome, e.g. Brave, Edge or Opera, chances are that the browser option match with the example above, just using a different capability name, e.g. ms:edgeOptions
.
If you compile your extension as .crx
file using e.g. the crx NPM package, you can also inject the bundled extension via:
import path from 'node:path'
import url from 'node:url'
const __dirname = url.fileURLToPath(new URL('.', import.meta.url))
const extPath = path.join(__dirname, `web-extension-chrome.crx`)
const chromeExtension = (await fs.readFile(extPath)).toString('base64')
export const config = {
// ...
capabilities: [{
browserName,
'goog:chromeOptions': {
extensions: [chromeExtension]
}
}]
}
Firefox
To create a Firefox profile that includes extensions you can use the Firefox Profile Service to set up your session accordingly. However you might run into issues where your local developed extension can't be loaded due to signing issues. In this case you can also load an extension in the before
hook via the installAddOn
command, e.g.:
import path from 'node:path'
import url from 'node:url'
const __dirname = url.fileURLToPath(new URL('.', import.meta.url))
const extensionPath = path.resolve(__dirname, `web-extension.xpi`)
export const config = {
// ...
before: async (capabilities) => {
const browserName = (capabilities as WebdriverIO.Capabilities).browserName
if (browserName === 'firefox') {
const extension = await fs.readFile(extensionPath)
await browser.installAddOn(extension.toString('base64'), true)
}
}
}
In order to generate an .xpi
file, it is recommended to use the web-ext
NPM package. You can bundle your extension using the following example command:
npx web-ext build -s dist/ -a . -n web-extension-firefox.xpi
Tips & Tricks
The following section contains a set useful tips and tricks that can be helpful when testing a web extension.
Test Popup Modal in Chrome
If you define a default_popup
browser action entry in your extension manifest you can test that HTML page directly, since clicking on the extension icon in the browser top bar won't work. Instead, you have to open the popup html file directly.
In Chrome this works by retrieving the extension ID and opening the popup page through browser.url('...')
. The behavior on that page will be the same as within the popup. To do so we recommend to write the following custom command:
export async function openExtensionPopup (this: WebdriverIO.Browser, extensionName: string, popupUrl = 'index.html') {
if ((this.capabilities as WebdriverIO.Capabilities).browserName !== 'chrome') {
throw new Error('This command only works with Chrome')
}
await this.url('chrome://extensions/')
const extensions = await this.$$('>>> extensions-item')
const extension = await extensions.find(async (ext) => (
await ext.$('#name').getText()) === extensionName
)
if (!extension) {
const installedExtensions = await extensions.map((ext) => ext.$('#name').getText())
throw new Error(`Couldn't find extension "${extensionName}", available installed extensions are "${installedExtensions.join('", "')}"`)
}
const extId = await extension.getAttribute('id')
await this.url(`chrome-extension://${extId}/popup/${popupUrl}`)
}
declare global {
namespace WebdriverIO {
interface Browser {
openExtensionPopup: typeof openExtensionPopup
}
}
}
In your wdio.conf.js
you can import this file and register the custom command in your before
hook, e.g.:
import type { Options } from '@wdio/testrunner'
import { browser } from '@wdio/globals'
import { openExtensionPopup } from './support/customCommands'
export const config: Options.Testrunner = {
// ...
before: () => {
browser.addCommand('openExtensionPopup', openExtensionPopup)
}
}
Now, in your test, you can access the popup page via:
await browser.openExtensionPopup('My Web Extension')