0

我正在尝试围绕 Arquillian 进行思考,甚至​​可能开始在我的项目中使用它。我有一个简单的 Java Web 应用程序,它作为 WAR 部署到 Tomcat。

在我的项目中,我定义了一个ServletContextListenerimpl 以便在 Tomcat 启动和停止应用程序时执行代码。

我正在尝试编写一个使用 ShrinkWrap 的超级简单的Arquillian 测试类,并且:

  1. 确认我捆绑的 WAR 可以部署到 Tomcat 并启动而不会抛出异常;和
  2. 一旦应用程序运行(ServletContextListener检查),就可以访问一个简单的系统属性;和
  3. 确认当Tomcat关闭时,没有抛出异常(clean shutdown)

另外,我的实现类ServletContextListener被称为AppLifecycleManager

public class AppLifeCycleManager implements ServletContextListener {
    private String logLevel;

    // Injected by Guice, but that's not really relevant for this question.
    @Inject
    private Logger logger;

    // Getter and setter for logLevel and logger

    @Override
    public void contextInitialized(ServletContextEvent event) {
        logLevel = System.getProperty("log.level");
    }

    @Override
    public void contextDestroyed(ServletContextEvent event) {
        logger.info("Peacefully shutting down the application.");
    }
}

到目前为止,这是我最好的尝试:

@RunWith(Arquillian.class)
public class MyFirstRealIntegrationTest {
    @Deployment
    public static Archive<?> createDeployment() {
        // Haven't figured this part out yet, but for the sake of
        // this question lets pretend this returns a properly-packaged
        // WAR of my web app, the same that my Ant build currently produces.
    }

    @Test
    public void shouldBeAbleToStartTomcatWithoutExceptions() {
        // Given
        Archive war = createDeployment();

        // When - deploy war to Tomcat container
        try {
            // ??? how to access/init a Tomcat container?
            TomcatContainer tomcat = new TomcatContainer(); // this is wrong
            tomcat.start();
        } catch(Throwable throwable) {
            // Starting the container should not throw exceptions
            Assert.fail();
        }
    }

    @Test
    public void shouldBeAbleToStopTomcatWithoutExceptions {
        // Same setup as above test but stops tomcat and checks for
        // thrown exceptions. Omitted for brevity.
    }

    @Test
    public void shouldHaveAccessToSysPropsOnceRunning() {
        // Here, deploy to the container and start it.
        // Then, confirm that AppLifecycleManager correctly read
        // the log.level system property.

        // Given
        Archive war = createDeployment();
        TomcatContainer tomcat = new TomcatContainer();

        // When - AppLifeycleManager should now read the system property
        tomcat.start();

        // Then - make sure log.level was set to "DEBUG" and that it was
        // correctly read by AppLifeCycleManager.
        Assert.assertTrue(war.getClass(AppLifeCycleManager.class)
                .getLogLevel().equals("DEBUG"));
    }
}

因此,鉴于我在这里的方法,我立即遇到了几个问题:

  1. 我不确定如何访问/实例化我的 Tomcat 容器,以便它甚至可以启动/停止
  2. 我不确定如何从我正在运行/部署的 Web 应用程序中实际执行测试。在上面的第三个测试中,我曾经war.getClass(AppLifeCycleManager.class).getLogLevel()尝试访问“实时”类实例并检查其logLevel属性的运行时值,但我知道这是错误的。

所以我问:一个身经百战的 Arquillian 退伍军人将如何编写这 3 个简单的测试,以及我实际上如何从 JUnit 测试中对我的“正在运行”的 Web 应用程序执行测试?提前致谢。

4

2 回答 2

2

我认为您不应该在测试中处理 tomcat 的启动/关闭。如果您使用嵌入式 tomcat 容器,我会容易得多:https ://docs.jboss.org/author/display/ARQ/Tomcat+7.0+-+Embedded 。在这种情况下,arquillian 将处理 tomcat 的启动和关闭。

您应该在使用@Deployment 注释的方法中创建部署。只需阅读以下指南:http ://arquillian.org/guides/getting_started/

于 2013-02-05T07:23:46.250 回答
1

您快到了。createDeployment为您管理嵌入式容器的生命周期(自动启动和停止虚拟容器)。这样,您只需专注于测试本身。要编写集成测试,没有“框架”或“Arquillian API”可供编码。您只需按照主代码的方式调用类和方法。

这里的关键是:您的测试实际上是在容器内运行的。如果发生异常,或者您的断言失败,Arquillian 运行程序会抛出异常并停止容器。对于您的代码示例,您正在测试以查看是否可以读取系统属性:

@RunsWith(Arquillian.class)
public class MyArquillianTest {
    @Deployment
    public Archive<?> createDeployment() { ... }

    @Test
    public void shouldBeAbleToReadSysPropAtStartup() {
        Assert.assertTrue(System.getProperty("log.level") != null);
    }
}

请记住,Assert.assertTrue(System.getProperty("log.level") != null)Arquillian 正在将您的代码“传输”到您为其配置的容器中。所以这个断言实际上是在你部署的容器中运行的。

于 2013-02-11T19:13:05.563 回答