精品欧美一区二区三区在线观看 _久久久久国色av免费观看性色_国产精品久久在线观看_亚洲第一综合网站_91精品又粗又猛又爽_小泽玛利亚一区二区免费_91亚洲精品国偷拍自产在线观看 _久久精品视频在线播放_美女精品久久久_欧美日韩国产成人在线

基于Jenkins Pipeline構建企業級CI/CD

開發 前端
Lark Notice Plugin 是一個用于 Jenkins 的構建通知插件,可以將 Jenkins構建過程以及結果通知推送到 Lark、飛書、釘釘 協作平臺。 可配置多個的通知時機,包括 構建啟動時、構建中斷、構建失敗、構建成功時、構建不穩定等。

案例介紹

本案例通過若依項目作為案例,通過Jenkins構建企業級CI/CD平臺。

若依服務列表:

  • ruoyi-auth
  • ruoyi-system
  • ruoyi-gateway
  • ruoyi-ui

若依環境列表:

  • DEV
  • UAT
  • PROD

環境準備工作:

  • nacos安裝并配置完成
  • MySQL部署完成并初始化
  • Redis部署完成
  • Harbor鏡像倉庫
  • Gitlab部署完成
  • Kubernetes部署完成
  • Ingress部署完成

設計思路

觸發構建設計:

本設計通過Jenkins Generic Webhook Trigger 插件實現了基于Webhook自動觸發流水線構建。

圖片圖片

流程說明:

  • 研發項目負責人代碼開發完成后進行合并代碼并生成Tag
  • Gitlab通過Webhook自動觸發Jenkins Pipeline構建

流水線設計:

圖片圖片

Jenkins流水線完整圖:

圖片圖片

自定義基礎鏡像

在實際企業環境中,基礎鏡像都會根據具體得需求定義適合自己得基礎鏡像。

定義Maven鏡像:

用于代碼構建編譯打包,會把Ruoyi相關依賴包打到基礎鏡像內,避免分層構建失敗。

# 拉取源代碼并切換分支
$ https://gitee.com/y_project/RuoYi-Cloud.git
$ git checkout v3.6.3
$ cd ..

# 定義Dockerfile
$ cat Dockerfile
FROM maven:3.8.6-openjdk-8
ADD  RuoYi-Cloud /opt/RuoYi-Cloud
RUN cd /opt/RuoYi-Cloud  && mvn clean install -DskipTests
RUN rm -rf /opt/RuoYi-Cloud

# 構建鏡像
$ docker build uhub.service.ucloud.cn/kubesre/maven:jdk8 .
$ docker push uhub.service.ucloud.cn/kubesre/maven:jdk8

定義Java基礎鏡像:

根據需求定義適合自己的基礎鏡像。通過變量傳遞讓配置變得更靈活!

# 創建個目錄
$ mkdir base && cd base

# 創建啟動腳本
$ cat docker-entrypoint.sh
#!/bin/sh
java -server -Xms$JVM_XMS -Xmx$JVM_XMX -XX:+PrintGCDetails -XX:+PrintGCDateStamps -Xloggc:/data/gc.log -XX:+HeapDumpOnOutOfMemoryError -XX:HeapDumpPath=/data/heapdump.hprof  -jar app.jar --server.port=$SERVICE_PORT --spring.profiles.active=$PROFILES_ACTIVE --spring.cloud.nacos.config.server-addr=$NACOS_ADDRESS --spring.cloud.nacos.config.namespace=$NACOS_NAMESPACE_ID --spring.cloud.nacos.config.username=$NACOS_USERNAME --spring.cloud.nacos.config.password=$NACOS_PASSWORD --spring.cloud.nacos.discovery.server-addr=$NACOS_ADDRESS --spring.cloud.nacos.discovery.namespace=$NACOS_NAMESPACE_ID --spring.cloud.nacos.discovery.username=$NACOS_USERNAME --spring.cloud.nacos.discovery.password=$NACOS_PASSWORD

# 創建down-nacos腳本
cat down-nacos.sh 
#!/bin/sh
ipTrue=false
java_service_ip=""
code=false

getPodIp() {
    java_service_ip=`ip a  | grep inet | grep -v inet6 | grep -v '127.0.0.1' | awk '{print $2}' | awk -F / '{print$1}'`
    grep -w "${java_service_ip}" /etc/hosts > /dev/null
    if [ $? -eq 0 ];then
      echo "get java service ip success"
      ipTrue=true
    else
      echo "get java service ip failed"
    fi
}

downService(){
    accessToken=`curl -s -X POST http://$NACOS_ADDRESS/nacos/v1/auth/users/login --form username=$NACOS_USERNAME --form password=$NACOS_PASSWORD|jq -r .accessToken`
    curl -s -X  PUT "$NACOS_ADDRESS/nacos/v1/ns/instance?language=zh-CN&accessToken=$accessToken&username=$NACOS_USERNAME&serviceName=$JAVA_SERVICE_NAME&ip=$java_service_ip&port=$SERVICE_PORT&enabled=false&namespaceId=$NACOS_NAMESPACE_ID"
    if [ "$code" = "ok" ];then
      echo "java service down from nacos success"
      code=true
    else
      echo "java service down from nacos failed"
    fi
}

start(){
  getPodIp
  if $ipTrue;then
    downService
    sleep 30
  else
    echo "down $JAVA_SERVICE_NAME failed" >> down_service.log
  fi   
}
start


# 定義Dockerfile
$ cat Dockerfile 
FROM  openjdk:8-jre
WORKDIR /data
COPY ./down-nacos.sh .
COPY ./docker-entrypoint.sh .
RUN chmod +x docker-entrypoint.sh && chmod +x  down-nacos.sh
ENTRYPOINT ["./docker-entrypoint.sh"]

# 構建鏡像
$ docker build uhub.service.ucloud.cn/kubesre/java-base:v8 .
$ docker push uhub.service.ucloud.cn/kubesre/java-base:v8

變量說明:

  • JVM_XMS:最小JVM堆棧內存
  • JVM_XMX:最大JVM堆棧內存
  • SERVICE_PORT:應用服務端口
  • NACOS_ADDRESS:Nacos地址
  • NACOS_USERNAME:Nacos用戶名
  • NACOS_PASSWORD:Nacos密碼
  • NACOS_NAMESPACE_ID:Nacos命名空間ID
  • PROFILES_ACTIVE:環境名稱

Dockerfile編寫

分層構建好處:

  • 不依賴本地環境
  • 減小容器鏡像大小

Java Dockerfile(分層構建):

FROM uhub.service.ucloud.cn/kubesre/maven:jdk8 AS build
COPY  src /opt/src/
COPY pom.xml /opt/
RUN  cd /opt/ && mvn clean install -DskipTests

FROM  uhub.service.ucloud.cn/kubesre/java-base:v8
# 復制jar文件到路徑
COPY  --from=build /opt/target/*.jar /data/app.jar

Vue Dockerfile(分層構建):

FROM node:16 AS builder

# 設置工作目錄
WORKDIR /usr/src/app

# 將項目文件復制到 Docker 鏡像中
COPY . .

# 安裝項目依賴
RUN npm install --registry=https://registry.npmmirror.com

# 構建靜態文件
RUN npm run build:prod

# 使用 Nginx 鏡像作為基礎鏡像,用于托管靜態文件
FROM nginx:stable-alpine

WORKDIR /home/ruoyi/projects/ruoyi-ui
# 將 VuePress 構建的靜態文件復制到 Nginx 的網站目錄
COPY --from=builder /usr/src/app/dist /home/ruoyi/projects/ruoyi-ui
COPY ./nginx/conf/nginx.conf /etc/nginx/nginx.conf

# 暴露 80 端口
EXPOSE 80

# 啟動 Nginx
CMD ["nginx", "-g", "daemon off;"]

Pipeline編寫

如下所有Pipeline文件,需要自行修改內容:

  • credentialsId
  • robot
  • 鏡像倉庫地址

如何查找credentialsId:

圖片圖片

如何查找robot:

圖片圖片

Java Pipeline:

pipeline {
    agent any
    triggers {
        GenericTrigger(
            genericVariables: [
                [key: 'ref', value: '$.ref'],  //獲取分支
                [key: 'user_username', value: '$.user_username'],     //獲取自動構建用戶名
                [key: 'GitRepository', value: '$.project.git_http_url'],  //獲取gitlab ssh項目地址
                [key: 'project', value: '$.project.name'],      //獲取項目名稱
                [key: 'repository', value: '$.repository.name'],
            ],
            token: "$JOB_NAME",
            causeString: 'Triggered on $branch',
            printContributedVariables: true,
            printPostContent: true,
            silentResponse: false,
        )
    }
    environment {
        // pipeline配置路徑
        pipeline_dir="/var/lib/jenkins/workspace/pipeline"
        // 項目版本
        Tag=sh(script: 'echo "${ref}" | awk -F"/" \'{print $3}\'', returnStdout: true).trim()
        // 項目名稱
        Project_Name="${project}"
        // 上一次版本
        Revsion_Prod=''
        //Depolyment名稱
        DeploymentName=''
        // 生產名稱空間
        Namespace_Prod=''
        // 灰度模式
        GrayHeaderMode=''
        // 灰度Depolyment名稱
        GrayDeploymentName=''
        // 灰度Service名稱
        GrayServiceName=''
        // 灰度Ingress名稱
        GrayIngressName=''
        // 是否灰度
        GrayEnable='yes'
    }
    options {
        // 表示保留10次構建歷史
        buildDiscarder(logRotator(numToKeepStr: '10'))
    }
    stages {
        stage('Pull Code') {
            // 拉取代碼
            steps {
                checkout([
                    $class: 'GitSCM', 
                    branches: [[name: "$ref"]],
                    doGenerateSubmoduleConfigurations: false,
                    extensions: [], 
                    userRemoteConfigs: [[
                        credentialsId: 'ac66550d-6999-485c-af3a-7e6189f765f0',
                        url: "$GitRepository"
                    ]]
                ])

                script{
                    currentBuild.displayName = "#${BUILD_NUMBER} - ${Project_Name} - ${Tag}"
                }
            }
        }
        //     // 代碼構建
        //   stage('Build Code') {
        //         steps {
        //              sh '/application/maven/bin/mvn -f pom.xml -s settings.xml clean package -DskipTests'
        //         }
        //     } 
        // 鏡像構建
        stage('Build Image') {
            steps {
                sh '''
                /usr/bin/docker build -t uhub.service.ucloud.cn/kubesre/$Project_Name:$Tag .
                /usr/bin/docker push  uhub.service.ucloud.cn/kubesre/$Project_Name:$Tag
                '''
            }
            post {
                success {
                    wrap([$class: 'BuildUser']) {
                        lark (
                            robot: "9f7c94cd-491e-4309-83b4-9290d01fc285",
                            type: "CARD",
                            title: "??  Jenkins 鏡像構建成功",
                            text: [
                                "?? **任務名稱**:[${JOB_NAME}](${JOB_URL})",
                                "?? **任務編號**:[${BUILD_DISPLAY_NAME}](${BUILD_URL})",
                                "?? **構建狀態**: <font color='green'>成功</font>",
                                "?? **鏡像版本**: $Tag",
                                "?? **鏡像倉庫**: uhub.service.ucloud.cn/kubesre/$Project_Name",
                                "?? **構建用時**: ${currentBuild.duration} ms",
                                "?? **執  行 者**: ${env.BUILD_USER}",
                                "<at id=all></at>"
                            ],
                            buttons: [
                                [
                                    title: "更改記錄",
                                    url: "${BUILD_URL}changes"
                                ],
                                [
                                    title: "控制臺",
                                    type: "danger",
                                    url: "${BUILD_URL}console"
                                ]
                            ]
                        )}
                }
            }
        }
        stage('DeployDev'){
            steps {
                echo "部署開發環境"
                script {
                    def userInput = input (
                        message: '確定要發布到DEV環境嗎?',
                        parameters:[
                            choice(name: '操作', choices: ['發布', '跳過'])
                        ],
                        ok: '確定',
                        submitter: 'admin',
                        submitterParameter: 'APPROVER'
                    )
                    if (userInput['操作'] == '發布'){
                        echo "部署Dev環境開始"
                        sh '''
                        echo $pipeline_dir
                        echo "打印編排文件詳細信息"
                        if [ -e "$pipeline_dir/dev/$Project_Name/deployment.yml" ]; then
                        cat $pipeline_dir/dev/$Project_Name/deployment.yml | sed  "s/TAG/${Tag}/g"
                        cat $pipeline_dir/dev/$Project_Name/deployment.yml | sed  "s/TAG/${Tag}/g" | /usr/bin/kubectl apply -f  -
                        fi

                        if [ -e "$pipeline_dir/dev/$Project_Name/service.yml" ]; then
                        cat $pipeline_dir/dev/$Project_Name/service.yml
                        cat $pipeline_dir/dev/$Project_Name/service.yml  | /usr/bin/kubectl apply -f  -
                        fi

                        if [ -e "$pipeline_dir/dev/$Project_Name/ingress.yml" ]; then
                        cat $pipeline_dir/dev/$Project_Name/ingress.yml 
                        cat $pipeline_dir/dev/$Project_Name/ingress.yml  | /usr/bin/kubectl apply -f  -
                        fi
                        '''
                    } else {
                        echo "不發布"
                    }
                }
            }
        } 
        stage('DeployUat'){
            steps {
                echo "部署測試環境"
                script {
                    def userInput = input (
                        message: '確定要發布到UAT環境嗎?',
                        parameters:[
                            choice(name: '操作', choices: ['發布', '跳過'])
                        ],
                        ok: '確定',
                        submitter: 'admin',
                        submitterParameter: 'APPROVER'
                    )
                    if (userInput['操作'] == '發布'){
                        echo "發布"
                        sh '''
                        echo $pipeline_dir
                        echo "打印編排文件詳細信息"

                        if [ -e "$pipeline_dir/uat/$Project_Name/deployment.yml" ]; then
                        cat $pipeline_dir/uat/$Project_Name/deployment.yml | sed  "s/TAG/${Tag}/g"
                        cat $pipeline_dir/uat/$Project_Name/deployment.yml | sed  "s/TAG/${Tag}/g" | /usr/bin/kubectl apply -f  -
                        fi

                        if [ -e "$pipeline_dir/uat/$Project_Name/service.yml" ]; then
                        cat $pipeline_dir/uat/$Project_Name/service.yml
                        cat $pipeline_dir/uat/$Project_Name/service.yml  | /usr/bin/kubectl apply -f  -
                        fi

                        if [ -e "$pipeline_dir/uat/$Project_Name/ingress.yml" ]; then
                        cat $pipeline_dir/uat/$Project_Name/ingress.yml 
                        cat $pipeline_dir/uat/$Project_Name/ingress.yml  | /usr/bin/kubectl apply -f  -
                        fi
                        '''
                    } else {
                        echo "不發布"
                    }
                }
            }
        } 
        stage('DeployGray'){
            steps {
                echo "部署灰度環境"
                script {
                    def GraysMode = input (
                        message: '確定要灰度驗證嗎?',
                        parameters:[
                            choice(name: 'operation', choices: ['基于權重灰度','基于請求頭灰度','跳過'])
                        ],
                        ok: '確定',
                        submitter: 'admin',
                        submitterParameter: 'APPROVER'
                    )
                    if (GraysMode['operation'] == '基于權重灰度'){
                        def WeightMode = input (
                            message: '請輸入權重比例!',
                            parameters:[
                                string(name: 'workload_weight',defaultValue: '',description: ''),
                                string(name: 'grayload_weight',defaultValue: '',description: '')
                            ],
                            ok: '確定',
                            submitter: 'admin',
                            submitterParameter: 'APPROVER'
                        )
                        sh """
                        echo $pipeline_dir
                        echo "打印編排文件詳細信息"
                        if [ -e "$pipeline_dir/prod/$Project_Name/deployment-gray.yml" ]; then
                        cat $pipeline_dir/prod/$Project_Name/deployment-gray.yml | sed  "s/TAG/${Tag}/g" 
                        cat $pipeline_dir/prod/$Project_Name/deployment-gray.yml | sed  "s/TAG/${Tag}/g" | /usr/bin/kubectl apply -f  -
                        fi

                        echo "配置權重"

                        echo ${WeightMode['grayload_weight']}
                        if [ -e "$pipeline_dir/prod/$Project_Name/ingress-gray-weight.yml" ]; then
                        cat $pipeline_dir/prod/$Project_Name/ingress-gray-weight.yml | sed  "s/WEIGHT-VALUE/${WeightMode['grayload_weight']}/g" 
                        cat $pipeline_dir/prod/$Project_Name/ingress-gray-weight.yml | sed  "s/WEIGHT-VALUE/${WeightMode['grayload_weight']}/g" | /usr/bin/kubectl apply -f  -
                        fi
                        """
                    }
                    if (GraysMode['operation'] == '基于請求頭灰度'){
                        GrayHeaderMode = input (
                            message: '請輸入請求頭!',
                            parameters:[
                                string(name: 'header_key',defaultValue: '',description: ''),
                                string(name: 'header_value',defaultValue: '',description: '')
                            ],
                            ok: '確定',
                            submitter: 'admin',
                            submitterParameter: 'APPROVER'
                        )

                        sh """
                        echo ${GrayHeaderMode['header_value']}
                        echo $pipeline_dir
                        echo "打印編排文件詳細信息"

                        if [ -e "$pipeline_dir/prod/$Project_Name/deployment-gray.yml" ]; then
                        cat $pipeline_dir/prod/$Project_Name/deployment-gray.yml | sed  "s/TAG/${Tag}/g" 
                        cat $pipeline_dir/prod/$Project_Name/deployment-gray.yml | sed  "s/TAG/${Tag}/g" | /usr/bin/kubectl apply -f  -
                        fi

                        echo "配置請求頭"
                        echo ${GrayHeaderMode['header_key']}
                        echo ${GrayHeaderMode['header_value']}

                        if [ -e "$pipeline_dir/prod/$Project_Name/ingress-gray-header.yml" ]; then
                        cat $pipeline_dir/prod/$Project_Name/ingress-gray-header.yml | sed  "s/header-key/${GrayHeaderMode['header_key']}/g" | sed  "s/header-value/${GrayHeaderMode['header_value']}/g"
                        cat $pipeline_dir/prod/$Project_Name/ingress-gray-header.yml | sed  "s/header-value/${GrayHeaderMode['header_key']}/g" | sed  "s/header-value/${GrayHeaderMode['header_value']}/g" | /usr/bin/kubectl apply -f  -
                        fi
                        """
                    }
                    // 默認模式為yes,如果跳過為no
                    if (GraysMode['operation'] == '跳過'){
                        GrayEnable='no'
                    }
                }
            }
        } 
        stage('DeployProd'){
            steps {
                echo "部署生產環境"
                script {
                    def userInput = input (
                        message: '確定要發布到生產環境嗎?',
                        parameters:[
                            choice(name: '操作', choices: ['發布', '跳過'])
                        ],
                        ok: '確定',
                        submitter: 'test',
                        submitterParameter: 'APPROVER'
                    )
                    if (userInput['操作'] == '發布'){
                        echo "發布"
                        Namespace_Prod = sh(script: "cat $pipeline_dir/prod/$Project_Name/deployment.yml | grep namespace | awk -F ':' '{print \$2}'", returnStdout: true).trim()
                        DeploymentName = sh(script: "cat $pipeline_dir/prod/$Project_Name/deployment.yml | grep name: |  head -n 1 | awk -F ':' '{print \$2}'", returnStdout: true).trim()
                        Revsion_Prod = sh(script: "kubectl get deployment $DeploymentName -n ${Namespace_Prod} -o=jsnotallow='{.spec.template.spec.containers[*].image}' | awk -F ':' '{print \$NF}'", returnStdout: true).trim()
                        GrayDeploymentName = sh(script: "cat $pipeline_dir/prod/$Project_Name/deployment-gray.yml | grep name: |  head -n 1 | awk -F ':' '{print \$2}'", returnStdout: true).trim()
                        GrayServiceName = sh(script: "cat $pipeline_dir/prod/$Project_Name/service-gray.yml | grep name: |  head -n 1 | awk -F ':' '{print \$2}'", returnStdout: true).trim()
                        GrayIngressName = sh(script: "cat $pipeline_dir/prod/$Project_Name/ingress-gray-header.yml | grep name: |  head -n 1 | awk -F ':' '{print \$2}'", returnStdout: true).trim()

                        sh '''
                        echo $pipeline_dir
                        echo "開始部署生產環境"
                        echo "打印編排文件詳細信息"

                        if [ -e "$pipeline_dir/prod/$Project_Name/deployment.yml" ]; then
                          cat $pipeline_dir/prod/$Project_Name/deployment.yml | sed  "s/TAG/${Tag}/g"
                          cat $pipeline_dir/prod/$Project_Name/deployment.yml | sed  "s/TAG/${Tag}/g" | /usr/bin/kubectl apply -f  -
                        fi

                        if [ -e "$pipeline_dir/prod/$Project_Name/service.yml" ]; then
                           cat $pipeline_dir/prod/$Project_Name/service.yml
                           cat $pipeline_dir/prod/$Project_Name/service.yml  | /usr/bin/kubectl apply -f  -
                        fi

                        if [ -e "$pipeline_dir/prod/$Project_Name/ingress.yml" ]; then
                          cat $pipeline_dir/prod/$Project_Name/ingress.yml
                          cat $pipeline_dir/prod/$Project_Name/ingress.yml  | /usr/bin/kubectl apply -f  -
                        fi
                        '''
                        if (GrayEnable == 'yes'){
                            sh """
                        kubectl delete deployment ${GrayDeploymentName} -n ${Namespace_Prod}
                        kubectl delete service  ${GrayServiceName} -n ${Namespace_Prod}
                        kubectl delete ingress  ${GrayIngressName} -n ${Namespace_Prod}
                        """
                        }

                    } else {
                        echo "不發布"
                    }
                }
            }
            post {
                success {
                    wrap([$class: 'BuildUser']) {
                    lark (
                        robot: "9f7c94cd-491e-4309-83b4-9290d01fc285",
                        type: "CARD",
                        title: "??  Jenkins 應用發布成功",
                        text: [
                            "?? **應用名稱**:[${JOB_NAME}](${JOB_URL})",
                            "?? **應用環境**:Prod",
                            "?? **任務編號**:[${BUILD_DISPLAY_NAME}](${BUILD_URL})",
                            "?? **發布狀態**: <font color='green'>成功</font>",
                            "?? **鏡像版本**: $Tag",
                            "?? **鏡像倉庫**: harbor.kubesre.com:8443/kubesre/$Project_Name",
                            "?? **執  行 者**: ${env.BUILD_USER}",
                            "<at id=all></at>"
                        ],
                        buttons: [
                           [
                              title: "更改記錄",
                              url: "${BUILD_URL}changes"
                           ],
                           [
                              title: "控制臺",
                              type: "danger",
                              url: "${BUILD_URL}console"
                           ]
                        ]
                    )}
                }
            }
        } 

        stage('RollBack'){
            steps {
                echo "版本回滾操作"
                script {
                  def RollBack = input (
                        message: '確定要執行回滾操作嗎?',
                        parameters:[
                            choice(name: '是否回滾', choices: ['是', '否']),
                            string(name: '回滾版本', defaultValue: Revsion_Prod,description: '默認上一個版本')
                        ],
                        ok: '確定',
                        submitter: 'admin',
                        submitterParameter: 'APPROVER'
                    )
                  if (RollBack['是否回滾'] == '是'){
                        echo "版本回滾成功"
                        echo RollBack['回滾版本']
                        sh """
                        if [ -e "$pipeline_dir/prod/$Project_Name/deployment.yml" ]; then
                          cat $pipeline_dir/prod/$Project_Name/deployment.yml | sed  "s/TAG/${RollBack['回滾版本']}/g"
                          cat $pipeline_dir/prod/$Project_Name/deployment.yml | sed  "s/TAG/${RollBack['回滾版本']}/g" | /usr/bin/kubectl apply -f  -
                        fi
                        """
                  } else {
                         echo "放棄版本回滾"
                         echo RollBack['回滾版本']
                  }
            }
         }
        }
    }
}

Vue Pipeline:

pipeline {
    agent any
    triggers {
            GenericTrigger(
                genericVariables: [
                    [key: 'ref', value: '$.ref'],  //獲取分支
                    [key: 'user_username', value: '$.user_username'],     //獲取自動構建用戶名
                    [key: 'GitRepository', value: '$.project.git_http_url'],  //獲取gitlab ssh項目地址
                    [key: 'project', value: '$.project.name'],      //獲取項目名稱
                    [key: 'repository', value: '$.repository.name'],
                ],
                token: "$JOB_NAME",
                causeString: 'Triggered on $branch',
                printContributedVariables: true,
                printPostContent: true,
                silentResponse: false,
            )
        }
    environment {
        // pipeline配置路徑
        pipeline_dir="/var/lib/jenkins/workspace/pipeline"
        // 項目版本
        Tag=sh(script: 'echo "${ref}" | awk -F"/" \'{print $3}\'', returnStdout: true).trim()
        // 項目名稱
        Project_Name="${project}"
        // 上一次版本
        Revsion_Prod=''
        //Depolyment名稱
        DeploymentName=''
        // 生產名稱空間
        Namespace_Prod=''
        // 灰度模式
        GrayHeaderMode=''
        // 灰度Depolyment名稱
        GrayDeploymentName=''
        // 灰度Service名稱
        GrayServiceName=''
        // 灰度Ingress名稱
        GrayIngressName=''
        // 是否灰度
        GrayEnable='yes'
    }
    options {
  // 表示保留10次構建歷史
  buildDiscarder(logRotator(numToKeepStr: '10'))
    }
    stages {
        stage('Pull Code') {
             // 拉取代碼
            steps {
                checkout([
                    $class: 'GitSCM', 
                    branches: [[name: "$ref"]],
                    doGenerateSubmoduleConfigurations: false,
                    extensions: [], 
                    userRemoteConfigs: [[
                        credentialsId: 'ac66550d-6999-485c-af3a-7e6189f765f0',
                        url: "$GitRepository"
                    ]]
                ])

                script{
                    currentBuild.displayName = "#${BUILD_NUMBER} - ${Project_Name} - ${Tag}"
                }
            }
        }
    //     // 代碼構建
    //   stage('Build Code') {
    //         steps {
    //              sh '/application/maven/bin/mvn -f pom.xml -s settings.xml clean package -DskipTests'
    //         }
    //     } 
         // 鏡像構建
        stage('Build Image') {
            steps {
                 sh '''
                   /usr/bin/docker build -t uhub.service.ucloud.cn/kubesre/$Project_Name:$Tag .
                   /usr/bin/docker push  uhub.service.ucloud.cn/kubesre/$Project_Name:$Tag
                 '''
            }
            post {
                success {
                    wrap([$class: 'BuildUser']) {
                    lark (
                        robot: "9f7c94cd-491e-4309-83b4-9290d01fc285",
                        type: "CARD",
                        title: "??  Jenkins 鏡像構建成功",
                        text: [
                            "?? **任務名稱**:[${JOB_NAME}](${JOB_URL})",
                            "?? **任務編號**:[${BUILD_DISPLAY_NAME}](${BUILD_URL})",
                            "?? **構建狀態**: <font color='green'>成功</font>",
                            "?? **鏡像版本**: $Tag",
                            "?? **鏡像倉庫**: uhub.service.ucloud.cn/kubesre/$Project_Name",
                            "?? **構建用時**: ${currentBuild.duration} ms",
                            "?? **執  行 者**: ${env.BUILD_USER}",
                            "<at id=all></at>"
                        ],
                        buttons: [
                           [
                              title: "更改記錄",
                              url: "${BUILD_URL}changes"
                           ],
                           [
                              title: "控制臺",
                              type: "danger",
                              url: "${BUILD_URL}console"
                           ]
                        ]
                    )}
                }
            }
        }
        stage('DeployDev'){
            steps {
                echo "部署開發環境"
                script {
                    def userInput = input (
                        message: '確定要發布到DEV環境嗎?',
                        parameters:[
                            choice(name: '操作', choices: ['發布', '跳過'])
                        ],
                        ok: '確定',
                        submitter: 'admin',
                        submitterParameter: 'APPROVER'
                    )
                    if (userInput['操作'] == '發布'){
                        echo "部署Dev環境開始"
                        sh '''
                        echo $pipeline_dir
                        echo "打印編排文件詳細信息"
                        if [ -e "$pipeline_dir/dev/$Project_Name/deployment.yml" ]; then
                            cat $pipeline_dir/dev/$Project_Name/deployment.yml | sed  "s/TAG/${Tag}/g"
                            cat $pipeline_dir/dev/$Project_Name/deployment.yml | sed  "s/TAG/${Tag}/g" | /usr/bin/kubectl apply -f  -
                        fi
                        
                        if [ -e "$pipeline_dir/dev/$Project_Name/service.yml" ]; then
                            cat $pipeline_dir/dev/$Project_Name/service.yml
                            cat $pipeline_dir/dev/$Project_Name/service.yml  | /usr/bin/kubectl apply -f  -
                        fi

                        if [ -e "$pipeline_dir/dev/$Project_Name/ingress.yml" ]; then
                            cat $pipeline_dir/dev/$Project_Name/ingress.yml 
                            cat $pipeline_dir/dev/$Project_Name/ingress.yml  | /usr/bin/kubectl apply -f  -
                        fi
                        '''
                    } else {
                        echo "不發布"
                    }
                }
            }
        } 
        stage('DeployUat'){
            steps {
                echo "部署測試環境"
                 script {
                    def userInput = input (
                        message: '確定要發布到UAT環境嗎?',
                        parameters:[
                            choice(name: '操作', choices: ['發布', '跳過'])
                        ],
                        ok: '確定',
                        submitter: 'admin',
                        submitterParameter: 'APPROVER'
                    )
                    if (userInput['操作'] == '發布'){
                        echo "發布"
                         sh '''
                        echo $pipeline_dir
                        echo "打印編排文件詳細信息"

                        if [ -e "$pipeline_dir/uat/$Project_Name/deployment.yml" ]; then
                            cat $pipeline_dir/uat/$Project_Name/deployment.yml | sed  "s/TAG/${Tag}/g"
                            cat $pipeline_dir/uat/$Project_Name/deployment.yml | sed  "s/TAG/${Tag}/g" | /usr/bin/kubectl apply -f  -
                        fi
                        
                        if [ -e "$pipeline_dir/uat/$Project_Name/service.yml" ]; then
                            cat $pipeline_dir/uat/$Project_Name/service.yml
                            cat $pipeline_dir/uat/$Project_Name/service.yml  | /usr/bin/kubectl apply -f  -
                        fi

                        if [ -e "$pipeline_dir/uat/$Project_Name/ingress.yml" ]; then
                            cat $pipeline_dir/uat/$Project_Name/ingress.yml 
                            cat $pipeline_dir/uat/$Project_Name/ingress.yml  | /usr/bin/kubectl apply -f  -
                        fi
                        '''
                    } else {
                        echo "不發布"
                    }
                }
            }
        } 
        stage('DeployProd'){
            steps {
                echo "部署生產環境"
                 script {
                    def userInput = input (
                        message: '確定要發布到生產環境嗎?',
                        parameters:[
                            choice(name: '操作', choices: ['發布', '跳過'])
                        ],
                        ok: '確定',
                        submitter: 'test',
                        submitterParameter: 'APPROVER'
                    )
                    if (userInput['操作'] == '發布'){
                        echo "發布"
                        Namespace_Prod = sh(script: "cat $pipeline_dir/prod/$Project_Name/deployment.yml | grep namespace | awk -F ':' '{print \$2}'", returnStdout: true).trim()
                        DeploymentName = sh(script: "cat $pipeline_dir/prod/$Project_Name/deployment.yml | grep name: |  head -n 1 | awk -F ':' '{print \$2}'", returnStdout: true).trim()
                        Revsion_Prod = sh(script: "kubectl get deployment $DeploymentName -n ${Namespace_Prod} -o=jsnotallow='{.spec.template.spec.containers[*].image}' | awk -F ':' '{print \$NF}'", returnStdout: true).trim()
                        GrayDeploymentName = sh(script: "cat $pipeline_dir/prod/$Project_Name/deployment-gray.yml | grep name: |  head -n 1 | awk -F ':' '{print \$2}'", returnStdout: true).trim()
                        GrayServiceName = sh(script: "cat $pipeline_dir/prod/$Project_Name/service-gray.yml | grep name: |  head -n 1 | awk -F ':' '{print \$2}'", returnStdout: true).trim()
                        GrayIngressName = sh(script: "cat $pipeline_dir/prod/$Project_Name/ingress-gray-header.yml | grep name: |  head -n 1 | awk -F ':' '{print \$2}'", returnStdout: true).trim()

                        sh '''
                        echo $pipeline_dir
                        echo "開始部署生產環境"
                        echo "打印編排文件詳細信息"

                        if [ -e "$pipeline_dir/prod/$Project_Name/deployment.yml" ]; then
                          cat $pipeline_dir/prod/$Project_Name/deployment.yml | sed  "s/TAG/${Tag}/g"
                          cat $pipeline_dir/prod/$Project_Name/deployment.yml | sed  "s/TAG/${Tag}/g" | /usr/bin/kubectl apply -f  -
                        fi

                        if [ -e "$pipeline_dir/prod/$Project_Name/service.yml" ]; then
                           cat $pipeline_dir/prod/$Project_Name/service.yml
                           cat $pipeline_dir/prod/$Project_Name/service.yml  | /usr/bin/kubectl apply -f  -
                        fi

                        if [ -e "$pipeline_dir/prod/$Project_Name/ingress.yml" ]; then
                          cat $pipeline_dir/prod/$Project_Name/ingress.yml
                          cat $pipeline_dir/prod/$Project_Name/ingress.yml  | /usr/bin/kubectl apply -f  -
                        fi
                        '''

                    } else {
                        echo "不發布"
                    }
                }
            }
            post {
                success {
                    wrap([$class: 'BuildUser']) {
                    lark (
                        robot: "9f7c94cd-491e-4309-83b4-9290d01fc285",
                        type: "CARD",
                        title: "??  Jenkins 應用發布成功",
                        text: [
                            "?? **應用名稱**:[${JOB_NAME}](${JOB_URL})",
                            "?? **應用環境**:Prod",
                            "?? **任務編號**:[${BUILD_DISPLAY_NAME}](${BUILD_URL})",
                            "?? **發布狀態**: <font color='green'>成功</font>",
                            "?? **鏡像版本**: $Tag",
                            "?? **鏡像倉庫**: harbor.kubesre.com:8443/kubesre/$Project_Name",
                            "?? **執  行 者**: ${env.BUILD_USER}",
                            "<at id=all></at>"
                        ],
                        buttons: [
                           [
                              title: "更改記錄",
                              url: "${BUILD_URL}changes"
                           ],
                           [
                              title: "控制臺",
                              type: "danger",
                              url: "${BUILD_URL}console"
                           ]
                        ]
                    )}
                }
            }
        } 
    }
}

配置Jenkins

依賴的組件(自行安裝):

  • Generic Webhook Trigger
  • Pipeline(所有以Pipeline開頭的組件)
  • build user vars
  • Blue Ocean
  • Lark Notice(通過上傳文件的方式安裝)https://721806280.github.io/lark-notice-plugin-doc/

配置Lark Notice:

Lark Notice Plugin 是一個用于 Jenkins 的構建通知插件,可以將 Jenkins構建過程以及結果通知推送到 Lark、飛書、釘釘 協作平臺。 可配置多個的通知時機,包括 構建啟動時、構建中斷、構建失敗、構建成功時、構建不穩定 等。 支持多種不同類型的消息,包括 文本消息、圖片消息, 群名片消息、富文本消息、卡片消息; 同時該插件還提供了自定義模板和變量的功能,使您能夠根據自己的需求來定制通知消息的內容和格式。(本次案例是基于飛書進行驗證)

準備工作,在飛書群新建一個機器人(https://open.feishu.cn/document/client-docs/bot-v3/add-custom-bot):

在飛書群,點擊設置:

圖片圖片

然后點擊群機器人:

圖片圖片

在飛書群,選擇添加機器人

圖片圖片

填寫相應配置信息并點擊保存:

圖片圖片

圖片圖片

選擇系統管理-Lark Notice:

圖片圖片

通知時機全部勾選:

圖片圖片

配置機器人信息并保存

圖片圖片

配置Java Pipeline

新建任務:

圖片圖片

填寫任務名稱,并選擇流水線,點擊確定:

圖片圖片

配置Pipeline SCM:

圖片圖片

修改腳本路徑,點擊確定:

圖片圖片

點擊構建讓配置生效:

圖片圖片

其他Java項目配置都一樣!

配置Vue Pipeline

新建任務:

圖片圖片

填寫任務名稱,選擇流水線:

圖片圖片

配置Pipeline SCM:

圖片圖片

修改腳本路徑,點擊確定:

圖片圖片

點擊構建讓配置生效:

圖片圖片

其他Vue項目配置都一樣!

配置Gitlab Webhook

進入項目,選擇webhook:

圖片圖片

選擇添加Webhook:

圖片圖片

配置webhook URL,Token以Job Name:

圖片圖片

勾選標簽推送事件,也就是說只有標簽推送事件才會觸發流水線:

圖片圖片

到此為止,Webook已配置完畢!所有項目配置都一樣。

觸發驗證

觸發Java Pipeline:

進入標簽管理:

圖片圖片

新建標簽:

圖片圖片

填寫信息并點擊創建標簽(此標簽名稱也是容器鏡像的Tag):

圖片圖片

進入Jenkins可以看到Gateway Pipeline已經觸發了:

圖片圖片

圖片圖片

選擇發布,并點擊確定,將新版本發布到Dev環境:

圖片圖片

選擇發布,并點擊確定,將新版本發布到Uat環境:

圖片圖片

選擇對應的灰度發布方式或者跳過:

圖片圖片

選擇發布,并點擊確定,將新版本發布到Prod環境:

圖片圖片

也可以回滾,默認是上一個版本也可修改成想要回滾到的版本:

圖片圖片

觸發 Vue流水線:

進入標簽管理:

圖片圖片

創建標簽:

圖片圖片

填寫信息并點擊創建標簽(此標簽名稱也是容器鏡像的Tag):

圖片圖片

圖片圖片

進入Jenkins可以看到ruoyi-ui Pipeline已經觸發了:

圖片圖片

選擇發布,并點擊確定,將新版本發布到Dev環境:

圖片圖片

選擇發布,并點擊確定,將新版本發布到Uat環境:

圖片圖片

選擇發布,并點擊確定,將新版本發布到Prod環境:

圖片圖片

構建通知

圖片圖片

責任編輯:武曉燕 來源: 云原生運維圈
相關推薦

2020-10-21 14:10:28

工具測試開發

2019-11-07 09:00:39

Jenkins流水線開源

2022-08-31 22:25:53

微服務架構DevOPs

2022-02-22 09:00:00

軟件開發CI/CD 管道工具

2021-10-11 14:28:25

TypeScript企業級應用

2009-01-03 14:54:36

ibmdwWebSphere

2009-06-03 14:24:12

ibmdwWebSphere

2015-11-10 15:14:20

CIO時代網

2023-02-19 15:28:39

CI/CD 管道集成開發

2020-06-05 07:20:41

測試自動化環境

2021-07-27 08:01:22

CICD平臺

2018-08-24 09:00:00

DevOps持續集成連續部署

2023-09-11 12:57:00

大數據大數據中臺

2021-08-31 09:00:00

開發Azure DevOp集成

2023-03-13 14:46:32

CI/CD工具軟件開發

2020-10-12 07:00:00

JenkinsDevOps測試工具

2010-04-07 08:55:00

OSGiSpring

2014-09-09 14:10:01

企業級HadoopSpark

2016-10-12 17:18:26

私有云持續交付華為

2018-02-02 11:21:25

云計算標準和應用大會
點贊
收藏

51CTO技術棧公眾號

国产一区精品| 日韩av黄色片| avtt久久| 午夜精品久久久久久久蜜桃app| 国产精品99久久久久久久| 日韩欧美三级视频| 日本女优一区| 日韩精品在线一区| 日本va中文字幕| gogo在线观看| 久久蜜桃av一区二区天堂| 成人免费在线网址| 91精品国产综合久久久蜜臀九色| 日韩中文在线电影| 亚洲国产精品久久91精品| 日韩av片网站| 成年人黄色大片在线| 国产精品网曝门| 国内视频一区| av 一区二区三区| 久久精品网址| 欧美国产一区二区三区| 欧美极品jizzhd欧美18| 精品久久97| 日韩午夜在线播放| 99国产精品久久久久久| 这里有精品可以观看| 伊人性伊人情综合网| 日本一区视频在线播放| 蜜臀av午夜精品| 国产精品影视在线| 国产美女被下药99| 精产国品一区二区| aa国产精品| 欧美精品18videos性欧| 亚洲怡红院在线观看| 国产欧美一区| 国产视频久久久久| 给我免费观看片在线电影的| 日本免费精品| 91精品国产品国语在线不卡| 日韩中文字幕a| 日本成人福利| 欧美视频日韩视频| 北条麻妃av高潮尖叫在线观看| 理论不卡电影大全神| 亚洲一二三区在线观看| 男人j进女人j| 国产激情视频在线| 亚洲色图19p| 欧美一级黄色录像片| 日本黄色片在线观看| 国产精品美女久久久久高潮| 日韩国产精品一区二区| 你懂的视频在线| 久久伊99综合婷婷久久伊| 久久av免费一区| 天天干在线观看| 99精品在线免费| 国产一区自拍视频| 日本高清中文字幕二区在线| 久久先锋资源网| 欧美在线一二三区| 91网页在线观看| 一色桃子久久精品亚洲| 人人妻人人澡人人爽精品欧美一区| 91se在线| 一区二区三区日本| 欧美视频免费看欧美视频| 麻豆网站免费在线观看| 欧美日韩亚洲一区二| 国产成人精品视频ⅴa片软件竹菊| 成人日韩精品| 51精品秘密在线观看| 蜜桃色一区二区三区| 国产日韩三级| 亚洲午夜小视频| 成人做爰视频网站| 亚洲高清资源| 青草青草久热精品视频在线观看| 无码人妻aⅴ一区二区三区有奶水| 视频一区二区欧美| 91久久久久久国产精品| 国产成人手机在线| 久久网站热最新地址| 亚洲国产一区二区在线| 18网站在线观看| 日韩欧美国产视频| 在线视频观看91| 色吊丝一区二区| 日韩中文字幕不卡视频| 青娱乐国产在线| 亚洲自啪免费| 成人免费视频a| 婷婷久久久久久| 国产精品成人免费精品自在线观看| 黄色激情在线视频| 精品免费av一区二区三区| 欧美一区二区三区免费视频| 中文字幕一二三四区| 天天综合精品| 庆余年2免费日韩剧观看大牛| 97av免费视频| 久久婷婷国产综合精品青草| 老汉色影院首页| 蜜桃成人精品| 日韩成人在线视频| 亚洲综合视频网站| 亚洲一区视频| 波多野结衣精品久久| 国产精品一区二区三区四区色| 亚洲欧美日韩小说| 亚洲 中文字幕 日韩 无码| 日本高清久久| 久久精品久久久久电影| 欧美 日韩 精品| 国产a精品视频| 亚洲成人午夜在线| 久草在线资源站手机版| 日韩午夜精品视频| 天堂网av2018| 天堂午夜影视日韩欧美一区二区| 成人午夜电影免费在线观看| 欧洲不卡av| 五月婷婷综合在线| 国产精品二区视频| 国产精品精品国产一区二区| 国产精品18久久久久久首页狼 | 日本黄色动态图| 久久五月天小说| 国产成人久久精品| 可以在线观看的av| 好吊成人免视频| 日本三级日本三级日本三级极| 亚洲成av人片乱码色午夜| 国产精品爱久久久久久久| 天堂在线一二区| 亚洲成人自拍偷拍| 亚洲欧美高清在线| 亚洲视频观看| 国产精品毛片一区视频| 天堂av中文在线| 日韩亚洲欧美在线| 久久精品www人人爽人人| 国产精品一二一区| 美女在线免费视频| 久久久久久久久久久久电影| 久久天天躁狠狠躁夜夜av| 一区二区三区播放| 国产精品久久久久久户外露出 | 中日韩在线视频| 福利一区二区三区视频在线观看 | 成年人免费观看视频网站 | 亚洲一区二区三区四区在线免费观看| 中文字幕一区二区三区四| 这里只有精品在线| 亚洲一区二区三区四区视频| 超碰在线观看免费版| 4438x成人网最大色成网站| 欧美激情精品久久久久久免费| 久草精品在线观看| 黄色成人在线免费观看| 国产精品zjzjzj在线观看| 777777777亚洲妇女| 欧美日韩在线精品一区二区三区激情综 | 精品免费视频.| 国产在线视频卡一卡二| 99久久久免费精品国产一区二区| 久久久久久久久久久99| 亚洲深夜福利在线观看| 国产精品免费福利| 中文字幕资源网在线观看| 精品久久久久久久一区二区蜜臀| 久久精品女人毛片国产| 91色视频在线| 一区二区成人网| 亚洲最大黄网| 久久66热这里只有精品| 欧美××××黑人××性爽| 中文字幕一精品亚洲无线一区| 91福利在线观看视频| 亚洲激情中文1区| 免费的av网站| 久久国内精品视频| 男人添女荫道口女人有什么感觉| 台湾亚洲精品一区二区tv| 国产精品免费小视频| 天堂8中文在线| 亚洲色图第三页| av男人天堂av| 色呦呦日韩精品| 欧美成人一区二区三区高清| 久久综合丝袜日本网| 亚洲黄色片免费| 先锋影音国产一区| 天天做天天爱天天高潮| 亚洲国产国产| 亚洲在线视频福利| 二吊插入一穴一区二区| 欧美成人在线免费| 国产免费a∨片在线观看不卡| 91麻豆精品国产91久久久久久 | 欧美剧在线免费观看网站| 日韩高清精品免费观看| 国产精品人妖ts系列视频| 色哟哟无码精品一区二区三区| 日本亚洲三级在线| 欧美久久久久久久久久久久久| 97精品国产| 久久久久久久久四区三区| 国产情侣一区在线| 国产精品久久久久91| 国产精品一区二区日韩| 久久影视电视剧免费网站清宫辞电视 | 亚洲私拍视频| 久久99国产精品自在自在app| 北条麻妃在线| 亚洲欧美日韩一区二区三区在线| 成 人片 黄 色 大 片| 欧美日韩国产综合草草| 免费看日批视频| 亚瑟在线精品视频| 国产这里有精品| 自拍偷拍国产精品| 成年人视频软件| 国产欧美日韩在线| 一区二区三区四区免费| 91美女片黄在线观看91美女| 69xxx免费视频| 国产酒店精品激情| 欧美日韩免费一区| 黑森林福利视频导航| 欧美日韩福利| 男女激烈动态图| 日韩在线二区| 亚洲一区二区三区精品在线观看| 国产精品片aa在线观看| 美日韩精品免费| 日韩欧美在线精品| 精选一区二区三区四区五区| 国产精品超碰| 国产精品大全| 国产精品传媒| 久久99久久精品国产| 亚洲国产欧美日韩在线观看第一区 | 国产人妻互换一区二区| 99久久精品网站| 一区二区三区观看| 欧美亚洲国产激情| 亚洲欧美日韩精品久久久| 久久一区二区中文字幕| 影音先锋欧美资源| 亚洲精品国产首次亮相| 欧美日韩亚洲国产成人| 午夜天堂精品久久久久| 精品一区二区三区无码视频| 亚洲激情另类| 春日野结衣av| 热久久久久久久| 亚洲精品中文字幕乱码无线| 国产精品888| 女教师高潮黄又色视频| 不卡视频一二三四| 中国美女乱淫免费看视频| 国产欧美精品区一区二区三区 | 久久精品色图| 中文字幕欧美日韩精品 | 668精品在线视频| 黑人巨大亚洲一区二区久| 国产精品爱久久久久久久| 国产精品免费精品自在线观看 | 天天爱天天干天天操| 精品亚洲aⅴ在线观看| 久久久pmvav| 久久精品亚洲一区| 97超碰在线免费| 国产成人久久久精品一区| 成人污版视频| 国产综合 伊人色| 欧美一级精品| 欧美国产视频一区| 麻豆久久精品| 五月天六月丁香| 久久影院视频免费| 国产日韩欧美在线观看视频| 五月婷婷激情综合| 中文字幕av久久爽| 欧美精品一区在线观看| 春暖花开成人亚洲区| 欧美高清无遮挡| 在线免费日韩片| 亚洲精品欧美日韩专区| 婷婷综合电影| 国产精品国三级国产av| 久久精品1区| 久久久久亚洲av成人网人人软件| 国产视频911| 国产精品 欧美 日韩| 欧美日韩一区二区电影| 天天操天天操天天操| 久久精品亚洲94久久精品| 成人欧美一区二区三区的电影| 亚洲一区二区免费| 激情五月综合| 国产伦精品一区二区三区四区视频_| 麻豆91精品视频| 亚洲区免费视频| 午夜不卡av免费| 国产夫妻性生活视频| 中文字幕日韩在线播放| 欧美gv在线| 成人动漫视频在线观看完整版| 欧美激情电影| www.国产区| 99re6这里只有精品视频在线观看| 搜索黄色一级片| 欧洲激情一区二区| 亚州av在线播放| 久久99精品视频一区97| 少妇高潮一区二区三区99| 欧美一区国产一区| 欧美一级二区| 免费看毛片的网站| 亚洲精品高清在线| 国产精品九九九九| 最近免费中文字幕视频2019| 欧美片第1页| 免费看污久久久| 中文字幕一区二区三区四区五区 | 综合激情网站| 亚洲少妇久久久| 久久久精品日韩欧美| 久久久久久久黄色片| 亚洲电影在线看| 女人天堂av在线播放| 99精彩视频在线观看免费| 亚洲精品久久久| 成人不卡免费视频| 国产精品福利影院| 亚洲天堂中文字幕在线| 自拍偷拍亚洲精品| 欧美极品在线| 亚洲一区二区三区午夜| 蜜桃久久精品一区二区| 人妻精品久久久久中文| 91久久精品日日躁夜夜躁欧美| 暖暖视频在线免费观看| 国产成人精品优优av| 国产一区二区精品福利地址| 亚洲色图38p| 中日韩免费视频中文字幕| 亚洲 日本 欧美 中文幕| 亚洲人午夜精品免费| 亚洲国产尤物| 中文字幕剧情在线观看一区| 韩国成人在线视频| 欧美成人精品激情在线视频| 精品粉嫩aⅴ一区二区三区四区| ririsao久久精品一区| 久久99导航| 日本不卡视频在线观看| 九九热视频在线免费观看| 欧美一区二区私人影院日本| av在线下载| 国产日韩欧美一区二区| 午夜在线精品偷拍| 中文字幕av久久爽一区| 欧美二区乱c少妇| 51xtv成人影院| 精品国产一区二区三区四区精华 | 99久久.com| 色综合久久久无码中文字幕波多| 午夜视频在线观看一区二区三区 | 久久精品国产亚洲5555| www.浪潮av.com| 国产精品美女久久久久aⅴ国产馆| 国产免费一区二区三区最新不卡 | 欧美激情自拍偷拍| 国产特级aaaaaa大片| 97精品久久久| 清纯唯美亚洲综合一区| 久久久久亚洲av无码麻豆| 第一福利永久视频精品| 137大胆人体在线观看| 99精品欧美一区二区三区| 免费日韩精品中文字幕视频在线| 香蕉久久久久久久| 亚洲韩国欧洲国产日产av| 成人一区视频| 夜夜添无码一区二区三区| 国产精品午夜在线观看| 国产综合在线播放| 国产精品情侣自拍| 亚洲国内欧美| 欧美性猛交xxxx乱大交少妇| 日韩精品视频在线观看网址 | 日韩一区二区三| 日韩精品第一| 人妻av中文系列| 亚洲欧美日韩中文播放|