shrinkwrap_Java EE 6测试第二部分– Arquillian和ShrinkWrap简介

shrinkwrap

在Java EE 6测试的第一部分中,我简要介绍了使用Glassfish嵌入式容器的EJB 3.1 Embeddable API,以演示如何启动该容器,如何在项目类路径中查找bean以及运行非常简单的集成测试。

这篇文章重点介绍Arquillian和ShrinkWrap以及为什么它们是用于企业Java应用程序集成测试的出色工具。

GitHub上的arquillian-shrinkwrap文件夹下提供了用于此文章的源代码。

工具

Arquillian
Arquillian将测试执行带入目标运行时,从而减轻了开发人员从测试或项目构建中管理运行时的负担。 为了反转此控件,Arquillian将生命周期包装在执行以下操作的测试执行周围:
  • 管理一个或多个容器的生命周期
  • 将测试用例,相关类和资源捆绑为ShrinkWrap档案
  • 将档案部署到容器
  • 通过依赖注入和其他声明式服务丰富测试用例
  • 在容器内部(或针对容器)执行测试
  • 将结果返回给测试跑步者进行报告

收缩包装

ShrinkWrap是Arquillian的核心组件,它提供了一种简单的机制,可以通过友好,流利的API来组合JAR,WAR和EAR等档案。

使用Arquillian的主要好处之一是,您可以在远程容器(即应用程序服务器)中运行测试。 这意味着您将测试真实的交易 。 没有嘲笑。 甚至没有嵌入式运行时!

议程

这篇文章将涵盖以下主题:

  • 在基于Maven的Java项目中配置Arquillian基础架构
  • 直接在测试实例中注入EJB和托管Bean(CDI)
  • 测试Java持久性API(JPA)层
  • 在客户端模式下运行Arquillian
  • 在IDE中运行和调试Arquillian测试

配置Maven以运行集成测试

为了与Maven进行集成测试,我们需要一种不同的方法。 通过不同的方法,我的意思是不同的插件: Maven故障安全插件 。

故障安全插件是Maven Surefire插件的分支,旨在运行集成测试。

Failsafe插件目标旨在在集成测试阶段的程序包阶段之后运行。

Maven生命周期有四个运行集成测试的阶段:

  • 集成前测试:在此阶段,我们可以启动任何所需的服务或执行任何操作,例如启动数据库或启动Web服务器,等等。
  • 集成测试:故障安全将在此阶段运行测试,因此在所有必需的服务启动之后。
  • 集成后测试:关闭所有服务的时间…
  • 验证: failsafe运行另一个在此处解释测试结果的目标,如果没有通过任何测试,failsafe将显示结果并退出构建。

在POM中配置故障保护:

<!-- clip -->
<plugin><groupId>org.apache.maven.plugins</groupId><artifactId>maven-surefire-plugin</artifactId><version>2.12</version><configuration><skipTests>true</skipTests></configuration>
</plugin>
<plugin><groupId>org.apache.maven.plugins</groupId><artifactId>maven-failsafe-plugin</artifactId><version>2.12</version><configuration><encoding>UTF-8</encoding></configuration><executions><execution><id>integration-test</id><goals><goal>integration-test</goal></goals></execution><execution><id>verify</id><goals><goal>verify</goal></goals></execution></executions>
</plugin>
<!-- clip -->

默认情况下,Surefire插件执行**/Test*.java**/*Test.java**/*TestCase.java测试类。 Failsafe插件将查找**/IT*.java**/*IT.java**/*ITCase.java 。 如果您同时使用Surefire和Failsafe插件,请确保使用此命名约定,以使其更容易识别哪个插件正在执行哪些测试。

在Maven中配置Arquillian基础架构

通过附加以下XML片段,将Maven项目描述符配置为使用Arquillian:

<!-- clip -->
<repositories><repository><id>jboss-public-repository-group</id><name>JBoss Public Repository Group</name><url>http://repository.jboss.org/nexus/content/groups/public/</url></repository>
</repositories><dependencyManagement><dependencies><dependency><groupId>org.jboss.arquillian</groupId><artifactId>arquillian-bom</artifactId><version>1.0.1.Final</version><scope>import</scope><type>pom</type></dependency></dependencies>
</dependencyManagement><dependencies><dependency><groupId>org.jboss.arquillian.testng</groupId><artifactId>arquillian-testng-container</artifactId><scope>test</scope></dependency><dependency><groupId>org.testng</groupId><artifactId>testng</artifactId><version>6.4</version><scope>test</scope></dependency><dependency><groupId>org.jboss.spec</groupId><artifactId>jboss-javaee-6.0</artifactId><version>3.0.1.Final</version><scope>provided</scope><type>pom</type></dependency>
</dependencies><profiles><profile><id>jbossas-remote-7</id><activation><activeByDefault>true</activeByDefault></activation><dependencies><dependency><groupId>org.jboss.as</groupId><artifactId>jboss-as-arquillian-container-remote</artifactId><version>7.1.1.Final</version><scope>test</scope></dependency></dependencies></profile>
</profiles>
<!-- clip -->

Arquillian有大量的容器适配器 。 Arquillian测试可以在与测试中使用的编程模型兼容的任何容器中执行。 但是,在本文中,仅使用JBoss AS 7。
与Java EE 6测试第1部分类似,我选择使用TestNG测试框架,但同样, JUnit应该也能正常工作。

创建可测试的组件

在研究如何使用Arquillian编写集成测试之前,我们首先需要有一个要测试的组件。
会话Bean是Java EE堆栈中的常见组件,将用作测试主题。 在本文中,我将创建一个非常基本的后端,用于向数据库中添加新用户。

@Stateless
public class UserServiceBean {@PersistenceContextprivate EntityManager em;public User addUser(User user) {em.persist(user);return user;}// Annotation says that we do not need to open a transaction@TransactionAttribute(TransactionAttributeType.SUPPORTS)public User findUserById(Long id) {return em.find(User.class, id);}
}

在上面的代码中,我使用JPA ,因此我们需要一个持久性单元。
持久性单元定义由应用程序中的EntityManager实例管理的所有实体类的集合。 这组实体类表示单个数据存储中包含的数据。
持久性单元由persistence.xml配置文件定义:

<?xml version="1.0" encoding="UTF-8"?>
<persistence xmlns="http://java.sun.com/xml/ns/persistence"xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"xsi:schemaLocation="http://java.sun.com/xml/ns/persistencehttp://java.sun.com/xml/ns/persistence/persistence_2_0.xsd"version="2.0"><persistence-unit name="example"><jta-data-source>java:jboss/datasources/ExampleDS</jta-data-source><properties><property name="hibernate.hbm2ddl.auto" value="create-drop" /><property name="hibernate.show_sql" value="true" /></properties></persistence-unit>
</persistence>

在此示例中,我使用的示例数据源使用H2数据库,并且已经使用JBoss AS 7进行了配置。

最后,我们还需要一个映射到数据库中表的实体:

@Entity
public class User {@Id@GeneratedValueprivate Long id;@NotNullprivate String name;// Removed constructors, getters and setters for brevity@Overridepublic String toString() {return "User [id=" + id + ", name=" + name + "]";}
}

使用Arquillian测试JPA

现在,我们都准备编写我们的第一个Arquillian测试。
一个Arquillian测试用例看起来就像一个带有一些额外功能的单元测试。 它必须具有三件事:

  • 扩展Arquillian类(这特定于TestNG,对于JUnit,您需要在类上使用@RunWith(Arquillian.class)批注)
  • 用@Deployment注释的公共静态方法,该方法返回ShrinkWrap存档
  • 至少一种使用@Test注释的方法
public class UserServiceBeanIT extends Arquillian {private static final Logger LOGGER = Logger.getLogger(UserServiceBeanIT.class.getName());@Injectprivate UserServiceBean service;@Deploymentpublic static JavaArchive createTestableDeployment() {final JavaArchive jar = ShrinkWrap.create(JavaArchive.class, "example.jar").addClasses(User.class, UserServiceBean.class).addAsManifestResource("META-INF/persistence.xml", "persistence.xml")// Enable CDI.addAsManifestResource(EmptyAsset.INSTANCE, ArchivePaths.create("beans.xml"));LOGGER.info(jar.toString(Formatters.VERBOSE));return jar;}@Testpublic void callServiceToAddNewUserToDB() {final User user = new User("Ike");service.addUser(user);assertNotNull(user.getId(), "User id should not be null!");}
}

该测试很简单,它插入一个新用户并检查id属性是否已被数据库生成的值填充。
由于Arquillian丰富了该测试,因此您通常可以使用@EJB@Inject批注来注入EJB和受管bean。
@Deployment注释的方法使用ShrinkWrap来构建一个JAR归档文件,该归档文件将部署到容器中并对其进行测试。 ShrinkWrap将测试所需的类和资源与类路径的其余部分隔离开来,您应该包括测试所需的每个组件才能在部署存档中运行。

客户端模式

Arquillian支持三种测试运行模式:

  • 容器内模式用于测试您的应用程序内部。 这使Arquillian能够与测试进行通信,丰富测试并远程运行测试。 在这种模式下,测试在远程容器中执行; Arquillian默认使用此模式。
  • 客户端模式用于测试客户端如何使用您的应用程序。 与重新包装并覆盖测试执行的容器内模式相反,客户端模式的作用尽可能小。 它不会重新打包@Deployment也不会将测试执行转发到远程服务器。 您的测试用例正在JVM中按预期运行,并且您可以自由地从外部测试容器,如客户所见。 Arquillian唯一要做的就是控制@Deployment的生命周期。
  • 混合模式允许在同一测试类中混合两种运行模式。

要以客户端模式运行Arquillian,首先要构建要测试的servlet:

@WebServlet("/User")
public class UserServlet extends HttpServlet {private static final long serialVersionUID = -7125652220750352874L;@Injectprivate UserServiceBean service;@Overridepublic void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {response.setContentType("text/plain");PrintWriter out = response.getWriter();out.println(service.addUser(new User("Ike")).toString());out.close();}@Overrideprotected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException,IOException {doGet(request, response);}
}

现在让我们测试一下:

public class UserServletIT extends Arquillian {private static final Logger LOGGER = Logger.getLogger(UserServletIT.class.getName());// Not managed, should be used for external calls (e.g. HTTP)@Deployment(testable = false)public static WebArchive createNotTestableDeployment() {final WebArchive war = ShrinkWrap.create(WebArchive.class, "example.war").addClasses(User.class, UserServiceBean.class, UserServlet.class).addAsResource("META-INF/persistence.xml")// Enable CDI.addAsWebInfResource(EmptyAsset.INSTANCE, ArchivePaths.create("beans.xml"));LOGGER.info(war.toString(Formatters.VERBOSE));return war;}@RunAsClient // Same as @Deployment(testable = false), should only be used in mixed mode@Test(dataProvider = Arquillian.ARQUILLIAN_DATA_PROVIDER)public void callServletToAddNewUserToDB(@ArquillianResource URL baseURL) throws IOException {// Servlet is listening at <context_path>/Userfinal URL url = new URL(baseURL, "User");final User user = new User(1L, "Ike");StringBuilder builder = new StringBuilder();BufferedReader reader = new BufferedReader(new InputStreamReader(url.openStream()));String line;while ((line = reader.readLine()) != null) {builder.append(line);}reader.close();assertEquals(builder.toString(), user.toString());}
}

尽管此测试非常简单,但是它允许您使用单个方法调用来测试应用程序的多个层。

在Eclipse中运行测试

您可以从IDE内部运行Arquillian测试,就像单元测试一样。

运行Arquillian测试

(点击图片以放大)

  • 安装TestNG和JBoss Tools Eclipse插件。
  • 将新的JBoss AS服务器添加到Eclipse:
  • 启动JBoss AS服务器:
  • 从Eclipse运行测试用例,右键单击Project Explorer上的测试文件,然后选择

Run As > TestNG Test

结果应类似于以下内容:

调试Arquillian测试

(点击图片以放大)

由于我们使用的是远程容器,因此Debug As > TestNG Test不会导致断点被激活。
相反,我们需要以调试模式启动容器并附加调试器。 这是因为测试是在与原始测试运行器不同的JVM中运行的。
调试测试所需要做的唯一更改是在调试模式下启动JBoss AS服务器:

  • 启动JBoss AS服务器调试模式:
  • 将所需的断点添加到代码中。
  • 并通过右键单击Project Explorer上的测试文件并选择它来调试它

Run As > TestNG Test

    更多资源

    我希望能够强调Arquillian的一些好处。
    有关Arquillian的更多信息,请查看以下资源:

    • Arquillian指南
    • Arquillian社区
    • Arquillian Git存储库

    相关文章

    • 单元测试JBoss 5服务
    • Java EE 6测试第I部分– EJB 3.1可嵌入API
    • Maven 2 Cobertura插件–更新
    • JBoss PojoCache配置
    • JBoss AS 5.0已经发布!
    • 上一篇文章:Java EE 6测试第I部分– EJB 3.1可嵌入API
    • 下一篇文章:比较OpenDDR与WURFL

    参考: Java EE 6测试第二部分–来自我们JCG合作伙伴 Samuel Santos的Arquillian和ShrinkWrap简介,位于Samaxes博客上。


    翻译自: https://www.javacodegeeks.com/2012/06/java-ee-6-testing-part-ii-introduction.html

    shrinkwrap

    本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/354958.shtml

    如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

    相关文章

    java中三个基本框架_对于Java基础者应该如何理解Java中的三大框架!

    三大框架&#xff1a;StrutsHibernateSpringJava三大框架主要用来做WEN应用。Struts主要负责表示层的显示Spring利用它的IOC和AOP来处理控制业务(负责对数据库的操作)Hibernate主要是数据持久化到数据库再用jsp的servlet做网页开发的时候有个 web.xml的映射文件&#xff0c;里面…

    Apache Camel的性能调整思路

    时不时地&#xff0c;我会以Camel速度较慢的说法来询问有关优化Camel应用程序的问题。 骆驼只是连接不同系统的粘合剂&#xff0c;路由引擎全都在内存中&#xff0c;并且不需要任何持久状态。 因此&#xff0c;在99&#xff05;的情况下&#xff0c;性能问题是由于其他系统的瓶…

    java虚拟机源码怎么看_java虚拟机JVM第4讲:从源代码到机器码,发生了什么?

    在上篇文章我们聊到&#xff0c;无论什么语言写的代码&#xff0c;其到最后都是通过机器码运行的&#xff0c;无一例外。那么对于 Java 语言来说&#xff0c;其从源代码到机器码&#xff0c;这中间到底发生了什么呢&#xff1f;这就是今天我们要聊的。如下图所示&#xff0c;编…

    java构建内存池队列_池化技术(线程池、连接池、内存池等)

    一、池化技术 -简单点来说&#xff0c;就是提前保存大量的资源&#xff0c;以备不时之需。对于线程&#xff0c;内存&#xff0c;oracle的连接对象等等&#xff0c;这些都是资源&#xff0c;程序中当你创建一个线程或者在堆上申请一块内存时&#xff0c;都涉及到很多系统调用&a…

    java 堆大小_适当的Java堆大小的5个技巧

    java 堆大小确定生产系统合适的Java堆大小不是一件容易的事。 在我的Java EE企业经验中&#xff0c;由于Java堆容量和调整不足&#xff0c;我遇到了多个性能问题案例。 本文将为您提供5个技巧&#xff0c;这些技巧可以帮助您确定当前或新生产环境的最佳Java堆大小。 这些技巧中…

    jcmd:一个可以全部统治的JDK命令行工具

    我在过去的几篇文章中都引用了方便的JDK工具jcmd &#xff0c;但是像我以前对jps所做的那样&#xff0c;仅专注于其实用性 。 jcmd工具是随Oracle Java 7引入的&#xff0c;在通过使用Java标识Java进程的ID &#xff08;与jps相似&#xff09;&#xff0c;获取堆转储 &#xff…

    ansible-playbook实操之一键搭建lnmp+wordpress

    目录 1、架构和准备&#xff1a; 2、配置nginx角色&#xff1a; 3、配置mariadb角色&#xff1a; 4、配置php角色&#xff1a; 5、配置完之后&#xff0c;写脚本调用roles 6、配置完之后浏览器搭建wordpress&#xff1a; 1、架构和准备&#xff1a; 操控节点&#xff1a;…

    pivot 与 unpivot 函数是SQL05新提供的2个函数

    pivot 与 unpivot 函数是SQL05新提供的2个函数 ------------------------------------------------------------------------------ pivot函数&#xff1a; create table test(id int,name varchar(20),quarter int,profile int)insert into test values(1,a,1,1000)insert i…

    python SimpleHTTPServer 快速共享文件

    简单介绍 通过一个python命令快速共享文件给他人。 操作步骤 1、打开cmd命令行&#xff0c;切换到需要共享文件的目录&#xff0c;执行命令 python -m SimpleHTTPServer 。 2、打开浏览器&#xff0c;在地址栏中输入http://10.10.11.164:8000或者http://localhost:8000/可以看到…

    mysql数据库设计实现工作流_工作流activiti部署到数据库(1)

    1.工作流定义(workflow):指"业务过程的部分或整体在计算机应用环境下的自动化".普通框架要有一个请假单,要有一个字段来标识请假单的状态,至少有三个,还有请假单的状态是走到那个经理审批还是老板审批,这个时候不便管理.BPM:业务流程管理框架,是用来管理流程的框架.B…

    abd.exe 需要下java吗_Abd.exe文件下载|

    abd.exe在哪个位置&#xff1f;abd.exe是一款很重要的电脑文件&#xff0c;如果这个文件丢失了&#xff0c;电脑部分程序将会无法正常运行&#xff0c;所以大家应该重视这类文件&#xff0c;小编已经将这个文件打包好了&#xff0c;欢迎大家来当易网下载。exe文件简介EXE File …

    Spark面对OOM问题的解决方法及优化总结 (转载)

    Spark面对OOM问题的解决方法及优化总结 (转载) 转载地址&#xff1a; http://blog.csdn.net/yhb315279058/article/details/51035631 Spark中的OOM问题不外乎以下两种情况map执行中内存溢出shuffle后内存溢出map执行中内存溢出代表了所有map类型的操作&#xff0c;包括&#xf…

    通过此注释改善您的JUnit体验

    JUnit可能是所有Java项目中90&#xff05;的一部分。 令人兴奋的是&#xff0c;我们很快将拥有支持Java 8的JUnit 5 。 我们最近在博客上发表了一项改进 。 回到JUnit 4领域&#xff0c;有一个小技巧&#xff0c;我只能建议您进行所有单元测试。 只需在此处添加这个小注释&…

    jdeveloper_JDeveloper中的Java反编译器

    jdeveloperJava Decompiler是一个独立的图形实用程序&#xff0c;显示“ .class”文件的Java源代码。 下面是Java Decompiler程序的快照 您可以从这里下载该程序 我将说明如何在Jdeveloper中将此程序用作外部工具 Java Decompiler和Jdeveloper之间的集成 您可以将此程序添加…

    文件上传 java 完美,vue+java实现文件上传(excel等),会出现跨域问题,直接用form表单提交就不会有问题了(new FormData())...

    vuejava实现文件上传(excel等)&#xff0c;会出现跨域问题&#xff0c;直接用form表单提交就不会有问题了(new FormData())地址&#xff1a;https://www.cnblogs.com/muscles/p/9503103.html一&#xff1a;首先说一下什么是跨域&#xff0c;跨域就是解决浏览器同源策略的问题。…

    php private方法,php如何调用private方法

    php调用private方法&#xff1a;首先定义一个parent类&#xff1b;然后在类的内部使用私有函数&#xff1b;接着实例化parent类&#xff0c;让其变成一个对象并赋值给“$obj”即可。将一个类实例化后就变成对象&#xff0c;私有函数只能在类内部使用&#xff0c;不能在类外&…

    java flux api,SpringBoot学习系列-WebFlux REST API 全局异常处理

    本文内容为什么要全局异常处理&#xff1f;WebFlux REST 全局异常处理实战小结摘录&#xff1a;只有不断培养好习惯&#xff0c;同时不断打破坏习惯&#xff0c;我们的行为举止才能够自始至终都是正确的。一、为什么要全局异常处理&#xff1f;前后端分离开发&#xff0c;一般提…

    mvvm 后端_ZK实际应用:MVVM –与ZK客户端API一起使用

    mvvm 后端在以前的文章中&#xff0c;我们已经使用ZK的MVVM实现了以下功能&#xff1a; 将数据加载到表中 使用表单绑定保存数据 删除条目并以编程方式更新视图 ZK MVVM和ZK MVC实现方式之间的主要区别是&#xff0c;我们不直接在controller&#xff08;ViewModel&#xff0…

    微信分享朋友圈固定缩略图 php,微信转发或分享朋友圈带缩略图、标题和描述的实现方法...

    自己做博客以来&#xff0c;很早之前分享过文章至朋友圈&#xff0c;那个时候分享过去的文章自动获取页面的比例适合的图片为所缩略图&#xff1a;后期就很少分享至朋友圈&#xff0c; 近来分享文章给朋友后&#xff0c;发现不带缩略图和简介了&#xff0c;觉得这样很不好看&am…

    java监控rabbitMq服务状态,SpringCloud-Turbine【RabbitMQ服务监控】

    前面我们介绍了通过turbine直接聚合多个服务的监控信息&#xff0c;实现了服务的监控&#xff0c;但是这种方式有个不太好的地方就是turbine和服务的耦合性太强了&#xff0c;针对这个问题&#xff0c;我们可以将服务的监控消息发送到RabbitMQ中&#xff0c;然后turbine中Rabbi…