Only return latest instance fault for instances
This patch addresses slowness that can occur when doing a list servers API operation when there are many thousands of records in the instance_faults table. Previously, in the Instance.fill_faults() method, we were getting all instance fault records for a set of instances having one of a set of supplied instance UUIDs and then iterating over those faults and returning a dict of instance UUID to the first fault returned (which happened to be the latest fault because of ordering the SQL query by created_at). This patch adds a new InstanceFaultList.get_latest_by_instance_uuids() method that does some SQL-fu to only return the latest fault records for each instance being inspected. Closes-Bug: #1632247 Co-Authored-By: Roman Podoliaka <rpodolyaka@mirantis.com> Change-Id: I8f2227b3969791ebb2d04d74a316b9d97a4b1571
This commit is contained in:
parent
0280b36666
commit
e24798a02e
@ -1901,9 +1901,11 @@ def instance_fault_create(context, values):
|
|||||||
return IMPL.instance_fault_create(context, values)
|
return IMPL.instance_fault_create(context, values)
|
||||||
|
|
||||||
|
|
||||||
def instance_fault_get_by_instance_uuids(context, instance_uuids):
|
def instance_fault_get_by_instance_uuids(context, instance_uuids,
|
||||||
|
latest=False):
|
||||||
"""Get all instance faults for the provided instance_uuids."""
|
"""Get all instance faults for the provided instance_uuids."""
|
||||||
return IMPL.instance_fault_get_by_instance_uuids(context, instance_uuids)
|
return IMPL.instance_fault_get_by_instance_uuids(context, instance_uuids,
|
||||||
|
latest=latest)
|
||||||
|
|
||||||
|
|
||||||
####################
|
####################
|
||||||
|
@ -6036,24 +6036,63 @@ def instance_fault_create(context, values):
|
|||||||
|
|
||||||
|
|
||||||
@pick_context_manager_reader
|
@pick_context_manager_reader
|
||||||
def instance_fault_get_by_instance_uuids(context, instance_uuids):
|
def instance_fault_get_by_instance_uuids(context, instance_uuids,
|
||||||
"""Get all instance faults for the provided instance_uuids."""
|
latest=False):
|
||||||
|
"""Get all instance faults for the provided instance_uuids.
|
||||||
|
|
||||||
|
:param instance_uuids: List of UUIDs of instances to grab faults for
|
||||||
|
:param latest: Optional boolean indicating we should only return the latest
|
||||||
|
fault for the instance
|
||||||
|
"""
|
||||||
if not instance_uuids:
|
if not instance_uuids:
|
||||||
return {}
|
return {}
|
||||||
|
|
||||||
rows = model_query(context, models.InstanceFault, read_deleted='no').\
|
faults_tbl = models.InstanceFault.__table__
|
||||||
filter(models.InstanceFault.instance_uuid.in_(
|
# NOTE(rpodolyaka): filtering by instance_uuids is performed in both
|
||||||
instance_uuids)).\
|
# code branches below for the sake of a better query plan. On change,
|
||||||
order_by(desc("created_at"), desc("id")).\
|
# make sure to update the other one as well.
|
||||||
all()
|
query = model_query(context, models.InstanceFault,
|
||||||
|
[faults_tbl],
|
||||||
|
read_deleted='no')
|
||||||
|
|
||||||
|
if latest:
|
||||||
|
# NOTE(jaypipes): We join instance_faults to a derived table of the
|
||||||
|
# latest faults per instance UUID. The SQL produced below looks like
|
||||||
|
# this:
|
||||||
|
#
|
||||||
|
# SELECT instance_faults.*
|
||||||
|
# FROM instance_faults
|
||||||
|
# JOIN (
|
||||||
|
# SELECT instance_uuid, MAX(id) AS max_id
|
||||||
|
# FROM instance_faults
|
||||||
|
# WHERE instance_uuid IN ( ... )
|
||||||
|
# AND deleted = 0
|
||||||
|
# GROUP BY instance_uuid
|
||||||
|
# ) AS latest_faults
|
||||||
|
# ON instance_faults.id = latest_faults.max_id;
|
||||||
|
latest_faults = model_query(
|
||||||
|
context, models.InstanceFault,
|
||||||
|
[faults_tbl.c.instance_uuid,
|
||||||
|
sql.func.max(faults_tbl.c.id).label('max_id')],
|
||||||
|
read_deleted='no'
|
||||||
|
).filter(
|
||||||
|
faults_tbl.c.instance_uuid.in_(instance_uuids)
|
||||||
|
).group_by(
|
||||||
|
faults_tbl.c.instance_uuid
|
||||||
|
).subquery(name="latest_faults")
|
||||||
|
|
||||||
|
query = query.join(latest_faults,
|
||||||
|
faults_tbl.c.id == latest_faults.c.max_id)
|
||||||
|
else:
|
||||||
|
query = query.filter(models.InstanceFault.instance_uuid.in_(
|
||||||
|
instance_uuids)).order_by(desc("id"))
|
||||||
|
|
||||||
output = {}
|
output = {}
|
||||||
for instance_uuid in instance_uuids:
|
for instance_uuid in instance_uuids:
|
||||||
output[instance_uuid] = []
|
output[instance_uuid] = []
|
||||||
|
|
||||||
for row in rows:
|
for row in query:
|
||||||
data = dict(row)
|
output[row.instance_uuid].append(row._asdict())
|
||||||
output[row['instance_uuid']].append(data)
|
|
||||||
|
|
||||||
return output
|
return output
|
||||||
|
|
||||||
|
Loading…
x
Reference in New Issue
Block a user