Merge "Simple service for singleton threshold eval"
This commit is contained in:
commit
04bbab3971
90
ceilometer/alarm/service.py
Normal file
90
ceilometer/alarm/service.py
Normal file
@ -0,0 +1,90 @@
|
|||||||
|
# -*- encoding: utf-8 -*-
|
||||||
|
#
|
||||||
|
# Copyright © 2013 Red Hat, Inc
|
||||||
|
#
|
||||||
|
# Author: Eoghan Glynn <eglynn@redhat.com>
|
||||||
|
#
|
||||||
|
# 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 oslo.config import cfg
|
||||||
|
from stevedore import extension
|
||||||
|
|
||||||
|
from ceilometer.service import prepare_service
|
||||||
|
from ceilometer.openstack.common import log
|
||||||
|
from ceilometer.openstack.common import service as os_service
|
||||||
|
from ceilometerclient import client as ceiloclient
|
||||||
|
|
||||||
|
OPTS = [
|
||||||
|
cfg.IntOpt('threshold_evaluation_interval',
|
||||||
|
default=60,
|
||||||
|
help='Period of threshold evaluation cycle, should'
|
||||||
|
' be >= than configured pipeline interval for'
|
||||||
|
' collection of underlying metrics.'),
|
||||||
|
]
|
||||||
|
|
||||||
|
cfg.CONF.register_opts(OPTS, group='alarm')
|
||||||
|
|
||||||
|
LOG = log.getLogger(__name__)
|
||||||
|
|
||||||
|
|
||||||
|
class SingletonAlarmService(os_service.Service):
|
||||||
|
|
||||||
|
ALARM_NAMESPACE = 'ceilometer.alarm'
|
||||||
|
|
||||||
|
def __init__(self):
|
||||||
|
super(SingletonAlarmService, self).__init__()
|
||||||
|
self.extension_manager = extension.ExtensionManager(
|
||||||
|
namespace=self.ALARM_NAMESPACE,
|
||||||
|
invoke_on_load=True,
|
||||||
|
)
|
||||||
|
|
||||||
|
def start(self):
|
||||||
|
super(SingletonAlarmService, self).start()
|
||||||
|
for ext in self.extension_manager.extensions:
|
||||||
|
if ext.name == 'threshold_eval':
|
||||||
|
self.threshold_eval = ext.obj
|
||||||
|
interval = cfg.CONF.alarm.threshold_evaluation_interval
|
||||||
|
args = [ext.obj, self._client()]
|
||||||
|
self.tg.add_timer(
|
||||||
|
interval,
|
||||||
|
self._evaluate_all_alarms,
|
||||||
|
0,
|
||||||
|
*args)
|
||||||
|
break
|
||||||
|
# Add a dummy thread to have wait() working
|
||||||
|
self.tg.add_timer(604800, lambda: None)
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def _client():
|
||||||
|
auth_config = cfg.CONF.service_credentials
|
||||||
|
creds = dict(
|
||||||
|
os_auth_url=auth_config.os_auth_url,
|
||||||
|
os_tenant_name=auth_config.os_tenant_name,
|
||||||
|
os_password=auth_config.os_password,
|
||||||
|
os_username=auth_config.os_username
|
||||||
|
)
|
||||||
|
return ceiloclient.get_client(2, **creds)
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def _evaluate_all_alarms(threshold_eval, api_client):
|
||||||
|
try:
|
||||||
|
alarms = api_client.alarms.list()
|
||||||
|
threshold_eval.assign_alarms(alarms)
|
||||||
|
threshold_eval.evaluate()
|
||||||
|
except Exception:
|
||||||
|
LOG.exception(_('threshold evaluation cycle failed'))
|
||||||
|
|
||||||
|
|
||||||
|
def singleton_alarm():
|
||||||
|
prepare_service()
|
||||||
|
os_service.launch(SingletonAlarmService()).wait()
|
@ -53,7 +53,7 @@ class Evaluator(object):
|
|||||||
# avoid unknown state
|
# avoid unknown state
|
||||||
quorum = 1
|
quorum = 1
|
||||||
|
|
||||||
def __init__(self, notifier):
|
def __init__(self, notifier=None):
|
||||||
self.alarms = []
|
self.alarms = []
|
||||||
self.notifier = notifier
|
self.notifier = notifier
|
||||||
self.api_client = None
|
self.api_client = None
|
||||||
|
@ -526,6 +526,18 @@
|
|||||||
#pool_timeout=<None>
|
#pool_timeout=<None>
|
||||||
|
|
||||||
|
|
||||||
|
[alarm]
|
||||||
|
|
||||||
|
#
|
||||||
|
# Options defined in ceilometer.alarm.service
|
||||||
|
#
|
||||||
|
|
||||||
|
# Period of threshold evaluation cycle, should be >= than
|
||||||
|
# configured pipeline interval for collection of underlying
|
||||||
|
# metrics. (integer value)
|
||||||
|
#threshold_evaluation_interval=60
|
||||||
|
|
||||||
|
|
||||||
[rpc_notifier2]
|
[rpc_notifier2]
|
||||||
|
|
||||||
#
|
#
|
||||||
@ -612,4 +624,4 @@
|
|||||||
#password=<None>
|
#password=<None>
|
||||||
|
|
||||||
|
|
||||||
# Total option count: 118
|
# Total option count: 119
|
||||||
|
@ -83,6 +83,9 @@ ceilometer.publisher =
|
|||||||
rpc = ceilometer.publisher.rpc:RPCPublisher
|
rpc = ceilometer.publisher.rpc:RPCPublisher
|
||||||
udp = ceilometer.publisher.udp:UDPPublisher
|
udp = ceilometer.publisher.udp:UDPPublisher
|
||||||
|
|
||||||
|
ceilometer.alarm =
|
||||||
|
threshold_eval = ceilometer.alarm.threshold_evaluation:Evaluator
|
||||||
|
|
||||||
paste.filter_factory =
|
paste.filter_factory =
|
||||||
swift = ceilometer.objectstore.swift_middleware:filter_factory
|
swift = ceilometer.objectstore.swift_middleware:filter_factory
|
||||||
|
|
||||||
@ -93,6 +96,7 @@ console_scripts =
|
|||||||
ceilometer-dbsync = ceilometer.storage:dbsync
|
ceilometer-dbsync = ceilometer.storage:dbsync
|
||||||
ceilometer-collector = ceilometer.collector.service:collector
|
ceilometer-collector = ceilometer.collector.service:collector
|
||||||
ceilometer-collector-udp = ceilometer.collector.service:udp_collector
|
ceilometer-collector-udp = ceilometer.collector.service:udp_collector
|
||||||
|
ceilometer-alarm-singleton = ceilometer.alarm.service:singleton_alarm
|
||||||
|
|
||||||
[build_sphinx]
|
[build_sphinx]
|
||||||
all_files = 1
|
all_files = 1
|
||||||
|
85
tests/alarm/test_singleton_alarm_svc.py
Normal file
85
tests/alarm/test_singleton_alarm_svc.py
Normal file
@ -0,0 +1,85 @@
|
|||||||
|
# -*- encoding: utf-8 -*-
|
||||||
|
#
|
||||||
|
# Copyright © 2013 Red Hat, Inc
|
||||||
|
#
|
||||||
|
# Author: Eoghan Glynn <eglynn@redhat.com>
|
||||||
|
#
|
||||||
|
# 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.
|
||||||
|
"""Tests for ceilometer/alarm/service.py
|
||||||
|
"""
|
||||||
|
import mock
|
||||||
|
import uuid
|
||||||
|
|
||||||
|
from stevedore import extension
|
||||||
|
from stevedore.tests import manager as extension_tests
|
||||||
|
|
||||||
|
from ceilometer.alarm import service
|
||||||
|
from ceilometer.storage import models
|
||||||
|
from ceilometer.tests import base
|
||||||
|
|
||||||
|
|
||||||
|
class TestSingletonAlarmService(base.TestCase):
|
||||||
|
def setUp(self):
|
||||||
|
super(TestSingletonAlarmService, self).setUp()
|
||||||
|
self.threshold_eval = mock.Mock()
|
||||||
|
self.extension_mgr = extension_tests.TestExtensionManager(
|
||||||
|
[
|
||||||
|
extension.Extension(
|
||||||
|
'threshold_eval',
|
||||||
|
None,
|
||||||
|
None,
|
||||||
|
self.threshold_eval, ),
|
||||||
|
])
|
||||||
|
self.api_client = mock.MagicMock()
|
||||||
|
self.singleton = service.SingletonAlarmService()
|
||||||
|
self.singleton.tg = mock.Mock()
|
||||||
|
self.singleton.extension_manager = self.extension_mgr
|
||||||
|
|
||||||
|
def test_start(self):
|
||||||
|
with mock.patch('ceilometerclient.client.get_client',
|
||||||
|
return_value=self.api_client):
|
||||||
|
self.singleton.start()
|
||||||
|
expected = [
|
||||||
|
mock.call(60,
|
||||||
|
self.singleton._evaluate_all_alarms,
|
||||||
|
0,
|
||||||
|
self.threshold_eval,
|
||||||
|
self.api_client),
|
||||||
|
mock.call(604800, mock.ANY),
|
||||||
|
]
|
||||||
|
actual = self.singleton.tg.add_timer.call_args_list
|
||||||
|
self.assertEqual(actual, expected)
|
||||||
|
|
||||||
|
def test_evaluation_cycle(self):
|
||||||
|
alarms = [
|
||||||
|
models.Alarm(name='instance_running_hot',
|
||||||
|
counter_name='cpu_util',
|
||||||
|
comparison_operator='gt',
|
||||||
|
threshold=80.0,
|
||||||
|
evaluation_periods=5,
|
||||||
|
statistic='avg',
|
||||||
|
user_id='foobar',
|
||||||
|
project_id='snafu',
|
||||||
|
period=60,
|
||||||
|
alarm_id=str(uuid.uuid4())),
|
||||||
|
]
|
||||||
|
self.api_client.alarms.list.return_value = alarms
|
||||||
|
with mock.patch('ceilometerclient.client.get_client',
|
||||||
|
return_value=self.api_client):
|
||||||
|
self.singleton.start()
|
||||||
|
self.singleton._evaluate_all_alarms(
|
||||||
|
self.threshold_eval,
|
||||||
|
self.api_client
|
||||||
|
)
|
||||||
|
self.threshold_eval.assign_alarms.assert_called_once_with(alarms)
|
||||||
|
self.threshold_eval.evaluate.assert_called_once_with()
|
Loading…
Reference in New Issue
Block a user