pandorafms/pandora_console/include/lib/Module.php

577 lines
14 KiB
PHP
Raw Normal View History

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
}