我们正在将Selenium与Java API和一些Javascript用户扩展一起使用。我们在应用程序中使用了很多AJAX调用。我们的许多测试都是随机失败的,因为有时AJAX调用的完成速度比其他时间慢,因此页面未完全加载。我们通过等待特定元素或Thread.sleep来解决此问题。我试图找到一种方法来代替等待网络流量完成。这样我们就可以做到这一点:
selenium.click("some JS button"); selenium.waitForNetwork(); assertTrue(something);
这样,我们可以摆脱线程休眠,并在服务器响应速度更快时使测试通过更快,并且不会因为计时问题而导致太多测试失败。
我一直无法找到一种方法来搜索Google。有谁知道我们如何做到这一点?(最好通过Javascript或Java API,但欢迎所有建议)。
注意:“ waitFor”的其他变体不是我想要的。我们已经在点击和其他方面使用了这些内容。我正在寻找等待网络流量的东西。感谢您的所有反馈,我将尝试一些建议,但我仍然欢迎其他想法。
谢谢。
更新:我与Selenium2以相同的方式使用相同的方法(wrapRequest),但仍然可以使用。 唯一的区别是我不使用用户扩展名。我只是执行javascript来查询JSP加载的代码。
这就是我们最终要做的。(请注意,如果您始终只使用单个框架(如JQuery),则有一种更简便的方法来获取是否正在运行Ajax调用。我们不得不进行更多的jimmy- rig操作,因为我们有几个页面使用JQuery,使用GWT)
我创建了一个Javascript文件,只有在进行测试时,每个页面才会加载。(我通过在JSP的模板中包含一个代码段来完成此操作:如果将测试查询字符串参数作为true传递,则设置一个cookie。如果设置了cookie,则包括javascript文件)这个javascript文件实际上包装了XMLHttpRequest对象以保持计数所有发送请求中:
function wrapRequest(xmlRequest) { var oldSend = xmlRequest.prototype.send; xmlRequest.prototype.send = function() { var oldOnReady = this.onreadystatechange; oldOnReady = function() { oldOnReady.apply(this, arguments); // if call is complete, decrement counter. } // increment counter if(oldSend.apply) oldSend.apply(this, arguments); else if (oldOnReady.handleEvent) { //Firefox sometimes will need this oldOnReady.handleEvent.apply(this, arguments); } } }
如果async == false,它还不止这些,但是应该很容易找出来。
然后,我在selenium user-extensions.js中添加了一个简单的函数:
Selenium.prototype.getIsNetworkReady = function() { if(this.browserbot.topWindow.isNetworkReady) { //sometimes this method is called before the page has loaded the isNetworkReady function return this.browserbot.topWindow.isNetworkReady(); } return false; }
另外,请注意对topWindow的检查:这是因为在选择iframe时遇到问题。
另外,您将必须在要加载的每个iFrame的XMLHttpRequest对象上调用wrapRequest方法。我现在使用:进行此操作document.addEventListener("DOMNodeInserted", onElementAdded, true);,然后在onElementAdded中,仅在iframe加载时抓取并传递XMLHttpRequest对象。但这在IE中不起作用,因此我正在寻找替代方法。如果有人知道有更好的方法可以在IE和FF中使用,那么我很想听听。
document.addEventListener("DOMNodeInserted", onElementAdded, true);
无论如何,这就是实现的要旨。希望这对将来的任何人有帮助。