繁体   English   中英

TestNG 是否保证 @BeforeSuite 方法在 @BeforeTest 方法之前执行?

[英]Does TestNG guarantee @BeforeSuite methods are executed before @BeforeTest methods?

背景:我的目标是编写一个运行自包含的 TestNG-Selenium 系统(没有 Maven 或 Ant 插件的字符串;只有 Java)。 它必须允许测试用例接受包括浏览器和域 URL 在内的参数。 TestRunner实例化这些测试用例时,浏览器和域用于获取 Selenium 对象以执行其测试。

问题:在尝试获取 Selenium 对象(在@BeforeTest )之前,每个套件只有一个测试类成功获取域参数(在@BeforeSuite方法中)。 不接收域的测试类有一个null selenium 对象 b/c,它不能被实例化。

代码:XmlClasses 每个都包含在它们自己的 XmlTest 中,并且所有三个都在一个 XmlSuite 中。 该套件包含 TestClass1、TestClass2、TestClass3 的顺序。 测试类本身是 2 层抽象基类的子类,其中包括初始化注入变量并随后获取 Selenium 实例的功能。 这样做的目的是用尽可能少的重复代码来测试一个或多个应用程序(在多个域上)(即:Selenium 实例化在根基类中,因为它对所有测试都是通用的)。 有关详细信息,请参阅以下方法。

// Top-most custom base class
abstract public class WebAppTestBase extends SeleneseTestBase
{
        private static Logger logger = Logger.getLogger(WebAppTestBase.class);
        protected static Selenium selenium = null;
        protected String domain = null;
        protected String browser = null;

        @BeforeTest(alwaysRun = true)
        @Parameters({ "selenium.browser" })
        public void setupTest(String browser)
        {
                this.browser = browser;
                logger.debug(this.getClass().getName()
                                + " acquiring Selenium instance ('" + this.browser + " : " + domain + "').");
                selenium = new DefaultSelenium("localhost", 4444, browser, domain);
                selenium.start();
        }

}

// Second level base class.
public abstract class App1TestBase extends WebAppTestBase
{

        @BeforeSuite(alwaysRun = true)
        @Parameters({"app1.domain" })
        public void setupSelenium(String domain)
        {
                // This should execute for each test case prior to instantiating any Selenium objects in @BeforeTest
                logger.debug(this.getClass().getName() + " starting selenium on domain '" + domain+ "'.");
                this.domain = domain;
        }
}

// Leaf level test class
public class TestClass1 extends App1TestBase
{
        @Test
        public void validateFunctionality() throws Exception
        {
                // Code for tests go here...
        }
}

// Leaf level test class
public class TestClass2 extends App1TestBase
{
        @Test
        public void validateFunctionality() throws Exception
        {
                selenium.isElementPresent( ...
                // Rest of code for tests go here...
                // ....
        }
}


// Leaf level test class
public class TestClass3 extends App1TestBase
{
        @Test
        public void validateFunctionality() throws Exception
        {
                // Code for tests go here...
        }
}

输出:TestCase3 正确运行。 TestCase1 和 TestCase2 失败。 生成堆栈跟踪...

 10:08:23 [DEBUG RunTestCommand.java:63] - Running Tests.
 10:08:23 [Parser] Running:
  Command line suite
  Command line suite

[DEBUG App1TestBase.java:49] - TestClass3 starting selenium on domain 'http://localhost:8080'.
 10:08:24 [DEBUG WebAppTestBase.java:46] - TestClass2 acquiring Selenium instance ('*firefox : null').
 10:08:24 [ERROR SeleniumCoreCommand.java:40] - Exception running 'isElementPresent 'command on session null
 10:08:24 java.lang.NullPointerException: sessionId should not be null; has this session been started yet?
        at org.openqa.selenium.server.FrameGroupCommandQueueSet.getQueueSet(FrameGroupCommandQueueSet.java:216)
        at org.openqa.selenium.server.commands.SeleniumCoreCommand.execute(SeleniumCoreCommand.java:34)
        at org.openqa.selenium.server.SeleniumDriverResourceHandler.doCommand(SeleniumDriverResourceHandler.java:562)
        at org.openqa.selenium.server.SeleniumDriverResourceHandler.handleCommandRequest(SeleniumDriverResourceHandler.java:370)
        at org.openqa.selenium.server.SeleniumDriverResourceHandler.handle(SeleniumDriverResourceHandler.java:129)
        at org.openqa.jetty.http.HttpContext.handle(HttpContext.java:1530)
        at org.openqa.jetty.http.HttpContext.handle(HttpContext.java:1482)
        at org.openqa.jetty.http.HttpServer.service(HttpServer.java:909)
        at org.openqa.jetty.http.HttpConnection.service(HttpConnection.java:820)
        at org.openqa.jetty.http.HttpConnection.handleNext(HttpConnection.java:986)
        at org.openqa.jetty.http.HttpConnection.handle(HttpConnection.java:837)
        at org.openqa.jetty.http.SocketListener.handleConnection(SocketListener.java:245)
        at org.openqa.jetty.util.ThreadedServer.handle(ThreadedServer.java:357)
        at org.openqa.jetty.util.ThreadPool$PoolThread.run(ThreadPool.java:534)

我很感激您在此问题上可能拥有的任何信息。

我认为问题在于您的@BeforeSuite方法正在为一个字段分配一个值,但您有三个不同的实例,因此其他两个从未被初始化。

请记住, @BeforeSuite只运行一次,无论它属于哪个类。 因此, @Before/AfterSuite方法通常定义在整个测试环境之外的类上。 这些方法应该是static但我决定不强制执行这个要求,因为它有时是不切实际的。

我认为解决您的问题的更好方法是将您的域字段视为每个测试将从 Guice 或其他依赖项注入框架接收的注入资源。

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM