这是一个Maven提高篇的系列,包含有以下文章:

  1. Maven提高篇系列之(一)——多模块 vs 继承
  2. Maven提高篇系列之(二)——配置Plugin到某个Phase(以Selenium集成测试为例)
  3. Maven提高篇系列之(三)——使用自己的Repository(Nexus)
  4. Maven提高篇系列之(四)——使用Profile
  5. Maven提高篇系列之(五)——处理依赖冲突
  6. Maven提高篇系列之(六)——编写自己的Plugin(本系列完)

持续交付要“自动化所有东西”,对于集成测试也是一样。集成测试和单元测试相比需要更多的环境准备工作,包括测试数据的准备和启动服务器等。在本篇中我们设想以下一种场景:

  • 你开发了一个web应用,集成测试使用了Selenium,你希望通过一个Maven命令跑完所有的测试,包括集成测试。

Maven的plugin包含了一个或多个goal,每一个goal表示plugin的一个操作单位,在plugin开发中,一个goal通常对应于Java类的一个方法(Mojo的execute方法,请参考本系列之六)。一个goal可以默认绑定到Mavan的某个phase,比如Jar plugin的jar这个goal便默认绑定在了package这个phase上。当Maven执行到package时,Jar的jar goal将自动执行。当然,在默认情况下plugin的goal也可以不绑定在任何一个phase上,此时Maven将不做任何操作。但是,我们可以显式地手动将某个plugin的某个goal绑定在一个phase中。

对于上面的场景,我们的解决方案是:在集成测试之前(对应Maven的phase为pre-integration-test),我们使用jetty-maven-plugin启动web应用,在集成测试时通过Selenium访问网站进行验证,集成测试完毕之后(对应Maven的phase为post-integration-test),同样使用jetty-maven-plugin关闭web应用。

Github示例代码:https://github.com/davenkin/maven-plugin-binding.git

通过上一篇的方式创建一个web工程,向其中添加一个helloworld.html用于测试:

<html>

<body>

<h2>Hello World!</h2>

</body>

</html>

添加Selelium Webdriver集成测试如下:

public class SafariIntegrationTest {

   @Test

   public void testHelloWorldIndexPage() {

       WebDriver driver = new HtmlUnitDriver();

       driver.get("http://localhost:8080/helloworld.html");

       WebElement element = driver.findElement(By.tagName("h2"));

       assertThat(element.getText(), is("Hello World!"));

   }

}

为了使用Selenium,我们需要将Selenium依赖加入pom.xml文件中:

       <dependency>

           <groupId>org.seleniumhq.selenium</groupId>

           <artifactId>selenium-java</artifactId>

           <version>2.33.0</version>

       </dependency>

接下来就可以配置jetty-maven-plugin了:

           <plugin>

               <groupId>org.mortbay.jetty</groupId>

               <artifactId>jetty-maven-plugin</artifactId>

               <version>8.1.12.v20130726</version>

               <configuration>

                   <scanintervalseconds>0</scanintervalseconds>

                   <stopKey>stop</stopKey>

                   <stopPort>9999</stopPort>

               </configuration>

               <executions>

                   <execution>

                       <id>start-jetty</id>

                       <phase>pre-integration-test</phase>

                       <goals>

                           <goal>run</goal>

                       </goals>

                       <configuration>

                           <scanintervalseconds>0</scanintervalseconds>

                           <daemon>true</daemon>

                       </configuration>

                   </execution>

                   <execution>

                       <id>stop-jetty</id>

                       <phase>post-integration-test</phase>

                       <goals>

                           <goal>stop</goal>

                       </goals>

                   </execution>

               </executions>

           </plugin>

可以看出,在pre-integration-test阶段,我们调用了jetty-maven-plugin的run,此时web服务器启动,在post-integration-test阶段,我们调用了jetty-maven-plugin的stop来关闭web服务器。

但是这里有个问题,在运行mvn clean install时,Maven会先运行单元测试,再运行集成测试,并且在默认情况下这两种测试都会运行以*Test.java结尾的测试类,结果在单元测试阶段也会运行上面的SafariIntegrationTest,结果还没有执行到集成测试阶段就挂了。

此时,我们需要将单元测试和集成测试分开,Maven使用maven-surefire-plugin执行测试,我们可以先将SafariIntegrationTest排除在测试之外,这样单元测试将不会运行该测试,然后在集成测试中,在将SafariIntegrationTest包含进来,此时我们需要修改maven-surefire-plugin的配置:

           <plugin>

               <groupId>org.apache.maven.plugins</groupId>

               <artifactId>maven-surefire-plugin</artifactId>

               <configuration>

                   <includes>

                       <include>**/unit/*Test.java</include>

                   </includes>

               </configuration>

               <executions>

                   <execution>

                       <id>surefire-it</id>

                       <phase>integration-test</phase>

                       <goals>

                           <goal>test</goal>

                       </goals>

                       <configuration>

                           <includes>

                               <include>**/*IntegrationTest.java</include>

                           </includes>

                       </configuration>

                   </execution>

               </executions>

           </plugin>

现在运行mvn clean install,成功。在上面的例子中,我们使用了一个小trick,先将只有在unit目录下的以*Test.java结尾的类看做测试类,此时不包含SafariIntegrationTest,而在itegration-test阶段,我们将运行以*IntegrationTest.java结尾的测试类。

你可能还是不怎么放心,除非自己看到了实际的页面为止。要达到这样的目的,我们可以将SafariIntegrationTest中的:

WebDriver driver = new HtmlUnitDriver();//使用HtmlUnit

修改为:

 WebDriver driver = new SafariDriver();//打开Safari浏览器

或者:

WebDriver driver = new InternetExplorerDriver();//打开IE浏览器

此时再运行mvn clean install,浏览器窗口将打开。

下一篇中,我们将讲到如何使用自己的repository——Nexus

Maven提高篇系列之(二)——配置Plugin到某个Phase(以Selenium集成测试为例)的更多相关文章

  1. Maven提高篇系列之(六)——编写自己的Plugin(本系列完)

    这是一个Maven提高篇的系列,包含有以下文章: Maven提高篇系列之(一)——多模块 vs 继承 Maven提高篇系列之(二)——配置Plugin到某个Phase(以Selenium集成测试为例) ...

  2. Maven提高篇系列之(五)——处理依赖冲突

    这是一个Maven提高篇的系列,包含有以下文章: Maven提高篇系列之(一)——多模块 vs 继承 Maven提高篇系列之(二)——配置Plugin到某个Phase(以Selenium集成测试为例) ...

  3. Maven提高篇系列之(四)——使用Profile

    这是一个Maven提高篇的系列,包含有以下文章: Maven提高篇系列之(一)——多模块 vs 继承 Maven提高篇系列之(二)——配置Plugin到某个Phase(以Selenium集成测试为例) ...

  4. Maven提高篇系列之(三)——使用自己的Repository(Nexus)

    这是一个Maven提高篇的系列,包含有以下文章: Maven提高篇系列之(一)——多模块 vs 继承 Maven提高篇系列之(二)——配置Plugin到某个Phase(以Selenium集成测试为例) ...

  5. Maven提高篇系列之(一)——多模块 vs 继承

    这是一个Maven提高篇的系列,包含有以下文章: Maven提高篇系列之(一)——多模块 vs 继承 Maven提高篇系列之(二)——配置Plugin到某个Phase(以Selenium集成测试为例) ...

  6. Maven提高篇系列之五——处理依赖冲突

    个人分类: Maven 不知道你在使用Maven时是否遇到过诸如"NoSuchMethodError"或"ClassNotFoundException"之类的问 ...

  7. chessy 提高篇系列 阅读笔记

    java提高篇(一)—–理解java的三大特性之封装 封装的好处, 汇聚属性和方法 减少修改对 其他处的影响 控制get和set方法. java提高篇(二)—–理解java的三大特性之继承 继承的好处 ...

  8. java提高篇(十七)-----异常(二)

          承接上篇博文:java提高篇-----异常(一) 五.自定义异常 Java确实给我们提供了非常多的异常,但是异常体系是不可能预见所有的希望加以报告的错误,所以Java允许我们自定义异常来表 ...

  9. Java提高篇(三二)-----List总结

    前面LZ已经充分介绍了有关于List接口的大部分知识,如ArrayList.LinkedList.Vector.Stack,通过这几个知识点能够对List接口有了比較深的了解了.仅仅有通过归纳总结的知 ...

随机推荐

  1. C#函数式编程之序列

    过了许久的时间,终于趁闲暇的时间来继续将函数式编程这个专辑连载下去,这段时间开头是为IOS这个新方向做准备,将OC的教程写成了SWIFT版,当然我个人是支持Xamarin,但是我一般会先掌握原生态的开 ...

  2. asp.net identity 2.2.0 中角色启用和基本使用(四)

    创建角色相关视图 第一步:添加视图   打开RolesAdminController.cs   将鼠标移动到public ActionResult Index()上  右键>添加视图   系统会 ...

  3. NoSQL:从关系型数据库到非关系型数据库

    关系型数据库 所谓关系型数据库,,就是指采用了关系模型来组织数据的数据库. 什么是关系模型,简单说,关系模型就是二维表格模型,而一个关系型数据库就是由二维表及其之间的联系所组成的一个数据组织. 关系模 ...

  4. 【月末轻松篇】--- 那些奇葩的Bugs

    不能说所有的bug都是纸老虎,但往往那种看似很奇葩的bug,导致的原因确实很简单,烦了你一段时间,找到真相又让你忍不住一笑.什么是奇葩的bug呢.我的定义是:代码逻辑都一样,但在A处是好的,到了B处就 ...

  5. [HIMCM暑期班]第3课:一个博弈问题

    在一个街道平面图上,住着n个住户.有两个贩卖热狗的商贩,各自想要在街区里摆设一个小摊.每天住户都会去离他家50米范围内的最近的摊点消费.问: 1. 如果两位小贩摆设小摊的顺序有先后(设A先摆,然后B再 ...

  6. AI-随机迷宫&迷宫求解

    本文记录了,人工智能中简单的搜索策略中的路径搜索策略中的A*算法,来实现迷宫寻路的问题.(这只是一次本人的课外作业) 完整的程序源码已经发送到我的Git.这里只记录了我的思路和感想以及收获. 产生随机 ...

  7. 关于python的最大递归层数详解

    在阅读http://www.cnblogs.com/skabyy/p/3451780.html这篇文章的时候,实验yield的流式迭代素数的时候发现有个问题,故详细记录下来. 首先来看看python默 ...

  8. OutputCache属性详解(一)一Duration、VaryByParam

    目录 OutputCache概念学习 OutputCache属性详解(一) OutputCache属性详解(二) OutputCache属性详解(三) OutputCache属性详解(四)— SqlD ...

  9. 【java并发】传统线程技术中创建线程的两种方式

    传统的线程技术中有两种创建线程的方式:一是继承Thread类,并重写run()方法:二是实现Runnable接口,覆盖接口中的run()方法,并把Runnable接口的实现扔给Thread.这两种方式 ...

  10. css_04之显示、定位

    1.显示方式:display:取值:none(隐藏,不占页面空间,脱离文档流)/block(元素变为块级)/inline(元素变为行内)/inline-block(元素变为行内块): 2.显示效果:v ...