2015-05-04 17:04:50 +02:00
|
|
|
<?php
|
2016-02-08 15:41:00 +01:00
|
|
|
/* Icinga Web 2 | (c) 2015 Icinga Development Team | GPLv2+ */
|
2015-05-04 17:04:50 +02:00
|
|
|
|
2015-08-27 14:16:17 +02:00
|
|
|
namespace Icinga\Controllers;
|
|
|
|
|
2015-08-27 14:17:19 +02:00
|
|
|
use Exception;
|
2015-05-13 13:58:40 +02:00
|
|
|
use Icinga\Application\Logger;
|
2019-09-25 10:39:46 +02:00
|
|
|
use Icinga\Authentication\AdmissionLoader;
|
2017-06-07 15:37:35 +02:00
|
|
|
use Icinga\Authentication\User\DomainAwareInterface;
|
2015-08-27 14:16:02 +02:00
|
|
|
use Icinga\Data\DataArray\ArrayDatasource;
|
2015-05-26 17:03:10 +02:00
|
|
|
use Icinga\Exception\ConfigurationError;
|
2015-05-28 16:28:43 +02:00
|
|
|
use Icinga\Exception\NotFoundError;
|
2015-05-26 17:03:10 +02:00
|
|
|
use Icinga\Forms\Config\User\CreateMembershipForm;
|
2015-05-26 10:23:40 +02:00
|
|
|
use Icinga\Forms\Config\User\UserForm;
|
2015-05-22 15:53:47 +02:00
|
|
|
use Icinga\User;
|
2015-05-26 10:12:10 +02:00
|
|
|
use Icinga\Web\Controller\AuthBackendController;
|
2015-05-08 09:54:45 +02:00
|
|
|
use Icinga\Web\Form;
|
2015-05-13 13:50:19 +02:00
|
|
|
use Icinga\Web\Notification;
|
2015-05-20 13:54:05 +02:00
|
|
|
use Icinga\Web\Url;
|
2015-05-04 17:04:50 +02:00
|
|
|
use Icinga\Web\Widget;
|
|
|
|
|
2015-05-26 10:12:10 +02:00
|
|
|
class UserController extends AuthBackendController
|
2015-05-04 17:04:50 +02:00
|
|
|
{
|
2019-07-12 10:22:01 +02:00
|
|
|
public function init()
|
|
|
|
{
|
|
|
|
$this->view->title = $this->translate('Users');
|
|
|
|
|
|
|
|
parent::init();
|
|
|
|
}
|
|
|
|
|
2015-05-04 17:04:50 +02:00
|
|
|
/**
|
|
|
|
* List all users of a single backend
|
|
|
|
*/
|
|
|
|
public function listAction()
|
|
|
|
{
|
2021-02-18 08:52:57 +01:00
|
|
|
$this->assertPermission('config/access-control/users');
|
2015-06-03 08:54:56 +02:00
|
|
|
$this->createListTabs()->activate('user/list');
|
2015-05-08 09:54:45 +02:00
|
|
|
$backendNames = array_map(
|
2017-01-27 14:48:59 +01:00
|
|
|
function ($b) {
|
|
|
|
return $b->getName();
|
|
|
|
},
|
2015-05-08 09:54:45 +02:00
|
|
|
$this->loadUserBackends('Icinga\Data\Selectable')
|
|
|
|
);
|
2015-06-08 13:27:45 +02:00
|
|
|
if (empty($backendNames)) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2015-05-08 09:54:45 +02:00
|
|
|
$this->view->backendSelection = new Form();
|
2019-07-17 07:57:11 +02:00
|
|
|
$this->view->backendSelection->setAttrib('class', 'backend-selection icinga-controls');
|
2015-05-08 09:54:45 +02:00
|
|
|
$this->view->backendSelection->setUidDisabled();
|
|
|
|
$this->view->backendSelection->setMethod('GET');
|
|
|
|
$this->view->backendSelection->setTokenDisabled();
|
|
|
|
$this->view->backendSelection->addElement(
|
|
|
|
'select',
|
|
|
|
'backend',
|
|
|
|
array(
|
|
|
|
'autosubmit' => true,
|
2015-10-15 17:06:02 +02:00
|
|
|
'label' => $this->translate('User Backend'),
|
2015-05-08 09:54:45 +02:00
|
|
|
'multiOptions' => array_combine($backendNames, $backendNames),
|
|
|
|
'value' => $this->params->get('backend')
|
|
|
|
)
|
|
|
|
);
|
|
|
|
|
2015-05-04 17:04:50 +02:00
|
|
|
$backend = $this->getUserBackend($this->params->get('backend'));
|
|
|
|
if ($backend === null) {
|
|
|
|
$this->view->backend = null;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2015-05-20 15:00:29 +02:00
|
|
|
$query = $backend->select(array('user_name'));
|
2015-05-04 17:04:50 +02:00
|
|
|
|
2015-05-27 08:53:34 +02:00
|
|
|
$this->view->users = $query;
|
2015-05-04 17:04:50 +02:00
|
|
|
$this->view->backend = $backend;
|
|
|
|
|
2015-05-27 08:53:34 +02:00
|
|
|
$this->setupPaginationControl($query);
|
2015-08-13 17:05:13 +02:00
|
|
|
$this->setupFilterControl($query);
|
2015-05-04 17:04:50 +02:00
|
|
|
$this->setupLimitControl();
|
2015-05-12 15:49:45 +02:00
|
|
|
$this->setupSortControl(
|
|
|
|
array(
|
|
|
|
'user_name' => $this->translate('Username'),
|
|
|
|
'is_active' => $this->translate('Active'),
|
|
|
|
'created_at' => $this->translate('Created at'),
|
|
|
|
'last_modified' => $this->translate('Last modified')
|
|
|
|
),
|
|
|
|
$query
|
|
|
|
);
|
2015-05-04 17:04:50 +02:00
|
|
|
}
|
|
|
|
|
2015-05-20 15:41:02 +02:00
|
|
|
/**
|
|
|
|
* Show a user
|
|
|
|
*/
|
|
|
|
public function showAction()
|
|
|
|
{
|
2021-02-18 08:52:57 +01:00
|
|
|
$this->assertPermission('config/access-control/users');
|
2015-05-20 15:41:02 +02:00
|
|
|
$userName = $this->params->getRequired('user');
|
|
|
|
$backend = $this->getUserBackend($this->params->getRequired('backend'));
|
|
|
|
|
|
|
|
$user = $backend->select(array(
|
|
|
|
'user_name',
|
|
|
|
'is_active',
|
|
|
|
'created_at',
|
|
|
|
'last_modified'
|
|
|
|
))->where('user_name', $userName)->fetchRow();
|
|
|
|
if ($user === false) {
|
|
|
|
$this->httpNotFound(sprintf($this->translate('User "%s" not found'), $userName));
|
|
|
|
}
|
|
|
|
|
2017-06-07 15:37:35 +02:00
|
|
|
$userObj = new User($userName);
|
|
|
|
if ($backend instanceof DomainAwareInterface) {
|
|
|
|
$userObj->setDomain($backend->getDomain());
|
|
|
|
}
|
|
|
|
|
|
|
|
$memberships = $this->loadMemberships($userObj)->select();
|
2015-05-22 15:53:47 +02:00
|
|
|
|
2015-08-13 17:05:13 +02:00
|
|
|
$this->setupFilterControl(
|
|
|
|
$memberships,
|
|
|
|
array('group_name' => t('User Group')),
|
2015-11-10 11:52:54 +01:00
|
|
|
array('group'),
|
2015-11-10 11:51:26 +01:00
|
|
|
array('user')
|
2015-08-13 17:05:13 +02:00
|
|
|
);
|
2015-05-22 15:53:47 +02:00
|
|
|
$this->setupPaginationControl($memberships);
|
|
|
|
$this->setupLimitControl();
|
|
|
|
$this->setupSortControl(
|
|
|
|
array(
|
|
|
|
'group_name' => $this->translate('Group')
|
|
|
|
),
|
|
|
|
$memberships
|
|
|
|
);
|
|
|
|
|
2021-02-18 08:52:57 +01:00
|
|
|
if ($this->hasPermission('config/access-control/groups')) {
|
2015-05-27 10:34:10 +02:00
|
|
|
$extensibleBackends = $this->loadUserGroupBackends('Icinga\Data\Extensible');
|
|
|
|
$this->view->showCreateMembershipLink = ! empty($extensibleBackends);
|
|
|
|
} else {
|
|
|
|
$this->view->showCreateMembershipLink = false;
|
|
|
|
}
|
2015-05-26 17:03:10 +02:00
|
|
|
|
2015-05-20 15:41:02 +02:00
|
|
|
$this->view->user = $user;
|
|
|
|
$this->view->backend = $backend;
|
2015-05-22 15:53:47 +02:00
|
|
|
$this->view->memberships = $memberships;
|
2015-05-22 16:13:20 +02:00
|
|
|
$this->createShowTabs($backend->getName(), $userName)->activate('user/show');
|
2015-05-22 15:53:47 +02:00
|
|
|
|
2021-02-18 08:52:57 +01:00
|
|
|
if ($this->hasPermission('config/access-control/groups')) {
|
2015-05-27 10:38:02 +02:00
|
|
|
$removeForm = new Form();
|
|
|
|
$removeForm->setUidDisabled();
|
2019-07-26 13:51:00 +02:00
|
|
|
$removeForm->setAttrib('class', 'inline');
|
2015-05-27 10:38:02 +02:00
|
|
|
$removeForm->addElement('hidden', 'user_name', array(
|
|
|
|
'isArray' => true,
|
|
|
|
'value' => $userName,
|
|
|
|
'decorators' => array('ViewHelper')
|
|
|
|
));
|
|
|
|
$removeForm->addElement('hidden', 'redirect', array(
|
|
|
|
'value' => Url::fromPath('user/show', array(
|
|
|
|
'backend' => $backend->getName(),
|
|
|
|
'user' => $userName
|
|
|
|
)),
|
|
|
|
'decorators' => array('ViewHelper')
|
|
|
|
));
|
|
|
|
$removeForm->addElement('button', 'btn_submit', array(
|
|
|
|
'escape' => false,
|
|
|
|
'type' => 'submit',
|
2015-10-15 14:35:30 +02:00
|
|
|
'class' => 'link-button spinner',
|
2015-05-27 10:38:02 +02:00
|
|
|
'value' => 'btn_submit',
|
|
|
|
'decorators' => array('ViewHelper'),
|
|
|
|
'label' => $this->view->icon('trash'),
|
|
|
|
'title' => $this->translate('Cancel this membership')
|
|
|
|
));
|
|
|
|
$this->view->removeForm = $removeForm;
|
|
|
|
}
|
2019-09-25 10:39:46 +02:00
|
|
|
|
|
|
|
$admissionLoader = new AdmissionLoader();
|
|
|
|
$admissionLoader->applyRoles($userObj);
|
|
|
|
$this->view->userObj = $userObj;
|
2021-02-18 08:52:57 +01:00
|
|
|
$this->view->allowedToEditRoles = $this->hasPermission('config/access-control/groups');
|
2015-05-20 15:41:02 +02:00
|
|
|
}
|
|
|
|
|
2015-05-20 10:54:42 +02:00
|
|
|
/**
|
|
|
|
* Add a user
|
|
|
|
*/
|
|
|
|
public function addAction()
|
|
|
|
{
|
2021-02-18 08:52:57 +01:00
|
|
|
$this->assertPermission('config/access-control/users');
|
2015-05-20 13:54:05 +02:00
|
|
|
$backend = $this->getUserBackend($this->params->getRequired('backend'), 'Icinga\Data\Extensible');
|
2015-05-20 10:54:42 +02:00
|
|
|
$form = new UserForm();
|
2015-05-20 13:54:05 +02:00
|
|
|
$form->setRedirectUrl(Url::fromPath('user/list', array('backend' => $backend->getName())));
|
|
|
|
$form->setRepository($backend);
|
2015-05-20 10:54:42 +02:00
|
|
|
$form->add()->handleRequest();
|
|
|
|
|
2015-10-01 09:39:21 +02:00
|
|
|
$this->renderForm($form, $this->translate('New User'));
|
2015-05-20 10:54:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Edit a user
|
|
|
|
*/
|
|
|
|
public function editAction()
|
|
|
|
{
|
2021-02-18 08:52:57 +01:00
|
|
|
$this->assertPermission('config/access-control/users');
|
2015-05-20 10:54:42 +02:00
|
|
|
$userName = $this->params->getRequired('user');
|
|
|
|
$backend = $this->getUserBackend($this->params->getRequired('backend'), 'Icinga\Data\Updatable');
|
|
|
|
|
|
|
|
$form = new UserForm();
|
2015-05-20 15:41:02 +02:00
|
|
|
$form->setRedirectUrl(Url::fromPath('user/show', array('backend' => $backend->getName(), 'user' => $userName)));
|
2015-05-20 10:54:42 +02:00
|
|
|
$form->setRepository($backend);
|
2015-05-28 16:28:43 +02:00
|
|
|
|
|
|
|
try {
|
|
|
|
$form->edit($userName)->handleRequest();
|
|
|
|
} catch (NotFoundError $_) {
|
|
|
|
$this->httpNotFound(sprintf($this->translate('User "%s" not found'), $userName));
|
|
|
|
}
|
2015-05-20 10:54:42 +02:00
|
|
|
|
2015-10-01 09:39:21 +02:00
|
|
|
$this->renderForm($form, $this->translate('Update User'));
|
2015-05-20 10:54:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Remove a user
|
|
|
|
*/
|
|
|
|
public function removeAction()
|
|
|
|
{
|
2021-02-18 08:52:57 +01:00
|
|
|
$this->assertPermission('config/access-control/users');
|
2015-05-20 10:54:42 +02:00
|
|
|
$userName = $this->params->getRequired('user');
|
|
|
|
$backend = $this->getUserBackend($this->params->getRequired('backend'), 'Icinga\Data\Reducible');
|
|
|
|
|
|
|
|
$form = new UserForm();
|
2015-05-20 13:54:05 +02:00
|
|
|
$form->setRedirectUrl(Url::fromPath('user/list', array('backend' => $backend->getName())));
|
2015-05-20 10:54:42 +02:00
|
|
|
$form->setRepository($backend);
|
2015-05-28 16:28:43 +02:00
|
|
|
|
|
|
|
try {
|
|
|
|
$form->remove($userName)->handleRequest();
|
|
|
|
} catch (NotFoundError $_) {
|
|
|
|
$this->httpNotFound(sprintf($this->translate('User "%s" not found'), $userName));
|
|
|
|
}
|
2015-05-20 10:54:42 +02:00
|
|
|
|
2015-10-01 09:39:21 +02:00
|
|
|
$this->renderForm($form, $this->translate('Remove User'));
|
2015-05-20 10:54:42 +02:00
|
|
|
}
|
|
|
|
|
2015-05-26 17:03:10 +02:00
|
|
|
/**
|
|
|
|
* Create a membership for a user
|
|
|
|
*/
|
|
|
|
public function createmembershipAction()
|
|
|
|
{
|
2021-02-18 08:52:57 +01:00
|
|
|
$this->assertPermission('config/access-control/groups');
|
2015-05-26 17:03:10 +02:00
|
|
|
$userName = $this->params->getRequired('user');
|
|
|
|
$backend = $this->getUserBackend($this->params->getRequired('backend'));
|
|
|
|
|
|
|
|
if ($backend->select()->where('user_name', $userName)->count() === 0) {
|
|
|
|
$this->httpNotFound(sprintf($this->translate('User "%s" not found'), $userName));
|
|
|
|
}
|
|
|
|
|
|
|
|
$backends = $this->loadUserGroupBackends('Icinga\Data\Extensible');
|
|
|
|
if (empty($backends)) {
|
|
|
|
throw new ConfigurationError($this->translate(
|
|
|
|
'You\'ll need to configure at least one user group backend first that allows to create new memberships'
|
|
|
|
));
|
|
|
|
}
|
|
|
|
|
|
|
|
$form = new CreateMembershipForm();
|
|
|
|
$form->setBackends($backends)
|
|
|
|
->setUsername($userName)
|
|
|
|
->setRedirectUrl(Url::fromPath('user/show', array('backend' => $backend->getName(), 'user' => $userName)))
|
|
|
|
->handleRequest();
|
|
|
|
|
|
|
|
$this->view->form = $form;
|
|
|
|
$this->render('form');
|
|
|
|
}
|
|
|
|
|
2015-05-22 15:53:47 +02:00
|
|
|
/**
|
|
|
|
* Fetch and return the given user's groups from all user group backends
|
|
|
|
*
|
|
|
|
* @param User $user
|
|
|
|
*
|
|
|
|
* @return ArrayDatasource
|
|
|
|
*/
|
|
|
|
protected function loadMemberships(User $user)
|
|
|
|
{
|
2015-05-27 09:03:00 +02:00
|
|
|
$groups = $alreadySeen = array();
|
2015-05-26 10:12:10 +02:00
|
|
|
foreach ($this->loadUserGroupBackends() as $backend) {
|
2015-05-27 08:55:14 +02:00
|
|
|
try {
|
|
|
|
foreach ($backend->getMemberships($user) as $groupName) {
|
2015-05-27 09:03:00 +02:00
|
|
|
if (array_key_exists($groupName, $alreadySeen)) {
|
|
|
|
continue; // Ignore duplicate memberships
|
|
|
|
}
|
|
|
|
|
|
|
|
$alreadySeen[$groupName] = null;
|
2015-05-27 08:55:14 +02:00
|
|
|
$groups[] = (object) array(
|
|
|
|
'group_name' => $groupName,
|
2015-11-10 11:52:54 +01:00
|
|
|
'group' => $groupName,
|
2015-05-27 08:55:14 +02:00
|
|
|
'backend' => $backend
|
|
|
|
);
|
|
|
|
}
|
|
|
|
} catch (Exception $e) {
|
|
|
|
Logger::error($e);
|
|
|
|
Notification::warning(sprintf(
|
|
|
|
$this->translate('Failed to fetch memberships from backend %s. Please check your log'),
|
|
|
|
$backend->getName()
|
|
|
|
));
|
2015-05-22 15:53:47 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return new ArrayDatasource($groups);
|
|
|
|
}
|
|
|
|
|
2015-05-22 16:13:20 +02:00
|
|
|
/**
|
|
|
|
* Create the tabs to display when showing a user
|
|
|
|
*
|
|
|
|
* @param string $backendName
|
|
|
|
* @param string $userName
|
|
|
|
*/
|
|
|
|
protected function createShowTabs($backendName, $userName)
|
|
|
|
{
|
|
|
|
$tabs = $this->getTabs();
|
|
|
|
$tabs->add(
|
|
|
|
'user/show',
|
|
|
|
array(
|
|
|
|
'title' => sprintf($this->translate('Show user %s'), $userName),
|
|
|
|
'label' => $this->translate('User'),
|
|
|
|
'url' => Url::fromPath('user/show', array('backend' => $backendName, 'user' => $userName))
|
|
|
|
)
|
|
|
|
);
|
|
|
|
|
|
|
|
return $tabs;
|
2015-05-05 07:36:14 +02:00
|
|
|
}
|
2015-06-22 13:26:24 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Create the tabs to display when listing users
|
|
|
|
*/
|
|
|
|
protected function createListTabs()
|
|
|
|
{
|
|
|
|
$tabs = $this->getTabs();
|
2015-10-01 09:01:03 +02:00
|
|
|
|
2021-02-18 08:52:57 +01:00
|
|
|
if ($this->hasPermission('config/access-control/roles')) {
|
|
|
|
$tabs->add(
|
|
|
|
'role/list',
|
|
|
|
array(
|
|
|
|
'baseTarget' => '_main',
|
|
|
|
'label' => $this->translate('Roles'),
|
|
|
|
'title' => $this->translate(
|
|
|
|
'Configure roles to permit or restrict users and groups accessing Icinga Web 2'
|
|
|
|
),
|
|
|
|
'url' => 'role/list'
|
|
|
|
)
|
|
|
|
);
|
2021-03-18 17:03:12 +01:00
|
|
|
|
|
|
|
$tabs->add(
|
|
|
|
'role/audit',
|
|
|
|
[
|
2021-03-22 16:34:51 +01:00
|
|
|
'title' => $this->translate('Audit a user\'s or group\'s privileges'),
|
|
|
|
'label' => $this->translate('Audit'),
|
|
|
|
'url' => 'role/audit',
|
|
|
|
'baseTarget' => '_main'
|
2021-03-18 17:03:12 +01:00
|
|
|
]
|
|
|
|
);
|
2021-02-18 08:52:57 +01:00
|
|
|
}
|
|
|
|
|
2015-06-22 13:26:24 +02:00
|
|
|
$tabs->add(
|
|
|
|
'user/list',
|
|
|
|
array(
|
|
|
|
'title' => $this->translate('List users of authentication backends'),
|
|
|
|
'label' => $this->translate('Users'),
|
|
|
|
'url' => 'user/list'
|
|
|
|
)
|
|
|
|
);
|
2021-02-18 08:52:57 +01:00
|
|
|
|
|
|
|
if ($this->hasPermission('config/access-control/groups')) {
|
|
|
|
$tabs->add(
|
|
|
|
'group/list',
|
|
|
|
array(
|
|
|
|
'title' => $this->translate('List groups of user group backends'),
|
|
|
|
'label' => $this->translate('User Groups'),
|
|
|
|
'url' => 'group/list'
|
|
|
|
)
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2015-06-22 13:26:24 +02:00
|
|
|
return $tabs;
|
|
|
|
}
|
2015-05-04 17:04:50 +02:00
|
|
|
}
|