Implement dot in matching_metadata key for mongodb

Before the dict matching_metadata was stored as dict in mongo.
But mongo doesn't allow dot in dictionary key

This change converts the matching_metadata dict into a array like this:

[ { 'key': 'info.key.dotted', 'value': 'the_value'} ]

Fixes bug #1201886

Change-Id: Iab94c675749331ff7bfb0f74728dbd8f947f26f6
This commit is contained in:
Mehdi Abaakouk 2013-07-18 10:22:32 +02:00
parent 0b50ac2f59
commit a958d6c2ce
3 changed files with 65 additions and 4 deletions

View File

@ -616,6 +616,26 @@ class Connection(base.Connection):
for r in results['results']),
key=operator.attrgetter('period_start'))
@staticmethod
def _decode_matching_metadata(matching_metadata):
if isinstance(matching_metadata, dict):
#note(sileht): keep compatibility with old db format
return matching_metadata
else:
new_matching_metadata = {}
for elem in matching_metadata:
new_matching_metadata[elem['key']] = elem['value']
return new_matching_metadata
@staticmethod
def _encode_matching_metadata(matching_metadata):
if matching_metadata:
new_matching_metadata = []
for k, v in matching_metadata.iteritems():
new_matching_metadata.append({'key': k, 'value': v})
return new_matching_metadata
return matching_metadata
def get_alarms(self, name=None, user=None,
project=None, enabled=True, alarm_id=None):
"""Yields a lists of alarms that match filters
@ -636,6 +656,8 @@ class Connection(base.Connection):
a = {}
a.update(alarm)
del a['_id']
a['matching_metadata'] = \
self._decode_matching_metadata(a['matching_metadata'])
yield models.Alarm(**a)
def update_alarm(self, alarm):
@ -645,6 +667,9 @@ class Connection(base.Connection):
# This is an insert, generate an id
alarm.alarm_id = str(uuid.uuid1())
data = alarm.as_dict()
data['matching_metadata'] = \
self._encode_matching_metadata(data['matching_metadata'])
self.db.alarm.update(
{'alarm_id': alarm.alarm_id},
{'$set': data},
@ -652,6 +677,8 @@ class Connection(base.Connection):
stored_alarm = self.db.alarm.find({'alarm_id': alarm.alarm_id})[0]
del stored_alarm['_id']
stored_alarm['matching_metadata'] = \
self._decode_matching_metadata(stored_alarm['matching_metadata'])
return models.Alarm(**stored_alarm)
def delete_alarm(self, alarm_id):

View File

@ -801,16 +801,21 @@ class AlarmTest(DBTestBase):
'me', 'and-da-boys',
evaluation_periods=1,
period=60,
alarm_actions=['http://nowhere/alarms']),
alarm_actions=['http://nowhere/alarms'],
matching_metadata={'key': 'value'}),
models.Alarm('orange-alert',
'test.fourty', 'gt', 75, 'avg',
'me', 'and-da-boys',
period=60,
alarm_actions=['http://nowhere/alarms']),
alarm_actions=['http://nowhere/alarms'],
matching_metadata={'key2': 'value2'}),
models.Alarm('yellow-alert',
'test.five', 'lt', 10, 'min',
'me', 'and-da-boys',
alarm_actions=['http://nowhere/alarms'])]
alarm_actions=['http://nowhere/alarms'],
matching_metadata=
{'key2': 'value2',
'user_metadata.key3': 'value3'})]
for a in alarms:
self.conn.update_alarm(a)
@ -832,15 +837,23 @@ class AlarmTest(DBTestBase):
self.assertEquals(yellow.state, models.Alarm.ALARM_INSUFFICIENT_DATA)
self.assertEquals(yellow.ok_actions, [])
self.assertEquals(yellow.insufficient_data_actions, [])
self.assertEquals(yellow.matching_metadata,
{'key2': 'value2',
'user_metadata.key3': 'value3'})
def test_update(self):
self.add_some_alarms()
orange = list(self.conn.get_alarms(name='orange-alert'))[0]
orange.enabled = False
orange.state = models.Alarm.ALARM_INSUFFICIENT_DATA
orange.matching_metadata = {'new': 'value',
'user_metadata.new2': 'value4'}
updated = self.conn.update_alarm(orange)
self.assertEquals(updated.enabled, False)
self.assertEquals(updated.state, models.Alarm.ALARM_INSUFFICIENT_DATA)
self.assertEquals(updated.matching_metadata,
{'new': 'value',
'user_metadata.new2': 'value4'})
def test_update_llu(self):
llu = models.Alarm('llu',

View File

@ -26,6 +26,8 @@
import copy
import datetime
import uuid
from oslo.config import cfg
from tests.storage import base
@ -33,6 +35,7 @@ from tests.storage import base
from ceilometer.publisher import rpc
from ceilometer import counter
from ceilometer.storage import impl_mongodb
from ceilometer.storage import models
class MongoDBEngineTestBase(base.DBTestBase):
@ -143,7 +146,25 @@ class StatisticsTest(base.StatisticsTest, MongoDBEngineTestBase):
class AlarmTest(base.AlarmTest, MongoDBEngineTestBase):
pass
def prepare_old_matching_metadata_alarm(self):
alarm = models.Alarm('old-alert',
'test.one', 'eq', 36, 'count',
'me', 'and-da-boys',
evaluation_periods=1,
period=60,
alarm_actions=['http://nowhere/alarms'],
matching_metadata={'key': 'value'})
alarm.alarm_id = str(uuid.uuid1())
data = alarm.as_dict()
self.conn.db.alarm.update(
{'alarm_id': alarm.alarm_id},
{'$set': data},
upsert=True)
def test_alarm_get_old_matching_metadata_format(self):
self.prepare_old_matching_metadata_alarm()
old = list(self.conn.get_alarms(name='old-alert'))[0]
self.assertEquals(old.matching_metadata, {'key': 'value'})
class CompatibilityTest(MongoDBEngineTestBase):