【发布时间】: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