Deprecate VolumeDetail and BackupDetail classes

We have introduced altering base_path for operations on resources. 
Volumes and Backups were not switched to this new method and were still 
relying on returning inherited classes VolumeDetail and BackupDetail. 
This change allow doing any supported operation on a resource returned 
by list immediately without any conversion or re-fetch.

Change-Id: Ia095c53f1d04f1c119c6a6f8a38c5bfd60dc8a67
This commit is contained in:
Artem Goncharov 2019-03-09 11:12:38 +01:00
parent 1066c9385f
commit 0eaf2c95d3
15 changed files with 111 additions and 273 deletions

View File

@ -10,12 +10,3 @@ The ``Backup`` class inherits from :class:`~openstack.resource.Resource`.
.. autoclass:: openstack.block_storage.v2.backup.Backup .. autoclass:: openstack.block_storage.v2.backup.Backup
:members: :members:
The BackupDetail Class
----------------------
The ``BackupDetail`` class inherits from
:class:`~openstack.block_storage.v2.backup.Backup`.
.. autoclass:: openstack.block_storage.v2.backup.BackupDetail
:members:

View File

@ -10,12 +10,3 @@ The ``Volume`` class inherits from :class:`~openstack.resource.Resource`.
.. autoclass:: openstack.block_storage.v2.volume.Volume .. autoclass:: openstack.block_storage.v2.volume.Volume
:members: :members:
The VolumeDetail Class
----------------------
The ``VolumeDetail`` class inherits from
:class:`~openstack.block_storage.v2.volume.Volume`.
.. autoclass:: openstack.block_storage.v2.volume.VolumeDetail
:members:

View File

@ -147,11 +147,9 @@ class Proxy(_base_proxy.BaseBlockStorageProxy):
def volumes(self, details=True, **query): def volumes(self, details=True, **query):
"""Retrieve a generator of volumes """Retrieve a generator of volumes
:param bool details: When set to ``False`` :param bool details: When set to ``False`` no extended attributes
:class:`~openstack.block_storage.v2.volume.Volume` objects will be returned. The default, ``True``, will cause objects with
will be returned. The default, ``True``, will cause additional attributes to be returned.
:class:`~openstack.block_storage.v2.volume.VolumeDetail`
objects to be returned.
:param kwargs query: Optional query parameters to be sent to limit :param kwargs query: Optional query parameters to be sent to limit
the volumes being returned. Available parameters include: the volumes being returned. Available parameters include:
@ -162,8 +160,8 @@ class Proxy(_base_proxy.BaseBlockStorageProxy):
:returns: A generator of volume objects. :returns: A generator of volume objects.
""" """
volume = _volume.VolumeDetail if details else _volume.Volume base_path = '/volumes/detail' if details else None
return self._list(volume, **query) return self._list(_volume.Volume, base_path=base_path, **query)
def create_volume(self, **attrs): def create_volume(self, **attrs):
"""Create a new volume from attributes """Create a new volume from attributes
@ -214,11 +212,9 @@ class Proxy(_base_proxy.BaseBlockStorageProxy):
def backups(self, details=True, **query): def backups(self, details=True, **query):
"""Retrieve a generator of backups """Retrieve a generator of backups
:param bool details: When set to ``False`` :param bool details: When set to ``False`` no additional details will
:class:`~openstack.block_storage.v2.backup.Backup` objects be returned. The default, ``True``, will cause objects with
will be returned. The default, ``True``, will cause additional attributes to be returned.
:class:`~openstack.block_storage.v2.backup.BackupDetail`
objects to be returned.
:param dict query: Optional query parameters to be sent to limit the :param dict query: Optional query parameters to be sent to limit the
resources being returned: resources being returned:
@ -239,8 +235,8 @@ class Proxy(_base_proxy.BaseBlockStorageProxy):
raise exceptions.SDKException( raise exceptions.SDKException(
'Object-store service is required for block-store backups' 'Object-store service is required for block-store backups'
) )
backup = _backup.BackupDetail if details else _backup.Backup base_path = '/backups/detail' if details else None
return self._list(backup, **query) return self._list(_backup.Backup, base_path=base_path, **query)
def get_backup(self, backup): def get_backup(self, backup):
"""Get a backup """Get a backup

View File

@ -90,11 +90,4 @@ class Backup(resource.Resource):
return self return self
class BackupDetail(Backup): BackupDetail = Backup
"""Volume Backup with Details"""
base_path = "/backups/detail"
# capabilities
allow_list = True
#: Properties

View File

@ -75,32 +75,6 @@ class Volume(resource.Resource):
#: The timestamp of this volume creation. #: The timestamp of this volume creation.
created_at = resource.Body("created_at") created_at = resource.Body("created_at")
def _action(self, session, body):
"""Preform volume actions given the message body."""
# NOTE: This is using Volume.base_path instead of self.base_path
# as both Volume and VolumeDetail instances can be acted on, but
# the URL used is sans any additional /detail/ part.
url = utils.urljoin(Volume.base_path, self.id, 'action')
headers = {'Accept': ''}
return session.post(url, json=body, headers=headers)
def extend(self, session, size):
"""Extend a volume size."""
body = {'os-extend': {'new_size': size}}
self._action(session, body)
class VolumeDetail(Volume):
base_path = "/volumes/detail"
# capabilities
allow_fetch = False
allow_create = False
allow_delete = False
allow_commit = False
allow_list = True
#: The volume's current back-end. #: The volume's current back-end.
host = resource.Body("os-vol-host-attr:host") host = resource.Body("os-vol-host-attr:host")
#: The project ID associated with current back-end. #: The project ID associated with current back-end.
@ -123,3 +97,20 @@ class VolumeDetail(Volume):
#: ``True`` if this volume is encrypted, ``False`` if not. #: ``True`` if this volume is encrypted, ``False`` if not.
#: *Type: bool* #: *Type: bool*
is_encrypted = resource.Body("encrypted", type=format.BoolStr) is_encrypted = resource.Body("encrypted", type=format.BoolStr)
def _action(self, session, body):
"""Preform volume actions given the message body."""
# NOTE: This is using Volume.base_path instead of self.base_path
# as both Volume and VolumeDetail instances can be acted on, but
# the URL used is sans any additional /detail/ part.
url = utils.urljoin(Volume.base_path, self.id, 'action')
headers = {'Accept': ''}
return session.post(url, json=body, headers=headers)
def extend(self, session, size):
"""Extend a volume size."""
body = {'os-extend': {'new_size': size}}
self._action(session, body)
VolumeDetail = Volume

View File

@ -147,11 +147,9 @@ class Proxy(_base_proxy.BaseBlockStorageProxy):
def volumes(self, details=True, **query): def volumes(self, details=True, **query):
"""Retrieve a generator of volumes """Retrieve a generator of volumes
:param bool details: When set to ``False`` :param bool details: When set to ``False`` no extended attributes
:class:`~openstack.block_storage.v3.volume.Volume` objects will be returned. The default, ``True``, will cause objects with
will be returned. The default, ``True``, will cause additional attributes to be returned.
:class:`~openstack.block_storage.v3.volume.VolumeDetail`
objects to be returned.
:param kwargs query: Optional query parameters to be sent to limit :param kwargs query: Optional query parameters to be sent to limit
the volumes being returned. Available parameters include: the volumes being returned. Available parameters include:
@ -162,8 +160,8 @@ class Proxy(_base_proxy.BaseBlockStorageProxy):
:returns: A generator of volume objects. :returns: A generator of volume objects.
""" """
volume = _volume.VolumeDetail if details else _volume.Volume base_path = '/volumes/detail' if details else None
return self._list(volume, **query) return self._list(_volume.Volume, base_path=base_path, **query)
def create_volume(self, **attrs): def create_volume(self, **attrs):
"""Create a new volume from attributes """Create a new volume from attributes
@ -215,10 +213,8 @@ class Proxy(_base_proxy.BaseBlockStorageProxy):
"""Retrieve a generator of backups """Retrieve a generator of backups
:param bool details: When set to ``False`` :param bool details: When set to ``False``
:class:`~openstack.block_storage.v3.backup.Backup` objects no additional details will be returned. The default, ``True``,
will be returned. The default, ``True``, will cause will cause objects with additional attributes to be returned.
:class:`~openstack.block_storage.v3.backup.BackupDetail`
objects to be returned.
:param dict query: Optional query parameters to be sent to limit the :param dict query: Optional query parameters to be sent to limit the
resources being returned: resources being returned:
@ -239,8 +235,8 @@ class Proxy(_base_proxy.BaseBlockStorageProxy):
raise exceptions.SDKException( raise exceptions.SDKException(
'Object-store service is required for block-store backups' 'Object-store service is required for block-store backups'
) )
backup = _backup.BackupDetail if details else _backup.Backup base_path = '/backups/detail' if details else None
return self._list(backup, **query) return self._list(_backup.Backup, base_path=base_path, **query)
def get_backup(self, backup): def get_backup(self, backup):
"""Get a backup """Get a backup

View File

@ -90,11 +90,4 @@ class Backup(resource.Resource):
return self return self
class BackupDetail(Backup): BackupDetail = Backup
"""Volume Backup with Details"""
base_path = "/backups/detail"
# capabilities
allow_list = True
#: Properties

View File

@ -75,32 +75,6 @@ class Volume(resource.Resource):
#: The timestamp of this volume creation. #: The timestamp of this volume creation.
created_at = resource.Body("created_at") created_at = resource.Body("created_at")
def _action(self, session, body):
"""Preform volume actions given the message body."""
# NOTE: This is using Volume.base_path instead of self.base_path
# as both Volume and VolumeDetail instances can be acted on, but
# the URL used is sans any additional /detail/ part.
url = utils.urljoin(Volume.base_path, self.id, 'action')
headers = {'Accept': ''}
return session.post(url, json=body, headers=headers)
def extend(self, session, size):
"""Extend a volume size."""
body = {'os-extend': {'new_size': size}}
self._action(session, body)
class VolumeDetail(Volume):
base_path = "/volumes/detail"
# capabilities
allow_fetch = False
allow_create = False
allow_delete = False
allow_commit = False
allow_list = True
#: The volume's current back-end. #: The volume's current back-end.
host = resource.Body("os-vol-host-attr:host") host = resource.Body("os-vol-host-attr:host")
#: The project ID associated with current back-end. #: The project ID associated with current back-end.
@ -123,3 +97,20 @@ class VolumeDetail(Volume):
#: ``True`` if this volume is encrypted, ``False`` if not. #: ``True`` if this volume is encrypted, ``False`` if not.
#: *Type: bool* #: *Type: bool*
is_encrypted = resource.Body("encrypted", type=format.BoolStr) is_encrypted = resource.Body("encrypted", type=format.BoolStr)
def _action(self, session, body):
"""Preform volume actions given the message body."""
# NOTE: This is using Volume.base_path instead of self.base_path
# as both Volume and VolumeDetail instances can be acted on, but
# the URL used is sans any additional /detail/ part.
url = utils.urljoin(Volume.base_path, self.id, 'action')
headers = {'Accept': ''}
return session.post(url, json=body, headers=headers)
def extend(self, session, size):
"""Extend a volume size."""
body = {'os-extend': {'new_size': size}}
self._action(session, body)
VolumeDetail = Volume

View File

@ -10,7 +10,6 @@
# License for the specific language governing permissions and limitations # License for the specific language governing permissions and limitations
# under the License. # under the License.
import copy
import mock import mock
from keystoneauth1 import adapter from keystoneauth1 import adapter
@ -38,12 +37,6 @@ BACKUP = {
"has_dependent_backups": False "has_dependent_backups": False
} }
DETAILS = {
}
BACKUP_DETAIL = copy.copy(BACKUP)
BACKUP_DETAIL.update(DETAILS)
class TestBackup(base.TestCase): class TestBackup(base.TestCase):
@ -92,30 +85,3 @@ class TestBackup(base.TestCase):
self.assertEqual(BACKUP["size"], sot.size) self.assertEqual(BACKUP["size"], sot.size)
self.assertEqual(BACKUP["has_dependent_backups"], self.assertEqual(BACKUP["has_dependent_backups"],
sot.has_dependent_backups) sot.has_dependent_backups)
class TestBackupDetail(base.TestCase):
def test_basic(self):
sot = backup.BackupDetail(BACKUP_DETAIL)
self.assertIsInstance(sot, backup.Backup)
self.assertEqual("/backups/detail", sot.base_path)
def test_create(self):
sot = backup.Backup(**BACKUP_DETAIL)
self.assertEqual(BACKUP_DETAIL["id"], sot.id)
self.assertEqual(BACKUP_DETAIL["name"], sot.name)
self.assertEqual(BACKUP_DETAIL["status"], sot.status)
self.assertEqual(BACKUP_DETAIL["container"], sot.container)
self.assertEqual(BACKUP_DETAIL["availability_zone"],
sot.availability_zone)
self.assertEqual(BACKUP_DETAIL["created_at"], sot.created_at)
self.assertEqual(BACKUP_DETAIL["updated_at"], sot.updated_at)
self.assertEqual(BACKUP_DETAIL["description"], sot.description)
self.assertEqual(BACKUP_DETAIL["fail_reason"], sot.fail_reason)
self.assertEqual(BACKUP_DETAIL["volume_id"], sot.volume_id)
self.assertEqual(BACKUP_DETAIL["object_count"], sot.object_count)
self.assertEqual(BACKUP_DETAIL["is_incremental"], sot.is_incremental)
self.assertEqual(BACKUP_DETAIL["size"], sot.size)
self.assertEqual(BACKUP_DETAIL["has_dependent_backups"],
sot.has_dependent_backups)

View File

@ -70,9 +70,10 @@ class TestVolumeProxy(test_proxy_base.TestProxyBase):
self.verify_get(self.proxy.get_volume, volume.Volume) self.verify_get(self.proxy.get_volume, volume.Volume)
def test_volumes_detailed(self): def test_volumes_detailed(self):
self.verify_list(self.proxy.volumes, volume.VolumeDetail, self.verify_list(self.proxy.volumes, volume.Volume,
method_kwargs={"details": True, "query": 1}, method_kwargs={"details": True, "query": 1},
expected_kwargs={"query": 1}) expected_kwargs={"query": 1,
"base_path": "/volumes/detail"})
def test_volumes_not_detailed(self): def test_volumes_not_detailed(self):
self.verify_list(self.proxy.volumes, volume.Volume, self.verify_list(self.proxy.volumes, volume.Volume,
@ -101,9 +102,10 @@ class TestVolumeProxy(test_proxy_base.TestProxyBase):
# NOTE: mock has_service # NOTE: mock has_service
self.proxy._connection = mock.Mock() self.proxy._connection = mock.Mock()
self.proxy._connection.has_service = mock.Mock(return_value=True) self.proxy._connection.has_service = mock.Mock(return_value=True)
self.verify_list(self.proxy.backups, backup.BackupDetail, self.verify_list(self.proxy.backups, backup.Backup,
method_kwargs={"details": True, "query": 1}, method_kwargs={"details": True, "query": 1},
expected_kwargs={"query": 1}) expected_kwargs={"query": 1,
"base_path": "/backups/detail"})
def test_backups_not_detailed(self): def test_backups_not_detailed(self):
# NOTE: mock has_service # NOTE: mock has_service

View File

@ -10,7 +10,6 @@
# License for the specific language governing permissions and limitations # License for the specific language governing permissions and limitations
# under the License. # under the License.
import copy
import mock import mock
from openstack.tests.unit import base from openstack.tests.unit import base
@ -42,10 +41,7 @@ VOLUME = {
"metadata": {}, "metadata": {},
"volume_image_metadata": IMAGE_METADATA, "volume_image_metadata": IMAGE_METADATA,
"id": FAKE_ID, "id": FAKE_ID,
"size": 10 "size": 10,
}
DETAILS = {
"os-vol-host-attr:host": "127.0.0.1", "os-vol-host-attr:host": "127.0.0.1",
"os-vol-tenant-attr:tenant_id": "some tenant", "os-vol-tenant-attr:tenant_id": "some tenant",
"os-vol-mig-status-attr:migstat": "done", "os-vol-mig-status-attr:migstat": "done",
@ -55,12 +51,9 @@ DETAILS = {
"consistencygroup_id": "123asf-asdf123", "consistencygroup_id": "123asf-asdf123",
"os-volume-replication:driver_data": "ahasadfasdfasdfasdfsdf", "os-volume-replication:driver_data": "ahasadfasdfasdfasdfsdf",
"snapshot_id": "93c2e2aa-7744-4fd6-a31a-80c4726b08d7", "snapshot_id": "93c2e2aa-7744-4fd6-a31a-80c4726b08d7",
"encrypted": "false", "encrypted": "false"
} }
VOLUME_DETAIL = copy.copy(VOLUME)
VOLUME_DETAIL.update(DETAILS)
class TestVolume(base.TestCase): class TestVolume(base.TestCase):
@ -108,6 +101,23 @@ class TestVolume(base.TestCase):
sot.volume_image_metadata) sot.volume_image_metadata)
self.assertEqual(VOLUME["size"], sot.size) self.assertEqual(VOLUME["size"], sot.size)
self.assertEqual(VOLUME["imageRef"], sot.image_id) self.assertEqual(VOLUME["imageRef"], sot.image_id)
self.assertEqual(VOLUME["os-vol-host-attr:host"], sot.host)
self.assertEqual(VOLUME["os-vol-tenant-attr:tenant_id"],
sot.project_id)
self.assertEqual(VOLUME["os-vol-mig-status-attr:migstat"],
sot.migration_status)
self.assertEqual(VOLUME["os-vol-mig-status-attr:name_id"],
sot.migration_id)
self.assertEqual(VOLUME["replication_status"],
sot.replication_status)
self.assertEqual(
VOLUME["os-volume-replication:extended_status"],
sot.extended_replication_status)
self.assertEqual(VOLUME["consistencygroup_id"],
sot.consistency_group_id)
self.assertEqual(VOLUME["os-volume-replication:driver_data"],
sot.replication_driver_data)
self.assertFalse(sot.is_encrypted)
def test_extend(self): def test_extend(self):
sot = volume.Volume(**VOLUME) sot = volume.Volume(**VOLUME)
@ -118,36 +128,3 @@ class TestVolume(base.TestCase):
body = {"os-extend": {"new_size": "20"}} body = {"os-extend": {"new_size": "20"}}
headers = {'Accept': ''} headers = {'Accept': ''}
self.sess.post.assert_called_with(url, json=body, headers=headers) self.sess.post.assert_called_with(url, json=body, headers=headers)
class TestVolumeDetail(base.TestCase):
def test_basic(self):
sot = volume.VolumeDetail(VOLUME_DETAIL)
self.assertIsInstance(sot, volume.Volume)
self.assertEqual("/volumes/detail", sot.base_path)
self.assertFalse(sot.allow_fetch)
self.assertFalse(sot.allow_commit)
self.assertFalse(sot.allow_create)
self.assertFalse(sot.allow_delete)
self.assertTrue(sot.allow_list)
def test_create(self):
sot = volume.VolumeDetail(**VOLUME_DETAIL)
self.assertEqual(VOLUME_DETAIL["os-vol-host-attr:host"], sot.host)
self.assertEqual(VOLUME_DETAIL["os-vol-tenant-attr:tenant_id"],
sot.project_id)
self.assertEqual(VOLUME_DETAIL["os-vol-mig-status-attr:migstat"],
sot.migration_status)
self.assertEqual(VOLUME_DETAIL["os-vol-mig-status-attr:name_id"],
sot.migration_id)
self.assertEqual(VOLUME_DETAIL["replication_status"],
sot.replication_status)
self.assertEqual(
VOLUME_DETAIL["os-volume-replication:extended_status"],
sot.extended_replication_status)
self.assertEqual(VOLUME_DETAIL["consistencygroup_id"],
sot.consistency_group_id)
self.assertEqual(VOLUME_DETAIL["os-volume-replication:driver_data"],
sot.replication_driver_data)
self.assertFalse(sot.is_encrypted)

View File

@ -10,7 +10,6 @@
# License for the specific language governing permissions and limitations # License for the specific language governing permissions and limitations
# under the License. # under the License.
import copy
import mock import mock
from keystoneauth1 import adapter from keystoneauth1 import adapter
@ -38,12 +37,6 @@ BACKUP = {
"has_dependent_backups": False "has_dependent_backups": False
} }
DETAILS = {
}
BACKUP_DETAIL = copy.copy(BACKUP)
BACKUP_DETAIL.update(DETAILS)
class TestBackup(base.TestCase): class TestBackup(base.TestCase):
@ -92,30 +85,3 @@ class TestBackup(base.TestCase):
self.assertEqual(BACKUP["size"], sot.size) self.assertEqual(BACKUP["size"], sot.size)
self.assertEqual(BACKUP["has_dependent_backups"], self.assertEqual(BACKUP["has_dependent_backups"],
sot.has_dependent_backups) sot.has_dependent_backups)
class TestBackupDetail(base.TestCase):
def test_basic(self):
sot = backup.BackupDetail(BACKUP_DETAIL)
self.assertIsInstance(sot, backup.Backup)
self.assertEqual("/backups/detail", sot.base_path)
def test_create(self):
sot = backup.Backup(**BACKUP_DETAIL)
self.assertEqual(BACKUP_DETAIL["id"], sot.id)
self.assertEqual(BACKUP_DETAIL["name"], sot.name)
self.assertEqual(BACKUP_DETAIL["status"], sot.status)
self.assertEqual(BACKUP_DETAIL["container"], sot.container)
self.assertEqual(BACKUP_DETAIL["availability_zone"],
sot.availability_zone)
self.assertEqual(BACKUP_DETAIL["created_at"], sot.created_at)
self.assertEqual(BACKUP_DETAIL["updated_at"], sot.updated_at)
self.assertEqual(BACKUP_DETAIL["description"], sot.description)
self.assertEqual(BACKUP_DETAIL["fail_reason"], sot.fail_reason)
self.assertEqual(BACKUP_DETAIL["volume_id"], sot.volume_id)
self.assertEqual(BACKUP_DETAIL["object_count"], sot.object_count)
self.assertEqual(BACKUP_DETAIL["is_incremental"], sot.is_incremental)
self.assertEqual(BACKUP_DETAIL["size"], sot.size)
self.assertEqual(BACKUP_DETAIL["has_dependent_backups"],
sot.has_dependent_backups)

View File

@ -70,9 +70,10 @@ class TestVolumeProxy(test_proxy_base.TestProxyBase):
self.verify_get(self.proxy.get_volume, volume.Volume) self.verify_get(self.proxy.get_volume, volume.Volume)
def test_volumes_detailed(self): def test_volumes_detailed(self):
self.verify_list(self.proxy.volumes, volume.VolumeDetail, self.verify_list(self.proxy.volumes, volume.Volume,
method_kwargs={"details": True, "query": 1}, method_kwargs={"details": True, "query": 1},
expected_kwargs={"query": 1}) expected_kwargs={"query": 1,
"base_path": "/volumes/detail"})
def test_volumes_not_detailed(self): def test_volumes_not_detailed(self):
self.verify_list(self.proxy.volumes, volume.Volume, self.verify_list(self.proxy.volumes, volume.Volume,
@ -101,9 +102,10 @@ class TestVolumeProxy(test_proxy_base.TestProxyBase):
# NOTE: mock has_service # NOTE: mock has_service
self.proxy._connection = mock.Mock() self.proxy._connection = mock.Mock()
self.proxy._connection.has_service = mock.Mock(return_value=True) self.proxy._connection.has_service = mock.Mock(return_value=True)
self.verify_list(self.proxy.backups, backup.BackupDetail, self.verify_list(self.proxy.backups, backup.Backup,
method_kwargs={"details": True, "query": 1}, method_kwargs={"details": True, "query": 1},
expected_kwargs={"query": 1}) expected_kwargs={"query": 1,
"base_path": "/backups/detail"})
def test_backups_not_detailed(self): def test_backups_not_detailed(self):
# NOTE: mock has_service # NOTE: mock has_service

View File

@ -10,7 +10,6 @@
# License for the specific language governing permissions and limitations # License for the specific language governing permissions and limitations
# under the License. # under the License.
import copy
import mock import mock
from openstack.tests.unit import base from openstack.tests.unit import base
@ -42,10 +41,7 @@ VOLUME = {
"metadata": {}, "metadata": {},
"volume_image_metadata": IMAGE_METADATA, "volume_image_metadata": IMAGE_METADATA,
"id": FAKE_ID, "id": FAKE_ID,
"size": 10 "size": 10,
}
DETAILS = {
"os-vol-host-attr:host": "127.0.0.1", "os-vol-host-attr:host": "127.0.0.1",
"os-vol-tenant-attr:tenant_id": "some tenant", "os-vol-tenant-attr:tenant_id": "some tenant",
"os-vol-mig-status-attr:migstat": "done", "os-vol-mig-status-attr:migstat": "done",
@ -55,12 +51,9 @@ DETAILS = {
"consistencygroup_id": "123asf-asdf123", "consistencygroup_id": "123asf-asdf123",
"os-volume-replication:driver_data": "ahasadfasdfasdfasdfsdf", "os-volume-replication:driver_data": "ahasadfasdfasdfasdfsdf",
"snapshot_id": "93c2e2aa-7744-4fd6-a31a-80c4726b08d7", "snapshot_id": "93c2e2aa-7744-4fd6-a31a-80c4726b08d7",
"encrypted": "false", "encrypted": "false"
} }
VOLUME_DETAIL = copy.copy(VOLUME)
VOLUME_DETAIL.update(DETAILS)
class TestVolume(base.TestCase): class TestVolume(base.TestCase):
@ -108,6 +101,23 @@ class TestVolume(base.TestCase):
sot.volume_image_metadata) sot.volume_image_metadata)
self.assertEqual(VOLUME["size"], sot.size) self.assertEqual(VOLUME["size"], sot.size)
self.assertEqual(VOLUME["imageRef"], sot.image_id) self.assertEqual(VOLUME["imageRef"], sot.image_id)
self.assertEqual(VOLUME["os-vol-host-attr:host"], sot.host)
self.assertEqual(VOLUME["os-vol-tenant-attr:tenant_id"],
sot.project_id)
self.assertEqual(VOLUME["os-vol-mig-status-attr:migstat"],
sot.migration_status)
self.assertEqual(VOLUME["os-vol-mig-status-attr:name_id"],
sot.migration_id)
self.assertEqual(VOLUME["replication_status"],
sot.replication_status)
self.assertEqual(
VOLUME["os-volume-replication:extended_status"],
sot.extended_replication_status)
self.assertEqual(VOLUME["consistencygroup_id"],
sot.consistency_group_id)
self.assertEqual(VOLUME["os-volume-replication:driver_data"],
sot.replication_driver_data)
self.assertFalse(sot.is_encrypted)
def test_extend(self): def test_extend(self):
sot = volume.Volume(**VOLUME) sot = volume.Volume(**VOLUME)
@ -118,36 +128,3 @@ class TestVolume(base.TestCase):
body = {"os-extend": {"new_size": "20"}} body = {"os-extend": {"new_size": "20"}}
headers = {'Accept': ''} headers = {'Accept': ''}
self.sess.post.assert_called_with(url, json=body, headers=headers) self.sess.post.assert_called_with(url, json=body, headers=headers)
class TestVolumeDetail(base.TestCase):
def test_basic(self):
sot = volume.VolumeDetail(VOLUME_DETAIL)
self.assertIsInstance(sot, volume.Volume)
self.assertEqual("/volumes/detail", sot.base_path)
self.assertFalse(sot.allow_fetch)
self.assertFalse(sot.allow_commit)
self.assertFalse(sot.allow_create)
self.assertFalse(sot.allow_delete)
self.assertTrue(sot.allow_list)
def test_create(self):
sot = volume.VolumeDetail(**VOLUME_DETAIL)
self.assertEqual(VOLUME_DETAIL["os-vol-host-attr:host"], sot.host)
self.assertEqual(VOLUME_DETAIL["os-vol-tenant-attr:tenant_id"],
sot.project_id)
self.assertEqual(VOLUME_DETAIL["os-vol-mig-status-attr:migstat"],
sot.migration_status)
self.assertEqual(VOLUME_DETAIL["os-vol-mig-status-attr:name_id"],
sot.migration_id)
self.assertEqual(VOLUME_DETAIL["replication_status"],
sot.replication_status)
self.assertEqual(
VOLUME_DETAIL["os-volume-replication:extended_status"],
sot.extended_replication_status)
self.assertEqual(VOLUME_DETAIL["consistencygroup_id"],
sot.consistency_group_id)
self.assertEqual(VOLUME_DETAIL["os-volume-replication:driver_data"],
sot.replication_driver_data)
self.assertFalse(sot.is_encrypted)

View File

@ -0,0 +1,6 @@
---
deprecations:
- |
Requesting volumes or backups with details from block_storage will return
objects of classes Volume and Backup correspondingly, instead
of VolumeDetail and BackupDetail.