You can not select more than 25 topics
			Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
		
		
		
		
		
			
		
			
				
					
					
						
							279 lines
						
					
					
						
							9.8 KiB
						
					
					
				
			
		
		
	
	
							279 lines
						
					
					
						
							9.8 KiB
						
					
					
				| <?php | |
| 
 | |
| declare(strict_types=1); | |
| 
 | |
| namespace PhpMyAdmin\Controllers\Preferences; | |
| 
 | |
| use PhpMyAdmin\Config; | |
| use PhpMyAdmin\Config\ConfigFile; | |
| use PhpMyAdmin\Config\Forms\User\UserFormList; | |
| use PhpMyAdmin\ConfigStorage\Relation; | |
| use PhpMyAdmin\Controllers\AbstractController; | |
| use PhpMyAdmin\Core; | |
| use PhpMyAdmin\File; | |
| use PhpMyAdmin\Message; | |
| use PhpMyAdmin\ResponseRenderer; | |
| use PhpMyAdmin\Template; | |
| use PhpMyAdmin\ThemeManager; | |
| use PhpMyAdmin\UserPreferences; | |
| use PhpMyAdmin\Util; | |
| 
 | |
| use function __; | |
| use function array_merge; | |
| use function define; | |
| use function file_exists; | |
| use function is_array; | |
| use function is_uploaded_file; | |
| use function json_decode; | |
| use function json_encode; | |
| use function mb_strpos; | |
| use function mb_substr; | |
| use function parse_url; | |
| use function str_replace; | |
| use function urlencode; | |
| use function var_export; | |
| 
 | |
| use const JSON_PRETTY_PRINT; | |
| use const PHP_URL_PATH; | |
| use const UPLOAD_ERR_OK; | |
| 
 | |
| /** | |
|  * User preferences management page. | |
|  */ | |
| class ManageController extends AbstractController | |
| { | |
|     /** @var UserPreferences */ | |
|     private $userPreferences; | |
| 
 | |
|     /** @var Relation */ | |
|     private $relation; | |
| 
 | |
|     /** @var Config */ | |
|     private $config; | |
| 
 | |
|     public function __construct( | |
|         ResponseRenderer $response, | |
|         Template $template, | |
|         UserPreferences $userPreferences, | |
|         Relation $relation, | |
|         Config $config | |
|     ) { | |
|         parent::__construct($response, $template); | |
|         $this->userPreferences = $userPreferences; | |
|         $this->relation = $relation; | |
|         $this->config = $config; | |
|     } | |
| 
 | |
|     public function __invoke(): void | |
|     { | |
|         global $cf, $error, $filename, $json, $lang; | |
|         global $new_config, $return_url, $form_display, $all_ok, $params, $query, $route; | |
| 
 | |
|         $cf = new ConfigFile($this->config->baseSettings); | |
|         $this->userPreferences->pageInit($cf); | |
| 
 | |
|         $error = ''; | |
|         if (isset($_POST['submit_export'], $_POST['export_type']) && $_POST['export_type'] === 'text_file') { | |
|             // export to JSON file | |
|             $this->response->disable(); | |
|             $filename = 'phpMyAdmin-config-' . urlencode(Core::getenv('HTTP_HOST')) . '.json'; | |
|             Core::downloadHeader($filename, 'application/json'); | |
|             $settings = $this->userPreferences->load(); | |
|             echo json_encode($settings['config_data'], JSON_PRETTY_PRINT); | |
| 
 | |
|             return; | |
|         } | |
| 
 | |
|         if (isset($_POST['submit_export'], $_POST['export_type']) && $_POST['export_type'] === 'php_file') { | |
|             // export to JSON file | |
|             $this->response->disable(); | |
|             $filename = 'phpMyAdmin-config-' . urlencode(Core::getenv('HTTP_HOST')) . '.php'; | |
|             Core::downloadHeader($filename, 'application/php'); | |
|             $settings = $this->userPreferences->load(); | |
|             echo '/* ' . __('phpMyAdmin configuration snippet') . " */\n\n"; | |
|             echo '/* ' . __('Paste it to your config.inc.php') . " */\n\n"; | |
|             foreach ($settings['config_data'] as $key => $val) { | |
|                 echo '$cfg[\'' . str_replace('/', '\'][\'', $key) . '\'] = '; | |
|                 echo var_export($val, true) . ";\n"; | |
|             } | |
| 
 | |
|             return; | |
|         } | |
| 
 | |
|         if (isset($_POST['submit_get_json'])) { | |
|             $settings = $this->userPreferences->load(); | |
|             $this->response->addJSON('prefs', json_encode($settings['config_data'])); | |
|             $this->response->addJSON('mtime', $settings['mtime']); | |
| 
 | |
|             return; | |
|         } | |
| 
 | |
|         if (isset($_POST['submit_import'])) { | |
|             // load from JSON file | |
|             $json = ''; | |
|             if ( | |
|                 isset($_POST['import_type'], $_FILES['import_file']) | |
|                 && $_POST['import_type'] === 'text_file' | |
|                 && $_FILES['import_file']['error'] == UPLOAD_ERR_OK | |
|                 && is_uploaded_file($_FILES['import_file']['tmp_name']) | |
|             ) { | |
|                 $importHandle = new File($_FILES['import_file']['tmp_name']); | |
|                 $importHandle->checkUploadedFile(); | |
|                 if ($importHandle->isError()) { | |
|                     $error = $importHandle->getError(); | |
|                 } else { | |
|                     // read JSON from uploaded file | |
|                     $json = $importHandle->getRawContent(); | |
|                 } | |
|             } else { | |
|                 // read from POST value (json) | |
|                 $json = $_POST['json'] ?? null; | |
|             } | |
| 
 | |
|             // hide header message | |
|             $_SESSION['userprefs_autoload'] = true; | |
| 
 | |
|             $configuration = json_decode($json, true); | |
|             $return_url = $_POST['return_url'] ?? null; | |
|             if (! is_array($configuration)) { | |
|                 if (! isset($error)) { | |
|                     $error = __('Could not import configuration'); | |
|                 } | |
|             } else { | |
|                 // sanitize input values: treat them as though | |
|                 // they came from HTTP POST request | |
|                 $form_display = new UserFormList($cf); | |
|                 $new_config = $cf->getFlatDefaultConfig(); | |
|                 if (! empty($_POST['import_merge'])) { | |
|                     $new_config = array_merge($new_config, $cf->getConfigArray()); | |
|                 } | |
| 
 | |
|                 $new_config = array_merge($new_config, $configuration); | |
|                 $_POST_bak = $_POST; | |
|                 foreach ($new_config as $k => $v) { | |
|                     $_POST[str_replace('/', '-', (string) $k)] = $v; | |
|                 } | |
| 
 | |
|                 $cf->resetConfigData(); | |
|                 $all_ok = $form_display->process(true, false); | |
|                 $all_ok = $all_ok && ! $form_display->hasErrors(); | |
|                 $_POST = $_POST_bak; | |
| 
 | |
|                 if (! $all_ok && isset($_POST['fix_errors'])) { | |
|                     $form_display->fixErrors(); | |
|                     $all_ok = true; | |
|                 } | |
| 
 | |
|                 if (! $all_ok) { | |
|                     // mimic original form and post json in a hidden field | |
|                     $relationParameters = $this->relation->getRelationParameters(); | |
| 
 | |
|                     echo $this->template->render('preferences/header', [ | |
|                         'route' => $route, | |
|                         'is_saved' => ! empty($_GET['saved']), | |
|                         'has_config_storage' => $relationParameters->userPreferencesFeature !== null, | |
|                     ]); | |
| 
 | |
|                     echo $this->template->render('preferences/manage/error', [ | |
|                         'form_errors' => $form_display->displayErrors(), | |
|                         'json' => $json, | |
|                         'import_merge' => $_POST['import_merge'] ?? null, | |
|                         'return_url' => $return_url, | |
|                     ]); | |
| 
 | |
|                     return; | |
|                 } | |
| 
 | |
|                 // check for ThemeDefault | |
|                 $params = []; | |
|                 $tmanager = ThemeManager::getInstance(); | |
|                 if ( | |
|                     isset($configuration['ThemeDefault']) | |
|                     && $tmanager->theme->getId() != $configuration['ThemeDefault'] | |
|                     && $tmanager->checkTheme($configuration['ThemeDefault']) | |
|                 ) { | |
|                     $tmanager->setActiveTheme($configuration['ThemeDefault']); | |
|                     $tmanager->setThemeCookie(); | |
|                 } | |
| 
 | |
|                 if (isset($configuration['lang']) && $configuration['lang'] != $lang) { | |
|                     $params['lang'] = $configuration['lang']; | |
|                 } | |
| 
 | |
|                 // save settings | |
|                 $result = $this->userPreferences->save($cf->getConfigArray()); | |
|                 if ($result === true) { | |
|                     if ($return_url) { | |
|                         $query = Util::splitURLQuery($return_url); | |
|                         $return_url = parse_url($return_url, PHP_URL_PATH); | |
| 
 | |
|                         foreach ($query as $q) { | |
|                             $pos = mb_strpos($q, '='); | |
|                             $k = mb_substr($q, 0, (int) $pos); | |
|                             if ($k === 'token') { | |
|                                 continue; | |
|                             } | |
| 
 | |
|                             $params[$k] = mb_substr($q, $pos + 1); | |
|                         } | |
|                     } else { | |
|                         $return_url = 'index.php?route=/preferences/manage'; | |
|                     } | |
| 
 | |
|                     // reload config | |
|                     $this->config->loadUserPreferences(); | |
|                     $this->userPreferences->redirect($return_url ?? '', $params); | |
| 
 | |
|                     return; | |
|                 } | |
| 
 | |
|                 $error = $result; | |
|             } | |
|         } elseif (isset($_POST['submit_clear'])) { | |
|             $result = $this->userPreferences->save([]); | |
|             if ($result === true) { | |
|                 $params = []; | |
|                 $this->config->removeCookie('pma_collaction_connection'); | |
|                 $this->config->removeCookie('pma_lang'); | |
|                 $this->userPreferences->redirect('index.php?route=/preferences/manage', $params); | |
| 
 | |
|                 return; | |
|             } else { | |
|                 $error = $result; | |
|             } | |
| 
 | |
|             return; | |
|         } | |
| 
 | |
|         $this->addScriptFiles(['config.js']); | |
| 
 | |
|         $relationParameters = $this->relation->getRelationParameters(); | |
| 
 | |
|         echo $this->template->render('preferences/header', [ | |
|             'route' => $route, | |
|             'is_saved' => ! empty($_GET['saved']), | |
|             'has_config_storage' => $relationParameters->userPreferencesFeature !== null, | |
|         ]); | |
| 
 | |
|         if ($error) { | |
|             if (! $error instanceof Message) { | |
|                 $error = Message::error($error); | |
|             } | |
| 
 | |
|             $error->getDisplay(); | |
|         } | |
| 
 | |
|         echo $this->template->render('preferences/manage/main', [ | |
|             'error' => $error, | |
|             'max_upload_size' => $GLOBALS['config']->get('max_upload_size'), | |
|             'exists_setup_and_not_exists_config' => @file_exists(ROOT_PATH . 'setup/index.php') | |
|                 && ! @file_exists(CONFIG_FILE), | |
|         ]); | |
| 
 | |
|         if ($this->response->isAjax()) { | |
|             $this->response->addJSON('disableNaviSettings', true); | |
|         } else { | |
|             define('PMA_DISABLE_NAVI_SETTINGS', true); | |
|         } | |
|     } | |
| }
 | |
| 
 |