blob: d1315b207567981d5f6522e75e8365a313d83c19 [file] [log] [blame]
Jiri Broulik2c00f4c2017-10-26 13:23:11 +02001/**
2 *
3 * Add Ceph node to existing cluster
4 *
5 * Requred parameters:
6 * SALT_MASTER_URL URL of Salt master
7 * SALT_MASTER_CREDENTIALS Credentials to the Salt API
8 * HOST Host (minion id) to be added
Tomek Jaroszykd085e512020-11-09 13:58:02 +01009 * CLUSTER_FLAGS Expected flags on the cluster during job run
10 * OSD_ONLY Add only new osds while keep rest intact
11 * USE_UPMAP Use upmap for rebalance the data after node was added
Jiri Broulik2c00f4c2017-10-26 13:23:11 +020012 *
13 */
14
Tomek Jaroszykd085e512020-11-09 13:58:02 +010015def common = new com.mirantis.mk.Common()
16def salt = new com.mirantis.mk.Salt()
17def ceph = new com.mirantis.mk.Ceph()
18def orchestrate = new com.mirantis.mk.Orchestrate()
Jiri Broulik2c00f4c2017-10-26 13:23:11 +020019def python = new com.mirantis.mk.Python()
20
21def pepperEnv = "pepperEnv"
Tomek Jaroszykd085e512020-11-09 13:58:02 +010022def flags = CLUSTER_FLAGS.tokenize(',').toSet()
23def osdOnly = OSD_ONLY.toBoolean()
24def useUpmap = USE_UPMAP.toBoolean()
25
Jakub Josefa63f9862018-01-11 17:58:38 +010026timeout(time: 12, unit: 'HOURS') {
27 node("python") {
Jiri Broulik2c00f4c2017-10-26 13:23:11 +020028
Jakub Josefa63f9862018-01-11 17:58:38 +010029 // create connection to salt master
30 python.setupPepperVirtualenv(pepperEnv, SALT_MASTER_URL, SALT_MASTER_CREDENTIALS)
Jiri Broulik2c00f4c2017-10-26 13:23:11 +020031
Tomek Jaroszykd085e512020-11-09 13:58:02 +010032 def target = salt.getMinions(pepperEnv, HOST)
33 if(target.isEmpty()) {
Mateusz Losb5663e62020-05-11 09:51:13 +020034 common.errorMsg("Host not found")
35 throw new InterruptedException()
36 }
Tomek Jaroszykd085e512020-11-09 13:58:02 +010037 else if(target.size() > 1) {
38 common.warningMsg("$HOST targeted more than one minion")
39 }
Mateusz Losb5663e62020-05-11 09:51:13 +020040
Tomek Jaroszykd085e512020-11-09 13:58:02 +010041 if(useUpmap) {
42 stage('enable upmap balancer') {
43 def features = ceph.cmdRun(pepperEnv, "ceph features --format json", false)
44 features = common.parseJSON(features)
45 for(group in features['client']) {
46 if(group instanceof java.util.HashMap$Node) { // Luminous
47 if(group.getValue()['release'] != 'luminous') {
48 throw new Exception("Some of installed clients does not support upmap. Update all clients to luminous or newer before using upmap")
49 }
50 }
51 else if(group['release'] != 'luminous') { // Nautilus
52 throw new Exception("Some of installed clients does not support upmap. Update all clients to luminous or newer before using upmap")
53 }
54 }
55 ceph.cmdRun(pepperEnv, 'ceph osd set-require-min-compat-client luminous')
56 ceph.cmdRun(pepperEnv, 'ceph balancer on')
57 ceph.cmdRun(pepperEnv, 'ceph balancer mode upmap')
58 }
59 }
Jakub Josefa63f9862018-01-11 17:58:38 +010060
Tomek Jaroszykd085e512020-11-09 13:58:02 +010061 salt.fullRefresh(pepperEnv, HOST)
62
63 stage("set flags") {
64 if(useUpmap) {
65 flags.add('norebalance')
66 }
67 ceph.setFlags(pepperEnv, flags)
68 }
69
70 try {
Jakub Josefa63f9862018-01-11 17:58:38 +010071 stage('Launch VMs') {
Tomek Jaroszykd085e512020-11-09 13:58:02 +010072 if(salt.testTarget(pepperEnv, "$HOST and not I@ceph:osd")) {
73 // launch VMs
Tomek Jaroszyk646cd722021-02-08 14:29:28 +010074 salt.enforceState([saltId: pepperEnv, target: "I@salt:control", state: 'salt.control'])
Jakub Josefa63f9862018-01-11 17:58:38 +010075
Tomek Jaroszykd085e512020-11-09 13:58:02 +010076 // wait till the HOST appears in salt-key on salt-master
77 salt.minionPresent(pepperEnv, 'I@salt:master', HOST)
78 }
79 else {
80 common.infoMsg("No VM require for a osd node.")
Jakub Josefa63f9862018-01-11 17:58:38 +010081 }
82 }
Tomek Jaroszykd085e512020-11-09 13:58:02 +010083
84 stage('Install infra') {
85 if(!osdOnly) {
86 // run basic states
87 orchestrate.installFoundationInfraOnTarget(pepperEnv, HOST)
88 }
89 else {
90 common.infoMsg('Stage skipped due to OSD_ONLY.')
91 }
92 }
93
94 stage('Install ceph components') {
95 if(salt.testTarget(pepperEnv, "$HOST and I@ceph:mon")) {
96 ceph.installMon(pepperEnv, HOST)
97 }
98 if(salt.testTarget(pepperEnv, "$HOST and I@ceph:radosgw")) {
99 ceph.installRgw(pepperEnv, HOST)
100 }
101 if(salt.testTarget(pepperEnv, "$HOST and I@ceph:osd")) {
102 ceph.installOsd(pepperEnv, HOST, !osdOnly) //skip setup while osdOnly
103 }
104 else if(osdOnly) {
105 common.infoMsg('Stage skipped due to OSD_ONLY.')
106 }
107 }
108
109 stage("Update/Install monitoring and hosts files") {
110 if(!osdOnly) {
111 ceph.updateMonitoring(pepperEnv, HOST)
112 salt.enforceState([saltId: pepperEnv, target: "I@ceph:common", state: 'linux.network.host'])
113 }
114 else {
115 common.infoMsg('Stage skipped due to OSD_ONLY.')
116 }
117 }
118
119 if(useUpmap) {
120 stage("update mappings") {
121 def mapping = []
122 def pgmap
123 for (int x = 1; x <= 3; x++) {
124 pgmap = ceph.cmdRun(pepperEnv, 'ceph pg ls remapped --format=json', false)
125 if (pgmap.trim()) {
126 pgmap = "{\"pgs\":$pgmap}" // common.parseJSON() can't parse a list of maps
127 pgmap = common.parseJSON(pgmap)['pgs']
Denis Egorenko21e1bdc2021-04-02 17:53:49 +0400128 if (!pgmap.get('pg_ready', false)) {
129 ceph.generateMapping(pgmap, mapping)
130 for(map in mapping) {
131 ceph.cmdRun(pepperEnv, map)
132 }
133 sleep(30)
Tomek Jaroszykd085e512020-11-09 13:58:02 +0100134 }
Tomek Jaroszykd085e512020-11-09 13:58:02 +0100135 }
136 }
137 }
138
139 stage('Unset norebalance') {
140 ceph.unsetFlags(pepperEnv, 'norebalance')
141 flags.removeElement('norebalance')
142 }
143 }
144 stage('Wait for healthy cluster status') {
145 ceph.waitForHealthy(pepperEnv, flags)
Jakub Josefa63f9862018-01-11 17:58:38 +0100146 }
Jiri Broulik2c00f4c2017-10-26 13:23:11 +0200147 }
Tomek Jaroszykd085e512020-11-09 13:58:02 +0100148 finally {
149 stage('Unset cluster flags') {
150 ceph.unsetFlags(pepperEnv, flags)
Ivan Berezovskiy2325dcb2019-11-05 17:42:57 +0400151 }
Ildar Svetlov9e9aa952018-04-13 23:13:07 +0400152 }
Jiri Broulik2c00f4c2017-10-26 13:23:11 +0200153 }
154}