blob: 964c78e613963314329b9e75993faf111965e5d2 [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 = {
29 'classes': ['service.runtest.tempest'],
30 'parameters': {
31 '_param': {
32 'runtest_tempest_cfg_dir': TEMPEST_CFG_DIR,
33 'runtest_tempest_cfg_name': 'tempest.conf',
34 'runtest_tempest_public_net': 'net04_ext',
35 'tempest_test_target': 'gtw01*'
36 },
37 'neutron': {
38 'client': {
39 'enabled': True
40 }
41 },
42 'runtest': {
43 'enabled': True,
44 'keystonerc_node': 'ctl01*',
45 'tempest': {
46 'enabled': True,
47 'cfg_dir': '${_param:runtest_tempest_cfg_dir}',
48 'cfg_name': '${_param:runtest_tempest_cfg_name}',
49 'DEFAULT': {
50 'log_file': 'tempest.log'
51 },
52 'compute': {
53 'build_timeout': 600,
54 'max_microversion': 2.53,
55 'min_compute_nodes': 2,
56 'min_microversion': 2.1,
57 'volume_device_name': 'vdc'
58 },
59 'convert_to_uuid': {
60 'network': {
61 'public_network_id':
62 '${_param:runtest_tempest_public_net}'
63 }
64 },
65 'dns_feature_enabled': {
66 'api_admin': False,
67 'api_v1': False,
68 'api_v2': True,
69 'api_v2_quotas': True,
70 'api_v2_root_recordsets': True,
71 'bug_1573141_fixed': True
72 },
73 'heat_plugin': {
74 'floating_network_name':
75 '${_param:runtest_tempest_public_net}'
76 },
77 'network': {
78 'floating_network_name':
79 '${_param:runtest_tempest_public_net}'
80 },
81 'share': {
Dmitry Tyzhnenkoc56b77e2018-05-21 11:01:43 +030082 'capability_snapshot_support': True,
Dmitry Tyzhnenkoc56b77e2018-05-21 11:01:43 +030083 'run_driver_assisted_migration_tests': False,
Dmitry Tyzhnenkoc56b77e2018-05-21 11:01:43 +030084 'run_manage_unmanage_snapshot_tests': False,
85 'run_manage_unmanage_tests': False,
86 'run_migration_with_preserve_snapshots_tests': False,
Dmitry Tyzhnenkoc56b77e2018-05-21 11:01:43 +030087 'run_quota_tests': True,
88 'run_replication_tests': False,
Dmitry Tyzhnenkoc56b77e2018-05-21 11:01:43 +030089 'run_snapshot_tests': True,
Dmitry Tyzhnenkoc56b77e2018-05-21 11:01:43 +030090 }}}}}
91
92
93class RuntestManager(object):
94 """Helper manager for execution tempest via runtest-formula"""
95
96 image_name = settings.TEMPEST_IMAGE
97 image_version = settings.TEMPEST_IMAGE_VERSION
98 container_name = 'run-tempest-ci'
99 master_host = "cfg01"
100 master_tgt = "{}*".format(master_host)
101 class_name = "runtest"
102 run_cmd = '/bin/bash -c "run-tempest"'
103
104 def __init__(self, underlay, salt_api, cluster_name, domain_name,
105 tempest_threads, tempest_exclude_test_args,
106 tempest_pattern=settings.TEMPEST_PATTERN,
107 run_cmd=None, target='gtw01'):
108 self.underlay = underlay
109 self.__salt_api = salt_api
110 self.target = target
111 self.cluster_name = cluster_name
112 self.domain_name = domain_name
113 self.tempest_threads = tempest_threads
114 self.tempest_exclude_test_args = tempest_exclude_test_args
115 self.tempest_pattern = tempest_pattern
116 self.run_cmd = run_cmd or self.run_cmd
117
118 @property
119 def salt_api(self):
120 return self.__salt_api
121
122 def install_python_lib(self):
123 return self.salt_api.local(
124 "{}*".format(self.target),
125 'pip.install', 'docker'), None
126
Oleksii Butenko5cd0a162018-06-14 18:18:10 +0300127 def run_salt_minion_state(self):
Oleksii Butenkoe41d39f2018-06-22 17:12:41 +0300128 return self.salt_api.local('cfg01*', 'state.sls', 'salt.minion')
Dmitry Tyzhnenkoc56b77e2018-05-21 11:01:43 +0300129
130 def create_networks(self):
131 return self.salt_api.enforce_state(self.master_tgt, 'neutron.client')
132
133 def create_flavors(self):
134 return self.salt_api.enforce_state(self.master_tgt, 'nova.client')
135
136 def create_cirros(self):
137 return self.salt_api.enforce_state(self.master_tgt, 'glance.client')
138
139 def generate_config(self):
140 return self.salt_api.enforce_state(self.master_tgt, 'runtest')
141
Oleksii Butenko71d76f32018-06-05 17:46:34 +0300142 def fetch_arficats(self, username=None, file_format='xml'):
Dmitry Tyzhnenkoc56b77e2018-05-21 11:01:43 +0300143 target_name = next(node_name for node_name
144 in self.underlay.node_names() if
145 self.target in node_name)
146 with self.underlay.remote(node_name=target_name, username=None) as tgt:
Oleksii Butenko71d76f32018-06-05 17:46:34 +0300147 result = tgt.execute('find {} -name "report_*.{}"'.format(
148 TEMPEST_CFG_DIR, file_format))
149 LOG.debug("Find result {0}".format(result))
150 assert len(result['stdout']) > 0, ('No report found, please check'
151 ' if test run was successful.')
152 report = result['stdout'][0].rstrip()
153 LOG.debug("Found files {0}".format(report))
Dmitry Tyzhnenkoc56b77e2018-05-21 11:01:43 +0300154 tgt.download(
Oleksii Butenko71d76f32018-06-05 17:46:34 +0300155 destination=report, # noqa
156 target=os.getcwd())
Dmitry Tyzhnenkoc56b77e2018-05-21 11:01:43 +0300157
158 def store_runtest_model(self, config=CONFIG):
159 master_name = next(node_name for node_name
160 in self.underlay.node_names() if
161 self.master_host in node_name)
162 with self.underlay.yaml_editor(
163 file_path="/srv/salt/reclass/classes/cluster/"
164 "{cluster_name}/infra/"
165 "{class_name}.yml".format(
166 cluster_name=self.cluster_name,
167 class_name=self.class_name),
168 node_name=master_name) as editor:
169 editor.content = config
Dmitry Tyzhnenkoc56b77e2018-05-21 11:01:43 +0300170 with self.underlay.yaml_editor(
171 file_path="/srv/salt/reclass/nodes/_generated/"
172 "cfg01.{domain_name}.yml".format(
173 domain_name=self.domain_name),
174 node_name=master_name) as editor:
175 editor.content['classes'].append(
176 'cluster.{cluster_name}.infra.{class_name}'.format(
177 cluster_name=self.cluster_name,
178 class_name=self.class_name))
179
180 self.salt_api.local('*', 'saltutil.refresh_pillar')
181 self.salt_api.local('*', 'saltutil.sync_all')
182
183 def save_runtime_logs(self, logs=None, inspect=None):
184 if logs:
185 with open("{path}/{target}_tempest_run.log".format(
186 path=settings.LOGS_DIR, target=self.target), 'w') as f:
187 LOG.info("Save tempest console log")
188 container_log = logs
Oleksii Butenko71d76f32018-06-05 17:46:34 +0300189 f.write(container_log.encode('ascii', 'ignore'))
Dmitry Tyzhnenkoc56b77e2018-05-21 11:01:43 +0300190
191 if inspect:
Oleksii Butenko71d76f32018-06-05 17:46:34 +0300192 with open("{path}/{target}_tempest_container_info.json.log".format(
Dmitry Tyzhnenkoc56b77e2018-05-21 11:01:43 +0300193 path=settings.LOGS_DIR, target=self.target), 'w') as f:
Oleksii Butenko71d76f32018-06-05 17:46:34 +0300194 LOG.info("Save tempest container inspect data")
Dmitry Tyzhnenkoc56b77e2018-05-21 11:01:43 +0300195
196 container_inspect = json.dumps(inspect,
197 indent=4, sort_keys=True)
198 f.write(container_inspect)
199
200 def prepare(self):
201 self.store_runtest_model()
Oleksii Butenko5cd0a162018-06-14 18:18:10 +0300202
Dmitry Tyzhnenkoc56b77e2018-05-21 11:01:43 +0300203 res = self.install_python_lib()
204 LOG.info(json.dumps(res, indent=4))
Oleksii Butenko5cd0a162018-06-14 18:18:10 +0300205
206 res = self.run_salt_minion_state()
207 LOG.info(json.dumps(res, indent=4))
Oleksii Butenkoe41d39f2018-06-22 17:12:41 +0300208 time.sleep(10)
Oleksii Butenko5cd0a162018-06-14 18:18:10 +0300209
Dmitry Tyzhnenkoc56b77e2018-05-21 11:01:43 +0300210 res = self.create_networks()
211 LOG.info(json.dumps(res, indent=4))
Oleksii Butenko5cd0a162018-06-14 18:18:10 +0300212
Dmitry Tyzhnenkoc56b77e2018-05-21 11:01:43 +0300213 res = self.create_flavors()
214 LOG.info(json.dumps(res, indent=4))
Oleksii Butenko5cd0a162018-06-14 18:18:10 +0300215
Dmitry Tyzhnenkoc56b77e2018-05-21 11:01:43 +0300216 res = self.create_cirros()
217 LOG.info(json.dumps(res, indent=4))
Oleksii Butenko5cd0a162018-06-14 18:18:10 +0300218
Dmitry Tyzhnenkoc56b77e2018-05-21 11:01:43 +0300219 res = self.generate_config()
220 LOG.info(json.dumps(res, indent=4))
221
222 def run_tempest(self, timeout=600):
223 tgt = "{}*".format(self.target)
224 params = {
225 "name": self.container_name,
Oleksii Butenko71d76f32018-06-05 17:46:34 +0300226 "image": "{}:{}".format(self.image_name, self.image_version),
Dmitry Tyzhnenkoc56b77e2018-05-21 11:01:43 +0300227 "environment": {
228 "ARGS": "-r {tempest_pattern} -w "
229 "{tempest_threads} "
230 "{tempest_exclude_test_args}".format(
231 tempest_pattern=self.tempest_pattern,
232 tempest_threads=self.tempest_threads,
233 tempest_exclude_test_args=self.tempest_exclude_test_args) # noqa
234 },
235 "binds": [
236 "{cfg_dir}/tempest.conf:/etc/tempest/tempest.conf".format(cfg_dir=TEMPEST_CFG_DIR), # noqa
237 "/tmp/:/tmp/",
238 "{cfg_dir}:/root/tempest".format(cfg_dir=TEMPEST_CFG_DIR),
239 "/etc/ssl/certs/:/etc/ssl/certs/"
240 ],
241 "auto_remove": False,
242 "cmd": self.run_cmd
243 }
244
Oleksii Butenko71d76f32018-06-05 17:46:34 +0300245 res = self.salt_api.local(tgt, 'dockerng.pull', "{}:{}".format(
246 self.image_name, self.image_version))
Dmitry Tyzhnenkoc56b77e2018-05-21 11:01:43 +0300247 LOG.info("Tempest image has beed pulled- \n{}".format(
248 json.dumps(res, indent=4)))
249
250 res = self.salt_api.local(tgt, 'dockerng.create', kwargs=params)
251 LOG.info("Tempest container has been created - \n{}".format(
252 json.dumps(res, indent=4)))
253
254 res = self.salt_api.local(tgt, 'dockerng.start', self.container_name)
255 LOG.info("Tempest container has been started - \n{}".format(
256 json.dumps(res, indent=4)))
257
258 def wait_status(s):
259 inspect_res = self.salt_api.local(tgt,
260 'dockerng.inspect',
261 self.container_name)
262 if 'return' in inspect_res:
263 inspect = inspect_res['return']
264 inspect = inspect[0]
265 inspect = next(inspect.iteritems())[1]
266 status = inspect['State']['Status']
267
268 return status.lower() == s.lower()
269
270 return False
271
272 helpers.wait(lambda: wait_status('exited'),
273 timeout=timeout,
274 timeout_msg=('Tempest run didnt finished '
275 'in {}'.format(timeout)))
276
277 inspect_res = self.salt_api.local(tgt,
278 'dockerng.inspect',
279 self.container_name)
280 inspect = inspect_res['return'][0]
281 inspect = next(inspect.iteritems())[1]
282 if inspect['State']['ExitCode'] != 0:
283 LOG.error("Tempest running failed")
284 LOG.info("Tempest tests have been finished - \n{}".format(
285 json.dumps(res, indent=4)))
286
287 logs_res = self.salt_api.local(tgt,
288 'dockerng.logs',
289 self.container_name)
290 logs = logs_res['return'][0]
291 logs = next(logs.iteritems())[1]
Oleksii Butenko71d76f32018-06-05 17:46:34 +0300292 LOG.info("Tempest result - \n{}".format(
293 logs.encode('ascii', 'ignore')))
Dmitry Tyzhnenkoc56b77e2018-05-21 11:01:43 +0300294
295 res = self.salt_api.local(tgt, 'dockerng.rm', self.container_name)
296 LOG.info("Tempest container was removed".format(
297 json.dumps(res, indent=4)))
298
299 return {'inspect': inspect,
300 'logs': logs}
Oleksii Butenkoe82441d2018-06-12 16:01:33 +0300301
302 def prepare_and_run_tempest(self, username='root'):
303 """
304 Run tempest tests
305 """
306 tempest_timeout = settings.TEMPEST_TIMEOUT
307 self.prepare()
308 test_res = self.run_tempest(tempest_timeout)
309 self.fetch_arficats(username=username)
310 self.save_runtime_logs(**test_res)