2020-05-14 17:49:02 +02:00
|
|
|
<?php
|
|
|
|
/**
|
|
|
|
* Entity class.
|
|
|
|
*
|
|
|
|
* @category Abstract class
|
|
|
|
* @package Pandora FMS
|
|
|
|
* @subpackage OpenSource
|
|
|
|
* @version 1.0.0
|
|
|
|
* @license See below
|
|
|
|
*
|
|
|
|
* ______ ___ _______ _______ ________
|
|
|
|
* | __ \.-----.--.--.--| |.-----.----.-----. | ___| | | __|
|
|
|
|
* | __/| _ | | _ || _ | _| _ | | ___| |__ |
|
|
|
|
* |___| |___._|__|__|_____||_____|__| |___._| |___| |__|_|__|_______|
|
|
|
|
*
|
|
|
|
* ============================================================================
|
2020-11-27 13:52:35 +01:00
|
|
|
* Copyright (c) 2005-2021 Artica Soluciones Tecnologicas
|
2020-05-14 17:49:02 +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.
|
|
|
|
* ============================================================================
|
|
|
|
*/
|
|
|
|
|
|
|
|
// Begin.
|
|
|
|
namespace PandoraFMS;
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Defines common methods for all PandoraFMS entity objects.
|
|
|
|
*/
|
|
|
|
abstract class Entity
|
|
|
|
{
|
|
|
|
|
2021-10-06 19:52:46 +02:00
|
|
|
/**
|
|
|
|
* Load from DB or new one.
|
|
|
|
*
|
|
|
|
* @var boolean
|
|
|
|
*/
|
|
|
|
protected $existsInDB;
|
|
|
|
|
2021-12-09 14:57:37 +01:00
|
|
|
/**
|
|
|
|
* Fields to identify register.
|
|
|
|
*
|
|
|
|
* @var array
|
|
|
|
*/
|
|
|
|
protected $primaryKeys;
|
|
|
|
|
2020-05-14 17:49:02 +02:00
|
|
|
/**
|
|
|
|
* Entity fields (from table).
|
|
|
|
*
|
|
|
|
* @var array
|
|
|
|
*/
|
|
|
|
protected $fields = [];
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Target table.
|
|
|
|
*
|
|
|
|
* @var string
|
|
|
|
*/
|
|
|
|
protected $table = '';
|
|
|
|
|
2020-07-10 13:38:42 +02:00
|
|
|
/**
|
|
|
|
* Enterprise capabilities object.
|
|
|
|
*
|
|
|
|
* @var object
|
|
|
|
*/
|
|
|
|
private $enterprise;
|
|
|
|
|
2021-02-03 14:21:59 +01:00
|
|
|
/**
|
|
|
|
* MC Node id.
|
|
|
|
*
|
|
|
|
* @var integer|null
|
|
|
|
*/
|
|
|
|
protected $nodeId = null;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Connected to external node.
|
|
|
|
*
|
|
|
|
* @var boolean
|
|
|
|
*/
|
|
|
|
private $connected = false;
|
|
|
|
|
2020-05-14 17:49:02 +02:00
|
|
|
|
2020-09-16 15:54:43 +02:00
|
|
|
/**
|
|
|
|
* Instances a new object using array definition.
|
|
|
|
*
|
|
|
|
* @param array $data Fields data.
|
2021-01-19 18:00:28 +01:00
|
|
|
* @param string $class_str Class name.
|
2020-09-16 15:54:43 +02:00
|
|
|
*
|
|
|
|
* @return object With current definition.
|
|
|
|
*/
|
2021-01-19 18:00:28 +01:00
|
|
|
public static function build(array $data=[], string $class_str=__CLASS__)
|
2020-09-16 15:54:43 +02:00
|
|
|
{
|
|
|
|
$obj = new $class_str();
|
|
|
|
// Set values.
|
|
|
|
foreach ($data as $k => $v) {
|
|
|
|
$obj->{$k}($v);
|
|
|
|
}
|
|
|
|
|
2021-10-06 19:52:46 +02:00
|
|
|
$obj->existsInDB = true;
|
2020-09-16 15:54:43 +02:00
|
|
|
return $obj;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2021-11-15 19:29:37 +01:00
|
|
|
/**
|
|
|
|
* Duplicates an object.
|
|
|
|
*
|
|
|
|
* @param array $field_exceptions Fields to skip.
|
|
|
|
* @param string $class_str Class name.
|
|
|
|
*
|
|
|
|
* @return object
|
|
|
|
*/
|
|
|
|
public function duplicate(
|
|
|
|
array $field_exceptions=[],
|
|
|
|
string $class_str=__CLASS__
|
|
|
|
) {
|
|
|
|
$keys = array_keys($this->toArray());
|
|
|
|
|
|
|
|
$new = new $class_str();
|
|
|
|
foreach ($keys as $k) {
|
|
|
|
if (in_array($k, $field_exceptions) === false) {
|
|
|
|
$new->$k($this->$k());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return $new;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2020-05-14 17:49:02 +02:00
|
|
|
/**
|
|
|
|
* Defines a generic constructor to extract information of the object.
|
|
|
|
*
|
2020-07-13 13:56:41 +02:00
|
|
|
* @param string $table Table.
|
|
|
|
* @param array|null $filters Filters, for instance ['id' => $id].
|
|
|
|
* @param string|null $enterprise_class Enterprise class name.
|
2021-09-30 12:23:37 +02:00
|
|
|
* @param boolean $cache Use cache or not.
|
2020-05-14 17:49:02 +02:00
|
|
|
*
|
|
|
|
* @throws \Exception On error.
|
|
|
|
*/
|
2020-07-10 13:38:42 +02:00
|
|
|
public function __construct(
|
|
|
|
string $table,
|
|
|
|
?array $filters=null,
|
2021-09-30 12:23:37 +02:00
|
|
|
?string $enterprise_class=null,
|
|
|
|
bool $cache=true
|
2020-07-10 13:38:42 +02:00
|
|
|
) {
|
2020-05-14 17:49:02 +02:00
|
|
|
if (empty($table) === true) {
|
|
|
|
throw new \Exception(
|
|
|
|
get_class($this).' error, table name is not defined'
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
$this->table = $table;
|
|
|
|
|
|
|
|
if (is_array($filters) === true) {
|
|
|
|
// New one.
|
2021-12-09 14:57:37 +01:00
|
|
|
$this->primaryKeys = array_keys($filters);
|
|
|
|
|
2021-09-30 12:23:37 +02:00
|
|
|
$data = \db_get_row_filter(
|
|
|
|
$this->table,
|
|
|
|
$filters,
|
|
|
|
false,
|
|
|
|
'AND',
|
|
|
|
false,
|
|
|
|
$cache
|
|
|
|
);
|
2020-05-14 17:49:02 +02:00
|
|
|
|
|
|
|
if ($data === false) {
|
|
|
|
throw new \Exception(
|
|
|
|
get_class($this).' error, entity not found'
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Map fields.
|
|
|
|
foreach ($data as $k => $v) {
|
|
|
|
$this->fields[$k] = $v;
|
|
|
|
}
|
2021-10-06 19:52:46 +02:00
|
|
|
|
|
|
|
// Mark as existing object.
|
|
|
|
$this->existsInDB = true;
|
2020-05-14 17:49:02 +02:00
|
|
|
} else {
|
|
|
|
// Empty one.
|
|
|
|
$data = \db_get_all_rows_sql(
|
|
|
|
sprintf(
|
|
|
|
'SHOW COLUMNS FROM %s',
|
|
|
|
$this->table
|
|
|
|
)
|
|
|
|
);
|
|
|
|
|
|
|
|
foreach ($data as $row) {
|
|
|
|
$this->fields[$row['Field']] = null;
|
|
|
|
}
|
2021-10-06 19:52:46 +02:00
|
|
|
|
|
|
|
// Mark as virtual object.
|
|
|
|
$this->existsInDB = false;
|
2020-05-14 17:49:02 +02:00
|
|
|
}
|
2020-07-10 13:38:42 +02:00
|
|
|
|
|
|
|
if (\enterprise_installed() === true
|
|
|
|
&& $enterprise_class !== null
|
|
|
|
) {
|
|
|
|
$this->enterprise = new $enterprise_class($this);
|
|
|
|
}
|
2020-05-14 17:49:02 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Dynamically call methods in this object.
|
|
|
|
*
|
2021-09-16 21:16:49 +02:00
|
|
|
* To dynamically switch between community methods and prioritize
|
|
|
|
* enterprise ones, define method visibility as 'protected' in both
|
|
|
|
* classes.
|
|
|
|
*
|
|
|
|
* For instance, in following situation:
|
|
|
|
* protected PandoraFMS\Agent::test()
|
|
|
|
* protected PandoraFMS\Enterprise\Agent::test()
|
|
|
|
*
|
|
|
|
* If enterprise is available, then PandoraFMS\Enterprise\Agent::test()
|
|
|
|
* will be executed, community method otherwise.
|
|
|
|
*
|
2020-05-14 17:49:02 +02:00
|
|
|
* @param string $methodName Name of target method or attribute.
|
|
|
|
* @param array $params Arguments for target method.
|
|
|
|
*
|
|
|
|
* @return mixed Return of method.
|
|
|
|
* @throws \Exception On error.
|
|
|
|
*/
|
|
|
|
public function __call(string $methodName, ?array $params=null)
|
|
|
|
{
|
2020-07-10 13:38:42 +02:00
|
|
|
// Enterprise capabilities.
|
2021-09-16 21:16:49 +02:00
|
|
|
// Prioritize enterprise written methods over dynamic fields.
|
2020-07-10 13:38:42 +02:00
|
|
|
if (\enterprise_installed() === true
|
|
|
|
&& $this->enterprise !== null
|
|
|
|
&& method_exists($this->enterprise, $methodName) === true
|
|
|
|
) {
|
2021-09-16 21:54:12 +02:00
|
|
|
return $this->enterprise->$methodName(...$params);
|
2020-07-10 13:38:42 +02:00
|
|
|
}
|
|
|
|
|
2021-09-16 21:16:49 +02:00
|
|
|
if (method_exists($this, $methodName) === false) {
|
|
|
|
if (array_key_exists($methodName, $this->fields) === true) {
|
|
|
|
if (empty($params) === true) {
|
|
|
|
return $this->fields[$methodName];
|
|
|
|
} else {
|
|
|
|
$this->fields[$methodName] = $params[0];
|
|
|
|
}
|
|
|
|
|
|
|
|
return null;
|
2020-05-14 17:49:02 +02:00
|
|
|
}
|
|
|
|
|
2021-09-16 21:16:49 +02:00
|
|
|
throw new \Exception(
|
|
|
|
get_class($this).' error, method '.$methodName.' does not exist'
|
|
|
|
);
|
2020-05-14 17:49:02 +02:00
|
|
|
}
|
|
|
|
|
2021-09-16 21:16:49 +02:00
|
|
|
// Do not return nor throw exceptions after this point, allow php
|
|
|
|
// default __call behaviour to continue working with object method
|
|
|
|
// defined.
|
2021-09-16 21:57:23 +02:00
|
|
|
// If you're receiving NULL as result of the method invocation, ensure
|
|
|
|
// it is not private, take in mind this method will mask any access
|
|
|
|
// level error or notification since it is public and has limited access
|
|
|
|
// to the object (public|protected).
|
2020-05-14 17:49:02 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2020-05-22 12:14:01 +02:00
|
|
|
/**
|
|
|
|
* Returns current object as array.
|
|
|
|
*
|
|
|
|
* @return array Of fields.
|
|
|
|
*/
|
|
|
|
public function toArray()
|
|
|
|
{
|
|
|
|
return $this->fields;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2021-02-03 14:21:59 +01:00
|
|
|
/**
|
|
|
|
* Connects to current nodeId target.
|
|
|
|
* If no nodeId is defined, then returns without doing anything.
|
|
|
|
*
|
|
|
|
* @return void
|
|
|
|
* @throws \Exception On error.
|
|
|
|
*/
|
|
|
|
public function connectNode()
|
|
|
|
{
|
|
|
|
if ($this->nodeId === null) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
\enterprise_include_once('include/functions_metaconsole.php');
|
|
|
|
$r = \enterprise_hook(
|
|
|
|
'metaconsole_connect',
|
|
|
|
[
|
|
|
|
null,
|
|
|
|
$this->nodeId,
|
|
|
|
]
|
|
|
|
);
|
|
|
|
|
|
|
|
if ($r !== NOERR) {
|
|
|
|
throw new \Exception(
|
|
|
|
__('Cannot connect to node %d', $this->nodeId)
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
$this->connected = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Restore connection after connectNode.
|
|
|
|
*
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
public function restoreConnection()
|
|
|
|
{
|
|
|
|
if ($this->connected === true) {
|
|
|
|
\enterprise_include_once('include/functions_metaconsole.php');
|
|
|
|
\enterprise_hook('metaconsole_restore_db');
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2020-05-14 17:49:02 +02:00
|
|
|
/**
|
|
|
|
* Saves current object definition to database.
|
|
|
|
*
|
|
|
|
* @return boolean Success or not.
|
2021-12-09 14:57:37 +01:00
|
|
|
* @throws \Exception On error.
|
|
|
|
*/
|
|
|
|
public function save()
|
|
|
|
{
|
|
|
|
$updates = $this->fields;
|
|
|
|
// Clean null fields.
|
|
|
|
foreach ($updates as $k => $v) {
|
|
|
|
if ($v === null) {
|
|
|
|
unset($updates[$k]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if ($this->existsInDB === true) {
|
|
|
|
// Update.
|
|
|
|
$where = [];
|
|
|
|
|
|
|
|
foreach ($this->primaryKeys as $key) {
|
|
|
|
$where[$key] = $this->fields[$key];
|
|
|
|
}
|
|
|
|
|
|
|
|
if (empty($where) === true) {
|
|
|
|
throw new \Exception(
|
|
|
|
__METHOD__.' error: Cannot identify object'
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
$rs = \db_process_sql_update(
|
|
|
|
$this->table,
|
|
|
|
$updates,
|
|
|
|
$where
|
|
|
|
);
|
|
|
|
|
|
|
|
if ($rs === false) {
|
|
|
|
global $config;
|
|
|
|
throw new \Exception(
|
|
|
|
__METHOD__.' error: '.$config['dbconnection']->error
|
|
|
|
);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
// New register.
|
|
|
|
$rs = \db_process_sql_insert(
|
|
|
|
$this->table,
|
|
|
|
$updates
|
|
|
|
);
|
|
|
|
|
|
|
|
if ($rs === false) {
|
|
|
|
global $config;
|
|
|
|
|
|
|
|
throw new \Exception(
|
|
|
|
__METHOD__.' error: '.$config['dbconnection']->error
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
$this->existsInDB = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Remove this entity.
|
|
|
|
*
|
|
|
|
* @return void
|
|
|
|
* @throws \Exception If no primary keys are defined.
|
2020-05-14 17:49:02 +02:00
|
|
|
*/
|
2021-12-09 14:57:37 +01:00
|
|
|
public function delete()
|
|
|
|
{
|
|
|
|
if ($this->existsInDB === true) {
|
|
|
|
$where = [];
|
|
|
|
|
|
|
|
foreach ($this->primaryKeys as $key) {
|
|
|
|
$where[$key] = $this->fields[$key];
|
|
|
|
}
|
|
|
|
|
|
|
|
if (empty($where) === true) {
|
|
|
|
throw new \Exception(
|
|
|
|
__METHOD__.' error: Cannot identify object on deletion'
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
\db_process_sql_delete(
|
|
|
|
$this->table,
|
|
|
|
$where
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
2020-05-14 17:49:02 +02:00
|
|
|
|
|
|
|
|
|
|
|
}
|