Merge "sqlalchemy: fix performance issue on get_meters()"
This commit is contained in:
commit
bd594cbde1
@ -24,6 +24,7 @@ import os
|
|||||||
import uuid
|
import uuid
|
||||||
from sqlalchemy import func
|
from sqlalchemy import func
|
||||||
from sqlalchemy import desc
|
from sqlalchemy import desc
|
||||||
|
from sqlalchemy.orm import aliased
|
||||||
|
|
||||||
from ceilometer.openstack.common import log
|
from ceilometer.openstack.common import log
|
||||||
from ceilometer.openstack.common import timeutils
|
from ceilometer.openstack.common import timeutils
|
||||||
@ -313,7 +314,31 @@ class Connection(base.Connection):
|
|||||||
:param metaquery: Optional dict with metadata to match on.
|
:param metaquery: Optional dict with metadata to match on.
|
||||||
"""
|
"""
|
||||||
session = sqlalchemy_session.get_session()
|
session = sqlalchemy_session.get_session()
|
||||||
query = session.query(Resource)
|
|
||||||
|
# Meter table will store large records and join with resource
|
||||||
|
# will be very slow.
|
||||||
|
# subquery_meter is used to reduce meter records
|
||||||
|
# by selecting a record for each (resource_id, counter_name).
|
||||||
|
# max() is used to choice a meter record, so the latest record
|
||||||
|
# is selected for each (resource_id, counter_name).
|
||||||
|
#
|
||||||
|
subquery_meter = session.query(func.max(Meter.id).label('id')).\
|
||||||
|
group_by(Meter.resource_id, Meter.counter_name).subquery()
|
||||||
|
|
||||||
|
# The SQL of query_meter is essentially:
|
||||||
|
#
|
||||||
|
# SELECT meter.* FROM meter INNER JOIN
|
||||||
|
# (SELECT max(meter.id) AS id FROM meter
|
||||||
|
# GROUP BY meter.resource_id, meter.counter_name) AS anon_2
|
||||||
|
# ON meter.id = anon_2.id
|
||||||
|
#
|
||||||
|
query_meter = session.query(Meter).\
|
||||||
|
join(subquery_meter, Meter.id == subquery_meter.c.id)
|
||||||
|
|
||||||
|
alias_meter = aliased(Meter, query_meter.subquery())
|
||||||
|
query = session.query(Resource, alias_meter).join(
|
||||||
|
alias_meter, Resource.id == alias_meter.resource_id)
|
||||||
|
|
||||||
if user is not None:
|
if user is not None:
|
||||||
query = query.filter(Resource.user_id == user)
|
query = query.filter(Resource.user_id == user)
|
||||||
if source is not None:
|
if source is not None:
|
||||||
@ -322,26 +347,18 @@ class Connection(base.Connection):
|
|||||||
query = query.filter(Resource.id == resource)
|
query = query.filter(Resource.id == resource)
|
||||||
if project is not None:
|
if project is not None:
|
||||||
query = query.filter(Resource.project_id == project)
|
query = query.filter(Resource.project_id == project)
|
||||||
query = query.options(
|
|
||||||
sqlalchemy_session.sqlalchemy.orm.joinedload('meters'))
|
|
||||||
if metaquery:
|
if metaquery:
|
||||||
raise NotImplementedError('metaquery not implemented')
|
raise NotImplementedError('metaquery not implemented')
|
||||||
|
|
||||||
for resource in query.all():
|
for resource, meter in query.all():
|
||||||
meter_names = set()
|
yield api_models.Meter(
|
||||||
for meter in resource.meters:
|
name=meter.counter_name,
|
||||||
if meter.counter_name in meter_names:
|
type=meter.counter_type,
|
||||||
continue
|
unit=meter.counter_unit,
|
||||||
meter_names.add(meter.counter_name)
|
resource_id=resource.id,
|
||||||
yield api_models.Meter(
|
project_id=resource.project_id,
|
||||||
name=meter.counter_name,
|
source=resource.sources[0].id,
|
||||||
type=meter.counter_type,
|
user_id=resource.user_id)
|
||||||
unit=meter.counter_unit,
|
|
||||||
resource_id=resource.id,
|
|
||||||
project_id=resource.project_id,
|
|
||||||
source=resource.sources[0].id,
|
|
||||||
user_id=resource.user_id,
|
|
||||||
)
|
|
||||||
|
|
||||||
@staticmethod
|
@staticmethod
|
||||||
def get_samples(sample_filter, limit=None):
|
def get_samples(sample_filter, limit=None):
|
||||||
|
@ -0,0 +1,34 @@
|
|||||||
|
# -*- encoding: utf-8 -*-
|
||||||
|
#
|
||||||
|
#
|
||||||
|
# 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 sqlalchemy import MetaData
|
||||||
|
from sqlalchemy import Index
|
||||||
|
from ceilometer.storage.sqlalchemy.models import Meter
|
||||||
|
|
||||||
|
meta = MetaData()
|
||||||
|
|
||||||
|
|
||||||
|
def upgrade(migrate_engine):
|
||||||
|
meta.bind = migrate_engine
|
||||||
|
index = Index('idx_meter_rid_cname', Meter.resource_id,
|
||||||
|
Meter.counter_name)
|
||||||
|
index.create(bind=migrate_engine)
|
||||||
|
|
||||||
|
|
||||||
|
def downgrade(migrate_engine):
|
||||||
|
meta.bind = migrate_engine
|
||||||
|
index = Index('idx_meter_rid_cname', Meter.resource_id,
|
||||||
|
Meter.counter_name)
|
||||||
|
index.drop(bind=migrate_engine)
|
Loading…
Reference in New Issue
Block a user