10

除了sleep()在我的测试中使用之外,我想知道是否有人知道在继续我的断言之前明确等待表单提交(POST)完成的更好策略。这是我的测试的一个非常精简的版本,使用来自 Facebook 的 phpunit 和php-webdriver)。

function test_form_submission()
{   
    // setup
    $web_driver = new WebDriver();
    $session = $web_driver->session();
    $session->open('http://example.com/login');

    // enter data
    $session->element('css selector', 'input[name=email]')->value(array('value' => str_split('test@example.com')));
    $session->element('css selector', 'input[name=password]')->value(array('value' => str_split('testpassword')));

    // click button to submit form
    $session->element('css selector', 'button[name=submit]')->click();

    // How do I wait here until the click() above submits the form
    // so I can check that we correctly arrives at the destination below
    // without resorting to sleep()?

    // Confirm that login was successful because we landed on account page
    $this->assertSame('http://example.com/account', $session->url());

    // teardown
    $session->close();
}
4

4 回答 4

20

Facebook的 php-webdriver已于 2013 年 6 月重写。您可以像这样轻松地等待 URL。

// wait for at most 10 seconds until the URL is 'http://example.com/account'.
// check again 500ms after the previous attempt.
$driver->wait(10, 500)->until(function ($driver) {
  return $driver->getCurrentURL() === 'http://example.com/account';
})
于 2013-08-14T23:01:03.570 回答
2

好的,所以我知道这是一个非常古老的问题,但是当我通过 Google 偶然发现它时,我希望这对某人仍然有用(甚至可能是 OP?:-))。

在谷歌代码上搜索和阅读文章和消息后,我发现了一个我认为非常丑陋的解决方案,但实际上并没有一个好的选择。您可以在此处阅读,WebDriver 无法检测到页面何时加载。所以,我们必须回到等待。现在,OP 正在使用 Facebook 的 PHP WebDriver,它不包括等待实用程序 AFAIK,但是当使用这个维护的克隆时,你会得到一个 WebDriverWait 的实现。

现在的问题是:我们还在等什么?您可以等待 URL 更改,但这绝不是一种可靠的方法,因为当 URL 已经更改时,页面可能尚未加载。下一个选项:等待您知道要加载的页面上的元素出现。这行得通,但是如果您想要更通用的方法怎么办?幸运的是,我在上面提到的线程中读到了一个非常好的主意。您可以等待两个页面上的元素消失并重新出现,例如页脚。

我刚刚实现了这个,它似乎工作。下面的函数可以传递给你$session,并且只会在新页面加载时返回(或者至少,页脚再次可用)。

public static function waitForNextPage($pWDSession) {
    $aWait = new PHPWebDriver_WebDriverWait($pWDSession);
    // wait for footer to not be available anymore
    $aWait->until(
            function($pWDSession) {
                return (0 === count($pWDSession->elements(PHPWebDriver_WebDriverBy::CSS_SELECTOR, "#footer,#mx-footer")));
            }
        );
    // wait for footer to be available again
    $aWait->until(
            function($pWDSession) {
                return (0 < count($pWDSession->elements(PHPWebDriver_WebDriverBy::CSS_SELECTOR, "#footer,#mx-footer")));
            }
        );
}

您显然可以更改选择器或使用其他元素,想法保持不变。

我希望它有帮助!干杯!

于 2013-07-31T12:27:10.263 回答
0

而不是click()应该可以使用clickAndWait(), 以便等到下一页已加载。

于 2013-01-22T20:38:30.323 回答
0

您可以使用Implicit Wait and Explicit Wait等待特定的 Web 元素,直到它出现在页面中。您可以定义的等待期取决于应用程序。

显式等待:

显式等待是您定义的代码,用于等待特定条件发生,然后再继续执行代码。如果条件达到,它将终止等待并继续执行进一步的步骤。

代码:

 WebDriverWait wait = new WebDriverWait(driver,30);
 wait.until(ExpectedConditions.visibilityOfElementLocated(By.id(strEdit)));

或者

 WebElement myDynamicElement = (new WebDriverWait(driver, 30))
 .until(new ExpectedCondition<WebElement>(){
@Override
public WebElement apply(WebDriver d) {
    return d.findElement(By.id("myDynamicElement"));
}});

这会在抛出 TimeoutException 之前等待最多 30 秒,或者如果它发现元素将在 0 - 30 秒内返回它。默认情况下,WebDriverWait 每 500 毫秒调用一次 ExpectedCondition,直到它成功返回。对于 ExpectedCondition 类型的成功返回是 Boolean 返回 true 或所有其他 ExpectedCondition 类型的非空返回值。

您可以根据应用程序的需要使用 ExpectedConditions 类。

隐式等待:

隐式等待是告诉 WebDriver 在尝试查找一个或多个元素时轮询 DOM 一段时间(如果它们不是立即可用的)

代码:

 driver.manage().timeouts().implicitlyWait(10, TimeUnit.SECONDS);

要记住的一件事是,一旦设置了隐式等待 - 它将在 WebDriver 对象实例的生命周期内保持不变

有关更多信息,请使用此链接http://seleniumhq.org/docs/04_webdriver_advanced.jsp

于 2013-01-23T04:09:53.503 回答