2013-06-07 11:44:37 +02:00
|
|
|
<?php
|
2014-07-15 13:39:22 +02:00
|
|
|
// {{{ICINGA_LICENSE_HEADER}}}
|
|
|
|
// {{{ICINGA_LICENSE_HEADER}}}
|
2013-06-07 11:44:37 +02:00
|
|
|
|
|
|
|
namespace Icinga\Protocol\Livestatus;
|
2013-07-12 13:41:48 +02:00
|
|
|
|
|
|
|
use Icinga\Application\Benchmark;
|
2014-11-16 15:51:26 +01:00
|
|
|
use Icinga\Exception\ConfigurationError;
|
|
|
|
use Icinga\Exception\SystemPermissionException;
|
2014-08-27 16:03:15 +02:00
|
|
|
use Icinga\Exception\IcingaException;
|
2013-07-12 13:41:48 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Backend class managing handling MKI Livestatus connections
|
|
|
|
*
|
|
|
|
* Usage example:
|
|
|
|
*
|
|
|
|
* <code>
|
|
|
|
* $lconf = new Connection((object) array(
|
|
|
|
* 'hostname' => 'localhost',
|
|
|
|
* 'root_dn' => 'dc=monitoring,dc=...',
|
|
|
|
* 'bind_dn' => 'cn=Mangager,dc=monitoring,dc=...',
|
|
|
|
* 'bind_pw' => '***'
|
|
|
|
* ));
|
|
|
|
* </code>
|
|
|
|
*
|
|
|
|
* @copyright Copyright (c) 2013 Icinga-Web Team <info@icinga.org>
|
|
|
|
* @author Icinga-Web Team <info@icinga.org>
|
|
|
|
* @license http://www.gnu.org/copyleft/gpl.html GNU General Public License
|
|
|
|
*/
|
2013-06-07 11:44:37 +02:00
|
|
|
class Connection
|
|
|
|
{
|
|
|
|
const TYPE_UNIX = 1;
|
|
|
|
const TYPE_TCP = 2;
|
|
|
|
|
|
|
|
protected $available_tables = array(
|
|
|
|
'hosts', // hosts
|
|
|
|
'services', // services, joined with all data from hosts
|
|
|
|
'hostgroups', // hostgroups
|
|
|
|
'servicegroups', // servicegroups
|
|
|
|
'contactgroups', // contact groups
|
|
|
|
'servicesbygroup', // all services grouped by service groups
|
|
|
|
'servicesbyhostgroup', // all services grouped by host groups
|
|
|
|
'hostsbygroup', // all hosts grouped by host groups
|
|
|
|
'contacts', // contacts
|
|
|
|
'commands', // defined commands
|
|
|
|
'timeperiods', // time period definitions (currently only name
|
|
|
|
// and alias)
|
|
|
|
'downtimes', // all scheduled host and service downtimes,
|
|
|
|
// joined with data from hosts and services.
|
|
|
|
'comments', // all host and service comments
|
|
|
|
'log', // a transparent access to the nagios logfiles
|
|
|
|
// (include archived ones)ones
|
|
|
|
'status', // general performance and status information.
|
|
|
|
// This table contains exactly one dataset.
|
|
|
|
'columns', // a complete list of all tables and columns
|
|
|
|
// available via Livestatus, including
|
|
|
|
// descriptions!
|
|
|
|
'statehist', // 1.2.1i2 sla statistics for hosts and services,
|
|
|
|
// joined with data from hosts, services and log.
|
|
|
|
);
|
|
|
|
|
|
|
|
protected $socket_path;
|
|
|
|
protected $socket_host;
|
|
|
|
protected $socket_port;
|
|
|
|
protected $socket_type;
|
|
|
|
protected $connection;
|
|
|
|
|
|
|
|
public function hasTable($name)
|
|
|
|
{
|
|
|
|
return in_array($name, $this->available_tables);
|
|
|
|
}
|
|
|
|
|
|
|
|
public function __construct($socket = '/var/lib/icinga/rw/live')
|
|
|
|
{
|
|
|
|
$this->assertPhpExtensionLoaded('sockets');
|
|
|
|
if ($socket[0] === '/') {
|
|
|
|
if (! is_writable($socket)) {
|
2014-11-16 15:51:26 +01:00
|
|
|
throw new SystemPermissionException(
|
2014-08-27 16:03:15 +02:00
|
|
|
'Cannot write to livestatus socket "%s"',
|
|
|
|
$socket
|
2013-07-12 13:41:48 +02:00
|
|
|
);
|
2013-06-07 11:44:37 +02:00
|
|
|
}
|
|
|
|
$this->socket_type = self::TYPE_UNIX;
|
|
|
|
$this->socket_path = $socket;
|
|
|
|
} else {
|
|
|
|
if (! preg_match('~^tcp://([^:]+):(\d+)~', $socket, $m)) {
|
2014-11-16 15:51:26 +01:00
|
|
|
throw new ConfigurationError(
|
2014-08-27 16:03:15 +02:00
|
|
|
'Invalid TCP socket syntax: "%s"',
|
|
|
|
$socket
|
2013-07-12 13:41:48 +02:00
|
|
|
);
|
2013-06-07 11:44:37 +02:00
|
|
|
}
|
2014-11-16 15:51:26 +01:00
|
|
|
// TODO: Better config syntax checks
|
2013-06-07 11:44:37 +02:00
|
|
|
$this->socket_host = $m[1];
|
|
|
|
$this->socket_port = (int) $m[2];
|
|
|
|
$this->socket_type = self::TYPE_TCP;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-11-16 15:53:19 +01:00
|
|
|
/**
|
|
|
|
* Count unlimited rows matching the query filter
|
|
|
|
*
|
|
|
|
* TODO: Currently hardcoded value, as the old variant was stupid
|
|
|
|
* Create a working variant doing this->execute(query->renderCount())...
|
|
|
|
*
|
|
|
|
* @param Query $query the query object
|
|
|
|
*
|
|
|
|
* @return int
|
|
|
|
*/
|
2013-06-07 11:44:37 +02:00
|
|
|
public function count(Query $query)
|
|
|
|
{
|
2014-11-16 15:20:12 +01:00
|
|
|
return 100;
|
2013-06-07 11:44:37 +02:00
|
|
|
$count = clone($query);
|
|
|
|
$count->count();
|
2013-07-12 13:41:48 +02:00
|
|
|
Benchmark::measure('Sending Livestatus Count Query');
|
|
|
|
$data = $this->doFetch((string) $count);
|
|
|
|
Benchmark::measure('Got Livestatus count result');
|
2013-06-07 11:44:37 +02:00
|
|
|
return $data[0][0];
|
|
|
|
}
|
|
|
|
|
2014-11-16 15:33:17 +01:00
|
|
|
/**
|
|
|
|
* Fetch a single row
|
|
|
|
*
|
|
|
|
* TODO: Currently based on fetchAll, that's bullshit
|
|
|
|
*
|
|
|
|
* @param Query $query the query object
|
|
|
|
*
|
|
|
|
* @return object the first result row
|
|
|
|
*/
|
|
|
|
public function fetchRow(Query $query)
|
|
|
|
{
|
|
|
|
$all = $this->fetchAll($query);
|
|
|
|
return array_shift($all);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Fetch key/value pairs
|
|
|
|
*
|
|
|
|
* TODO: Currently slow, needs improvement
|
|
|
|
*
|
|
|
|
* @param Query $query the query object
|
|
|
|
*
|
|
|
|
* @return array
|
|
|
|
*/
|
|
|
|
public function fetchPairs(Query $query)
|
|
|
|
{
|
|
|
|
$res = array();
|
|
|
|
$all = $this->fetchAll($query);
|
|
|
|
foreach ($all as $row) {
|
|
|
|
// slow
|
|
|
|
$keys = array_keys((array) $row);
|
|
|
|
$res[$row->{$keys[0]}] = $row->{$keys[1]};
|
|
|
|
}
|
|
|
|
return $res;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Fetch all result rows
|
|
|
|
*
|
|
|
|
* @param Query $query the query object
|
|
|
|
*
|
|
|
|
* @return array
|
|
|
|
*/
|
2013-06-07 11:44:37 +02:00
|
|
|
public function fetchAll(Query $query)
|
|
|
|
{
|
2013-07-12 13:41:48 +02:00
|
|
|
Benchmark::measure('Sending Livestatus Query');
|
|
|
|
$data = $this->doFetch((string) $query);
|
|
|
|
Benchmark::measure('Got Livestatus Data');
|
2013-06-07 11:44:37 +02:00
|
|
|
if ($query->hasColumns()) {
|
|
|
|
$headers = $query->getColumnAliases();
|
|
|
|
} else {
|
|
|
|
$headers = array_shift($data);
|
|
|
|
}
|
|
|
|
$result = array();
|
|
|
|
foreach ($data as $row) {
|
|
|
|
$result_row = & $result[];
|
|
|
|
$result_row = (object) array();
|
|
|
|
foreach ($row as $key => $val) {
|
|
|
|
$result_row->{$headers[$key]} = $val;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if ($query->hasOrder()) {
|
|
|
|
usort($result, array($query, 'compare'));
|
|
|
|
}
|
|
|
|
if ($query->hasLimit()) {
|
|
|
|
$result = array_slice(
|
|
|
|
$result,
|
|
|
|
$query->getOffset(),
|
|
|
|
$query->getLimit()
|
|
|
|
);
|
|
|
|
}
|
2013-07-12 13:41:48 +02:00
|
|
|
Benchmark::measure('Data sorted, limits applied');
|
2013-09-04 18:27:16 +02:00
|
|
|
|
2013-06-07 11:44:37 +02:00
|
|
|
return $result;
|
|
|
|
}
|
|
|
|
|
2013-07-12 13:41:48 +02:00
|
|
|
protected function doFetch($raw_query)
|
2013-06-07 11:44:37 +02:00
|
|
|
{
|
|
|
|
$conn = $this->getConnection();
|
|
|
|
$this->writeToSocket($raw_query);
|
|
|
|
$header = $this->readFromSocket(16);
|
|
|
|
$status = (int) substr($header, 0, 3);
|
|
|
|
$length = (int) trim(substr($header, 4));
|
|
|
|
$body = $this->readFromSocket($length);
|
|
|
|
if ($status !== 200) {
|
2014-08-27 16:03:15 +02:00
|
|
|
throw new IcingaException(
|
|
|
|
'Problem while reading %d bytes from livestatus: %s',
|
|
|
|
$length,
|
|
|
|
$body
|
2013-07-12 13:41:48 +02:00
|
|
|
);
|
2013-06-07 11:44:37 +02:00
|
|
|
}
|
|
|
|
$result = json_decode($body);
|
|
|
|
if ($result === null) {
|
2014-08-27 16:03:15 +02:00
|
|
|
throw new IcingaException('Got invalid response body from livestatus');
|
2013-06-07 11:44:37 +02:00
|
|
|
}
|
2013-09-04 18:27:16 +02:00
|
|
|
|
2013-06-07 11:44:37 +02:00
|
|
|
return $result;
|
|
|
|
}
|
|
|
|
|
|
|
|
protected function readFromSocket($length)
|
|
|
|
{
|
|
|
|
$offset = 0;
|
|
|
|
$buffer = '';
|
2013-09-04 18:27:16 +02:00
|
|
|
|
2013-07-12 13:41:48 +02:00
|
|
|
while ($offset < $length) {
|
2013-06-07 11:44:37 +02:00
|
|
|
$data = socket_read($this->connection, $length - $offset);
|
|
|
|
if ($data === false) {
|
2014-08-27 16:03:15 +02:00
|
|
|
throw new IcingaException(
|
|
|
|
'Failed to read from livestatus socket: %s',
|
|
|
|
socket_strerror(socket_last_error($this->connection))
|
2013-07-12 13:41:48 +02:00
|
|
|
);
|
2013-06-07 11:44:37 +02:00
|
|
|
}
|
|
|
|
$size = strlen($data);
|
|
|
|
$offset += $size;
|
|
|
|
$buffer .= $data;
|
2013-09-04 18:27:16 +02:00
|
|
|
|
2013-06-07 11:44:37 +02:00
|
|
|
if ($size === 0) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if ($offset !== $length) {
|
2014-08-27 16:03:15 +02:00
|
|
|
throw new IcingaException(
|
|
|
|
'Got only %d instead of %d bytes from livestatus socket',
|
|
|
|
$offset,
|
|
|
|
$length
|
2013-07-12 13:41:48 +02:00
|
|
|
);
|
2013-06-07 11:44:37 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
return $buffer;
|
|
|
|
}
|
|
|
|
|
|
|
|
protected function writeToSocket($data)
|
|
|
|
{
|
|
|
|
$res = socket_write($this->connection, $data);
|
|
|
|
if ($res === false) {
|
2014-08-27 16:03:15 +02:00
|
|
|
throw new IcingaException('Writing to livestatus socket failed');
|
2013-06-07 11:44:37 +02:00
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
protected function assertPhpExtensionLoaded($name)
|
|
|
|
{
|
|
|
|
if (! extension_loaded($name)) {
|
2014-08-27 16:03:15 +02:00
|
|
|
throw new IcingaException(
|
|
|
|
'The extension "%s" is not loaded',
|
|
|
|
$name
|
2013-07-12 13:41:48 +02:00
|
|
|
);
|
2013-06-07 11:44:37 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
protected function getConnection()
|
|
|
|
{
|
|
|
|
if ($this->connection === null) {
|
|
|
|
if ($this->socket_type === self::TYPE_TCP) {
|
|
|
|
$this->establishTcpConnection();
|
|
|
|
} else {
|
|
|
|
$this->establishSocketConnection();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return $this->connection;
|
|
|
|
}
|
|
|
|
|
|
|
|
protected function establishTcpConnection()
|
|
|
|
{
|
|
|
|
// TODO: find a bedder place for this
|
|
|
|
if (! defined('TCP_NODELAY')) {
|
|
|
|
define('TCP_NODELAY', 1);
|
|
|
|
}
|
2013-09-04 18:27:16 +02:00
|
|
|
|
2013-06-07 11:44:37 +02:00
|
|
|
$this->connection = socket_create(AF_INET, SOCK_STREAM, SOL_TCP);
|
|
|
|
if (! @socket_connect($this->connection, $this->socket_host, $this->socket_port)) {
|
2014-08-27 16:03:15 +02:00
|
|
|
throw new IcingaException(
|
|
|
|
'Cannot connect to livestatus TCP socket "%s:%d": %s',
|
|
|
|
$this->socket_host,
|
|
|
|
$this->socket_port,
|
|
|
|
socket_strerror(socket_last_error($this->connection))
|
2013-07-12 13:41:48 +02:00
|
|
|
);
|
2013-06-07 11:44:37 +02:00
|
|
|
}
|
|
|
|
socket_set_option($this->connection, SOL_TCP, TCP_NODELAY, 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
protected function establishSocketConnection()
|
|
|
|
{
|
2013-07-12 13:41:48 +02:00
|
|
|
$this->connection = socket_create(AF_UNIX, SOCK_STREAM, 0);
|
2013-06-07 11:44:37 +02:00
|
|
|
if (! socket_connect($this->connection, $this->socket_path)) {
|
2014-08-27 16:03:15 +02:00
|
|
|
throw new IcingaException(
|
|
|
|
'Cannot connect to livestatus local socket "%s"',
|
|
|
|
$this->socket_path
|
2013-07-12 13:41:48 +02:00
|
|
|
);
|
2013-06-07 11:44:37 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-04-16 16:41:23 +02:00
|
|
|
public function connect()
|
|
|
|
{
|
|
|
|
if (!$this->connection) {
|
|
|
|
$this->getConnection();
|
|
|
|
}
|
|
|
|
|
|
|
|
return $this;
|
|
|
|
}
|
|
|
|
|
2013-06-07 11:44:37 +02:00
|
|
|
public function disconnect()
|
|
|
|
{
|
|
|
|
if ($this->connection) {
|
|
|
|
socket_close($this->connection);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public function __destruct()
|
|
|
|
{
|
|
|
|
$this->disconnect();
|
|
|
|
}
|
|
|
|
}
|