CI/CD-云主机场景CI-[前端项目库|后端项目库]

root
233
文章
0
评论
2021年11月7日19:21:15 评论 16291字阅读54分18秒

CI/CD-云主机场景CI-[前端项目库|后端项目库]

实践1规划的2个CI项目内容

hwf-app-service项目

jenkins配置

  • 凭据
  • git项目地址

  • 字符参数:
    • branchName
  • 选项参数:
    • srcUrl
      • http://192.168.1.110:9080/hwf-app/hwf-app-service.git
    • buildTool
      • maven
    • type
      • jar
      • war

gitlab配置

Build.groovy

package hwf.devops


//Maven
def MavenBuild(){
    def buildTools = ["maven": "/data/apache-maven-3.8.3"]
    sh "${buildTools["maven"]}/bin/mvn clean package -DskipTests -f demo/pom.xml \
                        -s settings.xml"
}

//Gradle
def GradleBuild(){
    def buildTools = ["gradle": "/data/gradle-7.2"]
    sh "${buildTools["gradle"]}/bin/gradle clean && ${buildTools["gradle"]}/bin/gradle build -x test"
}

//Ant
def AntBuild(configPath="./build.xml"){
    sh "ant -f ${configPath}"
}

//Golang
def GoBuild(){
    def buildTools = ["go": "/data/go" ]
    sh "${buildTools["go"]}/bin/go build demo.go"
}

//Npm
def NpmBuild(){
    def buildTools = ["npm": "/data/node-v14.16.1-linux-x64" ]
    sh """
                    export PATH=\$PATH:${buildTools["npm"]}/bin
                    ${buildTools["npm"]}/bin/npm install
                    ${buildTools["npm"]}/bin/npm run build
    """
}

//Yarn
def YarnBuild(){
    sh "yarn install && yarn build "
}

//Main
def CodeBuild(type){
    switch(type){
        case "maven":
            MavenBuild()
            break;
        case "gradle":
            GradleBuild()
            break;
        case "npm":
            NpmBuild()
            break;
        case "yarn":
            YarnBuild()
            break;
        default:
            error "No such tools ... [maven/ant/gradle/npm/yarn/go]"
            break
    }
}

Checkout.groovy

package hwf.devops
//下载代码
 def GetCode(srcUrl, branchName){
    checkout([$class: 'GitSCM', 
            branches: [[name: branchName]], 
            extensions: [], 
            userRemoteConfigs: [[
            credentialsId: '6392297d-d77d-42a3-9d98-ea128dce6837', 
            url: srcUrl]]])
}

Nexuspackage.groovy

package hwf.devops

def Upload(prository,groupId,artifaceId,file,version,type,pkgName){
     withCredentials([usernamePassword(credentialsId: 'c2b82aec-d6d7-42c4-a97d-b78fc07c050b', passwordVariable: 'TOKEN', usernameVariable: 'USER')]) {
        sh """
            curl -X 'POST' \
            "http://192.168.1.110:8081/nexus/service/rest/v1/components?repository=${prository}" \
            -H 'accept: application/json' \
            -H 'Content-Type: multipart/form-data' \
            -F "maven2.groupId=${groupId}" \
            -F "maven2.artifactId=${artifaceId}" \
            -F "maven2.version=${version}" \
            -F "maven2.packaging=${type}" \
            -F "maven2.asset1=@${file}" \
            -F "maven2.asset1.extension=${pkgName}" \
            -u "${USER}":"${TOKEN}"
        """
    }
}

def PushRawArtifacts(repoName,targetDir, filePath, newPkgName, type ){
    withCredentials([usernamePassword(credentialsId: 'c2b82aec-d6d7-42c4-a97d-b78fc07c050b', passwordVariable: 'TOKEN', usernameVariable: 'USER')]
    ) {
        repoUrl = "http://192.168.1.110:8081/nexus/repository"
        pkgPath = "${repoUrl}/${repoName}/${projectName}/${version}/${projectName}-${version}.${type}"
        sh "wget --http-user=${USER} --http-passwd=${TOKEN} ${pkgPath}  -q"


    }
}

Sonar.groovy

package hwf.devops

def SonarRequest(apiUrl,method){
    withCredentials([string(credentialsId: "cb50abd7-ed16-4130-b5c8-8f609253b43d", variable: 'SONAR_TOKEN')]) {
        sonarApi = "http://192.168.1.110:9000/api"
        response = sh  returnStdout: true, 
            script: """
            curl --location \
                 --request ${method} \
                 "${sonarApi}/${apiUrl}" \
                 --header "Authorization: Basic ${SONAR_TOKEN}"
            """
        // json格式化
        try {
            response = readJSON text: """ ${response - "\n"} """
        } catch(e){
            response = readJSON text: """{"errors" : true}"""
        }
        return response
        
    }
}

//查找项目
def SearchProject(projectName){
    apiUrl = "projects/search?projects=${projectName}"
    response = SonarRequest(apiUrl,"GET")
    if (response.paging.total == 0){
        return false
    }
    return true 
}


//创建项目
def CreateProject(projectName){
    apiUrl = "projects/create?name=${projectName}&project=${projectName}"
    response = SonarRequest(apiUrl,"POST")
    try{
        if (response.project.key == projectName ) {
            println("Project Create success!...")
            return true
        }
    }catch(e){
        println(response.errors)
        return false
    }
}

// 更新质量阈
def UpdateQualityProfiles(lang, projectName, profileName){
    apiUrl = "qualityprofiles/add_project?language=${lang}&project=${projectName}&qualityProfile=${profileName}"
    response = SonarRequest(apiUrl,"POST")
    
    if (response.errors != true){
        println("ERROR: UpdateQualityProfiles ${response.errors}...")
        return false
    } else {
        println("SUCCESS: UpdateQualityProfiles ${lang} > ${projectName} > ${profileName}" )
        return true
    }
}

def SonarJava(projectName, groupName ){
    withCredentials([string(credentialsId: '40e88177-bd39-4ac4-92bd-3124e4fc9171', variable: 'AUTH_TOKEN')]) {
        sh """
            sonar-scanner -Dsonar.host.url=http://192.168.1.110:9000 \
                -Dsonar.projectKey=${projectName} \
                -Dsonar.projectName=${projectName} \
                -Dsonar.projectVersion=${BUILD_ID} \
                -Dsonar.login=${AUTH_TOKEN} \
                -Dsonar.ws.timeout=30 \
                -Dsonar.projectDescription="my first project!" \
                -Dsonar.links.homepage=http://192.168.1.110:9080/${groupName}/${projectName} \
                -Dsonar.links.ci=http://192.168.1.110:8080/job/${groupName}/job/${projectName}/ \
                -Dsonar.sources=./demo/src \
                -Dsonar.sourceEncoding=UTF-8 \
                -Dsonar.java.binaries=./demo/target/classes \
                -Dsonar.java.test.binaries=./demo/target/test-classes 
        """
    }
}


def SonarWeb(projectName, groupName ){
    withCredentials([string(credentialsId: '3edb6a73-c10f-4d36-b6aa-c37abd4570b6', variable: 'AUTH_TOKEN')]) {
        sh """
            sonar-scanner -Dsonar.host.url=http://192.168.1.110:9000 \
                -Dsonar.projectKey=${projectName} \
                -Dsonar.projectName=${projectName} \
                -Dsonar.projectVersion=${BUILD_ID} \
                -Dsonar.login=${AUTH_TOKEN} \
                -Dsonar.ws.timeout=30 \
                -Dsonar.projectDescription="my first project!" \
                -Dsonar.links.homepage=http://192.168.1.200/${groupName}/${projectName} \
                -Dsonar.links.ci=http://192.168.1.200:8080/job/${groupName}/job/${projectName}/ \
                -Dsonar.sources=src \
                -Dsonar.sourceEncoding=UTF-8 
        """
    }
}

//Main
def CodeSonar(type, projectName, groupName){
    if (SearchProject(projectName)){
        println("${projectName} exists....")
    } else {
        // 项目不存在
        println("${projectName} not found....")

        //创建项目
        CreateProject(projectName)
    }
    //质量配置
    profileMap = ["maven": "java", "npm": "js"]
    UpdateQualityProfiles(profileMap[type], projectName, groupName)

    switch(type){
        case "maven":
            
            SonarJava(projectName, groupName)
            break;
        case "gradle":
            SonarJava(projectName, groupName)
            break;
        case "npm":
            SonarWeb(projectName, groupName)
            break;
        default:
            println("No such tools ... [maven/ant/gradle/npm/yarn/go]")
            break
    }
}

 UnitTest.groovy 

package hwf.devops

//Maven
def MavenTest(){
    def buildTools = ["maven": "/data/apache-maven-3.8.3"]
    sh """
    ${buildTools["maven"]}/bin/mvn test -f demo/pom.xml -s settings.xml

    """
}

//Gradle
def GradleTest(){
    def buildTools = ["gradle": "/data/gradle-7.2"]
    sh """
        ${buildTools["gradle"]}/bin/gradle test
      junit 'build/test-results/test/*.xml'
    """
}

//Ant
//def AntBuild(configPath="./build.xml"){
//    sh "ant -f ${configPath}"
//}

//Golang
//def GoTest(){
    //sh " go test"
//}

//Npm
def NpmTest(){
    def buildTools = ["npm": "/data/node-v14.18.0-linux-x64" ]
    sh """
            export PATH=\$PATH:${buildTools["npm"]}/bin
            ${buildTools["npm"]}/bin/npm test"
    """
}

//Yarn
//def YarnTest(){
//    sh "yarn test "
//}

//Main
def CodeTest(type){
    switch(type){
        case "maven":
            MavenTest()
            break;
        case "gradle":
            GradleTest()
            break;
        default:
            println("No such tools ... [maven/ant/gradle/npm/yarn/go]")
            break
    }
}

jenkinsfile

@Library("hwf@master") _    

import hwf.devops.*           // 导入库

def checkout = new Checkout()
def build = new Build()
def unittest = new UnitTest()
def sonar = new Sonar()
def nexuspackage =  new Nexuspackage()

pipeline {
    agent { label "build" }

    stages{
        stage("Checkout"){
            steps{
                script {
                    println("GetCode")
                    checkout.GetCode("${env.srcUrl}", "${env.branchName}")
                    sh "ls -l"
                }
            }
        }

        stage("Build"){
            steps{
                script{
                    println("Build")
                    build.CodeBuild("${env.buildTool}")
                    sh "ls -l demo/target/"
                }
            }
        }

        stage("UnitTest"){
            steps{
                script{
                    println("test")
                    unittest.CodeTest("${env.buildTool}")
                }
            }
        }

        stage("SonarScan"){
            steps {
                script{
                    println("sonarqube")
                    groupName = "${JOB_NAME}".split("/")[0]
                    projectName = "${JOB_NAME}".split("/")[-1] //devops03/devops03-maven-service
                    sonar.CodeSonar("${env.buildTool}", projectName, groupName)
                }
            }
        }

        
        stage("PullArtifact"){
            steps{
                script{
                    println("upload nexus")
                    if ("${env.buildTool}" == "maven"){	
                    sh "ls -l demo/target/"
                    env.artifaceId='app-test'
                    env.file=sh returnStdout: true, script: "ls demo/target/*.jar"
                    env.file = env.file.trim()
                    env.pkgName=sh returnStdout: true, script: "cd demo/target/;ls *.jar;cd ../../"
                    env.pkgName = env.pkgName.trim()
                    env.groupId='hwf.app'
                    env.prository='hwf-app-service'
                    env.version='1.12'
                    nexuspackage.Upload("${env.prository}",
                                        "${env.groupId}",
                                        "${env.artifaceId}",
                                        "${env.file}",
                                        "${env.version}",
                                        "${env.type}",
                                        "${env.pkgName}")
                   }

                    if ("${env.buildTool}" == "npm"){
                    repoName = "${JOB_NAME}".split("/")[0]
                    projectName = "${JOB_NAME}".split("/")[-1].split("_")[0]
                    version = "${env.branchName}".split("-")[-1]     //RELEASE-1.1.1
                    targetDir = "${projectName}/${version}/"

                        filePath="dist"
                        newPkgName = "${projectName}-${version}.tar.gz"
                        sh """
                            cd ${filePath}
                            [ -f ${newPkgName} ] && rm -fr ${newPkgName}
                            echo "${version}"  >index.html
                            tar zcf ${newPkgName} *
                        """
                        type = "application/x-gzip"
                    nexuspackage.PushRawArtifacts(repoName,targetDir, filePath, newPkgName, type )
                    }
                }
            }
        }
    }
}

hwf-app-web项目

jenkins配置

  • 凭据
  • git项目地址
  • 字符参数:
    • branchName
  • 选项参数:
    • srcUrl
      • http://192.168.1.110:9080/hwf-app/hwf-app-service.git
    • buildTool
      • maven
    • type
      • jar
      • war

gitlab配置

Build.groovy

package hwf.devops


//Maven
def MavenBuild(){
    def buildTools = ["maven": "/data/apache-maven-3.8.3"]
    sh "${buildTools["maven"]}/bin/mvn clean package -DskipTests -f demo/pom.xml \
                        -s settings.xml"
}

//Gradle
def GradleBuild(){
    def buildTools = ["gradle": "/data/gradle-7.2"]
    sh "${buildTools["gradle"]}/bin/gradle clean && ${buildTools["gradle"]}/bin/gradle build -x test"
}

//Ant
def AntBuild(configPath="./build.xml"){
    sh "ant -f ${configPath}"
}

//Golang
def GoBuild(){
    def buildTools = ["go": "/data/go" ]
    sh "${buildTools["go"]}/bin/go build demo.go"
}

//Npm
def NpmBuild(){
    def buildTools = ["npm": "/data/node-v14.16.1-linux-x64" ]
    sh """
                    cd npm
                    export PATH=\$PATH:${buildTools["npm"]}/bin
                    ${buildTools["npm"]}/bin/npm install
                    ${buildTools["npm"]}/bin/npm run build
    """
}

//Yarn
def YarnBuild(){
    sh "yarn install && yarn build "
}

//Main
def CodeBuild(type){
    switch(type){
        case "maven":
            MavenBuild()
            break;
        case "gradle":
            GradleBuild()
            break;
        case "npm":
            NpmBuild()
            break;
        case "yarn":
            YarnBuild()
            break;
        default:
            error "No such tools ... [maven/ant/gradle/npm/yarn/go]"
            break
    }
}

Checkout.groovy

package hwf.devops
//下载代码
 def GetCode(srcUrl, branchName){
    checkout([$class: 'GitSCM', 
            branches: [[name: branchName]], 
            extensions: [], 
            userRemoteConfigs: [[
            credentialsId: 'fe149331-7ec9-4a5b-b3bf-557beb0caa42', 
            url: srcUrl]]])
}

Nexuspackage.groovy

package hwf.devops
 
def Upload(prository,groupId,artifaceId,file,version,type,pkgName){
     withCredentials([usernamePassword(credentialsId: 'c2b82aec-d6d7-42c4-a97d-b78fc07c050b', passwordVariable: 'TOKEN', usernameVariable: 'USER')]) {
        sh """
            curl -X 'POST' \
            "http://192.168.1.110:8081/nexus/service/rest/v1/components?repository=${prository}" \
            -H 'accept: application/json' \
            -H 'Content-Type: multipart/form-data' \
            -F "maven2.groupId=${groupId}" \
            -F "maven2.artifactId=${artifaceId}" \
            -F "maven2.version=${version}" \
            -F "maven2.packaging=${type}" \
            -F "maven2.asset1=@${file}" \
            -F "maven2.asset1.extension=${pkgName}" \
            -u "${USER}":"${TOKEN}"
        """
    }
}

def PushRawArtifacts(repoName,targetDir, filePath, pkgName, type ){
     withCredentials([usernamePassword(credentialsId: 'c2b82aec-d6d7-42c4-a97d-b78fc07c050b', passwordVariable: 'TOKEN', usernameVariable: 'USER')]) {
        sh """
            curl -X POST \
            "http://192.168.1.110:8081/nexus/service/rest/v1/components?repository=${repoName}" \
            -H "accept: application/json" \
            -H "Content-Type: multipart/form-data" \
            -F "raw.directory=${targetDir}" \
            -F "raw.asset1=@${filePath}/${pkgName};type=${type}" \
            -F "raw.asset1.filename=${pkgName}" \
            -u "${USER}":"${TOKEN}"
        """
    }
}

Sonar.groovy 

package hwf.devops

def SonarRequest(apiUrl,method){
    withCredentials([string(credentialsId: "01c25ddf-7ca7-4784-95f6-d4722eb3ab14", variable: 'SONAR_TOKEN')]) {
        sonarApi = "http://192.168.1.110:9000/api"
        response = sh  returnStdout: true, 
            script: """
            curl --location \
                 --request ${method} \
                 "${sonarApi}/${apiUrl}" \
                 --header "Authorization: Basic ${SONAR_TOKEN}"
            """
        // json格式化
        try {
            response = readJSON text: """ ${response - "\n"} """
        } catch(e){
            response = readJSON text: """{"errors" : true}"""
        }
        return response
        
    }
}

//查找项目
def SearchProject(projectName){
    apiUrl = "projects/search?projects=${projectName}"
    response = SonarRequest(apiUrl,"GET")
    if (response.paging.total == 0){
        return false
    }
    return true 
}


//创建项目
def CreateProject(projectName){
    apiUrl = "projects/create?name=${projectName}&project=${projectName}"
    response = SonarRequest(apiUrl,"POST")
    try{
        if (response.project.key == projectName ) {
            println("Project Create success!...")
            return true
        }
    }catch(e){
        println(response.errors)
        return false
    }
}

// 更新质量阈
def UpdateQualityProfiles(lang, projectName, profileName){
    apiUrl = "qualityprofiles/add_project?language=${lang}&project=${projectName}&qualityProfile=${profileName}"
    response = SonarRequest(apiUrl,"POST")
    
    if (response.errors != true){
        println("ERROR: UpdateQualityProfiles ${response.errors}...")
        return false
    } else {
        println("SUCCESS: UpdateQualityProfiles ${lang} > ${projectName} > ${profileName}" )
        return true
    }
}

def SonarJava(projectName, groupName ){
    withCredentials([string(credentialsId: '40e88177-bd39-4ac4-92bd-3124e4fc9171', variable: 'AUTH_TOKEN')]) {
        sh """
            sonar-scanner -Dsonar.host.url=http://192.168.1.110:9000 \
                -Dsonar.projectKey=${projectName} \
                -Dsonar.projectName=${projectName} \
                -Dsonar.projectVersion=${BUILD_ID} \
                -Dsonar.login=${AUTH_TOKEN} \
                -Dsonar.ws.timeout=30 \
                -Dsonar.projectDescription="my first project!" \
                -Dsonar.links.homepage=http://192.168.1.110:9080/${groupName}/${projectName} \
                -Dsonar.links.ci=http://192.168.1.110:8080/job/${groupName}/job/${projectName}/ \
                -Dsonar.sources=./demo/src \
                -Dsonar.sourceEncoding=UTF-8 \
                -Dsonar.java.binaries=./demo/target/classes \
                -Dsonar.java.test.binaries=./demo/target/test-classes 
        """
    }
}


def SonarWeb(projectName, groupName ){
    withCredentials([string(credentialsId: '40e88177-bd39-4ac4-92bd-3124e4fc9171', variable: 'AUTH_TOKEN')]) {
        sh """
            sonar-scanner -Dsonar.host.url=http://192.168.1.110:9000 \
                -Dsonar.projectKey=${projectName} \
                -Dsonar.projectName=${projectName} \
                -Dsonar.projectVersion=${BUILD_ID} \
                -Dsonar.login=${AUTH_TOKEN} \
                -Dsonar.ws.timeout=30 \
                -Dsonar.projectDescription="my first project!" \
                -Dsonar.links.homepage=http://192.168.1.110:9080/${groupName}/${projectName} \
                -Dsonar.links.ci=http://192.168.1.110:8080/job/${groupName}/job/${projectName}/ \
                -Dsonar.sources=./npm/src \
                -Dsonar.sourceEncoding=UTF-8 
        """
    }
}

//Main
def CodeSonar(type, projectName, groupName){
    if (SearchProject(projectName)){
        println("${projectName} exists....")
    } else {
        // 项目不存在
        println("${projectName} not found....")

        //创建项目
        CreateProject(projectName)
    }
    //质量配置
    profileMap = ["maven": "java", "npm": "js"]
    UpdateQualityProfiles(profileMap[type], projectName, groupName)

    switch(type){
        case "maven":
            
            SonarJava(projectName, groupName)
            break;
        case "gradle":
            SonarJava(projectName, groupName)
            break;
        case "npm":
            SonarWeb(projectName, groupName)
            break;
        default:
            println("No such tools ... [maven/ant/gradle/npm/yarn/go]")
            break
    }
}

UnitTest.groovy 

package hwf.devops

//Maven
def MavenTest(){
    def buildTools = ["maven": "/data/apache-maven-3.8.3"]
    sh """
    ${buildTools["maven"]}/bin/mvn test -f demo/pom.xml -s settings.xml

    """
}

//Gradle
def GradleTest(){
    def buildTools = ["gradle": "/data/gradle-7.2"]
    sh """
        ${buildTools["gradle"]}/bin/gradle test
      junit 'build/test-results/test/*.xml'
    """
}

//Ant
//def AntBuild(configPath="./build.xml"){
//    sh "ant -f ${configPath}"
//}

//Golang
//def GoTest(){
    //sh " go test"
//}

//Npm
def NpmTest(){
    def buildTools = ["npm": "/data/node-v14.16.1-linux-x64" ]
    sh """
            export PATH=\$PATH:${buildTools["npm"]}/bin
            ls -l
            pwd
            cd npm
            ${buildTools["npm"]}/bin/npm test
    """
}

//Yarn
//def YarnTest(){
//    sh "yarn test "
//}

//Main
def CodeTest(type){
    switch(type){
        case "maven":
            MavenTest()
            break;
        case "gradle":
            GradleTest()
            break;
        default:
            println("No such tools ... [maven/ant/gradle/npm/yarn/go]")
            break
    }
}

jenkinsfile

@Library("hwf-app@master") _    

import hwf.devops.*           // 导入库

def checkout = new Checkout()
def build = new Build()
def unittest = new UnitTest()
def sonar = new Sonar()
def nexuspackage =  new Nexuspackage()

pipeline {
    agent { label "build" }

    stages{
        stage("Checkout"){
            steps{
                script {
                    println("GetCode")
                    checkout.GetCode("${env.srcUrl}", "${env.branchName}")
                }
            }
        }

        stage("Build"){
            steps{
                script{
                    println("Build")
                    build.CodeBuild("${env.buildTool}")
                }
            }
        }

        stage("UnitTest"){
            steps{
                script{
                    unittest.CodeTest("${env.buildTool}")
                }
            }
        }

        stage("SonarScan"){
            steps {
                script{
                    groupName = "${JOB_NAME}".split("/")[0]
                    projectName = "${JOB_NAME}".split("/")[-1] //devops03/devops03-maven-service
                    sonar.CodeSonar("${env.buildTool}", projectName, groupName)

                }
            }
        }

        stage("Nexuspackage"){
            steps{
                script{
                    println("upload nexus")
                    if ("${env.buildTool}" == "maven"){	
                    sh "ls -l demo/target/"
                    env.artifaceId='app-test'
                    env.file=sh returnStdout: true, script: "ls demo/target/*.jar"
                    env.file = env.file.trim()
                    env.pkgName=sh returnStdout: true, script: "cd demo/target/;ls *.jar;cd ../../"
                    env.pkgName = env.pkgName.trim()
                    env.groupId='hwf.app'
                    env.prository='hwf-maven-release'
                    env.version='1.55'
                    nexuspackage.Upload("${env.prository}",
                                        "${env.groupId}",
                                        "${env.artifaceId}",
                                        "${env.file}",
                                        "${env.version}",
                                        "${env.type}",
                                        "${env.pkgName}")
                   }

                    if ("${env.buildTool}" == "npm"){
                    repoName = "${JOB_NAME}".split("/")[0]
                    projectName = "${JOB_NAME}".split("/")[-1].split("_")[0]
                    version = "${env.branchName}".split("-")[-1]     //RELEASE-1.1.1
                    targetDir = "${projectName}/${version}/"

                        filePath="npm/dist"
                        newPkgName = "${projectName}-${version}.tar.gz"
                        sh """
                            ls -l 
                            pwd
                            cd ${filePath}
                            [ -f ${newPkgName} ] && rm -fr ${newPkgName}
                            echo "${version}"  >index.html
                            tar zcf ${newPkgName} *
                        """
                        type = "application/x-gzip"
                    nexuspackage.PushRawArtifacts(repoName,targetDir, filePath, newPkgName, type )
                    }
                }
            }
        }
    }
}

 

继续阅读
weinxin
我的微信
这是我的微信扫一扫
  • 文本由 发表于 2021年11月7日19:21:15
  • 除非特殊声明,本站文章均为原创,转载请务必保留本文链接
CI/CD-云主机场景-[持续部署|cicd设计] CI/CD实践

CI/CD-云主机场景-[持续部署|cicd设计]

CI/CD-云主机场景的持续部署实践 0. 准备工作项目标准化 公司里面要使用流水线要做持续集成CI/CD的项目越来越多,这对流水线的设计和开发有不同的要求。我们经常听到用户的反馈: 各种不同语言的技...
匿名

发表评论

匿名网友 填写信息

:?: :razz: :sad: :evil: :!: :smile: :oops: :grin: :eek: :shock: :???: :cool: :lol: :mad: :twisted: :roll: :wink: :idea: :arrow: :neutral: :cry: :mrgreen: