Selenium屏幕截图侦听器捕获错误的浏览器

时间:2016-09-23 17:26:15

标签: java selenium selenium-webdriver testng

我有一个通过testng运行并行测试的selenium项目。当测试失败时,我有一个捕获截图的监听器类。该课程如下

public class ScreenshotOnFailure extends TestListenerAdapter {

@Override
public void onTestFailure(ITestResult tr) {
    WebDriver driver = SeleniumSetup.driverrunning;
    boolean hasQuit = driver.toString().contains("(null)");
    if(!hasQuit){
        System.out.println(driver);
        File scrFile = ((TakesScreenshot)driver).getScreenshotAs(OutputType.FILE);
        DateFormat dateFormat = new SimpleDateFormat("dd_MMM_yyyy__hh_mm_ssaa");
        Date date = new Date();
        String NewFileNamePath = null;
        File directory = new File(".");
        String methodName = tr.getMethod().getMethodName();
        try {
            NewFileNamePath =directory.getCanonicalPath() + "\\target\\surefire-reports\\html\\Screenshots\\"+methodName+"_"+ dateFormat.format(date) +"Listener.png";
        } catch (IOException e1) {
            // TODO Auto-generated catch block
            e1.printStackTrace();
        }


        try {
            FileUtils.copyFile(scrFile, new File(NewFileNamePath));
        } catch (IOException e) {
            e.printStackTrace();
        }
        String reportFilePath = ".\\Screenshots\\"+methodName+"_"+ dateFormat.format(date) +".png";
        System.setProperty("org.uncommons.reportng.escape-output", "false");    
        Reporter.log("<a href=" + reportFilePath + ">Click to open screenshot</a><img src=" + reportFilePath +  " height='350' width='700'>");      
    }
}}

在我的测试中,我有一个可以清理测试的AfterMethod

    @AfterMethod(alwaysRun = true)
public void tearDown() throws Exception
{
    driver.quit();
}

如果逐个运行测试,则会捕获正确的浏览器屏幕截图,但是如果我运行并行测试,则会捕获错误的测试浏览器。 我认为问题可能是以下之一

  • after方法已经退出浏览器(这是一个案例 有时为什么我必须添加hasQuit boolean)
  • 侦听器引用了错误的驱动程序对象。我相信这是问题,但我不确定如何确保它引用正确的驱动程序。

我有一个解决方法,几乎​​可以创建一个静态屏幕捕获对象,然后将其添加到AfterMethod,但这不太理想,因为我想使用一个监听器。

1 个答案:

答案 0 :(得分:3)

从您的代码WebDriver driver = SeleniumSetup.driverrunning,似乎driverrunning是SeleniumSetup类中的静态驱动程序实例。因此,在并行执行中,它可能引用了错误的驱动程序对象。

ThreadLocal可以帮助您创建线程安全的驱动程序对象,下面是一个示例。

public class DriverFactory
{

   private DriverFactory()
   {
      //Do-nothing..Do not allow to initialize this class from outside
   }
   private static DriverFactory instance = new DriverFactory();

   public static DriverFactory getInstance()
   {
      return instance;
   }

   ThreadLocal<WebDriver> driver = new ThreadLocal<WebDriver>() // thread local driver object for webdriver
   {
      @Override
      protected WebDriver initialValue()
      {
         return new FirefoxDriver(); // can be replaced with other browser drivers
      }
   };

   public WebDriver getDriver() // call this method to get the driver object and launch the browser
   {
      return driver.get();
   }

   public void removeDriver() // Quits the driver and closes the browser
   {
      driver.get().quit();
      driver.remove();
   }
}

使用DriverFactory获取驱动程序实例。

WebDriver driver = DriverFactory.getInstance().getDriver();