blob: 04ce85e60efb979d3efa57fdcbf0270164368e28 [file] [log] [blame]
package com.mirantis.mk
/**
*
* Openstack functions
*
*/
/**
* Convert maps
*
*/
@NonCPS def entries(m) {
return m.collect {k, v -> [k, v]}
}
/**
* Install OpenStack service clients in isolated environment
*
* @param path Path where virtualenv is created
* @param version Version of the OpenStack clients
*/
def setupOpenstackVirtualenv(path, version = 'latest') {
def python = new com.mirantis.mk.Python()
python.setupDocutilsVirtualenv(path)
def openstack_kilo_packages = [
//XXX: hack to fix https://bugs.launchpad.net/ubuntu/+source/python-pip/+bug/1635463
'cliff==2.8',
'python-cinderclient>=1.3.1,<1.4.0',
'python-glanceclient>=0.19.0,<0.20.0',
'python-heatclient>=0.6.0,<0.7.0',
'python-keystoneclient>=1.6.0,<1.7.0',
'python-neutronclient>=2.2.6,<2.3.0',
'python-novaclient>=2.19.0,<2.20.0',
'python-swiftclient>=2.5.0,<2.6.0',
'python-openstackclient>=1.7.0,<1.8.0',
'oslo.config>=2.2.0,<2.3.0',
'oslo.i18n>=2.3.0,<2.4.0',
'oslo.serialization>=1.8.0,<1.9.0',
'oslo.utils>=1.4.0,<1.5.0',
'docutils'
]
def openstack_latest_packages = [
//XXX: hack to fix https://bugs.launchpad.net/ubuntu/+source/python-pip/+bug/1635463
'cliff==2.8',
// NOTE(vsaienko): cmd2 is dependency for cliff, since we don't using upper-contstraints
// we have to pin cmd2 < 0.9.0 as later versions are not compatible with python2.
// TODO(vsaienko): use upper-constraints here, as in requirements we set only lowest library
// versions.
'cmd2<0.9.0;python_version=="2.7"',
'cmd2>=0.9.1;python_version=="3.4"',
'cmd2>=0.9.1;python_version=="3.5"',
'python-openstackclient',
'python-heatclient',
'docutils'
]
if (version == 'kilo') {
requirements = openstack_kilo_packages
} else if (version == 'liberty') {
requirements = openstack_kilo_packages
} else if (version == 'mitaka') {
requirements = openstack_kilo_packages
} else {
requirements = openstack_latest_packages
}
python.setupVirtualenv(path, 'python2', requirements, null, true)
}
/**
* create connection to OpenStack API endpoint
*
* @param path Path to created venv
* @param url OpenStack API endpoint address
* @param credentialsId Credentials to the OpenStack API
* @param project OpenStack project to connect to
*/
def createOpenstackEnv(path, url, credentialsId, project, project_domain="default",
project_id="", user_domain="default", api_ver="2", cacert="/etc/ssl/certs/ca-certificates.crt") {
def common = new com.mirantis.mk.Common()
rcFile = "${path}/keystonerc"
creds = common.getPasswordCredentials(credentialsId)
rc = """set +x
export OS_USERNAME=${creds.username}
export OS_PASSWORD=${creds.password.toString()}
export OS_TENANT_NAME=${project}
export OS_AUTH_URL=${url}
export OS_AUTH_STRATEGY=keystone
export OS_PROJECT_NAME=${project}
export OS_PROJECT_ID=${project_id}
export OS_PROJECT_DOMAIN_ID=${project_domain}
export OS_USER_DOMAIN_NAME=${user_domain}
export OS_IDENTITY_API_VERSION=${api_ver}
export OS_CACERT=${cacert}
set -x
"""
writeFile file: rcFile, text: rc
return rcFile
}
/**
* Run command with OpenStack env params and optional python env
*
* @param cmd Command to be executed
* @param env Environmental parameters with endpoint credentials
* @param path Optional path to virtualenv with specific clients
*/
def runOpenstackCommand(cmd, venv, path = null) {
def python = new com.mirantis.mk.Python()
openstackCmd = ". ${venv}; ${cmd}"
if (path) {
output = python.runVirtualenvCommand(path, openstackCmd)
}
else {
echo("[Command]: ${openstackCmd}")
output = sh (
script: openstackCmd,
returnStdout: true
).trim()
}
return output
}
/**
* Get OpenStack Keystone token for current credentials
*
* @param env Connection parameters for OpenStack API endpoint
* @param path Optional path to the custom virtualenv
*/
def getKeystoneToken(client, path = null) {
def python = new com.mirantis.mk.Python()
cmd = "openstack token issue"
outputTable = runOpenstackCommand(cmd, client, path)
output = python.parseTextTable(outputTable, 'item', 'prettytable', path)
return output
}
/**
* Create OpenStack environment file
*
* @param env Connection parameters for OpenStack API endpoint
* @param path Optional path to the custom virtualenv
*/
def createHeatEnv(file, environment = [], original_file = null) {
if (original_file) {
envString = readFile file: original_file
} else {
envString = "parameters:\n"
}
p = entries(environment)
for (int i = 0; i < p.size(); i++) {
envString = "${envString} ${p.get(i)[0]}: ${p.get(i)[1]}\n"
}
echo("writing to env file:\n${envString}")
writeFile file: file, text: envString
}
/**
* Create new OpenStack Heat stack
*
* @param env Connection parameters for OpenStack API endpoint
* @param template HOT template for the new Heat stack
* @param environment Environmentale parameters of the new Heat stack
* @param name Name of the new Heat stack
* @param path Optional path to the custom virtualenv
*/
def createHeatStack(client, name, template, params = [], environment = null, path = null, action="create") {
def python = new com.mirantis.mk.Python()
def templateFile = "${env.WORKSPACE}/template/template/${template}.hot"
def envFile
def envSource
if (environment) {
envFile = "${env.WORKSPACE}/template/env/${name}.env"
if (environment.contains("/")) {
//init() returns all elements but the last in a collection.
def envPath = environment.tokenize("/").init().join("/")
if (envPath) {
envFile = "${env.WORKSPACE}/template/env/${envPath}/${name}.env"
}
}
envSource = "${env.WORKSPACE}/template/env/${environment}.env"
createHeatEnv(envFile, params, envSource)
} else {
envFile = "${env.WORKSPACE}/template/${name}.env"
createHeatEnv(envFile, params)
}
def cmd
def waitState
if (action == "create") {
cmd = "heat stack-create -f ${templateFile} -e ${envFile} ${name}"
waitState = "CREATE_COMPLETE"
} else {
cmd = "heat stack-update -f ${templateFile} -e ${envFile} ${name}"
waitState = "UPDATE_COMPLETE"
}
dir("${env.WORKSPACE}/template/template") {
outputTable = runOpenstackCommand(cmd, client, path)
}
output = python.parseTextTable(outputTable, 'item', 'prettytable', path)
def heatStatusCheckerCount = 1
while (heatStatusCheckerCount <= 250) {
status = getHeatStackStatus(client, name, path)
echo("[Heat Stack] Status: ${status}, Check: ${heatStatusCheckerCount}")
if (status.contains('CREATE_FAILED')) {
info = getHeatStackInfo(client, name, path)
throw new Exception(info.stack_status_reason)
} else if (status.contains(waitState)) {
info = getHeatStackInfo(client, name, path)
echo(info.stack_status_reason)
break
}
sleep(30)
heatStatusCheckerCount++
}
echo("[Heat Stack] Status: ${status}")
}
/**
* Returns list of stacks for stack name filter
*
* @param client Connection parameters for OpenStack API endpoint
* @param filter Stack name filter
* @param path Optional path to the custom virtualenv
*/
def getStacksForNameContains(client, filter, path = null){
cmd = 'heat stack-list | awk \'NR>3 {print $4}\' | sed \'$ d\' | grep ' + filter + '|| true'
return runOpenstackCommand(cmd, client, path).trim().tokenize("\n")
}
/**
* Get list of stack names with given stack status
*
* @param client Connection parameters for OpenStack API endpoint
* @param status Stack status
* @param path Optional path to the custom virtualenv
*/
def getStacksWithStatus(client, status, path = null) {
cmd = 'heat stack-list -f stack_status='+status+' | awk \'NR>3 {print $4}\' | sed \'$ d\''
return runOpenstackCommand(cmd, client, path).trim().tokenize("\n")
}
/**
* Get life cycle status for existing OpenStack Heat stack
*
* @param env Connection parameters for OpenStack API endpoint
* @param name Name of the managed Heat stack instance
* @param path Optional path to the custom virtualenv
*/
def getHeatStackStatus(client, name, path = null) {
cmd = 'heat stack-list | awk -v stack='+name+' \'{if ($4==stack) print $6}\''
return runOpenstackCommand(cmd, client, path)
}
/**
* Get info about existing OpenStack Heat stack
*
* @param env Connection parameters for OpenStack API endpoint
* @param name Name of the managed Heat stack instance
* @param path Optional path to the custom virtualenv
*/
def getHeatStackInfo(env, name, path = null) {
def python = new com.mirantis.mk.Python()
cmd = "heat stack-show ${name}"
outputTable = runOpenstackCommand(cmd, env, path)
output = python.parseTextTable(outputTable, 'item', 'prettytable', path)
return output
}
/**
* Get existing OpenStack Heat stack output parameter
*
* @param env Connection parameters for OpenStack API endpoint
* @param name Name of the managed Heat stack
* @param parameter Name of the output parameter
* @param path Optional path to the custom virtualenv
*/
def getHeatStackOutputParam(env, name, outputParam, path = null) {
cmd = "heat output-show ${name} ${outputParam}"
output = runOpenstackCommand(cmd, env, path)
echo("${cmd}: ${output}")
// NOTE(vsaienko) heatclient 1.5.1 returns output in "", while later
// versions returns string without "".
// TODO Use openstack 'stack output show' when all jobs using at least Mitaka heatclient
return "${output}".replaceAll('"', '')
}
/**
* List all resources from existing OpenStack Heat stack
*
* @param env Connection parameters for OpenStack API endpoint
* @param name Name of the managed Heat stack instance
* @param path Optional path to the custom virtualenv
* @param depth Optional depth of stack for listing resources,
* 0 - do not list nested resources
*/
def getHeatStackResources(env, name, path = null, depth = 0) {
def python = new com.mirantis.mk.Python()
cmd = "heat resource-list --nested-depth ${depth} ${name}"
outputTable = runOpenstackCommand(cmd, env, path)
output = python.parseTextTable(outputTable, 'list', 'prettytable', path)
return output
}
/**
* Get info about resource from existing OpenStack Heat stack
*
* @param env Connection parameters for OpenStack API endpoint
* @param name Name of the managed Heat stack instance
* @param path Optional path to the custom virtualenv
*/
def getHeatStackResourceInfo(env, name, resource, path = null) {
def python = new com.mirantis.mk.Python()
cmd = "heat resource-show ${name} ${resource}"
outputTable = runOpenstackCommand(cmd, env, path)
output = python.parseTextTable(outputTable, 'item', 'prettytable', path)
return output
}
/**
* Update existing OpenStack Heat stack
*
* @param env Connection parameters for OpenStack API endpoint
* @param name Name of the managed Heat stack instance
* @param path Optional path to the custom virtualenv
*/
def updateHeatStack(env, name, path = null) {
def python = new com.mirantis.mk.Python()
cmd = "heat stack-update ${name}"
outputTable = runOpenstackCommand(cmd, env, path)
output = python.parseTextTable(outputTable, 'item', 'prettytable', path)
return output
}
/**
* Delete existing OpenStack Heat stack
*
* @param env Connection parameters for OpenStack API endpoint
* @param name Name of the managed Heat stack instance
* @param path Optional path to the custom virtualenv
*/
def deleteHeatStack(env, name, path = null) {
cmd = "heat stack-delete ${name}"
outputTable = runOpenstackCommand(cmd, env, path)
}
/**
* Return hashmap of hashes server_id:server_name of servers from OpenStack Heat stack
*
* @param env Connection parameters for OpenStack API endpoint
* @param name Name of the managed Heat stack instance
* @param path Optional path to the custom virtualenv
*/
def getHeatStackServers(env, name, path = null) {
// set depth to 1000 to ensure all nested resources are shown
resources = getHeatStackResources(env, name, path, 1000)
servers = [:]
for (resource in resources) {
if (resource.resource_type == 'OS::Nova::Server') {
server = getHeatStackResourceInfo(env, resource.stack_name, resource.resource_name, path)
servers[server.attributes.id] = server.attributes.name
}
}
echo("[Stack ${name}] Servers: ${servers}")
return servers
}
/**
* Stops all services that contain specific string (for example nova,heat, etc.)
* @param env Salt Connection object or pepperEnv
* @param probe single node on which to list service names
* @param target all targeted nodes
* @param services lists of type of services to be stopped
* @param confirm enable/disable manual service stop confirmation
* @return output of salt commands
*/
def stopServices(env, probe, target, services=[], confirm=false) {
def salt = new com.mirantis.mk.Salt()
def common = new com.mirantis.mk.Common()
for (s in services) {
def outputServicesStr = salt.getReturnValues(salt.cmdRun(env, "${probe}*", "service --status-all | grep ${s} | awk \'{print \$4}\'"))
def servicesList = outputServicesStr.tokenize("\n").init()
if (confirm) {
if (servicesList) {
try {
input message: "Click PROCEED to stop ${servicesList}. Otherwise click ABORT to skip stopping them."
for (name in servicesList) {
if (!name.contains('Salt command')) {
salt.runSaltProcessStep(env, "${target}*", 'service.stop', ["${name}"])
}
}
} catch (Exception er) {
common.infoMsg("skipping stopping ${servicesList} services")
}
}
} else {
if (servicesList) {
for (name in servicesList) {
if (!name.contains('Salt command')) {
salt.runSaltProcessStep(env, "${target}*", 'service.stop', ["${name}"])
}
}
}
}
}
}
/**
* Restores Galera database
* @param env Salt Connection object or pepperEnv
* @return output of salt commands
*/
def restoreGaleraDb(env) {
def salt = new com.mirantis.mk.Salt()
def common = new com.mirantis.mk.Common()
try {
salt.runSaltProcessStep(env, 'I@galera:slave', 'service.stop', ['mysql'])
} catch (Exception er) {
common.warningMsg('Mysql service already stopped')
}
try {
salt.runSaltProcessStep(env, 'I@galera:master', 'service.stop', ['mysql'])
} catch (Exception er) {
common.warningMsg('Mysql service already stopped')
}
try {
salt.cmdRun(env, 'I@galera:slave', "rm /var/lib/mysql/ib_logfile*")
} catch (Exception er) {
common.warningMsg('Files are not present')
}
try {
salt.cmdRun(env, 'I@galera:master', "mkdir /root/mysql/mysql.bak")
} catch (Exception er) {
common.warningMsg('Directory already exists')
}
try {
salt.cmdRun(env, 'I@galera:master', "rm -rf /root/mysql/mysql.bak/*")
} catch (Exception er) {
common.warningMsg('Directory already empty')
}
try {
salt.cmdRun(env, 'I@galera:master', "mv /var/lib/mysql/* /root/mysql/mysql.bak")
} catch (Exception er) {
common.warningMsg('Files were already moved')
}
try {
salt.runSaltProcessStep(env, 'I@galera:master', 'file.remove', ["/var/lib/mysql/.galera_bootstrap"])
} catch (Exception er) {
common.warningMsg('File is not present')
}
salt.cmdRun(env, 'I@galera:master', "sed -i '/gcomm/c\\wsrep_cluster_address=\"gcomm://\"' /etc/mysql/my.cnf")
def backup_dir = salt.getReturnValues(salt.getPillar(env, "I@galera:master", 'xtrabackup:client:backup_dir'))
if(backup_dir == null || backup_dir.isEmpty()) { backup_dir='/var/backups/mysql/xtrabackup' }
salt.runSaltProcessStep(env, 'I@galera:master', 'file.remove', ["${backup_dir}/dbrestored"])
salt.cmdRun(env, 'I@xtrabackup:client', "su root -c 'salt-call state.sls xtrabackup'")
salt.runSaltProcessStep(env, 'I@galera:master', 'service.start', ['mysql'])
// wait until mysql service on galera master is up
try {
salt.commandStatus(env, 'I@galera:master', 'service mysql status', 'running')
} catch (Exception er) {
input message: "Database is not running please fix it first and only then click on PROCEED."
}
salt.runSaltProcessStep(env, 'I@galera:slave', 'service.start', ['mysql'])
}
/**
* Recovers
* @param master Salt master
* @param recoverHost Hostname of the node to be recovered
* @param healthyHost Hostname of healthy node from the same cluster
* @return output of salt commands
*/
def recoverGluster(master, recoverHost, healthyHost) {
def salt = new com.mirantis.mk.Salt()
// Recover glusterfs
if (salt.testTarget(master, 'I@glusterfs:server')) {
salt.enforceState(master, 'I@glusterfs:server', 'glusterfs.server.service')
if (healthyHost != 'none' && recoverHost != 'none') {
salt.runSaltCommand(master, 'local', ['expression': "E@${healthyHost}", 'type': 'compound'], "cp.push /var/lib/glusterd/vols/ upload_path='/tmp/'")
salt.runSaltCommand(master, 'local', ['expression': "E@${recoverHost}", 'type': 'compound'], "get_dir salt://tmp/vols/ /var/lib/glusterd/")
}
salt.enforceState(master, 'I@glusterfs:server and *01*', 'glusterfs.server.setup', true, true, null, false, -1, 5)
sleep(10)
salt.cmdRun(master, 'I@glusterfs:server', "gluster peer status; gluster volume status")
}
// Ensure glusterfs clusters is ready
if (salt.testTarget(master, 'I@glusterfs:client')) {
salt.enforceState(master, 'I@glusterfs:client', 'glusterfs.client')
}
}