blob: c5aaca0e67e823ef2524f412fb2341c52cb71185 [file] [log] [blame]
Jakub Josef79ecec32017-02-17 14:36:28 +01001package com.mirantis.mk
2
3/**
4 *
5 * Aptly functions
6 *
7 */
8
9/**
10 * Upload package into local repo
11 *
12 * @param file File path
13 * @param server Server host
14 * @param repo Repository name
15 */
16def uploadPackage(file, server, repo, skipExists=false) {
17 def pkg = file.split('/')[-1].split('_')[0]
18 def jobName = currentBuild.build().environment.JOB_NAME
19
20 sh("curl -v -f -F file=@${file} ${server}/api/files/${pkg}")
21 sh("curl -v -o curl_out_${pkg}.log -f -X POST ${server}/api/repos/${repo}/file/${pkg}")
22
23 try {
24 sh("cat curl_out_${pkg}.log | json_pp | grep 'Unable to add package to repo' && exit 1 || exit 0")
25 } catch (err) {
26 sh("curl -s -f -X DELETE ${server}/api/files/${pkg}")
27 if (skipExists == true) {
28 println "[WARN] Package ${pkg} already exists in repo so skipping it's upload as requested"
29 } else {
30 error("Package ${pkg} already exists in repo, did you forget to add changelog entry and raise version?")
31 }
32 }
33}
34
35/**
36 * Build step to upload package. For use with eg. parallel
37 *
38 * @param file File path
39 * @param server Server host
40 * @param repo Repository name
41 */
42def uploadPackageStep(file, server, repo, skipExists=false) {
43 return {
44 uploadPackage(
45 file,
46 server,
47 repo,
48 skipExists
49 )
50 }
51}
52
53def snapshotRepo(server, repo, timestamp = null) {
54 // XXX: timestamp parameter is obsoleted, time of snapshot creation is
55 // what we should always use, not what calling pipeline provides
56 def now = new Date();
57 def ts = now.format("yyyyMMddHHmmss", TimeZone.getTimeZone('UTC'));
58
59 def snapshot = "${repo}-${ts}"
60 sh("curl -f -X POST -H 'Content-Type: application/json' --data '{\"Name\":\"$snapshot\"}' ${server}/api/repos/${repo}/snapshots")
61}
62
63def cleanupSnapshots(server, config='/etc/aptly-publisher.yaml', opts='-d --timeout 600') {
64 sh("aptly-publisher -c ${config} ${opts} --url ${server} cleanup")
65}
66
67def diffPublish(server, source, target, components=null, opts='--timeout 600') {
68 if (components) {
69 def componentsStr = components.join(' ')
70 opts = "${opts} --components ${componentsStr}"
71 }
72 sh("aptly-publisher --dry --url ${server} promote --source ${source} --target ${target} --diff ${opts}")
73}
74
chnyda751ac402017-11-28 17:32:43 +010075def promotePublish(server, source, target, recreate=false, components=null, packages=null, diff=false, opts='-d --timeout 600', dump_publish=false, storage="") {
chnydabe46ffa2017-03-21 13:19:08 +010076 if (components && components != "all" && components != "") {
77 def componentsStr = components.replaceAll(",", " ")
Jakub Josef79ecec32017-02-17 14:36:28 +010078 opts = "${opts} --components ${componentsStr}"
79 }
chnydabe46ffa2017-03-21 13:19:08 +010080 if (packages && packages != "all" && packages != "") {
81 def packagesStr = packages.replaceAll(",", " ")
Jakub Josef79ecec32017-02-17 14:36:28 +010082 opts = "${opts} --packages ${packagesStr}"
83 }
84 if (recreate.toBoolean() == true) {
85 opts = "${opts} --recreate"
86 }
87 if (diff.toBoolean() == true) {
chnyda5de5d432017-04-10 15:35:47 +020088 opts = "${opts} --dry --diff"
Jakub Josef79ecec32017-02-17 14:36:28 +010089 }
chnydabe46ffa2017-03-21 13:19:08 +010090
chnyda751ac402017-11-28 17:32:43 +010091 if (storage && storage != "") {
92 opts = "${opts} --storage ${storage}"
93 }
94
chnyda4906ce02017-08-22 14:45:13 +020095 if (dump_publish) {
96 def now = new Date();
chnydaf2f95792017-12-11 17:09:27 +010097 dumpTarget = target
chnyda4906ce02017-08-22 14:45:13 +020098 def timestamp = now.format("yyyyMMddHHmmss", TimeZone.getTimeZone('UTC'));
chnydaf2f95792017-12-11 17:09:27 +010099
chnyda7bdfeab2017-12-18 17:10:14 +0100100 if (source.contains(')') || source.contains('*')) {
chnydaf2f95792017-12-11 17:09:27 +0100101 sourceTarget = source.split('/')
102 dumpTarget = target.split('/')[-1]
103 sourceTarget[-1] = dumpTarget
104 dumpTarget = sourceTarget.join('/')
105 }
106
107 dumpPublishes(server, timestamp, dumpTarget)
chnyda4906ce02017-08-22 14:45:13 +0200108 }
chnydac78dfa52017-08-22 14:39:43 +0200109
chnydadc7292f2018-01-17 14:30:22 +0100110 sh("aptly-publisher --url ${server} promote --acquire-by-hash --source '${source}' --target '${target}' --force-overwrite ${opts}")
chnydac78dfa52017-08-22 14:39:43 +0200111
Jakub Josef79ecec32017-02-17 14:36:28 +0100112}
113
Filip Pytloun15fbbc92017-11-27 14:33:22 +0100114def publish(server, config='/etc/aptly-publisher.yaml', recreate=false, only_latest=true, force_overwrite=true, opts='-d --timeout 3600') {
Jakub Josef79ecec32017-02-17 14:36:28 +0100115 if (recreate == true) {
116 opts = "${opts} --recreate"
117 }
Filip Pytloun62fe3a42017-11-27 14:30:19 +0100118 if (only_latest == true) {
119 opts = "${opts} --only-latest"
120 }
121 if (force_overwrite == true) {
122 opts = "${opts} --force-overwrite"
123 }
chnydadc7292f2018-01-17 14:30:22 +0100124 sh("aptly-publisher --url ${server} -c ${config} ${opts} --acquire-by-hash publish")
chnydad8f51af2017-07-24 10:39:48 +0200125}
126
127/**
128 * Dump publishes
129 *
130 * @param server Server host
131 * @param save-dir Directory where publishes are to be serialized
132 * @param publishes Publishes to be serialized
133 * @param prefix Prefix of dump files
134 * @param opts Options: debug, timeout, ...
135 */
chnyda5fe91ca2017-09-19 09:19:46 +0200136def dumpPublishes(server, prefix, publishes='all', opts='-d --timeout 600') {
chnydaf2f95792017-12-11 17:09:27 +0100137 sh("aptly-publisher dump --url ${server} --save-dir . --prefix ${prefix} -p '${publishes}' ${opts}")
chnyda5fe91ca2017-09-19 09:19:46 +0200138 archiveArtifacts artifacts: "${prefix}*"
chnydad8f51af2017-07-24 10:39:48 +0200139}
140
141/**
142 * Restore publish from YAML file
143 *
144 * @param server Server host
145 * @param recreate Recreate publishes
146 * @param publish Serialized YAML of Publish
147 * @param components Components to restore
148 */
149def restorePublish(server, recreate, publish, components='all') {
150
151 opts = ""
152 if (recreate) {
153 opts << " --recreate"
154 }
155
156 sh("rm tmpFile || true")
157 writeFile(file: "tmpFile", text: publish)
158 sh("aptly-publisher restore --url ${server} --restore-file tmpFile --components ${components} ${opts}")
159}
Oleg Iurchenko85bb5d72018-02-15 16:45:22 +0200160
161/**
162 * The function return name of the snapshot belongs to repo considering the prefix with storage by REST API.
163 *
164 * @param server URI the server to connect to aptly API
165 * @param destribution Destiribution of the repo which have to be found
166 * @param prefix Prefix of the repo including storage eg. prefix or s3:aptcdn:prefix
167 * @param component Component of the repo
168 *
169 * @return snapshot name
170 **/
171def getSnapshotByAPI(server, distribution, prefix, component) {
172 http = new com.mirantis.mk.Http()
173 def list_published = http.restGet(server, '/api/publish')
174 def storage
175 for (items in list_published) {
176 for (row in items) {
177 if (prefix.tokenize(':')[1]) {
178 storage = prefix.tokenize(':')[0] + ':' + prefix.tokenize(':')[1]
179 } else {
180 storage = ''
181 }
182 if (row.key == 'Distribution' && row.value == distribution && items['Prefix'] == prefix.tokenize(':').last() && items['Storage'] == storage) {
183 for (source in items['Sources']){
184 if (source['Component'] == component) {
185 if(env.getEnvironment().containsKey('DEBUG') && env['DEBUG'] == "true"){
186 println ('Snapshot belongs to ' + distribution + '/' + prefix + ': ' + source['Name'])
187 }
188 return source['Name']
189 }
190 }
191 }
192 }
193 }
194 return false
195}
196
197/**
198 * Returns list of the packages matched to pattern and
199 * belonged to particular snapshot by REST API
200 *
201 * @param server URI of the server insluding port and protocol
202 * @param snapshotName Snapshot to check
203 * @param packagesList Pattern of the components to be compared
204 **/
205def snapshotPackagesByAPI(server, snapshotName, packagesList) {
206 http = new com.mirantis.mk.Http()
207 def pkgs = http.restGet(server, "/api/snapshots/${snapshotName}/packages")
208 def packages = []
209
210 for (package_pattern in packagesList.tokenize(',')) {
211 def pkg = pkgs.find { item -> item.contains(package_pattern) }
212 packages.add(pkg)
213 }
214
215 return packages
216}
217
218
219/**
220 * Creates snapshot of the repo or package refs by REST API
221 * @param server URI of the server insluding port and protocol
222 * @param repo Local repo name
223 * @param snapshotName Snapshot name is going to be created
224 * @param snapshotDescription Snapshot description
225 * @param packageRefs List of the packages are going to be included into the snapshot
226 **/
227def snapshotCreateByAPI(server, repo, snapshotName, snapshotDescription = null, packageRefs = null) {
228 http = new com.mirantis.mk.Http()
229 def data = [:]
230 data['Name'] = snapshotName
231 if (snapshotDescription) {
232 data['Description'] = snapshotDescription
233 } else {
234 data['Description'] = "Snapshot of ${repo} repo"
235 }
236 if (packageRefs) {
237 data['PackageRefs'] = packageRefs
238 http.restPost(server, '/api/snapshots', data)
239 } else {
240 http.restPost(server + "/api/repos/${repo}/snapshots", data)
241 }
242}
243
244/**
245 * Publishes the snapshot accodgin to distribution, components and prefix by REST API
246 * @param server URI of the server insluding port and protocol
247 * @param snapshotName Snapshot is going to be published
248 * @param distribution Distribution for the published repo
249 * @param components Component for the published repo
250 * @param prefix Prefix for thepubslidhed repo including storage
251 **/
252def snapshotPublishByAPI(server, snapshotName, distribution, components, prefix) {
253 http = new com.mirantis.mk.Http()
254 def source = [:]
255 source['Name'] = snapshotName
256 source['Component'] = components
257 def data = [:]
258 data['SourceKind'] = 'snapshot'
259 data['Sources'] = [source]
260 data['Architectures'] = ['amd64']
261 data['Distribution'] = distribution
262 return http.restPost(server, "/api/publish/${prefix}", data)
263}
264
265/**
266 * Unpublish Aptly repo by REST API
267 *
268 * @param aptlyServer Aptly connection object
269 * @param aptlyPrefix Aptly prefix where need to delete a repo
270 * @param aptlyRepo Aptly repo name
271 */
272def unpublishByAPI(aptlyServer, aptlyPrefix, aptlyRepo){
273 http = new com.mirantis.mk.Http()
274 http.restDelete(aptlyServer, "/api/publish/${aptlyPrefix}/${aptlyRepo}")
275}
276
277/**
278 * Delete Aptly repo by REST API
279 *
280 * @param aptlyServer Aptly connection object
281 * @param aptlyRepo Aptly repo name
282 */
283def deleteRepoByAPI(aptlyServer, aptlyRepo){
284 http = new com.mirantis.mk.Http()
285 http.restDelete(aptlyServer, "/api/repos/${aptlyRepo}")
286}