SonarQube waitForQualityGate() 返回 401

标签 sonarqube jenkins-pipeline

我有一个 SonarQube 服务器和一个 Jenkins 服务器。 Jenkins 服务器安装了以下插件。

  • 质量门插件
  • Sonar 质量门插件
  • SonarQubeScanner

  • 我已经使用以下 URL 在 SonarQube UI 中配置了一个 webhook。

    http://myjenkins.com:8083/sonarqube-webhook/

    我有一个 Jenkins 管道文件,如下所示。
    pipeline {
        /*
        * Run everything on an existing agent configured with a label 'windows'.
        * TODO : Once we have enough agents, distribute to get fastest feedback!
        */
        agent any
        /*
        agent {
            node {
                label 'windows'
            }
        }
        */
        /*
        * We are getting a lot of values from BitBucket notifier plugin.
        */
        parameters {
            string(defaultValue: '', description: '', name: 'PULL_REQUEST_URL')
            string(defaultValue: '', description: '', name: 'PULL_REQUEST_VERSION')
            string(defaultValue: '', description: '', name: 'PULL_REQUEST_AUTHOR_SLUG')
            string(defaultValue: '', description: '', name: 'PULL_REQUEST_TO_SSH_CLONE_URL')
            string(defaultValue: '', description: '', name: 'PULL_REQUEST_FROM_BRANCH')
            string(defaultValue: '', description: '', name: 'PULL_REQUEST_TO_BRANCH')
            string(defaultValue: '', description: '', name: 'PULL_REQUEST_TO_REPO_NAME')
            string(defaultValue: '', description: '', name: 'PULL_REQUEST_ID')
            string(defaultValue: '', description: '', name: 'PULL_REQUEST_TITLE')
            string(defaultValue: '', description: '', name: 'PULL_REQUEST_ACTION')
            string(defaultValue: '', description: '', name: 'PULL_REQUEST_REVIEWERS_SLUG')
            string(defaultValue: '', description: '', name: 'PULL_REQUEST_AUTHOR_EMAIL')
            string(defaultValue: '', description: '', name: 'PULL_REQUEST_AUTHOR_DISPLAY_NAME')
            string(defaultValue: '', description: '', name: 'PULL_REQUEST_USER_EMAIL_ADDRESS')
            string(defaultValue: '', description: '', name: 'PULL_REQUEST_USER_DISPLAY_NAME')
            string(defaultValue: '', description: '', name: 'PULL_REQUEST_STATE')
            string(defaultValue: '', description: '', name: 'PULL_REQUEST_REVIEWERS_EMAIL')
            string(defaultValue: '', description: '', name: 'PULL_REQUEST_REVIEWERS_APPROVED_COUNT')
            string(defaultValue: '', description: '', name: 'PULL_REQUEST_FROM_BRANCH')
            string(defaultValue: '', description: '', name: 'PULL_REQUEST_TO_REPO_PROJECT_KEY')
        }
    
        /* Add timestamps to the console log. Discard old builds.
        options {
            timestamps()
            buildDiscarder(logRotator(numToKeepStr: '20'))
        }
        */
        /* Stages can be parallel or sequential */
        stages {
            stage('Source-Code-Management') {
                steps {
                    git(url: 'ssh://mybitbucketserver.com:7999/dev/sitecore.git', branch: '${PULL_REQUEST_FROM_BRANCH}', credentialsId: 'root')
                    powershell "git merge origin/${PULL_REQUEST_TO_BRANCH}"
                }
            }
            stage('Build and Analyze') {
                steps {
                    script {
                        // Todo we should be using environment variables or properties for sonar scanner and not have hardcoded values
                        bat "\"${tool 'nuget-4.3.0'}\" restore src/Nemlig.sln -NoCache"
                        withSonarQubeEnv('Local') {
                            bat "\"${tool 'SonarScanner.MSBuild.exe'}\" begin /d:sonar.login=mytoken /k:TEST /n:Website /v: /d:sonar.host.url=${SONAR_URL} /d:sonar.cs.nunit.reportsPaths=${WORKSPACE}\\NUnitResult.xml /d:sonar.cs.opencover.reportsPaths=${WORKSPACE}\\OpenCoverResult.xml"
                            bat "\"${tool 'MSBuild-v15'}\" src/Nemlig.sln /p:Configuration=debug"
                            // Note that this shell can retrieve value with CRLF and NUNIT will puke. Must replace with spaces...
                            def dlls = powershell(returnStdout: true, script: '(ls -Recurse src\\*\\bin\\*.Test.dll | % FullName)')
                            dlls = dlls.replaceAll('\r\n', ' ')
                            mycmd = "OpenCover.Console.exe -register:path64 -target:\"nunit3-console.exe\" -returntargetcode -targetargs:\"$dlls --result=NUnitResult.xml\" -output:OpenCoverResult.xml"
                            powershell (returnStatus: true, script: "$mycmd")
                            bat "\"${tool 'SonarScanner.MSBuild.exe'}\" end /d:sonar.login=mytoken"
                        }
                    }
                }
            }
            stage('SonarQube Quality Gate') {
                steps {
                    script {
                        timeout(time: 1, unit: 'HOURS') { // Just in case something goes wrong, pipeline will be killed after a timeout
                            def qg = waitForQualityGate() // Reuse taskId previously collected by withSonarQubeEnv
                            if (qg.status != 'OK') {
                                error "Pipeline aborted due to quality gate failure: ${qg.status}"
                            }
                        }
                    }
                }
            }
            stage('Deploy Team Environment') {
                parallel {
                    stage('Deploy Site') {
                        steps {
                            powershell 'Write-Output "Here we could do a deploy of the site"'
                        }
                    }
                    stage('Deploy Backend') {
                        steps {
                            powershell 'Write-Output "Here we could do a deploy of the backend and maybe database changes?"'
                        }
                    }
                }
            }
            stage('Smoke') {
                parallel {
                    stage('Chrome') {
                        steps {
                            powershell 'Write-Output "Execute Smoke on Chrome"'
                        }
                    }
                    stage('Firefox') {
                        steps {
                            powershell 'Write-Output "Execute Smoke on Firefox"'
                        }
                    }
                    stage('Safari') {
                        steps {
                            powershell 'Write-Output "Execute Smoke on Safari"'
                        }
                    }
                }
            }
            stage('integrate') {
                steps {
                    powershell 'Write-Output "Push merge if all is success!"'
                }
            }
            stage('Store Binary') {
                steps {
                    powershell 'Write-Output "Put the tested binary in Artifactory. Maybe tag it with SHA and Build Number?"'
                }
            }
        }
        post {
            // Add the publish merge steps...
            // Add the notifiy Jira steps
            always {
                script {
                    currentBuild.result = currentBuild.result ?: 'SUCCESS'
                    notifyBitbucket()
                }
                nunit testResultsPattern: "NUnitResult.xml"
            }
        }
    }
    

    我执行管道,分析是正确处理并上传到 SonarQube 服务器 .然而,waitForQualityGate 总是返回 401。
    Checking status of SonarQube task 'AWKwODGjKw9CRrvU3S27' on server 'Local'
    
    Error 401 on http://mysonarserver.com:9000/api/ce/task?id=AWKwODGjKw9CRrvU3S27
    

    从日志中可以看出分析时生成的taskID是一样的。
    ANALYSIS SUCCESSFUL, you can browse http://10.100.1.189:9000/dashboard/index/TEST
    INFO: Note that you will be able to access the updated dashboard once the server has processed the submitted analysis report
    INFO: More about the report processing at http://mysonarserver.com:9000/api/ce/task?id=AWKwODGjKw9CRrvU3S27
    INFO: Task total time: 3:00.752 s
    

    我启动了一个本地 RequestBin 服务器,为此设置了一个新的 webhook 并看到了 它确实收到了来自 SonarQube 的数据。
    {
        "serverUrl": "http://localhost:9000", "taskId": "AWKwODGjKw9CRrvU3S27", "status": "SUCCESS", "analysedAt":
        "2018-04-11T07:41:40+0000", "changedAt": "2018-04-11T07:41:40+0000", "project":
        { "key": "TEST", "name": "Website", "url": "http://localhost:9000/dashboard?id=TEST" }, "branch":
        { "name": "master", "type": "LONG", "isMain": true, "url": "http://localhost:9000/dashboard?id=TEST" },
        "qualityGate": {
        "name": "Website", "status": "OK", "conditions": [{
                                                              "metric": "new_reliability_rating", "operator":
                                                              "GREATER_THAN", "value": "3", "status": "OK", "onLeakPeriod":
                                                              true, "errorThreshold": "3"
                                                          }, {
                                                              "metric": "new_duplicated_lines_density", "operator":
                                                              "GREATER_THAN", "status": "NO_VALUE", "onLeakPeriod": true,
                                                              "errorThreshold": "3"
                                                          }, {
                                                              "metric": "new_security_rating", "operator": "GREATER_THAN",
                                                              "value": "4", "status": "OK", "onLeakPeriod": true,
                                                              "errorThreshold": "4"
                                                          }, {
                                                              "metric": "blocker_violations", "operator": "GREATER_THAN",
                                                              "value": "-2", "status": "OK", "onLeakPeriod": true,
                                                              "errorThreshold": "35"
                                                          }, {
                                                              "metric": "new_maintainability_rating", "operator":
                                                              "GREATER_THAN", "value": "1", "status": "OK", "onLeakPeriod":
                                                              true, "errorThreshold": "1"
                                                          }]
    }, "properties": {}
    }
    

    所以我现在不知道去哪里?

    任何帮助,将不胜感激!

    最佳答案

    waitForQualityGate() step 需要一些凭据才能从 SQ 服务器获取质量门详细信息。凭据应在 Jenkins 全局配置中设置,适用于您的 'Local'服务器。

    但是根据您的管道片段,我看到您正在手动传递 /d:sonar.login=mytoken到扫描仪。这不受支持。请在全局服务器配置中设置 token 。

    关于SonarQube waitForQualityGate() 返回 401,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/49769197/

    相关文章:

    java - 如何使用 Maven 的 sonarQube 在提交之前收集增量分析

    matlab - Sonarqube 中有解析和集成 Matlab 的插件吗?

    ios - SonarQube 7.4 使用 faSTLane 扫描 ObjectiveC

    jenkins - Jenkins xUnit 插件的控制参数

    jenkins - 如何在具有一个或多个 SCM 的 Jenkins 工作流程中找到罪魁祸首或提交者

    java - 我想知道使用 Sonar 的最佳方法是什么?

    gradle - jacoco 和 gradle 多模块项目的跨模块代码覆盖率

    jenkins - 多个项目共享一个 jenkinsfile

    git - Jenkins 管道中的 changeSet 错误(错误 :java. io.NotSerializableException : hudson. plugins.git.GitChangeSetList)

    include - 如何 "include"另一个文件作为 Jenkins 管道定义的一部分