api: update for WSME 0.5b6 compliance

This makes use of the mandatory option of WSME, that now works, to
remove some of our custom validation code. This is needed for new
versions of WSME that do more validation on their own.

Fixes-Bug: #1240741

Change-Id: Icb66d17066b515bebf3f3a326d84e18cbfce01ef
Signed-off-by: Julien Danjou <julien@danjou.info>
This commit is contained in:
Julien Danjou 2013-09-30 16:43:59 +02:00
parent c0b7e4984a
commit e4a1a4fcef
5 changed files with 52 additions and 74 deletions

View File

@ -524,19 +524,19 @@ class Sample(_Base):
source = wtypes.text source = wtypes.text
"The ID of the source that identifies where the sample comes from" "The ID of the source that identifies where the sample comes from"
counter_name = wtypes.text counter_name = wsme.wsattr(wtypes.text, mandatory=True)
"The name of the meter" "The name of the meter"
# FIXME(dhellmann): Make this meter_name? # FIXME(dhellmann): Make this meter_name?
counter_type = wtypes.text counter_type = wsme.wsattr(wtypes.text, mandatory=True)
"The type of the meter (see :ref:`measurements`)" "The type of the meter (see :ref:`measurements`)"
# FIXME(dhellmann): Make this meter_type? # FIXME(dhellmann): Make this meter_type?
counter_unit = wtypes.text counter_unit = wsme.wsattr(wtypes.text, mandatory=True)
"The unit of measure for the value in counter_volume" "The unit of measure for the value in counter_volume"
# FIXME(dhellmann): Make this meter_unit? # FIXME(dhellmann): Make this meter_unit?
counter_volume = float counter_volume = wsme.wsattr(float, mandatory=True)
"The actual measured value" "The actual measured value"
user_id = wtypes.text user_id = wtypes.text
@ -545,7 +545,7 @@ class Sample(_Base):
project_id = wtypes.text project_id = wtypes.text
"The ID of the project or tenant that owns the resource" "The ID of the project or tenant that owns the resource"
resource_id = wtypes.text resource_id = wsme.wsattr(wtypes.text, mandatory=True)
"The ID of the :class:`Resource` for which the measurements are taken" "The ID of the :class:`Resource` for which the measurements are taken"
timestamp = datetime.datetime timestamp = datetime.datetime
@ -569,11 +569,6 @@ class Sample(_Base):
super(Sample, self).__init__(counter_volume=counter_volume, super(Sample, self).__init__(counter_volume=counter_volume,
resource_metadata=resource_metadata, resource_metadata=resource_metadata,
timestamp=timestamp, **kwds) timestamp=timestamp, **kwds)
# Seems the mandatory option doesn't work so do it manually
for m in ('counter_volume', 'counter_unit',
'counter_name', 'counter_type', 'resource_id'):
if getattr(self, m) in (wsme.Unset, None):
raise wsme.exc.MissingArgument(m)
if self.resource_metadata in (wtypes.Unset, None): if self.resource_metadata in (wtypes.Unset, None):
self.resource_metadata = {} self.resource_metadata = {}
@ -717,20 +712,12 @@ class MeterController(rest.RestController):
for e in pecan.request.storage_conn.get_samples(f, limit=limit) for e in pecan.request.storage_conn.get_samples(f, limit=limit)
] ]
@wsme.validate([Sample])
@wsme_pecan.wsexpose([Sample], body=[Sample]) @wsme_pecan.wsexpose([Sample], body=[Sample])
def post(self, body): def post(self, samples):
"""Post a list of new Samples to Ceilometer. """Post a list of new Samples to Ceilometer.
:param body: a list of samples within the request body. :param samples: a list of samples within the request body.
""" """
# Note:
# 1) the above validate decorator seems to do nothing. LP#1220678
# 2) the mandatory options seems to also do nothing. LP#1227004
# 3) the body should already be in a list of Sample's LP#1233219
samples = [Sample(**b) for b in body]
now = timeutils.utcnow() now = timeutils.utcnow()
auth_project = acl.get_limited_to_project(pecan.request.headers) auth_project = acl.get_limited_to_project(pecan.request.headers)
def_source = pecan.request.cfg.sample_source def_source = pecan.request.cfg.sample_source
@ -1010,14 +997,6 @@ class AlarmThresholdRule(_Base):
@staticmethod @staticmethod
def validate(threshold_rule): def validate(threshold_rule):
#note(sileht): wsme mandatory doesn't work as expected
#workaround for https://bugs.launchpad.net/wsme/+bug/1227004
for field in ['meter_name', 'threshold']:
if not getattr(threshold_rule, field):
error = _("threshold_rule/%s is mandatory") % field
pecan.response.translatable_error = error
raise wsme.exc.ClientSideError(unicode(error))
#note(sileht): wsme default doesn't work in some case #note(sileht): wsme default doesn't work in some case
#workaround for https://bugs.launchpad.net/wsme/+bug/1227039 #workaround for https://bugs.launchpad.net/wsme/+bug/1227039
if not threshold_rule.query: if not threshold_rule.query:
@ -1082,17 +1061,6 @@ class AlarmCombinationRule(_Base):
alarm_ids=['739e99cb-c2ec-4718-b900-332502355f38', alarm_ids=['739e99cb-c2ec-4718-b900-332502355f38',
'153462d0-a9b8-4b5b-8175-9e4b05e9b856']) '153462d0-a9b8-4b5b-8175-9e4b05e9b856'])
@staticmethod
def validate(combination_rule):
#note(sileht): wsme mandatory doesn't works as expected
#workaround for https://bugs.launchpad.net/wsme/+bug/1227004
if not combination_rule.alarm_ids:
error = _("combination_rule/alarm_ids is mandatory")
pecan.response.translatable_error = error
raise wsme.exc.ClientSideError(unicode(error))
return combination_rule
class Alarm(_Base): class Alarm(_Base):
"""Representation of an alarm. """Representation of an alarm.
@ -1173,13 +1141,18 @@ class Alarm(_Base):
@staticmethod @staticmethod
def validate(alarm): def validate(alarm):
#note(sileht): wsme mandatory doesn't work as expected if (alarm.threshold_rule == wtypes.Unset
#workaround for https://bugs.launchpad.net/wsme/+bug/1227004 and alarm.combination_rule == wtypes.Unset):
for field in ['name', 'type']: error = _("either threshold_rule or combination_rule "
if not getattr(alarm, field): "must be set")
error = _("%s is mandatory") % field pecan.response.translatable_error = error
pecan.response.translatable_error = error raise wsme.exc.ClientSideError(unicode(error))
raise wsme.exc.ClientSideError(unicode(error))
if alarm.threshold_rule and alarm.combination_rule:
error = _("threshold_rule and combination_rule "
"cannot be set at the same time")
pecan.response.translatable_error = error
raise wsme.exc.ClientSideError(unicode(error))
if alarm.threshold_rule: if alarm.threshold_rule:
# ensure an implicit constraint on project_id is added to # ensure an implicit constraint on project_id is added to
@ -1190,20 +1163,17 @@ class Alarm(_Base):
on_behalf_of=alarm.project_id on_behalf_of=alarm.project_id
) )
elif alarm.combination_rule: elif alarm.combination_rule:
auth_project = _get_auth_project(alarm.project_id) project = _get_auth_project(alarm.project_id
if alarm.project_id != wtypes.Unset
else None)
for id in alarm.combination_rule.alarm_ids: for id in alarm.combination_rule.alarm_ids:
alarms = list(pecan.request.storage_conn.get_alarms( alarms = list(pecan.request.storage_conn.get_alarms(
alarm_id=id, project=auth_project)) alarm_id=id, project=project))
if not alarms: if not alarms:
error = _("Alarm %s doesn't exist") % id error = _("Alarm %s doesn't exist") % id
pecan.response.translatable_error = error pecan.response.translatable_error = error
raise wsme.exc.ClientSideError(unicode(error)) raise wsme.exc.ClientSideError(unicode(error))
if alarm.threshold_rule and alarm.combination_rule:
error = _("threshold_rule and combination_rule "
"cannot be set at the same time")
pecan.response.translatable_error = error
raise wsme.exc.ClientSideError(unicode(error))
return alarm return alarm
@classmethod @classmethod
@ -1333,7 +1303,6 @@ class AlarmController(rest.RestController):
"""Return this alarm.""" """Return this alarm."""
return Alarm.from_db_model(self._alarm()) return Alarm.from_db_model(self._alarm())
@wsme.validate(Alarm)
@wsme_pecan.wsexpose(Alarm, wtypes.text, body=Alarm) @wsme_pecan.wsexpose(Alarm, wtypes.text, body=Alarm)
def put(self, data): def put(self, data):
"""Modify this alarm.""" """Modify this alarm."""
@ -1344,18 +1313,20 @@ class AlarmController(rest.RestController):
data.alarm_id = self._id data.alarm_id = self._id
user, project = acl.get_limited_to(pecan.request.headers) user, project = acl.get_limited_to(pecan.request.headers)
data.user_id = user or data.user_id or alarm_in.user_id if user:
data.project_id = project or data.project_id or alarm_in.project_id data.user_id = user
elif data.user_id == wtypes.Unset:
data.user_id = alarm_in.user_id
if project:
data.project_id = project
elif data.project_id == wtypes.Unset:
data.project_id = alarm_in.project_id
data.timestamp = now data.timestamp = now
if alarm_in.state != data.state: if alarm_in.state != data.state:
data.state_timestamp = now data.state_timestamp = now
else: else:
data.state_timestamp = alarm_in.state_timestamp data.state_timestamp = alarm_in.state_timestamp
#note(sileht): workaround for
#https://bugs.launchpad.net/wsme/+bug/1220678
Alarm.validate(data)
old_alarm = Alarm.from_db_model(alarm_in).as_dict(storage.models.Alarm) old_alarm = Alarm.from_db_model(alarm_in).as_dict(storage.models.Alarm)
updated_alarm = data.as_dict(storage.models.Alarm) updated_alarm = data.as_dict(storage.models.Alarm)
try: try:
@ -1462,7 +1433,6 @@ class AlarmsController(rest.RestController):
payload['detail'] = scrubbed_data payload['detail'] = scrubbed_data
_send_notification(type, payload) _send_notification(type, payload)
@wsme.validate(Alarm)
@wsme_pecan.wsexpose(Alarm, body=Alarm, status_code=201) @wsme_pecan.wsexpose(Alarm, body=Alarm, status_code=201)
def post(self, data): def post(self, data):
"""Create a new alarm.""" """Create a new alarm."""
@ -1471,17 +1441,17 @@ class AlarmsController(rest.RestController):
data.alarm_id = str(uuid.uuid4()) data.alarm_id = str(uuid.uuid4())
user, project = acl.get_limited_to(pecan.request.headers) user, project = acl.get_limited_to(pecan.request.headers)
data.user_id = (user or data.user_id or if user:
pecan.request.headers.get('X-User-Id')) data.user_id = user
data.project_id = (project or data.project_id or elif data.user_id == wtypes.Unset:
pecan.request.headers.get('X-Project-Id')) data.user_id = pecan.request.headers.get('X-User-Id')
if project:
data.project_id = project
elif data.project_id == wtypes.Unset:
data.project_id = pecan.request.headers.get('X-Project-Id')
data.timestamp = now data.timestamp = now
data.state_timestamp = now data.state_timestamp = now
#note(sileht): workaround for
#https://bugs.launchpad.net/wsme/+bug/1220678
Alarm.validate(data)
change = data.as_dict(storage.models.Alarm) change = data.as_dict(storage.models.Alarm)
# make sure alarms are unique by name per project. # make sure alarms are unique by name per project.

View File

@ -104,6 +104,9 @@ class FunctionalTest(db_test_base.TestBase):
'template_path': '%s/ceilometer/api/templates' % root_dir, 'template_path': '%s/ceilometer/api/templates' % root_dir,
'enable_acl': enable_acl, 'enable_acl': enable_acl,
}, },
'wsme': {
'debug': True,
},
} }
return pecan.testing.load_test_app(self.config) return pecan.testing.load_test_app(self.config)

View File

@ -279,7 +279,9 @@ class TestAlarms(FunctionalTest,
status=400, headers=self.auth_headers) status=400, headers=self.auth_headers)
self.assertEqual( self.assertEqual(
resp.json['error_message']['faultstring'], resp.json['error_message']['faultstring'],
'%s is mandatory' % field) "Invalid input for field/attribute %s."
" Value: \'None\'. Mandatory field missing."
% field.split('/', 1)[-1])
alarms = list(self.conn.get_alarms()) alarms = list(self.conn.get_alarms())
self.assertEqual(4, len(alarms)) self.assertEqual(4, len(alarms))

View File

@ -134,7 +134,8 @@ class TestApiMiddleware(FunctionalTest):
# Ensure translated messages get placed properly into json faults # Ensure translated messages get placed properly into json faults
self.stubs.Set(gettextutils, 'get_localized_message', self.stubs.Set(gettextutils, 'get_localized_message',
self._fake_get_localized_message) self._fake_get_localized_message)
response = self.post_json('/alarms', params={}, response = self.post_json('/alarms', params={'name': 'foobar',
'type': 'threshold'},
expect_errors=True, expect_errors=True,
headers={"Accept": headers={"Accept":
"application/json"} "application/json"}
@ -169,7 +170,8 @@ class TestApiMiddleware(FunctionalTest):
self.stubs.Set(gettextutils, 'get_localized_message', self.stubs.Set(gettextutils, 'get_localized_message',
self._fake_get_localized_message) self._fake_get_localized_message)
response = self.post_json('/alarms', params={}, response = self.post_json('/alarms', params={'name': 'foobar',
'type': 'threshold'},
expect_errors=True, expect_errors=True,
headers={"Accept": headers={"Accept":
"application/xml,*/*"} "application/xml,*/*"}
@ -186,7 +188,8 @@ class TestApiMiddleware(FunctionalTest):
self.stubs.Set(gettextutils, 'get_localized_message', self.stubs.Set(gettextutils, 'get_localized_message',
self._fake_get_localized_message) self._fake_get_localized_message)
response = self.post_json('/alarms', params={}, response = self.post_json('/alarms', params={'name': 'foobar',
'type': 'threshold'},
expect_errors=True, expect_errors=True,
headers={"Accept": headers={"Accept":
"application/xml,*/*", "application/xml,*/*",

View File

@ -198,7 +198,7 @@ class TestPostSamples(FunctionalTest,
s_broke = copy.copy(s1) s_broke = copy.copy(s1)
del s_broke[0][m] del s_broke[0][m]
print('posting without %s' % m) print('posting without %s' % m)
data = self.post_json('/meters/my_counter_name/', s_broke, data = self.post_json('/meters/my_counter_name', s_broke,
expect_errors=True) expect_errors=True)
self.assertEqual(data.status_int, 400) self.assertEqual(data.status_int, 400)