blob: 9c33b716a650c2bd0a43a47b4c1cc1b8177b156e [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
Sean Dague52abbd92016-03-01 09:38:09 -050013from oslo_log import log as logging
zhufl6b7040a2017-01-18 16:38:34 +080014import testtools
Sean Dague52abbd92016-03-01 09:38:09 -050015
Fei Long Wangd39431f2015-05-14 11:30:48 +120016from tempest.common.utils import data_utils
Ken'ichi Ohmichi0eb153c2015-07-13 02:18:25 +000017from tempest.common import waiters
Matthew Treinish6c072292014-01-29 19:15:52 +000018from tempest import config
Ken'ichi Ohmichic85a9512017-01-27 18:34:24 -080019from tempest.lib import decorators
fujioka yuuichi636f8db2013-08-09 12:05:24 +090020from tempest.scenario import manager
Masayuki Igawa4ded9f02014-02-17 15:05:59 +090021from tempest import test
fujioka yuuichi636f8db2013-08-09 12:05:24 +090022
Matthew Treinish6c072292014-01-29 19:15:52 +000023CONF = config.CONF
Sean Dague52abbd92016-03-01 09:38:09 -050024LOG = logging.getLogger(__name__)
fujioka yuuichi636f8db2013-08-09 12:05:24 +090025
Nachi Ueno95b41282014-01-15 06:54:21 -080026
Joseph Lanouxeef192f2014-08-01 14:32:53 +000027class TestVolumeBootPattern(manager.ScenarioTest):
fujioka yuuichi636f8db2013-08-09 12:05:24 +090028
Sean Dague02620fd2016-03-02 15:52:51 -050029 # Boot from volume scenario is quite slow, and needs extra
30 # breathing room to get through deletes in the time allotted.
31 TIMEOUT_SCALING_FACTOR = 2
32
JordanPbce55532014-03-19 12:10:32 +010033 @classmethod
Emily Hugenbruch5e2d2a22015-02-25 21:35:45 +000034 def skip_checks(cls):
35 super(TestVolumeBootPattern, cls).skip_checks()
JordanPbce55532014-03-19 12:10:32 +010036 if not CONF.volume_feature_enabled.snapshot:
37 raise cls.skipException("Cinder volume snapshots are disabled")
fujioka yuuichi636f8db2013-08-09 12:05:24 +090038
39 def _create_volume_from_image(self):
Matthew Treinish6c072292014-01-29 19:15:52 +000040 img_uuid = CONF.compute.image_ref
zhuflc6ce5392016-08-17 14:34:37 +080041 vol_name = data_utils.rand_name(
42 self.__class__.__name__ + '-volume-origin')
fujioka yuuichi636f8db2013-08-09 12:05:24 +090043 return self.create_volume(name=vol_name, imageRef=img_uuid)
44
lkuchlan8789c552017-01-08 11:40:27 +020045 def _get_bdm(self, source_id, source_type, delete_on_termination=False):
fujioka yuuichi636f8db2013-08-09 12:05:24 +090046 # NOTE(gfidente): the syntax for block_device_mapping is
47 # dev_name=id:type:size:delete_on_terminate
48 # where type needs to be "snap" if the server is booted
49 # from a snapshot, size instead can be safely left empty
lkuchlan8789c552017-01-08 11:40:27 +020050
Joseph Lanouxeef192f2014-08-01 14:32:53 +000051 bd_map = [{
52 'device_name': 'vda',
lkuchlan8789c552017-01-08 11:40:27 +020053 '{}_id'.format(source_type): source_id,
Andrey Pavlovc8bd4b12015-08-17 10:20:17 +030054 'delete_on_termination': str(int(delete_on_termination))}]
55 return {'block_device_mapping': bd_map}
56
lkuchlan8789c552017-01-08 11:40:27 +020057 def _boot_instance_from_resource(self, source_id,
58 source_type,
59 keypair=None,
60 security_group=None,
61 delete_on_termination=False):
Andrey Pavlovc8bd4b12015-08-17 10:20:17 +030062 create_kwargs = dict()
63 if keypair:
64 create_kwargs['key_name'] = keypair['name']
65 if security_group:
66 create_kwargs['security_groups'] = [
67 {'name': security_group['name']}]
68 create_kwargs.update(self._get_bdm(
lkuchlan8789c552017-01-08 11:40:27 +020069 source_id,
70 source_type,
71 delete_on_termination=delete_on_termination))
72
zhufl13c9c892017-02-10 12:04:07 +080073 return self.create_server(image_id='', **create_kwargs)
fujioka yuuichi636f8db2013-08-09 12:05:24 +090074
75 def _create_snapshot_from_volume(self, vol_id):
zhuflc6ce5392016-08-17 14:34:37 +080076 snap_name = data_utils.rand_name(
77 self.__class__.__name__ + '-snapshot')
melanie witt87412222015-01-21 04:32:17 +000078 snap = self.snapshots_client.create_snapshot(
Joseph Lanouxeef192f2014-08-01 14:32:53 +000079 volume_id=vol_id,
80 force=True,
John Warrenff7faf62015-08-17 16:59:06 +000081 display_name=snap_name)['snapshot']
Yaroslav Lobankov46a78c32015-04-08 13:45:27 +030082 self.addCleanup(
83 self.snapshots_client.wait_for_resource_deletion, snap['id'])
84 self.addCleanup(self.snapshots_client.delete_snapshot, snap['id'])
lkuchlan52d7b0d2016-11-07 20:53:19 +020085 waiters.wait_for_volume_resource_status(self.snapshots_client,
86 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
fujioka yuuichi636f8db2013-08-09 12:05:24 +090096 def _delete_server(self, server):
Joseph Lanouxeef192f2014-08-01 14:32:53 +000097 self.servers_client.delete_server(server['id'])
Ken'ichi Ohmichie91a0c62015-08-13 02:09:16 +000098 waiters.wait_for_server_termination(self.servers_client, server['id'])
fujioka yuuichi636f8db2013-08-09 12:05:24 +090099
Ken'ichi Ohmichic85a9512017-01-27 18:34:24 -0800100 @decorators.idempotent_id('557cd2c2-4eb8-4dce-98be-f86765ff311b')
Sean Dague3c634d12015-04-27 12:09:19 -0400101 @test.attr(type='smoke')
zhufl6b7040a2017-01-18 16:38:34 +0800102 @testtools.skipUnless(CONF.network.public_network_id,
103 'The public_network_id option must be specified.')
Masayuki Igawa4ded9f02014-02-17 15:05:59 +0900104 @test.services('compute', 'volume', 'image')
fujioka yuuichi636f8db2013-08-09 12:05:24 +0900105 def test_volume_boot_pattern(self):
lkuchlan2041cdd2016-08-15 13:50:43 +0300106
107 """This test case attempts to reproduce the following steps:
108
109 * Create in Cinder some bootable volume importing a Glance image
110 * Boot an instance from the bootable volume
111 * Write content to the volume
112 * Delete an instance and Boot a new instance from the volume
113 * Check written content in the instance
114 * Create a volume snapshot while the instance is running
115 * Boot an additional instance from the new snapshot based volume
116 * Check written content in the instance booted from snapshot
117 """
118
Sean Dague52abbd92016-03-01 09:38:09 -0500119 LOG.info("Creating keypair and security group")
fujioka yuuichi636f8db2013-08-09 12:05:24 +0900120 keypair = self.create_keypair()
Yaroslav Lobankov0089af52015-07-02 19:14:40 +0300121 security_group = self._create_security_group()
fujioka yuuichi636f8db2013-08-09 12:05:24 +0900122
123 # create an instance from volume
Sean Dague52abbd92016-03-01 09:38:09 -0500124 LOG.info("Booting instance 1 from volume")
fujioka yuuichi636f8db2013-08-09 12:05:24 +0900125 volume_origin = self._create_volume_from_image()
lkuchlan8789c552017-01-08 11:40:27 +0200126 instance_1st = self._boot_instance_from_resource(
127 source_id=volume_origin['id'],
128 source_type='volume',
129 keypair=keypair,
130 security_group=security_group)
Jordan Pittier525ec712016-12-07 17:51:26 +0100131 LOG.info("Booted first instance: %s", instance_1st)
fujioka yuuichi636f8db2013-08-09 12:05:24 +0900132
133 # write content to volume on instance
Jordan Pittier525ec712016-12-07 17:51:26 +0100134 LOG.info("Setting timestamp in instance %s", instance_1st)
Sean Dague20e98612016-01-06 14:33:28 -0500135 ip_instance_1st = self.get_server_ip(instance_1st)
Alexander Gubanov59cc3032015-11-05 11:58:03 +0200136 timestamp = self.create_timestamp(ip_instance_1st,
137 private_key=keypair['private_key'])
fujioka yuuichi636f8db2013-08-09 12:05:24 +0900138
139 # delete instance
Jordan Pittier525ec712016-12-07 17:51:26 +0100140 LOG.info("Deleting first instance: %s", instance_1st)
fujioka yuuichi636f8db2013-08-09 12:05:24 +0900141 self._delete_server(instance_1st)
142
143 # create a 2nd instance from volume
lkuchlan8789c552017-01-08 11:40:27 +0200144 instance_2nd = self._boot_instance_from_resource(
145 source_id=volume_origin['id'],
146 source_type='volume',
147 keypair=keypair,
148 security_group=security_group)
Jordan Pittier525ec712016-12-07 17:51:26 +0100149 LOG.info("Booted second instance %s", instance_2nd)
fujioka yuuichi636f8db2013-08-09 12:05:24 +0900150
151 # check the content of written file
Jordan Pittier525ec712016-12-07 17:51:26 +0100152 LOG.info("Getting timestamp in instance %s", instance_2nd)
Sean Dague20e98612016-01-06 14:33:28 -0500153 ip_instance_2nd = self.get_server_ip(instance_2nd)
Alexander Gubanov59cc3032015-11-05 11:58:03 +0200154 timestamp2 = self.get_timestamp(ip_instance_2nd,
155 private_key=keypair['private_key'])
156 self.assertEqual(timestamp, timestamp2)
fujioka yuuichi636f8db2013-08-09 12:05:24 +0900157
158 # snapshot a volume
Jordan Pittier525ec712016-12-07 17:51:26 +0100159 LOG.info("Creating snapshot from volume: %s", volume_origin['id'])
Joseph Lanouxeef192f2014-08-01 14:32:53 +0000160 snapshot = self._create_snapshot_from_volume(volume_origin['id'])
fujioka yuuichi636f8db2013-08-09 12:05:24 +0900161
162 # create a 3rd instance from snapshot
Jordan Pittier525ec712016-12-07 17:51:26 +0100163 LOG.info("Creating third instance from snapshot: %s", snapshot['id'])
Nuno Santosda899622016-11-17 12:32:53 -0500164 volume = self.create_volume(snapshot_id=snapshot['id'],
165 size=snapshot['size'])
166 LOG.info("Booting third instance from snapshot")
Alexander Gubanovc8829f82015-11-12 10:35:13 +0200167 server_from_snapshot = (
lkuchlan8789c552017-01-08 11:40:27 +0200168 self._boot_instance_from_resource(source_id=volume['id'],
169 source_type='volume',
170 keypair=keypair,
171 security_group=security_group))
Nuno Santosda899622016-11-17 12:32:53 -0500172 LOG.info("Booted third instance %s", server_from_snapshot)
fujioka yuuichi636f8db2013-08-09 12:05:24 +0900173
174 # check the content of written file
Jordan Pittier525ec712016-12-07 17:51:26 +0100175 LOG.info("Logging into third instance to get timestamp: %s",
Sean Dague52abbd92016-03-01 09:38:09 -0500176 server_from_snapshot)
Sean Dague20e98612016-01-06 14:33:28 -0500177 server_from_snapshot_ip = self.get_server_ip(server_from_snapshot)
Alexander Gubanovc8829f82015-11-12 10:35:13 +0200178 timestamp3 = self.get_timestamp(server_from_snapshot_ip,
Alexander Gubanov59cc3032015-11-05 11:58:03 +0200179 private_key=keypair['private_key'])
180 self.assertEqual(timestamp, timestamp3)
fujioka yuuichi636f8db2013-08-09 12:05:24 +0900181
Ken'ichi Ohmichic85a9512017-01-27 18:34:24 -0800182 @decorators.idempotent_id('05795fb2-b2a7-4c9f-8fac-ff25aedb1489')
lkuchlan8789c552017-01-08 11:40:27 +0200183 @test.services('compute', 'image', 'volume')
184 def test_create_server_from_volume_snapshot(self):
185 # Create a volume from an image
186 boot_volume = self._create_volume_from_image()
187
188 # Create a snapshot
189 boot_snapshot = self._create_snapshot_from_volume(boot_volume['id'])
190
191 # Create a server from a volume snapshot
192 server = self._boot_instance_from_resource(
193 source_id=boot_snapshot['id'],
194 source_type='snapshot',
195 delete_on_termination=True)
196
197 server_info = self.servers_client.show_server(server['id'])['server']
198
199 # The created volume when creating a server from a snapshot
200 created_volume = server_info['os-extended-volumes:volumes_attached']
201
lkuchlan9e22b852017-02-05 15:38:29 +0200202 self.assertNotEmpty(created_volume, "No volume attachment found.")
203
lkuchlan8789c552017-01-08 11:40:27 +0200204 created_volume_info = self.volumes_client.show_volume(
205 created_volume[0]['id'])['volume']
206
207 # Verify the server was created from the snapshot
208 self.assertEqual(
209 boot_volume['volume_image_metadata']['image_id'],
210 created_volume_info['volume_image_metadata']['image_id'])
211 self.assertEqual(boot_snapshot['id'],
212 created_volume_info['snapshot_id'])
213 self.assertEqual(server['id'],
214 created_volume_info['attachments'][0]['server_id'])
215 self.assertEqual(created_volume[0]['id'],
216 created_volume_info['attachments'][0]['volume_id'])
217
Ken'ichi Ohmichic85a9512017-01-27 18:34:24 -0800218 @decorators.idempotent_id('36c34c67-7b54-4b59-b188-02a2f458a63b')
Andrey Pavlovc8bd4b12015-08-17 10:20:17 +0300219 @test.services('compute', 'volume', 'image')
220 def test_create_ebs_image_and_check_boot(self):
221 # create an instance from volume
222 volume_origin = self._create_volume_from_image()
lkuchlan8789c552017-01-08 11:40:27 +0200223 instance = self._boot_instance_from_resource(
224 source_id=volume_origin['id'],
225 source_type='volume',
226 delete_on_termination=True)
Andrey Pavlovc8bd4b12015-08-17 10:20:17 +0300227 # create EBS image
zhuflf9d95722016-10-19 16:06:17 +0800228 image = self.create_server_snapshot(instance)
Andrey Pavlovc8bd4b12015-08-17 10:20:17 +0300229
230 # delete instance
231 self._delete_server(instance)
232
233 # boot instance from EBS image
zhufl13c9c892017-02-10 12:04:07 +0800234 instance = self.create_server(image_id=image['id'])
Andrey Pavlovc8bd4b12015-08-17 10:20:17 +0300235 # just ensure that instance booted
236
237 # delete instance
238 self._delete_server(instance)
239
Nikola Dipanov7cff03f2014-03-12 14:06:25 +0100240
241class TestVolumeBootPatternV2(TestVolumeBootPattern):
lkuchlan8789c552017-01-08 11:40:27 +0200242 def _get_bdm(self, source_id, source_type, delete_on_termination=False):
Yaroslav Lobankov0089af52015-07-02 19:14:40 +0300243 bd_map_v2 = [{
lkuchlan8789c552017-01-08 11:40:27 +0200244 'uuid': source_id,
245 'source_type': source_type,
Yaroslav Lobankov0089af52015-07-02 19:14:40 +0300246 'destination_type': 'volume',
247 'boot_index': 0,
Andrey Pavlovc8bd4b12015-08-17 10:20:17 +0300248 'delete_on_termination': delete_on_termination}]
249 return {'block_device_mapping_v2': bd_map_v2}