2019-10-11 14:55:20 +02:00
< ? php
/**
* Extension to self monitor Pandora FMS Console
*
* @ category Console Class
* @ package Pandora FMS
2019-10-17 15:17:28 +02:00
* @ subpackage Diagnostics
2019-10-11 14:55:20 +02:00
* @ version 1.0 . 0
* @ license See below
*
* ______ ___ _______ _______ ________
* | __ \ .-----.--.--.--| |.-----.----.-----. | ___ | | | __ |
* | __ /| _ | | _ || _ | _ | _ | | ___ | | __ |
* | ___ | | ___ . _ | __ | __ | _____ || _____ | __ | | ___ . _ | | ___ | | __ | _ | __ | _______ |
*
* ============================================================================
2020-11-27 13:52:35 +01:00
* Copyright ( c ) 2005 - 2021 Artica Soluciones Tecnologicas
2019-10-11 14:55:20 +02:00
* Please see http :// pandorafms . org for full contribution list
* 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 for 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 .
* ============================================================================
*/
global $config ;
require_once $config [ 'homedir' ] . '/include/functions_db.php' ;
require_once $config [ 'homedir' ] . '/include/functions_io.php' ;
2019-10-21 18:11:04 +02:00
require_once $config [ 'homedir' ] . '/godmode/wizards/Wizard.main.php' ;
2019-10-11 14:55:20 +02:00
/**
* Base class Diagnostics .
*/
2019-10-21 18:11:04 +02:00
class Diagnostics extends Wizard
2019-10-11 14:55:20 +02:00
{
2022-01-26 18:40:45 +01:00
const INNODB_FLUSH_LOG_AT_TRX_COMMIT = 2 ;
2022-01-26 17:34:56 +01:00
2019-10-11 14:55:20 +02:00
/**
2019-10-17 15:17:28 +02:00
* Ajax controller page .
2019-10-11 14:55:20 +02:00
*
2019-10-17 15:17:28 +02:00
* @ var string
2019-10-11 14:55:20 +02:00
*/
2019-10-17 15:17:28 +02:00
public $ajaxController ;
2019-10-11 14:55:20 +02:00
/**
2019-10-21 18:11:04 +02:00
* Print Html or Pdf view .
2019-10-17 15:17:28 +02:00
*
2019-10-21 18:11:04 +02:00
* @ var boolean
*/
public $pdf ;
/**
* Constructor .
2019-10-11 14:55:20 +02:00
*
2019-10-21 18:11:04 +02:00
* @ param string $page Page .
* @ param boolean $pdf PDF View .
2019-10-11 14:55:20 +02:00
*/
2019-10-23 19:50:57 +02:00
public function __construct (
string $page = 'tools/diagnostics' ,
bool $pdf = false
) {
2019-10-11 14:55:20 +02:00
global $config ;
2019-10-17 15:17:28 +02:00
// Check access.
check_login ();
2019-10-11 14:55:20 +02:00
2019-11-21 18:21:46 +01:00
$this -> url = ui_get_full_url (
'index.php?sec=gextensions&sec2=tools/diagnostics'
);
2019-10-17 15:17:28 +02:00
$this -> ajaxController = $page ;
2019-10-21 18:11:04 +02:00
$this -> pdf = $pdf ;
2019-11-13 15:52:09 +01:00
$this -> product_name = io_safe_output ( get_product_name ());
2019-10-17 15:17:28 +02:00
}
2019-10-11 14:55:20 +02:00
2019-10-17 15:17:28 +02:00
/**
* Allowed methods to be called using AJAX request .
*
* @ var array
*/
public $AJAXMethods = [
'getStatusInfo' ,
'getPHPSetup' ,
'getDatabaseSizeStats' ,
'getDatabaseHealthStatus' ,
'getDatabaseStatusInfo' ,
'getSystemInfo' ,
'getMySQLPerformanceMetrics' ,
'getTablesFragmentation' ,
'getPandoraFMSLogsDates' ,
'getLicenceInformation' ,
'getAttachmentFolder' ,
'getInfoTagenteDatos' ,
'getServerThreads' ,
2019-10-23 19:50:57 +02:00
'getShowEngine' ,
2019-10-17 15:17:28 +02:00
'datatablesDraw' ,
'getChartAjax' ,
2019-10-21 18:11:04 +02:00
'formFeedback' ,
'createdScheduleFeedbackTask' ,
2023-02-07 10:30:22 +01:00
'getSystemDate' ,
2019-10-17 15:17:28 +02:00
];
2019-10-11 14:55:20 +02:00
2019-10-17 15:17:28 +02:00
/**
* Checks if target method is available to be called using AJAX .
*
* @ param string $method Target method .
*
* @ return boolean True allowed , false not .
*/
public function ajaxMethod ( string $method ) : bool
{
return in_array ( $method , $this -> AJAXMethods );
}
2019-10-11 14:55:20 +02:00
2019-10-17 15:17:28 +02:00
/**
* Show view diagnostics .
*
* @ return void
*/
public function run ()
{
global $config ;
2019-10-14 17:21:38 +02:00
2019-11-21 18:21:46 +01:00
if ( $this -> pdf === true ) {
$this -> exportPDF ();
exit ;
}
2019-10-23 19:50:57 +02:00
ui_require_css_file ( 'diagnostics' );
2019-10-14 17:21:38 +02:00
2019-11-21 18:21:46 +01:00
$pdf_url = $this -> url . '&pdf=true' ;
$pdf_img = html_print_image (
'images/pdf.png' ,
true ,
2021-03-11 15:40:23 +01:00
[
2022-12-02 15:40:47 +01:00
'title' => __ ( 'Export to PDF' ),
'class' => 'invert_filter' ,
'onclick' => 'blockResubmit($(this))' ,
2021-03-11 15:40:23 +01:00
]
2019-11-21 18:21:46 +01:00
);
$header_buttons = [
'csv' => [
'active' => false ,
'text' => '<a target="_new" href="' . $pdf_url . '">' . $pdf_img . '</a>' ,
],
];
2019-10-17 15:17:28 +02:00
// Header.
ui_print_page_header (
2019-11-13 15:52:09 +01:00
__ ( '%s Diagnostic tool' , $this -> product_name ),
2019-10-17 15:17:28 +02:00
'images/gm_massive_operations.png' ,
false ,
2020-05-29 11:37:06 +02:00
'' ,
2019-11-21 18:21:46 +01:00
true ,
$header_buttons
2019-10-17 15:17:28 +02:00
);
2019-10-14 17:21:38 +02:00
2019-10-21 18:11:04 +02:00
// Print all Methods Diagnostic Info.
echo $this -> printMethodsDiagnostigsInfo ();
// Print all charts Monitoring.
echo $this -> printCharts ();
echo '<div class="footer-self-monitoring">' ;
echo $this -> checkPandoraDB ();
echo '</div>' ;
}
/**
* Print Methods :
* Info status pandoraFms .
* PHP setup .
* Database size stats .
* Database health status .
* Database status info .
* System Info .
* MySQL Performance metrics .
* Tables fragmentation in the Pandora FMS database .
* Pandora FMS logs dates .
* Pandora FMS Licence Information .
* Status of the attachment folder .
* Information from the tagente_datos table .
* Pandora FMS server threads .
*
* @ return string Html .
*/
public function printMethodsDiagnostigsInfo () : string
{
$infoMethods = [
'getStatusInfo' ,
'getPHPSetup' ,
'getDatabaseSizeStats' ,
'getDatabaseHealthStatus' ,
'getDatabaseStatusInfo' ,
'getSystemInfo' ,
'getMySQLPerformanceMetrics' ,
'getTablesFragmentation' ,
'getPandoraFMSLogsDates' ,
'getLicenceInformation' ,
'getAttachmentFolder' ,
'getInfoTagenteDatos' ,
'getServerThreads' ,
2023-02-07 10:30:22 +01:00
'getSystemDate' ,
2019-10-21 18:11:04 +02:00
];
2019-10-14 17:21:38 +02:00
2019-10-23 19:50:57 +02:00
if ( $this -> pdf === true ) {
$infoMethods [] = 'getShowEngine' ;
}
2019-10-21 18:11:04 +02:00
$return = '' ;
foreach ( $infoMethods as $key => $method ) {
2019-10-17 15:17:28 +02:00
switch ( $method ) {
case 'getStatusInfo' :
2019-11-13 15:52:09 +01:00
$title = __ ( 'Info status %s' , $this -> product_name );
2019-10-17 15:17:28 +02:00
break ;
case 'getPHPSetup' :
$title = __ ( 'PHP setup' );
break ;
case 'getDatabaseSizeStats' :
$title = __ ( 'Database size stats' );
break ;
case 'getDatabaseHealthStatus' :
$title = __ ( 'Database health status' );
break ;
case 'getDatabaseStatusInfo' :
$title = __ ( 'Database status info' );
break ;
case 'getSystemInfo' :
$title = __ ( 'System Info' );
break ;
case 'getMySQLPerformanceMetrics' :
$title = __ ( 'MySQL Performance metrics' );
break ;
case 'getTablesFragmentation' :
$title = __ (
2019-11-13 15:52:09 +01:00
'Tables fragmentation in the %s database' ,
$this -> product_name
2019-10-17 15:17:28 +02:00
);
break ;
case 'getPandoraFMSLogsDates' :
2019-11-13 15:52:09 +01:00
$title = __ ( '%s logs dates' , $this -> product_name );
2019-10-17 15:17:28 +02:00
break ;
case 'getLicenceInformation' :
2019-11-13 15:52:09 +01:00
$title = __ ( '%s Licence Information' , $this -> product_name );
2019-10-17 15:17:28 +02:00
break ;
case 'getAttachmentFolder' :
$title = __ ( 'Status of the attachment folder' );
break ;
case 'getInfoTagenteDatos' :
$title = __ ( 'Information from the tagente_datos table' );
break ;
case 'getServerThreads' :
2019-11-13 15:52:09 +01:00
$title = __ ( '%s server threads' , $this -> product_name );
2019-10-17 15:17:28 +02:00
break ;
2019-10-23 19:50:57 +02:00
case 'getShowEngine' :
$title = __ ( 'SQL show engine innodb status' );
break ;
2023-02-07 10:30:22 +01:00
case 'getSystemDate' :
$title = __ ( 'Date system' );
break ;
2019-10-17 15:17:28 +02:00
default :
// Not possible.
$title = '' ;
break ;
}
2019-10-14 17:21:38 +02:00
2021-03-11 15:40:23 +01:00
$return .= '<div class="mrgn_btn_30px">' ;
2019-10-21 18:11:04 +02:00
$return .= $this -> printData ( $method , $title );
$return .= '</div>' ;
2019-10-17 15:17:28 +02:00
}
2019-10-14 17:21:38 +02:00
2021-04-21 11:07:20 +02:00
if ( $this -> pdf === true ) {
return $return ;
} else {
return false ;
}
2019-10-21 18:11:04 +02:00
}
/**
* Print table graps :
* Graph of the Agents Unknown module .
* Graph of the Database Maintenance module .
* Graph of the Free Disk Spool Dir module .
* Graph of the Free RAM module .
* Graph of the Queued Modules module .
* Graph of the Status module .
* Graph of the System Load AVG module .
* Graph of the Execution Time module .
*
* @ return string
*/
public function printCharts ()
{
2019-10-14 17:21:38 +02:00
/*
2019-10-17 15:17:28 +02:00
* Agent id with name Master Server .
2019-10-14 17:21:38 +02:00
*/
2019-10-17 15:17:28 +02:00
$agentIdMasterServer = $this -> getAgentIdMasterServer ();
2019-10-14 17:21:38 +02:00
2019-10-21 18:11:04 +02:00
$result = '' ;
2019-10-17 15:17:28 +02:00
if ( $agentIdMasterServer !== 0 ) {
$agentMonitoring = [
'chartAgentsUnknown' => [
'title' => __ (
'Graph of the Agents Unknown module.'
),
'nameModule' => 'Agents_Unknown' ,
'idAgent' => $agentIdMasterServer ,
],
'chartDatabaseMain' => [
'title' => __ (
'Graph of the Database Maintenance module.'
),
'nameModule' => 'Database Maintenance' ,
'idAgent' => $agentIdMasterServer ,
],
'chartFreeDiskSpoolDir' => [
'title' => __ (
'Graph of the Free Disk Spool Dir module.'
),
'nameModule' => 'FreeDisk_SpoolDir' ,
'idAgent' => $agentIdMasterServer ,
],
'chartFreeRAM' => [
'title' => __ ( 'Graph of the Free RAM module.' ),
'nameModule' => 'Free_RAM' ,
'idAgent' => $agentIdMasterServer ,
],
'chartQueuedModules' => [
'title' => __ (
'Graph of the Queued Modules module.'
),
'nameModule' => 'Queued_Modules' ,
'idAgent' => $agentIdMasterServer ,
],
'chartStatus' => [
'title' => __ ( 'Graph of the Status module.' ),
'nameModule' => 'Status' ,
'idAgent' => $agentIdMasterServer ,
],
'chartSystemLoadAVG' => [
'title' => __ (
'Graph of the System Load AVG module.'
),
'nameModule' => 'System_Load_AVG' ,
'idAgent' => $agentIdMasterServer ,
],
'chartExecutionTime' => [
'title' => __ (
'Graph of the Execution Time module.'
),
'nameModule' => 'Execution_time' ,
'idAgent' => $agentIdMasterServer ,
],
];
2019-10-14 17:21:38 +02:00
2019-10-21 18:11:04 +02:00
$return .= '<div class="title-self-monitoring">' ;
2019-10-23 19:50:57 +02:00
$return .= __ (
'Graphs modules that represent the self-monitoring system'
);
2019-10-21 18:11:04 +02:00
$return .= '</div>' ;
$return .= '<div class="container-self-monitoring">' ;
2019-10-17 15:17:28 +02:00
foreach ( $agentMonitoring as $key => $value ) {
2019-10-21 18:11:04 +02:00
$return .= $this -> printDataCharts ( $value );
2019-10-17 15:17:28 +02:00
}
2019-10-14 17:21:38 +02:00
2019-10-21 18:11:04 +02:00
$return .= '</div>' ;
2019-10-14 17:21:38 +02:00
}
2019-10-17 15:17:28 +02:00
2019-10-21 18:11:04 +02:00
return $return ;
2019-10-11 14:55:20 +02:00
}
/**
* Info status pandoraFms .
*
* @ return string
*/
public function getStatusInfo () : string
{
global $config ;
global $build_version ;
global $pandora_version ;
$sql = sprintf (
" SELECT `key`, `value`
FROM `tupdate_settings`
WHERE `key` = '%s'
OR `key` = '%s'
OR `key` = '%s' " ,
'current_update' ,
'customer_key' ,
'updating_code_path'
);
$values_key = db_get_all_rows_sql ( $sql );
$values_key = array_reduce (
$values_key ,
function ( $carry , $item ) {
if ( $item [ 'key' ] === 'customer_key' ) {
$customer = substr ( $item [ 'value' ], 0 , 5 );
$customer .= '...' ;
$customer .= substr ( $item [ 'value' ], - 5 );
$item [ 'value' ] = $customer ;
}
$carry [ $item [ 'key' ]] = $item [ 'value' ];
return $carry ;
}
);
$result = [
'error' => false ,
'data' => [
'buildVersion' => [
2019-11-13 15:52:09 +01:00
'name' => __ ( '%s Build' , $this -> product_name ),
2019-10-11 14:55:20 +02:00
'value' => $build_version ,
],
'version' => [
2019-11-13 15:52:09 +01:00
'name' => __ ( '%s Version' , $this -> product_name ),
2019-10-11 14:55:20 +02:00
'value' => $pandora_version ,
],
'mr' => [
'name' => __ ( 'Minor Release' ),
'value' => $config [ 'MR' ],
],
'homeDir' => [
'name' => __ ( 'Homedir' ),
'value' => $config [ 'homedir' ],
],
'homeUrl' => [
'name' => __ ( 'HomeUrl' ),
'value' => $config [ 'homeurl' ],
],
'isEnterprise' => [
'name' => __ ( 'Enterprise installed' ),
'value' => ( enterprise_installed ()) ? __ ( 'true' ) : __ ( 'false' ),
],
'customerKey' => [
'name' => __ ( 'Update Key' ),
'value' => $values_key [ 'customer_key' ],
],
'updatingCode' => [
'name' => __ ( 'Updating code path' ),
'value' => $values_key [ 'updating_code_path' ],
],
'currentUpdate' => [
'name' => __ ( 'Current Update #' ),
'value' => $values_key [ 'current_update' ],
],
],
];
return json_encode ( $result );
}
/**
* PHP Status .
*
* @ return string
*/
public function getPHPSetup () : string
{
global $config ;
$result = [
'error' => false ,
'data' => [
'phpVersion' => [
'name' => __ ( 'PHP Version' ),
'value' => phpversion (),
],
'maxExecutionTime' => [
'name' => __ ( 'PHP Max execution time' ),
'value' => ini_get ( 'max_execution_time' ),
],
'maxInputTime' => [
'name' => __ ( 'PHP Max input time' ),
'value' => ini_get ( 'max_input_time' ),
],
'memoryLimit' => [
'name' => __ ( 'PHP Memory limit' ),
'value' => ini_get ( 'memory_limit' ),
],
'sessionLifetime' => [
'name' => __ ( 'Session cookie lifetime' ),
'value' => ini_get ( 'session.cookie_lifetime' ),
],
],
];
return json_encode ( $result );
}
2023-02-07 10:21:45 +01:00
/**
* Date system
*
* @ return string
*/
public function getSystemDate () : string
{
$result = [
'error' => false ,
'data' => [
'date' => [
'name' => __ ( 'System Date (Console)' ),
'value' => date ( 'H:i:s Y-m-d' ),
],
],
];
return json_encode ( $result );
}
2019-10-11 14:55:20 +02:00
/**
* Database size stats .
*
* @ return string
*/
public function getDatabaseSizeStats () : string
{
global $config ;
$countAgents = db_get_value_sql ( 'SELECT COUNT(*) FROM tagente' );
$countModules = db_get_value_sql ( 'SELECT COUNT(*) FROM tagente_modulo' );
$countGroups = db_get_value_sql ( 'SELECT COUNT(*) FROM tgrupo' );
$countModuleData = db_get_value_sql (
'SELECT COUNT(*) FROM tagente_datos'
);
$countAgentAccess = db_get_value_sql (
'SELECT COUNT(*) FROM tagent_access'
);
$countEvents = db_get_value_sql ( 'SELECT COUNT(*) FROM tevento' );
if ( enterprise_installed () === true ) {
$countTraps = db_get_value_sql ( 'SELECT COUNT(*) FROM ttrap' );
}
$countUsers = db_get_value_sql ( 'SELECT COUNT(*) FROM tusuario' );
$countSessions = db_get_value_sql ( 'SELECT COUNT(*) FROM tsesion' );
$result = [
'error' => false ,
'data' => [
'countAgents' => [
2019-11-13 15:52:09 +01:00
'name' => __ ( 'Total agents' ),
2019-10-11 14:55:20 +02:00
'value' => $countAgents ,
],
'countModules' => [
'name' => __ ( 'Total modules' ),
'value' => $countModules ,
],
'countGroups' => [
'name' => __ ( 'Total groups' ),
'value' => $countGroups ,
],
'countModuleData' => [
'name' => __ ( 'Total module data records' ),
'value' => $countModuleData ,
],
'countAgentAccess' => [
'name' => __ ( 'Total agent access record' ),
'value' => $countAgentAccess ,
],
'countEvents' => [
'name' => __ ( 'Total events' ),
'value' => $countEvents ,
],
'countTraps' => [
'name' => __ ( 'Total traps' ),
'value' => $countTraps ,
],
'countUsers' => [
'name' => __ ( 'Total users' ),
'value' => $countUsers ,
],
'countSessions' => [
'name' => __ ( 'Total sessions' ),
'value' => $countSessions ,
],
],
];
return json_encode ( $result );
}
2019-10-14 17:21:38 +02:00
/**
* Database health status .
*
* @ return string
*/
public function getDatabaseHealthStatus () : string
{
global $config ;
// Count agents unknowns.
$sqlUnknownAgents = ' SELECT COUNT ( DISTINCT tagente . id_agente )
FROM tagente_estado , tagente , tagente_modulo
WHERE tagente . disabled = 0
AND tagente_modulo . id_agente_modulo = tagente_estado . id_agente_modulo
AND tagente_modulo . disabled = 0
AND tagente_estado . id_agente = tagente . id_agente
AND tagente_estado . estado = 3 ' ;
$unknownAgents = db_get_sql ( $sqlUnknownAgents );
// Count modules not initialize.
$sqlNotInitAgents = ' SELECT COUNT ( tagente_estado . estado )
FROM tagente_estado
WHERE tagente_estado . estado = 4 ' ;
$notInitAgents = db_get_sql ( $sqlNotInitAgents );
$dateDbMantenaince = $config [ 'db_maintance' ];
$currentTime = time ();
2021-11-04 16:28:15 +01:00
$pandoraDbLastRun = __ ( 'Pandora DB has never been executed' );
2023-02-03 12:23:47 +01:00
if ( $dateDbMantenaince !== false && empty ( $dateDbMantenaince ) === false ) {
2019-10-14 17:21:38 +02:00
$difference = ( $currentTime - $dateDbMantenaince );
2019-11-13 15:52:09 +01:00
$pandoraDbLastRun = human_time_description_raw (
$difference ,
true
);
2019-10-14 17:21:38 +02:00
$pandoraDbLastRun .= ' ' . __ ( 'Ago' );
}
$result = [
'error' => false ,
'data' => [
'unknownAgents' => [
'name' => __ ( 'Total unknown agents' ),
'value' => $unknownAgents ,
],
'notInitAgents' => [
'name' => __ ( 'Total not-init modules' ),
'value' => $notInitAgents ,
],
'pandoraDbLastRun' => [
2021-11-04 16:28:15 +01:00
'name' => __ ( 'Pandora DB Last run' ),
2019-10-14 17:21:38 +02:00
'value' => $pandoraDbLastRun ,
],
],
];
return json_encode ( $result );
}
/**
* Database status info .
*
* @ return string
*/
public function getDatabaseStatusInfo () : string
{
global $config ;
$result = [
'error' => false ,
'data' => [
'dbSchemeFirstVersion' => [
'name' => __ ( 'DB Schema Version (first installed)' ),
'value' => $config [ 'db_scheme_first_version' ],
],
'dbSchemeVersion' => [
'name' => __ ( 'DB Schema Version (actual)' ),
'value' => $config [ 'db_scheme_version' ],
],
'dbSchemeBuild' => [
'name' => __ ( 'DB Schema Build' ),
'value' => $config [ 'db_scheme_build' ],
],
],
];
return json_encode ( $result );
}
/**
* Database status info .
*
* @ return string
*/
public function getSystemInfo () : string
{
global $config ;
2019-10-17 15:17:28 +02:00
$result = [];
if ( strtoupper ( substr ( PHP_OS , 0 , 3 )) !== 'WIN' ) {
$cpuModelName = 'cat /proc/cpuinfo | grep "model name" | tail -1 | cut -f 2 -d ":"' ;
$cpuProcessor = 'cat /proc/cpuinfo | grep "processor" | wc -l' ;
$ramMemTotal = 'cat /proc/meminfo | grep "MemTotal"' ;
2019-10-14 17:21:38 +02:00
2019-11-13 17:50:22 +01:00
exec (
" ifconfig | awk ' { print $ 2}' | grep -E -o '([0-9] { 1,3}[ \ .]) { 3}[0-9] { 1,3}' " ,
$output
);
$ips = implode ( ', ' , $output );
2019-10-17 15:17:28 +02:00
$result = [
'error' => false ,
'data' => [
2019-11-13 16:43:32 +01:00
'cpuInfo' => [
2019-10-17 15:17:28 +02:00
'name' => __ ( 'CPU' ),
'value' => exec ( $cpuModelName ) . ' x ' . exec ( $cpuProcessor ),
],
2019-11-13 16:43:32 +01:00
'ramInfo' => [
2019-10-17 15:17:28 +02:00
'name' => __ ( 'RAM' ),
'value' => exec ( $ramMemTotal ),
],
2019-11-13 16:43:32 +01:00
'osInfo' => [
'name' => __ ( 'Os' ),
'value' => exec ( 'uname -a' ),
],
'hostnameInfo' => [
'name' => __ ( 'Hostname' ),
'value' => exec ( 'hostname' ),
],
'ipInfo' => [
'name' => __ ( 'Ip' ),
2019-11-13 17:50:22 +01:00
'value' => $ips ,
2019-11-13 16:43:32 +01:00
],
],
];
} else {
$result = [
'error' => false ,
'data' => [
'osInfo' => [
'name' => __ ( 'OS' ),
'value' => exec ( 'ver' ),
],
'hostnameInfo' => [
'name' => __ ( 'Hostname' ),
'value' => exec ( 'hostname' ),
],
'ipInfo' => [
'name' => __ ( 'Ip' ),
'value' => exec ( 'ipconfig | findstr IPv4' ),
],
2019-10-14 17:21:38 +02:00
],
2019-10-17 15:17:28 +02:00
];
}
2019-10-14 17:21:38 +02:00
return json_encode ( $result );
}
/**
* MySQL Performance metrics .
*
* @ return string
*/
public function getMySQLPerformanceMetrics () : string
{
global $config ;
2019-10-17 15:17:28 +02:00
$variablesMsql = db_get_all_rows_sql ( 'SHOW VARIABLES' );
2019-10-14 17:21:38 +02:00
$variablesMsql = array_reduce (
$variablesMsql ,
function ( $carry , $item ) {
$bytes = 1048576 ;
$mega = 1024 ;
switch ( $item [ 'Variable_name' ]) {
2019-11-12 16:22:58 +01:00
case 'innodb_buffer_pool_size' :
$name = __ ( 'InnoDB buffer pool size' );
2019-10-14 17:21:38 +02:00
$value = ( $item [ 'Value' ] / $bytes );
2019-11-12 16:22:58 +01:00
$status = (( $item [ 'Value' ] / $bytes ) >= 250 ) ? 1 : 0 ;
$message = __ (
'It has to be 40% of the server memory not recommended to be greater or less'
);
2019-10-14 17:21:38 +02:00
break ;
2019-11-12 16:22:58 +01:00
case 'innodb_file_per_table' :
$name = __ ( 'InnoDB file per table' );
$value = $item [ 'Value' ];
$status = ( $item [ 'Value' ] === 'ON' ) ? 1 : 0 ;
$message = __ ( 'Recommended ON' );
2019-10-14 17:21:38 +02:00
break ;
case 'innodb_flush_log_at_trx_commit' :
$name = __ ( 'InnoDB flush log at trx-commit' );
$value = $item [ 'Value' ];
2022-01-26 17:34:56 +01:00
$status = (( int ) $item [ 'Value' ] === self :: INNODB_FLUSH_LOG_AT_TRX_COMMIT ) ? 1 : 0 ;
$message = __ ( 'Recommended Value %d' , self :: INNODB_FLUSH_LOG_AT_TRX_COMMIT );
2019-10-14 17:21:38 +02:00
break ;
2019-11-12 16:22:58 +01:00
case 'innodb_lock_wait_timeout' :
$name = __ ( 'InnoDB lock wait timeout' );
$value = $item [ 'Value' ];
$status = ( $item [ 'Value' ] >= 90 ) ? 1 : 0 ;
$message = __ ( 'Min. Recommended Value' ) . ' 90s' ;
2019-10-14 17:21:38 +02:00
break ;
2019-11-12 16:22:58 +01:00
case 'innodb_log_buffer_size' :
$name = __ ( 'InnoDB log buffer size' );
$value = ( $item [ 'Value' ] / $bytes );
$status = (( $item [ 'Value' ] / $bytes ) >= 16 ) ? 1 : 0 ;
$message = __ ( 'Min. Recommended Value' ) . ' 16M' ;
2019-10-14 17:21:38 +02:00
break ;
2019-11-12 16:22:58 +01:00
case 'innodb_log_file_size' :
$name = __ ( 'InnoDB log file size' );
$value = ( $item [ 'Value' ] / $bytes );
$status = (( $item [ 'Value' ] / $bytes ) >= 64 ) ? 1 : 0 ;
$message = __ ( 'Min. Recommended Value' ) . ' 64M' ;
2019-10-14 17:21:38 +02:00
break ;
2019-11-12 16:22:58 +01:00
case 'max_allowed_packet' :
$name = __ ( 'Maximun allowed packet' );
2019-10-14 17:21:38 +02:00
$value = ( $item [ 'Value' ] / $bytes );
2019-10-17 15:17:28 +02:00
$status = (( $item [ 'Value' ] / $bytes ) >= 32 ) ? 1 : 0 ;
2019-11-12 16:22:58 +01:00
$message = __ ( 'Min. Recommended Value' ) . ' 32M' ;
break ;
case 'max_connections' :
$name = __ ( 'Maximun connections' );
$value = $item [ 'Value' ];
$status = (( $item [ 'Value' ]) >= 90 ) ? 1 : 0 ;
$message = __ ( 'Min. Recommended Value' );
$message .= ' 90 ' ;
$message .= __ ( 'conections' );
2019-10-14 17:21:38 +02:00
break ;
case 'query_cache_limit' :
$name = __ ( 'Query cache limit' );
$value = ( $item [ 'Value' ] / $bytes );
2019-11-12 16:22:58 +01:00
$status = (( $item [ 'Value' ] / $bytes ) >= 8 ) ? 1 : 0 ;
$message = __ ( 'Min. Recommended Value' ) . ' 8M' ;
2019-10-14 17:21:38 +02:00
break ;
2019-11-12 16:22:58 +01:00
case 'query_cache_min_res_unit' :
$name = __ ( 'Query cache min-res-unit' );
$value = ( $item [ 'Value' ] / $mega );
$status = (( $item [ 'Value' ] / $mega ) >= 2 ) ? 1 : 0 ;
$message = __ ( 'Min. Recommended Value' ) . ' 2M' ;
2019-10-14 17:21:38 +02:00
break ;
2019-11-12 16:22:58 +01:00
case 'query_cache_size' :
$name = __ ( 'Query cache size' );
2019-10-14 17:21:38 +02:00
$value = ( $item [ 'Value' ] / $bytes );
2019-11-12 16:22:58 +01:00
$status = (( $item [ 'Value' ] / $bytes ) >= 32 ) ? 1 : 0 ;
$message = __ ( 'Min. Recommended Value' ) . ' 32M' ;
2019-10-14 17:21:38 +02:00
break ;
2019-11-12 16:22:58 +01:00
case 'query_cache_type' :
$name = __ ( 'Query cache type' );
2019-10-14 17:21:38 +02:00
$value = $item [ 'Value' ];
2019-11-12 16:22:58 +01:00
$status = ( $item [ 'Value' ] === 'ON' ) ? 1 : 0 ;
$message = __ ( 'Recommended ON' );
2019-10-14 17:21:38 +02:00
break ;
case 'read_buffer_size' :
$name = __ ( 'Read buffer size' );
2019-11-12 16:22:58 +01:00
$value = ( $item [ 'Value' ] / $mega );
$status = (( $item [ 'Value' ] / $mega ) >= 32 ) ? 1 : 0 ;
$message = __ ( 'Min. Recommended Value' ) . ' 32K' ;
2019-10-14 17:21:38 +02:00
break ;
case 'read_rnd_buffer_size' :
$name = __ ( 'Read rnd-buffer size' );
2019-11-12 16:22:58 +01:00
$value = ( $item [ 'Value' ] / $mega );
$status = (( $item [ 'Value' ] / $mega ) >= 32 ) ? 1 : 0 ;
$message = __ ( 'Min. Recommended Value' ) . ' 32K' ;
2019-10-14 17:21:38 +02:00
break ;
2019-11-12 16:22:58 +01:00
case 'sort_buffer_size' :
$name = __ ( 'Sort buffer size' );
$value = ( $item [ 'Value' ] / $mega );
$status = (( $item [ 'Value' ] / $mega ) >= 32 ) ? 1 : 0 ;
$message = __ ( 'Min. Recommended Value' ) . ' 32K' ;
2019-10-14 17:21:38 +02:00
break ;
2019-11-12 16:22:58 +01:00
case 'sql_mode' :
$name = __ ( 'Sql mode' );
$value = ( $item [ 'Value' ]);
$status = ( empty ( $item [ 'Value' ]) === true ) ? 1 : 0 ;
$message = __ ( 'Must be empty' );
break ;
case 'thread_cache_size' :
$name = __ ( 'Thread cache size' );
2019-10-14 17:21:38 +02:00
$value = $item [ 'Value' ];
2019-11-12 16:22:58 +01:00
$status = ( $item [ 'Value' ] >= 8 ) ? 1 : 0 ;
$message = __ ( 'Min. Recommended Value' ) . ' 8' ;
break ;
case 'thread_stack' :
$name = __ ( 'Thread stack' );
$value = ( $item [ 'Value' ] / $mega );
$status = (( $item [ 'Value' ] / $mega ) >= 256 ) ? 1 : 0 ;
$message = __ ( 'Min. Recommended Value' ) . ' 256' ;
2019-10-14 17:21:38 +02:00
break ;
default :
$name = '' ;
$value = 0 ;
break ;
}
if ( empty ( $name ) !== true ) {
$carry [ $item [ 'Variable_name' ]] = [
'name' => $name ,
'value' => $value ,
'status' => $status ,
'message' => $message ,
];
}
return $carry ;
},
[]
);
$result = [
'error' => false ,
'data' => $variablesMsql ,
];
return json_encode ( $result );
}
/**
* Tables fragmentation in the Pandora FMS database .
*
* @ return string
*/
public function getTablesFragmentation () : string
{
global $config ;
// Estimated fragmentation percentage as maximum.
$tFragmentationMax = 10 ;
// Extract the fragmentation value.
$tFragmentationValue = db_get_sql (
sprintf (
" SELECT (data_free/(index_length+data_length)) as frag_ratio
FROM information_schema . tables
WHERE DATA_FREE > 0
AND table_name = 'tagente_datos'
AND table_schema = '%s' " ,
$config [ 'dbname' ]
)
);
// Check if it meets the fragmentation value.
$status_tables_frag = '' ;
if ( $tFragmentationValue > $tFragmentationMax ) {
$tFragmentationMsg = __ (
'Table fragmentation is higher than recommended. They should be defragmented.'
);
$tFragmentationStatus = 0 ;
} else {
$tFragmentationMsg = __ ( 'Table fragmentation is correct.' );
$tFragmentationStatus = 1 ;
}
$result = [
'error' => false ,
'data' => [
'tablesFragmentationMax' => [
'name' => __ (
'Tables fragmentation (maximum recommended value)'
),
'value' => $tFragmentationMax . '%' ,
],
'tablesFragmentationValue' => [
'name' => __ ( 'Tables fragmentation (current value)' ),
2022-10-26 18:25:16 +02:00
'value' => number_format ( $tFragmentationValue , 2 , $config [ 'decimal_separator' ], $config [ 'thousand_separator' ]) . '%' ,
2019-10-14 17:21:38 +02:00
],
'tablesFragmentationStatus' => [
'name' => __ ( 'Table fragmentation status' ),
'value' => $status_tables_frag ,
'status' => $tFragmentationStatus ,
],
],
];
return json_encode ( $result );
}
/**
* Pandora FMS logs dates .
*
* @ return string
*/
public function getPandoraFMSLogsDates () : string
{
global $config ;
$unit = 'M' ;
2019-11-13 15:52:09 +01:00
$pathServerLogs = '/var/log/pandora/pandora_server.log' ;
2019-10-14 17:21:38 +02:00
$servers = $this -> getLogInfo ( $pathServerLogs );
2019-11-13 15:52:09 +01:00
$pathErrLogs = '/var/log/pandora/pandora_server.error' ;
2019-10-14 17:21:38 +02:00
$errors = $this -> getLogInfo ( $pathErrLogs );
2020-06-12 09:40:12 +02:00
$pathConsoleLogs = $config [ 'homedir' ] . '/log/console.log' ;
2019-10-14 17:21:38 +02:00
$console = $this -> getLogInfo ( $pathConsoleLogs );
$result = [
'error' => false ,
'data' => [
'sizeServerLog' => [
'name' => __ ( 'Size server logs (current value)' ),
'value' => $servers [ 'value' ] . ' ' . $unit ,
],
'statusServerLog' => [
'name' => __ ( 'Status server logs' ),
'value' => $servers [ 'message' ],
'status' => $servers [ 'status' ],
],
'sizeErrorLog' => [
'name' => __ ( 'Size error logs (current value)' ),
'value' => $errors [ 'value' ] . ' ' . $unit ,
],
'statusErrorLog' => [
'name' => __ ( 'Status error logs' ),
'value' => $errors [ 'message' ],
'status' => $errors [ 'status' ],
],
'sizeConsoleLog' => [
'name' => __ ( 'Size console logs (current value)' ),
'value' => $console [ 'value' ] . ' ' . $unit ,
],
'statusConsoleLog' => [
'name' => __ ( 'Status console logs' ),
'value' => $console [ 'message' ],
'status' => $console [ 'status' ],
],
],
];
return json_encode ( $result );
}
/**
* Pandora FMS Licence Information .
*
* @ return string
*/
public function getLicenceInformation () : string
{
global $config ;
// Extract customer key.
$sql = sprintf (
" SELECT `value`
FROM `tupdate_settings`
WHERE `key` = '%s' " ,
'customer_key'
);
$customerKey = db_get_value_sql ( $sql );
// Extract Info license.
2019-11-13 15:52:09 +01:00
enterprise_include_once ( 'include/functions_license.php' );
2019-10-14 17:21:38 +02:00
$license = enterprise_hook ( 'license_get_info' );
// Agent Capacity.
$agentCount = db_get_value_sql ( 'SELECT count(*) FROM tagente' );
$agentsCapacity = __ ( 'License capacity is less than 90 percent' );
$agentsCapacitySt = 1 ;
if ( $agentCount > ( $license [ 'limit' ] * 90 / 100 )) {
$agentsCapacity = __ ( 'License capacity exceeds 90 percent' );
$agentsCapacitySt = 0 ;
}
// Modules average.
$modulesCount = db_get_value_sql ( 'SELECT count(*) FROM tagente_modulo' );
$average = ( $modulesCount / $agentCount );
$averageMsg = __ (
'The average of modules per agent is more than 40. You can have performance problems'
);
$averageSt = 0 ;
if ( $average <= 40 ) {
$averageMsg = __ (
'The average of modules per agent is less than 40'
);
$averageSt = 1 ;
}
// Modules Networks average.
$totalNetworkModules = db_get_value_sql (
' SELECT count ( * )
FROM tagente_modulo
WHERE id_tipo_modulo
BETWEEN 6 AND 18 '
);
2019-11-13 15:52:09 +01:00
2019-10-14 17:21:38 +02:00
$totalModuleIntervalTime = db_get_value_sql (
' SELECT SUM ( module_interval )
FROM tagente_modulo
WHERE id_tipo_modulo
BETWEEN 6 AND 18 '
);
2019-10-17 15:17:28 +02:00
$averageTime = 0 ;
if ( $totalModuleIntervalTime !== false ) {
$averageTime = number_format (
2019-11-13 15:52:09 +01:00
(( int ) $totalNetworkModules / ( int ) $totalModuleIntervalTime ),
2022-10-26 18:25:16 +02:00
3 ,
$config [ 'decimal_separator' ],
$config [ 'thousand_separator' ]
2019-10-17 15:17:28 +02:00
);
}
2019-10-14 17:21:38 +02:00
$moduleNetworkmsg = __ (
sprintf (
2019-11-13 15:52:09 +01:00
'The system is not overloaded (average time %f)' ,
$averageTime
2019-10-14 17:21:38 +02:00
)
);
$moduleNetworkst = 1 ;
2019-11-13 15:52:09 +01:00
if ( $averageTime === 0 ) {
2019-10-14 17:21:38 +02:00
$moduleNetworkmsg = __ ( 'The system has no load' );
$moduleNetworkst = 0 ;
2019-11-13 15:52:09 +01:00
} else if ( $averageTime > 180 ) {
2019-10-14 17:21:38 +02:00
$moduleNetworkmsg = __ (
sprintf (
2019-11-13 15:52:09 +01:00
'The system is overloaded (average time %f) and a very fine configuration is required' ,
$averageTime
2019-10-14 17:21:38 +02:00
)
);
$moduleNetworkst = 0 ;
}
$result = [
'error' => false ,
'data' => [
'customerKey' => [
'name' => __ ( 'Customer key' ),
'value' => $customerKey ,
],
'customerExpires' => [
'name' => __ ( 'Support expires' ),
'value' => $license [ 'expiry_date' ],
],
'customerLimit' => [
'name' => __ ( 'Platform Limit' ),
'value' => $license [ 'limit' ] . ' ' . __ ( 'Agents' ),
],
'customerPfCount' => [
'name' => __ ( 'Current Platform Count' ),
'value' => $license [ 'count' ] . ' ' . __ ( 'Agents' ),
],
'customerPfCountEnabled' => [
'name' => __ ( 'Current Platform Count (enabled: items)' ),
'value' => $license [ 'count_enabled' ] . ' ' . __ ( 'Agents' ),
],
'customerPfCountDisabled' => [
'name' => __ ( 'Current Platform Count (disabled: items)' ),
'value' => $license [ 'count_disabled' ] . ' ' . __ ( 'Agents' ),
],
'customerMode' => [
'name' => __ ( 'License Mode' ),
'value' => $license [ 'license_mode' ],
],
'customerNMS' => [
'name' => __ ( 'Network Management System' ),
'value' => ( $license [ 'nms' ] > 0 ) ? __ ( 'On' ) : __ ( 'Off' ),
],
'customerSatellite' => [
'name' => __ ( 'Satellite' ),
'value' => ( $license [ 'dhpm' ] > 0 ) ? __ ( 'On' ) : __ ( 'Off' ),
],
'customerLicenseTo' => [
'name' => __ ( 'Licensed to' ),
'value' => $license [ 'licensed_to' ],
],
'customerCapacity' => [
'name' => __ ( 'Status of agents capacity' ),
'value' => $agentsCapacity ,
'status' => $agentsCapacitySt ,
],
'customerAverage' => [
'name' => __ ( 'Status of average modules per agent' ),
'value' => $averageMsg ,
'status' => $averageSt ,
],
'customerAverageNetwork' => [
'name' => __ ( 'Interval average of the network modules' ),
'value' => $moduleNetworkmsg ,
'status' => $moduleNetworkst ,
],
],
];
return json_encode ( $result );
}
/**
* Status of the attachment folder .
*
* @ return string
*/
public function getAttachmentFolder () : string
{
global $config ;
// Count files in attachment.
$attachmentFiles = count (
glob (
$config [ 'homedir' ] . '/attachment/{*.*}' ,
GLOB_BRACE
)
);
// Check status attachment.
$attachmentMsg = __ (
'The attached folder contains more than 700 files.'
);
$attachmentSt = 0 ;
if ( $attachmentFiles <= 700 ) {
$attachmentMsg = __ (
'The attached folder contains less than 700 files.'
);
$attachmentSt = 1 ;
}
$result = [
'error' => false ,
'data' => [
'attachFiles' => [
'name' => __ ( 'Total files in the attached folder' ),
'value' => $attachmentFiles ,
],
'attachStatus' => [
'name' => __ ( 'Status of the attachment folder' ),
'value' => $attachmentMsg ,
'status' => $attachmentSt ,
],
],
];
return json_encode ( $result );
}
/**
* Information from the tagente_datos table .
*
* @ return string
*/
public function getInfoTagenteDatos () : string
{
global $config ;
$agentDataCount = db_get_value_sql (
' SELECT COUNT ( * )
FROM tagente_datos '
);
2023-01-04 11:16:51 +01:00
$modulesDataCount = db_get_value_sql (
2023-01-04 11:36:26 +01:00
'SELECT count(*) * 300 FROM (SELECT * FROM tagente_datos GROUP BY id_agente_modulo) AS totalmodules'
2023-01-04 11:16:51 +01:00
);
$modulesDataCount = ( $modulesDataCount >= 500000 ) ? $modulesDataCount : 500000 ;
2019-10-14 17:21:38 +02:00
$taMsg = __ (
'The tagente_datos table contains too much data. A historical database is recommended.'
);
$taStatus = 0 ;
2023-01-04 11:16:51 +01:00
if ( $agentDataCount <= $modulesDataCount ) {
2019-10-14 17:21:38 +02:00
$taMsg = __ (
'The tagente_datos table contains an acceptable amount of data.'
);
$taStatus = 1 ;
}
$result = [
'error' => false ,
'data' => [
2019-10-17 15:17:28 +02:00
'agentDataCount' => [
2019-10-14 17:21:38 +02:00
'name' => __ ( 'Total data in tagente_datos table' ),
'value' => $agentDataCount ,
],
2019-10-17 15:17:28 +02:00
'agentDataStatus' => [
2019-10-14 17:21:38 +02:00
'name' => __ ( 'Tagente_datos table status' ),
'value' => $taMsg ,
'status' => $taStatus ,
],
],
];
return json_encode ( $result );
}
/**
* Pandora FMS server threads .
*
* @ return string
*/
public function getServerThreads () : string
{
global $config ;
2019-11-14 08:53:45 +01:00
$result = [];
2019-11-14 09:52:56 +01:00
$totalServerThreads = 0 ;
2019-11-14 08:53:45 +01:00
if ( strtoupper ( substr ( PHP_OS , 0 , 3 )) !== 'WIN' ) {
$totalServerThreads = shell_exec (
'ps -T aux | grep pandora_server | grep -v grep | wc -l'
);
2019-10-17 15:17:28 +02:00
}
2019-11-14 08:53:45 +01:00
include_once $config [ 'homedir' ] . '/include/functions_servers.php' ;
$sql = 'SELECT `name`, server_type, threads FROM tserver' ;
$servers = db_get_all_rows_sql ( $sql );
2019-10-14 17:21:38 +02:00
2019-11-14 08:53:45 +01:00
if ( isset ( $servers ) === true && is_array ( $servers ) === true ) {
$sum_threads = 0 ;
foreach ( $servers as $key => $value ) {
$result [ 'data' ][ 'threads_server_' . $value [ 'server_type' ]] = [
'name' => __ ( 'Threads' ) . ' ' . \servers_get_server_string_name (
$value [ 'server_type' ]
),
'value' => $value [ 'threads' ],
];
$sum_threads += $value [ 'threads' ];
}
$result [ 'data' ][ 'total_threads' ] = [
2019-11-14 09:52:56 +01:00
'name' => __ ( 'Total threads' ),
'value' => $sum_threads ,
'status' => ( $sum_threads < $totalServerThreads ) ? 2 : 1 ,
2019-11-14 08:53:45 +01:00
];
2019-11-14 09:52:56 +01:00
2019-11-14 09:58:10 +01:00
if ( $sum_threads < $totalServerThreads ) {
2019-11-14 09:52:56 +01:00
$result [ 'data' ][ 'total_threads' ][ 'message' ] = __ (
'Current pandora_server running threads'
);
2019-11-14 09:58:10 +01:00
} else {
__ (
'There\'s more pandora_server threads than configured, are you running multiple servers simultaneusly?.'
);
2019-11-14 09:52:56 +01:00
}
2019-11-14 08:53:45 +01:00
}
2019-10-14 17:21:38 +02:00
return json_encode ( $result );
}
2019-10-23 19:50:57 +02:00
/**
* SQL show engine innodb status .
*
* @ return string
*/
public function getShowEngine () : string
{
global $config ;
try {
// Trick to avoid showing error in case
// you don't have enough permissions.
$backup = error_reporting ();
error_reporting ( 0 );
$innodb = db_get_all_rows_sql ( 'show engine innodb status' );
error_reporting ( $backup );
} catch ( Exception $e ) {
$innodb [ 'Status' ] = $e -> getMessage ();
}
$result = [];
if ( isset ( $innodb [ 0 ][ 'Status' ]) === true
&& $innodb [ 0 ][ 'Status' ] !== false
) {
$lenght = strlen ( $innodb [ 0 ][ 'Status' ]);
$data = [];
2019-11-13 16:43:32 +01:00
for ( $i = 0 ; $i < $lenght ; $i = ( $i + 300 )) {
$str = substr ( $innodb [ 0 ][ 'Status' ], $i , ( $i + 300 ));
2019-10-23 19:50:57 +02:00
$data [ 'showEngine-' . $i ] = [
'name' => '' ,
'value' => '<pre>' . $str . '</pre>' ,
];
}
$result = [
'error' => false ,
'data' => $data ,
'id' => 'showEngine' ,
];
}
return json_encode ( $result );
}
2019-10-11 14:55:20 +02:00
/**
2019-10-17 15:17:28 +02:00
* Agent Id whit name is equal to Server Name .
*
* @ return integer Id agent module .
*/
public function getAgentIdMasterServer () : int
{
global $config ;
$serverName = db_get_value_sql (
' SELECT `name`
FROM tserver
WHERE `name` IS NOT NULL
AND `master` > 0
ORDER BY `master` DESC '
);
$agentId = ( int ) db_get_value_sql (
sprintf (
' SELECT id_agente
FROM tagente
WHERE nombre = " %s " ' ,
$serverName
)
);
if ( isset ( $agentId ) === false || is_numeric ( $agentId ) === false ) {
$agentId = 0 ;
}
return $agentId ;
}
/**
* Graph .
*
* @ param integer $id Id agent .
* @ param string $name Name module .
* @ param boolean $image Chart interactive or only image .
* @ param boolean $base64 Image or base64 .
*
* @ return string
*/
public function getChart (
int $id ,
string $name ,
bool $image = false ,
bool $base64 = false
) : string {
global $config ;
include_once $config [ 'homedir' ] . '/include/functions_graph.php' ;
$data = modules_get_agentmodule_id ( $name , $id );
$params = [
'agent_module_id' => $data [ 'id_agente_modulo' ],
'period' => SECONDS_1MONTH ,
'date' => time (),
'height' => '200' ,
'only_image' => $image ,
'return_img_base_64' => $base64 ,
];
return grafico_modulo_sparse ( $params );
}
/**
* Check pandoradb installed .
*
* @ return string
*/
public function checkPandoraDB () : string
{
global $config ;
$result = '' ;
if ( isset ( $config [ 'db_maintenance' ]) === false ) {
$result .= '(*) ' ;
$result .= __ (
2021-11-04 16:28:15 +01:00
'Please check your Pandora Server setup and make sure that the database maintenance daemon is running.'
2019-10-17 15:17:28 +02:00
);
$result .= ' ' ;
$result .= __ (
2021-11-04 16:28:15 +01:00
'It\' is very important to keep the database up-to-date to get the best performance and results in Pandora'
2019-10-17 15:17:28 +02:00
);
}
return $result ;
}
/**
* Draw table .
*
* @ param string $method Method .
* @ param string $title Title .
*
2019-10-21 18:11:04 +02:00
* @ return string Return html .
2019-10-17 15:17:28 +02:00
*/
2019-10-21 18:11:04 +02:00
public function printData ( string $method , string $title ) : string
2019-10-17 15:17:28 +02:00
{
global $config ;
if ( is_ajax ()) {
// TODO: Call method.
2019-10-21 18:11:04 +02:00
$result = $method ;
2019-10-17 15:17:28 +02:00
} else {
// Datatables list.
try {
$columns = [
[
'class' => 'datatables-td-title' ,
'text' => 'name' ,
],
[
'class' => 'datatables-td-max' ,
'text' => 'value' ,
],
'message' ,
];
$columnNames = [
[
'style' => 'display:none;' ,
'text' => '' ,
],
[
'style' => 'display:none' ,
'text' => '' ,
],
[
'style' => 'display:none' ,
'text' => '' ,
],
];
$tableId = $method . '_' . uniqid ();
// Load datatables user interface.
2019-10-21 18:11:04 +02:00
if ( $this -> pdf === false ) {
$result = ui_print_datatable (
[
'id' => $tableId ,
'class' => 'info_table caption_table' ,
'style' => 'width: 100%' ,
'columns' => $columns ,
'column_names' => $columnNames ,
'ajax_data' => [
'method' => 'datatablesDraw' ,
'name' => $method ,
],
'ajax_url' => $this -> ajaxController ,
'paging' => 0 ,
'no_sortable_columns' => [ - 1 ],
'caption' => $title ,
'print' => true ,
]
);
} else {
$data = json_decode (
$this -> datatablesDraw ( $method , true ),
true
);
$table = new stdClass ();
$table -> width = '100%' ;
2019-10-23 19:50:57 +02:00
$table -> class = 'pdf-report' ;
$table -> style = [];
$table -> style [ 0 ] = 'font-weight: bolder;' ;
// FIX tables break content.
if ( $data [ 'idTable' ] === 'showEngine' ) {
$table -> styleTable = 'page-break-inside: auto;' ;
} else {
$table -> autosize = 1 ;
}
2019-10-21 18:11:04 +02:00
$table -> head = [];
$table -> head_colspan [ 0 ] = 3 ;
$table -> head [ 0 ] = $title ;
$table -> data = [];
2019-10-23 19:50:57 +02:00
if ( isset ( $data ) === true
&& is_array ( $data ) === true
&& count ( $data ) > 0
) {
2019-10-21 18:11:04 +02:00
$i = 0 ;
foreach ( $data [ 'data' ] as $key => $value ) {
$table -> data [ $i ][ 0 ] = $value [ 'name' ];
$table -> data [ $i ][ 1 ] = $value [ 'value' ];
$table -> data [ $i ][ 2 ] = $value [ 'message' ];
$i ++ ;
}
}
$result = html_print_table ( $table , true );
}
2019-10-17 15:17:28 +02:00
} catch ( Exception $e ) {
2019-10-21 18:11:04 +02:00
$result = $e -> getMessage ();
2019-10-17 15:17:28 +02:00
}
}
2019-10-21 18:11:04 +02:00
2021-04-21 11:07:20 +02:00
return $result ;
2019-10-17 15:17:28 +02:00
}
/**
* Prepare params for getCarts .
*
* @ return void
*/
public function getChartAjax () : void
{
global $config ;
$params = json_decode (
io_safe_output ( get_parameter ( 'params' , '' )),
true
);
$return = '' ;
if ( isset ( $params [ 'idAgent' ]) === true
&& empty ( $params [ 'idAgent' ]) === false
&& isset ( $params [ 'nameModule' ])
&& empty ( $params [ 'nameModule' ]) === false
) {
$return = $this -> getChart (
$params [ 'idAgent' ],
$params [ 'nameModule' ]
);
}
exit ( $return );
}
/**
* Paint table with charts .
2019-10-11 14:55:20 +02:00
*
2019-10-17 15:17:28 +02:00
* @ param array $params Info charts .
2019-10-11 14:55:20 +02:00
*
2019-10-21 18:11:04 +02:00
* @ return string Html .
2019-10-11 14:55:20 +02:00
*/
2019-10-21 18:11:04 +02:00
public function printDataCharts ( array $params ) : string
2019-10-11 14:55:20 +02:00
{
global $config ;
2019-10-17 15:17:28 +02:00
if ( ! $params ) {
$params = get_parameter ( 'params' );
}
if ( is_ajax ()) {
// TODO: Call method.
2019-10-21 18:11:04 +02:00
$return = $method ;
2019-10-17 15:17:28 +02:00
} else {
// Datatables list.
try {
$id = str_replace (
' ' ,
'' ,
io_safe_output ( $params [ 'nameModule' ])
);
2019-10-21 18:11:04 +02:00
if ( $this -> pdf === false ) {
$return = '<div id="' . $id . '" class="element-self-monitoring"></div>' ;
$settings = [
'type' => 'POST' ,
'dataType' => 'html' ,
'url' => ui_get_full_url (
'ajax.php' ,
false ,
false ,
false
),
'data' => [
'page' => $this -> ajaxController ,
'method' => 'getChartAjax' ,
'params' => json_encode ( $params ),
],
];
?>
< script type = " text/javascript " >
ajaxRequest (
'<?php echo $id; ?>' ,
< ? php echo json_encode ( $settings ); ?>
);
</ script >
< ? php
} else {
2019-10-25 10:43:03 +02:00
$table = new stdClass ();
$table -> width = '100%' ;
$table -> class = 'pdf-report' ;
$table -> style = [];
$table -> style [ 0 ] = 'font-weight: bolder;' ;
$table -> autosize = 1 ;
$table -> head = [];
$table -> head [ 0 ] = $params [ 'nameModule' ];
$table -> data = [];
$table -> data [ 0 ] = $this -> getChart (
2019-10-21 18:11:04 +02:00
$params [ 'idAgent' ],
$params [ 'nameModule' ],
true ,
false
);
2019-10-25 10:43:03 +02:00
$return = html_print_table ( $table , true );
2019-10-21 18:11:04 +02:00
}
2019-10-17 15:17:28 +02:00
} catch ( Exception $e ) {
2019-10-21 18:11:04 +02:00
$return = $e -> getMessage ();
2019-10-17 15:17:28 +02:00
}
}
2019-10-21 18:11:04 +02:00
return $return ;
2019-10-11 14:55:20 +02:00
}
2019-10-14 17:21:38 +02:00
/**
* Private function for Info size path .
*
* @ param string $path Route file .
*
* @ return array With values size file and message and status .
*/
private function getLogInfo ( string $path ) : array
{
global $config ;
// Vars.
$mega = 1048576 ;
$tenMega = 10485760 ;
$result = [
'value' => 0 ,
'message' => '' ,
'status' => 0 ,
];
2019-11-13 15:52:09 +01:00
if ( file_exists ( $path ) === true ) {
2019-10-14 17:21:38 +02:00
$fileSize = filesize ( $path );
$sizeServerLog = number_format ( $fileSize );
$sizeServerLog = ( 0 + str_replace ( ',' , '' , $sizeServerLog ));
2022-10-26 18:25:16 +02:00
$value = number_format (( $fileSize / $mega ), 3 , $config [ 'decimal_separator' ], $config [ 'thousand_separator' ]);
2019-10-14 17:21:38 +02:00
$message = __ ( 'You have more than 10 MB of logs' );
$status = 0 ;
if ( $sizeServerLog <= $tenMega ) {
$message = __ ( 'You have less than 10 MB of logs' );
$status = 1 ;
}
$result = [
'value' => $value ,
'message' => $message ,
'status' => $status ,
];
}
return $result ;
}
2019-10-17 15:17:28 +02:00
/**
2019-10-21 18:11:04 +02:00
* Undocumented function
2019-10-17 15:17:28 +02:00
*
2019-10-21 18:11:04 +02:00
* @ param string | null $method Method data requested .
* @ param boolean $return Type return .
*
* @ return string | null
2019-10-17 15:17:28 +02:00
*/
2019-10-21 18:11:04 +02:00
public function datatablesDraw (
? string $method = null ,
bool $return = false
) : ? string {
if ( isset ( $method ) === false ) {
$method = get_parameter ( 'name' , '' );
}
2019-10-17 15:17:28 +02:00
if ( method_exists ( $this , $method ) === true ) {
$data = json_decode ( $this -> { $method }(), true );
}
2019-10-23 19:50:57 +02:00
$result = [];
if ( isset ( $data ) === true
&& is_array ( $data ) === true
&& count ( $data ) > 0
) {
2019-10-17 15:17:28 +02:00
$items = $data [ 'data' ];
$dataReduce = array_reduce (
array_keys ( $data [ 'data' ]),
function ( $carry , $key ) use ( $items ) {
// Transforms array of arrays $data into an array
// of objects, making a post-process of certain fields.
if ( isset ( $items [ $key ][ 'status' ]) === true ) {
$acumValue = $items [ $key ][ 'value' ];
2019-11-14 09:52:56 +01:00
if ( $items [ $key ][ 'status' ] === 2 ) {
$items [ $key ][ 'value' ] = html_print_image (
'images/icono-warning.png' ,
true ,
[
'title' => __ ( 'Warning' ),
'style' => 'width:15px;' ,
]
);
} else if ( $items [ $key ][ 'status' ] === 1 ) {
2019-10-17 15:17:28 +02:00
$items [ $key ][ 'value' ] = html_print_image (
'images/exito.png' ,
true ,
[
2022-04-20 14:51:01 +02:00
'title' => __ ( 'Successfully' ),
2019-10-17 15:17:28 +02:00
'style' => 'width:15px;' ,
]
);
} else {
$items [ $key ][ 'value' ] = html_print_image (
'images/error_1.png' ,
true ,
[
'title' => __ ( 'Error' ),
'style' => 'width:15px;' ,
]
);
}
$items [ $key ][ 'value' ] .= ' ' . $acumValue ;
}
// FIX for customer key.
if ( $key === 'customerKey' ) {
2019-10-24 17:42:37 +02:00
$customerKey = ui_print_truncate_text (
$items [ $key ][ 'value' ],
30 ,
false ,
true ,
false
);
$spanValue = '<span>' . $customerKey . '</span>' ;
2019-10-17 15:17:28 +02:00
$items [ $key ][ 'value' ] = $spanValue ;
}
if ( isset ( $items [ $key ][ 'message' ]) === false ) {
$items [ $key ][ 'message' ] = '' ;
}
$carry [] = ( object ) $items [ $key ];
return $carry ;
}
);
2019-10-23 19:50:57 +02:00
$result = [
2019-10-17 15:17:28 +02:00
'data' => $dataReduce ,
'recordsTotal' => count ( $dataReduce ),
'recordsFiltered' => count ( $dataReduce ),
2019-10-23 19:50:57 +02:00
'idTable' => ( isset ( $data [ 'id' ]) === true ) ? $data [ 'id' ] : '' ,
];
}
2019-10-21 18:11:04 +02:00
// Datatables format: RecordsTotal && recordsfiltered.
if ( $return === false ) {
2019-10-23 19:50:57 +02:00
echo json_encode ( $result );
2019-10-21 18:11:04 +02:00
return null ;
} else {
2019-10-23 19:50:57 +02:00
return json_encode ( $result );
2019-10-21 18:11:04 +02:00
}
}
2019-10-25 10:43:03 +02:00
/**
* Print Diagnostics Form feedback .
*
* @ return void
*/
public function formFeedback () : void
{
$form = [
'action' => '#' ,
'id' => 'modal_form_feedback' ,
'onsubmit' => 'return false;' ,
'class' => 'modal' ,
2019-11-05 15:12:24 +01:00
'extra' => 'novalidate' ,
2019-10-25 10:43:03 +02:00
];
$inputs = [];
$inputs [] = [
2020-09-01 12:18:10 +02:00
'label' => __ ( 'What happened?' ),
2019-10-25 10:43:03 +02:00
'id' => 'div-what-happened' ,
'class' => 'flex-row' ,
'arguments' => [
2019-10-25 13:28:01 +02:00
'name' => 'what-happened' ,
'type' => 'textarea' ,
'value' => '' ,
'return' => true ,
'rows' => 1 ,
'columns' => 1 ,
'size' => 25 ,
'attributes' => 'required="required"' ,
2019-10-25 10:43:03 +02:00
],
];
$inputs [] = [
'label' => __ ( 'Your email' ),
'class' => 'flex-row-baseline' ,
'arguments' => [
'name' => 'email' ,
'id' => 'email' ,
'type' => 'email' ,
2019-10-25 13:28:01 +02:00
'size' => 42 ,
2019-10-25 10:43:03 +02:00
'required' => 'required' ,
],
];
$inputs [] = [
'label' => __ ( 'Include installation data' ),
'class' => 'flex-row-vcenter' ,
'arguments' => [
'name' => 'include_installation_data' ,
'id' => 'include_installation_data' ,
'type' => 'switch' ,
'value' => 1 ,
],
];
exit (
$this -> printForm (
[
'form' => $form ,
'inputs' => $inputs ,
],
true
)
);
}
2019-10-23 19:50:57 +02:00
/**
* Create cron task form feedback .
*
* @ return void Json result AJAX request .
*/
public function createdScheduleFeedbackTask () : void
2019-10-21 18:11:04 +02:00
{
global $config ;
2019-10-24 17:42:37 +02:00
$mail_feedback = 'feedback@artica.es' ;
2019-10-29 09:13:48 +01:00
$email = $mail_feedback ;
2019-11-13 15:52:09 +01:00
$subject = $this -> product_name . ' Report ' . $config [ 'pandora_uid' ];
2019-10-21 18:11:04 +02:00
$text = get_parameter ( 'what-happened' , '' );
2019-10-23 19:50:57 +02:00
$attachment = get_parameter_switch ( 'include_installation_data' , 0 );
2019-10-24 17:42:37 +02:00
$email_from = get_parameter_switch ( 'email' , '' );
2019-10-25 10:43:03 +02:00
$title = __ ( 'Hello Feedback-Men' );
2019-10-24 17:42:37 +02:00
2019-10-25 10:43:03 +02:00
$product_name = io_safe_output ( get_product_name ());
if ( check_acl ( $config [ 'id_user' ], 0 , 'PM' ) !== 1 ) {
2019-10-24 17:42:37 +02:00
$email = get_mail_admin ();
$name_admin = get_name_admin ();
$subject = __ ( 'Feedback' ) . ' ' . $product_name . ' ' . $config [ 'pandora_uid' ];
$title = __ ( 'Hello' ) . ' ' . $name_admin ;
2019-10-25 10:43:03 +02:00
}
2019-10-24 17:42:37 +02:00
2019-10-25 10:43:03 +02:00
$p1 = __ (
'User %s is reporting an issue in its %s experience' ,
$email_from ,
$product_name
);
$p1 .= ':' ;
2019-10-24 17:42:37 +02:00
2019-10-25 10:43:03 +02:00
$p2 = $text ;
2019-10-24 17:42:37 +02:00
2019-10-25 10:43:03 +02:00
if ( $attachment === 1 ) {
$msg_attch = __ ( 'Find some files attached to this mail' );
$msg_attch .= '. ' ;
$msg_attch .= __ (
'PDF is the diagnostic information retrieved at report time'
2019-10-24 17:42:37 +02:00
);
2019-10-25 10:43:03 +02:00
$msg_attch .= '. ' ;
$msg_attch .= __ ( 'CSV contains the statuses of every product file' );
$msg_attch .= '. ' ;
}
2019-10-24 17:42:37 +02:00
2019-10-25 10:43:03 +02:00
$p3 = __ (
'If you think this report must be escalated, feel free to forward this mail to "%s"' ,
$mail_feedback
);
2019-10-24 17:42:37 +02:00
2019-10-25 10:43:03 +02:00
$legal = __ ( 'LEGAL WARNING' );
$legal1 = __ (
'The information contained in this transmission is privileged and confidential information intended only for the use of the individual or entity named above'
);
$legal1 .= '. ' ;
$legal2 = __ (
'If the reader of this message is not the intended recipient, you are hereby notified that any dissemination, distribution or copying of this communication is strictly prohibited'
);
$legal2 .= '. ' ;
$legal3 = __ (
'If you have received this transmission in error, do not read it'
);
$legal3 .= '. ' ;
$legal4 = __ (
'Please immediately reply to the sender that you have received this communication in error and then delete it'
);
$legal4 .= '.' ;
$patterns = [
'/__title__/' ,
'/__p1__/' ,
'/__p2__/' ,
'/__attachment__/' ,
'/__p3__/' ,
'/__legal__/' ,
'/__legal1__/' ,
'/__legal2__/' ,
'/__legal3__/' ,
'/__legal4__/' ,
];
2019-10-24 17:42:37 +02:00
2019-10-25 10:43:03 +02:00
$substitutions = [
$title ,
$p1 ,
$p2 ,
$msg_attch ,
$p3 ,
$legal ,
$legal1 ,
$legal2 ,
$legal3 ,
$legal4 ,
];
2019-10-24 17:42:37 +02:00
2019-10-25 10:43:03 +02:00
$html_template = file_get_contents (
$config [ 'homedir' ] . '/include/templates/feedback_send_mail.html'
);
$text = preg_replace ( $patterns , $substitutions , $html_template );
2019-10-21 18:11:04 +02:00
$idUserTask = db_get_value (
'id' ,
'tuser_task' ,
'function_name' ,
'cron_task_feedback_send_mail'
);
2019-10-23 19:50:57 +02:00
// Params for send mail with cron.
2019-10-21 18:11:04 +02:00
$parameters = [
0 => '0' ,
1 => $email ,
2 => $subject ,
3 => $text ,
2019-10-23 19:50:57 +02:00
4 => $attachment ,
2019-10-21 18:11:04 +02:00
'first_execution' => strtotime ( 'now' ),
];
2019-10-23 19:50:57 +02:00
// Values insert task cron.
2019-10-21 18:11:04 +02:00
$values = [
'id_usuario' => $config [ 'id_user' ],
'id_user_task' => $idUserTask ,
'args' => serialize ( $parameters ),
'scheduled' => 'no' ,
'id_grupo' => 0 ,
];
2019-10-23 19:50:57 +02:00
$result = db_process_sql_insert (
'tuser_task_scheduled' ,
$values
);
$error = 1 ;
if ( $result === false ) {
$error = 0 ;
}
$return = [
'error' => $error ,
'title' => [
__ ( 'Failed' ),
__ ( 'Success' ),
],
'text' => [
2019-10-25 10:43:03 +02:00
ui_print_error_message ( __ ( 'Invalid cron task' ), '' , true ),
2020-06-29 10:57:19 +02:00
ui_print_success_message ( __ ( 'Sending of information has been processed' ), '' , true ),
2019-10-23 19:50:57 +02:00
],
];
exit ( json_encode ( $return ));
2019-10-17 15:17:28 +02:00
}
/**
2019-10-21 18:11:04 +02:00
* Print Diagnostics PDF report .
2019-10-17 15:17:28 +02:00
*
2019-10-23 19:50:57 +02:00
* @ param string | null $filename Filename .
*
2019-11-27 15:22:39 +01:00
* @ return mixed
2019-10-17 15:17:28 +02:00
*/
2019-11-27 15:22:39 +01:00
public function exportPDF ( ? string $filename = null )
2019-10-17 15:17:28 +02:00
{
2019-10-18 15:23:23 +02:00
global $config ;
2019-10-17 15:17:28 +02:00
2019-10-23 19:50:57 +02:00
$this -> pdf = true ;
2019-10-18 15:23:23 +02:00
enterprise_include_once ( '/include/class/Pdf.class.php' );
2019-10-23 19:50:57 +02:00
$mpdf = new Pdf ([]);
2019-11-21 18:21:46 +01:00
// Ignore pending HTML outputs.
while ( @ ob_end_clean ()) {
$ignore_me ;
}
2019-10-23 19:50:57 +02:00
// ADD style.
$mpdf -> addStyle ( $config [ 'homedir' ] . '/include/styles/diagnostics.css' );
// ADD Metadata.
2019-10-21 18:11:04 +02:00
$product_name = io_safe_output ( get_product_name ());
2019-10-23 19:50:57 +02:00
$mpdf -> setMetadata (
2019-10-21 18:11:04 +02:00
__ ( 'Diagnostics Info' ),
$product_name . ' Enteprise' ,
$product_name ,
__ (
'Automated %s report for user defined report' ,
$product_name
)
);
2019-10-23 19:50:57 +02:00
// ADD Header.
$mpdf -> setHeaderHTML ( __ ( 'Diagnostics Info' ));
// ADD content to report.
$mpdf -> addHTML (
$this -> printMethodsDiagnostigsInfo ()
2019-10-21 18:11:04 +02:00
);
2019-10-23 19:50:57 +02:00
$mpdf -> addHTML (
$this -> printCharts ()
2019-10-21 18:11:04 +02:00
);
2019-10-23 19:50:57 +02:00
// ADD Footer.
$mpdf -> setFooterHTML ();
// Write html filename.
$mpdf -> writePDFfile ( $filename );
2019-11-21 18:21:46 +01:00
2019-11-27 15:22:39 +01:00
return ;
2019-10-21 18:11:04 +02:00
}
2019-10-23 19:50:57 +02:00
/**
* Send Csv md5 files .
*
* @ return string
*/
public function csvMd5Files () : string
{
global $config ;
// Extract files.
$files = $this -> recursiveDirValidation ( $config [ 'homedir' ]);
// Type divider.
$divider = html_entity_decode ( $config [ 'csv_divider' ]);
// BOM.
$result = pack ( 'C*' , 0xEF , 0xBB , 0xBF );
$result .= __ ( 'Path' ) . $divider . __ ( 'MD5' ) . " \n " ;
foreach ( $files as $key => $value ) {
$result .= $key . $divider . $value . " \n " ;
}
return $result ;
}
/**
* Function to return array with name file -> MD %.
*
* @ param string $dir Directory .
*
* @ return array Result all files in directory recursively .
*/
private function recursiveDirValidation ( string $dir ) : array
{
$result = [];
$dir_content = scandir ( $dir );
// Dont check attachment.
if ( strpos ( $dir , $config [ 'homedir' ] . '/attachment' ) === false ) {
if ( is_array ( $dir_content ) === true ) {
foreach ( scandir ( $dir ) as $file ) {
if ( '.' === $file || '..' === $file ) {
continue ;
}
if ( is_dir ( $dir . '/' . $file ) === true ) {
$result += $this -> recursiveDirValidation (
$dir . '/' . $file
);
} else {
$result [ $dir . '/' . $file ] = md5_file ( $dir . '/' . $file );
}
}
}
}
return $result ;
}
2019-10-25 10:43:03 +02:00
/**
* Send PHP info in report .
*
* @ param string $filename Download dir report .
*
* @ return void
*/
public function phpInfoReports ( string $filename )
{
global $config ;
$this -> pdf = true ;
// Extract info PHP.
ob_start ();
phpinfo ();
$php_info = ob_get_clean ();
enterprise_include_once ( '/include/class/Pdf.class.php' );
$mpdf = new Pdf ([]);
// ADD Metadata.
$product_name = io_safe_output ( get_product_name ());
$mpdf -> setMetadata (
__ ( 'PHP Info' ),
$product_name . ' Enteprise' ,
$product_name ,
__ (
'Automated %s report for user defined report' ,
$product_name
)
);
// ADD Header.
$mpdf -> setHeaderHTML ( __ ( 'PHP Info' ));
// ADD content to report.
$mpdf -> addHTML ( $php_info );
// ADD Footer.
$mpdf -> setFooterHTML ();
// Write html filename.
$mpdf -> writePDFfile ( $filename );
}
2019-10-11 14:55:20 +02:00
}