blob: 1f8d601c72fbde3e8e844fa38cf0d8c197d157b9 [file] [log] [blame]
Oleksandr Bryndzii13bb1362019-06-14 14:46:39 +00001/**
2 * Upgrade MySQL and Galera packages on dbs nodes.
3 * Update packages on given nodes
4 *
5 * Expected parameters:
6 * SALT_MASTER_CREDENTIALS Credentials to the Salt API.
7 * SALT_MASTER_URL Full Salt API address [http://10.10.10.15:6969].
8 * SHUTDOWN_CLUSTER Shutdown all mysql instances on target nodes at the same time.
9 * OS_DIST_UPGRADE Upgrade system packages including kernel (apt-get dist-upgrade).
10 * OS_UPGRADE Upgrade all installed applications (apt-get upgrade)
11 * TARGET_SERVERS Comma separated list of salt compound definitions to upgrade.
12 * INTERACTIVE Ask interactive questions during pipeline run (bool).
13 *
14**/
15
16def common = new com.mirantis.mk.Common()
17def salt = new com.mirantis.mk.Salt()
18def python = new com.mirantis.mk.Python()
19def debian = new com.mirantis.mk.Debian()
20def openstack = new com.mirantis.mk.Openstack()
21def galera = new com.mirantis.mk.Galera()
22def shutdownCluster = SHUTDOWN_CLUSTER.toBoolean()
23def interactive = INTERACTIVE.toBoolean()
24def LinkedHashMap upgradeStageMap = [:]
25
26upgradeStageMap.put('Pre upgrade',
27 [
28 'Description': 'Only non destructive actions will be applied during this phase. Basic service verification will be performed.',
29 'Status': 'NOT_LAUNCHED',
30 'Expected behaviors': '''
31 * No service downtime
32 * No workload downtime''',
33 'Launched actions': '''
34 * Verify API, perform basic CRUD operations for services.
35 * Verify MySQL is running and Galera cluster is operational.''',
36 'State result': 'Basic checks around wsrep Galera status are passed.'
37 ])
38
39upgradeStageMap.put('Stop MySQL service',
40 [
41 'Description': 'All MySQL services will be stopped on All TARGET_SERVERS nodes.',
42 'Status': 'NOT_LAUNCHED',
43 'Expected behaviors': '''
44 * MySQL services are stopped.
45 * OpenStack APIs are not accessible from this point.
46 * No workload downtime''',
47 'Launched actions': '''
48 * Stop MySQL services''',
49 'State result': 'MySQL service is stopped',
50 ])
51
52upgradeStageMap.put('Upgrade OS',
53 [
54 'Description': 'Optional step. OS packages will be upgraded during this phase, depending on the job parameters dist-upgrade might be called. And reboot of node executed.',
55 'Status': 'NOT_LAUNCHED',
56 'Expected behaviors': '''
57 * No workload downtime
58 * The nodes might be rebooted''',
59 'Launched actions': '''
60 * Install new version of system packages
61 * If doing dist-upgrade new kernel might be installed and node rebooted
62 * System packages are updated
63 * Node might be rebooted
64'''
65 ])
66
67upgradeStageMap.put('Upgrade MySQL server',
68 [
69 'Description': 'MySQL and Erlang code will be upgraded during this stage. No workload downtime is expected.',
70 'Status': 'NOT_LAUNCHED',
71 'Expected behaviors': '''
72 * OpenStack services loose connection to MySQL server
73 * No workload downtime''',
74 'Launched actions': '''
75 * Install new version of MySQL and Galera packages
76 * Render version of configs''',
77 'State result': '''
78 * MySQL packages are upgraded''',
79 ])
80
81upgradeStageMap.put('Start MySQL service',
82 [
83 'Description': 'All MySQL services will be running on All TARGET_SERVERS nodes.',
84 'Status': 'NOT_LAUNCHED',
85 'Expected behaviors': '''
86 * MySQL service is running.
87 * OpenStack API are accessible from this point.
88 * No workload downtime''',
89 'Launched actions': '''
90 * Start MySQL service''',
91 'State result': 'MySQL service is running',
92 ])
93
94def env = "env"
95timeout(time: 12, unit: 'HOURS') {
96 node() {
97
98 stage('Setup virtualenv for Pepper') {
99 python.setupPepperVirtualenv(env, SALT_MASTER_URL, SALT_MASTER_CREDENTIALS)
100 }
101
102 def upgradeTargets = salt.getMinionsSorted(env, TARGET_SERVERS)
103
104 if (upgradeTargets.isEmpty()) {
105 error("No servers for upgrade matched by ${TARGET_SERVERS}")
106 }
107
108 def targetSecMapping = [:]
109 def secNoList = []
110 def out
111 def stopTargets = upgradeTargets.reverse()
112 common.printStageMap(upgradeStageMap)
113
114 if (interactive){
115 input message: common.getColorizedString(
116 "Above you can find detailed info this pipeline will execute.\nThe info provides brief description of each stage, actions that will be performed and service/workload impact during each stage.\nPlease read it carefully.", "yellow")
117 }
118
119 for (target in upgradeTargets) {
120 salt.runSaltProcessStep(env, target, 'saltutil.refresh_pillar', [], null, true)
121 salt.enforceState(env, target, ['linux.system.repo'])
122 common.stageWrapper(upgradeStageMap, "Pre upgrade", target, interactive) {
123 openstack.runOpenStackUpgradePhase(env, target, 'pre')
124 openstack.runOpenStackUpgradePhase(env, target, 'verify')
125 }
126 }
127
128 if (shutdownCluster){
129 for (target in stopTargets) {
130 common.stageWrapper(upgradeStageMap, "Stop MySQL service", target, interactive) {
131 openstack.runOpenStackUpgradePhase(env, target, 'service_stopped')
132 }
133 }
134 }
135
Roman Lubianyi34603b32022-01-26 15:48:36 +0200136 def masterNode = salt.getMinionsSorted(env, galera.getGaleraLastShutdownNode(env))[0]
Oleksandr Bryndzii13bb1362019-06-14 14:46:39 +0000137 common.infoMsg("Master node is: ${masterNode}")
138
139 // Make sure we start upgrade always from master node
140 upgradeTargets.remove(masterNode)
141 upgradeTargets = [masterNode] + upgradeTargets
142 common.infoMsg("Upgrade targets are: ${upgradeTargets}")
143
144 for (target in upgradeTargets) {
145
146 common.stageWrapper(upgradeStageMap, "Stop MySQL service", target, interactive) {
147 openstack.runOpenStackUpgradePhase(env, target, 'service_stopped')
148 }
149
150 common.stageWrapper(upgradeStageMap, "Upgrade OS", target, interactive) {
151 if (OS_DIST_UPGRADE.toBoolean() == true){
152 upgrade_mode = 'dist-upgrade'
153 } else if (OS_UPGRADE.toBoolean() == true){
154 upgrade_mode = 'upgrade'
155 }
156 if (OS_DIST_UPGRADE.toBoolean() == true || OS_UPGRADE.toBoolean() == true) {
157 debian.osUpgradeNode(env, target, upgrade_mode, false)
158 }
159 }
160
161 common.stageWrapper(upgradeStageMap, "Upgrade MySQL server", target, interactive) {
162 openstack.runOpenStackUpgradePhase(env, target, 'pkgs_latest')
163 openstack.runOpenStackUpgradePhase(env, target, 'render_config')
164 }
165
166 if (shutdownCluster && target == masterNode){
167 //Start first node.
168 common.stageWrapper(upgradeStageMap, "Start MySQL service", target, interactive) {
169 galera.startFirstNode(env, target)
170 }
171 }
172
173 common.stageWrapper(upgradeStageMap, "Start MySQL service", target, interactive) {
174 openstack.runOpenStackUpgradePhase(env, target, 'service_running')
175 openstack.runOpenStackUpgradePhase(env, target, 'verify')
176 }
177 }
178
179 // restart first node by applying state.
180
181 if (shutdownCluster) {
182 openstack.runOpenStackUpgradePhase(env, masterNode, 'render_config')
Roman Lubianyi10d70fd2022-01-26 16:00:01 +0200183 salt.cmdRun(env, masterNode, "systemctl restart mysql")
Oleksandr Bryndzii13bb1362019-06-14 14:46:39 +0000184 openstack.runOpenStackUpgradePhase(env, masterNode, 'verify')
185 }
186
187 for (target in upgradeTargets) {
188 ensureClusterState = galera.getWsrepParameters(env, target, 'wsrep_evs_state')
189 if (ensureClusterState['wsrep_evs_state'] == 'OPERATIONAL') {
190 common.infoMsg('Node is in OPERATIONAL state.')
191 } else {
192 throw new Exception("Node is NOT in OPERATIONAL state.")
193 }
194 }
195 }
196}