2015-01-23 15:05:37 +01:00
< ? php
2023-06-08 13:19:01 +02:00
// Pandora FMS - https://pandorafms.com
2015-01-23 15:05:37 +01:00
// ==================================================
2023-06-08 11:53:13 +02:00
// Copyright (c) 2005-2023 Pandora FMS
2023-06-08 13:19:01 +02:00
// Please see https://pandorafms.com/community/ for full contribution list
2015-01-23 15:05:37 +01:00
// This program is free software; you can redistribute it and/or
// modify it under the terms of the GNU General Public License
// as published by the Free Software Foundation; version 2
// This program is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
// GNU General Public License for more details.
2019-01-30 16:18:44 +01:00
function extension_db_status ()
{
global $config ;
$db_user = get_parameter ( 'db_user' , '' );
$db_password = get_parameter ( 'db_password' , '' );
$db_host = get_parameter ( 'db_host' , '' );
$db_name = get_parameter ( 'db_name' , '' );
$db_status_execute = ( bool ) get_parameter ( 'db_status_execute' , false );
2023-03-09 15:14:47 +01:00
ui_print_standard_header (
2019-01-30 16:18:44 +01:00
__ ( 'DB Schema check' ),
'images/extensions.png' ,
false ,
2019-04-29 14:36:02 +02:00
'db_status_tab' ,
2019-01-30 16:18:44 +01:00
true ,
2023-03-09 15:14:47 +01:00
[],
[
[
'link' => '' ,
'label' => __ ( 'Admin tools' ),
],
[
'link' => '' ,
'label' => __ ( 'Run test' ),
],
]
2019-01-30 16:18:44 +01:00
);
if ( ! is_user_admin ( $config [ 'id_user' ])) {
db_pandora_audit (
2022-01-20 10:55:23 +01:00
AUDIT_LOG_ACL_VIOLATION ,
2019-01-30 16:18:44 +01:00
'Trying to access db status'
);
include 'general/noaccess.php' ;
return ;
}
ui_print_info_message (
2024-04-12 23:43:08 +02:00
" - " .
__ ( '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.' ) .
" <br> " .
" - " .
__ ( 'At the moment the checks is for MySQL/MariaDB.' ) .
" <br> " .
" - " .
__ ( 'User must have Select, Drop, Create and References privileges.' )
2019-01-30 16:18:44 +01:00
);
2023-03-09 15:14:47 +01:00
echo " <form method='post' class='max_floating_element_size'> " ;
2019-01-30 16:18:44 +01:00
echo '<fieldset>' ;
echo '<legend>' . __ ( 'DB settings' ) . '</legend>' ;
$table = new stdClass ();
$table -> data = [];
$row = [];
2023-03-09 15:14:47 +01:00
$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'
)
);
2019-01-30 16:18:44 +01:00
$table -> data [] = $row ;
$row = [];
2023-03-09 15:14:47 +01:00
$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'
)
);
2019-01-30 16:18:44 +01:00
$table -> data [] = $row ;
html_print_table ( $table );
echo '</fieldset>' ;
2023-03-09 15:14:47 +01:00
html_print_action_buttons (
html_print_submit_button (
__ ( 'Execute Test' ),
'submit' ,
false ,
[ 'icon' => 'cog' ],
true
)
2022-11-04 13:00:00 +01:00
);
2019-01-30 16:18:44 +01:00
2022-11-04 13:00:00 +01:00
html_print_input_hidden ( 'db_status_execute' , 1 );
2019-01-30 16:18:44 +01:00
echo '</form>' ;
if ( $db_status_execute ) {
extension_db_status_execute_checks (
$db_user ,
$db_password ,
$db_host ,
$db_name
);
}
2015-01-23 15:05:37 +01:00
}
2019-01-30 16:18:44 +01:00
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 ];
2024-04-12 23:43:08 +02:00
if ( ! $db_host ) {
ui_print_error_message ( __ ( 'A host must be provided' ));
return ;
}
if ( ! $db_name ) {
ui_print_error_message ( __ ( 'A DB name must be provided' ));
return ;
}
2024-04-10 16:30:25 +02:00
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 ;
2019-01-30 16:18:44 +01:00
}
if ( ! $connection_test ) {
2024-04-10 16:30:25 +02:00
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' ];
2019-01-30 16:18:44 +01:00
} else {
2024-04-10 16:30:25 +02:00
$exist_db = mysql_fetch_assoc ( mysqli_query ( $connection_test , $query ))[ 'result' ];
2019-01-30 16:18:44 +01:00
}
2024-04-10 16:30:25 +02:00
} 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 ;
}
2019-01-30 16:18:44 +01:00
2024-04-12 23:43:08 +02:00
if ( check_drop_privileges ( $connection_test ) == 0 ) {
return ;
}
2024-04-10 16:30:25 +02:00
try {
if ( $config [ 'mysqli' ]) {
$create_db = mysqli_query ( $connection_test , " CREATE DATABASE ` $db_name ` " );
2019-01-30 16:18:44 +01:00
} else {
2024-04-10 16:30:25 +02:00
$create_db = mysql_query ( " CREATE DATABASE ` $db_name ` " );
2019-01-30 16:18:44 +01:00
}
2024-04-10 16:30:25 +02:00
} catch ( Exception $e ) {
$connection_test = false ;
}
if ( ! $create_db ) {
ui_print_error_message ( __ ( 'Unsuccessful created the testing DB' ));
return ;
}
2024-04-12 23:43:08 +02:00
if ( check_ref_privileges ( $connection_test ) == 0 ) {
drop_database ( $connection_test , $db_name );
ui_print_error_message ( __ ( " Unable to <b>create references</b> 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 <b>explain</b> 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 ;
2024-04-10 16:30:25 +02:00
}
$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' ]
);
2024-04-12 23:43:08 +02:00
drop_database ( $connection_test , $db_name );
2015-01-23 15:05:37 +01:00
}
2019-01-30 16:18:44 +01:00
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 (
2019-02-28 16:44:37 +01:00
empty ( $diff_tables ),
2019-01-30 16:18:44 +01:00
__ ( '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 );
$tables_test = [];
while ( $row = mysql_fetch_array ( $result )) {
ui_print_info_message (
__ ( 'You can execute this SQL query for to fix.' ) . '<br />' . '<pre>' . $row [ 1 ] . '</pre>'
);
}
mysqli_free_result ( $result );
} else {
mysql_select_db ( $db_name_test , $connection_test );
$result = mysql_query ( 'SHOW CREATE TABLE ' . $table , $connection_test );
$tables_test = [];
while ( $row = mysql_fetch_array ( $result )) {
ui_print_info_message (
__ ( 'You can execute this SQL query for to fix.' ) . '<br />' . '<pre>' . $row [ 1 ] . '</pre>'
);
}
mysql_free_result ( $result );
}
}
}
// --------------- 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.' ) . '<br />' . '<pre>' . 'ALTER TABLE ' . $table . ' ADD COLUMN ' . $name_field . ' text;' . '</pre>'
);
} 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' ] . '<br>' ;
}
if ( $diff [ 'null' ]) {
$error_message .= " Unsuccessful the field $name_field in the table $table must be null: ( " . $diff [ 'null' ] . ').<br>' ;
}
if ( $diff [ 'default' ]) {
$error_message .= " Unsuccessful the field $name_field in the table $table must be set " . $diff [ 'default' ] . ' as default value.<br>' ;
}
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.<br> " ;
$info_message .= '<br><br>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.<br> " ;
$info_message .= '<br><br>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' )
);
}
2015-01-23 15:05:37 +01:00
}
2019-01-30 16:18:44 +01:00
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 ) {
2024-04-12 23:43:08 +02:00
$result = mysqli_query ( $connection , $query );
2019-01-30 16:18:44 +01:00
} else {
2024-04-12 23:43:08 +02:00
$result = mysql_query ( $query , $connection );
}
if ( ! $result ) {
echo mysqli_error ( $connection );
// Uncomment for debug
echo " <i><br> $query <br></i> " ;
return 0 ;
2019-01-30 16:18:44 +01:00
}
$query = '' ;
}
}
}
return 1 ;
} else {
return 0 ;
}
2015-01-23 15:05:37 +01:00
}
2019-01-30 16:18:44 +01:00
2024-04-12 23:43:08 +02:00
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 <b>create databases</b> 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 <b>drop databases</b> with the provided user please check its privileges. " ) .
" <br> " .
__ ( " 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 );
}
}
2015-01-23 15:05:37 +01:00
extensions_add_godmode_function ( 'extension_db_status' );
2019-01-30 16:18:44 +01:00
extensions_add_godmode_menu_option ( __ ( 'DB Schema check' ), 'DM' , 'gextensions' , null , 'v1r1' , 'gdbman' );