blob: 81ecda0c7d0c56011057ba94d01cd706559ca6db [file] [log] [blame]
fujioka yuuichi636f8db2013-08-09 12:05:24 +09001# Licensed under the Apache License, Version 2.0 (the "License"); you may
2# not use this file except in compliance with the License. You may obtain
3# a copy of the License at
4#
5# http://www.apache.org/licenses/LICENSE-2.0
6#
7# Unless required by applicable law or agreed to in writing, software
8# distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
9# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
10# License for the specific language governing permissions and limitations
11# under the License.
12
Doug Hellmann583ce2c2015-03-11 14:55:46 +000013from oslo_log import log
Matthew Treinishc49fcbe2015-02-05 23:37:34 -050014
Fei Long Wangd39431f2015-05-14 11:30:48 +120015from tempest.common.utils import data_utils
Ken'ichi Ohmichi0eb153c2015-07-13 02:18:25 +000016from tempest.common import waiters
Matthew Treinish6c072292014-01-29 19:15:52 +000017from tempest import config
fujioka yuuichi636f8db2013-08-09 12:05:24 +090018from tempest.scenario import manager
Masayuki Igawa4ded9f02014-02-17 15:05:59 +090019from tempest import test
fujioka yuuichi636f8db2013-08-09 12:05:24 +090020
Matthew Treinish6c072292014-01-29 19:15:52 +000021CONF = config.CONF
fujioka yuuichi636f8db2013-08-09 12:05:24 +090022
Nachi Ueno95b41282014-01-15 06:54:21 -080023LOG = log.getLogger(__name__)
24
25
Joseph Lanouxeef192f2014-08-01 14:32:53 +000026class TestVolumeBootPattern(manager.ScenarioTest):
fujioka yuuichi636f8db2013-08-09 12:05:24 +090027
Ken'ichi Ohmichic4e4f1c2015-11-17 08:16:12 +000028 """This test case attempts to reproduce the following steps:
fujioka yuuichi636f8db2013-08-09 12:05:24 +090029
30 * Create in Cinder some bootable volume importing a Glance image
31 * Boot an instance from the bootable volume
32 * Write content to the volume
33 * Delete an instance and Boot a new instance from the volume
34 * Check written content in the instance
35 * Create a volume snapshot while the instance is running
36 * Boot an additional instance from the new snapshot based volume
37 * Check written content in the instance booted from snapshot
38 """
JordanPbce55532014-03-19 12:10:32 +010039 @classmethod
Emily Hugenbruch5e2d2a22015-02-25 21:35:45 +000040 def skip_checks(cls):
41 super(TestVolumeBootPattern, cls).skip_checks()
JordanPbce55532014-03-19 12:10:32 +010042 if not CONF.volume_feature_enabled.snapshot:
43 raise cls.skipException("Cinder volume snapshots are disabled")
fujioka yuuichi636f8db2013-08-09 12:05:24 +090044
45 def _create_volume_from_image(self):
Matthew Treinish6c072292014-01-29 19:15:52 +000046 img_uuid = CONF.compute.image_ref
Masayuki Igawa259c1132013-10-31 17:48:44 +090047 vol_name = data_utils.rand_name('volume-origin')
fujioka yuuichi636f8db2013-08-09 12:05:24 +090048 return self.create_volume(name=vol_name, imageRef=img_uuid)
49
Andrey Pavlovc8bd4b12015-08-17 10:20:17 +030050 def _get_bdm(self, vol_id, delete_on_termination=False):
fujioka yuuichi636f8db2013-08-09 12:05:24 +090051 # NOTE(gfidente): the syntax for block_device_mapping is
52 # dev_name=id:type:size:delete_on_terminate
53 # where type needs to be "snap" if the server is booted
54 # from a snapshot, size instead can be safely left empty
Joseph Lanouxeef192f2014-08-01 14:32:53 +000055 bd_map = [{
56 'device_name': 'vda',
57 'volume_id': vol_id,
Andrey Pavlovc8bd4b12015-08-17 10:20:17 +030058 'delete_on_termination': str(int(delete_on_termination))}]
59 return {'block_device_mapping': bd_map}
60
61 def _boot_instance_from_volume(self, vol_id, keypair=None,
62 security_group=None,
63 delete_on_termination=False):
64 create_kwargs = dict()
65 if keypair:
66 create_kwargs['key_name'] = keypair['name']
67 if security_group:
68 create_kwargs['security_groups'] = [
69 {'name': security_group['name']}]
70 create_kwargs.update(self._get_bdm(
71 vol_id, delete_on_termination=delete_on_termination))
lanoux5fc14522015-09-21 08:17:35 +000072 return self.create_server(
73 image='',
74 wait_until='ACTIVE',
75 **create_kwargs)
fujioka yuuichi636f8db2013-08-09 12:05:24 +090076
77 def _create_snapshot_from_volume(self, vol_id):
Masayuki Igawa259c1132013-10-31 17:48:44 +090078 snap_name = data_utils.rand_name('snapshot')
melanie witt87412222015-01-21 04:32:17 +000079 snap = self.snapshots_client.create_snapshot(
Joseph Lanouxeef192f2014-08-01 14:32:53 +000080 volume_id=vol_id,
81 force=True,
John Warrenff7faf62015-08-17 16:59:06 +000082 display_name=snap_name)['snapshot']
Yaroslav Lobankov46a78c32015-04-08 13:45:27 +030083 self.addCleanup(
84 self.snapshots_client.wait_for_resource_deletion, snap['id'])
85 self.addCleanup(self.snapshots_client.delete_snapshot, snap['id'])
Joseph Lanouxeef192f2014-08-01 14:32:53 +000086 self.snapshots_client.wait_for_snapshot_status(snap['id'], 'available')
Ivan Kolodyazhnybcfc32e2015-08-06 13:31:36 +030087
88 # NOTE(e0ne): Cinder API v2 uses name instead of display_name
89 if 'display_name' in snap:
90 self.assertEqual(snap_name, snap['display_name'])
91 else:
92 self.assertEqual(snap_name, snap['name'])
93
fujioka yuuichi636f8db2013-08-09 12:05:24 +090094 return snap
95
96 def _create_volume_from_snapshot(self, snap_id):
Masayuki Igawa259c1132013-10-31 17:48:44 +090097 vol_name = data_utils.rand_name('volume')
fujioka yuuichi636f8db2013-08-09 12:05:24 +090098 return self.create_volume(name=vol_name, snapshot_id=snap_id)
99
fujioka yuuichi636f8db2013-08-09 12:05:24 +0900100 def _delete_server(self, server):
Joseph Lanouxeef192f2014-08-01 14:32:53 +0000101 self.servers_client.delete_server(server['id'])
Ken'ichi Ohmichie91a0c62015-08-13 02:09:16 +0000102 waiters.wait_for_server_termination(self.servers_client, server['id'])
fujioka yuuichi636f8db2013-08-09 12:05:24 +0900103
Chris Hoge7579c1a2015-02-26 14:12:15 -0800104 @test.idempotent_id('557cd2c2-4eb8-4dce-98be-f86765ff311b')
Sean Dague3c634d12015-04-27 12:09:19 -0400105 @test.attr(type='smoke')
Masayuki Igawa4ded9f02014-02-17 15:05:59 +0900106 @test.services('compute', 'volume', 'image')
fujioka yuuichi636f8db2013-08-09 12:05:24 +0900107 def test_volume_boot_pattern(self):
108 keypair = self.create_keypair()
Yaroslav Lobankov0089af52015-07-02 19:14:40 +0300109 security_group = self._create_security_group()
fujioka yuuichi636f8db2013-08-09 12:05:24 +0900110
111 # create an instance from volume
112 volume_origin = self._create_volume_from_image()
Joseph Lanouxeef192f2014-08-01 14:32:53 +0000113 instance_1st = self._boot_instance_from_volume(volume_origin['id'],
Yaroslav Lobankov0089af52015-07-02 19:14:40 +0300114 keypair, security_group)
fujioka yuuichi636f8db2013-08-09 12:05:24 +0900115
116 # write content to volume on instance
Alexander Gubanovc8829f82015-11-12 10:35:13 +0200117 ip_instance_1st = self.get_server_or_ip(instance_1st)
Alexander Gubanov59cc3032015-11-05 11:58:03 +0200118 timestamp = self.create_timestamp(ip_instance_1st,
119 private_key=keypair['private_key'])
fujioka yuuichi636f8db2013-08-09 12:05:24 +0900120
121 # delete instance
122 self._delete_server(instance_1st)
123
124 # create a 2nd instance from volume
Joseph Lanouxeef192f2014-08-01 14:32:53 +0000125 instance_2nd = self._boot_instance_from_volume(volume_origin['id'],
Yaroslav Lobankov0089af52015-07-02 19:14:40 +0300126 keypair, security_group)
fujioka yuuichi636f8db2013-08-09 12:05:24 +0900127
128 # check the content of written file
Alexander Gubanovc8829f82015-11-12 10:35:13 +0200129 ip_instance_2nd = self.get_server_or_ip(instance_2nd)
Alexander Gubanov59cc3032015-11-05 11:58:03 +0200130 timestamp2 = self.get_timestamp(ip_instance_2nd,
131 private_key=keypair['private_key'])
132 self.assertEqual(timestamp, timestamp2)
fujioka yuuichi636f8db2013-08-09 12:05:24 +0900133
134 # snapshot a volume
Joseph Lanouxeef192f2014-08-01 14:32:53 +0000135 snapshot = self._create_snapshot_from_volume(volume_origin['id'])
fujioka yuuichi636f8db2013-08-09 12:05:24 +0900136
137 # create a 3rd instance from snapshot
Joseph Lanouxeef192f2014-08-01 14:32:53 +0000138 volume = self._create_volume_from_snapshot(snapshot['id'])
Alexander Gubanovc8829f82015-11-12 10:35:13 +0200139 server_from_snapshot = (
Yaroslav Lobankov0089af52015-07-02 19:14:40 +0300140 self._boot_instance_from_volume(volume['id'],
141 keypair, security_group))
fujioka yuuichi636f8db2013-08-09 12:05:24 +0900142
143 # check the content of written file
Alexander Gubanovc8829f82015-11-12 10:35:13 +0200144 server_from_snapshot_ip = self.get_server_or_ip(server_from_snapshot)
145 timestamp3 = self.get_timestamp(server_from_snapshot_ip,
Alexander Gubanov59cc3032015-11-05 11:58:03 +0200146 private_key=keypair['private_key'])
147 self.assertEqual(timestamp, timestamp3)
fujioka yuuichi636f8db2013-08-09 12:05:24 +0900148
Andrey Pavlovc8bd4b12015-08-17 10:20:17 +0300149 @test.idempotent_id('36c34c67-7b54-4b59-b188-02a2f458a63b')
150 @test.services('compute', 'volume', 'image')
151 def test_create_ebs_image_and_check_boot(self):
152 # create an instance from volume
153 volume_origin = self._create_volume_from_image()
154 instance = self._boot_instance_from_volume(volume_origin['id'],
155 delete_on_termination=True)
156 # create EBS image
157 name = data_utils.rand_name('image')
158 image = self.create_server_snapshot(instance, name=name)
159
160 # delete instance
161 self._delete_server(instance)
162
163 # boot instance from EBS image
lanoux5fc14522015-09-21 08:17:35 +0000164 instance = self.create_server(
165 image_id=image['id'])
Andrey Pavlovc8bd4b12015-08-17 10:20:17 +0300166 # just ensure that instance booted
167
168 # delete instance
169 self._delete_server(instance)
170
Nikola Dipanov7cff03f2014-03-12 14:06:25 +0100171
172class TestVolumeBootPatternV2(TestVolumeBootPattern):
Andrey Pavlovc8bd4b12015-08-17 10:20:17 +0300173 def _get_bdm(self, vol_id, delete_on_termination=False):
Yaroslav Lobankov0089af52015-07-02 19:14:40 +0300174 bd_map_v2 = [{
175 'uuid': vol_id,
176 'source_type': 'volume',
177 'destination_type': 'volume',
178 'boot_index': 0,
Andrey Pavlovc8bd4b12015-08-17 10:20:17 +0300179 'delete_on_termination': delete_on_termination}]
180 return {'block_device_mapping_v2': bd_map_v2}