Гарантирует ли TestNG методы @BeforeSuite перед методами @BeforeTest? - PullRequest
5 голосов
/ 04 августа 2010

ОБЩАЯ ИНФОРМАЦИЯ: Моя цель - написать код для системы TestNG-Selenium, которая работает автономно (без строк для maven или ant plugins; только java).Он должен позволять тестовым примерам принимать параметры, включая браузер и URL домена.Когда TestRunner создает экземпляры этих тестовых случаев, браузер и домен используются для получения объекта Selenium для выполнения его тестирования.

ПРОБЛЕМА: только один тестовый класс на комплект успешно получает параметр домена (в методе @BeforeSuite) перед попыткой получить объект Selenium (в @BeforeTest).Тестовые классы, которые не получают домен, имеют нулевой объект селена, т.к. он не может быть создан.

CODE: каждый из классов XmlClasses содержится в своем собственном XmlTest, и все три находятся в одном XmlSuite.Набор содержит в порядке TestClass1, TestClass2, затем TestClass3.Сами тестовые классы являются подклассами двухуровневых абстрактных базовых классов, которые включают в себя функциональность для инициализации введенных переменных и последующего получения экземпляра 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)

Я ценю любую информацию, которую вы можете иметь по этому вопросу.

1 Ответ

9 голосов
/ 05 августа 2010

Думаю, проблема в том, что ваш метод @BeforeSuite присваивает значение полю, но у вас есть три разных экземпляра, поэтому два других никогда не инициализируются.

Помните, что @BeforeSuite запускается только один раз, независимо от того, к какому классу он принадлежит. Таким образом, методы @ Before / AfterSuite обычно определяются в классах, находящихся вне всей тестовой среды. Эти методы действительно должны быть статичными, но я решил не применять это требование, потому что оно иногда нецелесообразно.

Я думаю, что лучший способ подойти к вашей проблеме - это рассматривать поле вашего домена как внедренный ресурс, который каждый ваш тест будет получать от Guice или другой инфраструктуры внедрения зависимостей.

...