blob: 26c2818b295c5a9b7f985c1184d235ea8f9023bf [file] [log] [blame]
Filip Pytloun0a07f702017-02-24 18:26:18 +01001/**
2 *
3 * Launch heat stack with CI/CD lab infrastructure
4 *
5 * Expected parameters:
6 * HEAT_TEMPLATE_URL URL to git repo with Heat templates
7 * HEAT_TEMPLATE_CREDENTIALS Credentials to the Heat templates repo
8 * HEAT_TEMPLATE_BRANCH Heat templates repo branch
9 * HEAT_STACK_NAME Heat stack name
10 * HEAT_STACK_TEMPLATE Heat stack HOT template
11 * HEAT_STACK_ENVIRONMENT Heat stack environmental parameters
12 * HEAT_STACK_ZONE Heat stack availability zone
13 * HEAT_STACK_PUBLIC_NET Heat stack floating IP pool
14 * HEAT_STACK_DELETE Delete Heat stack when finished (bool)
15 * HEAT_STACK_CLEANUP_JOB Name of job for deleting Heat stack
16 * HEAT_STACK_REUSE Reuse Heat stack (don't create one)
17 *
18 * SALT_MASTER_CREDENTIALS Credentials to the Salt API
Filip Pytloune32fda82017-02-24 18:26:18 +010019 * SALT_MASTER_PORT Port of salt-api, defaults to 8000
Filip Pytloun0a07f702017-02-24 18:26:18 +010020 *
21 * OPENSTACK_API_URL OpenStack API address
22 * OPENSTACK_API_CREDENTIALS Credentials to the OpenStack API
23 * OPENSTACK_API_PROJECT OpenStack project to connect to
24 * OPENSTACK_API_CLIENT Versions of OpenStack python clients
25 * OPENSTACK_API_VERSION Version of the OpenStack API (2/3)
26 *
27 */
28
29git = new com.mirantis.mk.Git()
30openstack = new com.mirantis.mk.Openstack()
31salt = new com.mirantis.mk.Salt()
32orchestrate = new com.mirantis.mk.Orchestrate()
33
Filip Pytloun8fe1f7e2017-03-01 18:45:09 +010034def waitForServices(saltMaster) {
35 retry(30) {
36 out = salt.cmdRun(saltMaster, 'I@docker:swarm:role:master', """/bin/bash -c 'docker service ls | grep -E "0/[0-9]+"' && echo 'Some services are not running'""")
37 for (int a = 0; a < out['return'].size(); a++) {
38 def entry = out['return'].get(a)
39 for (int i = 0; i < entry.size(); i++) {
40 def node = entry.get(i)
41 if (node) {
42 if (node.value =~ /Some services are not running/) {
43 sleep(10)
44 throw new Exception("$node.key: $node.value")
45 } else {
46 print out
47 }
48 }
49 }
50 }
51 }
52}
53
Filip Pytlounbfce09d2017-03-01 19:00:43 +010054timestamps {
55 node {
56 try {
57 // connection objects
58 def openstackCloud
59 def saltMaster
Filip Pytloun0a07f702017-02-24 18:26:18 +010060
Filip Pytlounbfce09d2017-03-01 19:00:43 +010061 // value defaults
62 def openstackVersion = OPENSTACK_API_CLIENT ? OPENSTACK_API_CLIENT : 'liberty'
63 def openstackEnv = "${env.WORKSPACE}/venv"
Filip Pytloun0a07f702017-02-24 18:26:18 +010064
Filip Pytloun794ad952017-03-03 10:39:26 +010065 if (HEAT_STACK_REUSE.toBoolean() == true && HEAT_STACK_NAME == '') {
66 error("If you want to reuse existing stack you need to provide it's name")
67 }
68
69 if (HEAT_STACK_REUSE.toBoolean() == false) {
70 // Don't allow to set custom heat stack name
71 wrap([$class: 'BuildUser']) {
72 HEAT_STACK_NAME = "${env.BUILD_USER_ID}-${JOB_NAME}-${BUILD_NUMBER}"
73 currentBuild.description = HEAT_STACK_NAME
74 }
Filip Pytlounfd6726a2017-02-28 19:31:16 +010075 }
Filip Pytloun5b0954b2017-03-01 10:10:18 +010076
Filip Pytloun3d045f82017-03-01 09:44:52 +010077 //
Filip Pytlounbfce09d2017-03-01 19:00:43 +010078 // Bootstrap
Filip Pytloun3d045f82017-03-01 09:44:52 +010079 //
Filip Pytlounbfce09d2017-03-01 19:00:43 +010080
81 stage ('Download Heat templates') {
82 git.checkoutGitRepository('template', HEAT_TEMPLATE_URL, HEAT_TEMPLATE_BRANCH, HEAT_TEMPLATE_CREDENTIALS)
Filip Pytloun3d045f82017-03-01 09:44:52 +010083 }
Filip Pytloun3d045f82017-03-01 09:44:52 +010084
Filip Pytlounbfce09d2017-03-01 19:00:43 +010085 stage('Install OpenStack CLI') {
86 openstack.setupOpenstackVirtualenv(openstackEnv, openstackVersion)
87 }
Filip Pytloun64123cd2017-03-01 11:26:17 +010088
Filip Pytlounbfce09d2017-03-01 19:00:43 +010089 stage('Connect to OpenStack cloud') {
90 openstackCloud = openstack.createOpenstackEnv(OPENSTACK_API_URL, OPENSTACK_API_CREDENTIALS, OPENSTACK_API_PROJECT)
91 openstack.getKeystoneToken(openstackCloud, openstackEnv)
92 }
93
Filip Pytloun794ad952017-03-03 10:39:26 +010094 if (HEAT_STACK_REUSE.toBoolean() == false) {
Filip Pytlounbfce09d2017-03-01 19:00:43 +010095 stage('Launch new Heat stack') {
96 envParams = [
97 'instance_zone': HEAT_STACK_ZONE,
98 'public_net': HEAT_STACK_PUBLIC_NET
99 ]
100 openstack.createHeatStack(openstackCloud, HEAT_STACK_NAME, HEAT_STACK_TEMPLATE, envParams, HEAT_STACK_ENVIRONMENT, openstackEnv)
101 }
102 }
103
104 stage('Connect to Salt master') {
105 def saltMasterPort
106 try {
107 saltMasterPort = SALT_MASTER_PORT
108 } catch (MissingPropertyException e) {
109 saltMasterPort = 8000
110 }
111 saltMasterHost = openstack.getHeatStackOutputParam(openstackCloud, HEAT_STACK_NAME, 'salt_master_ip', openstackEnv)
112 saltMasterUrl = "http://${saltMasterHost}:${saltMasterPort}"
113 saltMaster = salt.connection(saltMasterUrl, SALT_MASTER_CREDENTIALS)
114 }
115
116 //
117 // Install
118 //
119
120 stage('Install core infra') {
121 // salt.master, reclass
122 // refresh_pillar
123 // sync_all
124 // linux,openssh,salt.minion.ntp
125
126 orchestrate.installFoundationInfra(saltMaster)
127 orchestrate.validateFoundationInfra(saltMaster)
128 }
129
130 stage("Deploy GlusterFS") {
131 salt.enforceState(saltMaster, 'I@glusterfs:server', 'glusterfs.server.service', true)
132 salt.enforceState(saltMaster, 'ci01*', 'glusterfs.server.setup', true)
133 sleep(5)
134 salt.enforceState(saltMaster, 'I@glusterfs:client', 'glusterfs.client', true)
135 print salt.cmdRun(saltMaster, 'I@glusterfs:client', 'mount|grep fuse.glusterfs || echo "Command failed"')
136 }
137
138 stage("Deploy GlusterFS") {
139 salt.enforceState(saltMaster, 'I@haproxy:proxy', 'haproxy,keepalived')
140 }
141
142 stage("Setup Docker Swarm") {
143 salt.enforceState(saltMaster, 'I@docker:host', 'docker.host', true)
144 salt.enforceState(saltMaster, 'I@docker:swarm:role:master', 'docker.swarm', true)
145 salt.enforceState(saltMaster, 'I@docker:swarm:role:master', 'salt', true)
146 salt.runSaltProcessStep(saltMaster, 'I@docker:swarm:role:master', 'mine.flush')
147 salt.runSaltProcessStep(saltMaster, 'I@docker:swarm:role:master', 'mine.update')
148 salt.enforceState(saltMaster, 'I@docker:swarm', 'docker.swarm', true)
149 print salt.cmdRun(saltMaster, 'I@docker:swarm:role:master', 'docker node ls')
150 }
151
152 stage("Deploy Docker services") {
153 salt.enforceState(saltMaster, 'I@docker:swarm:role:master', 'docker.client')
154
155 // XXX: Hack to fix dependency of gerrit on mysql
156 print salt.cmdRun(saltMaster, 'I@docker:swarm:role:master', "docker service rm gerrit; sleep 5; rm -rf /srv/volumes/gerrit/*")
157 waitForServices(saltMaster)
158
159 timeout(10) {
160 salt.cmdRun(saltMaster, 'I@docker:swarm:role:master', 'apt-get install -y mysql-client')
161 println "Waiting for MySQL to come up.."
162 salt.cmdRun(saltMaster, 'I@docker:swarm:role:master', 'while true; do mysql -h172.16.10.254 -ppassword -e"show status;" >/dev/null && break; done')
163 }
164 salt.enforceState(saltMaster, 'I@docker:swarm:role:master', 'docker.client')
165 // ---- cut here (end of hack) ----
166
167 waitForServices(saltMaster)
168 }
169
170 stage("Configure CI/CD services") {
171 // Aptly
172 salt.enforceState(saltMaster, 'I@aptly:server', 'aptly', true)
173
174 // Gerrit
175 timeout(10) {
176 println "Waiting for Gerrit to come up.."
177 salt.cmdRun(saltMaster, 'I@gerrit:client', 'while true; do curl -svf 172.16.10.254:8080 >/dev/null && break; done')
178 }
179 retry(2) {
180 // Needs to run twice to pass __virtual__ method of gerrit module
181 // after installation of dependencies
182 try {
183 salt.enforceState(saltMaster, 'I@gerrit:client', 'gerrit', true)
184 } catch (Exception e) {
185 print "Restarting Salt minion"
186 salt.cmdRun(saltMaster, 'I@gerrit:client', "exec 0>&-; exec 1>&-; exec 2>&-; nohup /bin/sh -c 'salt-call --local service.restart salt-minion' &")
187 sleep(5)
188 throw e
189 }
190 }
191
192 // Jenkins
193 timeout(10) {
194 println "Waiting for Jenkins to come up.."
195 salt.cmdRun(saltMaster, 'I@jenkins:client', 'while true; do curl -svf 172.16.10.254:8081 >/dev/null && break; done')
196 }
197 retry(2) {
198 // Same for jenkins
199 try {
200 salt.enforceState(saltMaster, 'I@jenkins:client', 'jenkins', true)
201 } catch (Exception e) {
202 print "Restarting Salt minion"
203 salt.cmdRun(saltMaster, 'I@jenkins:client', "exec 0>&-; exec 1>&-; exec 2>&-; nohup /bin/sh -c 'salt-call --local service.restart salt-minion' &")
204 sleep(5)
205 throw e
206 }
207 }
208 }
209
210 stage("Finalize") {
211 //
212 // Generate docs
213 //
214 try {
215 retry(3) {
Filip Pytloun27e8fa02017-03-01 20:02:46 +0100216 // TODO: fix salt.orchestrateSystem
217 // print salt.orchestrateSystem(saltMaster, ['expression': '*', 'type': 'compound'], 'sphinx.orch.generate_doc')
Filip Pytlounc17161d2017-03-03 09:50:54 +0100218 def out = salt.cmdRun(saltMaster, 'I@salt:master', 'salt-run state.orchestrate sphinx.orch.generate_doc || echo "Command execution failed"')
219 print out
220 if (out =~ /Command execution failed/) {
221 throw new Exception("Command execution failed")
222 }
Filip Pytlounbfce09d2017-03-01 19:00:43 +0100223 }
224 } catch (Throwable e) {
225 // We don't want sphinx docs to ruin whole build, so possible
226 // errors are just ignored here
227 true
228 }
229 salt.enforceState(saltMaster, 'I@nginx:server', 'nginx')
230
Filip Pytloun794ad952017-03-03 10:39:26 +0100231 print """
232 ============================================================
Filip Pytlounbfce09d2017-03-01 19:00:43 +0100233 Your CI/CD lab has been deployed and you can enjoy it:
234 Use sshuttle -r ubuntu@${saltMasterHost} 172.16.10.0/24
235 to connect to your private subnet and visit services
236 running at 172.16.10.254 (vip address):
237 9600 haproxy stats
238 8080 gerrit
239 8081 jenkins
240 8091 Docker swarm visualizer
241 8090 Reclass-generated documentation
242
Filip Pytloun27e8fa02017-03-01 20:02:46 +0100243 Don't forget to terminate your stack when you don't need it!
Filip Pytlounbfce09d2017-03-01 19:00:43 +0100244 ============================================================"""
245 }
246 } catch (Throwable e) {
247 // If there was an error or exception thrown, the build failed
248 currentBuild.result = "FAILURE"
249 throw e
250 } finally {
251 // Cleanup
Filip Pytloun794ad952017-03-03 10:39:26 +0100252 if (HEAT_STACK_DELETE.toBoolean() == true) {
Filip Pytlounbfce09d2017-03-01 19:00:43 +0100253 stage('Trigger cleanup job') {
254 build job: 'deploy_heat_cleanup', parameters: [[$class: 'StringParameterValue', name: 'HEAT_STACK_NAME', value: HEAT_STACK_NAME]]
255 }
Filip Pytlounfd6726a2017-02-28 19:31:16 +0100256 }
Filip Pytloun23741982017-02-27 17:43:00 +0100257 }
Filip Pytlounf6e877f2017-02-28 19:38:16 +0100258 }
Filip Pytloun0a07f702017-02-24 18:26:18 +0100259}