Add support for deleting volumes with associated snapshots

OSC doesn't support deleting volumes with associated snapshots.
This patch provides support for deleting volumes with associated
snapshots by adding an optional argument.

Change-Id: I7e74f251574993ff13a38e508fd2f9debeda8d0a
Closes-Bug: #1589332
Co-Authored-By: Rui Chen <chenrui.momo@gmail.com>
This commit is contained in:
wuyuting 2016-06-05 22:55:43 -04:00 committed by Rui Chen
parent 7a667d700f
commit 954c28dfa2
4 changed files with 76 additions and 11 deletions

View File

@ -88,13 +88,19 @@ Delete volume(s)
.. code:: bash .. code:: bash
os volume delete os volume delete
[--force] [--force | --purge]
<volume> [<volume> ...] <volume> [<volume> ...]
.. option:: --force .. option:: --force
Attempt forced removal of volume(s), regardless of state (defaults to False) Attempt forced removal of volume(s), regardless of state (defaults to False)
.. option:: --purge
Remove any snapshots along with volume(s) (defaults to False)
*Volume version 2 only*
.. _volume_delete-volume: .. _volume_delete-volume:
.. describe:: <volume> .. describe:: <volume>

View File

@ -420,13 +420,16 @@ class TestVolumeDelete(TestVolume):
volumes[0].id volumes[0].id
] ]
verifylist = [ verifylist = [
("volumes", [volumes[0].id]) ("force", False),
("purge", False),
("volumes", [volumes[0].id]),
] ]
parsed_args = self.check_parser(self.cmd, arglist, verifylist) parsed_args = self.check_parser(self.cmd, arglist, verifylist)
result = self.cmd.take_action(parsed_args) result = self.cmd.take_action(parsed_args)
self.volumes_mock.delete.assert_called_with(volumes[0].id) self.volumes_mock.delete.assert_called_once_with(
volumes[0].id, cascade=False)
self.assertIsNone(result) self.assertIsNone(result)
def test_volume_delete_multi_volumes(self): def test_volume_delete_multi_volumes(self):
@ -434,13 +437,15 @@ class TestVolumeDelete(TestVolume):
arglist = [v.id for v in volumes] arglist = [v.id for v in volumes]
verifylist = [ verifylist = [
('force', False),
('purge', False),
('volumes', arglist), ('volumes', arglist),
] ]
parsed_args = self.check_parser(self.cmd, arglist, verifylist) parsed_args = self.check_parser(self.cmd, arglist, verifylist)
result = self.cmd.take_action(parsed_args) result = self.cmd.take_action(parsed_args)
calls = [call(v.id) for v in volumes] calls = [call(v.id, cascade=False) for v in volumes]
self.volumes_mock.delete.assert_has_calls(calls) self.volumes_mock.delete.assert_has_calls(calls)
self.assertIsNone(result) self.assertIsNone(result)
@ -452,6 +457,8 @@ class TestVolumeDelete(TestVolume):
'unexist_volume', 'unexist_volume',
] ]
verifylist = [ verifylist = [
('force', False),
('purge', False),
('volumes', arglist), ('volumes', arglist),
] ]
parsed_args = self.check_parser(self.cmd, arglist, verifylist) parsed_args = self.check_parser(self.cmd, arglist, verifylist)
@ -471,8 +478,46 @@ class TestVolumeDelete(TestVolume):
self.assertEqual(2, find_mock.call_count) self.assertEqual(2, find_mock.call_count)
self.volumes_mock.delete.assert_called_once_with( self.volumes_mock.delete.assert_called_once_with(
volumes[0].id volumes[0].id, cascade=False)
)
def test_volume_delete_with_purge(self):
volumes = self.setup_volumes_mock(count=1)
arglist = [
'--purge',
volumes[0].id,
]
verifylist = [
('force', False),
('purge', True),
('volumes', [volumes[0].id]),
]
parsed_args = self.check_parser(self.cmd, arglist, verifylist)
result = self.cmd.take_action(parsed_args)
self.volumes_mock.delete.assert_called_once_with(
volumes[0].id, cascade=True)
self.assertIsNone(result)
def test_volume_delete_with_force(self):
volumes = self.setup_volumes_mock(count=1)
arglist = [
'--force',
volumes[0].id,
]
verifylist = [
('force', True),
('purge', False),
('volumes', [volumes[0].id]),
]
parsed_args = self.check_parser(self.cmd, arglist, verifylist)
result = self.cmd.take_action(parsed_args)
self.volumes_mock.force_delete.assert_called_once_with(volumes[0].id)
self.assertIsNone(result)
class TestVolumeList(TestVolume): class TestVolumeList(TestVolume):

View File

@ -166,13 +166,19 @@ class DeleteVolume(command.Command):
nargs="+", nargs="+",
help=_("Volume(s) to delete (name or ID)") help=_("Volume(s) to delete (name or ID)")
) )
parser.add_argument( group = parser.add_mutually_exclusive_group()
group.add_argument(
"--force", "--force",
action="store_true", action="store_true",
default=False,
help=_("Attempt forced removal of volume(s), regardless of state " help=_("Attempt forced removal of volume(s), regardless of state "
"(defaults to False)") "(defaults to False)")
) )
group.add_argument(
"--purge",
action="store_true",
help=_("Remove any snapshots along with volume(s) "
"(defaults to False)")
)
return parser return parser
def take_action(self, parsed_args): def take_action(self, parsed_args):
@ -186,12 +192,13 @@ class DeleteVolume(command.Command):
if parsed_args.force: if parsed_args.force:
volume_client.volumes.force_delete(volume_obj.id) volume_client.volumes.force_delete(volume_obj.id)
else: else:
volume_client.volumes.delete(volume_obj.id) volume_client.volumes.delete(volume_obj.id,
cascade=parsed_args.purge)
except Exception as e: except Exception as e:
result += 1 result += 1
LOG.error(_("Failed to delete volume with " LOG.error(_("Failed to delete volume with "
"name or ID '%(volume)s': %(e)s") "name or ID '%(volume)s': %(e)s"),
% {'volume': i, 'e': e}) {'volume': i, 'e': e})
if result > 0: if result > 0:
total = len(parsed_args.volumes) total = len(parsed_args.volumes)

View File

@ -0,0 +1,7 @@
---
features:
- |
Add ``--purge`` option to ``volume delete`` command (Volume v2 only) in
order to removing any snapshots along with volume automatically when user
delete the volume.
[Bug `1589332 <https://bugs.launchpad.net/python-openstackclient/+bug/1589332>`_]