Python Selenium - 类似身份验证的警报弹出窗口

Python Selenium - Alert Like Authentication Pop Up

嘿 Brilliant Stack 溢出社区,

我遇到了一个有趣的场景,涉及处理弹出用户身份验证框等警报。

1:使用的网站(练习网站):https://the-internet.herokuapp.com/

2:当我点击 Basic Auth 时,出现了类似弹出 window 的警告(见下文)。

3:到目前为止我尝试了什么。

from selenium import webdriver
from selenium.webdriver.chrome.service import Service
from selenium.webdriver.common.by import By
from selenium.webdriver.support.ui import WebDriverWait
from selenium.webdriver.support import expected_conditions

chrome_path = '.\chromedriver.exe'
chrome_service = Service(chrome_path)
chrome = webdriver.Chrome(service=chrome_service)
chrome.maximize_window()
chrome.get('https://the-internet.herokuapp.com/')
basic_auth = WebDriverWait(chrome,timeout=30).until(expected_conditions.element_to_be_clickable((By.PARTIAL_LINK_TEXT,'Basic Auth')))
basic_auth.click()

3-1: 将其视为警报 (switch_to.alert) & 使用发送键输入用户名和密码。但是得到了 NoAlertPresentException。

chrome.switch_to.alert.send_keys('user\npass')
# get selenium.common.exceptions.NoAlertPresentException

3-2: 将其视为活动元素并使用发送键。这次没有错误,但也没有任何内容发送到输入框。在我看来,这个弹出窗口不能成为一个元素。

chrome.switch_to.active_element.send_keys('user\npass')
# No error msg, but nothing show up in the pop up input field too. 

3-3: 将其视为一个新的window,但在检查有多少windows 活跃(driver.window_handles)时,只有一个window。所以这个弹出窗口也不是 window。

print(len(chrome.window_handles)) # return 1, so the pop up is not a window too

3-4:我偶然发现了这个 post,但我不确定我是否理解解决方案。

如果有人能帮助我解决如何在像这样的非 Web 元素中导航的问题,我将非常高兴。

非常感谢!

=======

问题更新

非常感谢@Tyzeron 和@Nic Laforge 的帮助!

post中提到的两种方法我都试过了,请问我做的对不对?

仅供参考:该网站本身是一个练习网站,因此每次您输入用户名和密码然后登录时,该网站都会生成一个新的基本身份验证弹出窗口。很难说我试过的方法是否有效。

方法一:使用get方法时,将Basic Authentication放在URL中

chrome_path = '.\chromedriver.exe'
chrome_service = Service(chrome_path)
chrome = webdriver.Chrome(service=chrome_service)
chrome.maximize_window()
chrome.get('https://username:pwd123@the-internet.herokuapp.com/basic_auth')

方法二:使用Selenium Wire(不知我的代码是否正确)

from seleniumwire import webdriver as wire_driver
import base64

def request_interceptor(req):
# add Authentication Here. 
# VXNlck5hbWU6UHNkMTIz is the base64 encoded str for "UserName:Psd123"
    req.headers['Authorization'] = 'Basic VXNlck5hbWU6UHNkMTIz=='
    print(req.headers)

chrome_path = '.\chromedriver.exe'
chrome_service = Service(chrome_path)
chrome = wire_driver.Chrome(service=chrome_service)
chrome.maximize_window()
chrome.request_interceptor = request_interceptor
chrome.get('https://the-internet.herokuapp.com/basic_auth')

selenium-wire 的第二种方法也没有产生错误,但是页面上没有可见的线索表明成功与否。

另外,函数中req对象的打印结果如下:

现在,这两种方法都没有产生错误,也没有可见的线索表明代码成功。所以我想知道如何判断它是否有效?

您无法“找到”元素的原因

警告框不是 HTML 元素。它不是网页的一部分,因此您无法在 HTML 代码中看到它。警告框是浏览器的一部分。

一些背景

您看到的是 Basic access authentication 的一个例子。正如 wiki 所述,通常会发生的情况是您的 app/browser 通过请求中的 header 字段(Authorization header)自动提供用户名和密码。在这种情况下,您的浏览器还不知道用户名和密码,因此它会通过浏览器的警告框询问。

我提出的解决方案

我相信使用 selenium 进行身份验证的最干净、最简单的方法是在您的 get 方法中提供凭据,如下所示:

chrome.get('http://username:password@domain')

在您的具体情况下,它将是 http://admin:admin@the-internet.herokuapp.com/basic_auth

然而,正如@Nic-Laforge提到的,这个解决方案依赖于浏览器支持它的事实。 (截至发文最新Chrome支持)

其他Whosebug的解决方案post

与我提出的解决方案不同,similar Whosebug post 中提出的两种解决方案都需要额外的库。

@Evander 先生提出的第一个解决方案是使用pynput 库来模拟键盘输入。此解决方案要求用户使用 non-headless 浏览器,使浏览器聚焦,并且在输入过程中不与键盘交互。

第二种解决方案要好得多。如上所述,Basic access authentication 需要您在请求中 Authorization header 中的凭据。 @Evander 先生所做的是使用 selenium-wire 库拦截 selenium 的请求并添加带有凭据的 header。