背景
- 业务需求根据用户的一些配置利用模板引擎来生成代码并实时打包。
- 将来需要支持二方GAV的支持,所以采用maven来编译,不采用javac的方式执行
- 项目采用docker容器化部署,外部不给予拥有maven环境的镜像,并且不开放上传自定义镜像的功能
思考过程
- 既然不能提供maven环境的镜像,我就联想到springboot中能够内嵌tomcat,尝试是否能够直接将maven.tar.gz的解压缩文件放在项目的resource中
- 利用
java.lang.Process
来运行mvn命令
开战
-
首先创建一个maven工程,将解压的maven文件放在
WEB-INF/maven
下(这里需要注意一下,之前我直接放在resource中,因为这里的项目是多Module项目,并且每个module是通过jar包的方式放在WEB-INF/lib
中,后面这样的方式存在一定的问题)。为了方便测试,我直接将模板生成的java文件以maven工程的结构放在WEB-INF/sourceCode/{$projectName}/
中,结构如下:
这里的1表示projectName
修改
WEB-INF/maven/conf/settings.xml
中的镜像地址为公司的私有仓库直接编写service代码,这里涉及到自定义的几个路径,
WEB_INF_PATH
、EMBEDDED_MAVEN_PATH
、EMBEDDED_MAVEN_COMMAND
、SOURCE_CODE_PATH
,大家参照代码和命名理解一下
package com.xxx.xxx.xxx.xxx.biz.service.impl;
import java.io.IOException;
import java.io.InputStream;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.beans.factory.InitializingBean;
import org.springframework.stereotype.Service;
import com.xxx.xxx.xxx.xxx.biz.service.IndexService;
/**
* @author coderyao
* @version IndexServiceImpl, v 0.1 2018/7/24 上午11:18 coderyao Exp
*/
@Service
public class IndexServiceImpl implements IndexService, InitializingBean {
private static final Logger LOGGER = LoggerFactory.getLogger(IndexServiceImpl.class);
private final String WEB_INF_PATH = IndexServiceImpl.class.getClassLoader().getResource("../").getPath();
private final String COMPILE_PATH = WEB_INF_PATH + "sourceCode/";
private final String EMBEDDED_MAVEN_PATH = WEB_INF_PATH + "maven/";
private final String EMBEDDED_MAVEN_COMMAND = EMBEDDED_MAVEN_PATH + "bin/mvn";
@Override
public void maven() throws IOException {
ProcessBuilder pb = new ProcessBuilder();
//构建命令
//若maven的路径在jar包中这里拿到的EMBEDDED_MAVEN_COMMAND是不能被执行的,因为是以file://打头的
pb.command(EMBEDDED_MAVEN_COMMAND, "-DskipTests=true", "clean", "package", "-f", IndexServiceImpl.class.getClassLoader().getResource("../sourceCode/1/pom.xml").getPath());
//执行命令
Process process = pb.start();
InputStream inputStream = process.getInputStream();
byte[] buffer = new byte[512];
int readSize;
while ((readSize = inputStream.read(buffer)) > 0) {
System.out.write(buffer, 0, readSize);
}
}
/**
* 修改权限(这里粗犷的修改为777,如有精细化的权限控制,自己调整一下)
* 因为一些原因,虽然线上默认的执行用户是root,并且权限为rwx,依然会报权限不足的错误
* 如果有大神指导原因请指点一二
* @throws Exception
*/
@Override
public void afterPropertiesSet() throws Exception {
//修改maven目录权限
Process chmodMaven = new ProcessBuilder("chmod", "-R", "777", EMBEDDED_MAVEN_PATH).start();
//等待完成
chmodMaven.waitFor();
LOGGER.info("修改权限完成:{}", EMBEDDED_MAVEN_PATH);
//修改编译目录权限
Process chmodCompile = new ProcessBuilder("chmod", "-R", "777", COMPILE_PATH).start();
chmodCompile.waitFor();
LOGGER.info("修改权限完成:{}", COMPILE_PATH);
}
}
- 提交代码,通过jenkins生成镜像并生成容器。
- 观察到控制台有陆续的命令出来,并在指定的文件夹中找到了jar包
踩到的坑
- 因为maven包中存在bin目录,而我的
.gitignore
文件中配置了下图内容,导致运行的时候老是报NPE,之前还以为公司平台考虑安全把bin
文件夹都给屏蔽了,还是运维同学提醒了我,感到特别的尴尬
#
# Eclipse and Maven idea output folders.
#
bin/
*/**/bin
target/
out/
*.class
.svn/
build/
- 另外一个目录结构确实也困扰我很久,如何获取到WEB项目的目录让我很头疼。。。还是自身不足
- 权限问题需要理解一下
总结
- 第一次写博客,可能并不是特别能够说明问题,或者说这个问题很浅显,后期会补上github地址
- 参考idea执行的mvn命令,之前也一直没有理解,这里顺便给大家参考一下
/Library/Java/JavaVirtualMachines/jdk1.8.0_60.jdk/Contents/Home/bin/java -Dmaven.multiModuleProjectDirectory=/Users/coderyao/projects/xx/xxx -Dmaven.home=/Library/maven -Dclassworlds.conf=/Library/maven/bin/m2.conf -Dfile.encoding=UTF-8 -classpath /Library/maven/boot/plexus-classworlds-2.5.2.jar org.codehaus.classworlds.Launcher -Didea.version=2018.1 -s /Library/maven/conf/settings.xml -DskipTests=true clean package -f pom.xml -P dev
- 希望能够在某些方面帮到大家