blob: 18553864d9069f2e6fba5bc96a2413ab56c35ef4 [file] [log] [blame]
Attila Fazekas36b1fcf2013-01-31 16:41:04 +01001# 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
Benny Kopilovcdcf53c2017-03-29 07:25:05 +030013import testtools
lkuchland8277372017-02-22 15:07:52 +020014from testtools import matchers
15
Sean Dague1937d092013-05-17 16:36:38 -040016from tempest.api.volume import base
Andrea Frittolicd368412017-08-14 21:37:56 +010017from tempest.common import utils
jeremy.zhangd1be5012018-06-06 17:25:39 +080018from tempest.common import waiters
Xiao Chen47fcbf42014-01-13 16:42:41 +080019from tempest import config
Ken'ichi Ohmichief1c1ce2017-03-10 11:07:10 -080020from tempest.lib.common.utils import data_utils
Ken'ichi Ohmichi6b279c72017-01-27 18:26:59 -080021from tempest.lib import decorators
lianghao6c011992017-04-24 20:50:59 +080022from tempest.lib import exceptions as lib_exc
Attila Fazekas36b1fcf2013-01-31 16:41:04 +010023
Xiao Chen47fcbf42014-01-13 16:42:41 +080024CONF = config.CONF
Giulio Fidente3a465e32013-05-07 13:38:18 +020025
Attila Fazekas36b1fcf2013-01-31 16:41:04 +010026
Ken'ichi Ohmichie8afb8c2017-03-27 11:25:37 -070027class VolumesSnapshotTestJSON(base.BaseVolumeTest):
Attila Fazekas36b1fcf2013-01-31 16:41:04 +010028
Giulio Fidente73332932013-05-03 18:04:09 +020029 @classmethod
Rohan Kanade05749152015-01-30 17:15:18 +053030 def skip_checks(cls):
Ken'ichi Ohmichie8afb8c2017-03-27 11:25:37 -070031 super(VolumesSnapshotTestJSON, cls).skip_checks()
Rohan Kanade05749152015-01-30 17:15:18 +053032 if not CONF.volume_feature_enabled.snapshot:
33 raise cls.skipException("Cinder volume snapshots are disabled")
34
35 @classmethod
Andrea Frittoli61a12e22014-09-15 13:14:54 +010036 def resource_setup(cls):
Ken'ichi Ohmichie8afb8c2017-03-27 11:25:37 -070037 super(VolumesSnapshotTestJSON, cls).resource_setup()
Zhi Kun Liu43f9af12014-03-19 21:01:35 +080038 cls.volume_origin = cls.create_volume()
Giulio Fidente73332932013-05-03 18:04:09 +020039
zhufl529eefa2017-05-12 16:00:32 +080040 @decorators.idempotent_id('8567b54c-4455-446d-a1cf-651ddeaa3ff2')
Andrea Frittolicd368412017-08-14 21:37:56 +010041 @utils.services('compute')
zhufl529eefa2017-05-12 16:00:32 +080042 def test_snapshot_create_delete_with_volume_in_use(self):
Xiao Chen47fcbf42014-01-13 16:42:41 +080043 # Create a test instance
lkuchland4ecd0e2017-06-11 12:01:27 +030044 server = self.create_server()
zhufla2bffbd2018-03-02 15:04:06 +080045 # NOTE(zhufl) Here we create volume from self.image_ref for adding
46 # coverage for "creating snapshot from non-blank volume".
47 volume = self.create_volume(image_ref=self.image_ref)
48 self.attach_volume(server['id'], volume['id'])
Erlon R. Cruzba19bc72016-09-28 14:32:11 -030049
lianghao6c011992017-04-24 20:50:59 +080050 # Snapshot a volume which attached to an instance with force=False
51 self.assertRaises(lib_exc.BadRequest, self.create_snapshot,
zhufla2bffbd2018-03-02 15:04:06 +080052 volume['id'], force=False)
lianghao6c011992017-04-24 20:50:59 +080053
Erlon R. Cruzba19bc72016-09-28 14:32:11 -030054 # Snapshot a volume attached to an instance
zhufla2bffbd2018-03-02 15:04:06 +080055 snapshot1 = self.create_snapshot(volume['id'], force=True)
56 snapshot2 = self.create_snapshot(volume['id'], force=True)
57 snapshot3 = self.create_snapshot(volume['id'], force=True)
Erlon R. Cruzba19bc72016-09-28 14:32:11 -030058
59 # Delete the snapshots. Some snapshot implementations can take
60 # different paths according to order they are deleted.
lkuchlan5b2b3622017-02-14 15:48:36 +020061 self.delete_snapshot(snapshot1['id'])
62 self.delete_snapshot(snapshot3['id'])
63 self.delete_snapshot(snapshot2['id'])
Erlon R. Cruzba19bc72016-09-28 14:32:11 -030064
Ken'ichi Ohmichi6b279c72017-01-27 18:26:59 -080065 @decorators.idempotent_id('5210a1de-85a0-11e6-bb21-641c676a5d61')
Andrea Frittolicd368412017-08-14 21:37:56 +010066 @utils.services('compute')
Erlon R. Cruzba19bc72016-09-28 14:32:11 -030067 def test_snapshot_create_offline_delete_online(self):
68
69 # Create a snapshot while it is not attached
70 snapshot1 = self.create_snapshot(self.volume_origin['id'])
71
72 # Create a server and attach it
lkuchland4ecd0e2017-06-11 12:01:27 +030073 server = self.create_server()
Erlon R. Cruzba19bc72016-09-28 14:32:11 -030074 self.attach_volume(server['id'], self.volume_origin['id'])
75
76 # Now that the volume is attached, create another snapshots
77 snapshot2 = self.create_snapshot(self.volume_origin['id'], force=True)
78 snapshot3 = self.create_snapshot(self.volume_origin['id'], force=True)
79
80 # Delete the snapshots. Some snapshot implementations can take
81 # different paths according to order they are deleted.
lkuchlan5b2b3622017-02-14 15:48:36 +020082 self.delete_snapshot(snapshot3['id'])
83 self.delete_snapshot(snapshot1['id'])
84 self.delete_snapshot(snapshot2['id'])
Erlon R. Cruzba19bc72016-09-28 14:32:11 -030085
Ken'ichi Ohmichi6b279c72017-01-27 18:26:59 -080086 @decorators.idempotent_id('2a8abbe4-d871-46db-b049-c41f5af8216e')
QingXin Mengdc95f5e2013-09-16 19:06:44 -070087 def test_snapshot_create_get_list_update_delete(self):
lkuchland8277372017-02-22 15:07:52 +020088 # Create a snapshot with metadata
89 metadata = {"snap-meta1": "value1",
90 "snap-meta2": "value2",
91 "snap-meta3": "value3"}
92 snapshot = self.create_snapshot(self.volume_origin['id'],
93 metadata=metadata)
Giulio Fidente73332932013-05-03 18:04:09 +020094
Giulio Fidentef41b8ee2013-05-21 11:07:21 +020095 # Get the snap and check for some of its details
John Warrenff7faf62015-08-17 16:59:06 +000096 snap_get = self.snapshots_client.show_snapshot(
97 snapshot['id'])['snapshot']
Giulio Fidentef41b8ee2013-05-21 11:07:21 +020098 self.assertEqual(self.volume_origin['id'],
99 snap_get['volume_id'],
100 "Referred volume origin mismatch")
jeremy.zhang681dff82017-04-06 19:21:01 +0800101 self.assertEqual(self.volume_origin['size'], snap_get['size'])
Giulio Fidentef41b8ee2013-05-21 11:07:21 +0200102
lkuchland8277372017-02-22 15:07:52 +0200103 # Verify snapshot metadata
104 self.assertThat(snap_get['metadata'].items(),
105 matchers.ContainsAll(metadata.items()))
106
Giulio Fidentef41b8ee2013-05-21 11:07:21 +0200107 # Compare also with the output from the list action
zhufla57530c2017-03-23 11:38:12 +0800108 tracking_data = (snapshot['id'], snapshot['name'])
John Warrenff7faf62015-08-17 16:59:06 +0000109 snaps_list = self.snapshots_client.list_snapshots()['snapshots']
zhufla57530c2017-03-23 11:38:12 +0800110 snaps_data = [(f['id'], f['name']) for f in snaps_list]
Giulio Fidentef41b8ee2013-05-21 11:07:21 +0200111 self.assertIn(tracking_data, snaps_data)
112
QingXin Mengdc95f5e2013-09-16 19:06:44 -0700113 # Updates snapshot with new values
zhuflc6ce5392016-08-17 14:34:37 +0800114 new_s_name = data_utils.rand_name(
115 self.__class__.__name__ + '-new-snap')
QingXin Mengdc95f5e2013-09-16 19:06:44 -0700116 new_desc = 'This is the new description of snapshot.'
zhufla57530c2017-03-23 11:38:12 +0800117 params = {'name': new_s_name,
118 'description': new_desc}
John Warrenff7faf62015-08-17 16:59:06 +0000119 update_snapshot = self.snapshots_client.update_snapshot(
120 snapshot['id'], **params)['snapshot']
QingXin Mengdc95f5e2013-09-16 19:06:44 -0700121 # Assert response body for update_snapshot method
zhufla57530c2017-03-23 11:38:12 +0800122 self.assertEqual(new_s_name, update_snapshot['name'])
123 self.assertEqual(new_desc, update_snapshot['description'])
Ken'ichi Ohmichi35798fb2015-04-06 01:22:41 +0000124 # Assert response body for show_snapshot method
John Warrenff7faf62015-08-17 16:59:06 +0000125 updated_snapshot = self.snapshots_client.show_snapshot(
126 snapshot['id'])['snapshot']
zhufla57530c2017-03-23 11:38:12 +0800127 self.assertEqual(new_s_name, updated_snapshot['name'])
128 self.assertEqual(new_desc, updated_snapshot['description'])
QingXin Mengdc95f5e2013-09-16 19:06:44 -0700129
Giulio Fidentef41b8ee2013-05-21 11:07:21 +0200130 # Delete the snapshot
lkuchlan5b2b3622017-02-14 15:48:36 +0200131 self.delete_snapshot(snapshot['id'])
Giulio Fidentef41b8ee2013-05-21 11:07:21 +0200132
lkuchlan91d5fe02018-09-02 15:52:39 +0300133 def _create_volume_from_snapshot(self, extra_size=0):
134 src_size = CONF.volume.volume_size
135 size = src_size + extra_size
136
137 src_vol = self.create_volume(size=src_size)
138 src_snap = self.create_snapshot(src_vol['id'])
139
140 dst_vol = self.create_volume(snapshot_id=src_snap['id'],
141 size=size)
142 # NOTE(zhufl): dst_vol is created based on snapshot, so dst_vol
143 # should be deleted before deleting snapshot, otherwise deleting
144 # snapshot will end with status 'error-deleting'. This depends on
145 # the implementation mechanism of vendors, generally speaking,
146 # some verdors will use "virtual disk clone" which will promote
147 # disk clone speed, and in this situation the "disk clone"
148 # is just a relationship between volume and snapshot.
149 self.addCleanup(self.delete_volume, self.volumes_client, dst_vol['id'])
150
151 volume = self.volumes_client.show_volume(dst_vol['id'])['volume']
152 # Should allow
153 self.assertEqual(volume['snapshot_id'], src_snap['id'])
154 self.assertEqual(volume['size'], size)
155
Ken'ichi Ohmichi6b279c72017-01-27 18:26:59 -0800156 @decorators.idempotent_id('677863d1-3142-456d-b6ac-9924f667a7f4')
Giulio Fidente73332932013-05-03 18:04:09 +0200157 def test_volume_from_snapshot(self):
jeremy.zhang681dff82017-04-06 19:21:01 +0800158 # Creates a volume from a snapshot passing a size
159 # different from the source
lkuchlan91d5fe02018-09-02 15:52:39 +0300160 self._create_volume_from_snapshot(extra_size=1)
Benny Kopilovcdcf53c2017-03-29 07:25:05 +0300161
Sean McGinnisfde21672018-08-01 09:08:21 -0500162 @decorators.idempotent_id('053d8870-8282-4fff-9dbb-99cb58bb5e0a')
163 def test_volume_from_snapshot_no_size(self):
164 # Creates a volume from a snapshot defaulting to original size
lkuchlan91d5fe02018-09-02 15:52:39 +0300165 self._create_volume_from_snapshot()
Sean McGinnisfde21672018-08-01 09:08:21 -0500166
Benny Kopilovcdcf53c2017-03-29 07:25:05 +0300167 @decorators.idempotent_id('bbcfa285-af7f-479e-8c1a-8c34fc16543c')
168 @testtools.skipUnless(CONF.volume_feature_enabled.backup,
169 "Cinder backup is disabled")
170 def test_snapshot_backup(self):
171 # Create a snapshot
172 snapshot = self.create_snapshot(volume_id=self.volume_origin['id'])
173
174 backup = self.create_backup(volume_id=self.volume_origin['id'],
175 snapshot_id=snapshot['id'])
jeremy.zhangd1be5012018-06-06 17:25:39 +0800176 waiters.wait_for_volume_resource_status(self.snapshots_client,
177 snapshot['id'], 'available')
Benny Kopilovcdcf53c2017-03-29 07:25:05 +0300178 backup_info = self.backups_client.show_backup(backup['id'])['backup']
179 self.assertEqual(self.volume_origin['id'], backup_info['volume_id'])
180 self.assertEqual(snapshot['id'], backup_info['snapshot_id'])