0 ) { $cluster_id = db_get_value( 'id', 'tcluster', 'id_agent', $id_agent ); return new self($cluster_id, $load_members); } return null; } /** * Builds a PandoraFMS\ClusterViewer\Cluster object from a cluster id. * * @param integer $id_cluster Cluster Id. * @param boolean $load_members Load members or not. * * @throws \Exception On error. */ public function __construct(?int $id_cluster=null, ?bool $load_members=true) { if (is_numeric($id_cluster) === true && $id_cluster > 0 ) { try { parent::__construct('tcluster', ['id' => $id_cluster]); } catch (\Exception $e) { throw new \Exception('Cluster id not found.'); } if ($load_members === true) { // Retrieve members. $data = \db_get_all_rows_filter( 'tcluster_agent', ['id_cluster' => $id_cluster] ); if (is_array($data) === true) { foreach ($data as $row) { $this->addMember($row['id_agent']); } } } // Retrieve items. $data = \db_get_all_rows_filter( 'tcluster_item', ['id_cluster' => $id_cluster] ); if (is_array($data) === true) { foreach ($data as $row) { if ($row['item_type'] === 'AA') { $this->aaModules[$row['name']] = new ClusterModule( $row['id'] ); } else if ($row['item_type'] === 'AP') { $this->apModules[$row['name']] = new ClusterModule( $row['id'] ); } } } } else { parent::__construct('tcluster'); } // Customize certain fields. try { $this->fields['group'] = new Group($this->group()); } catch (\Exception $e) { $this->fields['group'] = new Group(); } if ($this->id_agent() !== null) { try { $this->fields['agent'] = new Agent($this->id_agent(), true); } catch (\Exception $e) { $this->fields['agent'] = new Agent(); } } else { $this->fields['agent'] = new Agent(); } if ($this->id_agent() !== null) { $this->clusterStatus = Module::search( [ 'nombre' => io_safe_input('Cluster status'), 'id_agente' => $this->id_agent(), ], 1 ); } } /** * Return an array of PandoraFMS\Agents as members of current cluster. * * @return array Of agents. */ public function getMembers() { if (is_array($this->members) === true) { return $this->members; } return []; } /** * Counters modules involved status. * * @return array */ public function getCounters() :array { $id_agent_modules = $this->getIdsModulesInvolved(); if (empty($id_agent_modules) === true) { return []; } $sql = sprintf( 'SELECT SUM( IF(estado = 1, 1, 0) ) AS critical, SUM( IF(estado = 2, 1, 0) ) AS warning, SUM( IF(estado = 0, 1, 0) ) AS normal, SUM( IF(estado = 3, 1, 0) ) AS unknown, SUM( IF(estado = 4 OR estado = 5, 1, 0) ) AS not_init, COUNT(id_agente_modulo) AS total FROM tagente_estado WHERE id_agente_modulo IN (%s)', implode(',', $id_agent_modules) ); $counters = db_get_row_sql($sql); if ($counters === false) { $counters = []; } return $counters; } /** * Return Ids modules involved. * * @return array */ public function getIdsModulesInvolved(): array { $members = $this->getMembers(); $modules_ids = []; $modules_names = $this->getItemNames(); if (empty($members) === false) { foreach ($members as $agent) { $modules_filtered = $agent->searchModules(['nombre' => $modules_names], 0); if (empty($modules_filtered) === false) { foreach ($modules_filtered as $idAgent => $module) { $modules_ids[] = $module->id_agente_modulo(); } } } } return $modules_ids; } /** * Return names modules involved. * * @return array */ public function getItemNames(): array { $result = []; if (empty($this->getItems()) === false) { $result = array_keys($this->getItems()); } return $result; } /** * Return name type. * * @return string */ public function getStringTypeName(): string { $result = __('Active').' / '.__('Active'); if ($this->cluster_type() === 'AP') { $result = __('Active').' / '.__('Pasive'); } return $result; } /** * Cleans members from cluster object. * * @return void */ public function cleanMembers() { unset($this->members); } /** * Register a new agent in the cluster. * * @param integer $id_agent New id_agent to be added. * * @return mixed * @throws \Exception On error. */ public function addMember(int $id_agent) { if (isset($this->members[$id_agent]) === true) { // Already joining. return; } try { $agent = new Agent($id_agent); } catch (\Exception $e) { return; } if ($agent->id_agente() === null) { throw new \Exception('Invalid agent id.'); } $this->members[$agent->id_agente()] = $agent; return $agent; } /** * Remove an agent from the cluster. * * @param integer $id_agent New id_agent to be removed. * * @return void * @throws \Exception On error. */ public function removeMember(int $id_agent) { if (isset($this->members[$id_agent]) === false) { return; } unset($this->members[$id_agent]); $rs = \db_process_sql_delete( 'tcluster_agent', [ 'id_cluster' => $this->fields['id'], 'id_agent' => $id_agent, ] ); return true; } /** * Return AA modules associated to current cluster. * * @param integer $type AA or AP (use constants) * MODULE_PREDICTION_CLUSTER_AA * MODULE_PREDICTION_CLUSTER_AP. * * @return array Of items. */ public function getItems(?int $type=null) { $items = []; if ($type === MODULE_PREDICTION_CLUSTER_AA) { if (is_array($this->aaModules) === true) { return $this->aaModules; } } if ($type === MODULE_PREDICTION_CLUSTER_AP) { if (is_array($this->apModules) === true) { return $this->apModules; } } if (is_array($this->apModules) === true ) { $items = array_merge($items, $this->apModules); } if (is_array($this->aaModules) === true ) { $items = array_merge($items, $this->aaModules); } return $items; } /** * Retrieve AA modules. * * @return array Of ClusterItem definition. */ public function getAAModules() { return $this->getItems(MODULE_PREDICTION_CLUSTER_AA); } /** * Retrieve AP modules. * * @return array Of ClusterItem definition. */ public function getAPModules() { return $this->getItems(MODULE_PREDICTION_CLUSTER_AP); } /** * Retrieves module definition from current members matching name. * * @param string $name Target name to retrieve. * * @return array Module fields. * @throws \Exception On error. */ public function getModuleSkel(string $name) { foreach ($this->members as $member) { $module = $member->searchModules( ['nombre' => $name] ); if ($module !== null && empty($module) === false) { if (count($module) > 1) { $msg = __METHOD__.' error: Multiple occurrences of "'; $msg .= $name.'", please remove duplicates from agent "'; $msg .= $member->alias().'".'; throw new \Exception( $msg ); } // Method searchModules returns multiple occurrences. $module = $module[0]; $module = $module->toArray(); break; } } // Remove specific fields. unset($module['id_agente_modulo']); unset($module['id_agente']); return $module; } /** * Add an item to the cluster. * * @param string $name Target name. * @param integer $type Item type. * @param array $definition Module definition. * * @return ClusterModule Created module. * @throws \Exception On error. */ public function addItem(string $name, int $type, array $definition) { $item = new ClusterModule(); $item->name($name); $item->id_cluster($this->id()); // Skel values. $module_skel = $this->getModuleSkel($name); // Customize definition. $definition = array_merge($module_skel, $definition); // Store in cluster agent. $definition['id_agente'] = $this->id_agent(); if ($type === MODULE_PREDICTION_CLUSTER_AA) { $item->item_type('AA'); } else if ($type === MODULE_PREDICTION_CLUSTER_AP) { $item->item_type('AP'); } else { throw new \Exception(__METHOD__.' error: Invalid item type'); } // Set module definition. $item->setModule($definition); // Default values. $item->critical_limit(0); $item->warning_limit(0); $item->is_critical(0); return $item; } /** * Add AA module to the cluster. * * @param string $name Target name. * * @return void */ public function addAAModule(string $name) { if (empty($this->aaModules[$name]) === true) { $main_id = $this->clusterStatus->id_agente_modulo(); // Register module in agent. // id_modulo = 0, // tcp_port = 1, // prediction_moddule = 6. // Set thresholds while updating. $this->aaModules[$name] = $this->addItem( $name, MODULE_PREDICTION_CLUSTER_AA, [ 'nombre' => $name, 'id_modulo' => 0, 'prediction_module' => 6, 'tcp_port' => 1, 'id_tipo_modulo' => 1, 'custom_integer_1' => $this->id(), 'parent_module_id' => $main_id, ] ); \db_pandora_audit( AUDIT_LOG_AGENT_MANAGEMENT, 'Module '.io_safe_output( $name ).' added to cluster'.io_safe_output( $this->fields['name'] ).' as Active-Active module' ); } } /** * Add AP module to the cluster. * * @param string $name Target name. * * @return void */ public function addAPModule(string $name) { if (empty($this->apModules[$name]) === true) { $main_id = $this->clusterStatus->id_agente_modulo(); $type = db_get_value( 'id_tipo_modulo', 'tagente_modulo', 'nombre', $name ); if (empty($type) === true) { $type = 1; } // Register module in agent. // id_modulo = 5, // tcp_port = 1, // prediction_moddule = 7. // Set thresholds while updating. $this->apModules[$name] = $this->addItem( $name, MODULE_PREDICTION_CLUSTER_AP, [ 'nombre' => $name, 'id_modulo' => 5, 'prediction_module' => 7, 'tcp_port' => 1, 'id_tipo_modulo' => $type, 'custom_integer_1' => $this->id(), 'parent_module_id' => $main_id, ] ); \db_pandora_audit( AUDIT_LOG_AGENT_MANAGEMENT, 'Module '.io_safe_output( $name ).' added to cluster'.io_safe_output( $this->fields['name'] ).' as Active-Passive module' ); } } /** * Removes AA module from the cluster. * * @param string $name Target name. * * @return void */ public function removeAAModule(string $name) { if (empty($this->aaModules[$name]) === false) { // Mark item for db elimination. $this->removedItems[] = [ 'id' => $this->aaModules[$name]->id(), 'item_type' => $this->aaModules[$name]->item_type(), ]; $this->aaModules[$name]->delete(); unset($this->aaModules[$name]); } } /** * Removes AP module from the cluster. * * @param string $name Target name. * * @return void */ public function removeAPModule(string $name) { if (empty($this->apModules[$name]) === false) { // Mark item for db elimination. $this->removedItems[] = [ 'id' => $this->apModules[$name]->id(), 'item_type' => $this->apModules[$name]->item_type(), ]; $this->apModules[$name]->delete(); unset($this->apModules[$name]); } } /** * Return found cluster definitions. * * @param array $filter Conditions. * * @return mixed Array or false. */ public static function search(array $filter) { return \db_get_all_rows_filter( 'tcluster', $filter ); } /** * Operates with group. * * @param integer|null $id_group Target group to update. Retrieve group obj * if null. * * @return mixed Void if set, PandoraFMS\Group if argument is null. */ public function group(?int $id_group=null) { if (is_numeric($id_group) === true && $id_group > 0) { $this->fields['group'] = new Group($id_group); } else { return $this->fields['group']; } } /** * Returns AA modules as nodes for a map if any, if not, retrieves members. * * @return array Of PandoraFMS\Networkmap nodes. */ public function getNodes() { // Parse agents. $nodes = []; $node_count = 0; $parent = $node_count; $id_node = $node_count++; $status = \agents_get_status_from_counts($this->agent()->toArray()); $image = 'images/networkmap/'.os_get_icon($this->agent()->id_os()); if (empty($this->aaModules) === true) { // No AA modules, use members. $parent = $this->agent()->id_agente(); // Add node. foreach ($this->members as $agent) { $node = []; foreach ($agent->toArray() as $k => $v) { $node[$k] = $v; } $node['id_agente'] = $agent->id_agente(); $node['id_parent'] = $parent; $node['id_node'] = $node_count; $node['image'] = 'images/networkmap/'.os_get_icon( $agent->id_os() ); $node['status'] = \agents_get_status_from_counts( $agent->toArray() ); $nodes[$node_count++] = $node; } } else { foreach ($this->aaModules as $cl_item) { $cl_module = $cl_item->getModule(); if ($cl_module === null) { continue; } foreach ($this->members as $agent) { $module = $agent->searchModules( ['nombre' => $cl_module->nombre()] ); if (empty($module) === true) { // AA Module not found in member. continue; } // Transform multi array to get first occurrence. // Warning. Here must only be 1 result. $module = array_shift($module); $node = []; $node['type'] = NODE_GENERIC; $node['label'] = $agent->alias().' » '; $node['label'] .= $module->nombre(); $node['id_agente'] = $module->id_agente(); $node['id_agente_modulo'] = $module->id_agente_modulo(); $node['id_parent'] = $parent; $node['id_node'] = $node_count; $node['image'] = 'images/networkmap/'.os_get_icon( $agent->id_os() ); $node['status'] = $module->getStatus()->last_known_status(); $nodes[$node_count++] = $node; } } } $nodes[$parent] = $this->agent()->toArray(); $nodes[$parent] = ($nodes[$parent] + [ 'id_parent' => $parent, 'id_node' => $id_node, 'status' => $status, 'id_agente' => $this->agent()->id_agente(), 'image' => $image, ]); return $nodes; } /** * Saves current group definition to database. * * @return mixed Affected rows of false in case of error. * @throws \Exception On error. */ public function save() { $values = $this->fields; unset($values['agent']); $values['group'] = $this->group()->id_grupo(); if (isset($values['id']) === true && $values['id'] > 0) { // Update. $rs = \db_process_sql_update( 'tcluster', $values, ['id' => $this->fields['id']] ); if ($rs === false) { global $config; throw new \Exception( __METHOD__.' error: '.$config['dbconnection']->error ); } \db_pandora_audit( AUDIT_LOG_AGENT_MANAGEMENT, 'Cluster '.io_safe_output($this->fields['name']).' modified' ); } else { // New. $rs = \db_process_sql_insert( 'tcluster', $values ); if ($rs === false) { global $config; throw new \Exception( __METHOD__.' error: '.$config['dbconnection']->error ); } $this->fields['id'] = $rs; \db_pandora_audit( AUDIT_LOG_AGENT_MANAGEMENT, 'Cluster '.io_safe_output($this->fields['name']).' created' ); } $this->saveMembers(); $this->saveItems(); return true; } /** * Updates entries in tcluster_agent. * * @return void * @throws \Exception On error. */ public function saveMembers() { $err = __METHOD__.' error: '; $values = []; foreach ($this->members as $agent) { $values[$agent->id_agente()] = [ 'id_cluster' => $this->fields['id'], 'id_agent' => $agent->id_agente(), ]; } if (empty($values) === true) { return; } // Clean previous relationships. $rs = \db_process_sql_delete( 'tcluster_agent', [ 'id_cluster' => $this->fields['id'] ] ); foreach ($values as $set) { // Add current relationships. $rs = \db_process_sql_insert( 'tcluster_agent', $set ); if ($rs === false) { global $config; throw new \Exception( $err.$config['dbconnection']->error ); } } } /** * Undocumented function * * @return void */ public function saveItems() { $items = $this->getItems(); foreach ($this->removedItems as $item) { \db_process_sql_delete( 'tcluster_item', $item ); } // Save cluster modules. foreach ($items as $item) { $item->save(); } } /** * Force cluster status module to be executed. * * @param boolean $get_informed Throw exception if clusterStatus is null. * * @return void * @throws \Exception On error. */ public function force(?bool $get_informed=true) { if ($this->clusterStatus === null) { if ($get_informed === true) { throw new \Exception( __METHOD__.' error: Cluster status module does not exist' ); } } else { $this->clusterStatus->flag(1); $this->clusterStatus->save(); } } /** * Delete cluster from db. * * @return void * @throws \Exception On error. */ public function delete() { global $config; if ($this->agent() !== null) { // Delete agent and modules. $this->agent()->delete(); } // Remove entries from db. // Table tcluster_agent. $rs = \db_process_sql_delete( 'tcluster_agent', ['id_cluster' => $this->fields['id']] ); if ($rs === false) { throw new \Exception( __METHOD__.' error: '.$config['dbconnection']->error ); } // Table tcluster_item. $rs = \db_process_sql_delete( 'tcluster_item', ['id_cluster' => $this->fields['id']] ); if ($rs === false) { throw new \Exception( __METHOD__.' error: '.$config['dbconnection']->error ); } // Table tcluster. $rs = \db_process_sql_delete( 'tcluster', ['id' => $this->fields['id']] ); if ($rs === false) { throw new \Exception( __METHOD__.' error: '.$config['dbconnection']->error ); } \db_pandora_audit( AUDIT_LOG_AGENT_MANAGEMENT, 'Cluster '.io_safe_output($this->fields['name']).' deleted' ); unset($this->aaModules); unset($this->apModules); unset($this->fields); } }