Add commands for "consistency group snapshot"
Add commands:
    consistency group snapshot create
    consistency group snapshot delete
    consistency group snapshot list
    consistency group snapshot show
in volume v2 (v2 only)
Change-Id: Ib4115f8ff00fb5aa8194588223032657eb1346b5
Closes-Bug: #1642238
Implements: bp cinder-command-support
			
			
This commit is contained in:
		
							
								
								
									
										96
									
								
								doc/source/command-objects/consistency-group-snapshot.rst
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										96
									
								
								doc/source/command-objects/consistency-group-snapshot.rst
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,96 @@ | ||||
| ========================== | ||||
| consistency group snapshot | ||||
| ========================== | ||||
|  | ||||
| Block Storage v2 | ||||
|  | ||||
| consistency group snapshot create | ||||
| --------------------------------- | ||||
|  | ||||
| Create new consistency group snapshot. | ||||
|  | ||||
| .. program:: consistency group snapshot create | ||||
| .. code:: bash | ||||
|  | ||||
|     os consistency group snapshot create | ||||
|         [--consistency-group <consistency-group>] | ||||
|         [--description <description>] | ||||
|         [<snapshot-name>] | ||||
|  | ||||
| .. option:: --consistency-group <consistency-group> | ||||
|  | ||||
|     Consistency group to snapshot (name or ID) | ||||
|     (default to be the same as <snapshot-name>) | ||||
|  | ||||
| .. option:: --description <description> | ||||
|  | ||||
|     Description of this consistency group snapshot | ||||
|  | ||||
| .. _consistency_group_snapshot_create-snapshot-name: | ||||
| .. option:: <snapshot-name> | ||||
|  | ||||
|     Name of new consistency group snapshot (default to None) | ||||
|  | ||||
| consistency group snapshot delete | ||||
| --------------------------------- | ||||
|  | ||||
| Delete consistency group snapshot(s) | ||||
|  | ||||
| .. program:: consistency group snapshot delete | ||||
| .. code:: bash | ||||
|  | ||||
|     os consistency group snapshot delete | ||||
|         <consistency-group-snapshot> [<consistency-group-snapshot> ...] | ||||
|  | ||||
| .. _consistency_group_snapshot_delete-consistency-group-snapshot: | ||||
| .. describe:: <consistency-group-snapshot> | ||||
|  | ||||
|     Consistency group snapshot(s) to delete (name or ID) | ||||
|  | ||||
| consistency group snapshot list | ||||
| ------------------------------- | ||||
|  | ||||
| List consistency group snapshots. | ||||
|  | ||||
| .. program:: consistency group snapshot list | ||||
| .. code:: bash | ||||
|  | ||||
|     os consistency group snapshot list | ||||
|         [--all-projects] | ||||
|         [--long] | ||||
|         [--status <status>] | ||||
|         [--consistency-group <consistency-group>] | ||||
|  | ||||
| .. option:: --all-projects | ||||
|  | ||||
|     Show detail for all projects. Admin only. | ||||
|     (defaults to False) | ||||
|  | ||||
| .. option:: --long | ||||
|  | ||||
|     List additional fields in output | ||||
|  | ||||
| .. option:: --status <status> | ||||
|  | ||||
|     Filters results by a status | ||||
|     ("available", "error", "creating", "deleting" or "error_deleting") | ||||
|  | ||||
| .. option:: --consistency-group <consistency-group> | ||||
|  | ||||
|     Filters results by a consistency group (name or ID) | ||||
|  | ||||
| consistency group snapshot show | ||||
| ------------------------------- | ||||
|  | ||||
| Display consistency group snapshot details. | ||||
|  | ||||
| .. program:: consistency group snapshot show | ||||
| .. code:: bash | ||||
|  | ||||
|     os consistency group snapshot show | ||||
|         <consistency-group-snapshot> | ||||
|  | ||||
| .. _consistency_group_snapshot_show-consistency-group-snapshot: | ||||
| .. describe:: <consistency-group-snapshot> | ||||
|  | ||||
|     Consistency group snapshot to display (name or ID) | ||||
| @@ -81,6 +81,7 @@ referring to both Compute and Volume quotas. | ||||
| * ``compute service``: (**Compute**) a cloud Compute process running on a host | ||||
| * ``configuration``: (**Internal**) OpenStack client configuration | ||||
| * ``consistency group``: (**Volume**) a consistency group of volumes | ||||
| * ``consistency group snapshot``: (**Volume**) a point-in-time copy of a consistency group | ||||
| * ``console log``: (**Compute**) server console text dump | ||||
| * ``console url``: (**Compute**) server remote console URL | ||||
| * ``consumer``: (**Identity**) OAuth-based delegatee | ||||
|   | ||||
| @@ -224,6 +224,8 @@ class FakeVolumeClient(object): | ||||
|         self.quota_classes.resource_class = fakes.FakeResource(None, {}) | ||||
|         self.consistencygroups = mock.Mock() | ||||
|         self.consistencygroups.resource_class = fakes.FakeResource(None, {}) | ||||
|         self.cgsnapshots = mock.Mock() | ||||
|         self.cgsnapshots.resource_class = fakes.FakeResource(None, {}) | ||||
|         self.auth_token = kwargs['token'] | ||||
|         self.management_url = kwargs['endpoint'] | ||||
|  | ||||
| @@ -548,6 +550,82 @@ class FakeConsistencyGroup(object): | ||||
|         return consistency_groups | ||||
|  | ||||
|  | ||||
| class FakeConsistencyGroupSnapshot(object): | ||||
|     """Fake one or more consistency group snapshot.""" | ||||
|  | ||||
|     @staticmethod | ||||
|     def create_one_consistency_group_snapshot(attrs=None): | ||||
|         """Create a fake consistency group snapshot. | ||||
|  | ||||
|         :param Dictionary attrs: | ||||
|             A dictionary with all attributes | ||||
|         :return: | ||||
|             A FakeResource object with id, name, description, etc. | ||||
|         """ | ||||
|         attrs = attrs or {} | ||||
|  | ||||
|         # Set default attributes. | ||||
|         consistency_group_snapshot_info = { | ||||
|             "id": 'id-' + uuid.uuid4().hex, | ||||
|             "name": 'backup-name-' + uuid.uuid4().hex, | ||||
|             "description": 'description-' + uuid.uuid4().hex, | ||||
|             "status": "error", | ||||
|             "consistencygroup_id": 'consistency-group-id' + uuid.uuid4().hex, | ||||
|             "created_at": 'time-' + uuid.uuid4().hex, | ||||
|         } | ||||
|  | ||||
|         # Overwrite default attributes. | ||||
|         consistency_group_snapshot_info.update(attrs) | ||||
|  | ||||
|         consistency_group_snapshot = fakes.FakeResource( | ||||
|             info=copy.deepcopy(consistency_group_snapshot_info), | ||||
|             loaded=True) | ||||
|         return consistency_group_snapshot | ||||
|  | ||||
|     @staticmethod | ||||
|     def create_consistency_group_snapshots(attrs=None, count=2): | ||||
|         """Create multiple fake consistency group snapshots. | ||||
|  | ||||
|         :param Dictionary attrs: | ||||
|             A dictionary with all attributes | ||||
|         :param int count: | ||||
|             The number of consistency group snapshots to fake | ||||
|         :return: | ||||
|             A list of FakeResource objects faking the | ||||
|             consistency group snapshots | ||||
|         """ | ||||
|         consistency_group_snapshots = [] | ||||
|         for i in range(0, count): | ||||
|             consistency_group_snapshot = ( | ||||
|                 FakeConsistencyGroupSnapshot. | ||||
|                 create_one_consistency_group_snapshot(attrs) | ||||
|             ) | ||||
|             consistency_group_snapshots.append(consistency_group_snapshot) | ||||
|  | ||||
|         return consistency_group_snapshots | ||||
|  | ||||
|     @staticmethod | ||||
|     def get_consistency_group_snapshots(snapshots=None, count=2): | ||||
|         """Get an iterable MagicMock object with a list of faked cgsnapshots. | ||||
|  | ||||
|         If consistenct group snapshots list is provided, then initialize | ||||
|         the Mock object with the list. Otherwise create one. | ||||
|  | ||||
|         :param List snapshots: | ||||
|             A list of FakeResource objects faking consistency group snapshots | ||||
|         :param Integer count: | ||||
|             The number of consistency group snapshots to be faked | ||||
|         :return | ||||
|             An iterable Mock object with side_effect set to a list of faked | ||||
|             consistency groups | ||||
|         """ | ||||
|         if snapshots is None: | ||||
|             snapshots = (FakeConsistencyGroupSnapshot. | ||||
|                          create_consistency_group_snapshots(count)) | ||||
|  | ||||
|         return mock.Mock(side_effect=snapshots) | ||||
|  | ||||
|  | ||||
| class FakeExtension(object): | ||||
|     """Fake one or more extension.""" | ||||
|  | ||||
|   | ||||
| @@ -0,0 +1,351 @@ | ||||
| # | ||||
| #   Licensed under the Apache License, Version 2.0 (the "License"); you may | ||||
| #   not use this file except in compliance with the License. You may obtain | ||||
| #   a copy of the License at | ||||
| # | ||||
| #        http://www.apache.org/licenses/LICENSE-2.0 | ||||
| # | ||||
| #   Unless required by applicable law or agreed to in writing, software | ||||
| #   distributed under the License is distributed on an "AS IS" BASIS, WITHOUT | ||||
| #   WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the | ||||
| #   License for the specific language governing permissions and limitations | ||||
| #   under the License. | ||||
| # | ||||
|  | ||||
| from mock import call | ||||
|  | ||||
| from openstackclient.tests.unit.volume.v2 import fakes as volume_fakes | ||||
| from openstackclient.volume.v2 import consistency_group_snapshot | ||||
|  | ||||
|  | ||||
| class TestConsistencyGroupSnapshot(volume_fakes.TestVolume): | ||||
|  | ||||
|     def setUp(self): | ||||
|         super(TestConsistencyGroupSnapshot, self).setUp() | ||||
|  | ||||
|         # Get a shortcut to the TransferManager Mock | ||||
|         self.cgsnapshots_mock = ( | ||||
|             self.app.client_manager.volume.cgsnapshots) | ||||
|         self.cgsnapshots_mock.reset_mock() | ||||
|         self.consistencygroups_mock = ( | ||||
|             self.app.client_manager.volume.consistencygroups) | ||||
|         self.consistencygroups_mock.reset_mock() | ||||
|  | ||||
|  | ||||
| class TestConsistencyGroupSnapshotCreate(TestConsistencyGroupSnapshot): | ||||
|  | ||||
|     _consistency_group_snapshot = ( | ||||
|         volume_fakes. | ||||
|         FakeConsistencyGroupSnapshot. | ||||
|         create_one_consistency_group_snapshot() | ||||
|     ) | ||||
|     consistency_group = ( | ||||
|         volume_fakes.FakeConsistencyGroup.create_one_consistency_group()) | ||||
|  | ||||
|     columns = ( | ||||
|         'consistencygroup_id', | ||||
|         'created_at', | ||||
|         'description', | ||||
|         'id', | ||||
|         'name', | ||||
|         'status', | ||||
|     ) | ||||
|     data = ( | ||||
|         _consistency_group_snapshot.consistencygroup_id, | ||||
|         _consistency_group_snapshot.created_at, | ||||
|         _consistency_group_snapshot.description, | ||||
|         _consistency_group_snapshot.id, | ||||
|         _consistency_group_snapshot.name, | ||||
|         _consistency_group_snapshot.status, | ||||
|     ) | ||||
|  | ||||
|     def setUp(self): | ||||
|         super(TestConsistencyGroupSnapshotCreate, self).setUp() | ||||
|         self.cgsnapshots_mock.create.return_value = ( | ||||
|             self._consistency_group_snapshot) | ||||
|         self.consistencygroups_mock.get.return_value = ( | ||||
|             self.consistency_group) | ||||
|  | ||||
|         # Get the command object to test | ||||
|         self.cmd = (consistency_group_snapshot. | ||||
|                     CreateConsistencyGroupSnapshot(self.app, None)) | ||||
|  | ||||
|     def test_consistency_group_snapshot_create(self): | ||||
|         arglist = [ | ||||
|             '--consistency-group', self.consistency_group.id, | ||||
|             '--description', self._consistency_group_snapshot.description, | ||||
|             self._consistency_group_snapshot.name, | ||||
|         ] | ||||
|         verifylist = [ | ||||
|             ('consistency_group', self.consistency_group.id), | ||||
|             ('description', self._consistency_group_snapshot.description), | ||||
|             ('snapshot_name', self._consistency_group_snapshot.name), | ||||
|         ] | ||||
|         parsed_args = self.check_parser(self.cmd, arglist, verifylist) | ||||
|  | ||||
|         columns, data = self.cmd.take_action(parsed_args) | ||||
|  | ||||
|         self.consistencygroups_mock.get.assert_called_once_with( | ||||
|             self.consistency_group.id) | ||||
|         self.cgsnapshots_mock.create.assert_called_once_with( | ||||
|             self.consistency_group.id, | ||||
|             name=self._consistency_group_snapshot.name, | ||||
|             description=self._consistency_group_snapshot.description, | ||||
|         ) | ||||
|  | ||||
|         self.assertEqual(self.columns, columns) | ||||
|         self.assertEqual(self.data, data) | ||||
|  | ||||
|     def test_consistency_group_snapshot_create_no_consistency_group(self): | ||||
|         arglist = [ | ||||
|             '--description', self._consistency_group_snapshot.description, | ||||
|             self._consistency_group_snapshot.name, | ||||
|         ] | ||||
|         verifylist = [ | ||||
|             ('description', self._consistency_group_snapshot.description), | ||||
|             ('snapshot_name', self._consistency_group_snapshot.name), | ||||
|         ] | ||||
|         parsed_args = self.check_parser(self.cmd, arglist, verifylist) | ||||
|  | ||||
|         columns, data = self.cmd.take_action(parsed_args) | ||||
|  | ||||
|         self.consistencygroups_mock.get.assert_called_once_with( | ||||
|             self._consistency_group_snapshot.name) | ||||
|         self.cgsnapshots_mock.create.assert_called_once_with( | ||||
|             self.consistency_group.id, | ||||
|             name=self._consistency_group_snapshot.name, | ||||
|             description=self._consistency_group_snapshot.description, | ||||
|         ) | ||||
|  | ||||
|         self.assertEqual(self.columns, columns) | ||||
|         self.assertEqual(self.data, data) | ||||
|  | ||||
|  | ||||
| class TestConsistencyGroupSnapshotDelete(TestConsistencyGroupSnapshot): | ||||
|  | ||||
|     consistency_group_snapshots = ( | ||||
|         volume_fakes.FakeConsistencyGroupSnapshot. | ||||
|         create_consistency_group_snapshots(count=2) | ||||
|     ) | ||||
|  | ||||
|     def setUp(self): | ||||
|         super(TestConsistencyGroupSnapshotDelete, self).setUp() | ||||
|  | ||||
|         self.cgsnapshots_mock.get = ( | ||||
|             volume_fakes.FakeConsistencyGroupSnapshot. | ||||
|             get_consistency_group_snapshots(self.consistency_group_snapshots) | ||||
|         ) | ||||
|         self.cgsnapshots_mock.delete.return_value = None | ||||
|  | ||||
|         # Get the command object to mock | ||||
|         self.cmd = (consistency_group_snapshot. | ||||
|                     DeleteConsistencyGroupSnapshot(self.app, None)) | ||||
|  | ||||
|     def test_consistency_group_snapshot_delete(self): | ||||
|         arglist = [ | ||||
|             self.consistency_group_snapshots[0].id | ||||
|         ] | ||||
|         verifylist = [ | ||||
|             ("consistency_group_snapshot", | ||||
|              [self.consistency_group_snapshots[0].id]) | ||||
|         ] | ||||
|         parsed_args = self.check_parser(self.cmd, arglist, verifylist) | ||||
|  | ||||
|         result = self.cmd.take_action(parsed_args) | ||||
|  | ||||
|         self.cgsnapshots_mock.delete.assert_called_once_with( | ||||
|             self.consistency_group_snapshots[0].id) | ||||
|         self.assertIsNone(result) | ||||
|  | ||||
|     def test_multiple_consistency_group_snapshots_delete(self): | ||||
|         arglist = [] | ||||
|         for c in self.consistency_group_snapshots: | ||||
|             arglist.append(c.id) | ||||
|         verifylist = [ | ||||
|             ('consistency_group_snapshot', arglist), | ||||
|         ] | ||||
|  | ||||
|         parsed_args = self.check_parser(self.cmd, arglist, verifylist) | ||||
|         result = self.cmd.take_action(parsed_args) | ||||
|  | ||||
|         calls = [] | ||||
|         for c in self.consistency_group_snapshots: | ||||
|             calls.append(call(c.id)) | ||||
|         self.cgsnapshots_mock.delete.assert_has_calls(calls) | ||||
|         self.assertIsNone(result) | ||||
|  | ||||
|  | ||||
| class TestConsistencyGroupSnapshotList(TestConsistencyGroupSnapshot): | ||||
|  | ||||
|     consistency_group_snapshots = ( | ||||
|         volume_fakes.FakeConsistencyGroupSnapshot. | ||||
|         create_consistency_group_snapshots(count=2) | ||||
|     ) | ||||
|     consistency_group = ( | ||||
|         volume_fakes.FakeConsistencyGroup.create_one_consistency_group() | ||||
|     ) | ||||
|  | ||||
|     columns = [ | ||||
|         'ID', | ||||
|         'Status', | ||||
|         'Name', | ||||
|     ] | ||||
|     columns_long = [ | ||||
|         'ID', | ||||
|         'Status', | ||||
|         'ConsistencyGroup ID', | ||||
|         'Name', | ||||
|         'Description', | ||||
|         'Created At', | ||||
|     ] | ||||
|     data = [] | ||||
|     for c in consistency_group_snapshots: | ||||
|         data.append(( | ||||
|             c.id, | ||||
|             c.status, | ||||
|             c.name, | ||||
|         )) | ||||
|     data_long = [] | ||||
|     for c in consistency_group_snapshots: | ||||
|         data_long.append(( | ||||
|             c.id, | ||||
|             c.status, | ||||
|             c.consistencygroup_id, | ||||
|             c.name, | ||||
|             c.description, | ||||
|             c.created_at, | ||||
|         )) | ||||
|  | ||||
|     def setUp(self): | ||||
|         super(TestConsistencyGroupSnapshotList, self).setUp() | ||||
|  | ||||
|         self.cgsnapshots_mock.list.return_value = ( | ||||
|             self.consistency_group_snapshots) | ||||
|         self.consistencygroups_mock.get.return_value = self.consistency_group | ||||
|         # Get the command to test | ||||
|         self.cmd = ( | ||||
|             consistency_group_snapshot. | ||||
|             ListConsistencyGroupSnapshot(self.app, None) | ||||
|         ) | ||||
|  | ||||
|     def test_consistency_group_snapshot_list_without_options(self): | ||||
|         arglist = [] | ||||
|         verifylist = [ | ||||
|             ("all_projects", False), | ||||
|             ("long", False), | ||||
|             ("status", None), | ||||
|             ("consistency_group", None), | ||||
|         ] | ||||
|  | ||||
|         parsed_args = self.check_parser(self.cmd, arglist, verifylist) | ||||
|         columns, data = self.cmd.take_action(parsed_args) | ||||
|  | ||||
|         search_opts = { | ||||
|             'all_tenants': False, | ||||
|             'status': None, | ||||
|             'consistencygroup_id': None, | ||||
|         } | ||||
|         self.cgsnapshots_mock.list.assert_called_once_with( | ||||
|             detailed=True, search_opts=search_opts) | ||||
|         self.assertEqual(self.columns, columns) | ||||
|         self.assertEqual(self.data, list(data)) | ||||
|  | ||||
|     def test_consistency_group_snapshot_list_with_long(self): | ||||
|         arglist = [ | ||||
|             "--long", | ||||
|         ] | ||||
|         verifylist = [ | ||||
|             ("all_projects", False), | ||||
|             ("long", True), | ||||
|             ("status", None), | ||||
|             ("consistency_group", None), | ||||
|         ] | ||||
|  | ||||
|         parsed_args = self.check_parser(self.cmd, arglist, verifylist) | ||||
|         columns, data = self.cmd.take_action(parsed_args) | ||||
|  | ||||
|         search_opts = { | ||||
|             'all_tenants': False, | ||||
|             'status': None, | ||||
|             'consistencygroup_id': None, | ||||
|         } | ||||
|         self.cgsnapshots_mock.list.assert_called_once_with( | ||||
|             detailed=True, search_opts=search_opts) | ||||
|         self.assertEqual(self.columns_long, columns) | ||||
|         self.assertEqual(self.data_long, list(data)) | ||||
|  | ||||
|     def test_consistency_group_snapshot_list_with_options(self): | ||||
|         arglist = [ | ||||
|             "--all-project", | ||||
|             "--status", self.consistency_group_snapshots[0].status, | ||||
|             "--consistency-group", self.consistency_group.id, | ||||
|         ] | ||||
|         verifylist = [ | ||||
|             ("all_projects", True), | ||||
|             ("long", False), | ||||
|             ("status", self.consistency_group_snapshots[0].status), | ||||
|             ("consistency_group", self.consistency_group.id), | ||||
|         ] | ||||
|  | ||||
|         parsed_args = self.check_parser(self.cmd, arglist, verifylist) | ||||
|         columns, data = self.cmd.take_action(parsed_args) | ||||
|  | ||||
|         search_opts = { | ||||
|             'all_tenants': True, | ||||
|             'status': self.consistency_group_snapshots[0].status, | ||||
|             'consistencygroup_id': self.consistency_group.id, | ||||
|         } | ||||
|         self.consistencygroups_mock.get.assert_called_once_with( | ||||
|             self.consistency_group.id) | ||||
|         self.cgsnapshots_mock.list.assert_called_once_with( | ||||
|             detailed=True, search_opts=search_opts) | ||||
|         self.assertEqual(self.columns, columns) | ||||
|         self.assertEqual(self.data, list(data)) | ||||
|  | ||||
|  | ||||
| class TestConsistencyGroupSnapshotShow(TestConsistencyGroupSnapshot): | ||||
|  | ||||
|     _consistency_group_snapshot = ( | ||||
|         volume_fakes. | ||||
|         FakeConsistencyGroupSnapshot. | ||||
|         create_one_consistency_group_snapshot() | ||||
|     ) | ||||
|  | ||||
|     columns = ( | ||||
|         'consistencygroup_id', | ||||
|         'created_at', | ||||
|         'description', | ||||
|         'id', | ||||
|         'name', | ||||
|         'status', | ||||
|     ) | ||||
|     data = ( | ||||
|         _consistency_group_snapshot.consistencygroup_id, | ||||
|         _consistency_group_snapshot.created_at, | ||||
|         _consistency_group_snapshot.description, | ||||
|         _consistency_group_snapshot.id, | ||||
|         _consistency_group_snapshot.name, | ||||
|         _consistency_group_snapshot.status, | ||||
|     ) | ||||
|  | ||||
|     def setUp(self): | ||||
|         super(TestConsistencyGroupSnapshotShow, self).setUp() | ||||
|  | ||||
|         self.cgsnapshots_mock.get.return_value = ( | ||||
|             self._consistency_group_snapshot) | ||||
|         self.cmd = (consistency_group_snapshot. | ||||
|                     ShowConsistencyGroupSnapshot(self.app, None)) | ||||
|  | ||||
|     def test_consistency_group_snapshot_show(self): | ||||
|         arglist = [ | ||||
|             self._consistency_group_snapshot.id | ||||
|         ] | ||||
|         verifylist = [ | ||||
|             ("consistency_group_snapshot", self._consistency_group_snapshot.id) | ||||
|         ] | ||||
|         parsed_args = self.check_parser(self.cmd, arglist, verifylist) | ||||
|         columns, data = self.cmd.take_action(parsed_args) | ||||
|         self.cgsnapshots_mock.get.assert_called_once_with( | ||||
|             self._consistency_group_snapshot.id) | ||||
|         self.assertEqual(self.columns, columns) | ||||
|         self.assertEqual(self.data, data) | ||||
							
								
								
									
										190
									
								
								openstackclient/volume/v2/consistency_group_snapshot.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										190
									
								
								openstackclient/volume/v2/consistency_group_snapshot.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,190 @@ | ||||
| # | ||||
| #   Licensed under the Apache License, Version 2.0 (the "License"); you may | ||||
| #   not use this file except in compliance with the License. You may obtain | ||||
| #   a copy of the License at | ||||
| # | ||||
| #        http://www.apache.org/licenses/LICENSE-2.0 | ||||
| # | ||||
| #   Unless required by applicable law or agreed to in writing, software | ||||
| #   distributed under the License is distributed on an "AS IS" BASIS, WITHOUT | ||||
| #   WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the | ||||
| #   License for the specific language governing permissions and limitations | ||||
| #   under the License. | ||||
| # | ||||
|  | ||||
| """Volume v2 consistency group snapshot action implementations""" | ||||
|  | ||||
| import logging | ||||
|  | ||||
| from osc_lib.command import command | ||||
| from osc_lib import exceptions | ||||
| from osc_lib import utils | ||||
| import six | ||||
|  | ||||
| from openstackclient.i18n import _ | ||||
|  | ||||
|  | ||||
| LOG = logging.getLogger(__name__) | ||||
|  | ||||
|  | ||||
| class CreateConsistencyGroupSnapshot(command.ShowOne): | ||||
|     _description = _("Create new consistency group snapshot.") | ||||
|  | ||||
|     def get_parser(self, prog_name): | ||||
|         parser = super( | ||||
|             CreateConsistencyGroupSnapshot, self).get_parser(prog_name) | ||||
|         parser.add_argument( | ||||
|             "snapshot_name", | ||||
|             metavar="<snapshot-name>", | ||||
|             nargs="?", | ||||
|             help=_("Name of new consistency group snapshot (default to None)") | ||||
|         ) | ||||
|         parser.add_argument( | ||||
|             "--consistency-group", | ||||
|             metavar="<consistency-group>", | ||||
|             help=_("Consistency group to snapshot (name or ID) " | ||||
|                    "(default to be the same as <snapshot-name>)") | ||||
|         ) | ||||
|         parser.add_argument( | ||||
|             "--description", | ||||
|             metavar="<description>", | ||||
|             help=_("Description of this consistency group snapshot") | ||||
|         ) | ||||
|         return parser | ||||
|  | ||||
|     def take_action(self, parsed_args): | ||||
|         volume_client = self.app.client_manager.volume | ||||
|         consistency_group = parsed_args.consistency_group | ||||
|         if not parsed_args.consistency_group: | ||||
|             # If "--consistency-group" not specified, then consistency_group | ||||
|             # will be the same as the new consistency group snapshot name | ||||
|             consistency_group = parsed_args.snapshot_name | ||||
|         consistency_group_id = utils.find_resource( | ||||
|             volume_client.consistencygroups, | ||||
|             consistency_group).id | ||||
|         consistency_group_snapshot = volume_client.cgsnapshots.create( | ||||
|             consistency_group_id, | ||||
|             name=parsed_args.snapshot_name, | ||||
|             description=parsed_args.description, | ||||
|         ) | ||||
|  | ||||
|         return zip(*sorted(six.iteritems(consistency_group_snapshot._info))) | ||||
|  | ||||
|  | ||||
| class DeleteConsistencyGroupSnapshot(command.Command): | ||||
|     _description = _("Delete consistency group snapshot(s).") | ||||
|  | ||||
|     def get_parser(self, prog_name): | ||||
|         parser = super( | ||||
|             DeleteConsistencyGroupSnapshot, self).get_parser(prog_name) | ||||
|         parser.add_argument( | ||||
|             "consistency_group_snapshot", | ||||
|             metavar="<consistency-group-snapshot>", | ||||
|             nargs="+", | ||||
|             help=_("Consistency group snapshot(s) to delete (name or ID)") | ||||
|         ) | ||||
|         return parser | ||||
|  | ||||
|     def take_action(self, parsed_args): | ||||
|         volume_client = self.app.client_manager.volume | ||||
|         result = 0 | ||||
|  | ||||
|         for snapshot in parsed_args.consistency_group_snapshot: | ||||
|             try: | ||||
|                 snapshot_id = utils.find_resource(volume_client.cgsnapshots, | ||||
|                                                   snapshot).id | ||||
|  | ||||
|                 volume_client.cgsnapshots.delete(snapshot_id) | ||||
|             except Exception as e: | ||||
|                 result += 1 | ||||
|                 LOG.error(_("Failed to delete consistency group snapshot " | ||||
|                             "with name or ID '%(snapshot)s': %(e)s") | ||||
|                           % {'snapshot': snapshot, 'e': e}) | ||||
|  | ||||
|         if result > 0: | ||||
|             total = len(parsed_args.consistency_group_snapshot) | ||||
|             msg = (_("%(result)s of %(total)s consistency group snapshots " | ||||
|                    "failed to delete.") % {'result': result, 'total': total}) | ||||
|             raise exceptions.CommandError(msg) | ||||
|  | ||||
|  | ||||
| class ListConsistencyGroupSnapshot(command.Lister): | ||||
|     _description = _("List consistency group snapshots.") | ||||
|  | ||||
|     def get_parser(self, prog_name): | ||||
|         parser = super( | ||||
|             ListConsistencyGroupSnapshot, self).get_parser(prog_name) | ||||
|         parser.add_argument( | ||||
|             '--all-projects', | ||||
|             action="store_true", | ||||
|             help=_('Show detail for all projects (admin only) ' | ||||
|                    '(defaults to False)') | ||||
|         ) | ||||
|         parser.add_argument( | ||||
|             '--long', | ||||
|             action="store_true", | ||||
|             help=_('List additional fields in output') | ||||
|         ) | ||||
|         parser.add_argument( | ||||
|             '--status', | ||||
|             metavar="<status>", | ||||
|             choices=['available', 'error', 'creating', 'deleting', | ||||
|                      'error-deleting'], | ||||
|             help=_('Filters results by a status ("available", "error", ' | ||||
|                    '"creating", "deleting" or "error_deleting")') | ||||
|         ) | ||||
|         parser.add_argument( | ||||
|             '--consistency-group', | ||||
|             metavar="<consistency-group>", | ||||
|             help=_('Filters results by a consistency group (name or ID)') | ||||
|         ) | ||||
|         return parser | ||||
|  | ||||
|     def take_action(self, parsed_args): | ||||
|         if parsed_args.long: | ||||
|             columns = ['ID', 'Status', 'ConsistencyGroup ID', | ||||
|                        'Name', 'Description', 'Created At'] | ||||
|         else: | ||||
|             columns = ['ID', 'Status', 'Name'] | ||||
|         volume_client = self.app.client_manager.volume | ||||
|         consistency_group_id = None | ||||
|         if parsed_args.consistency_group: | ||||
|             consistency_group_id = utils.find_resource( | ||||
|                 volume_client.consistencygroups, | ||||
|                 parsed_args.consistency_group, | ||||
|             ).id | ||||
|         search_opts = { | ||||
|             'all_tenants': parsed_args.all_projects, | ||||
|             'status': parsed_args.status, | ||||
|             'consistencygroup_id': consistency_group_id, | ||||
|         } | ||||
|         consistency_group_snapshots = volume_client.cgsnapshots.list( | ||||
|             detailed=True, | ||||
|             search_opts=search_opts, | ||||
|         ) | ||||
|  | ||||
|         return (columns, ( | ||||
|             utils.get_item_properties( | ||||
|                 s, columns) | ||||
|             for s in consistency_group_snapshots)) | ||||
|  | ||||
|  | ||||
| class ShowConsistencyGroupSnapshot(command.ShowOne): | ||||
|     _description = _("Display consistency group snapshot details") | ||||
|  | ||||
|     def get_parser(self, prog_name): | ||||
|         parser = super( | ||||
|             ShowConsistencyGroupSnapshot, self).get_parser(prog_name) | ||||
|         parser.add_argument( | ||||
|             "consistency_group_snapshot", | ||||
|             metavar="<consistency-group-snapshot>", | ||||
|             help=_("Consistency group snapshot to display (name or ID)") | ||||
|         ) | ||||
|         return parser | ||||
|  | ||||
|     def take_action(self, parsed_args): | ||||
|         volume_client = self.app.client_manager.volume | ||||
|         consistency_group_snapshot = utils.find_resource( | ||||
|             volume_client.cgsnapshots, | ||||
|             parsed_args.consistency_group_snapshot) | ||||
|         return zip(*sorted(six.iteritems(consistency_group_snapshot._info))) | ||||
							
								
								
									
										6
									
								
								releasenotes/notes/bug-1642238-3032c7fe7f0ce29d.yaml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										6
									
								
								releasenotes/notes/bug-1642238-3032c7fe7f0ce29d.yaml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,6 @@ | ||||
| --- | ||||
| features: | ||||
|   - Add ``consistency group snapshot create``, ``consistency group snapshot delete``, | ||||
|     ``consistency group snapshot list`` and ``consistency group snapshot show`` commands | ||||
|     in volume v2. | ||||
|     [Bug `1642238 <https://bugs.launchpad.net/python-openstackclient/+bug/1642238>`_] | ||||
| @@ -508,6 +508,11 @@ openstack.volume.v2 = | ||||
|  | ||||
|     consistency_group_list = openstackclient.volume.v2.consistency_group:ListConsistencyGroup | ||||
|  | ||||
|     consistency_group_snapshot_create = openstackclient.volume.v2.consistency_group_snapshot:CreateConsistencyGroupSnapshot | ||||
|     consistency_group_snapshot_delete = openstackclient.volume.v2.consistency_group_snapshot:DeleteConsistencyGroupSnapshot | ||||
|     consistency_group_snapshot_list = openstackclient.volume.v2.consistency_group_snapshot:ListConsistencyGroupSnapshot | ||||
|     consistency_group_snapshot_show = openstackclient.volume.v2.consistency_group_snapshot:ShowConsistencyGroupSnapshot | ||||
|  | ||||
|     snapshot_create = openstackclient.volume.v2.snapshot:CreateSnapshot | ||||
|     snapshot_delete = openstackclient.volume.v2.snapshot:DeleteSnapshot | ||||
|     snapshot_list = openstackclient.volume.v2.snapshot:ListSnapshot | ||||
|   | ||||
		Reference in New Issue
	
	Block a user
	 Huanxuan Ao
					Huanxuan Ao