diff --git a/application/controllers/AuthenticationController.php b/application/controllers/AuthenticationController.php
index 402ad3411..662ad6143 100644
--- a/application/controllers/AuthenticationController.php
+++ b/application/controllers/AuthenticationController.php
@@ -122,7 +122,7 @@ class AuthenticationController extends ActionController
if ($backendsWithError) {
$this->view->form->getElement('username')->addError(
$this->translate(
- 'Please note that not all authentication methods where available.'
+ 'Please note that not all authentication methods were available.'
. ' Check the system log or Icinga Web 2 log for more information.'
)
);
@@ -135,7 +135,9 @@ class AuthenticationController extends ActionController
$authenticated = $backend->authenticate($user);
if ($authenticated === true) {
$auth->setAuthenticated($user);
- $this->rerenderLayout()->redirectNow($redirectUrl);
+ $this->rerenderLayout()->redirectNow(
+ Url::fromPath(Url::fromRequest()->getParam('redirect', 'dashboard'))
+ );
}
}
}
@@ -151,14 +153,16 @@ class AuthenticationController extends ActionController
public function logoutAction()
{
$auth = $this->Auth();
+ if (! $auth->isAuthenticated()) {
+ $this->redirectToLogin();
+ }
$isRemoteUser = $auth->getUser()->isRemoteUser();
$auth->removeAuthorization();
-
if ($isRemoteUser === true) {
$this->_helper->layout->setLayout('login');
$this->_response->setHttpResponseCode(401);
} else {
- $this->rerenderLayout()->redirectToLogin();
+ $this->redirectToLogin();
}
}
}
diff --git a/application/forms/Authentication/LoginForm.php b/application/forms/Authentication/LoginForm.php
index d878475f2..0afc56749 100644
--- a/application/forms/Authentication/LoginForm.php
+++ b/application/forms/Authentication/LoginForm.php
@@ -24,7 +24,7 @@ class LoginForm extends Form
/**
* @see Form::createElements()
*/
- public function createElements($formData)
+ public function createElements(array $formData)
{
$this->addElement(
'text',
diff --git a/application/forms/Config/Resource/StatusdatResourceForm.php b/application/forms/Config/Resource/StatusdatResourceForm.php
deleted file mode 100644
index d440f139d..000000000
--- a/application/forms/Config/Resource/StatusdatResourceForm.php
+++ /dev/null
@@ -1,63 +0,0 @@
-setName('form_config_resource_statusdat');
- }
-
- /**
- * @see Form::createElements()
- */
- public function createElements(array $formData)
- {
- $this->addElement(
- 'text',
- 'name',
- array(
- 'required' => true,
- 'label' => t('Resource Name'),
- 'description' => t('The unique name of this resource')
- )
- );
- $this->addElement(
- 'text',
- 'status_file',
- array(
- 'required' => true,
- 'label' => t('Filepath'),
- 'description' => t('Location of your icinga status.dat file'),
- 'value' => realpath(Icinga::app()->getApplicationDir() . '/../var/status.dat'),
- 'validators' => array(new ReadablePathValidator())
- )
- );
- $this->addElement(
- 'text',
- 'object_file',
- array(
- 'required' => true,
- 'label' => t('Filepath'),
- 'description' => t('Location of your icinga objects.cache file'),
- 'value' => realpath(Icinga::app()->getApplicationDir() . '/../var/objects.cache'),
- 'validators' => array(new ReadablePathValidator())
- )
- );
-
- return $this;
- }
-}
diff --git a/application/forms/Config/ResourceConfigForm.php b/application/forms/Config/ResourceConfigForm.php
index 9624b9e6a..e4e561ed8 100644
--- a/application/forms/Config/ResourceConfigForm.php
+++ b/application/forms/Config/ResourceConfigForm.php
@@ -12,7 +12,6 @@ use Icinga\Form\Config\Resource\DbResourceForm;
use Icinga\Form\Config\Resource\FileResourceForm;
use Icinga\Form\Config\Resource\LdapResourceForm;
use Icinga\Form\Config\Resource\LivestatusResourceForm;
-use Icinga\Form\Config\Resource\StatusdatResourceForm;
use Icinga\Application\Platform;
use Icinga\Exception\ConfigurationError;
@@ -42,8 +41,6 @@ class ResourceConfigForm extends ConfigForm
return new LdapResourceForm();
} elseif ($type === 'livestatus') {
return new LivestatusResourceForm();
- } elseif ($type === 'statusdat') {
- return new StatusdatResourceForm();
} elseif ($type === 'file') {
return new FileResourceForm();
} else {
@@ -214,7 +211,6 @@ class ResourceConfigForm extends ConfigForm
$resourceTypes = array(
'file' => t('File'),
- 'statusdat' => 'Status.dat',
'livestatus' => 'Livestatus',
);
if ($resourceType === 'ldap' || Platform::extensionLoaded('ldap')) {
diff --git a/config/authentication.ini.in b/config/authentication.ini.in
index 71d9e4402..2a2d2a969 100644
--- a/config/authentication.ini.in
+++ b/config/authentication.ini.in
@@ -18,6 +18,10 @@ backend = ldap
resource = internal_ldap
user_class = @ldap_user_objectclass@
user_name_attribute = @ldap_attribute_username@
+group_base_dn = @ldap_group_base_dn@
+group_attribute = @ldap_group_attribute@
+group_member_attribute = @ldap_group_member_attribute@
+group_class = @ldap_group_class@
[internal_db_authentication]
@internal_auth_disabled@
diff --git a/library/Icinga/Authentication/AdmissionLoader.php b/library/Icinga/Authentication/AdmissionLoader.php
index f961fb87c..098afda81 100644
--- a/library/Icinga/Authentication/AdmissionLoader.php
+++ b/library/Icinga/Authentication/AdmissionLoader.php
@@ -52,12 +52,8 @@ class AdmissionLoader
return $permissions;
}
foreach ($config as $section) {
- if ($this->match($section, $username, $groups)) {
- foreach ($section as $key => $value) {
- if (strpos($key, 'permission') === 0) {
- $permissions = array_merge($permissions, String::trimSplit($value));
- }
- }
+ if ($this->match($section, $username, $groups) && isset($section->permissions)) {
+ $permissions += String::trimSplit($section->permissions);
}
}
return $permissions;
@@ -79,12 +75,12 @@ class AdmissionLoader
} catch (NotReadableError $e) {
return $restrictions;
}
- foreach ($config as $section) {
+ foreach ($config as $name => $section) {
if ($this->match($section, $username, $groups)) {
if (!array_key_exists($section->name, $restrictions)) {
$restrictions[$section->name] = array();
}
- $restrictions[$section->name][] = $section->restriction;
+ $restrictions[$section->name][$name] = $section->restriction;
}
}
return $restrictions;
diff --git a/library/Icinga/Authentication/Backend/LdapUserBackend.php b/library/Icinga/Authentication/Backend/LdapUserBackend.php
index 8e5307fc8..0bf503dee 100644
--- a/library/Icinga/Authentication/Backend/LdapUserBackend.php
+++ b/library/Icinga/Authentication/Backend/LdapUserBackend.php
@@ -4,6 +4,7 @@
namespace Icinga\Authentication\Backend;
+use Icinga\Logger\Logger;
use Icinga\User;
use Icinga\Authentication\UserBackend;
use Icinga\Protocol\Ldap\Connection;
@@ -23,11 +24,14 @@ class LdapUserBackend extends UserBackend
protected $userNameAttribute;
- public function __construct(Connection $conn, $userClass, $userNameAttribute)
+ protected $groupOptions;
+
+ public function __construct(Connection $conn, $userClass, $userNameAttribute, $groupOptions = null)
{
$this->conn = $conn;
$this->userClass = $userClass;
$this->userNameAttribute = $userNameAttribute;
+ $this->groupOptions = $groupOptions;
}
/**
@@ -83,6 +87,43 @@ class LdapUserBackend extends UserBackend
}
}
+ /**
+ * Retrieve the user groups
+ *
+ * @TODO: Subject to change, see #7343
+ *
+ * @param string $dn
+ *
+ * @return array|null
+ */
+ public function getGroups($dn)
+ {
+ if (empty($this->groupOptions) || ! isset($this->groupOptions['group_base_dn'])) {
+ return null;
+ }
+
+ $q = $this->conn->select()
+ ->setBase($this->groupOptions['group_base_dn'])
+ ->from(
+ $this->groupOptions['group_class'],
+ array($this->groupOptions['group_attribute'])
+ )
+ ->where(
+ $this->groupOptions['group_member_attribute'],
+ $dn
+ );
+
+ $result = $this->conn->fetchAll($q);
+
+ $groups = array();
+
+ foreach ($result as $group) {
+ $groups[] = $group->{$this->groupOptions['group_attribute']};
+ }
+
+ return $groups;
+ }
+
/**
* Test whether the given user exists
*
@@ -127,10 +168,15 @@ class LdapUserBackend extends UserBackend
return false;
}
try {
- return $this->conn->testCredentials(
- $this->conn->fetchDN($this->createQuery($user->getUsername())),
+ $userDn = $this->conn->fetchDN($this->createQuery($user->getUsername()));
+ $authenticated = $this->conn->testCredentials(
+ $userDn,
$password
);
+ if ($authenticated) {
+ $user->setGroups($this->getGroups($userDn));
+ }
+ return $authenticated;
} catch (LdapException $e) {
// Error during authentication of this specific user
throw new AuthenticationException(
@@ -177,4 +223,3 @@ class LdapUserBackend extends UserBackend
return $users;
}
}
-
diff --git a/library/Icinga/Authentication/UserBackend.php b/library/Icinga/Authentication/UserBackend.php
index 9e7abd62a..7829210fd 100644
--- a/library/Icinga/Authentication/UserBackend.php
+++ b/library/Icinga/Authentication/UserBackend.php
@@ -93,26 +93,44 @@ abstract class UserBackend implements Countable
$backend = new DbUserBackend($resource);
break;
case 'msldap':
+ $groupOptions = array(
+ 'group_base_dn' => $backendConfig->group_base_dn,
+ 'group_attribute' => $backendConfig->group_attribute,
+ 'group_member_attribute' => $backendConfig->group_member_attribute,
+ 'group_class' => $backendConfig->group_class
+ );
$backend = new LdapUserBackend(
$resource,
$backendConfig->get('user_class', 'user'),
- $backendConfig->get('user_name_attribute', 'sAMAccountName')
+ $backendConfig->get('user_name_attribute', 'sAMAccountName'),
+ $groupOptions
);
break;
case 'ldap':
- if (($userClass = $backendConfig->user_class) === null) {
+ if ($backendConfig->user_class === null) {
throw new ConfigurationError(
'Authentication configuration for backend "%s" is missing the user_class directive',
$name
);
}
- if (($userNameAttribute = $backendConfig->user_name_attribute) === null) {
+ if ($backendConfig->user_name_attribute === null) {
throw new ConfigurationError(
'Authentication configuration for backend "%s" is missing the user_name_attribute directive',
$name
);
}
- $backend = new LdapUserBackend($resource, $userClass, $userNameAttribute);
+ $groupOptions = array(
+ 'group_base_dn' => $backendConfig->group_base_dn,
+ 'group_attribute' => $backendConfig->group_attribute,
+ 'group_member_attribute' => $backendConfig->group_member_attribute,
+ 'group_class' => $backendConfig->group_class
+ );
+ $backend = new LdapUserBackend(
+ $resource,
+ $backendConfig->user_class,
+ $backendConfig->user_name_attribute,
+ $groupOptions
+ );
break;
default:
throw new ConfigurationError(
diff --git a/library/Icinga/Cli/Params.php b/library/Icinga/Cli/Params.php
index 7236dcca2..5c6cbfe8b 100644
--- a/library/Icinga/Cli/Params.php
+++ b/library/Icinga/Cli/Params.php
@@ -108,6 +108,18 @@ class Params
return $this->standalone;
}
+ /**
+ * Support isset() and empty() checks on options
+ *
+ * @param $name
+ *
+ * @return bool
+ */
+ public function __isset($name)
+ {
+ return isset($this->params[$name]);
+ }
+
/**
* @see Params::get()
*/
diff --git a/library/Icinga/Data/Db/DbQuery.php b/library/Icinga/Data/Db/DbQuery.php
index b701f3ac0..94765d98f 100644
--- a/library/Icinga/Data/Db/DbQuery.php
+++ b/library/Icinga/Data/Db/DbQuery.php
@@ -59,6 +59,13 @@ class DbQuery extends SimpleQuery
*/
protected $count;
+ /**
+ * GROUP BY clauses
+ *
+ * @var string|array
+ */
+ protected $group;
+
protected function init()
{
$this->db = $this->ds->getDbAdapter();
@@ -89,17 +96,21 @@ class DbQuery extends SimpleQuery
public function getSelectQuery()
{
$select = $this->dbSelect();
-
// Add order fields to select for postgres distinct queries (#6351)
if ($this->hasOrder()
&& $this->getDatasource()->getDbType() === 'pgsql'
&& $select->getPart(Zend_Db_Select::DISTINCT) === true) {
foreach ($this->getOrder() as $fieldAndDirection) {
- list($alias, $field) = explode('.', $fieldAndDirection[0]);
- $this->columns[$field] = $fieldAndDirection[0];
+ if (array_search($fieldAndDirection[0], $this->columns) === false) {
+ $this->columns[] = $fieldAndDirection[0];
+ }
}
}
+ if ($this->group) {
+ $select->group($this->group);
+ }
+
$select->columns($this->columns);
$this->applyFilterSql($select);
@@ -117,7 +128,7 @@ class DbQuery extends SimpleQuery
return $select;
}
- protected function applyFilterSql($query)
+ public function applyFilterSql($query)
{
$where = $this->renderFilter($this->filter);
if ($where !== '') {
@@ -253,6 +264,7 @@ class DbQuery extends SimpleQuery
$this->applyFilterSql($count);
if ($this->useSubqueryCount) {
+ $count->columns($this->columns);
$columns = array('cnt' => 'COUNT(*)');
return $this->db->select()->from($count, $columns);
}
@@ -300,4 +312,17 @@ class DbQuery extends SimpleQuery
{
return (string) $this->getSelectQuery();
}
+
+ /**
+ * Add a GROUP BY clause
+ *
+ * @param string|array $group
+ *
+ * @return $this
+ */
+ public function group($group)
+ {
+ $this->group = $group;
+ return $this;
+ }
}
diff --git a/library/Icinga/Data/Filter/FilterExpression.php b/library/Icinga/Data/Filter/FilterExpression.php
index c29414166..8378b5ae3 100644
--- a/library/Icinga/Data/Filter/FilterExpression.php
+++ b/library/Icinga/Data/Filter/FilterExpression.php
@@ -12,6 +12,8 @@ class FilterExpression extends Filter
public function __construct($column, $sign, $expression)
{
+ $column = trim($column);
+ $expression = is_array($expression) ? array_map('trim', $expression) : trim($expression);
$this->column = $column;
$this->sign = $sign;
$this->expression = $expression;
diff --git a/library/Icinga/Data/ResourceFactory.php b/library/Icinga/Data/ResourceFactory.php
index 30597ad2d..e85413615 100644
--- a/library/Icinga/Data/ResourceFactory.php
+++ b/library/Icinga/Data/ResourceFactory.php
@@ -10,7 +10,6 @@ use Icinga\Util\ConfigAwareFactory;
use Icinga\Exception\ConfigurationError;
use Icinga\Data\Db\DbConnection;
use Icinga\Protocol\Livestatus\Connection as LivestatusConnection;
-use Icinga\Protocol\Statusdat\Reader as StatusdatReader;
use Icinga\Protocol\Ldap\Connection as LdapConnection;
use Icinga\Protocol\File\FileReader;
@@ -102,7 +101,7 @@ class ResourceFactory implements ConfigAwareFactory
*
* @param Zend_Config $config The configuration for the created resource.
*
- * @return DbConnection|LdapConnection|LivestatusConnection|StatusdatReader An objects that can be used to access
+ * @return DbConnection|LdapConnection|LivestatusConnection An object that can be used to access
* the given resource. The returned class depends on the configuration property 'type'.
* @throws ConfigurationError When an unsupported type is given
*/
@@ -115,9 +114,6 @@ class ResourceFactory implements ConfigAwareFactory
case 'ldap':
$resource = new LdapConnection($config);
break;
- case 'statusdat':
- $resource = new StatusdatReader($config);
- break;
case 'livestatus':
$resource = new LivestatusConnection($config->socket);
break;
@@ -137,7 +133,7 @@ class ResourceFactory implements ConfigAwareFactory
* Create a resource from name
*
* @param string $resourceName
- * @return DbConnection|LdapConnection|LivestatusConnection|StatusdatReader
+ * @return DbConnection|LdapConnection|LivestatusConnection
*/
public static function create($resourceName)
{
diff --git a/library/Icinga/Exception/QueryException.php b/library/Icinga/Exception/QueryException.php
new file mode 100644
index 000000000..beab3fe53
--- /dev/null
+++ b/library/Icinga/Exception/QueryException.php
@@ -0,0 +1,12 @@
+filename, $this->fields)
- );
+ return new FileIterator($this->filename, $this->fields);
}
/**
diff --git a/library/Icinga/Protocol/Statusdat/Exception/ParsingException.php b/library/Icinga/Protocol/Statusdat/Exception/ParsingException.php
deleted file mode 100644
index 0947e65bc..000000000
--- a/library/Icinga/Protocol/Statusdat/Exception/ParsingException.php
+++ /dev/null
@@ -1,13 +0,0 @@
-ref = & $obj;
- $this->reader = & $reader;
- }
-
- /**
- * @param $attribute
- * @return \stdClass
- */
- public function __get($attribute)
- {
- $exploded = explode(".", $attribute);
- $result = $this->ref;
-
- foreach ($exploded as $elem) {
- if (isset($result->$elem)) {
- $result = $result->$elem;
- } else {
- return null;
- }
- }
- return $result;
- }
-}
diff --git a/library/Icinga/Protocol/Statusdat/Parser.php b/library/Icinga/Protocol/Statusdat/Parser.php
deleted file mode 100644
index 5b3694518..000000000
--- a/library/Icinga/Protocol/Statusdat/Parser.php
+++ /dev/null
@@ -1,422 +0,0 @@
-file = $file;
- $this->icingaState = $baseState;
- }
-
- /**
- * Parse the given file handle as an objects file and read object information
- */
- public function parseObjectsFile()
- {
- $DEFINE = strlen('define ');
- $this->icingaState = array();
- foreach ($this->file as $line) {
- $line = trim($line);
- $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 as an status.dat file and read runtime information
- *
- * @param File $file The file to parse or null to parse the one passed to the constructor
- */
- public function parseRuntimeState(File $file = null)
- {
- if ($file != null) {
- $this->file = $file;
- } else {
- $file = $this->file;
- }
-
- if (!$this->icingaState) {
- throw new ProgrammingError('Tried to read runtime state without existing objects data');
- }
- $this->overwrites = array();
- foreach ($file as $line) {
- $line = trim($line);
- $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()
- {
- $monitoringObject = new PrintableObject();
- foreach ($this->file as $line) {
- $line = explode("\t", trim($line), 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()
- {
- $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($this->file->fgets()) !== '}') {
- }
- return null;
- } else {
- $str = '';
- while (($val = trim($this->file->fgets())) !== '}') {
- $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;
- }
-}
diff --git a/library/Icinga/Protocol/Statusdat/PrintableObject.php b/library/Icinga/Protocol/Statusdat/PrintableObject.php
deleted file mode 100644
index 840a7aa2a..000000000
--- a/library/Icinga/Protocol/Statusdat/PrintableObject.php
+++ /dev/null
@@ -1,20 +0,0 @@
-contact_name)) {
- return $this->contact_name;
- } elseif (isset($this->service_description)) {
- return $this->service_description;
- } elseif (isset($this->host_name)) {
- return $this->host_name;
- }
- return '';
- }
-}
diff --git a/library/Icinga/Protocol/Statusdat/Query.php b/library/Icinga/Protocol/Statusdat/Query.php
deleted file mode 100644
index f3b5ea85e..000000000
--- a/library/Icinga/Protocol/Statusdat/Query.php
+++ /dev/null
@@ -1,460 +0,0 @@
- array('host'),
- 'services' => array('service'),
- 'downtimes' => array('downtime'),
- 'groups' => array('hostgroup', 'servicegroup'),
- 'hostgroups' => array('hostgroup'),
- 'servicegroups' => array('servicegroup'),
- 'comments' => array('comment'),
- 'contacts' => array('contact'),
- 'contactgroups' => array('contactgroup')
- );
-
- /**
- * The current StatusDat query that will be applied upon calling fetchAll
- *
- * @var IQueryPart
- */
- private $queryFilter = null;
-
- /**
- * The current query source being used
- *
- * @var string
- */
- private $source = '';
-
- /**
- * An array containing all columns used for sorting
- *
- * @var array
- */
- protected $orderColumns = array();
-
- /**
- * An array containig all columns used for (simple) grouping
- *
- * @var array
- */
- private $groupColumns = array();
-
- /**
- * An optional function callback to use for more specific grouping
- *
- * @var array
- */
- private $groupByFn = null;
-
- /**
- * The scope index for the callback function
- */
- const FN_SCOPE = 0;
-
- /**
- * The name index for the callback function
- */
- const FN_NAME = 1;
-
- /**
- * Return true if columns are set for this query
- *
- * @return bool
- */
- public function hasColumns()
- {
- $columns = $this->getColumns();
- return !empty($columns);
- }
-
- /**
- * Set the status.dat specific IQueryPart filter to use
- *
- * @param IQueryPart $filter
- */
- public function setQueryFilter($filter)
- {
- $this->queryFilter = $filter;
- }
-
- /**
- * Order the query result by the given columns
- *
- * @param String|array $columns An array of columns to order by
- * @param String $dir The direction (asc or desc) in string form
- *
- * @return $this Fluent interface
- */
- public function order($columns, $dir = null, $isFunction = false)
- {
- if ($dir && strtolower($dir) == 'desc') {
- $dir = self::SORT_DESC;
- } else {
- $dir = self::SORT_ASC;
- }
- if (!is_array($columns)) {
- $columns = array($columns);
- }
-
- foreach ($columns as $col) {
- if (($pos = strpos($col, ' ')) !== false) {
- $dir = strtoupper(substr($col, $pos + 1));
- if ($dir === 'DESC') {
- $dir = self::SORT_DESC;
- } else {
- $dir = self::SORT_ASC;
- }
- $col = substr($col, 0, $pos);
- } else {
- $col = $col;
- }
-
- $this->orderColumns[] = array($col, $dir);
- }
- return $this;
- }
-
- /**
- * Order the query result using the callback to retrieve values for items
- *
- * @param array $columns A scope, function array to use for retrieving the values when ordering
- * @param String $dir The direction (asc or desc) in string form
- *
- * @return $this Fluent interface
- */
- public function orderByFn(array $callBack, $dir = null)
- {
- if ($dir && strtolower($dir) == 'desc') {
- $dir = self::SORT_DESC;
- } else {
- $dir = self::SORT_ASC;
- }
- $this->orderColumns[] = array($callBack, $dir);
- }
-
-
-
- /**
- * Set the query target
- *
- * @param String $table The table/target to select the query from
- * @param array $columns An array of attributes to use (required for fetchPairs())
- *
- * @return $this Fluent interface
- * @throws IcingaException If the target is unknonw
- */
- public function from($table, array $attributes = null)
- {
- if (!$this->getColumns() && $attributes) {
- $this->setColumns($attributes);
- }
- if (isset(self::$VALID_TARGETS[$table])) {
- $this->source = $table;
- } else {
- throw new IcingaException(
- 'Unknown from target for status.dat :%s',
- $table
- );
- }
- return $this;
- }
-
- /**
- * Return an index of all objects matching the filter of this query
- *
- * This index will be used for ordering, grouping and limiting
- */
- private function getFilteredIndices($classType = '\Icinga\Protocol\Statusdat\Query\Group')
- {
- $baseGroup = $this->queryFilter;
- $state = $this->ds->getState();
- $result = array();
- $source = self::$VALID_TARGETS[$this->source];
-
- foreach ($source as $target) {
-
- if (! isset($state[$target])) {
- continue;
- }
-
- $indexes = array_keys($state[$target]);
- if ($baseGroup) {
- $baseGroup->setQuery($this);
- $idx = array_keys($state[$target]);
- $indexes = $baseGroup->filter($state[$target], $idx);
- }
- if (!isset($result[$target])) {
- $result[$target] = $indexes;
- } else {
- array_merge($result[$target], $indexes);
- }
- }
- return $result;
- }
-
- /**
- * Order the given result set
- *
- * @param array $indices The result set of the query that should be ordered
- */
- private function orderIndices(array &$indices)
- {
- if (!empty($this->orderColumns)) {
- foreach ($indices as $type => &$subindices) {
- $this->currentType = $type;
- usort($subindices, array($this, 'orderResult'));
- }
- }
- }
-
- /**
- * Start a query
- *
- * This is just a dummy function to allow a more convenient syntax
- *
- * @return self Fluent interface
- */
- public function select()
- {
- return $this;
- }
-
- /**
- * Order implementation called by usort
- *
- * @param String $a The left object index
- * @param Strinv $b The right object index
- * @return int 0, 1 or -1, see usort for detail
- */
- private function orderResult($a, $b)
- {
- $o1 = $this->ds->getObjectByName($this->currentType, $a);
- $o2 = $this->ds->getObjectByName($this->currentType, $b);
- $result = 0;
-
- foreach ($this->orderColumns as &$col) {
- if (is_array($col[0])) {
- // sort by function
- $result += $col[1] * strnatcasecmp(
- $col[0][0]->$col[0][1]($o1),
- $col[0][0]->$col[0][1]($o2)
- );
- } else {
- $result += $col[1] * strnatcasecmp($o1->{$col[0]}, $o2->{$col[0]});
- }
- }
- return $result;
- }
-
- /**
- * Limit the given resultset
- *
- * @param array $indices The filtered, ordered indices
- */
- private function limitIndices(array &$indices)
- {
- foreach ($indices as $type => $subindices) {
- $indices[$type] = array_slice($subindices, $this->getOffset(), $this->getLimit());
- }
- }
-
- /**
- * Register the given function for grouping the result
- *
- * @param String $fn The function to use for grouping
- * @param Object $scope An optional scope to use instead of $this
- *
- * @return self Fluent interface
- */
- public function groupByFunction($fn, $scope = null)
- {
- $this->groupByFn = array($scope ? $scope : $this, $fn);
- return $this;
- }
-
- /**
- * Group by the given column
- *
- * @param array|string $columns The columns to use for grouping
- * @return self Fluent interface
- * @see Query::columnGroupFn() The implementation used for grouping
- */
- public function groupByColumns($columns)
- {
- if (!is_array($columns)) {
- $columns = array($columns);
- }
- $this->groupColumns = $columns;
- $this->groupByFn = array($this, 'columnGroupFn');
- return $this;
- }
-
- /**
- * The internal handler function used by the group function
- *
- * @param array $indices The indices to group
- * @return array The grouped result set
- */
- private function columnGroupFn(array &$indices)
- {
- $cols = $this->groupColumns;
- $result = array();
- foreach ($indices as $type => $subindices) {
- foreach ($subindices as $objectIndex) {
- $r = $this->ds->getObjectByName($type, $objectIndex);
- $hash = '';
- $cols = array();
- foreach ($this->groupColumns as $col) {
- $hash = md5($hash . $r->$col);
- $cols[$col] = $r->$col;
- }
- if (!isset($result[$hash])) {
- $result[$hash] = (object)array(
- 'columns' => (object)$cols,
- 'count' => 0
- );
- }
- $result[$hash]->count++;
- }
- }
- return array_values($result);
- }
-
- /**
- * Query Filter, Order, Group, Limit and return the result set
- *
- * @return array The resultset matching this query
- */
- public function getResult()
- {
- $indices = $this->getFilteredIndices();
- $this->orderIndices($indices);
- if ($this->groupByFn) {
- $scope = $this->groupByFn[self::FN_SCOPE];
- $fn = $this->groupByFn[self::FN_NAME];
-
- return $scope->$fn($indices);
- }
-
- $this->limitIndices($indices);
-
- $result = array();
- $state = $this->ds->getState();
-
- foreach ($indices as $type => $subindices) {
- foreach ($subindices as $index) {
- $result[] = & $state[$type][$index];
- }
- }
- return $result;
- }
-
-
- /**
- * Apply all filters of this filterable on the datasource
- */
- public function applyFilter()
- {
- $parser = new TreeToStatusdatQueryParser();
- if ($this->getFilter()) {
- $query = $parser->treeToQuery($this->getFilter(), $this);
- $this->setQueryFilter($query);
- }
-
- }
-
- /**
- * Return only the first row fetched from the result set
- *
- * @return MonitoringObjectList The monitoring object matching this query
- */
- public function fetchRow()
- {
- $rs = $this->fetchAll();
- $rs->rewind();
- return $rs->current();
- }
-
- /**
- * Fetch the result as an associative array using the first column as the key and the second as the value
- *
- * @return array An associative array with the result
- * @throws IcingaException If no attributes are defined
- */
- public function fetchPairs()
- {
- $result = array();
- if (count($this->getColumns()) < 2) {
- throw new IcingaException(
- 'Status.dat "fetchPairs()" query expects at least columns to be set in the query expression'
- );
- }
- $attributes = $this->getColumns();
-
- $param1 = $attributes[0];
- $param2 = $attributes[1];
- foreach ($this->fetchAll() as $resultList) {
- $result[$resultList->$param1] = $resultList->$param2;
- }
-
- return $result;
- }
-
- /**
- * Fetch all results
- *
- * @return MonitoringObjectList An MonitoringObjectList wrapping the given resultset
- */
- public function fetchAll()
- {
- $this->applyFilter();
- if (!isset($this->cursor)) {
- $result = $this->getResult();
- $this->cursor = new MonitoringObjectList($result, $this);
- }
- return $this->cursor;
- }
-
- /**
- * Return the value of the first column for the first row fetched from the result set
- */
- public function fetchOne()
- {
- throw new ProgrammingError('Statusdat/Query::fetchOne() is not implemented yet');
- }
-
- /**
- * Count the number of results
- *
- * @return int
- */
- public function count()
- {
- $q = clone $this;
- $q->limit(null, null);
- return count($q->fetchAll());
- }
-}
diff --git a/library/Icinga/Protocol/Statusdat/Query/Expression.php b/library/Icinga/Protocol/Statusdat/Query/Expression.php
deleted file mode 100644
index 1db48df33..000000000
--- a/library/Icinga/Protocol/Statusdat/Query/Expression.php
+++ /dev/null
@@ -1,415 +0,0 @@
-":
- $this->CB = "isGreater";
- break;
- case "<":
- $this->CB = "isLess";
- break;
- case ">=":
- $this->CB = "isGreaterEq";
- break;
- case "<=":
- $this->CB = "isLessEq";
- break;
- case "=":
- $this->CB = "isEqual";
- break;
- case "LIKE":
- $this->CB = "isLike";
- break;
- case "NOT_LIKE":
- $this->CB = "isNotLike";
- break;
- case "!=":
- $this->CB = "isNotEqual";
- break;
- case "IN":
- $this->CB = "isIn";
- break;
- case "NOT_IN":
- $this->CB = "isNotIn";
- break;
- default:
- throw new IcingaException(
- 'Unknown operator %s in expression %s !',
- $token,
- $this->expression
- );
- }
- }
-
- /**
- * @param $tokens
- * @return mixed
- */
- private function extractAggregationFunction(&$tokens)
- {
- $token = $tokens[0];
- $value = array();
- if (preg_match("/COUNT\{(.*)\}/", $token, $value) == false) {
- return $token;
- }
- $this->function = "count";
- $tokens[0] = $value[1];
-
- return null;
- }
-
- /**
- * @param $values
- */
- private function parseExpression(&$values)
- {
- $tokenized = preg_split("/ +/", trim($this->expression), 3);
- $this->extractAggregationFunction($tokenized);
- if (count($tokenized) != 3) {
- echo(
- "Currently statusdat query expressions must be in "
- . "the format FIELD OPERATOR ? or FIELD OPERATOR :value_name"
- );
- }
-
- $this->fields = explode(".", trim($tokenized[0]));
- $this->field = $this->fields[count($this->fields) - 1];
-
- $this->getOperatorType(trim($tokenized[1]));
- $tokenized[2] = trim($tokenized[2]);
-
- if ($tokenized[2][0] === ":") {
- $this->name = substr($tokenized, 1);
- $this->value = $values[$this->name];
- } else {
- if ($tokenized[2] === "?") {
- $this->value = array_shift($values);
- } else {
- $this->value = trim($tokenized[2]);
- }
- }
-
- }
-
- /**
- * @param $expression
- * @param $values
- * @return $this
- */
- public function fromString($expression, &$values)
- {
- $this->expression = $expression;
- $this->parseExpression($values);
- return $this;
- }
-
- /**
- * @param null $expression
- * @param array $values
- */
- public function __construct($expression = null, &$values = array())
- {
- if ($expression) {
- if (!is_array($values)) {
- $values = array($values);
- }
- $this->fromString($expression, $values);
- }
-
- }
-
- /**
- * @param array $base
- * @param array $idx
- * @return array|mixed
- */
- public function filter(array &$base, &$idx = array())
- {
- if (!$idx) {
- $idx = array_keys($base);
- }
- $this->basedata = $base;
- return array_filter($idx, array($this, "filterFn"));
- }
-
- /**
- * @return string
- */
- public function getValue()
- {
- return $this->value;
- }
-
- /**
- * @return null
- */
- public function getField()
- {
- return $this->field;
- }
-
- /**
- * @param $idx
- * @return bool
- */
- protected function filterFn($idx)
- {
- $values = $this->getFieldValues($idx);
-
- if ($values === false) {
- return false;
- }
-
- if ($this->CB == "isIn" || $this->CB == "isNotIn") {
- $cmpValues = is_array($this->value) ? $this->value : array($this->value);
- foreach ($cmpValues as $cmpValue) {
- $this->value = $cmpValue;
- foreach ($values as $value) {
- if ($this->CB == "isIn" && $this->isLike($value)) {
- $this->value = $cmpValues;
- return true;
- } elseif ($this->CB == "isNotIn" && $this->isNotLike($value)) {
- $this->value = $cmpValues;
- return true;
- }
- }
- }
- $this->value = $cmpValues;
- return false;
- }
-
- if ($this->function) {
- $values = call_user_func($this->function, $values);
- if (!is_array($values)) {
- $values = array($values);
- }
- }
- foreach ($values as $val) {
-
- if (!is_string($val) && !is_numeric($val) && is_object($val)) {
- if (isset($val->service_description)) {
- $val = $val->service_description;
- } elseif (isset($val->host_name)) {
- $val = $val->host_name;
- } else {
- return false;
- }
- }
- if ($this->{$this->CB}($val)) {
- return true;
- }
- }
- return false;
- }
-
- /**
- * @param $idx
- * @return array
- */
- private function getFieldValues($idx)
- {
- $res = $this->basedata[$idx];
-
- foreach ($this->fields as $field) {
- if (!is_array($res)) {
- if ($this->query) {
- $res = $this->query->get($res, $field);
- continue;
- }
-
- if (!isset($res->$field)) {
- $res = array();
- break;
- }
- $res = $res->$field;
- continue;
- }
-
- // it can be that an element contains more than one value, like it
- // happens when using comments, in this case we have to create a new
- // array that contains the values/objects we're searching
- $swap = array();
- foreach ($res as $sub) {
- if ($this->query) {
- $swap[] = $this->query->get($sub, $field);
- continue;
- }
- if (!isset($sub->$field)) {
- continue;
- }
- if (!is_array($sub->$field)) {
- $swap[] = $sub->$field;
- } else {
- $swap = array_merge($swap, $sub->$field);
- }
- }
- $res = $swap;
- }
- if (!is_array($res)) {
- return array($res);
- }
-
- return $res;
- }
-
- /**
- * @param $value
- * @return bool
- */
- public function isGreater($value)
- {
- return $value > $this->value;
- }
-
- /**
- * @param $value
- * @return bool
- */
- public function isLess($value)
- {
- return $value < $this->value;
- }
-
- /**
- * @param $value
- * @return bool
- */
- public function isLike($value)
- {
- return preg_match("/^" . str_replace("%", ".*", $this->value) . "$/", $value) ? true : false;
- }
-
- /**
- * @param $value
- * @return bool
- */
- public function isNotLike($value)
- {
- return !preg_match("/^" . str_replace("%", ".*", $this->value) . "$/", $value) ? true : false;
- }
-
- /**
- * @param $value
- * @return bool
- */
- public function isEqual($value)
- {
- if (!is_numeric($value)) {
- return strtolower($value) == strtolower($this->value);
- }
- return $value == $this->value;
- }
-
- /**
- * @param $value
- * @return bool
- */
- public function isNotEqual($value)
- {
- return $value != $this->value;
- }
-
- /**
- * @param $value
- * @return bool
- */
- public function isGreaterEq($value)
- {
- return $value >= $this->value;
- }
-
- /**
- * @param $value
- * @return bool
- */
- public function isLessEq($value)
- {
- return $value <= $this->value;
- }
-
- /**
- * Add additional information about the query this filter belongs to
- *
- * @param $query
- * @return mixed
- */
- public function setQuery($query)
- {
- $this->query = $query;
- }
-}
diff --git a/library/Icinga/Protocol/Statusdat/Query/Group.php b/library/Icinga/Protocol/Statusdat/Query/Group.php
deleted file mode 100644
index 2e110843f..000000000
--- a/library/Icinga/Protocol/Statusdat/Query/Group.php
+++ /dev/null
@@ -1,397 +0,0 @@
-value = $value;
- }
-
- /**
- * @return array
- */
- public function getItems()
- {
- return $this->items;
- }
-
- /**
- * @return string
- */
- public function getType()
- {
- return $this->type ? $this->type : self::TYPE_AND;
- }
-
- /**
- * @param $type
- */
- public function setType($type)
- {
- $this->type = $type;
- }
-
- /**
- * @throws IcingaException
- */
- private function tokenize()
- {
- $token = 0;
- $subgroupCount = 0;
- while ($token != self::EOF) {
-
- $token = $this->getNextToken();
-
- if ($token === self::GROUP_BEGIN) {
-
- /**
- * check if this is a nested group, if so then it's
- * considered part of the subexpression
- */
- if ($subgroupCount == 0) {
- $this->startNewSubExpression();
- }
- $subgroupCount++;
- continue;
- }
- if ($token === self::GROUP_END) {
- if ($subgroupCount < 1) {
- throw new IcingaException(
- 'Invalid Query: unexpected \')\' at pos %s',
- $this->parsePos
- );
- }
- $subgroupCount--;
- /*
- * check if this is a nested group, if so then it's
- * considered part of the subexpression
- */
- if ($subgroupCount == 0) {
- $this->addSubgroupFromExpression();
- }
- continue;
- }
-
- if ($token === self::CONJUNCTION_AND && $subgroupCount == 0) {
- $this->startNewSubExpression();
- if ($this->type != self::TYPE_AND && $this->type != "") {
- $this->createImplicitGroup(self::TYPE_AND);
- break;
- } else {
- $this->type = self::TYPE_AND;
- }
- continue;
- }
- if ($token === self::CONJUNCTION_OR && $subgroupCount == 0) {
- $this->startNewSubExpression();
- if ($this->type != self::TYPE_OR && $this->type != "") {
- $this->createImplicitGroup(self::TYPE_OR);
- break;
- } else {
-
- $this->type = self::TYPE_OR;
- }
- continue;
- }
-
- $this->subExpressionLength = $this->parsePos - $this->subExpressionStart;
- }
- if ($subgroupCount > 0) {
- throw new IcingaException('Unexpected end of query, are you missing a parenthesis?');
- }
-
- $this->startNewSubExpression();
- }
-
- /**
- * @param $type
- */
- private function createImplicitGroup($type)
- {
- $group = new Group();
- $group->setType($type);
- $group->addItem(array_pop($this->items));
-
- $group->fromString(substr($this->expression, $this->parsePos), $this->value, $this->expressionClass);
- $this->items[] = $group;
- $this->parsePos = strlen($this->expression);
-
- }
-
- /**
- *
- */
- private function startNewSubExpression()
- {
- if ($this->getCurrentSubExpression() != "") {
- if (!$this->expressionClass) {
- $this->items[] = new Expression($this->getCurrentSubExpression(), $this->value);
- } else {
- $this->items[] = new $this->expressionClass($this->getCurrentSubExpression(), $this->value);
- }
- }
-
- $this->subExpressionStart = $this->parsePos;
- $this->subExpressionLength = 0;
- }
-
- /**
- * @return string
- */
- private function getCurrentSubExpression()
- {
-
- return substr($this->expression, $this->subExpressionStart, $this->subExpressionLength);
- }
-
- /**
- *
- */
- private function addSubgroupFromExpression()
- {
-
- if (!$this->expressionClass) {
- $this->items[] = new Group($this->getCurrentSubExpression(), $this->value);
- } else {
- $group = new Group();
- $group->fromString($this->getCurrentSubExpression(), $this->value, $this->expressionClass);
- $this->items[] = $group;
- }
- $this->subExpressionStart = $this->parsePos;
- $this->subExpressionLength = 0;
- }
-
- /**
- * @return bool
- */
- private function isEOF()
- {
- if ($this->parsePos >= strlen($this->expression)) {
- return true;
- }
- return false;
- }
-
- /**
- * @return int|string
- */
- private function getNextToken()
- {
- if ($this->isEOF()) {
- return self::EOF;
- }
-
- // skip whitespaces
- while ($this->expression[$this->parsePos] == " ") {
- $this->parsePos++;
- if ($this->isEOF()) {
- return self::EOF;
- }
- }
- if ($this->expression[$this->parsePos] == self::GROUP_BEGIN) {
- $this->parsePos++;
- return self::GROUP_BEGIN;
- }
- if ($this->expression[$this->parsePos] == self::GROUP_END) {
- $this->parsePos++;
- return self::GROUP_END;
- }
- if (substr_compare(
- $this->expression,
- self::CONJUNCTION_AND,
- $this->parsePos,
- strlen(self::CONJUNCTION_AND),
- true
- ) === 0) {
- $this->parsePos += strlen(self::CONJUNCTION_AND);
- return self::CONJUNCTION_AND;
- }
- if (substr_compare(
- $this->expression,
- self::CONJUNCTION_OR,
- $this->parsePos,
- strlen(self::CONJUNCTION_OR),
- true
- ) === 0) {
- $this->parsePos += strlen(self::CONJUNCTION_OR);
- return self::CONJUNCTION_OR;
- }
- $this->parsePos++;
- return self::EXPRESSION;
- }
-
- /**
- * @param $ex
- * @return $this
- */
- public function addItem($ex)
- {
- $this->items[] = $ex;
- return $this;
- }
-
- /**
- * @param $expression
- * @param array $value
- * @param null $expressionClass
- * @return $this
- */
- public function fromString($expression, &$value = array(), $expressionClass = null)
- {
- $this->expression = $expression;
- $this->value = & $value;
- $this->expressionClass = $expressionClass;
-
- $this->tokenize();
- return $this;
- }
-
- /**
- * @param null $expression
- * @param array $value
- */
- public function __construct($expression = null, &$value = array())
- {
- if ($expression) {
- $this->fromString($expression, $value);
- }
- }
-
- /**
- * @param array $base
- * @param null $idx
- * @return array|null
- */
- public function filter(array &$base, &$idx = null)
- {
- if ($this->type == self::TYPE_OR) {
- $idx = array();
- foreach ($this->items as &$subFilter) {
- $baseKeys = array_keys($base);
- $subFilter->setQuery($this->query);
- $idx += $subFilter->filter($base, $baseKeys);
- }
- } else {
- if (!$idx) {
- $idx = array_keys($base);
- }
- foreach ($this->items as $subFilter) {
- $subFilter->setQuery($this->query);
- $idx = array_intersect($idx, $subFilter->filter($base, $idx));
- }
- }
-
- return $idx;
- }
-
- /**
- * Add additional information about the query this filter belongs to
- *
- * @param $query
- * @return mixed
- */
- public function setQuery($query)
- {
- $this->query = $query;
- }
-}
diff --git a/library/Icinga/Protocol/Statusdat/Query/IQueryPart.php b/library/Icinga/Protocol/Statusdat/Query/IQueryPart.php
deleted file mode 100644
index 1dcd434df..000000000
--- a/library/Icinga/Protocol/Statusdat/Query/IQueryPart.php
+++ /dev/null
@@ -1,36 +0,0 @@
-noCache = $noCache;
- if (isset($config->no_cache)) {
- $this->noCache = $config->no_cache;
- }
- $this->config = $config;
- $this->parser = $parser;
-
- if (!$this->noCache) {
- $this->cache = $this->initializeCaches($config);
- if ($this->fromCache()) {
- $this->createHostServiceConnections();
- return;
- }
- }
-
- if (!$this->lastState) {
- $this->parseObjectsCacheFile();
- }
- if (!$this->hasRuntimeState) {
-
- }
- $this->parseStatusDatFile();
- if (!$noCache && $this->newState) {
- $this->statusCache->save($this->parser->getRuntimeState(), 'object' . md5($this->config->object_file));
- }
- $this->createHostServiceConnections();
-
- }
-
- /**
- * Initialize the internal caches if enabled
- *
- * @throws ConfigurationError
- */
- private function initializeCaches()
- {
- $defaultCachePath = self::STATUSDAT_DEFAULT_CACHE_PATH;
- $cachePath = $this->config->get('cache_path', $defaultCachePath);
- $maxCacheLifetime = intval($this->config->get('cache_path', self::DEFAULT_CACHE_LIFETIME));
- $cachingEnabled = true;
- if (!is_writeable($cachePath)) {
- Logger::warning(
- 'Can\'t cache Status.dat backend; make sure cachepath %s is writable by the web user. '
- . 'Caching is now disabled',
- $cachePath
- );
- $cachePath = null;
- }
- $backendOptions = array(
- 'cache_dir' => $cachePath
- );
- // the object cache might exist for months and is still valid
- $this->objectCache = $this->initCache($this->config->object_file, $backendOptions, null, $cachingEnabled);
- $this->statusCache = $this->initCache(
- $this->config->status_file,
- $backendOptions,
- $maxCacheLifetime,
- $cachingEnabled
- );
- }
-
- /**
- * Init the Cache backend in Zend
- *
- * @param String $file The file to use as the cache master file
- * @param Zend_Config $backend The backend configuration to use
- * @param integer $lifetime The lifetime of the cache
- *
- * @return \Zend_Cache_Core|\Zend_Cache_Frontend
- */
- private function initCache($file, $backendConfig, $lifetime)
- {
- $frontendOptions = array(
- 'lifetime' => $lifetime,
- 'automatic_serialization' => true,
- 'master_files' => array($file)
- );
- return \Zend_Cache::factory('Core', 'File', $frontendOptions, $backendConfig);
- }
-
- /**
- * Read the current cache state
- *
- * @return bool True if the state is the same as the icinga state
- */
- private function fromCache()
- {
- if (!$this->readObjectsCache()) {
- $this->newState = true;
- return false;
- }
- if (!$this->readStatusCache()) {
- $this->newState = true;
- return false;
- }
- return true;
- }
-
- /**
- * Read the object.cache file from the Zend_Cache backend
- *
- * @return bool True if the file could be loaded from cache
- */
- private function readObjectsCache()
- {
- $this->lastState = $this->objectCache->load('object' . md5($this->config->object_file));
- if ($this->lastState == false) {
- return false;
- }
-
- return true;
- }
-
- /**
- * Read the status.dat file from the Zend_Cache backend
- *
- * @return bool True if the file could be loaded from cache
- */
- private function readStatusCache()
- {
- if (!isset($this->stateCache)) {
- return true;
- }
- $statusInfo = $this->stateCache->load('state' . md5($this->config->status_file));
- if ($statusInfo == false) {
- return false;
- }
-
- $this->hasRuntimeState = true;
- return true;
- }
-
- /**
- * Take the status.dat and objects.cache and connect all services to hosts
- *
- */
- private function createHostServiceConnections()
- {
- if (!isset($this->lastState["service"])) {
- return;
- }
- foreach ($this->lastState["host"] as &$host) {
- $host->host = $host;
- }
- foreach ($this->lastState["service"] as &$service) {
- $service->service = &$service; // allow easier querying
- $host = &$this->lastState["host"][$service->host_name];
- if (!isset($host->services)) {
- $host->services = array();
- }
- $host->services[$service->service_description] = & $service;
- $service->host = & $host;
- }
- }
-
- /**
- * Parse the object.cache file and update the current state
- *
- * @throws ConfigurationError If the object.cache couldn't be read
- */
- private function parseObjectsCacheFile()
- {
- if (!is_readable($this->config->object_file)) {
- throw new ConfigurationError(
- 'Can\'t read object-file "%s", check your configuration',
- $this->config->object_file
- );
- }
- if (!$this->parser) {
- $this->parser = new Parser(new File($this->config->object_file, 'r'));
- }
- $this->parser->parseObjectsFile();
- $this->lastState = $this->parser->getRuntimeState();
- }
-
- /**
- * Parse the status.dat file and update the current state
- *
- * @throws ConfigurationError If the status.dat couldn't be read
- */
- private function parseStatusDatFile()
- {
- if (!is_readable($this->config->status_file)) {
- throw new ConfigurationError(
- 'Can\'t read status-file %s, check your configuration',
- $this->config->status_file
- );
- }
- if (!$this->parser) {
- $this->parser = new Parser(new File($this->config->status_file, 'r'), $this->lastState);
- }
- $this->parser->parseRuntimeState(new File($this->config->status_file, 'r'));
- $this->lastState = $this->parser->getRuntimeState();
- if (!$this->noCache) {
- $this->statusCache->save(array("true" => true), "state" . md5($this->config->object_file));
- }
- }
-
- /**
- * Create a new Query
- *
- * @return Query The query to operate on
- */
- public function select()
- {
- return new Query($this);
- }
-
- /**
- * Return the internal state of the status.dat
- *
- * @return mixed The internal status.dat representation
- */
- public function getState()
- {
- return $this->lastState;
- }
-
-
- /**
- * Return the object with the given name and type
- *
- * @param String $type The type of the object to return (service, host, servicegroup...)
- * @param String $name The name of the object
- *
- * @return ObjectContainer An object container wrapping the result or null if the object doesn't exist
- */
- public function getObjectByName($type, $name)
- {
- if (isset($this->lastState[$type]) && isset($this->lastState[$type][$name])) {
- return new ObjectContainer($this->lastState[$type][$name], $this);
- }
- return null;
- }
-
- /**
- * Get an array containing all names of monitoring objects with the given type
- *
- * @param String $type The type of object to get the names for
- * @return array An array of names or null if the type does not exist
- */
- public function getObjectNames($type)
- {
- return isset($this->lastState[$type]) ? array_keys($this->lastState[$type]) : null;
- }
-}
diff --git a/library/Icinga/Protocol/Statusdat/RuntimeStateContainer.php b/library/Icinga/Protocol/Statusdat/RuntimeStateContainer.php
deleted file mode 100644
index b15dfe4bb..000000000
--- a/library/Icinga/Protocol/Statusdat/RuntimeStateContainer.php
+++ /dev/null
@@ -1,72 +0,0 @@
-runtimeState = $str;
- }
-
- /**
- * Return true if the argument exists
- *
- * @param String $attr The argument to retrieve
- * @return bool True if it exists, otherwise false
- */
- public function __isset($attr)
- {
- try {
- $this->__get($attr);
- return true;
- } catch (\InvalidArgumentException $e) {
- return false;
- }
- }
-
- /**
- * Return the given attribute
- *
- * If the container string is not yet parsed, this will happen here
- *
- * @param String $attr The attribute to retrieve
- * @return mixed The value of the attribute
- * @throws \InvalidArgumentException When the attribute does not exist
- */
- public function __get($attr)
- {
- $start = strpos($this->runtimeState, $attr . "=");
- if ($start === false) {
- throw new \InvalidArgumentException("Unknown property $attr");
- }
-
- $start += strlen($attr . "=");
- $len = strpos($this->runtimeState, "\n", $start) - $start;
- $this->$attr = trim(substr($this->runtimeState, $start, $len));
-
- return $this->$attr;
- }
-}
diff --git a/library/Icinga/Protocol/Statusdat/View/AccessorStrategy.php b/library/Icinga/Protocol/Statusdat/View/AccessorStrategy.php
deleted file mode 100644
index 4e4bc76a1..000000000
--- a/library/Icinga/Protocol/Statusdat/View/AccessorStrategy.php
+++ /dev/null
@@ -1,42 +0,0 @@
-dataSet = $dataset;
- $this->position = 0;
- $this->dataView = $dataView;
- }
-
- public function count()
- {
- return count($this->dataSet);
- }
-
- public function setPosition($pos)
- {
- $this->position = $pos;
- }
-
- /**
- * (PHP 5 >= 5.0.0)
- * Return the current element
- * @link http://php.net/manual/en/iterator.current.php
- * @return mixed Can return any type.
- */
- public function current()
- {
- if ($this->dataView) {
- return $this;
- }
-
- return $this->dataSet[$this->position];
- }
-
- /**
- * (PHP 5 >= 5.0.0)
- * Move forward to next element
- * @link http://php.net/manual/en/iterator.next.php
- * @return void Any returned value is ignored.
- */
- public function next()
- {
- $this->position++;
- }
-
- /**
- * (PHP 5 >= 5.0.0)
- * Return the key of the current element
- * @link http://php.net/manual/en/iterator.key.php
- * @return mixed scalar on success, or null on failure.
- */
- public function key()
- {
- return $this->position;
- }
-
- /**
- * (PHP 5 >= 5.0.0)
- * Checks if current position is valid
- * @link http://php.net/manual/en/iterator.valid.php
- * @return boolean The return value will be casted to boolean and then evaluated.
- * Returns true on success or false on failure.
- */
- public function valid()
- {
- return $this->position < count($this->dataSet);
- }
-
- /**
- * (PHP 5 >= 5.0.0)
- * Rewind the Iterator to the first element
- * @link http://php.net/manual/en/iterator.rewind.php
- * @return void Any returned value is ignored.
- */
- public function rewind()
- {
- $this->position = 0;
- }
-
- public function __isset($name)
- {
- return $this->dataView->exists($this->dataSet[$this->position], $name);
- }
-
- public function __get($name)
- {
- return $this->dataView->get($this->dataSet[$this->position], $name);
- }
-
- public function __set($name, $value)
- {
- throw new IcingaException('Setting is currently not available for objects');
- }
-
- public function offsetExists($offset)
- {
- return count($this->dataSet) < $offset;
- }
-
- public function offsetGet($offset)
- {
- $res = new MonitoringObjectList($this->dataSet, $this->dataView);
- $res->position = $offset;
- return $res;
- }
-
- public function offsetSet($offset, $value)
- {
- // non mutable
- }
-
- public function offsetUnset($offset)
- {
- // non mutable
- }
-}
diff --git a/library/Icinga/User.php b/library/Icinga/User.php
index 03aa1bcd2..659767644 100644
--- a/library/Icinga/User.php
+++ b/library/Icinga/User.php
@@ -198,7 +198,9 @@ class User
public function setPermissions(array $permissions)
{
natcasesort($permissions);
- $this->permissions = array_combine($permissions, $permissions);
+ if (! empty($permissions)) {
+ $this->permissions = array_combine($permissions, $permissions);
+ }
return $this;
}
diff --git a/library/Icinga/Util/Enumerate.php b/library/Icinga/Util/Enumerate.php
deleted file mode 100644
index 0861f7f65..000000000
--- a/library/Icinga/Util/Enumerate.php
+++ /dev/null
@@ -1,62 +0,0 @@
-iterator = $iterator;
- }
-
- public function rewind()
- {
- $this->iterator->rewind();
- $this->key = 0;
- }
-
- public function next()
- {
- $this->iterator->next();
- ++$this->key;
- }
-
- public function valid()
- {
- return $this->iterator->valid();
- }
-
- public function current()
- {
- return $this->iterator->current();
- }
-
- public function key()
- {
- return $this->key;
- }
-}
diff --git a/library/Icinga/Util/EnumeratingFilterIterator.php b/library/Icinga/Util/EnumeratingFilterIterator.php
new file mode 100644
index 000000000..44ef9b0c8
--- /dev/null
+++ b/library/Icinga/Util/EnumeratingFilterIterator.php
@@ -0,0 +1,37 @@
+index = 0;
+ }
+
+ /**
+ * @return int
+ */
+ public function key()
+ {
+ return $this->index++;
+ }
+}
diff --git a/library/Icinga/Util/String.php b/library/Icinga/Util/String.php
index b64d9e524..0bebb06e7 100644
--- a/library/Icinga/Util/String.php
+++ b/library/Icinga/Util/String.php
@@ -21,4 +21,18 @@ class String
{
return array_map('trim', explode($delimiter, $value));
}
+
+ /**
+ * Uppercase the first character of each word in a string assuming and removing the underscore as word separator
+ *
+ * Converts 'first_name' to 'firstName' for example.
+ *
+ * @param string $name
+ *
+ * @return string
+ */
+ public static function cname($name)
+ {
+ return str_replace(' ', '', ucwords(str_replace('_', ' ', strtolower($name))));
+ }
}
diff --git a/library/Icinga/Web/Form.php b/library/Icinga/Web/Form.php
index 2763e073e..b32a03075 100644
--- a/library/Icinga/Web/Form.php
+++ b/library/Icinga/Web/Form.php
@@ -5,6 +5,7 @@
namespace Icinga\Web;
use LogicException;
+use Zend_Config;
use Zend_Form;
use Zend_View_Interface;
use Icinga\Application\Icinga;
@@ -81,6 +82,19 @@ class Form extends Zend_Form
*/
protected $uidElementName = 'formUID';
+ /**
+ * Default element decorators
+ *
+ * @var array
+ */
+ public static $defaultElementDecorators = array(
+ 'ViewHelper',
+ 'Errors',
+ array('Description', array('tag' => 'span', 'class' => 'description')),
+ 'Label',
+ array('HtmlTag', array('tag' => 'div'))
+ );
+
/**
* Create a new form
*
@@ -105,16 +119,6 @@ class Form extends Zend_Form
throw new LogicException('The option `onSuccess\' is not callable');
}
- if (! isset($options['elementDecorators'])) {
- $options['elementDecorators'] = array(
- 'ViewHelper',
- 'Errors',
- array('Description', array('tag' => 'span', 'class' => 'description')),
- 'Label',
- array('HtmlTag', array('tag' => 'div'))
- );
- }
-
parent::__construct($options);
}
@@ -417,23 +421,35 @@ class Form extends Zend_Form
/**
* Create a new element
*
- * Additionally, all structural form element decorators by Zend are replaced with our own ones.
+ * Icinga Web 2 loads its own default element decorators. For loading Zend's default element decorators set the
+ * `disableLoadDefaultDecorators' option to any other value than `true'. For loading custom element decorators use
+ * the 'decorators' option.
*
* @param string $type String element type
* @param string $name The name of the element to add
- * @param array $options The options for the element
+ * @param mixed $options The options for the element
*
* @return Zend_Form_Element
*
- * @see Zend_Form::createElement()
+ * @see Form::$defaultElementDecorators For Icinga Web 2's default element decorators.
*/
public function createElement($type, $name, $options = null)
{
- if (is_array($options) && ! isset($options['disableLoadDefaultDecorators'])) {
- $options['disableLoadDefaultDecorators'] = true;
+ if ($options !== null) {
+ if ($options instanceof Zend_Config) {
+ $options = $options->toArray();
+ }
+ if (! isset($options['decorators'])
+ && ! array_key_exists('disabledLoadDefaultDecorators', $options)
+ ) {
+ $options['decorators'] = static::$defaultElementDecorators;
+ }
+ } else {
+ $options = array('decorators' => static::$defaultElementDecorators);
}
$el = parent::createElement($type, $name, $options);
+
if ($el && $el->getAttrib('autosubmit')) {
$el->addDecorator(new NoScriptApply()); // Non-JS environments
$class = $el->getAttrib('class');
diff --git a/library/Icinga/Web/Form/Element/CsrfCounterMeasure.php b/library/Icinga/Web/Form/Element/CsrfCounterMeasure.php
index 722b1d323..e9bc37edd 100644
--- a/library/Icinga/Web/Form/Element/CsrfCounterMeasure.php
+++ b/library/Icinga/Web/Form/Element/CsrfCounterMeasure.php
@@ -4,8 +4,8 @@
namespace Icinga\Web\Form\Element;
-use Zend_Form_Element_Xhtml;
use Icinga\Web\Session;
+use Icinga\Web\Form\FormElement;
use Icinga\Web\Form\InvalidCSRFTokenException;
/**
@@ -13,7 +13,7 @@ use Icinga\Web\Form\InvalidCSRFTokenException;
*
* You must not set a value to successfully use this element, just give it a name and you're good to go.
*/
-class CsrfCounterMeasure extends Zend_Form_Element_Xhtml
+class CsrfCounterMeasure extends FormElement
{
/**
* Default form view helper to use for rendering
@@ -22,14 +22,26 @@ class CsrfCounterMeasure extends Zend_Form_Element_Xhtml
*/
public $helper = 'formHidden';
+ /**
+ * Counter measure element is required
+ *
+ * @var bool
+ */
+ protected $_ignore = true;
+
+ /**
+ * Ignore element when retrieving values at form level
+ *
+ * @var bool
+ */
+ protected $_required = true;
+
/**
* Initialize this form element
*/
public function init()
{
- $this->setRequired(true); // Not requiring this element would not make any sense
- $this->setIgnore(true); // We do not want this element's value being retrieved by Form::getValues()
- $this->setDecorators(array('ViewHelper'));
+ $this->addDecorator('ViewHelper');
$this->setValue($this->generateCsrfToken());
}
diff --git a/library/Icinga/Web/Form/Element/DateTimePicker.php b/library/Icinga/Web/Form/Element/DateTimePicker.php
index 47f9b26e9..728262827 100644
--- a/library/Icinga/Web/Form/Element/DateTimePicker.php
+++ b/library/Icinga/Web/Form/Element/DateTimePicker.php
@@ -5,27 +5,15 @@
namespace Icinga\Web\Form\Element;
use DateTime;
-use Zend_Form_Element;
+use Icinga\Web\Form;
+use Icinga\Web\Form\FormElement;
use Icinga\Web\Form\Validator\DateTimeValidator;
/**
* A date-and-time input control
- *
- * @method DateTime getValue()
*/
-class DateTimePicker extends Zend_Form_Element
+class DateTimePicker extends FormElement
{
- /**
- * Disable default decorators
- *
- * \Icinga\Web\Form sets default decorators for elements.
- *
- * @var bool
- *
- * @see \Icinga\Web\Form::__construct() For default element decorators.
- */
- protected $_disableLoadDefaultDecorators = true;
-
/**
* Form view helper to use for rendering
*
@@ -54,7 +42,7 @@ class DateTimePicker extends Zend_Form_Element
/**
* (non-PHPDoc)
- * @see \Zend_Form_Element::init() For the method documentation.
+ * @see Zend_Form_Element::init() For the method documentation.
*/
public function init()
{
diff --git a/library/Icinga/Web/Form/Element/Note.php b/library/Icinga/Web/Form/Element/Note.php
index 700bea107..78881ab44 100644
--- a/library/Icinga/Web/Form/Element/Note.php
+++ b/library/Icinga/Web/Form/Element/Note.php
@@ -5,35 +5,42 @@
namespace Icinga\Web\Form\Element;
use Zend_Form_Element;
+use Icinga\Web\Form;
/**
- * Implements note element for Zend forms
+ * A note
*/
class Note extends Zend_Form_Element
{
/**
- * Disable default decorators
- *
- * \Icinga\Web\Form sets default decorators for elements.
- *
- * @var bool
- *
- * @see \Icinga\Web\Form::__construct() For default element decorators.
- */
- protected $_disableLoadDefaultDecorators = true;
-
- /**
- * Name of the view helper
+ * Form view helper to use for rendering
*
* @var string
*/
public $helper = 'formNote';
/**
- * Return true to ensure redrawing
+ * Ignore element when retrieving values at form level
*
- * @param mixed $value The value of to validate (ignored)
- * @return bool Always true
+ * @var bool
+ */
+ protected $_ignore = true;
+
+ /**
+ * (non-PHPDoc)
+ * @see Zend_Form_Element::init() For the method documentation.
+ */
+ public function init()
+ {
+ $this->setDecorators(Form::$defaultElementDecorators);
+ }
+
+ /**
+ * Validate element value (pseudo)
+ *
+ * @param mixed $value Ignored
+ *
+ * @return bool Always true
*/
public function isValid($value)
{
diff --git a/library/Icinga/Web/Form/Element/Number.php b/library/Icinga/Web/Form/Element/Number.php
index 75c08165c..836142514 100644
--- a/library/Icinga/Web/Form/Element/Number.php
+++ b/library/Icinga/Web/Form/Element/Number.php
@@ -4,24 +4,13 @@
namespace Icinga\Web\Form\Element;
-use Zend_Form_Element;
+use Icinga\Web\Form\FormElement;
/**
* A number input control
*/
-class Number extends Zend_Form_Element
+class Number extends FormElement
{
- /**
- * Disable default decorators
- *
- * \Icinga\Web\Form sets default decorators for elements.
- *
- * @var bool
- *
- * @see \Icinga\Web\Form::__construct() For default element decorators.
- */
- protected $_disableLoadDefaultDecorators = true;
-
/**
* Form view helper to use for rendering
*
diff --git a/library/Icinga/Web/Form/FormElement.php b/library/Icinga/Web/Form/FormElement.php
new file mode 100644
index 000000000..690be1acf
--- /dev/null
+++ b/library/Icinga/Web/Form/FormElement.php
@@ -0,0 +1,62 @@
+_disableLoadDefaultDecorators === true;
+ }
+
+ /**
+ * Load default decorators
+ *
+ * Icinga Web 2 loads its own default element decorators. For loading Zend's default element decorators set
+ * FormElement::$_disableLoadDefaultDecorators to false.
+ *
+ * @return this
+ * @see Form::$defaultElementDecorators For Icinga Web 2's default element decorators.
+ */
+ public function loadDefaultDecorators()
+ {
+ if ($this->loadDefaultDecoratorsIsDisabled()) {
+ return $this;
+ }
+
+ if (! isset($this->_disableLoadDefaultDecorators)) {
+ $decorators = $this->getDecorators();
+ if (empty($decorators)) {
+ // Load Icinga Web 2's default element decorators
+ $this->addDecorators(Form::$defaultElementDecorators);
+ }
+ } else {
+ // Load Zend's default decorators
+ parent::loadDefaultDecorators();
+ }
+ return $this;
+ }
+}
diff --git a/library/Icinga/Web/Widget/Tabs.php b/library/Icinga/Web/Widget/Tabs.php
index 716d5a959..d79453a01 100644
--- a/library/Icinga/Web/Widget/Tabs.php
+++ b/library/Icinga/Web/Widget/Tabs.php
@@ -23,6 +23,7 @@ class Tabs extends AbstractWidget implements Countable
= sprintf($this->translate('%u Hosts'), array_sum(array_values($hostStates))) ?> | -|
---|---|
- = $this->hostStatesPieChart ?> - | -
-
+ = sprintf($this->translate('Hosts (%u)'), array_sum(array_values($hostStates))) ?>
+
+
+ = $this->hostStatesPieChart ?>
+
+
+ $count) {
+ echo sprintf('%s: %u
- foreach (array_filter($hostStates) as $text => $count) {
- echo sprintf('%s: %u', strtoupper($text), $count); + } ?> + ', strtoupper($text), $count); - } - - ?> - |
-
= sprintf($this->translate('%u Services'), array_sum(array_values($serviceStates))) ?> | -= sprintf($this->translate('%u Hosts'), array_sum(array_values($hostStates))) ?> | -||
---|---|---|---|
- = $this->serviceStatesPieChart ?> - | -
- $count) {
- echo sprintf('%s: %u ', strtoupper($text), $count); - } +
+
- ?>
-
+ = sprintf($this->translate('Services (%u)'), array_sum(array_values($serviceStates))) ?>
+
+
+ = $this->serviceStatesPieChart ?>
+
+
+ $count) {
+ echo sprintf(' %s: %u
+ ', strtoupper($text), $count); + } ?> + |
- - = $this->hostStatesPieChart ?> - | -
- $count) {
- echo sprintf('%s: %u ', strtoupper($text), $count); - } - - ?> - |
-