blob: 68ec6e708b8fce137f95c8fe4d21f609b5134129 [file] [log] [blame]
Andrea Frittolif5da28b2013-12-06 07:08:07 +00001# Copyright 2013 Hewlett-Packard, Ltd.
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
Matthew Treinishbeae8602014-04-08 17:53:06 -040015
16import json
17import re
18import string
19import unicodedata
20
Matthew Treinisha0048cb2014-04-08 17:44:42 -040021import testscenarios
22import testtools
23
Andrea Frittolif9cde7e2014-02-18 09:57:04 +000024from tempest import clients
Andrea Frittoli9efbe952015-01-29 12:43:09 +000025from tempest.common import cred_provider
Andrea Frittolif5da28b2013-12-06 07:08:07 +000026from tempest.common.utils import misc
27from tempest import config
Matthew Treinisha0f820f2014-09-16 11:25:34 -040028from tempest import exceptions
Andrea Frittolif5da28b2013-12-06 07:08:07 +000029
Andrea Frittolif5da28b2013-12-06 07:08:07 +000030CONF = config.CONF
31
32
33@misc.singleton
34class ImageUtils(object):
35
36 default_ssh_user = 'root'
37
38 def __init__(self):
39 # Load configuration items
40 self.ssh_users = json.loads(CONF.input_scenario.ssh_user_regex)
41 self.non_ssh_image_pattern = \
42 CONF.input_scenario.non_ssh_image_regex
43 # Setup clients
Andrea Frittolife1e2432014-09-25 10:39:37 +010044 os = clients.Manager()
45 self.images_client = os.images_client
46 self.flavors_client = os.flavors_client
Andrea Frittolif5da28b2013-12-06 07:08:07 +000047
48 def ssh_user(self, image_id):
David Kranza5299eb2015-01-15 17:24:05 -050049 _image = self.images_client.get_image(image_id)
Andrea Frittolif5da28b2013-12-06 07:08:07 +000050 for regex, user in self.ssh_users:
51 # First match wins
Andrea Frittolife1e2432014-09-25 10:39:37 +010052 if re.match(regex, _image['name']) is not None:
Andrea Frittolif5da28b2013-12-06 07:08:07 +000053 return user
54 else:
55 return self.default_ssh_user
56
57 def _is_sshable_image(self, image):
58 return not re.search(pattern=self.non_ssh_image_pattern,
Andrea Frittolife1e2432014-09-25 10:39:37 +010059 string=str(image['name']))
Andrea Frittolif5da28b2013-12-06 07:08:07 +000060
61 def is_sshable_image(self, image_id):
David Kranza5299eb2015-01-15 17:24:05 -050062 _image = self.images_client.get_image(image_id)
Andrea Frittolif5da28b2013-12-06 07:08:07 +000063 return self._is_sshable_image(_image)
64
65 def _is_flavor_enough(self, flavor, image):
Andrea Frittolife1e2432014-09-25 10:39:37 +010066 return image['minDisk'] <= flavor['disk']
Andrea Frittolif5da28b2013-12-06 07:08:07 +000067
68 def is_flavor_enough(self, flavor_id, image_id):
David Kranza5299eb2015-01-15 17:24:05 -050069 _image = self.images_client.get_image(image_id)
David Kranz2fa77b22015-02-09 11:39:50 -050070 _flavor = self.flavors_client.get_flavor_details(flavor_id)
Andrea Frittolif5da28b2013-12-06 07:08:07 +000071 return self._is_flavor_enough(_flavor, _image)
72
73
74@misc.singleton
75class InputScenarioUtils(object):
76
77 """
78 Example usage:
79
80 import testscenarios
81 (...)
82 load_tests = testscenarios.load_tests_apply_scenarios
83
84
Andrea Frittolife1e2432014-09-25 10:39:37 +010085 class TestInputScenario(manager.ScenarioTest):
Andrea Frittolif5da28b2013-12-06 07:08:07 +000086
Matthew Treinishd75edef2014-04-11 15:57:16 -040087 scenario_utils = utils.InputScenarioUtils()
Andrea Frittolif5da28b2013-12-06 07:08:07 +000088 scenario_flavor = scenario_utils.scenario_flavors
89 scenario_image = scenario_utils.scenario_images
90 scenarios = testscenarios.multiply_scenarios(scenario_image,
91 scenario_flavor)
92
93 def test_create_server_metadata(self):
94 name = rand_name('instance')
Andrea Frittolife1e2432014-09-25 10:39:37 +010095 self.servers_client.create_server(name=name,
96 flavor_ref=self.flavor_ref,
97 image_ref=self.image_ref)
Andrea Frittolif5da28b2013-12-06 07:08:07 +000098 """
99 validchars = "-_.{ascii}{digit}".format(ascii=string.ascii_letters,
100 digit=string.digits)
101
102 def __init__(self):
Andrea Frittolife1e2432014-09-25 10:39:37 +0100103 os = clients.Manager(
Andrea Frittoli9efbe952015-01-29 12:43:09 +0000104 cred_provider.get_configured_credentials('user', fill_in=False))
Andrea Frittolife1e2432014-09-25 10:39:37 +0100105 self.images_client = os.images_client
106 self.flavors_client = os.flavors_client
Andrea Frittolif5da28b2013-12-06 07:08:07 +0000107 self.image_pattern = CONF.input_scenario.image_regex
108 self.flavor_pattern = CONF.input_scenario.flavor_regex
109
110 def _normalize_name(self, name):
111 nname = unicodedata.normalize('NFKD', name).encode('ASCII', 'ignore')
112 nname = ''.join(c for c in nname if c in self.validchars)
113 return nname
114
115 @property
116 def scenario_images(self):
117 """
118 :return: a scenario with name and uuid of images
119 """
Andrea Frittolie9674c32014-02-07 20:01:46 +0000120 if not CONF.service_available.glance:
121 return []
Andrea Frittolif5da28b2013-12-06 07:08:07 +0000122 if not hasattr(self, '_scenario_images'):
Matthew Treinish67e570c2015-02-18 16:59:39 +0000123 try:
124 images = self.images_client.list_images()
125 self._scenario_images = [
126 (self._normalize_name(i['name']), dict(image_ref=i['id']))
127 for i in images if re.search(self.image_pattern,
128 str(i['name']))
129 ]
130 except Exception:
131 self._scenario_images = []
Andrea Frittolif5da28b2013-12-06 07:08:07 +0000132 return self._scenario_images
133
134 @property
135 def scenario_flavors(self):
136 """
137 :return: a scenario with name and uuid of flavors
138 """
139 if not hasattr(self, '_scenario_flavors'):
Matthew Treinish67e570c2015-02-18 16:59:39 +0000140 try:
141 flavors = self.flavors_client.list_flavors()
142 self._scenario_flavors = [
143 (self._normalize_name(f['name']), dict(flavor_ref=f['id']))
144 for f in flavors if re.search(self.flavor_pattern,
145 str(f['name']))
146 ]
147 except Exception:
148 self._scenario_flavors = []
Andrea Frittolif5da28b2013-12-06 07:08:07 +0000149 return self._scenario_flavors
Matthew Treinisha0048cb2014-04-08 17:44:42 -0400150
151
152def load_tests_input_scenario_utils(*args):
153 """
154 Wrapper for testscenarios to set the scenarios to avoid running a getattr
155 on the CONF object at import.
156 """
157 if getattr(args[0], 'suiteClass', None) is not None:
158 loader, standard_tests, pattern = args
159 else:
160 standard_tests, module, loader = args
Matthew Treinisha0f820f2014-09-16 11:25:34 -0400161 try:
162 scenario_utils = InputScenarioUtils()
163 scenario_flavor = scenario_utils.scenario_flavors
164 scenario_image = scenario_utils.scenario_images
165 except exceptions.InvalidConfiguration:
166 return standard_tests
Matthew Treinisha0048cb2014-04-08 17:44:42 -0400167 for test in testtools.iterate_tests(standard_tests):
168 setattr(test, 'scenarios', testscenarios.multiply_scenarios(
169 scenario_image,
170 scenario_flavor))
171 return testscenarios.load_tests_apply_scenarios(*args)