1
0
mirror of https://github.com/SAP/jenkins-library.git synced 2025-01-18 05:18:24 +02:00
sap-jenkins-library/vars/multicloudDeploy.groovy
Kevin Hudemann 89e8209f8e
Add isolation for CF Deployments in multicloudDeploy step (#1742)
This change adds isolation to CF deployments, when using blue-green
deployments in multicloudDeploy step.
2020-07-01 16:08:12 +02:00

157 lines
5.6 KiB
Groovy

import com.sap.piper.GenerateDocumentation
import com.sap.piper.CloudPlatform
import com.sap.piper.DeploymentType
import com.sap.piper.k8s.ContainerMap
import com.sap.piper.ConfigurationHelper
import com.sap.piper.Utils
import com.sap.piper.JenkinsUtils
import groovy.transform.Field
import static com.sap.piper.Prerequisites.checkScript
@Field String STEP_NAME = getClass().getName()
@Field Set GENERAL_CONFIG_KEYS = [
/** Defines the targets to deploy on cloudFoundry.*/
'cfTargets',
/** Defines the targets to deploy on neo.*/
'neoTargets'
]
@Field Set STEP_CONFIG_KEYS = []
@Field Set PARAMETER_KEYS = GENERAL_CONFIG_KEYS.plus([
/** Defines the deployment type.*/
'enableZeroDowntimeDeployment',
/** Executes the deployments in parallel.*/
'parallelExecution',
/** The source file to deploy to SAP Cloud Platform.*/
'source'
])
/**
* Deploys an application to multiple platforms (Cloud Foundry, SAP Cloud Platform) or to multiple instances of multiple platforms or the same platform.
*/
@GenerateDocumentation
void call(parameters = [:]) {
handlePipelineStepErrors(stepName: STEP_NAME, stepParameters: parameters) {
def script = checkScript(this, parameters) ?: this
def utils = parameters.utils ?: new Utils()
def jenkinsUtils = parameters.jenkinsUtils ?: new JenkinsUtils()
def stageName = parameters.stage ?: env.STAGE_NAME
ConfigurationHelper configHelper = ConfigurationHelper.newInstance(this)
.loadStepDefaults()
.mixinGeneralConfig(script.commonPipelineEnvironment, GENERAL_CONFIG_KEYS)
.mixin(parameters, PARAMETER_KEYS)
Map config = configHelper.use()
configHelper
.withMandatoryProperty('source', null, { config.neoTargets })
utils.pushToSWA([
step: STEP_NAME,
stepParamKey1: 'enableZeroDowntimeDeployment',
stepParam1: config.enableZeroDowntimeDeployment
], config)
def index = 1
def deployments = [:]
if (config.cfTargets) {
def deploymentType = DeploymentType.selectFor(CloudPlatform.CLOUD_FOUNDRY, config.enableZeroDowntimeDeployment).toString()
def deployTool = script.commonPipelineEnvironment.configuration.isMta ? 'mtaDeployPlugin' : 'cf_native'
// An isolated workspace is only required when using blue-green deployment with multiple cfTargets,
// since the cloudFoundryDeploy step might edit the manifest.yml file in that case.
Boolean runInIsolatedWorkspace = config.cfTargets.size() > 1 && deploymentType == "blue-green"
for (int i = 0; i < config.cfTargets.size(); i++) {
def target = config.cfTargets[i]
Closure deployment = {
Utils deploymentUtils = new Utils()
if (runInIsolatedWorkspace) {
deploymentUtils.unstashStageFiles(script, stageName)
}
cloudFoundryDeploy(
script: script,
juStabUtils: utils,
jenkinsUtilsStub: jenkinsUtils,
deployType: deploymentType,
cloudFoundry: target,
mtaPath: script.commonPipelineEnvironment.mtarFilePath,
deployTool: deployTool
)
if (runInIsolatedWorkspace) {
deploymentUtils.stashStageFiles(script, stageName)
}
}
if (runInIsolatedWorkspace){
deployments["Deployment ${index}"] = {
if (env.POD_NAME) {
dockerExecuteOnKubernetes(script: script, containerMap: ContainerMap.instance.getMap().get(stageName) ?: [:]) {
deployment.call()
}
} else {
node(env.NODE_NAME) {
deployment.call()
}
}
}
} else {
deployments.put("Deployment ${index}", deployment)
}
index++
}
}
if (config.neoTargets) {
def deploymentType = DeploymentType.selectFor(CloudPlatform.NEO, config.enableZeroDowntimeDeployment).toString()
for (int i = 0; i < config.neoTargets.size(); i++) {
def target = config.neoTargets[i]
Closure deployment = {
neoDeploy (
script: script,
warAction: deploymentType,
source: config.source,
neo: target
)
}
deployments.put("Deployment ${index}", deployment)
index++
}
}
if (!config.cfTargets && !config.neoTargets) {
error "Deployment skipped because no targets defined!"
}
echo "Executing deployments"
if (config.parallelExecution) {
echo "Executing deployments in parallel"
parallel deployments
} else {
echo "Executing deployments in sequence"
def closuresToRun = deployments.values().asList()
Collections.shuffle(closuresToRun) // Shuffle the list so no one tries to rely on the order of execution
for (int i = 0; i < closuresToRun.size(); i++) {
(closuresToRun[i] as Closure)()
}
}
}
}