mirror of
https://github.com/librenms/librenms.git
synced 2024-10-07 16:52:45 +00:00
feature: Added elasticsearch transport and docs (#6687)
* Add elasticsearch transport and docs * Fix formatting * Fix formatting * Remove librenms string from sql file * Add missing states * Use ifAlias not ifDescr * Update Alerting.md * Remove unused es_enabled flag * Fix storage column names * Fix processor columns * Rename 194.sql to 193.sql
This commit is contained in:
@@ -34,6 +34,7 @@ Table of Content:
|
|||||||
- [Cisco Spark](#transports-ciscospark)
|
- [Cisco Spark](#transports-ciscospark)
|
||||||
- [SMSEagle](#transports-smseagle)
|
- [SMSEagle](#transports-smseagle)
|
||||||
- [Syslog](#transports-syslog)
|
- [Syslog](#transports-syslog)
|
||||||
|
- [Elasticsearch](#transports-elasticsearch)
|
||||||
- [Entities](#entities)
|
- [Entities](#entities)
|
||||||
- [Devices](#entity-devices)
|
- [Devices](#entity-devices)
|
||||||
- [BGP Peers](#entity-bgppeers)
|
- [BGP Peers](#entity-bgppeers)
|
||||||
@@ -679,15 +680,38 @@ $config['alert']['transports']['syslog']['syslog_facility'] = 3;
|
|||||||
```
|
```
|
||||||
~
|
~
|
||||||
|
|
||||||
|
## <a name="transports-elasticsearch">Elasticsearch</a>
|
||||||
|
|
||||||
|
You can have LibreNMS emit alerts to an elasticsearch database. Each fault will be sent as a separate document.
|
||||||
|
The index pattern uses strftime() formatting.
|
||||||
|
The proxy setting uses the proxy set in config.php if true and does not if false; this allows you to use local servers.
|
||||||
|
|
||||||
|
~
|
||||||
|
```php
|
||||||
|
$config['alert']['transports']['elasticsearch']['es_host'] = '127.0.0.1';
|
||||||
|
$config['alert']['transports']['elasticsearch']['es_port'] = 9200;
|
||||||
|
$config['alert']['transports']['elasticsearch']['es_index'] = 'librenms-%Y.%m.%d';
|
||||||
|
$config['alert']['transports']['elasticsearch']['es_proxy'] = false;
|
||||||
|
```
|
||||||
|
~
|
||||||
|
|
||||||
# <a name="entities">Entities
|
# <a name="entities">Entities
|
||||||
|
|
||||||
Entities as described earlier are based on the table and column names within the database, if you are unsure of what the entity is you want then have a browse around inside MySQL using `show tables` and `desc <tablename>`.
|
Entities as described earlier are based on the table and column names within the database, if you are unsure of what the entity is you want then have a browse around inside MySQL using `show tables` and `desc <tablename>`.
|
||||||
|
|
||||||
## <a name="entity-devices">Devices</a>
|
## <a name="entity-devices">Devices</a>
|
||||||
|
|
||||||
__devices.hostname__ = The devices hostname.
|
__devices.hostname__ = The device hostname.
|
||||||
|
|
||||||
__devices.location__ = The devices location.
|
__devices.sysName__ = The device sysName.
|
||||||
|
|
||||||
|
__devices.sysDescr__ = The device sysDescr.
|
||||||
|
|
||||||
|
__devices.hardware__ = The device hardware.
|
||||||
|
|
||||||
|
__devices.version__ = The device os version.
|
||||||
|
|
||||||
|
__devices.location__ = The device location.
|
||||||
|
|
||||||
__devices.status__ = The status of the device, 1 = up, 0 = down.
|
__devices.status__ = The status of the device, 1 = up, 0 = down.
|
||||||
|
|
||||||
|
|||||||
@@ -20,11 +20,15 @@ if (is_admin() === false) {
|
|||||||
$transport = mres($_POST['transport']);
|
$transport = mres($_POST['transport']);
|
||||||
|
|
||||||
require_once $config['install_dir'].'/includes/alerts.inc.php';
|
require_once $config['install_dir'].'/includes/alerts.inc.php';
|
||||||
$tmp = array(dbFetchRow('select device_id,hostname from devices order by device_id asc limit 1'));
|
$tmp = array(dbFetchRow('select device_id,hostname,sysDescr,version,hardware,location from devices order by device_id asc limit 1'));
|
||||||
$tmp['contacts'] = GetContacts($tmp);
|
$tmp['contacts'] = GetContacts($tmp);
|
||||||
$obj = array(
|
$obj = array(
|
||||||
"hostname" => $tmp[0]['hostname'],
|
"hostname" => $tmp[0]['hostname'],
|
||||||
"device_id" => $tmp[0]['device_id'],
|
"device_id" => $tmp[0]['device_id'],
|
||||||
|
"sysDescr" => $tmp[0]['sysDescr'],
|
||||||
|
"version" => $tmp[0]['version'],
|
||||||
|
"hardware" => $tmp[0]['hardware'],
|
||||||
|
"location" => $tmp[0]['location'],
|
||||||
"title" => "Testing transport from ".$config['project_name'],
|
"title" => "Testing transport from ".$config['project_name'],
|
||||||
"elapsed" => "11s",
|
"elapsed" => "11s",
|
||||||
"id" => "000",
|
"id" => "000",
|
||||||
@@ -33,6 +37,7 @@ $obj = array(
|
|||||||
"severity" => "critical",
|
"severity" => "critical",
|
||||||
"rule" => "%macros.device = 1",
|
"rule" => "%macros.device = 1",
|
||||||
"name" => "Test-Rule",
|
"name" => "Test-Rule",
|
||||||
|
"string" => "#1: test => string;",
|
||||||
"timestamp" => date("Y-m-d H:i:s"),
|
"timestamp" => date("Y-m-d H:i:s"),
|
||||||
"contacts" => $tmp['contacts'],
|
"contacts" => $tmp['contacts'],
|
||||||
"state" => "1",
|
"state" => "1",
|
||||||
|
|||||||
@@ -1348,6 +1348,58 @@ echo '
|
|||||||
<input id="syslog_facility" class="form-control" type="text" name="global-config-input" value="'.$syslog_facility['config_value'].'" data-config_id="'.$syslog_facility['config_id'].'">
|
<input id="syslog_facility" class="form-control" type="text" name="global-config-input" value="'.$syslog_facility['config_value'].'" data-config_id="'.$syslog_facility['config_id'].'">
|
||||||
<span class="form-control-feedback">
|
<span class="form-control-feedback">
|
||||||
<i class="fa" aria-hidden="true"></i>
|
<i class="fa" aria-hidden="true"></i>
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>';
|
||||||
|
$es_host = get_config_by_name('alert.transports.elasticsearch.es_host');
|
||||||
|
$es_port = get_config_by_name('alert.transports.elasticsearch.es_port');
|
||||||
|
$es_index = get_config_by_name('alert.transports.elasticsearch.es_index');
|
||||||
|
$es_proxy = get_config_by_name('alert.transports.elasticsearch.es_proxy');
|
||||||
|
echo '
|
||||||
|
<div class="panel panel-default">
|
||||||
|
<div class="panel-heading">
|
||||||
|
<h4 class="panel-title">
|
||||||
|
<a data-toggle="collapse" data-parent="#accordion" href="#es_transport_expand"><i class="fa fa-caret-down"></i> Elasticsearch transport</a> <button name="test-alert" id="test-alert" type="button" data-transport="elasticsearch" class="btn btn-primary btn-xs pull-right">Test transport</button>
|
||||||
|
</h4>
|
||||||
|
</div>
|
||||||
|
<div id="es_transport_expand" class="panel-collapse collapse">
|
||||||
|
<div class="panel-body">
|
||||||
|
<div class="form-group has-feedback">
|
||||||
|
<label for="es_host" class="col-sm-4 control-label">Elasticsearch Host </label>
|
||||||
|
<div class="col-sm-4">
|
||||||
|
<input id="es_host" class="form-control" type="text" name="global-config-input" value="'.$es_host['config_value'].'" data-config_id="'.$es_host['config_id'].'">
|
||||||
|
<span class="form-control-feedback">
|
||||||
|
<i class="fa" aria-hidden="true"></i>
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="form-group has-feedback">
|
||||||
|
<label for="es_port" class="col-sm-4 control-label">Elasticsearch Port </label>
|
||||||
|
<div class="col-sm-4">
|
||||||
|
<input id="es_port" class="form-control" type="text" name="global-config-input" value="'.$es_port['config_value'].'" data-config_id="'.$es_port['config_id'].'">
|
||||||
|
<span class="form-control-feedback">
|
||||||
|
<i class="fa" aria-hidden="true"></i>
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="form-group has-feedback">
|
||||||
|
<label for="es_index" class="col-sm-4 control-label">Elasticsearch Index Pattern </label>
|
||||||
|
<div class="col-sm-4">
|
||||||
|
<input id="es_index" class="form-control" type="text" name="global-config-input" value="'.$es_index['config_value'].'" data-config_id="'.$es_index['config_id'].'">
|
||||||
|
<span class="form-control-feedback">
|
||||||
|
<i class="fa" aria-hidden="true"></i>
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="form-group has-feedback">
|
||||||
|
<label for="es_proxy" class="col-sm-4 control-label">Use proxy if configured? </label>
|
||||||
|
<div class="col-sm-4">
|
||||||
|
<input id="es_proxy" type="checkbox" name="global-config-check" '.$es_proxy['config_value'].' data-on-text="Yes" data-off-text="No" data-size="small" data-config_id="'.$es_proxy['config_id'].'">
|
||||||
|
<span class="form-control-feedback">
|
||||||
|
<i class="fa" aria-hidden="true"></i>
|
||||||
</span>
|
</span>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|||||||
@@ -444,11 +444,14 @@ function DescribeAlert($alert)
|
|||||||
{
|
{
|
||||||
$obj = array();
|
$obj = array();
|
||||||
$i = 0;
|
$i = 0;
|
||||||
$device = dbFetchRow('SELECT hostname, sysName, location, purpose, notes, uptime FROM devices WHERE device_id = ?', array($alert['device_id']));
|
$device = dbFetchRow('SELECT hostname, sysName, sysDescr, hardware, version, location, purpose, notes, uptime FROM devices WHERE device_id = ?', array($alert['device_id']));
|
||||||
$tpl = dbFetchRow('SELECT `template`,`title`,`title_rec` FROM `alert_templates` JOIN `alert_template_map` ON `alert_template_map`.`alert_templates_id`=`alert_templates`.`id` WHERE `alert_template_map`.`alert_rule_id`=?', array($alert['rule_id']));
|
$tpl = dbFetchRow('SELECT `template`,`title`,`title_rec` FROM `alert_templates` JOIN `alert_template_map` ON `alert_template_map`.`alert_templates_id`=`alert_templates`.`id` WHERE `alert_template_map`.`alert_rule_id`=?', array($alert['rule_id']));
|
||||||
$default_tpl = "%title\r\nSeverity: %severity\r\n{if %state == 0}Time elapsed: %elapsed\r\n{/if}Timestamp: %timestamp\r\nUnique-ID: %uid\r\nRule: {if %name}%name{else}%rule{/if}\r\n{if %faults}Faults:\r\n{foreach %faults} #%key: %value.string\r\n{/foreach}{/if}Alert sent to: {foreach %contacts}%value <%key> {/foreach}";
|
$default_tpl = "%title\r\nSeverity: %severity\r\n{if %state == 0}Time elapsed: %elapsed\r\n{/if}Timestamp: %timestamp\r\nUnique-ID: %uid\r\nRule: {if %name}%name{else}%rule{/if}\r\n{if %faults}Faults:\r\n{foreach %faults} #%key: %value.string\r\n{/foreach}{/if}Alert sent to: {foreach %contacts}%value <%key> {/foreach}";
|
||||||
$obj['hostname'] = $device['hostname'];
|
$obj['hostname'] = $device['hostname'];
|
||||||
$obj['sysName'] = $device['sysName'];
|
$obj['sysName'] = $device['sysName'];
|
||||||
|
$obj['sysDescr'] = $device['sysDescr'];
|
||||||
|
$obj['hardware'] = $device['hardware'];
|
||||||
|
$obj['version'] = $device['version'];
|
||||||
$obj['location'] = $device['location'];
|
$obj['location'] = $device['location'];
|
||||||
$obj['uptime'] = $device['uptime'];
|
$obj['uptime'] = $device['uptime'];
|
||||||
$obj['uptime_short'] = formatUptime($device['uptime'], 'short');
|
$obj['uptime_short'] = formatUptime($device['uptime'], 'short');
|
||||||
|
|||||||
173
includes/alerts/transport.elasticsearch.php
Normal file
173
includes/alerts/transport.elasticsearch.php
Normal file
@@ -0,0 +1,173 @@
|
|||||||
|
/* LibreNMS
|
||||||
|
*
|
||||||
|
* Copyright (C) 2017 Paul Blasquez <pblasquez@gmail.com>
|
||||||
|
* This program is free software: you can redistribute it and/or modify
|
||||||
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
* the Free Software Foundation, either version 3 of the License, or
|
||||||
|
* (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.See the
|
||||||
|
* GNU General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License
|
||||||
|
* along with this program. If not, see <http://www.gnu.org/licenses/>. */
|
||||||
|
|
||||||
|
$es_host = '127.0.0.1';
|
||||||
|
$es_port = 9200;
|
||||||
|
$index = strftime("librenms-%Y.%m.%d");
|
||||||
|
$type = 'alert';
|
||||||
|
$severity = $obj['severity'];
|
||||||
|
$device = device_by_id_cache($obj['device_id']); // for event logging
|
||||||
|
|
||||||
|
if (!empty($opts['es_host'])) {
|
||||||
|
if (preg_match("/[a-zA-Z]/", $opts['es_host'])) {
|
||||||
|
$es_host = gethostbyname($opts['es_host']);
|
||||||
|
if ($es_host === $opts['es_host']) {
|
||||||
|
return "Alphanumeric hostname found but does not resolve to an IP.";
|
||||||
|
}
|
||||||
|
} elseif (filter_var($opts['es_host'], FILTER_VALIDATE_IP)) {
|
||||||
|
$es_host = $opts['es_host'];
|
||||||
|
} else {
|
||||||
|
return "Elasticsearch host is not a valid IP: " . $opts['es_host'];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!empty($opts['es_port']) && preg_match("/^\d+$/", $opts['es_port'])) {
|
||||||
|
$es_port = $opts['es_port'];
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!empty($opts['es_index'])) {
|
||||||
|
$index = strftime($opts['es_index']);
|
||||||
|
}
|
||||||
|
|
||||||
|
$host = $es_host.':'.$es_port.'/'.$index.'/'.$type;
|
||||||
|
|
||||||
|
switch( $obj['state'] ) {
|
||||||
|
case 0:
|
||||||
|
$state = "ok";
|
||||||
|
break;
|
||||||
|
case 1:
|
||||||
|
$state = $severity;
|
||||||
|
break;
|
||||||
|
case 2:
|
||||||
|
$state = "acknowledged";
|
||||||
|
break;
|
||||||
|
case 3:
|
||||||
|
$state = "worse";
|
||||||
|
break;
|
||||||
|
case 4:
|
||||||
|
$state = "better";
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
$data = array(
|
||||||
|
'@timestamp' => strftime("%Y-%m-%dT%T"),
|
||||||
|
"host" => gethostname(),
|
||||||
|
"location" => $obj['location'],
|
||||||
|
"title" => $obj['name'],
|
||||||
|
"message" => $obj['string'],
|
||||||
|
"device_id" => $obj['device_id'],
|
||||||
|
"device_name" => $obj['hostname'],
|
||||||
|
"device_hardware" => $obj['hardware'],
|
||||||
|
"device_version" => $obj['version'],
|
||||||
|
"state" => $state,
|
||||||
|
"severity" => $severity,
|
||||||
|
"first_occurrence" => $obj['timestamp'],
|
||||||
|
"entity_type" => "device",
|
||||||
|
"entity_id" => $obj['device_id'],
|
||||||
|
"entity_name" => $obj['hostname'],
|
||||||
|
"entity_descr" => $obj['sysDescr'],
|
||||||
|
);
|
||||||
|
|
||||||
|
if( !empty( $obj['faults'] ) ) {
|
||||||
|
foreach($obj['faults'] as $k => $v) {
|
||||||
|
$curl = curl_init();
|
||||||
|
$data['message'] = $v['string'];
|
||||||
|
switch (true) {
|
||||||
|
case (array_key_exists('port_id', $v)):
|
||||||
|
$data['entity_type'] = 'port';
|
||||||
|
$data['entity_id'] = $v['port_id'];
|
||||||
|
$data['entity_name'] = $v['ifName'];
|
||||||
|
$data['entity_descr'] = $v['ifAlias'];
|
||||||
|
break;
|
||||||
|
case (array_key_exists('sensor_id', $v)):
|
||||||
|
$data['entity_type'] = 'sensor';
|
||||||
|
$data['entity_id'] = $v['sensor_id'];
|
||||||
|
$data['entity_name'] = $v['sensor_class'];
|
||||||
|
$data['entity_descr'] = $v['sensor_descr'];
|
||||||
|
break;
|
||||||
|
case (array_key_exists('mempool_id', $v)):
|
||||||
|
$data['entity_type'] = 'mempool';
|
||||||
|
$data['entity_id'] = $v['mempool_id'];
|
||||||
|
$data['entity_name'] = $v['mempool_index'];
|
||||||
|
$data['entity_descr'] = $v['mempool_descr'];
|
||||||
|
break;
|
||||||
|
case (array_key_exists('storage_id', $v)):
|
||||||
|
$data['entity_type'] = 'storage';
|
||||||
|
$data['entity_id'] = $v['storage_id'];
|
||||||
|
$data['entity_name'] = $v['storage_index'];
|
||||||
|
$data['entity_descr'] = $v['storage_descr'];
|
||||||
|
break;
|
||||||
|
case (array_key_exists('processor_id', $v)):
|
||||||
|
$data['entity_type'] = 'processor';
|
||||||
|
$data['entity_id'] = $v['processor_id'];
|
||||||
|
$data['entity_name'] = $v['processor_type'];
|
||||||
|
$data['entity_descr'] = $v['processor_descr'];
|
||||||
|
break;
|
||||||
|
case (array_key_exists('bgpPeer_id', $v)):
|
||||||
|
$data['entity_type'] = 'bgp';
|
||||||
|
$data['entity_id'] = $v['bgpPeer_id'];
|
||||||
|
$data['entity_name'] = 'local: '.$v['bgpPeerLocalAddr'].' - AS'.$obj['bgpLocalAs'];
|
||||||
|
$data['entity_descr'] = 'remote: '.$v['bgpPeerIdentifier'].' - AS'.$v['bgpPeerRemoteAs'];
|
||||||
|
break;
|
||||||
|
case (array_key_exists('tunnel_id', $v)):
|
||||||
|
$data['entity_type'] = 'ipsec_tunnel';
|
||||||
|
$data['entity_id'] = $v['tunnel_id'];
|
||||||
|
$data['entity_name'] = $v['tunnel_name'];
|
||||||
|
$data['entity_descr'] = 'local: '.$v['local_addr'].':'.$v['local_port'].', remote: '.$v['peer_addr'].':'.$v['peer_port'];
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
$data['entity_type'] = 'generic';
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
$alert_message = json_encode($data);
|
||||||
|
curl_setopt($curl, CURLOPT_HTTPHEADER, array(
|
||||||
|
'Content-Type: application/json')
|
||||||
|
);
|
||||||
|
if ($opts['es_proxy'] === true) {
|
||||||
|
set_curl_proxy($curl);
|
||||||
|
}
|
||||||
|
curl_setopt($curl, CURLOPT_URL, $host);
|
||||||
|
curl_setopt($curl, CURLOPT_RETURNTRANSFER, 1);
|
||||||
|
curl_setopt($curl, CURLOPT_POST,true);
|
||||||
|
curl_setopt($curl, CURLOPT_POSTFIELDS, $alert_message );
|
||||||
|
|
||||||
|
$ret = curl_exec($curl);
|
||||||
|
$code = curl_getinfo($curl, CURLINFO_HTTP_CODE);
|
||||||
|
if( $code != 200 && $code != 201 ) {
|
||||||
|
return $host.' returned HTTP Status code '.$code.' for '.$alert_message;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
$curl = curl_init();
|
||||||
|
$alert_message = json_encode($data);
|
||||||
|
curl_setopt($curl, CURLOPT_HTTPHEADER, array(
|
||||||
|
'Content-Type: application/json')
|
||||||
|
);
|
||||||
|
if ($opts['es_proxy'] === true) {
|
||||||
|
set_curl_proxy($curl);
|
||||||
|
}
|
||||||
|
curl_setopt($curl, CURLOPT_URL, $host);
|
||||||
|
curl_setopt($curl, CURLOPT_RETURNTRANSFER, 1);
|
||||||
|
curl_setopt($curl, CURLOPT_POST,true);
|
||||||
|
curl_setopt($curl, CURLOPT_POSTFIELDS, $alert_message );
|
||||||
|
|
||||||
|
$ret = curl_exec($curl);
|
||||||
|
$code = curl_getinfo($curl, CURLINFO_HTTP_CODE);
|
||||||
|
if( $code != 200 && $code != 201 ) {
|
||||||
|
return $host.' returned HTTP Status code '.$code.' for '.$alert_message;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return true;
|
||||||
1
sql-schema/193.sql
Normal file
1
sql-schema/193.sql
Normal file
@@ -0,0 +1 @@
|
|||||||
|
INSERT INTO config (config_name,config_value,config_default,config_descr,config_group,config_group_order,config_sub_group,config_sub_group_order,config_hidden,config_disabled) values ('alert.transports.elasticsearch.es_host','','127.0.0.1','Elasticsearch Host','alerting',0,'transports',0,'0','0'),('alert.transports.elasticsearch.es_port','','9200','Elasticsearch Port','alerting',0,'transports',0,'0','0'),('alert.transports.elasticsearch.es_index','','','Elasticsearch Index','alerting',0,'transports',0,'0','0'),('alert.transports.elasticsearch.es_proxy','','FALSE','Use Proxy?','alerting',0,'transports',0,'0','0');
|
||||||
Reference in New Issue
Block a user