【问题标题】:Python Selenium Explicit WaitPython Selenium 显式等待
【发布时间】:2017-10-22 12:26:04
【问题描述】:

在我的 Python Selenium 测试中,我很难找到一种干净可靠的等待方式。我已经使用隐式等待很长时间了,由于某种原因它开始不稳定,所以我切换到显式等待。出于某种原因,我无法找到一种方法让我的测试在 100% 的时间内都能正常工作。现在,即使正在测试的代码和测试本身没有改变,但问题却一次又一次地发生。 (我会说它有 50% 的时间有效)。当问题发生时,通常是相同的行。以下是其中一些:

for th in td_list:
    if(th.text == "18"):
        th.click()
        break
    time.sleep(3)
    element = WebDriverWait(driver, 20).until(
      EC.element_to_be_clickable((By.ID, "closeModalDisplayBill"))
    )

这里的“closeModalDisplayBill”是一个必须按下的按钮。它提出的问题如下:

Traceback (most recent call last):
  File "./scripts/test-sl/SeleniumTest.py", line 54, in test_selenium_launcher
    frontAdminErr = SlFrontAdminErr(driver, self.add)
  File "/Users/JNEB/WebstormProjects/backend/backend/scripts/test-sl/SlFrontAdminErr.py", line 45, in __init__
    driver.find_element_by_id("closeModalAddComp").click()
  File "/Library/Python/2.7/site-packages/selenium/webdriver/remote/webelement.py", line 78, in click
    self._execute(Command.CLICK_ELEMENT)
  File "/Library/Python/2.7/site-packages/selenium/webdriver/remote/webelement.py", line 499, in _execute
    return self._parent.execute(command, params)
  File "/Library/Python/2.7/site-packages/selenium/webdriver/remote/webdriver.py", line 297, in execute
    self.error_handler.check_response(response)
  File "/Library/Python/2.7/site-packages/selenium/webdriver/remote/errorhandler.py", line 194, in check_response
    raise exception_class(message, screen, stacktrace)
ElementNotInteractableException: Message: 

另一个经常出现问题的地方是 Selenium 尝试打开模式来编辑某些内容(单击或清除输入非常不稳定)。这是代码:

time.sleep(5)
driver.implicitly_wait(15)
driver.find_element_by_id("btnLogin").click()
time.sleep(1)
driver.find_element_by_id("login-email").clear()

这是错误发生时的日志(试图清除登录电子邮件输入字段):

Traceback (most recent call last):
  File "./scripts/test-sl/SeleniumTest.py", line 39, in test_selenium_launcher
    frontUserTest = SlFrontUser(self, driver, self.add)
  File "/Users/JNEB/WebstormProjects/backend/backend/scripts/test-sl/SlFrontUser.py", line 21, in __init__
    driver.find_element_by_id("login-email").clear()
  File "/Library/Python/2.7/site-packages/selenium/webdriver/remote/webelement.py", line 93, in clear
    self._execute(Command.CLEAR_ELEMENT)
  File "/Library/Python/2.7/site-packages/selenium/webdriver/remote/webelement.py", line 499, in _execute
    return self._parent.execute(command, params)
  File "/Library/Python/2.7/site-packages/selenium/webdriver/remote/webdriver.py", line 297, in execute
    self.error_handler.check_response(response)
  File "/Library/Python/2.7/site-packages/selenium/webdriver/remote/errorhandler.py", line 194, in check_response
    raise exception_class(message, screen, stacktrace)
InvalidElementStateException: Message: Element is not currently interactable and may not be manipulated

相同的测试可以连续成功四到五次,然后在相同的时间内出错。 (这里的一切都是在我自己的电脑上测试的)。当这些测试通过 Jenkins 完成时,情况会变得更糟。有时整套测试可以在 10 或 8 分钟内完成,但有时这些测试将在 30 分钟后完成/失败。我认为 Jenkins 的缓慢因素可能是我的测试不稳定的原因之一,但这并不能解释为什么这些错误经常出现在我自己的计算机上。

所有这些测试都是从另一个启动 firefox 驱动程序实例的 Python 脚本启动的,然后像这样启动所有测试:

class SeleniumTestLauncher(unittest.TestCase):
    environmnent = "envName"
    add = ""
    quickbooks_url = "https://developer.intuit.com/"
    port = ""
    driver = webdriver.Firefox()
    base_url = ""

    def setUpAdd(self):
        self.driver.implicitly_wait(30)
        self.verificationErrors = []
        self.accept_next_alert = True

    def test_selenium_launcher(self):
        driver = self.driver
        ### -- Here just call every selenium test -- ###

        ## -- Test Front User -- ##
        frontUserTest = SlFrontUser(self, driver, self.add)

第二次编辑: 正如建议的那样,我删除了所有显式等待的隐式等待()。似乎更稳定,但我仍然遇到错误。例如:

element = WebDriverWait(driver, 20).until(
      EC.element_to_be_clickable((By.ID, "closeModalDisplayBill"))
)
driver.find_element_by_id("closeModalDisplayBill").click()

即使驱动程序应该等待“closeModalDisplayBill”可点击才能实际尝试点击,我还是会收到此错误:

Traceback (most recent call last):
  File "./scripts/test-sl/SeleniumTest.py", line 53, in test_selenium_launcher
    frontUserTest = SlFrontUser(self, driver, self.add)
  File "/Users/JNEB/WebstormProjects/backend/backend/scripts/test-sl/SlFrontUser.py", line 37, in __init__
    driver.find_element_by_id("closeModalDisplayBill").click()
  File "/Library/Python/2.7/site-packages/selenium/webdriver/remote/webelement.py", line 78, in click
    self._execute(Command.CLICK_ELEMENT)
  File "/Library/Python/2.7/site-packages/selenium/webdriver/remote/webelement.py", line 499, in _execute
    return self._parent.execute(command, params)
  File "/Library/Python/2.7/site-packages/selenium/webdriver/remote/webdriver.py", line 297, in execute
    self.error_handler.check_response(response)
  File "/Library/Python/2.7/site-packages/selenium/webdriver/remote/errorhandler.py", line 194, in check_response
    raise exception_class(message, screen, stacktrace)
ElementNotInteractableException: Message: 

编辑 3: 只使用显式等待和以前一样不稳定。我遇到的大多数错误是:“InvalidElementStateException:消息:元素当前不可交互并且可能无法操作”。我不明白为什么 Selenium 会尝试单击“输入字段”,即使我使用了显式等待。例如:

time.sleep(3)
element = WebDriverWait(driver, 20).until(      
  EC.presence_of_element_located((By.ID, "login-email"))
)
driver.find_element_by_id("login-email").clear()

在这里,我要求 selenium 等待“登录电子邮件”出现,如果是,请单击它。等待在 20 秒超时之前返回,但 .clear() 函数抛出“InvalidElementStateException:消息:元素当前不可交互并且可能无法操作”。

【问题讨论】:

  • 你用的是什么浏览器?跨浏览器是否一致?
  • @LeviNoecker 我一直在使用 Firefox。我正在使用 webdriver 的 init 编辑帖子
  • @jineb92 不建议混合隐式和显式等待,因为您会遇到不可预知的超时。
  • @Grasshopper 通过混合隐式和显式超时,您的意思是我应该使用“time.sleep(x)”还是“implicitly_wait()”?如果我选择在某个时候使用显式超时,我应该将所有隐式超时更改为显式超时?
  • @jineb92 使用显式或隐式等待。如果您根本不使用隐式等待,代码是否有效?

标签: python python-2.7 selenium jenkins


【解决方案1】:

closeModalDialog 表明您的应用程序基于某些 Ajax 框架。

使用 Selenium 自动化 Ajax 应用程序有时可能是一个巨大的挑战。当 Webriver 尝试定位元素时,元素似乎在时间 X 可见且可点击,但仍有一些待处理的 javascript 代码修改 DOM,并将在测试尝试单击时在时间 X+10 毫秒更改此元素在元素上,并且在这种情况下,Webdriver 通常会抛出以下之一:ElementNotVisibleException、InvalidElementStateException、StaleElementReferenceException、ElementNotInteractableException。
该测试在本地(快速)机器上运行良好,但在速度较慢的虚拟机上失败,或者当应用程序处于高负载且响应时间增加时。


首先与您的开发人员讨论这个问题。
我们的应用程序基于 JSF + Primefaces framework,在尝试自动化我们的产品时,我们遇到了许多类似上述的问题。我们已经与我们的开发人员讨论了这些问题,他们在每个页面的标题中添加了简单的Status component,这表明是否有正在进行的(活动的)ajax 请求。其次,由于 Primefaces 使用 jQuery(许多 ajax 框架都使用 jQuery),我们也在检查 jQuery 状态是否准备就绪(详情请参阅this answer
我们创建了一个简单的 waitForPrimefacesAndJQuery 函数,它等待 Unitil Primefaces 状态和 jQuery 状态准备就绪,并且一直在随处使用这个函数以及标准的 WebDriverWait。
这已经解决了 99.9% 的问题。强>


但仍有一些组件不想合作。
对于他们,我们通常使用蛮力方法 - 简单地排除异常并重试几次,例如以这种方式:

for(int i=0; i <= 5; i++ ){
  try{
    findElement( element ).click();
    break;
  } catch( ElementNotVisibleException | InvalidElementStateException 
          | StaleElementReferenceException | ElementNotInteractableException )
  {
    // swallow the exception, wait some time and try again
    sleep( 500 );
  }
}

【讨论】:

  • 非常感谢这个答案,我想我会为经常出现的问题实施蛮力方法。我们没有使用 jQuery,而是使用 Angular/Type Script,但我觉得 jQuery 中的问题是我遇到的问题。只有当我在慢速机器上运行测试时才会出现问题。再次感谢
猜你喜欢
  • 1970-01-01
  • 2023-03-11
  • 2013-12-03
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
相关资源
最近更新 更多