====== ======
====== SpringBoot打包部署最佳实践 ======
[[https://segmentfault.com/u/wls1036|{{https://avatar-static.segmentfault.com/116/471/1164713654-5f22adf620393_big64?nolink&32}}**wls1036**]]发布于 2019-09-15
{{https://sponsor.segmentfault.com/lg.php?nolink&0x0}}
===== spring boot介绍 =====
[[https://spring.io/projects/spring-boot/|Spring Boot]]目前流行的java web应用开发框架,相比传统的spring开发,spring boot极大简化了配置,并且遵守[[https://facilethings.com/blog/en/convention-over-configuration|约定优于配置]]的原则即使0配置也能正常运行,这在spring中是难以想象的。spring boot应用程序可以独立运行,框架内嵌web容器,使得web应用程序可以像本地程序一样启动和调试,十分的方便,这种设计方式也使得spring boot应用程序非常适合容器化进行大规模部署。生态方面,spring boot提供了非常丰富的组件,目前流行的java web框架基本都有spring boot版本,生态十分庞大,是目前java web开发最好的方案。
===== spring boot部署问题 =====
Springboot应用程序有两种运行方式
* 以jar包方式运行
* 以war包方式运行
两种方式应用场景不一样,各有优缺点
==== jar包运行 ====
通过maven插件''spring-boot-maven-plugin'' ,在进行打包时,会动态生成jar的启动类''org.springframework.boot.loader.JarLauncher'' ,借助该类对springboot应用程序进行启动。
=== 优点 ===
* 本地无需搭建web容器,方便开发和调试。
* 因为自带web容器,可以避免由于web容器的差异造成不同环境结果不一致问题。
* 一个jar包就是全部,方便应用扩展。
* 借助容器化,可以进行大规模的部署。
=== 缺点 ===
* 应用过于独立,难以统一管理。
* 数据源无法通过界面进行管理。
* 应用体积过大。
* 修改web容器相关配置较为困难,需要借助代码实现。
==== war包运行 ====
以war包方式运行,通过maven插件''spring-boot-maven-plugin'' 进行相关配置后,最终生成一个可运行在tomcat,weblogic等java web容器中的war包。
=== 优点 ===
* 可以借助web容器管理界面对应用进行管理。
* 可以管理JNDI数据源。
* web容器配置较为灵活,配置和程序分离。
* 应用体积较小,甚至可以借助web容器的包管理功能(比如weblogic Library)进一步减小应用大小。
=== 缺点 ===
* 本地需要搭建web容器,对本地环境要求更高点,学习成本也响应更高。
* 调试较为困难,需要借助web容器。
* 无法兼容所有web容器(比如spring boot2.x无法运行在weblogic 11g上)。
* 部署较为困难(比如和weblogic有较多的类冲突)
在实际的项目中,并没有哪一种方式是最好的,根据客户不同的需求制定不同的部署方案,比如有些客户比较看中管理功能,要求数据源和tomcat相关配置必须由管理员进行管理,那么选择war包方式,有些客户希望借助容器化进行大规模部署,那么jar方式更适合。不管选择哪种方式,在部署时都会遇到下面的问题
* 如果需要打war包,那么不仅是pom文件需要修改,应用程序也要做相应的改动,改动完后,应用程序就无法本地运行,需要打完包后将配置信息修改回来,这样不仅麻烦,还容易出错。
* 不管是war包还是jar包,如何管理不同环境的配置文件,保证不会出错,虽然spring boot有提供''spring.profiles.active'' 配置设置不同的环境,但一方面需要人为修改配置文件,只要是人为的就有可能出错,另一方面,客户有时出于安全考虑不会提供生产环境配置信息,那么这时候就无法指定''prifiles.active'' 。
* 如何将多个spring boot模块打包在一起。
* jar包需要配合容器化才能发挥出最大的优势,如果没有容器,spring boot jar包就是一个''玩具'' ,随处运行的jar包,缺少统一管理,是达不到生产的要求,那么如果从jar包到容器也是一个问题。
早期碰到这些问题,都是人工解决,不仅效率十分低下,部署一次都需要十几分钟,而且很容易出错,一百次出错一次算是概率低了,但是生产出错一次都是重大事件,所以我们也在思考如何通过自动化解决以上问题,如何将开发和部署分离,开发人员只关心开发,开发完提交代码,打包和部署都是后台透明的完成。以下就是我们的解决方案。
===== 打包 =====
==== war包打包问题解决 ====
spring boot打war包的步骤如下
* 在''pom.xml'' 中将打包方式改为war。
''
* 设置''spring-boot-starter-tomcat'' 范围为''provided''
''
* 修改spring boot的启动类,继承''SpringBootServletInitializer''
''public class DemoApplication extends SpringBootServletInitializer{
@Override
protected SpringApplicationBuilder configure(SpringApplicationBuilder builder) {
return builder.sources(DemoApplication.class);
}
}''
每打包一次都要修改''pom.xml'' 和启动类,打包完再修改回来,十分的繁琐,因为,我们提出以下整改方案
* 从''pom.xml'' 复制一个''pom-war.xml'' 文件,将''pom-war.xml'' 修改为war包配置
* 在根目录下(除了src目录外都可以)复制一份启动类的代码,修改为war包的配置方式。
* 编写shell脚本进行打包。
shell脚本打包过程为
- 备份当前启动类的java代码。
- 将war包启动类的代码替换掉当前启动类的代码。
- maven指定''pom-war.xml'' 文件进行打包。
- 打包结束后恢复启动类文件。
以下就是参考脚本
**//app-war.sh// **
''#!/usr/bin/env bash
v1=src/main/java/com/definesys/demo/DemoApplication.java
v2=war/DemoApplication.java
v3=war/DemoApplication-bak.java
cp -rf $v2 $v1
mvn clean package -Dmaven.test.skip=true -f war-pom.xml
#recovery
cp -rf $v3 $v1''
通过预先配置好pom文件和启动类文件,开发人员只要运行''app-war.sh'' 脚本无需修改任何文件即可生成war包。
=== 更优的方案 ===
以上方案pom文件和启动类文件都需要预先准备好,未实现完全的自动化,通过优化方案做到完全自动化。
* 脚本可以通过find命令搜索以''*Application.java'' 结尾的文件,作为启动类文件,读取文件名获取类名,通过字符串替换方式动态生成war包启动类文件。
* 在pom.xml中用注释设置好锚点,脚本通过替换锚点动态生成pom.xml文件。
* 如果不希望通过锚点实现,可以借助更高级的脚本语言,比如python对xml进行解析,再动态生成xml。
==== 多模块打包 ====
这里的多模块指的是maven中的多模块,项目工程中的代码多模块,一个项目按功能划分模块后,在创建工程时一般也按照功能层面上的模块进行创建,这样避免一个模块代码过于庞大,也利于任务的分工,但打包却更麻烦了。
* 每个模块都是独立的spring boot程序,整合到一个包的时候会出现多个启动类,多个配置文件冲突的问题。
* 每个模块有引用相同的依赖,依赖包版本升级后,需要每个pom文件都做修改。
通过优化项目结构解决以上问题
* 父项目的pom指定spring boot的依赖和公共的依赖。
* 创建一个spring boot的子项目,作为启动项目,我们称为''start'' 项目。
* 其余子项目为普通的java maven项目,parent设置为第一步创建的spring boot父项目。
* start项目的pom引用其他子项目的依赖。
* 本地调试可以直接运行start的启动类,ide会自动编译其他模块并引用。
* 打包可以在父项目上进行''install'' 后再进入start项目进行打包,脚本参考如下
''mvn clean install
cd start
mvn clean package''
//目录结构如下//
''.
├── pom.xml
├── role
│ ├── pom.xml
│ └── src
│ ├── main
│ │ ├── java
│ │ │ └── com
│ │ │ └── definesys
│ │ │ └── demo
│ │ │ └── controller
│ │ │ └── RoleController.java
│ │ └── resources
├── start
│ ├── pom.xml
│ ├── src
│ │ ├── main
│ │ │ ├── java
│ │ │ │ └── com
│ │ │ │ └── definesys
│ │ │ │ └── demo
│ │ │ │ └── DemoApplication.java
│ │ │ └── resources
│ │ │ └── application.properties
└── user
├── pom.xml
└── src
└── main
├── java
│ └── com
│ └── definesys
│ └── demo
│ └── controller
│ └── UserController.java
└── resources
''
* start项目包含包含启动类和配置文件,pom文件引用其余子项目。
**//start pom.xml// **
''
* 父项目parent为spring boot,引用spring boot相关依赖和各个子项目公共的依赖
**//父项目 pom.xml// **
''
* 所有非start的子项目需要指定版本号并且父项目都设为根目录项目。
**//子项目 pom.xml// **
''
* 所有子项目的包路径前缀必须一样,并且以start项目作为基本路径。
==== 配置文件问题 ====
spring boot提供''spring.profiles.active'' 指定配置文件,但生产环境有时候客户出于安全考虑不提供配置信息给开发人员,而是预先将配置文件上传到服务器指定路径,程序需要在运行时去引用该配置文件,如果运行环境是''kubernetes'' ,则会提供一个config map作为配置文件,这时候就要求spring boot程序读取外部配置文件。
这里讨论的是线上环境配置文件方案,本地调试参考子模块打包相关内容,可以将配置文件统一写在start项目中。=== jar包外部配置文件读取 === jar运行可以通过指定参数''spring.config.location'' 引用外部文件,命令参考如下:
''java -jar start-1.0-SNAPSHOT.jar --spring.config.location=/Users/asan/workspace/config''
''config'' 目录存放properties配置文件
可以通过配合''spring.profiles.active'' 参数可以指定目录下配置文件,如:
''java -jar start-1.0-SNAPSHOT.jar --spring.profiles.active=prod --spring.config.location=/Users/asan/workspace/config''
则会读取''/Users/asan/workspace/config/appliction-prod.properties'' 文件作为配置文件。
=== war包外部配置文件读取 ===
以''tomcat'' 为例,需要在tomcat启动时指定''-Dspring.config.location'' 参数,可以设置服务器环境变量''CATALINA_OPTS'' 达到目的。可以编辑用户 prifile文件
''export CATALINA_OPTS=/Users/asan/workspace/config''
同样,也可以通过''-Dspring.profiles.active'' 指定配置文件名称。
==== 容器化 ====
spring boot借助容器化,可以如虎添翼,发挥出更大的威力,也只有通过容器化,才能体会到spring boot开发的高效。通过以上的介绍,你可以很顺利的打好一个jar包或者war包,那么可以通过编写dockerfile文件进行镜像的构建。spring boot在构建镜像时有两个地方需要考虑
* 时区问题,基础镜像的时区默认是''UTC'' ,比北京时间早8小时,需要指定镜像时区。
* 配置文件问题,需要指定外部配置文件(根据项目具体情况选择)。
**//app-jar-dockerfile.Dockerfile// **
''FROM openjdk:8-jdk-alpine
MAINTAINER definesys.com
VOLUME /tmp
ADD start-1.0-SNAPSHOT.jar app.jar
RUN echo "Asia/Shanghai" > /etc/timezone
EXPOSE 8080
ENTRYPOINT ["java","-Djava.security.egd=file:/dev/./urandom","-jar","--spring.config.location=/Users/asan/workspace/config","/app.jar"]''
**//app-war.dockerfile.Dockerfile// **
''FROM tomcat
MAINTAINER definesys.com
ENV CATALINA_OPTS -Dspring.config.location=file:/middleware/config/
ADD start-1.0-SNAPSHOT.war /usr/local/tomcat/webapps/app.jar
RUN echo "Asia/Shanghai" > /etc/timezone
EXPOSE 8080
EOF''
===== 部署 =====
早期我们采用的是以下部署过程
{{https://segmentfault.com/img/remote/1460000020388317?nolink&}}
* 首先构建测试环境的镜像,上传到镜像仓库,应用重新部署。
* 接着构建UAT环境的镜像,上传到镜像仓库,应用重新部署。
* 最后构建生产环境的镜像,上传到镜像仓库,应用重新部署。
每一次发布都是一个新的镜像,但这种方式有个问题就是如何保证前一个环境验证没问题,后一个环境就一定没问题,因为两个镜像是不一样的,虽然可能两次构建都是基于同一版本代码,但因为是重新构建,中间可能因为各种原因,如maven包版本更新等,无法保证两次构建就是完全一样的镜像。因此我们优化了构建的流程,如下:
{{https://segmentfault.com/img/remote/1460000020388318?nolink&}}
所有的环境都是用同一个镜像,环境之间只有配置文件不同,文件通过configmap或者外部配置文件方式进行挂载,这样保证了配置文件没问题的前提下,每个环境的程序一定是一样的。
===== jenkins自动打包部署 =====
打包和部署在本地进行也是有问题的,本地jdk版本取决于个人电脑,甚至有黑客污染jdk导致编译的class文件自带后门,个人电脑环境也是随着用户不同操作可能改变,构建出来的包不能保证是稳定的包。因此需要一个远程服务器用于打包和部署,能够实现从源码到镜像过程。jenkins是一个基于java开发的持续集成工具,通过配置插件和编写脚本实现程序从代码到制品再到线上运行的过程。jenkins在spring boot开发中主要完成了以下工作。
* 通过gitlab插件实现源代码的获取。
* 基于以上介绍的脚本,实现从源码到制品的过程。
* 通过docker工具实现从制品到镜像的过程。
* 通过kubectl工具,实现从镜像到上云的过程。
jenkins在构建镜像时需要借助docker工具,但jenkins本身也是有docker版本的,所以就面临着''docker in docker'' 的问题,这里选择的方案是用二进制文件安装jenkin而非镜像方式,虽然丧失了docker的便利性,但可以简化docker方案,降低集成的复杂度。