# -*- encoding: utf-8 -*- # # Copyright © 2013 Red Hat, Inc # Copyright © 2013 eNovance # # Authors: Eoghan Glynn # Julien Danjou # # 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.alarm import rpc as rpc_alarm from ceilometer.service import prepare_service from ceilometer.openstack.common import log from ceilometer.openstack.common import network_utils from ceilometer.openstack.common import service as os_service from ceilometer.openstack.common.gettextutils import _ from ceilometer.openstack.common.rpc import service as rpc_service from ceilometer.openstack.common.rpc import dispatcher as rpc_dispatcher 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') cfg.CONF.import_opt('notifier_rpc_topic', 'ceilometer.alarm.rpc', 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, invoke_args=(rpc_alarm.RPCAlarmNotifier(),) ) 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, endpoint_type=auth_config.os_endpoint_type, ) 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() cfg.CONF.import_opt('host', 'ceilometer.service') class AlarmNotifierService(rpc_service.Service): EXTENSIONS_NAMESPACE = "ceilometer.alarm.notifier" def __init__(self, host, topic): super(AlarmNotifierService, self).__init__(host, topic, self) self.notifiers = extension.ExtensionManager(self.EXTENSIONS_NAMESPACE, invoke_on_load=True) def start(self): super(AlarmNotifierService, self).start() # Add a dummy thread to have wait() working self.tg.add_timer(604800, lambda: None) def initialize_service_hook(self, service): LOG.debug('initialize_service_hooks') self.conn.create_worker( cfg.CONF.alarm.notifier_rpc_topic, rpc_dispatcher.RpcDispatcher([self]), 'ceilometer.alarm.' + cfg.CONF.alarm.notifier_rpc_topic, ) def _handle_action(self, action, alarm, state, reason): try: action = network_utils.urlsplit(action) except Exception: LOG.error( _("Unable to parse action %(action)s for alarm %(alarm)s"), locals()) return try: notifier = self.notifiers[action.scheme].obj except KeyError: scheme = action.scheme LOG.error( _("Action %(scheme)s for alarm %(alarm)s is unknown, " "cannot notify"), locals()) return try: LOG.debug("Notifying alarm %s with action %s", alarm, action) notifier.notify(action, alarm, state, reason) except Exception: LOG.exception(_("Unable to notify alarm %s"), alarm) return def notify_alarm(self, context, data): """Notify that alarm has been triggered. data should be a dict with the following keys: - actions, the URL of the action to run; this is a mapped to extensions automatically - alarm, the alarm that has been triggered - state, the new state the alarm transitionned to - reason, the reason the alarm changed its state :param context: Request context. :param data: A dict as described above. """ actions = data.get('actions') if not actions: LOG.error(_("Unable to notify for an alarm with no action")) return for action in actions: self._handle_action(action, data.get('alarm'), data.get('state'), data.get('reason')) def alarm_notifier(): prepare_service() os_service.launch(AlarmNotifierService( cfg.CONF.host, 'ceilometer.alarm')).wait()