From 8e763e22513f5511ba50bed4d83c51080c671682 Mon Sep 17 00:00:00 2001 From: Terri Yu Date: Wed, 21 Aug 2013 04:27:27 +0000 Subject: [PATCH] Adds group by statistics for MongoDB driver Implements: blueprint api-group-by Adds group by meter statistics in the MongoDB driver for the case where the groupby fields are a combination of 'user-id', 'resource-id', 'project-id', and 'source' (metadata fields not implemented) The group by meter statistics can be requested with query filtering and/or period parameters. In this case, the query filtering and/or period grouping is applied first, followed by the calculation of group by meter statistics. Refactors the meter statistics map functions MAP_STATS and MAP_STATS_PERIOD to remove duplicate code Modifies test_group_by_unknown_field in class StatisticsGroupByTest so that it works for both the SQL Alchemy driver and the MongoDB driver; the previous version only worked for the SQL Alchemy driver Change-Id: I6954abe9d8c86c780ad96bbf04849d068b360b1c --- ceilometer/storage/impl_mongodb.py | 136 +++++++++++++++++------- tests/storage/test_storage_scenarios.py | 12 ++- 2 files changed, 105 insertions(+), 43 deletions(-) diff --git a/ceilometer/storage/impl_mongodb.py b/ceilometer/storage/impl_mongodb.py index 14dab014b..442f25fa3 100644 --- a/ceilometer/storage/impl_mongodb.py +++ b/ceilometer/storage/impl_mongodb.py @@ -28,6 +28,7 @@ import weakref import bson.code import bson.objectid +import json import pymongo from oslo.config import cfg @@ -172,40 +173,86 @@ class Connection(base.Connection): } """) - MAP_STATS = bson.code.Code(""" - function () { - emit('statistics', { unit: this.counter_unit, - min : this.counter_volume, - max : this.counter_volume, - sum : this.counter_volume, - count : NumberInt(1), - duration_start : this.timestamp, - duration_end : this.timestamp, - period_start : this.timestamp, - period_end : this.timestamp} ) - } - """) + EMIT_STATS_COMMON = """ + emit(%(key_val)s, { unit: this.counter_unit, + min : this.counter_volume, + max : this.counter_volume, + sum : this.counter_volume, + count : NumberInt(1), + groupby : %(groupby_val)s, + duration_start : this.timestamp, + duration_end : this.timestamp, + period_start : %(period_start_val)s, + period_end : %(period_end_val)s} ) + """ - MAP_STATS_PERIOD = bson.code.Code(""" - function () { - var period = %d * 1000; - var period_first = %d * 1000; + MAP_STATS_PERIOD_VAR = """ + var period = %(period)d * 1000; + var period_first = %(period_first)d * 1000; var period_start = period_first + (Math.floor(new Date(this.timestamp.getTime() - period_first) / period) * period); - emit(period_start, - { unit: this.counter_unit, - min : this.counter_volume, - max : this.counter_volume, - sum : this.counter_volume, - count : NumberInt(1), - duration_start : this.timestamp, - duration_end : this.timestamp, - period_start : new Date(period_start), - period_end : new Date(period_start + period) } ) + """ + + MAP_STATS_GROUPBY_VAR = """ + var groupby_fields = %(groupby_fields)s; + var groupby = {}; + var groupby_key = {}; + + for ( var i=0; i