dbi = $dbi; } public function index(): void { global $err_url; $params = ['filter' => $_GET['filter'] ?? null]; $err_url = Url::getFromRoute('/'); if ($this->dbi->isSuperUser()) { $this->dbi->selectDb('mysql'); } $filterValue = ! empty($params['filter']) ? $params['filter'] : ''; $this->addScriptFiles(['server/variables.js']); $variables = []; $serverVarsResult = $this->dbi->tryQuery('SHOW SESSION VARIABLES;'); if ($serverVarsResult !== false) { $serverVarsSession = []; while ($arr = $this->dbi->fetchRow($serverVarsResult)) { $serverVarsSession[$arr[0]] = $arr[1]; } $this->dbi->freeResult($serverVarsResult); $serverVars = $this->dbi->fetchResult('SHOW GLOBAL VARIABLES;', 0, 1); // list of static (i.e. non-editable) system variables $staticVariables = ServerVariablesProvider::getImplementation()->getStaticVariables(); foreach ($serverVars as $name => $value) { $hasSessionValue = isset($serverVarsSession[$name]) && $serverVarsSession[$name] !== $value; $docLink = Generator::linkToVarDocumentation( $name, $this->dbi->isMariaDB(), str_replace('_', ' ', $name) ); [$formattedValue, $isEscaped] = $this->formatVariable($name, $value); if ($hasSessionValue) { [$sessionFormattedValue] = $this->formatVariable( $name, $serverVarsSession[$name] ); } $variables[] = [ 'name' => $name, 'is_editable' => ! in_array(strtolower($name), $staticVariables), 'doc_link' => $docLink, 'value' => $formattedValue, 'is_escaped' => $isEscaped, 'has_session_value' => $hasSessionValue, 'session_value' => $sessionFormattedValue ?? null, ]; } } $this->render('server/variables/index', [ 'variables' => $variables, 'filter_value' => $filterValue, 'is_superuser' => $this->dbi->isSuperUser(), 'is_mariadb' => $this->dbi->isMariaDB(), ]); } /** * Handle the AJAX request for a single variable value * * @param array $params Request parameters */ public function getValue(array $params): void { if (! $this->response->isAjax()) { return; } // Send with correct charset header('Content-Type: text/html; charset=UTF-8'); // Do not use double quotes inside the query to avoid a problem // when server is running in ANSI_QUOTES sql_mode $varValue = $this->dbi->fetchSingleRow( 'SHOW GLOBAL VARIABLES WHERE Variable_name=\'' . $this->dbi->escapeString($params['name']) . '\';', 'NUM' ); $json = [ 'message' => $varValue[1], ]; $variableType = ServerVariablesProvider::getImplementation()->getVariableType($params['name']); if ($variableType === 'byte') { $json['message'] = implode( ' ', Util::formatByteDown($varValue[1], 3, 3) ); } $this->response->addJSON($json); } /** * Handle the AJAX request for setting value for a single variable * * @param array $vars Request parameters */ public function setValue(array $vars): void { $params = [ 'varName' => $vars['name'], 'varValue' => $_POST['varValue'] ?? null, ]; if (! $this->response->isAjax()) { return; } $value = (string) $params['varValue']; $variableName = (string) $params['varName']; $matches = []; $variableType = ServerVariablesProvider::getImplementation()->getVariableType($variableName); if ($variableType === 'byte' && preg_match( '/^\s*(\d+(\.\d+)?)\s*(mb|kb|mib|kib|gb|gib)\s*$/i', $value, $matches )) { $exp = [ 'kb' => 1, 'kib' => 1, 'mb' => 2, 'mib' => 2, 'gb' => 3, 'gib' => 3, ]; $value = (float) $matches[1] * pow( 1024, $exp[mb_strtolower($matches[3])] ); } else { $value = $this->dbi->escapeString($value); } if (! is_numeric($value)) { $value = "'" . $value . "'"; } $json = []; if (! preg_match('/[^a-zA-Z0-9_]+/', $params['varName']) && $this->dbi->query( 'SET GLOBAL ' . $params['varName'] . ' = ' . $value ) ) { // Some values are rounded down etc. $varValue = $this->dbi->fetchSingleRow( 'SHOW GLOBAL VARIABLES WHERE Variable_name="' . $this->dbi->escapeString($params['varName']) . '";', 'NUM' ); [$formattedValue, $isHtmlFormatted] = $this->formatVariable( $params['varName'], $varValue[1] ); if ($isHtmlFormatted === false) { $json['variable'] = htmlspecialchars($formattedValue); } else { $json['variable'] = $formattedValue; } } else { $this->response->setRequestStatus(false); $json['error'] = __('Setting variable failed'); } $this->response->addJSON($json); } /** * Format Variable * * @param string $name variable name * @param int|string $value variable value * * @return array formatted string and bool if string is HTML formatted */ private function formatVariable($name, $value): array { $isHtmlFormatted = false; $formattedValue = $value; if (is_numeric($value)) { $variableType = ServerVariablesProvider::getImplementation()->getVariableType($name); if ($variableType === 'byte') { $isHtmlFormatted = true; $formattedValue = trim( $this->template->render( 'server/variables/format_variable', [ 'valueTitle' => Util::formatNumber($value, 0), 'value' => implode(' ', Util::formatByteDown($value, 3, 3)), ] ) ); } else { $formattedValue = Util::formatNumber($value, 0); } } return [ $formattedValue, $isHtmlFormatted, ]; } }