教坏小朋友:Maven 的一些姿势

最近遇到了一个 Maven 的小问题,着实让有2年多的 Java 实战开发经验的笔者留了一身汗。自以为对 Maven 的常用东西还是比较熟悉的,实际还是停留在 “知其然而不知其所以然” 的阶段。习惯了 “临摹官方示例+遇到问题就搜索” 的快餐式手法,自然就渐渐失去刨根问底的心,是吧。

所以本篇旨在记录所遇到问题的解决过程,以及从 “技术小白” (我还是很谦虚的好不 🐶) 的视角来看看 Maven 工具的一些“小技巧”。

问题:咦,集成测试挑机器,死活跑不起来?

为了测试 redis 实现的 repository,使用了 embedded-redis,在测试前可以开启一个嵌入式的 redis server,跑完测试只需要 server.stop() 就好了。这一套在我的 Macbook Pro 上跑得好好的,换到另一个同事的 Windows 上就死活报错说 server 起不来,但其实 embedded-redis 是支持 windows的。

本着不想折腾 embedded-redis 配置的原则,咱来个曲线救国吧,让一部分测试成为可选项,只按需的跑一跑(不在开发本地跑,只在 Jenkins 上跑)。自然的就想到了将一部分测试作为 Integration Test,通过配置来控制它们的运行。

然而,这次运气就没那么好了,除非在 maven surefire plugin 中配置 Skip Test 或者使用 mvn install -DskipTests,似乎没有办法在 mvn install 的时候不去运行集成测试。

1. 无痛起步:mvn clean install ?

从我两年前第一天上班起,这个三个英文字母就跟打了烙印一样深刻。拿到项目代码,啥事儿不管,咱先实现个小目标,比如一次性跑通 mvn clean install。所以说 “入门教程” 他害人不浅呐,直到几天前,我都还以为本地编译打包用这 “一招鲜” 就可以了。

首先我们来看看 Maven 的 Lifecycle 吧,完整的 Lifecycle 还包含很多中间步骤。

  • validate:验证一些东西,我也不知道是啥 🐱
  • compile:编译
  • test:单元测试
  • package:打包成 jar 或者其他形式
  • verify:集成测试
  • install:安装到 maven repository
  • deploy:部署上传到仓库

看到这里,读者可能意识到了什么吧,其实有个最直截了当的做法,丫的你只运行 mvn package 不就好了🐶。可是……俺就想跑个 mvn install 怎么办!(参考后面的“打包所有的 jar”,因为某插件是在 install 的阶段才执行)

2. Test 原来可以优雅的分为两类

补充一点,前面提到了单元测试 (UnitTest) 和集成测试 (IntegrationTest),如何让一部分测试成为集成测试呢。有一种简单直接的办法就是手动配置标记那些测试文件(文件夹)只在集成测试的时候才运行。

其实工具已经提供了一个优雅的办法来区分哪些是单元测试,哪些是集成测试。在后文中要用的集成测试插件maven-surefire-plugin、maven-failsafe-plugin 都有个 include的约定:
Unit Test 文件命名约定:凡是以 Test 作为文件名的开头或结尾的那些测试自动归类为单元测试。
Integration Test 文件命名约定:凡是以 IT 作为文件名的开头或结尾的那些测试自动归类为集成测试。

如此,在 “约定大于配置” 之下,很多事儿就自然优雅了。

3. 解法:Profile

下面来说说解法,这里笔者选择了 Maven 的 profile 来控制是否加入集成测试插件,在 all 的 profile 配置下,failsafe插件才生效,mvn install 才会去运行那些集成测试。

<profiles>
    <profile>
        <id>default</id>
        <activation>
            <activeByDefault>true</activeByDefault>
        </activation>
    </profile>

    <profile>
        <id>all</id>
        <build>
            <plugins>
                <plugin>
                    <groupId>org.apache.maven.plugins</groupId>
                    <artifactId>maven-failsafe-plugin</artifactId>
                    <version>2.19.1</version>
                    <executions>
                        <execution>
                            <goals>
                                <goal>integration-test</goal>
                                <goal>verify</goal>
                            </goals>
                        </execution>
                    </executions>
                </plugin>
            </plugins>
        </build>
    </profile>
</profiles>

大部分时候:我只想默默地完成 Build

mvn clean install

因为前面配置了 activeByDefault 的空 Profile,所以无需额外的说明,它只会跑所有的单元测试。

集成的时候:我也想来个大宝剑

mvn clean install -P all

在 Jenkins 配置上这个命令就可以跑完所有的测试流程了。

Maven 的一些常用配置补完

1. 打包所有的 jar

在我们的项目中,虽然使用了 spring-boot,但并没有使用插件来生产 fat-jar。 而是把所有的 jar 文件(包括众多的 dependencies)打包成一个文件归档,用于部署。下面就讲讲如何在 Maven 里来完成这个步骤。

使用插件打包所有的jar

这里使用了 “maven-dependency-plugin” 来复制所有的依赖 jar 到 target/lib 目录下, 然后 “maven-assembly-plugin” 插件会将可执行 jar 以及所有的依赖库 jar 按照 zip-package.xml 的配置去执行打包。如果你和我一样配置了 “finalName” 那么最终的输出文件名就会是例子中的 demo-service.tar.gz,否则它会默认根据你的 {artifactId}-{version} 来生成。

<plugin>
    <groupId>org.apache.maven.plugins</groupId>
    <artifactId>maven-assembly-plugin</artifactId>
    <version>2.6</version>
    <executions>
        <execution>
            <id>create-distribution</id>
            <phase>install</phase>
            <goals>
                <goal>single</goal>
            </goals>
            <configuration>
                <finalName>demo-service</finalName>
                <descriptors>
                    <descriptor>zip-package.xml</descriptor>
                </descriptors>
            </configuration>
        </execution>
    </executions>
</plugin>

<plugin>
    <artifactId>maven-dependency-plugin</artifactId>
    <executions>
        <execution>
            <phase>package</phase>
            <goals>
                <goal>copy-dependencies</goal>
            </goals>
            <configuration>
                <outputDirectory>${project.build.directory}/lib</outputDirectory>
                <includeScope>runtime</includeScope>
            </configuration>
        </execution>
    </executions>
</plugin>

assembly 的配置

<?xml version="1.0" encoding="UTF-8"?>
<assembly
  xmlns="http://maven.apache.org/plugins/maven-assembly-plugin/assembly/1.1.2"
  xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
  xsi:schemaLocation="
    http://maven.apache.org/plugins/maven-assembly-plugin/assembly/1.1.2
      http://maven.apache.org/xsd/assembly-1.1.2.xsd">
  <id>dist</id>
  <formats>
    <format>tar.gz</format>
  </formats>
  <fileSets>
    <fileSet>
      <directory>${project.build.directory}</directory>
      <outputDirectory>build</outputDirectory>
      <includes>
        <include>*.jar</include>
      </includes>
      <excludes>
        <exclude>*-sources.jar</exclude>
      </excludes>
    </fileSet>
    <fileSet>
      <directory>${project.build.directory}/lib</directory>
      <outputDirectory>lib</outputDirectory>
      <includes>
        <include>*.jar</include>
      </includes>
    </fileSet>
  </fileSets>
</assembly>

上面贴的是我使用的 zip-package.xml ,打包的输出 format 使用 gzip 压缩,在 fileSet 里设置了 include/exclude 的文件,outputDirectory 可以用于设置文件的输出路径,上面例子中的配置生成的结构就是:

build/
    |-- service.jar
lib/
    |-- tomcat.jar
    |-- tool.jar
    |-- ...

2. 手动上传依赖的 jar 到 私有的 Nexus 里

某些时候,你项目里依赖的 jar lib 可能并没有在 public repository 里面,比如外包厂商生成的 jar。有一种做法是直接添加 sytem scope 的依赖:

<dependency>
    <groupId>com.demo.www</groupId>
    <artifactId>demo</artifactId>
    <version>0.0.1</version>
    <scope>system</scope>
    <systemPath>${project.basedir}/lib/demo-0.0.1.jar</systemPath>
</dependency>

这么做不是不可以,但会看着很奇怪,依赖管理起来也很麻烦,某些时候还会遇到 bug (maven-dependency-plugin 里配置的 runtime includeScope 并不会把这个 system scope 的依赖放进去)

另一种做法就是把这个 jar 上传到你的 私有 Nexus 里面就行管理,用的时候保持平常的 dependency 规则就好了:

<dependency>
    <groupId>com.demo.www</groupId>
    <artifactId>demo</artifactId>
    <version>0.0.1</version>
</dependency>

下面就给出笔者所使用的 bash 上传脚本,读者请按照你的实际情况修改即可,从此 pom 里无需system scope的依赖了。

mvn deploy:deploy-file \
-DgroupId=com.demo.www \
-DartifactId=demo \
-Dversion=0.0.1 \
-DgeneratePom=true \
-Dpackaging=jar \
-DrepositoryId=myrepository \
-Durl=http://myrepository.com/content/repositories/releases \
-Dfile=demo-sdk-java-0.0.1.jar

3. 其他的一些工具

  • error-prone:google 出品必属精品,帮你做代码的静态检查,老司机(毕竟使用神器 Intellij)应该不怎么有机会触发它的警报。
  • distributionManager: 当你使用私有的 Nexus 的时候需要配置的。
  • aspectj-maven-plugin: 用于在编译期 weave 你的 AspectJ AOP,例子可以参考之前的一篇文章《Java AOP 实例踩坑记》。

最后:请手下留情

别问我 “你为啥不用 Gradle 呀”,我怎么好意思机智的回答说 “那是因为公司的技术太老旧了呀,只会用 Maven” 😅。就算我技多不压身,但精力有限,术业有专攻,我知道 Gradle 可以很方便的做很多 DevOps 的事情,可我只想做个安静的 Coder,也没必要折腾这么多种构建工具吧,以后用到了学起来就好了。

参考

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,125评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,293评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 162,054评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,077评论 1 291
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,096评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,062评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,988评论 3 417
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,817评论 0 273
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,266评论 1 310
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,486评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,646评论 1 347
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,375评论 5 342
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,974评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,621评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,796评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,642评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,538评论 2 352

推荐阅读更多精彩内容