python-client icon indicating copy to clipboard operation
python-client copied to clipboard

mypy error "Incompatible return value type" on webdriver.find_element

Open bandophahita opened this issue 5 months ago • 7 comments

Mypy errors on the fact that appium.webdriver.WebDriver.find_element isn't overridden but indeed returns a different WebElement.

Reproducible Code

# no need to actually run this code, just send it through MyPy to see the error.

from appium.webdriver.common.appiumby import AppiumBy

from typing import TYPE_CHECKING

if TYPE_CHECKING:
    from appium.webdriver import Remote
    from appium.webdriver.webelement import WebElement

def find_element(driver: Remote) -> WebElement:
    # should get a mypy error on next line
    return driver.find_element(AppiumBy.ACCESSIBILITY_ID, "demo button")


def find_elements(driver: Remote) -> WebElement:
    # should get a mypy error on next line
    return driver.find_elements(AppiumBy.ACCESSIBILITY_ID, "demo confetti")

mypy output:

mypy .
screenpy_appium/target.py:89: error: Incompatible return value type (got "selenium.webdriver.remote.webelement.WebElement", expected "appium.webdriver.webelement.WebElement")  [return-value]
screenpy_appium/target.py:98: error: Incompatible return value type (got "list[selenium.webdriver.remote.webelement.WebElement]", expected "list[appium.webdriver.webelement.WebElement]")  [return-value]
Found 2 errors in 1 file (checked 35 source files)

Solution

Annotations need to be added to appium.webdriver.WebDriver

class WebDriver(
    webdriver.Remote,
    ...
):
    if TYPE_CHECKING:
        def find_element(self, by=By.ID, value: Optional[str] = None) -> MobileWebElement:
            ...
        def find_elements(self, by=By.ID, value: Optional[str] = None) -> list[MobileWebElement]:
            ...

bandophahita avatar Jul 23 '25 18:07 bandophahita

Hm, let me check. Here returns appium.webdriver.webelement.WebElement as their types:

https://github.com/appium/python-client/blob/d43a190cc99dcae31f13c074b7f7ac9e7552f9d9/appium/protocols/webdriver/can_find_elements.py#L15-L24

KazuCocoa avatar Jul 24 '25 02:07 KazuCocoa

What about this? https://github.com/appium/python-client/pull/1148

KazuCocoa avatar Jul 24 '25 05:07 KazuCocoa

Please try out 5.1.2

KazuCocoa avatar Jul 24 '25 06:07 KazuCocoa

The inheritance of the protocol in this manner doesn't result in mypy understanding that the methods return the appium subclassed WebElement. As a result, the sample code in this issue still errors in mypy. (though, for a moment it seemed like clever way to solve the problem)

In this case, I think you have to include the TYPE_CHECKING logic.

bandophahita avatar Jul 25 '25 15:07 bandophahita

Hm, it looks like Protocol https://github.com/appium/python-client/blob/d43a190cc99dcae31f13c074b7f7ac9e7552f9d9/appium/protocols/webdriver/can_find_elements.py#L15-L24 needs some adjustment for mypy. I'll explore/learn a bit more

KazuCocoa avatar Aug 02 '25 03:08 KazuCocoa

https://github.com/appium/python-client/pull/1153

KazuCocoa avatar Aug 02 '25 05:08 KazuCocoa

5.1.3 has the change. cc @bandophahita

https://github.com/appium/python-client/pull/1153

It has a change like you suggested. Protocol usage for mypy needed some work, so the suggested one was the simplest so far.

KazuCocoa avatar Aug 07 '25 02:08 KazuCocoa