2014-09-29 12:29:36 +02:00
|
|
|
<?php
|
2016-02-08 15:41:00 +01:00
|
|
|
/* Icinga Web 2 | (c) 2014 Icinga Development Team | GPLv2+ */
|
2014-09-29 12:29:36 +02:00
|
|
|
|
2014-11-14 11:01:16 +01:00
|
|
|
namespace Icinga\Module\Setup\Forms;
|
2014-09-29 12:29:36 +02:00
|
|
|
|
|
|
|
use Exception;
|
2015-07-29 15:07:41 +02:00
|
|
|
use Icinga\Application\Config;
|
2016-10-17 16:19:26 +02:00
|
|
|
use Icinga\Authentication\User\ExternalBackend;
|
2015-06-25 15:26:56 +02:00
|
|
|
use Icinga\Authentication\User\UserBackend;
|
2015-06-30 09:20:52 +02:00
|
|
|
use Icinga\Authentication\User\DbUserBackend;
|
|
|
|
use Icinga\Authentication\User\LdapUserBackend;
|
2015-07-29 17:13:06 +02:00
|
|
|
use Icinga\Authentication\UserGroup\UserGroupBackend;
|
|
|
|
use Icinga\Authentication\UserGroup\LdapUserGroupBackend;
|
2014-11-18 13:11:52 +01:00
|
|
|
use Icinga\Data\ConfigObject;
|
2015-07-29 15:07:41 +02:00
|
|
|
use Icinga\Data\ResourceFactory;
|
2015-07-29 17:18:41 +02:00
|
|
|
use Icinga\Data\Selectable;
|
|
|
|
use Icinga\Exception\NotImplementedError;
|
2015-06-25 15:26:56 +02:00
|
|
|
use Icinga\Web\Form;
|
2014-09-29 12:29:36 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Wizard page to define the initial administrative account
|
|
|
|
*/
|
|
|
|
class AdminAccountPage extends Form
|
|
|
|
{
|
|
|
|
/**
|
|
|
|
* The resource configuration to use
|
|
|
|
*
|
|
|
|
* @var array
|
|
|
|
*/
|
|
|
|
protected $resourceConfig;
|
|
|
|
|
|
|
|
/**
|
2015-07-29 17:13:06 +02:00
|
|
|
* The user backend configuration to use
|
2014-09-29 12:29:36 +02:00
|
|
|
*
|
|
|
|
* @var array
|
|
|
|
*/
|
|
|
|
protected $backendConfig;
|
|
|
|
|
2015-07-29 17:13:06 +02:00
|
|
|
/**
|
|
|
|
* The user group backend configuration to use
|
|
|
|
*
|
|
|
|
* @var array
|
|
|
|
*/
|
|
|
|
protected $groupConfig;
|
|
|
|
|
2014-09-29 12:29:36 +02:00
|
|
|
/**
|
|
|
|
* Initialize this page
|
|
|
|
*/
|
|
|
|
public function init()
|
|
|
|
{
|
|
|
|
$this->setName('setup_admin_account');
|
2015-06-30 11:38:59 +02:00
|
|
|
$this->setTitle($this->translate('Administration', 'setup.page.title'));
|
2015-06-30 10:58:17 +02:00
|
|
|
$this->addDescription($this->translate(
|
2015-07-29 17:13:06 +02:00
|
|
|
'Now it\'s time to configure your first administrative account or group for Icinga Web 2.'
|
2015-06-30 10:58:17 +02:00
|
|
|
));
|
2014-09-29 12:29:36 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Set the resource configuration to use
|
|
|
|
*
|
|
|
|
* @param array $config
|
|
|
|
*
|
2015-04-07 14:23:26 +02:00
|
|
|
* @return $this
|
2014-09-29 12:29:36 +02:00
|
|
|
*/
|
|
|
|
public function setResourceConfig(array $config)
|
|
|
|
{
|
|
|
|
$this->resourceConfig = $config;
|
|
|
|
return $this;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2015-07-29 17:13:06 +02:00
|
|
|
* Set the user backend configuration to use
|
2014-09-29 12:29:36 +02:00
|
|
|
*
|
|
|
|
* @param array $config
|
|
|
|
*
|
2015-04-07 14:23:26 +02:00
|
|
|
* @return $this
|
2014-09-29 12:29:36 +02:00
|
|
|
*/
|
|
|
|
public function setBackendConfig(array $config)
|
|
|
|
{
|
|
|
|
$this->backendConfig = $config;
|
|
|
|
return $this;
|
|
|
|
}
|
|
|
|
|
2015-07-29 17:13:06 +02:00
|
|
|
/**
|
|
|
|
* Set the user group backend configuration to use
|
|
|
|
*
|
|
|
|
* @param array $config
|
|
|
|
*
|
|
|
|
* @return $this
|
|
|
|
*/
|
|
|
|
public function setGroupConfig(array $config = null)
|
|
|
|
{
|
|
|
|
$this->groupConfig = $config;
|
|
|
|
return $this;
|
|
|
|
}
|
|
|
|
|
2014-09-29 12:29:36 +02:00
|
|
|
/**
|
|
|
|
* @see Form::createElements()
|
|
|
|
*/
|
|
|
|
public function createElements(array $formData)
|
|
|
|
{
|
|
|
|
$choices = array();
|
|
|
|
if ($this->backendConfig['backend'] !== 'db') {
|
2015-01-19 11:07:39 +01:00
|
|
|
$choices['by_name'] = $this->translate('By Name', 'setup.admin');
|
2015-06-30 10:58:17 +02:00
|
|
|
$choice = isset($formData['user_type']) ? $formData['user_type'] : 'by_name';
|
2015-07-29 17:13:06 +02:00
|
|
|
|
|
|
|
if (in_array($this->backendConfig['backend'], array('ldap', 'msldap'))) {
|
|
|
|
$groups = $this->fetchGroups();
|
|
|
|
if (! empty($groups)) {
|
|
|
|
$choices['user_group'] = $this->translate('User Group', 'setup.admin');
|
|
|
|
}
|
|
|
|
}
|
2015-06-30 10:58:17 +02:00
|
|
|
} else {
|
|
|
|
$choices['new_user'] = $this->translate('New User', 'setup.admin');
|
|
|
|
$choice = isset($formData['user_type']) ? $formData['user_type'] : 'new_user';
|
|
|
|
}
|
|
|
|
|
|
|
|
if (in_array($this->backendConfig['backend'], array('db', 'ldap', 'msldap'))) {
|
|
|
|
$users = $this->fetchUsers();
|
|
|
|
if (! empty($users)) {
|
|
|
|
$choices['existing_user'] = $this->translate('Existing User', 'setup.admin');
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (count($choices) > 1) {
|
|
|
|
$this->addElement(
|
|
|
|
'select',
|
|
|
|
'user_type',
|
|
|
|
array(
|
|
|
|
'required' => true,
|
|
|
|
'autosubmit' => true,
|
|
|
|
'label' => $this->translate('Type Of Definition'),
|
2015-06-30 15:10:17 +02:00
|
|
|
'description' => $this->translate('Choose how to define the desired account.'),
|
2015-06-30 10:58:17 +02:00
|
|
|
'multiOptions' => $choices,
|
|
|
|
'value' => $choice
|
|
|
|
)
|
|
|
|
);
|
|
|
|
} else {
|
|
|
|
$this->addElement(
|
|
|
|
'hidden',
|
|
|
|
'user_type',
|
|
|
|
array(
|
|
|
|
'required' => true,
|
|
|
|
'value' => key($choices)
|
|
|
|
)
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
if ($choice === 'by_name') {
|
2014-09-29 12:29:36 +02:00
|
|
|
$this->addElement(
|
|
|
|
'text',
|
|
|
|
'by_name',
|
|
|
|
array(
|
2015-06-30 10:58:17 +02:00
|
|
|
'required' => true,
|
2014-09-29 12:29:36 +02:00
|
|
|
'value' => $this->getUsername(),
|
2015-01-19 11:07:39 +01:00
|
|
|
'label' => $this->translate('Username'),
|
|
|
|
'description' => $this->translate(
|
2014-09-29 12:29:36 +02:00
|
|
|
'Define the initial administrative account by providing a username that reflects'
|
2015-06-30 15:10:17 +02:00
|
|
|
. ' a user created later or one that is authenticated using external mechanisms.'
|
2014-09-29 12:29:36 +02:00
|
|
|
)
|
|
|
|
)
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2015-07-29 17:13:06 +02:00
|
|
|
if ($choice === 'user_group') {
|
|
|
|
$this->addElement(
|
|
|
|
'select',
|
|
|
|
'user_group',
|
|
|
|
array(
|
|
|
|
'required' => true,
|
|
|
|
'label' => $this->translate('Group Name'),
|
|
|
|
'description' => $this->translate(
|
|
|
|
'Choose a user group reported by the LDAP backend'
|
|
|
|
. ' to permit its members administrative access.',
|
|
|
|
'setup.admin'
|
|
|
|
),
|
|
|
|
'multiOptions' => array_combine($groups, $groups)
|
|
|
|
)
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2015-06-30 10:58:17 +02:00
|
|
|
if ($choice === 'existing_user') {
|
|
|
|
$this->addElement(
|
|
|
|
'select',
|
|
|
|
'existing_user',
|
|
|
|
array(
|
|
|
|
'required' => true,
|
|
|
|
'label' => $this->translate('Username'),
|
|
|
|
'description' => sprintf(
|
|
|
|
$this->translate(
|
2015-06-30 15:10:17 +02:00
|
|
|
'Choose a user reported by the %s backend as the initial administrative account.',
|
2015-06-30 10:58:17 +02:00
|
|
|
'setup.admin'
|
2014-09-29 12:29:36 +02:00
|
|
|
),
|
2015-06-30 10:58:17 +02:00
|
|
|
$this->backendConfig['backend'] === 'db'
|
|
|
|
? $this->translate('database', 'setup.admin.authbackend')
|
|
|
|
: 'LDAP'
|
|
|
|
),
|
|
|
|
'multiOptions' => array_combine($users, $users)
|
|
|
|
)
|
|
|
|
);
|
2014-09-29 12:29:36 +02:00
|
|
|
}
|
|
|
|
|
2015-06-30 10:58:17 +02:00
|
|
|
if ($choice === 'new_user') {
|
2014-09-29 12:29:36 +02:00
|
|
|
$this->addElement(
|
|
|
|
'text',
|
|
|
|
'new_user',
|
|
|
|
array(
|
2015-06-30 10:58:17 +02:00
|
|
|
'required' => true,
|
2015-01-19 11:07:39 +01:00
|
|
|
'label' => $this->translate('Username'),
|
|
|
|
'description' => $this->translate(
|
2015-06-30 15:10:17 +02:00
|
|
|
'Enter the username to be used when creating an initial administrative account.'
|
2014-09-29 12:29:36 +02:00
|
|
|
)
|
|
|
|
)
|
|
|
|
);
|
|
|
|
$this->addElement(
|
|
|
|
'password',
|
|
|
|
'new_user_password',
|
|
|
|
array(
|
2015-07-02 11:50:23 +02:00
|
|
|
'required' => true,
|
|
|
|
'renderPassword' => true,
|
|
|
|
'label' => $this->translate('Password'),
|
2015-07-20 15:52:10 +02:00
|
|
|
'description' => $this->translate(
|
|
|
|
'Enter the password to assign to the newly created account.'
|
|
|
|
)
|
2014-09-29 12:29:36 +02:00
|
|
|
)
|
|
|
|
);
|
|
|
|
$this->addElement(
|
|
|
|
'password',
|
|
|
|
'new_user_2ndpass',
|
|
|
|
array(
|
2015-07-02 11:50:23 +02:00
|
|
|
'required' => true,
|
|
|
|
'renderPassword' => true,
|
|
|
|
'label' => $this->translate('Repeat password'),
|
|
|
|
'description' => $this->translate(
|
2015-06-30 15:10:17 +02:00
|
|
|
'Please repeat the password given above to avoid typing errors.'
|
2015-06-30 10:58:17 +02:00
|
|
|
),
|
2015-07-02 11:50:23 +02:00
|
|
|
'validators' => array(
|
2014-09-29 12:29:36 +02:00
|
|
|
array('identical', false, array('new_user_password'))
|
|
|
|
)
|
|
|
|
)
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-10-08 10:38:04 +02:00
|
|
|
/**
|
|
|
|
* Validate the given request data and ensure that any new user does not already exist
|
|
|
|
*
|
|
|
|
* @param array $data The request data to validate
|
|
|
|
*
|
|
|
|
* @return bool
|
|
|
|
*/
|
|
|
|
public function isValid($data)
|
|
|
|
{
|
|
|
|
if (false === parent::isValid($data)) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2015-07-01 14:25:43 +02:00
|
|
|
if ($data['user_type'] === 'new_user' && $this->hasUser($data['new_user'])) {
|
2015-01-19 11:07:39 +01:00
|
|
|
$this->getElement('new_user')->addError($this->translate('Username already exists.'));
|
2014-10-08 10:38:04 +02:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2014-09-29 12:29:36 +02:00
|
|
|
/**
|
|
|
|
* Return the name of the externally authenticated user
|
|
|
|
*
|
|
|
|
* @return string
|
|
|
|
*/
|
|
|
|
protected function getUsername()
|
|
|
|
{
|
2016-11-16 12:06:58 +01:00
|
|
|
list($name, $_) = ExternalBackend::getRemoteUserInformation();
|
2016-10-17 16:19:26 +02:00
|
|
|
if ($name === null) {
|
2014-09-29 12:29:36 +02:00
|
|
|
return '';
|
|
|
|
}
|
|
|
|
|
2014-10-20 15:14:14 +02:00
|
|
|
if (isset($this->backendConfig['strip_username_regexp']) && $this->backendConfig['strip_username_regexp']) {
|
2014-09-29 12:29:36 +02:00
|
|
|
// No need to silence or log anything here because the pattern has
|
|
|
|
// already been successfully compiled during backend configuration
|
|
|
|
$name = preg_replace($this->backendConfig['strip_username_regexp'], '', $name);
|
|
|
|
}
|
|
|
|
|
|
|
|
return $name;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2015-07-29 17:13:06 +02:00
|
|
|
* Return the names of all users the user backend currently provides
|
2014-09-29 12:29:36 +02:00
|
|
|
*
|
|
|
|
* @return array
|
|
|
|
*/
|
|
|
|
protected function fetchUsers()
|
|
|
|
{
|
|
|
|
try {
|
2015-10-01 15:47:11 +02:00
|
|
|
$query = $this
|
2015-07-29 17:13:06 +02:00
|
|
|
->createUserBackend()
|
|
|
|
->select(array('user_name'))
|
2015-10-01 15:47:11 +02:00
|
|
|
->order('user_name', 'asc', true);
|
|
|
|
if (in_array($this->backendConfig['backend'], array('ldap', 'msldap'))) {
|
|
|
|
$query->getQuery()->setUsePagedResults();
|
|
|
|
}
|
|
|
|
|
|
|
|
return $query->fetchColumn();
|
2015-05-04 12:21:17 +02:00
|
|
|
} catch (Exception $_) {
|
2014-09-29 12:29:36 +02:00
|
|
|
// No need to handle anything special here. Error means no users found.
|
|
|
|
return array();
|
|
|
|
}
|
|
|
|
}
|
2015-06-30 09:20:52 +02:00
|
|
|
|
|
|
|
/**
|
2015-07-29 17:13:06 +02:00
|
|
|
* Return whether the user backend provides a user with the given name
|
2015-06-30 09:20:52 +02:00
|
|
|
*
|
|
|
|
* @param string $username
|
|
|
|
*
|
|
|
|
* @return bool
|
|
|
|
*/
|
|
|
|
protected function hasUser($username)
|
|
|
|
{
|
2015-07-01 14:25:43 +02:00
|
|
|
try {
|
2015-07-29 17:13:06 +02:00
|
|
|
return $this
|
|
|
|
->createUserBackend()
|
|
|
|
->select()
|
|
|
|
->where('user_name', $username)
|
|
|
|
->count() > 1;
|
2015-07-01 14:25:43 +02:00
|
|
|
} catch (Exception $_) {
|
2015-07-29 17:13:06 +02:00
|
|
|
return false;
|
2015-07-01 14:25:43 +02:00
|
|
|
}
|
2015-06-30 09:20:52 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2015-07-29 17:13:06 +02:00
|
|
|
* Create and return the user backend
|
2015-06-30 09:20:52 +02:00
|
|
|
*
|
|
|
|
* @return DbUserBackend|LdapUserBackend
|
|
|
|
*/
|
2015-07-29 17:13:06 +02:00
|
|
|
protected function createUserBackend()
|
2015-06-30 09:20:52 +02:00
|
|
|
{
|
2015-07-29 15:07:41 +02:00
|
|
|
$resourceConfig = new Config();
|
|
|
|
$resourceConfig->setSection($this->resourceConfig['name'], $this->resourceConfig);
|
|
|
|
ResourceFactory::setConfig($resourceConfig);
|
|
|
|
|
2015-06-30 09:20:52 +02:00
|
|
|
$config = new ConfigObject($this->backendConfig);
|
2015-07-29 15:07:41 +02:00
|
|
|
$config->resource = $this->resourceConfig['name'];
|
2015-06-30 09:20:52 +02:00
|
|
|
return UserBackend::create(null, $config);
|
|
|
|
}
|
2015-07-29 17:13:06 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Return the names of all user groups the user group backend currently provides
|
|
|
|
*
|
|
|
|
* @return array
|
|
|
|
*/
|
|
|
|
protected function fetchGroups()
|
|
|
|
{
|
|
|
|
try {
|
2015-10-01 15:47:11 +02:00
|
|
|
$query = $this
|
2015-07-29 17:13:06 +02:00
|
|
|
->createUserGroupBackend()
|
2015-10-01 15:47:11 +02:00
|
|
|
->select(array('group_name'));
|
|
|
|
if (in_array($this->backendConfig['backend'], array('ldap', 'msldap'))) {
|
|
|
|
$query->getQuery()->setUsePagedResults();
|
|
|
|
}
|
|
|
|
|
|
|
|
return $query->fetchColumn();
|
2015-07-29 17:13:06 +02:00
|
|
|
} catch (Exception $_) {
|
|
|
|
// No need to handle anything special here. Error means no groups found.
|
|
|
|
return array();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Return whether the user group backend provides a user group with the given name
|
|
|
|
*
|
|
|
|
* @param string $groupname
|
|
|
|
*
|
|
|
|
* @return bool
|
|
|
|
*/
|
|
|
|
protected function hasGroup($groupname)
|
|
|
|
{
|
|
|
|
try {
|
|
|
|
return $this
|
|
|
|
->createUserGroupBackend()
|
|
|
|
->select()
|
|
|
|
->where('group_name', $groupname)
|
|
|
|
->count() > 1;
|
|
|
|
} catch (Exception $_) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Create and return the user group backend
|
|
|
|
*
|
|
|
|
* @return LdapUserGroupBackend
|
|
|
|
*/
|
|
|
|
protected function createUserGroupBackend()
|
|
|
|
{
|
|
|
|
$resourceConfig = new Config();
|
|
|
|
$resourceConfig->setSection($this->resourceConfig['name'], $this->resourceConfig);
|
|
|
|
ResourceFactory::setConfig($resourceConfig);
|
|
|
|
|
|
|
|
$backendConfig = new Config();
|
|
|
|
$backendConfig->setSection($this->backendConfig['name'], array_merge(
|
|
|
|
$this->backendConfig,
|
|
|
|
array('resource' => $this->resourceConfig['name'])
|
|
|
|
));
|
|
|
|
UserBackend::setConfig($backendConfig);
|
|
|
|
|
|
|
|
if (empty($this->groupConfig)) {
|
|
|
|
$groupConfig = new ConfigObject(array(
|
|
|
|
'backend' => $this->backendConfig['backend'], // _Should_ be "db" or "msldap"
|
|
|
|
'resource' => $this->resourceConfig['name'],
|
|
|
|
'user_backend' => $this->backendConfig['name'] // Gets ignored if 'backend' is "db"
|
|
|
|
));
|
|
|
|
} else {
|
|
|
|
$groupConfig = new ConfigObject($this->groupConfig);
|
|
|
|
}
|
|
|
|
|
2015-07-29 17:18:41 +02:00
|
|
|
$backend = UserGroupBackend::create(null, $groupConfig);
|
|
|
|
if (! $backend instanceof Selectable) {
|
|
|
|
throw new NotImplementedError('Unsupported, until #9772 has been resolved');
|
|
|
|
}
|
|
|
|
|
|
|
|
return $backend;
|
2015-07-29 17:13:06 +02:00
|
|
|
}
|
2014-09-29 12:29:36 +02:00
|
|
|
}
|