jenkins运行日志时间与linux,持续集成之Jenkins结合脚本实现代码自动化部署及一键回滚至上一版本...

一:本文通过jenkins调用shell脚本的的方式完成从Git服务器获取代码、打包、部署到web服务器、将web服务器从负载均衡器删除、解压、复制配置文件、创建软连接、测试每一台web服务器、将web服务器添加至负载均衡、回滚到任意指定版本、一键回滚到上一版本等功能,脚本放在www用户家目录并使用www用户身份执行,每个web服务器也都使用www用户运行web服务,且UID相同web目录和权限都一致,更严格的标准化可以带来更安全的生产环境和更高的效率:

1.1:在jenkins项目配置中调用shell脚本与环境准备:

1.1.1:#jenkins-项目-配置:

15ad43d0974cc37d9a2487b70a09b1b0.png

1.1.2:www用户家目录中的脚本内容:

$ cat code_deploy.sh

#!/bin/bash

#Dir List 部署节点(即部署节点需要做的操作)

# mkdir -p /deploy/code/web-demo

# mkdir -p /deploy/config/web-demo/base

# mkdir -p /deploy/config/web-demo/other

# mkdir /deploy/tmp

# mkdir /deploy/tar

# chown -R www.www /deploy

# chown -R www.www /webroot

# chown -R www.www /opt/webroot/

# chown -R www.www /webroot

# 需要在客户端节点做的操作

# mkdir /opt/webroot

# mkdir /webroot

# chown -R www.www /webroot

# chown -R www.www /opt/webroot/

# chown -R www.www /webroot

# [www@ ~]$ touch /webroot/web-dem

# Node List 服务器节点

PRE_LIST="192.168.3.12"        # 预生产节点

GROUP1_LIST="192.168.3.12 192.168.3.13"

GROUP2_LIST="192.168.3.13"

ROLLBACK_LIST="192.168.3.12 192.168.3.13"

# 日志日期和时间变量

LOG_DATE='date "+%Y-%m-%d"' # 如果执行的话后面执行的时间,此时间是不固定的,这是记录日志使用的时间

LOG_TIME='date "+%H-%M-%S"'

# 代码打包时间变量

CDATE=$(date "+%Y-%m-%d") # 脚本一旦执行就会取一个固定时间赋值给变量,此时间是固定的

CTIME=$(date +"%H-%M-%S")

# shell env 脚本位置等变量

SHELL_NAME="deploy.sh"    # 脚本名称

SHELL_DIR="/home/www/"  # 脚本路径

SHELL_LOG="${SHELL_DIR}/${SHELL_NAME}.log" # 脚本执行日志文件路径

# code env 代码变量

PRO_NAME="web-demo"    # 项目名称的函数

CODE_DIR="/deploy/code/web-demo"    # 从版本管理系统更新的代码目录

CONFIG_DIR="/deploy/config/web-demo"    # 保存不同项目的配置文件,一个目录里面就是一个项目的一个配置文件或多个配置文件

TMP_DIR="/deploy/tmp"            # 临时目录

TAR_DIR="/deploy/tar"            # 打包目录

LOCK_FILE="/tmp/deploy.lock" # 锁文件路径

usage(){ # 使用帮助函数

echo $"Usage: $0 [ deploy | rollback [ list | emergency | version ]"

}

writelog(){ # 写入日志的函数

LOGINFO=$1 # 将参数作为日志输入

echo "${CDATE} ${CTIME} : ${SEHLL_NAME} : ${LOGINFO}" >> ${SHELL_LOG}

}

# 锁函数

shell_lock(){

touch ${LOCK_FILE}

}

# 解锁函数

shell_unlock(){

rm -f ${LOCK_FILE}

}

# 获取代码的函数

code_get(){

echo "code_get"

writelog code_get

cd $CODE_DIR && git pull # 进入到代码目录更新代码,此处必须免密码更新,此目录仅用于代码更新不能放其他任何文件

cp -rf ${CODE_DIR} ${TMP_DIR}/ # 临时保存代码并重命名,包名为时间+版本号,准备复制到web服务器

API_VERL=$(git show | grep commit | cut -d ' ' -f2)

API_VER=$(echo ${API_VERL:0:8})        # 版本号

}

code_build(){ # 代码编译函数

echo code_build

}

code_config(){ # 配置文件函数

writelog "code_config"

/bin/cp -rf ${CONFIG_DIR}/base/* ${TMP_DIR}/"${PRO_NAME}" # 将配置文件放在本机保存配置文件的临时目录,用于暂时保存代码项目

PKG_NAME="${PRO_NAME}"_"$API_VER"_"${CDATE}-${CTIME}"    # 定义代码目录名称

cd ${TMP_DIR} && mv ${PRO_NAME} ${PKG_NAME}    # 重命名代码文件为web-demo_123-20170629-11-19-10格式

}

code_tar(){    # 对代码打包函数

writelog code_tar

cd ${TMP_DIR} && tar czf ${PKG_NAME}.tar.gz ${PKG_NAME} --exclude=".git" # 将目录打包成压缩文件,便于网络传输

writelog "${PKG_NAME}.tar.gz packaged success"    # 记录打包成功的日志

}

code_scp(){ # 代码压缩包scp到客户端的函数

writelog  "code_scp"

for node in $PRE_LIST;do # 循环服务器节点列表

scp ${TMP_DIR}/${PKG_NAME}.tar.gz $node:/opt/webroot/ # 将压缩后的代码包复制到web服务器的/opt/webroot

done

for node in $GROUP1_LIST;do # 循环服务器节点列表

scp ${TMP_DIR}/${PKG_NAME}.tar.gz $node:/opt/webroot/ # 将压缩后的代码包复制到web服务器的/opt/webroot

done

}

cluster_node_add(){ #将web服务器添加至前端负载

echo cluster_node_add

}

cluster_node_remove(){ # 将web服务器从集群移除函数(正在部署的时候应该不处理业务)

writelog "cluster_node_remove"

}

url_test(){

URL=$1

curl -s --head $URL |grep '200 OK'

if [ $? -ne 0 ];then

shell_unlock;

writelog "test error" && exit;

fi

}

pre_deploy(){ # 代码解压部署函数,预生产节点

writelog "pre_deploy"

for node in ${PRE_LIST};do # 循环预生产服务器节点列表

cluster_node_remove  ${node} # 部署之前将节点从前端负载删除

echo  "pre_deploy, cluster_node_remove ${node}"

ssh ${node} "cd /opt/webroot && tar zxf ${PKG_NAME}.tar.gz" #分别到web服务器执行压缩包解压命令

ssh ${node} "rm -f /webroot/web-demo && ln -s /opt/webroot/${PKG_NAME} /webroot/web-demo" # 整个自动化的核心,创建软连接

done

}

pre_test(){ # 预生产主机测试函数

for node in ${PRE_LIST};do # 循环预生产主机列表

curl -s --head http://${node}:9999/index.html | grep "200 OK" # 测试web界面访问

if [ $? -eq 0 ];then  # 如果访问成功

writelog " ${node} Web Test OK!" # 记录日志

echo " ${node} Web Test OK!"

cluster_node_add ${node} # 测试成功之后调用添加函数把服务器添加至节点,

writelog "pre,${node} add to cluster OK!" # 记录添加服务器到集群的日志

else # 如果访问失败

writelog "${node} test no OK" # 记录日志

echo "${node} test not OK"

shell_unlock # 调用删除锁文件函数

break # 结束部署

fi

done

}

group1_deploy(){ # 代码解压部署函数

writelog "group1_code_deploy"

for node in ${GROUP1_LIST};do # 循环生产服务器节点列表

cluster_node_remove $node

echo "group1, cluster_node_remove $node"

ssh ${node} "cd /opt/webroot && tar zxf ${PKG_NAME}.tar.gz" # 分别到各web服务器节点执行压缩包解压命令

ssh ${node} "rm -f /webroot/web-demo && ln -s /opt/webroot/${PKG_NAME} /webroot/web-demo" # 整个自动化的核心,创建软连接

done

scp ${CONFIG_DIR}/other/192.168.3.13.server.xml 192.168.3.13:/webroot/web-demo/server.xml  # 将差异项目的配置文件scp到此web服务器并以项目结尾

}

group1_test(){ # 生产主机测试函数

for node in ${PRE_LIST};do # 循环生产主机列表

curl -s --head http://${node}:9999/index.html | grep "200 OK" #测试web界面访问

if [ $? -eq 0 ];then  #如果访问成功

writelog " ${node} Web Test OK!" #记录日志

echo "group1_test,${node} Web Test OK!"

cluster_node_add

writelog " ${node} add to cluster OK!" #记录将服务器 添加至集群的日志

else #如果访问失败

writelog "${node} test no OK" #记录日志

echo "${node} test no OK"

shell_unlock # 调用删除锁文件函数

break # 结束部署

fi

done

}

emergency_code_get(){ #获取代码的函数

writelog "code_get"

cd ${CODE_DIR} && git reset --hard HEAD^  #进入到代码目录更新代码,此处必须免密码更新,此目录仅用于代码更新不能放其他任何文件

/bin/cp -rf ${CODE_DIR} ${TMP_DIR}/ #临时保存代码并重命名,包名为时间+版本号,准备复制到web服务器

API_VERL=$(git show | grep commit | cut -d ' ' -f2)

API_VER=$(echo ${API_VERL:0:8}) #取八位

}

emergency(){  #紧急回退到上一个版本函数

emergency_code_get #执行将代码回退到上一个版本函数

code_build; #如果要编译执行编译函数

code_config; #cp配置文件

code_tar; #打包

code_scp; #scp到服务器

cluster_node_remove;

pre_deploy; #预生产环境部署

pre_test; #预生产环境测试

group1_deploy; #生产环境部署

group1_test; #生产环境测试

code_config; #cp差异文件

#code_test; #代码测试

shell_unlock #执行完成后删除锁文件

}

rollback_fun(){

for node in $ROLLBACK_LIST;do # 循环服务器节点列表

# 注意一定要加"号,否则无法在远程执行命令

ssh $node "rm -f /webroot/web-demo && ln -s /opt/webroot/$1 /webroot/web-demo" # 立即回滚到指定的版本,$1即指定的版本参数

echo "${node} rollback success!"

done

}

rollback(){ # 代码回滚主函数

if [ -z $1 ];then

shell_unlock # 删除锁文件

echo "Please input rollback version" && exit 3;

fi

case $1 in # 把第二个参数做当自己的第一个参数

list)

ls -l /opt/webroot/*.tar.gz

;;

*)

rollback_fun $1

esac

}

main(){

if [ -f $LOCK_FILE ] ;then # 先判断锁文件在不在,如果有锁文件直接退出

echo "Deploy is running" && exit 10

fi

DEPLOY_METHOD=$1 # 避免出错误将脚本的第一个参数作为变量

ROLLBACK_VER=$2

case $DEPLOY_METHOD in

deploy) # 如果第一个参数是deploy就执行以下操作

shell_lock; # 执行部署之前创建锁。如果同时有其他人执行则提示锁文件存在

code_get; # 获取代码

code_build; # 如果要编译执行编译函数

code_config; # cp配置文件

code_tar;    # 打包

code_scp;    # scp到服务器

pre_deploy;  # 预生产环境部署

pre_test;    # 预生产环境测试

group1_deploy; # 生产环境部署

group1_test;  # 生产环境测试

shell_unlock; # 执行完成后删除锁文件

;;

rollback) # 如果第一个参数是rollback就执行以下操作

shell_lock; # 回滚之前也是先创建锁文件

rollback $ROLLBACK_VER;

shell_unlock; # 执行完成删除锁文件

;;

emergency)

emergency; #紧急回退就不需要参数了,但是在执行的时候要确认一下是否要紧急回退,避免输入错误

;;

*)

usage;

esac

}

main $1 $2

1.1.4:修改当前web页面:

[www@master ~]$ cd web-demo

[www@master web-demo]$ echo "

jenkins deploy test" > index.html

[www@master web-demo]$ git add index.html

[www@master web-demo]$ git commit -m "jenkins deploy test"

[master 9a43cf5] jenkins deploy test

1 file changed, 1 insertion(+), 1 deletion(-)

[www@master web-demo]$ git push -u origin master

Counting objects: 5, done.

Compressing objects: 100% (2/2), done.

Writing objects: 100% (3/3), 276 bytes | 0 bytes/s, done.

Total 3 (delta 1), reused 0 (delta 0)

To git@192.168.3.198:web/web-demo.git

beb37cb..9a43cf5 master -> master

Branch master set up to track remote branch master from origin.

a287224186ea330d313617e095042449.png

1.4:回滚到任意版本:

1.4.1:在哪看回滚到的版本?:

$ ll /deploy/tmp/ #部署服务器,web服务器在nginx定义的目录查看版本

1.4.3:在jenkins执行回滚:

[root@slave01 ~]# ll /opt/webroot/

total 20672

drwxr-xr-x 5 www www 4096 Jun 26 11:36 web-demo_123_2017-06-26-11-36-44

-rw-rw-r-- 1 www www 1243347 Jun 28 22:03 web-demo_123_2017-06-26-11-36-44.tar.gz

drwxr-xr-x 5 www www 4096 Jun 26 11:39 web-demo_123_2017-06-26-11-39-02

-rw-rw-r-- 1 www www 1243347 Jun 28 22:06 web-demo_123_2017-06-26-11-39-02.tar.gz

drwxr-xr-x 5 www www 4096 Jun 26 12:04 web-demo_123_2017-06-26-12-04-19

-rw-rw-r-- 1 www www 1243351 Jun 28 22:31 web-demo_123_2017-06-26-12-04-19.tar.gz

drwxr-xr-x 5 www www 4096 Jun 26 12:16 web-demo_123_2017-06-26-12-16-49

-rw-rw-r-- 1 www www 1243347 Jun 28 22:43 web-demo_123_2017-06-26-12-16-49.tar.gz

drwxr-xr-x 5 www www 4096 Jun 26 12:18 web-demo_123_2017-06-26-12-18-09

-rw-rw-r-- 1 www www 1243347 Jun 28 22:45 web-demo_123_2017-06-26-12-18-09.tar.gz

drwxr-xr-x 5 www www 4096 Jun 26 12:18 web-demo_123_2017-06-26-12-18-57

-rw-rw-r-- 1 www www 1243369 Jun 28 22:46 web-demo_123_2017-06-26-12-18-57.tar.gz

-rw-rw-r-- 1 www www 45 Jun 29 06:21 web-demo__2017-06-30-14-28-54.tar.gz

drwxrwxr-x 2 www www 4096 Jun 30 14:30 web-demo__2017-06-30-14-30-22

-rw-rw-r-- 1 www www 130 Jun 29 06:23 web-demo__2017-06-30-14-30-22.tar.gz

drwxrwxr-x 2 www www 4096 Jun 30 14:31 web-demo__2017-06-30-14-31-53

-rw-rw-r-- 1 www www 219 Jun 29 06:24 web-demo__2017-06-30-14-31-53.tar.gz

drwxrwxr-x 4 www www 4096 Jun 30 14:59 web-demo_75463f1b_2017-06-30-14-59-58

-rw-rw-r-- 1 www www 1236456 Jun 29 06:52 web-demo_75463f1b_2017-06-30-14-59-58.tar.gz

drwxrwxr-x 4 www www 4096 Jun 30 15:05 web-demo_75463f1b_2017-06-30-15-05-57

-rw-rw-r-- 1 www www 1236450 Jun 29 06:58 web-demo_75463f1b_2017-06-30-15-05-57.tar.gz

drwxrwxr-x 4 www www 4096 Jul 10 14:01 web-demo_75463f1b_2017-07-10-14-01-34

-rw-rw-r-- 1 www www 1236446 Jul 10 14:01 web-demo_75463f1b_2017-07-10-14-01-34.tar.gz

drwxrwxr-x 4 www www 4096 Jun 30 15:18 web-demo_78869143_2017-06-30-15-18-29

-rw-rw-r-- 1 www www 1236465 Jun 29 07:11 web-demo_78869143_2017-06-30-15-18-29.tar.gz

drwxrwxr-x 4 www www 4096 Jul 10 14:00 web-demo_78869143_2017-07-10-14-00-35

-rw-rw-r-- 1 www www 1236453 Jul 10 14:00 web-demo_78869143_2017-07-10-14-00-35.tar.gz

drwxrwxr-x 3 www www 4096 Jun 30 14:14 web-demo_91d09cc2_2017-06-30-14-14-42

-rw-rw-r-- 1 www www 1236371 Jun 29 06:06 web-demo_91d09cc2_2017-06-30-14-14-42.tar.gz

drwxrwxr-x 3 www www 4096 Jun 30 14:15 web-demo_91d09cc2_2017-06-30-14-15-16

-rw-rw-r-- 1 www www 1236382 Jun 29 06:08 web-demo_91d09cc2_2017-06-30-14-15-16.tar.gz

drwxrwxr-x 4 www www 4096 Jul 10 14:08 web-demo_9a43cf55_2017-07-10-14-08-46

-rw-rw-r-- 1 www www 1233708 Jul 10 14:08 web-demo_9a43cf55_2017-07-10-14-08-46.tar.gz

drwxrwxr-x 4 www www 4096 Jun 30 15:21 web-demo_b8f3be43_2017-06-30-15-21-55

-rw-rw-r-- 1 www www 1236454 Jun 29 07:14 web-demo_b8f3be43_2017-06-30-15-21-55.tar.gz

drwxrwxr-x 4 www www 4096 Jul 10 12:34 web-demo_b8f3be43_2017-07-10-12-34-00

-rw-rw-r-- 1 www www 1236462 Jul 10 12:34 web-demo_b8f3be43_2017-07-10-12-34-00.tar.gz

drwxrwxr-x 4 www www 4096 Jun 30 14:57 web-demo_dcfb44f0_2017-06-30-14-57-10

-rw-rw-r-- 1 www www 1236447 Jun 29 06:50 web-demo_dcfb44f0_2017-06-30-14-57-10.tar.gz

1.4.2:回滚任意版本就将版本的参数传递给脚本,脚本会将web-demo的链接重新指向传递的版本(参数),比如我要回滚到web-demo_78869143_2017-06-30-15-18-29这个版本,则jenkins的配置为:

d5df53f477c451d0610bc53b69d61e89.png

1.4.3:在jenkins执行回滚:

1.4.4:执行回滚的信息:

502f682cdb5ff3c07d9d1826d14bd823.png

1.4.5:访问web界面测试任意版本回滚是否成功:

0bc8556ace2479e6230333d500225285.png0b1331709591d260c1c78e86d0c51c18.png

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值