This is an implementation of the WebDriver BiDi protocol with some extensions (BiDi+) for Chromium, implemented as a JavaScript layer translating between BiDi and CDP, running inside a Chrome tab.
Current status can be checked at WPT WebDriver BiDi status.
"BiDi+" is an extension of the WebDriver BiDi protocol. In addition to WebDriver BiDi it has:
CdpSendCommandCommand = {
method: "cdp.sendCommand",
params: CdpSendCommandParameters,
}
CdpSendCommandParameters = {
method: text,
params: any,
session?: text,
}
CdpSendCommandResult = {
result: any,
session: text,
}
The command runs the described CDP command and returns the result.
CdpGetSessionCommand = {
method: "cdp.getSession",
params: CdpGetSessionParameters,
}
CdpGetSessionParameters = {
context: BrowsingContext,
}
CdpGetSessionResult = {
session: text,
}
The command returns the default CDP session for the selected browsing context.
CdpResolveRealmCommand = {
method: "cdp.resolveRealm",
params: CdpResolveRealmParameters,
}
CdpResolveRealmParameters = {
realm: Script.Realm,
}
CdpResolveRealmResult = {
executionContextId: text,
}
The command returns resolves a BiDi realm to its CDP execution context ID.
CdpEventReceivedEvent = {
method: "cdp.<CDP Event Name>",
params: CdpEventReceivedParameters,
}
CdpEventReceivedParameters = {
event: text,
params: any,
session: text,
}
The event contains a CDP event.
Each command can be extended with a channel:
Command = {
id: js-uint,
channel?: text,
CommandData,
Extensible,
}
If provided and non-empty string, the very same channel is added to the response:
CommandResponse = {
id: js-uint,
channel?: text,
result: ResultData,
Extensible,
}
ErrorResponse = {
id: js-uint / null,
channel?: text,
error: ErrorCode,
message: text,
?stacktrace: text,
Extensible
}
When client uses
commands session.subscribe
and session.unsubscribe
with channel, the subscriptions are handled per channel, and the corresponding
channel filed is added to the event message:
Event = {
channel?: text,
EventData,
Extensible,
}
This is a Node.js project, so install dependencies as usual:
npm installWe use cddlconv to generate our WebDriverBidi types before building.
- Install Rust.
- Run
cargo install --git https://github.com/google/cddlconv.git cddlconv
Refer to the documentation at .pre-commit-config.yaml.
pre-commit install --hook-type pre-pushThis will run the server on port 8080:
npm run serverUse the PORT= environment variable or --port= argument to run it on another port:
PORT=8081 npm run server
npm run server -- --port=8081Use the DEBUG environment variable to see debug info:
DEBUG=* npm run serverUse the DEBUG_DEPTH (default: 10) environment variable to see debug deeply nested objects:
DEBUG_DEPTH=100 DEBUG=* npm run serverUse the CHANNEL=... environment variable with one of the following values to run
the specific Chrome channel: stable, beta, canary, dev, local. Default is
local. The local channel means the pinned in .browser Chrome version will be
downloaded if it is not yet in cache. Otherwise, the requested Chrome version should
be installed.
CHANNEL=dev npm run serverUse the CLI argument --verbose to have CDP events printed to the console. Note: you have to enable debugging output bidi:mapper:debug:* as well.
DEBUG=bidi:mapper:debug:* npm run server -- --verboseor
DEBUG=* npm run server -- --verboseTODO: verify it works on Windows.
You can also run the server by using npm run server. It will write
output to the file log.txt:
npm run server -- --port=8081 --headless=falseSometimes it good to verify that a change will not affect thing downstream for other packages.
There is a useful puppeteer label you can add to any PR to run Puppeteer test with your changes.
It will bundle chromium-bidi and install it in Puppeteer project then run that package test.
Running:
npm run unitThe E2E tests are written using Python, in order to learn how to eventually do this in web-platform-tests.
Python 3.10+ and some dependencies are required:
python -m pip install --user pipenv
pipenv installThe E2E tests require BiDi server running on the same host. By default, tests
try to connect to the port 8080. The server can be run from the project root:
npm run e2e # alias to to e2e:headless
npm run e2e:headful
npm run e2e:headlessThis commands will run ./tools/run-e2e.mjs, which will log the PyTest output to console,
Additionally the output is also recorded under ./logs/<DATE>.e2e.log, this will contain
both the PyTest logs and in the event of FAILED test all the Chromium-BiDi logs.
If you need to see the logs for all test run the command with VERBOSE=true.
Simply pass npm run e2e -- tests/<PathOrFile> and the e2e will run only the selected one.
You run a specific test by running npm run e2e -- -k <TestName>.
Use CHROMEDRIVER environment to run tests in chromedriver instead of NodeJS runner:
CHROMEDRIVER=true npm run e2eUse the PORT environment variable to connect to another port:
PORT=8081 npm run e2eUse the HEADLESS to run the tests in headless (new or old) or headful modes.
Values: new, old, false, default: new.
HEADLESS=new npm run e2enpm run e2e -- --snapshot-updateSee https://github.com/tophat/syrupy for more information.
E2E tests use local http
server pytest-httpserver, which is run
automatically with the tests. However,
sometimes it is useful to run the http server outside the test
case, for example for manual debugging. This can be done by running:
pipenv run local_http_server...or directly:
python tests/tools/local_http_server.pyRefer to examples/README.md.
WPT is added as a git submodule. To get run WPT tests:
git submodule update --initcd wptFollow the System Setup instructions.
Follow
the hosts File Setup
instructions.
./wpt make-hosts-file | sudo tee -a /etc/hostsThis must be run in a PowerShell session with Administrator privileges:
python wpt make-hosts-file | Out-File $env:SystemRoot\System32\drivers\etc\hosts -Encoding ascii -AppendIf you are behind a proxy, you also need to make sure the domains above are excluded from your proxy lookups.
Set the BROWSER_BIN environment variable to a Chrome, Edge or Chromium binary to launch.
For example, on macOS:
# Chrome
export BROWSER_BIN="/Applications/Google Chrome Canary.app/Contents/MacOS/Google Chrome Canary"
export BROWSER_BIN="/Applications/Google Chrome Dev.app/Contents/MacOS/Google Chrome Dev"
export BROWSER_BIN="/Applications/Google Chrome Beta.app/Contents/MacOS/Google Chrome Beta"
export BROWSER_BIN="/Applications/Google Chrome.app/Contents/MacOS/Google Chrome"
export BROWSER_BIN="/Applications/Chromium.app/Contents/MacOS/Chromium"
# Edge
export BROWSER_BIN="/Applications/Microsoft Edge Canary.app/Contents/MacOS/Microsoft Edge Canary"
export BROWSER_BIN="/Applications/Microsoft Edge.app/Contents/MacOS/Microsoft Edge"https://www.google.com/chrome/dev/
Oneshot:
npm run buildContinuously:
npm run build --watchnpm run wpt -- webdriver/tests/bidi/UPDATE_EXPECTATIONS=true npm run wpt -- webdriver/tests/bidi/The architecture is described in the WebDriver BiDi in Chrome Context implementation plan .
There are 2 main modules:
- backend WS server in
src. It runs webSocket server, and for each ws connection runs an instance of browser with BiDi Mapper. - front-end BiDi Mapper in
src/bidiMapper. Gets BiDi commands from the backend, and map them to CDP commands.
The BiDi commands are processed in the src/bidiMapper/commandProcessor.ts. To add a
new command, add it to _processCommand, write and call processor for it.
We use release-please to automate releases. When a release should be done, check for the release PR in our pull requests and merge it.
-
Dry-run
npm publish --dry-run
-
Open a PR bumping the chromium-bidi version number in
package.jsonfor review:npm version patch -m 'chore: Release v%s' --no-git-tag-versionInstead of
patch, useminorormajoras needed. -
After the PR is reviewed, create a GitHub release specifying the tag name matching the bumped version. Our CI then automatically publishes the new release to npm based on the tag name.
This section assumes you already have a Chromium set-up locally, and knowledge on how to submit changes to the repo. Otherwise submit an issue for a project maintainer.
- Create a new branch in chromium
src/. - Update the mapper version:
third_party/bidimapper/roll_bidmapper-
Submit a CL with bug
chromedriver:4226. -
Regenerate WPT expectations or baselines:
4.1. Trigger a build and test run:
third_party/blink/tools/blink_tool.py rebaseline-cl --build="linux-blink-rel" --verbose4.2. Once the test completes on the builder, rerun that command to update the baselines. Update test expectations if there are any crashes or timeouts. Commit the changes (if any), and upload the new patch to the CL.
-
Add appropriate reviewers or comment the CL link on the PR.