前段时间,本人参与了某项目的从“零”开始的开发与运维。真的是从零开始啊……从项目设计到开发,再到发布、运维,说多了都是泪……还好现在有好多现成的工具可以使用,省了很多时间和精力。
此项目使用AWS,Web 端架构采用 ELB + AutoScalling group,数据库使用RDS,文件存储使用了S3。使用这个架构可以节省很多的运维时间和精力,可以拿更多的时间关注项目的开发。但是这个架构并不包括代码部署的方面,本文主要介绍在代码部署方面自动化运维道路上的各种进化。
项目主要软件环境: Java EE, Spring 4 MVC, maven, tomcat8, gitlab
项目分测试环境和生产环境,生产环境采用ELB+AutoScalling,测试环境只有一台服务器跑tomcat,虽然不是很严谨,但是在前期还是能省(qian)则省了
……
在代码部署方面大体经历了以下几个阶段。
石器时代
最开始时在本地开发测试,然后idea 打包上传到服务器上,然后ssh 登陆服务器手动部署代码。每次代码部署都要执行n多操作和命令。有段时间网络不是很好,光上传war 包就耗费十几分钟,对耐心是一场很大的考验。实在受不了这种繁琐的操作时候开始了一步步简化操作。
服务器上部署war 时需要先停止tomcat,然后删除tomcat webapps 目录下ROOT.war 文件和ROOT 目录,然后移动新的ROOT.war 到webapps 下,最后启动tomcat 服务。首先对这个步骤写了个shell 脚本:
“石头锤子” deployWar.sh
- #! /bin/bash
-
- if [[ $# -eq 0 ]]; then
- warFile="/home/ec2-user/target/hs-0.1-SNAPSHOT.war"
- elif [[ $# -eq 1 ]]; then
- warFile="$1"
- else
- echo "Parameter Error!"
- exit 1
- fi
-
- if [[ -f "$warFile" ]]; then
- service tomcat8 stop
- rm -f "/usr/share/tomcat8/webapps/ROOT.war"
- rm -rf "/usr/share/tomcat8/webapps/ROOT"
- cp "$warFile" "/usr/share/tomcat8/webapps/ROOT.war"
- service tomcat8 start
- mv "$warFile" "${warFile%'.war'}_`date +'%Y-%m-%d_%H:%M:%S'`.war"
- echo
- echo Done!
- else
- echo "No file: $warFile!"
- fi
此“石头锤子”能实现上述war包的部署步骤,并对当前部署的war包进行备份。
然后又出现一个问题,如果改动只有一个或几个文件,完整部署太麻烦,这时可以只上传改动的文件,然后部署就可以了。
“石头镰刀1” updateClasses.sh
- #! /bin/bash
-
- service tomcat8 stop
- cp -R /home/ec2-user/target/classes/ /home/ec2-user/webapps/ROOT/WEB-INF/
- service tomcat8 start
- echo Done!
tomcat 的class 文件更新后需要重启tomcat 才能生效,而静态文件如js、css 文件等直接覆盖即可。所以针对静态文件有:
“石头镰刀2” updateStatic.sh
- #! /bin/bash
-
- cp -R /home/ec2-user/src/main/webapp/WEB-INF/ /home/ec2-user/webapps/ROOT/
- echo Done!
“铁器时代”
当测试情况良好需要部署到生产坏境时,就涉及到从原来的单点部署到集群部署了。原来的脚本和架构也不太适合了,幸好我们有还有铸就“金刚”之身的原料--S3。首先我们将需要部署的war包上传的到S3 的指定目录,登陆需要部署的服务器,下载该war 包并部署。流程很简单,但是需要执行的命令也是繁杂和重复。
“小铁铲” cpWarToS3.sh
- #! /bin/bash
-
- if [[ $# -eq 0 ]]; then
- warFile="/home/ec2-user/target/hs-0.1-SNAPSHOT.war"
- elif [[ $# -eq 1 ]]; then
- warFile="$1"
- else
- echo "Parameter Error!"
- exit 1
- fi
-
- if [[ -f "$warFile" ]]; then
- echo Copy $warFile to S3...
- aws s3 cp "$warFile" "s3://config.ziyoufang.cn/war/ROOT.war"
- echo Done!
- else
- echo "No file: $warFile!"
- fi
“大铁锤” deployFromS3.sh
- #! /bin/bash
- if [[ $# -eq 0 ]]; then
- WAR=ROOT.war
- elif [[ $# -eq 1 ]]; then
- WAR=$1
- else
- echo "Parameter Error!"
- exit 1
- fi
-
- WAR=`echo $WAR | awk -F '.' '{print $1}'`
-
- service tomcat8 stop
- rm -f "/usr/share/tomcat8/webapps/ROOT.war"
- rm -rf "/usr/share/tomcat8/webapps/ROOT"
- aws s3 cp s3://config.ziyoufang.cn/war/"$WAR".war "/usr/share/tomcat8/webapps/ROOT.war"
- service tomcat8 start
- log="`date +'%Y-%m-%d %H:%M:%S'` Deploy war from s3. done!"
- echo $log > deploy.log
- echo
- echo Done!
从文件名上就可以看出这两个脚本一个是用来将war 上传到S3,一个是从S3 下载war包并部署的。
“工业时代”
上面还面临着一个重要的问题,就是每次部署都要打包上传完整的war 包到S3,这也是一个比较耗时耗力的过程,对于一个能坐就不站,能躺就不坐的”懒货“来说是一种巨大的折磨。
“烈火” gitlab +“鼓风” maven
gitlab 作为一款优秀的git server 系统,maven 作为一款最常用的包管理软件之一,各位前辈已经提供的了丰富的工具我们就得充分利用。在开发时使用git 做版本控制,gitlab 部署在AWS 上,开发只需要和gitlab 进行sync 即可。然后在服务器上使用mvn clean install 进行打包,并上传到S3上。
“小卡车” updateWarToS3.sh
- #! /bin/bash
-
- if [[ $# -eq 0 ]]; then
- BRANCH=master
- elif [[ $# -eq 1 ]]; then
- BRANCH=$1
- TARGET=ROOT.war
- elif [[ $# -eq 2 ]]; then
- BRANCH=$1
- TARGET=$2
- else
- echo "Parameter Error!"
- exit 1
- fi
- TARGET=`echo $TARGET | awk -F '.' '{print $1}'`
-
- cd /home/ec2-user/Web_server
- git checkout $BRANCH
- git pull
- mvn clean install
-
- S3_Prefix="s3://config.ziyoufang.cn/war/"
- S3_War=$S3_Prefix"$TARGET".war
- S3_WarBack=$S3_Prefix"$TARGET""_`date +'%Y-%m-%d_%H:%M:%S'`.war"
- warFile="/home/ec2-user/Web_server/target/hs-0.1-SNAPSHOT.war"
-
- echo Backup "$TARGET".war on S3...
- aws s3 mv $S3_War $S3_WarBack
-
- echo upload new "$TARGET".war...
- aws s3 cp $warFile $S3_War
-
- echo "upload done."
此时部署时只需先执行
updateWarToS3.sh
,然后登陆需要部署的服务器执行
deployFromS3.sh
即可。这下干感觉就是从原始社会走出来了~~爽啊~~~
可是……(哎……就怕有可是……)在部署生产环境时,每次都需要执行多个流程:
从ELB 中移除一台EC2 -> 等待connection draining -> 登陆该EC2 -> 执行deployFromS3.sh -> 等待tomcat启动起来 -> 添加该EC2 回ELB -> 等待监控状态检查到InService -> 下一台EC2……
在压力小的情况下执行该操作ELB 后端实例较少,部署几次之后我烦了,交给了另外一个人去部署,(嗯,以邻为壑的感觉挺爽~~)结果他部署了几次之后他也烦了,威胁说撂挑子不干了……无奈只好继续利用我大shell 铸造大杀器了……
“巨型铲车” deployIntoELBBackendInstance.sh
该脚本能实现自动将指定ELB 下的后端健康实例进行部署,最后会提示部署成功和部署失败的实例。
至此,整个部署流程在updateWarToS3.sh 之后只需要执行deployIntoELBBackendInstance.sh 就可以了。
手执“大铁锤”,开着“巨型铲车”,慵懒的日子~舒坦~~~