Giulio Fidente | 74b08ad | 2014-01-18 04:02:51 +0100 | [diff] [blame] | 1 | # Copyright 2014 OpenStack Foundation |
| 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 | |
Gorka Eguileor | 9de20c1 | 2015-08-13 11:16:43 +0200 | [diff] [blame] | 16 | import base64 |
| 17 | import six |
| 18 | |
| 19 | from oslo_serialization import jsonutils as json |
| 20 | |
Masayuki Igawa | 1edf94f | 2014-03-04 18:34:16 +0900 | [diff] [blame] | 21 | from tempest.api.volume import base |
Fei Long Wang | d39431f | 2015-05-14 11:30:48 +1200 | [diff] [blame] | 22 | from tempest.common.utils import data_utils |
Yaroslav Lobankov | ed3a35b | 2016-03-24 22:41:30 -0500 | [diff] [blame] | 23 | from tempest.common import waiters |
Giulio Fidente | 74b08ad | 2014-01-18 04:02:51 +0100 | [diff] [blame] | 24 | from tempest import config |
Giulio Fidente | 74b08ad | 2014-01-18 04:02:51 +0100 | [diff] [blame] | 25 | from tempest import test |
| 26 | |
| 27 | CONF = config.CONF |
Giulio Fidente | 74b08ad | 2014-01-18 04:02:51 +0100 | [diff] [blame] | 28 | |
| 29 | |
lkuchlan | 9dea88e | 2016-06-07 17:12:01 +0300 | [diff] [blame] | 30 | class VolumesBackupsAdminV2Test(base.BaseVolumeAdminTest): |
Giulio Fidente | 74b08ad | 2014-01-18 04:02:51 +0100 | [diff] [blame] | 31 | |
| 32 | @classmethod |
Rohan Kanade | 0574915 | 2015-01-30 17:15:18 +0530 | [diff] [blame] | 33 | def skip_checks(cls): |
lkuchlan | 9dea88e | 2016-06-07 17:12:01 +0300 | [diff] [blame] | 34 | super(VolumesBackupsAdminV2Test, cls).skip_checks() |
Giulio Fidente | 74b08ad | 2014-01-18 04:02:51 +0100 | [diff] [blame] | 35 | if not CONF.volume_feature_enabled.backup: |
| 36 | raise cls.skipException("Cinder backup feature disabled") |
| 37 | |
Rohan Kanade | 0574915 | 2015-01-30 17:15:18 +0530 | [diff] [blame] | 38 | @classmethod |
| 39 | def resource_setup(cls): |
lkuchlan | 9dea88e | 2016-06-07 17:12:01 +0300 | [diff] [blame] | 40 | super(VolumesBackupsAdminV2Test, cls).resource_setup() |
Rohan Kanade | 0574915 | 2015-01-30 17:15:18 +0530 | [diff] [blame] | 41 | |
Giulio Fidente | 74b08ad | 2014-01-18 04:02:51 +0100 | [diff] [blame] | 42 | cls.volume = cls.create_volume() |
| 43 | |
ravikumar-venkatesan | 18c1d0e | 2015-03-18 11:28:37 +0000 | [diff] [blame] | 44 | def _delete_backup(self, backup_id): |
bkopilov | 62d2175 | 2016-06-08 10:16:11 +0300 | [diff] [blame] | 45 | self.admin_backups_client.delete_backup(backup_id) |
| 46 | self.admin_backups_client.wait_for_backup_deletion(backup_id) |
ravikumar-venkatesan | 18c1d0e | 2015-03-18 11:28:37 +0000 | [diff] [blame] | 47 | |
Gorka Eguileor | 9de20c1 | 2015-08-13 11:16:43 +0200 | [diff] [blame] | 48 | def _decode_url(self, backup_url): |
| 49 | return json.loads(base64.decodestring(backup_url)) |
| 50 | |
| 51 | def _encode_backup(self, backup): |
| 52 | retval = json.dumps(backup) |
| 53 | if six.PY3: |
| 54 | retval = retval.encode('utf-8') |
| 55 | return base64.encodestring(retval) |
| 56 | |
| 57 | def _modify_backup_url(self, backup_url, changes): |
| 58 | backup = self._decode_url(backup_url) |
| 59 | backup.update(changes) |
| 60 | return self._encode_backup(backup) |
| 61 | |
Chris Hoge | 7579c1a | 2015-02-26 14:12:15 -0800 | [diff] [blame] | 62 | @test.idempotent_id('a66eb488-8ee1-47d4-8e9f-575a095728c6') |
raiesmh08 | f04da34 | 2014-02-28 17:14:43 +0530 | [diff] [blame] | 63 | def test_volume_backup_create_get_detailed_list_restore_delete(self): |
| 64 | # Create backup |
Giulio Fidente | 74b08ad | 2014-01-18 04:02:51 +0100 | [diff] [blame] | 65 | backup_name = data_utils.rand_name('Backup') |
bkopilov | 62d2175 | 2016-06-08 10:16:11 +0300 | [diff] [blame] | 66 | create_backup = self.admin_backups_client.create_backup |
Ghanshyam | 4a9e303 | 2015-12-10 17:50:54 +0900 | [diff] [blame] | 67 | backup = create_backup(volume_id=self.volume['id'], |
John Warren | 6cadeee | 2015-08-13 17:00:56 +0000 | [diff] [blame] | 68 | name=backup_name)['backup'] |
bkopilov | 62d2175 | 2016-06-08 10:16:11 +0300 | [diff] [blame] | 69 | self.addCleanup(self.admin_backups_client.delete_backup, |
Giulio Fidente | 74b08ad | 2014-01-18 04:02:51 +0100 | [diff] [blame] | 70 | backup['id']) |
raiesmh08 | f04da34 | 2014-02-28 17:14:43 +0530 | [diff] [blame] | 71 | self.assertEqual(backup_name, backup['name']) |
Yaroslav Lobankov | ed3a35b | 2016-03-24 22:41:30 -0500 | [diff] [blame] | 72 | waiters.wait_for_volume_status(self.admin_volume_client, |
| 73 | self.volume['id'], 'available') |
bkopilov | 62d2175 | 2016-06-08 10:16:11 +0300 | [diff] [blame] | 74 | self.admin_backups_client.wait_for_backup_status(backup['id'], |
| 75 | 'available') |
Giulio Fidente | 74b08ad | 2014-01-18 04:02:51 +0100 | [diff] [blame] | 76 | |
raiesmh08 | f04da34 | 2014-02-28 17:14:43 +0530 | [diff] [blame] | 77 | # Get a given backup |
bkopilov | 62d2175 | 2016-06-08 10:16:11 +0300 | [diff] [blame] | 78 | backup = self.admin_backups_client.show_backup(backup['id'])['backup'] |
raiesmh08 | f04da34 | 2014-02-28 17:14:43 +0530 | [diff] [blame] | 79 | self.assertEqual(backup_name, backup['name']) |
Giulio Fidente | 74b08ad | 2014-01-18 04:02:51 +0100 | [diff] [blame] | 80 | |
raiesmh08 | f04da34 | 2014-02-28 17:14:43 +0530 | [diff] [blame] | 81 | # Get all backups with detail |
bkopilov | 62d2175 | 2016-06-08 10:16:11 +0300 | [diff] [blame] | 82 | backups = self.admin_backups_client.list_backups( |
| 83 | detail=True)['backups'] |
raiesmh08 | f04da34 | 2014-02-28 17:14:43 +0530 | [diff] [blame] | 84 | self.assertIn((backup['name'], backup['id']), |
| 85 | [(m['name'], m['id']) for m in backups]) |
| 86 | |
| 87 | # Restore backup |
bkopilov | 62d2175 | 2016-06-08 10:16:11 +0300 | [diff] [blame] | 88 | restore = self.admin_backups_client.restore_backup( |
John Warren | 6cadeee | 2015-08-13 17:00:56 +0000 | [diff] [blame] | 89 | backup['id'])['restore'] |
raiesmh08 | f04da34 | 2014-02-28 17:14:43 +0530 | [diff] [blame] | 90 | |
| 91 | # Delete backup |
Chandan Kumar | ee3f4bd | 2014-10-29 23:09:29 +0530 | [diff] [blame] | 92 | self.addCleanup(self.admin_volume_client.delete_volume, |
Giulio Fidente | 74b08ad | 2014-01-18 04:02:51 +0100 | [diff] [blame] | 93 | restore['volume_id']) |
raiesmh08 | f04da34 | 2014-02-28 17:14:43 +0530 | [diff] [blame] | 94 | self.assertEqual(backup['id'], restore['backup_id']) |
bkopilov | 62d2175 | 2016-06-08 10:16:11 +0300 | [diff] [blame] | 95 | self.admin_backups_client.wait_for_backup_status(backup['id'], |
| 96 | 'available') |
Yaroslav Lobankov | ed3a35b | 2016-03-24 22:41:30 -0500 | [diff] [blame] | 97 | waiters.wait_for_volume_status(self.admin_volume_client, |
| 98 | restore['volume_id'], 'available') |
jun xie | ebc3da3 | 2014-11-18 14:34:56 +0800 | [diff] [blame] | 99 | |
ravikumar-venkatesan | 18c1d0e | 2015-03-18 11:28:37 +0000 | [diff] [blame] | 100 | @test.idempotent_id('a99c54a1-dd80-4724-8a13-13bf58d4068d') |
| 101 | def test_volume_backup_export_import(self): |
Gorka Eguileor | 9de20c1 | 2015-08-13 11:16:43 +0200 | [diff] [blame] | 102 | """Test backup export import functionality. |
| 103 | |
| 104 | Cinder allows exporting DB backup information through its API so it can |
| 105 | be imported back in case of a DB loss. |
| 106 | """ |
ravikumar-venkatesan | 18c1d0e | 2015-03-18 11:28:37 +0000 | [diff] [blame] | 107 | # Create backup |
| 108 | backup_name = data_utils.rand_name('Backup') |
bkopilov | 62d2175 | 2016-06-08 10:16:11 +0300 | [diff] [blame] | 109 | backup = (self.admin_backups_client.create_backup( |
Ghanshyam | 4a9e303 | 2015-12-10 17:50:54 +0900 | [diff] [blame] | 110 | volume_id=self.volume['id'], name=backup_name)['backup']) |
ravikumar-venkatesan | 18c1d0e | 2015-03-18 11:28:37 +0000 | [diff] [blame] | 111 | self.addCleanup(self._delete_backup, backup['id']) |
| 112 | self.assertEqual(backup_name, backup['name']) |
bkopilov | 62d2175 | 2016-06-08 10:16:11 +0300 | [diff] [blame] | 113 | self.admin_backups_client.wait_for_backup_status(backup['id'], |
| 114 | 'available') |
ravikumar-venkatesan | 18c1d0e | 2015-03-18 11:28:37 +0000 | [diff] [blame] | 115 | |
| 116 | # Export Backup |
bkopilov | 62d2175 | 2016-06-08 10:16:11 +0300 | [diff] [blame] | 117 | export_backup = (self.admin_backups_client.export_backup(backup['id']) |
John Warren | 6cadeee | 2015-08-13 17:00:56 +0000 | [diff] [blame] | 118 | ['backup-record']) |
ravikumar-venkatesan | 18c1d0e | 2015-03-18 11:28:37 +0000 | [diff] [blame] | 119 | self.assertIn('backup_service', export_backup) |
| 120 | self.assertIn('backup_url', export_backup) |
| 121 | self.assertTrue(export_backup['backup_service'].startswith( |
| 122 | 'cinder.backup.drivers')) |
| 123 | self.assertIsNotNone(export_backup['backup_url']) |
| 124 | |
Gorka Eguileor | 9de20c1 | 2015-08-13 11:16:43 +0200 | [diff] [blame] | 125 | # NOTE(geguileo): Backups are imported with the same backup id |
| 126 | # (important for incremental backups among other things), so we cannot |
| 127 | # import the exported backup information as it is, because that Backup |
| 128 | # ID already exists. So we'll fake the data by changing the backup id |
| 129 | # in the exported backup DB info we have retrieved before importing it |
| 130 | # back. |
| 131 | new_id = data_utils.rand_uuid() |
| 132 | new_url = self._modify_backup_url( |
| 133 | export_backup['backup_url'], {'id': new_id}) |
| 134 | |
ravikumar-venkatesan | 18c1d0e | 2015-03-18 11:28:37 +0000 | [diff] [blame] | 135 | # Import Backup |
bkopilov | 62d2175 | 2016-06-08 10:16:11 +0300 | [diff] [blame] | 136 | import_backup = self.admin_backups_client.import_backup( |
ravikumar-venkatesan | 18c1d0e | 2015-03-18 11:28:37 +0000 | [diff] [blame] | 137 | backup_service=export_backup['backup_service'], |
Gorka Eguileor | 9de20c1 | 2015-08-13 11:16:43 +0200 | [diff] [blame] | 138 | backup_url=new_url)['backup'] |
| 139 | |
| 140 | # NOTE(geguileo): We delete both backups, but only one of those |
| 141 | # deletions will delete data from the backup back-end because they |
| 142 | # were both pointing to the same backend data. |
| 143 | self.addCleanup(self._delete_backup, new_id) |
ravikumar-venkatesan | 18c1d0e | 2015-03-18 11:28:37 +0000 | [diff] [blame] | 144 | self.assertIn("id", import_backup) |
Gorka Eguileor | 9de20c1 | 2015-08-13 11:16:43 +0200 | [diff] [blame] | 145 | self.assertEqual(new_id, import_backup['id']) |
bkopilov | 62d2175 | 2016-06-08 10:16:11 +0300 | [diff] [blame] | 146 | self.admin_backups_client.wait_for_backup_status(import_backup['id'], |
| 147 | 'available') |
ravikumar-venkatesan | 18c1d0e | 2015-03-18 11:28:37 +0000 | [diff] [blame] | 148 | |
| 149 | # Verify Import Backup |
bkopilov | 62d2175 | 2016-06-08 10:16:11 +0300 | [diff] [blame] | 150 | backups = self.admin_backups_client.list_backups( |
| 151 | detail=True)['backups'] |
Gorka Eguileor | 9de20c1 | 2015-08-13 11:16:43 +0200 | [diff] [blame] | 152 | self.assertIn(new_id, [b['id'] for b in backups]) |
ravikumar-venkatesan | 18c1d0e | 2015-03-18 11:28:37 +0000 | [diff] [blame] | 153 | |
| 154 | # Restore backup |
bkopilov | 62d2175 | 2016-06-08 10:16:11 +0300 | [diff] [blame] | 155 | restore = self.admin_backups_client.restore_backup( |
Gorka Eguileor | 9de20c1 | 2015-08-13 11:16:43 +0200 | [diff] [blame] | 156 | backup['id'])['restore'] |
ravikumar-venkatesan | 18c1d0e | 2015-03-18 11:28:37 +0000 | [diff] [blame] | 157 | self.addCleanup(self.admin_volume_client.delete_volume, |
| 158 | restore['volume_id']) |
Gorka Eguileor | 9de20c1 | 2015-08-13 11:16:43 +0200 | [diff] [blame] | 159 | self.assertEqual(backup['id'], restore['backup_id']) |
Yaroslav Lobankov | ed3a35b | 2016-03-24 22:41:30 -0500 | [diff] [blame] | 160 | waiters.wait_for_volume_status(self.admin_volume_client, |
| 161 | restore['volume_id'], 'available') |
ravikumar-venkatesan | 18c1d0e | 2015-03-18 11:28:37 +0000 | [diff] [blame] | 162 | |
| 163 | # Verify if restored volume is there in volume list |
John Warren | 6177c9e | 2015-08-19 20:00:17 +0000 | [diff] [blame] | 164 | volumes = self.admin_volume_client.list_volumes()['volumes'] |
ravikumar-venkatesan | 18c1d0e | 2015-03-18 11:28:37 +0000 | [diff] [blame] | 165 | self.assertIn(restore['volume_id'], [v['id'] for v in volumes]) |
bkopilov | 62d2175 | 2016-06-08 10:16:11 +0300 | [diff] [blame] | 166 | self.admin_backups_client.wait_for_backup_status(import_backup['id'], |
| 167 | 'available') |
ravikumar-venkatesan | 18c1d0e | 2015-03-18 11:28:37 +0000 | [diff] [blame] | 168 | |
bkopilov | 19fc5fd | 2016-07-06 12:02:18 +0300 | [diff] [blame] | 169 | @test.idempotent_id('47a35425-a891-4e13-961c-c45deea21e94') |
| 170 | def test_volume_backup_reset_status(self): |
| 171 | # Create a backup |
| 172 | backup_name = data_utils.rand_name('Backup') |
| 173 | backup = self.admin_backups_client.create_backup( |
| 174 | volume_id=self.volume['id'], name=backup_name)['backup'] |
| 175 | self.addCleanup(self.admin_backups_client.delete_backup, |
| 176 | backup['id']) |
| 177 | self.assertEqual(backup_name, backup['name']) |
| 178 | self.admin_backups_client.wait_for_backup_status(backup['id'], |
| 179 | 'available') |
| 180 | # Reset backup status to error |
| 181 | self.admin_backups_client.reset_backup_status(backup_id=backup['id'], |
| 182 | status="error") |
| 183 | self.admin_backups_client.wait_for_backup_status(backup['id'], |
| 184 | 'error') |
| 185 | |
jun xie | ebc3da3 | 2014-11-18 14:34:56 +0800 | [diff] [blame] | 186 | |
lkuchlan | 9dea88e | 2016-06-07 17:12:01 +0300 | [diff] [blame] | 187 | class VolumesBackupsAdminV1Test(VolumesBackupsAdminV2Test): |
jun xie | ebc3da3 | 2014-11-18 14:34:56 +0800 | [diff] [blame] | 188 | _api_version = 1 |