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) |
Ken'ichi Ohmichi | 153b3d9 | 2016-08-03 14:35:46 -0700 | [diff] [blame] | 46 | self.admin_backups_client.wait_for_resource_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 | |
ravikumar-venkatesan | 18c1d0e | 2015-03-18 11:28:37 +0000 | [diff] [blame] | 62 | @test.idempotent_id('a99c54a1-dd80-4724-8a13-13bf58d4068d') |
| 63 | def test_volume_backup_export_import(self): |
Gorka Eguileor | 9de20c1 | 2015-08-13 11:16:43 +0200 | [diff] [blame] | 64 | """Test backup export import functionality. |
| 65 | |
| 66 | Cinder allows exporting DB backup information through its API so it can |
| 67 | be imported back in case of a DB loss. |
| 68 | """ |
ravikumar-venkatesan | 18c1d0e | 2015-03-18 11:28:37 +0000 | [diff] [blame] | 69 | # Create backup |
| 70 | backup_name = data_utils.rand_name('Backup') |
bkopilov | 62d2175 | 2016-06-08 10:16:11 +0300 | [diff] [blame] | 71 | backup = (self.admin_backups_client.create_backup( |
Ghanshyam | 4a9e303 | 2015-12-10 17:50:54 +0900 | [diff] [blame] | 72 | volume_id=self.volume['id'], name=backup_name)['backup']) |
ravikumar-venkatesan | 18c1d0e | 2015-03-18 11:28:37 +0000 | [diff] [blame] | 73 | self.addCleanup(self._delete_backup, backup['id']) |
| 74 | self.assertEqual(backup_name, backup['name']) |
bkopilov | 62d2175 | 2016-06-08 10:16:11 +0300 | [diff] [blame] | 75 | self.admin_backups_client.wait_for_backup_status(backup['id'], |
| 76 | 'available') |
ravikumar-venkatesan | 18c1d0e | 2015-03-18 11:28:37 +0000 | [diff] [blame] | 77 | |
| 78 | # Export Backup |
bkopilov | 62d2175 | 2016-06-08 10:16:11 +0300 | [diff] [blame] | 79 | export_backup = (self.admin_backups_client.export_backup(backup['id']) |
John Warren | 6cadeee | 2015-08-13 17:00:56 +0000 | [diff] [blame] | 80 | ['backup-record']) |
ravikumar-venkatesan | 18c1d0e | 2015-03-18 11:28:37 +0000 | [diff] [blame] | 81 | self.assertIn('backup_service', export_backup) |
| 82 | self.assertIn('backup_url', export_backup) |
| 83 | self.assertTrue(export_backup['backup_service'].startswith( |
| 84 | 'cinder.backup.drivers')) |
| 85 | self.assertIsNotNone(export_backup['backup_url']) |
| 86 | |
Gorka Eguileor | 9de20c1 | 2015-08-13 11:16:43 +0200 | [diff] [blame] | 87 | # NOTE(geguileo): Backups are imported with the same backup id |
| 88 | # (important for incremental backups among other things), so we cannot |
| 89 | # import the exported backup information as it is, because that Backup |
| 90 | # ID already exists. So we'll fake the data by changing the backup id |
| 91 | # in the exported backup DB info we have retrieved before importing it |
| 92 | # back. |
| 93 | new_id = data_utils.rand_uuid() |
| 94 | new_url = self._modify_backup_url( |
| 95 | export_backup['backup_url'], {'id': new_id}) |
| 96 | |
ravikumar-venkatesan | 18c1d0e | 2015-03-18 11:28:37 +0000 | [diff] [blame] | 97 | # Import Backup |
bkopilov | 62d2175 | 2016-06-08 10:16:11 +0300 | [diff] [blame] | 98 | import_backup = self.admin_backups_client.import_backup( |
ravikumar-venkatesan | 18c1d0e | 2015-03-18 11:28:37 +0000 | [diff] [blame] | 99 | backup_service=export_backup['backup_service'], |
Gorka Eguileor | 9de20c1 | 2015-08-13 11:16:43 +0200 | [diff] [blame] | 100 | backup_url=new_url)['backup'] |
| 101 | |
| 102 | # NOTE(geguileo): We delete both backups, but only one of those |
| 103 | # deletions will delete data from the backup back-end because they |
| 104 | # were both pointing to the same backend data. |
| 105 | self.addCleanup(self._delete_backup, new_id) |
ravikumar-venkatesan | 18c1d0e | 2015-03-18 11:28:37 +0000 | [diff] [blame] | 106 | self.assertIn("id", import_backup) |
Gorka Eguileor | 9de20c1 | 2015-08-13 11:16:43 +0200 | [diff] [blame] | 107 | self.assertEqual(new_id, import_backup['id']) |
bkopilov | 62d2175 | 2016-06-08 10:16:11 +0300 | [diff] [blame] | 108 | self.admin_backups_client.wait_for_backup_status(import_backup['id'], |
| 109 | 'available') |
ravikumar-venkatesan | 18c1d0e | 2015-03-18 11:28:37 +0000 | [diff] [blame] | 110 | |
| 111 | # Verify Import Backup |
bkopilov | 62d2175 | 2016-06-08 10:16:11 +0300 | [diff] [blame] | 112 | backups = self.admin_backups_client.list_backups( |
| 113 | detail=True)['backups'] |
Gorka Eguileor | 9de20c1 | 2015-08-13 11:16:43 +0200 | [diff] [blame] | 114 | self.assertIn(new_id, [b['id'] for b in backups]) |
ravikumar-venkatesan | 18c1d0e | 2015-03-18 11:28:37 +0000 | [diff] [blame] | 115 | |
| 116 | # Restore backup |
bkopilov | 62d2175 | 2016-06-08 10:16:11 +0300 | [diff] [blame] | 117 | restore = self.admin_backups_client.restore_backup( |
Gorka Eguileor | 9de20c1 | 2015-08-13 11:16:43 +0200 | [diff] [blame] | 118 | backup['id'])['restore'] |
ravikumar-venkatesan | 18c1d0e | 2015-03-18 11:28:37 +0000 | [diff] [blame] | 119 | self.addCleanup(self.admin_volume_client.delete_volume, |
| 120 | restore['volume_id']) |
Gorka Eguileor | 9de20c1 | 2015-08-13 11:16:43 +0200 | [diff] [blame] | 121 | self.assertEqual(backup['id'], restore['backup_id']) |
Yaroslav Lobankov | ed3a35b | 2016-03-24 22:41:30 -0500 | [diff] [blame] | 122 | waiters.wait_for_volume_status(self.admin_volume_client, |
| 123 | restore['volume_id'], 'available') |
ravikumar-venkatesan | 18c1d0e | 2015-03-18 11:28:37 +0000 | [diff] [blame] | 124 | |
| 125 | # Verify if restored volume is there in volume list |
John Warren | 6177c9e | 2015-08-19 20:00:17 +0000 | [diff] [blame] | 126 | volumes = self.admin_volume_client.list_volumes()['volumes'] |
ravikumar-venkatesan | 18c1d0e | 2015-03-18 11:28:37 +0000 | [diff] [blame] | 127 | self.assertIn(restore['volume_id'], [v['id'] for v in volumes]) |
bkopilov | 62d2175 | 2016-06-08 10:16:11 +0300 | [diff] [blame] | 128 | self.admin_backups_client.wait_for_backup_status(import_backup['id'], |
| 129 | 'available') |
ravikumar-venkatesan | 18c1d0e | 2015-03-18 11:28:37 +0000 | [diff] [blame] | 130 | |
bkopilov | 19fc5fd | 2016-07-06 12:02:18 +0300 | [diff] [blame] | 131 | @test.idempotent_id('47a35425-a891-4e13-961c-c45deea21e94') |
| 132 | def test_volume_backup_reset_status(self): |
| 133 | # Create a backup |
| 134 | backup_name = data_utils.rand_name('Backup') |
| 135 | backup = self.admin_backups_client.create_backup( |
| 136 | volume_id=self.volume['id'], name=backup_name)['backup'] |
| 137 | self.addCleanup(self.admin_backups_client.delete_backup, |
| 138 | backup['id']) |
| 139 | self.assertEqual(backup_name, backup['name']) |
| 140 | self.admin_backups_client.wait_for_backup_status(backup['id'], |
| 141 | 'available') |
| 142 | # Reset backup status to error |
| 143 | self.admin_backups_client.reset_backup_status(backup_id=backup['id'], |
| 144 | status="error") |
| 145 | self.admin_backups_client.wait_for_backup_status(backup['id'], |
| 146 | 'error') |
| 147 | |
jun xie | ebc3da3 | 2014-11-18 14:34:56 +0800 | [diff] [blame] | 148 | |
lkuchlan | 9dea88e | 2016-06-07 17:12:01 +0300 | [diff] [blame] | 149 | class VolumesBackupsAdminV1Test(VolumesBackupsAdminV2Test): |
jun xie | ebc3da3 | 2014-11-18 14:34:56 +0800 | [diff] [blame] | 150 | _api_version = 1 |