456 lines
14 KiB
PHP
Executable File
456 lines
14 KiB
PHP
Executable File
<?php
|
|
// {{{ICINGA_LICENSE_HEADER}}}
|
|
/**
|
|
* This file is part of Icinga 2 Web.
|
|
*
|
|
* Icinga 2 Web - Head for multiple monitoring backends.
|
|
* Copyright (C) 2013 Icinga Development Team
|
|
*
|
|
* 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 2
|
|
* 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, write to the Free Software
|
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
|
*
|
|
* @copyright 2013 Icinga Development Team <info@icinga.org>
|
|
* @license http://www.gnu.org/licenses/gpl-2.0.txt GPL, version 2
|
|
* @author Icinga Development Team <info@icinga.org>
|
|
*/
|
|
// {{{ICINGA_LICENSE_HEADER}}}
|
|
|
|
namespace Icinga\Protocol\Statusdat;
|
|
|
|
use Icinga\Application\Logger;
|
|
use Icinga\Exception\ConfigurationError;
|
|
use Icinga\Exception\ProgrammingError;
|
|
use Icinga\Protocol\Statusdat\Exception\ParsingException as ParsingException;
|
|
|
|
/**
|
|
* Status.dat and object.cache parser implementation
|
|
*/
|
|
class Parser
|
|
{
|
|
/**
|
|
* An array of objects that couldn't be resolved yet due to missing dependencies
|
|
*
|
|
* @var array
|
|
*/
|
|
private $deferred = array();
|
|
|
|
/**
|
|
* The resource pointing to the currently read file
|
|
*
|
|
* @var resource
|
|
*/
|
|
private $filehandle;
|
|
|
|
/**
|
|
* String representation of the currently parsed object type
|
|
*
|
|
* @var string
|
|
*/
|
|
private $currentObjectType;
|
|
|
|
/**
|
|
* The current state type (host, service)
|
|
*
|
|
* @var string
|
|
*/
|
|
private $currentStateType;
|
|
|
|
/**
|
|
* The internal representation of the icinga statue
|
|
*
|
|
* @var array
|
|
*/
|
|
private $icingaState;
|
|
|
|
/**
|
|
* The current line being read
|
|
*
|
|
* @var int
|
|
*/
|
|
private $lineCtr = 0;
|
|
|
|
/**
|
|
* Create a new parser using the given file
|
|
*
|
|
* @param resource $filehandle The file handle to usefor parsing
|
|
* @param array $baseState The state using for the base
|
|
*
|
|
* @throws ConfigurationError When the file can't be used
|
|
*/
|
|
public function __construct($filehandle = null, $baseState = null)
|
|
{
|
|
if (!is_resource($filehandle)) {
|
|
throw new ConfigurationError("Statusdat parser can't find $filehandle");
|
|
}
|
|
|
|
$this->filehandle = $filehandle;
|
|
$this->icingaState = $baseState;
|
|
}
|
|
|
|
/**
|
|
* Parse the given file handle as an objects file and read object information
|
|
*/
|
|
public function parseObjectsFile()
|
|
{
|
|
$DEFINE = strlen("define ");
|
|
$filehandle = $this->filehandle;
|
|
$this->icingaState = array();
|
|
while (!feof($filehandle)) {
|
|
|
|
$line = trim(fgets($filehandle));
|
|
|
|
$this->lineCtr++;
|
|
if ($line === "" || $line[0] === "#") {
|
|
continue;
|
|
}
|
|
$this->currentObjectType = trim(substr($line, $DEFINE, -1));
|
|
if (!isset($this->icingaState[$this->currentObjectType])) {
|
|
$this->icingaState[$this->currentObjectType] = array();
|
|
}
|
|
$this->readCurrentObject();
|
|
}
|
|
$this->processDeferred();
|
|
}
|
|
|
|
/**
|
|
* Parse the given file handle as an status.dat file and read runtime information
|
|
*/
|
|
public function parseRuntimeState($filehandle = null)
|
|
{
|
|
if ($filehandle != null) {
|
|
$this->filehandle = $filehandle;
|
|
} else {
|
|
$filehandle = $this->filehandle;
|
|
}
|
|
|
|
if (!$this->icingaState) {
|
|
throw new ProgrammingError("Tried to read runtime state without existing objects data");
|
|
}
|
|
$this->overwrites = array();
|
|
while (!feof($filehandle)) {
|
|
$line = trim(fgets($filehandle));
|
|
$this->lineCtr++;
|
|
if ($line === "" || $line[0] === "#") {
|
|
continue;
|
|
}
|
|
$this->currentStateType = trim(substr($line, 0, -1));
|
|
$this->readCurrentState();
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Read the next object from the object.cache file handle
|
|
*
|
|
* @throws ParsingException
|
|
*/
|
|
private function readCurrentObject()
|
|
{
|
|
$filehandle = $this->filehandle;
|
|
$monitoringObject = new PrintableObject();
|
|
while (!feof($filehandle)) {
|
|
$line = explode("\t", trim(fgets($filehandle)), 2);
|
|
$this->lineCtr++;
|
|
if (!$line) {
|
|
continue;
|
|
}
|
|
|
|
// End of object
|
|
if ($line[0] === "}") {
|
|
$this->registerObject($monitoringObject);
|
|
return;
|
|
}
|
|
if (!isset($line[1])) {
|
|
$line[1] = "";
|
|
}
|
|
$monitoringObject->{$line[0]} = trim($line[1]);
|
|
}
|
|
throw new ParsingException("Unexpected EOF in objects.cache, line " . $this->lineCtr);
|
|
}
|
|
|
|
/**
|
|
* Read the next state from the status.dat file handler
|
|
*
|
|
* @throws Exception\ParsingException
|
|
*/
|
|
private function readCurrentState()
|
|
{
|
|
$filehandle = $this->filehandle;
|
|
$statusdatObject = new RuntimeStateContainer();
|
|
|
|
$objectType = $this->getObjectTypeForState();
|
|
|
|
if ($objectType != "host" && $objectType != "service") {
|
|
$this->skipObject(); // ignore unknown objects
|
|
return;
|
|
}
|
|
if (!isset($this->icingaState[$this->currentObjectType])) {
|
|
throw new ParsingException("No $this->currentObjectType objects registered in objects.cache");
|
|
}
|
|
$base = & $this->icingaState[$this->currentObjectType];
|
|
$state = $this->skipObject(true);
|
|
$statusdatObject->runtimeState = & $state;
|
|
$name = $this->getObjectIdentifier($statusdatObject);
|
|
|
|
if (!isset($base[$name])) {
|
|
throw new ParsingException(
|
|
"Unknown object $name " . $this->currentObjectType . " - "
|
|
. print_r(
|
|
$statusdatObject,
|
|
true
|
|
)
|
|
. "\n" . print_r($base, true)
|
|
);
|
|
}
|
|
$type = substr($this->currentStateType, strlen($objectType));
|
|
|
|
if ($type == "status") {
|
|
// directly set the status to the status field of the given object
|
|
$base[$name]->status = & $statusdatObject;
|
|
} else {
|
|
if (!isset($base[$name]->$type) || !in_array($base[$name]->$type, $this->overwrites)) {
|
|
$base[$name]->$type = array();
|
|
$this->overwrites[] = & $base[$name]->$type;
|
|
}
|
|
array_push($base[$name]->$type, $statusdatObject);
|
|
$this->currentObjectType = $type;
|
|
if (!isset($this->icingaState[$type])) {
|
|
$this->icingaState[$type] = array();
|
|
}
|
|
$this->icingaState[$type][] = &$statusdatObject;
|
|
$id = $this->getObjectIdentifier($statusdatObject);
|
|
if ($id !== false && isset($this->icingaState[$objectType][$id])) {
|
|
$statusdatObject->$objectType = $this->icingaState[$objectType][$id];
|
|
}
|
|
}
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
/**
|
|
* Get the corresponding object type name for the given state
|
|
*
|
|
* @return string
|
|
*/
|
|
private function getObjectTypeForState()
|
|
{
|
|
$pos = strpos($this->currentStateType, "service");
|
|
|
|
if ($pos === false) {
|
|
$pos = strpos($this->currentStateType, "host");
|
|
} else {
|
|
$this->currentObjectType = "service";
|
|
return "service";
|
|
}
|
|
|
|
if ($pos === false) {
|
|
return $this->currentStateType;
|
|
} else {
|
|
$this->currentObjectType = "host";
|
|
return "host";
|
|
}
|
|
|
|
return $this->currentObjectType;
|
|
}
|
|
|
|
/**
|
|
* Skip the current object definition
|
|
*
|
|
* @param bool $returnString If true, the object string will be returned
|
|
* @return string The skipped object if $returnString is true
|
|
*/
|
|
protected function skipObject($returnString = false)
|
|
{
|
|
if (!$returnString) {
|
|
while (trim(fgets($this->filehandle)) !== "}") {
|
|
}
|
|
return null;
|
|
} else {
|
|
$str = "";
|
|
while (($val = trim(fgets($this->filehandle))) !== "}") {
|
|
$str .= $val . "\n";
|
|
}
|
|
return $str;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Register the given object in the icinga state
|
|
*
|
|
* @param object $object The monitoring object to register
|
|
*/
|
|
protected function registerObject(&$object)
|
|
{
|
|
|
|
$name = $this->getObjectIdentifier($object);
|
|
if ($name !== false) {
|
|
$this->icingaState[$this->currentObjectType][$name] = &$object;
|
|
}
|
|
$this->registerObjectAsProperty($object);
|
|
}
|
|
|
|
/**
|
|
* Register the given object as a property in related objects
|
|
*
|
|
* This registers for example hosts underneath their hostgroup and vice cersa
|
|
*
|
|
* @param object $object The object to register as a property
|
|
*/
|
|
protected function registerObjectAsProperty(&$object)
|
|
{
|
|
if ($this->currentObjectType == 'service'
|
|
|| $this->currentObjectType == 'host'
|
|
|| $this->currentObjectType == 'contact') {
|
|
return null;
|
|
}
|
|
$isService = strpos($this->currentObjectType, "service") !== false;
|
|
$isHost = strpos($this->currentObjectType, "host") !== false;
|
|
$isContact = strpos($this->currentObjectType, "contact") !== false;
|
|
$name = $this->getObjectIdentifier($object);
|
|
|
|
if ($isService === false && $isHost === false && $isContact === false) { // this would be error in the parser implementation
|
|
return null;
|
|
}
|
|
$property = $this->currentObjectType;
|
|
if ($isService) {
|
|
$this->currentObjectType = "service";
|
|
$property = substr($property, strlen("service"));
|
|
} elseif ($isHost) {
|
|
$this->currentObjectType = "host";
|
|
$property = substr($property, strlen("host"));
|
|
} elseif ($isContact) {
|
|
$this->currentObjectType = "contact";
|
|
$property = substr($property, strlen("contact"));
|
|
}
|
|
|
|
if (!isset($this->icingaState[$this->currentObjectType])) {
|
|
return $this->deferRegistration($object, $this->currentObjectType . $property);
|
|
}
|
|
|
|
// @TODO: Clean up, this differates between 1:n and 1:1 references
|
|
if (strpos($property, "group") !== false) {
|
|
$sourceIdentifier = $this->getMembers($object);
|
|
foreach ($sourceIdentifier as $id) {
|
|
$source = $this->icingaState[$this->currentObjectType][$id];
|
|
if (!isset($source->$property)) {
|
|
$source->$property = array();
|
|
}
|
|
$type = $this->currentObjectType;
|
|
if (!isset($object->$type)) {
|
|
$object->$type = array();
|
|
}
|
|
// add the member to the group object
|
|
array_push($object->$type, $source);
|
|
// add the group to the member object
|
|
array_push($source->$property, $name);
|
|
}
|
|
} else {
|
|
$source = $this->icingaState[$this->currentObjectType][$this->getObjectIdentifier($object)];
|
|
if (!isset($source->$property)) {
|
|
$source->$property = array();
|
|
}
|
|
|
|
array_push($source->$property, $object);
|
|
}
|
|
|
|
return null;
|
|
}
|
|
|
|
/**
|
|
* Defer registration of the given object
|
|
*
|
|
* @param object $object The object to defer
|
|
* @param String $objType The name of the object type
|
|
*/
|
|
protected function deferRegistration($object, $objType)
|
|
{
|
|
$this->deferred[] = array($object, $objType);
|
|
}
|
|
|
|
/**
|
|
* Process deferred objects
|
|
*/
|
|
protected function processDeferred()
|
|
{
|
|
foreach ($this->deferred as $obj) {
|
|
$this->currentObjectType = $obj[1];
|
|
$this->registerObjectAsProperty($obj[0]);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Return the resolved members directive of an object
|
|
*
|
|
* @param object $object The object to get the members from
|
|
* @return array An array of member names
|
|
*/
|
|
protected function getMembers(&$object)
|
|
{
|
|
if (!isset($object->members)) {
|
|
return array();
|
|
}
|
|
|
|
$members = explode(",", $object->members);
|
|
|
|
if ($this->currentObjectType == "service") {
|
|
$res = array();
|
|
for ($i = 0; $i < count($members); $i += 2) {
|
|
$res[] = $members[$i] . ";" . $members[$i + 1];
|
|
}
|
|
return $res;
|
|
} else {
|
|
return $members;
|
|
}
|
|
|
|
}
|
|
|
|
/**
|
|
* Return the unique name of the given object
|
|
*
|
|
* @param object $object The object to retrieve the name from
|
|
* @return string The name of the object or null if no name can be retrieved
|
|
*/
|
|
protected function getObjectIdentifier(&$object)
|
|
{
|
|
if ($this->currentObjectType == 'contact') {
|
|
return $object->contact_name;
|
|
}
|
|
|
|
if ($this->currentObjectType == "service") {
|
|
return $object->host_name . ";" . $object->service_description;
|
|
}
|
|
$name = $this->currentObjectType . "_name";
|
|
if (isset($object->{$name})) {
|
|
return $object->{$name};
|
|
}
|
|
if (isset($object->service_description)) {
|
|
return $object->host_name . ";" . $object->service_description;
|
|
} elseif (isset($object->host_name)) {
|
|
return $object->host_name;
|
|
}
|
|
return null;
|
|
|
|
}
|
|
|
|
/**
|
|
* Return the internal state of the parser
|
|
*
|
|
* @return null
|
|
*/
|
|
public function getRuntimeState()
|
|
{
|
|
return $this->icingaState;
|
|
}
|
|
}
|