mypy error "Incompatible return value type" on webdriver.find_element
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]:
...
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
What about this? https://github.com/appium/python-client/pull/1148
Please try out 5.1.2
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.
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
https://github.com/appium/python-client/pull/1153
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.