Merge branch 'master' into feature/user-and-group-management-8826
This commit is contained in:
commit
cd0c418854
|
@ -22,7 +22,7 @@ Vagrant.configure(VAGRANTFILE_API_VERSION) do |config|
|
|||
config.vm.provision :shell, :path => ".puppet/manifests/puppet.sh"
|
||||
|
||||
config.vm.provider :virtualbox do |v, override|
|
||||
override.vm.box = "puppetlabs/centos-6.5-64-puppet"
|
||||
override.vm.box = "puppetlabs/centos-6.6-64-puppet"
|
||||
|
||||
v.customize ["modifyvm", :id, "--memory", "1024"]
|
||||
end
|
||||
|
|
|
@ -84,7 +84,7 @@ class ConfigController extends Controller
|
|||
public function indexAction()
|
||||
{
|
||||
if ($this->firstAllowedAction === null) {
|
||||
throw new SecurityException('No permission for configuration');
|
||||
throw new SecurityException($this->translate('No permission for configuration'));
|
||||
}
|
||||
$action = $this->getTabs()->get($this->firstAllowedAction);
|
||||
if (substr($action->getUrl()->getPath(), 0, 7) === 'config/') {
|
||||
|
@ -271,7 +271,7 @@ class ConfigController extends Controller
|
|||
$configForm->remove($authBackend);
|
||||
} catch (InvalidArgumentException $e) {
|
||||
Notification::error($e->getMessage());
|
||||
return;
|
||||
return false;
|
||||
}
|
||||
|
||||
if ($configForm->save()) {
|
||||
|
@ -353,7 +353,7 @@ class ConfigController extends Controller
|
|||
$configForm->remove($resource);
|
||||
} catch (InvalidArgumentException $e) {
|
||||
Notification::error($e->getMessage());
|
||||
return;
|
||||
return false;
|
||||
}
|
||||
|
||||
if ($configForm->save()) {
|
||||
|
|
|
@ -78,7 +78,7 @@ class DashboardController extends ActionController
|
|||
$dashboard = $this->dashboard;
|
||||
$form = new DashletForm();
|
||||
$form->setDashboard($dashboard);
|
||||
$form->setSubmitLabel(t('Update Dashlet'));
|
||||
$form->setSubmitLabel($this->translate('Update Dashlet'));
|
||||
if (! $this->_request->getParam('pane')) {
|
||||
throw new Zend_Controller_Action_Exception(
|
||||
'Missing parameter "pane"',
|
||||
|
|
|
@ -37,7 +37,7 @@ class ErrorController extends ActionController
|
|||
$path = array_shift($path);
|
||||
$this->getResponse()->setHttpResponseCode(404);
|
||||
$this->view->message = $this->translate('Page not found.');
|
||||
if ($modules->hasInstalled($path) && ! $modules->hasEnabled($path)) {
|
||||
if ($this->Auth()->isAuthenticated() && $modules->hasInstalled($path) && ! $modules->hasEnabled($path)) {
|
||||
$this->view->message .= ' ' . sprintf(
|
||||
$this->translate('Enabling the "%s" module might help!'),
|
||||
$path
|
||||
|
|
|
@ -1,7 +1,7 @@
|
|||
<?php
|
||||
/* Icinga Web 2 | (c) 2013-2015 Icinga Development Team | GPLv2+ */
|
||||
|
||||
use Icinga\Module\Monitoring\Controller;
|
||||
use Icinga\Web\Controller;
|
||||
use Icinga\Web\Url;
|
||||
use Icinga\Web\Widget\Tabextension\DashboardAction;
|
||||
use Icinga\Web\Widget\Tabextension\OutputFormat;
|
||||
|
|
|
@ -149,7 +149,7 @@ class ResourceConfigForm extends ConfigForm
|
|||
}
|
||||
} catch (InvalidArgumentException $e) {
|
||||
Notification::error($e->getMessage());
|
||||
return;
|
||||
return false;
|
||||
}
|
||||
|
||||
if ($this->save()) {
|
||||
|
|
|
@ -0,0 +1,15 @@
|
|||
# <a id="configuration"></a> Configuration
|
||||
|
||||
## Overview
|
||||
|
||||
Apart from its web configuration capabilities, the local configuration is
|
||||
stored in `/etc/icingaweb2` by default (depending on your config setup).
|
||||
|
||||
Location | File | Description
|
||||
------------------------------|-----------------------|---------------------------
|
||||
. | config.ini | General configuration (logging, preferences)
|
||||
. | resources.ini | Global resources (Icinga Web 2 database for preferences and authentication, icinga ido database)
|
||||
. | roles.ini | User specific roles (e.g. `administrators`) and permissions
|
||||
. | [authentication.ini](authentication.md) | Authentication backends (e.g. database)
|
||||
enabledModules | Symlink | Contains symlinks to enabled modules from `/usr/share/icingaweb2/modules/*`. Defaults to [monitoring](modules/monitoring/doc/configuration.md) and `doc`.
|
||||
modules | Directory | Module specific configuration
|
|
@ -71,7 +71,7 @@ class Manager
|
|||
$user
|
||||
);
|
||||
$preferences = new Preferences($preferencesStore->load());
|
||||
} catch (NotReadableError $e) {
|
||||
} catch (Exception $e) {
|
||||
Logger::error(
|
||||
new IcingaException(
|
||||
'Cannot load preferences for user "%s". An exception was thrown: %s',
|
||||
|
|
|
@ -62,7 +62,7 @@ class IniEditor
|
|||
$this->commentIndentation = array_key_exists('commentIndentation', $options)
|
||||
? $options['commentIndentation'] : 43;
|
||||
$this->sectionSeparators = array_key_exists('sectionSeparators', $options)
|
||||
? $options['sectionSeparators'] : 2;
|
||||
? $options['sectionSeparators'] : 1;
|
||||
}
|
||||
|
||||
/**
|
||||
|
@ -314,7 +314,7 @@ class IniEditor
|
|||
public function getText()
|
||||
{
|
||||
$this->cleanUpWhitespaces();
|
||||
return implode(PHP_EOL, $this->text);
|
||||
return rtrim(implode(PHP_EOL, $this->text)) . PHP_EOL;
|
||||
}
|
||||
|
||||
/**
|
||||
|
|
|
@ -216,25 +216,22 @@ class FilterEditor extends AbstractWidget
|
|||
$filter = $this->getFilter();
|
||||
|
||||
if ($search !== null) {
|
||||
if (empty($this->searchColumns)) {
|
||||
if (strpos($search, '=') === false) {
|
||||
Notification::error(mt('monitoring', 'Cannot search here'));
|
||||
return $this;
|
||||
} else {
|
||||
list($k, $v) = preg_split('/=/', $search);
|
||||
$filter = $this->mergeRootExpression($filter, trim($k), '=', ltrim($v));
|
||||
}
|
||||
} else {
|
||||
if (false === $this->resetSearchColumns($filter)) {
|
||||
if (strpos($search, '=') !== false) {
|
||||
list($k, $v) = preg_split('/=/', $search);
|
||||
$filter = $this->mergeRootExpression($filter, trim($k), '=', ltrim($v));
|
||||
} elseif (! empty($this->searchColumns)) {
|
||||
if (! $this->resetSearchColumns($filter)) {
|
||||
$filter = Filter::matchAll();
|
||||
}
|
||||
|
||||
$filters = array();
|
||||
$search = ltrim($search);
|
||||
foreach ($this->searchColumns as $searchColumn) {
|
||||
$filters[] = Filter::expression($searchColumn, '=', "*$search*");
|
||||
}
|
||||
$filter->andFilter(new FilterOr($filters));
|
||||
$filter = $filter->andFilter(new FilterOr($filters));
|
||||
} else {
|
||||
Notification::error(mt('monitoring', 'Cannot search here'));
|
||||
return $this;
|
||||
}
|
||||
|
||||
$url = $this->url()->setQueryString(
|
||||
|
|
|
@ -56,6 +56,7 @@ class Monitoring_HostsController extends Controller
|
|||
'host_icon_image',
|
||||
'host_icon_image_alt',
|
||||
'host_name',
|
||||
'host_address',
|
||||
'host_state',
|
||||
'host_problem',
|
||||
'host_handled',
|
||||
|
@ -94,6 +95,7 @@ class Monitoring_HostsController extends Controller
|
|||
'host_icon_image',
|
||||
'host_icon_image_alt',
|
||||
'host_name',
|
||||
'host_address',
|
||||
'host_state',
|
||||
'host_problem',
|
||||
'host_handled',
|
||||
|
@ -131,14 +133,14 @@ class Monitoring_HostsController extends Controller
|
|||
$this->view->objects = $this->hostList;
|
||||
$this->view->unhandledObjects = $this->hostList->getUnhandledObjects();
|
||||
$this->view->problemObjects = $this->hostList->getProblemObjects();
|
||||
|
||||
$this->view->acknowledgeUnhandledLink = Url::fromPath('monitoring/hosts/acknowledge-problem')
|
||||
->setQueryString($this->hostList->getUnhandledObjects()->objectsFilter());
|
||||
$this->view->downtimeUnhandledLink = Url::fromPath('monitoring/hosts/schedule-downtime')
|
||||
->setQueryString($this->hostList->getUnhandledObjects()->objectsFilter());
|
||||
->setQueryString($this->hostList->getUnhandledObjects()->objectsFilter()->toQueryString());
|
||||
$this->view->downtimeLink = Url::fromPath('monitoring/hosts/schedule-downtime')
|
||||
->setQueryString($this->hostList->getProblemObjects()->objectsFilter());
|
||||
->setQueryString($this->hostList->getProblemObjects()->objectsFilter()->toQueryString());
|
||||
$this->view->acknowledgedObjects = $this->hostList->getAcknowledgedObjects();
|
||||
$this->view->acknowledgeLink = Url::fromPath('monitoring/hosts/acknowledge-problem')
|
||||
->setQueryString($this->hostList->getUnacknowledgedObjects()->objectsFilter()->toQueryString());
|
||||
$this->view->unacknowledgedObjects = $this->hostList->getUnacknowledgedObjects();
|
||||
$this->view->objectsInDowntime = $this->hostList->getObjectsInDowntime();
|
||||
$this->view->inDowntimeLink = Url::fromPath('monitoring/list/hosts')
|
||||
->setQueryString(
|
||||
|
|
|
@ -53,6 +53,7 @@ class Monitoring_ServicesController extends Controller
|
|||
'host_icon_image',
|
||||
'host_icon_image_alt',
|
||||
'host_name',
|
||||
'host_address',
|
||||
'host_output',
|
||||
'host_state',
|
||||
'host_problem',
|
||||
|
@ -100,6 +101,7 @@ class Monitoring_ServicesController extends Controller
|
|||
'host_icon_image',
|
||||
'host_icon_image_alt',
|
||||
'host_name',
|
||||
'host_address',
|
||||
'host_output',
|
||||
'host_state',
|
||||
'host_problem',
|
||||
|
@ -147,13 +149,14 @@ class Monitoring_ServicesController extends Controller
|
|||
$this->view->objects = $this->serviceList;
|
||||
$this->view->unhandledObjects = $this->serviceList->getUnhandledObjects();
|
||||
$this->view->problemObjects = $this->serviceList->getProblemObjects();
|
||||
$this->view->acknowledgeUnhandledLink = Url::fromPath('monitoring/services/acknowledge-problem')
|
||||
->setQueryString($this->serviceList->getUnhandledObjects()->objectsFilter()->toQueryString());
|
||||
$this->view->downtimeUnhandledLink = Url::fromPath('monitoring/services/schedule-downtime')
|
||||
->setQueryString($this->serviceList->getUnhandledObjects()->objectsFilter()->toQueryString());
|
||||
$this->view->downtimeLink = Url::fromPath('monitoring/services/schedule-downtime')
|
||||
->setQueryString($this->serviceList->getProblemObjects()->objectsFilter()->toQueryString());
|
||||
$this->view->acknowledgedObjects = $acknowledgedObjects;
|
||||
$this->view->acknowledgeLink = Url::fromPath('monitoring/services/acknowledge-problem')
|
||||
->setQueryString($this->serviceList->getUnacknowledgedObjects()->objectsFilter()->toQueryString());
|
||||
$this->view->unacknowledgedObjects = $this->serviceList->getUnacknowledgedObjects();
|
||||
$this->view->objectsInDowntime = $this->serviceList->getObjectsInDowntime();
|
||||
$this->view->inDowntimeLink = Url::fromPath('monitoring/list/services')
|
||||
->setQueryString($this->serviceList->getObjectsInDowntime()
|
||||
|
|
|
@ -1,6 +1,8 @@
|
|||
<?php
|
||||
/* Icinga Web 2 | (c) 2013-2015 Icinga Development Team | GPLv2+ */
|
||||
|
||||
use Icinga\Module\Monitoring\Object\Macro;
|
||||
|
||||
/**
|
||||
* Generate icons to describe a given hosts state
|
||||
*/
|
||||
|
@ -26,7 +28,7 @@ class Zend_View_Helper_IconImage extends Zend_View_Helper_Abstract
|
|||
{
|
||||
if ($object->host_icon_image && ! preg_match('/[\'"]/', $object->host_icon_image)) {
|
||||
return $this->view->img(
|
||||
'img/icons/' . $this->view->resolveMacros($object->host_icon_image, $object),
|
||||
'img/icons/' . Macro::resolveMacros($object->host_icon_image, $object),
|
||||
null,
|
||||
array(
|
||||
'alt' => $object->host_icon_image_alt,
|
||||
|
@ -48,7 +50,7 @@ class Zend_View_Helper_IconImage extends Zend_View_Helper_Abstract
|
|||
{
|
||||
if ($object->service_icon_image && ! preg_match('/[\'"]/', $object->service_icon_image)) {
|
||||
return $this->view->img(
|
||||
'img/icons/' . $this->view->resolveMacros($object->service_icon_image, $object),
|
||||
'img/icons/' . Macro::resolveMacros($object->service_icon_image, $object),
|
||||
null,
|
||||
array(
|
||||
'alt' => $object->service_icon_image_alt,
|
||||
|
|
|
@ -11,6 +11,7 @@
|
|||
|
||||
<table class="avp newsection">
|
||||
<tbody>
|
||||
<?= $this->render('show/components/notes.phtml') ?>
|
||||
<?= $this->render('show/components/acknowledgement.phtml') ?>
|
||||
<?= $this->render('show/components/comments.phtml') ?>
|
||||
<?= $this->render('show/components/notifications.phtml') ?>
|
||||
|
|
|
@ -72,9 +72,10 @@
|
|||
<?php
|
||||
$unhandledCount = count($unhandledObjects);
|
||||
$problemCount = count($problemObjects);
|
||||
$unackCount = count($unacknowledgedObjects);
|
||||
?>
|
||||
|
||||
<?php if ($problemCount || $unhandledCount): ?>
|
||||
<?php if ($problemCount || $unhandledCount || $unackCount): ?>
|
||||
<h3>
|
||||
<?= $this->icon('attention-alt') ?>
|
||||
<?= $this->translatePlural(
|
||||
|
@ -110,6 +111,24 @@
|
|||
); ?>
|
||||
<?php endif; ?>
|
||||
|
||||
<?php
|
||||
if ($unackCount > 0): ?>
|
||||
<br>
|
||||
<?= $this->qlink(
|
||||
sprintf(
|
||||
$this->translatePlural(
|
||||
'Acknowledge %u unacknowledged problem hosts',
|
||||
'Acknowledge %u unacknowledged problem hosts',
|
||||
$unackCount
|
||||
),
|
||||
$unackCount
|
||||
),
|
||||
$acknowledgeLink,
|
||||
null,
|
||||
array('icon' => 'ok')
|
||||
); ?>
|
||||
<?php endif; ?>
|
||||
|
||||
<?php if ($unhandledCount): ?>
|
||||
<p>
|
||||
<?= sprintf(
|
||||
|
@ -136,21 +155,6 @@
|
|||
array('icon' => 'plug')
|
||||
); ?>
|
||||
|
||||
<br>
|
||||
<?= $this->qlink(
|
||||
sprintf(
|
||||
$this->translatePlural(
|
||||
'Acknowledge %u unhandled problem host',
|
||||
'Acknowledge %u unhandled problem hosts',
|
||||
$unhandledCount
|
||||
),
|
||||
$unhandledCount
|
||||
),
|
||||
$acknowledgeUnhandledLink,
|
||||
null,
|
||||
array('icon' => 'ok')
|
||||
); ?>
|
||||
|
||||
<?php endif; ?>
|
||||
<?php endif;?>
|
||||
|
||||
|
|
|
@ -15,7 +15,15 @@ $this->baseFilter = isset($this->baseFilter) ? $this->baseFilter : null;
|
|||
$selfUrl = 'monitoring/list/hosts';
|
||||
$currentUrl = Url::fromRequest()->getRelativeUrl();
|
||||
?><div class="tinystatesummary" <?= $this->compact ? ' data-base-target="col1"' : ''; ?>>
|
||||
<?= sprintf($this->translate('%s hosts:'), $this->stats->hosts_total); ?>
|
||||
<?= $this->qlink(
|
||||
sprintf($this->translatePlural('%u Host', '%u Hosts', $this->stats->hosts_total), $this->stats->hosts_total),
|
||||
$selfUrl,
|
||||
null,
|
||||
array('title' => sprintf(
|
||||
$this->translatePlural('List %u host', 'List all %u hosts', $this->stats->hosts_total),
|
||||
$this->stats->hosts_total
|
||||
))
|
||||
) ?>:
|
||||
<span class="badges">
|
||||
<?php if ($this->stats->hosts_up): ?>
|
||||
<span class="state ok<?= $currentUrl === Url::fromPath($selfUrl, array('host_state' => 0))->getRelativeUrl() ? ' active' : ''; ?>">
|
||||
|
@ -156,4 +164,4 @@ $currentUrl = Url::fromRequest()->getRelativeUrl();
|
|||
</span>
|
||||
<?php endif; ?>
|
||||
</span>
|
||||
</div>
|
||||
</div>
|
||||
|
|
|
@ -17,7 +17,18 @@ $this->baseFilter = isset($this->baseFilter) ? $this->baseFilter : null;
|
|||
$selfUrl = 'monitoring/list/services';
|
||||
$currentUrl = Url::fromRequest()->getRelativeUrl();
|
||||
?><div class="tinystatesummary" <?= $this->compact ? ' data-base-target="col1"' : ''; ?>>
|
||||
<?= sprintf($this->translate('%s services:'), $this->stats->services_total); ?>
|
||||
<?= $this->qlink(
|
||||
sprintf($this->translatePlural(
|
||||
'%u Service', '%u Services', $this->stats->services_total),
|
||||
$this->stats->services_total
|
||||
),
|
||||
$selfUrl,
|
||||
null,
|
||||
array('title' => sprintf(
|
||||
$this->translatePlural('List %u service', 'List all %u services', $this->stats->services_total),
|
||||
$this->stats->services_total
|
||||
))
|
||||
) ?>:
|
||||
<span class="badges">
|
||||
<?php if ($this->stats->services_ok): ?>
|
||||
<span class="state ok<?= $currentUrl === Url::fromPath($selfUrl, array('service_state' => 0))->getRelativeUrl() ? ' active' : ''; ?>">
|
||||
|
@ -63,7 +74,7 @@ foreach (array(2 => 'critical', 3 => 'unknown', 1 => 'warning') as $stateId => $
|
|||
} else {
|
||||
$active = '';
|
||||
}
|
||||
|
||||
|
||||
echo '<span class="state ' . $state . $active . ($this->stats->$unhandled ? '' : ' handled') . '">';
|
||||
if ($this->stats->$unhandled) {
|
||||
|
||||
|
|
|
@ -11,6 +11,7 @@
|
|||
|
||||
<table class="avp newsection">
|
||||
<tbody>
|
||||
<?= $this->render('show/components/notes.phtml') ?>
|
||||
<?= $this->render('show/components/acknowledgement.phtml') ?>
|
||||
<?= $this->render('show/components/comments.phtml') ?>
|
||||
<?= $this->render('show/components/notifications.phtml') ?>
|
||||
|
|
|
@ -70,9 +70,10 @@
|
|||
<?php
|
||||
$unhandledCount = count($unhandledObjects);
|
||||
$problemCount = count($problemObjects);
|
||||
$unackCount = count($unacknowledgedObjects);
|
||||
?>
|
||||
|
||||
<?php if ($problemCount || $unhandledCount): ?>
|
||||
<?php if ($problemCount || $unhandledCount || $unackCount): ?>
|
||||
<div>
|
||||
<h3>
|
||||
<?= $this->icon('attention-alt') ?>
|
||||
|
@ -105,6 +106,23 @@
|
|||
); ?>
|
||||
<?php endif ?>
|
||||
|
||||
<?php if ($unackCount > 0): ?>
|
||||
<br>
|
||||
<?= $this->qlink(
|
||||
sprintf(
|
||||
$this->translatePlural(
|
||||
'Acknowledge %u unacknowledged problem service',
|
||||
'Acknowledge %u unacknowledged problem services',
|
||||
$unackCount
|
||||
),
|
||||
$unackCount
|
||||
),
|
||||
$acknowledgeLink,
|
||||
null,
|
||||
array('icon' => 'ok')
|
||||
); ?>
|
||||
<?php endif; ?>
|
||||
|
||||
<?php if ($unhandledCount): ?>
|
||||
<p>
|
||||
<?= sprintf($this->translate('There are %s unhandled problem services. ' .
|
||||
|
@ -126,20 +144,6 @@
|
|||
array('icon' => 'plug')
|
||||
); ?>
|
||||
|
||||
<br>
|
||||
<?= $this->qlink(
|
||||
sprintf(
|
||||
$this->translatePlural(
|
||||
'Acknowledge %u unhandled problem service',
|
||||
'Acknowledge %u unhandled problem services',
|
||||
$unhandledCount
|
||||
),
|
||||
$unhandledCount
|
||||
),
|
||||
$acknowledgeUnhandledLink,
|
||||
null,
|
||||
array('icon' => 'ok')
|
||||
); ?>
|
||||
<?php endif ?>
|
||||
|
||||
</div>
|
||||
|
|
|
@ -1,41 +1,16 @@
|
|||
<?php
|
||||
|
||||
$links = array();
|
||||
|
||||
// add warning to links that open in new tabs to improve accessibility, as recommended by WCAG20 G201
|
||||
$newTabInfo = sprintf('<span class="info-box display-on-hover"> %s </span>', $this->translate('opens in new window'));
|
||||
|
||||
$linkText = '<a href="%s" target="_blank">%s ' . $newTabInfo . '</a>';
|
||||
$localLinkText = '<a href="%s">%s</a>';
|
||||
|
||||
if ($object->notes_url) {
|
||||
if (strpos($object->notes_url, "' ") === false) {
|
||||
$links[] = sprintf($linkText, $this->resolveMacros($object->notes_url, $object), 'Notes');
|
||||
} else {
|
||||
// TODO: We should find out document what's going on here. Looks strange :p
|
||||
foreach(explode("' ", $object->notes_url) as $url) {
|
||||
$url = strpos($url, "'") === 0 ? substr($url, 1) : $url;
|
||||
$url = strrpos($url, "'") === strlen($url) - 1 ? substr($url, 0, strlen($url) - 1) : $url;
|
||||
$links[] = sprintf($linkText, $this->resolveMacros($url, $object), 'Notes');
|
||||
}
|
||||
}
|
||||
}
|
||||
if ($object->action_url) {
|
||||
if (strpos($object->action_url, "' ") === false) {
|
||||
$links[] = sprintf($linkText, $this->resolveMacros($object->action_url, $object), 'Action');
|
||||
} else {
|
||||
// TODO: We should find out document what's going on here. Looks strange :p
|
||||
foreach(explode("' ", $object->action_url) as $url) {
|
||||
$url = strpos($url, "'") === 0 ? substr($url, 1) : $url;
|
||||
$url = strrpos($url, "'") === strlen($url) - 1 ? substr($url, 0, strlen($url) - 1) : $url;
|
||||
$links[] = sprintf($linkText, $this->resolveMacros($url, $object), 'Action');
|
||||
}
|
||||
}
|
||||
$links = $object->getActionUrls();
|
||||
foreach ($links as $i => $link) {
|
||||
$links[$i] = sprintf('<a href="%s" target="_blank">%s ' . $newTabInfo . '</a>', $link, 'Action');
|
||||
}
|
||||
|
||||
if (isset($this->actions)) {
|
||||
foreach ($this->actions as $id => $action) {
|
||||
$links[] = sprintf($localLinkText, $action, $id);
|
||||
$links[] = sprintf('<a href="%s">%s</a>', $action, $id);
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -46,5 +21,5 @@ if (empty($links)) {
|
|||
?>
|
||||
<tr>
|
||||
<th><?= $this->translate('Actions') ?></th>
|
||||
<td><?= implode("\n ", $links) . "\n" ?></td>
|
||||
<td><?= implode("<br>", $links) ?></td>
|
||||
</tr>
|
||||
|
|
|
@ -0,0 +1,26 @@
|
|||
<?php
|
||||
$notes = trim($object->getNotes());
|
||||
$links = $object->getNotesUrls();
|
||||
|
||||
if (! empty($links) || ! empty($notes)): ?>
|
||||
<tr>
|
||||
<th><?= $this->translate('Notes') ?></th>
|
||||
<td>
|
||||
<?php
|
||||
if (! empty($notes)) {
|
||||
echo $notes . '<br>';
|
||||
}
|
||||
// add warning to links that open in new tabs to improve accessibility, as recommended by WCAG20 G201
|
||||
$newTabInfo = sprintf(
|
||||
'<span class="info-box display-on-hover"> %s </span>',
|
||||
$this->translate('opens in new window')
|
||||
);
|
||||
$linkText = '<a href="%s" target="_blank">%s ' . $newTabInfo . '</a>';
|
||||
foreach ($links as $i => $link) {
|
||||
$links[$i] = sprintf($linkText, $this->escape($link), $this->escape($link));
|
||||
}
|
||||
echo implode('<br>', $links);
|
||||
?>
|
||||
</td>
|
||||
</tr>
|
||||
<?php endif ?>
|
|
@ -0,0 +1,16 @@
|
|||
# <a id="monitoring-configuration"></a> Monitoring Module Configuration
|
||||
|
||||
## Overview
|
||||
|
||||
Apart from its web configuration capabilities, the local configuration is
|
||||
stored in `/etc/icingaweb2` by default (depending on your config setup).
|
||||
|
||||
Location | File | Description
|
||||
------------------------------|-----------------------|---------------------------
|
||||
modules/monitoring | Directory | `monitoring` module specific configuration
|
||||
modules/monitoring | config.ini | Security settings (e.g. protected custom vars) for the `monitoring` module
|
||||
modules/monitoring | backends.ini | Backend type and resources (e.g. Icinga IDO DB)
|
||||
modules/monitoring | [instances.ini](instances.md#instances) | Instances and their transport (e.g. local external command pipe)
|
||||
|
||||
|
||||
|
|
@ -43,6 +43,7 @@ class StatusQuery extends IdoQuery
|
|||
'host_icon_image' => 'h.icon_image',
|
||||
'host_icon_image_alt' => 'h.icon_image_alt',
|
||||
'host_action_url' => 'h.action_url',
|
||||
'host_notes' => 'h.notes',
|
||||
'host_notes_url' => 'h.notes_url'
|
||||
),
|
||||
'hoststatus' => array(
|
||||
|
@ -179,6 +180,7 @@ class StatusQuery extends IdoQuery
|
|||
'service_icon_image' => 's.icon_image',
|
||||
'service_icon_image_alt' => 's.icon_image_alt',
|
||||
'service_action_url' => 's.action_url',
|
||||
'service_notes' => 's.notes',
|
||||
'service_notes_url' => 's.notes_url',
|
||||
'object_type' => '(\'service\')'
|
||||
),
|
||||
|
|
|
@ -4,6 +4,7 @@
|
|||
namespace Icinga\Module\Monitoring\Command\Transport;
|
||||
|
||||
use Exception;
|
||||
use RuntimeException;
|
||||
use Icinga\Application\Logger;
|
||||
use Icinga\Exception\ConfigurationError;
|
||||
use Icinga\Module\Monitoring\Command\Exception\TransportException;
|
||||
|
@ -123,11 +124,15 @@ class LocalCommandFile implements CommandTransportInterface
|
|||
$file->fwrite($commandString . "\n");
|
||||
$file->fflush();
|
||||
} catch (Exception $e) {
|
||||
$message = $e->getMessage();
|
||||
if ($e instanceof RuntimeException && ($pos = strrpos($message, ':')) !== false) {
|
||||
// Assume RuntimeException thrown by SplFileObject in the format: __METHOD__ . "({$filename}): Message"
|
||||
$message = substr($message, $pos + 1);
|
||||
}
|
||||
throw new TransportException(
|
||||
'Can\'t send external Icinga command "%s" to the local command file "%s": %s',
|
||||
$commandString,
|
||||
'Can\'t send external Icinga command to the local command file "%s": %s',
|
||||
$this->path,
|
||||
$e
|
||||
$message
|
||||
);
|
||||
}
|
||||
}
|
||||
|
|
|
@ -201,8 +201,7 @@ class RemoteCommandFile implements CommandTransportInterface
|
|||
exec($ssh, $output, $status);
|
||||
if ($status !== 0) {
|
||||
throw new TransportException(
|
||||
'Can\'t send external Icinga command "%s": %s',
|
||||
$ssh,
|
||||
'Can\'t send external Icinga command: %s',
|
||||
implode(' ', $output)
|
||||
);
|
||||
}
|
||||
|
|
|
@ -7,7 +7,7 @@ use InvalidArgumentException;
|
|||
use Icinga\Module\Monitoring\Backend\MonitoringBackend;
|
||||
|
||||
/**
|
||||
* A Icinga host
|
||||
* An Icinga host
|
||||
*/
|
||||
class Host extends MonitoredObject
|
||||
{
|
||||
|
@ -119,6 +119,7 @@ class Host extends MonitoredObject
|
|||
'host_max_check_attempts',
|
||||
'host_name',
|
||||
'host_next_check',
|
||||
'host_notes',
|
||||
'host_notes_url',
|
||||
'host_notifications_enabled',
|
||||
'host_notifications_enabled_changed',
|
||||
|
@ -188,4 +189,16 @@ class Host extends MonitoredObject
|
|||
}
|
||||
return $text;
|
||||
}
|
||||
|
||||
public function getNotesUrls()
|
||||
{
|
||||
return $this->resolveAllStrings(
|
||||
MonitoredObject::parseAttributeUrls($this->host_notes_url)
|
||||
);
|
||||
}
|
||||
|
||||
public function getNotes()
|
||||
{
|
||||
return $this->host_notes;
|
||||
}
|
||||
}
|
||||
|
|
|
@ -99,4 +99,19 @@ class HostList extends ObjectList
|
|||
->applyFilter(clone $this->filter)
|
||||
->where('downtime_objecttype', 'host');
|
||||
}
|
||||
|
||||
/**
|
||||
* @return ObjectList
|
||||
*/
|
||||
public function getUnacknowledgedObjects()
|
||||
{
|
||||
$unhandledObjects = array();
|
||||
foreach ($this as $object) {
|
||||
if (! in_array((int) $object->state, array(0, 99)) &&
|
||||
(bool) $object->host_acknowledged === false) {
|
||||
$unhandledObjects[] = $object;
|
||||
}
|
||||
}
|
||||
return $this->newFromArray($unhandledObjects);
|
||||
}
|
||||
}
|
||||
|
|
|
@ -1,36 +1,41 @@
|
|||
<?php
|
||||
/* Icinga Web 2 | (c) 2013-2015 Icinga Development Team | GPLv2+ */
|
||||
|
||||
use \Zend_View_Helper_Abstract;
|
||||
use Icinga\Module\Monitoring\Object\MonitoredObject;
|
||||
namespace Icinga\Module\Monitoring\Object;
|
||||
|
||||
class Zend_View_Helper_ResolveMacros extends Zend_View_Helper_Abstract
|
||||
/**
|
||||
* Expand macros in string in the context of MonitoredObjects
|
||||
*/
|
||||
class Macro
|
||||
{
|
||||
/**
|
||||
* Known icinga macros
|
||||
*
|
||||
* @var array
|
||||
*/
|
||||
private $icingaMacros = array(
|
||||
'HOSTNAME' => 'host_name',
|
||||
'HOSTADDRESS' => 'host_address',
|
||||
'SERVICEDESC' => 'service_description'
|
||||
private static $icingaMacros = array(
|
||||
'HOSTNAME' => 'host_name',
|
||||
'HOSTADDRESS' => 'host_address',
|
||||
'SERVICEDESC' => 'service_description',
|
||||
'host.name' => 'host_name',
|
||||
'host.address' => 'host_address',
|
||||
'service.description' => 'service_description'
|
||||
);
|
||||
|
||||
/**
|
||||
* Return the given string with macros being resolved
|
||||
*
|
||||
* @param string $input The string in which to look for macros
|
||||
* @param MonitoredObject|stdClass $object The host or service used to resolve macros
|
||||
* @param MonitoredObject|stdClass $object The host or service used to resolve macros
|
||||
*
|
||||
* @return string The substituted or unchanged string
|
||||
*/
|
||||
public function resolveMacros($input, $object)
|
||||
public static function resolveMacros($input, $object)
|
||||
{
|
||||
$matches = array();
|
||||
if (preg_match_all('@\$([^\$\s]+)\$@', $input, $matches)) {
|
||||
foreach ($matches[1] as $key => $value) {
|
||||
$newValue = $this->resolveMacro($value, $object);
|
||||
$newValue = self::resolveMacro($value, $object);
|
||||
if ($newValue !== $value) {
|
||||
$input = str_replace($matches[0][$key], $newValue, $input);
|
||||
}
|
||||
|
@ -44,14 +49,14 @@ class Zend_View_Helper_ResolveMacros extends Zend_View_Helper_Abstract
|
|||
* Resolve a macro based on the given object
|
||||
*
|
||||
* @param string $macro The macro to resolve
|
||||
* @param MonitoredObject|stdClass $object The object used to resolve the macro
|
||||
* @param MonitoredObject|stdClass $object The object used to resolve the macro
|
||||
*
|
||||
* @return string The new value or the macro if it cannot be resolved
|
||||
*/
|
||||
public function resolveMacro($macro, $object)
|
||||
public static function resolveMacro($macro, $object)
|
||||
{
|
||||
if (array_key_exists($macro, $this->icingaMacros) && $object->{$this->icingaMacros[$macro]} !== false) {
|
||||
return $object->{$this->icingaMacros[$macro]};
|
||||
if (array_key_exists($macro, self::$icingaMacros) && $object->{self::$icingaMacros[$macro]} !== false) {
|
||||
return $object->{self::$icingaMacros[$macro]};
|
||||
}
|
||||
if (array_key_exists($macro, $object->customvars)) {
|
||||
return $object->customvars[$macro];
|
|
@ -563,4 +563,73 @@ abstract class MonitoredObject implements Filterable
|
|||
}
|
||||
return null;
|
||||
}
|
||||
|
||||
/**
|
||||
* The notes for this monitored object
|
||||
*
|
||||
* @return string The notes as a string
|
||||
*/
|
||||
public abstract function getNotes();
|
||||
|
||||
/**
|
||||
* Get all note urls configured for this monitored object
|
||||
*
|
||||
* @return array All note urls as a string
|
||||
*/
|
||||
public abstract function getNotesUrls();
|
||||
|
||||
/**
|
||||
* Get all action urls configured for this monitored object
|
||||
*
|
||||
* @return array All note urls as a string
|
||||
*/
|
||||
public function getActionUrls()
|
||||
{
|
||||
return $this->resolveAllStrings(
|
||||
MonitoredObject::parseAttributeUrls($this->action_url)
|
||||
);
|
||||
}
|
||||
|
||||
/**
|
||||
* Resolve macros in all given strings in the current object context
|
||||
*
|
||||
* @param array $strs An array of urls as string
|
||||
* @return type
|
||||
*/
|
||||
protected function resolveAllStrings(array $strs)
|
||||
{
|
||||
foreach ($strs as $i => $str) {
|
||||
$strs[$i] = Macro::resolveMacros($str, $this);
|
||||
}
|
||||
return $strs;
|
||||
}
|
||||
|
||||
/**
|
||||
* Parse the content of the action_url or notes_url attributes
|
||||
*
|
||||
* Find all occurences of http links, separated by whitespaces and quoted
|
||||
* by single or double-ticks.
|
||||
*
|
||||
* @link http://docs.icinga.org/latest/de/objectdefinitions.html
|
||||
*
|
||||
* @param string $urlString A string containing one or more urls
|
||||
* @return array Array of urls as strings
|
||||
*/
|
||||
public static function parseAttributeUrls($urlString)
|
||||
{
|
||||
if (empty($urlString)) {
|
||||
return array();
|
||||
}
|
||||
if (strpos($urlString, "' ") === false) {
|
||||
$links[] = $urlString;
|
||||
} else {
|
||||
// parse notes-url format
|
||||
foreach (explode("' ", $urlString) as $url) {
|
||||
$url = strpos($url, "'") === 0 ? substr($url, 1) : $url;
|
||||
$url = strrpos($url, "'") === strlen($url) - 1 ? substr($url, 0, strlen($url) - 1) : $url;
|
||||
$links[] = $url;
|
||||
}
|
||||
}
|
||||
return $links;
|
||||
}
|
||||
}
|
||||
|
|
|
@ -193,6 +193,11 @@ abstract class ObjectList implements Countable, IteratorAggregate
|
|||
return $this->newFromArray($handledObjects);
|
||||
}
|
||||
|
||||
/**
|
||||
* @return ObjectList
|
||||
*/
|
||||
public abstract function getUnacknowledgedObjects();
|
||||
|
||||
/**
|
||||
* Create a ObjectList from an array of hosts without querying a backend
|
||||
*
|
||||
|
|
|
@ -7,7 +7,7 @@ use InvalidArgumentException;
|
|||
use Icinga\Module\Monitoring\Backend\MonitoringBackend;
|
||||
|
||||
/**
|
||||
* A Icinga service
|
||||
* An Icinga service
|
||||
*/
|
||||
class Service extends MonitoredObject
|
||||
{
|
||||
|
@ -147,6 +147,7 @@ class Service extends MonitoredObject
|
|||
'service_last_state_change',
|
||||
'service_long_output',
|
||||
'service_next_check',
|
||||
'service_notes',
|
||||
'service_notes_url',
|
||||
'service_notifications_enabled',
|
||||
'service_notifications_enabled_changed',
|
||||
|
@ -198,4 +199,16 @@ class Service extends MonitoredObject
|
|||
}
|
||||
return $text;
|
||||
}
|
||||
|
||||
public function getNotesUrls()
|
||||
{
|
||||
return $this->resolveAllStrings(
|
||||
MonitoredObject::parseAttributeUrls($this->service_notes_url)
|
||||
);
|
||||
}
|
||||
|
||||
public function getNotes()
|
||||
{
|
||||
return $this->service_notes;
|
||||
}
|
||||
}
|
||||
|
|
|
@ -147,4 +147,19 @@ class ServiceList extends ObjectList
|
|||
->applyFilter(clone $this->filter)
|
||||
->where('downtime_objecttype', 'service');
|
||||
}
|
||||
|
||||
/**
|
||||
* @return ObjectList
|
||||
*/
|
||||
public function getUnacknowledgedObjects()
|
||||
{
|
||||
$unhandledObjects = array();
|
||||
foreach ($this as $object) {
|
||||
if (! in_array((int) $object->state, array(0, 99)) &&
|
||||
(bool) $object->service_acknowledged === false) {
|
||||
$unhandledObjects[] = $object;
|
||||
}
|
||||
}
|
||||
return $this->newFromArray($unhandledObjects);
|
||||
}
|
||||
}
|
||||
|
|
|
@ -1,78 +0,0 @@
|
|||
<?php
|
||||
/* Icinga Web 2 | (c) 2013-2015 Icinga Development Team | GPLv2+ */
|
||||
|
||||
namespace Tests\Icinga\Modules\Monitoring\Application\Views\Helpers;
|
||||
|
||||
use Mockery;
|
||||
use Zend_View_Helper_ResolveMacros;
|
||||
use Icinga\Test\BaseTestCase;
|
||||
|
||||
require_once realpath(BaseTestCase::$moduleDir . '/monitoring/application/views/helpers/ResolveMacros.php');
|
||||
|
||||
class ResolveMacrosTest extends BaseTestCase
|
||||
{
|
||||
public function testHostMacros()
|
||||
{
|
||||
$hostMock = Mockery::mock('host');
|
||||
$hostMock->host_name = 'test';
|
||||
$hostMock->host_address = '1.1.1.1';
|
||||
|
||||
$helper = new Zend_View_Helper_ResolveMacros();
|
||||
$this->assertEquals($helper->resolveMacros('$HOSTNAME$', $hostMock), $hostMock->host_name);
|
||||
$this->assertEquals($helper->resolveMacros('$HOSTADDRESS$', $hostMock), $hostMock->host_address);
|
||||
}
|
||||
|
||||
public function testServiceMacros()
|
||||
{
|
||||
$svcMock = Mockery::mock('service');
|
||||
$svcMock->host_name = 'test';
|
||||
$svcMock->host_address = '1.1.1.1';
|
||||
$svcMock->service_description = 'a service';
|
||||
|
||||
$helper = new Zend_View_Helper_ResolveMacros();
|
||||
$this->assertEquals($helper->resolveMacros('$HOSTNAME$', $svcMock), $svcMock->host_name);
|
||||
$this->assertEquals($helper->resolveMacros('$HOSTADDRESS$', $svcMock), $svcMock->host_address);
|
||||
$this->assertEquals($helper->resolveMacros('$SERVICEDESC$', $svcMock), $svcMock->service_description);
|
||||
}
|
||||
|
||||
public function testCustomvars()
|
||||
{
|
||||
$objectMock = Mockery::mock('object');
|
||||
$objectMock->customvars = array(
|
||||
'CUSTOMVAR' => 'test'
|
||||
);
|
||||
|
||||
$helper = new Zend_View_Helper_ResolveMacros();
|
||||
$this->assertEquals($helper->resolveMacros('$CUSTOMVAR$', $objectMock), $objectMock->customvars['CUSTOMVAR']);
|
||||
}
|
||||
|
||||
public function testFaultyMacros()
|
||||
{
|
||||
$hostMock = Mockery::mock('host');
|
||||
$hostMock->host_name = 'test';
|
||||
$hostMock->customvars = array(
|
||||
'HOST' => 'te',
|
||||
'NAME' => 'st'
|
||||
);
|
||||
|
||||
$helper = new Zend_View_Helper_ResolveMacros();
|
||||
$this->assertEquals(
|
||||
$helper->resolveMacros('$$HOSTNAME$ $ HOSTNAME$ $HOST$NAME$', $hostMock),
|
||||
'$test $ HOSTNAME$ teNAME$'
|
||||
);
|
||||
}
|
||||
|
||||
public function testMacrosWithSpecialCharacters()
|
||||
{
|
||||
$objectMock = Mockery::mock('object');
|
||||
$objectMock->customvars = array(
|
||||
'V€RY_SP3C|@L' => 'not too special!'
|
||||
);
|
||||
|
||||
$helper = new Zend_View_Helper_ResolveMacros();
|
||||
$this->assertEquals(
|
||||
$helper->resolveMacros('$V€RY_SP3C|@L$', $objectMock),
|
||||
$objectMock->customvars['V€RY_SP3C|@L']
|
||||
);
|
||||
}
|
||||
}
|
|
@ -0,0 +1,78 @@
|
|||
<?php
|
||||
/* Icinga Web 2 | (c) 2013-2015 Icinga Development Team | GPLv2+ */
|
||||
|
||||
namespace Tests\Icinga\Modules\Monitoring\Application\Views\Helpers;
|
||||
|
||||
use Mockery;
|
||||
use Icinga\Test\BaseTestCase;
|
||||
use Icinga\Module\Monitoring\Object\Macro;
|
||||
|
||||
require_once realpath(BaseTestCase::$moduleDir . '/monitoring/library/Monitoring/Object/Macro.php');
|
||||
|
||||
class MacroTest extends BaseTestCase
|
||||
{
|
||||
public function testHostMacros()
|
||||
{
|
||||
$hostMock = Mockery::mock('host');
|
||||
$hostMock->host_name = 'test';
|
||||
$hostMock->host_address = '1.1.1.1';
|
||||
|
||||
$this->assertEquals(Macro::resolveMacros('$HOSTNAME$', $hostMock), $hostMock->host_name);
|
||||
$this->assertEquals(Macro::resolveMacros('$HOSTADDRESS$', $hostMock), $hostMock->host_address);
|
||||
$this->assertEquals(Macro::resolveMacros('$host.name$', $hostMock), $hostMock->host_name);
|
||||
$this->assertEquals(Macro::resolveMacros('$host.address$', $hostMock), $hostMock->host_address);
|
||||
}
|
||||
|
||||
public function testServiceMacros()
|
||||
{
|
||||
$svcMock = Mockery::mock('service');
|
||||
$svcMock->host_name = 'test';
|
||||
$svcMock->host_address = '1.1.1.1';
|
||||
$svcMock->service_description = 'a service';
|
||||
|
||||
$this->assertEquals(Macro::resolveMacros('$HOSTNAME$', $svcMock), $svcMock->host_name);
|
||||
$this->assertEquals(Macro::resolveMacros('$HOSTADDRESS$', $svcMock), $svcMock->host_address);
|
||||
$this->assertEquals(Macro::resolveMacros('$SERVICEDESC$', $svcMock), $svcMock->service_description);
|
||||
$this->assertEquals(Macro::resolveMacros('$host.name$', $svcMock), $svcMock->host_name);
|
||||
$this->assertEquals(Macro::resolveMacros('$host.address$', $svcMock), $svcMock->host_address);
|
||||
$this->assertEquals(Macro::resolveMacros('$service.description$', $svcMock), $svcMock->service_description);
|
||||
}
|
||||
|
||||
public function testCustomvars()
|
||||
{
|
||||
$objectMock = Mockery::mock('object');
|
||||
$objectMock->customvars = array(
|
||||
'CUSTOMVAR' => 'test'
|
||||
);
|
||||
|
||||
$this->assertEquals(Macro::resolveMacros('$CUSTOMVAR$', $objectMock), $objectMock->customvars['CUSTOMVAR']);
|
||||
}
|
||||
|
||||
public function testFaultyMacros()
|
||||
{
|
||||
$hostMock = Mockery::mock('host');
|
||||
$hostMock->host_name = 'test';
|
||||
$hostMock->customvars = array(
|
||||
'HOST' => 'te',
|
||||
'NAME' => 'st'
|
||||
);
|
||||
|
||||
$this->assertEquals(
|
||||
Macro::resolveMacros('$$HOSTNAME$ $ HOSTNAME$ $HOST$NAME$', $hostMock),
|
||||
'$test $ HOSTNAME$ teNAME$'
|
||||
);
|
||||
}
|
||||
|
||||
public function testMacrosWithSpecialCharacters()
|
||||
{
|
||||
$objectMock = Mockery::mock('object');
|
||||
$objectMock->customvars = array(
|
||||
'V€RY_SP3C|@L' => 'not too special!'
|
||||
);
|
||||
|
||||
$this->assertEquals(
|
||||
Macro::resolveMacros('$V€RY_SP3C|@L$', $objectMock),
|
||||
$objectMock->customvars['V€RY_SP3C|@L']
|
||||
);
|
||||
}
|
||||
}
|
|
@ -96,11 +96,6 @@ select::-moz-focus-inner {
|
|||
outline: 0;
|
||||
}
|
||||
|
||||
input:disabled, select:disabled {
|
||||
background-color: #fff;
|
||||
border-color: white;
|
||||
}
|
||||
|
||||
form.inline {
|
||||
margin: 0;
|
||||
padding: 0;
|
||||
|
@ -263,4 +258,4 @@ form > div.header {
|
|||
h1, h2, h3, h4, h5, h6 {
|
||||
display: inline-block;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
|
|
@ -191,7 +191,6 @@ tr[href]:hover {
|
|||
}
|
||||
|
||||
tr.state[href]:hover td.state {
|
||||
color: white;
|
||||
background-color: #eee;
|
||||
}
|
||||
|
||||
|
|
|
@ -274,6 +274,10 @@
|
|||
}
|
||||
}
|
||||
|
||||
// Disable all form controls to prevent resubmission except for our search input
|
||||
// Note that disabled form inputs will not be enabled via JavaScript again
|
||||
$form.find(':input:not(#search):not(:disabled)').prop('disabled', true);
|
||||
|
||||
icinga.loader.loadUrl(url, $target, data, method);
|
||||
|
||||
return false;
|
||||
|
|
|
@ -521,12 +521,10 @@ EOD
|
|||
key1 = "1"
|
||||
key2 = "2"
|
||||
|
||||
|
||||
[two]
|
||||
a.b = "c"
|
||||
d.e = "f"
|
||||
|
||||
|
||||
[three]
|
||||
key = "value"
|
||||
foo.bar = "raboof"
|
||||
|
@ -537,12 +535,10 @@ EOD;
|
|||
key = "value"
|
||||
foo.bar = "raboof"
|
||||
|
||||
|
||||
[two]
|
||||
a.b = "c"
|
||||
d.e = "f"
|
||||
|
||||
|
||||
[one]
|
||||
key1 = "1"
|
||||
key2 = "2"
|
||||
|
@ -589,7 +585,6 @@ EOD;
|
|||
; comment 1
|
||||
[one]
|
||||
|
||||
|
||||
; comment 2
|
||||
[two]
|
||||
EOD;
|
||||
|
@ -598,7 +593,6 @@ EOD;
|
|||
; comment 2
|
||||
[two]
|
||||
|
||||
|
||||
; comment 1
|
||||
[one]
|
||||
EOD;
|
||||
|
@ -637,8 +631,8 @@ EOD;
|
|||
);
|
||||
|
||||
$this->assertEquals(
|
||||
$config,
|
||||
$writer->render(),
|
||||
trim($config),
|
||||
trim($writer->render()),
|
||||
'IniWriter does not preserve comments on empty lines'
|
||||
);
|
||||
}
|
||||
|
@ -667,8 +661,8 @@ EOD;
|
|||
);
|
||||
|
||||
$this->assertEquals(
|
||||
$config,
|
||||
$writer->render(),
|
||||
trim($config),
|
||||
trim($writer->render()),
|
||||
'IniWriter does not preserve comments on property lines'
|
||||
);
|
||||
}
|
||||
|
@ -686,8 +680,8 @@ EOD;
|
|||
);
|
||||
|
||||
$this->assertEquals(
|
||||
$config,
|
||||
$writer->render(),
|
||||
trim($config),
|
||||
trim($writer->render()),
|
||||
'IniWriter does not preserve comments on empty section lines'
|
||||
);
|
||||
}
|
||||
|
@ -719,8 +713,8 @@ EOD;
|
|||
);
|
||||
|
||||
$this->assertEquals(
|
||||
$config,
|
||||
$writer->render(),
|
||||
trim($config),
|
||||
trim($writer->render()),
|
||||
'IniWriter does not preserve comments on property lines'
|
||||
);
|
||||
}
|
||||
|
|
Loading…
Reference in New Issue