stage_4 support
[osm/devops.git] / jenkins / ci-pipelines / ci_stage_3.groovy
1 /* Copyright 2017 Sandvine
2  *
3  * All Rights Reserved.
4  * 
5  *   Licensed under the Apache License, Version 2.0 (the "License"); you may
6  *   not use this file except in compliance with the License. You may obtain
7  *   a copy of the License at
8  *
9  *        http://www.apache.org/licenses/LICENSE-2.0
10  *
11  *   Unless required by applicable law or agreed to in writing, software
12  *   distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
13  *   WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
14  *   License for the specific language governing permissions and limitations
15  *   under the License.
16  */
17
18 properties([
19     parameters([
20         string(defaultValue: env.GERRIT_BRANCH, description: '', name: 'GERRIT_BRANCH'),
21         string(defaultValue: 'system', description: '', name: 'NODE'),
22         string(defaultValue: '', description: '', name: 'BUILD_FROM_SOURCE'),
23         string(defaultValue: 'unstable', description: '', name: 'REPO_DISTRO'),
24         string(defaultValue: '', description: '', name: 'COMMIT_ID'),
25         string(defaultValue: '-stage_2', description: '', name: 'UPSTREAM_SUFFIX'),
26         string(defaultValue: 'pubkey.asc', description: '', name: 'REPO_KEY_NAME'),
27         string(defaultValue: 'release', description: '', name: 'RELEASE'),
28         string(defaultValue: '', description: '', name: 'UPSTREAM_JOB_NAME'),
29         string(defaultValue: '', description: '', name: 'UPSTREAM_JOB_NUMBER'),
30         string(defaultValue: '', description: '', name: 'UPSTREAM_JOB_NUMBER'),
31         string(defaultValue: 'dpkg1', description: '', name: 'GPG_KEY_NAME'),
32         string(defaultValue: 'artifactory-osm', description: '', name: 'ARTIFACTORY_SERVER'),
33         string(defaultValue: 'osm-stage_4', description: '', name: 'DOWNSTREAM_STAGE_NAME'),
34         booleanParam(defaultValue: false, description: '', name: 'SAVE_CONTAINER_ON_FAIL'),
35         booleanParam(defaultValue: false, description: '', name: 'SAVE_CONTAINER_ON_PASS'),
36         booleanParam(defaultValue: false, description: '', name: 'DO_STAGE_4'),
37     ])
38 ])
39
40 node("${params.NODE}") {
41
42     sh 'env'
43
44     tag_or_branch = params.GERRIT_BRANCH.replaceAll(/\./,"")
45
46     stage("Checkout") {
47         checkout scm
48     }
49
50     ci_helper = load "jenkins/ci-pipelines/ci_helper.groovy"
51
52     def upstream_main_job = params.UPSTREAM_SUFFIX
53     def save_artifacts = false
54
55     // upstream jobs always use merged artifacts
56     upstream_main_job += '-merge'
57     container_name_prefix = "osm-${tag_or_branch}"
58     container_name = "${container_name_prefix}"
59     if ( JOB_NAME.contains('merge') ) {
60         save_artifacts = true
61         println("merge job, saving artifacts")
62         container_name += "-merge"
63     }
64     container_name += "-${BUILD_NUMBER}"
65
66     // Copy the artifacts from the upstream jobs
67     stage("Copy Artifacts") {
68         // cleanup any previous repo
69         sh 'rm -rf repo'
70         dir("repo") {
71             // grab all stable upstream builds based on the
72
73             dir("${RELEASE}") {
74                 def list = ["SO", "UI", "RO", "openvim", "osmclient", "IM"]
75                 for (component in list) {
76                     step ([$class: 'CopyArtifact',
77                            projectName: "${component}${upstream_main_job}/${GERRIT_BRANCH}"])
78
79                     // grab the build name/number
80                     //options = get_env_from_build('build.env')
81                     build_num = ci_helper.get_env_value('build.env','BUILD_NUMBER')
82
83                     // grab the archives from the stage_2 builds (ie. this will be the artifacts stored based on a merge)
84                     ci_helper.get_archive(params.ARTIFACTORY_SERVER,component,GERRIT_BRANCH, "${component}${upstream_main_job} :: ${GERRIT_BRANCH}", build_num)
85
86                     // cleanup any prevously defined dists
87                     sh "rm -rf dists"
88                 }
89
90                 // check if an upstream artifact based on specific build number has been requested
91                 // This is the case of a merge build and the upstream merge build is not yet complete (it is not deemed
92                 // a successful build yet). The upstream job is calling this downstream job (with the its build artifiact)
93                 if ( params.UPSTREAM_JOB_NAME ) {
94                     step ([$class: 'CopyArtifact',
95                            projectName: "${params.UPSTREAM_JOB_NAME}",
96                            selector: [$class: 'SpecificBuildSelector', buildNumber: "${params.UPSTREAM_JOB_NUMBER}"]
97                           ])
98
99                     //options = get_env_from_build('build.env')
100                     // grab the build name/number
101                     //build_num = sh(returnStdout:true,  script: "cat build.env | awk -F= '/BUILD_NUMBER/{print \$2}'").trim()
102                     build_num = ci_helper.get_env_value('build.env','BUILD_NUMBER')
103                     component = ci_helper.get_mdg_from_project(ci_helper.get_env_value('build.env','GERRIT_PROJECT'))
104
105                     // the upstream job name contains suffix with the project. Need this stripped off
106                     def project_without_branch = params.UPSTREAM_JOB_NAME.split('/')[0]
107
108                     ci_helper.get_archive(params.ARTIFACTORY_SERVER,component,GERRIT_BRANCH, "${project_without_branch} :: ${GERRIT_BRANCH}", build_num)
109
110                     sh "rm -rf dists"
111                 }
112                 
113                 // sign all the components
114                 for (component in list) {
115                     sh "dpkg-sig --sign builder -k ${GPG_KEY_NAME} pool/${component}/*"
116                 }
117
118                 // now create the distro
119                 for (component in list) {
120                     sh "mkdir -p dists/${params.REPO_DISTRO}/${component}/binary-amd64/"
121                     sh "apt-ftparchive packages pool/${component} > dists/${params.REPO_DISTRO}/${component}/binary-amd64/Packages"
122                     sh "gzip -9fk dists/${params.REPO_DISTRO}/${component}/binary-amd64/Packages"
123                 }
124
125                 // create and sign the release file
126                 sh "apt-ftparchive release dists/${params.REPO_DISTRO} > dists/${params.REPO_DISTRO}/Release"
127                 sh "gpg --yes -abs -u ${GPG_KEY_NAME} -o dists/${params.REPO_DISTRO}/Release.gpg dists/${params.REPO_DISTRO}/Release"
128
129                 // copy the public key into the release folder
130                 // this pulls the key from the home dir of the current user (jenkins)
131                 sh "cp ~/${REPO_KEY_NAME} ."
132             }
133             // start an apache server to serve up the images
134             http_server_name = "${container_name}-apache"
135
136             pwd = sh(returnStdout:true,  script: 'pwd').trim()
137             repo_base_url = ci_helper.start_http_server(pwd,http_server_name)
138         }
139     }
140
141     error = null
142
143     try {
144         stage("Install") {
145
146             //will by default always delete containers on complete
147             //sh "jenkins/system/delete_old_containers.sh ${container_name_prefix}"
148
149             commit_id = ''
150             repo_distro = ''
151             repo_key_name = ''
152             release = ''
153
154             if ( params.COMMIT_ID )
155             {
156                 commit_id = "-b ${params.COMMIT_ID}"
157             }
158
159             if ( params.REPO_DISTRO )
160             {
161                 repo_distro = "-r ${params.REPO_DISTRO}"
162             }
163
164             if ( params.REPO_KEY_NAME )
165             {
166                 repo_key_name = "-k ${params.REPO_KEY_NAME}"
167             }
168
169             if ( params.RELEASE )
170             {
171                 release = "-R ${params.RELEASE}"
172             }
173      
174             sh """
175                 export OSM_USE_LOCAL_DEVOPS=true
176                 jenkins/host/start_build system --build-container ${container_name} \
177                                                 ${commit_id} \
178                                                 ${repo_distro} \
179                                                 ${repo_base_url} \
180                                                 ${repo_key_name} \
181                                                 ${release} \
182                                                 ${params.BUILD_FROM_SOURCE}
183                """
184         }
185
186         stage("Smoke") {
187             ci_helper.systest_run(container_name, 'smoke')
188             junit '*.xml'
189         }
190
191         if ( params.DO_STAGE_4 ) {
192             stage("stage_4") {
193                 def downstream_params = [
194                     string(name: 'CONTAINER_NAME', value: container_name),
195                 ]
196                 stage_4_result = build job: "${params.DOWNSTREAM_STAGE_NAME}/${GERRIT_BRANCH}", parameters: downstream_params, propagate: false 
197                
198                 currentBuild.result = stage_4_result.result
199             }
200         }
201
202         // save the artifacts of this build if this is a merge job
203         if ( save_artifacts ) {
204             stage("Archive") {
205                 sh "echo ${container_name} > build_version.txt"
206                 archiveArtifacts artifacts: "build_version.txt", fingerprint: true
207
208                 // Archive the tested repo
209                 dir("repo/${RELEASE}") {
210                     ci_helper.archive(params.ARTIFACTORY_SERVER,RELEASE,GERRIT_BRANCH,'tested')
211                 }
212             }
213         }
214     }
215     catch(caughtError) {
216         println("Caught error!")
217         error = caughtError
218         currentBuild.result = 'FAILURE'
219     }
220     finally {
221         sh "docker stop ${http_server_name}"
222
223         if (error) {
224             if ( !params.SAVE_CONTAINER_ON_FAIL ) {
225                 sh "lxc delete ${container_name} --force"
226             }
227             throw error 
228         }
229         else {
230             if ( !params.SAVE_CONTAINER_ON_PASS ) {
231                 sh "lxc delete ${container_name} --force"
232             }
233         }
234     }
235 }