While it took a bit longer than expected the WebdriverIO team is excited to announce that we finally released v8
today! 🎉 🎉 🎉
As with almost all of the last major updates we again had to touch every single file of the project. This time our major goal for the new version was to finally transition from CommonJS to ESM which enables us to continue with important dependency updates and avoid security issues. Furthermore, we cleaned up some technical debt, e.g. removed all code related to synchronous command execution which was deprecated last year, as well as implemented a new Action API interface and streamlined the way WebdriverIO deals with global objects using the test runner.
In this blog post, we go through every important change and explain what you need to do to upgrade to v8
. Spoiler alert: in most cases no updates to your tests are necessary 😉
Drop Node.js v12, v13 and v14 Support
We've dropped support for Node v12 and v14, latter was moved into a maintenance LTS phase by the Node.js team in October 2021. While it is technically fine to continue using Node.js v14, we don't see a reason why you shouldn't update to Node.js v16 or ideally v18 directly.
To update Node.js, it is important to know how it was installed in the first place. If you are in a Docker environment, you can just upgrade the base image like:
- FROM mhart/alpine-node:14
+ FROM mhart/alpine-node:18
We recommend using NVM (Node Version Manager) to install and manage Node.js versions. You can find a detailed description of how to install NVM and update Node in their project readme.
CommonJS to ESM Transition
The transition to the new module system has been the biggest chunk of work related to this release. It required us to update all module imports, transitioning from Jest to Vitest as a unit test framework and rewrite various parts within the code base. While this affected every single file it "should" be unrecognizable to you. If your project still uses CommonJS, WebdriverIO will work just fine as both module systems continue to be supported. This is also the case when using webdriver
, devtools
or webdriverio
as a module.
If you have been using Babel only to use import
statements in your tests, you can remove the integration as this is now supported by ESM natively. If you like to continue using CommonJS and require
, that is fine too, no changes are needed to update to v8
.
A new Runner for Unit and Component Testing in the Browser
If it comes to one feature we are really excited about in this release it is the new browser runner 🙌 I've been writing and testing a lot of web components in this past year and was always frustrated about the fact that they would be tested against JSDOM rather than an actual browser. JSDOM is a re-implementation of many Web APIs in Node.js and is a great tool for simple testing but it doesn't replace an actual DOM implementation within a browser. Especially using JSDOM for component testing has various disadvantages compared to running tests in the browser.
Furthermore running component tests through WebdriverIO allows to use the WebdriverIO API seamlessly and enables real user interaction with your components through the WebDriver protocol. This makes those interactions more realistic compared to emitting them through JavaScript. It comes also with 1st class support for popular utility frameworks such as Testing Library and allows to use both APIs interchangeably. Check out how you can use Testing Library for rendering and fetching elements while using WebdriverIO for interacting with the component:
import { $, expect } from '@wdio/globals'
import { render } from '@testing-library/vue'
import Component from './components/Component.vue'
describe('Vue Component Testing', () => {
it('increments value on click', async () => {
// The render method returns a collection of utilities to query your component.
const { getByText } = render(Component)
// getByText returns the first matching node for the provided text, and
// throws an error if no elements match or if more than one match is found.
getByText('Times clicked: 0')
const button = await $(getByText('increment'))
// Dispatch a native click event to our button element.
await button.click()
await button.click()
getByText('Times clicked: 2') // assert with Testing Library
await expect($('p=Times clicked: 2')).toExist() // assert with WebdriverIO
})
})
The new browser runner allows you to load and execute tests within the browser rather than in Node.js. This allows you to access all Web APIs to render web components or to run unit tests for your frontend modules. Under the hood it uses Vite to load all dependencies and make the integration seamless.
If you have been using Karma for running unit tests in the browser you can switch over to WebdriverIO which provides the same capabilities but offers better support and integration to other services and reporters. It also seems that the Karma project is not much maintained these days and has unresovled security vulnerabilities.
New Action Interface
For many years users that liked to run more complex interactions on their applications using WebDriver's actions API had to know many details about the command to construct the correct payload. With v8
of WebdriverIO, we now ship a new interface that makes executing various actions much easier.
With two new browser commands: action
and actions
, it is now much simpler and type-safe to run the right action, e.g. sending key events to the browser:
await browser.action('key')
.down('f').up('f')
.down('o').up('o')
.down('o').up('o')
.perform()
Read more on the new action interface in the WebdriverIO API.
WebDriver BiDi Support
A strong argument to use WebdriverIO as opposed to other tools is the fact that it is based on the WebDriver protocol, which is a web standard for automating browsers. It guarantees the ability to run tests in browsers that are used by your users as opposed to a browser engine, which can be very different from a feature and security aspect. The W3C Working Group has been working on a new version of the protocol that will enable better introspection capabilities and new automation primitives.
With this release, users can start accessing these new protocol features as they become available in browsers. Over time more commands will transition to the new protocol under the hood while the interface remains the same. We are all very excited about the new capabilities and opportunities this protocol will provide, e.g. listening to log events while running tests, e.g.:
await browser.send({
method: 'session.subscribe',
params: { events: ['log.entryAdded'] }
})
/**
* returns:
* {
* "method":"log.entryAdded",
* "params":{
* "type":"console",
* "method":"log",
* "realm":null,
* "args":[
* {
* "type":"string",
* "value":"Hello Bidi"
* }
* ],
* "level":"info",
* "text":"Hello Bidi",
* "timestamp":1657282076037
* }
* }
*/
browser.on('message', (data) => console.log('received %s', data))
/**
* trigger listener
*/
await browser.execute(() => console.log("Hello Bidi"))
We are following and supporting the development of all browser vendors to ensure new features are working as expected and can be used through a lean user interface. For more information on this topic check out my talk on The Evolution of Browser Automation.
Optional Globals
When using the WebdriverIO test runner it would usually register the browser
object or the $
and $$
command to the global scope as these are usually often used when writing tests. However, attaching objects to the global scope is not seen as best practice and can cause side effects when other modules decide to do the same. Therefore with v8
, it is now up to the user whether they like to continue attaching these objects and methods to the global scope or prefer importing them directly, via:
import { browser, $, $$, expect } from '@wdio/globals'
A new configuration property called injectGlobals
(defaults: true
) handles whether the test runner modifies the global scope or not. If your setup works fine using global objects, no change is needed to update to v8
. However, we recommend importing WebdriverIO-related interfaces directly to ensure no side effects can happen.
Note: If you are using TypeScript, updates to the tsconfig.json
are necessary to reflect changes made to the location of the WebdriverIO types. These are now part of the @wdio/globals
package:
{
"compilerOptions": {
"types": [
"node",
- "webdriverio/async"
+ "@wdio/globals/types"
]
}
}
Miscellaneous
Aside from these major updates, the team has spent time improving the documentation and introduced new API docs around WebdriverIO objects like browser
, element
and mock
. Furthermore, we removed the config
property from the browser
object. If you have been using it to access data from the WDIO config, we suggest replacing it with options
, e.g.:
- browser.config.hostname
+ browser.options.hostname
Furthermore did we fix the behavior of relative spec or exclude paths. Before v8
every path within specs
, exclude
or --spec
was always seen relative from the users working directory. This behavior was confusing especially when the wdio.conf.js
was not within the root of you project. This got fixed now so that specs
and exclude
path will be always seen as relative to the config file and --spec
arguments, relative from the working directory.
Lastly, we had to remove support for tsconfig-paths as we haven't found a way to get it working within an ESM context. We believe this integration hasn't been used much anyway and a lot of it is nowadays natively supported in TypeScript. Let us know if this assumption is wrong and you would like to see it being supported again.
What's Next?
The WebdriverIO team is very excited about this release as it frees up time to start working on some new cool features we put on the roadmap. For many months we have been working secretly on a VS Code Extension that makes authoring and debugging tests much easier. Aside from that, there is always plenty more work to do and opportunities to explore to make this project better. We are welcoming and supporting everyone who likes to join us.
Lastly, I would like to say thank you to everyone who supports the project. Not only the folks who contribute financially through Open Collective or Tidelift but also everyone who contributes code, ideas, reports issues or supports folks in our support chat, occasionally or on regular basis. Without contributions from the community, this project can't go anywhere. Aside from many alternative projects WebdriverIO is not funded, nor driven by any corporate interest and stays 100% community governed. No lack of funding or need for capital gains will have an impact on this project. It has been like this since its inception more than 10 years ago and will continue to be like this for many many more years. Therefore we are always looking for interested folks who like to help us hack on the project. If you haven't, join our Open Office Hours and consider giving back to the project.
I am looking forward to more years and great features ahead. Thanks for reading!