From 89b64d7e7318c5607700d9a7dfd12b22384b35ad Mon Sep 17 00:00:00 2001 From: Marcel Wilson Date: Wed, 7 Feb 2024 16:29:29 -0600 Subject: [PATCH] Ruff q-z (#46) * RET * TCH * UP * YTT * fixing mypy configuration --- mypy.ini | 2 +- pyproject.toml | 26 ++++++++-------- screenpy_selenium/abilities/browse_the_web.py | 18 ++++++----- screenpy_selenium/actions/accept_alert.py | 6 +++- screenpy_selenium/actions/chain.py | 6 +++- screenpy_selenium/actions/clear.py | 14 +++++---- screenpy_selenium/actions/click.py | 18 ++++++----- screenpy_selenium/actions/dismiss_alert.py | 6 +++- screenpy_selenium/actions/double_click.py | 19 +++++++----- screenpy_selenium/actions/enter.py | 23 +++++++------- screenpy_selenium/actions/enter_2fa_token.py | 15 ++++++---- screenpy_selenium/actions/go_back.py | 6 +++- screenpy_selenium/actions/go_forward.py | 6 +++- screenpy_selenium/actions/hold_down.py | 23 +++++++------- screenpy_selenium/actions/move_mouse.py | 29 ++++++++++-------- screenpy_selenium/actions/open.py | 12 ++++---- screenpy_selenium/actions/pause.py | 8 +++-- screenpy_selenium/actions/refresh_page.py | 6 +++- screenpy_selenium/actions/release.py | 18 +++++------ .../actions/respond_to_the_prompt.py | 8 +++-- screenpy_selenium/actions/right_click.py | 19 +++++++----- screenpy_selenium/actions/save_console_log.py | 10 ++++--- screenpy_selenium/actions/save_screenshot.py | 10 ++++--- screenpy_selenium/actions/select.py | 26 ++++++++-------- screenpy_selenium/actions/switch_to.py | 17 ++++++----- screenpy_selenium/actions/switch_to_tab.py | 6 +++- screenpy_selenium/actions/wait.py | 18 ++++++----- screenpy_selenium/protocols.py | 7 +++-- screenpy_selenium/questions/attribute.py | 16 +++++----- screenpy_selenium/questions/browser_title.py | 6 +++- screenpy_selenium/questions/browser_url.py | 6 +++- screenpy_selenium/questions/cookies.py | 6 +++- screenpy_selenium/questions/element.py | 15 ++++++---- screenpy_selenium/questions/list.py | 20 +++++++------ screenpy_selenium/questions/number.py | 10 ++++--- screenpy_selenium/questions/selected.py | 18 ++++++----- screenpy_selenium/questions/text.py | 18 ++++++----- .../questions/text_of_the_alert.py | 6 +++- .../custom_matchers/is_clickable_element.py | 12 ++++---- .../custom_matchers/is_invisible_element.py | 12 ++++---- .../custom_matchers/is_present_element.py | 12 ++++---- .../custom_matchers/is_visible_element.py | 12 ++++---- screenpy_selenium/resolutions/is_clickable.py | 6 +++- screenpy_selenium/resolutions/is_invisible.py | 6 +++- screenpy_selenium/resolutions/is_present.py | 6 +++- screenpy_selenium/resolutions/is_visible.py | 6 +++- screenpy_selenium/target.py | 30 +++++++++---------- setup.py | 4 +-- tests/test_questions.py | 2 +- tests/test_resolutions.py | 7 +++-- tests/test_target.py | 6 +++- tests/useful_mocks.py | 14 ++++----- 52 files changed, 384 insertions(+), 254 deletions(-) diff --git a/mypy.ini b/mypy.ini index 15b03d9..ff3681c 100644 --- a/mypy.ini +++ b/mypy.ini @@ -6,7 +6,7 @@ exclude = (?x)( | docs/ ) -[mypy-screenpy.*] +[mypy-screenpy_selenium.*] disallow_untyped_defs = True [mypy-tests.*] diff --git a/pyproject.toml b/pyproject.toml index 80f3518..7f84720 100644 --- a/pyproject.toml +++ b/pyproject.toml @@ -138,18 +138,18 @@ select = [ "PIE", # flake8-pie "PL", # pylint "PT", # flake8-pytest-style -# "Q", # flake8-quotes -# "RET", # flake8-return -# "RSE", # flake8-raise -# "RUF", # ruff specific -# "SIM", # flake8-simplify -# "T10", # flake8-debugger -# "T20", # flake8-print -# "TCH", # flake8-type-checking -# "TRY", # tryceratops -# "UP", # python upgrade + "Q", # flake8-quotes + "RET", # flake8-return + "RSE", # flake8-raise + "RUF", # ruff specific + "SIM", # flake8-simplify + "T10", # flake8-debugger + "T20", # flake8-print + "TCH", # flake8-type-checking + "TRY", # tryceratops + "UP", # python upgrade "W", # pycodestyle warning -# "YTT", # flake8-2020 + "YTT", # flake8-2020 # we would like these someday, but not yet # "FURB", # refurb @@ -165,13 +165,15 @@ ignore = [ extend-safe-fixes = [ "EM101", "EM102", -# "TCH001", "TCH002", "TCH003", "TCH004", + "TCH001", "TCH002", "TCH003", "TCH004", # "SIM108" # maybe? # "F841", "C419", "D200", "D205", "D415", "PT003", "PT006", "PT018", + "RET504", + "UP007", ] [tool.ruff.lint.per-file-ignores] diff --git a/screenpy_selenium/abilities/browse_the_web.py b/screenpy_selenium/abilities/browse_the_web.py index f401f8f..7a8cd2a 100644 --- a/screenpy_selenium/abilities/browse_the_web.py +++ b/screenpy_selenium/abilities/browse_the_web.py @@ -3,13 +3,15 @@ from __future__ import annotations import os -from typing import Type, TypeVar +from typing import TYPE_CHECKING, TypeVar from selenium.webdriver import Chrome, Firefox, Remote, Safari -from selenium.webdriver.remote.webdriver import WebDriver from ..exceptions import BrowsingError +if TYPE_CHECKING: + from selenium.webdriver.remote.webdriver import WebDriver + DEFAULT_APPIUM_HUB_URL = "http://localhost:4723/wd/hub" @@ -33,22 +35,22 @@ class BrowseTheWeb: browser: WebDriver @classmethod - def using_chrome(cls: Type[SelfBrowseTheWeb]) -> SelfBrowseTheWeb: + def using_chrome(cls: type[SelfBrowseTheWeb]) -> SelfBrowseTheWeb: """Create and use a default Chrome Selenium webdriver instance.""" return cls.using(browser=Chrome()) @classmethod - def using_firefox(cls: Type[SelfBrowseTheWeb]) -> SelfBrowseTheWeb: + def using_firefox(cls: type[SelfBrowseTheWeb]) -> SelfBrowseTheWeb: """Create and use a default Firefox Selenium webdriver instance.""" return cls.using(browser=Firefox()) @classmethod - def using_safari(cls: Type[SelfBrowseTheWeb]) -> SelfBrowseTheWeb: + def using_safari(cls: type[SelfBrowseTheWeb]) -> SelfBrowseTheWeb: """Create and use a default Safari Selenium webdriver instance.""" return cls.using(browser=Safari()) @classmethod - def using_ios(cls: Type[SelfBrowseTheWeb]) -> SelfBrowseTheWeb: + def using_ios(cls: type[SelfBrowseTheWeb]) -> SelfBrowseTheWeb: """ Create and use a default Remote driver instance. @@ -81,7 +83,7 @@ def using_ios(cls: Type[SelfBrowseTheWeb]) -> SelfBrowseTheWeb: return cls.using(browser=Remote(hub_url, IOS_CAPABILITIES)) @classmethod - def using_android(cls: Type[SelfBrowseTheWeb]) -> SelfBrowseTheWeb: + def using_android(cls: type[SelfBrowseTheWeb]) -> SelfBrowseTheWeb: """ Create and use a default Remote driver instance. @@ -114,7 +116,7 @@ def using_android(cls: Type[SelfBrowseTheWeb]) -> SelfBrowseTheWeb: return cls.using(browser=Remote(hub_url, ANDROID_CAPABILITIES)) @classmethod - def using(cls: Type[SelfBrowseTheWeb], browser: WebDriver) -> SelfBrowseTheWeb: + def using(cls: type[SelfBrowseTheWeb], browser: WebDriver) -> SelfBrowseTheWeb: """Provide an already-set-up WebDriver to use to browse the web.""" return cls(browser=browser) diff --git a/screenpy_selenium/actions/accept_alert.py b/screenpy_selenium/actions/accept_alert.py index e087d90..c59572a 100644 --- a/screenpy_selenium/actions/accept_alert.py +++ b/screenpy_selenium/actions/accept_alert.py @@ -2,11 +2,15 @@ from __future__ import annotations -from screenpy.actor import Actor +from typing import TYPE_CHECKING + from screenpy.pacing import aside, beat from ..abilities import BrowseTheWeb +if TYPE_CHECKING: + from screenpy.actor import Actor + class AcceptAlert: """Accept an alert! diff --git a/screenpy_selenium/actions/chain.py b/screenpy_selenium/actions/chain.py index ee7da10..75dcdfb 100644 --- a/screenpy_selenium/actions/chain.py +++ b/screenpy_selenium/actions/chain.py @@ -2,7 +2,8 @@ from __future__ import annotations -from screenpy.actor import Actor +from typing import TYPE_CHECKING + from screenpy.exceptions import UnableToAct from screenpy.pacing import beat from selenium.webdriver.common.action_chains import ActionChains @@ -10,6 +11,9 @@ from ..abilities import BrowseTheWeb from ..protocols import Chainable +if TYPE_CHECKING: + from screenpy.actor import Actor + class Chain: """Group a series of chainable Actions together. diff --git a/screenpy_selenium/actions/clear.py b/screenpy_selenium/actions/clear.py index 1cac23c..ea0a621 100644 --- a/screenpy_selenium/actions/clear.py +++ b/screenpy_selenium/actions/clear.py @@ -2,14 +2,16 @@ from __future__ import annotations -from typing import Type, TypeVar +from typing import TYPE_CHECKING, TypeVar -from screenpy.actor import Actor from screenpy.exceptions import DeliveryError from screenpy.pacing import beat from selenium.common.exceptions import WebDriverException -from ..target import Target +if TYPE_CHECKING: + from screenpy.actor import Actor + + from ..target import Target SelfClear = TypeVar("SelfClear", bound="Clear") @@ -26,7 +28,7 @@ class Clear: """ @classmethod - def the_text_from_the(cls: Type[SelfClear], target: Target) -> SelfClear: + def the_text_from_the(cls: type[SelfClear], target: Target) -> SelfClear: """ Specify the Target from which to clear the text. @@ -37,13 +39,13 @@ def the_text_from_the(cls: Type[SelfClear], target: Target) -> SelfClear: return cls(target=target) @classmethod - def the_text_from(cls: Type[SelfClear], target: Target) -> SelfClear: + def the_text_from(cls: type[SelfClear], target: Target) -> SelfClear: """Alias for :meth:`~screenpy_selenium.actions.Clear.the_text_from_the`.""" return cls.the_text_from_the(target=target) @classmethod def the_text_from_the_first_of_the( - cls: Type[SelfClear], target: Target + cls: type[SelfClear], target: Target ) -> SelfClear: """Alias for :meth:`~screenpy_selenium.actions.Clear.the_text_from_the`.""" return cls.the_text_from_the(target=target) diff --git a/screenpy_selenium/actions/click.py b/screenpy_selenium/actions/click.py index 995c45f..c9093ec 100644 --- a/screenpy_selenium/actions/click.py +++ b/screenpy_selenium/actions/click.py @@ -2,15 +2,17 @@ from __future__ import annotations -from typing import Optional, Type, TypeVar +from typing import TYPE_CHECKING, TypeVar -from screenpy.actor import Actor from screenpy.exceptions import DeliveryError, UnableToAct from screenpy.pacing import beat from selenium.common.exceptions import WebDriverException -from selenium.webdriver.common.action_chains import ActionChains -from ..target import Target +if TYPE_CHECKING: + from screenpy.actor import Actor + from selenium.webdriver.common.action_chains import ActionChains + + from ..target import Target SelfClick = TypeVar("SelfClick", bound="Click") @@ -31,7 +33,7 @@ class Click: """ @classmethod - def on_the(cls: Type[SelfClick], target: Target) -> SelfClick: + def on_the(cls: type[SelfClick], target: Target) -> SelfClick: """ Target the element to click on. @@ -42,12 +44,12 @@ def on_the(cls: Type[SelfClick], target: Target) -> SelfClick: return cls(target=target) @classmethod - def on(cls: Type[SelfClick], target: Target) -> SelfClick: + def on(cls: type[SelfClick], target: Target) -> SelfClick: """Alias for :meth:`~screenpy_selenium.actions.Click.on_the`.""" return cls.on_the(target=target) @classmethod - def on_the_first_of_the(cls: Type[SelfClick], target: Target) -> SelfClick: + def on_the_first_of_the(cls: type[SelfClick], target: Target) -> SelfClick: """Alias for :meth:`~screenpy_selenium.actions.Click.on_the`.""" return cls.on_the(target=target) @@ -88,6 +90,6 @@ def add_to_chain( the_chain.click(on_element=the_element) - def __init__(self: SelfClick, target: Optional[Target] = None) -> None: + def __init__(self: SelfClick, target: Target | None = None) -> None: self.target = target self.description = f" on the {target}" if target is not None else "" diff --git a/screenpy_selenium/actions/dismiss_alert.py b/screenpy_selenium/actions/dismiss_alert.py index e1eb8d7..de05d92 100644 --- a/screenpy_selenium/actions/dismiss_alert.py +++ b/screenpy_selenium/actions/dismiss_alert.py @@ -2,11 +2,15 @@ from __future__ import annotations -from screenpy.actor import Actor +from typing import TYPE_CHECKING + from screenpy.pacing import aside, beat from ..abilities import BrowseTheWeb +if TYPE_CHECKING: + from screenpy.actor import Actor + class DismissAlert: """Dismiss an alert. diff --git a/screenpy_selenium/actions/double_click.py b/screenpy_selenium/actions/double_click.py index b42dd64..c22728a 100644 --- a/screenpy_selenium/actions/double_click.py +++ b/screenpy_selenium/actions/double_click.py @@ -2,14 +2,17 @@ from __future__ import annotations -from typing import Optional, Type, TypeVar +from typing import TYPE_CHECKING, TypeVar -from screenpy.actor import Actor from screenpy.pacing import beat from selenium.webdriver.common.action_chains import ActionChains from ..abilities import BrowseTheWeb -from ..target import Target + +if TYPE_CHECKING: + from screenpy.actor import Actor + + from ..target import Target SelfDoubleClick = TypeVar("SelfDoubleClick", bound="DoubleClick") @@ -27,10 +30,10 @@ class DoubleClick: the_actor.attempts_to(Chain(DoubleClick())) """ - target: Optional[Target] + target: Target | None @classmethod - def on_the(cls: Type[SelfDoubleClick], target: Target) -> SelfDoubleClick: + def on_the(cls: type[SelfDoubleClick], target: Target) -> SelfDoubleClick: """ Target the element to double-click on. @@ -41,13 +44,13 @@ def on_the(cls: Type[SelfDoubleClick], target: Target) -> SelfDoubleClick: return cls(target=target) @classmethod - def on(cls: Type[SelfDoubleClick], target: Target) -> SelfDoubleClick: + def on(cls: type[SelfDoubleClick], target: Target) -> SelfDoubleClick: """Alias for :meth:`~screenpy_selenium.actions.DoubleClick.on_the`.""" return cls.on_the(target=target) @classmethod def on_the_first_of_the( - cls: Type[SelfDoubleClick], target: Target + cls: type[SelfDoubleClick], target: Target ) -> SelfDoubleClick: """Alias for :meth:`~screenpy_selenium.actions.DoubleClick.on_the`.""" return cls.on_the(target=target) @@ -82,6 +85,6 @@ def add_to_chain( """Add the DoubleClick Action to a Chain of Actions.""" self._add_action_to_chain(the_actor, the_chain) - def __init__(self: SelfDoubleClick, target: Optional[Target] = None) -> None: + def __init__(self: SelfDoubleClick, target: Target | None = None) -> None: self.target = target self.description = f" on the {target}" if target is not None else "" diff --git a/screenpy_selenium/actions/enter.py b/screenpy_selenium/actions/enter.py index 8ab586b..fcdba36 100644 --- a/screenpy_selenium/actions/enter.py +++ b/screenpy_selenium/actions/enter.py @@ -3,16 +3,19 @@ from __future__ import annotations from functools import partial -from typing import List, Optional, Type, TypeVar +from typing import TYPE_CHECKING, TypeVar -from screenpy import Actor from screenpy.exceptions import DeliveryError, UnableToAct from screenpy.pacing import aside, beat from selenium.common.exceptions import WebDriverException -from selenium.webdriver.common.action_chains import ActionChains from ..speech_tools import KEY_NAMES -from ..target import Target + +if TYPE_CHECKING: + from screenpy import Actor + from selenium.webdriver.common.action_chains import ActionChains + + from ..target import Target SelfEnter = TypeVar("SelfEnter", bound="Enter") @@ -30,13 +33,13 @@ class Enter: ) """ - target: Optional[Target] - following_keys: List[str] + target: Target | None + following_keys: list[str] text: str text_to_log: str @classmethod - def the_text(cls: Type[SelfEnter], text: str) -> SelfEnter: + def the_text(cls: type[SelfEnter], text: str) -> SelfEnter: """Provide the text to enter into the field. Aliases: @@ -45,12 +48,12 @@ def the_text(cls: Type[SelfEnter], text: str) -> SelfEnter: return cls(text=text) @classmethod - def the_keys(cls: Type[SelfEnter], text: str) -> SelfEnter: + def the_keys(cls: type[SelfEnter], text: str) -> SelfEnter: """Alias for :meth:`~screenpy_selenium.actions.Enter.the_text`.""" return cls.the_text(text=text) @classmethod - def the_secret(cls: Type[SelfEnter], text: str) -> SelfEnter: + def the_secret(cls: type[SelfEnter], text: str) -> SelfEnter: """ Provide the text to enter into the field, but mask it in logging. @@ -62,7 +65,7 @@ def the_secret(cls: Type[SelfEnter], text: str) -> SelfEnter: return cls(text=text, mask=True) @classmethod - def the_password(cls: Type[SelfEnter], text: str) -> SelfEnter: + def the_password(cls: type[SelfEnter], text: str) -> SelfEnter: """Alias for :meth:`~screenpy_selenium.actions.Enter.the_secret`.""" return cls.the_secret(text=text) diff --git a/screenpy_selenium/actions/enter_2fa_token.py b/screenpy_selenium/actions/enter_2fa_token.py index f6b2e7e..c490e01 100644 --- a/screenpy_selenium/actions/enter_2fa_token.py +++ b/screenpy_selenium/actions/enter_2fa_token.py @@ -2,16 +2,19 @@ from __future__ import annotations -from typing import Type, TypeVar +from typing import TYPE_CHECKING, TypeVar -from screenpy.actor import Actor from screenpy.pacing import beat from screenpy_pyotp.abilities import AuthenticateWith2FA -from selenium.webdriver.common.action_chains import ActionChains -from ..target import Target from .enter import Enter +if TYPE_CHECKING: + from screenpy.actor import Actor + from selenium.webdriver.common.action_chains import ActionChains + + from ..target import Target + SelfEnter2FAToken = TypeVar("SelfEnter2FAToken", bound="Enter2FAToken") @@ -28,7 +31,7 @@ class Enter2FAToken: """ @classmethod - def into_the(cls: Type[SelfEnter2FAToken], target: Target) -> SelfEnter2FAToken: + def into_the(cls: type[SelfEnter2FAToken], target: Target) -> SelfEnter2FAToken: """ Target the element into which to enter the 2FA token. @@ -38,7 +41,7 @@ def into_the(cls: Type[SelfEnter2FAToken], target: Target) -> SelfEnter2FAToken: return cls(target) @classmethod - def into(cls: Type[SelfEnter2FAToken], target: Target) -> SelfEnter2FAToken: + def into(cls: type[SelfEnter2FAToken], target: Target) -> SelfEnter2FAToken: """Alias for :meth:`~screenpy_selenium.actions.Enter2FAToken.into_the`.""" return cls.into_the(target=target) diff --git a/screenpy_selenium/actions/go_back.py b/screenpy_selenium/actions/go_back.py index 5658e04..93afa41 100644 --- a/screenpy_selenium/actions/go_back.py +++ b/screenpy_selenium/actions/go_back.py @@ -2,11 +2,15 @@ from __future__ import annotations -from screenpy.actor import Actor +from typing import TYPE_CHECKING + from screenpy.pacing import beat from ..abilities import BrowseTheWeb +if TYPE_CHECKING: + from screenpy.actor import Actor + class GoBack: """Press the browser back button. diff --git a/screenpy_selenium/actions/go_forward.py b/screenpy_selenium/actions/go_forward.py index 673395c..f3e5de0 100644 --- a/screenpy_selenium/actions/go_forward.py +++ b/screenpy_selenium/actions/go_forward.py @@ -2,11 +2,15 @@ from __future__ import annotations -from screenpy.actor import Actor +from typing import TYPE_CHECKING + from screenpy.pacing import beat from ..abilities import BrowseTheWeb +if TYPE_CHECKING: + from screenpy.actor import Actor + class GoForward: """Press the browser forward button. diff --git a/screenpy_selenium/actions/hold_down.py b/screenpy_selenium/actions/hold_down.py index 91cbf1c..9acbac5 100644 --- a/screenpy_selenium/actions/hold_down.py +++ b/screenpy_selenium/actions/hold_down.py @@ -3,16 +3,19 @@ from __future__ import annotations import platform -from typing import Optional, Type, TypeVar +from typing import TYPE_CHECKING, TypeVar -from screenpy import Actor from screenpy.exceptions import UnableToAct from screenpy.pacing import beat -from selenium.webdriver.common.action_chains import ActionChains from selenium.webdriver.common.keys import Keys from ..speech_tools import KEY_NAMES -from ..target import Target + +if TYPE_CHECKING: + from screenpy import Actor + from selenium.webdriver.common.action_chains import ActionChains + + from ..target import Target SelfHoldDown = TypeVar("SelfHoldDown", bound="HoldDown") @@ -38,13 +41,13 @@ class HoldDown: ) """ - target: Optional[Target] - key: Optional[str] + target: Target | None + key: str | None lmb: bool description: str @classmethod - def command_or_control_key(cls: Type[SelfHoldDown]) -> SelfHoldDown: + def command_or_control_key(cls: type[SelfHoldDown]) -> SelfHoldDown: """ A convenience method for supporting multiple operating systems. @@ -56,7 +59,7 @@ def command_or_control_key(cls: Type[SelfHoldDown]) -> SelfHoldDown: return cls(Keys.CONTROL) @classmethod - def left_mouse_button(cls: Type[SelfHoldDown]) -> SelfHoldDown: + def left_mouse_button(cls: type[SelfHoldDown]) -> SelfHoldDown: """Hold down the left mouse button.""" return cls(lmb=True) @@ -85,9 +88,7 @@ def add_to_chain( msg = "HoldDown must be told what to hold down." raise UnableToAct(msg) - def __init__( - self: SelfHoldDown, key: Optional[str] = None, lmb: bool = False - ) -> None: + def __init__(self: SelfHoldDown, key: str | None = None, lmb: bool = False) -> None: self.key = key self.lmb = lmb self.target = None diff --git a/screenpy_selenium/actions/move_mouse.py b/screenpy_selenium/actions/move_mouse.py index 311a908..78152ed 100644 --- a/screenpy_selenium/actions/move_mouse.py +++ b/screenpy_selenium/actions/move_mouse.py @@ -2,15 +2,18 @@ from __future__ import annotations -from typing import Optional, Tuple, Type, TypeVar +from typing import TYPE_CHECKING, TypeVar -from screenpy.actor import Actor from screenpy.exceptions import UnableToAct from screenpy.pacing import beat from selenium.webdriver.common.action_chains import ActionChains from ..abilities import BrowseTheWeb -from ..target import Target + +if TYPE_CHECKING: + from screenpy.actor import Actor + + from ..target import Target SelfMoveMouse = TypeVar("SelfMoveMouse", bound="MoveMouse") @@ -40,12 +43,12 @@ class MoveMouse: ) """ - offset: Optional[Tuple[int, int]] - target: Optional[Target] + offset: tuple[int, int] | None + target: Target | None description: str @classmethod - def to_the(cls: Type[SelfMoveMouse], target: Target) -> SelfMoveMouse: + def to_the(cls: type[SelfMoveMouse], target: Target) -> SelfMoveMouse: """ Target an element to move the mouse to. @@ -58,30 +61,30 @@ def to_the(cls: Type[SelfMoveMouse], target: Target) -> SelfMoveMouse: return cls(target=target, description=f"to the {target}") @classmethod - def on_the(cls: Type[SelfMoveMouse], target: Target) -> SelfMoveMouse: + def on_the(cls: type[SelfMoveMouse], target: Target) -> SelfMoveMouse: """Alias for :meth:`~screenpy_selenium.actions.MoveMouse.to_the`.""" return cls.to_the(target=target) @classmethod - def over_the(cls: Type[SelfMoveMouse], target: Target) -> SelfMoveMouse: + def over_the(cls: type[SelfMoveMouse], target: Target) -> SelfMoveMouse: """Alias for :meth:`~screenpy_selenium.actions.MoveMouse.to_the`.""" return cls.to_the(target=target) @classmethod def over_the_first_of_the( - cls: Type[SelfMoveMouse], target: Target + cls: type[SelfMoveMouse], target: Target ) -> SelfMoveMouse: """Alias for :meth:`~screenpy_selenium.actions.MoveMouse.to_the`.""" return cls.to_the(target=target) @classmethod - def to_the_first_of_the(cls: Type[SelfMoveMouse], target: Target) -> SelfMoveMouse: + def to_the_first_of_the(cls: type[SelfMoveMouse], target: Target) -> SelfMoveMouse: """Alias for :meth:`~screenpy_selenium.actions.MoveMouse.to_the`.""" return cls.to_the(target=target) @classmethod def by_offset( - cls: Type[SelfMoveMouse], x_offset: int, y_offset: int + cls: type[SelfMoveMouse], x_offset: int, y_offset: int ) -> SelfMoveMouse: """Specify the offset by which to move the mouse.""" return cls( @@ -135,8 +138,8 @@ def add_to_chain( def __init__( self: SelfMoveMouse, - offset: Optional[Tuple[int, int]] = None, - target: Optional[Target] = None, + offset: tuple[int, int] | None = None, + target: Target | None = None, description: str = "", ) -> None: self.offset = offset diff --git a/screenpy_selenium/actions/open.py b/screenpy_selenium/actions/open.py index 71302f4..5980d0b 100644 --- a/screenpy_selenium/actions/open.py +++ b/screenpy_selenium/actions/open.py @@ -3,13 +3,15 @@ from __future__ import annotations import os -from typing import Type, TypeVar, Union +from typing import TYPE_CHECKING, TypeVar -from screenpy import Actor from screenpy.pacing import beat from ..abilities import BrowseTheWeb +if TYPE_CHECKING: + from screenpy import Actor + SelfOpen = TypeVar("SelfOpen", bound="Open") @@ -40,7 +42,7 @@ class Open: """ @classmethod - def their_browser_on(cls: Type[SelfOpen], location: Union[str, object]) -> SelfOpen: + def their_browser_on(cls: type[SelfOpen], location: str | object) -> SelfOpen: """ Provide a URL to visit. @@ -50,7 +52,7 @@ def their_browser_on(cls: Type[SelfOpen], location: Union[str, object]) -> SelfO return cls(location=location) @classmethod - def browser_on(cls: Type[SelfOpen], location: Union[str, object]) -> SelfOpen: + def browser_on(cls: type[SelfOpen], location: str | object) -> SelfOpen: """Alias for :meth:`~screenpy_selenium.actions.Open.their_browser_on`.""" return cls.their_browser_on(location=location) @@ -64,7 +66,7 @@ def perform_as(self: SelfOpen, the_actor: Actor) -> None: browser = the_actor.ability_to(BrowseTheWeb).browser browser.get(self.url) - def __init__(self: SelfOpen, location: Union[str, object]) -> None: + def __init__(self: SelfOpen, location: str | object) -> None: url = getattr(location, "url", location) url = f'{os.getenv("BASE_URL", "")}{url}' self.url = url diff --git a/screenpy_selenium/actions/pause.py b/screenpy_selenium/actions/pause.py index 5c2310d..ac473ae 100644 --- a/screenpy_selenium/actions/pause.py +++ b/screenpy_selenium/actions/pause.py @@ -2,10 +2,14 @@ from __future__ import annotations -from screenpy import Actor +from typing import TYPE_CHECKING + from screenpy.actions import Pause as BasePause from screenpy.pacing import beat -from selenium.webdriver.common.action_chains import ActionChains + +if TYPE_CHECKING: + from screenpy import Actor + from selenium.webdriver.common.action_chains import ActionChains class Pause(BasePause): diff --git a/screenpy_selenium/actions/refresh_page.py b/screenpy_selenium/actions/refresh_page.py index 1864877..2d52f95 100644 --- a/screenpy_selenium/actions/refresh_page.py +++ b/screenpy_selenium/actions/refresh_page.py @@ -2,11 +2,15 @@ from __future__ import annotations -from screenpy import Actor +from typing import TYPE_CHECKING + from screenpy.pacing import beat from ..abilities import BrowseTheWeb +if TYPE_CHECKING: + from screenpy import Actor + class RefreshPage: """Refresh the browser page! diff --git a/screenpy_selenium/actions/release.py b/screenpy_selenium/actions/release.py index 49a9036..d7afc04 100644 --- a/screenpy_selenium/actions/release.py +++ b/screenpy_selenium/actions/release.py @@ -3,16 +3,18 @@ from __future__ import annotations import platform -from typing import Optional, Type, TypeVar +from typing import TYPE_CHECKING, TypeVar -from screenpy import Actor from screenpy.exceptions import UnableToAct from screenpy.pacing import beat -from selenium.webdriver.common.action_chains import ActionChains from selenium.webdriver.common.keys import Keys from ..speech_tools import KEY_NAMES +if TYPE_CHECKING: + from screenpy import Actor + from selenium.webdriver.common.action_chains import ActionChains + SelfRelease = TypeVar("SelfRelease", bound="Release") @@ -35,13 +37,13 @@ class Release: the_actor.attempts_to(Release.command_or_control_key()) """ - key: Optional[str] + key: str | None lmb: bool description: str the_kraken: str @classmethod - def command_or_control_key(cls: Type[SelfRelease]) -> SelfRelease: + def command_or_control_key(cls: type[SelfRelease]) -> SelfRelease: """ A convenience method for supporting multiple operating systems. @@ -53,7 +55,7 @@ def command_or_control_key(cls: Type[SelfRelease]) -> SelfRelease: return cls(key=Keys.CONTROL) @classmethod - def left_mouse_button(cls: Type[SelfRelease]) -> SelfRelease: + def left_mouse_button(cls: type[SelfRelease]) -> SelfRelease: """Release the left mouse button.""" return cls(lmb=True) @@ -73,9 +75,7 @@ def add_to_chain(self: SelfRelease, _: Actor, the_chain: ActionChains) -> None: msg = "Release must be told what to release." raise UnableToAct(msg) - def __init__( - self: SelfRelease, key: Optional[str] = None, lmb: bool = False - ) -> None: + def __init__(self: SelfRelease, key: str | None = None, lmb: bool = False) -> None: self.key = key self.lmb = lmb self.description = "LEFT MOUSE BUTTON" if lmb else KEY_NAMES[key] diff --git a/screenpy_selenium/actions/respond_to_the_prompt.py b/screenpy_selenium/actions/respond_to_the_prompt.py index 97ca97b..9d201de 100644 --- a/screenpy_selenium/actions/respond_to_the_prompt.py +++ b/screenpy_selenium/actions/respond_to_the_prompt.py @@ -2,13 +2,15 @@ from __future__ import annotations -from typing import Type, TypeVar +from typing import TYPE_CHECKING, TypeVar -from screenpy.actor import Actor from screenpy.pacing import aside, beat from ..abilities import BrowseTheWeb +if TYPE_CHECKING: + from screenpy.actor import Actor + SelfRespondToThePrompt = TypeVar("SelfRespondToThePrompt", bound="RespondToThePrompt") @@ -26,7 +28,7 @@ class RespondToThePrompt: """ @classmethod - def with_(cls: Type[SelfRespondToThePrompt], text: str) -> SelfRespondToThePrompt: + def with_(cls: type[SelfRespondToThePrompt], text: str) -> SelfRespondToThePrompt: """Provide the text to enter into the prompt.""" return cls(text) diff --git a/screenpy_selenium/actions/right_click.py b/screenpy_selenium/actions/right_click.py index bdf9baf..d915370 100644 --- a/screenpy_selenium/actions/right_click.py +++ b/screenpy_selenium/actions/right_click.py @@ -2,14 +2,17 @@ from __future__ import annotations -from typing import Optional, Type, TypeVar +from typing import TYPE_CHECKING, TypeVar -from screenpy import Actor from screenpy.pacing import beat from selenium.webdriver.common.action_chains import ActionChains from ..abilities import BrowseTheWeb -from ..target import Target + +if TYPE_CHECKING: + from screenpy import Actor + + from ..target import Target SelfRightClick = TypeVar("SelfRightClick", bound="RightClick") @@ -32,10 +35,10 @@ class RightClick: context menu made of web elements to be able to interact with it. """ - target: Optional[Target] + target: Target | None @classmethod - def on_the(cls: Type[SelfRightClick], target: Target) -> SelfRightClick: + def on_the(cls: type[SelfRightClick], target: Target) -> SelfRightClick: """Target an element to right-click on. Aliases: @@ -45,13 +48,13 @@ def on_the(cls: Type[SelfRightClick], target: Target) -> SelfRightClick: return cls(target=target) @classmethod - def on(cls: Type[SelfRightClick], target: Target) -> SelfRightClick: + def on(cls: type[SelfRightClick], target: Target) -> SelfRightClick: """Alias for :meth:`~screenpy_selenium.actions.RightClick.on_the`.""" return cls.on_the(target=target) @classmethod def on_the_first_of_the( - cls: Type[SelfRightClick], target: Target + cls: type[SelfRightClick], target: Target ) -> SelfRightClick: """Alias for :meth:`~screenpy_selenium.actions.RightClick.on_the`.""" return cls.on_the(target=target) @@ -86,6 +89,6 @@ def add_to_chain( """Add the RightClick Action to a Chain of Actions.""" self._add_action_to_chain(the_actor, the_chain) - def __init__(self: SelfRightClick, target: Optional[Target] = None) -> None: + def __init__(self: SelfRightClick, target: Target | None = None) -> None: self.target = target self.description = f" on the {target}" if target is not None else "" diff --git a/screenpy_selenium/actions/save_console_log.py b/screenpy_selenium/actions/save_console_log.py index dcf6eba..fad9612 100644 --- a/screenpy_selenium/actions/save_console_log.py +++ b/screenpy_selenium/actions/save_console_log.py @@ -3,14 +3,16 @@ from __future__ import annotations import os -from typing import Any, Optional, Type, TypeVar +from typing import TYPE_CHECKING, Any, TypeVar -from screenpy import Actor from screenpy.actions import AttachTheFile from screenpy.pacing import beat from ..abilities import BrowseTheWeb +if TYPE_CHECKING: + from screenpy import Actor + SelfSaveConsoleLog = TypeVar("SelfSaveConsoleLog", bound="SaveConsoleLog") @@ -47,7 +49,7 @@ class SaveConsoleLog: ) """ - attach_kwargs: Optional[dict] + attach_kwargs: dict | None path: str filename: str @@ -56,7 +58,7 @@ def describe(self: SelfSaveConsoleLog) -> str: return f"Save browser console log as {self.filename}" @classmethod - def as_(cls: Type[SelfSaveConsoleLog], path: str) -> SelfSaveConsoleLog: + def as_(cls: type[SelfSaveConsoleLog], path: str) -> SelfSaveConsoleLog: """Supply the name and/or filepath for the saved text file. If only a name is supplied, the text file will appear in the current diff --git a/screenpy_selenium/actions/save_screenshot.py b/screenpy_selenium/actions/save_screenshot.py index 12905e9..d582a4e 100644 --- a/screenpy_selenium/actions/save_screenshot.py +++ b/screenpy_selenium/actions/save_screenshot.py @@ -3,14 +3,16 @@ from __future__ import annotations import os -from typing import Any, Optional, Type, TypeVar +from typing import TYPE_CHECKING, Any, TypeVar -from screenpy import Actor from screenpy.actions import AttachTheFile from screenpy.pacing import beat from ..abilities import BrowseTheWeb +if TYPE_CHECKING: + from screenpy import Actor + SelfSaveScreenshot = TypeVar("SelfSaveScreenshot", bound="SaveScreenshot") @@ -43,7 +45,7 @@ class SaveScreenshot: ) """ - attach_kwargs: Optional[dict] + attach_kwargs: dict | None path: str filename: str @@ -52,7 +54,7 @@ def describe(self: SelfSaveScreenshot) -> str: return f"Save screenshot as {self.filename}" @classmethod - def as_(cls: Type[SelfSaveScreenshot], path: str) -> SelfSaveScreenshot: + def as_(cls: type[SelfSaveScreenshot], path: str) -> SelfSaveScreenshot: """Supply the name and/or filepath for the screenshot. If only a name is supplied, the screenshot will appear in the current diff --git a/screenpy_selenium/actions/select.py b/screenpy_selenium/actions/select.py index 16523a6..2c70a0b 100644 --- a/screenpy_selenium/actions/select.py +++ b/screenpy_selenium/actions/select.py @@ -2,15 +2,17 @@ from __future__ import annotations -from typing import Optional, TypeVar, Union +from typing import TYPE_CHECKING, TypeVar -from screenpy import Actor from screenpy.exceptions import DeliveryError, UnableToAct from screenpy.pacing import beat from selenium.common.exceptions import WebDriverException from selenium.webdriver.support.ui import Select as SeleniumSelect -from ..target import Target +if TYPE_CHECKING: + from screenpy import Actor + + from ..target import Target SelfSelectByText = TypeVar("SelfSelectByText", bound="SelectByText") SelfSelectByIndex = TypeVar("SelfSelectByIndex", bound="SelectByIndex") @@ -42,17 +44,17 @@ class Select: """ @staticmethod - def the_option_named(text: str) -> "SelectByText": + def the_option_named(text: str) -> SelectByText: """Select the option by its text.""" return SelectByText(text) @staticmethod - def the_option_at_index(index: Union[int, str]) -> "SelectByIndex": + def the_option_at_index(index: int | str) -> SelectByIndex: """Select the option by its index. This index is 0-based.""" return SelectByIndex(index) @staticmethod - def the_option_with_value(value: Union[int, str]) -> "SelectByValue": + def the_option_with_value(value: int | str) -> SelectByValue: """Select the option by its value.""" return SelectByValue(value) @@ -67,7 +69,7 @@ class SelectByText: :class:`~screenpy_selenium.abilities.BrowseTheWeb` """ - target: Optional[Target] + target: Target | None text: str def from_the(self: SelfSelectByText, target: Target) -> SelfSelectByText: @@ -103,7 +105,7 @@ def perform_as(self: SelfSelectByText, the_actor: Actor) -> None: raise DeliveryError(msg) from e def __init__( - self: SelfSelectByText, text: str, target: Optional[Target] = None + self: SelfSelectByText, text: str, target: Target | None = None ) -> None: self.target = target self.text = text @@ -119,7 +121,7 @@ class SelectByIndex: :class:`~screenpy_selenium.abilities.BrowseTheWeb` """ - target: Optional[Target] + target: Target | None index: int def from_the(self: SelfSelectByIndex, target: Target) -> SelfSelectByIndex: @@ -155,7 +157,7 @@ def perform_as(self: SelfSelectByIndex, the_actor: Actor) -> None: raise DeliveryError(msg) from e def __init__( - self: SelfSelectByIndex, index: Union[int, str], target: Optional[Target] = None + self: SelfSelectByIndex, index: int | str, target: Target | None = None ) -> None: self.target = target self.index = int(index) @@ -171,7 +173,7 @@ class SelectByValue: :class:`~screenpy_selenium.abilities.BrowseTheWeb` """ - target: Optional[Target] + target: Target | None value: str def from_the(self: SelfSelectByValue, target: Target) -> SelfSelectByValue: @@ -207,7 +209,7 @@ def perform_as(self: SelfSelectByValue, the_actor: Actor) -> None: raise DeliveryError(msg) from e def __init__( - self: SelfSelectByValue, value: Union[int, str], target: Optional[Target] = None + self: SelfSelectByValue, value: int | str, target: Target | None = None ) -> None: self.target = target self.value = str(value) diff --git a/screenpy_selenium/actions/switch_to.py b/screenpy_selenium/actions/switch_to.py index 327c1ff..eecde46 100644 --- a/screenpy_selenium/actions/switch_to.py +++ b/screenpy_selenium/actions/switch_to.py @@ -2,13 +2,16 @@ from __future__ import annotations -from typing import Optional, Type, TypeVar +from typing import TYPE_CHECKING, TypeVar -from screenpy.actor import Actor from screenpy.pacing import beat from ..abilities import BrowseTheWeb -from ..target import Target + +if TYPE_CHECKING: + from screenpy.actor import Actor + + from ..target import Target SelfSwitchTo = TypeVar("SelfSwitchTo", bound="SwitchTo") @@ -29,12 +32,12 @@ class SwitchTo: """ @classmethod - def the(cls: Type[SelfSwitchTo], target: Target) -> SelfSwitchTo: + def the(cls: type[SelfSwitchTo], target: Target) -> SelfSwitchTo: """Target an element, probably an iframe, to switch to.""" return cls(target=target, frame_to_log=str(target)) @classmethod - def default(cls: Type[SelfSwitchTo]) -> SelfSwitchTo: + def default(cls: type[SelfSwitchTo]) -> SelfSwitchTo: """Switch back to the default frame, the browser window.""" return cls(target=None, frame_to_log="default frame") @@ -51,8 +54,6 @@ def perform_as(self: SelfSwitchTo, the_actor: Actor) -> None: else: browser.switch_to.frame(self.target.found_by(the_actor)) - def __init__( - self: SelfSwitchTo, target: Optional[Target], frame_to_log: str - ) -> None: + def __init__(self: SelfSwitchTo, target: Target | None, frame_to_log: str) -> None: self.target = target self.frame_to_log = frame_to_log diff --git a/screenpy_selenium/actions/switch_to_tab.py b/screenpy_selenium/actions/switch_to_tab.py index 3f3f45c..b0da2e9 100644 --- a/screenpy_selenium/actions/switch_to_tab.py +++ b/screenpy_selenium/actions/switch_to_tab.py @@ -2,11 +2,15 @@ from __future__ import annotations -from screenpy import Actor +from typing import TYPE_CHECKING + from screenpy.pacing import beat from ..abilities import BrowseTheWeb +if TYPE_CHECKING: + from screenpy import Actor + class SwitchToTab: """Switch to a specified tab or window. diff --git a/screenpy_selenium/actions/wait.py b/screenpy_selenium/actions/wait.py index f1feaa7..94ca3a6 100644 --- a/screenpy_selenium/actions/wait.py +++ b/screenpy_selenium/actions/wait.py @@ -2,7 +2,7 @@ from __future__ import annotations -from typing import Any, Callable, Iterable, Optional, Type, TypeVar +from typing import TYPE_CHECKING, Any, Callable, Iterable, TypeVar from screenpy import Actor, settings from screenpy.exceptions import DeliveryError @@ -12,7 +12,9 @@ from selenium.webdriver.support.ui import WebDriverWait from ..abilities import BrowseTheWeb -from ..target import Target + +if TYPE_CHECKING: + from ..target import Target SelfWait = TypeVar("SelfWait", bound="Wait") @@ -50,10 +52,10 @@ class Wait: args: Iterable[Any] timeout: float - log_detail: Optional[str] + log_detail: str | None @classmethod - def for_the(cls: Type[SelfWait], target: Target) -> SelfWait: + def for_the(cls: type[SelfWait], target: Target) -> SelfWait: """Set the Target to wait for. Aliases: @@ -62,7 +64,7 @@ def for_the(cls: Type[SelfWait], target: Target) -> SelfWait: return cls(seconds=settings.TIMEOUT, args=[target]) @classmethod - def for_(cls: Type[SelfWait], target: Target) -> SelfWait: + def for_(cls: type[SelfWait], target: Target) -> SelfWait: """Alias for :meth:`~screenpy_selenium.actions.Wait.for_the`.""" return cls.for_the(target=target) @@ -74,7 +76,7 @@ def seconds_for_the(self: SelfWait, target: Target) -> SelfWait: second_for = second_for_the = seconds_for = seconds_for_the def using( - self: SelfWait, strategy: Callable[..., Any], log_detail: Optional[str] = None + self: SelfWait, strategy: Callable[..., Any], log_detail: str | None = None ) -> SelfWait: """Use the given strategy to wait for the Target. @@ -147,8 +149,8 @@ def perform_as(self: SelfWait, the_actor: Actor) -> None: def __init__( self: SelfWait, - seconds: Optional[float] = None, - args: Optional[Iterable[Any]] = None, + seconds: float | None = None, + args: Iterable[Any] | None = None, ) -> None: self.args = args if args is not None else [] self.timeout = seconds if seconds is not None else settings.TIMEOUT diff --git a/screenpy_selenium/protocols.py b/screenpy_selenium/protocols.py index 0119c04..47ad3f1 100644 --- a/screenpy_selenium/protocols.py +++ b/screenpy_selenium/protocols.py @@ -2,10 +2,11 @@ from __future__ import annotations -from typing import Protocol, runtime_checkable +from typing import TYPE_CHECKING, Protocol, runtime_checkable -from screenpy import Actor -from selenium.webdriver.common.action_chains import ActionChains +if TYPE_CHECKING: + from screenpy import Actor + from selenium.webdriver.common.action_chains import ActionChains @runtime_checkable diff --git a/screenpy_selenium/questions/attribute.py b/screenpy_selenium/questions/attribute.py index 36cf4d7..1fb414b 100644 --- a/screenpy_selenium/questions/attribute.py +++ b/screenpy_selenium/questions/attribute.py @@ -2,13 +2,15 @@ from __future__ import annotations -from typing import List, Optional, Union +from typing import TYPE_CHECKING -from screenpy import Actor from screenpy.exceptions import UnableToAnswer from screenpy.pacing import beat -from ..target import Target +if TYPE_CHECKING: + from screenpy import Actor + + from ..target import Target class Attribute: @@ -29,16 +31,16 @@ class Attribute: ) """ - target: Optional[Target] + target: Target | None - def of_the(self, target: Target) -> "Attribute": + def of_the(self, target: Target) -> Attribute: """Target the element to get the attribute from.""" self.target = target return self of = of_the_first_of_the = of_the - def of_all(self, target: Target) -> "Attribute": + def of_all(self, target: Target) -> Attribute: """Target the elements, plural, to get the attribute from.""" self.target = target self.multi = True @@ -49,7 +51,7 @@ def describe(self) -> str: return f'The "{self.attribute}" attribute of the {self.target}.' @beat('{} examines the "{attribute}" attribute of the {target}...') - def answered_by(self, the_actor: Actor) -> Union[str, List[Union[str, None]], None]: + def answered_by(self, the_actor: Actor) -> str | list[str | None] | None: """Direct the actor to investigate the attribute on the element.""" if self.target is None: msg = ( diff --git a/screenpy_selenium/questions/browser_title.py b/screenpy_selenium/questions/browser_title.py index 2b93122..c7523f3 100644 --- a/screenpy_selenium/questions/browser_title.py +++ b/screenpy_selenium/questions/browser_title.py @@ -2,11 +2,15 @@ from __future__ import annotations -from screenpy import Actor +from typing import TYPE_CHECKING + from screenpy.pacing import beat from ..abilities import BrowseTheWeb +if TYPE_CHECKING: + from screenpy import Actor + class BrowserTitle: """Ask what the title of the browser's active window is. diff --git a/screenpy_selenium/questions/browser_url.py b/screenpy_selenium/questions/browser_url.py index c99d637..db469a7 100644 --- a/screenpy_selenium/questions/browser_url.py +++ b/screenpy_selenium/questions/browser_url.py @@ -2,11 +2,15 @@ from __future__ import annotations -from screenpy import Actor +from typing import TYPE_CHECKING + from screenpy.pacing import beat from ..abilities import BrowseTheWeb +if TYPE_CHECKING: + from screenpy import Actor + class BrowserURL: """Ask what the url of the browser's active window is. diff --git a/screenpy_selenium/questions/cookies.py b/screenpy_selenium/questions/cookies.py index 53687e0..6fdc875 100644 --- a/screenpy_selenium/questions/cookies.py +++ b/screenpy_selenium/questions/cookies.py @@ -2,11 +2,15 @@ from __future__ import annotations -from screenpy import Actor +from typing import TYPE_CHECKING + from screenpy.pacing import beat from ..abilities import BrowseTheWeb +if TYPE_CHECKING: + from screenpy import Actor + class Cookies: """Ask about the cookies on the Actor's web browsing session. diff --git a/screenpy_selenium/questions/element.py b/screenpy_selenium/questions/element.py index 6219151..4b8ee10 100644 --- a/screenpy_selenium/questions/element.py +++ b/screenpy_selenium/questions/element.py @@ -2,14 +2,17 @@ from __future__ import annotations -from typing import Optional +from typing import TYPE_CHECKING -from screenpy import Actor from screenpy.pacing import beat -from selenium.webdriver.remote.webelement import WebElement from ..exceptions import TargetingError -from ..target import Target + +if TYPE_CHECKING: + from screenpy import Actor + from selenium.webdriver.remote.webelement import WebElement + + from ..target import Target class Element: @@ -23,14 +26,14 @@ class Element: the_actor.should(See.the(Element(WELCOME_BANNER), IsVisible())) """ - caught_exception: Optional[TargetingError] + caught_exception: TargetingError | None def describe(self) -> str: """Describe the Question.""" return f"The {self.target}." @beat("{} inspects the {target}.") - def answered_by(self, the_actor: Actor) -> Optional[WebElement]: + def answered_by(self, the_actor: Actor) -> WebElement | None: """Direct the Actor to find the element.""" try: return self.target.found_by(the_actor) diff --git a/screenpy_selenium/questions/list.py b/screenpy_selenium/questions/list.py index c100607..d6af340 100644 --- a/screenpy_selenium/questions/list.py +++ b/screenpy_selenium/questions/list.py @@ -2,13 +2,15 @@ from __future__ import annotations -from typing import List as ListType, Type, TypeVar +from typing import TYPE_CHECKING, TypeVar -from screenpy import Actor from screenpy.pacing import beat -from selenium.webdriver.remote.webdriver import WebElement -from ..target import Target +if TYPE_CHECKING: + from screenpy import Actor + from selenium.webdriver.remote.webdriver import WebElement + + from ..target import Target SelfList = TypeVar("SelfList", bound="List") @@ -25,7 +27,7 @@ class List: """ @classmethod - def of_the(cls: Type[SelfList], target: Target) -> SelfList: + def of_the(cls: type[SelfList], target: Target) -> SelfList: """Target the element(s) to list. Aliases: @@ -36,17 +38,17 @@ def of_the(cls: Type[SelfList], target: Target) -> SelfList: return cls(target=target) @classmethod - def of_all_the(cls: Type[SelfList], target: Target) -> SelfList: + def of_all_the(cls: type[SelfList], target: Target) -> SelfList: """Alias for :meth:`~screenpy_selenium.actions.List.of_the`.""" return cls.of_the(target=target) @classmethod - def of_all(cls: Type[SelfList], target: Target) -> SelfList: + def of_all(cls: type[SelfList], target: Target) -> SelfList: """Alias for :meth:`~screenpy_selenium.actions.List.of_the`.""" return cls.of_the(target=target) @classmethod - def of(cls: Type[SelfList], target: Target) -> SelfList: + def of(cls: type[SelfList], target: Target) -> SelfList: """Alias for :meth:`~screenpy_selenium.actions.List.of_the`.""" return cls.of_the(target=target) @@ -55,7 +57,7 @@ def describe(self: SelfList) -> str: return f"The list of {self.target}." @beat("{} lists off the {target}.") - def answered_by(self: SelfList, the_actor: Actor) -> ListType[WebElement]: + def answered_by(self: SelfList, the_actor: Actor) -> list[WebElement]: """Direct the Actor to rattle off the specified elements.""" return self.target.all_found_by(the_actor) diff --git a/screenpy_selenium/questions/number.py b/screenpy_selenium/questions/number.py index eb581cb..20c3976 100644 --- a/screenpy_selenium/questions/number.py +++ b/screenpy_selenium/questions/number.py @@ -2,12 +2,14 @@ from __future__ import annotations -from typing import Type, TypeVar +from typing import TYPE_CHECKING, TypeVar -from screenpy import Actor from screenpy.pacing import beat -from ..target import Target +if TYPE_CHECKING: + from screenpy import Actor + + from ..target import Target SelfNumber = TypeVar("SelfNumber", bound="Number") @@ -24,7 +26,7 @@ class Number: """ @classmethod - def of(cls: Type[SelfNumber], target: Target) -> SelfNumber: + def of(cls: type[SelfNumber], target: Target) -> SelfNumber: """Target the element to be counted.""" return cls(target=target) diff --git a/screenpy_selenium/questions/selected.py b/screenpy_selenium/questions/selected.py index 110dd48..026f36f 100644 --- a/screenpy_selenium/questions/selected.py +++ b/screenpy_selenium/questions/selected.py @@ -2,13 +2,15 @@ from __future__ import annotations -from typing import List, Type, TypeVar, Union +from typing import TYPE_CHECKING, TypeVar -from screenpy import Actor from screenpy.pacing import beat from selenium.webdriver.support.ui import Select as SeleniumSelect -from ..target import Target +if TYPE_CHECKING: + from screenpy import Actor + + from ..target import Target SelfSelected = TypeVar("SelfSelected", bound="Selected") @@ -32,7 +34,7 @@ class Selected: multi: bool @classmethod - def option_from_the(cls: Type[SelfSelected], target: Target) -> SelfSelected: + def option_from_the(cls: type[SelfSelected], target: Target) -> SelfSelected: """ Get the option. @@ -48,13 +50,13 @@ def option_from_the(cls: Type[SelfSelected], target: Target) -> SelfSelected: return cls(target=target) @classmethod - def option_from(cls: Type[SelfSelected], target: Target) -> SelfSelected: + def option_from(cls: type[SelfSelected], target: Target) -> SelfSelected: """Alias of :meth:`~screenpy_selenium.actions.Selected.option_from_the`.""" return cls.option_from_the(target=target) @classmethod def options_from_the( - cls: Type[SelfSelected], multiselect_target: Target + cls: type[SelfSelected], multiselect_target: Target ) -> SelfSelected: """ Get all the options that are currently selected in a multi-select field. @@ -70,7 +72,7 @@ def options_from_the( @classmethod def options_from( - cls: Type[SelfSelected], multiselect_target: Target + cls: type[SelfSelected], multiselect_target: Target ) -> SelfSelected: """Alias of :meth:`~screenpy_selenium.actions.Selected.options_from_the`.""" return cls.options_from_the(multiselect_target=multiselect_target) @@ -80,7 +82,7 @@ def describe(self: SelfSelected) -> str: return f"The selected option(s) from the {self.target}." @beat("{} checks the selected option(s) from the {target}.") - def answered_by(self: SelfSelected, the_actor: Actor) -> Union[str, List[str]]: + def answered_by(self: SelfSelected, the_actor: Actor) -> str | list[str]: """Direct the Actor to name the selected option(s).""" select = SeleniumSelect(self.target.found_by(the_actor)) diff --git a/screenpy_selenium/questions/text.py b/screenpy_selenium/questions/text.py index aff245c..5e9eab5 100644 --- a/screenpy_selenium/questions/text.py +++ b/screenpy_selenium/questions/text.py @@ -2,12 +2,14 @@ from __future__ import annotations -from typing import List, Type, TypeVar, Union +from typing import TYPE_CHECKING, TypeVar -from screenpy import Actor from screenpy.pacing import beat -from ..target import Target +if TYPE_CHECKING: + from screenpy import Actor + + from ..target import Target SelfText = TypeVar("SelfText", bound="Text") @@ -33,7 +35,7 @@ class Text: multi: bool @classmethod - def of_the(cls: Type[SelfText], target: Target) -> SelfText: + def of_the(cls: type[SelfText], target: Target) -> SelfText: """Target the element to extract the text from. Aliases: @@ -43,17 +45,17 @@ def of_the(cls: Type[SelfText], target: Target) -> SelfText: return cls(target=target) @classmethod - def of(cls: Type[SelfText], target: Target) -> SelfText: + def of(cls: type[SelfText], target: Target) -> SelfText: """Alias of :meth:`~screenpy_selenium.actions.Text.of_the`.""" return cls.of_the(target=target) @classmethod - def of_the_first_of_the(cls: Type[SelfText], target: Target) -> SelfText: + def of_the_first_of_the(cls: type[SelfText], target: Target) -> SelfText: """Alias of :meth:`~screenpy_selenium.actions.Text.of_the`.""" return cls.of_the(target=target) @classmethod - def of_all(cls: Type[SelfText], multi_target: Target) -> SelfText: + def of_all(cls: type[SelfText], multi_target: Target) -> SelfText: """Target the elements, plural, to extract the text from.""" return cls(target=multi_target, multi=True) @@ -62,7 +64,7 @@ def describe(self: SelfText) -> str: return f"The text from the {self.target}." @beat("{} reads the text from the {target}.") - def answered_by(self: SelfText, the_actor: Actor) -> Union[str, List[str]]: + def answered_by(self: SelfText, the_actor: Actor) -> str | list[str]: """Direct the Actor to read off the text of the element(s).""" if self.multi: return [e.text for e in self.target.all_found_by(the_actor)] diff --git a/screenpy_selenium/questions/text_of_the_alert.py b/screenpy_selenium/questions/text_of_the_alert.py index 333fc26..815e1b1 100644 --- a/screenpy_selenium/questions/text_of_the_alert.py +++ b/screenpy_selenium/questions/text_of_the_alert.py @@ -2,11 +2,15 @@ from __future__ import annotations -from screenpy.actor import Actor +from typing import TYPE_CHECKING + from screenpy.pacing import beat from ..abilities import BrowseTheWeb +if TYPE_CHECKING: + from screenpy.actor import Actor + class TextOfTheAlert: """Ask what text appears in the alert. diff --git a/screenpy_selenium/resolutions/custom_matchers/is_clickable_element.py b/screenpy_selenium/resolutions/custom_matchers/is_clickable_element.py index 0aa5cbf..7d57a16 100644 --- a/screenpy_selenium/resolutions/custom_matchers/is_clickable_element.py +++ b/screenpy_selenium/resolutions/custom_matchers/is_clickable_element.py @@ -8,17 +8,19 @@ from __future__ import annotations -from typing import Optional +from typing import TYPE_CHECKING, Optional from hamcrest.core.base_matcher import BaseMatcher -from hamcrest.core.description import Description from selenium.webdriver.remote.webelement import WebElement +if TYPE_CHECKING: + from hamcrest.core.description import Description + class IsClickableElement(BaseMatcher[Optional[WebElement]]): """Matches an element which both ``is_enabled`` and ``is_displayed``.""" - def _matches(self, item: Optional[WebElement]) -> bool: + def _matches(self, item: WebElement | None) -> bool: if item is None: return False return item.is_displayed() and item.is_enabled() @@ -28,13 +30,13 @@ def describe_to(self, description: Description) -> None: description.append_text("the element is enabled/clickable") def describe_match( - self, _: Optional[WebElement], match_description: Description + self, _: WebElement | None, match_description: Description ) -> None: """Describe the matching case.""" match_description.append_text("it was enabled/clickable") def describe_mismatch( - self, item: Optional[WebElement], mismatch_description: Description + self, item: WebElement | None, mismatch_description: Description ) -> None: """Describe the failing case.""" if item is None or not item.is_displayed(): diff --git a/screenpy_selenium/resolutions/custom_matchers/is_invisible_element.py b/screenpy_selenium/resolutions/custom_matchers/is_invisible_element.py index 3c045c6..f9871e9 100644 --- a/screenpy_selenium/resolutions/custom_matchers/is_invisible_element.py +++ b/screenpy_selenium/resolutions/custom_matchers/is_invisible_element.py @@ -8,17 +8,19 @@ from __future__ import annotations -from typing import Optional +from typing import TYPE_CHECKING, Optional from hamcrest.core.base_matcher import BaseMatcher -from hamcrest.core.description import Description from selenium.webdriver.remote.webelement import WebElement +if TYPE_CHECKING: + from hamcrest.core.description import Description + class IsInvisibleElement(BaseMatcher[Optional[WebElement]]): """Matches an element whose ``is_displayed`` method returns False.""" - def _matches(self, item: Optional[WebElement]) -> bool: + def _matches(self, item: WebElement | None) -> bool: if item is None: return True return item.is_displayed() is False @@ -28,13 +30,13 @@ def describe_to(self, description: Description) -> None: description.append_text("the element is invisible") def describe_match( - self, _: Optional[WebElement], match_description: Description + self, _: WebElement | None, match_description: Description ) -> None: """Describe the matching case.""" match_description.append_text("it was invisible") def describe_mismatch( - self, _: Optional[WebElement], mismatch_description: Description + self, _: WebElement | None, mismatch_description: Description ) -> None: """Describe the failing case.""" mismatch_description.append_text("was not invisible") diff --git a/screenpy_selenium/resolutions/custom_matchers/is_present_element.py b/screenpy_selenium/resolutions/custom_matchers/is_present_element.py index b8e454a..3c977ba 100644 --- a/screenpy_selenium/resolutions/custom_matchers/is_present_element.py +++ b/screenpy_selenium/resolutions/custom_matchers/is_present_element.py @@ -8,17 +8,19 @@ from __future__ import annotations -from typing import Optional +from typing import TYPE_CHECKING, Optional from hamcrest.core.base_matcher import BaseMatcher -from hamcrest.core.description import Description from selenium.webdriver.remote.webelement import WebElement +if TYPE_CHECKING: + from hamcrest.core.description import Description + class IsPresentElement(BaseMatcher[Optional[WebElement]]): """Matches an element to be a present WebElement.""" - def _matches(self, item: Optional[WebElement]) -> bool: + def _matches(self, item: WebElement | None) -> bool: if item is None: return False return isinstance(item, WebElement) @@ -28,13 +30,13 @@ def describe_to(self, description: Description) -> None: description.append_text("the element is present") def describe_match( - self, _: Optional[WebElement], match_description: Description + self, _: WebElement | None, match_description: Description ) -> None: """Describe the matching case.""" match_description.append_text("it was present") def describe_mismatch( - self, _: Optional[WebElement], mismatch_description: Description + self, _: WebElement | None, mismatch_description: Description ) -> None: """Describe the failing case.""" mismatch_description.append_text("was not present") diff --git a/screenpy_selenium/resolutions/custom_matchers/is_visible_element.py b/screenpy_selenium/resolutions/custom_matchers/is_visible_element.py index 3ab5a2b..b4188ba 100644 --- a/screenpy_selenium/resolutions/custom_matchers/is_visible_element.py +++ b/screenpy_selenium/resolutions/custom_matchers/is_visible_element.py @@ -8,17 +8,19 @@ from __future__ import annotations -from typing import Optional +from typing import TYPE_CHECKING, Optional from hamcrest.core.base_matcher import BaseMatcher -from hamcrest.core.description import Description from selenium.webdriver.remote.webelement import WebElement +if TYPE_CHECKING: + from hamcrest.core.description import Description + class IsVisibleElement(BaseMatcher[Optional[WebElement]]): """Matches an element whose ``is_displayed`` method returns True.""" - def _matches(self, item: Optional[WebElement]) -> bool: + def _matches(self, item: WebElement | None) -> bool: if item is None: return False return item.is_displayed() @@ -28,13 +30,13 @@ def describe_to(self, description: Description) -> None: description.append_text("the element is visible") def describe_match( - self, _: Optional[WebElement], match_description: Description + self, _: WebElement | None, match_description: Description ) -> None: """Describe the matching case.""" match_description.append_text("it was visible") def describe_mismatch( - self, item: Optional[WebElement], mismatch_description: Description + self, item: WebElement | None, mismatch_description: Description ) -> None: """Describe the failing case.""" if item is None: diff --git a/screenpy_selenium/resolutions/is_clickable.py b/screenpy_selenium/resolutions/is_clickable.py index 13bd725..ad5746a 100644 --- a/screenpy_selenium/resolutions/is_clickable.py +++ b/screenpy_selenium/resolutions/is_clickable.py @@ -2,10 +2,14 @@ from __future__ import annotations +from typing import TYPE_CHECKING + from screenpy.resolutions.base_resolution import BaseResolution from .custom_matchers import is_clickable_element -from .custom_matchers.is_clickable_element import IsClickableElement + +if TYPE_CHECKING: + from .custom_matchers.is_clickable_element import IsClickableElement class IsClickable(BaseResolution): diff --git a/screenpy_selenium/resolutions/is_invisible.py b/screenpy_selenium/resolutions/is_invisible.py index 07e1df7..cd1e1a5 100644 --- a/screenpy_selenium/resolutions/is_invisible.py +++ b/screenpy_selenium/resolutions/is_invisible.py @@ -2,10 +2,14 @@ from __future__ import annotations +from typing import TYPE_CHECKING + from screenpy.resolutions.base_resolution import BaseResolution from .custom_matchers import is_invisible_element -from .custom_matchers.is_invisible_element import IsInvisibleElement + +if TYPE_CHECKING: + from .custom_matchers.is_invisible_element import IsInvisibleElement class IsInvisible(BaseResolution): diff --git a/screenpy_selenium/resolutions/is_present.py b/screenpy_selenium/resolutions/is_present.py index 9e4f230..4055794 100644 --- a/screenpy_selenium/resolutions/is_present.py +++ b/screenpy_selenium/resolutions/is_present.py @@ -2,10 +2,14 @@ from __future__ import annotations +from typing import TYPE_CHECKING + from screenpy.resolutions.base_resolution import BaseResolution from .custom_matchers import is_present_element -from .custom_matchers.is_present_element import IsPresentElement + +if TYPE_CHECKING: + from .custom_matchers.is_present_element import IsPresentElement class IsPresent(BaseResolution): diff --git a/screenpy_selenium/resolutions/is_visible.py b/screenpy_selenium/resolutions/is_visible.py index 195e5ed..274d0bd 100644 --- a/screenpy_selenium/resolutions/is_visible.py +++ b/screenpy_selenium/resolutions/is_visible.py @@ -2,10 +2,14 @@ from __future__ import annotations +from typing import TYPE_CHECKING + from screenpy.resolutions.base_resolution import BaseResolution from .custom_matchers import is_visible_element -from .custom_matchers.is_visible_element import IsVisibleElement + +if TYPE_CHECKING: + from .custom_matchers.is_visible_element import IsVisibleElement class IsVisible(BaseResolution): diff --git a/screenpy_selenium/target.py b/screenpy_selenium/target.py index 5310e23..aefe684 100644 --- a/screenpy_selenium/target.py +++ b/screenpy_selenium/target.py @@ -8,16 +8,18 @@ from __future__ import annotations -from typing import Iterator, List, Optional, Tuple, Type, TypeVar, Union +from typing import TYPE_CHECKING, Iterator, TypeVar -from screenpy.actor import Actor from selenium.common.exceptions import WebDriverException from selenium.webdriver.common.by import By -from selenium.webdriver.remote.webdriver import WebElement from .abilities.browse_the_web import BrowseTheWeb from .exceptions import TargetingError +if TYPE_CHECKING: + from screenpy.actor import Actor + from selenium.webdriver.remote.webdriver import WebElement + SelfTarget = TypeVar("SelfTarget", bound="Target") @@ -36,11 +38,11 @@ class Target: Target().located_by((By.ID, "username-field")) """ - _description: Optional[str] = None - locator: Optional[Tuple[str, str]] = None + _description: str | None = None + locator: tuple[str, str] | None = None @property - def target_name(self: SelfTarget) -> Optional[str]: + def target_name(self: SelfTarget) -> str | None: """Return the description when set or the 2nd half of the locator.""" if self._description is not None: return self._description @@ -55,16 +57,14 @@ def target_name(self: SelfTarget) -> None: del self._description @classmethod - def the(cls: Type[SelfTarget], desc: str) -> SelfTarget: + def the(cls: type[SelfTarget], desc: str) -> SelfTarget: """Name this Target. Beginning with a lower-case letter makes the logs look the nicest. """ return cls(desc=desc) - def located_by( - self: SelfTarget, locator: Union[Tuple[str, str], str] - ) -> SelfTarget: + def located_by(self: SelfTarget, locator: tuple[str, str] | str) -> SelfTarget: """Set the locator for this Target. Possible values for locator: @@ -91,11 +91,11 @@ def located_by( return self - def located(self: SelfTarget, locator: Union[Tuple[str, str], str]) -> SelfTarget: + def located(self: SelfTarget, locator: tuple[str, str] | str) -> SelfTarget: """Alias for :meth:~screenpy_selenium.Target.located_by.""" return self.located_by(locator) - def get_locator(self: SelfTarget) -> Tuple[str, str]: + def get_locator(self: SelfTarget) -> tuple[str, str]: """Return the stored locator. Raises: @@ -118,7 +118,7 @@ def found_by(self: SelfTarget, the_actor: Actor) -> WebElement: msg = f"{e} raised while trying to find {self}." raise TargetingError(msg) from e - def all_found_by(self: SelfTarget, the_actor: Actor) -> List[WebElement]: + def all_found_by(self: SelfTarget, the_actor: Actor) -> list[WebElement]: """Retrieve a list of |WebElement| objects as viewed by the Actor.""" browser = the_actor.ability_to(BrowseTheWeb).browser try: @@ -143,8 +143,8 @@ def __getitem__(self: SelfTarget, index: int) -> str: def __init__( self: SelfTarget, - desc: Optional[str] = None, - locator: Optional[Tuple[str, str]] = None, + desc: str | None = None, + locator: tuple[str, str] | None = None, ) -> None: self.target_name = desc self.locator = locator diff --git a/setup.py b/setup.py index 7c6c6f8..efcc520 100644 --- a/setup.py +++ b/setup.py @@ -11,10 +11,10 @@ repo_dir = path.abspath(path.dirname(__file__)) about = {} -with open(path.join(repo_dir, "screenpy_selenium", "__version__.py"), "r") as f: +with open(path.join(repo_dir, "screenpy_selenium", "__version__.py")) as f: exec(f.read(), about) -with open("README.md", "r") as f: +with open("README.md") as f: readme = f.read() setup( diff --git a/tests/test_questions.py b/tests/test_questions.py index d7474f0..6517afd 100644 --- a/tests/test_questions.py +++ b/tests/test_questions.py @@ -170,7 +170,7 @@ def test_caught_exception_annotation(self) -> None: e = Element(TARGET) annotation = e.__annotations__["caught_exception"] - assert annotation == "Optional[TargetingError]" + assert annotation == "TargetingError | None" def test_question_returns_none_if_no_element_found(self, Tester: Actor) -> None: test_target = Target.the("foo").located_by("//bar") diff --git a/tests/test_resolutions.py b/tests/test_resolutions.py index 23cf502..2502168 100644 --- a/tests/test_resolutions.py +++ b/tests/test_resolutions.py @@ -1,11 +1,10 @@ from __future__ import annotations from dataclasses import dataclass +from typing import TYPE_CHECKING import pytest from hamcrest.core.string_description import StringDescription -from screenpy import BaseResolution -from selenium.webdriver.remote.webelement import WebElement from screenpy_selenium import IsClickable, IsInvisible, IsPresent, IsVisible from screenpy_selenium.resolutions.custom_matchers.is_clickable_element import ( @@ -23,6 +22,10 @@ from .useful_mocks import get_mocked_element +if TYPE_CHECKING: + from screenpy import BaseResolution + from selenium.webdriver.remote.webelement import WebElement + @dataclass class ExpectedDescriptions: diff --git a/tests/test_target.py b/tests/test_target.py index 76eb8ba..6449dc2 100644 --- a/tests/test_target.py +++ b/tests/test_target.py @@ -1,7 +1,8 @@ from __future__ import annotations +from typing import TYPE_CHECKING + import pytest -from screenpy import Actor from selenium.common.exceptions import WebDriverException from selenium.webdriver.common.by import By @@ -9,6 +10,9 @@ from .useful_mocks import get_mocked_browser +if TYPE_CHECKING: + from screenpy import Actor + def test_can_be_instantiated() -> None: t1 = Target.the("test") diff --git a/tests/useful_mocks.py b/tests/useful_mocks.py index af44ce6..10a3892 100644 --- a/tests/useful_mocks.py +++ b/tests/useful_mocks.py @@ -1,15 +1,17 @@ from __future__ import annotations -from typing import Tuple, cast +from typing import TYPE_CHECKING, cast from unittest import mock -from screenpy import Actor from selenium.webdriver.common.action_chains import ActionChains from selenium.webdriver.remote.webdriver import WebDriver from selenium.webdriver.remote.webelement import WebElement from screenpy_selenium import BrowseTheWeb, Target +if TYPE_CHECKING: + from screenpy import Actor + def get_mocked_element() -> mock.Mock: return mock.create_autospec(WebElement, instance=True) @@ -22,13 +24,12 @@ def get_mocked_chain() -> mock.Mock: def get_mock_target_class() -> type: class FakeTarget(Target): def __new__(cls, *args: object, **kwargs: object) -> FakeTarget: # noqa: ARG003 - rt = mock.create_autospec(FakeTarget, instance=True) - return rt + return mock.create_autospec(FakeTarget, instance=True) return FakeTarget -def get_mocked_target_and_element() -> Tuple[mock.Mock, mock.Mock]: +def get_mocked_target_and_element() -> tuple[mock.Mock, mock.Mock]: """Get a mocked target which returns a mocked element.""" target = get_mock_target_class()() element = get_mocked_element() @@ -38,8 +39,7 @@ def get_mocked_target_and_element() -> Tuple[mock.Mock, mock.Mock]: def get_mocked_browser(actor: Actor) -> mock.Mock: - browser = cast(mock.Mock, actor.ability_to(BrowseTheWeb).browser) - return browser + return cast(mock.Mock, actor.ability_to(BrowseTheWeb).browser) def get_mocked_webdriver() -> mock.Mock: