Richard Felkl | e90ef1c | 2017-12-08 00:13:56 +0100 | [diff] [blame] | 1 | /** |
| 2 | * |
| 3 | * Mirror Docker images |
| 4 | * |
| 5 | * Expected parameters: |
| 6 | * TARGET_REGISTRY_CREDENTIALS_ID Credentials for target Docker Registry |
| 7 | * TARGET_REGISTRY Target Docker Registry name |
| 8 | * REGISTRY_URL Target Docker Registry URL |
| 9 | * IMAGE_TAG Tag to use when pushing images |
azvyagintsev | d162b91 | 2018-07-26 10:52:14 +0200 | [diff] [blame] | 10 | * SOURCE_IMAGE_TAG Tag to use when pulling images(optional,if SOURCE_IMAGE_TAG has been found) |
azvyagintsev | 388a1bd | 2018-09-06 12:39:49 +0300 | [diff] [blame] | 11 | * SET_DEFAULT_ARTIFACTORY_PROPERTIES Add extra props. directly to artifactory, |
Richard Felkl | e90ef1c | 2017-12-08 00:13:56 +0100 | [diff] [blame] | 12 | * IMAGE_LIST List of images to mirror |
azvyagintsev | 388a1bd | 2018-09-06 12:39:49 +0300 | [diff] [blame] | 13 | * Example: docker.elastic.co/elasticsearch/elasticsearch:5.4.1 docker-prod-local.docker.mirantis.net/mirantis/external/docker.elastic.co/elasticsearch |
| 14 | * docker.elastic.co/elasticsearch/elasticsearch:SUBS_SOURCE_IMAGE_TAG docker-prod-local.docker.mirantis.net/mirantis/external/elasticsearch:${IMAGE_TAG}* Will be proceed like: |
| 15 | * docker tag docker.elastic.co/elasticsearch/elasticsearch:5.4.1 docker-prod-local.docker.mirantis.net/mirantis/external/docker.elastic.co/elasticsearch/elasticsearch:5.4.1 |
| 16 | * |
Richard Felkl | e90ef1c | 2017-12-08 00:13:56 +0100 | [diff] [blame] | 17 | * |
| 18 | */ |
azvyagintsev | 388a1bd | 2018-09-06 12:39:49 +0300 | [diff] [blame] | 19 | import java.util.regex.Pattern |
Alexander Evseev | 89b0da4 | 2018-12-10 13:14:04 +0000 | [diff] [blame] | 20 | import groovy.json.JsonSlurper |
Richard Felkl | e90ef1c | 2017-12-08 00:13:56 +0100 | [diff] [blame] | 21 | |
azvyagintsev | 388a1bd | 2018-09-06 12:39:49 +0300 | [diff] [blame] | 22 | common = new com.mirantis.mk.Common() |
Kirill Mashchenko | 343973c | 2018-12-25 15:52:24 +0400 | [diff] [blame] | 23 | jenkinsUtils = new com.mirantis.mk.JenkinsUtils() |
azvyagintsev | 388a1bd | 2018-09-06 12:39:49 +0300 | [diff] [blame] | 24 | external = false |
| 25 | externalMarker = '/mirantis/external/' |
Richard Felkl | e90ef1c | 2017-12-08 00:13:56 +0100 | [diff] [blame] | 26 | |
azvyagintsev | aa7b43a | 2019-02-02 10:17:58 +0200 | [diff] [blame] | 27 | // slaveNode = env.SLAVE_NODE ?: 'docker' |
| 28 | slaveNode = env.SLAVE_NODE ?: 'hardware' |
azvyagintsev | 388a1bd | 2018-09-06 12:39:49 +0300 | [diff] [blame] | 29 | setDefaultArtifactoryProperties = env.SET_DEFAULT_ARTIFACTORY_PROPERTIES ?: true |
| 30 | |
Kirill Mashchenko | 343973c | 2018-12-25 15:52:24 +0400 | [diff] [blame] | 31 | |
| 32 | |
Richard Felkl | e90ef1c | 2017-12-08 00:13:56 +0100 | [diff] [blame] | 33 | def getImageName(String image) { |
| 34 | def regex = Pattern.compile('(?:.+/)?([^:]+)(?::.+)?') |
| 35 | def matcher = regex.matcher(image) |
azvyagintsev | 388a1bd | 2018-09-06 12:39:49 +0300 | [diff] [blame] | 36 | if (matcher.find()) { |
Richard Felkl | e90ef1c | 2017-12-08 00:13:56 +0100 | [diff] [blame] | 37 | def imageName = matcher.group(1) |
| 38 | return imageName |
azvyagintsev | 388a1bd | 2018-09-06 12:39:49 +0300 | [diff] [blame] | 39 | } else { |
| 40 | error("Wrong format of image name.") |
Richard Felkl | e90ef1c | 2017-12-08 00:13:56 +0100 | [diff] [blame] | 41 | } |
| 42 | } |
azvyagintsev | 388a1bd | 2018-09-06 12:39:49 +0300 | [diff] [blame] | 43 | |
Denis Egorenko | 343a571 | 2018-12-10 19:08:45 +0400 | [diff] [blame] | 44 | def getImageInfo(String imageName) { |
| 45 | String unique_image_id = sh( |
Kyrylo Mashchenko | a86912d | 2018-12-21 08:31:47 +0000 | [diff] [blame] | 46 | script: "docker inspect --format='{{index .RepoDigests 0}}' '${imageName}'", |
| 47 | returnStdout: true, |
Denis Egorenko | 343a571 | 2018-12-10 19:08:45 +0400 | [diff] [blame] | 48 | ).trim() |
| 49 | String imageSha256 = unique_image_id.tokenize(':')[1] |
| 50 | common.infoMsg("Docker ${imageName} image sha256 is ${imageSha256}") |
Kyrylo Mashchenko | a86912d | 2018-12-21 08:31:47 +0000 | [diff] [blame] | 51 | return [ 'id': unique_image_id, 'sha256': imageSha256 ] |
Denis Egorenko | 343a571 | 2018-12-10 19:08:45 +0400 | [diff] [blame] | 52 | } |
| 53 | |
| 54 | def imageURL(String registry, String imageName, String sha256) { |
| 55 | def ret = new URL("https://${registry}/artifactory/api/search/checksum?sha256=${sha256}").getText() |
| 56 | // Most probably, we would get many images, especially for external images. We need to guess |
| 57 | // exactly one, which we pushing now |
| 58 | def tgtGuessImage = imageName.replace(':', '/').replace(registry, '') |
| 59 | ArrayList img_data = new JsonSlurper().parseText(ret)['results'] |
| 60 | def tgtImgUrl = img_data*.uri.find { it.contains(tgtGuessImage) } |
| 61 | if (tgtImgUrl) { |
| 62 | return tgtImgUrl |
| 63 | } else { |
| 64 | error("Can't find image ${imageName} in registry ${registry} with sha256: ${sha256}!") |
| 65 | } |
| 66 | } |
| 67 | |
azvyagintsev | 388a1bd | 2018-09-06 12:39:49 +0300 | [diff] [blame] | 68 | timeout(time: 4, unit: 'HOURS') { |
| 69 | node(slaveNode) { |
Kirill Mashchenko | 2c67fe6 | 2018-12-26 02:18:19 +0400 | [diff] [blame] | 70 | def user = jenkinsUtils.currentUsername() |
Denis Egorenko | 899ee32 | 2018-11-19 17:35:50 +0400 | [diff] [blame] | 71 | currentBuild.description = "${user}: [${env.SOURCE_IMAGE_TAG} => ${env.IMAGE_TAG}]\n${env.IMAGE_LIST}" |
Jakub Josef | a63f986 | 2018-01-11 17:58:38 +0100 | [diff] [blame] | 72 | try { |
Kirill Mashchenko | 343973c | 2018-12-25 15:52:24 +0400 | [diff] [blame] | 73 | allowedGroups = ['release-engineering'] |
Roman Vialov | d6b9e83 | 2019-02-04 12:46:54 +0300 | [diff] [blame^] | 74 | releaseTags = ['proposed', 'release', '2018', '2019', '2020'] |
Kirill Mashchenko | 343973c | 2018-12-25 15:52:24 +0400 | [diff] [blame] | 75 | tags = [env.SOURCE_IMAGE_TAG, env.IMAGE_TAG] |
| 76 | tagInRelease = tags.any { tag -> releaseTags.any { tag.contains(it) } } |
| 77 | if (tagInRelease) { |
| 78 | if (!jenkinsUtils.currentUserInGroups(allowedGroups)) { |
Kirill Mashchenko | dfa2e59 | 2019-01-11 11:41:12 +0400 | [diff] [blame] | 79 | throw new Exception("You - ${user} - don't have permissions to run this job with tags ${tags}!") |
Kirill Mashchenko | 343973c | 2018-12-25 15:52:24 +0400 | [diff] [blame] | 80 | } else { |
Kirill Mashchenko | 2c67fe6 | 2018-12-26 02:18:19 +0400 | [diff] [blame] | 81 | echo "User `${user}` belongs to one of groups `${allowedGroups}`. Proceeding..." |
Kirill Mashchenko | 343973c | 2018-12-25 15:52:24 +0400 | [diff] [blame] | 82 | } |
| 83 | } |
azvyagintsev | 388a1bd | 2018-09-06 12:39:49 +0300 | [diff] [blame] | 84 | stage("Mirror Docker Images") { |
| 85 | |
Jakub Josef | a63f986 | 2018-01-11 17:58:38 +0100 | [diff] [blame] | 86 | def images = IMAGE_LIST.tokenize('\n') |
azvyagintsev | 388a1bd | 2018-09-06 12:39:49 +0300 | [diff] [blame] | 87 | def imageName, sourceImage, targetRegistryPath, imageArray |
| 88 | for (image in images) { |
| 89 | if (image.trim().indexOf(' ') == -1) { |
| 90 | error("Wrong format of image and target repository input") |
Jakub Josef | a63f986 | 2018-01-11 17:58:38 +0100 | [diff] [blame] | 91 | } |
| 92 | imageArray = image.trim().tokenize(' ') |
azvyagintsev | 388a1bd | 2018-09-06 12:39:49 +0300 | [diff] [blame] | 93 | sourceImage = imageArray[0] |
| 94 | if (sourceImage.contains('SUBS_SOURCE_IMAGE_TAG')) { |
| 95 | common.warningMsg("Replacing SUBS_SOURCE_IMAGE_TAG => ${env.SOURCE_IMAGE_TAG}") |
| 96 | sourceImage = sourceImage.replace('SUBS_SOURCE_IMAGE_TAG', env.SOURCE_IMAGE_TAG) |
azvyagintsev | d162b91 | 2018-07-26 10:52:14 +0200 | [diff] [blame] | 97 | } |
azvyagintsev | 388a1bd | 2018-09-06 12:39:49 +0300 | [diff] [blame] | 98 | targetRegistryPath = imageArray[1] |
| 99 | targetRegistry = imageArray[1].split('/')[0] |
| 100 | imageName = getImageName(sourceImage) |
| 101 | targetImageFull = "${targetRegistryPath}/${imageName}:${env.IMAGE_TAG}" |
Denis Egorenko | 2ee651c | 2018-11-23 17:41:38 +0400 | [diff] [blame] | 102 | |
| 103 | def mcp_artifactory = new com.mirantis.mcp.MCPArtifactory() |
| 104 | if (targetImageFull.contains(externalMarker)) { |
| 105 | external = true |
| 106 | // check if images exists - raise error, as we don't want to rewrite existing one |
| 107 | def imageRepo = targetRegistryPath - targetRegistry |
Denis Egorenko | 1b207d1 | 2018-11-26 15:35:21 +0400 | [diff] [blame] | 108 | if (mcp_artifactory.imageExists(env.REGISTRY_URL, "${imageRepo}/${imageName}", env.IMAGE_TAG)) { |
Denis Egorenko | 2ee651c | 2018-11-23 17:41:38 +0400 | [diff] [blame] | 109 | error("Image ${targetImageFull} already exists!") |
| 110 | } |
| 111 | } |
| 112 | |
azvyagintsev | 388a1bd | 2018-09-06 12:39:49 +0300 | [diff] [blame] | 113 | srcImage = docker.image(sourceImage) |
azvyagintsev | 86ba4a0 | 2018-10-03 19:21:28 +0300 | [diff] [blame] | 114 | common.retry(3, 5) { |
| 115 | srcImage.pull() |
| 116 | } |
Denis Egorenko | 343a571 | 2018-12-10 19:08:45 +0400 | [diff] [blame] | 117 | source_image_sha256 = getImageInfo(sourceImage)['sha256'] |
azvyagintsev | 388a1bd | 2018-09-06 12:39:49 +0300 | [diff] [blame] | 118 | // Use sh-docker call for tag, due magic code in plugin: |
| 119 | // https://github.com/jenkinsci/docker-workflow-plugin/blob/docker-workflow-1.17/src/main/resources/org/jenkinsci/plugins/docker/workflow/Docker.groovy#L168-L170 |
| 120 | sh("docker tag ${srcImage.id} ${targetImageFull}") |
| 121 | common.infoMsg("Attempt to push docker image into remote registry: ${env.REGISTRY_URL}") |
azvyagintsev | 86ba4a0 | 2018-10-03 19:21:28 +0300 | [diff] [blame] | 122 | common.retry(3, 5) { |
| 123 | docker.withRegistry(env.REGISTRY_URL, env.TARGET_REGISTRY_CREDENTIALS_ID) { |
Alexander Evseev | 3a37d08 | 2018-12-10 12:46:39 +0000 | [diff] [blame] | 124 | sh("docker push ${targetImageFull}") |
azvyagintsev | 86ba4a0 | 2018-10-03 19:21:28 +0300 | [diff] [blame] | 125 | } |
Alexander Evseev | d70ef00 | 2018-10-02 13:01:27 +0200 | [diff] [blame] | 126 | } |
Denis Egorenko | a0f1b2f | 2018-11-23 17:24:35 +0000 | [diff] [blame] | 127 | def buildTime = new Date().format("yyyyMMdd-HH:mm:ss.SSS", TimeZone.getTimeZone('UTC')) |
azvyagintsev | 388a1bd | 2018-09-06 12:39:49 +0300 | [diff] [blame] | 128 | |
| 129 | if (setDefaultArtifactoryProperties) { |
| 130 | common.infoMsg("Processing artifactory props for : ${targetImageFull}") |
| 131 | LinkedHashMap artifactoryProperties = [:] |
Denis Egorenko | 343a571 | 2018-12-10 19:08:45 +0400 | [diff] [blame] | 132 | def tgtImageInfo = getImageInfo(targetImageFull) |
| 133 | def tgt_image_sha256 = tgtImageInfo['sha256'] |
| 134 | def unique_image_id = tgtImageInfo['id'] |
| 135 | def tgtImgUrl = imageURL(targetRegistry, targetImageFull, tgt_image_sha256) - '/manifest.json' |
Denis Egorenko | a0f1b2f | 2018-11-23 17:24:35 +0000 | [diff] [blame] | 136 | artifactoryProperties = [ |
Kyrylo Mashchenko | a86912d | 2018-12-21 08:31:47 +0000 | [diff] [blame] | 137 | 'com.mirantis.targetTag' : env.IMAGE_TAG, |
| 138 | 'com.mirantis.uniqueImageId': unique_image_id, |
Denis Egorenko | a0f1b2f | 2018-11-23 17:24:35 +0000 | [diff] [blame] | 139 | ] |
| 140 | if (external) { |
| 141 | artifactoryProperties << ['com.mirantis.externalImage': external] |
| 142 | } |
Denis Egorenko | a0f1b2f | 2018-11-23 17:24:35 +0000 | [diff] [blame] | 143 | def historyProperties = [] |
Denis Egorenko | 4c9b3c5 | 2018-12-24 16:11:54 +0400 | [diff] [blame] | 144 | try { |
| 145 | def sourceRegistry = sourceImage.split('/')[0] |
| 146 | def sourceImgUrl = imageURL(sourceRegistry, sourceImage, source_image_sha256) - '/manifest.json' |
| 147 | def existingProps = mcp_artifactory.getPropertiesForArtifact(sourceImgUrl) |
| 148 | // check does the source image have already history props |
| 149 | if (existingProps) { |
| 150 | historyProperties = existingProps.get('com.mirantis.versionHistory', []) |
| 151 | } |
| 152 | } catch (Exception e) { |
| 153 | common.warningMsg("Can't find history for ${sourceImage}.") |
Denis Egorenko | a0f1b2f | 2018-11-23 17:24:35 +0000 | [diff] [blame] | 154 | } |
| 155 | // %5C - backslash symbol is needed |
| 156 | historyProperties.add("${buildTime}%5C=${sourceImage}") |
Kyrylo Mashchenko | a86912d | 2018-12-21 08:31:47 +0000 | [diff] [blame] | 157 | artifactoryProperties << [ 'com.mirantis.versionHistory': historyProperties.join(',') ] |
Denis Egorenko | a0f1b2f | 2018-11-23 17:24:35 +0000 | [diff] [blame] | 158 | common.infoMsg("artifactoryProperties=> ${artifactoryProperties}") |
| 159 | common.retry(3, 5) { |
Denis Egorenko | 0191839 | 2018-12-10 12:28:46 +0400 | [diff] [blame] | 160 | mcp_artifactory.setProperties(tgtImgUrl, artifactoryProperties) |
azvyagintsev | 388a1bd | 2018-09-06 12:39:49 +0300 | [diff] [blame] | 161 | } |
| 162 | } |
Richard Felkl | 7c920d0 | 2017-12-11 15:28:18 +0100 | [diff] [blame] | 163 | } |
Richard Felkl | e90ef1c | 2017-12-08 00:13:56 +0100 | [diff] [blame] | 164 | } |
Jakub Josef | a63f986 | 2018-01-11 17:58:38 +0100 | [diff] [blame] | 165 | } catch (Throwable e) { |
azvyagintsev | 388a1bd | 2018-09-06 12:39:49 +0300 | [diff] [blame] | 166 | // Stub for future processing |
Jakub Josef | a63f986 | 2018-01-11 17:58:38 +0100 | [diff] [blame] | 167 | currentBuild.result = "FAILURE" |
| 168 | throw e |
Richard Felkl | e90ef1c | 2017-12-08 00:13:56 +0100 | [diff] [blame] | 169 | } |
Richard Felkl | e90ef1c | 2017-12-08 00:13:56 +0100 | [diff] [blame] | 170 | } |
azvyagintsev | d162b91 | 2018-07-26 10:52:14 +0200 | [diff] [blame] | 171 | } |