blob: 3f6d9c405d94646261621003f9940309853db9ba [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
14
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
Sean Dague52abbd92016-03-01 09:38:09 -050022LOG = logging.getLogger(__name__)
fujioka yuuichi636f8db2013-08-09 12:05:24 +090023
Nachi Ueno95b41282014-01-15 06:54:21 -080024
Joseph Lanouxeef192f2014-08-01 14:32:53 +000025class TestVolumeBootPattern(manager.ScenarioTest):
fujioka yuuichi636f8db2013-08-09 12:05:24 +090026
Ken'ichi Ohmichic4e4f1c2015-11-17 08:16:12 +000027 """This test case attempts to reproduce the following steps:
fujioka yuuichi636f8db2013-08-09 12:05:24 +090028
29 * Create in Cinder some bootable volume importing a Glance image
30 * Boot an instance from the bootable volume
31 * Write content to the volume
32 * Delete an instance and Boot a new instance from the volume
33 * Check written content in the instance
34 * Create a volume snapshot while the instance is running
35 * Boot an additional instance from the new snapshot based volume
36 * Check written content in the instance booted from snapshot
37 """
Sean Dague02620fd2016-03-02 15:52:51 -050038
39 # Boot from volume scenario is quite slow, and needs extra
40 # breathing room to get through deletes in the time allotted.
41 TIMEOUT_SCALING_FACTOR = 2
42
JordanPbce55532014-03-19 12:10:32 +010043 @classmethod
Emily Hugenbruch5e2d2a22015-02-25 21:35:45 +000044 def skip_checks(cls):
45 super(TestVolumeBootPattern, cls).skip_checks()
JordanPbce55532014-03-19 12:10:32 +010046 if not CONF.volume_feature_enabled.snapshot:
47 raise cls.skipException("Cinder volume snapshots are disabled")
fujioka yuuichi636f8db2013-08-09 12:05:24 +090048
49 def _create_volume_from_image(self):
Matthew Treinish6c072292014-01-29 19:15:52 +000050 img_uuid = CONF.compute.image_ref
zhuflc6ce5392016-08-17 14:34:37 +080051 vol_name = data_utils.rand_name(
52 self.__class__.__name__ + '-volume-origin')
fujioka yuuichi636f8db2013-08-09 12:05:24 +090053 return self.create_volume(name=vol_name, imageRef=img_uuid)
54
Andrey Pavlovc8bd4b12015-08-17 10:20:17 +030055 def _get_bdm(self, vol_id, delete_on_termination=False):
fujioka yuuichi636f8db2013-08-09 12:05:24 +090056 # NOTE(gfidente): the syntax for block_device_mapping is
57 # dev_name=id:type:size:delete_on_terminate
58 # where type needs to be "snap" if the server is booted
59 # from a snapshot, size instead can be safely left empty
Joseph Lanouxeef192f2014-08-01 14:32:53 +000060 bd_map = [{
61 'device_name': 'vda',
62 'volume_id': vol_id,
Andrey Pavlovc8bd4b12015-08-17 10:20:17 +030063 'delete_on_termination': str(int(delete_on_termination))}]
64 return {'block_device_mapping': bd_map}
65
66 def _boot_instance_from_volume(self, vol_id, keypair=None,
67 security_group=None,
68 delete_on_termination=False):
69 create_kwargs = dict()
70 if keypair:
71 create_kwargs['key_name'] = keypair['name']
72 if security_group:
73 create_kwargs['security_groups'] = [
74 {'name': security_group['name']}]
75 create_kwargs.update(self._get_bdm(
76 vol_id, delete_on_termination=delete_on_termination))
lanoux5fc14522015-09-21 08:17:35 +000077 return self.create_server(
Anusha Ramineni9aaef8b2016-01-19 10:56:40 +053078 image_id='',
lanoux5fc14522015-09-21 08:17:35 +000079 wait_until='ACTIVE',
80 **create_kwargs)
fujioka yuuichi636f8db2013-08-09 12:05:24 +090081
82 def _create_snapshot_from_volume(self, vol_id):
zhuflc6ce5392016-08-17 14:34:37 +080083 snap_name = data_utils.rand_name(
84 self.__class__.__name__ + '-snapshot')
melanie witt87412222015-01-21 04:32:17 +000085 snap = self.snapshots_client.create_snapshot(
Joseph Lanouxeef192f2014-08-01 14:32:53 +000086 volume_id=vol_id,
87 force=True,
John Warrenff7faf62015-08-17 16:59:06 +000088 display_name=snap_name)['snapshot']
Yaroslav Lobankov46a78c32015-04-08 13:45:27 +030089 self.addCleanup(
90 self.snapshots_client.wait_for_resource_deletion, snap['id'])
91 self.addCleanup(self.snapshots_client.delete_snapshot, snap['id'])
Yaroslav Lobankov667aaa22016-03-24 23:13:28 -050092 waiters.wait_for_snapshot_status(self.snapshots_client,
93 snap['id'], 'available')
Ivan Kolodyazhnybcfc32e2015-08-06 13:31:36 +030094
95 # NOTE(e0ne): Cinder API v2 uses name instead of display_name
96 if 'display_name' in snap:
97 self.assertEqual(snap_name, snap['display_name'])
98 else:
99 self.assertEqual(snap_name, snap['name'])
100
fujioka yuuichi636f8db2013-08-09 12:05:24 +0900101 return snap
102
103 def _create_volume_from_snapshot(self, snap_id):
zhuflc6ce5392016-08-17 14:34:37 +0800104 vol_name = data_utils.rand_name(
105 self.__class__.__name__ + '-volume')
fujioka yuuichi636f8db2013-08-09 12:05:24 +0900106 return self.create_volume(name=vol_name, snapshot_id=snap_id)
107
fujioka yuuichi636f8db2013-08-09 12:05:24 +0900108 def _delete_server(self, server):
Joseph Lanouxeef192f2014-08-01 14:32:53 +0000109 self.servers_client.delete_server(server['id'])
Ken'ichi Ohmichie91a0c62015-08-13 02:09:16 +0000110 waiters.wait_for_server_termination(self.servers_client, server['id'])
fujioka yuuichi636f8db2013-08-09 12:05:24 +0900111
Chris Hoge7579c1a2015-02-26 14:12:15 -0800112 @test.idempotent_id('557cd2c2-4eb8-4dce-98be-f86765ff311b')
Sean Dague3c634d12015-04-27 12:09:19 -0400113 @test.attr(type='smoke')
Masayuki Igawa4ded9f02014-02-17 15:05:59 +0900114 @test.services('compute', 'volume', 'image')
fujioka yuuichi636f8db2013-08-09 12:05:24 +0900115 def test_volume_boot_pattern(self):
Sean Dague52abbd92016-03-01 09:38:09 -0500116 LOG.info("Creating keypair and security group")
fujioka yuuichi636f8db2013-08-09 12:05:24 +0900117 keypair = self.create_keypair()
Yaroslav Lobankov0089af52015-07-02 19:14:40 +0300118 security_group = self._create_security_group()
fujioka yuuichi636f8db2013-08-09 12:05:24 +0900119
120 # create an instance from volume
Sean Dague52abbd92016-03-01 09:38:09 -0500121 LOG.info("Booting instance 1 from volume")
fujioka yuuichi636f8db2013-08-09 12:05:24 +0900122 volume_origin = self._create_volume_from_image()
Joseph Lanouxeef192f2014-08-01 14:32:53 +0000123 instance_1st = self._boot_instance_from_volume(volume_origin['id'],
Yaroslav Lobankov0089af52015-07-02 19:14:40 +0300124 keypair, security_group)
Sean Dague52abbd92016-03-01 09:38:09 -0500125 LOG.info("Booted first instance: %s" % instance_1st)
fujioka yuuichi636f8db2013-08-09 12:05:24 +0900126
127 # write content to volume on instance
Sean Dague52abbd92016-03-01 09:38:09 -0500128 LOG.info("Setting timestamp in instance %s" % instance_1st)
Sean Dague20e98612016-01-06 14:33:28 -0500129 ip_instance_1st = self.get_server_ip(instance_1st)
Alexander Gubanov59cc3032015-11-05 11:58:03 +0200130 timestamp = self.create_timestamp(ip_instance_1st,
131 private_key=keypair['private_key'])
fujioka yuuichi636f8db2013-08-09 12:05:24 +0900132
133 # delete instance
Sean Dague52abbd92016-03-01 09:38:09 -0500134 LOG.info("Deleting first instance: %s" % instance_1st)
fujioka yuuichi636f8db2013-08-09 12:05:24 +0900135 self._delete_server(instance_1st)
136
137 # create a 2nd instance from volume
Joseph Lanouxeef192f2014-08-01 14:32:53 +0000138 instance_2nd = self._boot_instance_from_volume(volume_origin['id'],
Yaroslav Lobankov0089af52015-07-02 19:14:40 +0300139 keypair, security_group)
Sean Dague52abbd92016-03-01 09:38:09 -0500140 LOG.info("Booted second instance %s" % instance_2nd)
fujioka yuuichi636f8db2013-08-09 12:05:24 +0900141
142 # check the content of written file
Sean Dague52abbd92016-03-01 09:38:09 -0500143 LOG.info("Getting timestamp in instance %s" % instance_2nd)
Sean Dague20e98612016-01-06 14:33:28 -0500144 ip_instance_2nd = self.get_server_ip(instance_2nd)
Alexander Gubanov59cc3032015-11-05 11:58:03 +0200145 timestamp2 = self.get_timestamp(ip_instance_2nd,
146 private_key=keypair['private_key'])
147 self.assertEqual(timestamp, timestamp2)
fujioka yuuichi636f8db2013-08-09 12:05:24 +0900148
149 # snapshot a volume
Sean Dague52abbd92016-03-01 09:38:09 -0500150 LOG.info("Creating snapshot from volume: %s" % volume_origin['id'])
Joseph Lanouxeef192f2014-08-01 14:32:53 +0000151 snapshot = self._create_snapshot_from_volume(volume_origin['id'])
fujioka yuuichi636f8db2013-08-09 12:05:24 +0900152
153 # create a 3rd instance from snapshot
Sean Dague52abbd92016-03-01 09:38:09 -0500154 LOG.info("Creating third instance from snapshot: %s" % snapshot['id'])
Joseph Lanouxeef192f2014-08-01 14:32:53 +0000155 volume = self._create_volume_from_snapshot(snapshot['id'])
Alexander Gubanovc8829f82015-11-12 10:35:13 +0200156 server_from_snapshot = (
Yaroslav Lobankov0089af52015-07-02 19:14:40 +0300157 self._boot_instance_from_volume(volume['id'],
158 keypair, security_group))
fujioka yuuichi636f8db2013-08-09 12:05:24 +0900159
160 # check the content of written file
Sean Dague52abbd92016-03-01 09:38:09 -0500161 LOG.info("Logging into third instance to get timestamp: %s" %
162 server_from_snapshot)
Sean Dague20e98612016-01-06 14:33:28 -0500163 server_from_snapshot_ip = self.get_server_ip(server_from_snapshot)
Alexander Gubanovc8829f82015-11-12 10:35:13 +0200164 timestamp3 = self.get_timestamp(server_from_snapshot_ip,
Alexander Gubanov59cc3032015-11-05 11:58:03 +0200165 private_key=keypair['private_key'])
166 self.assertEqual(timestamp, timestamp3)
fujioka yuuichi636f8db2013-08-09 12:05:24 +0900167
Andrey Pavlovc8bd4b12015-08-17 10:20:17 +0300168 @test.idempotent_id('36c34c67-7b54-4b59-b188-02a2f458a63b')
169 @test.services('compute', 'volume', 'image')
170 def test_create_ebs_image_and_check_boot(self):
171 # create an instance from volume
172 volume_origin = self._create_volume_from_image()
173 instance = self._boot_instance_from_volume(volume_origin['id'],
174 delete_on_termination=True)
175 # create EBS image
zhuflc6ce5392016-08-17 14:34:37 +0800176 name = data_utils.rand_name(self.__class__.__name__ + '-image')
Andrey Pavlovc8bd4b12015-08-17 10:20:17 +0300177 image = self.create_server_snapshot(instance, name=name)
178
179 # delete instance
180 self._delete_server(instance)
181
182 # boot instance from EBS image
lanoux5fc14522015-09-21 08:17:35 +0000183 instance = self.create_server(
184 image_id=image['id'])
Andrey Pavlovc8bd4b12015-08-17 10:20:17 +0300185 # just ensure that instance booted
186
187 # delete instance
188 self._delete_server(instance)
189
Nikola Dipanov7cff03f2014-03-12 14:06:25 +0100190
191class TestVolumeBootPatternV2(TestVolumeBootPattern):
Andrey Pavlovc8bd4b12015-08-17 10:20:17 +0300192 def _get_bdm(self, vol_id, delete_on_termination=False):
Yaroslav Lobankov0089af52015-07-02 19:14:40 +0300193 bd_map_v2 = [{
194 'uuid': vol_id,
195 'source_type': 'volume',
196 'destination_type': 'volume',
197 'boot_index': 0,
Andrey Pavlovc8bd4b12015-08-17 10:20:17 +0300198 'delete_on_termination': delete_on_termination}]
199 return {'block_device_mapping_v2': bd_map_v2}