首页 > 编程语言 >基于Jenkins+Docker的自动化部署实践——整合Git与Python脚本实现远程部署

基于Jenkins+Docker的自动化部署实践——整合Git与Python脚本实现远程部署

时间:2024-12-24 14:10:50浏览次数:10  
标签:INFO 10 Git 部署 JenkinsDeployer self Python url config

环境说明:

  • Ubuntu:v24.04.1 LTS
  • Jekins:v2.491
  • Docker:v27.4.0
  • Gogs:v0.14.0 - 可选。可以选择Github,Gitlab或者Gitea等Git仓库,不限仓库类型
  • 1Panel: v1.10.21-lts - 可选。这里主要用于查看和管理Docker容器

Jenkins实现参数化构建

这里通过Docker进行安装

【系统管理】【插件管理】,安装“Publish Over SSH”

【系统管理】【系统配置】,配置“SSH Servers”

填写配置信息后点击“Test Configuration​”,显示“Success”说明配置成功,保存配置

新建任务

新建“test”任务,选择“流水线”

脚本编写如下:

pipeline {
    agent any
  
    parameters {
        string(name: 'GIT_REPO_URL', description: 'Git仓库地址', trim: true)
        string(name: 'BRANCH', defaultValue: 'main', description: '分支名称', trim: true)
        choice(name: 'DEPLOY_ENV', choices: ['deploy'], description: '部署环境')  // 改为 deploy 匹配 SSH 配置
    }
  
    environment {
        APP_NAME = 'myapp'
        IMAGE_TAG = "${BUILD_NUMBER}"
        REMOTE_DIR = '/opt'  // 修改为你配置的远程目录
    }
  
    stages {
        stage('拉取代码') {
            steps {
                git url: "${params.GIT_REPO_URL}", branch: "${params.BRANCH}"
            }
        }
      
        stage('远程构建和部署') {
            steps {
                script {
                    sshPublisher(
                        publishers: [
                            sshPublisherDesc(
                                configName: 'deploy',  // 修改为你配置的 Name
                                verbose: true,
                                transfers: [
                                    sshTransfer(
                                        sourceFiles: "**/*",
                                        remoteDirectory: "${APP_NAME}-${BUILD_NUMBER}",
                                        execCommand: """
                                            cd ${REMOTE_DIR}/${APP_NAME}-${BUILD_NUMBER}
                                          
                                            # 构建和部署
                                            docker build -t ${APP_NAME}:${IMAGE_TAG} .
                                            docker stop ${APP_NAME} || true
                                            docker rm ${APP_NAME} || true
                                            docker run -d --name ${APP_NAME} \
                                                -p 8880:5000 \
                                                --restart unless-stopped \
                                                ${APP_NAME}:${IMAGE_TAG}
                                          
                                            # 清理
                                            cd ..
                                            rm -rf ${APP_NAME}-${BUILD_NUMBER}
                                            docker system prune -f
                                        """
                                    )
                                ]
                            )
                        ]
                    )
                }
            }
        }
      
        stage('健康检查') {
            steps {
                sleep 15
                sshPublisher(
                    publishers: [
                        sshPublisherDesc(
                            configName: 'deploy',
                            transfers: [
                                sshTransfer(
                                    execCommand: '''
                                        max_attempts=5
                                        attempt=1
                                        while [ $attempt -le $max_attempts ]; do
                                            response=$(curl -s -o /dev/null -w "%{http_code}" http://localhost:8880/health)
                                            if [ "$response" = "200" ]; then
                                                echo "Health check succeeded"
                                                exit 0
                                            fi
                                            echo "Attempt $attempt failed, waiting... (Status code: $response)"
                                            sleep 10
                                            attempt=$((attempt + 1))
                                        done
                                        echo "Health check failed after $max_attempts attempts"
                                        exit 1
                                    '''
                                )
                            ]
                        )
                    ]
                )
            }
        }
    }
  
    post {
        success {
            echo "部署成功: ${APP_NAME}:${IMAGE_TAG}"
        }
        failure {
            echo "部署失败"
        }
        cleanup {
            cleanWs()
        }
    }
}

打开【Build with Parameters】填写Git仓库地址和仓库分支,点击【Build】

打开【Console Output】查看构建日志

日志显示如下,代表构建完成

...
[Pipeline] step
SSH: Connecting from host [94e37d92d688]
SSH: Connecting with configuration [deploy] ...
SSH: EXEC: completed after 200 ms
SSH: Disconnecting configuration [deploy] ...
SSH: Transferred 0 file(s)
[Pipeline] }
[Pipeline] // stage
[Pipeline] stage
[Pipeline] { (Declarative: Post Actions)
[Pipeline] echo
部署成功: myapp:27
[Pipeline] cleanWs
[WS-CLEANUP] Deleting project workspace...
[WS-CLEANUP] Deferred wipeout is used...
[WS-CLEANUP] done
[Pipeline] }
[Pipeline] // stage
[Pipeline] }
[Pipeline] // withEnv
[Pipeline] }
[Pipeline] // node
[Pipeline] End of Pipeline
Finished: SUCCESS

这时在部署的主机上可以看到自动化部署的Docker容器

健康检查

由于在项目docker_net8_webapi_fortran中已经实现/health接口,所以在流水线中健康检查逻辑如下,这里支持重试机制

stage('健康检查') {
    steps {
        sleep 15
        sshPublisher(
            publishers: [
                sshPublisherDesc(
                    configName: 'deploy',
                    transfers: [
                        sshTransfer(
                            execCommand: '''
                                max_attempts=5
                                attempt=1
                                while [ $attempt -le $max_attempts ]; do
                                    response=$(curl -s -o /dev/null -w "%{http_code}" http://localhost:8880/health)
                                    if [ "$response" = "200" ]; then
                                        echo "Health check succeeded"
                                        exit 0
                                    fi
                                    echo "Attempt $attempt failed, waiting... (Status code: $response)"
                                    sleep 10
                                    attempt=$((attempt + 1))
                                done
                                echo "Health check failed after $max_attempts attempts"
                                exit 1
                            '''
                        )
                    ]
                )
            ]
        )
    }
}

参数化配置Python脚本实现远程部署

刚刚我们通过手动填写参数的方式完成项目自动化构建,进一步,可以使用python脚本远程进行触发构建过程,并且实现参数化配置

import requests
import json
import time
import yaml
import logging
import urllib3
from typing import Optional, Dict, Any
from urllib.parse import quote
from tenacity import retry, stop_after_attempt, wait_exponential

# 配置日志
logging.basicConfig(
    level=logging.INFO,
    format='%(asctime)s - %(name)s - %(levelname)s - %(message)s',
    handlers=[
        logging.FileHandler('deployment.log'),
        logging.StreamHandler()
    ]
)
logger = logging.getLogger('JenkinsDeployer')

class Config:
    def __init__(self, config_file: str = "config.yaml"):
        with open(config_file, 'r', encoding='utf-8') as f:
            self.config = yaml.safe_load(f)
      
        # Jenkins配置
        self.jenkins = self.config['jenkins']
        self.deployment = self.config['deployment']

class JenkinsDeployer:
    def __init__(self, config: Config):
        self.config = config
        self.JENKINS_URL = config.jenkins['url']
        self.USER = config.jenkins['user']
        self.API_TOKEN = config.jenkins['token']
        self.JOB_NAME = config.jenkins['job_name']

    @retry(
        stop=stop_after_attempt(3),
        wait=wait_exponential(multiplier=1, min=4, max=10)
    )
    def _make_request(self, method: str, url: str, **kwargs) -> requests.Response:
        """通用的请求方法,带重试机制"""
        logger.info(f"发送 {method} 请求到 {url}")
        response = requests.request(
            method,
            url,
            auth=(self.USER, self.API_TOKEN),
            timeout=30,
            verify=False,
            **kwargs
        )
        response.raise_for_status()
        return response

    def trigger_deploy(self, 
                      git_repo_url: str, 
                      branch: str = "main", 
                      deploy_env: str = "deploy",
                      wait: bool = True) -> Dict[str, Any]:
        """触发Jenkins部署"""
        try:
             # 参数不进行 URL 编码
            params = {
                "GIT_REPO_URL": git_repo_url,
                "BRANCH": branch, 
                "DEPLOY_ENV": deploy_env
            }
          
            # 构建 URL
            url = f"{self.JENKINS_URL}/job/{self.JOB_NAME}/buildWithParameters"
          
            logger.info(f"触发构建: {url}")
            logger.info(f"参数: {json.dumps(params, indent=2, ensure_ascii=False)}")
          
            # 发送请求
            response = self._make_request('POST', url, params=params)
          
            if response.status_code == 201:
                queue_url = response.headers.get('Location')
                if not queue_url:
                    return {"error": "未获取到队列URL"}
              
                logger.info(f"构建队列 URL: {queue_url}")
                build_number = self._get_build_number(queue_url)
              
                if build_number:
                    logger.info(f"部署已触发,构建号: {build_number}")
                  
                    if wait:
                        return self.wait_for_completion(build_number)
                    return {"build_number": build_number, "status": "STARTED"}
                else:
                    return {"error": "未能获取构建号"}
          
            return {
                "error": f"触发失败: {response.status_code}",
                "details": response.text
            }
          
        except Exception as e:
            logger.error(f"部署触发失败: {str(e)}", exc_info=True)
            return {"error": f"部署触发失败: {str(e)}"}

    def _get_build_number(self, queue_url: str) -> Optional[int]:
        """获取构建号"""
        max_attempts = 10
        for attempt in range(max_attempts):
            try:
                response = self._make_request('GET', f"{queue_url}api/json")
              
                logger.info(f"尝试获取构建号 ({attempt + 1}/{max_attempts})")
                logger.debug(f"队列响应: {response.text}")
              
                data = response.json()
                if "executable" in data and "number" in data["executable"]:
                    return data["executable"]["number"]
                elif "why" in data:
                    logger.info(f"构建等待中: {data['why']}")
                  
                time.sleep(2)
            except Exception as e:
                logger.error(f"获取构建号失败 ({attempt + 1}/{max_attempts}): {e}")
        return None

    def get_build_status(self, build_number: int) -> Dict[str, Any]:
        """获取构建状态"""
        url = f"{self.JENKINS_URL}/job/{self.JOB_NAME}/{build_number}/api/json"
        try:
            response = self._make_request('GET', url)
            build_info = response.json()
          
            return {
                "number": build_info["number"],
                "result": build_info.get("result", "IN_PROGRESS"),
                "url": build_info["url"],
                "duration": build_info["duration"],
                "timestamp": build_info["timestamp"]
            }
        except Exception as e:
            logger.error(f"获取构建状态失败: {e}", exc_info=True)
            return {"error": f"获取状态失败: {str(e)}"}

    def wait_for_completion(self, build_number: int, timeout: int = 300) -> Dict[str, Any]:
        """等待部署完成"""
        start_time = time.time()
        while time.time() - start_time < timeout:
            status = self.get_build_status(build_number)
          
            if "error" in status:
                return status
          
            if status["result"] and status["result"] != "IN_PROGRESS":
                return status
          
            logger.info(f"部署进行中... ({int(time.time() - start_time)}s)")
            time.sleep(10)
          
        return {"error": "部署超时"}

    def check_deployment_health(self, host: str, port: int, max_attempts: int = 5) -> bool:
        """检查部署的应用是否健康"""
        health_url = f"http://{host}:{port}/health"
      
        for attempt in range(max_attempts):
            try:
                response = requests.get(health_url, timeout=5)
                if response.status_code == 200:
                    logger.info(f"健康检查成功 (尝试 {attempt + 1}/{max_attempts})")
                    return True
            except Exception as e:
                logger.warning(f"健康检查失败 (尝试 {attempt + 1}/{max_attempts}): {e}")
          
            if attempt < max_attempts - 1:
                time.sleep(10)
      
        return False

def main():
    # 禁用 SSL 警告
    urllib3.disable_warnings()
  
    try:
        # 初始化配置
        config = Config("config.yaml")
        deployer = JenkinsDeployer(config)
      
        # 触发部署
        result = deployer.trigger_deploy(
            git_repo_url=config.deployment['git_repo'],
            branch=config.deployment['branch'],
            deploy_env="deploy",
            wait=True
        )
      
        if "error" in result:
            logger.error(f"部署失败: {result['error']}")
            return
      
        # 执行健康检查
        health_config = config.deployment['health_check']
        is_healthy = deployer.check_deployment_health(
            host=health_config['host'],
            port=health_config['port'],
            max_attempts=health_config['max_attempts']
        )
      
        if is_healthy:
            logger.info("部署成功且应用程序运行正常")
        else:
            logger.warning("部署可能成功但健康检查失败")
          
    except Exception as e:
        logger.error(f"部署过程出错: {str(e)}", exc_info=True)

if __name__ == "__main__":
    main()

config.yaml配置文件如下:

jenkins:
  url: "http://192.168.1.140:8563"
  user: "admin"
  token: "1144e4584a109badf5051a42a960aef11d"
  job_name: "test"

deployment:
  git_repo: "http://192.168.1.140:10880/root/docker_net8_webapi_fortran.git"
  branch: "master"
  health_check:
    host: "192.168.1.140"
    port: 8880
    max_attempts: 5

运行python脚本,日志结果如下:

$ python .\main.py
2024-12-24 10:22:06,538 - JenkinsDeployer - INFO - 触发构建: http://192.168.1.140:8563/job/test/buildWithParameters
2024-12-24 10:22:06,539 - JenkinsDeployer - INFO - 参数: {
  "GIT_REPO_URL": "http://192.168.1.140:10880/root/docker_net8_webapi_fortran.git",
  "BRANCH": "master",
  "DEPLOY_ENV": "deploy"
}
2024-12-24 10:22:06,539 - JenkinsDeployer - INFO - 发送 POST 请求到 http://192.168.1.140:8563/job/test/buildWithParameters
2024-12-24 10:22:06,636 - JenkinsDeployer - INFO - 构建队列 URL: http://192.168.1.140:8563/queue/item/62/
2024-12-24 10:22:06,637 - JenkinsDeployer - INFO - 发送 GET 请求到 http://192.168.1.140:8563/queue/item/62/api/json
2024-12-24 10:22:06,788 - JenkinsDeployer - INFO - 尝试获取构建号 (1/10)
2024-12-24 10:22:06,789 - JenkinsDeployer - INFO - 构建等待中: In the quiet period. Expires in 4.8 sec
2024-12-24 10:22:08,790 - JenkinsDeployer - INFO - 发送 GET 请求到 http://192.168.1.140:8563/queue/item/62/api/json
2024-12-24 10:22:08,921 - JenkinsDeployer - INFO - 尝试获取构建号 (2/10)
2024-12-24 10:22:08,921 - JenkinsDeployer - INFO - 构建等待中: In the quiet period. Expires in 2.7 sec
2024-12-24 10:22:10,929 - JenkinsDeployer - INFO - 发送 GET 请求到 http://192.168.1.140:8563/queue/item/62/api/json
2024-12-24 10:22:11,029 - JenkinsDeployer - INFO - 尝试获取构建号 (3/10)
2024-12-24 10:22:11,029 - JenkinsDeployer - INFO - 构建等待中: In the quiet period. Expires in 0.6 sec
2024-12-24 10:22:13,032 - JenkinsDeployer - INFO - 发送 GET 请求到 http://192.168.1.140:8563/queue/item/62/api/json
2024-12-24 10:22:13,207 - JenkinsDeployer - INFO - 尝试获取构建号 (4/10)
2024-12-24 10:22:13,207 - JenkinsDeployer - INFO - 部署已触发,构建号: 26
2024-12-24 10:22:13,208 - JenkinsDeployer - INFO - 发送 GET 请求到 http://192.168.1.140:8563/job/test/26/api/json
2024-12-24 10:22:13,357 - JenkinsDeployer - INFO - 部署进行中... (0s)
2024-12-24 10:22:23,364 - JenkinsDeployer - INFO - 发送 GET 请求到 http://192.168.1.140:8563/job/test/26/api/json
2024-12-24 10:22:23,640 - JenkinsDeployer - INFO - 部署进行中... (10s)
2024-12-24 10:22:33,644 - JenkinsDeployer - INFO - 发送 GET 请求到 http://192.168.1.140:8563/job/test/26/api/json
2024-12-24 10:22:33,907 - JenkinsDeployer - INFO - 健康检查成功 (尝试 1/5)
2024-12-24 10:22:33,908 - JenkinsDeployer - INFO - 部署成功且应用程序运行正常

参考

标签:INFO,10,Git,部署,JenkinsDeployer,self,Python,url,config
From: https://www.cnblogs.com/vinciyan/p/18627291

相关文章

  • 雷池社区版 SSL 证书安装部署
    雷池社区版SSL证书安装部署示例说明域名:rivers.chaitin.cnWeb服务为Nginx安装SSL证书前,请在Nginx服务器上开启HTTPS默认端口443,避免证书安装后无法启用HTTPS。证书安装在控制台服务器类型选择Nginx(请注意基于自己Web服务类型进行选择),并点击下载文件解压。......
  • 用 Python 构建一个简单的爬虫:抓取豆瓣电影信息
    用Python构建一个简单的爬虫:抓取豆瓣电影信息爬虫是Python开发者的必备技能之一。本文将带你从零开始,构建一个简单的爬虫,用来抓取豆瓣电影Top250的信息,包括电影名称、评分和简介。目录项目简介环境配置与依赖安装解析目标网站结构编写爬虫代码保存数据到CSV文......
  • [机器人机构学]课设四:moveit配置及python接口控制
    运行环境:ROS操作系统需要安装:VMwareWorkstationProROS操作系统不想自己安装ROS的看这里系列文章目录[机器人机构学]课设一:通过杆件四大参数确定机械臂模型[机器人机构学]课设二:三维建模及爆炸图动画制作[机器人机构学]课设三:SOLIDWORKS模型转URDF[机器人机构......
  • Springboot进口零食销售网站74r3o(程序+源码+数据库+调试部署+开发环境)
    本系统(程序+源码+数据库+调试部署+开发环境)带论文文档1万字以上,文末可获取,系统界面在最后面。系统程序文件列表用户,零食信息,类型开题报告内容研究背景随着互联网技术的飞速发展和消费者购物习惯的深刻变革,电子商务已成为推动全球经济增长的重要力量。进口零食作为日常消......
  • Springboot紧急自救知识教学与交流平台9c75u(程序+源码+数据库+调试部署+开发环境)
    本系统(程序+源码+数据库+调试部署+开发环境)带论文文档1万字以上,文末可获取,系统界面在最后面。系统程序文件列表用户,灾害类型,历史案例,教学课程,课程购买,紧急通知开题报告内容一、课题来源及研究目的和意义在现代社会,自然灾害与突发事件频发,公众对于紧急自救知识的需求......
  • ssm九价预约t4s1p程序+源码+数据库+调试部署+开发环境
    本系统(程序+源码+数据库+调试部署+开发环境)带论文文档1万字以上,文末可获取,系统界面在最后面。系统程序文件列表开题报告内容一、项目背景近年来,九价HPV疫苗因其高效预防宫颈癌等恶性肿瘤而受到广泛关注。然而,由于疫苗供应紧张,预约接种过程繁琐,使得许多有接种需求的人群面......
  • ssm教职工新冠疫苗预约x0i30--(程序+源码+数据库+调试部署+开发环境)
    本系统(程序+源码+数据库+调试部署+开发环境)带论文文档1万字以上,文末可获取,系统界面在最后面。系统程序文件列表开题报告内容一、项目背景随着新冠疫情的全球蔓延,疫苗接种已成为控制疫情、保障教职工健康安全的重要手段。然而,传统的疫苗接种预约方式存在诸多不便,如信息不透......
  • Python机器学习笔记(十一、特征提取)
    特征提取PCA的另一个应用是特征提取。特征提取背后的思想是,可以找到一种数据表示,比给定的原始表示更适合于分析。特征提取很有用,它的一个很好的应用实例就是图像。图像由像素组成,通常存储为红绿蓝(RGB)强度。图像中的对象通常由上千个像素组成,它们只有放在一起才有意义。现在......
  • python web知识点梳理
    目录1、第1章Django概述(1)环境搭建:需要安装django,使用国内镜像(2)创建项目和应用2、第2章路由系统(1)路由工作原理:(2)内置路由转换器(3)路由分发:include函数(4)向视图传递额外参数(5)命名空间①URL命名和reverse解析函数②应用命名空间,app_name属性3、第3章模型(1)定义和使......
  • (2024最新毕设合集)基于SpringBoot的小说在线阅读网咖+86615|可做计算机毕业设计JAVA、P
    目 录摘要1绪论1.1 选题背景1.2研究内容1.3本文的组织结构2相关技术介绍2.1MySQL数据库2.2Java编程语言2.3SpringBoot框架介绍3 系统需求分析与设计3.1可行性分析3.1.1技术可行性分析3.1.2经济可行性分析3.1.3法律可行性分析3.2需......