Xing Yang | 0ddf83e | 2015-11-17 22:15:25 -0500 | [diff] [blame] | 1 | # Copyright (C) 2017 Dell Inc. or its subsidiaries. |
| 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 | |
| 16 | from tempest.api.volume import base |
| 17 | from tempest.common import waiters |
| 18 | from tempest import config |
| 19 | from tempest.lib.common.utils import data_utils |
xing-yang | d56edc3 | 2017-07-01 14:00:06 -0700 | [diff] [blame] | 20 | from tempest.lib.common.utils import test_utils |
Xing Yang | 0ddf83e | 2015-11-17 22:15:25 -0500 | [diff] [blame] | 21 | from tempest.lib import decorators |
| 22 | |
| 23 | CONF = config.CONF |
| 24 | |
| 25 | |
jeremy.zhang | cca9974 | 2017-08-20 01:08:24 +0800 | [diff] [blame^] | 26 | class BaseGroupsTest(base.BaseVolumeAdminTest): |
Xing Yang | 0ddf83e | 2015-11-17 22:15:25 -0500 | [diff] [blame] | 27 | |
| 28 | def _delete_group(self, grp_id, delete_volumes=True): |
xing-yang | d56edc3 | 2017-07-01 14:00:06 -0700 | [diff] [blame] | 29 | self.groups_client.delete_group(grp_id, delete_volumes) |
| 30 | vols = self.volumes_client.list_volumes(detail=True)['volumes'] |
Xing Yang | 0ddf83e | 2015-11-17 22:15:25 -0500 | [diff] [blame] | 31 | for vol in vols: |
| 32 | if vol['group_id'] == grp_id: |
xing-yang | d56edc3 | 2017-07-01 14:00:06 -0700 | [diff] [blame] | 33 | self.volumes_client.wait_for_resource_deletion(vol['id']) |
| 34 | self.groups_client.wait_for_resource_deletion(grp_id) |
Xing Yang | 0ddf83e | 2015-11-17 22:15:25 -0500 | [diff] [blame] | 35 | |
xing-yang | 41ed715 | 2017-05-03 06:52:56 -0400 | [diff] [blame] | 36 | def _delete_group_snapshot(self, group_snapshot_id, grp_id): |
xing-yang | d56edc3 | 2017-07-01 14:00:06 -0700 | [diff] [blame] | 37 | self.group_snapshots_client.delete_group_snapshot( |
xing-yang | 41ed715 | 2017-05-03 06:52:56 -0400 | [diff] [blame] | 38 | group_snapshot_id) |
xing-yang | d56edc3 | 2017-07-01 14:00:06 -0700 | [diff] [blame] | 39 | vols = self.volumes_client.list_volumes(detail=True)['volumes'] |
| 40 | snapshots = self.snapshots_client.list_snapshots( |
xing-yang | 41ed715 | 2017-05-03 06:52:56 -0400 | [diff] [blame] | 41 | detail=True)['snapshots'] |
| 42 | for vol in vols: |
| 43 | for snap in snapshots: |
| 44 | if (vol['group_id'] == grp_id and |
| 45 | vol['id'] == snap['volume_id']): |
xing-yang | d56edc3 | 2017-07-01 14:00:06 -0700 | [diff] [blame] | 46 | self.snapshots_client.wait_for_resource_deletion( |
xing-yang | 41ed715 | 2017-05-03 06:52:56 -0400 | [diff] [blame] | 47 | snap['id']) |
xing-yang | d56edc3 | 2017-07-01 14:00:06 -0700 | [diff] [blame] | 48 | self.group_snapshots_client.wait_for_resource_deletion( |
xing-yang | 41ed715 | 2017-05-03 06:52:56 -0400 | [diff] [blame] | 49 | group_snapshot_id) |
| 50 | |
xing-yang | d56edc3 | 2017-07-01 14:00:06 -0700 | [diff] [blame] | 51 | def _create_group(self, group_type, volume_type, grp_name=None): |
| 52 | if not grp_name: |
| 53 | grp_name = data_utils.rand_name('Group') |
| 54 | grp = self.groups_client.create_group( |
| 55 | group_type=group_type['id'], |
| 56 | volume_types=[volume_type['id']], |
| 57 | name=grp_name)['group'] |
| 58 | self.addCleanup(test_utils.call_and_ignore_notfound_exc, |
| 59 | self._delete_group, grp['id']) |
| 60 | waiters.wait_for_volume_resource_status( |
| 61 | self.groups_client, grp['id'], 'available') |
| 62 | self.assertEqual(grp_name, grp['name']) |
| 63 | return grp |
| 64 | |
jeremy.zhang | cca9974 | 2017-08-20 01:08:24 +0800 | [diff] [blame^] | 65 | |
| 66 | class GroupsTest(BaseGroupsTest): |
| 67 | min_microversion = '3.14' |
| 68 | max_microversion = 'latest' |
| 69 | _api_version = 3 |
| 70 | |
Xing Yang | 0ddf83e | 2015-11-17 22:15:25 -0500 | [diff] [blame] | 71 | @decorators.idempotent_id('4b111d28-b73d-4908-9bd2-03dc2992e4d4') |
| 72 | def test_group_create_show_list_delete(self): |
| 73 | # Create volume type |
| 74 | volume_type = self.create_volume_type() |
| 75 | |
| 76 | # Create group type |
| 77 | group_type = self.create_group_type() |
| 78 | |
| 79 | # Create group |
| 80 | grp1_name = data_utils.rand_name('Group1') |
xing-yang | d56edc3 | 2017-07-01 14:00:06 -0700 | [diff] [blame] | 81 | grp1 = self._create_group(group_type, volume_type, |
| 82 | grp_name=grp1_name) |
Xing Yang | 0ddf83e | 2015-11-17 22:15:25 -0500 | [diff] [blame] | 83 | grp1_id = grp1['id'] |
| 84 | |
| 85 | grp2_name = data_utils.rand_name('Group2') |
xing-yang | d56edc3 | 2017-07-01 14:00:06 -0700 | [diff] [blame] | 86 | grp2 = self._create_group(group_type, volume_type, |
| 87 | grp_name=grp2_name) |
Xing Yang | 0ddf83e | 2015-11-17 22:15:25 -0500 | [diff] [blame] | 88 | grp2_id = grp2['id'] |
| 89 | |
| 90 | # Create volume |
| 91 | vol1_name = data_utils.rand_name("volume") |
| 92 | params = {'name': vol1_name, |
| 93 | 'volume_type': volume_type['id'], |
| 94 | 'group_id': grp1['id'], |
| 95 | 'size': CONF.volume.volume_size} |
xing-yang | d56edc3 | 2017-07-01 14:00:06 -0700 | [diff] [blame] | 96 | vol1 = self.volumes_client.create_volume(**params)['volume'] |
Xing Yang | 0ddf83e | 2015-11-17 22:15:25 -0500 | [diff] [blame] | 97 | self.assertEqual(grp1['id'], vol1['group_id']) |
| 98 | waiters.wait_for_volume_resource_status( |
xing-yang | d56edc3 | 2017-07-01 14:00:06 -0700 | [diff] [blame] | 99 | self.volumes_client, vol1['id'], 'available') |
Xing Yang | 0ddf83e | 2015-11-17 22:15:25 -0500 | [diff] [blame] | 100 | vol1_id = vol1['id'] |
| 101 | |
| 102 | # Get a given group |
xing-yang | d56edc3 | 2017-07-01 14:00:06 -0700 | [diff] [blame] | 103 | grp1 = self.groups_client.show_group(grp1['id'])['group'] |
Xing Yang | 0ddf83e | 2015-11-17 22:15:25 -0500 | [diff] [blame] | 104 | self.assertEqual(grp1_name, grp1['name']) |
| 105 | self.assertEqual(grp1_id, grp1['id']) |
| 106 | |
xing-yang | d56edc3 | 2017-07-01 14:00:06 -0700 | [diff] [blame] | 107 | grp2 = self.groups_client.show_group(grp2['id'])['group'] |
Xing Yang | 0ddf83e | 2015-11-17 22:15:25 -0500 | [diff] [blame] | 108 | self.assertEqual(grp2_name, grp2['name']) |
| 109 | self.assertEqual(grp2_id, grp2['id']) |
| 110 | |
| 111 | # Get all groups with detail |
xing-yang | d56edc3 | 2017-07-01 14:00:06 -0700 | [diff] [blame] | 112 | grps = self.groups_client.list_groups( |
Xing Yang | 0ddf83e | 2015-11-17 22:15:25 -0500 | [diff] [blame] | 113 | detail=True)['groups'] |
| 114 | filtered_grps = [g for g in grps if g['id'] in [grp1_id, grp2_id]] |
| 115 | self.assertEqual(2, len(filtered_grps)) |
| 116 | for grp in filtered_grps: |
| 117 | self.assertEqual([volume_type['id']], grp['volume_types']) |
| 118 | self.assertEqual(group_type['id'], grp['group_type']) |
| 119 | |
xing-yang | d56edc3 | 2017-07-01 14:00:06 -0700 | [diff] [blame] | 120 | vols = self.volumes_client.list_volumes( |
Xing Yang | 0ddf83e | 2015-11-17 22:15:25 -0500 | [diff] [blame] | 121 | detail=True)['volumes'] |
| 122 | filtered_vols = [v for v in vols if v['id'] in [vol1_id]] |
| 123 | self.assertEqual(1, len(filtered_vols)) |
| 124 | for vol in filtered_vols: |
| 125 | self.assertEqual(grp1_id, vol['group_id']) |
| 126 | |
| 127 | # Delete group |
| 128 | # grp1 has a volume so delete_volumes flag is set to True by default |
| 129 | self._delete_group(grp1_id) |
| 130 | # grp2 is empty so delete_volumes flag can be set to False |
| 131 | self._delete_group(grp2_id, delete_volumes=False) |
xing-yang | d56edc3 | 2017-07-01 14:00:06 -0700 | [diff] [blame] | 132 | grps = self.groups_client.list_groups( |
Xing Yang | 0ddf83e | 2015-11-17 22:15:25 -0500 | [diff] [blame] | 133 | detail=True)['groups'] |
| 134 | self.assertEmpty(grps) |
xing-yang | 41ed715 | 2017-05-03 06:52:56 -0400 | [diff] [blame] | 135 | |
| 136 | @decorators.idempotent_id('1298e537-f1f0-47a3-a1dd-8adec8168897') |
| 137 | def test_group_snapshot_create_show_list_delete(self): |
| 138 | # Create volume type |
| 139 | volume_type = self.create_volume_type() |
| 140 | |
| 141 | # Create group type |
| 142 | group_type = self.create_group_type() |
| 143 | |
| 144 | # Create group |
xing-yang | d56edc3 | 2017-07-01 14:00:06 -0700 | [diff] [blame] | 145 | grp = self._create_group(group_type, volume_type) |
xing-yang | 41ed715 | 2017-05-03 06:52:56 -0400 | [diff] [blame] | 146 | |
| 147 | # Create volume |
| 148 | vol = self.create_volume(volume_type=volume_type['id'], |
| 149 | group_id=grp['id']) |
| 150 | |
| 151 | # Create group snapshot |
| 152 | group_snapshot_name = data_utils.rand_name('group_snapshot') |
| 153 | group_snapshot = ( |
xing-yang | d56edc3 | 2017-07-01 14:00:06 -0700 | [diff] [blame] | 154 | self.group_snapshots_client.create_group_snapshot( |
xing-yang | 41ed715 | 2017-05-03 06:52:56 -0400 | [diff] [blame] | 155 | group_id=grp['id'], |
| 156 | name=group_snapshot_name)['group_snapshot']) |
xing-yang | d56edc3 | 2017-07-01 14:00:06 -0700 | [diff] [blame] | 157 | snapshots = self.snapshots_client.list_snapshots( |
xing-yang | 41ed715 | 2017-05-03 06:52:56 -0400 | [diff] [blame] | 158 | detail=True)['snapshots'] |
| 159 | for snap in snapshots: |
| 160 | if vol['id'] == snap['volume_id']: |
| 161 | waiters.wait_for_volume_resource_status( |
xing-yang | d56edc3 | 2017-07-01 14:00:06 -0700 | [diff] [blame] | 162 | self.snapshots_client, snap['id'], 'available') |
xing-yang | 41ed715 | 2017-05-03 06:52:56 -0400 | [diff] [blame] | 163 | waiters.wait_for_volume_resource_status( |
xing-yang | d56edc3 | 2017-07-01 14:00:06 -0700 | [diff] [blame] | 164 | self.group_snapshots_client, |
xing-yang | 41ed715 | 2017-05-03 06:52:56 -0400 | [diff] [blame] | 165 | group_snapshot['id'], 'available') |
| 166 | self.assertEqual(group_snapshot_name, group_snapshot['name']) |
| 167 | |
| 168 | # Get a given group snapshot |
xing-yang | d56edc3 | 2017-07-01 14:00:06 -0700 | [diff] [blame] | 169 | group_snapshot = self.group_snapshots_client.show_group_snapshot( |
xing-yang | 41ed715 | 2017-05-03 06:52:56 -0400 | [diff] [blame] | 170 | group_snapshot['id'])['group_snapshot'] |
| 171 | self.assertEqual(group_snapshot_name, group_snapshot['name']) |
| 172 | |
| 173 | # Get all group snapshots with detail |
| 174 | group_snapshots = ( |
xing-yang | d56edc3 | 2017-07-01 14:00:06 -0700 | [diff] [blame] | 175 | self.group_snapshots_client.list_group_snapshots( |
xing-yang | 41ed715 | 2017-05-03 06:52:56 -0400 | [diff] [blame] | 176 | detail=True)['group_snapshots']) |
| 177 | self.assertIn((group_snapshot['name'], group_snapshot['id']), |
| 178 | [(m['name'], m['id']) for m in group_snapshots]) |
| 179 | |
| 180 | # Delete group snapshot |
| 181 | self._delete_group_snapshot(group_snapshot['id'], grp['id']) |
| 182 | group_snapshots = ( |
xing-yang | d56edc3 | 2017-07-01 14:00:06 -0700 | [diff] [blame] | 183 | self.group_snapshots_client.list_group_snapshots( |
xing-yang | 41ed715 | 2017-05-03 06:52:56 -0400 | [diff] [blame] | 184 | detail=True)['group_snapshots']) |
| 185 | self.assertEmpty(group_snapshots) |
xing-yang | d56edc3 | 2017-07-01 14:00:06 -0700 | [diff] [blame] | 186 | |
| 187 | @decorators.idempotent_id('eff52c70-efc7-45ed-b47a-4ad675d09b81') |
| 188 | def test_create_group_from_group_snapshot(self): |
| 189 | # Create volume type |
| 190 | volume_type = self.create_volume_type() |
| 191 | |
| 192 | # Create group type |
| 193 | group_type = self.create_group_type() |
| 194 | |
| 195 | # Create Group |
| 196 | grp = self._create_group(group_type, volume_type) |
| 197 | |
| 198 | # Create volume |
| 199 | vol = self.create_volume(volume_type=volume_type['id'], |
| 200 | group_id=grp['id']) |
| 201 | |
| 202 | # Create group_snapshot |
| 203 | group_snapshot_name = data_utils.rand_name('group_snapshot') |
| 204 | group_snapshot = ( |
| 205 | self.group_snapshots_client.create_group_snapshot( |
| 206 | group_id=grp['id'], |
| 207 | name=group_snapshot_name)['group_snapshot']) |
| 208 | self.addCleanup(self._delete_group_snapshot, |
| 209 | group_snapshot['id'], grp['id']) |
| 210 | self.assertEqual(group_snapshot_name, group_snapshot['name']) |
| 211 | snapshots = self.snapshots_client.list_snapshots( |
| 212 | detail=True)['snapshots'] |
| 213 | for snap in snapshots: |
| 214 | if vol['id'] == snap['volume_id']: |
| 215 | waiters.wait_for_volume_resource_status( |
| 216 | self.snapshots_client, snap['id'], 'available') |
| 217 | waiters.wait_for_volume_resource_status( |
| 218 | self.group_snapshots_client, |
| 219 | group_snapshot['id'], 'available') |
| 220 | |
| 221 | # Create Group from Group snapshot |
| 222 | grp_name2 = data_utils.rand_name('Group_from_snap') |
| 223 | grp2 = self.groups_client.create_group_from_source( |
| 224 | group_snapshot_id=group_snapshot['id'], |
| 225 | name=grp_name2)['group'] |
| 226 | self.addCleanup(self._delete_group, grp2['id']) |
| 227 | self.assertEqual(grp_name2, grp2['name']) |
| 228 | vols = self.volumes_client.list_volumes(detail=True)['volumes'] |
| 229 | for vol in vols: |
| 230 | if vol['group_id'] == grp2['id']: |
| 231 | waiters.wait_for_volume_resource_status( |
| 232 | self.volumes_client, vol['id'], 'available') |
| 233 | waiters.wait_for_volume_resource_status( |
| 234 | self.groups_client, grp2['id'], 'available') |
| 235 | |
| 236 | @decorators.idempotent_id('2424af8c-7851-4888-986a-794b10c3210e') |
| 237 | def test_create_group_from_group(self): |
| 238 | # Create volume type |
| 239 | volume_type = self.create_volume_type() |
| 240 | |
| 241 | # Create group type |
| 242 | group_type = self.create_group_type() |
| 243 | |
| 244 | # Create Group |
| 245 | grp = self._create_group(group_type, volume_type) |
| 246 | |
| 247 | # Create volume |
| 248 | self.create_volume(volume_type=volume_type['id'], group_id=grp['id']) |
| 249 | |
| 250 | # Create Group from Group |
| 251 | grp_name2 = data_utils.rand_name('Group_from_grp') |
| 252 | grp2 = self.groups_client.create_group_from_source( |
| 253 | source_group_id=grp['id'], name=grp_name2)['group'] |
| 254 | self.addCleanup(self._delete_group, grp2['id']) |
| 255 | self.assertEqual(grp_name2, grp2['name']) |
| 256 | vols = self.volumes_client.list_volumes( |
| 257 | detail=True)['volumes'] |
| 258 | for vol in vols: |
| 259 | if vol['group_id'] == grp2['id']: |
| 260 | waiters.wait_for_volume_resource_status( |
| 261 | self.volumes_client, vol['id'], 'available') |
| 262 | waiters.wait_for_volume_resource_status( |
| 263 | self.groups_client, grp2['id'], 'available') |
xing-yang | 9ee9860 | 2017-07-01 14:12:54 -0700 | [diff] [blame] | 264 | |
| 265 | @decorators.idempotent_id('4a8a6fd2-8b3b-4641-8f54-6a6f99320006') |
| 266 | def test_group_update(self): |
| 267 | # Create volume type |
| 268 | volume_type = self.create_volume_type() |
| 269 | |
| 270 | # Create group type |
| 271 | group_type = self.create_group_type() |
| 272 | |
| 273 | # Create Group |
| 274 | grp = self._create_group(group_type, volume_type) |
| 275 | |
xing-yang | 5805d32 | 2017-08-11 04:31:27 -0700 | [diff] [blame] | 276 | # Create volumes |
| 277 | grp_vols = [] |
| 278 | for _ in range(2): |
| 279 | vol = self.create_volume(volume_type=volume_type['id'], |
| 280 | group_id=grp['id']) |
| 281 | grp_vols.append(vol) |
| 282 | vol2 = grp_vols[1] |
xing-yang | 9ee9860 | 2017-07-01 14:12:54 -0700 | [diff] [blame] | 283 | |
| 284 | # Remove a volume from group and update name and description |
| 285 | new_grp_name = 'new_group' |
| 286 | new_desc = 'This is a new group' |
| 287 | grp_params = {'name': new_grp_name, |
| 288 | 'description': new_desc, |
xing-yang | 5805d32 | 2017-08-11 04:31:27 -0700 | [diff] [blame] | 289 | 'remove_volumes': vol2['id']} |
xing-yang | 9ee9860 | 2017-07-01 14:12:54 -0700 | [diff] [blame] | 290 | self.groups_client.update_group(grp['id'], **grp_params) |
| 291 | |
| 292 | # Wait for group status to become available |
| 293 | waiters.wait_for_volume_resource_status( |
| 294 | self.groups_client, grp['id'], 'available') |
| 295 | |
| 296 | # Get the updated Group |
| 297 | grp = self.groups_client.show_group(grp['id'])['group'] |
| 298 | self.assertEqual(new_grp_name, grp['name']) |
| 299 | self.assertEqual(new_desc, grp['description']) |
| 300 | |
| 301 | # Get volumes in the group |
| 302 | vols = self.volumes_client.list_volumes( |
| 303 | detail=True)['volumes'] |
| 304 | grp_vols = [] |
| 305 | for vol in vols: |
| 306 | if vol['group_id'] == grp['id']: |
| 307 | grp_vols.append(vol) |
| 308 | self.assertEqual(1, len(grp_vols)) |
xing-yang | 5805d32 | 2017-08-11 04:31:27 -0700 | [diff] [blame] | 309 | |
| 310 | # Add a volume to the group |
| 311 | grp_params = {'add_volumes': vol2['id']} |
| 312 | self.groups_client.update_group(grp['id'], **grp_params) |
| 313 | |
| 314 | # Wait for group status to become available |
| 315 | waiters.wait_for_volume_resource_status( |
| 316 | self.groups_client, grp['id'], 'available') |
| 317 | |
| 318 | # Get volumes in the group |
| 319 | vols = self.volumes_client.list_volumes( |
| 320 | detail=True)['volumes'] |
| 321 | grp_vols = [] |
| 322 | for vol in vols: |
| 323 | if vol['group_id'] == grp['id']: |
| 324 | grp_vols.append(vol) |
| 325 | self.assertEqual(2, len(grp_vols)) |
jeremy.zhang | cca9974 | 2017-08-20 01:08:24 +0800 | [diff] [blame^] | 326 | |
| 327 | |
| 328 | class GroupsV320Test(BaseGroupsTest): |
| 329 | _api_version = 3 |
| 330 | min_microversion = '3.20' |
| 331 | max_microversion = 'latest' |
| 332 | |
| 333 | @decorators.idempotent_id('b20c696b-0cbc-49a5-8b3a-b1fb9338f45c') |
| 334 | def test_reset_group_status(self): |
| 335 | # Create volume type |
| 336 | volume_type = self.create_volume_type() |
| 337 | |
| 338 | # Create group type |
| 339 | group_type = self.create_group_type() |
| 340 | |
| 341 | # Create group |
| 342 | group = self._create_group(group_type, volume_type) |
| 343 | |
| 344 | # Reset group status |
| 345 | self.addCleanup(waiters.wait_for_volume_resource_status, |
| 346 | self.groups_client, group['id'], 'available') |
| 347 | self.addCleanup(self.admin_groups_client.reset_group_status, |
| 348 | group['id'], 'available') |
| 349 | for status in ['creating', 'available', 'error']: |
| 350 | self.admin_groups_client.reset_group_status(group['id'], status) |
| 351 | waiters.wait_for_volume_resource_status( |
| 352 | self.groups_client, group['id'], status) |