在上一节中我们分享了怎么开发和部署自己开发的jar包到maven私有库中,今天给大家介绍如何开发一个maven插件,我们在使用maven时,用的最多的就是maven的插件,而且这也是maven最核心的东西,maven的插件功能能帮助我们在构建、编译、发布项目的时候完成很多事情。有时我们在发布项目时可能会有一些我们定制化的处理,现有的插件都无法满足我们的需求,这时你可以自己开发一个插件来使用,下面我们用一个实际需求的案例来介绍如何开发自己的maven插件。
记得曾经给一个客户开发了一套系统,采用Java+groovy的模式,大部分核心的逻辑在groovy中来实现,而且为了实现能差量更新,groovy部分将作为升级包,而且不进行编译,直接采用文件加载模式来执行,但是groovy又不能用原文的形式提供给客户,需要进行加密处理,因此为了项目打包方便,便开发了一个自己maven插件,在每次打包后对groovy部分的脚本内容进行加密处理后再统一制作成升级包。当然也可以手工来处理,但是利用maven插件会方便很多,下面给大家具体讲解每一个过程,相关的资源也已上传。
开发maven插件其实和我们开发一个普通jar包过程是一样的,首先我们需要准备一个项目,
然后在pom.xml下添加如下两个依赖:
<dependency>
<groupId>org.apache.mavengroupId>
<artifactId>maven-plugin-apiartifactId>
<version>3.5.0version>
dependency>
<dependency>
<groupId>org.apache.maven.plugin-toolsgroupId>
<artifactId>maven-plugin-annotationsartifactId>
<version>3.5version>
<scope>providedscope>
dependency>
其次我们在项目中添加一个类,需要继承org.apache.maven.plugin.AbstractMojo,代码如下:
@Mojo(name="encoder",defaultPhase= LifecyclePhase.GENERATE_SOURCES)
public class EncodeMojo extends AbstractMojo {
@Parameter
private String fromPath;
@Parameter
private String toPath;
@Parameter
private String secret;
@Parameter
private String iv;
public EncodeMojo(String fromPath, String toPath,String secret, String iv) {
this.fromPath = fromPath;
this.toPath = toPath;
this.secret = secret;
this.iv = iv;
}
public EncodeMojo() {
}
@Override
public void execute() throws MojoExecutionException, MojoFailureException {
}
}
@Mojo注解
标识执行的动作,它对应的就是插件中的executions标签,我们可以定义多个Mojo,在使用时可以来执行不同的动作,比如上面我们定义的这个Mojo,在使用时的配置如下:
<executions>
<execution>
<phase>generate-resourcesphase>
<goals>
<goal>encodergoal>
goals>
execution>
executions>
这里的phase和goal就对应上面的Mojo的定义
@Parameter注解
这里是插件的参数,这里需要注意的是,参数需要通过构造函数来初始化,它在实际的使用时对应的是configuration标签,比如上面的定义的配置如下:
<configuration>
<fromPath>${project.basedir}/scriptfromPath>
<toPath>${project.basedir}/target/scripttoPath>
<secret>720cb14b3151490dad2afaf61d2919fbsecret>
<iv>af3c77c3c6b747e6ad1006e47a1b8422iv>
configuration>
execute方法
插件要具体做的事情就在这个方法中来实现,上面我们的需求就是把fromPath里面的groovy脚本进行加密处理,然后再输出到toPath里面,secret和iv是加密的参数,具体的实现过程大家可以到顶部的资源处下载。
参照上一节的jar包发布,其实过程都是一样的,如果不清楚可以点击:
在Linux下搭建自己的私有maven库并部署和发布自定义jar依赖和自定义maven插件(二)发布自己开发的jar包
查看,最重要的就是定义pom.xml里面的distributionManagement节点,和本地maven的settings.xml配置。具体不再详解,运行发布命令:
mvn clean -DskipTests deploy
执行完成后,我们登录Nexus后台,检查我们的插件是否发布成功。
已经发布成功了,下面我们在实际的项目中来使用。
我们在准备一个项目来测试刚才我们的插件,根据需求,刚才我们的插件的作用是,把项目中的groovy脚本内容进行加密处理,然后制作成升级包。项目的结构如下:
现在我们在项目的pom.xml中加入下面的配置:
加入插件仓库源:
<pluginRepositories>
<pluginRepository>
<id>test-nexusid>
<name>test-nexusname>
<url>http://192.168.101.170:8081/repository/maven-public/url>
<releases>
<enabled>trueenabled>
releases>
pluginRepository>
pluginRepositories>
添加插件:
<plugin>
<groupId>com.testgroupId>
<artifactId>test-mvn-pluginartifactId>
<version>1.0version>
<configuration>
<fromPath>${project.basedir}/groovyfromPath>
<toPath>${project.basedir}/target/scripttoPath>
<secret>720cb14b3151490dad2afaf61d2919fbsecret>
<iv>af3c77c3c6b747e6ad1006e47a1b8422iv>
configuration>
<executions>
<execution>
<phase>generate-resourcesphase>
<goals>
<goal>encodergoal>
goals>
execution>
executions>
plugin>
这里的作用就是把groovy目录下的文件,进行加密,然后输出到target/script目录,最后我们需要把script目录下的文件在打包成ZIP:
<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>packageid>
<formats>
<format>zipformat>
formats>
<includeBaseDirectory>trueincludeBaseDirectory>
<fileSets>
<fileSet>
<directory>${project.build.directory}/scriptdirectory>
<outputDirectory>\scriptoutputDirectory>
fileSet>
fileSets>
assembly>
在代码中我们这样使用:
package com.example.testmvnpkgexespringboot;
import com.example.testmvnpkgexespringboot.groovy.GroovyCompiler;
import org.apache.commons.io.FileUtils;
import org.springframework.beans.factory.annotation.Value;
import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.http.ResponseEntity;
import org.springframework.util.StringUtils;
import org.springframework.web.bind.annotation.PathVariable;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;
import java.io.File;
import java.io.IOException;
@SpringBootApplication
@RestController
public class TestMvnPkgExeSpringbootApplication {
static File script_dir=new File("script");
public static void main(String[] args) {
SpringApplication.run(TestMvnPkgExeSpringbootApplication.class, args);
}
@Value("${script.secret}")
private String secret;
@Value("${script.iv}")
private String iv;
private GroovyCompiler groovyCompiler=new GroovyCompiler();
@RequestMapping("/test/{script}")
public Object test(@PathVariable String script){
//读取script目录下的文件
File file=new File(script_dir,script);
if(!file.exists()){
return ResponseEntity
.notFound();
}
//解密groovy脚本
String content= null;
try {
content = FileUtils.readFileToString(file);
} catch (IOException e) {
return ResponseEntity
.badRequest().body("读取文件异常:"+e.getMessage())
;
}
String de_content = AESCBCUtil.decryptAES_CBC(content,script,iv);
if(StringUtils.isEmpty(de_content)){
return ResponseEntity
.badRequest().body("解密失败")
;
}
//执行脚本
return groovyCompiler.compile(script,de_content).run();
}
}
下面我们进行打包:
mvn clean -DskipTests package
打包完成后,我们来看下结果:
我们发现脚本已经加密了,而且文件的后缀也改变了,这些就是在插件中完成的
最后我们来看下运行结果,groovy的原文如下:
def user=["id":1,"name":"csdn"]
return user;
java -jar app.jar
然后访问:http://localhost:12345/test/get_user
结果和groovy中的代码一致,说明插件能正常使用
1、利用maven的插件功能,不仅能帮助我们来构建项目使用,在有的客户需求其实也只需要开发一个maven插件就能满足,完全可以配合maven来完成一个业务需求。
2、上面项目中大家可以自行根据情况来调整相关的参数,我这里只是为了测试功能
3、maven开发的插件在gradle中也能完全使用
4、maven插件有完全的生命周期,大家可以参照官网的文档