123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967 |
- <?php
- /**
- * Form management class, displays and processes forms
- *
- * Explanation of used terms:
- * o work_path - original field path, eg. Servers/4/verbose
- * o system_path - work_path modified so that it points to the first server,
- * eg. Servers/1/verbose
- * o translated_path - work_path modified for HTML field name, a path with
- * slashes changed to hyphens, eg. Servers-4-verbose
- */
- declare(strict_types=1);
- namespace PhpMyAdmin\Config;
- use PhpMyAdmin\Config\Forms\User\UserFormList;
- use PhpMyAdmin\Html\MySQLDocumentation;
- use PhpMyAdmin\Sanitize;
- use PhpMyAdmin\Util;
- use const E_USER_WARNING;
- use function array_flip;
- use function array_keys;
- use function array_search;
- use function count;
- use function explode;
- use function function_exists;
- use function gettype;
- use function implode;
- use function is_array;
- use function is_bool;
- use function is_numeric;
- use function mb_substr;
- use function preg_match;
- use function settype;
- use function sprintf;
- use function str_replace;
- use function trigger_error;
- use function trim;
- /**
- * Form management class, displays and processes forms
- */
- class FormDisplay
- {
- /**
- * ConfigFile instance
- *
- * @var ConfigFile
- */
- private $configFile;
- /**
- * Form list
- *
- * @var Form[]
- */
- private $forms = [];
- /**
- * Stores validation errors, indexed by paths
- * [ Form_name ] is an array of form errors
- * [path] is a string storing error associated with single field
- *
- * @var array
- */
- private $errors = [];
- /**
- * Paths changed so that they can be used as HTML ids, indexed by paths
- *
- * @var array
- */
- private $translatedPaths = [];
- /**
- * Server paths change indexes so we define maps from current server
- * path to the first one, indexed by work path
- *
- * @var array
- */
- private $systemPaths = [];
- /**
- * Language strings which will be sent to Messages JS variable
- * Will be looked up in $GLOBALS: str{value} or strSetup{value}
- *
- * @var array
- */
- private $jsLangStrings = [];
- /**
- * Tells whether forms have been validated
- *
- * @var bool
- */
- private $isValidated = true;
- /**
- * Dictionary with user preferences keys
- *
- * @var array|null
- */
- private $userprefsKeys;
- /**
- * Dictionary with disallowed user preferences keys
- *
- * @var array
- */
- private $userprefsDisallow;
- /** @var FormDisplayTemplate */
- private $formDisplayTemplate;
- /**
- * @param ConfigFile $cf Config file instance
- */
- public function __construct(ConfigFile $cf)
- {
- $this->formDisplayTemplate = new FormDisplayTemplate($GLOBALS['PMA_Config']);
- $this->jsLangStrings = [
- 'error_nan_p' => __('Not a positive number!'),
- 'error_nan_nneg' => __('Not a non-negative number!'),
- 'error_incorrect_port' => __('Not a valid port number!'),
- 'error_invalid_value' => __('Incorrect value!'),
- 'error_value_lte' => __('Value must be less than or equal to %s!'),
- ];
- $this->configFile = $cf;
- // initialize validators
- Validator::getValidators($this->configFile);
- }
- /**
- * Returns {@link ConfigFile} associated with this instance
- *
- * @return ConfigFile
- */
- public function getConfigFile()
- {
- return $this->configFile;
- }
- /**
- * Registers form in form manager
- *
- * @param string $formName Form name
- * @param array $form Form data
- * @param int $serverId 0 if new server, validation; >= 1 if editing a server
- *
- * @return void
- */
- public function registerForm($formName, array $form, $serverId = null)
- {
- $this->forms[$formName] = new Form(
- $formName,
- $form,
- $this->configFile,
- $serverId
- );
- $this->isValidated = false;
- foreach ($this->forms[$formName]->fields as $path) {
- $workPath = $serverId === null
- ? $path
- : str_replace('Servers/1/', 'Servers/' . $serverId . '/', $path);
- $this->systemPaths[$workPath] = $path;
- $this->translatedPaths[$workPath] = str_replace('/', '-', $workPath);
- }
- }
- /**
- * Processes forms, returns true on successful save
- *
- * @param bool $allowPartialSave allows for partial form saving
- * on failed validation
- * @param bool $checkFormSubmit whether check for $_POST['submit_save']
- *
- * @return bool whether processing was successful
- */
- public function process($allowPartialSave = true, $checkFormSubmit = true)
- {
- if ($checkFormSubmit && ! isset($_POST['submit_save'])) {
- return false;
- }
- // save forms
- if (count($this->forms) > 0) {
- return $this->save(array_keys($this->forms), $allowPartialSave);
- }
- return false;
- }
- /**
- * Runs validation for all registered forms
- *
- * @return void
- */
- private function validate()
- {
- if ($this->isValidated) {
- return;
- }
- $paths = [];
- $values = [];
- foreach ($this->forms as $form) {
- /** @var Form $form */
- $paths[] = $form->name;
- // collect values and paths
- foreach ($form->fields as $path) {
- $workPath = array_search($path, $this->systemPaths);
- $values[$path] = $this->configFile->getValue($workPath);
- $paths[] = $path;
- }
- }
- // run validation
- $errors = Validator::validate(
- $this->configFile,
- $paths,
- $values,
- false
- );
- // change error keys from canonical paths to work paths
- if (is_array($errors) && count($errors) > 0) {
- $this->errors = [];
- foreach ($errors as $path => $errorList) {
- $workPath = array_search($path, $this->systemPaths);
- // field error
- if (! $workPath) {
- // form error, fix path
- $workPath = $path;
- }
- $this->errors[$workPath] = $errorList;
- }
- }
- $this->isValidated = true;
- }
- /**
- * Outputs HTML for the forms under the menu tab
- *
- * @param bool $showRestoreDefault whether to show "restore default"
- * button besides the input field
- * @param array $jsDefault stores JavaScript code
- * to be displayed
- * @param array $js will be updated with javascript code
- * @param bool $showButtons whether show submit and reset button
- *
- * @return string
- */
- private function displayForms(
- $showRestoreDefault,
- array &$jsDefault,
- array &$js,
- $showButtons
- ) {
- $htmlOutput = '';
- $validators = Validator::getValidators($this->configFile);
- foreach ($this->forms as $form) {
- /** @var Form $form */
- $formErrors = $this->errors[$form->name] ?? null;
- $htmlOutput .= $this->formDisplayTemplate->displayFieldsetTop(
- Descriptions::get('Form_' . $form->name),
- Descriptions::get('Form_' . $form->name, 'desc'),
- $formErrors,
- ['id' => $form->name]
- );
- foreach ($form->fields as $field => $path) {
- $workPath = array_search($path, $this->systemPaths);
- $translatedPath = $this->translatedPaths[$workPath];
- // always true/false for user preferences display
- // otherwise null
- $userPrefsAllow = isset($this->userprefsKeys[$path])
- ? ! isset($this->userprefsDisallow[$path])
- : null;
- // display input
- $htmlOutput .= $this->displayFieldInput(
- $form,
- $field,
- $path,
- $workPath,
- $translatedPath,
- $showRestoreDefault,
- $userPrefsAllow,
- $jsDefault
- );
- // register JS validators for this field
- if (! isset($validators[$path])) {
- continue;
- }
- $this->formDisplayTemplate->addJsValidate($translatedPath, $validators[$path], $js);
- }
- $htmlOutput .= $this->formDisplayTemplate->displayFieldsetBottom($showButtons);
- }
- return $htmlOutput;
- }
- /**
- * Outputs HTML for forms
- *
- * @param bool $tabbedForm if true, use a form with tabs
- * @param bool $showRestoreDefault whether show "restore default" button
- * besides the input field
- * @param bool $showButtons whether show submit and reset button
- * @param string $formAction action attribute for the form
- * @param array|null $hiddenFields array of form hidden fields (key: field
- * name)
- *
- * @return string HTML for forms
- */
- public function getDisplay(
- $tabbedForm = false,
- $showRestoreDefault = false,
- $showButtons = true,
- $formAction = null,
- $hiddenFields = null
- ) {
- static $jsLangSent = false;
- $htmlOutput = '';
- $js = [];
- $jsDefault = [];
- $htmlOutput .= $this->formDisplayTemplate->displayFormTop($formAction, 'post', $hiddenFields);
- if ($tabbedForm) {
- $tabs = [];
- foreach ($this->forms as $form) {
- $tabs[$form->name] = Descriptions::get('Form_' . $form->name);
- }
- $htmlOutput .= $this->formDisplayTemplate->displayTabsTop($tabs);
- }
- // validate only when we aren't displaying a "new server" form
- $isNewServer = false;
- foreach ($this->forms as $form) {
- /** @var Form $form */
- if ($form->index === 0) {
- $isNewServer = true;
- break;
- }
- }
- if (! $isNewServer) {
- $this->validate();
- }
- // user preferences
- $this->loadUserprefsInfo();
- // display forms
- $htmlOutput .= $this->displayForms(
- $showRestoreDefault,
- $jsDefault,
- $js,
- $showButtons
- );
- if ($tabbedForm) {
- $htmlOutput .= $this->formDisplayTemplate->displayTabsBottom();
- }
- $htmlOutput .= $this->formDisplayTemplate->displayFormBottom();
- // if not already done, send strings used for validation to JavaScript
- if (! $jsLangSent) {
- $jsLangSent = true;
- $jsLang = [];
- foreach ($this->jsLangStrings as $strName => $strValue) {
- $jsLang[] = "'" . $strName . "': '" . Sanitize::jsFormat($strValue, false) . '\'';
- }
- $js[] = "$.extend(Messages, {\n\t"
- . implode(",\n\t", $jsLang) . '})';
- }
- $js[] = "$.extend(defaultValues, {\n\t"
- . implode(",\n\t", $jsDefault) . '})';
- return $htmlOutput . $this->formDisplayTemplate->displayJavascript($js);
- }
- /**
- * Prepares data for input field display and outputs HTML code
- *
- * @param Form $form Form object
- * @param string $field field name as it appears in $form
- * @param string $systemPath field path, eg. Servers/1/verbose
- * @param string $workPath work path, eg. Servers/4/verbose
- * @param string $translatedPath work path changed so that it can be
- * used as XHTML id
- * @param bool $showRestoreDefault whether show "restore default" button
- * besides the input field
- * @param bool|null $userPrefsAllow whether user preferences are enabled
- * for this field (null - no support,
- * true/false - enabled/disabled)
- * @param array $jsDefault array which stores JavaScript code
- * to be displayed
- *
- * @return string|null HTML for input field
- */
- private function displayFieldInput(
- Form $form,
- $field,
- $systemPath,
- $workPath,
- $translatedPath,
- $showRestoreDefault,
- $userPrefsAllow,
- array &$jsDefault
- ) {
- $name = Descriptions::get($systemPath);
- $description = Descriptions::get($systemPath, 'desc');
- $value = $this->configFile->get($workPath);
- $valueDefault = $this->configFile->getDefault($systemPath);
- $valueIsDefault = false;
- if ($value === null || $value === $valueDefault) {
- $value = $valueDefault;
- $valueIsDefault = true;
- }
- $opts = [
- 'doc' => $this->getDocLink($systemPath),
- 'show_restore_default' => $showRestoreDefault,
- 'userprefs_allow' => $userPrefsAllow,
- 'userprefs_comment' => Descriptions::get($systemPath, 'cmt'),
- ];
- if (isset($form->default[$systemPath])) {
- $opts['setvalue'] = (string) $form->default[$systemPath];
- }
- if (isset($this->errors[$workPath])) {
- $opts['errors'] = $this->errors[$workPath];
- }
- $type = '';
- switch ($form->getOptionType($field)) {
- case 'string':
- $type = 'text';
- break;
- case 'short_string':
- $type = 'short_text';
- break;
- case 'double':
- case 'integer':
- $type = 'number_text';
- break;
- case 'boolean':
- $type = 'checkbox';
- break;
- case 'select':
- $type = 'select';
- $opts['values'] = $form->getOptionValueList($form->fields[$field]);
- break;
- case 'array':
- $type = 'list';
- $value = (array) $value;
- $valueDefault = (array) $valueDefault;
- break;
- case 'group':
- // :group:end is changed to :group:end:{unique id} in Form class
- $htmlOutput = '';
- if (mb_substr($field, 7, 4) !== 'end:') {
- $htmlOutput .= $this->formDisplayTemplate->displayGroupHeader(
- mb_substr($field, 7)
- );
- } else {
- $this->formDisplayTemplate->displayGroupFooter();
- }
- return $htmlOutput;
- case 'NULL':
- trigger_error('Field ' . $systemPath . ' has no type', E_USER_WARNING);
- return null;
- }
- // detect password fields
- if ($type === 'text'
- && (mb_substr($translatedPath, -9) === '-password'
- || mb_substr($translatedPath, -4) === 'pass'
- || mb_substr($translatedPath, -4) === 'Pass')
- ) {
- $type = 'password';
- }
- // TrustedProxies requires changes before displaying
- if ($systemPath === 'TrustedProxies') {
- foreach ($value as $ip => &$v) {
- if (preg_match('/^-\d+$/', $ip)) {
- continue;
- }
- $v = $ip . ': ' . $v;
- }
- }
- $this->setComments($systemPath, $opts);
- // send default value to form's JS
- $jsLine = '\'' . $translatedPath . '\': ';
- switch ($type) {
- case 'text':
- case 'short_text':
- case 'number_text':
- case 'password':
- $jsLine .= '\'' . Sanitize::escapeJsString($valueDefault) . '\'';
- break;
- case 'checkbox':
- $jsLine .= $valueDefault ? 'true' : 'false';
- break;
- case 'select':
- $valueDefaultJs = is_bool($valueDefault)
- ? (int) $valueDefault
- : $valueDefault;
- $jsLine .= '[\'' . Sanitize::escapeJsString($valueDefaultJs) . '\']';
- break;
- case 'list':
- $val = $valueDefault;
- if (isset($val['wrapper_params'])) {
- unset($val['wrapper_params']);
- }
- $jsLine .= '\'' . Sanitize::escapeJsString(implode("\n", $val))
- . '\'';
- break;
- }
- $jsDefault[] = $jsLine;
- return $this->formDisplayTemplate->displayInput(
- $translatedPath,
- $name,
- $type,
- $value,
- $description,
- $valueIsDefault,
- $opts
- );
- }
- /**
- * Displays errors
- *
- * @return string|null HTML for errors
- */
- public function displayErrors()
- {
- $this->validate();
- if (count($this->errors) === 0) {
- return null;
- }
- $htmlOutput = '';
- foreach ($this->errors as $systemPath => $errorList) {
- if (isset($this->systemPaths[$systemPath])) {
- $name = Descriptions::get($this->systemPaths[$systemPath]);
- } else {
- $name = Descriptions::get('Form_' . $systemPath);
- }
- $htmlOutput .= $this->formDisplayTemplate->displayErrors($name, $errorList);
- }
- return $htmlOutput;
- }
- /**
- * Reverts erroneous fields to their default values
- *
- * @return void
- */
- public function fixErrors()
- {
- $this->validate();
- if (count($this->errors) === 0) {
- return;
- }
- $cf = $this->configFile;
- foreach (array_keys($this->errors) as $workPath) {
- if (! isset($this->systemPaths[$workPath])) {
- continue;
- }
- $canonicalPath = $this->systemPaths[$workPath];
- $cf->set($workPath, $cf->getDefault($canonicalPath));
- }
- }
- /**
- * Validates select field and casts $value to correct type
- *
- * @param string|bool $value Current value
- * @param array $allowed List of allowed values
- */
- private function validateSelect(&$value, array $allowed): bool
- {
- $valueCmp = is_bool($value)
- ? (int) $value
- : $value;
- foreach ($allowed as $vk => $v) {
- // equality comparison only if both values are numeric or not numeric
- // (allows to skip 0 == 'string' equalling to true)
- // or identity (for string-string)
- if (! (($vk == $value && ! (is_numeric($valueCmp) xor is_numeric($vk)))
- || $vk === $value)
- ) {
- continue;
- }
- // keep boolean value as boolean
- if (! is_bool($value)) {
- // phpcs:ignore Generic.PHP.ForbiddenFunctions
- settype($value, gettype($vk));
- }
- return true;
- }
- return false;
- }
- /**
- * Validates and saves form data to session
- *
- * @param array|string $forms array of form names
- * @param bool $allowPartialSave allows for partial form saving on
- * failed validation
- *
- * @return bool true on success (no errors and all saved)
- */
- public function save($forms, $allowPartialSave = true)
- {
- $result = true;
- $forms = (array) $forms;
- $values = [];
- $toSave = [];
- $isSetupScript = $GLOBALS['PMA_Config']->get('is_setup');
- if ($isSetupScript) {
- $this->loadUserprefsInfo();
- }
- $this->errors = [];
- foreach ($forms as $formName) {
- if (! isset($this->forms[$formName])) {
- continue;
- }
- /** @var Form $form */
- $form = $this->forms[$formName];
- // get current server id
- $changeIndex = $form->index === 0
- ? $this->configFile->getServerCount() + 1
- : false;
- // grab POST values
- foreach ($form->fields as $field => $systemPath) {
- $workPath = array_search($systemPath, $this->systemPaths);
- $key = $this->translatedPaths[$workPath];
- $type = (string) $form->getOptionType($field);
- // skip groups
- if ($type === 'group') {
- continue;
- }
- // ensure the value is set
- if (! isset($_POST[$key])) {
- // checkboxes aren't set by browsers if they're off
- if ($type !== 'boolean') {
- $this->errors[$form->name][] = sprintf(
- __('Missing data for %s'),
- '<i>' . Descriptions::get($systemPath) . '</i>'
- );
- $result = false;
- continue;
- }
- $_POST[$key] = false;
- }
- // user preferences allow/disallow
- if ($isSetupScript
- && isset($this->userprefsKeys[$systemPath])
- ) {
- if (isset($this->userprefsDisallow[$systemPath], $_POST[$key . '-userprefs-allow'])
- ) {
- unset($this->userprefsDisallow[$systemPath]);
- } elseif (! isset($_POST[$key . '-userprefs-allow'])) {
- $this->userprefsDisallow[$systemPath] = true;
- }
- }
- // cast variables to correct type
- switch ($type) {
- case 'double':
- $_POST[$key] = Util::requestString($_POST[$key]);
- // phpcs:ignore Generic.PHP.ForbiddenFunctions
- settype($_POST[$key], 'float');
- break;
- case 'boolean':
- case 'integer':
- if ($_POST[$key] !== '') {
- $_POST[$key] = Util::requestString($_POST[$key]);
- // phpcs:ignore Generic.PHP.ForbiddenFunctions
- settype($_POST[$key], $type);
- }
- break;
- case 'select':
- $successfullyValidated = $this->validateSelect(
- $_POST[$key],
- $form->getOptionValueList($systemPath)
- );
- if (! $successfullyValidated) {
- $this->errors[$workPath][] = __('Incorrect value!');
- $result = false;
- // "continue" for the $form->fields foreach-loop
- continue 2;
- }
- break;
- case 'string':
- case 'short_string':
- $_POST[$key] = Util::requestString($_POST[$key]);
- break;
- case 'array':
- // eliminate empty values and ensure we have an array
- $postValues = is_array($_POST[$key])
- ? $_POST[$key]
- : explode("\n", $_POST[$key]);
- $_POST[$key] = [];
- $this->fillPostArrayParameters($postValues, $key);
- break;
- }
- // now we have value with proper type
- $values[$systemPath] = $_POST[$key];
- if ($changeIndex !== false) {
- $workPath = str_replace(
- 'Servers/' . $form->index . '/',
- 'Servers/' . $changeIndex . '/',
- $workPath
- );
- }
- $toSave[$workPath] = $systemPath;
- }
- }
- // save forms
- if (! $allowPartialSave && ! empty($this->errors)) {
- // don't look for non-critical errors
- $this->validate();
- return $result;
- }
- foreach ($toSave as $workPath => $path) {
- // TrustedProxies requires changes before saving
- if ($path === 'TrustedProxies') {
- $proxies = [];
- $i = 0;
- foreach ($values[$path] as $value) {
- $matches = [];
- $match = preg_match(
- '/^(.+):(?:[ ]?)(\\w+)$/',
- $value,
- $matches
- );
- if ($match) {
- // correct 'IP: HTTP header' pair
- $ip = trim($matches[1]);
- $proxies[$ip] = trim($matches[2]);
- } else {
- // save also incorrect values
- $proxies['-' . $i] = $value;
- $i++;
- }
- }
- $values[$path] = $proxies;
- }
- $this->configFile->set($workPath, $values[$path], $path);
- }
- if ($isSetupScript) {
- $this->configFile->set(
- 'UserprefsDisallow',
- array_keys($this->userprefsDisallow)
- );
- }
- // don't look for non-critical errors
- $this->validate();
- return $result;
- }
- /**
- * Tells whether form validation failed
- *
- * @return bool
- */
- public function hasErrors()
- {
- return count($this->errors) > 0;
- }
- /**
- * Returns link to documentation
- *
- * @param string $path Path to documentation
- *
- * @return string
- */
- public function getDocLink($path)
- {
- $test = mb_substr($path, 0, 6);
- if ($test === 'Import' || $test === 'Export') {
- return '';
- }
- return MySQLDocumentation::getDocumentationLink(
- 'config',
- 'cfg_' . $this->getOptName($path),
- Sanitize::isSetup() ? '../' : './'
- );
- }
- /**
- * Changes path so it can be used in URLs
- *
- * @param string $path Path
- *
- * @return string
- */
- private function getOptName($path)
- {
- return str_replace(['Servers/1/', '/'], ['Servers/', '_'], $path);
- }
- /**
- * Fills out {@link userprefs_keys} and {@link userprefs_disallow}
- *
- * @return void
- */
- private function loadUserprefsInfo()
- {
- if ($this->userprefsKeys !== null) {
- return;
- }
- $this->userprefsKeys = array_flip(UserFormList::getFields());
- // read real config for user preferences display
- $userPrefsDisallow = $GLOBALS['PMA_Config']->get('is_setup')
- ? $this->configFile->get('UserprefsDisallow', [])
- : $GLOBALS['cfg']['UserprefsDisallow'];
- $this->userprefsDisallow = array_flip($userPrefsDisallow ?? []);
- }
- /**
- * Sets field comments and warnings based on current environment
- *
- * @param string $systemPath Path to settings
- * @param array $opts Chosen options
- *
- * @return void
- */
- private function setComments($systemPath, array &$opts)
- {
- // RecodingEngine - mark unavailable types
- if ($systemPath === 'RecodingEngine') {
- $comment = '';
- if (! function_exists('iconv')) {
- $opts['values']['iconv'] .= ' (' . __('unavailable') . ')';
- $comment = sprintf(
- __('"%s" requires %s extension'),
- 'iconv',
- 'iconv'
- );
- }
- if (! function_exists('recode_string')) {
- $opts['values']['recode'] .= ' (' . __('unavailable') . ')';
- $comment .= ($comment ? ', ' : '') . sprintf(
- __('"%s" requires %s extension'),
- 'recode',
- 'recode'
- );
- }
- /* mbstring is always there thanks to polyfill */
- $opts['comment'] = $comment;
- $opts['comment_warning'] = true;
- }
- // ZipDump, GZipDump, BZipDump - check function availability
- if ($systemPath === 'ZipDump'
- || $systemPath === 'GZipDump'
- || $systemPath === 'BZipDump'
- ) {
- $comment = '';
- $funcs = [
- 'ZipDump' => [
- 'zip_open',
- 'gzcompress',
- ],
- 'GZipDump' => [
- 'gzopen',
- 'gzencode',
- ],
- 'BZipDump' => [
- 'bzopen',
- 'bzcompress',
- ],
- ];
- if (! function_exists($funcs[$systemPath][0])) {
- $comment = sprintf(
- __(
- 'Compressed import will not work due to missing function %s.'
- ),
- $funcs[$systemPath][0]
- );
- }
- if (! function_exists($funcs[$systemPath][1])) {
- $comment .= ($comment ? '; ' : '') . sprintf(
- __(
- 'Compressed export will not work due to missing function %s.'
- ),
- $funcs[$systemPath][1]
- );
- }
- $opts['comment'] = $comment;
- $opts['comment_warning'] = true;
- }
- if ($GLOBALS['PMA_Config']->get('is_setup')) {
- return;
- }
- if ($systemPath !== 'MaxDbList' && $systemPath !== 'MaxTableList'
- && $systemPath !== 'QueryHistoryMax'
- ) {
- return;
- }
- $opts['comment'] = sprintf(
- __('maximum %s'),
- $GLOBALS['cfg'][$systemPath]
- );
- }
- /**
- * Copy items of an array to $_POST variable
- *
- * @param array $postValues List of parameters
- * @param string $key Array key
- *
- * @return void
- */
- private function fillPostArrayParameters(array $postValues, $key)
- {
- foreach ($postValues as $v) {
- $v = Util::requestString($v);
- if ($v === '') {
- continue;
- }
- $_POST[$key][] = $v;
- }
- }
- }
|