setName('form_config_authbackend'); $this->setSubmitLabel($this->translate('Save Changes')); } /** * Set the resource configuration to use * * @param Config $resources The resource configuration * * @return $this */ public function setResourceConfig(Config $resourceConfig) { $resources = array(); foreach ($resourceConfig as $name => $resource) { $resources[strtolower($resource->type)][] = $name; } $this->resources = $resources; return $this; } /** * Return a form object for the given backend type * * @param string $type The backend type for which to return a form * * @return Form */ public function getBackendForm($type) { if ($type === 'db') { $form = new DbBackendForm(); $form->setResources(isset($this->resources['db']) ? $this->resources['db'] : array()); } elseif ($type === 'ldap') { $form = new LdapBackendForm(); $form->setResources(isset($this->resources['ldap']) ? $this->resources['ldap'] : array()); } elseif ($type === 'external') { $form = new ExternalBackendForm(); } else { throw new InvalidArgumentException(sprintf($this->translate('Invalid backend type "%s" provided'), $type)); } return $form; } /** * Add a particular user backend * * The backend to add is identified by the array-key `name'. * * @param array $values The values to extend the configuration with * * @return $this * * @throws InvalidArgumentException In case the backend does already exist */ public function add(array $values) { $name = isset($values['name']) ? $values['name'] : ''; if (! $name) { throw new InvalidArgumentException($this->translate('User backend name missing')); } elseif ($this->config->hasSection($name)) { throw new InvalidArgumentException($this->translate('User backend already exists')); } unset($values['name']); $this->config->setSection($name, $values); return $this; } /** * Edit a particular user backend * * @param string $name The name of the backend to edit * @param array $values The values to edit the configuration with * * @return array The edited backend configuration * * @throws InvalidArgumentException In case the backend does not exist */ public function edit($name, array $values) { if (! $name) { throw new InvalidArgumentException($this->translate('Old user backend name missing')); } elseif (! ($newName = isset($values['name']) ? $values['name'] : '')) { throw new InvalidArgumentException($this->translate('New user backend name missing')); } elseif (! $this->config->hasSection($name)) { throw new InvalidArgumentException($this->translate('Unknown user backend provided')); } $backendConfig = $this->config->getSection($name); if ($newName !== $name) { // Only remove the old entry if it has changed as the order gets screwed when editing backend names $this->config->removeSection($name); } unset($values['name']); $this->config->setSection($newName, $backendConfig->merge($values)); return $backendConfig; } /** * Remove the given user backend * * @param string $name The name of the backend to remove * * @return array The removed backend configuration * * @throws InvalidArgumentException In case the backend does not exist */ public function remove($name) { if (! $name) { throw new InvalidArgumentException($this->translate('user backend name missing')); } elseif (! $this->config->hasSection($name)) { throw new InvalidArgumentException($this->translate('Unknown user backend provided')); } $backendConfig = $this->config->getSection($name); $this->config->removeSection($name); return $backendConfig; } /** * Move the given user backend up or down in order * * @param string $name The name of the backend to be moved * @param int $position The new (absolute) position of the backend * * @return $this * * @throws InvalidArgumentException In case the backend does not exist */ public function move($name, $position) { if (! $name) { throw new InvalidArgumentException($this->translate('User backend name missing')); } elseif (! $this->config->hasSection($name)) { throw new InvalidArgumentException($this->translate('Unknown user backend provided')); } $backendOrder = $this->config->keys(); array_splice($backendOrder, array_search($name, $backendOrder), 1); array_splice($backendOrder, $position, 0, $name); $newConfig = array(); foreach ($backendOrder as $backendName) { $newConfig[$backendName] = $this->config->getSection($backendName); } $config = Config::fromArray($newConfig); $this->config = $config->setConfigFile($this->config->getConfigFile()); return $this; } /** * Add or edit an user backend and save the configuration * * Performs a connectivity validation using the submitted values. A checkbox is * added to the form to skip the check if it fails and redirection is aborted. * * @see Form::onSuccess() */ public function onSuccess() { if (($el = $this->getElement('force_creation')) === null || false === $el->isChecked()) { $backendForm = $this->getBackendForm($this->getElement('type')->getValue()); if (false === $backendForm::isValidUserBackend($this)) { $this->addElement($this->getForceCreationCheckbox()); return false; } } $authBackend = $this->request->getQuery('backend'); try { if ($authBackend === null) { // create new backend $this->add($this->getValues()); $message = $this->translate('User backend "%s" has been successfully created'); } else { // edit existing backend $this->edit($authBackend, $this->getValues()); $message = $this->translate('User backend "%s" has been successfully changed'); } } catch (InvalidArgumentException $e) { Notification::error($e->getMessage()); return; } if ($this->save()) { Notification::success(sprintf($message, $this->getElement('name')->getValue())); } else { return false; } } /** * Populate the form in case an user backend is being edited * * @see Form::onRequest() * * @throws ConfigurationError In case the backend name is missing in the request or is invalid */ public function onRequest() { $authBackend = $this->request->getQuery('backend'); if ($authBackend !== null) { if ($authBackend === '') { throw new ConfigurationError($this->translate('User backend name missing')); } elseif (! $this->config->hasSection($authBackend)) { throw new ConfigurationError($this->translate('Unknown user backend provided')); } elseif ($this->config->getSection($authBackend)->backend === null) { throw new ConfigurationError( sprintf($this->translate('Backend "%s" has no `backend\' setting'), $authBackend) ); } $configValues = $this->config->getSection($authBackend)->toArray(); $configValues['type'] = $configValues['backend']; $configValues['name'] = $authBackend; $this->populate($configValues); } elseif (empty($this->resources)) { $externalBackends = array_filter( $this->config->toArray(), function ($authBackendCfg) { return isset($authBackendCfg['backend']) && $authBackendCfg['backend'] === 'external'; } ); if (false === empty($externalBackends)) { throw new ConfigurationError($this->translate('Could not find any resources for authentication')); } } } /** * Return a checkbox to be displayed at the beginning of the form * which allows the user to skip the connection validation * * @return Zend_Form_Element */ protected function getForceCreationCheckbox() { return $this->createElement( 'checkbox', 'force_creation', array( 'order' => 0, 'ignore' => true, 'label' => $this->translate('Force Changes'), 'description' => $this->translate('Check this box to enforce changes without connectivity validation') ) ); } /** * @see Form::createElements() */ public function createElements(array $formData) { $backendTypes = array(); $backendType = isset($formData['type']) ? $formData['type'] : null; if (isset($this->resources['db'])) { $backendTypes['db'] = $this->translate('Database'); } if (isset($this->resources['ldap']) && ($backendType === 'ldap' || Platform::extensionLoaded('ldap'))) { $backendTypes['ldap'] = 'LDAP'; } $externalBackends = array_filter( $this->config->toArray(), function ($authBackendCfg) { return isset($authBackendCfg['backend']) && $authBackendCfg['backend'] === 'external'; } ); if ($backendType === 'external' || empty($externalBackends)) { $backendTypes['external'] = $this->translate('External'); } if ($backendType === null) { $backendType = key($backendTypes); } $this->addElement( 'select', 'type', array( 'ignore' => true, 'required' => true, 'autosubmit' => true, 'label' => $this->translate('Backend Type'), 'description' => $this->translate( 'The type of the resource to use for this authenticaton provider' ), 'multiOptions' => $backendTypes ) ); if (isset($formData['force_creation']) && $formData['force_creation']) { // In case another error occured and the checkbox was displayed before $this->addElement($this->getForceCreationCheckbox()); } $this->addElements($this->getBackendForm($backendType)->createElements($formData)->getElements()); } /** * Return the configuration for the chosen resource * * @return ConfigObject */ public function getResourceConfig() { return ResourceFactory::getResourceConfig($this->getValue('resource')); } }