'', 'label' => __('Admin tools'), ], [ 'link' => '', 'label' => __('Run test'), ], ] ); if (!is_user_admin($config['id_user'])) { db_pandora_audit( AUDIT_LOG_ACL_VIOLATION, 'Trying to access db status' ); include 'general/noaccess.php'; return; } ui_print_info_message( ' - '.__('This extension checks the DB is correct. Because sometimes the old DB from a migration has not some fields in the tables or the data is changed.').'
'.' - '.__('At the moment the checks is for MySQL/MariaDB.').'
'.' - '.__('User must have Select, Drop, Create and References privileges.') ); echo "
"; echo '
'; echo ''.__('DB settings').''; $table = new stdClass(); $table->data = []; $row = []; $row[] = html_print_label_input_block( __('DB User with privileges'), html_print_input_text( 'db_user', $db_user, '', 50, 255, true, false, false, '', 'w100p mrgn_top_10px' ) ); $row[] = html_print_label_input_block( __('DB Password for this user'), html_print_input_password( 'db_password', $db_password, '', 50, 255, true, false, false, 'w100p mrgn_top_10px' ) ); $table->data[] = $row; $row = []; $row[] = html_print_label_input_block( __('DB Hostname'), html_print_input_text( 'db_host', $db_host, '', 50, 255, true, false, false, '', 'w100p mrgn_top_10px' ) ); $row[] = html_print_label_input_block( __('DB Name (temporal for testing)'), html_print_input_text( 'db_name', $db_name, '', 50, 255, true, false, false, '', 'w100p mrgn_top_10px' ) ); $table->data[] = $row; html_print_table($table); echo '
'; html_print_action_buttons( html_print_submit_button( __('Execute Test'), 'submit', false, [ 'icon' => 'cog' ], true ) ); html_print_input_hidden('db_status_execute', 1); echo '
'; if ($db_status_execute) { extension_db_status_execute_checks( $db_user, $db_password, $db_host, $db_name ); } } function extension_db_status_execute_checks($db_user, $db_password, $db_host, $db_name) { global $config; $connection_system = $config['dbconnection']; // Avoid SQL injection $db_name = io_safe_output($db_name); $db_name = str_replace(';', ' ', $db_name); $db_name = explode(' ', $db_name); $db_name = $db_name[0]; if (!$db_host) { ui_print_error_message(__('Host must be provided')); return; } if (!$db_name) { ui_print_error_message(__('DB name must be provided')); return; } try { if ($config['mysqli']) { $connection_test = mysqli_connect($db_host, $db_user, $db_password); } else { $connection_test = mysql_connect($db_host, $db_user, $db_password); } } catch (Exception $e) { $connection_test = false; } if (!$connection_test) { ui_print_error_message(__('Unsuccessful connected to the DB')); return; } try { $query = "SELECT IF(EXISTS(SELECT 1 FROM information_schema.SCHEMATA WHERE schema_name = '$db_name'), 'true', 'false') AS result"; if ($config['mysqli']) { $exist_db = mysqli_fetch_assoc(mysqli_query($connection_test, $query))['result']; } else { $exist_db = mysql_fetch_assoc(mysqli_query($connection_test, $query))['result']; } } catch (Exception $e) { ui_print_error_message(__("There was a problem during verification of the existence of the `$db_name` table")); return; } if ($exist_db == 'true') { ui_print_error_message(__("The testing DB `$db_name` already exists")); return; } if (check_drop_privileges($connection_test) == 0) { return; } try { if ($config['mysqli']) { $create_db = mysqli_query($connection_test, "CREATE DATABASE `$db_name`"); } else { $create_db = mysql_query("CREATE DATABASE `$db_name`"); } } catch (Exception $e) { $connection_test = false; } if (!$create_db) { ui_print_error_message(__('Unsuccessful created the testing DB')); return; } if (check_ref_privileges($connection_test) == 0) { drop_database($connection_test, $db_name); ui_print_error_message(__('Unable to create references with the provided user please check its privileges')); return; } if (check_explain_privileges($connection_test) == 0) { drop_database($connection_test, $db_name); ui_print_error_message(__('Unable to explain with the provided user please check its privileges')); return; } try { if ($config['mysqli'] === true) { mysqli_select_db($connection_test, $db_name); } else { mysql_select_db($db_name, $connection_test); } } catch (Exception $e) { drop_database($connection_test, $db_name); ui_print_error_message(__('There was an error selecting the DB')); return; } $install_tables = extension_db_status_execute_sql_file( $config['homedir'].'/pandoradb.sql', $connection_test ); if (!$install_tables) { ui_print_error_message(__('Unsuccessful installed tables into the testing DB')); return; } extension_db_check_tables_differences( $connection_test, $connection_system, $db_name, $config['dbname'] ); drop_database($connection_test, $db_name); } function extension_db_check_tables_differences( $connection_test, $connection_system, $db_name_test, $db_name_system ) { global $config; // --------- Check the tables -------------------------------------- if ($config['mysqli'] === true) { mysqli_select_db($connection_test, $db_name_test); $result = mysqli_query($connection_test, 'SHOW TABLES'); } else { mysql_select_db($db_name_test, $connection_test); $result = mysql_query('SHOW TABLES', $connection_test); } $tables_test = []; if ($config['mysqli'] === true) { while ($row = mysqli_fetch_array($result)) { $tables_test[] = $row[0]; } mysqli_free_result($result); mysqli_select_db($connection_test, $db_name_system); $result = mysqli_query($connection_test, 'SHOW TABLES'); } else { while ($row = mysql_fetch_array($result)) { $tables_test[] = $row[0]; } mysql_free_result($result); mysql_select_db($db_name_system, $connection_test); $result = mysql_query('SHOW TABLES', $connection_test); } $tables_system = []; if ($config['mysqli'] === true) { while ($row = mysqli_fetch_array($result)) { $tables_system[] = $row[0]; } mysqli_free_result($result); } else { while ($row = mysql_fetch_array($result)) { $tables_system[] = $row[0]; } mysql_free_result($result); } $diff_tables = array_diff($tables_test, $tables_system); ui_print_result_message( empty($diff_tables), __('Success! %s DB contains all tables', get_product_name()), __( '%s DB could not retrieve all tables. The missing tables are (%s)', get_product_name(), implode(', ', $diff_tables) ) ); if (!empty($diff_tables)) { foreach ($diff_tables as $table) { if ($config['mysqli'] === true) { mysqli_select_db($connection_test, $db_name_test); $result = mysqli_query($connection_test, 'SHOW CREATE TABLE '.$table); $create_query = mysqli_fetch_assoc($result)['Create Table']; mysqli_free_result($result); ui_print_info_message( __('You can execute this SQL query for to fix.').'
'.'
'.$create_query.'
' ); } else { mysql_select_db($db_name_test, $connection_test); $result = mysql_query('SHOW CREATE TABLE '.$table, $connection_test); $create_query = mysqli_fetch_assoc($result)['Create Table']; mysql_free_result($result); ui_print_info_message( __('You can execute this SQL query for to fix.').'
'.'
'.$create_query.'
' ); } } } // --------------- Check the fields ------------------------------- $correct_fields = true; foreach ($tables_system as $table) { if ($config['mysqli'] === true) { mysqli_select_db($connection_test, $db_name_test); $result = mysqli_query($connection_test, 'EXPLAIN '.$table); } else { mysql_select_db($db_name_test, $connection_test); $result = mysql_query('EXPLAIN '.$table, $connection_test); } $fields_test = []; if (!empty($result)) { if ($config['mysqli'] === true) { while ($row = mysqli_fetch_array($result)) { $fields_test[$row[0]] = [ 'field ' => $row[0], 'type' => $row[1], 'null' => $row[2], 'key' => $row[3], 'default' => $row[4], 'extra' => $row[5], ]; } mysqli_free_result($result); mysqli_select_db($connection_test, $db_name_system); } else { while ($row = mysql_fetch_array($result)) { $fields_test[$row[0]] = [ 'field ' => $row[0], 'type' => $row[1], 'null' => $row[2], 'key' => $row[3], 'default' => $row[4], 'extra' => $row[5], ]; } mysql_free_result($result); mysql_select_db($db_name_system, $connection_test); } } if ($config['mysqli'] === true) { $result = mysqli_query($connection_test, 'EXPLAIN '.$table); } else { $result = mysql_query('EXPLAIN '.$table, $connection_test); } $fields_system = []; if (!empty($result)) { if ($config['mysqli'] === true) { while ($row = mysqli_fetch_array($result)) { $fields_system[$row[0]] = [ 'field ' => $row[0], 'type' => $row[1], 'null' => $row[2], 'key' => $row[3], 'default' => $row[4], 'extra' => $row[5], ]; } mysqli_free_result($result); } else { while ($row = mysql_fetch_array($result)) { $fields_system[$row[0]] = [ 'field ' => $row[0], 'type' => $row[1], 'null' => $row[2], 'key' => $row[3], 'default' => $row[4], 'extra' => $row[5], ]; } mysql_free_result($result); } } foreach ($fields_test as $name_field => $field_test) { if (!isset($fields_system[$name_field])) { $correct_fields = false; ui_print_error_message( __( 'Unsuccessful the table %s has not the field %s', $table, $name_field ) ); ui_print_info_message( __('You can execute this SQL query for to fix.').'
'.'
'.'ALTER TABLE '.$table.' ADD COLUMN '.$name_field.' text;'.'
' ); } else { $correct_fields = false; $field_system = $fields_system[$name_field]; $diff = array_diff($field_test, $field_system); if (!empty($diff)) { $info_message = ''; $error_message = ''; if ($diff['type']) { $error_message .= 'Unsuccessful the field '.$name_field.' in the table '.$table.' must be set the type with '.$diff['type'].'
'; } if ($diff['null']) { $error_message .= "Unsuccessful the field $name_field in the table $table must be null: (".$diff['null'].').
'; } if ($diff['default']) { $error_message .= "Unsuccessful the field $name_field in the table $table must be set ".$diff['default'].' as default value.
'; } if ($field_test['null'] == 'YES' || !isset($field_test['null']) || $field_test['null'] == '') { $null_defect = ' NULL'; } else { $null_defect = ' NOT NULL'; } if (!isset($field_test['default']) || $field_test['default'] == '') { $default_value = ''; } else { $default_value = ' DEFAULT '.$field_test['default']; } if ($diff['type'] || $diff['null'] || $diff['default']) { $info_message .= 'ALTER TABLE '.$table.' MODIFY COLUMN '.$name_field.' '.$field_test['type'].$null_defect.$default_value.';'; } if ($diff['key']) { $error_message .= "Unsuccessful the field $name_field in the table $table must be set the key as defined in the SQL file.
"; $info_message .= '

Please check the SQL file for to know the kind of key needed.'; } if ($diff['extra']) { $error_message .= "Unsuccessful the field $name_field in the table $table must be set as defined in the SQL file.
"; $info_message .= '

Please check the SQL file for to know the kind of extra config needed.'; } ui_print_error_message(__($error_message)); ui_print_info_message(__($info_message)); } } } } if ($correct_fields) { ui_print_success_message( __('Successful all the tables have the correct fields') ); } } function extension_db_status_execute_sql_file($url, $connection) { global $config; if (file_exists($url)) { $file_content = file($url); $query = ''; foreach ($file_content as $sql_line) { if (trim($sql_line) != '' && strpos($sql_line, '--') === false) { $query .= $sql_line; if (preg_match("/;[\040]*\$/", $sql_line)) { if ($config['mysqli'] === true) { $result = mysqli_query($connection, $query); } else { $result = mysql_query($query, $connection); } if (!$result) { echo mysqli_error($connection); // Uncomment for debug echo "
$query
"; return 0; } $query = ''; } } } return 1; } else { return 0; } } function check_explain_privileges($connection) { global $config; $has_privileges = 1; $explain_check = 'EXPLAIN tb1'; $create_tb1 = 'CREATE TABLE tb1 ( id INT AUTO_INCREMENT PRIMARY KEY )'; drop_database($connection, 'pandora_tmp_privilege_check'); try { if ($config['mysqli']) { mysqli_query($connection, 'CREATE DATABASE `pandora_tmp_privilege_check`'); } else { mysql_query('CREATE DATABASE `pandora_tmp_privilege_check`', $connection); } } catch (Exception $e) { ui_print_error_message(__('There was an error creating the DB during reference check')); return 0; } try { if ($config['mysqli'] === true) { mysqli_select_db($connection, 'pandora_tmp_privilege_check'); } else { mysql_select_db('reference_check', $connection); } } catch (Exception $e) { ui_print_error_message(__('There was an error selecting the DB during reference check')); return 0; } try { if ($config['mysqli'] === true) { $result = mysqli_query($connection, $create_tb1); } else { $result = mysql_query($create_tb1, $connection); } if (!$result) { throw new Exception('Error on explain check: '.$connection->error); } if ($config['mysqli'] === true) { $result = mysqli_query($connection, $explain_check); } else { $result = mysql_query($explain_check, $connection); } if (!$result) { throw new Exception('Error on explain check: '.$connection->error); } } catch (Exception $e) { $has_privileges = 0; } finally { drop_database($connection, 'pandora_tmp_privilege_check'); return $has_privileges; } } function check_drop_privileges($connection) { global $config; $has_privileges = 1; try { if ($config['mysqli']) { $create_db = mysqli_query($connection, 'CREATE DATABASE IF NOT EXISTS`pandora_tmp_privilege_check`'); } else { $create_db = mysql_query('CREATE DATABASE IF NOT EXISTS `pandora_tmp_privilege_check`', $connection); } } catch (Exception $e) { $error_message = $e->getMessage(); } if (!$create_db) { if (stripos($error_message, 'access denied for user') !== false) { preg_match("/'.+?'\@'.+?'/", $error_message, $error_user); $error_user = $error_user[0]; ui_print_error_message(__('Unable to create databases with the provided user please check its privileges')); return 0; } ui_print_error_message(__('There was an error creating the DB during drop check')); return 0; } try { if ($config['mysqli'] === true) { mysqli_select_db($connection, 'pandora_tmp_privilege_check'); } else { mysql_select_db('reference_check', $connection); } } catch (Exception $e) { ui_print_error_message(__('There was an error selecting the DB during drop check')); return 0; } try { drop_database($connection, 'pandora_tmp_privilege_check'); } catch (Exception $e) { $has_privileges = 0; ui_print_error_message( __('Unable to drop databases with the provided user please check its privileges.').'
'.__('Test databases may have been left over due to lack of drop privileges.') ); } finally { return $has_privileges; } } function check_ref_privileges($connection) { global $config; $has_privileges = 1; drop_database($connection, 'pandora_tmp_privilege_check'); $create_tb1 = 'CREATE TABLE tb1 ( id INT AUTO_INCREMENT PRIMARY KEY )'; $create_tb2 = 'CREATE TABLE tb2 ( id INT AUTO_INCREMENT PRIMARY KEY, id_tb1 INT, FOREIGN KEY (id_tb1) REFERENCES tb2(id) )'; try { if ($config['mysqli']) { mysqli_query($connection, 'CREATE DATABASE `pandora_tmp_privilege_check`'); } else { mysql_query('CREATE DATABASE `pandora_tmp_privilege_check`', $connection); } } catch (Exception $e) { ui_print_error_message(__('There was an error creating the DB during reference check')); return 0; } try { if ($config['mysqli'] === true) { mysqli_select_db($connection, 'pandora_tmp_privilege_check'); } else { mysql_select_db('reference_check', $connection); } } catch (Exception $e) { ui_print_error_message(__('There was an error selecting the DB during reference check')); return 0; } try { if ($config['mysqli'] === true) { $result = mysqli_query($connection, $create_tb1); } else { $result = mysql_query($create_tb1, $connection); } if (!$result) { throw new Exception('Error on reference check: '.$connection->error); } if ($config['mysqli'] === true) { $result = mysqli_query($connection, $create_tb2); } else { $result = mysql_query($create_tb2, $connection); } if (!$result) { throw new Exception('Error on reference check: '.$connection->error); } } catch (Exception $e) { $has_privileges = 0; } finally { drop_database($connection, 'pandora_tmp_privilege_check'); return $has_privileges; } } function drop_database($connection, $database) { global $config; if ($config['mysqli'] === true) { mysqli_query($connection, "DROP DATABASE IF EXISTS `$database`"); } else { mysql_query("DROP DATABASE IF EXISTS `$database`", $connection); } } extensions_add_godmode_function('extension_db_status'); extensions_add_godmode_menu_option(__('DB Schema check'), 'DM', 'gextensions', null, 'v1r1', 'gdbman');