草庐IT

javascript - 如何调试等待异步 Angular 任务超时?无法在 Angular 页面上找到元素

coder 2024-07-24 原文

编辑:请注意,我在@ernst-zwingli 的帮助下找到了问题的根源,因此如果您遇到相同的错误,他的一个修复程序可能会对您有所帮助。我的问题是 Protractor 本身的一个已知问题,如果您认为这可能是您,我已经扩展了我的步骤以在我最初提出问题后查明问题的根源。

我正在尝试在使用 angular-cli 构建的 Angular2(只是 Angular)应用程序中使用 Protractor 。

我的问题:当 browser.waitForAngularEnabled 时找不到 Angular 应用程序页面上的元素处于默认设置 true (就像“我相信我在一个 Angular 页面上并且希望 Protractor 做到这一点很神奇”)。如果我设置 browser.waitForAngularEnabled,它们会被发现很好至 false (如“我不在一个 Angular 页面上,想自己处理这个问题,请坐 Protractor”)。如何在我的绝对 Angular 页面上追踪导致此问题的原因?

我有一个带有非 Angular Auth0 登录页面的产品,该页面可以访问用 Angular 编写的产品的其余部分(准确地说是 Angular 4.3.2)。我已经成功遍历了非 Angular 登录页面的登录。我翻转了 waitForAngularEnabled切换到 false方便非Angular登录。我把它转回 true在单击提交按钮后,我预计我的初始登录页面 (Angular) 将被加载。代码如下:

        browser.waitForAngularEnabled(false);
        browser.driver.get('https://dashboard.net/projects');
        browser.driver.sleep(10000);
        browser.driver.findElement(By.css("[type='email']"));
        browser.driver.findElement(By.css("[type='email']")).sendKeys("email@example.com");
        browser.driver.findElement(By.css(".auth0-label-submit")).click();

        browser.driver.findElement(By.id("passwordInput")).sendKeys("password");
        browser.driver.findElement(By.id("submitButton")).click();
        browser.driver.sleep(5000);  // needed if not waiting for Angular
        //browser.waitForAngularEnabled(true);  // Back to Protractor land we go
        let elementToFind = element(by.className("header-text"));
        elementToFind.isDisplayed().then(function() {grabTheDarnLocalStorage()});
        expect(elementToFind.isDisplayed()).toBeTruthy();

如果我取消注释 browser.waitForAngularEnabled(true);行声明我回到了 Angular 代码中,我得到如下错误跟踪:
Failed: Timed out waiting for asynchronous Angular tasks to finish after 30 seconds. This may be because the current page is not an Angular application. Please see the FAQ for more details: https://github.com/angular/protractor/blob/master/docs/timeouts.md#waiting-for-angular
While waiting for element with locator - Locator: By(css selector, .header-text)
ScriptTimeoutError: asynchronous script timeout: result was not received in 30 seconds
  (Session info: chrome=61.0.3163.100)
  (Driver info: chromedriver=2.32.498550 (9dec58e66c31bcc53a9ce3c7226f0c1c5810906a),platform=Windows NT 10.0.14393 x86_64)
    at WebDriverError (C:\Users\c-shouston\AppData\Roaming\npm\node_modules\protractor\node_modules\selenium-webdriver\lib\error.js:27:5)
    at ScriptTimeoutError (C:\Users\c-shouston\AppData\Roaming\npm\node_modules\protractor\node_modules\selenium-webdriver\lib\error.js:203:5)
    at Object.checkLegacyResponse (C:\Users\c-shouston\AppData\Roaming\npm\node_modules\protractor\node_modules\selenium-webdriver\lib\error.js:505:15)
    at parseHttpResponse (C:\Users\c-shouston\AppData\Roaming\npm\node_modules\protractor\node_modules\selenium-webdriver\lib\http.js:509:13)
    at doSend.then.response (C:\Users\c-shouston\AppData\Roaming\npm\node_modules\protractor\node_modules\selenium-webdriver\lib\http.js:440:13)
    at process._tickCallback (internal/process/next_tick.js:109:7)
From: Task: Protractor.waitForAngular() - Locator: By(css selector, .header-text)

我引用了常见问题解答:https://github.com/angular/protractor/blob/master/docs/timeouts.md#waiting-for-angular
我的开发人员说他们不使用 $timeout(他们使用(编辑:NOT $interval)Observable Interval,非常感谢)并且他们不确定 $http。

我找到了这个关于调试 Protractor Angular 同步问题的规范方法的解决方案:Canonical way to debug Protractor-to-Angular sync issues
但我不确定该解决方案在无法修改开发代码以运行程序化跟踪器的情况下是否有效。 (编辑:我从来没有想过如何让它工作)

我还发现这是关于您在每次测试之前添加的长时间超时,但我认为这是不必要的开销,它使您的整体测试执行时间比应有的时间更长,而无需了解问题的根本原因:https://stackoverflow.com/a/37217167/2718402 (编辑:是的,这是一个坏主意,会给您的测试增加不必要的时间,请不要这样做)

令人沮丧的是,这似乎很常见,而且似乎没有关于如何处理它的简化文档。使用非 Angular 页面登录只是为了转换到 Angular 页面。 Protractor 未正确拾取 Angular 页面。我在网上找到的所有示例都是一些代码,我没有引用它们应该在我的整体测试框架中的位置。我会杀死一个完整的例子,有人测试一个非 Angular 登录,该登录转换到一个完整的 Angular 网站,带有设置配置和真实世界的测试用例。 (编辑:这仍然是正确的,但我无法自己制作一个,因为我的应用程序处于糟糕的灰色区域,请注意下面的 RCA 以获取更多详细信息。)

我只希望能够进行登录,然后成功转换到我的 Angular 页面,并且能够依靠 Protractor 来处理我的 Angular 页面。我需要知道要查找的内容可能是长时间运行的异步进程(我可以在 Chrome 开发工具中具体检查什么?)。我很想了解 Protractor 需要什么作为默认值才能成功地处理我的应用程序/网站的 Angular 部分(HTML 中是否有 <app-root _nghost-c0="" ng-version="4.3.2"> 之外的东西?)。在这份工作之前,我在 Java 工作,所以所有这些异步性和 Angular 对我来说都是新的,所以我知道我错过了经验丰富的 Javascript 开发人员所知道的已知事物。

我的解决方案/根本原因分析

从@ernst-zwingli 建议的列表开始:

for Angular(2) Check if the object window.getAllAngularRootElements returns at least one value.



它至少返回了一个值,所以我继续前进。

useAllAngular2AppRoots: true,



我试过了,但仍然遇到了异步超时。

And if $interval or other long lasting asynchronous tasks are used, there can be issues, because of the zones



之前@ernst-zwingli 也提到了查看可测试性方法,只是它是旧方法。通过研究和测试,我找到了 window对象也有一个 getAllAngularTestabilities方法。这导致了一个有趣的兔子洞。 Chrome 控制台的输出示例(将 window.getAllAngularTestabilities() 放在 Chrome 控制台窗口中,查看结果列表)如下:
t:
  _callbacks:...,
  _didWork:true,
  _isZoneStable: true (this looks promising, but why isn't Protractor working then?!?)
  _ngZone:
    hasPendingMacrotasks: true,
    hasPendingMicrotasks: false,
    isStable: true

我认为 isZoneStable 就足够了,但对于 Protractor 显然不是这样。然后看到 Macrotasks 是真的,我不得不查一下 Macrotask 到底是什么:What does hasPendingMacrotasks and hasPendingMicrotasks check for? .

宏任务可以是:

i.e. setTimeout, setInterval, setImmediate



因此,@ernst-zwingli 关于间隔在区域中引起问题的注释被记住了,并且最终点击了一些东西。
First github issue, about zone instability

Another github issue提示使用 browser.driver 和 browser.waitForAngularEnabled 一起完成工作的必要性。显然这是预期的行为,它导致我发出 #3349

Issue #3349 - 我的问题的实际根本原因。我的开发人员不会主动进出可观察对象的区域。即使这些 observables 只有一个订阅者。由于此时它们处于 Angular 区域,因此它们是 Protractor 无限等待的长时间运行的“宏任务”。

我无法用这些包装器重写代码,因为我目前对 Angular 还不够精通,无法安全地做到这一点,而且我们目前正朝着 11 月的最后期限迈进。我想我暂时必须处理使用 browser.driver 并希望我以后不能修复它。希望我的 RCA 对您有所帮助。

最佳答案

在下面我列出了一组潜在的原因和修复/解决它们的可能性。

AngularJS 和 Angular(2) 如何工作/我可以在浏览器开发模式中检查什么

我无法解释它以及Andrey Agibalov in his Blog here ,所以检查一下(也适用于开发人员)。

基本上,Protractor 所需的对象可以在 Chrome Dev 中检查。

对于 AngularJS
检查对象是否window.angular正确定义,即查找 window.angular.version并尝试 window.angular.getTestability你的根元素

对于 Angular(2)
检查对象是否window.getAllAngularRootElements至少返回一个值。

根元素 (AngularJS)

可能你的 Angular 应用程序被包裹在 Body 中的某个地方,比如 <div ng-app="my-app"> .
在这种情况下,您必须调整您的 rootElement: bodyconfig.ts . Check this answer for details .

Angular (2)

如果您使用的是 Angular(又名 Angular2),那么引入了 ngZone。在这种情况下,您的 config.js还应包含以下内容:

exports.config = {
    framework: 'jasmine',
    seleniumAddress: 'http://localhost:4444/wd/hub',
    specs: ['spec.js'],
    useAllAngular2AppRoots: true,
    // rootElement: 'root-element'
};

在浏览器中查看 window.getAllAngularRootElements因为 conf.js 中的附加行是关于这个的。

如果可以,也许可以利用多个区域的优势。创建第二个区域,配置 rootElement: 'root-element'只关注一个区域,然后将一些异步任务移动到另一个区域,直到找到导致超时的任务。将这些任务(如果可能)放在单独的区域中,以便 Protractor 忽略这些任务。

如果 $interval或使用其他持久的异步任务,可能会出现问题,因为区域。重复或持久的任务应该在区域外开始,然后移动到区域中,否则 Protractor 可能会超时。有一种解决方法可供开发人员申请,以避免 Protractor 出现这些问题。
read all about it here

浏览器.驱动程序。 - 旁注
browser.driver.get()就像 ignoreSynchronization = true 一样工作,因为您直接分配浏览器驱动程序并且您绕过了 Protractor 的同步逻辑。
this answer here 中阅读更多相关信息.

希望我能给你更多的意见,你可以解决你的问题。请让我知道结果。

关于javascript - 如何调试等待异步 Angular 任务超时?无法在 Angular 页面上找到元素,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/46588749/

有关javascript - 如何调试等待异步 Angular 任务超时?无法在 Angular 页面上找到元素的更多相关文章

  1. ruby - 如何使用 Nokogiri 的 xpath 和 at_xpath 方法 - 2

    我正在学习如何使用Nokogiri,根据这段代码我遇到了一些问题:require'rubygems'require'mechanize'post_agent=WWW::Mechanize.newpost_page=post_agent.get('http://www.vbulletin.org/forum/showthread.php?t=230708')puts"\nabsolutepathwithtbodygivesnil"putspost_page.parser.xpath('/html/body/div/div/div/div/div/table/tbody/tr/td/div

  2. ruby - 如何从 ruby​​ 中的字符串运行任意对象方法? - 2

    总的来说,我对ruby​​还比较陌生,我正在为我正在创建的对象编写一些rspec测试用例。许多测试用例都非常基础,我只是想确保正确填充和返回值。我想知道是否有办法使用循环结构来执行此操作。不必为我要测试的每个方法都设置一个assertEquals。例如:describeitem,"TestingtheItem"doit"willhaveanullvaluetostart"doitem=Item.new#HereIcoulddotheitem.name.shouldbe_nil#thenIcoulddoitem.category.shouldbe_nilendend但我想要一些方法来使用

  3. ruby - 其他文件中的 Rake 任务 - 2

    我试图在一个项目中使用rake,如果我把所有东西都放到Rakefile中,它会很大并且很难读取/找到东西,所以我试着将每个命名空间放在lib/rake中它自己的文件中,我添加了这个到我的rake文件的顶部:Dir['#{File.dirname(__FILE__)}/lib/rake/*.rake'].map{|f|requiref}它加载文件没问题,但没有任务。我现在只有一个.rake文件作为测试,名为“servers.rake”,它看起来像这样:namespace:serverdotask:testdoputs"test"endend所以当我运行rakeserver:testid时

  4. python - 如何使用 Ruby 或 Python 创建一系列高音调和低音调的蜂鸣声? - 2

    关闭。这个问题是opinion-based.它目前不接受答案。想要改进这个问题?更新问题,以便editingthispost可以用事实和引用来回答它.关闭4年前。Improvethisquestion我想在固定时间创建一系列低音和高音调的哔哔声。例如:在150毫秒时发出高音调的蜂鸣声在151毫秒时发出低音调的蜂鸣声200毫秒时发出低音调的蜂鸣声250毫秒的高音调蜂鸣声有没有办法在Ruby或Python中做到这一点?我真的不在乎输出编码是什么(.wav、.mp3、.ogg等等),但我确实想创建一个输出文件。

  5. ruby-on-rails - 由于 "wkhtmltopdf",PDFKIT 显然无法正常工作 - 2

    我在从html页面生成PDF时遇到问题。我正在使用PDFkit。在安装它的过程中,我注意到我需要wkhtmltopdf。所以我也安装了它。我做了PDFkit的文档所说的一切......现在我在尝试加载PDF时遇到了这个错误。这里是错误:commandfailed:"/usr/local/bin/wkhtmltopdf""--margin-right""0.75in""--page-size""Letter""--margin-top""0.75in""--margin-bottom""0.75in""--encoding""UTF-8""--margin-left""0.75in""-

  6. ruby-on-rails - 如何验证 update_all 是否实际在 Rails 中更新 - 2

    给定这段代码defcreate@upgrades=User.update_all(["role=?","upgraded"],:id=>params[:upgrade])redirect_toadmin_upgrades_path,:notice=>"Successfullyupgradeduser."end我如何在该操作中实际验证它们是否已保存或未重定向到适当的页面和消息? 最佳答案 在Rails3中,update_all不返回任何有意义的信息,除了已更新的记录数(这可能取决于您的DBMS是否返回该信息)。http://ar.ru

  7. ruby-on-rails - 'compass watch' 是如何工作的/它是如何与 rails 一起使用的 - 2

    我在我的项目目录中完成了compasscreate.和compassinitrails。几个问题:我已将我的.sass文件放在public/stylesheets中。这是放置它们的正确位置吗?当我运行compasswatch时,它不会自动编译这些.sass文件。我必须手动指定文件:compasswatchpublic/stylesheets/myfile.sass等。如何让它自动运行?文件ie.css、print.css和screen.css已放在stylesheets/compiled。如何在编译后不让它们重新出现的情况下删除它们?我自己编译的.sass文件编译成compiled/t

  8. ruby - 如何将脚本文件的末尾读取为数据文件(Perl 或任何其他语言) - 2

    我正在寻找执行以下操作的正确语法(在Perl、Shell或Ruby中):#variabletoaccessthedatalinesappendedasafileEND_OF_SCRIPT_MARKERrawdatastartshereanditcontinues. 最佳答案 Perl用__DATA__做这个:#!/usr/bin/perlusestrict;usewarnings;while(){print;}__DATA__Texttoprintgoeshere 关于ruby-如何将脚

  9. ruby - 如何指定 Rack 处理程序 - 2

    Rackup通过Rack的默认处理程序成功运行任何Rack应用程序。例如:classRackAppdefcall(environment)['200',{'Content-Type'=>'text/html'},["Helloworld"]]endendrunRackApp.new但是当最后一行更改为使用Rack的内置CGI处理程序时,rackup给出“NoMethodErrorat/undefinedmethod`call'fornil:NilClass”:Rack::Handler::CGI.runRackApp.newRack的其他内置处理程序也提出了同样的反对意见。例如Rack

  10. ruby - 如何每月在 Heroku 运行一次 Scheduler 插件? - 2

    在选择我想要运行操作的频率时,唯一的选项是“每天”、“每小时”和“每10分钟”。谢谢!我想为我的Rails3.1应用程序运行调度程序。 最佳答案 这不是一个优雅的解决方案,但您可以安排它每天运行,并在实际开始工作之前检查日期是否为当月的第一天。 关于ruby-如何每月在Heroku运行一次Scheduler插件?,我们在StackOverflow上找到一个类似的问题: https://stackoverflow.com/questions/8692687/

随机推荐