mirror of
				https://github.com/Icinga/icingaweb2.git
				synced 2025-10-26 08:54:05 +01:00 
			
		
		
		
	
		
			
				
	
	
		
			380 lines
		
	
	
		
			13 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
			
		
		
	
	
			380 lines
		
	
	
		
			13 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
| <?php
 | |
| /* Icinga Web 2 | (c) 2014 Icinga Development Team | GPLv2+ */
 | |
| 
 | |
| namespace Icinga\Forms\Config;
 | |
| 
 | |
| use InvalidArgumentException;
 | |
| use Icinga\Application\Platform;
 | |
| use Icinga\Exception\ConfigurationError;
 | |
| use Icinga\Data\ConfigObject;
 | |
| use Icinga\Data\Inspectable;
 | |
| use Icinga\Data\Inspection;
 | |
| use Icinga\Data\ResourceFactory;
 | |
| use Icinga\Forms\ConfigForm;
 | |
| use Icinga\Forms\Config\Resource\DbResourceForm;
 | |
| use Icinga\Forms\Config\Resource\FileResourceForm;
 | |
| use Icinga\Forms\Config\Resource\LdapResourceForm;
 | |
| use Icinga\Forms\Config\Resource\LivestatusResourceForm;
 | |
| use Icinga\Forms\Config\Resource\SshResourceForm;
 | |
| use Icinga\Web\Form;
 | |
| use Icinga\Web\Notification;
 | |
| 
 | |
| class ResourceConfigForm extends ConfigForm
 | |
| {
 | |
|     /**
 | |
|      * Initialize this form
 | |
|      */
 | |
|     public function init()
 | |
|     {
 | |
|         $this->setName('form_config_resource');
 | |
|         $this->setSubmitLabel($this->translate('Save Changes'));
 | |
|         $this->setValidatePartial(true);
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Return a form object for the given resource type
 | |
|      *
 | |
|      * @param   string      $type      The resource type for which to return a form
 | |
|      *
 | |
|      * @return  Form
 | |
|      */
 | |
|     public function getResourceForm($type)
 | |
|     {
 | |
|         if ($type === 'db') {
 | |
|             return new DbResourceForm();
 | |
|         } elseif ($type === 'ldap') {
 | |
|             return new LdapResourceForm();
 | |
|         } elseif ($type === 'livestatus') {
 | |
|             return new LivestatusResourceForm();
 | |
|         } elseif ($type === 'file') {
 | |
|             return new FileResourceForm();
 | |
|         } elseif ($type === 'ssh') {
 | |
|             return new SshResourceForm();
 | |
|         } else {
 | |
|             throw new InvalidArgumentException(sprintf($this->translate('Invalid resource type "%s" provided'), $type));
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Add a particular resource
 | |
|      *
 | |
|      * 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 resource does already exist
 | |
|      */
 | |
|     public function add(array $values)
 | |
|     {
 | |
|         $name = isset($values['name']) ? $values['name'] : '';
 | |
|         if (! $name) {
 | |
|             throw new InvalidArgumentException($this->translate('Resource name missing'));
 | |
|         } elseif ($this->config->hasSection($name)) {
 | |
|             throw new InvalidArgumentException($this->translate('Resource already exists'));
 | |
|         }
 | |
| 
 | |
|         unset($values['name']);
 | |
|         $this->config->setSection($name, $values);
 | |
|         return $this;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Edit a particular resource
 | |
|      *
 | |
|      * @param   string      $name           The name of the resource to edit
 | |
|      * @param   array       $values         The values to edit the configuration with
 | |
|      *
 | |
|      * @return  array                       The edited configuration
 | |
|      *
 | |
|      * @throws  InvalidArgumentException    In case the resource does not exist
 | |
|      */
 | |
|     public function edit($name, array $values)
 | |
|     {
 | |
|         if (! $name) {
 | |
|             throw new InvalidArgumentException($this->translate('Old resource name missing'));
 | |
|         } elseif (! ($newName = isset($values['name']) ? $values['name'] : '')) {
 | |
|             throw new InvalidArgumentException($this->translate('New resource name missing'));
 | |
|         } elseif (! $this->config->hasSection($name)) {
 | |
|             throw new InvalidArgumentException($this->translate('Unknown resource provided'));
 | |
|         }
 | |
| 
 | |
|         $resourceConfig = $this->config->getSection($name);
 | |
|         $this->config->removeSection($name);
 | |
|         unset($values['name']);
 | |
|         $this->config->setSection($newName, $resourceConfig->merge($values));
 | |
|         return $resourceConfig;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Remove a particular resource
 | |
|      *
 | |
|      * @param   string      $name           The name of the resource to remove
 | |
|      *
 | |
|      * @return  array                       The removed resource configuration
 | |
|      *
 | |
|      * @throws  InvalidArgumentException    In case the resource does not exist
 | |
|      */
 | |
|     public function remove($name)
 | |
|     {
 | |
|         if (! $name) {
 | |
|             throw new InvalidArgumentException($this->translate('Resource name missing'));
 | |
|         } elseif (! $this->config->hasSection($name)) {
 | |
|             throw new InvalidArgumentException($this->translate('Unknown resource provided'));
 | |
|         }
 | |
| 
 | |
|         $resourceConfig = $this->config->getSection($name);
 | |
|         $resourceForm = $this->getResourceForm($resourceConfig->type);
 | |
|         if (method_exists($resourceForm, 'beforeRemove')) {
 | |
|             $resourceForm::beforeRemove($resourceConfig);
 | |
|         }
 | |
| 
 | |
|         $this->config->removeSection($name);
 | |
|         return $resourceConfig;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Add or edit a resource 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()
 | |
|     {
 | |
|         $resourceForm = $this->getResourceForm($this->getElement('type')->getValue());
 | |
| 
 | |
|         if (($el = $this->getElement('force_creation')) === null || false === $el->isChecked()) {
 | |
|             $inspection = static::inspectResource($this);
 | |
|             if ($inspection !== null && $inspection->hasError()) {
 | |
|                 $this->error($inspection->getError());
 | |
|                 $this->addElement($this->getForceCreationCheckbox());
 | |
|                 return false;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         $resource = $this->request->getQuery('resource');
 | |
|         try {
 | |
|             if ($resource === null) { // create new resource
 | |
|                 if (method_exists($resourceForm, 'beforeAdd')) {
 | |
|                     if (! $resourceForm::beforeAdd($this)) {
 | |
|                         return false;
 | |
|                     }
 | |
|                 }
 | |
|                 $this->add(array_filter($this->getValues()));
 | |
|                 $message = $this->translate('Resource "%s" has been successfully created');
 | |
|             } else { // edit existing resource
 | |
|                 $this->edit($resource, array_filter($this->getValues()));
 | |
|                 $message = $this->translate('Resource "%s" has been successfully changed');
 | |
|             }
 | |
|         } catch (InvalidArgumentException $e) {
 | |
|             Notification::error($e->getMessage());
 | |
|             return false;
 | |
|         }
 | |
| 
 | |
|         if ($this->save()) {
 | |
|             Notification::success(sprintf($message, $this->getElement('name')->getValue()));
 | |
|         } else {
 | |
|             return false;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Populate the form in case a resource is being edited
 | |
|      *
 | |
|      * @see Form::onRequest()
 | |
|      *
 | |
|      * @throws  ConfigurationError      In case the backend name is missing in the request or is invalid
 | |
|      */
 | |
|     public function onRequest()
 | |
|     {
 | |
|         $resource = $this->request->getQuery('resource');
 | |
|         if ($resource !== null) {
 | |
|             if ($resource === '') {
 | |
|                 throw new ConfigurationError($this->translate('Resource name missing'));
 | |
|             } elseif (! $this->config->hasSection($resource)) {
 | |
|                 throw new ConfigurationError($this->translate('Unknown resource provided'));
 | |
|             }
 | |
| 
 | |
|             $configValues = $this->config->getSection($resource)->toArray();
 | |
|             $configValues['name'] = $resource;
 | |
|             $this->populate($configValues);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * 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::createElemeents()
 | |
|      */
 | |
|     public function createElements(array $formData)
 | |
|     {
 | |
|         $resourceType = isset($formData['type']) ? $formData['type'] : 'db';
 | |
| 
 | |
|         $resourceTypes = array(
 | |
|             'file'          => $this->translate('File'),
 | |
|             'livestatus'    => 'Livestatus',
 | |
|             'ssh'           => $this->translate('SSH Identity'),
 | |
|         );
 | |
|         if ($resourceType === 'ldap' || Platform::hasLdapSupport()) {
 | |
|             $resourceTypes['ldap'] = 'LDAP';
 | |
|         }
 | |
|         if ($resourceType === 'db' || Platform::hasDatabaseSupport()) {
 | |
|             $resourceTypes['db'] = $this->translate('SQL Database');
 | |
|         }
 | |
| 
 | |
|         $this->addElement(
 | |
|             'select',
 | |
|             'type',
 | |
|             array(
 | |
|                 'required'          => true,
 | |
|                 'autosubmit'        => true,
 | |
|                 'label'             => $this->translate('Resource Type'),
 | |
|                 'description'       => $this->translate('The type of resource'),
 | |
|                 'multiOptions'      => $resourceTypes,
 | |
|                 'value'             => $resourceType
 | |
|             )
 | |
|         );
 | |
| 
 | |
|         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->getResourceForm($resourceType)->createElements($formData)->getElements());
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Create a resource by using the given form's values and return its inspection results
 | |
|      *
 | |
|      * @param   Form    $form
 | |
|      *
 | |
|      * @return  Inspection
 | |
|      */
 | |
|     public static function inspectResource(Form $form)
 | |
|     {
 | |
|         if ($form->getValue('type') !== 'ssh') {
 | |
|             $resource = ResourceFactory::createResource(new ConfigObject($form->getValues()));
 | |
|             if ($resource instanceof Inspectable) {
 | |
|                 return $resource->inspect();
 | |
|             }
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Run the configured resource's inspection checks and show the result, if necessary
 | |
|      *
 | |
|      * This will only run any validation if the user pushed the 'resource_validation' button.
 | |
|      *
 | |
|      * @param   array   $formData
 | |
|      *
 | |
|      * @return  bool
 | |
|      */
 | |
|     public function isValidPartial(array $formData)
 | |
|     {
 | |
|         if ($this->getElement('resource_validation')->isChecked() && parent::isValid($formData)) {
 | |
|             $inspection = static::inspectResource($this);
 | |
|             if ($inspection !== null) {
 | |
|                 $join = function ($e) use (& $join) {
 | |
|                     return is_string($e) ? $e : join("\n", array_map($join, $e));
 | |
|                 };
 | |
|                 $this->addElement(
 | |
|                     'note',
 | |
|                     'inspection_output',
 | |
|                     array(
 | |
|                         'order'         => 0,
 | |
|                         'value'         => '<strong>' . $this->translate('Validation Log') . "</strong>\n\n"
 | |
|                             . join("\n", array_map($join, $inspection->toArray())),
 | |
|                         'decorators'    => array(
 | |
|                             'ViewHelper',
 | |
|                             array('HtmlTag', array('tag' => 'pre', 'class' => 'log-output')),
 | |
|                         )
 | |
|                     )
 | |
|                 );
 | |
| 
 | |
|                 if ($inspection->hasError()) {
 | |
|                     $this->warning(sprintf(
 | |
|                         $this->translate('Failed to successfully validate the configuration: %s'),
 | |
|                         $inspection->getError()
 | |
|                     ));
 | |
|                     return false;
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             $this->info($this->translate('The configuration has been successfully validated.'));
 | |
|         }
 | |
| 
 | |
|         return true;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Add a submit button to this form and one to manually validate the configuration
 | |
|      *
 | |
|      * Calls parent::addSubmitButton() to add the submit button.
 | |
|      *
 | |
|      * @return  $this
 | |
|      */
 | |
|     public function addSubmitButton()
 | |
|     {
 | |
|         parent::addSubmitButton()
 | |
|             ->getElement('btn_submit')
 | |
|             ->setDecorators(array('ViewHelper'));
 | |
| 
 | |
|         $this->addElement(
 | |
|             'submit',
 | |
|             'resource_validation',
 | |
|             array(
 | |
|                 'ignore'                => true,
 | |
|                 'label'                 => $this->translate('Validate Configuration'),
 | |
|                 'data-progress-label'   => $this->translate('Validation In Progress'),
 | |
|                 'decorators'            => array('ViewHelper')
 | |
|             )
 | |
|         );
 | |
| 
 | |
|         $this->setAttrib('data-progress-element', 'resource-progress');
 | |
|         $this->addElement(
 | |
|             'note',
 | |
|             'resource-progress',
 | |
|             array(
 | |
|                 'decorators'    => array(
 | |
|                     'ViewHelper',
 | |
|                     array('Spinner', array('id' => 'resource-progress'))
 | |
|                 )
 | |
|             )
 | |
|         );
 | |
| 
 | |
|         $this->addDisplayGroup(
 | |
|             array('btn_submit', 'resource_validation', 'resource-progress'),
 | |
|             'submit_validation',
 | |
|             array(
 | |
|                 'decorators' => array(
 | |
|                     'FormElements',
 | |
|                     array('HtmlTag', array('tag' => 'div', 'class' => 'control-group form-controls'))
 | |
|                 )
 | |
|             )
 | |
|         );
 | |
| 
 | |
|         return $this;
 | |
|     }
 | |
| }
 |