You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

483 lines
20 KiB

zabbix_export:
version: '7.0'
media_types:
-
name: Opsgenie
type: WEBHOOK
parameters:
-
name: alert_message
value: '{ALERT.MESSAGE}'
-
name: alert_subject
value: '{ALERT.SUBJECT}'
-
name: event_id
value: '{EVENT.ID}'
-
name: event_nseverity
value: '{EVENT.NSEVERITY}'
-
name: event_source
value: '{EVENT.SOURCE}'
-
name: event_tags_json
value: '{EVENT.TAGSJSON}'
-
name: event_update_action
value: '{EVENT.UPDATE.ACTION}'
-
name: event_update_status
value: '{EVENT.UPDATE.STATUS}'
-
name: event_value
value: '{EVENT.VALUE}'
-
name: opsgenie_api
value: '<put your opsgenie api>'
-
name: opsgenie_tags
value: ''
-
name: opsgenie_teams
value: ''
-
name: opsgenie_token
value: '<put your token>'
-
name: opsgenie_web
value: '<put your opsgenie web>'
-
name: severity_average
value: P3
-
name: severity_default
value: P5
-
name: severity_disaster
value: P1
-
name: severity_high
value: P2
-
name: severity_information
value: P5
-
name: severity_not_classified
value: P5
-
name: severity_warning
value: P4
-
name: status_counter
value: '25'
-
name: trigger_id
value: '{TRIGGER.ID}'
-
name: zbxurl
value: '{$ZABBIX.URL}'
-
name: zbxuser
value: '{USER.FULLNAME}'
script: |
var method,
Media = {
params: {},
name: '',
labels: [],
HTTPProxy: '',
setParams: function (params) {
if (typeof params !== 'object') {
return;
}
Media.params = params;
Media.params.api += Media.params.api.endsWith('/') ? '' : '/';
Media.params.web += Media.params.web.endsWith('/') ? '' : '/';
},
setProxy: function (HTTPProxy) {
if (typeof HTTPProxy !== 'undefined' && HTTPProxy.trim() !== '') {
Media.HTTPProxy = HTTPProxy;
}
},
setTags: function(event_tags_json) {
if (typeof event_tags_json !== 'undefined' && event_tags_json !== ''
&& event_tags_json !== '{EVENT.TAGSJSON}') {
try {
var tags = JSON.parse(event_tags_json),
label;
tags.forEach(function (tag) {
if (typeof tag.tag === 'string') {
label = (tag.tag + (typeof tag.value !== 'undefined'
&& tag.value !== '' ? (':' + tag.value) : '')).replace(/\s/g, '_');
Media.labels.push(label);
}
});
}
catch (error) {
Zabbix.log(4, '[ ' + Media.name + ' Webhook ] Failed to parse "event_tags_json" param');
}
}
},
request: function (method, query, data, allow_404) {
if (typeof(allow_404) === 'undefined') {
allow_404 = false;
}
['api', 'token'].forEach(function (field) {
if (typeof Media.params !== 'object' || typeof Media.params[field] === 'undefined'
|| Media.params[field] === '') {
throw 'Required ' + Media.name + ' param is not set: "' + field + '".';
}
});
var response,
url = Media.params.api + query,
request = new HttpRequest();
request.addHeader('Content-Type: application/json');
request.addHeader('Authorization: ' + Media.params.token);
request.setProxy(Media.HTTPProxy);
if (typeof data !== 'undefined') {
data = JSON.stringify(data);
}
Zabbix.log(4, '[ ' + Media.name + ' Webhook ] Sending request: ' +
url + ((typeof data === 'string') ? ('\n' + data) : ''));
switch (method) {
case 'get':
response = request.get(url, data);
break;
case 'post':
response = request.post(url, data);
break;
case 'put':
response = request.put(url, data);
break;
default:
throw 'Unsupported HTTP request method: ' + method;
}
Zabbix.log(4, '[ ' + Media.name + ' Webhook ] Received response with status code ' +
request.getStatus() + '\n' + response);
if (response !== null) {
try {
response = JSON.parse(response);
}
catch (error) {
Zabbix.log(4, '[ ' + Media.name + ' Webhook ] Failed to parse response.');
response = null;
}
}
if ((request.getStatus() < 200 || request.getStatus() >= 300)
&& (!allow_404 || request.getStatus() !== 404)) {
var message = 'Request failed with status code ' + request.getStatus();
if (response !== null) {
if (typeof response.errors === 'object' && Object.keys(response.errors).length > 0) {
message += ': ' + JSON.stringify(response.errors);
}
else if (typeof response.errorMessages === 'object' && Object.keys(response.errorMessages).length > 0) {
message += ': ' + JSON.stringify(response.errorMessages);
}
else if (typeof response.message === 'string') {
message += ': ' + response.message;
}
}
throw message + ' Check debug log for more information.';
}
return {
status: request.getStatus(),
response: response
};
},
getAlertId: function (requestId) {
status_counter = params.status_counter || 25;
do {
resp = Media.request('get', 'requests/' + requestId, undefined, true);
status_counter -= 1;
}
while ( status_counter > 0 &&
(
typeof resp.response !== 'object' ||
typeof resp.response.data === 'undefined' ||
resp.response.data.success === false &&
!resp.response.data.status.includes("There is no open alert") &&
!resp.response.data.status.includes("Alert is already")
)
);
if (typeof resp.response !== 'object' || typeof resp.response.data === 'undefined') {
throw 'Cannot get ' + Media.name + ' issue ID. Check debug log for more information.';
}
else if (resp.response.data.success === false ) {
throw Media.name + ': Operation status (' + resp.response.data.status + ')';
}
return resp;
}
};
try {
var result = {tags: {}},
params = JSON.parse(value),
media = {},
fields = {},
resp = {},
responders = [],
tags = [],
required_params = [
'alert_subject',
'alert_message',
'event_id',
'event_source',
'event_value',
'event_update_status',
'opsgenie_api',
'opsgenie_web',
'opsgenie_token'
],
severities = [
'not_classified',
'information',
'warning',
'average',
'high',
'disaster',
'resolved',
'default'
],
priority;
Object.keys(params)
.forEach(function (key) {
if (required_params.indexOf(key) !== -1 && params[key].trim() === '') {
throw 'Parameter "' + key + '" cannot be empty.';
}
if (key.startsWith('opsgenie_')) {
media[key.substring(9)] = params[key];
}
});
// Possible values of event_source:
// 0 - Trigger, 1 - Discovery, 2 - Autoregistration, 3 - Internal.
if ([0, 1, 2, 3].indexOf(parseInt(params.event_source)) === -1) {
throw 'Incorrect "event_source" parameter given: "' + params.event_source + '".\nMust be 0-3.';
}
// Check event_value for trigger-based and internal events.
// Possible values: 1 for problem, 0 for recovering
if (params.event_value !== '0' && params.event_value !== '1'
&& (params.event_source === '0' || params.event_source === '3')) {
throw 'Incorrect "event_value" parameter given: ' + params.event_value + '\nMust be 0 or 1.';
}
// Check event_update_status only for trigger-based events.
// Possible values: 0 - Webhook was called because of problem/recovery event, 1 - Update operation.
if (params.event_source === '0' && params.event_update_status !== '0' && params.event_update_status !== '1') {
throw 'Incorrect "event_update_status" parameter given: ' + params.event_update_status + '\nMust be 0 or 1.';
}
// Check event_id for a numeric value.
if (isNaN(parseInt(params.event_id)) || params.event_id < 1) {
throw 'Incorrect "event_id" parameter given: ' + params.event_id + '\nMust be a positive number.';
}
if ((params.event_source === '1' || params.event_source === '2') && params.event_value === '0') {
throw 'Recovery operations are supported only for Trigger and Internal actions.';
}
if ([0, 1, 2, 3, 4, 5].indexOf(parseInt(params.event_nseverity)) === -1) {
params.event_nseverity = '7';
}
if (params.event_value === '0') {
params.event_nseverity = '6';
}
priority = params['severity_' + severities[params.event_nseverity]];
params.zbxurl = params.zbxurl + (params.zbxurl.endsWith('/') ? '' : '/');
Media.name = 'Opsgenie';
Media.setParams(media);
Media.params.token = 'GenieKey ' + Media.params.token;
Media.setProxy(params.HTTPProxy);
Media.setTags(params.event_tags_json); // Set Media.labels
// Create an issue.
// Numeric value of the event that triggered an action (1 for problem, 0 for recovering).
// Numeric value of the problem update status. Possible values:
// 0 - Webhook was called because of problem/recovery event, 1 - Update operation.
if ((params.event_source == 0 && params.event_value == 1 && params.event_update_status == 0)
|| (params.event_source == 3 && params.event_value == 1)
|| params.event_source == 1 || params.event_source == 2) {
fields.message = params.alert_subject;
fields.alias = params.event_id;
fields.description = params.alert_message;
fields.priority = priority;
fields.source = 'Zabbix';
if (params.event_source === '0') {
fields.details = {
'Zabbix server': params.zbxurl,
Problem: params.zbxurl + 'tr_events.php?triggerid=' + params.trigger_id + '&eventid=' + params.event_id
};
}
else {
fields.details = {'Zabbix server': params.zbxurl};
}
if (typeof params.opsgenie_teams === 'string') {
responders = params.opsgenie_teams.split(',');
fields.responders = responders.map(function(team) {
return {type: 'team', name: team.trim()};
});
}
fields.tags = Media.labels;
if (typeof params.opsgenie_tags === 'string') {
tags = params.opsgenie_tags.split(',');
tags.forEach(function(item) {
fields.tags.push(item.trim());
});
}
resp = Media.request('post', '', fields);
if (typeof resp.response !== 'object' || typeof resp.response.result === 'undefined') {
throw 'Cannot create ' + Media.name + ' issue. Check debug log for more information.';
}
if (resp.status === 202) {
resp = Media.getAlertId(resp.response.requestId);
if (params.event_source == 0 && params.event_value == 1 && params.event_update_status == 0) {
result.tags.__zbx_ops_issuekey = resp.response.data.alertId;
result.tags.__zbx_ops_issuelink = Media.params.web + 'alert/detail/' + resp.response.data.alertId;
}
}
else {
throw Media.name + ' response code is unexpected. Check debug log for more information.';
}
}
// Update or close the created issue.
else {
fields.user = (params.event_value != 0) ? params.zbxuser : '';
fields.note = params.alert_message;
if ( [0, 3].indexOf(parseInt(params.event_source)) > -1 && params.event_value == 0 ) {
// skip sending of close request from update operation(mandatory when both update & recovery operations are defined in action)
method = params.event_update_status == 0 ? "close" : "skip";
}
else if ( params.event_source == 0 && params.event_value == 1 && params.event_update_status == 1 && params.event_update_action.includes('acknowledged')) {
method = params.event_update_action.includes('unacknowledged') ? "unacknowledge" : "acknowledge";
}
else {
method = "notes";
}
if (method !== "skip") {
resp = Media.request('post', params.event_id + '/' + method +'?identifierType=alias', fields);
if (typeof resp.response !== 'object' || typeof resp.response.result === 'undefined') {
throw 'Cannot update ' + Media.name + ' issue. Check debug log for more information.';
}
if (resp.status === 202) {
resp = Media.getAlertId(resp.response.requestId);
}
else {
throw Media.name + ' response code is unexpected. Check debug log for more information.';
}
}
}
return JSON.stringify(result);
}
catch (error) {
Zabbix.log(3, '[ ' + Media.name + ' Webhook ] ERROR: ' + error);
throw 'Sending failed: ' + error;
}
process_tags: 'YES'
show_event_menu: 'YES'
event_menu_url: '{EVENT.TAGS.__zbx_ops_issuelink}'
event_menu_name: 'Opsgenie: {EVENT.TAGS.__zbx_ops_issuekey}'
description: |
Please refer to https://docs.opsgenie.com/docs/alert-api and https://www.zabbix.com/documentation/7.0/manual/config/notifications/media/webhook#example_scripts.
Set global macro {$ZABBIX.URL} with your Zabbix server URL.
Add dedicated user with media type "Opsgenie".
Change the values of the variables opsgenie_api (https://api.opsgenie.com/v2/alerts or https://api.eu.opsgenie.com/v2/alerts),
opsgenie_web (for example, https://myzabbix.app.opsgenie.com), opsgenie_token.
message_templates:
-
event_source: TRIGGERS
operation_mode: PROBLEM
subject: 'Problem: {EVENT.NAME}'
message: |
Problem started at {EVENT.TIME} on {EVENT.DATE}
Problem name: {EVENT.NAME}
Host: {HOST.NAME}
Severity: {EVENT.SEVERITY}
Operational data: {EVENT.OPDATA}
Original problem ID: {EVENT.ID}
{TRIGGER.URL}
-
event_source: TRIGGERS
operation_mode: UPDATE
subject: 'Updated problem in {EVENT.AGE}: {EVENT.NAME}'
message: |
{USER.FULLNAME} {EVENT.UPDATE.ACTION} problem at {EVENT.UPDATE.DATE} {EVENT.UPDATE.TIME}.
{EVENT.UPDATE.MESSAGE}
Current problem status is {EVENT.STATUS}, age is {EVENT.AGE}, acknowledged: {EVENT.ACK.STATUS}.
-
event_source: DISCOVERY
operation_mode: PROBLEM
subject: 'Discovery: {DISCOVERY.DEVICE.STATUS} {DISCOVERY.DEVICE.IPADDRESS}'
message: |
Discovery rule: {DISCOVERY.RULE.NAME}
Device IP: {DISCOVERY.DEVICE.IPADDRESS}
Device DNS: {DISCOVERY.DEVICE.DNS}
Device status: {DISCOVERY.DEVICE.STATUS}
Device uptime: {DISCOVERY.DEVICE.UPTIME}
Device service name: {DISCOVERY.SERVICE.NAME}
Device service port: {DISCOVERY.SERVICE.PORT}
Device service status: {DISCOVERY.SERVICE.STATUS}
Device service uptime: {DISCOVERY.SERVICE.UPTIME}
-
event_source: AUTOREGISTRATION
operation_mode: PROBLEM
subject: 'Autoregistration: {HOST.HOST}'
message: |
Host name: {HOST.HOST}
Host IP: {HOST.IP}
Agent port: {HOST.PORT}
-
event_source: TRIGGERS
operation_mode: RECOVERY
subject: 'Resolved in {EVENT.DURATION}: {EVENT.NAME}'
message: |
Problem has been resolved at {EVENT.RECOVERY.TIME} on {EVENT.RECOVERY.DATE}
Problem name: {EVENT.NAME}
Problem duration: {EVENT.DURATION}
Host: {HOST.NAME}
Severity: {EVENT.SEVERITY}
Original problem ID: {EVENT.ID}
{TRIGGER.URL}