blob: cbcbd582f543e50a29e082e51a0310f179cdcbde [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
34node {
Filip Pytlounfd6726a2017-02-28 19:31:16 +010035 try {
36 // connection objects
37 def openstackCloud
38 def saltMaster
Filip Pytloun0a07f702017-02-24 18:26:18 +010039
Filip Pytlounfd6726a2017-02-28 19:31:16 +010040 // value defaults
41 def openstackVersion = OPENSTACK_API_CLIENT ? OPENSTACK_API_CLIENT : 'liberty'
42 def openstackEnv = "${env.WORKSPACE}/venv"
Filip Pytloun0a07f702017-02-24 18:26:18 +010043
Filip Pytlounfd6726a2017-02-28 19:31:16 +010044 if (HEAT_STACK_NAME == '') {
45 HEAT_STACK_NAME = BUILD_TAG
Filip Pytloun0a07f702017-02-24 18:26:18 +010046 }
Filip Pytloun0a07f702017-02-24 18:26:18 +010047
Filip Pytlounfd6726a2017-02-28 19:31:16 +010048 //
49 // Bootstrap
50 //
51
52 stage ('Download Heat templates') {
53 git.checkoutGitRepository('template', HEAT_TEMPLATE_URL, HEAT_TEMPLATE_BRANCH, HEAT_TEMPLATE_CREDENTIALS)
Filip Pytloune32fda82017-02-24 18:26:18 +010054 }
Filip Pytloun0a07f702017-02-24 18:26:18 +010055
Filip Pytlounfd6726a2017-02-28 19:31:16 +010056 stage('Install OpenStack CLI') {
57 openstack.setupOpenstackVirtualenv(openstackEnv, openstackVersion)
58 }
Filip Pytloun0a07f702017-02-24 18:26:18 +010059
Filip Pytlounfd6726a2017-02-28 19:31:16 +010060 stage('Connect to OpenStack cloud') {
61 openstackCloud = openstack.createOpenstackEnv(OPENSTACK_API_URL, OPENSTACK_API_CREDENTIALS, OPENSTACK_API_PROJECT)
62 openstack.getKeystoneToken(openstackCloud, openstackEnv)
63 }
Filip Pytloun0a07f702017-02-24 18:26:18 +010064
Filip Pytlounfd6726a2017-02-28 19:31:16 +010065 if (HEAT_STACK_REUSE == 'false') {
66 stage('Launch new Heat stack') {
67 envParams = [
68 'instance_zone': HEAT_STACK_ZONE,
69 'public_net': HEAT_STACK_PUBLIC_NET
70 ]
71 openstack.createHeatStack(openstackCloud, HEAT_STACK_NAME, HEAT_STACK_TEMPLATE, envParams, HEAT_STACK_ENVIRONMENT, openstackEnv)
72 }
73 }
Filip Pytloun0a07f702017-02-24 18:26:18 +010074
Filip Pytlounfd6726a2017-02-28 19:31:16 +010075 stage('Connect to Salt master') {
76 def saltMasterPort
77 try {
78 saltMasterPort = SALT_MASTER_PORT
79 } catch (MissingPropertyException e) {
80 saltMasterPort = 8000
81 }
82 saltMasterHost = openstack.getHeatStackOutputParam(openstackCloud, HEAT_STACK_NAME, 'salt_master_ip', openstackEnv)
83 saltMasterUrl = "http://${saltMasterHost}:${saltMasterPort}"
84 saltMaster = salt.connection(saltMasterUrl, SALT_MASTER_CREDENTIALS)
85 }
Filip Pytloun0a07f702017-02-24 18:26:18 +010086
Filip Pytlounfd6726a2017-02-28 19:31:16 +010087 //
88 // Install
89 //
Filip Pytloun0a07f702017-02-24 18:26:18 +010090
Filip Pytlounfd6726a2017-02-28 19:31:16 +010091 stage('Install core infra') {
92 // salt.master, reclass
93 // refresh_pillar
94 // sync_all
95 // linux,openssh,salt.minion.ntp
Filip Pytloun0a07f702017-02-24 18:26:18 +010096
Filip Pytlounfd6726a2017-02-28 19:31:16 +010097 orchestrate.installFoundationInfra(saltMaster)
98 orchestrate.validateFoundationInfra(saltMaster)
99 }
Filip Pytloun23741982017-02-27 17:43:00 +0100100
Filip Pytlounfd6726a2017-02-28 19:31:16 +0100101 stage("Deploy GlusterFS") {
102 salt.enforceState(saltMaster, 'I@glusterfs:server', 'glusterfs.server.service', true)
103 salt.enforceState(saltMaster, 'ci01*', 'glusterfs.server.setup', true)
104 sleep(5)
105 salt.enforceState(saltMaster, 'I@glusterfs:client', 'glusterfs.client', true)
106 print salt.cmdRun(saltMaster, 'I@glusterfs:client', 'mount|grep fuse.glusterfs || echo "Command failed"')
107 }
Filip Pytloun0a07f702017-02-24 18:26:18 +0100108
Filip Pytlounfd6726a2017-02-28 19:31:16 +0100109 stage("Deploy GlusterFS") {
110 salt.enforceState(saltMaster, 'I@haproxy:proxy', 'haproxy,keepalived')
111 }
112
113 stage("Setup Docker Swarm") {
114 salt.enforceState(saltMaster, 'I@docker:host', 'docker.host', true)
115 salt.enforceState(saltMaster, 'I@docker:swarm:role:master', 'docker.swarm', true)
116 salt.enforceState(saltMaster, 'I@docker:swarm:role:master', 'salt', true)
117 salt.runSaltProcessStep(saltMaster, 'I@docker:swarm:role:master', 'mine.flush')
118 salt.runSaltProcessStep(saltMaster, 'I@docker:swarm:role:master', 'mine.update')
119 salt.enforceState(saltMaster, 'I@docker:swarm', 'docker.swarm', true)
120 print salt.cmdRun(saltMaster, 'I@docker:swarm:role:master', 'docker node ls')
121 }
122
123 stage("Deploy Docker services") {
124 salt.enforceState(saltMaster, 'I@docker:swarm:role:master', 'docker.client')
125
126 // XXX: Hack to fix dependency of gerrit on mysql
127 print salt.cmdRun(saltMaster, 'I@docker:swarm:role:master', "docker service rm gerrit; sleep 5; rm -rf /srv/volumes/gerrit/*")
128 sleep(10)
Filip Pytlounc6d3bbf2017-03-01 08:48:02 +0100129 print salt.cmdRun(saltMaster, 'I@docker:swarm:role:master', "apt-get install -y mysql-client; mysql -ppassword -h172.16.10.11 -P13306 -e'drop database gerrit;create database gerrit;'")
Filip Pytlounfd6726a2017-02-28 19:31:16 +0100130 salt.enforceState(saltMaster, 'I@docker:swarm:role:master', 'docker.client')
131 // ---- cut here (end of hack) ----
132
133 retry(30) {
134 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'""")
135 for (int a = 0; a < out['return'].size(); a++) {
136 def entry = out['return'].get(a)
137 for (int i = 0; i < entry.size(); i++) {
138 def node = entry.get(i)
139 if (node) {
140 if (node.value =~ /Some services are not running/) {
141 sleep(10)
142 throw new Exception("$node.key: $node.value")
143 } else {
144 print out
145 }
Filip Pytloun7e2a8af2017-02-28 15:36:29 +0100146 }
Filip Pytloun2a9d78d2017-02-27 19:53:21 +0100147 }
148 }
149 }
Filip Pytloune8657fc2017-02-28 21:46:13 +0100150 // Give services some time to settle
Filip Pytloundd3d8e22017-03-01 10:22:44 +0100151 sleep(30)
Filip Pytloun0a07f702017-02-24 18:26:18 +0100152 }
Filip Pytloun0a07f702017-02-24 18:26:18 +0100153
Filip Pytlounfd6726a2017-02-28 19:31:16 +0100154 stage("Configure CI/CD services") {
Filip Pytloun5b0954b2017-03-01 10:10:18 +0100155 // Aptly
Filip Pytloundd3d8e22017-03-01 10:22:44 +0100156 salt.enforceState(saltMaster, 'I@aptly:server', 'aptly', true)
Filip Pytloun5b0954b2017-03-01 10:10:18 +0100157
158 // Gerrit
Filip Pytloundd3d8e22017-03-01 10:22:44 +0100159 timeout(10) {
Filip Pytloun5b0954b2017-03-01 10:10:18 +0100160 println "Waiting for Gerrit to come up.."
Filip Pytloundd3d8e22017-03-01 10:22:44 +0100161 salt.cmdRun(saltMaster, 'I@gerrit:client', 'while true; do curl -svf 172.16.10.254:8080 >/dev/null && break; done')
Filip Pytloun5b0954b2017-03-01 10:10:18 +0100162 }
Filip Pytlounfd6726a2017-02-28 19:31:16 +0100163 retry(2) {
164 // Needs to run twice to pass __virtual__ method of gerrit module
165 // after installation of dependencies
Filip Pytloundd3d8e22017-03-01 10:22:44 +0100166 try {
167 salt.enforceState(saltMaster, 'I@gerrit:client', 'gerrit', true)
168 } catch (Exception e) {
Filip Pytloun22cc2382017-03-01 12:06:04 +0100169 print "Restarting Salt minion"
170 salt.cmdRun(saltMaster, 'I@gerrit:client', "exec 0>&-; exec 1>&-; exec 2>&-; nohup /bin/sh -c 'salt-call --local service.restart salt-minion' &")
Filip Pytloun199a2fb2017-03-01 12:47:47 +0100171 sleep(5)
Filip Pytloundd3d8e22017-03-01 10:22:44 +0100172 throw e
173 }
Filip Pytlounfd6726a2017-02-28 19:31:16 +0100174 }
Filip Pytloun5b0954b2017-03-01 10:10:18 +0100175
176 // Jenkins
Filip Pytloundd3d8e22017-03-01 10:22:44 +0100177 timeout(10) {
Filip Pytloun5b0954b2017-03-01 10:10:18 +0100178 println "Waiting for Jenkins to come up.."
Filip Pytloundd3d8e22017-03-01 10:22:44 +0100179 salt.cmdRun(saltMaster, 'I@jenkins:client', 'while true; do curl -svf 172.16.10.254:8081 >/dev/null && break; done')
Filip Pytloun5b0954b2017-03-01 10:10:18 +0100180 }
Filip Pytlounfd6726a2017-02-28 19:31:16 +0100181 retry(2) {
182 // Same for jenkins
Filip Pytloundd3d8e22017-03-01 10:22:44 +0100183 try {
184 salt.enforceState(saltMaster, 'I@jenkins:client', 'jenkins', true)
185 } catch (Exception e) {
Filip Pytloun22cc2382017-03-01 12:06:04 +0100186 print "Restarting Salt minion"
187 salt.cmdRun(saltMaster, 'I@jenkins:client', "exec 0>&-; exec 1>&-; exec 2>&-; nohup /bin/sh -c 'salt-call --local service.restart salt-minion' &")
Filip Pytloun199a2fb2017-03-01 12:47:47 +0100188 sleep(5)
Filip Pytloundd3d8e22017-03-01 10:22:44 +0100189 throw e
190 }
Filip Pytlounfd6726a2017-02-28 19:31:16 +0100191 }
Filip Pytloun0a07f702017-02-24 18:26:18 +0100192 }
Filip Pytloun3d045f82017-03-01 09:44:52 +0100193
194 stage("Finalize") {
195 //
196 // Generate docs
197 //
198 try {
199 retry(3) {
Filip Pytlounf2961622017-03-01 12:13:30 +0100200 print salt.orchestrateSystem(saltMaster, ['expression': '*', 'type': 'compound'], 'sphinx.orch.generate_doc')
Filip Pytloun3d045f82017-03-01 09:44:52 +0100201 }
202 } catch (Throwable e) {
203 // We don't want sphinx docs to ruin whole build, so possible
204 // errors are just ignored here
205 true
206 }
207 salt.enforceState(saltMaster, 'I@nginx:server', 'nginx')
208
Filip Pytlounf2961622017-03-01 12:13:30 +0100209 print """============================================================
Filip Pytloun64123cd2017-03-01 11:26:17 +0100210Your CI/CD lab has been deployed and you can enjoy it:
Filip Pytlounf2961622017-03-01 12:13:30 +0100211Use sshuttle -r ubuntu@${saltMasterHost} 172.16.10.0/24
212to connect to your private subnet and visit services
213running at 172.16.10.254 (vip address):
214 9600 haproxy stats
215 8080 gerrit
216 8081 jenkins
217 8091 Docker swarm visualizer
218 8090 Reclass-generated documentation
Filip Pytloun64123cd2017-03-01 11:26:17 +0100219
220Don't forget to terminate your stack when you don't needed!
221============================================================"""
Filip Pytloun3d045f82017-03-01 09:44:52 +0100222 }
Filip Pytlounfd6726a2017-02-28 19:31:16 +0100223 } catch (Throwable e) {
224 // If there was an error or exception thrown, the build failed
225 currentBuild.result = "FAILURE"
226 throw e
227 } finally {
228 // Cleanup
229 if (HEAT_STACK_DELETE == 'true') {
230 stage('Trigger cleanup job') {
231 build job: 'deploy_heat_cleanup', parameters: [[$class: 'StringParameterValue', name: 'HEAT_STACK_NAME', value: HEAT_STACK_NAME]]
232 }
Filip Pytloun23741982017-02-27 17:43:00 +0100233 }
Filip Pytlounf6e877f2017-02-28 19:38:16 +0100234 }
Filip Pytloun0a07f702017-02-24 18:26:18 +0100235}