【流水线入门】解放运维,一键部署

简介: 阿里云云效流水线,降低运维工作量,让运维解放出来做更有意义的事

本文只涉及到云效流水线的配置及部署,前提是linux服务器已经装好java环境。
云效提供任务、日程、统计、流水线等功能,丰富多样。由于文章篇幅有限,本文只涉及到流水线功能。

准备项目

小白写一个简单的springboot项目,功能是打印请求IP,写好后上传到github远程仓库(也可以是Gitlab,自建Gitlab,码云,Codeup等外网可以访问的代码托管平台)。这里注意一定要有空URL的映射,因为启动脚本要检测项目是否启动成功。或者可以自行修改启动脚本

@RestController
@RequestMapping("/")
public class ShowIpController {

    @GetMapping("/")
    public String index(HttpServletRequest request){
        String ip = getIp(request);
        System.out.println("ip:[" + ip + "]");
        return "thanks";
    }

    @GetMapping("/ip/show")
    public String show(HttpServletRequest request){
        String ip = getIp(request);
        System.out.println("ip:[" + ip + "]");
        return "thanks";
    }

    private static String getIp(HttpServletRequest request) {
        if (request == null) {
            return "unknown";
        }
        String ip = request.getHeader("x-forwarded-for");
        if (ip == null || ip.length() == 0 || "unknown".equalsIgnoreCase(ip)) {
            ip = request.getHeader("Proxy-Client-IP");
        }
        if (ip == null || ip.length() == 0 || "unknown".equalsIgnoreCase(ip)) {
            ip = request.getHeader("X-Forwarded-For");
        }
        if (ip == null || ip.length() == 0 || "unknown".equalsIgnoreCase(ip)) {
            ip = request.getHeader("WL-Proxy-Client-IP");
        }
        if (ip == null || ip.length() == 0 || "unknown".equalsIgnoreCase(ip)) {
            ip = request.getHeader("X-Real-IP");
        }

        if (ip == null || ip.length() == 0 || "unknown".equalsIgnoreCase(ip)) {
            ip = request.getRemoteAddr();
        }

        return "0:0:0:0:0:0:0:1".equals(ip) ? "127.0.0.1" : ip;
    }
}

创建启动脚本 deploy.sh 并放入resources目录下,下文能用得到,有了启动脚本我们就不需要手动去启动构建的项目

#!/bin/bash

# 修改APP_NAME为云效上的应用名
APP_NAME=ip
VERSION=1.0


PROG_NAME=$0
ACTION=$1
APP_START_TIMEOUT=20    # 等待应用启动的时间
APP_PORT=8080          # 应用端口
HEALTH_CHECK_URL=http://127.0.0.1:${APP_PORT}  # 应用健康检查URL
APP_HOME=/home/devops/${APP_NAME} # 从package.tgz中解压出来的jar包放到这个目录下
JAR_NAME=${APP_HOME}/${APP_NAME}-${VERSION}.jar # jar包的名字
JAVA_OUT=${APP_HOME}/logs/start.log  #应用的启动日志

# 创建出相关目录
mkdir -p ${APP_HOME}
mkdir -p ${APP_HOME}/logs
usage() {
    echo "Usage: $PROG_NAME {start|stop|restart}"
    exit 2
}

health_check() {
    exptime=0
    echo "checking ${HEALTH_CHECK_URL}"
    while true
        do
            status_code=`/usr/bin/curl -L -o /dev/null --connect-timeout 5 -s -w %{http_code}  ${HEALTH_CHECK_URL}`
            if [ "$?" != "0" ]; then
               echo -n -e "\rapplication not started"
            else
                echo "code is $status_code"
                if [ $status_code -eq 200 ];then
                    break
                fi
            fi
            sleep 1
            ((exptime++))

            echo -e "\rWait app to pass health check: $exptime..."

            if [ $exptime -gt ${APP_START_TIMEOUT} ]; then
                echo 'app start failed'
               exit 1
            fi
        done
    echo "check ${HEALTH_CHECK_URL} success"
}
start_application() {
    echo "starting java process"
    nohup java -jar ${JAR_NAME} > ${JAVA_OUT} 2>&1 &
    echo "started java process"
}

stop_application() {
   checkjavapid=`ps -ef | grep java | grep ${APP_NAME} | grep -v grep |grep -v 'deploy.sh'| awk '{print$2}'`
   
   if [[ ! $checkjavapid ]];then
      echo -e "\rno java process"
      return
   fi

   echo "stop java process"
   times=60
   for e in $(seq 60)
   do
        sleep 1
        COSTTIME=$(($times - $e ))
        checkjavapid=`ps -ef | grep java | grep ${APP_NAME} | grep -v grep |grep -v 'deploy.sh'| awk '{print$2}'`
        if [[ $checkjavapid ]];then
            kill -9 $checkjavapid
            echo -e  "\r        -- stopping java lasts `expr $COSTTIME` seconds."
        else
            echo -e "\rjava process has exited"
            break;
        fi
   done
   echo ""
}
start() {
    start_application
    health_check
}
stop() {
    stop_application
}
case "$ACTION" in
    start)
        start
    ;;
    stop)
        stop
    ;;
    restart)
        stop
        start
    ;;
    *)
        usage
    ;;
esac

开始配置

新建项目

登录阿里云云效。进到云效管理后台后,点击新建项目,云效提供产品研发,游戏研发等多种企业模板。此处我选择了空白模板,大家可以根据自己的需要选择不同的模板。输入项目名ip,选择ip项目分组,点击创建即可创建成功,操作十分简单。
image.png

新建流水线

创建完成后就自动跳转到了我们刚刚创建的ip项目,切到"流水线"TAB,点击新建流水线。选择第一项`
Java · 构建、部署到阿里云ECS/自有主机

![image.png](https://ucc.alicdn.com/pic/developer-ecology/aa42470e6d5342fdaf137e62b42a1a9e.png)

##关联远程仓库并设置代码源
代码源选择你已经上传的外网远程仓库,我关联github,完成授权后依次选择命名空间,仓库,填入分支名"master"。
![image.png](https://ucc.alicdn.com/pic/developer-ecology/dd4fea2290cd48f8a55fd79931f0dc35.png)

##编辑java构建上传
单工程项目直接默认的就可以了,如果是有父子层级的项目就需要配置 ***构建物上传-打包路径***,有几个需要发布的工程就需要几条打包路径,需要注意的是开头没有/,结尾有/。例如app/target/
还可以选择任务插件,将成功,失败等状态直接发送到你自定义的邮箱或者是钉钉群
![image.png](https://ucc.alicdn.com/pic/developer-ecology/99d05d691bca484a9e2e5b237b39414e.png)
##编辑主机部署
制品选择我们刚刚构建时创建的制品,新建主机组,选择 ***自有主机*** ,将页面中的命令复制到你的远程主机上运行一下,等待安装成功即可。注意此命令15分钟有效,超出后需要重新获取。
![image.png](https://ucc.alicdn.com/pic/developer-ecology/a5caa7dbc7b9435ba2fb186f6d74e1c6.png)
这样我们的列表中就出现了一台机器,这台机器已经与这条流水线关联了。代码会自动部署到这台机器上。主机组就选择刚刚创建的主机,你想把构建好的jar包放在哪,下载路径就填入你想放的路径,这里我填入*/home/devops/ip/ip.tgz*。部署脚本填入解压缩命令及运行 *启动脚本*
*tar zxvf /home/devops/ip/ip.tgz -C /home/devops/ip/
sh /home/devops/ip/classes/deploy.sh start*
[关于部署脚本例子的详细解释](https://thoughts.aliyun.com/sharespace/5e86a419546fd9001aee81f2/docs/5e86a416546fd9001aee81b9)
至此点击右上角保存并运行,然后静静的等待项目运行即可。

![image.png](https://ucc.alicdn.com/pic/developer-ecology/f826243431ae4d22943427112c933444.png)
##运行成功后效果
![image.png](https://ucc.alicdn.com/pic/developer-ecology/3dfdd1659d3f41edb0ecf20d6f645df2.png)


----
【云效官网】https://www.aliyun.com/product/yunxiao?channel=zhibo 
相关实践学习
2分钟自动化部署人生模拟器
本场景将带你借助云效流水线Flow实现人生模拟器小游戏的自动化部署
SVN版本控制系统
SVN是现在软件开发之中的主流软件版本控制工具,在工作之中利用SVN可以有效的解决多人开发的代码管理问题,本课程将为读者讲解SVN服务器的配置以及基于MyEclipse的SVN客户端插件的配置与使用,并且在讲解之中着重讲解了冲突的产生于解决。
相关文章
|
域名解析 Cloud Native jenkins
【Drone+Gitlab】一条龙服务,直接起飞 — 从介绍->部署->配置->写.drone.yml流水线+常见的报错解决
gitlab+drone部署安装,编写.drone.yml流水线 drone是一个持续集成化工具,gitlab是一个代码仓库,.drone.yml流水线编写 fatal: unable to access,could not resolve host 克隆地址连接不上(修改默认clone克隆),没有Trusted选项,启动drone-server时添加(--env=DRONE_USER_CREATE=username:root,admin:true) .drone.yml文件中sed命令报错
1806 0
【Drone+Gitlab】一条龙服务,直接起飞 — 从介绍->部署->配置->写.drone.yml流水线+常见的报错解决
|
3月前
|
Serverless
云效流水线部署函数计算任务时出现了错误
【1月更文挑战第18天】【1月更文挑战第90篇】云效流水线部署函数计算任务时出现了错误
41 1
|
4月前
|
测试技术 Serverless 持续交付
通过云效流水线进行部署任务
通过云效流水线进行部署任务
65 1
|
4月前
|
Java jenkins 持续交付
Jenkins Pipeline 流水线方式部署 SpringBoot 项目2
Jenkins Pipeline 流水线方式部署 SpringBoot 项目
129 0
|
4月前
|
jenkins Java 持续交付
Jenkins Pipeline 流水线方式部署 SpringBoot 项目1
Jenkins Pipeline 流水线方式部署 SpringBoot 项目
175 0
|
9月前
|
Kubernetes JavaScript 前端开发
k8s KubeSphere流水线部署Vue前端项目 详细教程
k8s KubeSphere流水线部署Vue前端项目 详细教程
|
9月前
|
Kubernetes 数据可视化 Java
k8s KubeSphere流水线部署SpringBoot后端项目 详细教程
k8s KubeSphere流水线部署SpringBoot后端项目 详细教程
|
9月前
|
存储 关系型数据库 MySQL
rancher服务部署之DevOps流水线(一)—基础数据库服务部署及rancher相关配置私服信息
rancher服务部署之DevOps流水线(一)—基础数据库服务部署及rancher相关配置私服信息
|
10月前
|
Kubernetes 数据可视化 网络协议
【Jenkins-初识篇】容器快速部署Jenkins,创建流水线-Blue Ocean可视化界面展示
【Jenkins-初识篇】容器快速部署Jenkins,创建流水线-Blue Ocean可视化界面展示
366 0
|
SQL Kubernetes Java
使用流水线插件实现持续集成、持续部署
本文将介绍使用流水线插件部署 RuoYi SpringBoot 项目,并实现提交代码后自动构建、自动部署。