geckodriver 中的“无法访问死对象”

Posted

技术标签:

【中文标题】geckodriver 中的“无法访问死对象”【英文标题】:"Can't access dead object" in geckodriver 【发布时间】:2017-10-15 17:54:41 【问题描述】:

我在 Java 中使用 Selenium 3.4。使用 Chrome,一切正常。但是我需要使用 Firefox,并且出现了一些问题。

我正在自动化 Dojo UI 的测试,并且需要等待 Dojo UI 进行大量渲染。所以这就是我所做的,它在 Chrome 中运行良好。请注意,我的代码中通常设置了 20 秒的隐式等待。

driver.switchTo().defaultContent();
driver.switchTo().frame(driver.findElement(By.id("contentframe"))); // relying on implicit wait 
driver.manage().timeouts().implicitlyWait(0, TimeUnit.SECONDS);
(new WebDriverWait(driver, 120)).
     until(ExpectedConditions.elementToBeClickable(By.id("some_id")));

我已经简化了代码,因此您看不到隐式等待是如何设置回 20 秒的。当问题发生时,无论如何它不会到达那里。 WebDriverWait 导致异常。异常说TypeError: can't access dead object

wait里面有对应的消息:

May 16, 2017 3:36:11 PM org.openqa.selenium.support.ui.ExpectedConditions findElement
WARNING: WebDriverException thrown by findElement(By.id: 
some_id)
org.openqa.selenium.WebDriverException: TypeError: can't access dead object

显然,geckodriver 还有一些 javascript 错误输出:

JavaScript error: chrome://marionette/content/listener.js, line 1555: TypeError: can't access dead object
*************************
A coding exception was thrown and uncaught in a Task.

Full message: TypeError: can't access dead object
Full stack: find_@chrome://marionette/content/element.js:284:7
element.find/</findElements<@chrome://marionette/content/element.js:255:15
implicitlyWaitFor/</elementSearch@chrome://marionette/content/element.js:600:15
implicitlyWaitFor/<@chrome://marionette/content/element.js:627:5
implicitlyWaitFor@chrome://marionette/content/element.js:593:10
element.find/<@chrome://marionette/content/element.js:254:24
element.find@chrome://marionette/content/element.js:253:10
findElementsContent@chrome://marionette/content/listener.js:1314:19
TaskImpl_run@resource://gre/modules/Task.jsm:319:42
TaskImpl@resource://gre/modules/Task.jsm:277:3
createAsyncFunction/asyncFunction@resource://gre/modules/Task.jsm:252:14
Task_spawn@resource://gre/modules/Task.jsm:166:12
TaskImpl_handleResultValue@resource://gre/modules/Task.jsm:389:16
TaskImpl_run@resource://gre/modules/Task.jsm:327:15
TaskImpl@resource://gre/modules/Task.jsm:277:3
createAsyncFunction/asyncFunction@resource://gre/modules/Task.jsm:252:14
Task_spawn@resource://gre/modules/Task.jsm:166:12
dispatch/<@chrome://marionette/content/listener.js:186:15

*************************
*************************
A coding exception was thrown and uncaught in a Task.

Full message: TypeError: can't access dead object
Full stack: find_@chrome://marionette/content/element.js:284:7
element.find/</findElements<@chrome://marionette/content/element.js:255:15
implicitlyWaitFor/</elementSearch@chrome://marionette/content/element.js:600:15
implicitlyWaitFor/<@chrome://marionette/content/element.js:627:5
implicitlyWaitFor@chrome://marionette/content/element.js:593:10
element.find/<@chrome://marionette/content/element.js:254:24
element.find@chrome://marionette/content/element.js:253:10
findElementsContent@chrome://marionette/content/listener.js:1314:19
TaskImpl_run@resource://gre/modules/Task.jsm:319:42
TaskImpl@resource://gre/modules/Task.jsm:277:3
createAsyncFunction/asyncFunction@resource://gre/modules/Task.jsm:252:14
Task_spawn@resource://gre/modules/Task.jsm:166:12
TaskImpl_handleResultValue@resource://gre/modules/Task.jsm:389:16
TaskImpl_run@resource://gre/modules/Task.jsm:327:15
TaskImpl@resource://gre/modules/Task.jsm:277:3
createAsyncFunction/asyncFunction@resource://gre/modules/Task.jsm:252:14
Task_spawn@resource://gre/modules/Task.jsm:166:12
dispatch/<@chrome://marionette/content/listener.js:186:15

此外,我的自动异常处理尝试截取屏幕截图,但失败并出现同样的错误。代码行是:

File snapshotTempFile = ((TakesScreenshot) driver).getScreenshotAs(OutputType.FILE); 而这次 geckodriver 的输出是:

A coding exception was thrown and uncaught in a Task.

Full message: TypeError: can't access dead object
Full stack: capture.viewport@chrome://marionette/content/capture.js:65:7
takeScreenshot@chrome://marionette/content/listener.js:1782:14
dispatch/</req<@chrome://marionette/content/listener.js:188:22
TaskImpl_run@resource://gre/modules/Task.jsm:319:42
TaskImpl@resource://gre/modules/Task.jsm:277:3
createAsyncFunction/asyncFunction@resource://gre/modules/Task.jsm:252:14
Task_spawn@resource://gre/modules/Task.jsm:166:12
dispatch/<@chrome://marionette/content/listener.js:186:15

那么,我可以做些什么来使它正常工作吗?这是我需要作为 geckodriver 错误提出的问题吗?

我唯一可以用谷歌搜索的是:https://github.com/mozilla/geckodriver/issues/614,唯一提出的解决方案是driver.switchTo().defaultContent()。这可能会修复我的屏幕截图例程,但我正在等待的元素位于内容框架内,因此我无法使用此修复来等待。

【问题讨论】:

【参考方案1】:

在您等待元素 some_id 时,看起来框架已重新加载了新的引用。 我将此问题归类为错误,因为驱动程序返回的错误不是由WebDriver 协议定义的。

让它工作的最佳机会可能是实现一个自定义服务员来定位框架/元素并跳过未处理的异常:

WebElement elem = waiter.Until(elementToBeClickableInFrame(By.id("contentframe"),
                                                           By.id("some_id")));
public static ExpectedCondition<WebElement> elementToBeClickableInFrame(final By locatorFrame, final By locator) 
  return new ExpectedCondition<WebElement>() 

    @Override
    public WebElement apply(WebDriver driver) 
      try 

        driver.switchTo().defaultContent();
        driver.switchTo().frame(driver.findElement(locatorFrame));

        WebElement elem = driver.findElement(locator);
        return elem.isDisplayed() && elem.isEnabled() ? elem : null;

       catch (Exception e) 
        return null;
      
    

    @Override
    public String toString() 
      return "element located by: " + locator + " in " + locatorFrame;
    
  ;

【讨论】:

谢谢!这已经奏效了。 (我已经修改它以使用单独的类定义而不是大括号初始化)。我最终在 geckodriver 跟踪器上找到了问题,有了这个答案,我知道如何发表评论。我已经做到了。问题在github.com/mozilla/geckodriver/issues/614【参考方案2】:

不确定这是否会对您有所帮助,但是当我遇到此错误消息时,我能够通过以下方式克服它:

driver.switchTo().defaultContent();

driver.switchTo().frame(0);

在与 iframe 中的元素的每次交互之间。

例子:

driver.switchTo().frame(0);
    myPage.selectElement(getCycleSummary());
    driver.switchTo().defaultContent();
    driver.switchTo().frame(0);
    myPage.selectDisplayedElement(this.getCycleBtn());

如果没有驱动程序开关,我会收到死对象错误。

【讨论】:

【参考方案3】:

看,

应该有我在下面给出的格式:

    首先切换到框架。 (如果您已经在另一个框架中,请先切换到默认值)。

    执行您的操作(点击任何元素)

    再次切换到默认内容。 (如果不切换回来,就会产生问题)。

         driver.switchTo().defaultContent();
    
         driver.switchTo().frame(locator or name of the frame);
    
    
         driver.click(your element locator);
    
    
         driver.switchTo().defaultContent();
    

【讨论】:

以上是关于geckodriver 中的“无法访问死对象”的主要内容,如果未能解决你的问题,请参考以下文章

Python3&selenium raise 消息:无法连接到服务 geckodriver

cx_freeze 可执行文件无法与 Geckodriver 配合使用

Ruby selenium webdriver 无法找到 Mozilla geckodriver

Selenium远程驱动程序无法找到geckodriver

如何使用 selenium firefox geckodriver 自动激活 Flash 插件?

如何从 Java 中的 BotD 中隐藏 Geckodriver 中的 WebDriver?