2013-06-27 10:14:15 +02:00
|
|
|
<?php
|
2013-08-08 17:42:34 +02:00
|
|
|
// {{{ICINGA_LICENSE_HEADER}}}
|
|
|
|
// {{{ICINGA_LICENSE_HEADER}}}
|
2013-06-27 10:14:15 +02:00
|
|
|
|
|
|
|
namespace Icinga\Web\Widget;
|
|
|
|
|
2014-02-18 19:11:44 +01:00
|
|
|
use Icinga\Application\Icinga;
|
2014-11-07 13:53:03 +01:00
|
|
|
use Icinga\Application\Config;
|
2014-11-04 16:15:06 +01:00
|
|
|
use Icinga\Data\ConfigObject;
|
2014-06-05 16:07:40 +02:00
|
|
|
use Icinga\Exception\ConfigurationError;
|
2014-11-18 16:28:04 +01:00
|
|
|
use Icinga\Exception\NotReadableError;
|
2014-03-04 12:18:28 +01:00
|
|
|
use Icinga\Exception\ProgrammingError;
|
2014-11-18 16:28:04 +01:00
|
|
|
use Icinga\Exception\SystemPermissionException;
|
2014-11-11 15:59:59 +01:00
|
|
|
use Icinga\File\Ini\IniWriter;
|
2014-11-11 14:44:38 +01:00
|
|
|
use Icinga\User;
|
2014-02-18 19:11:44 +01:00
|
|
|
use Icinga\Web\Widget\Dashboard\Pane;
|
|
|
|
use Icinga\Web\Widget\Dashboard\Component as DashboardComponent;
|
|
|
|
use Icinga\Web\Url;
|
2013-06-27 10:14:15 +02:00
|
|
|
|
2013-08-07 17:44:18 +02:00
|
|
|
/**
|
|
|
|
* Dashboards display multiple views on a single page
|
|
|
|
*
|
|
|
|
* The terminology is as follows:
|
|
|
|
* - Component: A single view showing a specific url
|
|
|
|
* - Pane: Aggregates one or more components on one page, displays it's title as a tab
|
|
|
|
* - Dashboard: Shows all panes
|
|
|
|
*
|
|
|
|
*/
|
2014-02-18 19:53:56 +01:00
|
|
|
class Dashboard extends AbstractWidget
|
2013-06-27 10:14:15 +02:00
|
|
|
{
|
2013-08-07 17:44:18 +02:00
|
|
|
/**
|
|
|
|
* An array containing all panes of this dashboard
|
|
|
|
*
|
|
|
|
* @var array
|
|
|
|
*/
|
2013-08-06 11:53:42 +02:00
|
|
|
private $panes = array();
|
2013-08-07 17:44:18 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* The @see Icinga\Web\Widget\Tabs object for displaying displayable panes
|
|
|
|
*
|
|
|
|
* @var Tabs
|
|
|
|
*/
|
2013-08-06 11:53:42 +02:00
|
|
|
private $tabs;
|
|
|
|
|
2013-08-07 17:44:18 +02:00
|
|
|
/**
|
|
|
|
* The parameter that will be added to identify panes
|
|
|
|
*
|
|
|
|
* @var string
|
|
|
|
*/
|
2013-08-06 11:53:42 +02:00
|
|
|
private $tabParam = 'pane';
|
2013-06-27 10:14:15 +02:00
|
|
|
|
2014-11-11 14:44:38 +01:00
|
|
|
/**
|
|
|
|
* @var User
|
|
|
|
*/
|
|
|
|
private $user;
|
|
|
|
|
2013-08-07 17:44:18 +02:00
|
|
|
/**
|
|
|
|
* Set the given tab name as active.
|
|
|
|
*
|
|
|
|
* @param string $name The tab name to activate
|
|
|
|
*
|
|
|
|
*/
|
2013-06-27 10:14:15 +02:00
|
|
|
public function activate($name)
|
|
|
|
{
|
2013-08-06 11:53:42 +02:00
|
|
|
$this->getTabs()->activate($name);
|
2013-06-27 10:14:15 +02:00
|
|
|
}
|
2013-08-07 17:44:18 +02:00
|
|
|
|
2014-08-26 10:13:49 +02:00
|
|
|
/**
|
|
|
|
* Load Pane items provided by all enabled modules
|
|
|
|
*
|
|
|
|
* @return self
|
|
|
|
*/
|
2014-11-11 14:44:38 +01:00
|
|
|
public function load()
|
2014-08-26 10:13:49 +02:00
|
|
|
{
|
|
|
|
$manager = Icinga::app()->getModuleManager();
|
|
|
|
foreach ($manager->getLoadedModules() as $module) {
|
|
|
|
/** @var $module \Icinga\Application\Modules\Module */
|
2014-11-11 14:44:38 +01:00
|
|
|
$this->mergePanes($module->getPaneItems());
|
2014-08-26 10:13:49 +02:00
|
|
|
|
|
|
|
}
|
2014-11-11 14:44:38 +01:00
|
|
|
if ($this->user !== null) {
|
|
|
|
$this->loadUserDashboards();
|
|
|
|
}
|
2014-11-18 16:28:04 +01:00
|
|
|
|
2014-11-11 14:44:38 +01:00
|
|
|
return $this;
|
|
|
|
}
|
|
|
|
|
2014-11-11 15:59:59 +01:00
|
|
|
/**
|
2014-11-19 11:47:31 +01:00
|
|
|
* Create a writer object
|
|
|
|
*
|
|
|
|
* @return IniWriter
|
2014-11-11 15:59:59 +01:00
|
|
|
*/
|
2014-11-19 11:47:31 +01:00
|
|
|
public function createWriter()
|
2014-11-11 15:59:59 +01:00
|
|
|
{
|
|
|
|
$configFile = $this->getConfigFile();
|
|
|
|
$output = array();
|
|
|
|
foreach ($this->panes as $pane) {
|
|
|
|
if ($pane->isUserWidget() === true) {
|
|
|
|
$output[$pane->getName()] = $pane->toArray();
|
|
|
|
}
|
|
|
|
foreach ($pane->getComponents() as $component) {
|
|
|
|
if ($component->isUserWidget() === true) {
|
|
|
|
$output[$pane->getName() . '.' . $component->getTitle()] = $component->toArray();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-11-04 16:15:06 +01:00
|
|
|
$co = new ConfigObject($output);
|
|
|
|
$config = new Config($co);
|
2014-11-19 11:47:31 +01:00
|
|
|
return new IniWriter(array('config' => $config, 'filename' => $configFile));
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Write user specific dashboards to disk
|
|
|
|
*/
|
|
|
|
public function write()
|
|
|
|
{
|
|
|
|
$this->createWriter()->write();
|
2014-11-11 15:59:59 +01:00
|
|
|
}
|
|
|
|
|
2014-11-11 14:44:38 +01:00
|
|
|
/**
|
|
|
|
* @return bool
|
|
|
|
*/
|
|
|
|
private function loadUserDashboards()
|
|
|
|
{
|
2014-11-19 10:33:41 +01:00
|
|
|
try {
|
|
|
|
$config = Config::fromIni($this->getConfigFile());
|
|
|
|
} catch (NotReadableError $e) {
|
|
|
|
return;
|
|
|
|
}
|
2014-11-11 14:44:38 +01:00
|
|
|
if (! count($config)) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
$panes = array();
|
|
|
|
$components = array();
|
|
|
|
foreach ($config as $key => $part) {
|
|
|
|
if (strpos($key, '.') === false) {
|
2014-11-18 16:28:04 +01:00
|
|
|
if ($this->hasPane($part->title)) {
|
|
|
|
$panes[$key] = $this->getPane($part->title);
|
|
|
|
} else {
|
|
|
|
$panes[$key] = new Pane($key);
|
|
|
|
$panes[$key]->setTitle($part->title);
|
|
|
|
}
|
2014-11-11 15:38:51 +01:00
|
|
|
$panes[$key]->setUserWidget();
|
2014-11-18 16:28:04 +01:00
|
|
|
if ((bool) $part->get('disabled', false) === true) {
|
|
|
|
$panes[$key]->setDisabled();
|
|
|
|
}
|
2014-11-11 14:44:38 +01:00
|
|
|
|
|
|
|
} else {
|
|
|
|
list($paneName, $componentName) = explode('.', $key, 2);
|
|
|
|
$part->pane = $paneName;
|
|
|
|
$part->component = $componentName;
|
|
|
|
$components[] = $part;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
foreach ($components as $componentData) {
|
|
|
|
$pane = null;
|
|
|
|
|
|
|
|
if (array_key_exists($componentData->pane, $panes) === true) {
|
|
|
|
$pane = $panes[$componentData->pane];
|
|
|
|
} elseif (array_key_exists($componentData->pane, $this->panes) === true) {
|
|
|
|
$pane = $this->panes[$componentData->pane];
|
|
|
|
} else {
|
|
|
|
continue;
|
|
|
|
}
|
2014-11-11 15:38:51 +01:00
|
|
|
$component = new DashboardComponent(
|
|
|
|
$componentData->title,
|
|
|
|
$componentData->url,
|
|
|
|
$pane
|
2014-11-11 14:44:38 +01:00
|
|
|
);
|
2014-11-11 15:38:51 +01:00
|
|
|
|
|
|
|
if ((bool) $componentData->get('disabled', false) === true) {
|
|
|
|
$component->setDisabled(true);
|
|
|
|
}
|
|
|
|
|
|
|
|
$component->setUserWidget();
|
|
|
|
$pane->addComponent($component);
|
2014-11-11 14:44:38 +01:00
|
|
|
}
|
2014-11-11 15:38:51 +01:00
|
|
|
|
2014-11-11 14:44:38 +01:00
|
|
|
$this->mergePanes($panes);
|
2014-11-11 15:38:51 +01:00
|
|
|
|
2014-11-11 14:44:38 +01:00
|
|
|
return true;
|
2014-08-26 10:13:49 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Merge panes with existing panes
|
|
|
|
*
|
2014-11-11 14:44:38 +01:00
|
|
|
* @param array $panes
|
|
|
|
*
|
|
|
|
* @return $this
|
2014-08-26 10:13:49 +02:00
|
|
|
*/
|
|
|
|
public function mergePanes(array $panes)
|
|
|
|
{
|
|
|
|
/** @var $pane Pane */
|
|
|
|
foreach ($panes as $pane) {
|
2014-11-18 16:28:04 +01:00
|
|
|
if ($pane->getDisabled()) {
|
|
|
|
if ($this->hasPane($pane->getTitle()) === true) {
|
|
|
|
$this->removePane($pane->getTitle());
|
|
|
|
}
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if ($this->hasPane($pane->getTitle()) === true) {
|
2014-08-26 10:13:49 +02:00
|
|
|
/** @var $current Pane */
|
|
|
|
$current = $this->panes[$pane->getName()];
|
|
|
|
$current->addComponents($pane->getComponents());
|
|
|
|
} else {
|
2014-09-03 14:36:04 +02:00
|
|
|
$this->panes[$pane->getName()] = $pane;
|
2014-08-26 10:13:49 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return $this;
|
|
|
|
}
|
|
|
|
|
2013-08-07 17:44:18 +02:00
|
|
|
/**
|
|
|
|
* Return the tab object used to navigate through this dashboard
|
|
|
|
*
|
|
|
|
* @return Tabs
|
|
|
|
*/
|
2013-08-06 11:53:42 +02:00
|
|
|
public function getTabs()
|
2013-06-27 10:14:15 +02:00
|
|
|
{
|
2014-11-18 09:50:10 +01:00
|
|
|
$url = Url::fromPath('dashboard')->getUrlWithout($this->tabParam);
|
2013-06-27 10:14:15 +02:00
|
|
|
if ($this->tabs === null) {
|
2013-08-06 18:00:33 +02:00
|
|
|
$this->tabs = new Tabs();
|
2013-08-06 11:53:42 +02:00
|
|
|
|
2013-08-07 17:40:18 +02:00
|
|
|
foreach ($this->panes as $key => $pane) {
|
2013-08-07 18:10:39 +02:00
|
|
|
$this->tabs->add(
|
|
|
|
$key,
|
|
|
|
array(
|
|
|
|
'title' => $pane->getTitle(),
|
|
|
|
'url' => clone($url),
|
|
|
|
'urlParams' => array($this->tabParam => $key)
|
|
|
|
)
|
|
|
|
);
|
2013-06-27 10:14:15 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return $this->tabs;
|
|
|
|
}
|
|
|
|
|
2014-09-03 14:36:04 +02:00
|
|
|
/**
|
|
|
|
* Return all panes of this dashboard
|
|
|
|
*
|
|
|
|
* @return array
|
|
|
|
*/
|
|
|
|
public function getPanes()
|
|
|
|
{
|
|
|
|
return $this->panes;
|
|
|
|
}
|
|
|
|
|
2013-06-27 10:14:15 +02:00
|
|
|
|
2013-08-07 17:44:18 +02:00
|
|
|
/**
|
|
|
|
* Creates a new empty pane with the given title
|
|
|
|
*
|
2013-08-08 17:42:34 +02:00
|
|
|
* @param string $title
|
|
|
|
*
|
2013-08-09 10:32:57 +02:00
|
|
|
* @return self
|
2013-08-07 17:44:18 +02:00
|
|
|
*/
|
2013-08-06 11:53:42 +02:00
|
|
|
public function createPane($title)
|
|
|
|
{
|
|
|
|
$pane = new Pane($title);
|
|
|
|
$pane->setTitle($title);
|
|
|
|
$this->addPane($pane);
|
2013-08-08 17:42:34 +02:00
|
|
|
|
|
|
|
return $this;
|
2013-08-06 11:53:42 +02:00
|
|
|
}
|
|
|
|
|
2014-08-26 10:13:49 +02:00
|
|
|
/**
|
|
|
|
* Checks if the current dashboard has any panes
|
|
|
|
*
|
|
|
|
* @return bool
|
|
|
|
*/
|
|
|
|
public function hasPanes()
|
|
|
|
{
|
|
|
|
return ! empty($this->panes);
|
|
|
|
}
|
|
|
|
|
2014-11-18 16:28:04 +01:00
|
|
|
/**
|
|
|
|
* Check if a panel exist
|
|
|
|
*
|
|
|
|
* @param string $pane
|
|
|
|
* @return bool
|
|
|
|
*/
|
|
|
|
public function hasPane($pane)
|
|
|
|
{
|
|
|
|
return $pane && array_key_exists($pane, $this->panes);
|
|
|
|
}
|
|
|
|
|
2013-08-07 17:44:18 +02:00
|
|
|
/**
|
|
|
|
* Add a pane object to this dashboard
|
|
|
|
*
|
|
|
|
* @param Pane $pane The pane to add
|
|
|
|
*
|
2013-08-09 10:32:57 +02:00
|
|
|
* @return self
|
2013-08-07 17:44:18 +02:00
|
|
|
*/
|
2013-06-27 10:14:15 +02:00
|
|
|
public function addPane(Pane $pane)
|
|
|
|
{
|
|
|
|
$this->panes[$pane->getName()] = $pane;
|
|
|
|
return $this;
|
|
|
|
}
|
|
|
|
|
2014-11-18 16:28:04 +01:00
|
|
|
public function removePane($title)
|
|
|
|
{
|
|
|
|
if ($this->hasPane($title) === true) {
|
|
|
|
$pane = $this->getPane($title);
|
|
|
|
if ($pane->isUserWidget() === true) {
|
|
|
|
unset($this->panes[$pane->getName()]);
|
|
|
|
} else {
|
|
|
|
$pane->setDisabled();
|
|
|
|
$pane->setUserWidget();
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
throw new ProgrammingError('Pane not found: ' . $title);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-08-07 17:44:18 +02:00
|
|
|
/**
|
2014-03-04 12:18:28 +01:00
|
|
|
* Return the pane with the provided name
|
2013-08-07 17:44:18 +02:00
|
|
|
*
|
|
|
|
* @param string $name The name of the pane to return
|
|
|
|
*
|
2014-03-04 12:18:28 +01:00
|
|
|
* @return Pane The pane or null if no pane with the given name exists
|
|
|
|
* @throws ProgrammingError
|
2013-08-07 17:44:18 +02:00
|
|
|
*/
|
2013-06-27 10:14:15 +02:00
|
|
|
public function getPane($name)
|
|
|
|
{
|
2014-03-04 12:18:28 +01:00
|
|
|
if (! array_key_exists($name, $this->panes)) {
|
|
|
|
throw new ProgrammingError(
|
2014-08-26 11:15:19 +02:00
|
|
|
'Trying to retrieve invalid dashboard pane "%s"',
|
|
|
|
$name
|
2014-03-04 12:18:28 +01:00
|
|
|
);
|
2013-08-07 17:40:18 +02:00
|
|
|
}
|
2013-06-27 10:14:15 +02:00
|
|
|
return $this->panes[$name];
|
|
|
|
}
|
2013-08-07 17:44:18 +02:00
|
|
|
|
2014-11-12 09:22:39 +01:00
|
|
|
/**
|
|
|
|
* Return an array with pane name=>title format used for comboboxes
|
|
|
|
*
|
|
|
|
* @return array
|
|
|
|
*/
|
|
|
|
public function getPaneKeyTitleArray()
|
|
|
|
{
|
|
|
|
$list = array();
|
|
|
|
foreach ($this->panes as $name => $pane) {
|
|
|
|
$list[$name] = $pane->getTitle();
|
|
|
|
}
|
|
|
|
return $list;
|
|
|
|
}
|
|
|
|
|
2013-08-07 17:44:18 +02:00
|
|
|
/**
|
|
|
|
* @see Icinga\Web\Widget::render
|
|
|
|
*/
|
2014-03-17 17:04:09 +01:00
|
|
|
public function render()
|
2013-06-27 10:14:15 +02:00
|
|
|
{
|
|
|
|
if (empty($this->panes)) {
|
|
|
|
return '';
|
|
|
|
}
|
2014-11-18 16:28:04 +01:00
|
|
|
|
2014-03-17 17:04:09 +01:00
|
|
|
return $this->determineActivePane()->render();
|
2013-08-06 11:53:42 +02:00
|
|
|
}
|
|
|
|
|
2013-08-07 17:44:18 +02:00
|
|
|
/**
|
|
|
|
* Activates the default pane of this dashboard and returns it's name
|
|
|
|
*
|
|
|
|
* @return mixed
|
|
|
|
*/
|
2013-08-06 11:53:42 +02:00
|
|
|
private function setDefaultPane()
|
|
|
|
{
|
|
|
|
reset($this->panes);
|
|
|
|
$active = key($this->panes);
|
|
|
|
$this->activate($active);
|
|
|
|
return $active;
|
2013-06-27 10:14:15 +02:00
|
|
|
}
|
|
|
|
|
2014-09-03 14:36:04 +02:00
|
|
|
/**
|
|
|
|
* @see determineActivePane()
|
|
|
|
*/
|
2014-03-04 12:18:28 +01:00
|
|
|
public function getActivePane()
|
|
|
|
{
|
|
|
|
return $this->determineActivePane();
|
|
|
|
}
|
|
|
|
|
2013-08-07 17:44:18 +02:00
|
|
|
/**
|
|
|
|
* Determine the active pane either by the selected tab or the current request
|
|
|
|
*
|
2014-11-11 14:44:38 +01:00
|
|
|
* @throws \Icinga\Exception\ConfigurationError
|
|
|
|
* @throws \Icinga\Exception\ProgrammingError
|
|
|
|
*
|
|
|
|
* @return Pane The currently active pane
|
2013-08-07 17:44:18 +02:00
|
|
|
*/
|
|
|
|
public function determineActivePane()
|
2013-06-27 10:14:15 +02:00
|
|
|
{
|
2013-08-06 11:53:42 +02:00
|
|
|
$active = $this->getTabs()->getActiveName();
|
2014-02-18 19:11:44 +01:00
|
|
|
if (! $active) {
|
2013-07-31 10:59:34 +02:00
|
|
|
if ($active = Url::fromRequest()->getParam($this->tabParam)) {
|
2014-09-03 14:36:04 +02:00
|
|
|
if ($this->hasPane($active)) {
|
2013-08-06 11:53:42 +02:00
|
|
|
$this->activate($active);
|
2014-09-03 14:36:04 +02:00
|
|
|
} else {
|
|
|
|
throw new ProgrammingError(
|
|
|
|
'Try to get an inexistent pane.'
|
|
|
|
);
|
2013-08-06 11:53:42 +02:00
|
|
|
}
|
2013-06-27 10:14:15 +02:00
|
|
|
} else {
|
2013-08-06 11:53:42 +02:00
|
|
|
$active = $this->setDefaultPane();
|
2013-06-27 10:14:15 +02:00
|
|
|
}
|
|
|
|
}
|
2014-06-05 16:07:40 +02:00
|
|
|
|
|
|
|
if (isset($this->panes[$active])) {
|
|
|
|
return $this->panes[$active];
|
|
|
|
}
|
|
|
|
|
|
|
|
throw new ConfigurationError('Could not determine active pane');
|
2013-06-27 10:14:15 +02:00
|
|
|
}
|
|
|
|
|
2013-08-07 17:44:18 +02:00
|
|
|
/**
|
2014-11-11 15:59:59 +01:00
|
|
|
* Setter for user object
|
|
|
|
*
|
|
|
|
* @param User $user
|
2013-08-07 17:44:18 +02:00
|
|
|
*/
|
2014-11-11 15:59:59 +01:00
|
|
|
public function setUser(User $user)
|
2013-06-27 10:14:15 +02:00
|
|
|
{
|
2014-11-11 14:44:38 +01:00
|
|
|
$this->user = $user;
|
2013-06-27 10:14:15 +02:00
|
|
|
}
|
2013-08-07 17:44:18 +02:00
|
|
|
|
|
|
|
/**
|
2014-11-11 15:59:59 +01:00
|
|
|
* Getter for user object
|
|
|
|
*
|
|
|
|
* @return User
|
2013-08-07 17:44:18 +02:00
|
|
|
*/
|
2014-11-11 14:44:38 +01:00
|
|
|
public function getUser()
|
2013-06-27 10:14:15 +02:00
|
|
|
{
|
2014-11-11 14:44:38 +01:00
|
|
|
return $this->user;
|
2013-06-27 10:14:15 +02:00
|
|
|
}
|
2014-11-11 15:38:51 +01:00
|
|
|
|
2014-11-11 15:59:59 +01:00
|
|
|
/**
|
|
|
|
* Get config file
|
|
|
|
*
|
|
|
|
* @return string
|
|
|
|
*/
|
2014-11-11 15:38:51 +01:00
|
|
|
public function getConfigFile()
|
|
|
|
{
|
|
|
|
if ($this->user === null) {
|
|
|
|
return '';
|
|
|
|
}
|
2014-11-18 16:28:04 +01:00
|
|
|
|
|
|
|
$baseDir = '/var/lib/icingaweb';
|
|
|
|
|
|
|
|
if (! file_exists($baseDir)) {
|
|
|
|
throw new NotReadableError('Could not read: ' . $baseDir);
|
|
|
|
}
|
|
|
|
|
|
|
|
$userDir = $baseDir . '/' . $this->user->getUsername();
|
|
|
|
|
|
|
|
if (! file_exists($userDir)) {
|
|
|
|
$success = @mkdir($userDir);
|
|
|
|
if (!$success) {
|
|
|
|
throw new SystemPermissionException('Could not create: ' . $userDir);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (! file_exists($userDir)) {
|
|
|
|
throw new NotReadableError('Could not read: ' . $userDir);
|
|
|
|
}
|
|
|
|
|
|
|
|
return $userDir . '/dashboard.ini';
|
2014-11-11 15:38:51 +01:00
|
|
|
}
|
2013-06-27 10:14:15 +02:00
|
|
|
}
|