python的selenium“send_keys”与chrome驱动程序删除字符

时间:2013-08-02 09:48:04

标签: python google-chrome selenium key send

我在Windows 7中使用带有Python(https://pypi.python.org/pypi/selenium)的selenium包。 当我尝试登录我的Facebook帐户时,我使用send_keys命令,例如

elem = browser.find_element_by_name("email")
elem.send_keys(email);
elem = browser.find_element_by_name("pass")
elem.send_keys(password);

登录失败显然是因为第二个send_keys删除了密码的第一个字符(我通过直接将密码字符发送到电子邮件字段找到了这个。

发生了什么事?为什么selenium不能像将密钥发送到输入字段那样简单? 这是一种由Facebook编码的拒绝自动脚本的保护措施吗?

试图发送整个字母并得到这个:

abcdefghijklmnopqrstuvwxyzBCDFGIKLNOQSTWX

注意缺少多少个字符......

更新

显然问题与facebook无关,而与Chrome驱动程序无关。

如果我发送以下简单代码

browser = webdriver.Chrome()
browser.get("https://www.google.com") # Load page
elem = browser.find_element_by_name("q") # Find the query box
query= "ABCDEFGHIJKLMNOPQRSTUVWXYZ"
elem.send_keys(query)

我得到镀铬驱动程序     BCDFGIKLNOQSTWX 注意A,E,H ... Y,Z是如何丢失的 使用firefox驱动程序(用browser = webdriver.Chrome()替换browser = webdriver.Firefox()我得到:     ABCDEFGHIJKLMNOPQRSTUVWXYZ

4 个答案:

答案 0 :(得分:0)

看起来Chrome webdriver中有一些错误:https://code.google.com/p/chromedriver/issues/detail?id=435

当问题的键盘配置为非英语语言,或者webdriver进程和chrome显示器在不同的语言/环境中运行时(例如,当通过远程显示器从一个主人到另一个,等等。)

答案 1 :(得分:0)

我已经解决了使用自定义方法send_keys的问题,该方法工作效率较低但足够快。

from selenium.webdriver.remote.webelement import WebElement

def send_keys(el: WebElement, keys: str):
    for i in range(len(keys)):
        el.send_keys(keys[i])

send_keys(el, keys)

答案 2 :(得分:0)

我遇到了与OP类似的问题,对这里写的答案以及我在该网站上其他任何地方都找不到的答案不满意。 自从我在研究期间读到该问题已在某个时间点解决后,我不得不假设它在如今已经再次出现。

以P.T.提到过,如果您遇到类似的问题,则chromedrivers / firefoxdrivers可能会再次出现故障。

由于我发现的所有解决方案都无法缓解问题,因此我选择完全避开硒及其驱动程序。因此,我使用javascript首先(通过其绝对Xpath)找到该元素,然后对其进行写入/设置。

from selenium import webdriver

def write_to_element(driver, element_xpath, input_string), 
        js_command = f'document.evaluate(\'{xpath}\', document, null, XPathResult.FIRST_ORDERED_NODE_TYPE, null).singleNodeValue.value = \'{input_string}\';'
        driver.execute_script(js_command)

driver = webdriver.Chrome()
driver.get('WebPage/With/Element/You/Want/To/Write/To')
xpath = 'Xpath/Of/Element/You/Want/To/Write/To'
write_to_element(driver, xpath, 'SomeRandomInput')

document.evaluate(\'{xpath}\', document, null, XPathResult.FIRST_ORDERED_NODE_TYPE, null)会评估元素的xpath,因此本质上是在网页上找到您的元素。

.singleNodeValue.value = \'{input_string}\';'将该元素的值设置为input_string

答案 3 :(得分:-1)

在python中使用selenium Ide和导出测试用例

from selenium import webdriver
from selenium.webdriver.common.by import By
from selenium.webdriver.support.ui import Select
from selenium.common.exceptions import NoSuchElementException
import unittest, time, re

class Test1(unittest.TestCase):
    def setUp(self):
        self.driver = webdriver.Firefox()
        self.driver.implicitly_wait(30)
        self.base_url = "https://www.facebook.com/"
        self.verificationErrors = []
        self.accept_next_alert = True

    def test_1(self):
        driver = self.driver
        driver.get(self.base_url + "/")
        driver.find_element_by_id("email").clear()
        driver.find_element_by_id("email").send_keys("username")
        driver.find_element_by_id("pass").clear()
        driver.find_element_by_id("pass").send_keys("password")
        driver.find_element_by_id("u_0_b").click()
        driver.find_element_by_xpath("//div[@id='u_ps_0_1_5']/div/div").click()
        driver.find_element_by_link_text("1 Requests").click()
        driver.find_element_by_id("globalContainer").click()

    def is_element_present(self, how, what):
        try: self.driver.find_element(by=how, value=what)
        except NoSuchElementException, e: return False
        return True

    def is_alert_present(self):
        try: self.driver.switch_to_alert()
        except NoAlertPresentException, e: return False
        return True

    def close_alert_and_get_its_text(self):
        try:
            alert = self.driver.switch_to_alert()
            alert_text = alert.text
            if self.accept_next_alert:
                alert.accept()
            else:
                alert.dismiss()
            return alert_text
        finally: self.accept_next_alert = True

    def tearDown(self):
        self.driver.quit()
        self.assertEqual([], self.verificationErrors)

if __name__ == "__main__":
    unittest.main()