1
0
mirror of https://github.com/SAP/jenkins-library.git synced 2024-12-14 11:03:09 +02:00
sap-jenkins-library/vars/setupCommonPipelineEnvironment.groovy

304 lines
12 KiB
Groovy
Raw Normal View History

import com.cloudbees.groovy.cps.NonCPS
import static com.sap.piper.Prerequisites.checkScript
import com.sap.piper.GenerateDocumentation
import com.sap.piper.ConfigurationHelper
import com.sap.piper.Utils
import com.sap.piper.analytics.InfluxData
import com.sap.piper.GitUtils
import groovy.transform.Field
@Field String STEP_NAME = getClass().getName()
@Field Set GENERAL_CONFIG_KEYS = [
/** */
'collectTelemetryData',
/** Credentials (username and password) used to download custom defaults if access is secured.*/
'customDefaultsCredentialsId',
/** Enable automatic inference of build tool (maven, npm, mta) based on existing project files.
* If this is set to true, it is not required to set the build tool by hand for those cases.
*/
'inferBuildTool'
]
@Field Set STEP_CONFIG_KEYS = []
@Field Set PARAMETER_KEYS = [
/** Path to the pipeline configuration file defining project specific settings.*/
'configFile',
/** A list of file names which will be extracted from library resources and which serve as source for
* default values for the pipeline configuration. These are merged with and override built-in defaults, with
* a parameter supplied by the last resource file taking precedence over the same parameter supplied in an
* earlier resource file or built-in default.*/
'customDefaults',
/** A list of file paths or URLs which must point to YAML content. These work exactly like
* `customDefaults`, but from local or remote files instead of library resources. They are merged with and
* take precedence over `customDefaults`.*/
'customDefaultsFromFiles',
/** The map returned from a Jenkins git checkout. Used to set the git information in the
* common pipeline environment */
'scmInfo'
]
/**
* Initializes the [`commonPipelineEnvironment`](commonPipelineEnvironment.md), which is used throughout the complete pipeline.
*
* !!! tip
* This step needs to run at the beginning of a pipeline right after the SCM checkout.
* Then subsequent pipeline steps consume the information from `commonPipelineEnvironment`; it does not need to be passed to pipeline steps explicitly.
*/
@GenerateDocumentation
2018-08-30 16:33:07 +02:00
void call(Map parameters = [:]) {
2017-07-11 15:12:03 +02:00
handlePipelineStepErrors (stepName: STEP_NAME, stepParameters: parameters) {
2017-07-11 15:12:03 +02:00
def script = checkScript(this, parameters)
2017-07-11 15:12:03 +02:00
String configFile = parameters.get('configFile')
loadConfigurationFromFile(script, configFile)
// Copy custom defaults from library resources to include them in the 'pipelineConfigAndTests' stash
List customDefaultsResources = Utils.appendParameterToStringList(
['default_pipeline_environment.yml'], parameters, 'customDefaults')
customDefaultsResources.each {
cd ->
writeFile file: ".pipeline/${cd}", text: libraryResource(cd)
}
List customDefaultsFiles = Utils.appendParameterToStringList(
[], parameters, 'customDefaultsFromFiles')
if (script.commonPipelineEnvironment.configuration.customDefaults) {
if (!script.commonPipelineEnvironment.configuration.customDefaults in List) {
// Align with Go side on supported parameter type.
error "You have defined the parameter 'customDefaults' in your project configuration " +
"but it is of an unexpected type. Please make sure that it is a list of strings, i.e. " +
"customDefaults = ['...']. See https://sap.github.io/jenkins-library/configuration/ for " +
"more details."
}
customDefaultsFiles = Utils.appendParameterToStringList(
customDefaultsFiles, script.commonPipelineEnvironment.configuration as Map, 'customDefaults')
}
String customDefaultsCredentialsId = script.commonPipelineEnvironment.configuration.general?.customDefaultsCredentialsId
customDefaultsFiles = copyOrDownloadCustomDefaultsIntoPipelineEnv(script, customDefaultsFiles, customDefaultsCredentialsId)
prepareDefaultValues([
script: script,
customDefaults: parameters.customDefaults,
customDefaultsFromFiles: customDefaultsFiles ])
piperLoadGlobalExtensions script: script, customDefaults: parameters.customDefaults, customDefaultsFromFiles: customDefaultsFiles
String stashIncludes = '.pipeline/**'
if (configFile && !configFile.startsWith('.pipeline/')) {
stashIncludes += ", $configFile"
}
stash name: 'pipelineConfigAndTests', includes: stashIncludes, allowEmpty: true
Map config = ConfigurationHelper.newInstance(this)
.loadStepDefaults()
.mixinGeneralConfig(script.commonPipelineEnvironment, GENERAL_CONFIG_KEYS)
.use()
inferBuildTool(script, config)
(parameters.utils ?: new Utils()).pushToSWA([
step: STEP_NAME,
stepParamKey4: 'customDefaults',
2019-03-26 15:13:03 +02:00
stepParam4: parameters.customDefaults?'true':'false'
], config)
InfluxData.addField('step_data', 'build_url', env.BUILD_URL)
InfluxData.addField('pipeline_data', 'build_url', env.BUILD_URL)
def scmInfo = parameters.scmInfo
if (scmInfo) {
setGitUrlsOnCommonPipelineEnvironment(script, scmInfo.GIT_URL)
script.commonPipelineEnvironment.setGitCommitId(scmInfo.GIT_COMMIT)
def gitUtils = parameters.gitUtils ?: new GitUtils()
setGitRefOnCommonPipelineEnvironment(script, scmInfo.GIT_COMMIT, scmInfo.GIT_BRANCH, gitUtils)
}
}
}
// Infer build tool (maven, npm, mta) based on existing build descriptor files in the project root.
private static void inferBuildTool(script, config) {
// For backwards compatibility, build tool inference must be enabled via inferBuildTool setting
boolean inferBuildTool = config?.inferBuildTool
if (inferBuildTool) {
boolean isMtaProject = script.fileExists('mta.yaml')
def isMavenProject = script.fileExists('pom.xml')
def isNpmProject = script.fileExists('package.json')
if (isMtaProject) {
script.commonPipelineEnvironment.buildTool = 'mta'
} else if (isMavenProject) {
script.commonPipelineEnvironment.buildTool = 'maven'
} else if (isNpmProject) {
script.commonPipelineEnvironment.buildTool = 'npm'
}
}
}
private static loadConfigurationFromFile(script, String configFile) {
if (!configFile) {
String defaultYmlConfigFile = '.pipeline/config.yml'
String defaultYamlConfigFile = '.pipeline/config.yaml'
if (script.fileExists(defaultYmlConfigFile)) {
configFile = defaultYmlConfigFile
} else if (script.fileExists(defaultYamlConfigFile)) {
configFile = defaultYamlConfigFile
}
}
// A file passed to the function is not checked for existence in order to fail the pipeline.
2019-03-26 15:13:03 +02:00
if (configFile) {
script.commonPipelineEnvironment.configuration = script.readYaml(file: configFile)
script.commonPipelineEnvironment.configurationFile = configFile
2017-07-11 15:12:03 +02:00
}
}
private static List copyOrDownloadCustomDefaultsIntoPipelineEnv(script, List customDefaults, String credentialsId) {
List fileList = []
int urlCount = 0
for (int i = 0; i < customDefaults.size(); i++) {
// copy retrieved file to .pipeline/ to make sure they are in the pipelineConfigAndTests stash
if (!(customDefaults[i] in CharSequence) || customDefaults[i] == '') {
script.echo "WARNING: Ignoring invalid entry in custom defaults from files: '${customDefaults[i]}'"
continue
}
String fileName
if (customDefaults[i].startsWith('http://') || customDefaults[i].startsWith('https://')) {
fileName = "custom_default_from_url_${urlCount}.yml"
Map httpRequestParameters = [
url: customDefaults[i],
validResponseCodes: '100:399,404' // Allow a more specific error message for 404 case
]
if (credentialsId) {
httpRequestParameters.authentication = credentialsId
}
def response = script.httpRequest(httpRequestParameters)
if (response.status == 404) {
error "URL for remote custom defaults (${customDefaults[i]}) appears to be incorrect. " +
"Server returned HTTP status code 404. " +
"Please make sure that the path is correct and no authentication is required to retrieve the file."
}
script.writeFile file: ".pipeline/$fileName", text: response.content
urlCount++
} else if (script.fileExists(customDefaults[i])) {
fileName = customDefaults[i]
script.writeFile file: ".pipeline/$fileName", text: script.readFile(file: fileName)
} else {
script.echo "WARNING: Custom default entry not found: '${customDefaults[i]}', it will be ignored"
continue
}
fileList.add(fileName)
}
return fileList
}
/*
* Returns the parts of an url.
* Valid keys for the retured map are:
* - protocol
* - auth
* - host
* - port
* - path
*/
@NonCPS
/* private */ Map parseUrl(String url) {
def urlMatcher = url =~ /^((http|https|git|ssh):\/\/)?((.*)@)?([^:\/]+)(:([\d]*))?(\/?(.*))$/
return [
protocol: urlMatcher[0][2],
auth: urlMatcher[0][4],
host: urlMatcher[0][5],
port: urlMatcher[0][7],
path: urlMatcher[0][9],
]
}
private void setGitUrlsOnCommonPipelineEnvironment(script, String gitUrl) {
Map url = parseUrl(gitUrl)
if (url.protocol in ['http', 'https']) {
script.commonPipelineEnvironment.setGitSshUrl("git@${url.host}:${url.path}")
script.commonPipelineEnvironment.setGitHttpsUrl(gitUrl)
} else if (url.protocol in [ null, 'ssh', 'git']) {
script.commonPipelineEnvironment.setGitSshUrl(gitUrl)
script.commonPipelineEnvironment.setGitHttpsUrl("https://${url.host}/${url.path}")
}
List gitPathParts = url.path.replaceAll('.git', '').split('/')
def gitFolder = 'N/A'
def gitRepo = 'N/A'
switch (gitPathParts.size()) {
case 0:
break
case 1:
gitRepo = gitPathParts[0]
break
case 2:
gitFolder = gitPathParts[0]
gitRepo = gitPathParts[1]
break
default:
gitRepo = gitPathParts[gitPathParts.size()-1]
gitPathParts.remove(gitPathParts.size()-1)
gitFolder = gitPathParts.join('/')
break
}
script.commonPipelineEnvironment.setGithubOrg(gitFolder)
script.commonPipelineEnvironment.setGithubRepo(gitRepo)
}
private void setGitRefOnCommonPipelineEnvironment(script, String gitCommit, String gitBranch, def gitUtils) {
if(!gitBranch){
return
}
if(gitBranch.contains("/")){
gitBranch = gitBranch.split("/")[1]
}
if (!gitBranch.contains("PR")) {
script.commonPipelineEnvironment.setGitRef("refs/heads/" + gitBranch)
script.commonPipelineEnvironment.setGitRemoteCommitId(gitCommit)
return
}
boolean isMergeCommit = gitUtils.isMergeCommit()
def mergeOrHead = isMergeCommit?"merge":"head"
def changeId = gitBranch.split("-")[1]
script.commonPipelineEnvironment.setGitRef("refs/pull/" + changeId + "/" + mergeOrHead)
if(!isMergeCommit){
script.commonPipelineEnvironment.setGitRemoteCommitId(gitCommit)
return
}
try{
String gitRemoteCommitId = gitUtils.getGitMergeCommitId(changeId)
if(gitRemoteCommitId?.trim() && gitUtils.compareParentsOfMergeAndHead(gitRemoteCommitId)){
script.commonPipelineEnvironment.setGitRemoteCommitId(gitRemoteCommitId)
return
}
}catch(Exception e){
echo "Exception in getting git merge commit id or comparing git merge commit parents: ${e}"
}
script.commonPipelineEnvironment.setGitRemoteCommitId("NA")
}