2015-07-03 10:49:12 +02:00
|
|
|
<?php
|
|
|
|
|
|
|
|
namespace Icinga\Module\Director\Forms;
|
|
|
|
|
2015-11-17 17:27:55 +01:00
|
|
|
use Icinga\Exception\ConfigurationError;
|
2016-10-13 16:02:51 +02:00
|
|
|
use Icinga\Module\Director\CustomVariable\CustomVariables;
|
2016-11-01 18:28:36 +01:00
|
|
|
use Icinga\Module\Director\Hook\DataTypeHook;
|
2015-07-03 13:17:05 +02:00
|
|
|
use Icinga\Module\Director\Web\Form\DirectorObjectForm;
|
2016-02-17 11:49:56 +01:00
|
|
|
use Icinga\Application\Hook;
|
2015-11-17 17:27:55 +01:00
|
|
|
use Exception;
|
2015-07-03 10:49:12 +02:00
|
|
|
|
2015-07-03 13:17:05 +02:00
|
|
|
class DirectorDatafieldForm extends DirectorObjectForm
|
2015-07-03 10:49:12 +02:00
|
|
|
{
|
2015-07-31 15:51:07 +02:00
|
|
|
protected $objectName = 'Data field';
|
|
|
|
|
2016-10-13 16:02:51 +02:00
|
|
|
protected function onRequest()
|
|
|
|
{
|
|
|
|
if ($this->hasBeenSent()) {
|
|
|
|
if ($this->shouldBeDeleted()) {
|
|
|
|
$varname = $this->getSentValue('varname');
|
|
|
|
if ($cnt = CustomVariables::countAll($varname, $this->getDb())) {
|
|
|
|
$this->askForVariableDeletion($varname, $cnt);
|
|
|
|
}
|
2017-07-11 15:28:04 +02:00
|
|
|
} elseif ($this->shouldBeRenamed()) {
|
|
|
|
$varname = $this->object()->getOriginalProperty('varname');
|
|
|
|
if ($cnt = CustomVariables::countAll($varname, $this->getDb())) {
|
|
|
|
$this->askForVariableRename(
|
|
|
|
$varname,
|
|
|
|
$this->getSentValue('varname'),
|
|
|
|
$cnt
|
|
|
|
);
|
|
|
|
}
|
2016-10-13 16:02:51 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return parent::onRequest();
|
|
|
|
}
|
|
|
|
|
|
|
|
protected function askForVariableDeletion($varname, $cnt)
|
|
|
|
{
|
|
|
|
$msg = $this->translate(
|
|
|
|
'Leaving custom variables in place while removing the related field is'
|
|
|
|
. ' perfectly legal and might be a desired operation. This way you can'
|
|
|
|
. ' no longer modify related custom variables in the Director GUI, but'
|
|
|
|
. ' the variables themselves will stay there and continue to be deployed.'
|
|
|
|
. ' When you re-add a field for the same variable later on, everything'
|
|
|
|
. ' will continue to work as before'
|
|
|
|
);
|
|
|
|
|
|
|
|
$this->addBoolean('wipe_vars', array(
|
|
|
|
'label' => $this->translate('Wipe related vars'),
|
|
|
|
'description' => sprintf($msg, $this->getSentValue('varname')),
|
|
|
|
'required' => true,
|
|
|
|
));
|
|
|
|
|
|
|
|
if ($wipe = $this->getSentValue('wipe_vars')) {
|
|
|
|
if ($wipe === 'y') {
|
|
|
|
CustomVariables::deleteAll($varname, $this->getDb());
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
$this->abortDeletion();
|
|
|
|
$this->addError(
|
|
|
|
sprintf(
|
|
|
|
$this->translate('Also wipe all "%s" custom variables from %d objects?'),
|
|
|
|
$varname,
|
|
|
|
$cnt
|
|
|
|
)
|
|
|
|
);
|
|
|
|
$this->getElement('wipe_vars')->addError(
|
|
|
|
sprintf(
|
|
|
|
$this->translate(
|
|
|
|
'There are %d objects with a related property. Should I also'
|
|
|
|
. ' remove the "%s" property from them?'
|
|
|
|
),
|
|
|
|
$cnt,
|
|
|
|
$varname
|
|
|
|
)
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-07-11 15:28:04 +02:00
|
|
|
protected function askForVariableRename($oldname, $newname, $cnt)
|
|
|
|
{
|
|
|
|
$msg = $this->translate(
|
|
|
|
'Leaving custom variables in place while renaming the related field is'
|
|
|
|
. ' perfectly legal and might be a desired operation. This way you can'
|
|
|
|
. ' no longer modify related custom variables in the Director GUI, but'
|
|
|
|
. ' the variables themselves will stay there and continue to be deployed.'
|
|
|
|
. ' When you re-add a field for the same variable later on, everything'
|
|
|
|
. ' will continue to work as before'
|
|
|
|
);
|
|
|
|
|
|
|
|
$this->addBoolean('rename_vars', array(
|
|
|
|
'label' => $this->translate('Rename related vars'),
|
|
|
|
'description' => sprintf($msg, $this->getSentValue('varname')),
|
|
|
|
'required' => true,
|
|
|
|
));
|
|
|
|
|
|
|
|
if ($wipe = $this->getSentValue('rename_vars')) {
|
|
|
|
if ($wipe === 'y') {
|
|
|
|
CustomVariables::renameAll($oldname, $newname, $this->getDb());
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
$this->abortDeletion();
|
|
|
|
$this->addError(
|
|
|
|
sprintf(
|
|
|
|
$this->translate('Also rename all "%s" custom variables to "%s" on %d objects?'),
|
|
|
|
$oldname,
|
|
|
|
$newname,
|
|
|
|
$cnt
|
|
|
|
)
|
|
|
|
);
|
|
|
|
$this->getElement('rename_vars')->addError(
|
|
|
|
sprintf(
|
|
|
|
$this->translate(
|
|
|
|
'There are %d objects with a related property. Should I also'
|
|
|
|
. ' rename the "%s" property to "%s" on them?'
|
|
|
|
),
|
|
|
|
$cnt,
|
|
|
|
$oldname,
|
|
|
|
$newname
|
|
|
|
)
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-07-03 10:49:12 +02:00
|
|
|
public function setup()
|
|
|
|
{
|
2015-07-31 15:49:41 +02:00
|
|
|
$this->addHtmlHint(
|
2016-10-17 19:48:04 +02:00
|
|
|
$this->translate(
|
|
|
|
'Data fields allow you to customize input controls for Icinga custom'
|
|
|
|
. ' variables. Once you defined them here, you can provide them through'
|
|
|
|
. ' your defined templates. This gives you a granular control over what'
|
|
|
|
. ' properties your users should be allowed to configure in which way.'
|
|
|
|
)
|
2015-07-31 15:49:41 +02:00
|
|
|
);
|
|
|
|
|
2015-07-03 10:49:12 +02:00
|
|
|
$this->addElement('text', 'varname', array(
|
2015-07-30 09:07:00 +02:00
|
|
|
'label' => $this->translate('Field name'),
|
2016-10-17 19:48:04 +02:00
|
|
|
'description' => $this->translate(
|
|
|
|
'The unique name of the field. This will be the name of the custom'
|
|
|
|
. ' variable in the rendered Icinga configuration.'
|
|
|
|
),
|
2015-07-30 09:07:00 +02:00
|
|
|
'required' => true,
|
2015-07-03 10:49:12 +02:00
|
|
|
));
|
|
|
|
|
|
|
|
$this->addElement('text', 'caption', array(
|
2015-07-30 09:07:00 +02:00
|
|
|
'label' => $this->translate('Caption'),
|
2015-07-31 16:00:25 +02:00
|
|
|
'required' => true,
|
2016-10-17 19:48:04 +02:00
|
|
|
'description' => $this->translate(
|
|
|
|
'The caption which should be displayed to your users when this field'
|
|
|
|
. ' is shown'
|
|
|
|
)
|
2015-07-03 10:49:12 +02:00
|
|
|
));
|
|
|
|
|
|
|
|
$this->addElement('textarea', 'description', array(
|
2015-07-30 09:07:00 +02:00
|
|
|
'label' => $this->translate('Description'),
|
2016-10-17 19:48:04 +02:00
|
|
|
'description' => $this->translate(
|
|
|
|
'An extended description for this field. Will be shown as soon as a'
|
|
|
|
. ' user puts the focus on this field'
|
|
|
|
),
|
2015-07-30 09:07:00 +02:00
|
|
|
'rows' => '3',
|
2015-07-03 10:49:12 +02:00
|
|
|
));
|
|
|
|
|
2015-11-17 17:27:55 +01:00
|
|
|
$error = false;
|
|
|
|
try {
|
|
|
|
$types = $this->enumDataTypes();
|
|
|
|
} catch (Exception $e) {
|
|
|
|
$error = $e->getMessage();
|
|
|
|
$types = $this->optionalEnum(array());
|
|
|
|
}
|
|
|
|
|
2015-07-24 15:29:27 +02:00
|
|
|
$this->addElement('select', 'datatype', array(
|
|
|
|
'label' => $this->translate('Data type'),
|
|
|
|
'description' => $this->translate('Field type'),
|
|
|
|
'required' => true,
|
2015-11-17 17:27:55 +01:00
|
|
|
'multiOptions' => $types,
|
2015-07-30 09:07:00 +02:00
|
|
|
'class' => 'autosubmit',
|
2015-07-03 10:49:12 +02:00
|
|
|
));
|
2015-11-17 17:27:55 +01:00
|
|
|
if ($error) {
|
|
|
|
$this->getElement('datatype')->addError($error);
|
|
|
|
}
|
2015-07-27 17:25:09 +02:00
|
|
|
|
2015-11-17 17:27:55 +01:00
|
|
|
try {
|
2017-07-11 15:28:04 +02:00
|
|
|
$object = $this->object();
|
2015-11-17 17:27:55 +01:00
|
|
|
if ($class = $this->getSentValue('datatype')) {
|
|
|
|
if ($class && array_key_exists($class, $types)) {
|
|
|
|
$this->addSettings($class);
|
|
|
|
}
|
2017-07-11 15:28:04 +02:00
|
|
|
} elseif ($class = $object->get('datatype')) {
|
2015-07-27 17:25:09 +02:00
|
|
|
$this->addSettings($class);
|
|
|
|
}
|
2015-11-17 17:27:55 +01:00
|
|
|
|
|
|
|
// TODO: next line looks like obsolete duplicate code to me
|
|
|
|
$this->addSettings();
|
|
|
|
} catch (Exception $e) {
|
|
|
|
$this->getElement('datatype')->addError($e->getMessage());
|
2015-07-31 16:00:25 +02:00
|
|
|
}
|
|
|
|
|
2017-07-11 15:28:04 +02:00
|
|
|
foreach ($object->getSettings() as $key => $val) {
|
2015-07-31 16:00:25 +02:00
|
|
|
if ($el = $this->getElement($key)) {
|
|
|
|
$el->setValue($val);
|
|
|
|
}
|
2015-07-27 17:25:09 +02:00
|
|
|
}
|
2016-04-20 22:38:28 +02:00
|
|
|
|
|
|
|
$this->setButtons();
|
2015-07-27 17:25:09 +02:00
|
|
|
}
|
|
|
|
|
2017-07-11 15:28:04 +02:00
|
|
|
public function shouldBeRenamed()
|
|
|
|
{
|
|
|
|
$object = $this->object();
|
|
|
|
return $object->hasBeenLoadedFromDb()
|
|
|
|
&& $object->getOriginalProperty('varname') !== $this->getSentValue('varname');
|
|
|
|
}
|
|
|
|
|
2015-07-27 17:25:09 +02:00
|
|
|
protected function addSettings($class = null)
|
|
|
|
{
|
|
|
|
if ($class === null) {
|
2015-11-17 17:27:55 +01:00
|
|
|
$class = $this->getValue('datatype');
|
|
|
|
}
|
|
|
|
|
|
|
|
if ($class !== null) {
|
|
|
|
if (! class_exists($class)) {
|
|
|
|
throw new ConfigurationError(
|
|
|
|
'The hooked class "%s" for this data field does no longer exist',
|
|
|
|
$class
|
|
|
|
);
|
2015-07-27 17:25:09 +02:00
|
|
|
}
|
2015-11-17 17:27:55 +01:00
|
|
|
|
2015-07-27 17:25:09 +02:00
|
|
|
$class::addSettingsFormFields($this);
|
|
|
|
}
|
2015-07-27 15:52:35 +02:00
|
|
|
}
|
2015-07-03 10:49:12 +02:00
|
|
|
|
2015-07-31 16:00:25 +02:00
|
|
|
protected function clearOutdatedSettings()
|
2015-07-27 15:52:35 +02:00
|
|
|
{
|
2015-07-31 16:00:25 +02:00
|
|
|
$names = array();
|
|
|
|
$object = $this->object();
|
|
|
|
$global = array('varname', 'description', 'caption', 'datatype');
|
|
|
|
|
2016-11-01 18:28:36 +01:00
|
|
|
/** @var \Zend_Form_Element $el */
|
2015-07-31 16:00:25 +02:00
|
|
|
foreach ($this->getElements() as $el) {
|
2016-02-26 12:42:21 +01:00
|
|
|
if ($el->getIgnore()) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2015-07-31 16:00:25 +02:00
|
|
|
$name = $el->getName();
|
2016-02-26 12:42:21 +01:00
|
|
|
if (in_array($name, $global)) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2015-07-31 16:00:25 +02:00
|
|
|
$names[$name] = $name;
|
2015-07-24 15:29:27 +02:00
|
|
|
}
|
2015-07-27 15:52:35 +02:00
|
|
|
|
2015-07-31 16:00:25 +02:00
|
|
|
|
|
|
|
foreach ($object->getSettings() as $setting => $value) {
|
|
|
|
if (! array_key_exists($setting, $names)) {
|
|
|
|
unset($object->$setting);
|
|
|
|
}
|
|
|
|
}
|
2015-07-24 15:29:27 +02:00
|
|
|
}
|
|
|
|
|
2015-07-31 16:00:25 +02:00
|
|
|
public function onSuccess()
|
2015-07-28 13:20:27 +02:00
|
|
|
{
|
2015-07-31 16:00:25 +02:00
|
|
|
$this->clearOutdatedSettings();
|
2015-07-28 13:20:27 +02:00
|
|
|
|
2015-07-31 16:00:25 +02:00
|
|
|
if ($class = $this->getValue('datatype')) {
|
|
|
|
if (array_key_exists($class, $this->enumDataTypes())) {
|
|
|
|
$this->addHidden('format', $class::getFormat());
|
2015-07-28 13:20:27 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-07-31 16:00:25 +02:00
|
|
|
parent::onSuccess();
|
2015-07-28 13:20:27 +02:00
|
|
|
}
|
|
|
|
|
2015-07-24 15:29:27 +02:00
|
|
|
protected function enumDataTypes()
|
|
|
|
{
|
|
|
|
$hooks = Hook::all('Director\\DataType');
|
|
|
|
$enum = array(null => '- please choose -');
|
2016-11-01 18:28:36 +01:00
|
|
|
/** @var DataTypeHook $hook */
|
2015-07-24 15:29:27 +02:00
|
|
|
foreach ($hooks as $hook) {
|
|
|
|
$enum[get_class($hook)] = $hook->getName();
|
|
|
|
}
|
|
|
|
|
|
|
|
return $enum;
|
2015-07-03 10:49:12 +02:00
|
|
|
}
|
|
|
|
}
|