diff --git a/dist/panel-triggers/migrations.js b/dist/panel-triggers/migrations.js index c6fd1ff..7c2999e 100644 --- a/dist/panel-triggers/migrations.js +++ b/dist/panel-triggers/migrations.js @@ -4,11 +4,20 @@ System.register([], function (_export, _context) { "use strict"; function migratePanelSchema(panel) { + if (isEmptyPanel(panel)) { + return panel; + } + var schemaVersion = getSchemaVersion(panel); switch (schemaVersion) { case 1: panel.datasources = [panel.datasource]; + panel.targets = {}; panel.targets[panel.datasources[0]] = panel.triggers; + + // delete old props + delete panel.triggers; + delete panel.datasource; break; } @@ -20,6 +29,10 @@ System.register([], function (_export, _context) { function getSchemaVersion(panel) { return panel.schemaVersion || 1; } + + function isEmptyPanel(panel) { + return !panel.datasource && !panel.datasources && !panel.triggers && !panel.targets; + } return { setters: [], execute: function () {} diff --git a/dist/panel-triggers/migrations.js.map b/dist/panel-triggers/migrations.js.map index 71c8945..29aca82 100644 --- a/dist/panel-triggers/migrations.js.map +++ b/dist/panel-triggers/migrations.js.map @@ -1 +1 @@ -{"version":3,"sources":["../../src/panel-triggers/migrations.js"],"names":["migratePanelSchema","panel","schemaVersion","getSchemaVersion","datasources","datasource","targets","triggers"],"mappings":";;;;;AAAO,WAASA,kBAAT,CAA4BC,KAA5B,EAAmC;AACxC,QAAMC,gBAAgBC,iBAAiBF,KAAjB,CAAtB;AACA,YAAQC,aAAR;AACE,WAAK,CAAL;AACED,cAAMG,WAAN,GAAoB,CAACH,MAAMI,UAAP,CAApB;AACAJ,cAAMK,OAAN,CAAcL,MAAMG,WAAN,CAAkB,CAAlB,CAAd,IAAsCH,MAAMM,QAA5C;AACA;AAJJ;;AAOA,WAAON,KAAP;AACD;;gCAVeD,kB;;AAYhB,WAASG,gBAAT,CAA0BF,KAA1B,EAAiC;AAC/B,WAAOA,MAAMC,aAAN,IAAuB,CAA9B;AACD","file":"migrations.js","sourcesContent":["export function migratePanelSchema(panel) {\n const schemaVersion = getSchemaVersion(panel);\n switch (schemaVersion) {\n case 1:\n panel.datasources = [panel.datasource];\n panel.targets[panel.datasources[0]] = panel.triggers;\n break;\n }\n\n return panel;\n}\n\nfunction getSchemaVersion(panel) {\n return panel.schemaVersion || 1;\n}\n"]} \ No newline at end of file +{"version":3,"sources":["../../src/panel-triggers/migrations.js"],"names":["migratePanelSchema","panel","isEmptyPanel","schemaVersion","getSchemaVersion","datasources","datasource","targets","triggers"],"mappings":";;;;;AAAO,WAASA,kBAAT,CAA4BC,KAA5B,EAAmC;AACxC,QAAIC,aAAaD,KAAb,CAAJ,EAAyB;AACvB,aAAOA,KAAP;AACD;;AAED,QAAME,gBAAgBC,iBAAiBH,KAAjB,CAAtB;AACA,YAAQE,aAAR;AACE,WAAK,CAAL;AACEF,cAAMI,WAAN,GAAoB,CAACJ,MAAMK,UAAP,CAApB;AACAL,cAAMM,OAAN,GAAgB,EAAhB;AACAN,cAAMM,OAAN,CAAcN,MAAMI,WAAN,CAAkB,CAAlB,CAAd,IAAsCJ,MAAMO,QAA5C;;AAEA;AACA,eAAOP,MAAMO,QAAb;AACA,eAAOP,MAAMK,UAAb;AACA;AATJ;;AAYA,WAAOL,KAAP;AACD;;gCAnBeD,kB;;AAqBhB,WAASI,gBAAT,CAA0BH,KAA1B,EAAiC;AAC/B,WAAOA,MAAME,aAAN,IAAuB,CAA9B;AACD;;AAED,WAASD,YAAT,CAAsBD,KAAtB,EAA6B;AAC3B,WAAO,CAACA,MAAMK,UAAP,IAAqB,CAACL,MAAMI,WAA5B,IAA2C,CAACJ,MAAMO,QAAlD,IAA8D,CAACP,MAAMM,OAA5E;AACD","file":"migrations.js","sourcesContent":["export function migratePanelSchema(panel) {\n if (isEmptyPanel(panel)) {\n return panel;\n }\n\n const schemaVersion = getSchemaVersion(panel);\n switch (schemaVersion) {\n case 1:\n panel.datasources = [panel.datasource];\n panel.targets = {};\n panel.targets[panel.datasources[0]] = panel.triggers;\n\n // delete old props\n delete panel.triggers;\n delete panel.datasource;\n break;\n }\n\n return panel;\n}\n\nfunction getSchemaVersion(panel) {\n return panel.schemaVersion || 1;\n}\n\nfunction isEmptyPanel(panel) {\n return !panel.datasource && !panel.datasources && !panel.triggers && !panel.targets;\n}\n"]} \ No newline at end of file diff --git a/dist/panel-triggers/module.js b/dist/panel-triggers/module.js index 6aed50a..be3f5da 100644 --- a/dist/panel-triggers/module.js +++ b/dist/panel-triggers/module.js @@ -1,545 +1,32 @@ 'use strict'; -System.register(['lodash', 'jquery', 'moment', 'app/plugins/sdk', '../datasource-zabbix/utils', './options_tab', './triggers_tab', './migrations', './ack-tooltip.directive'], function (_export, _context) { +System.register(['./triggers_panel_ctrl', 'app/plugins/sdk', './ack-tooltip.directive'], function (_export, _context) { "use strict"; - var _, $, moment, loadPluginCss, utils, PanelCtrl, triggerPanelOptionsTab, triggerPanelTriggersTab, migratePanelSchema, _createClass, ZABBIX_DS_ID, DEFAULT_TARGET, defaultSeverity, DEFAULT_TIME_FORMAT, panelDefaults, triggerStatusMap, TriggerPanelCtrl; - - function _classCallCheck(instance, Constructor) { - if (!(instance instanceof Constructor)) { - throw new TypeError("Cannot call a class as a function"); - } - } - - function _possibleConstructorReturn(self, call) { - if (!self) { - throw new ReferenceError("this hasn't been initialised - super() hasn't been called"); - } - - return call && (typeof call === "object" || typeof call === "function") ? call : self; - } - - function _inherits(subClass, superClass) { - if (typeof superClass !== "function" && superClass !== null) { - throw new TypeError("Super expression must either be null or a function, not " + typeof superClass); - } - - subClass.prototype = Object.create(superClass && superClass.prototype, { - constructor: { - value: subClass, - enumerable: false, - writable: true, - configurable: true - } - }); - if (superClass) Object.setPrototypeOf ? Object.setPrototypeOf(subClass, superClass) : subClass.__proto__ = superClass; - } - - function _filterTriggers(triggers, triggerFilter) { - if (utils.isRegex(triggerFilter)) { - return _.filter(triggers, function (trigger) { - return utils.buildRegex(triggerFilter).test(trigger.description); - }); - } else { - return _.filter(triggers, function (trigger) { - return trigger.description === triggerFilter; - }); - } - } - + var TriggerPanelCtrl, loadPluginCss; return { - setters: [function (_lodash) { - _ = _lodash.default; - }, function (_jquery) { - $ = _jquery.default; - }, function (_moment) { - moment = _moment.default; + setters: [function (_triggers_panel_ctrl) { + TriggerPanelCtrl = _triggers_panel_ctrl.TriggerPanelCtrl; }, function (_appPluginsSdk) { loadPluginCss = _appPluginsSdk.loadPluginCss; - PanelCtrl = _appPluginsSdk.PanelCtrl; - }, function (_datasourceZabbixUtils) { - utils = _datasourceZabbixUtils; - }, function (_options_tab) { - triggerPanelOptionsTab = _options_tab.triggerPanelOptionsTab; - }, function (_triggers_tab) { - triggerPanelTriggersTab = _triggers_tab.triggerPanelTriggersTab; - }, function (_migrations) { - migratePanelSchema = _migrations.migratePanelSchema; }, function (_ackTooltipDirective) {}], execute: function () { - _createClass = function () { - function defineProperties(target, props) { - for (var i = 0; i < props.length; i++) { - var descriptor = props[i]; - descriptor.enumerable = descriptor.enumerable || false; - descriptor.configurable = true; - if ("value" in descriptor) descriptor.writable = true; - Object.defineProperty(target, descriptor.key, descriptor); - } - } - - return function (Constructor, protoProps, staticProps) { - if (protoProps) defineProperties(Constructor.prototype, protoProps); - if (staticProps) defineProperties(Constructor, staticProps); - return Constructor; - }; - }(); - - /** - * Grafana-Zabbix - * Zabbix plugin for Grafana. - * http://github.com/alexanderzobnin/grafana-zabbix - * - * Trigger panel. - * This feature sponsored by CORE IT - * http://www.coreit.fr - * - * Copyright 2015 Alexander Zobnin alexanderzobnin@gmail.com - * Licensed under the Apache License, Version 2.0 - */ loadPluginCss({ dark: 'plugins/alexanderzobnin-zabbix-app/css/grafana-zabbix.dark.css', light: 'plugins/alexanderzobnin-zabbix-app/css/grafana-zabbix.light.css' - }); - - ZABBIX_DS_ID = 'alexanderzobnin-zabbix-datasource'; - - _export('DEFAULT_TARGET', DEFAULT_TARGET = { - group: { filter: "" }, - host: { filter: "" }, - application: { filter: "" }, - trigger: { filter: "" } - }); - - _export('DEFAULT_TARGET', DEFAULT_TARGET); - - defaultSeverity = [{ priority: 0, severity: 'Not classified', color: '#B7DBAB', show: true }, { priority: 1, severity: 'Information', color: '#82B5D8', show: true }, { priority: 2, severity: 'Warning', color: '#E5AC0E', show: true }, { priority: 3, severity: 'Average', color: '#C15C17', show: true }, { priority: 4, severity: 'High', color: '#BF1B00', show: true }, { priority: 5, severity: 'Disaster', color: '#890F02', show: true }]; - DEFAULT_TIME_FORMAT = "DD MMM YYYY HH:mm:ss"; - panelDefaults = { - schemaVersion: 2, - datasources: [], - targets: {}, - // Fields - hostField: true, - statusField: false, - severityField: false, - lastChangeField: true, - ageField: true, - infoField: true, - // Options - hideHostsInMaintenance: false, - showTriggers: 'all triggers', - sortTriggersBy: { text: 'last change', value: 'lastchange' }, - showEvents: { text: 'Problems', value: '1' }, - limit: 10, - // View options - fontSize: '100%', - pageSize: 10, - scroll: true, - customLastChangeFormat: false, - lastChangeFormat: "", - // Triggers severity and colors - triggerSeverity: defaultSeverity, - okEventColor: 'rgba(0, 245, 153, 0.45)', - ackEventColor: 'rgba(0, 0, 0, 0)' - }; - triggerStatusMap = { - '0': 'OK', - '1': 'Problem' - }; - - _export('PanelCtrl', _export('TriggerPanelCtrl', TriggerPanelCtrl = function (_PanelCtrl) { - _inherits(TriggerPanelCtrl, _PanelCtrl); - - /** @ngInject */ - function TriggerPanelCtrl($scope, $injector, $element, datasourceSrv, templateSrv, contextSrv, dashboardSrv) { - _classCallCheck(this, TriggerPanelCtrl); - - var _this = _possibleConstructorReturn(this, (TriggerPanelCtrl.__proto__ || Object.getPrototypeOf(TriggerPanelCtrl)).call(this, $scope, $injector)); - - _this.datasourceSrv = datasourceSrv; - _this.templateSrv = templateSrv; - _this.contextSrv = contextSrv; - _this.dashboardSrv = dashboardSrv; - - _this.editorTabIndex = 1; - _this.triggerStatusMap = triggerStatusMap; - _this.defaultTimeFormat = DEFAULT_TIME_FORMAT; - _this.pageIndex = 0; - _this.triggerList = []; - _this.currentTriggersPage = []; - _this.datasources = {}; - - _.defaults(_this.panel, _.cloneDeep(panelDefaults)); - _this.panel = migratePanelSchema(_this.panel); - - _this.available_datasources = _.map(_this.getZabbixDataSources(), 'name'); - if (_this.panel.datasources.length === 0) { - _this.panel.datasources.push(_this.available_datasources[0]); - } - if (_.isEmpty(_this.panel.targets)) { - _this.panel.targets[_this.panel.datasources[0]] = DEFAULT_TARGET; - } - - _this.initDatasources(); - _this.events.on('init-edit-mode', _this.onInitEditMode.bind(_this)); - _this.events.on('refresh', _this.onRefresh.bind(_this)); - return _this; - } - - _createClass(TriggerPanelCtrl, [{ - key: 'initDatasources', - value: function initDatasources() { - var _this2 = this; - - var promises = _.map(this.panel.datasources, function (ds) { - // Load datasource - return _this2.datasourceSrv.get(ds).then(function (datasource) { - _this2.datasources[ds] = datasource; - return datasource; - }); - }); - return Promise.all(promises); - } - }, { - key: 'getZabbixDataSources', - value: function getZabbixDataSources() { - return _.filter(this.datasourceSrv.getMetricSources(), function (datasource) { - return datasource.meta.id === ZABBIX_DS_ID && datasource.value; - }); - } - }, { - key: 'onInitEditMode', - value: function onInitEditMode() { - this.addEditorTab('Triggers', triggerPanelTriggersTab, 1); - this.addEditorTab('Options', triggerPanelOptionsTab, 2); - } - }, { - key: 'setTimeQueryStart', - value: function setTimeQueryStart() { - this.timing.queryStart = new Date().getTime(); - } - }, { - key: 'setTimeQueryEnd', - value: function setTimeQueryEnd() { - this.timing.queryEnd = new Date().getTime(); - } - }, { - key: 'onRefresh', - value: function onRefresh() { - var _this3 = this; - - // ignore fetching data if another panel is in fullscreen - if (this.otherPanelInFullscreenMode()) { - return; - } - - // clear loading/error state - delete this.error; - this.loading = true; - this.setTimeQueryStart(); - - return this.getTriggers().then(function (triggerList) { - // Notify panel that request is finished - _this3.loading = false; - _this3.setTimeQueryEnd(); - - // Limit triggers number - _this3.triggerList = triggerList.slice(0, _this3.panel.limit); - _this3.getCurrentTriggersPage(); - _this3.render(_this3.triggerList); - }).catch(function (err) { - // if cancelled keep loading set to true - if (err.cancelled) { - console.log('Panel request cancelled', err); - return; - } - - _this3.loading = false; - _this3.error = err.message || "Request Error"; - - if (err.data) { - if (err.data.message) { - _this3.error = err.data.message; - } - if (err.data.error) { - _this3.error = err.data.error; - } - } - - _this3.events.emit('data-error', err); - console.log('Panel data error:', err); - }); - } - }, { - key: 'getTriggers', - value: function getTriggers() { - var _this4 = this; - - var promises = _.map(this.panel.datasources, function (ds) { - return _this4.datasourceSrv.get(ds).then(function (datasource) { - var zabbix = datasource.zabbix; - var showEvents = _this4.panel.showEvents.value; - var triggerFilter = _this4.panel.targets[ds]; - var hideHostsInMaintenance = _this4.panel.hideHostsInMaintenance; - - // Replace template variables - var groupFilter = datasource.replaceTemplateVars(triggerFilter.group.filter); - var hostFilter = datasource.replaceTemplateVars(triggerFilter.host.filter); - var appFilter = datasource.replaceTemplateVars(triggerFilter.application.filter); - - var triggersOptions = { - showTriggers: showEvents, - hideHostsInMaintenance: hideHostsInMaintenance - }; - - return zabbix.getTriggers(groupFilter, hostFilter, appFilter, triggersOptions); - }).then(function (triggers) { - return _this4.getAcknowledges(triggers, ds); - }).then(function (triggers) { - return _this4.filterTriggers(triggers, ds); - }); - }); - - return Promise.all(promises).then(function (results) { - return _.flatten(results); - }).then(function (triggers) { - return _this4.sortTriggers(triggers); - }).then(function (triggers) { - return _.map(triggers, _this4.formatTrigger.bind(_this4)); - }); - } - }, { - key: 'getAcknowledges', - value: function getAcknowledges(triggerList, ds) { - var _this5 = this; - - // Request acknowledges for trigger - var eventids = _.map(triggerList, function (trigger) { - return trigger.lastEvent.eventid; - }); - - return this.datasources[ds].zabbix.getAcknowledges(eventids).then(function (events) { - - // Map events to triggers - _.each(triggerList, function (trigger) { - var event = _.find(events, function (event) { - return event.eventid === trigger.lastEvent.eventid; - }); - - if (event) { - trigger.acknowledges = _.map(event.acknowledges, function (ack) { - var timestamp = moment.unix(ack.clock); - if (_this5.panel.customLastChangeFormat) { - ack.time = timestamp.format(_this5.panel.lastChangeFormat); - } else { - ack.time = timestamp.format(_this5.defaultTimeFormat); - } - ack.user = ack.alias + ' (' + ack.name + ' ' + ack.surname + ')'; - return ack; - }); - - // Mark acknowledged triggers with different color - if (_this5.panel.markAckEvents && trigger.acknowledges.length) { - trigger.color = _this5.panel.ackEventColor; - } - } - }); - - return triggerList; - }); - } - }, { - key: 'filterTriggers', - value: function filterTriggers(triggerList, ds) { - var _this6 = this; - - // Filter triggers by description - var triggerFilter = this.panel.targets[ds].trigger.filter; - triggerFilter = this.datasources[ds].replaceTemplateVars(triggerFilter); - if (triggerFilter) { - triggerList = _filterTriggers(triggerList, triggerFilter); - } - - // Filter acknowledged triggers - if (this.panel.showTriggers === 'unacknowledged') { - triggerList = _.filter(triggerList, function (trigger) { - return !trigger.acknowledges; - }); - } else if (this.panel.showTriggers === 'acknowledged') { - triggerList = _.filter(triggerList, 'acknowledges'); - } else { - triggerList = triggerList; - } - - // Filter triggers by severity - triggerList = _.filter(triggerList, function (trigger) { - return _this6.panel.triggerSeverity[trigger.priority].show; - }); - - return triggerList; - } - }, { - key: 'sortTriggers', - value: function sortTriggers(triggerList) { - if (this.panel.sortTriggersBy.value === 'priority') { - triggerList = _.sortBy(triggerList, 'priority').reverse(); - } else { - triggerList = _.sortBy(triggerList, 'lastchangeUnix').reverse(); - } - return triggerList; - } - }, { - key: 'formatTrigger', - value: function formatTrigger(trigger) { - var triggerObj = trigger; - - // Format last change and age - trigger.lastchangeUnix = Number(trigger.lastchange); - var timestamp = moment.unix(trigger.lastchangeUnix); - if (this.panel.customLastChangeFormat) { - // User defined format - triggerObj.lastchange = timestamp.format(this.panel.lastChangeFormat); - } else { - triggerObj.lastchange = timestamp.format(this.defaultTimeFormat); - } - triggerObj.age = timestamp.fromNow(true); - - // Set host that the trigger belongs - if (trigger.hosts.length) { - triggerObj.host = trigger.hosts[0].name; - triggerObj.hostTechName = trigger.hosts[0].host; - } - - // Set color - if (trigger.value === '1') { - // Problem state - triggerObj.color = this.panel.triggerSeverity[trigger.priority].color; - } else { - // OK state - triggerObj.color = this.panel.okEventColor; - } - - triggerObj.severity = this.panel.triggerSeverity[trigger.priority].severity; - return triggerObj; - } - }, { - key: 'switchComment', - value: function switchComment(trigger) { - trigger.showComment = !trigger.showComment; - } - }, { - key: 'acknowledgeTrigger', - value: function acknowledgeTrigger(trigger, message) { - var eventid = trigger.lastEvent.eventid; - var grafana_user = this.contextSrv.user.name; - var ack_message = grafana_user + ' (Grafana): ' + message; - return this.datasourceSrv.get(this.panel.datasource).then(function (datasource) { - var zabbixAPI = datasource.zabbix.zabbixAPI; - return zabbixAPI.acknowledgeEvent(eventid, ack_message); - }).then(this.onRefresh.bind(this)); - } - }, { - key: 'getCurrentTriggersPage', - value: function getCurrentTriggersPage() { - var pageSize = this.panel.pageSize || 10; - var startPos = this.pageIndex * pageSize; - var endPos = Math.min(startPos + pageSize, this.triggerList.length); - this.currentTriggersPage = this.triggerList.slice(startPos, endPos); - return this.currentTriggersPage; - } - }, { - key: 'link', - value: function link(scope, elem, attrs, ctrl) { - var data; - var panel = ctrl.panel; - var pageCount = 0; - data = ctrl.triggerList; - - function getTableHeight() { - var panelHeight = ctrl.height; - - if (pageCount > 1) { - panelHeight -= 26; - } - - return panelHeight - 31 + 'px'; - } - - function switchPage(e) { - var el = $(e.currentTarget); - ctrl.pageIndex = parseInt(el.text(), 10) - 1; - - var pageSize = ctrl.panel.pageSize || 10; - var startPos = ctrl.pageIndex * pageSize; - var endPos = Math.min(startPos + pageSize, ctrl.triggerList.length); - ctrl.currentTriggersPage = ctrl.triggerList.slice(startPos, endPos); - - scope.$apply(function () { - renderPanel(); - }); - } - - function appendPaginationControls(footerElem) { - footerElem.empty(); - - var pageSize = ctrl.panel.pageSize || 5; - pageCount = Math.ceil(data.length / pageSize); - if (pageCount === 1) { - return; - } - - var startPage = Math.max(ctrl.pageIndex - 3, 0); - var endPage = Math.min(pageCount, startPage + 9); - - var paginationList = $(''); - - for (var i = startPage; i < endPage; i++) { - var activeClass = i === ctrl.pageIndex ? 'active' : ''; - var pageLinkElem = $('
  • ' + (i + 1) + '
  • '); - paginationList.append(pageLinkElem); - } - - footerElem.append(paginationList); - } - - function renderPanel() { - var panelElem = elem.parents('.panel'); - var rootElem = elem.find('.triggers-panel-scroll'); - var footerElem = elem.find('.triggers-panel-footer'); - - elem.css({ 'font-size': panel.fontSize }); - panelElem.addClass('triggers-panel-wrapper'); - appendPaginationControls(footerElem); - - rootElem.css({ 'max-height': panel.scroll ? getTableHeight() : '' }); - ctrl.renderingCompleted(); - } - - elem.on('click', '.triggers-panel-page-link', switchPage); - - var unbindDestroy = scope.$on('$destroy', function () { - elem.off('click', '.triggers-panel-page-link'); - unbindDestroy(); - }); - - ctrl.events.on('render', function (renderData) { - data = renderData || data; - if (data) { - scope.$apply(function () { - renderPanel(); - }); - } - }); - } - }]); - - return TriggerPanelCtrl; - }(PanelCtrl))); - - TriggerPanelCtrl.templateUrl = 'panel-triggers/module.html'; - _export('TriggerPanelCtrl', TriggerPanelCtrl); + }); /** + * Grafana-Zabbix + * Zabbix plugin for Grafana. + * http://github.com/alexanderzobnin/grafana-zabbix + * + * Trigger panel. + * This feature sponsored by CORE IT + * http://www.coreit.fr + * + * Copyright 2015 Alexander Zobnin alexanderzobnin@gmail.com + * Licensed under the Apache License, Version 2.0 + */ _export('PanelCtrl', TriggerPanelCtrl); } diff --git a/dist/panel-triggers/module.js.map b/dist/panel-triggers/module.js.map index 17028e8..e38db54 100644 --- a/dist/panel-triggers/module.js.map +++ b/dist/panel-triggers/module.js.map @@ -1 +1 @@ -{"version":3,"sources":["../../src/panel-triggers/module.js"],"names":["filterTriggers","triggers","triggerFilter","utils","isRegex","_","filter","trigger","buildRegex","test","description","$","moment","loadPluginCss","PanelCtrl","triggerPanelOptionsTab","triggerPanelTriggersTab","migratePanelSchema","dark","light","ZABBIX_DS_ID","DEFAULT_TARGET","group","host","application","defaultSeverity","priority","severity","color","show","DEFAULT_TIME_FORMAT","panelDefaults","schemaVersion","datasources","targets","hostField","statusField","severityField","lastChangeField","ageField","infoField","hideHostsInMaintenance","showTriggers","sortTriggersBy","text","value","showEvents","limit","fontSize","pageSize","scroll","customLastChangeFormat","lastChangeFormat","triggerSeverity","okEventColor","ackEventColor","triggerStatusMap","TriggerPanelCtrl","$scope","$injector","$element","datasourceSrv","templateSrv","contextSrv","dashboardSrv","editorTabIndex","defaultTimeFormat","pageIndex","triggerList","currentTriggersPage","defaults","panel","cloneDeep","available_datasources","map","getZabbixDataSources","length","push","isEmpty","initDatasources","events","on","onInitEditMode","bind","onRefresh","promises","ds","get","then","datasource","Promise","all","getMetricSources","meta","id","addEditorTab","timing","queryStart","Date","getTime","queryEnd","otherPanelInFullscreenMode","error","loading","setTimeQueryStart","getTriggers","setTimeQueryEnd","slice","getCurrentTriggersPage","render","catch","err","cancelled","console","log","message","data","emit","zabbix","groupFilter","replaceTemplateVars","hostFilter","appFilter","triggersOptions","getAcknowledges","flatten","results","sortTriggers","formatTrigger","eventids","lastEvent","eventid","each","event","find","acknowledges","timestamp","unix","ack","clock","time","format","user","alias","name","surname","markAckEvents","sortBy","reverse","triggerObj","lastchangeUnix","Number","lastchange","age","fromNow","hosts","hostTechName","showComment","grafana_user","ack_message","zabbixAPI","acknowledgeEvent","startPos","endPos","Math","min","scope","elem","attrs","ctrl","pageCount","getTableHeight","panelHeight","height","switchPage","e","el","currentTarget","parseInt","$apply","renderPanel","appendPaginationControls","footerElem","empty","ceil","startPage","max","endPage","paginationList","i","activeClass","pageLinkElem","append","panelElem","parents","rootElem","css","addClass","renderingCompleted","unbindDestroy","$on","off","renderData","templateUrl"],"mappings":"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","file":"module.js","sourcesContent":["/**\n * Grafana-Zabbix\n * Zabbix plugin for Grafana.\n * http://github.com/alexanderzobnin/grafana-zabbix\n *\n * Trigger panel.\n * This feature sponsored by CORE IT\n * http://www.coreit.fr\n *\n * Copyright 2015 Alexander Zobnin alexanderzobnin@gmail.com\n * Licensed under the Apache License, Version 2.0\n */\n\nimport _ from 'lodash';\nimport $ from 'jquery';\nimport moment from 'moment';\nimport {loadPluginCss} from 'app/plugins/sdk';\nimport * as utils from '../datasource-zabbix/utils';\nimport {PanelCtrl} from 'app/plugins/sdk';\nimport {triggerPanelOptionsTab} from './options_tab';\nimport {triggerPanelTriggersTab} from './triggers_tab';\nimport {migratePanelSchema} from './migrations';\nimport './ack-tooltip.directive';\n\nloadPluginCss({\n dark: 'plugins/alexanderzobnin-zabbix-app/css/grafana-zabbix.dark.css',\n light: 'plugins/alexanderzobnin-zabbix-app/css/grafana-zabbix.light.css'\n});\n\nconst ZABBIX_DS_ID = 'alexanderzobnin-zabbix-datasource';\n\nexport const DEFAULT_TARGET = {\n group: {filter: \"\"},\n host: {filter: \"\"},\n application: {filter: \"\"},\n trigger: {filter: \"\"}\n};\n\nconst defaultSeverity = [\n { priority: 0, severity: 'Not classified', color: '#B7DBAB', show: true },\n { priority: 1, severity: 'Information', color: '#82B5D8', show: true },\n { priority: 2, severity: 'Warning', color: '#E5AC0E', show: true },\n { priority: 3, severity: 'Average', color: '#C15C17', show: true },\n { priority: 4, severity: 'High', color: '#BF1B00', show: true },\n { priority: 5, severity: 'Disaster', color: '#890F02', show: true }\n];\n\nconst DEFAULT_TIME_FORMAT = \"DD MMM YYYY HH:mm:ss\";\n\nconst panelDefaults = {\n schemaVersion: 2,\n datasources: [],\n targets: {},\n // Fields\n hostField: true,\n statusField: false,\n severityField: false,\n lastChangeField: true,\n ageField: true,\n infoField: true,\n // Options\n hideHostsInMaintenance: false,\n showTriggers: 'all triggers',\n sortTriggersBy: { text: 'last change', value: 'lastchange' },\n showEvents: { text: 'Problems', value: '1' },\n limit: 10,\n // View options\n fontSize: '100%',\n pageSize: 10,\n scroll: true,\n customLastChangeFormat: false,\n lastChangeFormat: \"\",\n // Triggers severity and colors\n triggerSeverity: defaultSeverity,\n okEventColor: 'rgba(0, 245, 153, 0.45)',\n ackEventColor: 'rgba(0, 0, 0, 0)'\n};\n\nconst triggerStatusMap = {\n '0': 'OK',\n '1': 'Problem'\n};\n\nclass TriggerPanelCtrl extends PanelCtrl {\n\n /** @ngInject */\n constructor($scope, $injector, $element, datasourceSrv, templateSrv, contextSrv, dashboardSrv) {\n super($scope, $injector);\n this.datasourceSrv = datasourceSrv;\n this.templateSrv = templateSrv;\n this.contextSrv = contextSrv;\n this.dashboardSrv = dashboardSrv;\n\n this.editorTabIndex = 1;\n this.triggerStatusMap = triggerStatusMap;\n this.defaultTimeFormat = DEFAULT_TIME_FORMAT;\n this.pageIndex = 0;\n this.triggerList = [];\n this.currentTriggersPage = [];\n this.datasources = {};\n\n _.defaults(this.panel, _.cloneDeep(panelDefaults));\n this.panel = migratePanelSchema(this.panel);\n\n this.available_datasources = _.map(this.getZabbixDataSources(), 'name');\n if (this.panel.datasources.length === 0) {\n this.panel.datasources.push(this.available_datasources[0]);\n }\n if (_.isEmpty(this.panel.targets)) {\n this.panel.targets[this.panel.datasources[0]] = DEFAULT_TARGET;\n }\n\n this.initDatasources();\n this.events.on('init-edit-mode', this.onInitEditMode.bind(this));\n this.events.on('refresh', this.onRefresh.bind(this));\n }\n\n initDatasources() {\n let promises = _.map(this.panel.datasources, (ds) => {\n // Load datasource\n return this.datasourceSrv.get(ds)\n .then(datasource => {\n this.datasources[ds] = datasource;\n return datasource;\n });\n });\n return Promise.all(promises);\n }\n\n getZabbixDataSources() {\n return _.filter(this.datasourceSrv.getMetricSources(), datasource => {\n return datasource.meta.id === ZABBIX_DS_ID && datasource.value;\n });\n }\n\n onInitEditMode() {\n this.addEditorTab('Triggers', triggerPanelTriggersTab, 1);\n this.addEditorTab('Options', triggerPanelOptionsTab, 2);\n }\n\n setTimeQueryStart() {\n this.timing.queryStart = new Date().getTime();\n }\n\n setTimeQueryEnd() {\n this.timing.queryEnd = new Date().getTime();\n }\n\n onRefresh() {\n // ignore fetching data if another panel is in fullscreen\n if (this.otherPanelInFullscreenMode()) { return; }\n\n // clear loading/error state\n delete this.error;\n this.loading = true;\n this.setTimeQueryStart();\n\n return this.getTriggers()\n .then(triggerList => {\n // Notify panel that request is finished\n this.loading = false;\n this.setTimeQueryEnd();\n\n // Limit triggers number\n this.triggerList = triggerList.slice(0, this.panel.limit);\n this.getCurrentTriggersPage();\n this.render(this.triggerList);\n })\n .catch(err => {\n // if cancelled keep loading set to true\n if (err.cancelled) {\n console.log('Panel request cancelled', err);\n return;\n }\n\n this.loading = false;\n this.error = err.message || \"Request Error\";\n\n if (err.data) {\n if (err.data.message) {\n this.error = err.data.message;\n }\n if (err.data.error) {\n this.error = err.data.error;\n }\n }\n\n this.events.emit('data-error', err);\n console.log('Panel data error:', err);\n });\n }\n\n getTriggers() {\n let promises = _.map(this.panel.datasources, (ds) => {\n return this.datasourceSrv.get(ds)\n .then(datasource => {\n var zabbix = datasource.zabbix;\n var showEvents = this.panel.showEvents.value;\n var triggerFilter = this.panel.targets[ds];\n var hideHostsInMaintenance = this.panel.hideHostsInMaintenance;\n\n // Replace template variables\n var groupFilter = datasource.replaceTemplateVars(triggerFilter.group.filter);\n var hostFilter = datasource.replaceTemplateVars(triggerFilter.host.filter);\n var appFilter = datasource.replaceTemplateVars(triggerFilter.application.filter);\n\n let triggersOptions = {\n showTriggers: showEvents,\n hideHostsInMaintenance: hideHostsInMaintenance\n };\n\n return zabbix.getTriggers(groupFilter, hostFilter, appFilter, triggersOptions);\n }).then((triggers) => {\n return this.getAcknowledges(triggers, ds);\n }).then((triggers) => {\n return this.filterTriggers(triggers, ds);\n });\n });\n\n return Promise.all(promises)\n .then(results => _.flatten(results))\n .then((triggers) => {\n return this.sortTriggers(triggers);\n })\n .then(triggers => {\n return _.map(triggers, this.formatTrigger.bind(this));\n });\n }\n\n getAcknowledges(triggerList, ds) {\n // Request acknowledges for trigger\n var eventids = _.map(triggerList, trigger => {\n return trigger.lastEvent.eventid;\n });\n\n return this.datasources[ds].zabbix.getAcknowledges(eventids)\n .then(events => {\n\n // Map events to triggers\n _.each(triggerList, trigger => {\n var event = _.find(events, event => {\n return event.eventid === trigger.lastEvent.eventid;\n });\n\n if (event) {\n trigger.acknowledges = _.map(event.acknowledges, ack => {\n let timestamp = moment.unix(ack.clock);\n if (this.panel.customLastChangeFormat) {\n ack.time = timestamp.format(this.panel.lastChangeFormat);\n } else {\n ack.time = timestamp.format(this.defaultTimeFormat);\n }\n ack.user = ack.alias + ' (' + ack.name + ' ' + ack.surname + ')';\n return ack;\n });\n\n // Mark acknowledged triggers with different color\n if (this.panel.markAckEvents && trigger.acknowledges.length) {\n trigger.color = this.panel.ackEventColor;\n }\n }\n });\n\n return triggerList;\n });\n }\n\n filterTriggers(triggerList, ds) {\n // Filter triggers by description\n var triggerFilter = this.panel.targets[ds].trigger.filter;\n triggerFilter = this.datasources[ds].replaceTemplateVars(triggerFilter);\n if (triggerFilter) {\n triggerList = filterTriggers(triggerList, triggerFilter);\n }\n\n // Filter acknowledged triggers\n if (this.panel.showTriggers === 'unacknowledged') {\n triggerList = _.filter(triggerList, trigger => {\n return !trigger.acknowledges;\n });\n } else if (this.panel.showTriggers === 'acknowledged') {\n triggerList = _.filter(triggerList, 'acknowledges');\n } else {\n triggerList = triggerList;\n }\n\n // Filter triggers by severity\n triggerList = _.filter(triggerList, trigger => {\n return this.panel.triggerSeverity[trigger.priority].show;\n });\n\n return triggerList;\n }\n\n sortTriggers(triggerList) {\n if (this.panel.sortTriggersBy.value === 'priority') {\n triggerList = _.sortBy(triggerList, 'priority').reverse();\n } else {\n triggerList = _.sortBy(triggerList, 'lastchangeUnix').reverse();\n }\n return triggerList;\n }\n\n formatTrigger(trigger) {\n let triggerObj = trigger;\n\n // Format last change and age\n trigger.lastchangeUnix = Number(trigger.lastchange);\n let timestamp = moment.unix(trigger.lastchangeUnix);\n if (this.panel.customLastChangeFormat) {\n // User defined format\n triggerObj.lastchange = timestamp.format(this.panel.lastChangeFormat);\n } else {\n triggerObj.lastchange = timestamp.format(this.defaultTimeFormat);\n }\n triggerObj.age = timestamp.fromNow(true);\n\n // Set host that the trigger belongs\n if (trigger.hosts.length) {\n triggerObj.host = trigger.hosts[0].name;\n triggerObj.hostTechName = trigger.hosts[0].host;\n }\n\n // Set color\n if (trigger.value === '1') {\n // Problem state\n triggerObj.color = this.panel.triggerSeverity[trigger.priority].color;\n } else {\n // OK state\n triggerObj.color = this.panel.okEventColor;\n }\n\n triggerObj.severity = this.panel.triggerSeverity[trigger.priority].severity;\n return triggerObj;\n }\n\n switchComment(trigger) {\n trigger.showComment = !trigger.showComment;\n }\n\n acknowledgeTrigger(trigger, message) {\n let eventid = trigger.lastEvent.eventid;\n let grafana_user = this.contextSrv.user.name;\n let ack_message = grafana_user + ' (Grafana): ' + message;\n return this.datasourceSrv.get(this.panel.datasource)\n .then(datasource => {\n let zabbixAPI = datasource.zabbix.zabbixAPI;\n return zabbixAPI.acknowledgeEvent(eventid, ack_message);\n })\n .then(this.onRefresh.bind(this));\n }\n\n getCurrentTriggersPage() {\n let pageSize = this.panel.pageSize || 10;\n let startPos = this.pageIndex * pageSize;\n let endPos = Math.min(startPos + pageSize, this.triggerList.length);\n this.currentTriggersPage = this.triggerList.slice(startPos, endPos);\n return this.currentTriggersPage;\n }\n\n link(scope, elem, attrs, ctrl) {\n var data;\n var panel = ctrl.panel;\n var pageCount = 0;\n data = ctrl.triggerList;\n\n function getTableHeight() {\n var panelHeight = ctrl.height;\n\n if (pageCount > 1) {\n panelHeight -= 26;\n }\n\n return (panelHeight - 31) + 'px';\n }\n\n function switchPage(e) {\n let el = $(e.currentTarget);\n ctrl.pageIndex = (parseInt(el.text(), 10)-1);\n\n let pageSize = ctrl.panel.pageSize || 10;\n let startPos = ctrl.pageIndex * pageSize;\n let endPos = Math.min(startPos + pageSize, ctrl.triggerList.length);\n ctrl.currentTriggersPage = ctrl.triggerList.slice(startPos, endPos);\n\n scope.$apply(() => {\n renderPanel();\n });\n }\n\n function appendPaginationControls(footerElem) {\n footerElem.empty();\n\n var pageSize = ctrl.panel.pageSize || 5;\n pageCount = Math.ceil(data.length / pageSize);\n if (pageCount === 1) {\n return;\n }\n\n var startPage = Math.max(ctrl.pageIndex - 3, 0);\n var endPage = Math.min(pageCount, startPage + 9);\n\n var paginationList = $('');\n\n for (var i = startPage; i < endPage; i++) {\n var activeClass = i === ctrl.pageIndex ? 'active' : '';\n var pageLinkElem = $('
  • ' + (i+1) + '
  • ');\n paginationList.append(pageLinkElem);\n }\n\n footerElem.append(paginationList);\n }\n\n function renderPanel() {\n var panelElem = elem.parents('.panel');\n var rootElem = elem.find('.triggers-panel-scroll');\n var footerElem = elem.find('.triggers-panel-footer');\n\n elem.css({'font-size': panel.fontSize});\n panelElem.addClass('triggers-panel-wrapper');\n appendPaginationControls(footerElem);\n\n rootElem.css({'max-height': panel.scroll ? getTableHeight() : '' });\n ctrl.renderingCompleted();\n }\n\n elem.on('click', '.triggers-panel-page-link', switchPage);\n\n var unbindDestroy = scope.$on('$destroy', function() {\n elem.off('click', '.triggers-panel-page-link');\n unbindDestroy();\n });\n\n ctrl.events.on('render', (renderData) => {\n data = renderData || data;\n if (data) {\n scope.$apply(() => {\n renderPanel();\n });\n }\n });\n }\n}\n\nTriggerPanelCtrl.templateUrl = 'panel-triggers/module.html';\n\nfunction filterTriggers(triggers, triggerFilter) {\n if (utils.isRegex(triggerFilter)) {\n return _.filter(triggers, function(trigger) {\n return utils.buildRegex(triggerFilter).test(trigger.description);\n });\n } else {\n return _.filter(triggers, function(trigger) {\n return trigger.description === triggerFilter;\n });\n }\n}\n\nexport {\n TriggerPanelCtrl,\n TriggerPanelCtrl as PanelCtrl\n};\n"]} \ No newline at end of file +{"version":3,"sources":["../../src/panel-triggers/module.js"],"names":["TriggerPanelCtrl","loadPluginCss","dark","light"],"mappings":";;;;;;;;AAaQA,sB,wBAAAA,gB;;AACAC,mB,kBAAAA,a;;;;AAGRA,oBAAc;AACZC,cAAM,gEADM;AAEZC,eAAO;AAFK,OAAd,E,CAjBA;;;;;;;;;;;;;2BAuBEH,gB","file":"module.js","sourcesContent":["/**\n * Grafana-Zabbix\n * Zabbix plugin for Grafana.\n * http://github.com/alexanderzobnin/grafana-zabbix\n *\n * Trigger panel.\n * This feature sponsored by CORE IT\n * http://www.coreit.fr\n *\n * Copyright 2015 Alexander Zobnin alexanderzobnin@gmail.com\n * Licensed under the Apache License, Version 2.0\n */\n\nimport {TriggerPanelCtrl} from './triggers_panel_ctrl';\nimport {loadPluginCss} from 'app/plugins/sdk';\nimport './ack-tooltip.directive';\n\nloadPluginCss({\n dark: 'plugins/alexanderzobnin-zabbix-app/css/grafana-zabbix.dark.css',\n light: 'plugins/alexanderzobnin-zabbix-app/css/grafana-zabbix.light.css'\n});\n\nexport {\n TriggerPanelCtrl as PanelCtrl\n};\n"]} \ No newline at end of file diff --git a/dist/panel-triggers/specs/migrations.spec.js b/dist/panel-triggers/specs/migrations.spec.js new file mode 100644 index 0000000..f6bba36 --- /dev/null +++ b/dist/panel-triggers/specs/migrations.spec.js @@ -0,0 +1,110 @@ +import {TriggerPanelCtrl} from '../triggers_panel_ctrl'; +import {DEFAULT_TARGET} from '../triggers_panel_ctrl'; +import {DEFAULT_SEVERITY} from '../triggers_panel_ctrl'; + +describe('Triggers Panel schema migration', () => { + let ctx = {}; + let datasourceSrvMock = { + getMetricSources: () => { + return [{ meta: {id: 'alexanderzobnin-zabbix-datasource'}, value: {}, name: 'zabbix_default' }]; + }, + get: () => Promise.resolve({}) + }; + + beforeEach(() => { + ctx = { + scope: { + panel: { + datasource: 'zabbix', + triggers: DEFAULT_TARGET, + hostField: true, + statusField: false, + severityField: false, + lastChangeField: true, + ageField: true, + infoField: true, + limit: 10, + showTriggers: 'all triggers', + hideHostsInMaintenance: false, + sortTriggersBy: { text: 'last change', value: 'lastchange' }, + showEvents: { text: 'Problems', value: '1' }, + triggerSeverity: DEFAULT_SEVERITY, + okEventColor: 'rgba(0, 245, 153, 0.45)', + ackEventColor: 'rgba(0, 0, 0, 0)', + scroll: true, + pageSize: 10, + fontSize: '100%', + } + } + }; + }); + + it('should update old panel schema', (done) => { + let updatedPanelCtrl = new TriggerPanelCtrl(ctx.scope, {}, {}, datasourceSrvMock, {}, {}, {}); + + let expected = { + schemaVersion: 2, + datasources: ['zabbix'], + targets: { + 'zabbix': DEFAULT_TARGET + }, + hostField: true, + statusField: false, + severityField: false, + lastChangeField: true, + ageField: true, + infoField: true, + hideHostsInMaintenance: false, + showTriggers: 'all triggers', + sortTriggersBy: { text: 'last change', value: 'lastchange' }, + showEvents: { text: 'Problems', value: '1' }, + limit: 10, + fontSize: '100%', + pageSize: 10, + scroll: true, + customLastChangeFormat: false, + lastChangeFormat: "", + triggerSeverity: DEFAULT_SEVERITY, + okEventColor: 'rgba(0, 245, 153, 0.45)', + ackEventColor: 'rgba(0, 0, 0, 0)' + }; + + expect(updatedPanelCtrl.panel).toEqual(expected); + done(); + }); + + it('should create new panel with default schema', (done) => { + ctx.scope.panel = {}; + let updatedPanelCtrl = new TriggerPanelCtrl(ctx.scope, {}, {}, datasourceSrvMock, {}, {}, {}); + + let expected = { + schemaVersion: 2, + datasources: ['zabbix_default'], + targets: { + 'zabbix_default': DEFAULT_TARGET + }, + hostField: true, + statusField: false, + severityField: false, + lastChangeField: true, + ageField: true, + infoField: true, + hideHostsInMaintenance: false, + showTriggers: 'all triggers', + sortTriggersBy: { text: 'last change', value: 'lastchange' }, + showEvents: { text: 'Problems', value: '1' }, + limit: 10, + fontSize: '100%', + pageSize: 10, + scroll: true, + customLastChangeFormat: false, + lastChangeFormat: "", + triggerSeverity: DEFAULT_SEVERITY, + okEventColor: 'rgba(0, 245, 153, 0.45)', + ackEventColor: 'rgba(0, 0, 0, 0)' + }; + + expect(updatedPanelCtrl.panel).toEqual(expected); + done(); + }); +}); diff --git a/dist/panel-triggers/triggers_panel_ctrl.js b/dist/panel-triggers/triggers_panel_ctrl.js new file mode 100644 index 0000000..fd6d5db --- /dev/null +++ b/dist/panel-triggers/triggers_panel_ctrl.js @@ -0,0 +1,530 @@ +'use strict'; + +System.register(['lodash', 'jquery', 'moment', '../datasource-zabbix/utils', 'app/plugins/sdk', './options_tab', './triggers_tab', './migrations'], function (_export, _context) { + "use strict"; + + var _, $, moment, utils, PanelCtrl, triggerPanelOptionsTab, triggerPanelTriggersTab, migratePanelSchema, _createClass, ZABBIX_DS_ID, DEFAULT_TARGET, DEFAULT_SEVERITY, DEFAULT_TIME_FORMAT, panelDefaults, triggerStatusMap, TriggerPanelCtrl; + + function _classCallCheck(instance, Constructor) { + if (!(instance instanceof Constructor)) { + throw new TypeError("Cannot call a class as a function"); + } + } + + function _possibleConstructorReturn(self, call) { + if (!self) { + throw new ReferenceError("this hasn't been initialised - super() hasn't been called"); + } + + return call && (typeof call === "object" || typeof call === "function") ? call : self; + } + + function _inherits(subClass, superClass) { + if (typeof superClass !== "function" && superClass !== null) { + throw new TypeError("Super expression must either be null or a function, not " + typeof superClass); + } + + subClass.prototype = Object.create(superClass && superClass.prototype, { + constructor: { + value: subClass, + enumerable: false, + writable: true, + configurable: true + } + }); + if (superClass) Object.setPrototypeOf ? Object.setPrototypeOf(subClass, superClass) : subClass.__proto__ = superClass; + } + + function _filterTriggers(triggers, triggerFilter) { + if (utils.isRegex(triggerFilter)) { + return _.filter(triggers, function (trigger) { + return utils.buildRegex(triggerFilter).test(trigger.description); + }); + } else { + return _.filter(triggers, function (trigger) { + return trigger.description === triggerFilter; + }); + } + } + return { + setters: [function (_lodash) { + _ = _lodash.default; + }, function (_jquery) { + $ = _jquery.default; + }, function (_moment) { + moment = _moment.default; + }, function (_datasourceZabbixUtils) { + utils = _datasourceZabbixUtils; + }, function (_appPluginsSdk) { + PanelCtrl = _appPluginsSdk.PanelCtrl; + }, function (_options_tab) { + triggerPanelOptionsTab = _options_tab.triggerPanelOptionsTab; + }, function (_triggers_tab) { + triggerPanelTriggersTab = _triggers_tab.triggerPanelTriggersTab; + }, function (_migrations) { + migratePanelSchema = _migrations.migratePanelSchema; + }], + execute: function () { + _createClass = function () { + function defineProperties(target, props) { + for (var i = 0; i < props.length; i++) { + var descriptor = props[i]; + descriptor.enumerable = descriptor.enumerable || false; + descriptor.configurable = true; + if ("value" in descriptor) descriptor.writable = true; + Object.defineProperty(target, descriptor.key, descriptor); + } + } + + return function (Constructor, protoProps, staticProps) { + if (protoProps) defineProperties(Constructor.prototype, protoProps); + if (staticProps) defineProperties(Constructor, staticProps); + return Constructor; + }; + }(); + + ZABBIX_DS_ID = 'alexanderzobnin-zabbix-datasource'; + + _export('DEFAULT_TARGET', DEFAULT_TARGET = { + group: { filter: "" }, + host: { filter: "" }, + application: { filter: "" }, + trigger: { filter: "" } + }); + + _export('DEFAULT_TARGET', DEFAULT_TARGET); + + _export('DEFAULT_SEVERITY', DEFAULT_SEVERITY = [{ priority: 0, severity: 'Not classified', color: '#B7DBAB', show: true }, { priority: 1, severity: 'Information', color: '#82B5D8', show: true }, { priority: 2, severity: 'Warning', color: '#E5AC0E', show: true }, { priority: 3, severity: 'Average', color: '#C15C17', show: true }, { priority: 4, severity: 'High', color: '#BF1B00', show: true }, { priority: 5, severity: 'Disaster', color: '#890F02', show: true }]); + + _export('DEFAULT_SEVERITY', DEFAULT_SEVERITY); + + DEFAULT_TIME_FORMAT = "DD MMM YYYY HH:mm:ss"; + panelDefaults = { + schemaVersion: 2, + datasources: [], + targets: {}, + // Fields + hostField: true, + statusField: false, + severityField: false, + lastChangeField: true, + ageField: true, + infoField: true, + // Options + hideHostsInMaintenance: false, + showTriggers: 'all triggers', + sortTriggersBy: { text: 'last change', value: 'lastchange' }, + showEvents: { text: 'Problems', value: '1' }, + limit: 10, + // View options + fontSize: '100%', + pageSize: 10, + scroll: true, + customLastChangeFormat: false, + lastChangeFormat: "", + // Triggers severity and colors + triggerSeverity: DEFAULT_SEVERITY, + okEventColor: 'rgba(0, 245, 153, 0.45)', + ackEventColor: 'rgba(0, 0, 0, 0)' + }; + triggerStatusMap = { + '0': 'OK', + '1': 'Problem' + }; + + _export('TriggerPanelCtrl', TriggerPanelCtrl = function (_PanelCtrl) { + _inherits(TriggerPanelCtrl, _PanelCtrl); + + /** @ngInject */ + function TriggerPanelCtrl($scope, $injector, $element, datasourceSrv, templateSrv, contextSrv, dashboardSrv) { + _classCallCheck(this, TriggerPanelCtrl); + + var _this = _possibleConstructorReturn(this, (TriggerPanelCtrl.__proto__ || Object.getPrototypeOf(TriggerPanelCtrl)).call(this, $scope, $injector)); + + _this.datasourceSrv = datasourceSrv; + _this.templateSrv = templateSrv; + _this.contextSrv = contextSrv; + _this.dashboardSrv = dashboardSrv; + + _this.editorTabIndex = 1; + _this.triggerStatusMap = triggerStatusMap; + _this.defaultTimeFormat = DEFAULT_TIME_FORMAT; + _this.pageIndex = 0; + _this.triggerList = []; + _this.currentTriggersPage = []; + _this.datasources = {}; + + _this.panel = migratePanelSchema(_this.panel); + _.defaults(_this.panel, _.cloneDeep(panelDefaults)); + + _this.available_datasources = _.map(_this.getZabbixDataSources(), 'name'); + if (_this.panel.datasources.length === 0) { + _this.panel.datasources.push(_this.available_datasources[0]); + } + if (_.isEmpty(_this.panel.targets)) { + _this.panel.targets[_this.panel.datasources[0]] = DEFAULT_TARGET; + } + + _this.initDatasources(); + _this.events.on('init-edit-mode', _this.onInitEditMode.bind(_this)); + _this.events.on('refresh', _this.onRefresh.bind(_this)); + return _this; + } + + _createClass(TriggerPanelCtrl, [{ + key: 'initDatasources', + value: function initDatasources() { + var _this2 = this; + + var promises = _.map(this.panel.datasources, function (ds) { + // Load datasource + return _this2.datasourceSrv.get(ds).then(function (datasource) { + _this2.datasources[ds] = datasource; + return datasource; + }); + }); + return Promise.all(promises); + } + }, { + key: 'getZabbixDataSources', + value: function getZabbixDataSources() { + return _.filter(this.datasourceSrv.getMetricSources(), function (datasource) { + return datasource.meta.id === ZABBIX_DS_ID && datasource.value; + }); + } + }, { + key: 'onInitEditMode', + value: function onInitEditMode() { + this.addEditorTab('Triggers', triggerPanelTriggersTab, 1); + this.addEditorTab('Options', triggerPanelOptionsTab, 2); + } + }, { + key: 'setTimeQueryStart', + value: function setTimeQueryStart() { + this.timing.queryStart = new Date().getTime(); + } + }, { + key: 'setTimeQueryEnd', + value: function setTimeQueryEnd() { + this.timing.queryEnd = new Date().getTime(); + } + }, { + key: 'onRefresh', + value: function onRefresh() { + var _this3 = this; + + // ignore fetching data if another panel is in fullscreen + if (this.otherPanelInFullscreenMode()) { + return; + } + + // clear loading/error state + delete this.error; + this.loading = true; + this.setTimeQueryStart(); + + return this.getTriggers().then(function (triggerList) { + // Notify panel that request is finished + _this3.loading = false; + _this3.setTimeQueryEnd(); + + // Limit triggers number + _this3.triggerList = triggerList.slice(0, _this3.panel.limit); + _this3.getCurrentTriggersPage(); + _this3.render(_this3.triggerList); + }).catch(function (err) { + // if cancelled keep loading set to true + if (err.cancelled) { + console.log('Panel request cancelled', err); + return; + } + + _this3.loading = false; + _this3.error = err.message || "Request Error"; + + if (err.data) { + if (err.data.message) { + _this3.error = err.data.message; + } + if (err.data.error) { + _this3.error = err.data.error; + } + } + + _this3.events.emit('data-error', err); + console.log('Panel data error:', err); + }); + } + }, { + key: 'getTriggers', + value: function getTriggers() { + var _this4 = this; + + var promises = _.map(this.panel.datasources, function (ds) { + return _this4.datasourceSrv.get(ds).then(function (datasource) { + var zabbix = datasource.zabbix; + var showEvents = _this4.panel.showEvents.value; + var triggerFilter = _this4.panel.targets[ds]; + var hideHostsInMaintenance = _this4.panel.hideHostsInMaintenance; + + // Replace template variables + var groupFilter = datasource.replaceTemplateVars(triggerFilter.group.filter); + var hostFilter = datasource.replaceTemplateVars(triggerFilter.host.filter); + var appFilter = datasource.replaceTemplateVars(triggerFilter.application.filter); + + var triggersOptions = { + showTriggers: showEvents, + hideHostsInMaintenance: hideHostsInMaintenance + }; + + return zabbix.getTriggers(groupFilter, hostFilter, appFilter, triggersOptions); + }).then(function (triggers) { + return _this4.getAcknowledges(triggers, ds); + }).then(function (triggers) { + return _this4.filterTriggers(triggers, ds); + }); + }); + + return Promise.all(promises).then(function (results) { + return _.flatten(results); + }).then(function (triggers) { + return _this4.sortTriggers(triggers); + }).then(function (triggers) { + return _.map(triggers, _this4.formatTrigger.bind(_this4)); + }); + } + }, { + key: 'getAcknowledges', + value: function getAcknowledges(triggerList, ds) { + var _this5 = this; + + // Request acknowledges for trigger + var eventids = _.map(triggerList, function (trigger) { + return trigger.lastEvent.eventid; + }); + + return this.datasources[ds].zabbix.getAcknowledges(eventids).then(function (events) { + + // Map events to triggers + _.each(triggerList, function (trigger) { + var event = _.find(events, function (event) { + return event.eventid === trigger.lastEvent.eventid; + }); + + if (event) { + trigger.acknowledges = _.map(event.acknowledges, function (ack) { + var timestamp = moment.unix(ack.clock); + if (_this5.panel.customLastChangeFormat) { + ack.time = timestamp.format(_this5.panel.lastChangeFormat); + } else { + ack.time = timestamp.format(_this5.defaultTimeFormat); + } + ack.user = ack.alias + ' (' + ack.name + ' ' + ack.surname + ')'; + return ack; + }); + + // Mark acknowledged triggers with different color + if (_this5.panel.markAckEvents && trigger.acknowledges.length) { + trigger.color = _this5.panel.ackEventColor; + } + } + }); + + return triggerList; + }); + } + }, { + key: 'filterTriggers', + value: function filterTriggers(triggerList, ds) { + var _this6 = this; + + // Filter triggers by description + var triggerFilter = this.panel.targets[ds].trigger.filter; + triggerFilter = this.datasources[ds].replaceTemplateVars(triggerFilter); + if (triggerFilter) { + triggerList = _filterTriggers(triggerList, triggerFilter); + } + + // Filter acknowledged triggers + if (this.panel.showTriggers === 'unacknowledged') { + triggerList = _.filter(triggerList, function (trigger) { + return !trigger.acknowledges; + }); + } else if (this.panel.showTriggers === 'acknowledged') { + triggerList = _.filter(triggerList, 'acknowledges'); + } else { + triggerList = triggerList; + } + + // Filter triggers by severity + triggerList = _.filter(triggerList, function (trigger) { + return _this6.panel.triggerSeverity[trigger.priority].show; + }); + + return triggerList; + } + }, { + key: 'sortTriggers', + value: function sortTriggers(triggerList) { + if (this.panel.sortTriggersBy.value === 'priority') { + triggerList = _.sortBy(triggerList, 'priority').reverse(); + } else { + triggerList = _.sortBy(triggerList, 'lastchangeUnix').reverse(); + } + return triggerList; + } + }, { + key: 'formatTrigger', + value: function formatTrigger(trigger) { + var triggerObj = trigger; + + // Format last change and age + trigger.lastchangeUnix = Number(trigger.lastchange); + var timestamp = moment.unix(trigger.lastchangeUnix); + if (this.panel.customLastChangeFormat) { + // User defined format + triggerObj.lastchange = timestamp.format(this.panel.lastChangeFormat); + } else { + triggerObj.lastchange = timestamp.format(this.defaultTimeFormat); + } + triggerObj.age = timestamp.fromNow(true); + + // Set host that the trigger belongs + if (trigger.hosts.length) { + triggerObj.host = trigger.hosts[0].name; + triggerObj.hostTechName = trigger.hosts[0].host; + } + + // Set color + if (trigger.value === '1') { + // Problem state + triggerObj.color = this.panel.triggerSeverity[trigger.priority].color; + } else { + // OK state + triggerObj.color = this.panel.okEventColor; + } + + triggerObj.severity = this.panel.triggerSeverity[trigger.priority].severity; + return triggerObj; + } + }, { + key: 'switchComment', + value: function switchComment(trigger) { + trigger.showComment = !trigger.showComment; + } + }, { + key: 'acknowledgeTrigger', + value: function acknowledgeTrigger(trigger, message) { + var eventid = trigger.lastEvent.eventid; + var grafana_user = this.contextSrv.user.name; + var ack_message = grafana_user + ' (Grafana): ' + message; + return this.datasourceSrv.get(this.panel.datasource).then(function (datasource) { + var zabbixAPI = datasource.zabbix.zabbixAPI; + return zabbixAPI.acknowledgeEvent(eventid, ack_message); + }).then(this.onRefresh.bind(this)); + } + }, { + key: 'getCurrentTriggersPage', + value: function getCurrentTriggersPage() { + var pageSize = this.panel.pageSize || 10; + var startPos = this.pageIndex * pageSize; + var endPos = Math.min(startPos + pageSize, this.triggerList.length); + this.currentTriggersPage = this.triggerList.slice(startPos, endPos); + return this.currentTriggersPage; + } + }, { + key: 'link', + value: function link(scope, elem, attrs, ctrl) { + var data; + var panel = ctrl.panel; + var pageCount = 0; + data = ctrl.triggerList; + + function getTableHeight() { + var panelHeight = ctrl.height; + + if (pageCount > 1) { + panelHeight -= 26; + } + + return panelHeight - 31 + 'px'; + } + + function switchPage(e) { + var el = $(e.currentTarget); + ctrl.pageIndex = parseInt(el.text(), 10) - 1; + + var pageSize = ctrl.panel.pageSize || 10; + var startPos = ctrl.pageIndex * pageSize; + var endPos = Math.min(startPos + pageSize, ctrl.triggerList.length); + ctrl.currentTriggersPage = ctrl.triggerList.slice(startPos, endPos); + + scope.$apply(function () { + renderPanel(); + }); + } + + function appendPaginationControls(footerElem) { + footerElem.empty(); + + var pageSize = ctrl.panel.pageSize || 5; + pageCount = Math.ceil(data.length / pageSize); + if (pageCount === 1) { + return; + } + + var startPage = Math.max(ctrl.pageIndex - 3, 0); + var endPage = Math.min(pageCount, startPage + 9); + + var paginationList = $(''); + + for (var i = startPage; i < endPage; i++) { + var activeClass = i === ctrl.pageIndex ? 'active' : ''; + var pageLinkElem = $('
  • ' + (i + 1) + '
  • '); + paginationList.append(pageLinkElem); + } + + footerElem.append(paginationList); + } + + function renderPanel() { + var panelElem = elem.parents('.panel'); + var rootElem = elem.find('.triggers-panel-scroll'); + var footerElem = elem.find('.triggers-panel-footer'); + + elem.css({ 'font-size': panel.fontSize }); + panelElem.addClass('triggers-panel-wrapper'); + appendPaginationControls(footerElem); + + rootElem.css({ 'max-height': panel.scroll ? getTableHeight() : '' }); + ctrl.renderingCompleted(); + } + + elem.on('click', '.triggers-panel-page-link', switchPage); + + var unbindDestroy = scope.$on('$destroy', function () { + elem.off('click', '.triggers-panel-page-link'); + unbindDestroy(); + }); + + ctrl.events.on('render', function (renderData) { + data = renderData || data; + if (data) { + scope.$apply(function () { + renderPanel(); + }); + } + }); + } + }]); + + return TriggerPanelCtrl; + }(PanelCtrl)); + + _export('TriggerPanelCtrl', TriggerPanelCtrl); + + TriggerPanelCtrl.templateUrl = 'panel-triggers/module.html'; + } + }; +}); +//# sourceMappingURL=triggers_panel_ctrl.js.map diff --git a/dist/panel-triggers/triggers_panel_ctrl.js.map b/dist/panel-triggers/triggers_panel_ctrl.js.map new file mode 100644 index 0000000..aa30026 --- /dev/null +++ b/dist/panel-triggers/triggers_panel_ctrl.js.map @@ -0,0 +1 @@ +{"version":3,"sources":["../../src/panel-triggers/triggers_panel_ctrl.js"],"names":["filterTriggers","triggers","triggerFilter","utils","isRegex","_","filter","trigger","buildRegex","test","description","$","moment","PanelCtrl","triggerPanelOptionsTab","triggerPanelTriggersTab","migratePanelSchema","ZABBIX_DS_ID","DEFAULT_TARGET","group","host","application","DEFAULT_SEVERITY","priority","severity","color","show","DEFAULT_TIME_FORMAT","panelDefaults","schemaVersion","datasources","targets","hostField","statusField","severityField","lastChangeField","ageField","infoField","hideHostsInMaintenance","showTriggers","sortTriggersBy","text","value","showEvents","limit","fontSize","pageSize","scroll","customLastChangeFormat","lastChangeFormat","triggerSeverity","okEventColor","ackEventColor","triggerStatusMap","TriggerPanelCtrl","$scope","$injector","$element","datasourceSrv","templateSrv","contextSrv","dashboardSrv","editorTabIndex","defaultTimeFormat","pageIndex","triggerList","currentTriggersPage","panel","defaults","cloneDeep","available_datasources","map","getZabbixDataSources","length","push","isEmpty","initDatasources","events","on","onInitEditMode","bind","onRefresh","promises","ds","get","then","datasource","Promise","all","getMetricSources","meta","id","addEditorTab","timing","queryStart","Date","getTime","queryEnd","otherPanelInFullscreenMode","error","loading","setTimeQueryStart","getTriggers","setTimeQueryEnd","slice","getCurrentTriggersPage","render","catch","err","cancelled","console","log","message","data","emit","zabbix","groupFilter","replaceTemplateVars","hostFilter","appFilter","triggersOptions","getAcknowledges","flatten","results","sortTriggers","formatTrigger","eventids","lastEvent","eventid","each","event","find","acknowledges","timestamp","unix","ack","clock","time","format","user","alias","name","surname","markAckEvents","sortBy","reverse","triggerObj","lastchangeUnix","Number","lastchange","age","fromNow","hosts","hostTechName","showComment","grafana_user","ack_message","zabbixAPI","acknowledgeEvent","startPos","endPos","Math","min","scope","elem","attrs","ctrl","pageCount","getTableHeight","panelHeight","height","switchPage","e","el","currentTarget","parseInt","$apply","renderPanel","appendPaginationControls","footerElem","empty","ceil","startPage","max","endPage","paginationList","i","activeClass","pageLinkElem","append","panelElem","parents","rootElem","css","addClass","renderingCompleted","unbindDestroy","$on","off","renderData","templateUrl"],"mappings":"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","file":"triggers_panel_ctrl.js","sourcesContent":["import _ from 'lodash';\nimport $ from 'jquery';\nimport moment from 'moment';\nimport * as utils from '../datasource-zabbix/utils';\nimport {PanelCtrl} from 'app/plugins/sdk';\nimport {triggerPanelOptionsTab} from './options_tab';\nimport {triggerPanelTriggersTab} from './triggers_tab';\nimport {migratePanelSchema} from './migrations';\n\nconst ZABBIX_DS_ID = 'alexanderzobnin-zabbix-datasource';\n\nexport const DEFAULT_TARGET = {\n group: {filter: \"\"},\n host: {filter: \"\"},\n application: {filter: \"\"},\n trigger: {filter: \"\"}\n};\n\nexport const DEFAULT_SEVERITY = [\n { priority: 0, severity: 'Not classified', color: '#B7DBAB', show: true },\n { priority: 1, severity: 'Information', color: '#82B5D8', show: true },\n { priority: 2, severity: 'Warning', color: '#E5AC0E', show: true },\n { priority: 3, severity: 'Average', color: '#C15C17', show: true },\n { priority: 4, severity: 'High', color: '#BF1B00', show: true },\n { priority: 5, severity: 'Disaster', color: '#890F02', show: true }\n];\n\nconst DEFAULT_TIME_FORMAT = \"DD MMM YYYY HH:mm:ss\";\n\nconst panelDefaults = {\n schemaVersion: 2,\n datasources: [],\n targets: {},\n // Fields\n hostField: true,\n statusField: false,\n severityField: false,\n lastChangeField: true,\n ageField: true,\n infoField: true,\n // Options\n hideHostsInMaintenance: false,\n showTriggers: 'all triggers',\n sortTriggersBy: { text: 'last change', value: 'lastchange' },\n showEvents: { text: 'Problems', value: '1' },\n limit: 10,\n // View options\n fontSize: '100%',\n pageSize: 10,\n scroll: true,\n customLastChangeFormat: false,\n lastChangeFormat: \"\",\n // Triggers severity and colors\n triggerSeverity: DEFAULT_SEVERITY,\n okEventColor: 'rgba(0, 245, 153, 0.45)',\n ackEventColor: 'rgba(0, 0, 0, 0)'\n};\n\nconst triggerStatusMap = {\n '0': 'OK',\n '1': 'Problem'\n};\n\nexport class TriggerPanelCtrl extends PanelCtrl {\n\n /** @ngInject */\n constructor($scope, $injector, $element, datasourceSrv, templateSrv, contextSrv, dashboardSrv) {\n super($scope, $injector);\n this.datasourceSrv = datasourceSrv;\n this.templateSrv = templateSrv;\n this.contextSrv = contextSrv;\n this.dashboardSrv = dashboardSrv;\n\n this.editorTabIndex = 1;\n this.triggerStatusMap = triggerStatusMap;\n this.defaultTimeFormat = DEFAULT_TIME_FORMAT;\n this.pageIndex = 0;\n this.triggerList = [];\n this.currentTriggersPage = [];\n this.datasources = {};\n\n this.panel = migratePanelSchema(this.panel);\n _.defaults(this.panel, _.cloneDeep(panelDefaults));\n\n this.available_datasources = _.map(this.getZabbixDataSources(), 'name');\n if (this.panel.datasources.length === 0) {\n this.panel.datasources.push(this.available_datasources[0]);\n }\n if (_.isEmpty(this.panel.targets)) {\n this.panel.targets[this.panel.datasources[0]] = DEFAULT_TARGET;\n }\n\n this.initDatasources();\n this.events.on('init-edit-mode', this.onInitEditMode.bind(this));\n this.events.on('refresh', this.onRefresh.bind(this));\n }\n\n initDatasources() {\n let promises = _.map(this.panel.datasources, (ds) => {\n // Load datasource\n return this.datasourceSrv.get(ds)\n .then(datasource => {\n this.datasources[ds] = datasource;\n return datasource;\n });\n });\n return Promise.all(promises);\n }\n\n getZabbixDataSources() {\n return _.filter(this.datasourceSrv.getMetricSources(), datasource => {\n return datasource.meta.id === ZABBIX_DS_ID && datasource.value;\n });\n }\n\n onInitEditMode() {\n this.addEditorTab('Triggers', triggerPanelTriggersTab, 1);\n this.addEditorTab('Options', triggerPanelOptionsTab, 2);\n }\n\n setTimeQueryStart() {\n this.timing.queryStart = new Date().getTime();\n }\n\n setTimeQueryEnd() {\n this.timing.queryEnd = new Date().getTime();\n }\n\n onRefresh() {\n // ignore fetching data if another panel is in fullscreen\n if (this.otherPanelInFullscreenMode()) { return; }\n\n // clear loading/error state\n delete this.error;\n this.loading = true;\n this.setTimeQueryStart();\n\n return this.getTriggers()\n .then(triggerList => {\n // Notify panel that request is finished\n this.loading = false;\n this.setTimeQueryEnd();\n\n // Limit triggers number\n this.triggerList = triggerList.slice(0, this.panel.limit);\n this.getCurrentTriggersPage();\n this.render(this.triggerList);\n })\n .catch(err => {\n // if cancelled keep loading set to true\n if (err.cancelled) {\n console.log('Panel request cancelled', err);\n return;\n }\n\n this.loading = false;\n this.error = err.message || \"Request Error\";\n\n if (err.data) {\n if (err.data.message) {\n this.error = err.data.message;\n }\n if (err.data.error) {\n this.error = err.data.error;\n }\n }\n\n this.events.emit('data-error', err);\n console.log('Panel data error:', err);\n });\n }\n\n getTriggers() {\n let promises = _.map(this.panel.datasources, (ds) => {\n return this.datasourceSrv.get(ds)\n .then(datasource => {\n var zabbix = datasource.zabbix;\n var showEvents = this.panel.showEvents.value;\n var triggerFilter = this.panel.targets[ds];\n var hideHostsInMaintenance = this.panel.hideHostsInMaintenance;\n\n // Replace template variables\n var groupFilter = datasource.replaceTemplateVars(triggerFilter.group.filter);\n var hostFilter = datasource.replaceTemplateVars(triggerFilter.host.filter);\n var appFilter = datasource.replaceTemplateVars(triggerFilter.application.filter);\n\n let triggersOptions = {\n showTriggers: showEvents,\n hideHostsInMaintenance: hideHostsInMaintenance\n };\n\n return zabbix.getTriggers(groupFilter, hostFilter, appFilter, triggersOptions);\n }).then((triggers) => {\n return this.getAcknowledges(triggers, ds);\n }).then((triggers) => {\n return this.filterTriggers(triggers, ds);\n });\n });\n\n return Promise.all(promises)\n .then(results => _.flatten(results))\n .then((triggers) => {\n return this.sortTriggers(triggers);\n })\n .then(triggers => {\n return _.map(triggers, this.formatTrigger.bind(this));\n });\n }\n\n getAcknowledges(triggerList, ds) {\n // Request acknowledges for trigger\n var eventids = _.map(triggerList, trigger => {\n return trigger.lastEvent.eventid;\n });\n\n return this.datasources[ds].zabbix.getAcknowledges(eventids)\n .then(events => {\n\n // Map events to triggers\n _.each(triggerList, trigger => {\n var event = _.find(events, event => {\n return event.eventid === trigger.lastEvent.eventid;\n });\n\n if (event) {\n trigger.acknowledges = _.map(event.acknowledges, ack => {\n let timestamp = moment.unix(ack.clock);\n if (this.panel.customLastChangeFormat) {\n ack.time = timestamp.format(this.panel.lastChangeFormat);\n } else {\n ack.time = timestamp.format(this.defaultTimeFormat);\n }\n ack.user = ack.alias + ' (' + ack.name + ' ' + ack.surname + ')';\n return ack;\n });\n\n // Mark acknowledged triggers with different color\n if (this.panel.markAckEvents && trigger.acknowledges.length) {\n trigger.color = this.panel.ackEventColor;\n }\n }\n });\n\n return triggerList;\n });\n }\n\n filterTriggers(triggerList, ds) {\n // Filter triggers by description\n var triggerFilter = this.panel.targets[ds].trigger.filter;\n triggerFilter = this.datasources[ds].replaceTemplateVars(triggerFilter);\n if (triggerFilter) {\n triggerList = filterTriggers(triggerList, triggerFilter);\n }\n\n // Filter acknowledged triggers\n if (this.panel.showTriggers === 'unacknowledged') {\n triggerList = _.filter(triggerList, trigger => {\n return !trigger.acknowledges;\n });\n } else if (this.panel.showTriggers === 'acknowledged') {\n triggerList = _.filter(triggerList, 'acknowledges');\n } else {\n triggerList = triggerList;\n }\n\n // Filter triggers by severity\n triggerList = _.filter(triggerList, trigger => {\n return this.panel.triggerSeverity[trigger.priority].show;\n });\n\n return triggerList;\n }\n\n sortTriggers(triggerList) {\n if (this.panel.sortTriggersBy.value === 'priority') {\n triggerList = _.sortBy(triggerList, 'priority').reverse();\n } else {\n triggerList = _.sortBy(triggerList, 'lastchangeUnix').reverse();\n }\n return triggerList;\n }\n\n formatTrigger(trigger) {\n let triggerObj = trigger;\n\n // Format last change and age\n trigger.lastchangeUnix = Number(trigger.lastchange);\n let timestamp = moment.unix(trigger.lastchangeUnix);\n if (this.panel.customLastChangeFormat) {\n // User defined format\n triggerObj.lastchange = timestamp.format(this.panel.lastChangeFormat);\n } else {\n triggerObj.lastchange = timestamp.format(this.defaultTimeFormat);\n }\n triggerObj.age = timestamp.fromNow(true);\n\n // Set host that the trigger belongs\n if (trigger.hosts.length) {\n triggerObj.host = trigger.hosts[0].name;\n triggerObj.hostTechName = trigger.hosts[0].host;\n }\n\n // Set color\n if (trigger.value === '1') {\n // Problem state\n triggerObj.color = this.panel.triggerSeverity[trigger.priority].color;\n } else {\n // OK state\n triggerObj.color = this.panel.okEventColor;\n }\n\n triggerObj.severity = this.panel.triggerSeverity[trigger.priority].severity;\n return triggerObj;\n }\n\n switchComment(trigger) {\n trigger.showComment = !trigger.showComment;\n }\n\n acknowledgeTrigger(trigger, message) {\n let eventid = trigger.lastEvent.eventid;\n let grafana_user = this.contextSrv.user.name;\n let ack_message = grafana_user + ' (Grafana): ' + message;\n return this.datasourceSrv.get(this.panel.datasource)\n .then(datasource => {\n let zabbixAPI = datasource.zabbix.zabbixAPI;\n return zabbixAPI.acknowledgeEvent(eventid, ack_message);\n })\n .then(this.onRefresh.bind(this));\n }\n\n getCurrentTriggersPage() {\n let pageSize = this.panel.pageSize || 10;\n let startPos = this.pageIndex * pageSize;\n let endPos = Math.min(startPos + pageSize, this.triggerList.length);\n this.currentTriggersPage = this.triggerList.slice(startPos, endPos);\n return this.currentTriggersPage;\n }\n\n link(scope, elem, attrs, ctrl) {\n var data;\n var panel = ctrl.panel;\n var pageCount = 0;\n data = ctrl.triggerList;\n\n function getTableHeight() {\n var panelHeight = ctrl.height;\n\n if (pageCount > 1) {\n panelHeight -= 26;\n }\n\n return (panelHeight - 31) + 'px';\n }\n\n function switchPage(e) {\n let el = $(e.currentTarget);\n ctrl.pageIndex = (parseInt(el.text(), 10)-1);\n\n let pageSize = ctrl.panel.pageSize || 10;\n let startPos = ctrl.pageIndex * pageSize;\n let endPos = Math.min(startPos + pageSize, ctrl.triggerList.length);\n ctrl.currentTriggersPage = ctrl.triggerList.slice(startPos, endPos);\n\n scope.$apply(() => {\n renderPanel();\n });\n }\n\n function appendPaginationControls(footerElem) {\n footerElem.empty();\n\n var pageSize = ctrl.panel.pageSize || 5;\n pageCount = Math.ceil(data.length / pageSize);\n if (pageCount === 1) {\n return;\n }\n\n var startPage = Math.max(ctrl.pageIndex - 3, 0);\n var endPage = Math.min(pageCount, startPage + 9);\n\n var paginationList = $('');\n\n for (var i = startPage; i < endPage; i++) {\n var activeClass = i === ctrl.pageIndex ? 'active' : '';\n var pageLinkElem = $('
  • ' + (i+1) + '
  • ');\n paginationList.append(pageLinkElem);\n }\n\n footerElem.append(paginationList);\n }\n\n function renderPanel() {\n var panelElem = elem.parents('.panel');\n var rootElem = elem.find('.triggers-panel-scroll');\n var footerElem = elem.find('.triggers-panel-footer');\n\n elem.css({'font-size': panel.fontSize});\n panelElem.addClass('triggers-panel-wrapper');\n appendPaginationControls(footerElem);\n\n rootElem.css({'max-height': panel.scroll ? getTableHeight() : '' });\n ctrl.renderingCompleted();\n }\n\n elem.on('click', '.triggers-panel-page-link', switchPage);\n\n var unbindDestroy = scope.$on('$destroy', function() {\n elem.off('click', '.triggers-panel-page-link');\n unbindDestroy();\n });\n\n ctrl.events.on('render', (renderData) => {\n data = renderData || data;\n if (data) {\n scope.$apply(() => {\n renderPanel();\n });\n }\n });\n }\n}\n\nTriggerPanelCtrl.templateUrl = 'panel-triggers/module.html';\n\nfunction filterTriggers(triggers, triggerFilter) {\n if (utils.isRegex(triggerFilter)) {\n return _.filter(triggers, function(trigger) {\n return utils.buildRegex(triggerFilter).test(trigger.description);\n });\n } else {\n return _.filter(triggers, function(trigger) {\n return trigger.description === triggerFilter;\n });\n }\n}\n"]} \ No newline at end of file diff --git a/dist/panel-triggers/triggers_tab.js b/dist/panel-triggers/triggers_tab.js index 9112fd5..2a7147f 100644 --- a/dist/panel-triggers/triggers_tab.js +++ b/dist/panel-triggers/triggers_tab.js @@ -1,6 +1,6 @@ 'use strict'; -System.register(['lodash', '../datasource-zabbix/utils', './datasource-selector.directive', '../datasource-zabbix/css/query-editor.css!', './module'], function (_export, _context) { +System.register(['lodash', '../datasource-zabbix/utils', './datasource-selector.directive', '../datasource-zabbix/css/query-editor.css!', './triggers_panel_ctrl'], function (_export, _context) { "use strict"; var _, utils, DEFAULT_TARGET, _createClass, TriggersTabCtrl; @@ -27,8 +27,8 @@ System.register(['lodash', '../datasource-zabbix/utils', './datasource-selector. _ = _lodash.default; }, function (_datasourceZabbixUtils) { utils = _datasourceZabbixUtils; - }, function (_datasourceSelectorDirective) {}, function (_datasourceZabbixCssQueryEditorCss) {}, function (_module) { - DEFAULT_TARGET = _module.DEFAULT_TARGET; + }, function (_datasourceSelectorDirective) {}, function (_datasourceZabbixCssQueryEditorCss) {}, function (_triggers_panel_ctrl) { + DEFAULT_TARGET = _triggers_panel_ctrl.DEFAULT_TARGET; }], execute: function () { _createClass = function () { diff --git a/dist/panel-triggers/triggers_tab.js.map b/dist/panel-triggers/triggers_tab.js.map index 04f59e1..f6f3e98 100644 --- a/dist/panel-triggers/triggers_tab.js.map +++ b/dist/panel-triggers/triggers_tab.js.map @@ -1 +1 @@ -{"version":3,"sources":["../../src/panel-triggers/triggers_tab.js"],"names":["triggerPanelTriggersTab","restrict","scope","templateUrl","controller","TriggersTabCtrl","_","utils","DEFAULT_TARGET","$scope","$rootScope","uiSegmentSrv","templateSrv","editor","panelCtrl","ctrl","panel","datasources","scopeDefaults","getGroupNames","getHostNames","getApplicationNames","oldTarget","cloneDeep","targets","defaultsDeep","initDatasources","refresh","then","each","datasource","bindSuggestionFunctions","ds","name","bind","suggestGroups","suggestHosts","suggestApps","query","callback","zabbix","getAllGroups","map","groups","groupFilter","replaceTemplateVars","group","filter","getAllHosts","hosts","hostFilter","host","getAllApps","apps","parseTarget","newTarget","isEqual","str","isRegex","isTemplateVariable","variables"],"mappings":"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","file":"triggers_tab.js","sourcesContent":["import _ from 'lodash';\nimport * as utils from '../datasource-zabbix/utils';\nimport './datasource-selector.directive';\nimport '../datasource-zabbix/css/query-editor.css!';\nimport {DEFAULT_TARGET} from './module';\n\nclass TriggersTabCtrl {\n\n /** @ngInject */\n constructor($scope, $rootScope, uiSegmentSrv, templateSrv) {\n $scope.editor = this;\n this.panelCtrl = $scope.ctrl;\n this.panel = this.panelCtrl.panel;\n this.templateSrv = templateSrv;\n this.datasources = this.panelCtrl.datasources;\n\n // Load scope defaults\n var scopeDefaults = {\n getGroupNames: {},\n getHostNames: {},\n getApplicationNames: {},\n oldTarget: _.cloneDeep(this.panel.targets)\n };\n _.defaultsDeep(this, scopeDefaults);\n\n this.initDatasources();\n this.panelCtrl.refresh();\n }\n\n initDatasources() {\n return this.panelCtrl.initDatasources()\n .then((datasources) => {\n _.each(datasources, (datasource) => {\n this.bindSuggestionFunctions(datasource);\n });\n });\n }\n\n bindSuggestionFunctions(datasource) {\n // Map functions for bs-typeahead\n let ds = datasource.name;\n this.getGroupNames[ds] = _.bind(this.suggestGroups, this, datasource);\n this.getHostNames[ds] = _.bind(this.suggestHosts, this, datasource);\n this.getApplicationNames[ds] = _.bind(this.suggestApps, this, datasource);\n }\n\n suggestGroups(datasource, query, callback) {\n return datasource.zabbix.getAllGroups()\n .then(groups => {\n return _.map(groups, 'name');\n })\n .then(callback);\n }\n\n suggestHosts(datasource, query, callback) {\n let groupFilter = datasource.replaceTemplateVars(this.panel.targets[datasource.name].group.filter);\n return datasource.zabbix.getAllHosts(groupFilter)\n .then(hosts => {\n return _.map(hosts, 'name');\n })\n .then(callback);\n }\n\n suggestApps(datasource, query, callback) {\n let groupFilter = datasource.replaceTemplateVars(this.panel.targets[datasource.name].group.filter);\n let hostFilter = datasource.replaceTemplateVars(this.panel.targets[datasource.name].host.filter);\n return datasource.zabbix.getAllApps(groupFilter, hostFilter)\n .then(apps => {\n return _.map(apps, 'name');\n })\n .then(callback);\n }\n\n datasourcesChanged() {\n _.each(this.panel.datasources, (ds) => {\n if (!this.panel.targets[ds]) {\n this.panel.targets[ds] = DEFAULT_TARGET;\n }\n });\n this.parseTarget();\n }\n\n parseTarget() {\n this.initDatasources()\n .then(() => {\n var newTarget = _.cloneDeep(this.panel.targets);\n if (!_.isEqual(this.oldTarget, newTarget)) {\n this.oldTarget = newTarget;\n }\n this.panelCtrl.refresh();\n });\n }\n\n isRegex(str) {\n return utils.isRegex(str);\n }\n\n isVariable(str) {\n return utils.isTemplateVariable(str, this.templateSrv.variables);\n }\n}\n\nexport function triggerPanelTriggersTab() {\n return {\n restrict: 'E',\n scope: true,\n templateUrl: 'public/plugins/alexanderzobnin-zabbix-app/panel-triggers/partials/triggers_tab.html',\n controller: TriggersTabCtrl,\n };\n}\n"]} \ No newline at end of file +{"version":3,"sources":["../../src/panel-triggers/triggers_tab.js"],"names":["triggerPanelTriggersTab","restrict","scope","templateUrl","controller","TriggersTabCtrl","_","utils","DEFAULT_TARGET","$scope","$rootScope","uiSegmentSrv","templateSrv","editor","panelCtrl","ctrl","panel","datasources","scopeDefaults","getGroupNames","getHostNames","getApplicationNames","oldTarget","cloneDeep","targets","defaultsDeep","initDatasources","refresh","then","each","datasource","bindSuggestionFunctions","ds","name","bind","suggestGroups","suggestHosts","suggestApps","query","callback","zabbix","getAllGroups","map","groups","groupFilter","replaceTemplateVars","group","filter","getAllHosts","hosts","hostFilter","host","getAllApps","apps","parseTarget","newTarget","isEqual","str","isRegex","isTemplateVariable","variables"],"mappings":"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","file":"triggers_tab.js","sourcesContent":["import _ from 'lodash';\nimport * as utils from '../datasource-zabbix/utils';\nimport './datasource-selector.directive';\nimport '../datasource-zabbix/css/query-editor.css!';\nimport {DEFAULT_TARGET} from './triggers_panel_ctrl';\n\nclass TriggersTabCtrl {\n\n /** @ngInject */\n constructor($scope, $rootScope, uiSegmentSrv, templateSrv) {\n $scope.editor = this;\n this.panelCtrl = $scope.ctrl;\n this.panel = this.panelCtrl.panel;\n this.templateSrv = templateSrv;\n this.datasources = this.panelCtrl.datasources;\n\n // Load scope defaults\n var scopeDefaults = {\n getGroupNames: {},\n getHostNames: {},\n getApplicationNames: {},\n oldTarget: _.cloneDeep(this.panel.targets)\n };\n _.defaultsDeep(this, scopeDefaults);\n\n this.initDatasources();\n this.panelCtrl.refresh();\n }\n\n initDatasources() {\n return this.panelCtrl.initDatasources()\n .then((datasources) => {\n _.each(datasources, (datasource) => {\n this.bindSuggestionFunctions(datasource);\n });\n });\n }\n\n bindSuggestionFunctions(datasource) {\n // Map functions for bs-typeahead\n let ds = datasource.name;\n this.getGroupNames[ds] = _.bind(this.suggestGroups, this, datasource);\n this.getHostNames[ds] = _.bind(this.suggestHosts, this, datasource);\n this.getApplicationNames[ds] = _.bind(this.suggestApps, this, datasource);\n }\n\n suggestGroups(datasource, query, callback) {\n return datasource.zabbix.getAllGroups()\n .then(groups => {\n return _.map(groups, 'name');\n })\n .then(callback);\n }\n\n suggestHosts(datasource, query, callback) {\n let groupFilter = datasource.replaceTemplateVars(this.panel.targets[datasource.name].group.filter);\n return datasource.zabbix.getAllHosts(groupFilter)\n .then(hosts => {\n return _.map(hosts, 'name');\n })\n .then(callback);\n }\n\n suggestApps(datasource, query, callback) {\n let groupFilter = datasource.replaceTemplateVars(this.panel.targets[datasource.name].group.filter);\n let hostFilter = datasource.replaceTemplateVars(this.panel.targets[datasource.name].host.filter);\n return datasource.zabbix.getAllApps(groupFilter, hostFilter)\n .then(apps => {\n return _.map(apps, 'name');\n })\n .then(callback);\n }\n\n datasourcesChanged() {\n _.each(this.panel.datasources, (ds) => {\n if (!this.panel.targets[ds]) {\n this.panel.targets[ds] = DEFAULT_TARGET;\n }\n });\n this.parseTarget();\n }\n\n parseTarget() {\n this.initDatasources()\n .then(() => {\n var newTarget = _.cloneDeep(this.panel.targets);\n if (!_.isEqual(this.oldTarget, newTarget)) {\n this.oldTarget = newTarget;\n }\n this.panelCtrl.refresh();\n });\n }\n\n isRegex(str) {\n return utils.isRegex(str);\n }\n\n isVariable(str) {\n return utils.isTemplateVariable(str, this.templateSrv.variables);\n }\n}\n\nexport function triggerPanelTriggersTab() {\n return {\n restrict: 'E',\n scope: true,\n templateUrl: 'public/plugins/alexanderzobnin-zabbix-app/panel-triggers/partials/triggers_tab.html',\n controller: TriggersTabCtrl,\n };\n}\n"]} \ No newline at end of file diff --git a/package.json b/package.json index 143129c..5c0a129 100644 --- a/package.json +++ b/package.json @@ -5,6 +5,7 @@ "description": "Zabbix plugin for Grafana", "scripts": { "test": "jest", + "jest": "jest --notify --watch", "build": "grunt && jest", "watch": "grunt watch", "codecov": "grunt && jest --coverage && codecov", @@ -50,7 +51,8 @@ "tether-drop": "^1.4.2" }, "dependencies": { - "lodash": "~4.0.0" + "lodash": "~4.0.0", + "systemjs": "^0.20.19" }, "homepage": "http://grafana-zabbix.org" } diff --git a/src/panel-triggers/migrations.js b/src/panel-triggers/migrations.js index 163d79b..e932731 100644 --- a/src/panel-triggers/migrations.js +++ b/src/panel-triggers/migrations.js @@ -1,9 +1,18 @@ export function migratePanelSchema(panel) { + if (isEmptyPanel(panel)) { + return panel; + } + const schemaVersion = getSchemaVersion(panel); switch (schemaVersion) { case 1: panel.datasources = [panel.datasource]; + panel.targets = {}; panel.targets[panel.datasources[0]] = panel.triggers; + + // delete old props + delete panel.triggers; + delete panel.datasource; break; } @@ -13,3 +22,7 @@ export function migratePanelSchema(panel) { function getSchemaVersion(panel) { return panel.schemaVersion || 1; } + +function isEmptyPanel(panel) { + return !panel.datasource && !panel.datasources && !panel.triggers && !panel.targets; +} diff --git a/src/panel-triggers/module.js b/src/panel-triggers/module.js index 10fbd94..329bd74 100644 --- a/src/panel-triggers/module.js +++ b/src/panel-triggers/module.js @@ -11,15 +11,8 @@ * Licensed under the Apache License, Version 2.0 */ -import _ from 'lodash'; -import $ from 'jquery'; -import moment from 'moment'; +import {TriggerPanelCtrl} from './triggers_panel_ctrl'; import {loadPluginCss} from 'app/plugins/sdk'; -import * as utils from '../datasource-zabbix/utils'; -import {PanelCtrl} from 'app/plugins/sdk'; -import {triggerPanelOptionsTab} from './options_tab'; -import {triggerPanelTriggersTab} from './triggers_tab'; -import {migratePanelSchema} from './migrations'; import './ack-tooltip.directive'; loadPluginCss({ @@ -27,436 +20,6 @@ loadPluginCss({ light: 'plugins/alexanderzobnin-zabbix-app/css/grafana-zabbix.light.css' }); -const ZABBIX_DS_ID = 'alexanderzobnin-zabbix-datasource'; - -export const DEFAULT_TARGET = { - group: {filter: ""}, - host: {filter: ""}, - application: {filter: ""}, - trigger: {filter: ""} -}; - -const defaultSeverity = [ - { priority: 0, severity: 'Not classified', color: '#B7DBAB', show: true }, - { priority: 1, severity: 'Information', color: '#82B5D8', show: true }, - { priority: 2, severity: 'Warning', color: '#E5AC0E', show: true }, - { priority: 3, severity: 'Average', color: '#C15C17', show: true }, - { priority: 4, severity: 'High', color: '#BF1B00', show: true }, - { priority: 5, severity: 'Disaster', color: '#890F02', show: true } -]; - -const DEFAULT_TIME_FORMAT = "DD MMM YYYY HH:mm:ss"; - -const panelDefaults = { - schemaVersion: 2, - datasources: [], - targets: {}, - // Fields - hostField: true, - statusField: false, - severityField: false, - lastChangeField: true, - ageField: true, - infoField: true, - // Options - hideHostsInMaintenance: false, - showTriggers: 'all triggers', - sortTriggersBy: { text: 'last change', value: 'lastchange' }, - showEvents: { text: 'Problems', value: '1' }, - limit: 10, - // View options - fontSize: '100%', - pageSize: 10, - scroll: true, - customLastChangeFormat: false, - lastChangeFormat: "", - // Triggers severity and colors - triggerSeverity: defaultSeverity, - okEventColor: 'rgba(0, 245, 153, 0.45)', - ackEventColor: 'rgba(0, 0, 0, 0)' -}; - -const triggerStatusMap = { - '0': 'OK', - '1': 'Problem' -}; - -class TriggerPanelCtrl extends PanelCtrl { - - /** @ngInject */ - constructor($scope, $injector, $element, datasourceSrv, templateSrv, contextSrv, dashboardSrv) { - super($scope, $injector); - this.datasourceSrv = datasourceSrv; - this.templateSrv = templateSrv; - this.contextSrv = contextSrv; - this.dashboardSrv = dashboardSrv; - - this.editorTabIndex = 1; - this.triggerStatusMap = triggerStatusMap; - this.defaultTimeFormat = DEFAULT_TIME_FORMAT; - this.pageIndex = 0; - this.triggerList = []; - this.currentTriggersPage = []; - this.datasources = {}; - - _.defaults(this.panel, _.cloneDeep(panelDefaults)); - this.panel = migratePanelSchema(this.panel); - - this.available_datasources = _.map(this.getZabbixDataSources(), 'name'); - if (this.panel.datasources.length === 0) { - this.panel.datasources.push(this.available_datasources[0]); - } - if (_.isEmpty(this.panel.targets)) { - this.panel.targets[this.panel.datasources[0]] = DEFAULT_TARGET; - } - - this.initDatasources(); - this.events.on('init-edit-mode', this.onInitEditMode.bind(this)); - this.events.on('refresh', this.onRefresh.bind(this)); - } - - initDatasources() { - let promises = _.map(this.panel.datasources, (ds) => { - // Load datasource - return this.datasourceSrv.get(ds) - .then(datasource => { - this.datasources[ds] = datasource; - return datasource; - }); - }); - return Promise.all(promises); - } - - getZabbixDataSources() { - return _.filter(this.datasourceSrv.getMetricSources(), datasource => { - return datasource.meta.id === ZABBIX_DS_ID && datasource.value; - }); - } - - onInitEditMode() { - this.addEditorTab('Triggers', triggerPanelTriggersTab, 1); - this.addEditorTab('Options', triggerPanelOptionsTab, 2); - } - - setTimeQueryStart() { - this.timing.queryStart = new Date().getTime(); - } - - setTimeQueryEnd() { - this.timing.queryEnd = new Date().getTime(); - } - - onRefresh() { - // ignore fetching data if another panel is in fullscreen - if (this.otherPanelInFullscreenMode()) { return; } - - // clear loading/error state - delete this.error; - this.loading = true; - this.setTimeQueryStart(); - - return this.getTriggers() - .then(triggerList => { - // Notify panel that request is finished - this.loading = false; - this.setTimeQueryEnd(); - - // Limit triggers number - this.triggerList = triggerList.slice(0, this.panel.limit); - this.getCurrentTriggersPage(); - this.render(this.triggerList); - }) - .catch(err => { - // if cancelled keep loading set to true - if (err.cancelled) { - console.log('Panel request cancelled', err); - return; - } - - this.loading = false; - this.error = err.message || "Request Error"; - - if (err.data) { - if (err.data.message) { - this.error = err.data.message; - } - if (err.data.error) { - this.error = err.data.error; - } - } - - this.events.emit('data-error', err); - console.log('Panel data error:', err); - }); - } - - getTriggers() { - let promises = _.map(this.panel.datasources, (ds) => { - return this.datasourceSrv.get(ds) - .then(datasource => { - var zabbix = datasource.zabbix; - var showEvents = this.panel.showEvents.value; - var triggerFilter = this.panel.targets[ds]; - var hideHostsInMaintenance = this.panel.hideHostsInMaintenance; - - // Replace template variables - var groupFilter = datasource.replaceTemplateVars(triggerFilter.group.filter); - var hostFilter = datasource.replaceTemplateVars(triggerFilter.host.filter); - var appFilter = datasource.replaceTemplateVars(triggerFilter.application.filter); - - let triggersOptions = { - showTriggers: showEvents, - hideHostsInMaintenance: hideHostsInMaintenance - }; - - return zabbix.getTriggers(groupFilter, hostFilter, appFilter, triggersOptions); - }).then((triggers) => { - return this.getAcknowledges(triggers, ds); - }).then((triggers) => { - return this.filterTriggers(triggers, ds); - }); - }); - - return Promise.all(promises) - .then(results => _.flatten(results)) - .then((triggers) => { - return this.sortTriggers(triggers); - }) - .then(triggers => { - return _.map(triggers, this.formatTrigger.bind(this)); - }); - } - - getAcknowledges(triggerList, ds) { - // Request acknowledges for trigger - var eventids = _.map(triggerList, trigger => { - return trigger.lastEvent.eventid; - }); - - return this.datasources[ds].zabbix.getAcknowledges(eventids) - .then(events => { - - // Map events to triggers - _.each(triggerList, trigger => { - var event = _.find(events, event => { - return event.eventid === trigger.lastEvent.eventid; - }); - - if (event) { - trigger.acknowledges = _.map(event.acknowledges, ack => { - let timestamp = moment.unix(ack.clock); - if (this.panel.customLastChangeFormat) { - ack.time = timestamp.format(this.panel.lastChangeFormat); - } else { - ack.time = timestamp.format(this.defaultTimeFormat); - } - ack.user = ack.alias + ' (' + ack.name + ' ' + ack.surname + ')'; - return ack; - }); - - // Mark acknowledged triggers with different color - if (this.panel.markAckEvents && trigger.acknowledges.length) { - trigger.color = this.panel.ackEventColor; - } - } - }); - - return triggerList; - }); - } - - filterTriggers(triggerList, ds) { - // Filter triggers by description - var triggerFilter = this.panel.targets[ds].trigger.filter; - triggerFilter = this.datasources[ds].replaceTemplateVars(triggerFilter); - if (triggerFilter) { - triggerList = filterTriggers(triggerList, triggerFilter); - } - - // Filter acknowledged triggers - if (this.panel.showTriggers === 'unacknowledged') { - triggerList = _.filter(triggerList, trigger => { - return !trigger.acknowledges; - }); - } else if (this.panel.showTriggers === 'acknowledged') { - triggerList = _.filter(triggerList, 'acknowledges'); - } else { - triggerList = triggerList; - } - - // Filter triggers by severity - triggerList = _.filter(triggerList, trigger => { - return this.panel.triggerSeverity[trigger.priority].show; - }); - - return triggerList; - } - - sortTriggers(triggerList) { - if (this.panel.sortTriggersBy.value === 'priority') { - triggerList = _.sortBy(triggerList, 'priority').reverse(); - } else { - triggerList = _.sortBy(triggerList, 'lastchangeUnix').reverse(); - } - return triggerList; - } - - formatTrigger(trigger) { - let triggerObj = trigger; - - // Format last change and age - trigger.lastchangeUnix = Number(trigger.lastchange); - let timestamp = moment.unix(trigger.lastchangeUnix); - if (this.panel.customLastChangeFormat) { - // User defined format - triggerObj.lastchange = timestamp.format(this.panel.lastChangeFormat); - } else { - triggerObj.lastchange = timestamp.format(this.defaultTimeFormat); - } - triggerObj.age = timestamp.fromNow(true); - - // Set host that the trigger belongs - if (trigger.hosts.length) { - triggerObj.host = trigger.hosts[0].name; - triggerObj.hostTechName = trigger.hosts[0].host; - } - - // Set color - if (trigger.value === '1') { - // Problem state - triggerObj.color = this.panel.triggerSeverity[trigger.priority].color; - } else { - // OK state - triggerObj.color = this.panel.okEventColor; - } - - triggerObj.severity = this.panel.triggerSeverity[trigger.priority].severity; - return triggerObj; - } - - switchComment(trigger) { - trigger.showComment = !trigger.showComment; - } - - acknowledgeTrigger(trigger, message) { - let eventid = trigger.lastEvent.eventid; - let grafana_user = this.contextSrv.user.name; - let ack_message = grafana_user + ' (Grafana): ' + message; - return this.datasourceSrv.get(this.panel.datasource) - .then(datasource => { - let zabbixAPI = datasource.zabbix.zabbixAPI; - return zabbixAPI.acknowledgeEvent(eventid, ack_message); - }) - .then(this.onRefresh.bind(this)); - } - - getCurrentTriggersPage() { - let pageSize = this.panel.pageSize || 10; - let startPos = this.pageIndex * pageSize; - let endPos = Math.min(startPos + pageSize, this.triggerList.length); - this.currentTriggersPage = this.triggerList.slice(startPos, endPos); - return this.currentTriggersPage; - } - - link(scope, elem, attrs, ctrl) { - var data; - var panel = ctrl.panel; - var pageCount = 0; - data = ctrl.triggerList; - - function getTableHeight() { - var panelHeight = ctrl.height; - - if (pageCount > 1) { - panelHeight -= 26; - } - - return (panelHeight - 31) + 'px'; - } - - function switchPage(e) { - let el = $(e.currentTarget); - ctrl.pageIndex = (parseInt(el.text(), 10)-1); - - let pageSize = ctrl.panel.pageSize || 10; - let startPos = ctrl.pageIndex * pageSize; - let endPos = Math.min(startPos + pageSize, ctrl.triggerList.length); - ctrl.currentTriggersPage = ctrl.triggerList.slice(startPos, endPos); - - scope.$apply(() => { - renderPanel(); - }); - } - - function appendPaginationControls(footerElem) { - footerElem.empty(); - - var pageSize = ctrl.panel.pageSize || 5; - pageCount = Math.ceil(data.length / pageSize); - if (pageCount === 1) { - return; - } - - var startPage = Math.max(ctrl.pageIndex - 3, 0); - var endPage = Math.min(pageCount, startPage + 9); - - var paginationList = $(''); - - for (var i = startPage; i < endPage; i++) { - var activeClass = i === ctrl.pageIndex ? 'active' : ''; - var pageLinkElem = $('
  • ' + (i+1) + '
  • '); - paginationList.append(pageLinkElem); - } - - footerElem.append(paginationList); - } - - function renderPanel() { - var panelElem = elem.parents('.panel'); - var rootElem = elem.find('.triggers-panel-scroll'); - var footerElem = elem.find('.triggers-panel-footer'); - - elem.css({'font-size': panel.fontSize}); - panelElem.addClass('triggers-panel-wrapper'); - appendPaginationControls(footerElem); - - rootElem.css({'max-height': panel.scroll ? getTableHeight() : '' }); - ctrl.renderingCompleted(); - } - - elem.on('click', '.triggers-panel-page-link', switchPage); - - var unbindDestroy = scope.$on('$destroy', function() { - elem.off('click', '.triggers-panel-page-link'); - unbindDestroy(); - }); - - ctrl.events.on('render', (renderData) => { - data = renderData || data; - if (data) { - scope.$apply(() => { - renderPanel(); - }); - } - }); - } -} - -TriggerPanelCtrl.templateUrl = 'panel-triggers/module.html'; - -function filterTriggers(triggers, triggerFilter) { - if (utils.isRegex(triggerFilter)) { - return _.filter(triggers, function(trigger) { - return utils.buildRegex(triggerFilter).test(trigger.description); - }); - } else { - return _.filter(triggers, function(trigger) { - return trigger.description === triggerFilter; - }); - } -} - export { - TriggerPanelCtrl, TriggerPanelCtrl as PanelCtrl }; diff --git a/src/panel-triggers/specs/migrations.spec.js b/src/panel-triggers/specs/migrations.spec.js new file mode 100644 index 0000000..f6bba36 --- /dev/null +++ b/src/panel-triggers/specs/migrations.spec.js @@ -0,0 +1,110 @@ +import {TriggerPanelCtrl} from '../triggers_panel_ctrl'; +import {DEFAULT_TARGET} from '../triggers_panel_ctrl'; +import {DEFAULT_SEVERITY} from '../triggers_panel_ctrl'; + +describe('Triggers Panel schema migration', () => { + let ctx = {}; + let datasourceSrvMock = { + getMetricSources: () => { + return [{ meta: {id: 'alexanderzobnin-zabbix-datasource'}, value: {}, name: 'zabbix_default' }]; + }, + get: () => Promise.resolve({}) + }; + + beforeEach(() => { + ctx = { + scope: { + panel: { + datasource: 'zabbix', + triggers: DEFAULT_TARGET, + hostField: true, + statusField: false, + severityField: false, + lastChangeField: true, + ageField: true, + infoField: true, + limit: 10, + showTriggers: 'all triggers', + hideHostsInMaintenance: false, + sortTriggersBy: { text: 'last change', value: 'lastchange' }, + showEvents: { text: 'Problems', value: '1' }, + triggerSeverity: DEFAULT_SEVERITY, + okEventColor: 'rgba(0, 245, 153, 0.45)', + ackEventColor: 'rgba(0, 0, 0, 0)', + scroll: true, + pageSize: 10, + fontSize: '100%', + } + } + }; + }); + + it('should update old panel schema', (done) => { + let updatedPanelCtrl = new TriggerPanelCtrl(ctx.scope, {}, {}, datasourceSrvMock, {}, {}, {}); + + let expected = { + schemaVersion: 2, + datasources: ['zabbix'], + targets: { + 'zabbix': DEFAULT_TARGET + }, + hostField: true, + statusField: false, + severityField: false, + lastChangeField: true, + ageField: true, + infoField: true, + hideHostsInMaintenance: false, + showTriggers: 'all triggers', + sortTriggersBy: { text: 'last change', value: 'lastchange' }, + showEvents: { text: 'Problems', value: '1' }, + limit: 10, + fontSize: '100%', + pageSize: 10, + scroll: true, + customLastChangeFormat: false, + lastChangeFormat: "", + triggerSeverity: DEFAULT_SEVERITY, + okEventColor: 'rgba(0, 245, 153, 0.45)', + ackEventColor: 'rgba(0, 0, 0, 0)' + }; + + expect(updatedPanelCtrl.panel).toEqual(expected); + done(); + }); + + it('should create new panel with default schema', (done) => { + ctx.scope.panel = {}; + let updatedPanelCtrl = new TriggerPanelCtrl(ctx.scope, {}, {}, datasourceSrvMock, {}, {}, {}); + + let expected = { + schemaVersion: 2, + datasources: ['zabbix_default'], + targets: { + 'zabbix_default': DEFAULT_TARGET + }, + hostField: true, + statusField: false, + severityField: false, + lastChangeField: true, + ageField: true, + infoField: true, + hideHostsInMaintenance: false, + showTriggers: 'all triggers', + sortTriggersBy: { text: 'last change', value: 'lastchange' }, + showEvents: { text: 'Problems', value: '1' }, + limit: 10, + fontSize: '100%', + pageSize: 10, + scroll: true, + customLastChangeFormat: false, + lastChangeFormat: "", + triggerSeverity: DEFAULT_SEVERITY, + okEventColor: 'rgba(0, 245, 153, 0.45)', + ackEventColor: 'rgba(0, 0, 0, 0)' + }; + + expect(updatedPanelCtrl.panel).toEqual(expected); + done(); + }); +}); diff --git a/src/panel-triggers/triggers_panel_ctrl.js b/src/panel-triggers/triggers_panel_ctrl.js new file mode 100644 index 0000000..6fbfe9d --- /dev/null +++ b/src/panel-triggers/triggers_panel_ctrl.js @@ -0,0 +1,437 @@ +import _ from 'lodash'; +import $ from 'jquery'; +import moment from 'moment'; +import * as utils from '../datasource-zabbix/utils'; +import {PanelCtrl} from 'app/plugins/sdk'; +import {triggerPanelOptionsTab} from './options_tab'; +import {triggerPanelTriggersTab} from './triggers_tab'; +import {migratePanelSchema} from './migrations'; + +const ZABBIX_DS_ID = 'alexanderzobnin-zabbix-datasource'; + +export const DEFAULT_TARGET = { + group: {filter: ""}, + host: {filter: ""}, + application: {filter: ""}, + trigger: {filter: ""} +}; + +export const DEFAULT_SEVERITY = [ + { priority: 0, severity: 'Not classified', color: '#B7DBAB', show: true }, + { priority: 1, severity: 'Information', color: '#82B5D8', show: true }, + { priority: 2, severity: 'Warning', color: '#E5AC0E', show: true }, + { priority: 3, severity: 'Average', color: '#C15C17', show: true }, + { priority: 4, severity: 'High', color: '#BF1B00', show: true }, + { priority: 5, severity: 'Disaster', color: '#890F02', show: true } +]; + +const DEFAULT_TIME_FORMAT = "DD MMM YYYY HH:mm:ss"; + +const panelDefaults = { + schemaVersion: 2, + datasources: [], + targets: {}, + // Fields + hostField: true, + statusField: false, + severityField: false, + lastChangeField: true, + ageField: true, + infoField: true, + // Options + hideHostsInMaintenance: false, + showTriggers: 'all triggers', + sortTriggersBy: { text: 'last change', value: 'lastchange' }, + showEvents: { text: 'Problems', value: '1' }, + limit: 10, + // View options + fontSize: '100%', + pageSize: 10, + scroll: true, + customLastChangeFormat: false, + lastChangeFormat: "", + // Triggers severity and colors + triggerSeverity: DEFAULT_SEVERITY, + okEventColor: 'rgba(0, 245, 153, 0.45)', + ackEventColor: 'rgba(0, 0, 0, 0)' +}; + +const triggerStatusMap = { + '0': 'OK', + '1': 'Problem' +}; + +export class TriggerPanelCtrl extends PanelCtrl { + + /** @ngInject */ + constructor($scope, $injector, $element, datasourceSrv, templateSrv, contextSrv, dashboardSrv) { + super($scope, $injector); + this.datasourceSrv = datasourceSrv; + this.templateSrv = templateSrv; + this.contextSrv = contextSrv; + this.dashboardSrv = dashboardSrv; + + this.editorTabIndex = 1; + this.triggerStatusMap = triggerStatusMap; + this.defaultTimeFormat = DEFAULT_TIME_FORMAT; + this.pageIndex = 0; + this.triggerList = []; + this.currentTriggersPage = []; + this.datasources = {}; + + this.panel = migratePanelSchema(this.panel); + _.defaults(this.panel, _.cloneDeep(panelDefaults)); + + this.available_datasources = _.map(this.getZabbixDataSources(), 'name'); + if (this.panel.datasources.length === 0) { + this.panel.datasources.push(this.available_datasources[0]); + } + if (_.isEmpty(this.panel.targets)) { + this.panel.targets[this.panel.datasources[0]] = DEFAULT_TARGET; + } + + this.initDatasources(); + this.events.on('init-edit-mode', this.onInitEditMode.bind(this)); + this.events.on('refresh', this.onRefresh.bind(this)); + } + + initDatasources() { + let promises = _.map(this.panel.datasources, (ds) => { + // Load datasource + return this.datasourceSrv.get(ds) + .then(datasource => { + this.datasources[ds] = datasource; + return datasource; + }); + }); + return Promise.all(promises); + } + + getZabbixDataSources() { + return _.filter(this.datasourceSrv.getMetricSources(), datasource => { + return datasource.meta.id === ZABBIX_DS_ID && datasource.value; + }); + } + + onInitEditMode() { + this.addEditorTab('Triggers', triggerPanelTriggersTab, 1); + this.addEditorTab('Options', triggerPanelOptionsTab, 2); + } + + setTimeQueryStart() { + this.timing.queryStart = new Date().getTime(); + } + + setTimeQueryEnd() { + this.timing.queryEnd = new Date().getTime(); + } + + onRefresh() { + // ignore fetching data if another panel is in fullscreen + if (this.otherPanelInFullscreenMode()) { return; } + + // clear loading/error state + delete this.error; + this.loading = true; + this.setTimeQueryStart(); + + return this.getTriggers() + .then(triggerList => { + // Notify panel that request is finished + this.loading = false; + this.setTimeQueryEnd(); + + // Limit triggers number + this.triggerList = triggerList.slice(0, this.panel.limit); + this.getCurrentTriggersPage(); + this.render(this.triggerList); + }) + .catch(err => { + // if cancelled keep loading set to true + if (err.cancelled) { + console.log('Panel request cancelled', err); + return; + } + + this.loading = false; + this.error = err.message || "Request Error"; + + if (err.data) { + if (err.data.message) { + this.error = err.data.message; + } + if (err.data.error) { + this.error = err.data.error; + } + } + + this.events.emit('data-error', err); + console.log('Panel data error:', err); + }); + } + + getTriggers() { + let promises = _.map(this.panel.datasources, (ds) => { + return this.datasourceSrv.get(ds) + .then(datasource => { + var zabbix = datasource.zabbix; + var showEvents = this.panel.showEvents.value; + var triggerFilter = this.panel.targets[ds]; + var hideHostsInMaintenance = this.panel.hideHostsInMaintenance; + + // Replace template variables + var groupFilter = datasource.replaceTemplateVars(triggerFilter.group.filter); + var hostFilter = datasource.replaceTemplateVars(triggerFilter.host.filter); + var appFilter = datasource.replaceTemplateVars(triggerFilter.application.filter); + + let triggersOptions = { + showTriggers: showEvents, + hideHostsInMaintenance: hideHostsInMaintenance + }; + + return zabbix.getTriggers(groupFilter, hostFilter, appFilter, triggersOptions); + }).then((triggers) => { + return this.getAcknowledges(triggers, ds); + }).then((triggers) => { + return this.filterTriggers(triggers, ds); + }); + }); + + return Promise.all(promises) + .then(results => _.flatten(results)) + .then((triggers) => { + return this.sortTriggers(triggers); + }) + .then(triggers => { + return _.map(triggers, this.formatTrigger.bind(this)); + }); + } + + getAcknowledges(triggerList, ds) { + // Request acknowledges for trigger + var eventids = _.map(triggerList, trigger => { + return trigger.lastEvent.eventid; + }); + + return this.datasources[ds].zabbix.getAcknowledges(eventids) + .then(events => { + + // Map events to triggers + _.each(triggerList, trigger => { + var event = _.find(events, event => { + return event.eventid === trigger.lastEvent.eventid; + }); + + if (event) { + trigger.acknowledges = _.map(event.acknowledges, ack => { + let timestamp = moment.unix(ack.clock); + if (this.panel.customLastChangeFormat) { + ack.time = timestamp.format(this.panel.lastChangeFormat); + } else { + ack.time = timestamp.format(this.defaultTimeFormat); + } + ack.user = ack.alias + ' (' + ack.name + ' ' + ack.surname + ')'; + return ack; + }); + + // Mark acknowledged triggers with different color + if (this.panel.markAckEvents && trigger.acknowledges.length) { + trigger.color = this.panel.ackEventColor; + } + } + }); + + return triggerList; + }); + } + + filterTriggers(triggerList, ds) { + // Filter triggers by description + var triggerFilter = this.panel.targets[ds].trigger.filter; + triggerFilter = this.datasources[ds].replaceTemplateVars(triggerFilter); + if (triggerFilter) { + triggerList = filterTriggers(triggerList, triggerFilter); + } + + // Filter acknowledged triggers + if (this.panel.showTriggers === 'unacknowledged') { + triggerList = _.filter(triggerList, trigger => { + return !trigger.acknowledges; + }); + } else if (this.panel.showTriggers === 'acknowledged') { + triggerList = _.filter(triggerList, 'acknowledges'); + } else { + triggerList = triggerList; + } + + // Filter triggers by severity + triggerList = _.filter(triggerList, trigger => { + return this.panel.triggerSeverity[trigger.priority].show; + }); + + return triggerList; + } + + sortTriggers(triggerList) { + if (this.panel.sortTriggersBy.value === 'priority') { + triggerList = _.sortBy(triggerList, 'priority').reverse(); + } else { + triggerList = _.sortBy(triggerList, 'lastchangeUnix').reverse(); + } + return triggerList; + } + + formatTrigger(trigger) { + let triggerObj = trigger; + + // Format last change and age + trigger.lastchangeUnix = Number(trigger.lastchange); + let timestamp = moment.unix(trigger.lastchangeUnix); + if (this.panel.customLastChangeFormat) { + // User defined format + triggerObj.lastchange = timestamp.format(this.panel.lastChangeFormat); + } else { + triggerObj.lastchange = timestamp.format(this.defaultTimeFormat); + } + triggerObj.age = timestamp.fromNow(true); + + // Set host that the trigger belongs + if (trigger.hosts.length) { + triggerObj.host = trigger.hosts[0].name; + triggerObj.hostTechName = trigger.hosts[0].host; + } + + // Set color + if (trigger.value === '1') { + // Problem state + triggerObj.color = this.panel.triggerSeverity[trigger.priority].color; + } else { + // OK state + triggerObj.color = this.panel.okEventColor; + } + + triggerObj.severity = this.panel.triggerSeverity[trigger.priority].severity; + return triggerObj; + } + + switchComment(trigger) { + trigger.showComment = !trigger.showComment; + } + + acknowledgeTrigger(trigger, message) { + let eventid = trigger.lastEvent.eventid; + let grafana_user = this.contextSrv.user.name; + let ack_message = grafana_user + ' (Grafana): ' + message; + return this.datasourceSrv.get(this.panel.datasource) + .then(datasource => { + let zabbixAPI = datasource.zabbix.zabbixAPI; + return zabbixAPI.acknowledgeEvent(eventid, ack_message); + }) + .then(this.onRefresh.bind(this)); + } + + getCurrentTriggersPage() { + let pageSize = this.panel.pageSize || 10; + let startPos = this.pageIndex * pageSize; + let endPos = Math.min(startPos + pageSize, this.triggerList.length); + this.currentTriggersPage = this.triggerList.slice(startPos, endPos); + return this.currentTriggersPage; + } + + link(scope, elem, attrs, ctrl) { + var data; + var panel = ctrl.panel; + var pageCount = 0; + data = ctrl.triggerList; + + function getTableHeight() { + var panelHeight = ctrl.height; + + if (pageCount > 1) { + panelHeight -= 26; + } + + return (panelHeight - 31) + 'px'; + } + + function switchPage(e) { + let el = $(e.currentTarget); + ctrl.pageIndex = (parseInt(el.text(), 10)-1); + + let pageSize = ctrl.panel.pageSize || 10; + let startPos = ctrl.pageIndex * pageSize; + let endPos = Math.min(startPos + pageSize, ctrl.triggerList.length); + ctrl.currentTriggersPage = ctrl.triggerList.slice(startPos, endPos); + + scope.$apply(() => { + renderPanel(); + }); + } + + function appendPaginationControls(footerElem) { + footerElem.empty(); + + var pageSize = ctrl.panel.pageSize || 5; + pageCount = Math.ceil(data.length / pageSize); + if (pageCount === 1) { + return; + } + + var startPage = Math.max(ctrl.pageIndex - 3, 0); + var endPage = Math.min(pageCount, startPage + 9); + + var paginationList = $(''); + + for (var i = startPage; i < endPage; i++) { + var activeClass = i === ctrl.pageIndex ? 'active' : ''; + var pageLinkElem = $('
  • ' + (i+1) + '
  • '); + paginationList.append(pageLinkElem); + } + + footerElem.append(paginationList); + } + + function renderPanel() { + var panelElem = elem.parents('.panel'); + var rootElem = elem.find('.triggers-panel-scroll'); + var footerElem = elem.find('.triggers-panel-footer'); + + elem.css({'font-size': panel.fontSize}); + panelElem.addClass('triggers-panel-wrapper'); + appendPaginationControls(footerElem); + + rootElem.css({'max-height': panel.scroll ? getTableHeight() : '' }); + ctrl.renderingCompleted(); + } + + elem.on('click', '.triggers-panel-page-link', switchPage); + + var unbindDestroy = scope.$on('$destroy', function() { + elem.off('click', '.triggers-panel-page-link'); + unbindDestroy(); + }); + + ctrl.events.on('render', (renderData) => { + data = renderData || data; + if (data) { + scope.$apply(() => { + renderPanel(); + }); + } + }); + } +} + +TriggerPanelCtrl.templateUrl = 'panel-triggers/module.html'; + +function filterTriggers(triggers, triggerFilter) { + if (utils.isRegex(triggerFilter)) { + return _.filter(triggers, function(trigger) { + return utils.buildRegex(triggerFilter).test(trigger.description); + }); + } else { + return _.filter(triggers, function(trigger) { + return trigger.description === triggerFilter; + }); + } +} diff --git a/src/panel-triggers/triggers_tab.js b/src/panel-triggers/triggers_tab.js index 7212cfc..c033152 100644 --- a/src/panel-triggers/triggers_tab.js +++ b/src/panel-triggers/triggers_tab.js @@ -2,7 +2,7 @@ import _ from 'lodash'; import * as utils from '../datasource-zabbix/utils'; import './datasource-selector.directive'; import '../datasource-zabbix/css/query-editor.css!'; -import {DEFAULT_TARGET} from './module'; +import {DEFAULT_TARGET} from './triggers_panel_ctrl'; class TriggersTabCtrl { diff --git a/src/test-setup/jest-setup.js b/src/test-setup/jest-setup.js index ff528b8..439e185 100644 --- a/src/test-setup/jest-setup.js +++ b/src/test-setup/jest-setup.js @@ -2,6 +2,8 @@ /* globals global: false */ import {JSDOM} from 'jsdom'; +import System from 'systemjs'; +import {PanelCtrl} from './panelStub'; // Mock Grafana modules that are not available outside of the core project // Required for loading module.js @@ -17,9 +19,12 @@ jest.mock('angular', () => { }; }, {virtual: true}); +let mockPanelCtrl = PanelCtrl; jest.mock('app/plugins/sdk', () => { return { - QueryCtrl: null + QueryCtrl: null, + loadPluginCss: () => {}, + PanelCtrl: mockPanelCtrl }; }, {virtual: true}); @@ -48,3 +53,4 @@ let dom = new JSDOM(''); global.window = dom.window; global.document = global.window.document; global.Node = window.Node; +global.System = System; diff --git a/src/test-setup/panelStub.js b/src/test-setup/panelStub.js new file mode 100644 index 0000000..b5f601f --- /dev/null +++ b/src/test-setup/panelStub.js @@ -0,0 +1,97 @@ +// JSHint options +/* jshint ignore:start */ + +export class PanelCtrl { + constructor($scope, $injector) { + this.$injector = $injector; + this.$scope = $scope; + this.panel = $scope.panel; + this.events = { + on: () => {} + }; + } + + init() { + } + + renderingCompleted() { + } + + refresh() { + } + + publishAppEvent(evtName, evt) { + } + + changeView(fullscreen, edit) { + } + + viewPanel() { + this.changeView(true, false); + } + + editPanel() { + this.changeView(true, true); + } + + exitFullscreen() { + this.changeView(false, false); + } + + initEditMode() { + } + + changeTab(newIndex) { + } + + addEditorTab(title, directiveFn, index) { + } + + getMenu() { + return []; + } + + getExtendedMenu() { + return []; + } + + otherPanelInFullscreenMode() { + return false; + } + + calculatePanelHeight() { + } + + render(payload) { + } + + toggleEditorHelp(index) { + } + + duplicate() { + } + + updateColumnSpan(span) { + } + + removePanel() { + } + + editPanelJson() { + } + + replacePanel(newPanel, oldPanel) { + } + + sharePanel() { + } + + getInfoMode() { + } + + getInfoContent(options) { + } + + openInspector() { + } +}