Improve get all instances with share data

Share instance get all by host database query is being optimized by
performing join of share_instances table with share table.

Co-Authored-By: Maurice Escher <maurice.escher@sap.com>

Closes-bug: #2084783
Change-Id: Idbf4a99793b49699b9cd340f4e82e4610bd48eeb
This commit is contained in:
Kiran Pawar 2024-10-17 10:51:56 +00:00
parent 0990ddf73a
commit bc01cc1c1f
4 changed files with 53 additions and 38 deletions

View File

@ -1678,8 +1678,12 @@ def share_instance_update(context, share_instance_id, values,
return instance_ref return instance_ref
def _share_instance_update(context, share_instance_id, values): def _share_instance_update(
share_instance_ref = _share_instance_get(context, share_instance_id) context, share_instance_id, values, with_share_data=False
):
share_instance_ref = _share_instance_get(
context, share_instance_id,
with_share_data=with_share_data)
share_instance_ref.update(values) share_instance_ref.update(values)
share_instance_ref.save(session=context.session) share_instance_ref.save(session=context.session)
return share_instance_ref return share_instance_ref
@ -1955,19 +1959,14 @@ def update_share_instance_quota_usages(context, instance_id):
deferred_delete=True) deferred_delete=True)
def _set_instances_share_data(context, instances): def _set_instances_share_data(instances):
if instances and not isinstance(instances, list): instances = instances.options(
instances = [instances] orm.joinedload(models.ShareInstance.share)).all()
instances = [s for s in instances if s.share]
for s in instances:
s.set_share_data(s.share)
instances_with_share_data = [] return instances
for instance in instances:
try:
parent_share = _share_get(context, instance['share_id'])
except exception.NotFound:
continue
instance.set_share_data(parent_share)
instances_with_share_data.append(instance)
return instances_with_share_data
@require_admin_context @require_admin_context
@ -1985,11 +1984,12 @@ def share_instance_get_all_by_host(context, host, with_share_data=False,
) )
if status is not None: if status is not None:
instances = instances.filter(models.ShareInstance.status == status) instances = instances.filter(models.ShareInstance.status == status)
# Returns list of all instances that satisfy filters.
instances = instances.all()
if with_share_data: if with_share_data:
instances = _set_instances_share_data(context, instances) instances = _set_instances_share_data(instances)
else:
# Returns list of all instances that satisfy filters.
instances = instances.all()
return instances return instances
@ -2013,11 +2013,13 @@ def share_instance_get_all_by_share_server(context, share_server_id,
result = ( result = (
model_query(context, models.ShareInstance).filter( model_query(context, models.ShareInstance).filter(
models.ShareInstance.share_server_id == share_server_id, models.ShareInstance.share_server_id == share_server_id,
).all() )
) )
if with_share_data: if with_share_data:
result = _set_instances_share_data(context, result) result = _set_instances_share_data(result)
else:
result = result.all()
return result return result
@ -2098,10 +2100,12 @@ def share_replicas_get_all(context, with_share_data=False,
"""Returns replica instances for all available replicated shares.""" """Returns replica instances for all available replicated shares."""
result = _share_replica_get_with_filters( result = _share_replica_get_with_filters(
context, with_share_server=with_share_server, context, with_share_server=with_share_server,
).all() )
if with_share_data: if with_share_data:
result = _set_instances_share_data(context, result) result = _set_instances_share_data(result)
else:
result = result.all()
return result return result
@ -2115,10 +2119,12 @@ def share_replicas_get_all_by_share(context, share_id,
result = _share_replica_get_with_filters( result = _share_replica_get_with_filters(
context, with_share_server=with_share_server, context, with_share_server=with_share_server,
share_id=share_id, share_id=share_id,
).all() )
if with_share_data: if with_share_data:
result = _set_instances_share_data(context, result) result = _set_instances_share_data(result)
else:
result = result.all()
return result return result
@ -2133,10 +2139,12 @@ def share_replicas_get_available_active_replica(context, share_id,
context, with_share_server=with_share_server, share_id=share_id, context, with_share_server=with_share_server, share_id=share_id,
replica_state=constants.REPLICA_STATE_ACTIVE, replica_state=constants.REPLICA_STATE_ACTIVE,
status=constants.STATUS_AVAILABLE, status=constants.STATUS_AVAILABLE,
).first() )
if result and with_share_data: if result.first() and with_share_data:
result = _set_instances_share_data(context, result)[0] result = _set_instances_share_data(result)[0]
else:
result = result.first()
return result return result
@ -2150,14 +2158,16 @@ def share_replica_get(context, replica_id, with_share_data=False,
context, context,
with_share_server=with_share_server, with_share_server=with_share_server,
replica_id=replica_id, replica_id=replica_id,
).first() )
if result.first() and with_share_data:
result = _set_instances_share_data(result)[0]
else:
result = result.first()
if result is None: if result is None:
raise exception.ShareReplicaNotFound(replica_id=replica_id) raise exception.ShareReplicaNotFound(replica_id=replica_id)
if with_share_data:
result = _set_instances_share_data(context, result)[0]
return result return result
@ -2170,13 +2180,9 @@ def share_replica_update(context, share_replica_id, values,
"""Updates a share replica with specified values.""" """Updates a share replica with specified values."""
updated_share_replica = _share_instance_update( updated_share_replica = _share_instance_update(
context, share_replica_id, values, context, share_replica_id, values,
with_share_data=with_share_data
) )
if with_share_data:
updated_share_replica = _set_instances_share_data(
context, updated_share_replica,
)[0]
return updated_share_replica return updated_share_replica

View File

@ -428,6 +428,11 @@ class ShareInstance(BASE, ManilaBase):
primaryjoin='and_(' primaryjoin='and_('
'ShareInstance.share_type_id == ShareTypes.id, ' 'ShareInstance.share_type_id == ShareTypes.id, '
'ShareTypes.deleted == "False")') 'ShareTypes.deleted == "False")')
share = orm.relationship(
'Share',
foreign_keys=share_id,
primaryjoin='ShareInstance.share_id == Share.id'
)
class ShareInstanceExportLocations(BASE, ManilaBase): class ShareInstanceExportLocations(BASE, ManilaBase):

View File

@ -597,10 +597,8 @@ class ShareDatabaseAPITestCase(test.TestCase):
def test_share_instance_get_all_by_host_not_found_exception(self): def test_share_instance_get_all_by_host_not_found_exception(self):
db_utils.create_share() db_utils.create_share()
self.mock_object(db_api, '_share_get', mock.Mock(
side_effect=exception.NotFound))
instances = db_api.share_instance_get_all_by_host( instances = db_api.share_instance_get_all_by_host(
self.ctxt, 'fake_host', True) self.ctxt, 'not_found_host', True)
self.assertEqual(0, len(instances)) self.assertEqual(0, len(instances))

View File

@ -0,0 +1,6 @@
---
fixes:
- |
Share instance/replica get with share data query is being optimized to
improve the performance. Please check `launchpad bug 2084783
<https://bugs.launchpad.net/manila/+bug/2084783>`_ for more details.