chg: [instance:settings] Integrated actually save of settings

pull/70/head
mokaddem 2021-07-27 10:40:58 +02:00
parent 34fc7ad027
commit 7fc2c595d7
4 changed files with 77 additions and 26 deletions

View File

@ -7,6 +7,7 @@ use Cake\Utility\Hash;
use Cake\Utility\Text; use Cake\Utility\Text;
use \Cake\Database\Expression\QueryExpression; use \Cake\Database\Expression\QueryExpression;
use Cake\Event\EventInterface; use Cake\Event\EventInterface;
use Cake\Core\Configure;
class InstanceController extends AppController class InstanceController extends AppController
{ {
@ -125,11 +126,6 @@ class InstanceController extends AppController
if (empty($errors)) { if (empty($errors)) {
$message = __('Setting `{0}` saved', $data['name']); $message = __('Setting `{0}` saved', $data['name']);
$data = $this->Settings->getSetting($data['name']); $data = $this->Settings->getSetting($data['name']);
// TO DEL
$data['errorMessage'] = 'Test test test';
$data['error'] = true;
$data['error'] = false;
// TO DEL
} }
$this->CRUD->setResponseForController('saveSetting', empty($errors), $message, $data, $errors); $this->CRUD->setResponseForController('saveSetting', empty($errors), $message, $data, $errors);
$responsePayload = $this->CRUD->getResponsePayload(); $responsePayload = $this->CRUD->getResponsePayload();

View File

@ -124,22 +124,12 @@ class SettingsProviderTable extends AppTable
} }
} }
if (!$skipValidation) { if (!$skipValidation) {
$validationResult = false; $validationResult = true;
if (!isset($setting['value'])) { if (!isset($setting['value'])) {
$validationResult = $this->settingValidator->testEmptyBecomesDefault(null, $setting); $validationResult = $this->settingValidator->testEmptyBecomesDefault(null, $setting);
} else if (isset($setting['test'])) { } else if (isset($setting['test'])) {
$setting['value'] = $setting['value'] ?? ''; $setting['value'] = $setting['value'] ?? '';
if (is_callable($setting['test'])) { // Validate with anonymous function $validationResult = $this->evaluateFunctionForSetting($setting['test'], $setting);
$validationResult = $setting['test']($setting['value'], $setting, new Validator());
} else if (method_exists($this->settingValidator, $setting['test'])) { // Validate with function defined in settingValidator class
$validationResult = $this->settingValidator->{$setting['test']}($setting['value'], $setting);
} else {
$validator = new Validator();
if (method_exists($validator, $setting['test'])) { // Validate with cake's validator function
$validator->{$setting['test']};
$validationResult = $validator->validate($setting['value']);
}
}
} }
if ($validationResult !== true) { if ($validationResult !== true) {
$setting['severity'] = $setting['severity'] ?? 'warning'; $setting['severity'] = $setting['severity'] ?? 'warning';
@ -152,12 +142,37 @@ class SettingsProviderTable extends AppTable
} }
return $setting; return $setting;
} }
/**
* evaluateFunctionForSetting - evaluate the provided function. If function could not be evaluated, its result is defaulted to true
*
* @param mixed $fun
* @param array $setting
* @return mixed
*/
public function evaluateFunctionForSetting($fun, $setting)
{
$functionResult = true;
if (is_callable($fun)) { // Validate with anonymous function
$functionResult = $fun($setting['value'], $setting, new Validator());
} else if (method_exists($this->settingValidator, $fun)) { // Validate with function defined in settingValidator class
$functionResult = $this->settingValidator->{$fun}($setting['value'], $setting);
} else {
$validator = new Validator();
if (method_exists($validator, $fun)) { // Validate with cake's validator function
$validator->{$fun};
$functionResult = $validator->validate($setting['value']);
}
}
return $functionResult;
}
/** /**
* Support up to 3 level: * Support up to 3 level:
* Application -> Network -> Proxy -> Proxy.URL * Application -> Network -> Proxy -> Proxy.URL
* *
* Leave errorMessage empty to let the validator generate the error message * - Leave errorMessage empty to let the validator generate the error message
* - Default severity level is `info` if a `default` value is provided otherwise it becomes `critical`
*/ */
private function generateSettingsConfiguration() private function generateSettingsConfiguration()
{ {
@ -188,8 +203,16 @@ class SettingsProviderTable extends AppTable
'errorMessage' => 'to del', 'errorMessage' => 'to del',
'default' => 'A-default-value', 'default' => 'A-default-value',
'name' => 'To DEL', 'name' => 'To DEL',
'test' => function ($value) { 'test' => function($value) {
return empty($value) ? __('Oh not! it\'s not valid!') : ''; return empty($value) ? __('Oh not! it\'s not valid!') : true;
},
'beforeSave' => function($value, $setting) {
if ($value != 'foo') {
return 'value must be `foo`!';
}
return true;
},
'afterSave' => function($value, $setting) {
}, },
'type' => 'string' 'type' => 'string'
], ],

View File

@ -9,6 +9,9 @@ use Cake\ORM\TableRegistry;
class SettingsTable extends AppTable class SettingsTable extends AppTable
{ {
private static $FILENAME = 'cerebrate';
private static $CONFIG_KEY = 'Cerebrate';
public function initialize(array $config): void public function initialize(array $config): void
{ {
parent::initialize($config); parent::initialize($config);
@ -18,7 +21,7 @@ class SettingsTable extends AppTable
public function getSettings($full=false): array public function getSettings($full=false): array
{ {
$settings = Configure::read()['Cerebrate']; $settings = $this->readSettings();
if (empty($full)) { if (empty($full)) {
return $settings; return $settings;
} else { } else {
@ -36,7 +39,7 @@ class SettingsTable extends AppTable
public function getSetting($name=false): array public function getSetting($name=false): array
{ {
$settings = Configure::read()['Cerebrate']; $settings = $this->readSettings();
$settingsProvider = $this->SettingsProvider->getSettingsConfiguration($settings); $settingsProvider = $this->SettingsProvider->getSettingsConfiguration($settings);
$settingsFlattened = $this->SettingsProvider->flattenSettingsConfiguration($settingsProvider); $settingsFlattened = $this->SettingsProvider->flattenSettingsConfiguration($settingsProvider);
return $settingsFlattened[$name] ?? []; return $settingsFlattened[$name] ?? [];
@ -45,7 +48,36 @@ class SettingsTable extends AppTable
public function saveSetting(string $name, string $value): array public function saveSetting(string $name, string $value): array
{ {
$errors = []; $errors = [];
// Save setting here! $setting = $this->getSetting($name);
if (!empty($setting['beforeSave'])) {
$setting['value'] = $value ?? '';
$beforeSaveResult = $this->SettingsProvider->evaluateFunctionForSetting($setting['beforeSave'], $setting);
if ($beforeSaveResult !== true) {
$errors[] = $beforeSaveResult;
}
}
if (empty($errors)) {
$saveResult = $this->saveSettingOnDisk($name, $value);
if ($saveResult) {
if (!empty($setting['afterSave'])) {
$this->SettingsProvider->evaluateFunctionForSetting($setting['afterSave'], $setting);
}
}
}
return $errors; return $errors;
} }
private function readSettings()
{
return Configure::read()[$this::$CONFIG_KEY];
}
private function saveSettingOnDisk($name, $value)
{
$settings = $this->readSettings();
$settings[$name] = $value;
Configure::write($this::$CONFIG_KEY, $settings);
Configure::dump($this::$FILENAME, 'default', [$this::$CONFIG_KEY]);
return true;
}
} }

View File

@ -170,7 +170,7 @@ function isLeaf($setting)
$input.val(result.data.value) $input.val(result.data.value)
} }
handleSettingValueChange($input) handleSettingValueChange($input)
}) }).catch((e) => {})
} }
function handleSettingValueChange($input) { function handleSettingValueChange($input) {
@ -187,7 +187,7 @@ function isLeaf($setting)
const $inputGroup = $input.closest('.input-group') const $inputGroup = $input.closest('.input-group')
const $inputGroupAppend = $inputGroup.find('.input-group-append') const $inputGroupAppend = $inputGroup.find('.input-group-append')
const $saveButton = $inputGroup.find('button.btn-save-setting') const $saveButton = $inputGroup.find('button.btn-save-setting')
$input.removeClass(['is-invalid', 'border-warning', 'border-danger', 'border-info']) $input.removeClass(['is-invalid', 'border-warning', 'border-danger', 'border-info', 'warning', 'info'])
$inputGroupAppend.removeClass('d-none') $inputGroupAppend.removeClass('d-none')
if ($input.is('select') && $input.find('option:selected').data('is-empty-option') == 1) { if ($input.is('select') && $input.find('option:selected').data('is-empty-option') == 1) {
$inputGroupAppend.addClass('d-none') // hide save button if empty selection picked $inputGroupAppend.addClass('d-none') // hide save button if empty selection picked
@ -202,7 +202,7 @@ function isLeaf($setting)
const setting = settingsFlattened[$input.data('setting-name')] const setting = settingsFlattened[$input.data('setting-name')]
if (setting.error) { if (setting.error) {
borderVariant = setting.severity !== undefined ? variantFromSeverity[setting.severity] : 'warning' borderVariant = setting.severity !== undefined ? variantFromSeverity[setting.severity] : 'warning'
$input.addClass(['is-invalid', `border-${borderVariant}`]) $input.addClass(['is-invalid', `border-${borderVariant}`, borderVariant])
if (setting.severity == 'warning') { if (setting.severity == 'warning') {
$input.addClass('warning') $input.addClass('warning')
} }