Fix snapshot status is always backing-up
This patch adds a snapshot clean operation in the cleanup_incomplete_backup_operations process of c-bak. Co-Authored-by: Christian Rohmann <christian.rohmann@inovex.de> Closes-Bug: #1938488 Change-Id: Ifa3d572139fc37c94e3b50a02e61c9818a1b6501
This commit is contained in:
		 zhaoyixin
					zhaoyixin
				
			
				
					committed by
					
						 Christian Rohmann
						Christian Rohmann
					
				
			
			
				
	
			
			
			 Christian Rohmann
						Christian Rohmann
					
				
			
						parent
						
							f3a63ed6fa
						
					
				
				
					commit
					7f819854d2
				
			| @@ -233,11 +233,29 @@ class BackupManager(manager.SchedulerDependentManager): | |||||||
|             self.db.volume_update(ctxt, volume['id'], |             self.db.volume_update(ctxt, volume['id'], | ||||||
|                                   {'status': 'error_restoring'}) |                                   {'status': 'error_restoring'}) | ||||||
|  |  | ||||||
|  |     def _cleanup_one_snapshot(self, ctxt, snapshot_id): | ||||||
|  |         try: | ||||||
|  |             snapshot = objects.Snapshot.get_by_id(ctxt, snapshot_id) | ||||||
|  |         except exception.SnapshotNotFound: | ||||||
|  |             LOG.info('Snapshot %s does not exist anymore. Ignoring.', | ||||||
|  |                      snapshot_id) | ||||||
|  |             return | ||||||
|  |         if snapshot['status'] == 'backing-up': | ||||||
|  |             LOG.info('Resetting snapshot %(snap_id)s to previous ' | ||||||
|  |                      'status %(status)s (was backing-up).', | ||||||
|  |                      {'snap_id': snapshot['id'], | ||||||
|  |                       'status': fields.SnapshotStatus.AVAILABLE}) | ||||||
|  |  | ||||||
|  |             snapshot.status = fields.SnapshotStatus.AVAILABLE | ||||||
|  |             snapshot.save() | ||||||
|  |  | ||||||
|     def _cleanup_one_backup(self, ctxt, backup): |     def _cleanup_one_backup(self, ctxt, backup): | ||||||
|         if backup['status'] == fields.BackupStatus.CREATING: |         if backup['status'] == fields.BackupStatus.CREATING: | ||||||
|             LOG.info('Resetting backup %s to error (was creating).', |             LOG.info('Resetting backup %s to error (was creating).', | ||||||
|                      backup['id']) |                      backup['id']) | ||||||
|             self._cleanup_one_volume(ctxt, backup.volume_id) |             self._cleanup_one_volume(ctxt, backup.volume_id) | ||||||
|  |             if backup.snapshot_id: | ||||||
|  |                 self._cleanup_one_snapshot(ctxt, backup.snapshot_id) | ||||||
|             err = 'incomplete backup reset on manager restart' |             err = 'incomplete backup reset on manager restart' | ||||||
|             volume_utils.update_backup_error(backup, err) |             volume_utils.update_backup_error(backup, err) | ||||||
|         elif backup['status'] == fields.BackupStatus.RESTORING: |         elif backup['status'] == fields.BackupStatus.RESTORING: | ||||||
| @@ -245,6 +263,8 @@ class BackupManager(manager.SchedulerDependentManager): | |||||||
|                      'available (was restoring).', |                      'available (was restoring).', | ||||||
|                      backup['id']) |                      backup['id']) | ||||||
|             self._cleanup_one_volume(ctxt, backup.restore_volume_id) |             self._cleanup_one_volume(ctxt, backup.restore_volume_id) | ||||||
|  |             if backup.snapshot_id: | ||||||
|  |                 self._cleanup_one_snapshot(ctxt, backup.snapshot_id) | ||||||
|             backup.status = fields.BackupStatus.AVAILABLE |             backup.status = fields.BackupStatus.AVAILABLE | ||||||
|             backup.save() |             backup.save() | ||||||
|         elif backup['status'] == fields.BackupStatus.DELETING: |         elif backup['status'] == fields.BackupStatus.DELETING: | ||||||
|   | |||||||
| @@ -448,6 +448,18 @@ class BackupTestCase(BaseBackupTest): | |||||||
|         volume = db.volume_get(self.ctxt, volume_id) |         volume = db.volume_get(self.ctxt, volume_id) | ||||||
|         self.assertEqual('available', volume['status']) |         self.assertEqual('available', volume['status']) | ||||||
|  |  | ||||||
|  |     def test_cleanup_one_backing_up_snapshot(self): | ||||||
|  |         """Test cleanup_one_snapshot for snapshot status 'backing-up'.""" | ||||||
|  |  | ||||||
|  |         volume_id = str(uuid.uuid4()) | ||||||
|  |         snapshot_entry = self._create_snapshot_db_entry(status='backing-up', | ||||||
|  |                                                         volume_id=volume_id) | ||||||
|  |  | ||||||
|  |         self.backup_mgr._cleanup_one_snapshot(self.ctxt, snapshot_entry.id) | ||||||
|  |  | ||||||
|  |         snapshot = db.snapshot_get(self.ctxt, snapshot_entry.id) | ||||||
|  |         self.assertEqual('available', snapshot['status']) | ||||||
|  |  | ||||||
|     def test_cleanup_one_restoring_backup_volume(self): |     def test_cleanup_one_restoring_backup_volume(self): | ||||||
|         """Test cleanup_one_volume for volume status 'restoring-backup'.""" |         """Test cleanup_one_volume for volume status 'restoring-backup'.""" | ||||||
|  |  | ||||||
| @@ -478,6 +490,27 @@ class BackupTestCase(BaseBackupTest): | |||||||
|             'Volume %s does not exist anymore. Ignoring.', volume_id |             'Volume %s does not exist anymore. Ignoring.', volume_id | ||||||
|         ) |         ) | ||||||
|  |  | ||||||
|  |     @ddt.data(fields.BackupStatus.CREATING, | ||||||
|  |               fields.BackupStatus.RESTORING) | ||||||
|  |     def test_cleanup_one_backup_with_deleted_snapshot(self, backup_status): | ||||||
|  |         """Test cleanup_one_backup for non-existing volume.""" | ||||||
|  |  | ||||||
|  |         volume_id = str(uuid.uuid4()) | ||||||
|  |         snapshot_id = str(uuid.uuid4()) | ||||||
|  |         backup = self._create_backup_db_entry( | ||||||
|  |             status=backup_status, | ||||||
|  |             volume_id=volume_id, | ||||||
|  |             restore_volume_id=volume_id, | ||||||
|  |             snapshot_id=snapshot_id | ||||||
|  |         ) | ||||||
|  |  | ||||||
|  |         mock_log = self.mock_object(manager, 'LOG') | ||||||
|  |         self.backup_mgr._cleanup_one_backup(self.ctxt, backup) | ||||||
|  |  | ||||||
|  |         mock_log.info.assert_called_with( | ||||||
|  |             'Snapshot %s does not exist anymore. Ignoring.', snapshot_id | ||||||
|  |         ) | ||||||
|  |  | ||||||
|     def test_cleanup_one_creating_backup(self): |     def test_cleanup_one_creating_backup(self): | ||||||
|         """Test cleanup_one_backup for volume status 'creating'.""" |         """Test cleanup_one_backup for volume status 'creating'.""" | ||||||
|  |  | ||||||
|   | |||||||
							
								
								
									
										6
									
								
								releasenotes/notes/bug1938488-a528893c103c03af.yaml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										6
									
								
								releasenotes/notes/bug1938488-a528893c103c03af.yaml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,6 @@ | |||||||
|  | --- | ||||||
|  | fixes: | ||||||
|  |   - | | ||||||
|  |     `Bug #1938488 <https://bugs.launchpad.net/cinder/+bug/1938488>`_: | ||||||
|  |     When cleaning up a failed backup, clean up the snapshot status when the | ||||||
|  |     backup source is a snapshot | ||||||
		Reference in New Issue
	
	Block a user