--- id: release-notes title: "Release notes" toc_max_heading_level: 2 --- ## Version 1.51 ### Highlights * New option [`option: BrowserContext.storageState.indexedDB`] for [`method: BrowserContext.storageState`] allows to save and restore IndexedDB contents. Useful when your application uses [IndexedDB API](https://developer.mozilla.org/en-US/docs/Web/API/IndexedDB_API) to store authentication tokens, like Firebase Authentication. Here is an example following the [authentication guide](./auth.md#reusing-signed-in-state): ```python # Save storage state into the file. Make sure to include IndexedDB. storage = await context.storage_state(path="state.json", indexed_db=True) # Create a new context with the saved storage state. context = await browser.new_context(storage_state="state.json") ``` * New option [`option: Locator.filter.visible`] for [`method: Locator.filter`] allows matching only visible elements. ```python # Ignore invisible todo items. todo_items = page.get_by_test_id("todo-item").filter(visible=True) # Check there are exactly 3 visible ones. await expect(todo_items).to_have_count(3) ``` * New option `contrast` for methods [`method: Page.emulateMedia`] and [`method: Browser.newContext`] allows to emulate the `prefers-contrast` media feature. * New option [`option: APIRequest.newContext.failOnStatusCode`] makes all fetch requests made through the [APIRequestContext] throw on response codes other than 2xx and 3xx. ### Browser Versions * Chromium 134.0.6998.35 * Mozilla Firefox 135.0 * WebKit 18.4 This version was also tested against the following stable channels: * Google Chrome 133 * Microsoft Edge 133 ## Version 1.50 ### Async Pytest Plugin * [Playwright's Pytest plugin](./test-runners.md) now has support for [Async Fixtures](https://playwright.dev/python/docs/test-runners#async-fixtures). ### Miscellaneous * Added method [`method: LocatorAssertions.toHaveAccessibleErrorMessage`] to assert the Locator points to an element with a given [aria errormessage](https://w3c.github.io/aria/#aria-errormessage). ### UI updates * New button in Codegen for picking elements to produce aria snapshots. * Additional details (such as keys pressed) are now displayed alongside action API calls in traces. * Display of `canvas` content in traces is error-prone. Display is now disabled by default, and can be enabled via the `Display canvas content` UI setting. * `Call` and `Network` panels now display additional time information. ### Breaking * [`method: LocatorAssertions.toBeEditable`] and [`method: Locator.isEditable`] now throw if the target element is not ``, `` elements. ```python page.get_by_label("Upload directory").set_input_files('mydir') ``` - Multiple methods like [`method: Locator.click`] or [`method: Locator.press`] now support a `ControlOrMeta` modifier key. This key maps to `Meta` on macOS and maps to `Control` on Windows and Linux. ```python # Press the common keyboard shortcut Control+S or Meta+S to trigger a "Save" operation. page.keyboard.press("ControlOrMeta+S") ``` - New property `httpCredentials.send` in [`method: APIRequest.newContext`] that allows to either always send the `Authorization` header or only send it in response to `401 Unauthorized`. - Playwright now supports Chromium, Firefox and WebKit on Ubuntu 24.04. - v1.45 is the last release to receive WebKit update for macOS 12 Monterey. Please update macOS to keep using the latest WebKit. ### Browser Versions * Chromium 127.0.6533.5 * Mozilla Firefox 127.0 * WebKit 17.4 This version was also tested against the following stable channels: * Google Chrome 126 * Microsoft Edge 126 ## Version 1.44 ### New APIs **Accessibility assertions** - [`method: LocatorAssertions.toHaveAccessibleName`] checks if the element has the specified accessible name: ```python locator = page.get_by_role("button") expect(locator).to_have_accessible_name("Submit") ``` - [`method: LocatorAssertions.toHaveAccessibleDescription`] checks if the element has the specified accessible description: ```python locator = page.get_by_role("button") expect(locator).to_have_accessible_description("Upload a photo") ``` - [`method: LocatorAssertions.toHaveRole`] checks if the element has the specified ARIA role: ```python locator = page.get_by_test_id("save-button") expect(locator).to_have_role("button") ``` **Locator handler** - After executing the handler added with [`method: Page.addLocatorHandler`], Playwright will now wait until the overlay that triggered the handler is not visible anymore. You can opt-out of this behavior with the new `no_wait_after` option. - You can use new `times` option in [`method: Page.addLocatorHandler`] to specify maximum number of times the handler should be run. - The handler in [`method: Page.addLocatorHandler`] now accepts the locator as argument. - New [`method: Page.removeLocatorHandler`] method for removing previously added locator handlers. ```python locator = page.get_by_text("This interstitial covers the button") page.add_locator_handler(locator, lambda overlay: overlay.locator("#close").click(), times=3, no_wait_after=True) # Run your tests that can be interrupted by the overlay. # ... page.remove_locator_handler(locator) ``` **Miscellaneous options** - [`method: PageAssertions.toHaveURL`] now supports `ignore_case` [option](./api/class-pageassertions#page-assertions-to-have-url-option-ignore-case). ### Browser Versions * Chromium 125.0.6422.14 * Mozilla Firefox 125.0.1 * WebKit 17.4 This version was also tested against the following stable channels: * Google Chrome 124 * Microsoft Edge 124 ## Version 1.43 ### New APIs - Method [`method: BrowserContext.clearCookies`] now supports filters to remove only some cookies. ```python # Clear all cookies. context.clear_cookies() # New: clear cookies with a particular name. context.clear_cookies(name="session-id") # New: clear cookies for a particular domain. context.clear_cookies(domain="my-origin.com") ``` - New method [`method: Locator.contentFrame`] converts a [Locator] object to a [FrameLocator]. This can be useful when you have a [Locator] object obtained somewhere, and later on would like to interact with the content inside the frame. ```python locator = page.locator("iframe[name='embedded']") # ... frame_locator = locator.content_frame frame_locator.getByRole("button").click() ``` - New method [`method: FrameLocator.owner`] converts a [FrameLocator] object to a [Locator]. This can be useful when you have a [FrameLocator] object obtained somewhere, and later on would like to interact with the `iframe` element. ```python frame_locator = page.frame_locator("iframe[name='embedded']") # ... locator = frame_locator.owner expect(locator).to_be_visible() ``` - Conda builds are now published for macOS-arm64 and Linux-arm64. ### Browser Versions * Chromium 124.0.6367.8 * Mozilla Firefox 124.0 * WebKit 17.4 This version was also tested against the following stable channels: * Google Chrome 123 * Microsoft Edge 123 ## Version 1.42 ### New Locator Handler New method [`method: Page.addLocatorHandler`] registers a callback that will be invoked when specified element becomes visible and may block Playwright actions. The callback can get rid of the overlay. Here is an example that closes a cookie dialog when it appears. ```python # Setup the handler. page.add_locator_handler( page.get_by_role("heading", name="Hej! You are in control of your cookies."), lambda: page.get_by_role("button", name="Accept all").click(), ) # Write the test as usual. page.goto("https://www.ikea.com/") page.get_by_role("link", name="Collection of blue and white").click() expect(page.get_by_role("heading", name="Light and easy")).to_be_visible() ``` ### New APIs - [`method: Page.pdf`] accepts two new options [`option: Page.pdf.tagged`] and [`option: Page.pdf.outline`]. ### Announcements * ⚠ïļ Ubuntu 18 is not supported anymore. ### Browser Versions * Chromium 123.0.6312.4 * Mozilla Firefox 123.0 * WebKit 17.4 This version was also tested against the following stable channels: * Google Chrome 122 * Microsoft Edge 123 ## Version 1.41 ### New APIs - New method [`method: Page.unrouteAll`] removes all routes registered by [`method: Page.route`] and [`method: Page.routeFromHAR`]. Optionally allows to wait for ongoing routes to finish, or ignore any errors from them. - New method [`method: BrowserContext.unrouteAll`] removes all routes registered by [`method: BrowserContext.route`] and [`method: BrowserContext.routeFromHAR`]. Optionally allows to wait for ongoing routes to finish, or ignore any errors from them. - New options [`option: Page.screenshot.style`] in [`method: Page.screenshot`] and [`option: Locator.screenshot.style`] in [`method: Locator.screenshot`] to add custom CSS to the page before taking a screenshot. ### Browser Versions * Chromium 121.0.6167.57 * Mozilla Firefox 121.0 * WebKit 17.4 This version was also tested against the following stable channels: * Google Chrome 120 * Microsoft Edge 120 ## Version 1.40 ### Test Generator Update ![Playwright Test Generator](https://github.com/microsoft/playwright/assets/9881434/e8d67e2e-f36d-4301-8631-023948d3e190) New tools to generate assertions: - "Assert visibility" tool generates [`method: LocatorAssertions.toBeVisible`]. - "Assert value" tool generates [`method: LocatorAssertions.toHaveValue`]. - "Assert text" tool generates [`method: LocatorAssertions.toContainText`]. Here is an example of a generated test with assertions: ```python from playwright.sync_api import Page, expect def test_example(page: Page) -> None: page.goto("https://playwright.dev/") page.get_by_role("link", name="Get started").click() expect(page.get_by_label("Breadcrumbs").get_by_role("list")).to_contain_text("Installation") expect(page.get_by_label("Search")).to_be_visible() page.get_by_label("Search").click() page.get_by_placeholder("Search docs").fill("locator") expect(page.get_by_placeholder("Search docs")).to_have_value("locator"); ``` ### New APIs - Options [`option: Page.close.reason`] in [`method: Page.close`], [`option: BrowserContext.close.reason`] in [`method: BrowserContext.close`] and [`option: Browser.close.reason`] in [`method: Browser.close`]. Close reason is reported for all operations interrupted by the closure. - Option [`option: BrowserType.launchPersistentContext.firefoxUserPrefs`] in [`method: BrowserType.launchPersistentContext`]. ### Other Changes - Method [`method: Download.path`] throws an error for failed and cancelled downloads. ### Browser Versions * Chromium 120.0.6099.28 * Mozilla Firefox 119.0 * WebKit 17.4 This version was also tested against the following stable channels: * Google Chrome 119 * Microsoft Edge 119 ## Version 1.39 Evergreen browsers update. ### Browser Versions * Chromium 119.0.6045.9 * Mozilla Firefox 118.0.1 * WebKit 17.4 This version was also tested against the following stable channels: * Google Chrome 118 * Microsoft Edge 118 ## Version 1.38 ### Trace Viewer Updates ![Playwright Trace Viewer](https://github.com/microsoft/playwright/assets/746130/0c41e20d-c54b-4600-8ca8-1cbb6393ddef) 1. Zoom into time range. 1. Network panel redesign. ### New APIs - [`event: BrowserContext.webError`] - [`method: Locator.pressSequentially`] ### Deprecations * The following methods were deprecated: [`method: Page.type`], [`method: Frame.type`], [`method: Locator.type`] and [`method: ElementHandle.type`]. Please use [`method: Locator.fill`] instead which is much faster. Use [`method: Locator.pressSequentially`] only if there is a special keyboard handling on the page, and you need to press keys one-by-one. ### Browser Versions * Chromium 117.0.5938.62 * Mozilla Firefox 117.0 * WebKit 17.0 This version was also tested against the following stable channels: * Google Chrome 116 * Microsoft Edge 116 ## Version 1.37 ### Highlights * New [--full-page-screenshot](./test-runners.md#cli-arguments) command line flag allows taking a full page screenshot on failure. * It is now possible to override the context options for a single test by using the [browser_context_args](./test-runners.md#fixtures) marker. * `pytest-playwright` is now also getting published [on Anaconda](https://anaconda.org/Microsoft/pytest-playwright/) ### 📚 Debian 12 Bookworm Support Playwright now supports Debian 12 Bookworm on both x86_64 and arm64 for Chromium, Firefox and WebKit. Let us know if you encounter any issues! Linux support looks like this: | | Ubuntu 20.04 | Ubuntu 22.04 | Debian 11 | Debian 12 | | :--- | :---: | :---: | :---: | :---: | | Chromium | ✅ | ✅ | ✅ | ✅ | | WebKit | ✅ | ✅ | ✅ | ✅ | | Firefox | ✅ | ✅ | ✅ | ✅ | ### Browser Versions * Chromium 116.0.5845.82 * Mozilla Firefox 115.0 * WebKit 17.0 This version was also tested against the following stable channels: * Google Chrome 115 * Microsoft Edge 115 ## Version 1.36 🏝ïļ Summer maintenance release. ### Browser Versions * Chromium 115.0.5790.75 * Mozilla Firefox 115.0 * WebKit 17.0 This version was also tested against the following stable channels: * Google Chrome 114 * Microsoft Edge 114 ## Version 1.35 ### Highlights * New option `mask_color` for methods [`method: Page.screenshot`] and [`method: Locator.screenshot`] to change default masking color. * New `uninstall` CLI command to uninstall browser binaries: ```bash $ playwright uninstall # remove browsers installed by this installation $ playwright uninstall --all # remove all ever-install Playwright browsers ``` ### Browser Versions * Chromium 115.0.5790.13 * Mozilla Firefox 113.0 * WebKit 16.4 This version was also tested against the following stable channels: * Google Chrome 114 * Microsoft Edge 114 ## Version 1.34 ### Highlights * New [`method: Locator.and`] to create a locator that matches both locators. ```python button = page.get_by_role("button").and_(page.get_by_title("Subscribe")) ``` * New events [`event: BrowserContext.console`] and [`event: BrowserContext.dialog`] to subscribe to any dialogs and console messages from any page from the given browser context. Use the new methods [`method: ConsoleMessage.page`] and [`method: Dialog.page`] to pin-point event source. ### Browser Versions * Chromium 114.0.5735.26 * Mozilla Firefox 113.0 * WebKit 16.4 This version was also tested against the following stable channels: * Google Chrome 113 * Microsoft Edge 113 ## Version 1.33 ### Locators Update * Use [`method: Locator.or`] to create a locator that matches either of the two locators. Consider a scenario where you'd like to click on a "New email" button, but sometimes a security settings dialog shows up instead. In this case, you can wait for either a "New email" button, or a dialog and act accordingly: ```python new_email = page.get_by_role("button", name="New email") dialog = page.get_by_text("Confirm security settings") expect(new_email.or_(dialog)).is_visible() if (dialog.is_visible()): page.get_by_role("button", name="Dismiss").click() new_email.click() ``` * Use new options [`option: Locator.filter.hasNot`] and [`option: Locator.filter.hasNotText`] in [`method: Locator.filter`] to find elements that **do not match** certain conditions. ```python row_locator = page.locator("tr") row_locator.filter(has_not_text="text in column 1").filter( has_not=page.get_by_role("button", name="column 2 button") ).screenshot() ``` * Use new web-first assertion [`method: LocatorAssertions.toBeAttached`] to ensure that the element is present in the page's DOM. Do not confuse with the [`method: LocatorAssertions.toBeVisible`] that ensures that element is both attached & visible. ### New APIs - [`method: Locator.or`] - New option [`option: Locator.filter.hasNot`] in [`method: Locator.filter`] - New option [`option: Locator.filter.hasNotText`] in [`method: Locator.filter`] - [`method: LocatorAssertions.toBeAttached`] - New option [`option: Route.fetch.timeout`] in [`method: Route.fetch`] ### ⚠ïļ Breaking change * The `mcr.microsoft.com/playwright/python:v1.33.0` now serves a Playwright image based on Ubuntu Jammy. To use the focal-based image, please use `mcr.microsoft.com/playwright/python:v1.33.0-focal` instead. ### Browser Versions * Chromium 113.0.5672.53 * Mozilla Firefox 112.0 * WebKit 16.4 This version was also tested against the following stable channels: * Google Chrome 112 * Microsoft Edge 112 ## Version 1.32 ### New APIs - Custom expect message, see [test assertions documentation](./test-assertions.md#custom-expect-message). - New options [`option: Page.routeFromHAR.updateMode`] and [`option: Page.routeFromHAR.updateContent`] in [`method: Page.routeFromHAR`] and [`method: BrowserContext.routeFromHAR`]. - Chaining existing locator objects, see [locator docs](./locators.md#matching-inside-a-locator) for details. - New option [`option: Tracing.startChunk.name`] in method [`method: Tracing.startChunk`]. ### Browser Versions * Chromium 112.0.5615.29 * Mozilla Firefox 111.0 * WebKit 16.4 This version was also tested against the following stable channels: * Google Chrome 111 * Microsoft Edge 111 ## Version 1.31 ### New APIs - New assertion [`method: LocatorAssertions.toBeInViewport`] ensures that locator points to an element that intersects viewport, according to the [intersection observer API](https://developer.mozilla.org/en-US/docs/Web/API/Intersection_Observer_API). ```python from playwright.sync_api import expect locator = page.get_by_role("button") # Make sure at least some part of element intersects viewport. expect(locator).to_be_in_viewport() # Make sure element is fully outside of viewport. expect(locator).not_to_be_in_viewport() # Make sure that at least half of the element intersects viewport. expect(locator).to_be_in_viewport(ratio=0.5) ``` ### Miscellaneous - DOM snapshots in trace viewer can be now opened in a separate window. - New option [`option: Route.fetch.maxRedirects`] for method [`method: Route.fetch`]. - Playwright now supports Debian 11 arm64. - Official [docker images](./docker.md) now include Node 18 instead of Node 16. ### Browser Versions * Chromium 111.0.5563.19 * Mozilla Firefox 109.0 * WebKit 16.4 This version was also tested against the following stable channels: * Google Chrome 110 * Microsoft Edge 110 ## Version 1.30 ### Browser Versions * Chromium 110.0.5481.38 * Mozilla Firefox 108.0.2 * WebKit 16.4 This version was also tested against the following stable channels: * Google Chrome 109 * Microsoft Edge 109 ## Version 1.29 ### New APIs - New method [`method: Route.fetch`] and new option `json` for [`method: Route.fulfill`]: ```python def handle_route(route: Route): # Fetch original settings. response = route.fetch() # Force settings theme to a predefined value. json = response.json() json["theme"] = "Solorized" # Fulfill with modified data. route.fulfill(json=json) page.route("**/api/settings", handle_route) ``` - New method [`method: Locator.all`] to iterate over all matching elements: ```python # Check all checkboxes! checkboxes = page.get_by_role("checkbox") for checkbox in checkboxes.all(): checkbox.check() ``` - [`method: Locator.selectOption`] matches now by value or label: ```html ``` ```python element.select_option("Red") ``` ### Miscellaneous - Option `postData` in method [`method: Route.continue`] now supports [Serializable] values. ### Browser Versions * Chromium 109.0.5414.46 * Mozilla Firefox 107.0 * WebKit 16.4 This version was also tested against the following stable channels: * Google Chrome 108 * Microsoft Edge 108 ## Version 1.28 ### Playwright Tools * **Live Locators in CodeGen.** Generate a locator for any element on the page using "Explore" tool. ![Locator Explorer](https://user-images.githubusercontent.com/9798949/202293514-8e2eade6-c809-4b0a-864b-899dfcee3d84.png) ### New APIs - [`method: Locator.blur`] - [`method: Locator.clear`] ### Browser Versions * Chromium 108.0.5359.29 * Mozilla Firefox 106.0 * WebKit 16.4 This version was also tested against the following stable channels: * Google Chrome 107 * Microsoft Edge 107 ## Version 1.27 ### Locators With these new APIs writing locators is a joy: - [`method: Page.getByText`] to locate by text content. - [`method: Page.getByRole`] to locate by [ARIA role](https://www.w3.org/TR/wai-aria-1.2/#roles), [ARIA attributes](https://www.w3.org/TR/wai-aria-1.2/#aria-attributes) and [accessible name](https://w3c.github.io/accname/#dfn-accessible-name). - [`method: Page.getByLabel`] to locate a form control by associated label's text. - [`method: Page.getByTestId`] to locate an element based on its `data-testid` attribute (other attribute can be configured). - [`method: Page.getByPlaceholder`] to locate an input by placeholder. - [`method: Page.getByAltText`] to locate an element, usually image, by its text alternative. - [`method: Page.getByTitle`] to locate an element by its title. ```python page.get_by_label("User Name").fill("John") page.get_by_label("Password").fill("secret-password") page.get_by_role("button", name="Sign in").click() expect(page.get_by_text("Welcome, John!")).to_be_visible() ``` All the same methods are also available on [Locator], [FrameLocator] and [Frame] classes. ### Other highlights - As announced in v1.25, Ubuntu 18 will not be supported as of Dec 2022. In addition to that, there will be no WebKit updates on Ubuntu 18 starting from the next Playwright release. ### Behavior Changes - [`method: LocatorAssertions.toHaveAttribute`] with an empty value does not match missing attribute anymore. For example, the following snippet will succeed when `button` **does not** have a `disabled` attribute. ```python expect(page.get_by_role("button")).to_have_attribute("disabled", "") ``` ### Browser Versions * Chromium 107.0.5304.18 * Mozilla Firefox 105.0.1 * WebKit 16.0 This version was also tested against the following stable channels: * Google Chrome 106 * Microsoft Edge 106 ## Version 1.26 ### Assertions - New option `enabled` for [`method: LocatorAssertions.toBeEnabled`]. - [`method: LocatorAssertions.toHaveText`] now pierces open shadow roots. - New option `editable` for [`method: LocatorAssertions.toBeEditable`]. - New option `visible` for [`method: LocatorAssertions.toBeVisible`]. ### Other highlights - New option `max_redirects` for [`method: APIRequestContext.get`] and others to limit redirect count. - Python 3.11 is now supported. ### Behavior Change A bunch of Playwright APIs already support the `wait_until: "domcontentloaded"` option. For example: ```python page.goto("https://playwright.dev", wait_until="domcontentloaded") ``` Prior to 1.26, this would wait for all iframes to fire the `DOMContentLoaded` event. To align with web specification, the `'domcontentloaded'` value only waits for the target frame to fire the `'DOMContentLoaded'` event. Use `wait_until="load"` to wait for all iframes. ### Browser Versions * Chromium 106.0.5249.30 * Mozilla Firefox 104.0 * WebKit 16.0 This version was also tested against the following stable channels: * Google Chrome 105 * Microsoft Edge 105 ## Version 1.25 ### Announcements * 🎁 We now ship Ubuntu 22.04 Jammy Jellyfish docker image: `mcr.microsoft.com/playwright/python:v1.34.0-jammy`. * ðŸŠĶ This is the last release with macOS 10.15 support (deprecated as of 1.21). * ⚠ïļ Ubuntu 18 is now deprecated and will not be supported as of Dec 2022. ### Browser Versions * Chromium 105.0.5195.19 * Mozilla Firefox 103.0 * WebKit 16.0 This version was also tested against the following stable channels: * Google Chrome 104 * Microsoft Edge 104 ## Version 1.24
### 🐂 Debian 11 Bullseye Support Playwright now supports Debian 11 Bullseye on x86_64 for Chromium, Firefox and WebKit. Let us know if you encounter any issues! Linux support looks like this: | | Ubuntu 20.04 | Ubuntu 22.04 | Debian 11 | :--- | :---: | :---: | :---: | :---: | | Chromium | ✅ | ✅ | ✅ | | WebKit | ✅ | ✅ | ✅ | | Firefox | ✅ | ✅ | ✅ | ### New introduction docs We rewrote our Getting Started docs to be more end-to-end testing focused. Check them out on [playwright.dev](https://playwright.dev/python/docs/intro). ## Version 1.23 ### Network Replay Now you can record network traffic into a HAR file and re-use this traffic in your tests. To record network into HAR file: ```bash npx playwright open --save-har=github.har.zip https://github.com/microsoft ``` Alternatively, you can record HAR programmatically: ```python async context = await browser.new_context(record_har_path="github.har.zip") # ... do stuff ... await context.close() ``` ```python sync context = browser.new_context(record_har_path="github.har.zip") # ... do stuff ... context.close() ``` Use the new methods [`method: Page.routeFromHAR`] or [`method: BrowserContext.routeFromHAR`] to serve matching responses from the [HAR](http://www.softwareishard.com/blog/har-12-spec/) file: ```python async await context.route_from_har("github.har.zip") ``` ```python sync context.route_from_har("github.har.zip") ``` Read more in [our documentation](./mock.md#mocking-with-har-files). ### Advanced Routing You can now use [`method: Route.fallback`] to defer routing to other handlers. Consider the following example: ```python async # Remove a header from all requests async def remove_header_handler(route: Route) -> None: headers = await route.request.all_headers() if "if-none-match" in headers: del headers["if-none-match"] await route.fallback(headers=headers) await page.route("**/*", remove_header_handler) # Abort all images async def abort_images_handler(route: Route) -> None: if route.request.resource_type == "image": await route.abort() else: await route.fallback() await page.route("**/*", abort_images_handler) ``` ```python sync # Remove a header from all requests def remove_header_handler(route: Route) -> None: headers = route.request.all_headers() if "if-none-match" in headers: del headers["if-none-match"] route.fallback(headers=headers) page.route("**/*", remove_header_handler) # Abort all images def abort_images_handler(route: Route) -> None: if route.request.resource_type == "image": route.abort() else: route.fallback() page.route("**/*", abort_images_handler) ``` Note that the new methods [`method: Page.routeFromHAR`] and [`method: BrowserContext.routeFromHAR`] also participate in routing and could be deferred to. ### Web-First Assertions Update * New method [`method: LocatorAssertions.toHaveValues`] that asserts all selected values of `