Jakub Josef | 79ecec3 | 2017-02-17 14:36:28 +0100 | [diff] [blame] | 1 | package com.mirantis.mk |
| 2 | |
| 3 | /** |
| 4 | * |
| 5 | * Debian functions |
| 6 | * |
| 7 | */ |
| 8 | |
| 9 | def cleanup(image="debian:sid") { |
| 10 | def common = new com.mirantis.mk.Common() |
| 11 | def img = docker.image(image) |
| 12 | |
| 13 | workspace = common.getWorkspace() |
| 14 | sh("docker run -e DEBIAN_FRONTEND=noninteractive -v ${workspace}:${workspace} -w ${workspace} --rm=true --privileged ${image} /bin/bash -c 'rm -rf build-area || true'") |
| 15 | } |
| 16 | |
| 17 | /* |
| 18 | * Build binary Debian package from existing dsc |
| 19 | * |
| 20 | * @param file dsc file to build |
| 21 | * @param image Image name to use for build (default debian:sid) |
| 22 | */ |
| 23 | def buildBinary(file, image="debian:sid", extraRepoUrl=null, extraRepoKeyUrl=null) { |
| 24 | def common = new com.mirantis.mk.Common() |
Filip Pytloun | 81c864d | 2017-03-21 15:19:30 +0100 | [diff] [blame] | 25 | def jenkinsUID = common.getJenkinsUid() |
| 26 | def jenkinsGID = common.getJenkinsGid() |
Jakub Josef | 79ecec3 | 2017-02-17 14:36:28 +0100 | [diff] [blame] | 27 | def pkg = file.split('/')[-1].split('_')[0] |
chnyda | 1cf6f0d | 2017-06-02 11:01:04 +0200 | [diff] [blame] | 28 | def dockerLib = new com.mirantis.mk.Docker() |
| 29 | def imageArray = image.split(":") |
| 30 | def os = imageArray[0] |
| 31 | def dist = imageArray[1] |
Jakub Josef | d4887ab | 2018-05-10 16:13:33 +0200 | [diff] [blame] | 32 | def img = dockerLib.getImage("mirantis/debian-build-${os}-${dist}:latest", image) |
chnyda | 1cf6f0d | 2017-06-02 11:01:04 +0200 | [diff] [blame] | 33 | def workspace = common.getWorkspace() |
Jakub Josef | 1321431 | 2018-02-06 16:34:14 +0100 | [diff] [blame] | 34 | def debug = env.getEnvironment().containsKey("DEBUG") && env["DEBUG"].toBoolean() ? "true" : "" |
Jakub Josef | 79ecec3 | 2017-02-17 14:36:28 +0100 | [diff] [blame] | 35 | |
chnyda | 8ad962e | 2017-06-02 12:24:15 +0200 | [diff] [blame] | 36 | img.inside("-u root:root" ) { |
Petr Ruzicka | c9eaa7e | 2018-02-07 08:51:38 +0100 | [diff] [blame] | 37 | sh("""bash -x -c 'cd ${workspace} && (which eatmydata || (apt-get update && apt-get install -y eatmydata)) && |
Jakub Josef | 1321431 | 2018-02-06 16:34:14 +0100 | [diff] [blame] | 38 | export DEBUG="${debug}" && |
Jakub Josef | 79ecec3 | 2017-02-17 14:36:28 +0100 | [diff] [blame] | 39 | export LD_LIBRARY_PATH=\${LD_LIBRARY_PATH:+"\$LD_LIBRARY_PATH:"}/usr/lib/libeatmydata && |
| 40 | export LD_PRELOAD=\${LD_PRELOAD:+"\$LD_PRELOAD "}libeatmydata.so && |
Kirill Mashchenko | b4d2fba | 2018-04-20 16:23:04 +0300 | [diff] [blame] | 41 | export DEB_BUILD_OPTIONS=nocheck && |
Oleh Hryhorov | d28a974 | 2018-03-06 17:41:38 +0200 | [diff] [blame] | 42 | [[ -z "${extraRepoUrl}" && "${extraRepoUrl}" != "null" ]] || echo "${extraRepoUrl}" | tr ";" "\\\\n" >/etc/apt/sources.list.d/extra.list && |
Jakub Josef | 79ecec3 | 2017-02-17 14:36:28 +0100 | [diff] [blame] | 43 | [[ -z "${extraRepoKeyUrl}" && "${extraRepoKeyUrl}" != "null" ]] || ( |
| 44 | which curl || (apt-get update && apt-get install -y curl) && |
Oleh Hryhorov | d28a974 | 2018-03-06 17:41:38 +0200 | [diff] [blame] | 45 | EXTRAKEY=`echo "${extraRepoKeyUrl}" | tr ";" " "` && |
| 46 | for RepoKey in \${EXTRAKEY}; do curl --insecure -ss -f "\${RepoKey}" | apt-key add - ; done |
Jakub Josef | 79ecec3 | 2017-02-17 14:36:28 +0100 | [diff] [blame] | 47 | ) && |
Filip Pytloun | 81c864d | 2017-03-21 15:19:30 +0100 | [diff] [blame] | 48 | apt-get update && apt-get install -y build-essential devscripts equivs sudo && |
| 49 | groupadd -g ${jenkinsGID} jenkins && |
| 50 | useradd -s /bin/bash --uid ${jenkinsUID} --gid ${jenkinsGID} -m jenkins && |
Filip Pytloun | 78b9183 | 2017-06-30 13:16:31 +0200 | [diff] [blame] | 51 | chown -R ${jenkinsUID}:${jenkinsGID} /home/jenkins && |
Jakub Josef | 6bebf16 | 2017-05-10 14:21:00 +0200 | [diff] [blame] | 52 | [ ! -f pre_build_script.sh ] || bash ./pre_build_script.sh && |
Filip Pytloun | ff82fc0 | 2017-03-27 12:17:05 +0200 | [diff] [blame] | 53 | sudo -H -E -u jenkins dpkg-source -x ${file} build-area/${pkg} && cd build-area/${pkg} && |
Filip Pytloun | 2f7302a | 2017-06-30 14:59:30 +0200 | [diff] [blame] | 54 | mk-build-deps -t "apt-get -o Debug::pkgProblemResolver=yes -y" -i debian/control && |
Petr Ruzicka | 638e0bf | 2018-02-07 13:31:25 +0100 | [diff] [blame] | 55 | sudo -H -E -u jenkins debuild --preserve-envvar DEBUG --no-lintian -uc -us -b'""") |
chnyda | 1cf6f0d | 2017-06-02 11:01:04 +0200 | [diff] [blame] | 56 | } |
| 57 | |
Jakub Josef | 79ecec3 | 2017-02-17 14:36:28 +0100 | [diff] [blame] | 58 | } |
| 59 | |
| 60 | /* |
| 61 | * Build source package from directory |
| 62 | * |
| 63 | * @param dir Tree to build |
| 64 | * @param image Image name to use for build (default debian:sid) |
| 65 | * @param snapshot Generate snapshot version (default false) |
| 66 | */ |
Oleg Iurchenko | 68e17b0 | 2018-01-02 12:24:49 +0200 | [diff] [blame] | 67 | def buildSource(dir, image="debian:sid", snapshot=false, gitEmail='jenkins@dummy.org', gitName='Jenkins', revisionPostfix="", remote="origin/") { |
Jakub Josef | 79ecec3 | 2017-02-17 14:36:28 +0100 | [diff] [blame] | 68 | def isGit |
| 69 | try { |
| 70 | sh("test -d ${dir}/.git") |
| 71 | isGit = true |
| 72 | } catch (Exception e) { |
| 73 | isGit = false |
| 74 | } |
| 75 | |
| 76 | if (isGit == true) { |
Oleg Iurchenko | 68e17b0 | 2018-01-02 12:24:49 +0200 | [diff] [blame] | 77 | buildSourceGbp(dir, image, snapshot, gitEmail, gitName, revisionPostfix, remote) |
Jakub Josef | 79ecec3 | 2017-02-17 14:36:28 +0100 | [diff] [blame] | 78 | } else { |
| 79 | buildSourceUscan(dir, image) |
| 80 | } |
| 81 | } |
| 82 | |
| 83 | /* |
| 84 | * Build source package, fetching upstream code using uscan |
| 85 | * |
| 86 | * @param dir Tree to build |
| 87 | * @param image Image name to use for build (default debian:sid) |
| 88 | */ |
| 89 | def buildSourceUscan(dir, image="debian:sid") { |
| 90 | def common = new com.mirantis.mk.Common() |
chnyda | 1cf6f0d | 2017-06-02 11:01:04 +0200 | [diff] [blame] | 91 | def dockerLib = new com.mirantis.mk.Docker() |
| 92 | def imageArray = image.split(":") |
| 93 | def os = imageArray[0] |
| 94 | def dist = imageArray[1] |
Jakub Josef | d4887ab | 2018-05-10 16:13:33 +0200 | [diff] [blame] | 95 | def img = dockerLib.getImage("mirantis/debian-build-${os}-${dist}:latest", image) |
chnyda | 1cf6f0d | 2017-06-02 11:01:04 +0200 | [diff] [blame] | 96 | def workspace = common.getWorkspace() |
| 97 | |
chnyda | 8ad962e | 2017-06-02 12:24:15 +0200 | [diff] [blame] | 98 | img.inside("-u root:root" ) { |
chnyda | 12f3b3f | 2017-06-02 12:19:38 +0200 | [diff] [blame] | 99 | sh("""cd ${workspace} && apt-get update && apt-get install -y build-essential devscripts && |
chnyda | 1cf6f0d | 2017-06-02 11:01:04 +0200 | [diff] [blame] | 100 | cd ${dir} && uscan --download-current-version && |
| 101 | dpkg-buildpackage -S -nc -uc -us""") |
| 102 | } |
Jakub Josef | 79ecec3 | 2017-02-17 14:36:28 +0100 | [diff] [blame] | 103 | } |
| 104 | |
| 105 | /* |
| 106 | * Build source package using git-buildpackage |
| 107 | * |
| 108 | * @param dir Tree to build |
| 109 | * @param image Image name to use for build (default debian:sid) |
| 110 | * @param snapshot Generate snapshot version (default false) |
| 111 | */ |
Oleg Iurchenko | 68e17b0 | 2018-01-02 12:24:49 +0200 | [diff] [blame] | 112 | def buildSourceGbp(dir, image="debian:sid", snapshot=false, gitName='Jenkins', gitEmail='jenkins@dummy.org', revisionPostfix="", remote="origin/") { |
Jakub Josef | 79ecec3 | 2017-02-17 14:36:28 +0100 | [diff] [blame] | 113 | def common = new com.mirantis.mk.Common() |
Filip Pytloun | 81c864d | 2017-03-21 15:19:30 +0100 | [diff] [blame] | 114 | def jenkinsUID = common.getJenkinsUid() |
| 115 | def jenkinsGID = common.getJenkinsGid() |
Jakub Josef | 79ecec3 | 2017-02-17 14:36:28 +0100 | [diff] [blame] | 116 | |
| 117 | if (! revisionPostfix) { |
| 118 | revisionPostfix = "" |
| 119 | } |
| 120 | |
chnyda | 1cf6f0d | 2017-06-02 11:01:04 +0200 | [diff] [blame] | 121 | def workspace = common.getWorkspace() |
| 122 | def dockerLib = new com.mirantis.mk.Docker() |
| 123 | def imageArray = image.split(":") |
| 124 | def os = imageArray[0] |
| 125 | def dist = imageArray[1] |
Jakub Josef | d4887ab | 2018-05-10 16:13:33 +0200 | [diff] [blame] | 126 | def img = dockerLib.getImage("mirantis/debian-build-${os}-${dist}:latest", image) |
chnyda | 1cf6f0d | 2017-06-02 11:01:04 +0200 | [diff] [blame] | 127 | |
chnyda | 8ad962e | 2017-06-02 12:24:15 +0200 | [diff] [blame] | 128 | img.inside("-u root:root") { |
chnyda | 1cf6f0d | 2017-06-02 11:01:04 +0200 | [diff] [blame] | 129 | |
| 130 | withEnv(["DEBIAN_FRONTEND=noninteractive", "DEBFULLNAME='${gitName}'", "DEBEMAIL='${gitEmail}'"]) { |
Petr Ruzicka | c9eaa7e | 2018-02-07 08:51:38 +0100 | [diff] [blame] | 131 | sh("""bash -x -c 'cd ${workspace} && (which eatmydata || (apt-get update && apt-get install -y eatmydata)) && |
Jakub Josef | 79ecec3 | 2017-02-17 14:36:28 +0100 | [diff] [blame] | 132 | export LD_LIBRARY_PATH=\${LD_LIBRARY_PATH:+"\$LD_LIBRARY_PATH:"}/usr/lib/libeatmydata && |
| 133 | export LD_PRELOAD=\${LD_PRELOAD:+"\$LD_PRELOAD "}libeatmydata.so && |
Jakub Josef | 5de05f6 | 2017-05-30 15:25:55 +0200 | [diff] [blame] | 134 | apt-get update && apt-get install -y build-essential git-buildpackage dpkg-dev sudo && |
Jakub Josef | 79ecec3 | 2017-02-17 14:36:28 +0100 | [diff] [blame] | 135 | groupadd -g ${jenkinsGID} jenkins && |
| 136 | useradd -s /bin/bash --uid ${jenkinsUID} --gid ${jenkinsGID} -m jenkins && |
Filip Pytloun | 78b9183 | 2017-06-30 13:16:31 +0200 | [diff] [blame] | 137 | chown -R ${jenkinsUID}:${jenkinsGID} /home/jenkins && |
Jakub Josef | 79ecec3 | 2017-02-17 14:36:28 +0100 | [diff] [blame] | 138 | cd ${dir} && |
Filip Pytloun | ff82fc0 | 2017-03-27 12:17:05 +0200 | [diff] [blame] | 139 | sudo -H -E -u jenkins git config --global user.name "${gitName}" && |
| 140 | sudo -H -E -u jenkins git config --global user.email "${gitEmail}" && |
Jakub Josef | 79ecec3 | 2017-02-17 14:36:28 +0100 | [diff] [blame] | 141 | [[ "${snapshot}" == "false" ]] || ( |
Alexander Noskov | 2b6b4be | 2017-09-13 17:32:44 +0400 | [diff] [blame] | 142 | VERSION=`dpkg-parsechangelog --count 1 --show-field Version` && |
Jakub Josef | 79ecec3 | 2017-02-17 14:36:28 +0100 | [diff] [blame] | 143 | UPSTREAM_VERSION=`echo \$VERSION | cut -d "-" -f 1` && |
| 144 | REVISION=`echo \$VERSION | cut -d "-" -f 2` && |
| 145 | TIMESTAMP=`date +%Y%m%d%H%M` && |
| 146 | if [[ "`cat debian/source/format`" = *quilt* ]]; then |
| 147 | UPSTREAM_BRANCH=`(grep upstream-branch debian/gbp.conf || echo master) | cut -d = -f 2 | tr -d " "` && |
Oleg Iurchenko | 68e17b0 | 2018-01-02 12:24:49 +0200 | [diff] [blame] | 148 | UPSTREAM_REV=`git rev-parse --short ${remote}\$UPSTREAM_BRANCH` && |
Jakub Josef | 79ecec3 | 2017-02-17 14:36:28 +0100 | [diff] [blame] | 149 | NEW_UPSTREAM_VERSION="\$UPSTREAM_VERSION+\$TIMESTAMP.\$UPSTREAM_REV" && |
Alexander Noskov | 2b6b4be | 2017-09-13 17:32:44 +0400 | [diff] [blame] | 150 | NEW_UPSTREAM_VERSION_TAG=`echo \$NEW_UPSTREAM_VERSION | sed 's/.*://'` && |
Jakub Josef | 79ecec3 | 2017-02-17 14:36:28 +0100 | [diff] [blame] | 151 | NEW_VERSION=\$NEW_UPSTREAM_VERSION-\$REVISION$revisionPostfix && |
Alexander Noskov | 2b6b4be | 2017-09-13 17:32:44 +0400 | [diff] [blame] | 152 | echo "Generating new upstream version \$NEW_UPSTREAM_VERSION_TAG" && |
Oleg Iurchenko | 68e17b0 | 2018-01-02 12:24:49 +0200 | [diff] [blame] | 153 | sudo -H -E -u jenkins git tag \$NEW_UPSTREAM_VERSION_TAG ${remote}\$UPSTREAM_BRANCH && |
Alexander Noskov | 2b6b4be | 2017-09-13 17:32:44 +0400 | [diff] [blame] | 154 | sudo -H -E -u jenkins git merge -X theirs \$NEW_UPSTREAM_VERSION_TAG |
Jakub Josef | 79ecec3 | 2017-02-17 14:36:28 +0100 | [diff] [blame] | 155 | else |
| 156 | NEW_VERSION=\$VERSION+\$TIMESTAMP.`git rev-parse --short HEAD`$revisionPostfix |
| 157 | fi && |
Dmitry Burmistrov | 65098ac | 2018-05-04 14:33:01 +0400 | [diff] [blame] | 158 | sudo -H -E -u jenkins gbp dch \ |
| 159 | --auto \ |
| 160 | --git-author \ |
| 161 | --id-length=7 \ |
| 162 | --git-log='--reverse' \ |
| 163 | --ignore-branch \ |
| 164 | --new-version=\$NEW_VERSION \ |
| 165 | --distribution `lsb_release -c -s` \ |
| 166 | --force-distribution && |
Filip Pytloun | ff82fc0 | 2017-03-27 12:17:05 +0200 | [diff] [blame] | 167 | sudo -H -E -u jenkins git add -u debian/changelog && |
| 168 | sudo -H -E -u jenkins git commit -m "New snapshot version \$NEW_VERSION" |
Jakub Josef | 79ecec3 | 2017-02-17 14:36:28 +0100 | [diff] [blame] | 169 | ) && |
chnyda | a3603b4 | 2017-06-02 12:36:08 +0200 | [diff] [blame] | 170 | sudo -H -E -u jenkins gbp buildpackage -nc --git-force-create --git-notify=false --git-ignore-branch --git-ignore-new --git-verbose --git-export-dir=../build-area -sa -S -uc -us '""") |
chnyda | 1cf6f0d | 2017-06-02 11:01:04 +0200 | [diff] [blame] | 171 | } |
| 172 | } |
Jakub Josef | 79ecec3 | 2017-02-17 14:36:28 +0100 | [diff] [blame] | 173 | } |
| 174 | |
| 175 | /* |
| 176 | * Run lintian checks |
| 177 | * |
| 178 | * @param changes Changes file to test against |
| 179 | * @param profile Lintian profile to use (default debian) |
| 180 | * @param image Image name to use for build (default debian:sid) |
| 181 | */ |
| 182 | def runLintian(changes, profile="debian", image="debian:sid") { |
| 183 | def common = new com.mirantis.mk.Common() |
chnyda | 1cf6f0d | 2017-06-02 11:01:04 +0200 | [diff] [blame] | 184 | def workspace = common.getWorkspace() |
| 185 | def dockerLib = new com.mirantis.mk.Docker() |
| 186 | def imageArray = image.split(":") |
| 187 | def os = imageArray[0] |
| 188 | def dist = imageArray[1] |
Jakub Josef | d4887ab | 2018-05-10 16:13:33 +0200 | [diff] [blame] | 189 | def img = dockerLib.getImage("mirantis/debian-build-${os}-${dist}:latest", image) |
chnyda | 8ad962e | 2017-06-02 12:24:15 +0200 | [diff] [blame] | 190 | img.inside("-u root:root") { |
chnyda | 12f3b3f | 2017-06-02 12:19:38 +0200 | [diff] [blame] | 191 | sh("""cd ${workspace} && apt-get update && apt-get install -y lintian && |
chnyda | 1cf6f0d | 2017-06-02 11:01:04 +0200 | [diff] [blame] | 192 | lintian -Ii -E --pedantic --profile=${profile} ${changes}""") |
| 193 | } |
Jakub Josef | 79ecec3 | 2017-02-17 14:36:28 +0100 | [diff] [blame] | 194 | } |
chnyda | 4e5ac79 | 2017-03-14 15:24:18 +0100 | [diff] [blame] | 195 | |
| 196 | /* |
| 197 | * Import gpg key |
| 198 | * |
| 199 | * @param privateKeyCredId Public key jenkins credential id |
| 200 | */ |
| 201 | def importGpgKey(privateKeyCredId) |
| 202 | { |
| 203 | def common = new com.mirantis.mk.Common() |
| 204 | def workspace = common.getWorkspace() |
| 205 | def privKey = common.getCredentials(privateKeyCredId, "key") |
| 206 | def private_key = privKey.privateKeySource.privateKey |
chnyda | c684645 | 2017-03-21 16:50:43 +0100 | [diff] [blame] | 207 | def gpg_key_id = common.getCredentials(privateKeyCredId, "key").username |
| 208 | def retval = sh(script: "export GNUPGHOME=${workspace}/.gnupg; gpg --list-secret-keys | grep ${gpg_key_id}", returnStatus: true) |
| 209 | if (retval) { |
| 210 | writeFile file:"${workspace}/private.key", text: private_key |
| 211 | sh(script: "gpg --no-tty --allow-secret-key-import --homedir ${workspace}/.gnupg --import ./private.key") |
| 212 | } |
chnyda | 4e5ac79 | 2017-03-14 15:24:18 +0100 | [diff] [blame] | 213 | } |
| 214 | |
| 215 | /* |
| 216 | * upload source package to launchpad |
| 217 | * |
| 218 | * @param ppaRepo ppa repository on launchpad |
| 219 | * @param dirPath repository containing the source packages |
| 220 | */ |
| 221 | |
| 222 | def uploadPpa(ppaRepo, dirPath, privateKeyCredId) { |
| 223 | |
| 224 | def common = new com.mirantis.mk.Common() |
| 225 | def workspace = common.getWorkspace() |
| 226 | def gpg_key_id = common.getCredentials(privateKeyCredId, "key").username |
| 227 | |
| 228 | dir(dirPath) |
| 229 | { |
| 230 | def images = findFiles(glob: "*.orig*.tar.gz") |
| 231 | for (int i = 0; i < images.size(); ++i) { |
| 232 | def name = images[i].getName() |
| 233 | def orig_sha1 = common.cutOrDie("sha1sum ${name}", 0) |
| 234 | def orig_sha256 = common.cutOrDie("sha256sum ${name}", 0) |
| 235 | def orig_md5 = common.cutOrDie("md5sum ${name}", 0) |
| 236 | def orig_size = common.cutOrDie("ls -l ${name}", 4) |
| 237 | |
chnyda | 5d1e97f | 2017-03-17 15:53:47 +0100 | [diff] [blame] | 238 | def retval = sh(script: "wget --quiet -O orig-tmp https://launchpad.net/ubuntu/+archive/primary/+files/${name}", returnStatus: true) |
chnyda | 4e5ac79 | 2017-03-14 15:24:18 +0100 | [diff] [blame] | 239 | if (retval == 0) { |
| 240 | sh("mv orig-tmp ${name}") |
| 241 | def new_sha1 = common.cutOrDie("sha1sum ${name}", 0) |
| 242 | def new_sha256 = common.cutOrDie("sha256sum ${name}", 0) |
| 243 | def new_md5 = common.cutOrDie("md5sum ${name}", 0) |
| 244 | def new_size = common.cutOrDie("ls -l ${name}", 4) |
| 245 | |
| 246 | sh("sed -i -e s,$orig_sha1,$new_sha1,g -e s,$orig_sha256,$new_sha256,g -e s,$orig_size,$new_size,g -e s,$orig_md5,$new_md5,g *.dsc") |
chnyda | 3c93ff6 | 2017-03-23 10:11:36 +0100 | [diff] [blame] | 247 | sh("sed -i -e s,$orig_sha1,$new_sha1,g -e s,$orig_sha256,$new_sha256,g -e s,$orig_size,$new_size,g -e s,$orig_md5,$new_md5,g *_source.changes") |
chnyda | 4e5ac79 | 2017-03-14 15:24:18 +0100 | [diff] [blame] | 248 | } |
chnyda | 4e5ac79 | 2017-03-14 15:24:18 +0100 | [diff] [blame] | 249 | } |
chnyda | 34e5c94 | 2017-03-22 18:06:06 +0100 | [diff] [blame] | 250 | sh("export GNUPGHOME=${workspace}/.gnupg; debsign --re-sign -k ${gpg_key_id} *_source.changes") |
| 251 | sh("export GNUPGHOME=${workspace}/.gnupg; dput -f \"ppa:${ppaRepo}\" *_source.changes") |
chnyda | 4e5ac79 | 2017-03-14 15:24:18 +0100 | [diff] [blame] | 252 | } |
Filip Pytloun | fbbd168 | 2017-03-17 22:48:04 +0100 | [diff] [blame] | 253 | } |
Vasyl Saienko | be38d9b | 2018-09-03 12:17:09 +0300 | [diff] [blame] | 254 | |
| 255 | /** |
Vasyl Saienko | be38d9b | 2018-09-03 12:17:09 +0300 | [diff] [blame] | 256 | * Reboot specified target, and wait when minion is UP. |
| 257 | * |
| 258 | * @param env Salt Connection object or env Salt command map |
| 259 | * @param target Salt target to upgrade packages on. |
| 260 | * @param timeout Sleep timeout when doing retries. |
| 261 | * @param attempts Number of attemps to wait for. |
| 262 | */ |
Ivan Berezovskiy | 23e38d6 | 2020-01-27 14:22:10 +0400 | [diff] [blame] | 263 | def osReboot(env, target, timeout=30, attempts=10) { |
| 264 | def salt = new com.mirantis.mk.Salt() |
| 265 | def common = new com.mirantis.mk.Common() |
Vasyl Saienko | be38d9b | 2018-09-03 12:17:09 +0300 | [diff] [blame] | 266 | |
Ivan Berezovskiy | 23e38d6 | 2020-01-27 14:22:10 +0400 | [diff] [blame] | 267 | salt.runSaltProcessStep(env, target, 'cmd.run', ["touch /tmp/rebooting"]) |
| 268 | salt.runSaltProcessStep(env, target, 'system.reboot', [], null, true, 5) |
Vasyl Saienko | be38d9b | 2018-09-03 12:17:09 +0300 | [diff] [blame] | 269 | |
Ivan Berezovskiy | 23e38d6 | 2020-01-27 14:22:10 +0400 | [diff] [blame] | 270 | common.retry(timeout, attempts) { |
Denis V. Meltsaykin | 0bc092f | 2020-08-29 11:43:26 +0200 | [diff] [blame] | 271 | if (salt.runSaltProcessStep(env, target, 'cmd.run', ['test -e /tmp/rebooting || echo NOFILE'], null, true, 5)['return'][0].values()[0] != "NOFILE") { |
Ivan Berezovskiy | 23e38d6 | 2020-01-27 14:22:10 +0400 | [diff] [blame] | 272 | error("The system is still rebooting...") |
| 273 | } |
Vasyl Saienko | be38d9b | 2018-09-03 12:17:09 +0300 | [diff] [blame] | 274 | } |
Vasyl Saienko | be38d9b | 2018-09-03 12:17:09 +0300 | [diff] [blame] | 275 | } |
| 276 | |
| 277 | /** |
| 278 | * Upgrade OS on given node, wait when minion become reachable. |
| 279 | * |
| 280 | * @param env Salt Connection object or env Salt command map |
| 281 | * @param target Salt target to upgrade packages on. |
| 282 | * @param mode 'upgrade' or 'dist-upgrade' |
| 283 | * @param postponeReboot Boolean flag to specify if reboot have to be postponed. |
| 284 | * @param timeout Sleep timeout when doing retries. |
| 285 | * @param attempts Number of attemps to wait for. |
| 286 | */ |
Ivan Berezovskiy | 23e38d6 | 2020-01-27 14:22:10 +0400 | [diff] [blame] | 287 | def osUpgradeNode(env, target, mode, postponeReboot=false, timeout=30, attempts=10, batch=null) { |
Ivan Berezovskiy | 99e79aa | 2020-03-24 14:38:28 +0400 | [diff] [blame] | 288 | if(mode in ['upgrade', 'dist-upgrade']) { |
| 289 | def common = new com.mirantis.mk.Common() |
| 290 | def salt = new com.mirantis.mk.Salt() |
| 291 | def rebootRequired = false |
Vasyl Saienko | be38d9b | 2018-09-03 12:17:09 +0300 | [diff] [blame] | 292 | |
Ivan Berezovskiy | 99e79aa | 2020-03-24 14:38:28 +0400 | [diff] [blame] | 293 | common.infoMsg("Running apt ${mode} on ${target}") |
| 294 | common.retry(3, 5) { |
| 295 | salt.cmdRun(env, target, 'salt-call pkg.refresh_db failhard=true', true, batch) |
Ivan Berezovskiy | 23e38d6 | 2020-01-27 14:22:10 +0400 | [diff] [blame] | 296 | } |
Denis V. Meltsaykin | acc7d28 | 2020-06-03 21:58:32 +0200 | [diff] [blame] | 297 | |
Denis V. Meltsaykin | c05da4c | 2020-05-17 17:24:07 +0200 | [diff] [blame] | 298 | /* first try to upgrade salt components since they demand asynchronous upgrade */ |
| 299 | upgradeSaltPackages(env, target) |
Ivan Berezovskiy | 99e79aa | 2020-03-24 14:38:28 +0400 | [diff] [blame] | 300 | def cmd = "export DEBIAN_FRONTEND=noninteractive; apt-get -y -q --allow-downgrades -o Dpkg::Options::=\"--force-confdef\" -o Dpkg::Options::=\"--force-confold\" ${mode}" |
Denis V. Meltsaykin | acc7d28 | 2020-06-03 21:58:32 +0200 | [diff] [blame] | 301 | |
| 302 | /* |
| 303 | * This is a long running batch operation that may return empty response |
| 304 | * which is a pretty typical salt behavior. This does not represent an error |
| 305 | * but might hide the error if it's ignored. If there is no persistent error |
| 306 | * with the procedure itself, the consequent run will succeed. |
| 307 | */ |
| 308 | common.retry(2, 120) { |
| 309 | salt.cmdRun(env, target, cmd, true, batch) |
| 310 | } |
| 311 | |
Ivan Berezovskiy | 99e79aa | 2020-03-24 14:38:28 +0400 | [diff] [blame] | 312 | rebootRequired = salt.runSaltProcessStep(env, target, 'file.file_exists', ['/var/run/reboot-required'], batch, true, 5)['return'][0].values()[0].toBoolean() |
| 313 | if (rebootRequired) { |
| 314 | if (!postponeReboot) { |
| 315 | common.infoMsg("Reboot is required after upgrade on ${target} Rebooting...") |
| 316 | osReboot(env, target, timeout, attempts) |
| 317 | } else { |
| 318 | common.infoMsg("Postponing reboot on node ${target}") |
| 319 | } |
| 320 | } |
| 321 | } else { |
| 322 | common.errorMsg("Invalid upgrade mode specified: ${mode}. Has to be 'upgrade' or 'dist-upgrade'") |
Vasyl Saienko | be38d9b | 2018-09-03 12:17:09 +0300 | [diff] [blame] | 323 | } |
| 324 | } |
Denis V. Meltsaykin | c05da4c | 2020-05-17 17:24:07 +0200 | [diff] [blame] | 325 | |
| 326 | /** |
| 327 | * Upgrade salt packages on target asynchronously, wait minions' availability. |
| 328 | * |
| 329 | * @param env Salt Connection object or env Salt command map |
| 330 | * @param target Salt target to upgrade packages on. |
| 331 | * @param timeout Sleep timeout when doing retries. |
| 332 | * @param attempts Number of attemps to wait for. |
| 333 | */ |
| 334 | def upgradeSaltPackages(env, target, timeout=60, attempts=20) { |
| 335 | def common = new com.mirantis.mk.Common() |
| 336 | def salt = new com.mirantis.mk.Salt() |
| 337 | def saltUpgradeCmd = |
| 338 | 'export DEBIAN_FRONTEND=noninteractive; apt-get -y -q ' + |
| 339 | '-o Dpkg::Options::=\"--force-confdef\" -o Dpkg::Options::=\"--force-confold\" ' + |
| 340 | 'install --only-upgrade salt-master salt-common salt-api salt-minion' |
| 341 | |
| 342 | common.infoMsg("Upgrading SaltStack on ${target}") |
| 343 | salt.cmdRun(env, target, saltUpgradeCmd, false, null, true, [], [], true) |
| 344 | /* wait for 2 mins before checking the availability of minions to give |
| 345 | apt some time to finish updating so the dpkg releases its locks */ |
| 346 | sleep(120) |
| 347 | /* taken from upgrade-mcp-release */ |
| 348 | common.retry(attempts, timeout) { |
| 349 | salt.minionsReachable(env, 'I@salt:master', target) |
| 350 | def running = salt.runSaltProcessStep(env, target, 'saltutil.running', [], null, true, 5) |
| 351 | for (value in running.get("return")[0].values()) { |
| 352 | if (value != []) { |
| 353 | throw new Exception("Not all salt-minions are ready for execution") |
| 354 | } |
| 355 | } |
| 356 | } |
Denis V. Meltsaykin | 0bc092f | 2020-08-29 11:43:26 +0200 | [diff] [blame] | 357 | } |