Селен, параллельный запуск TestNG работает не так, как ожидалось

#java #selenium-webdriver #testng #parallel-testing

Вопрос:

У меня есть 3 тестовых класса, состоящих из нескольких методов тестирования, которые я хочу запускать параллельно. Я использую ThreadLocal для изоляции экземпляров webdriver в потоке. Когда я запускаю тесты последовательно, все выглядит нормально, но проблема возникает, когда я запускаю их параллельно. Ниже приведен мой файл suite

 <!DOCTYPE suite SYSTEM "http://testng.org/testng-1.0.dtd">
<suite name="platform" parallel="classes" thread-count="5">
    <test name="platform">
        <classes>
            <class name="com.sat.platform.mobile.PlatformMobileIdCaptureMonitoringWf11"></class>
            <class name="com.sat.platform.mobile.PlatformMobileIdVerificationMonitoringWf2"></class>
            <class name="com.sat.platform.mobile.PlatformMobileIdandIVMonitoringWf3"></class>
            <class name="com.sat.platform.mobile.PlatformMobileLivenessMonitoringWf6"></class>
            <class name="com.sat.platform.mixed.PlatformMixedIdSimilarityMonitoringWf2and5"></class>
        </classes>
    </test>
</suite>
 

Я инициализирую Webdriver в @BeforeClass в BrowserClient.java как показано ниже.

     protected WebDriver driver;
    private static int implicitWaitTime;
    private static int explicitWaitTime;
    private static int fluentWaitTime;
    private static int pollingTime;
    protected static WebDriverWait explicitWait;
    protected static Wait<WebDriver> fluentWait;
    private static String browser;
    protected static Browsers browsers;

    static {
        Properties prop = new Properties();
        ClassLoader loader = Thread.currentThread().getContextClassLoader();
        InputStream stream = loader.getResourceAsStream("browser.properties");
        try {
            prop.load(stream);
        } catch (IOException e) {
        }
        implicitWaitTime = Integer.parseInt(prop.getProperty("browser.implicit.wait.timeout"));
        explicitWaitTime = Integer.parseInt(prop.getProperty("browser.explicit.wait.timeout"));
        fluentWaitTime = Integer.parseInt(prop.getProperty("browser.fluent.wait.timeout"));
        pollingTime = Integer.parseInt(prop.getProperty("browser.wait.polling.time"));
        browser = System.getProperty("browser");
    }

    @BeforeClass
    public void initializeEnv() throws MalformedURLException {
        driver = BrowserFactory.createInstance(browser, implicitWaitTime);
        DriverFactory.getInstance().setDriver(driver);
        driver = DriverFactory.getInstance().getDriver();
        explicitWait = new WebDriverWait(driver, explicitWaitTime);
        fluentWait = new FluentWait(driver).withTimeout(Duration.of(fluentWaitTime, SECONDS))
                .pollingEvery(Duration.of(pollingTime, SECONDS))
                .ignoring(NoSuchElementException.class);
    }
 

используемый здесь класс, т. е. BrowserFactory.java

 public static WebDriver createInstance(String browser, int implicitWaitTime) throws MalformedURLException {
        WebDriver driver = null;
        Browsers browserEnum = Browsers.valueOf(browser);
        String testVideo = ImageProcessingUtils.getAbsolutePath("digital_copy.mjpeg", false);

        switch (browserEnum) {
            case chrome:
                ChromeOptions options = new ChromeOptions();
                options.addArguments("--use-fake-ui-for-media-stream", "--use-fake-device-for-media-stream",
                        "--use-file-for-fake-video-capture="   testVideo, "--start-maximized");
                driver = new ChromeDriver(options);
                break;

            case firefox:
                FirefoxProfile firefoxProfile = new FirefoxProfile();
                firefoxProfile.setPreference("media.navigator.permission.disabled", true);
                firefoxProfile.setPreference("media.navigator.streams.fake", true);
                firefoxProfile .setPreference("browser.private.browsing.autostart", false);
                FirefoxOptions firefoxOptions = new FirefoxOptions();
                firefoxOptions.setProfile(firefoxProfile);
                driver = new FirefoxDriver(firefoxOptions);
                break;
     }
}
 

DriverFactory.java

 public class DriverFactory {

    private static DriverFactory instance = new DriverFactory();
    private static ThreadLocal<WebDriver> driver = new ThreadLocal<>();
    private static List<WebDriver> driversList = new ArrayList();

    private DriverFactory(){

    }

    public static DriverFactory getInstance() {
        return instance;
    }

    public WebDriver getDriver(){
        WebDriver localDriver = driver.get();
        driversList.add(localDriver);
        return localDriver;
    }

    public void setDriver(WebDriver driver){
        this.driver.set(driver);
    }

    public static void removeDriver(){
        for(WebDriver driver : driversList) {
            driver.quit();
        }
    }
}
 

И мои тестовые занятия расширяются BrowserClient.java где я могу напрямую использовать драйвер. Одним из распространенных методов во всех 3 классах тестов является merchant_gets_oauth_token (), как показано ниже. Проблема в том, что при запуске набора тестов параллельно открываются 3 браузера firefox, и все они переходят на страницу входа, но только 1, а иногда и 2 теста проходят, в то время как 3-й не удается (не удается войти в систему).

     @Test
    public void merchant_gets_oauth_token() {
        OAuth2Client client = new OAuth2Client();
        String loginUrl = DslConfigFactory.getEnvConfig("portal.customer.url");
        driver.get(loginUrl);
        CustomerPortalLoginPage loginPage = new CustomerPortalLoginPage(driver);
        log.info("----------merchant logging to customer portal to get oauth token----------");
        loginPage.login(merchantUser.getEmail(), merchantUser.getPassword());
        CustomerPortalHomePage homePage = new CustomerPortalHomePage(driver);
        homePage.clickOnSettings();
        CustomerPortalSettingsPage settingsPage = new CustomerPortalSettingsPage(driver);
        settingsPage.clickOnApiCredentials();
        CustomerPortalApiCredentialsPage apiCredentialsPage = new CustomerPortalApiCredentialsPage(driver);
        clientCredentials = apiCredentialsPage.getOauth2ClientCredentials();
        oauthToken = client.getOauthToken(clientCredentials.get("token"), clientCredentials.get("secret"));
    }
 

Я некоторое время боролся с этой проблемой и без посторонней помощи просмотрел множество онлайн-ресурсов. Может быть, кто-нибудь здесь в состоянии сделать RCA. заранее спасибо!!

Ответ №1:

 public static void removeDriver(){
        for(WebDriver driver : driversList) {
            driver.quit();
        }
    }

Here, replace driver.quit() with driver.close().
driver.close() method it will close only current driver. driver.close() closes all drivers. 

When first instance executes removeDriver(), it will closes all windows/drivers. Then next instance try to execute removeDriver(),there driver is not available because it is already closed by previous instance of driver.so you get error.
 

Комментарии:

1. removeDriver вызывается в @AfterSuite , поэтому драйверы закрываются только после завершения всех тестов.