2020-05-20 19:04:42 +02:00
|
|
|
<?php
|
|
|
|
// phpcs:disable Squiz.NamingConventions.ValidFunctionName.ScopeNotCamelCaps
|
|
|
|
/**
|
|
|
|
* Module entity class.
|
|
|
|
*
|
|
|
|
* @category Class
|
|
|
|
* @package Pandora FMS
|
|
|
|
* @subpackage OpenSource
|
|
|
|
* @version 1.0.0
|
|
|
|
* @license See below
|
|
|
|
*
|
|
|
|
* ______ ___ _______ _______ ________
|
|
|
|
* | __ \.-----.--.--.--| |.-----.----.-----. | ___| | | __|
|
|
|
|
* | __/| _ | | _ || _ | _| _ | | ___| |__ |
|
|
|
|
* |___| |___._|__|__|_____||_____|__| |___._| |___| |__|_|__|_______|
|
|
|
|
*
|
|
|
|
* ============================================================================
|
|
|
|
* Copyright (c) 2005-2019 Artica Soluciones Tecnologicas
|
|
|
|
* 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.
|
|
|
|
* ============================================================================
|
|
|
|
*/
|
|
|
|
|
|
|
|
// Begin.
|
|
|
|
namespace PandoraFMS;
|
|
|
|
|
2020-06-26 15:48:58 +02:00
|
|
|
use PandoraFMS\Agent;
|
|
|
|
|
2020-05-20 19:04:42 +02:00
|
|
|
/**
|
|
|
|
* PandoraFMS agent entity.
|
|
|
|
*/
|
|
|
|
class Module extends Entity
|
|
|
|
{
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Module status (From tagente_estado).
|
|
|
|
*
|
|
|
|
* @var PandoraFMS\ModuleStatus
|
|
|
|
*/
|
|
|
|
private $status;
|
|
|
|
|
2020-06-26 15:48:58 +02:00
|
|
|
/**
|
|
|
|
* Agent where module is stored.
|
|
|
|
*
|
|
|
|
* @var PandoraFMS\Agent
|
|
|
|
*/
|
|
|
|
private $linkedAgent;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Metaconsole setup id.
|
|
|
|
*
|
|
|
|
* @var integer
|
|
|
|
*/
|
|
|
|
private $idNode;
|
|
|
|
|
2020-05-20 19:04:42 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Search a module in db.
|
|
|
|
*
|
|
|
|
* @param array $params Search parameters (fields from tagente_modulo).
|
|
|
|
* @param integer $limit Limit results to N rows.
|
|
|
|
*
|
|
|
|
* @return PandoraFMS\Module found or null if not found.
|
|
|
|
* @throws \Exception On error.
|
|
|
|
*/
|
|
|
|
public static function search(array $params, ?int $limit=0)
|
|
|
|
{
|
|
|
|
if (empty($params) === true) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
$rs = \db_get_all_rows_filter(
|
|
|
|
'tagente_modulo',
|
|
|
|
$params
|
|
|
|
);
|
|
|
|
|
|
|
|
if ($rs === false) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (empty($rs) === true) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ($limit !== 1) {
|
|
|
|
$modules = [];
|
|
|
|
$i = 0;
|
|
|
|
foreach ($rs as $row) {
|
|
|
|
if ($limit > 1 && (++$i) > $limit) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
$modules[] = self::build($row);
|
|
|
|
}
|
|
|
|
|
|
|
|
return $modules;
|
|
|
|
} else {
|
|
|
|
return self::build($rs[0]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns current object as array.
|
|
|
|
*
|
|
|
|
* @return array Of fields.
|
|
|
|
*/
|
|
|
|
public function toArray()
|
|
|
|
{
|
|
|
|
return $this->fields;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Creates a module object from given data. Avoid query duplication.
|
|
|
|
*
|
|
|
|
* @param array $data Module information.
|
|
|
|
*
|
|
|
|
* @return PandoraFMS\Module Object.
|
|
|
|
*/
|
|
|
|
public static function build(array $data=[])
|
|
|
|
{
|
|
|
|
$obj = new Module();
|
|
|
|
|
|
|
|
// Set values.
|
|
|
|
foreach ($data as $k => $v) {
|
|
|
|
$obj->{$k}($v);
|
|
|
|
}
|
|
|
|
|
|
|
|
if ($obj->nombre() === 'delete_pending') {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Customize certain fields.
|
|
|
|
$obj->status = new ModuleStatus($obj->id_agente_modulo());
|
|
|
|
|
|
|
|
return $obj;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Builds a PandoraFMS\Module object from given id.
|
|
|
|
*
|
2020-06-26 15:48:58 +02:00
|
|
|
* @param integer|null $id_agent_module Module id.
|
|
|
|
* @param boolean $link_agent Link agent object.
|
|
|
|
* @param integer|null $id_node Metaconsole only. ID node.
|
|
|
|
*
|
|
|
|
* @throws \Exception On error.
|
2020-05-20 19:04:42 +02:00
|
|
|
*/
|
2020-06-26 15:48:58 +02:00
|
|
|
public function __construct(
|
|
|
|
?int $id_agent_module=null,
|
|
|
|
bool $link_agent=false,
|
|
|
|
?int $id_node=null
|
|
|
|
) {
|
|
|
|
if ($id_node !== null) {
|
|
|
|
$this->idNode = $id_node;
|
|
|
|
|
|
|
|
enterprise_include_once('include/functions_metaconsole.php');
|
|
|
|
\metaconsole_connect(null, $this->idNode);
|
|
|
|
}
|
|
|
|
|
2020-05-20 19:04:42 +02:00
|
|
|
if (is_numeric($id_agent_module) === true
|
|
|
|
&& $id_agent_module > 0
|
|
|
|
) {
|
|
|
|
parent::__construct(
|
|
|
|
'tagente_modulo',
|
|
|
|
['id_agente_modulo' => $id_agent_module]
|
|
|
|
);
|
2020-06-26 15:48:58 +02:00
|
|
|
|
|
|
|
if ($this->nombre() === 'delete_pending') {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ($link_agent === true) {
|
|
|
|
try {
|
|
|
|
$this->linkedAgent = new Agent($this->id_agente());
|
|
|
|
} catch (\Exception $e) {
|
|
|
|
if ($this->idNode !== null) {
|
|
|
|
\metaconsole_restore_db();
|
|
|
|
}
|
|
|
|
|
|
|
|
// Unexistent agent.
|
|
|
|
throw new \Exception(
|
|
|
|
__METHOD__.__(
|
|
|
|
' error: Module has no agent assigned.'
|
|
|
|
)
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
2020-05-20 19:04:42 +02:00
|
|
|
} else {
|
|
|
|
// Create empty skel.
|
|
|
|
parent::__construct('tagente_modulo');
|
|
|
|
}
|
|
|
|
|
2020-06-26 15:48:58 +02:00
|
|
|
try {
|
|
|
|
// Customize certain fields.
|
|
|
|
$this->status = new ModuleStatus($this->fields['id_agente_modulo']);
|
|
|
|
} catch (\Exception $e) {
|
|
|
|
$this->status = new Modulestatus();
|
|
|
|
if ($this->idNode !== null) {
|
|
|
|
\metaconsole_restore_db();
|
|
|
|
}
|
2020-05-20 19:04:42 +02:00
|
|
|
}
|
2020-06-26 15:48:58 +02:00
|
|
|
}
|
2020-05-20 19:04:42 +02:00
|
|
|
|
2020-06-26 15:48:58 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Return agent object where module is defined.
|
|
|
|
*
|
|
|
|
* @return PandoraFMS\Agent Where module is defined.
|
|
|
|
*/
|
|
|
|
public function agent()
|
|
|
|
{
|
|
|
|
if ($this->linkedAgent === null) {
|
|
|
|
try {
|
|
|
|
$this->linkedAgent = new Agent($this->id_agente());
|
|
|
|
} catch (\Exception $e) {
|
|
|
|
// Unexistent agent.
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return $this->linkedAgent;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Return last value reported by the module.
|
|
|
|
*
|
|
|
|
* @return mixed Data depending on module type.
|
|
|
|
*/
|
|
|
|
public function lastValue()
|
|
|
|
{
|
|
|
|
return $this->status->datos();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Return last status reported by the module.
|
|
|
|
*
|
|
|
|
* @return mixed Data depending on module type.
|
|
|
|
*/
|
|
|
|
public function lastStatus()
|
|
|
|
{
|
|
|
|
return $this->status->estado();
|
2020-05-20 19:04:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns current status.
|
|
|
|
*
|
|
|
|
* @return PandoraFMS\ModuleStatus Status of the module.
|
|
|
|
*/
|
|
|
|
public function getStatus()
|
|
|
|
{
|
|
|
|
return $this->status;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2020-06-24 18:01:34 +02:00
|
|
|
/**
|
|
|
|
* Retrieve all alert templates (ids) assigned to current module.
|
|
|
|
*
|
|
|
|
* @return array Of ids.
|
|
|
|
*/
|
|
|
|
public function alertTemplatesAssigned()
|
|
|
|
{
|
|
|
|
if ($this->id_agente_modulo() === null) {
|
|
|
|
// Need to be stored first.
|
|
|
|
return [];
|
|
|
|
}
|
|
|
|
|
|
|
|
$result = db_get_all_rows_filter(
|
|
|
|
'talert_template_modules',
|
|
|
|
['id_agent_module' => $this->id_agente_modulo()],
|
|
|
|
'id_alert_template'
|
|
|
|
);
|
|
|
|
|
|
|
|
if ($result === false) {
|
|
|
|
return [];
|
|
|
|
}
|
|
|
|
|
|
|
|
return array_reduce(
|
|
|
|
$result,
|
|
|
|
function ($carry, $item) {
|
|
|
|
$carry[] = $item['id_alert_template'];
|
|
|
|
return $carry;
|
|
|
|
},
|
|
|
|
[]
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Remove a alert template assignment.
|
|
|
|
*
|
|
|
|
* @param integer $id_alert_template Target id.
|
|
|
|
*
|
|
|
|
* @return boolean Success or not.
|
|
|
|
*/
|
|
|
|
public function unassignAlertTemplate(int $id_alert_template)
|
|
|
|
{
|
|
|
|
if ($this->id_agente_modulo() === null) {
|
|
|
|
// Need to be stored first.
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (is_numeric($id_alert_template) === false
|
|
|
|
|| $id_alert_template <= 0
|
|
|
|
) {
|
|
|
|
// Invalid alert template.
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return (bool) \db_process_sql_delete(
|
|
|
|
'talert_template_modules',
|
|
|
|
[
|
|
|
|
'id_agent_module' => $this->id_agente_modulo(),
|
|
|
|
'id_alert_template' => $id_alert_template,
|
|
|
|
]
|
|
|
|
);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Add an alert template to this module.
|
|
|
|
*
|
|
|
|
* @param integer|null $id_alert_template Target alert template.
|
|
|
|
*
|
|
|
|
* @return boolean Status of adding process.
|
|
|
|
*/
|
|
|
|
public function addAlertTemplate(?int $id_alert_template=null)
|
|
|
|
{
|
|
|
|
if ($this->id_agente_modulo() === null) {
|
|
|
|
// Need to be stored first.
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (is_numeric($id_alert_template) === false
|
|
|
|
|| $id_alert_template <= 0
|
|
|
|
) {
|
|
|
|
// Invalid alert template.
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return (bool) \db_process_sql_insert(
|
|
|
|
'talert_template_modules',
|
|
|
|
[
|
|
|
|
'id_agent_module' => $this->id_agente_modulo(),
|
|
|
|
'id_alert_template' => $id_alert_template,
|
|
|
|
'last_reference' => time(),
|
|
|
|
]
|
|
|
|
);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2020-05-20 19:04:42 +02:00
|
|
|
/**
|
|
|
|
* Saves current definition to database.
|
|
|
|
*
|
|
|
|
* @return mixed Affected rows of false in case of error.
|
|
|
|
* @throws \Exception On error.
|
|
|
|
*/
|
|
|
|
public function save()
|
|
|
|
{
|
|
|
|
if (empty($this->fields['nombre']) === true) {
|
|
|
|
throw new \Exception(
|
|
|
|
get_class($this).' error, "nombre" is not set'
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (empty($this->fields['id_agente']) === true) {
|
|
|
|
throw new \Exception(
|
|
|
|
get_class($this).' error, "id_agente" is not set'
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
if ($this->fields['id_agente_modulo'] > 0) {
|
|
|
|
// Update.
|
|
|
|
$updates = $this->fields;
|
2020-06-26 15:48:58 +02:00
|
|
|
if ($this->idNode !== null) {
|
|
|
|
enterprise_include_once('include/functions_metaconsole.php');
|
|
|
|
\metaconsole_connect(null, $this->idNode);
|
|
|
|
}
|
2020-05-20 19:04:42 +02:00
|
|
|
|
|
|
|
$rs = \db_process_sql_update(
|
|
|
|
'tagente_modulo',
|
|
|
|
$updates,
|
|
|
|
['id_agente_modulo' => $this->fields['id_agente_modulo']]
|
|
|
|
);
|
|
|
|
|
2020-06-26 15:48:58 +02:00
|
|
|
global $config;
|
|
|
|
$error = $config['dbconnection']->error;
|
|
|
|
|
|
|
|
if ($this->idNode !== null) {
|
|
|
|
\metaconsole_restore_db();
|
|
|
|
}
|
|
|
|
|
2020-05-20 19:04:42 +02:00
|
|
|
if ($rs === false) {
|
|
|
|
throw new \Exception(
|
2020-06-26 15:48:58 +02:00
|
|
|
__METHOD__.' error: '.$error
|
2020-05-20 19:04:42 +02:00
|
|
|
);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
// Creation.
|
|
|
|
$updates = $this->fields;
|
|
|
|
|
|
|
|
// Clean null fields.
|
|
|
|
foreach ($updates as $k => $v) {
|
|
|
|
if ($v === null) {
|
|
|
|
unset($updates[$k]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-06-26 15:48:58 +02:00
|
|
|
if ($this->idNode !== null) {
|
|
|
|
enterprise_include_once('include/functions_metaconsole.php');
|
|
|
|
\metaconsole_connect(null, $this->idNode);
|
|
|
|
}
|
|
|
|
|
2020-05-20 19:04:42 +02:00
|
|
|
$rs = \modules_create_agent_module(
|
|
|
|
$this->fields['id_agente'],
|
|
|
|
$updates['nombre'],
|
|
|
|
$updates
|
|
|
|
);
|
|
|
|
|
2020-06-26 15:48:58 +02:00
|
|
|
global $config;
|
|
|
|
$error = $config['dbconnection']->error;
|
|
|
|
|
|
|
|
if ($this->idNode !== null) {
|
|
|
|
\metaconsole_restore_db();
|
|
|
|
}
|
|
|
|
|
2020-05-20 19:04:42 +02:00
|
|
|
if ($rs === false) {
|
|
|
|
throw new \Exception(
|
2020-06-26 15:48:58 +02:00
|
|
|
__METHOD__.' error: '.$error
|
2020-05-20 19:04:42 +02:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
$this->fields['id_agente_modulo'] = $rs;
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Erases this module.
|
|
|
|
*
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
public function delete()
|
|
|
|
{
|
2020-06-26 15:48:58 +02:00
|
|
|
if ($this->idNode !== null) {
|
|
|
|
enterprise_include_once('include/functions_metaconsole.php');
|
|
|
|
\metaconsole_connect(null, $this->idNode);
|
|
|
|
}
|
|
|
|
|
2020-05-20 19:04:42 +02:00
|
|
|
\modules_delete_agent_module(
|
|
|
|
$this->id_agente_modulo()
|
|
|
|
);
|
|
|
|
|
2020-06-26 15:48:58 +02:00
|
|
|
if ($this->idNode !== null) {
|
|
|
|
\metaconsole_restore_db();
|
|
|
|
}
|
|
|
|
|
2020-05-20 19:04:42 +02:00
|
|
|
unset($this->fields);
|
|
|
|
unset($this->status);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2020-06-24 18:01:34 +02:00
|
|
|
/**
|
|
|
|
* Transforms results from classic mode into modern exceptions.
|
|
|
|
*
|
|
|
|
* @param integer|boolean $result Result received from module management.
|
|
|
|
*
|
|
|
|
* @return integer Module id created or result.
|
|
|
|
* @throws \Exception On error.
|
|
|
|
*/
|
|
|
|
public static function errorToException($result)
|
|
|
|
{
|
|
|
|
if ($result === ERR_INCOMPLETE) {
|
|
|
|
throw new \Exception(
|
|
|
|
__('Module name empty.')
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
if ($result === ERR_GENERIC) {
|
|
|
|
throw new \Exception(
|
|
|
|
__('Invalid characters in module name')
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
if ($result === ERR_EXIST) {
|
|
|
|
throw new \Exception(
|
|
|
|
__('Module already exists please select another name or agent.')
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
if ($result === false) {
|
|
|
|
throw new \Exception(
|
|
|
|
__('Insufficent permissions to perform this action')
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
if ($result === ERR_DB) {
|
|
|
|
global $config;
|
|
|
|
throw new \Exception(
|
|
|
|
__('Error while processing: %s', $config['dbconnection']->error)
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
return $result;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2020-06-26 15:48:58 +02:00
|
|
|
/**
|
|
|
|
* Calculates CPS value and updates its content.
|
|
|
|
*
|
|
|
|
* @return integer CPS value set.
|
|
|
|
*/
|
|
|
|
public function calculateCPS()
|
|
|
|
{
|
|
|
|
$cps = 0;
|
|
|
|
|
|
|
|
if (class_exists('\PandoraFMS\Enterprise\Service') === false) {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check if is child of services in local console.
|
|
|
|
$service_parents = \db_get_all_rows_filter(
|
|
|
|
'tservice_element',
|
|
|
|
[ 'id_agente_modulo' => $this->id_agente_modulo() ],
|
|
|
|
'id_service'
|
|
|
|
);
|
|
|
|
|
|
|
|
if (is_array($service_parents) === true) {
|
|
|
|
foreach ($service_parents as $ref) {
|
|
|
|
$service = new \PandoraFMS\Enterprise\Service(
|
|
|
|
$ref['id_service']
|
|
|
|
);
|
|
|
|
if (((bool) $service->cascade_protection()) === true) {
|
|
|
|
$cps += $service->cps();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check if is child of services in metaconsole.
|
|
|
|
if (is_metaconsole() === false) {
|
|
|
|
global $config;
|
|
|
|
|
|
|
|
$nodo_connect = \metaconsole_load_external_db(
|
|
|
|
[
|
|
|
|
'dbhost' => $config['replication_dbhost'],
|
|
|
|
'dbuser' => $config['replication_dbuser'],
|
|
|
|
'dbpass' => \io_output_password(
|
|
|
|
$config['replication_dbpass']
|
|
|
|
),
|
|
|
|
'dbname' => $config['replication_dbname'],
|
|
|
|
]
|
|
|
|
);
|
|
|
|
|
|
|
|
\metaconsole_restore_db();
|
|
|
|
}
|
|
|
|
|
|
|
|
return $cps;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2020-05-20 19:04:42 +02:00
|
|
|
}
|