summaryrefslogtreecommitdiff
path: root/openstackclient/tests
diff options
context:
space:
mode:
authorStephen Finucane <sfinucan@redhat.com>2021-05-24 15:56:27 +0100
committerStephen Finucane <sfinucan@redhat.com>2021-06-03 15:52:18 +0100
commit6dc94e1fb85595653dcdd24185c914b9df1741df (patch)
tree7ca68a28d50afc5ff1394f1eb2d4a333a938cf19 /openstackclient/tests
parent0f28588e48c1e296f834e8684f293c2cdf4afc33 (diff)
downloadpython-openstackclient-6dc94e1fb85595653dcdd24185c914b9df1741df.tar.gz
volume: Add 'volume attachment *' commands
These mirror the 'cinder attachment-*' commands, with arguments copied across essentially verbatim. The only significant departure is the replacement of "tenant" terminology with "project". volume attachment create volume attachment delete volume attachment list volume attachment complete volume attachment set volume attachment show Full support for filtering is deferred for now since that's a more complicated change that requires additional commands be added first. TODOs are included to this effect. Change-Id: If47c2b56fe65ee2cee07c000d6ae3688d5ef3b42 Signed-off-by: Stephen Finucane <sfinucan@redhat.com>
Diffstat (limited to 'openstackclient/tests')
-rw-r--r--openstackclient/tests/unit/volume/v3/fakes.py155
-rw-r--r--openstackclient/tests/unit/volume/v3/test_volume_attachment.py560
2 files changed, 715 insertions, 0 deletions
diff --git a/openstackclient/tests/unit/volume/v3/fakes.py b/openstackclient/tests/unit/volume/v3/fakes.py
new file mode 100644
index 00000000..fb3b1b74
--- /dev/null
+++ b/openstackclient/tests/unit/volume/v3/fakes.py
@@ -0,0 +1,155 @@
+# 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.
+
+import random
+from unittest import mock
+import uuid
+
+from cinderclient import api_versions
+
+from openstackclient.tests.unit.compute.v2 import fakes as compute_fakes
+from openstackclient.tests.unit import fakes
+from openstackclient.tests.unit.identity.v3 import fakes as identity_fakes
+from openstackclient.tests.unit import utils
+from openstackclient.tests.unit.volume.v2 import fakes as volume_v2_fakes
+
+
+class FakeVolumeClient(object):
+
+ def __init__(self, **kwargs):
+ self.auth_token = kwargs['token']
+ self.management_url = kwargs['endpoint']
+ self.api_version = api_versions.APIVersion('3.0')
+
+ self.attachments = mock.Mock()
+ self.attachments.resource_class = fakes.FakeResource(None, {})
+ self.volumes = mock.Mock()
+ self.volumes.resource_class = fakes.FakeResource(None, {})
+
+
+class TestVolume(utils.TestCommand):
+
+ def setUp(self):
+ super().setUp()
+
+ self.app.client_manager.volume = FakeVolumeClient(
+ endpoint=fakes.AUTH_URL,
+ token=fakes.AUTH_TOKEN
+ )
+ self.app.client_manager.identity = identity_fakes.FakeIdentityv3Client(
+ endpoint=fakes.AUTH_URL,
+ token=fakes.AUTH_TOKEN
+ )
+ self.app.client_manager.compute = compute_fakes.FakeComputev2Client(
+ endpoint=fakes.AUTH_URL,
+ token=fakes.AUTH_TOKEN,
+ )
+
+
+# TODO(stephenfin): Check if the responses are actually the same
+FakeVolume = volume_v2_fakes.FakeVolume
+
+
+class FakeVolumeAttachment:
+ """Fake one or more volume attachments."""
+
+ @staticmethod
+ def create_one_volume_attachment(attrs=None):
+ """Create a fake volume attachment.
+
+ :param attrs: A dictionary with all attributes of volume attachment
+ :return: A FakeResource object with id, status, etc.
+ """
+ attrs = attrs or {}
+
+ attachment_id = uuid.uuid4().hex
+ volume_id = attrs.pop('volume_id', None) or uuid.uuid4().hex
+ server_id = attrs.pop('instance', None) or uuid.uuid4().hex
+
+ # Set default attribute
+ attachment_info = {
+ 'id': attachment_id,
+ 'volume_id': volume_id,
+ 'instance': server_id,
+ 'status': random.choice([
+ 'attached',
+ 'attaching',
+ 'detached',
+ 'reserved',
+ 'error_attaching',
+ 'error_detaching',
+ 'deleted',
+ ]),
+ 'attach_mode': random.choice(['ro', 'rw']),
+ 'attached_at': '2015-09-16T09:28:52.000000',
+ 'detached_at': None,
+ 'connection_info': {
+ 'access_mode': 'rw',
+ 'attachment_id': attachment_id,
+ 'auth_method': 'CHAP',
+ 'auth_password': 'AcUZ8PpxLHwzypMC',
+ 'auth_username': '7j3EZQWT3rbE6pcSGKvK',
+ 'cacheable': False,
+ 'driver_volume_type': 'iscsi',
+ 'encrypted': False,
+ 'qos_specs': None,
+ 'target_discovered': False,
+ 'target_iqn':
+ f'iqn.2010-10.org.openstack:volume-{attachment_id}',
+ 'target_lun': '1',
+ 'target_portal': '192.168.122.170:3260',
+ 'volume_id': volume_id,
+ },
+ }
+
+ # Overwrite default attributes if there are some attributes set
+ attachment_info.update(attrs)
+
+ attachment = fakes.FakeResource(
+ None,
+ attachment_info,
+ loaded=True)
+ return attachment
+
+ @staticmethod
+ def create_volume_attachments(attrs=None, count=2):
+ """Create multiple fake volume attachments.
+
+ :param attrs: A dictionary with all attributes of volume attachment
+ :param count: The number of volume attachments to be faked
+ :return: A list of FakeResource objects
+ """
+ attachments = []
+
+ for n in range(0, count):
+ attachments.append(
+ FakeVolumeAttachment.create_one_volume_attachment(attrs))
+
+ return attachments
+
+ @staticmethod
+ def get_volume_attachments(attachments=None, count=2):
+ """Get an iterable MagicMock object with a list of faked volumes.
+
+ If attachments list is provided, then initialize the Mock object with
+ the list. Otherwise create one.
+
+ :param attachments: A list of FakeResource objects faking volume
+ attachments
+ :param count: The number of volume attachments to be faked
+ :return An iterable Mock object with side_effect set to a list of faked
+ volume attachments
+ """
+ if attachments is None:
+ attachments = FakeVolumeAttachment.create_volume_attachments(count)
+
+ return mock.Mock(side_effect=attachments)
diff --git a/openstackclient/tests/unit/volume/v3/test_volume_attachment.py b/openstackclient/tests/unit/volume/v3/test_volume_attachment.py
new file mode 100644
index 00000000..09f698e7
--- /dev/null
+++ b/openstackclient/tests/unit/volume/v3/test_volume_attachment.py
@@ -0,0 +1,560 @@
+# 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 cinderclient import api_versions
+from osc_lib.cli import format_columns
+from osc_lib import exceptions
+
+from openstackclient.tests.unit.compute.v2 import fakes as compute_fakes
+from openstackclient.tests.unit.identity.v3 import fakes as identity_fakes
+from openstackclient.tests.unit.volume.v3 import fakes as volume_fakes
+from openstackclient.volume.v3 import volume_attachment
+
+
+class TestVolumeAttachment(volume_fakes.TestVolume):
+
+ def setUp(self):
+ super().setUp()
+
+ self.volumes_mock = self.app.client_manager.volume.volumes
+ self.volumes_mock.reset_mock()
+
+ self.volume_attachments_mock = \
+ self.app.client_manager.volume.attachments
+ self.volume_attachments_mock.reset_mock()
+
+ self.projects_mock = self.app.client_manager.identity.projects
+ self.projects_mock.reset_mock()
+
+ self.servers_mock = self.app.client_manager.compute.servers
+ self.servers_mock.reset_mock()
+
+
+class TestVolumeAttachmentCreate(TestVolumeAttachment):
+
+ volume = volume_fakes.FakeVolume.create_one_volume()
+ server = compute_fakes.FakeServer.create_one_server()
+ volume_attachment = \
+ volume_fakes.FakeVolumeAttachment.create_one_volume_attachment(
+ attrs={'instance': server.id, 'volume_id': volume.id})
+
+ columns = (
+ 'ID',
+ 'Volume ID',
+ 'Instance ID',
+ 'Status',
+ 'Attach Mode',
+ 'Attached At',
+ 'Detached At',
+ 'Properties',
+ )
+ data = (
+ volume_attachment.id,
+ volume_attachment.volume_id,
+ volume_attachment.instance,
+ volume_attachment.status,
+ volume_attachment.attach_mode,
+ volume_attachment.attached_at,
+ volume_attachment.detached_at,
+ format_columns.DictColumn(volume_attachment.connection_info),
+ )
+
+ def setUp(self):
+ super().setUp()
+
+ self.volumes_mock.get.return_value = self.volume
+ self.servers_mock.get.return_value = self.server
+ self.volume_attachments_mock.create.return_value = \
+ self.volume_attachment
+
+ self.cmd = volume_attachment.CreateVolumeAttachment(self.app, None)
+
+ def test_volume_attachment_create(self):
+ self.app.client_manager.volume.api_version = \
+ api_versions.APIVersion('3.27')
+
+ arglist = [
+ self.volume.id,
+ self.server.id,
+ ]
+ verifylist = [
+ ('volume', self.volume.id),
+ ('server', self.server.id),
+ ('connect', False),
+ ('initiator', None),
+ ('ip', None),
+ ('host', None),
+ ('platform', None),
+ ('os_type', None),
+ ('multipath', False),
+ ('mountpoint', None),
+ ('mode', None),
+ ]
+ parsed_args = self.check_parser(self.cmd, arglist, verifylist)
+
+ columns, data = self.cmd.take_action(parsed_args)
+
+ self.volumes_mock.get.assert_called_once_with(self.volume.id)
+ self.servers_mock.get.assert_called_once_with(self.server.id)
+ self.volume_attachments_mock.create.assert_called_once_with(
+ self.volume.id, {}, self.server.id, None,
+ )
+ self.assertEqual(self.columns, columns)
+ self.assertCountEqual(self.data, data)
+
+ def test_volume_attachment_create_with_connect(self):
+ self.app.client_manager.volume.api_version = \
+ api_versions.APIVersion('3.54')
+
+ arglist = [
+ self.volume.id,
+ self.server.id,
+ '--connect',
+ '--initiator', 'iqn.1993-08.org.debian:01:cad181614cec',
+ '--ip', '192.168.1.20',
+ '--host', 'my-host',
+ '--platform', 'x86_64',
+ '--os-type', 'linux2',
+ '--multipath',
+ '--mountpoint', '/dev/vdb',
+ '--mode', 'null',
+ ]
+ verifylist = [
+ ('volume', self.volume.id),
+ ('server', self.server.id),
+ ('connect', True),
+ ('initiator', 'iqn.1993-08.org.debian:01:cad181614cec'),
+ ('ip', '192.168.1.20'),
+ ('host', 'my-host'),
+ ('platform', 'x86_64'),
+ ('os_type', 'linux2'),
+ ('multipath', True),
+ ('mountpoint', '/dev/vdb'),
+ ('mode', 'null'),
+ ]
+ parsed_args = self.check_parser(self.cmd, arglist, verifylist)
+
+ columns, data = self.cmd.take_action(parsed_args)
+
+ connect_info = dict([
+ ('initiator', 'iqn.1993-08.org.debian:01:cad181614cec'),
+ ('ip', '192.168.1.20'),
+ ('host', 'my-host'),
+ ('platform', 'x86_64'),
+ ('os_type', 'linux2'),
+ ('multipath', True),
+ ('mountpoint', '/dev/vdb'),
+ ])
+
+ self.volumes_mock.get.assert_called_once_with(self.volume.id)
+ self.servers_mock.get.assert_called_once_with(self.server.id)
+ self.volume_attachments_mock.create.assert_called_once_with(
+ self.volume.id, connect_info, self.server.id, 'null',
+ )
+ self.assertEqual(self.columns, columns)
+ self.assertCountEqual(self.data, data)
+
+ def test_volume_attachment_create_pre_v327(self):
+ self.app.client_manager.volume.api_version = \
+ api_versions.APIVersion('3.26')
+
+ arglist = [
+ self.volume.id,
+ self.server.id,
+ ]
+ verifylist = [
+ ('volume', self.volume.id),
+ ('server', self.server.id),
+ ]
+ parsed_args = self.check_parser(self.cmd, arglist, verifylist)
+
+ exc = self.assertRaises(
+ exceptions.CommandError,
+ self.cmd.take_action,
+ parsed_args)
+ self.assertIn(
+ '--os-volume-api-version 3.27 or greater is required',
+ str(exc))
+
+ def test_volume_attachment_create_with_mode_pre_v354(self):
+ self.app.client_manager.volume.api_version = \
+ api_versions.APIVersion('3.53')
+
+ arglist = [
+ self.volume.id,
+ self.server.id,
+ '--mode', 'rw',
+ ]
+ verifylist = [
+ ('volume', self.volume.id),
+ ('server', self.server.id),
+ ('mode', 'rw'),
+ ]
+ parsed_args = self.check_parser(self.cmd, arglist, verifylist)
+
+ exc = self.assertRaises(
+ exceptions.CommandError,
+ self.cmd.take_action,
+ parsed_args)
+ self.assertIn(
+ '--os-volume-api-version 3.54 or greater is required',
+ str(exc))
+
+ def test_volume_attachment_create_with_connect_missing_arg(self):
+ self.app.client_manager.volume.api_version = \
+ api_versions.APIVersion('3.54')
+
+ arglist = [
+ self.volume.id,
+ self.server.id,
+ '--initiator', 'iqn.1993-08.org.debian:01:cad181614cec',
+ ]
+ verifylist = [
+ ('volume', self.volume.id),
+ ('server', self.server.id),
+ ('connect', False),
+ ('initiator', 'iqn.1993-08.org.debian:01:cad181614cec'),
+ ]
+ parsed_args = self.check_parser(self.cmd, arglist, verifylist)
+
+ exc = self.assertRaises(
+ exceptions.CommandError,
+ self.cmd.take_action,
+ parsed_args)
+ self.assertIn(
+ 'You must specify the --connect option for any',
+ str(exc))
+
+
+class TestVolumeAttachmentDelete(TestVolumeAttachment):
+
+ volume_attachment = \
+ volume_fakes.FakeVolumeAttachment.create_one_volume_attachment()
+
+ def setUp(self):
+ super().setUp()
+
+ self.volume_attachments_mock.delete.return_value = None
+
+ self.cmd = volume_attachment.DeleteVolumeAttachment(self.app, None)
+
+ def test_volume_attachment_delete(self):
+ self.app.client_manager.volume.api_version = \
+ api_versions.APIVersion('3.27')
+
+ arglist = [
+ self.volume_attachment.id,
+ ]
+ verifylist = [
+ ('attachment', self.volume_attachment.id),
+ ]
+ parsed_args = self.check_parser(self.cmd, arglist, verifylist)
+
+ result = self.cmd.take_action(parsed_args)
+
+ self.volume_attachments_mock.delete.assert_called_once_with(
+ self.volume_attachment.id,
+ )
+ self.assertIsNone(result)
+
+ def test_volume_attachment_delete_pre_v327(self):
+ self.app.client_manager.volume.api_version = \
+ api_versions.APIVersion('3.26')
+
+ arglist = [
+ self.volume_attachment.id,
+ ]
+ verifylist = [
+ ('attachment', self.volume_attachment.id),
+ ]
+ parsed_args = self.check_parser(self.cmd, arglist, verifylist)
+
+ exc = self.assertRaises(
+ exceptions.CommandError,
+ self.cmd.take_action,
+ parsed_args)
+ self.assertIn(
+ '--os-volume-api-version 3.27 or greater is required',
+ str(exc))
+
+
+class TestVolumeAttachmentSet(TestVolumeAttachment):
+
+ volume_attachment = \
+ volume_fakes.FakeVolumeAttachment.create_one_volume_attachment()
+
+ columns = (
+ 'ID',
+ 'Volume ID',
+ 'Instance ID',
+ 'Status',
+ 'Attach Mode',
+ 'Attached At',
+ 'Detached At',
+ 'Properties',
+ )
+ data = (
+ volume_attachment.id,
+ volume_attachment.volume_id,
+ volume_attachment.instance,
+ volume_attachment.status,
+ volume_attachment.attach_mode,
+ volume_attachment.attached_at,
+ volume_attachment.detached_at,
+ format_columns.DictColumn(volume_attachment.connection_info),
+ )
+
+ def setUp(self):
+ super().setUp()
+
+ self.volume_attachments_mock.update.return_value = \
+ self.volume_attachment
+
+ self.cmd = volume_attachment.SetVolumeAttachment(self.app, None)
+
+ def test_volume_attachment_set(self):
+ self.app.client_manager.volume.api_version = \
+ api_versions.APIVersion('3.27')
+
+ arglist = [
+ self.volume_attachment.id,
+ '--initiator', 'iqn.1993-08.org.debian:01:cad181614cec',
+ '--ip', '192.168.1.20',
+ '--host', 'my-host',
+ '--platform', 'x86_64',
+ '--os-type', 'linux2',
+ '--multipath',
+ '--mountpoint', '/dev/vdb',
+ ]
+ verifylist = [
+ ('attachment', self.volume_attachment.id),
+ ('initiator', 'iqn.1993-08.org.debian:01:cad181614cec'),
+ ('ip', '192.168.1.20'),
+ ('host', 'my-host'),
+ ('platform', 'x86_64'),
+ ('os_type', 'linux2'),
+ ('multipath', True),
+ ('mountpoint', '/dev/vdb'),
+ ]
+ parsed_args = self.check_parser(self.cmd, arglist, verifylist)
+
+ columns, data = self.cmd.take_action(parsed_args)
+
+ connect_info = dict([
+ ('initiator', 'iqn.1993-08.org.debian:01:cad181614cec'),
+ ('ip', '192.168.1.20'),
+ ('host', 'my-host'),
+ ('platform', 'x86_64'),
+ ('os_type', 'linux2'),
+ ('multipath', True),
+ ('mountpoint', '/dev/vdb'),
+ ])
+
+ self.volume_attachments_mock.update.assert_called_once_with(
+ self.volume_attachment.id, connect_info,
+ )
+ self.assertEqual(self.columns, columns)
+ self.assertCountEqual(self.data, data)
+
+ def test_volume_attachment_set_pre_v327(self):
+ self.app.client_manager.volume.api_version = \
+ api_versions.APIVersion('3.26')
+
+ arglist = [
+ self.volume_attachment.id,
+ '--initiator', 'iqn.1993-08.org.debian:01:cad181614cec',
+ ]
+ verifylist = [
+ ('attachment', self.volume_attachment.id),
+ ('initiator', 'iqn.1993-08.org.debian:01:cad181614cec'),
+ ]
+ parsed_args = self.check_parser(self.cmd, arglist, verifylist)
+
+ exc = self.assertRaises(
+ exceptions.CommandError,
+ self.cmd.take_action,
+ parsed_args)
+ self.assertIn(
+ '--os-volume-api-version 3.27 or greater is required',
+ str(exc))
+
+
+class TestVolumeAttachmentComplete(TestVolumeAttachment):
+
+ volume_attachment = \
+ volume_fakes.FakeVolumeAttachment.create_one_volume_attachment()
+
+ def setUp(self):
+ super().setUp()
+
+ self.volume_attachments_mock.complete.return_value = None
+
+ self.cmd = volume_attachment.CompleteVolumeAttachment(self.app, None)
+
+ def test_volume_attachment_complete(self):
+ self.app.client_manager.volume.api_version = \
+ api_versions.APIVersion('3.44')
+
+ arglist = [
+ self.volume_attachment.id,
+ ]
+ verifylist = [
+ ('attachment', self.volume_attachment.id),
+ ]
+ parsed_args = self.check_parser(self.cmd, arglist, verifylist)
+
+ result = self.cmd.take_action(parsed_args)
+
+ self.volume_attachments_mock.complete.assert_called_once_with(
+ self.volume_attachment.id,
+ )
+ self.assertIsNone(result)
+
+ def test_volume_attachment_complete_pre_v344(self):
+ self.app.client_manager.volume.api_version = \
+ api_versions.APIVersion('3.43')
+
+ arglist = [
+ self.volume_attachment.id,
+ ]
+ verifylist = [
+ ('attachment', self.volume_attachment.id),
+ ]
+ parsed_args = self.check_parser(self.cmd, arglist, verifylist)
+
+ exc = self.assertRaises(
+ exceptions.CommandError,
+ self.cmd.take_action,
+ parsed_args)
+ self.assertIn(
+ '--os-volume-api-version 3.44 or greater is required',
+ str(exc))
+
+
+class TestVolumeAttachmentList(TestVolumeAttachment):
+
+ project = identity_fakes.FakeProject.create_one_project()
+ volume_attachments = \
+ volume_fakes.FakeVolumeAttachment.create_volume_attachments()
+
+ columns = (
+ 'ID',
+ 'Volume ID',
+ 'Server ID',
+ 'Status',
+ )
+ data = [
+ (
+ volume_attachment.id,
+ volume_attachment.volume_id,
+ volume_attachment.instance,
+ volume_attachment.status,
+ ) for volume_attachment in volume_attachments
+ ]
+
+ def setUp(self):
+ super().setUp()
+
+ self.projects_mock.get.return_value = self.project
+ self.volume_attachments_mock.list.return_value = \
+ self.volume_attachments
+
+ self.cmd = volume_attachment.ListVolumeAttachment(self.app, None)
+
+ def test_volume_attachment_list(self):
+ self.app.client_manager.volume.api_version = \
+ api_versions.APIVersion('3.27')
+
+ arglist = []
+ verifylist = [
+ ('project', None),
+ ('all_projects', False),
+ ('volume_id', None),
+ ('status', None),
+ ('marker', None),
+ ('limit', None),
+ ]
+ parsed_args = self.check_parser(self.cmd, arglist, verifylist)
+
+ columns, data = self.cmd.take_action(parsed_args)
+
+ self.volume_attachments_mock.list.assert_called_once_with(
+ search_opts={
+ 'all_tenants': False,
+ 'project_id': None,
+ 'status': None,
+ 'volume_id': None,
+ },
+ marker=None,
+ limit=None,
+ )
+ self.assertEqual(self.columns, columns)
+ self.assertCountEqual(tuple(self.data), data)
+
+ def test_volume_attachment_list_with_options(self):
+ self.app.client_manager.volume.api_version = \
+ api_versions.APIVersion('3.27')
+
+ arglist = [
+ '--project', self.project.name,
+ '--volume-id', 'volume-id',
+ '--status', 'attached',
+ '--marker', 'volume-attachment-id',
+ '--limit', '2',
+ ]
+ verifylist = [
+ ('project', self.project.name),
+ ('all_projects', False),
+ ('volume_id', 'volume-id'),
+ ('status', 'attached'),
+ ('marker', 'volume-attachment-id'),
+ ('limit', 2),
+ ]
+ parsed_args = self.check_parser(self.cmd, arglist, verifylist)
+
+ columns, data = self.cmd.take_action(parsed_args)
+
+ self.volume_attachments_mock.list.assert_called_once_with(
+ search_opts={
+ 'all_tenants': True,
+ 'project_id': self.project.id,
+ 'status': 'attached',
+ 'volume_id': 'volume-id',
+ },
+ marker='volume-attachment-id',
+ limit=2,
+ )
+ self.assertEqual(self.columns, columns)
+ self.assertCountEqual(tuple(self.data), data)
+
+ def test_volume_attachment_list_pre_v327(self):
+ self.app.client_manager.volume.api_version = \
+ api_versions.APIVersion('3.26')
+
+ arglist = []
+ verifylist = [
+ ('project', None),
+ ('all_projects', False),
+ ('volume_id', None),
+ ('status', None),
+ ('marker', None),
+ ('limit', None),
+ ]
+ parsed_args = self.check_parser(self.cmd, arglist, verifylist)
+
+ exc = self.assertRaises(
+ exceptions.CommandError,
+ self.cmd.take_action,
+ parsed_args)
+ self.assertIn(
+ '--os-volume-api-version 3.27 or greater is required',
+ str(exc))