blob: 541188173909e4b105b948385bcfa67e7c5c3b1b [file] [log] [blame]
Dmitry Tyzhnenkoc56b77e2018-05-21 11:01:43 +03001# Copyright 2018 Mirantis, Inc.
2#
3# Licensed under the Apache License, Version 2.0 (the "License"); you may
4# not use this file except in compliance with the License. You may obtain
5# a copy of the License at
6#
7# http://www.apache.org/licenses/LICENSE-2.0
8#
9# Unless required by applicable law or agreed to in writing, software
10# distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
11# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
12# License for the specific language governing permissions and limitations
13# under the License.
14
Dmitry Tyzhnenkoc56b77e2018-05-21 11:01:43 +030015import json
Oleksii Butenko71d76f32018-06-05 17:46:34 +030016import os
Oleksii Butenkoe41d39f2018-06-22 17:12:41 +030017import time
Dmitry Tyzhnenkoc56b77e2018-05-21 11:01:43 +030018
19from devops.helpers import helpers
20
21from tcp_tests import logger
22from tcp_tests import settings
23
Dmitry Tyzhnenkoc56b77e2018-05-21 11:01:43 +030024LOG = logger.logger
25
26TEMPEST_CFG_DIR = '/tmp/test'
27
28CONFIG = {
Oleksii Butenko6f691d42018-07-19 15:34:15 +030029 'classes': ['service.runtest.tempest',
30 'service.runtest.tempest.services.manila.glance'],
Dmitry Tyzhnenkoc56b77e2018-05-21 11:01:43 +030031 'parameters': {
32 '_param': {
33 'runtest_tempest_cfg_dir': TEMPEST_CFG_DIR,
34 'runtest_tempest_cfg_name': 'tempest.conf',
35 'runtest_tempest_public_net': 'net04_ext',
36 'tempest_test_target': 'gtw01*'
37 },
38 'neutron': {
39 'client': {
40 'enabled': True
41 }
42 },
43 'runtest': {
44 'enabled': True,
45 'keystonerc_node': 'ctl01*',
46 'tempest': {
47 'enabled': True,
48 'cfg_dir': '${_param:runtest_tempest_cfg_dir}',
49 'cfg_name': '${_param:runtest_tempest_cfg_name}',
50 'DEFAULT': {
51 'log_file': 'tempest.log'
52 },
53 'compute': {
54 'build_timeout': 600,
55 'max_microversion': 2.53,
56 'min_compute_nodes': 2,
57 'min_microversion': 2.1,
58 'volume_device_name': 'vdc'
59 },
60 'convert_to_uuid': {
61 'network': {
62 'public_network_id':
63 '${_param:runtest_tempest_public_net}'
64 }
65 },
66 'dns_feature_enabled': {
67 'api_admin': False,
68 'api_v1': False,
69 'api_v2': True,
70 'api_v2_quotas': True,
71 'api_v2_root_recordsets': True,
72 'bug_1573141_fixed': True
73 },
74 'heat_plugin': {
75 'floating_network_name':
76 '${_param:runtest_tempest_public_net}'
77 },
78 'network': {
79 'floating_network_name':
80 '${_param:runtest_tempest_public_net}'
81 },
82 'share': {
Dmitry Tyzhnenkoc56b77e2018-05-21 11:01:43 +030083 'capability_snapshot_support': True,
Dmitry Tyzhnenkoc56b77e2018-05-21 11:01:43 +030084 'run_driver_assisted_migration_tests': False,
Dmitry Tyzhnenkoc56b77e2018-05-21 11:01:43 +030085 'run_manage_unmanage_snapshot_tests': False,
86 'run_manage_unmanage_tests': False,
87 'run_migration_with_preserve_snapshots_tests': False,
Dmitry Tyzhnenkoc56b77e2018-05-21 11:01:43 +030088 'run_quota_tests': True,
89 'run_replication_tests': False,
Dmitry Tyzhnenkoc56b77e2018-05-21 11:01:43 +030090 'run_snapshot_tests': True,
Dmitry Tyzhnenkoc56b77e2018-05-21 11:01:43 +030091 }}}}}
92
93
94class RuntestManager(object):
95 """Helper manager for execution tempest via runtest-formula"""
96
97 image_name = settings.TEMPEST_IMAGE
98 image_version = settings.TEMPEST_IMAGE_VERSION
99 container_name = 'run-tempest-ci'
100 master_host = "cfg01"
101 master_tgt = "{}*".format(master_host)
102 class_name = "runtest"
103 run_cmd = '/bin/bash -c "run-tempest"'
104
105 def __init__(self, underlay, salt_api, cluster_name, domain_name,
106 tempest_threads, tempest_exclude_test_args,
107 tempest_pattern=settings.TEMPEST_PATTERN,
108 run_cmd=None, target='gtw01'):
109 self.underlay = underlay
110 self.__salt_api = salt_api
111 self.target = target
112 self.cluster_name = cluster_name
113 self.domain_name = domain_name
114 self.tempest_threads = tempest_threads
115 self.tempest_exclude_test_args = tempest_exclude_test_args
116 self.tempest_pattern = tempest_pattern
117 self.run_cmd = run_cmd or self.run_cmd
118
119 @property
120 def salt_api(self):
121 return self.__salt_api
122
123 def install_python_lib(self):
124 return self.salt_api.local(
125 "{}*".format(self.target),
126 'pip.install', 'docker'), None
127
Oleksii Butenko5cd0a162018-06-14 18:18:10 +0300128 def run_salt_minion_state(self):
Oleksii Butenkoe41d39f2018-06-22 17:12:41 +0300129 return self.salt_api.local('cfg01*', 'state.sls', 'salt.minion')
Dmitry Tyzhnenkoc56b77e2018-05-21 11:01:43 +0300130
131 def create_networks(self):
Oleksii Butenkoe0184a02018-07-26 13:10:03 +0300132 return self.salt_api.local('cfg01*', 'state.sls', 'neutron.client')
Dmitry Tyzhnenkoc56b77e2018-05-21 11:01:43 +0300133
134 def create_flavors(self):
Oleksii Butenkoe0184a02018-07-26 13:10:03 +0300135 return self.salt_api.local('cfg01*', 'state.sls', 'nova.client')
Dmitry Tyzhnenkoc56b77e2018-05-21 11:01:43 +0300136
137 def create_cirros(self):
Oleksii Butenkoe0184a02018-07-26 13:10:03 +0300138 return self.salt_api.local('cfg01*', 'state.sls', 'glance.client')
Dmitry Tyzhnenkoc56b77e2018-05-21 11:01:43 +0300139
140 def generate_config(self):
Oleksii Butenkoe0184a02018-07-26 13:10:03 +0300141 return self.salt_api.local('cfg01*', 'state.sls', 'runtest')
Dmitry Tyzhnenkoc56b77e2018-05-21 11:01:43 +0300142
Oleksii Butenko71d76f32018-06-05 17:46:34 +0300143 def fetch_arficats(self, username=None, file_format='xml'):
Dmitry Tyzhnenkoc56b77e2018-05-21 11:01:43 +0300144 target_name = next(node_name for node_name
145 in self.underlay.node_names() if
146 self.target in node_name)
147 with self.underlay.remote(node_name=target_name, username=None) as tgt:
Oleksii Butenko71d76f32018-06-05 17:46:34 +0300148 result = tgt.execute('find {} -name "report_*.{}"'.format(
149 TEMPEST_CFG_DIR, file_format))
150 LOG.debug("Find result {0}".format(result))
151 assert len(result['stdout']) > 0, ('No report found, please check'
152 ' if test run was successful.')
153 report = result['stdout'][0].rstrip()
154 LOG.debug("Found files {0}".format(report))
Dmitry Tyzhnenkoc56b77e2018-05-21 11:01:43 +0300155 tgt.download(
Oleksii Butenko71d76f32018-06-05 17:46:34 +0300156 destination=report, # noqa
157 target=os.getcwd())
Dmitry Tyzhnenkoc56b77e2018-05-21 11:01:43 +0300158
159 def store_runtest_model(self, config=CONFIG):
160 master_name = next(node_name for node_name
161 in self.underlay.node_names() if
162 self.master_host in node_name)
163 with self.underlay.yaml_editor(
164 file_path="/srv/salt/reclass/classes/cluster/"
165 "{cluster_name}/infra/"
166 "{class_name}.yml".format(
167 cluster_name=self.cluster_name,
168 class_name=self.class_name),
169 node_name=master_name) as editor:
170 editor.content = config
Dmitry Tyzhnenkoc56b77e2018-05-21 11:01:43 +0300171 with self.underlay.yaml_editor(
172 file_path="/srv/salt/reclass/nodes/_generated/"
173 "cfg01.{domain_name}.yml".format(
174 domain_name=self.domain_name),
175 node_name=master_name) as editor:
176 editor.content['classes'].append(
177 'cluster.{cluster_name}.infra.{class_name}'.format(
178 cluster_name=self.cluster_name,
179 class_name=self.class_name))
180
181 self.salt_api.local('*', 'saltutil.refresh_pillar')
182 self.salt_api.local('*', 'saltutil.sync_all')
183
184 def save_runtime_logs(self, logs=None, inspect=None):
185 if logs:
186 with open("{path}/{target}_tempest_run.log".format(
187 path=settings.LOGS_DIR, target=self.target), 'w') as f:
188 LOG.info("Save tempest console log")
189 container_log = logs
Oleksii Butenko71d76f32018-06-05 17:46:34 +0300190 f.write(container_log.encode('ascii', 'ignore'))
Dmitry Tyzhnenkoc56b77e2018-05-21 11:01:43 +0300191
192 if inspect:
Oleksii Butenko71d76f32018-06-05 17:46:34 +0300193 with open("{path}/{target}_tempest_container_info.json.log".format(
Dmitry Tyzhnenkoc56b77e2018-05-21 11:01:43 +0300194 path=settings.LOGS_DIR, target=self.target), 'w') as f:
Oleksii Butenko71d76f32018-06-05 17:46:34 +0300195 LOG.info("Save tempest container inspect data")
Dmitry Tyzhnenkoc56b77e2018-05-21 11:01:43 +0300196
197 container_inspect = json.dumps(inspect,
198 indent=4, sort_keys=True)
199 f.write(container_inspect)
200
201 def prepare(self):
202 self.store_runtest_model()
Oleksii Butenko5cd0a162018-06-14 18:18:10 +0300203
Dmitry Tyzhnenkoc56b77e2018-05-21 11:01:43 +0300204 res = self.install_python_lib()
205 LOG.info(json.dumps(res, indent=4))
Oleksii Butenko5cd0a162018-06-14 18:18:10 +0300206
207 res = self.run_salt_minion_state()
208 LOG.info(json.dumps(res, indent=4))
Oleksii Butenkoe0184a02018-07-26 13:10:03 +0300209 time.sleep(20)
Oleksii Butenko5cd0a162018-06-14 18:18:10 +0300210
Dmitry Tyzhnenkoc56b77e2018-05-21 11:01:43 +0300211 res = self.create_networks()
212 LOG.info(json.dumps(res, indent=4))
Oleksii Butenkoe0184a02018-07-26 13:10:03 +0300213 time.sleep(20)
Oleksii Butenko5cd0a162018-06-14 18:18:10 +0300214
Dmitry Tyzhnenkoc56b77e2018-05-21 11:01:43 +0300215 res = self.create_flavors()
216 LOG.info(json.dumps(res, indent=4))
Oleksii Butenkoe0184a02018-07-26 13:10:03 +0300217 time.sleep(20)
Oleksii Butenko5cd0a162018-06-14 18:18:10 +0300218
Dmitry Tyzhnenkoc56b77e2018-05-21 11:01:43 +0300219 res = self.create_cirros()
220 LOG.info(json.dumps(res, indent=4))
Oleksii Butenkoe0184a02018-07-26 13:10:03 +0300221 time.sleep(20)
Oleksii Butenko5cd0a162018-06-14 18:18:10 +0300222
Dmitry Tyzhnenkoc56b77e2018-05-21 11:01:43 +0300223 res = self.generate_config()
224 LOG.info(json.dumps(res, indent=4))
Oleksii Butenkoe0184a02018-07-26 13:10:03 +0300225 time.sleep(20)
Dmitry Tyzhnenkoc56b77e2018-05-21 11:01:43 +0300226
227 def run_tempest(self, timeout=600):
228 tgt = "{}*".format(self.target)
229 params = {
230 "name": self.container_name,
Oleksii Butenko71d76f32018-06-05 17:46:34 +0300231 "image": "{}:{}".format(self.image_name, self.image_version),
Dmitry Tyzhnenkoc56b77e2018-05-21 11:01:43 +0300232 "environment": {
233 "ARGS": "-r {tempest_pattern} -w "
234 "{tempest_threads} "
235 "{tempest_exclude_test_args}".format(
236 tempest_pattern=self.tempest_pattern,
237 tempest_threads=self.tempest_threads,
238 tempest_exclude_test_args=self.tempest_exclude_test_args) # noqa
239 },
240 "binds": [
241 "{cfg_dir}/tempest.conf:/etc/tempest/tempest.conf".format(cfg_dir=TEMPEST_CFG_DIR), # noqa
242 "/tmp/:/tmp/",
243 "{cfg_dir}:/root/tempest".format(cfg_dir=TEMPEST_CFG_DIR),
244 "/etc/ssl/certs/:/etc/ssl/certs/"
245 ],
246 "auto_remove": False,
247 "cmd": self.run_cmd
248 }
249
Oleksii Butenko71d76f32018-06-05 17:46:34 +0300250 res = self.salt_api.local(tgt, 'dockerng.pull', "{}:{}".format(
251 self.image_name, self.image_version))
Dmitry Tyzhnenkoc56b77e2018-05-21 11:01:43 +0300252 LOG.info("Tempest image has beed pulled- \n{}".format(
253 json.dumps(res, indent=4)))
254
255 res = self.salt_api.local(tgt, 'dockerng.create', kwargs=params)
256 LOG.info("Tempest container has been created - \n{}".format(
257 json.dumps(res, indent=4)))
258
259 res = self.salt_api.local(tgt, 'dockerng.start', self.container_name)
260 LOG.info("Tempest container has been started - \n{}".format(
261 json.dumps(res, indent=4)))
262
263 def wait_status(s):
264 inspect_res = self.salt_api.local(tgt,
265 'dockerng.inspect',
266 self.container_name)
267 if 'return' in inspect_res:
268 inspect = inspect_res['return']
269 inspect = inspect[0]
270 inspect = next(inspect.iteritems())[1]
271 status = inspect['State']['Status']
272
273 return status.lower() == s.lower()
274
275 return False
276
277 helpers.wait(lambda: wait_status('exited'),
278 timeout=timeout,
279 timeout_msg=('Tempest run didnt finished '
280 'in {}'.format(timeout)))
281
282 inspect_res = self.salt_api.local(tgt,
283 'dockerng.inspect',
284 self.container_name)
285 inspect = inspect_res['return'][0]
286 inspect = next(inspect.iteritems())[1]
287 if inspect['State']['ExitCode'] != 0:
288 LOG.error("Tempest running failed")
289 LOG.info("Tempest tests have been finished - \n{}".format(
290 json.dumps(res, indent=4)))
291
292 logs_res = self.salt_api.local(tgt,
293 'dockerng.logs',
294 self.container_name)
295 logs = logs_res['return'][0]
296 logs = next(logs.iteritems())[1]
Oleksii Butenko71d76f32018-06-05 17:46:34 +0300297 LOG.info("Tempest result - \n{}".format(
298 logs.encode('ascii', 'ignore')))
Dmitry Tyzhnenkoc56b77e2018-05-21 11:01:43 +0300299
300 res = self.salt_api.local(tgt, 'dockerng.rm', self.container_name)
301 LOG.info("Tempest container was removed".format(
302 json.dumps(res, indent=4)))
303
304 return {'inspect': inspect,
305 'logs': logs}
Oleksii Butenkoe82441d2018-06-12 16:01:33 +0300306
307 def prepare_and_run_tempest(self, username='root'):
308 """
309 Run tempest tests
310 """
311 tempest_timeout = settings.TEMPEST_TIMEOUT
312 self.prepare()
313 test_res = self.run_tempest(tempest_timeout)
314 self.fetch_arficats(username=username)
315 self.save_runtime_logs(**test_res)