pandorafms/pandora_console/include/lib/Entity.php

421 lines
10 KiB
PHP
Raw Normal View History

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
*
* ______ ___ _______ _______ ________
2023-06-08 12:42:10 +02:00
* | __ \.-----.--.--.--| |.-----.----.-----. | ___| | | __|
* | __/| _ | | _ || _ | _| _ | | ___| |__ |
2020-05-14 17:49:02 +02:00
* |___| |___._|__|__|_____||_____|__| |___._| |___| |__|_|__|_______|
*
* ============================================================================
2023-06-08 11:53:13 +02:00
* Copyright (c) 2005-2023 Pandora FMS
2023-06-08 13:19:01 +02:00
* Please see https://pandorafms.com/community/ for full contribution list
2020-05-14 17:49:02 +02:00
* This program is free software; you can redistribute it and/or
* modify it under the terms of the GNU General Public License
* as published by the Free Software Foundation 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;
/**
* 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.
* @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,
?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);
$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;
}
/**
* 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
}