blob: c3882fecdd4041e186e2d788dd3442e922be408e [file] [log] [blame]
/**
* Update packages on given nodes
*
* Expected parameters:
* SALT_MASTER_CREDENTIALS Credentials to the Salt API.
* SALT_MASTER_URL Full Salt API address [https://10.10.10.1:8000].
* TARGET_SERVERS Salt compound target to match nodes to be updated [*, G@osfamily:debian].
* TARGET_PACKAGES Space delimited list of packages to be updates [package1=version package2=version], empty string means all updating all packages to the latest version.
* TARGET_SUBSET_TEST Number of nodes to list package updates, empty string means all targetted nodes.
* TARGET_SUBSET_LIVE Number of selected nodes to live apply selected package update.
* TARGET_BATCH_LIVE Batch size for the complete live package update on all nodes, empty string means apply to all targetted nodes.
*
**/
def common = new com.mirantis.mk.Common()
def salt = new com.mirantis.mk.Salt()
def python = new com.mirantis.mk.Python()
def pepperEnv = "pepperEnv"
def targetTestSubset
def targetLiveSubset
def targetLiveAll
def minions
def result
def packages
def command
def commandKwargs
timeout(time: 12, unit: 'HOURS') {
node() {
try {
stage('Setup virtualenv for Pepper') {
python.setupPepperVirtualenv(pepperEnv, SALT_MASTER_URL, SALT_MASTER_CREDENTIALS)
}
stage('List target servers') {
minions = salt.getMinions(pepperEnv, TARGET_SERVERS)
if (minions.isEmpty()) {
throw new Exception("No minion was targeted")
}
if (TARGET_SUBSET_TEST != "") {
targetTestSubset = minions.subList(0, Integer.valueOf(TARGET_SUBSET_TEST)).join(' or ')
} else {
targetTestSubset = minions.join(' or ')
}
targetLiveSubset = minions.subList(0, Integer.valueOf(TARGET_SUBSET_LIVE)).join(' or ')
targetLiveAll = minions.join(' or ')
common.infoMsg("Found nodes: ${targetLiveAll}")
common.infoMsg("Selected test nodes: ${targetTestSubset}")
common.infoMsg("Selected sample nodes: ${targetLiveSubset}")
}
stage("List package upgrades") {
common.infoMsg("Listing all the packages that have a new update available on test nodes: ${targetTestSubset}")
salt.runSaltProcessStep(pepperEnv, targetTestSubset, 'pkg.list_upgrades', [], null, true)
if(TARGET_PACKAGES != "" && TARGET_PACKAGES != "*"){
common.infoMsg("Note that only the ${TARGET_PACKAGES} would be installed from the above list of available updates on the ${targetTestSubset}")
}
}
stage('Confirm live package upgrades on sample') {
if(TARGET_PACKAGES==""){
timeout(time: 2, unit: 'HOURS') {
def userInput = input(
id: 'userInput', message: 'Insert package names for update', parameters: [
[$class: 'TextParameterDefinition', defaultValue: '', description: 'Package names (or *)', name: 'packages']
])
if(userInput!= "" && userInput!= "*"){
TARGET_PACKAGES = userInput
}
}
}else{
timeout(time: 2, unit: 'HOURS') {
input message: "Approve live package upgrades on ${targetLiveSubset} nodes?"
}
}
}
if (TARGET_PACKAGES != "") {
command = "pkg.install"
packages = TARGET_PACKAGES.tokenize(' ')
commandKwargs = ['only_upgrade': 'true']
}else {
command = "pkg.upgrade"
packages = null
}
stage('Apply package upgrades on sample') {
out = salt.runSaltCommand(pepperEnv, 'local', ['expression': targetLiveSubset, 'type': 'compound'], command, null, packages, commandKwargs)
salt.printSaltCommandResult(out)
for(value in out.get("return")[0].values()){
if (value.containsKey('result') && value.result == false) {
throw new Exception("The package upgrade on sample node has failed. Please check the Salt run result above for more information.")
}
}
}
stage('Confirm package upgrades on all nodes') {
timeout(time: 2, unit: 'HOURS') {
input message: "Approve live package upgrades on ${targetLiveAll} nodes?"
}
}
stage('Apply package upgrades on all nodes') {
out = salt.runSaltCommand(pepperEnv, 'local', ['expression': targetLiveAll, 'type': 'compound'], command, null, packages, commandKwargs)
salt.printSaltCommandResult(out)
for(value in out.get("return")[0].values()){
if (value.containsKey('result') && value.result == false) {
throw new Exception("The package upgrade on sample node has failed. Please check the Salt run result above for more information.")
}
}
common.warningMsg("Pipeline has finished successfully, but please, check if any packages have been kept back.")
}
} catch (Throwable e) {
// If there was an error or exception thrown, the build failed
currentBuild.result = "FAILURE"
currentBuild.description = currentBuild.description ? e.message + " " + currentBuild.description : e.message
throw e
}
}
}