From 4d31bdc3154653d3234457efa9cd97f0dadf83b5 Mon Sep 17 00:00:00 2001 From: Max Schmitt Date: Fri, 11 Oct 2024 11:33:33 +0200 Subject: [PATCH] fix(asyncio): already cancelled tasks ends up in 'InvalidStateError: invalid state' (#2593) --- playwright/_impl/_connection.py | 2 ++ tests/async/test_asyncio.py | 22 +++++++++++++++++++++- 2 files changed, 23 insertions(+), 1 deletion(-) diff --git a/playwright/_impl/_connection.py b/playwright/_impl/_connection.py index c15d82e79..19b68fb13 100644 --- a/playwright/_impl/_connection.py +++ b/playwright/_impl/_connection.py @@ -294,6 +294,8 @@ def cleanup(self, cause: str = None) -> None: # To prevent 'Future exception was never retrieved' we ignore all callbacks that are no_reply. if callback.no_reply: continue + if callback.future.cancelled(): + continue callback.future.set_exception(self._closed_error) self._callbacks.clear() self.emit("close") diff --git a/tests/async/test_asyncio.py b/tests/async/test_asyncio.py index 1d4423afb..33edc71ce 100644 --- a/tests/async/test_asyncio.py +++ b/tests/async/test_asyncio.py @@ -13,11 +13,12 @@ # limitations under the License. import asyncio import gc +import sys from typing import Dict import pytest -from playwright.async_api import async_playwright +from playwright.async_api import Page, async_playwright from tests.server import Server from tests.utils import TARGET_CLOSED_ERROR_MESSAGE @@ -67,3 +68,22 @@ async def test_cancel_pending_protocol_call_on_playwright_stop(server: Server) - with pytest.raises(Exception) as exc_info: await pending_task assert TARGET_CLOSED_ERROR_MESSAGE in str(exc_info.value) + + +async def test_should_not_throw_with_taskgroup(page: Page) -> None: + if sys.version_info < (3, 11): + pytest.skip("TaskGroup is only available in Python 3.11+") + + from builtins import ExceptionGroup # type: ignore + + async def raise_exception() -> None: + raise ValueError("Something went wrong") + + with pytest.raises(ExceptionGroup) as exc_info: + async with asyncio.TaskGroup() as group: # type: ignore + group.create_task(page.locator(".this-element-does-not-exist").inner_text()) + group.create_task(raise_exception()) + assert len(exc_info.value.exceptions) == 1 + assert "Something went wrong" in str(exc_info.value.exceptions[0]) + assert isinstance(exc_info.value.exceptions[0], ValueError) + assert await page.evaluate("() => 11 * 11") == 121