blob: 636c66637cc9c8004f7dffb2e0583e8fb5514153 [file] [log] [blame]
Richard Felkle90ef1c2017-12-08 00:13:56 +01001/**
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
azvyagintsevd162b912018-07-26 10:52:14 +020010 * SOURCE_IMAGE_TAG Tag to use when pulling images(optional,if SOURCE_IMAGE_TAG has been found)
azvyagintsev388a1bd2018-09-06 12:39:49 +030011 * SET_DEFAULT_ARTIFACTORY_PROPERTIES Add extra props. directly to artifactory,
Richard Felkle90ef1c2017-12-08 00:13:56 +010012 * IMAGE_LIST List of images to mirror
azvyagintsev388a1bd2018-09-06 12:39:49 +030013 * 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 Felkle90ef1c2017-12-08 00:13:56 +010017 *
18 */
azvyagintsev388a1bd2018-09-06 12:39:49 +030019import java.util.regex.Pattern
20import groovy.json.JsonSlurper
Richard Felkle90ef1c2017-12-08 00:13:56 +010021
azvyagintsev388a1bd2018-09-06 12:39:49 +030022common = new com.mirantis.mk.Common()
23external = false
24externalMarker = '/mirantis/external/'
Richard Felkle90ef1c2017-12-08 00:13:56 +010025
azvyagintsev388a1bd2018-09-06 12:39:49 +030026slaveNode = env.SLAVE_NODE ?: 'docker'
27setDefaultArtifactoryProperties = env.SET_DEFAULT_ARTIFACTORY_PROPERTIES ?: true
28
Richard Felkle90ef1c2017-12-08 00:13:56 +010029def getImageName(String image) {
30 def regex = Pattern.compile('(?:.+/)?([^:]+)(?::.+)?')
31 def matcher = regex.matcher(image)
azvyagintsev388a1bd2018-09-06 12:39:49 +030032 if (matcher.find()) {
Richard Felkle90ef1c2017-12-08 00:13:56 +010033 def imageName = matcher.group(1)
34 return imageName
azvyagintsev388a1bd2018-09-06 12:39:49 +030035 } else {
36 error("Wrong format of image name.")
Richard Felkle90ef1c2017-12-08 00:13:56 +010037 }
38}
azvyagintsev388a1bd2018-09-06 12:39:49 +030039
40timeout(time: 4, unit: 'HOURS') {
41 node(slaveNode) {
Denis Egorenko899ee322018-11-19 17:35:50 +040042 def user = ''
43 wrap([$class: 'BuildUser']) {
44 user = env.BUILD_USER_ID
45 }
46 currentBuild.description = "${user}: [${env.SOURCE_IMAGE_TAG} => ${env.IMAGE_TAG}]\n${env.IMAGE_LIST}"
Jakub Josefa63f9862018-01-11 17:58:38 +010047 try {
azvyagintsev388a1bd2018-09-06 12:39:49 +030048 stage("Mirror Docker Images") {
49
Jakub Josefa63f9862018-01-11 17:58:38 +010050 def images = IMAGE_LIST.tokenize('\n')
azvyagintsev388a1bd2018-09-06 12:39:49 +030051 def imageName, sourceImage, targetRegistryPath, imageArray
52 for (image in images) {
53 if (image.trim().indexOf(' ') == -1) {
54 error("Wrong format of image and target repository input")
Jakub Josefa63f9862018-01-11 17:58:38 +010055 }
56 imageArray = image.trim().tokenize(' ')
azvyagintsev388a1bd2018-09-06 12:39:49 +030057 sourceImage = imageArray[0]
58 if (sourceImage.contains('SUBS_SOURCE_IMAGE_TAG')) {
59 common.warningMsg("Replacing SUBS_SOURCE_IMAGE_TAG => ${env.SOURCE_IMAGE_TAG}")
60 sourceImage = sourceImage.replace('SUBS_SOURCE_IMAGE_TAG', env.SOURCE_IMAGE_TAG)
azvyagintsevd162b912018-07-26 10:52:14 +020061 }
azvyagintsev388a1bd2018-09-06 12:39:49 +030062 targetRegistryPath = imageArray[1]
63 targetRegistry = imageArray[1].split('/')[0]
64 imageName = getImageName(sourceImage)
65 targetImageFull = "${targetRegistryPath}/${imageName}:${env.IMAGE_TAG}"
Denis Egorenko2ee651c2018-11-23 17:41:38 +040066
67 def mcp_artifactory = new com.mirantis.mcp.MCPArtifactory()
68 if (targetImageFull.contains(externalMarker)) {
69 external = true
70 // check if images exists - raise error, as we don't want to rewrite existing one
71 def imageRepo = targetRegistryPath - targetRegistry
Denis Egorenko1b207d12018-11-26 15:35:21 +040072 if (mcp_artifactory.imageExists(env.REGISTRY_URL, "${imageRepo}/${imageName}", env.IMAGE_TAG)) {
Denis Egorenko2ee651c2018-11-23 17:41:38 +040073 error("Image ${targetImageFull} already exists!")
74 }
75 }
76
azvyagintsev388a1bd2018-09-06 12:39:49 +030077 srcImage = docker.image(sourceImage)
azvyagintsev86ba4a02018-10-03 19:21:28 +030078 common.retry(3, 5) {
79 srcImage.pull()
80 }
azvyagintsev388a1bd2018-09-06 12:39:49 +030081 // Use sh-docker call for tag, due magic code in plugin:
82 // https://github.com/jenkinsci/docker-workflow-plugin/blob/docker-workflow-1.17/src/main/resources/org/jenkinsci/plugins/docker/workflow/Docker.groovy#L168-L170
83 sh("docker tag ${srcImage.id} ${targetImageFull}")
84 common.infoMsg("Attempt to push docker image into remote registry: ${env.REGISTRY_URL}")
azvyagintsev86ba4a02018-10-03 19:21:28 +030085 common.retry(3, 5) {
86 docker.withRegistry(env.REGISTRY_URL, env.TARGET_REGISTRY_CREDENTIALS_ID) {
87 sh("docker push ${targetImageFull}")
88 }
Alexander Evseevd70ef002018-10-02 13:01:27 +020089 }
Denis Egorenkoa0f1b2f2018-11-23 17:24:35 +000090 def buildTime = new Date().format("yyyyMMdd-HH:mm:ss.SSS", TimeZone.getTimeZone('UTC'))
azvyagintsev388a1bd2018-09-06 12:39:49 +030091
92 if (setDefaultArtifactoryProperties) {
93 common.infoMsg("Processing artifactory props for : ${targetImageFull}")
94 LinkedHashMap artifactoryProperties = [:]
95 // Get digest of pushed image
96 String unique_image_id = sh(
97 script: "docker inspect --format='{{index .RepoDigests 0}}' '${targetImageFull}'",
98 returnStdout: true,
99 ).trim()
100 def image_sha256 = unique_image_id.tokenize(':')[1]
101 def ret = new URL("https://${targetRegistry}/artifactory/api/search/checksum?sha256=${image_sha256}").getText()
102 // Most probably, we would get many images, especially for external images. We need to guess
103 // exactly one, which we pushing now
104 guessImage = targetImageFull.replace(':', '/').replace(targetRegistry, '')
105 ArrayList img_data = new JsonSlurper().parseText(ret)['results']
Denis Egorenkoa0f1b2f2018-11-23 17:24:35 +0000106 def imgUrl = img_data*.uri.find { it.contains(guessImage) } - '/manifest.json'
107 artifactoryProperties = [
108 'com.mirantis.targetTag' : env.IMAGE_TAG,
109 'com.mirantis.uniqueImageId': unique_image_id,
110 ]
111 if (external) {
112 artifactoryProperties << ['com.mirantis.externalImage': external]
113 }
Denis Egorenkoa0f1b2f2018-11-23 17:24:35 +0000114 def existingProps = mcp_artifactory.getPropertiesForArtifact(imgUrl)
115 def historyProperties = []
116 // check does image have already some props
117 if (existingProps) {
118 historyProperties = existingProps.get('com.mirantis.versionHistory', [])
119 }
120 // %5C - backslash symbol is needed
121 historyProperties.add("${buildTime}%5C=${sourceImage}")
122 artifactoryProperties << [ 'com.mirantis.versionHistory': historyProperties.join(',') ]
123 common.infoMsg("artifactoryProperties=> ${artifactoryProperties}")
124 common.retry(3, 5) {
125 mcp_artifactory.setProperties(imgUrl, artifactoryProperties)
azvyagintsev388a1bd2018-09-06 12:39:49 +0300126 }
127 }
Richard Felkl7c920d02017-12-11 15:28:18 +0100128 }
Richard Felkle90ef1c2017-12-08 00:13:56 +0100129 }
Jakub Josefa63f9862018-01-11 17:58:38 +0100130 } catch (Throwable e) {
azvyagintsev388a1bd2018-09-06 12:39:49 +0300131 // Stub for future processing
Jakub Josefa63f9862018-01-11 17:58:38 +0100132 currentBuild.result = "FAILURE"
133 throw e
Richard Felkle90ef1c2017-12-08 00:13:56 +0100134 }
Richard Felkle90ef1c2017-12-08 00:13:56 +0100135 }
azvyagintsevd162b912018-07-26 10:52:14 +0200136}