11

我对 selenium chromedriver 有疑问,我无法弄清楚是什么原因造成的。几周前一切正常,突然这个错误开始出现。问题来自以下功能。

 def login_(browser):
    try:
        browser.get("some_url")
        # user credentials
        user = browser.find_element_by_xpath('//*[@id="username"]')
        user.send_keys(config('user'))
        password = browser.find_element_by_xpath('//*[@id="password"]')
        password.send_keys(config('pass'))
        login = browser.find_element_by_xpath('/html/body/div[1]/div/button')
        login.send_keys("\n")
        time.sleep(1)
        sidebar = browser.find_element_by_xpath('//*[@id="sidebar"]/ul/li[1]/a')
        sidebar.send_keys("\n")
        app_submit = browser.find_element_by_xpath('//*[@id="sidebar"]/ul/li[1]/ul/li[1]/a')
        app_submit.send_keys("\n")
    except TimeoutException or NoSuchElementException:
        raise LoginException

此函数在开发环境(macOS 10.11)中运行没有问题,但在生产环境中抛出以下错误:

Message: no such element: Unable to locate element: {"method":"xpath","selector":"//*[@id="sidebar"]/ul/li[1]/a"}
(Session info: headless chrome=67.0.3396.79)
(Driver info: chromedriver=2.40.565383 (76257d1ab79276b2d53ee97XXX),platform=Linux 4.4.0-116-generic x86_64)

我已经在每个环境中更新了 Chrome 和 chromedriver(分别为 v67 和 2.40)。我也给了更多time.sleep(15)。但问题仍然存在。我最近的猜测是,也许 webdriver 的初始化工作不正常:

def initiate_webdriver():
   option = webdriver.ChromeOptions()
   option.binary_location = config('GOOGLE_CHROME_BIN')
   option.add_argument('--disable-gpu')
   option.add_argument('window-size=1600,900')
   option.add_argument('--no-sandbox')
   if not config('DEBUG', cast=bool):
       display = Display(visible=0, size=(1600, 900))
       display.start()
       option.add_argument("--headless")
   else:
       option.add_argument("--incognito")
   return webdriver.Chrome(executable_path=config('CHROMEDRIVER_PATH'), chrome_options=option)

因为,如果Display不起作用,则可能没有提到的按钮,sidebar而是其他按钮。

所以我的问题是:有人遇到过类似的问题吗?有没有办法知道驱动程序在寻找这样的元素时显示的页面是什么?

4

4 回答 4

4

提供登录后报告元素未找到错误,所以我认为登录失败并且页面重定向到某个地方。您可以使用屏幕截图选项对页面进行屏幕截图,然后查看驱动程序加载的页面。

driver.save_screenshot("path to save screen.jpeg")

您还可以保存原始 html 代码并检查同一页面。

Webdriver 截图

在 Python 中使用 Selenium 在 Firefox 上保存网页

于 2018-06-22T18:57:37.310 回答
3

根据login_(browser)方法有几件事:

  • 当您通过以下方式识别登录按钮时:

    login = browser.find_element_by_xpath('/html/body/div[1]/div/button')
    

    我建议宁愿通过onclick()事件send_keys("\n")的帮助来模拟单击登录按钮,如下所示:login.click()

    login = browser.find_element_by_xpath('/html/body/div[1]/div/button')
    login.click()
    
  • 接下来,当您识别侧边栏时,将WebDriverWait诱导为可点击的元素,如下所示:

    WebDriverWait(driver, 20).until(EC.element_to_be_clickable((By.XPATH, '//*[@id="sidebar"]/ul/li[1]/a'))).click()
    
  • 正如您提到的,您的代码块在macOS 10.11环境中完美运行,但在生产环境 (Linux) 中引发以下错误,不同的浏览器很有可能在不同的操作系统架构中呈现不同的HTML DOM 。因此,您必须使用相对 xpath而不是绝对 xpath,如下所示:

    WebDriverWait(driver, 20).until(EC.element_to_be_clickable((By.XPATH, "//a[@attribute='value']"))).click()
    

根据initiate_webdriver()方法有几件事:

  • 根据Headless Chrome 入门,该参数--disable-gpu仅适用于Windows ,但不适用于Linux OS的有效配置。所以需要删除:

    option.add_argument('--disable-gpu')
    

注意:您必须添加以下导入:

from selenium.webdriver.support.ui import WebDriverWait
from selenium.webdriver.common.by import By
from selenium.webdriver.support import expected_conditions as EC
于 2018-06-25T06:50:33.467 回答
2

每当我在 Selenium 中遇到这样的奇怪问题时,我宁愿重试找到导致间歇性故障的特定元素。一种方法是将其包裹在 try-except 块中:

try:
   sidebar = browser.find_element_by_xpath('//*[@id="sidebar"]/ul/li[1]/a')
except NoSuchElementException:
   time.sleep(10)
   print("Unable to find element in first time, trying it again")
   sidebar = browser.find_element_by_xpath('//*[@id="sidebar"]/ul/li[1]/a')

您还可以将try代码放入带有合适计数变量的循环中,以使自动化代码工作。(检查这个)。根据我使用 JAVA 的经验,这个想法已经解决了多个问题。

于 2018-06-25T06:16:59.763 回答
2

您需要等到元素可见,否则您将收到此错误。尝试这样的事情:

from selenium.webdriver.support.wait import WebDriverWait
from selenium.webdriver.support.expected_conditions import visibility_of_element_located
from selenium.webdriver.common.by import By
TIMEOUT = 5

...
xpath = '//*[@id="sidebar"]/ul/li[1]/a'
WebDriverWait(self.selenium, TIMEOUT).until(visibility_of_element_located((By.XPATH, xpath)))
browser.find_element_by_xpath(xpath)
...
于 2018-06-28T15:39:17.740 回答