2015-09-07 16:58:22 +02:00
|
|
|
<?php
|
2016-02-08 15:41:00 +01:00
|
|
|
/* Icinga Web 2 | (c) 2015 Icinga Development Team | GPLv2+ */
|
2015-09-07 16:58:22 +02:00
|
|
|
|
|
|
|
namespace Icinga\Forms\Navigation;
|
|
|
|
|
|
|
|
use InvalidArgumentException;
|
2015-09-15 15:57:03 +02:00
|
|
|
use Icinga\Application\Config;
|
2015-09-17 08:24:23 +02:00
|
|
|
use Icinga\Application\Logger;
|
|
|
|
use Icinga\Application\Icinga;
|
2015-09-16 16:25:17 +02:00
|
|
|
use Icinga\Authentication\Auth;
|
2015-09-25 10:51:44 +02:00
|
|
|
use Icinga\Data\ConfigObject;
|
2015-09-07 16:58:22 +02:00
|
|
|
use Icinga\Exception\IcingaException;
|
|
|
|
use Icinga\Exception\NotFoundError;
|
2015-09-18 09:43:40 +02:00
|
|
|
use Icinga\Exception\ProgrammingError;
|
2015-09-07 16:58:22 +02:00
|
|
|
use Icinga\Forms\ConfigForm;
|
2015-09-15 15:57:03 +02:00
|
|
|
use Icinga\User;
|
2016-01-27 15:56:27 +01:00
|
|
|
use Icinga\Util\StringHelper;
|
2015-09-07 16:58:22 +02:00
|
|
|
use Icinga\Web\Form;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Form for managing navigation items
|
|
|
|
*/
|
|
|
|
class NavigationConfigForm extends ConfigForm
|
|
|
|
{
|
2015-09-17 08:24:23 +02:00
|
|
|
/**
|
|
|
|
* The class namespace where to locate navigation type forms
|
|
|
|
*
|
|
|
|
* @var string
|
|
|
|
*/
|
|
|
|
const FORM_NS = 'Forms\\Navigation';
|
|
|
|
|
2015-09-16 10:58:57 +02:00
|
|
|
/**
|
|
|
|
* The secondary configuration to write
|
|
|
|
*
|
|
|
|
* This is always the reduced configuration and is only written to
|
|
|
|
* disk once the main configuration has been successfully written.
|
|
|
|
*
|
|
|
|
* @var Config
|
|
|
|
*/
|
|
|
|
protected $secondaryConfig;
|
|
|
|
|
2015-09-07 16:58:22 +02:00
|
|
|
/**
|
|
|
|
* The navigation item to load when displaying the form for the first time
|
|
|
|
*
|
|
|
|
* @var string
|
|
|
|
*/
|
|
|
|
protected $itemToLoad;
|
|
|
|
|
2015-09-15 15:57:03 +02:00
|
|
|
/**
|
|
|
|
* The user for whom to manage navigation items
|
|
|
|
*
|
|
|
|
* @var User
|
|
|
|
*/
|
|
|
|
protected $user;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The user's navigation configuration
|
|
|
|
*
|
|
|
|
* @var Config
|
|
|
|
*/
|
|
|
|
protected $userConfig;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The shared navigation configuration
|
|
|
|
*
|
|
|
|
* @var Config
|
|
|
|
*/
|
|
|
|
protected $shareConfig;
|
|
|
|
|
2015-09-16 15:05:43 +02:00
|
|
|
/**
|
|
|
|
* The available navigation item types
|
|
|
|
*
|
|
|
|
* @var array
|
|
|
|
*/
|
|
|
|
protected $itemTypes;
|
|
|
|
|
2015-10-01 13:55:12 +02:00
|
|
|
private $defaultUrl;
|
|
|
|
|
2015-09-07 16:58:22 +02:00
|
|
|
/**
|
|
|
|
* Initialize this form
|
|
|
|
*/
|
|
|
|
public function init()
|
|
|
|
{
|
|
|
|
$this->setName('form_config_navigation');
|
|
|
|
$this->setSubmitLabel($this->translate('Save Changes'));
|
|
|
|
}
|
|
|
|
|
2015-09-15 15:57:03 +02:00
|
|
|
/**
|
|
|
|
* Set the user for whom to manage navigation items
|
|
|
|
*
|
|
|
|
* @param User $user
|
|
|
|
*
|
|
|
|
* @return $this
|
|
|
|
*/
|
|
|
|
public function setUser(User $user)
|
|
|
|
{
|
|
|
|
$this->user = $user;
|
|
|
|
return $this;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Return the user for whom to manage navigation items
|
|
|
|
*
|
|
|
|
* @return User
|
|
|
|
*/
|
|
|
|
public function getUser()
|
|
|
|
{
|
|
|
|
return $this->user;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Set the user's navigation configuration
|
|
|
|
*
|
|
|
|
* @param Config $config
|
|
|
|
*
|
|
|
|
* @return $this
|
|
|
|
*/
|
|
|
|
public function setUserConfig(Config $config)
|
|
|
|
{
|
2015-09-24 15:49:04 +02:00
|
|
|
$config->getConfigObject()->setKeyColumn('name');
|
2015-09-15 15:57:03 +02:00
|
|
|
$this->userConfig = $config;
|
|
|
|
return $this;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Return the user's navigation configuration
|
|
|
|
*
|
2015-09-29 17:06:22 +02:00
|
|
|
* @param string $type
|
|
|
|
*
|
2015-09-15 15:57:03 +02:00
|
|
|
* @return Config
|
|
|
|
*/
|
2015-09-29 17:06:22 +02:00
|
|
|
public function getUserConfig($type = null)
|
2015-09-15 15:57:03 +02:00
|
|
|
{
|
2015-10-01 10:57:00 +02:00
|
|
|
if ($this->userConfig === null || $type !== null) {
|
2015-09-29 17:06:22 +02:00
|
|
|
if ($type === null) {
|
|
|
|
throw new ProgrammingError('You need to pass a type if no user configuration is set');
|
|
|
|
}
|
|
|
|
|
2015-09-30 11:30:38 +02:00
|
|
|
$this->setUserConfig(Config::navigation($type, $this->getUser()->getUsername()));
|
2015-09-15 15:57:03 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
return $this->userConfig;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Set the shared navigation configuration
|
|
|
|
*
|
|
|
|
* @param Config $config
|
|
|
|
*
|
|
|
|
* @return $this
|
|
|
|
*/
|
|
|
|
public function setShareConfig(Config $config)
|
|
|
|
{
|
2015-09-24 15:49:04 +02:00
|
|
|
$config->getConfigObject()->setKeyColumn('name');
|
2015-09-15 15:57:03 +02:00
|
|
|
$this->shareConfig = $config;
|
|
|
|
return $this;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Return the shared navigation configuration
|
|
|
|
*
|
2015-09-29 17:07:04 +02:00
|
|
|
* @param string $type
|
|
|
|
*
|
2015-09-15 15:57:03 +02:00
|
|
|
* @return Config
|
|
|
|
*/
|
2015-09-29 17:07:04 +02:00
|
|
|
public function getShareConfig($type = null)
|
2015-09-15 15:57:03 +02:00
|
|
|
{
|
2015-09-29 17:07:04 +02:00
|
|
|
if ($this->shareConfig === null) {
|
|
|
|
if ($type === null) {
|
|
|
|
throw new ProgrammingError('You need to pass a type if no share configuration is set');
|
|
|
|
}
|
|
|
|
|
2015-09-30 11:30:38 +02:00
|
|
|
$this->setShareConfig(Config::navigation($type));
|
2015-09-29 17:07:04 +02:00
|
|
|
}
|
|
|
|
|
2015-09-15 15:57:03 +02:00
|
|
|
return $this->shareConfig;
|
|
|
|
}
|
|
|
|
|
2015-09-16 15:05:43 +02:00
|
|
|
/**
|
|
|
|
* Set the available navigation item types
|
|
|
|
*
|
|
|
|
* @param array $itemTypes
|
|
|
|
*
|
|
|
|
* @return $this
|
|
|
|
*/
|
|
|
|
public function setItemTypes(array $itemTypes)
|
|
|
|
{
|
|
|
|
$this->itemTypes = $itemTypes;
|
|
|
|
return $this;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Return the available navigation item types
|
|
|
|
*
|
|
|
|
* @return array
|
|
|
|
*/
|
|
|
|
public function getItemTypes()
|
|
|
|
{
|
|
|
|
return $this->itemTypes ?: array();
|
|
|
|
}
|
|
|
|
|
2015-09-17 13:45:47 +02:00
|
|
|
/**
|
|
|
|
* Return a list of available parent items for the given type of navigation item
|
|
|
|
*
|
2015-09-21 15:29:14 +02:00
|
|
|
* @param string $type
|
|
|
|
* @param string $owner
|
|
|
|
*
|
2015-09-17 13:45:47 +02:00
|
|
|
* @return array
|
|
|
|
*/
|
2015-09-21 15:29:14 +02:00
|
|
|
public function listAvailableParents($type, $owner = null)
|
2015-09-17 13:45:47 +02:00
|
|
|
{
|
2015-09-18 11:01:02 +02:00
|
|
|
$children = $this->itemToLoad ? $this->getFlattenedChildren($this->itemToLoad) : array();
|
2015-09-17 13:45:47 +02:00
|
|
|
|
|
|
|
$names = array();
|
2015-09-29 17:07:04 +02:00
|
|
|
foreach ($this->getShareConfig($type) as $sectionName => $sectionConfig) {
|
2019-06-26 11:40:30 +02:00
|
|
|
if ((string) $sectionName !== $this->itemToLoad
|
2015-09-21 15:29:14 +02:00
|
|
|
&& $sectionConfig->owner === ($owner ?: $this->getUser()->getUsername())
|
2017-11-09 14:17:46 +01:00
|
|
|
&& ! in_array($sectionName, $children, true)
|
2015-09-18 11:01:02 +02:00
|
|
|
) {
|
|
|
|
$names[] = $sectionName;
|
2015-09-17 13:45:47 +02:00
|
|
|
}
|
2015-09-18 11:01:02 +02:00
|
|
|
}
|
|
|
|
|
2015-09-29 17:06:22 +02:00
|
|
|
foreach ($this->getUserConfig($type) as $sectionName => $sectionConfig) {
|
2019-06-26 11:40:30 +02:00
|
|
|
if ((string) $sectionName !== $this->itemToLoad
|
2017-11-09 14:17:46 +01:00
|
|
|
&& ! in_array($sectionName, $children, true)
|
2015-09-18 11:01:02 +02:00
|
|
|
) {
|
|
|
|
$names[] = $sectionName;
|
2015-09-17 13:45:47 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return $names;
|
|
|
|
}
|
|
|
|
|
2015-09-17 15:53:12 +02:00
|
|
|
/**
|
|
|
|
* Recursively return all children of the given navigation item
|
|
|
|
*
|
|
|
|
* @param string $name
|
|
|
|
*
|
|
|
|
* @return array
|
|
|
|
*/
|
|
|
|
protected function getFlattenedChildren($name)
|
|
|
|
{
|
|
|
|
$config = $this->getConfigForItem($name);
|
|
|
|
if ($config === null) {
|
|
|
|
return array();
|
|
|
|
}
|
|
|
|
|
|
|
|
$children = array();
|
2015-09-18 12:20:44 +02:00
|
|
|
foreach ($config->toArray() as $sectionName => $sectionConfig) {
|
|
|
|
if (isset($sectionConfig['parent']) && $sectionConfig['parent'] === $name) {
|
2015-09-17 15:53:12 +02:00
|
|
|
$children[] = $sectionName;
|
|
|
|
$children = array_merge($children, $this->getFlattenedChildren($sectionName));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return $children;
|
|
|
|
}
|
|
|
|
|
2015-09-07 16:58:22 +02:00
|
|
|
/**
|
|
|
|
* Populate the form with the given navigation item's config
|
|
|
|
*
|
|
|
|
* @param string $name
|
|
|
|
*
|
|
|
|
* @return $this
|
|
|
|
*
|
|
|
|
* @throws NotFoundError In case no navigation item with the given name is found
|
|
|
|
*/
|
|
|
|
public function load($name)
|
|
|
|
{
|
2015-09-15 15:57:03 +02:00
|
|
|
if ($this->getConfigForItem($name) === null) {
|
2015-09-07 16:58:22 +02:00
|
|
|
throw new NotFoundError('No navigation item called "%s" found', $name);
|
|
|
|
}
|
|
|
|
|
|
|
|
$this->itemToLoad = $name;
|
|
|
|
return $this;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Add a new navigation item
|
|
|
|
*
|
|
|
|
* The navigation item to add is identified by the array-key `name'.
|
|
|
|
*
|
|
|
|
* @param array $data
|
|
|
|
*
|
|
|
|
* @return $this
|
|
|
|
*
|
2015-09-29 17:06:22 +02:00
|
|
|
* @throws InvalidArgumentException In case $data does not contain a navigation item name or type
|
2015-09-07 16:58:22 +02:00
|
|
|
* @throws IcingaException In case a navigation item with the same name already exists
|
|
|
|
*/
|
|
|
|
public function add(array $data)
|
|
|
|
{
|
|
|
|
if (! isset($data['name'])) {
|
|
|
|
throw new InvalidArgumentException('Key \'name\' missing');
|
2015-09-29 17:06:22 +02:00
|
|
|
} elseif (! isset($data['type'])) {
|
|
|
|
throw new InvalidArgumentException('Key \'type\' missing');
|
2015-09-07 16:58:22 +02:00
|
|
|
}
|
|
|
|
|
2015-09-24 15:50:19 +02:00
|
|
|
$shared = false;
|
2015-09-29 17:06:22 +02:00
|
|
|
$config = $this->getUserConfig($data['type']);
|
2015-09-16 11:51:57 +02:00
|
|
|
if ((isset($data['users']) && $data['users']) || (isset($data['groups']) && $data['groups'])) {
|
2021-02-18 08:52:57 +01:00
|
|
|
if ($this->getUser()->can('user/share/navigation')) {
|
2015-09-16 11:51:57 +02:00
|
|
|
$data['owner'] = $this->getUser()->getUsername();
|
2015-09-29 17:07:04 +02:00
|
|
|
$config = $this->getShareConfig($data['type']);
|
2015-09-24 15:50:19 +02:00
|
|
|
$shared = true;
|
2015-09-16 11:51:57 +02:00
|
|
|
} else {
|
|
|
|
unset($data['users']);
|
|
|
|
unset($data['groups']);
|
|
|
|
}
|
2015-09-29 17:07:04 +02:00
|
|
|
} elseif (isset($data['parent']) && $data['parent'] && $this->hasBeenShared($data['parent'], $data['type'])) {
|
2015-09-18 15:53:39 +02:00
|
|
|
$data['owner'] = $this->getUser()->getUsername();
|
2015-09-29 17:07:04 +02:00
|
|
|
$config = $this->getShareConfig($data['type']);
|
2015-09-24 15:50:19 +02:00
|
|
|
$shared = true;
|
2015-09-16 11:51:57 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
$itemName = $data['name'];
|
2015-09-24 15:50:19 +02:00
|
|
|
$exists = $config->hasSection($itemName);
|
|
|
|
if (! $exists) {
|
|
|
|
if ($shared) {
|
2015-09-29 17:06:22 +02:00
|
|
|
$exists = $this->getUserConfig($data['type'])->hasSection($itemName);
|
2015-09-24 15:50:19 +02:00
|
|
|
} else {
|
2015-09-29 17:07:04 +02:00
|
|
|
$exists = (bool) $this->getShareConfig($data['type'])
|
2015-09-24 15:50:19 +02:00
|
|
|
->select()
|
|
|
|
->where('name', $itemName)
|
|
|
|
->where('owner', $this->getUser()->getUsername())
|
|
|
|
->count();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if ($exists) {
|
2015-09-07 16:58:22 +02:00
|
|
|
throw new IcingaException(
|
|
|
|
$this->translate('A navigation item with the name "%s" does already exist'),
|
|
|
|
$itemName
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
unset($data['name']);
|
2015-09-15 15:57:03 +02:00
|
|
|
$config->setSection($itemName, $data);
|
|
|
|
$this->setIniConfig($config);
|
2015-09-07 16:58:22 +02:00
|
|
|
return $this;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Edit a navigation item
|
|
|
|
*
|
|
|
|
* @param string $name
|
|
|
|
* @param array $data
|
|
|
|
*
|
|
|
|
* @return $this
|
|
|
|
*
|
2015-09-24 12:58:11 +02:00
|
|
|
* @throws NotFoundError In case no navigation item with the given name is found
|
|
|
|
* @throws IcingaException In case a navigation item with the same name already exists
|
2015-09-07 16:58:22 +02:00
|
|
|
*/
|
|
|
|
public function edit($name, array $data)
|
|
|
|
{
|
2015-09-15 15:57:03 +02:00
|
|
|
$config = $this->getConfigForItem($name);
|
|
|
|
if ($config === null) {
|
2015-09-07 16:58:22 +02:00
|
|
|
throw new NotFoundError('No navigation item called "%s" found', $name);
|
2015-09-18 16:08:58 +02:00
|
|
|
} else {
|
|
|
|
$itemConfig = $config->getSection($name);
|
2015-09-07 16:58:22 +02:00
|
|
|
}
|
|
|
|
|
2015-09-18 16:08:58 +02:00
|
|
|
$shared = false;
|
2015-09-16 11:51:57 +02:00
|
|
|
if ($this->hasBeenShared($name)) {
|
2015-09-18 15:56:00 +02:00
|
|
|
if (isset($data['parent']) && $data['parent']
|
2017-11-09 14:17:46 +01:00
|
|
|
? ! $this->hasBeenShared($data['parent'])
|
|
|
|
: ((! isset($data['users']) || ! $data['users']) && (! isset($data['groups']) || ! $data['groups']))
|
2015-09-18 15:56:00 +02:00
|
|
|
) {
|
2015-09-16 11:51:57 +02:00
|
|
|
// It is shared but shouldn't anymore
|
2015-09-25 10:51:16 +02:00
|
|
|
$config = $this->unshare($name, isset($data['parent']) ? $data['parent'] : null);
|
2015-09-16 11:51:57 +02:00
|
|
|
}
|
|
|
|
} elseif ((isset($data['users']) && $data['users']) || (isset($data['groups']) && $data['groups'])) {
|
2021-02-18 08:52:57 +01:00
|
|
|
if ($this->getUser()->can('user/share/navigation')) {
|
2015-09-16 11:51:57 +02:00
|
|
|
// It is not shared yet but should be
|
|
|
|
$this->secondaryConfig = $config;
|
|
|
|
$config = $this->getShareConfig();
|
|
|
|
$data['owner'] = $this->getUser()->getUsername();
|
2015-09-18 16:08:58 +02:00
|
|
|
$shared = true;
|
2015-09-16 11:51:57 +02:00
|
|
|
} else {
|
|
|
|
unset($data['users']);
|
|
|
|
unset($data['groups']);
|
|
|
|
}
|
2015-09-18 15:53:39 +02:00
|
|
|
} elseif (isset($data['parent']) && $data['parent'] && $this->hasBeenShared($data['parent'])) {
|
|
|
|
// Its parent is shared so should it itself
|
|
|
|
$this->secondaryConfig = $config;
|
|
|
|
$config = $this->getShareConfig();
|
|
|
|
$data['owner'] = $this->getUser()->getUsername();
|
|
|
|
$shared = true;
|
2015-09-16 11:51:57 +02:00
|
|
|
}
|
|
|
|
|
2015-09-18 16:00:33 +02:00
|
|
|
$oldName = null;
|
2015-09-07 16:58:22 +02:00
|
|
|
if (isset($data['name'])) {
|
|
|
|
if ($data['name'] !== $name) {
|
2015-09-18 16:00:33 +02:00
|
|
|
$oldName = $name;
|
2015-09-07 16:58:22 +02:00
|
|
|
$name = $data['name'];
|
2015-09-24 12:58:11 +02:00
|
|
|
|
2015-09-24 15:50:19 +02:00
|
|
|
$exists = $config->hasSection($name);
|
|
|
|
if (! $exists) {
|
|
|
|
$ownerName = $itemConfig->owner ?: $this->getUser()->getUsername();
|
|
|
|
if ($shared || $this->hasBeenShared($oldName)) {
|
|
|
|
if ($ownerName === $this->getUser()->getUsername()) {
|
|
|
|
$exists = $this->getUserConfig()->hasSection($name);
|
|
|
|
} else {
|
2015-09-30 11:38:14 +02:00
|
|
|
$exists = Config::navigation($itemConfig->type, $ownerName)->hasSection($name);
|
2015-09-24 15:50:19 +02:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
$exists = (bool) $this->getShareConfig()
|
|
|
|
->select()
|
|
|
|
->where('name', $name)
|
|
|
|
->where('owner', $ownerName)
|
|
|
|
->count();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if ($exists) {
|
2015-09-24 12:58:11 +02:00
|
|
|
throw new IcingaException(
|
|
|
|
$this->translate('A navigation item with the name "%s" does already exist'),
|
|
|
|
$name
|
|
|
|
);
|
|
|
|
}
|
2015-09-07 16:58:22 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
unset($data['name']);
|
|
|
|
}
|
|
|
|
|
2015-09-18 16:08:58 +02:00
|
|
|
$itemConfig->merge($data);
|
|
|
|
|
|
|
|
if ($shared) {
|
|
|
|
// Share all descendant children
|
|
|
|
foreach ($this->getFlattenedChildren($oldName ?: $name) as $child) {
|
|
|
|
$childConfig = $this->secondaryConfig->getSection($child);
|
|
|
|
$this->secondaryConfig->removeSection($child);
|
|
|
|
$childConfig->owner = $this->getUser()->getUsername();
|
|
|
|
$config->setSection($child, $childConfig);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-09-18 16:00:33 +02:00
|
|
|
if ($oldName) {
|
|
|
|
// Update the parent name on all direct children
|
|
|
|
foreach ($config as $sectionConfig) {
|
|
|
|
if ($sectionConfig->parent === $oldName) {
|
|
|
|
$sectionConfig->parent = $name;
|
|
|
|
}
|
|
|
|
}
|
2015-09-18 16:08:58 +02:00
|
|
|
|
2015-09-18 16:25:06 +02:00
|
|
|
$config->removeSection($oldName);
|
2015-09-18 16:00:33 +02:00
|
|
|
}
|
|
|
|
|
2015-09-18 16:08:58 +02:00
|
|
|
if ($this->secondaryConfig !== null) {
|
|
|
|
$this->secondaryConfig->removeSection($oldName ?: $name);
|
2015-09-07 16:58:22 +02:00
|
|
|
}
|
|
|
|
|
2015-09-15 15:57:03 +02:00
|
|
|
$config->setSection($name, $itemConfig);
|
|
|
|
$this->setIniConfig($config);
|
2015-09-07 16:58:22 +02:00
|
|
|
return $this;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Remove a navigation item
|
|
|
|
*
|
|
|
|
* @param string $name
|
|
|
|
*
|
2015-09-25 10:51:44 +02:00
|
|
|
* @return ConfigObject The navigation item's config
|
2015-09-18 15:58:20 +02:00
|
|
|
*
|
2015-09-21 09:00:22 +02:00
|
|
|
* @throws NotFoundError In case no navigation item with the given name is found
|
2015-09-18 15:58:20 +02:00
|
|
|
* @throws IcingaException In case the navigation item has still children
|
2015-09-07 16:58:22 +02:00
|
|
|
*/
|
|
|
|
public function delete($name)
|
|
|
|
{
|
2015-09-15 15:57:03 +02:00
|
|
|
$config = $this->getConfigForItem($name);
|
|
|
|
if ($config === null) {
|
|
|
|
throw new NotFoundError('No navigation item called "%s" found', $name);
|
|
|
|
}
|
|
|
|
|
2015-09-18 15:58:20 +02:00
|
|
|
$children = $this->getFlattenedChildren($name);
|
|
|
|
if (! empty($children)) {
|
|
|
|
throw new IcingaException(
|
2015-09-24 15:50:40 +02:00
|
|
|
$this->translate(
|
|
|
|
'Unable to delete navigation item "%s". There'
|
|
|
|
. ' are other items dependent from it: %s'
|
|
|
|
),
|
2015-09-18 15:58:20 +02:00
|
|
|
$name,
|
|
|
|
join(', ', $children)
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2015-09-25 10:51:44 +02:00
|
|
|
$section = $config->getSection($name);
|
2015-09-15 15:57:03 +02:00
|
|
|
$config->removeSection($name);
|
|
|
|
$this->setIniConfig($config);
|
2015-09-25 10:51:44 +02:00
|
|
|
return $section;
|
2015-09-07 16:58:22 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Unshare the given navigation item
|
|
|
|
*
|
|
|
|
* @param string $name
|
2015-09-18 15:57:45 +02:00
|
|
|
* @param string $parent
|
2015-09-07 16:58:22 +02:00
|
|
|
*
|
2015-09-25 10:51:16 +02:00
|
|
|
* @return Config The new config of the given navigation item
|
2015-09-07 16:58:22 +02:00
|
|
|
*
|
2015-09-18 15:57:45 +02:00
|
|
|
* @throws NotFoundError In case no navigation item with the given name is found
|
|
|
|
* @throws IcingaException In case the navigation item has a parent assigned to it
|
2015-09-07 16:58:22 +02:00
|
|
|
*/
|
2015-09-18 15:57:45 +02:00
|
|
|
public function unshare($name, $parent = null)
|
2015-09-07 16:58:22 +02:00
|
|
|
{
|
2015-09-16 13:10:21 +02:00
|
|
|
$config = $this->getShareConfig();
|
|
|
|
if (! $config->hasSection($name)) {
|
|
|
|
throw new NotFoundError('No navigation item called "%s" found', $name);
|
|
|
|
}
|
|
|
|
|
|
|
|
$itemConfig = $config->getSection($name);
|
2015-09-18 15:57:45 +02:00
|
|
|
if ($parent === null) {
|
|
|
|
$parent = $itemConfig->parent;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ($parent && $this->hasBeenShared($parent)) {
|
|
|
|
throw new IcingaException(
|
2015-09-24 15:50:40 +02:00
|
|
|
$this->translate(
|
|
|
|
'Unable to unshare navigation item "%s". It is dependent from item "%s".'
|
|
|
|
. ' Dependent items can only be unshared by unsharing their parent'
|
|
|
|
),
|
2015-09-18 15:57:45 +02:00
|
|
|
$name,
|
|
|
|
$parent
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
$children = $this->getFlattenedChildren($name);
|
2015-09-16 13:10:21 +02:00
|
|
|
$config->removeSection($name);
|
|
|
|
$this->secondaryConfig = $config;
|
|
|
|
|
|
|
|
if (! $itemConfig->owner || $itemConfig->owner === $this->getUser()->getUsername()) {
|
|
|
|
$config = $this->getUserConfig();
|
|
|
|
} else {
|
2015-09-30 11:38:14 +02:00
|
|
|
$config = Config::navigation($itemConfig->type, $itemConfig->owner);
|
2015-09-16 13:10:21 +02:00
|
|
|
}
|
|
|
|
|
2015-09-18 15:57:45 +02:00
|
|
|
foreach ($children as $child) {
|
|
|
|
$childConfig = $this->secondaryConfig->getSection($child);
|
|
|
|
unset($childConfig->owner);
|
|
|
|
$this->secondaryConfig->removeSection($child);
|
|
|
|
$config->setSection($child, $childConfig);
|
|
|
|
}
|
|
|
|
|
2015-09-16 13:10:21 +02:00
|
|
|
unset($itemConfig->owner);
|
|
|
|
unset($itemConfig->users);
|
|
|
|
unset($itemConfig->groups);
|
|
|
|
|
|
|
|
$config->setSection($name, $itemConfig);
|
|
|
|
$this->setIniConfig($config);
|
2015-09-25 10:51:16 +02:00
|
|
|
return $config;
|
2015-09-07 16:58:22 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* {@inheritdoc}
|
|
|
|
*/
|
|
|
|
public function createElements(array $formData)
|
|
|
|
{
|
2015-09-18 10:52:03 +02:00
|
|
|
$shared = false;
|
2015-09-16 15:05:43 +02:00
|
|
|
$itemTypes = $this->getItemTypes();
|
2015-09-16 13:51:12 +02:00
|
|
|
$itemType = isset($formData['type']) ? $formData['type'] : key($itemTypes);
|
2015-09-29 17:07:56 +02:00
|
|
|
if ($itemType === null) {
|
|
|
|
throw new ProgrammingError(
|
2017-01-12 16:04:09 +01:00
|
|
|
'This should actually not happen. Create a bug report at https://github.com/icinga/icingaweb2'
|
2015-09-29 17:07:56 +02:00
|
|
|
. ' or remove this assertion if you know what you\'re doing'
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2015-09-18 09:48:22 +02:00
|
|
|
$itemForm = $this->getItemForm($itemType);
|
2015-09-07 16:58:22 +02:00
|
|
|
|
|
|
|
$this->addElement(
|
|
|
|
'text',
|
|
|
|
'name',
|
|
|
|
array(
|
|
|
|
'required' => true,
|
|
|
|
'label' => $this->translate('Name'),
|
|
|
|
'description' => $this->translate(
|
|
|
|
'The name of this navigation item that is used to differentiate it from others'
|
|
|
|
)
|
|
|
|
)
|
|
|
|
);
|
|
|
|
|
2017-11-09 14:17:46 +01:00
|
|
|
if ((! $itemForm->requiresParentSelection() || ! isset($formData['parent']) || ! $formData['parent'])
|
2021-02-18 08:52:57 +01:00
|
|
|
&& $this->getUser()->can('user/share/navigation')
|
2015-09-18 09:48:22 +02:00
|
|
|
) {
|
2015-09-16 11:51:57 +02:00
|
|
|
$checked = isset($formData['shared']) ? null : (isset($formData['users']) || isset($formData['groups']));
|
|
|
|
|
|
|
|
$this->addElement(
|
|
|
|
'checkbox',
|
|
|
|
'shared',
|
|
|
|
array(
|
|
|
|
'autosubmit' => true,
|
|
|
|
'ignore' => true,
|
|
|
|
'value' => $checked,
|
|
|
|
'label' => $this->translate('Shared'),
|
|
|
|
'description' => $this->translate('Tick this box to share this item with others')
|
|
|
|
)
|
|
|
|
);
|
|
|
|
|
|
|
|
if ($checked || (isset($formData['shared']) && $formData['shared'])) {
|
2015-09-18 10:52:03 +02:00
|
|
|
$shared = true;
|
2015-09-16 11:51:57 +02:00
|
|
|
$this->addElement(
|
2017-07-31 09:06:17 +02:00
|
|
|
'textarea',
|
2015-09-16 11:51:57 +02:00
|
|
|
'users',
|
|
|
|
array(
|
|
|
|
'label' => $this->translate('Users'),
|
|
|
|
'description' => $this->translate(
|
|
|
|
'Comma separated list of usernames to share this item with'
|
|
|
|
)
|
|
|
|
)
|
|
|
|
);
|
|
|
|
$this->addElement(
|
2017-07-31 09:06:17 +02:00
|
|
|
'textarea',
|
2015-09-16 11:51:57 +02:00
|
|
|
'groups',
|
|
|
|
array(
|
|
|
|
'label' => $this->translate('Groups'),
|
|
|
|
'description' => $this->translate(
|
|
|
|
'Comma separated list of group names to share this item with'
|
|
|
|
)
|
|
|
|
)
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-09-29 17:07:56 +02:00
|
|
|
if (empty($itemTypes) || count($itemTypes) === 1) {
|
|
|
|
$this->addElement(
|
|
|
|
'hidden',
|
|
|
|
'type',
|
|
|
|
array(
|
2015-09-30 12:33:13 +02:00
|
|
|
'value' => $itemType
|
2015-09-29 17:07:56 +02:00
|
|
|
)
|
|
|
|
);
|
|
|
|
} else {
|
|
|
|
$this->addElement(
|
|
|
|
'select',
|
|
|
|
'type',
|
|
|
|
array(
|
|
|
|
'required' => true,
|
|
|
|
'autosubmit' => true,
|
|
|
|
'label' => $this->translate('Type'),
|
|
|
|
'description' => $this->translate('The type of this navigation item'),
|
2015-09-30 11:30:38 +02:00
|
|
|
'multiOptions' => $itemTypes
|
2015-09-29 17:07:56 +02:00
|
|
|
)
|
|
|
|
);
|
|
|
|
}
|
2015-09-07 16:58:22 +02:00
|
|
|
|
2015-09-18 10:52:03 +02:00
|
|
|
if (! $shared && $itemForm->requiresParentSelection()) {
|
2015-09-21 15:29:14 +02:00
|
|
|
if ($this->itemToLoad && $this->hasBeenShared($this->itemToLoad)) {
|
|
|
|
$itemConfig = $this->getShareConfig()->getSection($this->itemToLoad);
|
|
|
|
$availableParents = $this->listAvailableParents($itemType, $itemConfig->owner);
|
|
|
|
} else {
|
|
|
|
$availableParents = $this->listAvailableParents($itemType);
|
|
|
|
}
|
|
|
|
|
2015-09-18 09:48:22 +02:00
|
|
|
$this->addElement(
|
|
|
|
'select',
|
|
|
|
'parent',
|
|
|
|
array(
|
|
|
|
'allowEmpty' => true,
|
|
|
|
'autosubmit' => true,
|
|
|
|
'label' => $this->translate('Parent'),
|
|
|
|
'description' => $this->translate(
|
|
|
|
'The parent item to assign this navigation item to. '
|
|
|
|
. 'Select "None" to make this a main navigation item'
|
|
|
|
),
|
2019-06-26 11:40:30 +02:00
|
|
|
'multiOptions' => ['' => $this->translate('None', 'No parent for a navigation item')]
|
|
|
|
+ (empty($availableParents) ? [] : array_combine($availableParents, $availableParents))
|
2015-09-18 09:48:22 +02:00
|
|
|
)
|
|
|
|
);
|
2019-06-26 11:41:01 +02:00
|
|
|
} else {
|
|
|
|
$this->addElement('hidden', 'parent', ['disabled' => true]);
|
2015-09-18 09:48:22 +02:00
|
|
|
}
|
|
|
|
|
2015-09-17 13:42:25 +02:00
|
|
|
$this->addSubForm($itemForm, 'item_form');
|
2015-09-18 09:48:22 +02:00
|
|
|
$itemForm->create($formData); // May require a parent which gets set by addSubForm()
|
2015-09-07 16:58:22 +02:00
|
|
|
}
|
|
|
|
|
2015-10-01 13:55:12 +02:00
|
|
|
/**
|
|
|
|
* DO NOT USE! This will be removed soon, very soon...
|
|
|
|
*/
|
|
|
|
public function setDefaultUrl($url)
|
|
|
|
{
|
|
|
|
$this->defaultUrl = $url;
|
|
|
|
}
|
|
|
|
|
2015-09-07 16:58:22 +02:00
|
|
|
/**
|
|
|
|
* Populate the configuration of the navigation item to load
|
|
|
|
*/
|
|
|
|
public function onRequest()
|
|
|
|
{
|
|
|
|
if ($this->itemToLoad) {
|
2015-09-15 15:57:03 +02:00
|
|
|
$data = $this->getConfigForItem($this->itemToLoad)->getSection($this->itemToLoad)->toArray();
|
2015-09-07 16:58:22 +02:00
|
|
|
$data['name'] = $this->itemToLoad;
|
|
|
|
$this->populate($data);
|
2015-10-01 13:55:12 +02:00
|
|
|
} elseif ($this->defaultUrl !== null) {
|
|
|
|
$this->populate(array('url' => $this->defaultUrl));
|
2015-09-07 16:58:22 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-09-16 16:25:17 +02:00
|
|
|
/**
|
|
|
|
* {@inheritdoc}
|
|
|
|
*/
|
|
|
|
public function isValid($formData)
|
|
|
|
{
|
|
|
|
if (! parent::isValid($formData)) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
$valid = true;
|
|
|
|
if (isset($formData['users']) && $formData['users']) {
|
|
|
|
$parsedUserRestrictions = array();
|
|
|
|
foreach (Auth::getInstance()->getRestrictions('application/share/users') as $userRestriction) {
|
|
|
|
$parsedUserRestrictions[] = array_map('trim', explode(',', $userRestriction));
|
|
|
|
}
|
|
|
|
|
|
|
|
if (! empty($parsedUserRestrictions)) {
|
|
|
|
$desiredUsers = array_map('trim', explode(',', $formData['users']));
|
|
|
|
array_unshift($parsedUserRestrictions, $desiredUsers);
|
|
|
|
$forbiddenUsers = call_user_func_array('array_diff', $parsedUserRestrictions);
|
|
|
|
if (! empty($forbiddenUsers)) {
|
|
|
|
$valid = false;
|
|
|
|
$this->getElement('users')->addError(
|
2017-11-09 14:17:46 +01:00
|
|
|
sprintf(
|
|
|
|
$this->translate(
|
|
|
|
'You are not permitted to share this navigation item with the following users: %s'
|
|
|
|
),
|
2015-09-16 16:25:17 +02:00
|
|
|
implode(', ', $forbiddenUsers)
|
2017-11-09 14:17:46 +01:00
|
|
|
)
|
2015-09-16 16:25:17 +02:00
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (isset($formData['groups']) && $formData['groups']) {
|
|
|
|
$parsedGroupRestrictions = array();
|
|
|
|
foreach (Auth::getInstance()->getRestrictions('application/share/groups') as $groupRestriction) {
|
|
|
|
$parsedGroupRestrictions[] = array_map('trim', explode(',', $groupRestriction));
|
|
|
|
}
|
|
|
|
|
|
|
|
if (! empty($parsedGroupRestrictions)) {
|
|
|
|
$desiredGroups = array_map('trim', explode(',', $formData['groups']));
|
|
|
|
array_unshift($parsedGroupRestrictions, $desiredGroups);
|
|
|
|
$forbiddenGroups = call_user_func_array('array_diff', $parsedGroupRestrictions);
|
|
|
|
if (! empty($forbiddenGroups)) {
|
|
|
|
$valid = false;
|
|
|
|
$this->getElement('groups')->addError(
|
2017-11-09 14:17:46 +01:00
|
|
|
sprintf(
|
|
|
|
$this->translate(
|
|
|
|
'You are not permitted to share this navigation item with the following groups: %s'
|
|
|
|
),
|
2015-09-16 16:25:17 +02:00
|
|
|
implode(', ', $forbiddenGroups)
|
2017-11-09 14:17:46 +01:00
|
|
|
)
|
2015-09-16 16:25:17 +02:00
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return $valid;
|
|
|
|
}
|
|
|
|
|
2015-09-16 10:58:57 +02:00
|
|
|
/**
|
|
|
|
* {@inheritdoc}
|
|
|
|
*/
|
|
|
|
protected function writeConfig(Config $config)
|
|
|
|
{
|
|
|
|
parent::writeConfig($config);
|
|
|
|
|
|
|
|
if ($this->secondaryConfig !== null) {
|
|
|
|
$this->config = $this->secondaryConfig; // Causes the config being displayed to the user in case of an error
|
|
|
|
parent::writeConfig($this->secondaryConfig);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-09-15 15:57:03 +02:00
|
|
|
/**
|
|
|
|
* Return the navigation configuration the given item is a part of
|
|
|
|
*
|
|
|
|
* @param string $name
|
|
|
|
*
|
|
|
|
* @return Config|null In case the item is not part of any configuration
|
|
|
|
*/
|
|
|
|
protected function getConfigForItem($name)
|
|
|
|
{
|
|
|
|
if ($this->getUserConfig()->hasSection($name)) {
|
|
|
|
return $this->getUserConfig();
|
|
|
|
} elseif ($this->getShareConfig()->hasSection($name)) {
|
2017-01-27 14:48:59 +01:00
|
|
|
if ($this->getShareConfig()->get($name, 'owner') === $this->getUser()->getUsername()
|
2021-02-18 08:52:57 +01:00
|
|
|
|| $this->getUser()->can('user/share/navigation')
|
2015-09-15 15:57:03 +02:00
|
|
|
) {
|
|
|
|
return $this->getShareConfig();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-09-16 11:51:57 +02:00
|
|
|
/**
|
|
|
|
* Return whether the given navigation item has been shared
|
|
|
|
*
|
|
|
|
* @param string $name
|
2015-09-29 17:07:04 +02:00
|
|
|
* @param string $type
|
2015-09-16 11:51:57 +02:00
|
|
|
*
|
|
|
|
* @return bool
|
|
|
|
*/
|
2015-09-29 17:07:04 +02:00
|
|
|
protected function hasBeenShared($name, $type = null)
|
2015-09-16 11:51:57 +02:00
|
|
|
{
|
2015-09-29 17:07:04 +02:00
|
|
|
return $this->getShareConfig($type) === $this->getConfigForItem($name);
|
2015-09-16 11:51:57 +02:00
|
|
|
}
|
|
|
|
|
2015-09-07 16:58:22 +02:00
|
|
|
/**
|
|
|
|
* Return the form for the given type of navigation item
|
|
|
|
*
|
|
|
|
* @param string $type
|
|
|
|
*
|
|
|
|
* @return Form
|
|
|
|
*/
|
|
|
|
protected function getItemForm($type)
|
|
|
|
{
|
2016-01-27 15:56:27 +01:00
|
|
|
$className = StringHelper::cname($type, '-') . 'Form';
|
2015-09-17 08:24:23 +02:00
|
|
|
|
|
|
|
$form = null;
|
|
|
|
foreach (Icinga::app()->getModuleManager()->getLoadedModules() as $module) {
|
|
|
|
$classPath = 'Icinga\\Module\\'
|
|
|
|
. ucfirst($module->getName())
|
|
|
|
. '\\'
|
|
|
|
. static::FORM_NS
|
|
|
|
. '\\'
|
|
|
|
. $className;
|
|
|
|
if (class_exists($classPath)) {
|
|
|
|
$form = new $classPath();
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if ($form === null) {
|
|
|
|
$classPath = 'Icinga\\' . static::FORM_NS . '\\' . $className;
|
|
|
|
if (class_exists($classPath)) {
|
|
|
|
$form = new $classPath();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if ($form === null) {
|
|
|
|
Logger::debug(
|
|
|
|
'Failed to find custom navigation item form %s for item %s. Using form NavigationItemForm now',
|
|
|
|
$className,
|
|
|
|
$type
|
|
|
|
);
|
|
|
|
|
|
|
|
$form = new NavigationItemForm();
|
2015-09-18 09:43:40 +02:00
|
|
|
} elseif (! $form instanceof NavigationItemForm) {
|
|
|
|
throw new ProgrammingError('Class %s must inherit from NavigationItemForm', $classPath);
|
2015-09-17 08:24:23 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
return $form;
|
2015-09-07 16:58:22 +02:00
|
|
|
}
|
|
|
|
}
|