ZhiQiang Fan | 39f9722 | 2013-09-20 04:49:44 +0800 | [diff] [blame] | 1 | # Copyright 2012 OpenStack Foundation |
dwalleck | e62b9f0 | 2012-10-10 23:34:42 -0500 | [diff] [blame] | 2 | # All Rights Reserved. |
| 3 | # |
| 4 | # Licensed under the Apache License, Version 2.0 (the "License"); you may |
| 5 | # not use this file except in compliance with the License. You may obtain |
| 6 | # a copy of the License at |
| 7 | # |
| 8 | # http://www.apache.org/licenses/LICENSE-2.0 |
| 9 | # |
| 10 | # Unless required by applicable law or agreed to in writing, software |
| 11 | # distributed under the License is distributed on an "AS IS" BASIS, WITHOUT |
| 12 | # WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the |
| 13 | # License for the specific language governing permissions and limitations |
| 14 | # under the License. |
| 15 | |
Tiago Mello | eda03b5 | 2012-08-22 23:47:29 -0300 | [diff] [blame] | 16 | import urllib |
| 17 | |
| 18 | from lxml import etree |
| 19 | |
vponomaryov | 960eeb4 | 2014-02-22 18:25:25 +0200 | [diff] [blame] | 20 | from tempest.common import rest_client |
Matthew Treinish | 28f164c | 2014-03-04 18:55:06 +0000 | [diff] [blame] | 21 | from tempest.common import xml_utils |
Matthew Treinish | 684d899 | 2014-01-30 16:27:40 +0000 | [diff] [blame] | 22 | from tempest import config |
Tiago Mello | eda03b5 | 2012-08-22 23:47:29 -0300 | [diff] [blame] | 23 | |
Matthew Treinish | 684d899 | 2014-01-30 16:27:40 +0000 | [diff] [blame] | 24 | CONF = config.CONF |
Tiago Mello | eda03b5 | 2012-08-22 23:47:29 -0300 | [diff] [blame] | 25 | |
| 26 | XMLNS_OS_FLV_EXT_DATA = \ |
rajalakshmi-ganesan | f344cc3 | 2012-12-31 20:02:27 +0530 | [diff] [blame] | 27 | "http://docs.openstack.org/compute/ext/flavor_extra_data/api/v1.1" |
| 28 | XMLNS_OS_FLV_ACCESS = \ |
ivan-zhu | ae7c7c5 | 2013-10-21 22:13:22 +0800 | [diff] [blame] | 29 | "http://docs.openstack.org/compute/ext/flavor_access/api/v2" |
Tiago Mello | eda03b5 | 2012-08-22 23:47:29 -0300 | [diff] [blame] | 30 | |
| 31 | |
vponomaryov | 960eeb4 | 2014-02-22 18:25:25 +0200 | [diff] [blame] | 32 | class FlavorsClientXML(rest_client.RestClient): |
| 33 | TYPE = "xml" |
Tiago Mello | eda03b5 | 2012-08-22 23:47:29 -0300 | [diff] [blame] | 34 | |
Andrea Frittoli | 8bbdb16 | 2014-01-06 11:06:13 +0000 | [diff] [blame] | 35 | def __init__(self, auth_provider): |
| 36 | super(FlavorsClientXML, self).__init__(auth_provider) |
Matthew Treinish | 684d899 | 2014-01-30 16:27:40 +0000 | [diff] [blame] | 37 | self.service = CONF.compute.catalog_type |
Tiago Mello | eda03b5 | 2012-08-22 23:47:29 -0300 | [diff] [blame] | 38 | |
| 39 | def _format_flavor(self, f): |
| 40 | flavor = {'links': []} |
| 41 | for k, v in f.items(): |
Ken'ichi Ohmichi | 3577260 | 2013-11-14 15:03:27 +0900 | [diff] [blame] | 42 | if k == 'id': |
| 43 | flavor['id'] = v |
| 44 | continue |
| 45 | |
Tiago Mello | eda03b5 | 2012-08-22 23:47:29 -0300 | [diff] [blame] | 46 | if k == 'link': |
| 47 | flavor['links'].append(v) |
| 48 | continue |
| 49 | |
| 50 | if k == '{%s}ephemeral' % XMLNS_OS_FLV_EXT_DATA: |
| 51 | k = 'OS-FLV-EXT-DATA:ephemeral' |
| 52 | |
ivan-zhu | ae7c7c5 | 2013-10-21 22:13:22 +0800 | [diff] [blame] | 53 | if k == '{%s}is_public' % XMLNS_OS_FLV_ACCESS: |
| 54 | k = 'os-flavor-access:is_public' |
| 55 | v = True if v == 'True' else False |
| 56 | |
Aditi Raveesh | ccfa653 | 2013-08-19 17:11:05 +0530 | [diff] [blame] | 57 | if k == 'extra_specs': |
Aarti Kriplani | c04a0fc | 2013-08-06 05:10:49 -0500 | [diff] [blame] | 58 | k = 'OS-FLV-WITH-EXT-SPECS:extra_specs' |
| 59 | flavor[k] = dict(v) |
| 60 | continue |
| 61 | |
Tiago Mello | eda03b5 | 2012-08-22 23:47:29 -0300 | [diff] [blame] | 62 | try: |
| 63 | v = int(v) |
| 64 | except ValueError: |
| 65 | try: |
| 66 | v = float(v) |
| 67 | except ValueError: |
| 68 | pass |
| 69 | |
| 70 | flavor[k] = v |
| 71 | |
| 72 | return flavor |
| 73 | |
| 74 | def _parse_array(self, node): |
Matthew Treinish | 28f164c | 2014-03-04 18:55:06 +0000 | [diff] [blame] | 75 | return [self._format_flavor(xml_utils.xml_to_json(x)) for x in node] |
Tiago Mello | eda03b5 | 2012-08-22 23:47:29 -0300 | [diff] [blame] | 76 | |
| 77 | def _list_flavors(self, url, params): |
Matthew Treinish | 26dd0fa | 2012-12-04 17:14:37 -0500 | [diff] [blame] | 78 | if params: |
Tiago Mello | eda03b5 | 2012-08-22 23:47:29 -0300 | [diff] [blame] | 79 | url += "?%s" % urllib.urlencode(params) |
| 80 | |
vponomaryov | f4c27f9 | 2014-02-18 10:56:42 +0200 | [diff] [blame] | 81 | resp, body = self.get(url) |
Tiago Mello | eda03b5 | 2012-08-22 23:47:29 -0300 | [diff] [blame] | 82 | flavors = self._parse_array(etree.fromstring(body)) |
| 83 | return resp, flavors |
| 84 | |
| 85 | def list_flavors(self, params=None): |
| 86 | url = 'flavors' |
| 87 | return self._list_flavors(url, params) |
| 88 | |
| 89 | def list_flavors_with_detail(self, params=None): |
| 90 | url = 'flavors/detail' |
| 91 | return self._list_flavors(url, params) |
| 92 | |
| 93 | def get_flavor_details(self, flavor_id): |
vponomaryov | f4c27f9 | 2014-02-18 10:56:42 +0200 | [diff] [blame] | 94 | resp, body = self.get("flavors/%s" % str(flavor_id)) |
Matthew Treinish | 28f164c | 2014-03-04 18:55:06 +0000 | [diff] [blame] | 95 | body = xml_utils.xml_to_json(etree.fromstring(body)) |
Tiago Mello | eda03b5 | 2012-08-22 23:47:29 -0300 | [diff] [blame] | 96 | flavor = self._format_flavor(body) |
| 97 | return resp, flavor |
| 98 | |
rajalakshmi-ganesan | f344cc3 | 2012-12-31 20:02:27 +0530 | [diff] [blame] | 99 | def create_flavor(self, name, ram, vcpus, disk, flavor_id, **kwargs): |
Sean Dague | f237ccb | 2013-01-04 15:19:14 -0500 | [diff] [blame] | 100 | """Creates a new flavor or instance type.""" |
Matthew Treinish | 28f164c | 2014-03-04 18:55:06 +0000 | [diff] [blame] | 101 | flavor = xml_utils.Element("flavor", |
| 102 | xmlns=xml_utils.XMLNS_11, |
| 103 | ram=ram, |
| 104 | vcpus=vcpus, |
| 105 | disk=disk, |
| 106 | id=flavor_id, |
| 107 | name=name) |
rajalakshmi-ganesan | f344cc3 | 2012-12-31 20:02:27 +0530 | [diff] [blame] | 108 | if kwargs.get('rxtx'): |
| 109 | flavor.add_attr('rxtx_factor', kwargs.get('rxtx')) |
| 110 | if kwargs.get('swap'): |
| 111 | flavor.add_attr('swap', kwargs.get('swap')) |
| 112 | if kwargs.get('ephemeral'): |
| 113 | flavor.add_attr('OS-FLV-EXT-DATA:ephemeral', |
| 114 | kwargs.get('ephemeral')) |
| 115 | if kwargs.get('is_public'): |
| 116 | flavor.add_attr('os-flavor-access:is_public', |
| 117 | kwargs.get('is_public')) |
Tiago Mello | eda03b5 | 2012-08-22 23:47:29 -0300 | [diff] [blame] | 118 | flavor.add_attr('xmlns:OS-FLV-EXT-DATA', XMLNS_OS_FLV_EXT_DATA) |
rajalakshmi-ganesan | f344cc3 | 2012-12-31 20:02:27 +0530 | [diff] [blame] | 119 | flavor.add_attr('xmlns:os-flavor-access', XMLNS_OS_FLV_ACCESS) |
Matthew Treinish | 28f164c | 2014-03-04 18:55:06 +0000 | [diff] [blame] | 120 | resp, body = self.post('flavors', str(xml_utils.Document(flavor))) |
| 121 | body = xml_utils.xml_to_json(etree.fromstring(body)) |
Tiago Mello | eda03b5 | 2012-08-22 23:47:29 -0300 | [diff] [blame] | 122 | flavor = self._format_flavor(body) |
| 123 | return resp, flavor |
| 124 | |
| 125 | def delete_flavor(self, flavor_id): |
Sean Dague | f237ccb | 2013-01-04 15:19:14 -0500 | [diff] [blame] | 126 | """Deletes the given flavor.""" |
vponomaryov | f4c27f9 | 2014-02-18 10:56:42 +0200 | [diff] [blame] | 127 | return self.delete("flavors/%s" % str(flavor_id)) |
rajalakshmi-ganesan | f344cc3 | 2012-12-31 20:02:27 +0530 | [diff] [blame] | 128 | |
| 129 | def is_resource_deleted(self, id): |
Attila Fazekas | a8b5fe7 | 2013-08-01 16:59:06 +0200 | [diff] [blame] | 130 | # Did not use get_flavor_details(id) for verification as it gives |
| 131 | # 200 ok even for deleted id. LP #981263 |
| 132 | # we can remove the loop here and use get by ID when bug gets sortedout |
rajalakshmi-ganesan | f344cc3 | 2012-12-31 20:02:27 +0530 | [diff] [blame] | 133 | resp, flavors = self.list_flavors_with_detail() |
| 134 | for flavor in flavors: |
| 135 | if flavor['id'] == id: |
| 136 | return False |
| 137 | return True |
rajalakshmi-ganesan | 6d0e7a2 | 2012-12-18 20:52:38 +0530 | [diff] [blame] | 138 | |
| 139 | def set_flavor_extra_spec(self, flavor_id, specs): |
| 140 | """Sets extra Specs to the mentioned flavor.""" |
Matthew Treinish | 28f164c | 2014-03-04 18:55:06 +0000 | [diff] [blame] | 141 | extra_specs = xml_utils.Element("extra_specs") |
rajalakshmi-ganesan | 6d0e7a2 | 2012-12-18 20:52:38 +0530 | [diff] [blame] | 142 | for key in specs.keys(): |
| 143 | extra_specs.add_attr(key, specs[key]) |
| 144 | resp, body = self.post('flavors/%s/os-extra_specs' % flavor_id, |
Matthew Treinish | 28f164c | 2014-03-04 18:55:06 +0000 | [diff] [blame] | 145 | str(xml_utils.Document(extra_specs))) |
| 146 | body = xml_utils.xml_to_json(etree.fromstring(body)) |
rajalakshmi-ganesan | 6d0e7a2 | 2012-12-18 20:52:38 +0530 | [diff] [blame] | 147 | return resp, body |
| 148 | |
| 149 | def get_flavor_extra_spec(self, flavor_id): |
| 150 | """Gets extra Specs of the mentioned flavor.""" |
vponomaryov | f4c27f9 | 2014-02-18 10:56:42 +0200 | [diff] [blame] | 151 | resp, body = self.get('flavors/%s/os-extra_specs' % flavor_id) |
Matthew Treinish | 28f164c | 2014-03-04 18:55:06 +0000 | [diff] [blame] | 152 | body = xml_utils.xml_to_json(etree.fromstring(body)) |
rajalakshmi-ganesan | 6d0e7a2 | 2012-12-18 20:52:38 +0530 | [diff] [blame] | 153 | return resp, body |
| 154 | |
lijunjbj | 9feaa21 | 2013-10-14 02:11:49 -0500 | [diff] [blame] | 155 | def get_flavor_extra_spec_with_key(self, flavor_id, key): |
Zhu Zhu | 2e1872d | 2013-10-08 21:23:29 -0500 | [diff] [blame] | 156 | """Gets extra Specs key-value of the mentioned flavor and key.""" |
| 157 | resp, xml_body = self.get('flavors/%s/os-extra_specs/%s' % |
vponomaryov | f4c27f9 | 2014-02-18 10:56:42 +0200 | [diff] [blame] | 158 | (str(flavor_id), key)) |
Zhu Zhu | 2e1872d | 2013-10-08 21:23:29 -0500 | [diff] [blame] | 159 | body = {} |
| 160 | element = etree.fromstring(xml_body) |
| 161 | key = element.get('key') |
Matthew Treinish | 28f164c | 2014-03-04 18:55:06 +0000 | [diff] [blame] | 162 | body[key] = xml_utils.xml_to_json(element) |
lijunjbj | 9feaa21 | 2013-10-14 02:11:49 -0500 | [diff] [blame] | 163 | return resp, body |
| 164 | |
ivan-zhu | ae7c7c5 | 2013-10-21 22:13:22 +0800 | [diff] [blame] | 165 | def update_flavor_extra_spec(self, flavor_id, key, **kwargs): |
Zhu Zhu | 2e1872d | 2013-10-08 21:23:29 -0500 | [diff] [blame] | 166 | """Update extra Specs details of the mentioned flavor and key.""" |
Matthew Treinish | 28f164c | 2014-03-04 18:55:06 +0000 | [diff] [blame] | 167 | doc = xml_utils.Document() |
ivan-zhu | ae7c7c5 | 2013-10-21 22:13:22 +0800 | [diff] [blame] | 168 | for (k, v) in kwargs.items(): |
Matthew Treinish | 28f164c | 2014-03-04 18:55:06 +0000 | [diff] [blame] | 169 | element = xml_utils.Element(k) |
ivan-zhu | ae7c7c5 | 2013-10-21 22:13:22 +0800 | [diff] [blame] | 170 | doc.append(element) |
Matthew Treinish | 28f164c | 2014-03-04 18:55:06 +0000 | [diff] [blame] | 171 | value = xml_utils.Text(v) |
ivan-zhu | ae7c7c5 | 2013-10-21 22:13:22 +0800 | [diff] [blame] | 172 | element.append(value) |
| 173 | |
| 174 | resp, body = self.put('flavors/%s/os-extra_specs/%s' % |
vponomaryov | f4c27f9 | 2014-02-18 10:56:42 +0200 | [diff] [blame] | 175 | (flavor_id, key), str(doc)) |
Matthew Treinish | 28f164c | 2014-03-04 18:55:06 +0000 | [diff] [blame] | 176 | body = xml_utils.xml_to_json(etree.fromstring(body)) |
ivan-zhu | ae7c7c5 | 2013-10-21 22:13:22 +0800 | [diff] [blame] | 177 | return resp, {key: body} |
| 178 | |
rajalakshmi-ganesan | 6d0e7a2 | 2012-12-18 20:52:38 +0530 | [diff] [blame] | 179 | def unset_flavor_extra_spec(self, flavor_id, key): |
| 180 | """Unsets an extra spec based on the mentioned flavor and key.""" |
| 181 | return self.delete('flavors/%s/os-extra_specs/%s' % (str(flavor_id), |
| 182 | key)) |
Mitsuhiko Yamazaki | f5f4da6 | 2013-03-29 17:40:03 +0900 | [diff] [blame] | 183 | |
| 184 | def _parse_array_access(self, node): |
Matthew Treinish | 28f164c | 2014-03-04 18:55:06 +0000 | [diff] [blame] | 185 | return [xml_utils.xml_to_json(x) for x in node] |
Mitsuhiko Yamazaki | f5f4da6 | 2013-03-29 17:40:03 +0900 | [diff] [blame] | 186 | |
Zhu Zhu | 705f24a | 2013-10-11 05:52:54 -0500 | [diff] [blame] | 187 | def list_flavor_access(self, flavor_id): |
| 188 | """Gets flavor access information given the flavor id.""" |
vponomaryov | f4c27f9 | 2014-02-18 10:56:42 +0200 | [diff] [blame] | 189 | resp, body = self.get('flavors/%s/os-flavor-access' % str(flavor_id)) |
Zhu Zhu | 705f24a | 2013-10-11 05:52:54 -0500 | [diff] [blame] | 190 | body = self._parse_array(etree.fromstring(body)) |
| 191 | return resp, body |
| 192 | |
Mitsuhiko Yamazaki | f5f4da6 | 2013-03-29 17:40:03 +0900 | [diff] [blame] | 193 | def add_flavor_access(self, flavor_id, tenant_id): |
| 194 | """Add flavor access for the specified tenant.""" |
Matthew Treinish | 28f164c | 2014-03-04 18:55:06 +0000 | [diff] [blame] | 195 | doc = xml_utils.Document() |
| 196 | server = xml_utils.Element("addTenantAccess") |
Mitsuhiko Yamazaki | f5f4da6 | 2013-03-29 17:40:03 +0900 | [diff] [blame] | 197 | doc.append(server) |
| 198 | server.add_attr("tenant", tenant_id) |
vponomaryov | f4c27f9 | 2014-02-18 10:56:42 +0200 | [diff] [blame] | 199 | resp, body = self.post('flavors/%s/action' % str(flavor_id), str(doc)) |
Mitsuhiko Yamazaki | f5f4da6 | 2013-03-29 17:40:03 +0900 | [diff] [blame] | 200 | body = self._parse_array_access(etree.fromstring(body)) |
| 201 | return resp, body |
| 202 | |
| 203 | def remove_flavor_access(self, flavor_id, tenant_id): |
| 204 | """Remove flavor access from the specified tenant.""" |
Matthew Treinish | 28f164c | 2014-03-04 18:55:06 +0000 | [diff] [blame] | 205 | doc = xml_utils.Document() |
| 206 | server = xml_utils.Element("removeTenantAccess") |
Mitsuhiko Yamazaki | f5f4da6 | 2013-03-29 17:40:03 +0900 | [diff] [blame] | 207 | doc.append(server) |
| 208 | server.add_attr("tenant", tenant_id) |
vponomaryov | f4c27f9 | 2014-02-18 10:56:42 +0200 | [diff] [blame] | 209 | resp, body = self.post('flavors/%s/action' % str(flavor_id), str(doc)) |
Mitsuhiko Yamazaki | f5f4da6 | 2013-03-29 17:40:03 +0900 | [diff] [blame] | 210 | body = self._parse_array_access(etree.fromstring(body)) |
| 211 | return resp, body |