使用dataprovider执行并行TestNG selenium测试的驱动程序行为

时间:2013-07-18 12:55:02

标签: selenium automated-tests selenium-webdriver testng selenium-grid

我想在TestNg中并行运行使用@dataprovider的selenium测试。理想情况下,测试是通过方法并行(一个测试=一个方法)而不是浏览器的简单套件并行性。我在某处读过,一次可以控制大约5个ChromeDriver实例,所以我认为这应该是可行的。后来我打算搬到grid2。为了开发,我通过在XML配置文件上右键单击+运行来运行IntelliJ Idea测试运行器。

我在并行运行测试时遇到了问题(在grid2和本地),所以我创建了一个或多或少我想做的样本。

这是我的测试类

package tests;

import org.openqa.selenium.By;
import org.openqa.selenium.Keys;
import org.openqa.selenium.WebDriver;
import org.openqa.selenium.WebElement;
import org.openqa.selenium.chrome.ChromeDriver;
import org.openqa.selenium.firefox.FirefoxDriver;
import org.openqa.selenium.interactions.Actions;
import org.testng.annotations.*;
import java.util.concurrent.TimeUnit;
import static org.testng.Assert.assertNotNull;

public class ParallelTest {
    public static final String SEARCH_TERMS = "search-terms";
    private WebDriver driver;

    @BeforeMethod
    @Parameters({"browser"})
    public void beforeMethod(@Optional("chrome") String browser){
        driver = getBrowser(browser);
        driver.manage().deleteAllCookies();
        driver.manage().timeouts().implicitlyWait(10, TimeUnit.SECONDS);
    }

    private WebDriver getBrowser(String browser) {
        if(browser.equals("chrome")){
            System.setProperty("webdriver.chrome.driver", "webdrivers\\chromedriver.exe");
            return new ChromeDriver();
        }
        return new FirefoxDriver();
    }

    @AfterMethod
    public void afterMethod(){
        driver.quit();
    }

    @Test(description = "Check parallel selenium works.",
          dataProvider = SEARCH_TERMS)
    public void parallelSeleniumTest(String searchTerm){
        driver.get("http://google.com");
        WebElement search = driver.findElement(By.id("gbqfq"));
        new Actions(driver)
                .sendKeys(search, searchTerm)
                .sendKeys(search, Keys.ENTER)
                .perform();
        String firstResult = driver.findElements(By.className("r")).get(0).getText();
        assertNotNull(firstResult);
        System.out.println(firstResult);
    }

    @DataProvider(name = SEARCH_TERMS, parallel = true)
    public Object[][] getSearchTerms(){
        return new Object[][]{
                {"google"},
                {"microsoft"},
                {"facebook"},
                {"amazon"},
                {"apple"},
                {"oracle"},
                {"yahoo"},
                {"jetbrains"},
                {"intellij idea"},
                {"selenium"},
                {"java"},
                {"testng"},
                {"code"}
        };
    }
}

我投入了一些原生事件,因为我在测试套件中大量使用它们。

这是TestNg xml配置文件

<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE suite SYSTEM "http://testng.org/testng-1.0.dtd">
<suite thread-count="4" name="vfr6-ui-tests" parallel="methods">
    <test name="parallel-test-firefox">
        <parameter name="browser" value="firefox"/>
        <classes>
            <class name="tests.ParallelTest"/>
        </classes>
    </test>
    <test name="parallel-test-chrome">
        <parameter name="browser" value="chrome"/>
        <classes>
            <class name="tests.ParallelTest"/>
        </classes>
    </test>
</suite>

我读过每次测试实例化一个驱动程序往往是最易维护的。问题是firefox测试是串行运行的,而chrome测试将所有数据点作为测试用例吐出,尝试打开大量浏览器实例,然后一切都失败了。我的测试将有10-25或300-500个数据点(在客户或客户x产品之间循环)。

设置驱动程序,数据提供程序和测试运行器以在运行测试中实现最佳并行性的最佳方法是什么?

2 个答案:

答案 0 :(得分:3)

我对dataProvider有相同的体验。在我的例子中,我使用了dataProvider(parallel = true)属性。您的问题有两种解决方案。

  1. 使用dataProvider并在测试类中使用工厂注释作为构造函数。 在工厂注释的属性中,使用dataProvider =“您的dataProvider的名称”。 在testng.xml中,使用parallel = instances。

    而不是parallel = methods

    上述方法的缺点是当你收到报告时;可能是maven- surefire,testng Eclipse报告或reportNG报告,你没有看到传递的参数 面前。要解决此问题,您可以使用以下方法。

  2. 使用a创建工厂类并在工厂方法中实例化测试类 for循环。 (从0开始循环)。在测试类中定义一个构造函数 从工厂类接收参数。在此测试类中定义dataProvider 可以使用构造函数中收到的参数(data-point)。定义BeforeMethod 或者可以使用该参数或数据点的BeforeClass和您的测试方法 将“dataProvider”属性指向所需的dataProvider。再次,在 testng.xml使用parallel =“instances”。

    此外,使用try / catch块实例化驱动程序对象并关闭浏览器。这个 将帮助您避免由于tearUown方法的setUp失败而跳过。

答案 1 :(得分:1)

你真的不需要使用工厂。如果我是你,我会在dataprovider方法中调用此代码:

driver = getBrowser(browser);

然后,将驱动程序实例作为第二列args返回到测试方法。这样做允许数据提供者生成浏览器实例。为了改进这一点,您可以使用DriverHelper类形式的构建器设计模式,它可以替换getBrowser方法,以便在将驱动程序实例传递给测试方法之前生成更具体的驱动程序配置。 / p>

注意:请记住,如果您以后想要使用Spring加载驱动程序,那么此方法根本不起作用。实际上,您根本无法使用DataProvider。但是,如果你不使用Spring,那么我会说这是最优雅的方式。