is really 'rawNodes' here you put an array like follows: * nodes => [ * 'some_id' => [ * 'type' => NODE_GENERIC/NODE_PANDORA/NODE_MODULE/NODE_AGENT, * behaviour and fields to be used are different depending * on the type. * 'id_agente' => 'some_id', * 'status' => 'agent status', * 'id_parent' => 'target parent to match in map, its no need to use the * real parent but must match some_id' * 'id_node' => incremental id (0,1,2,3...), * 'image' => relative path to image to use, * 'label' => label to use (in NODE_GENERIC) * ] * ] * * Tooltipster support: Keep in mind, using tooltipster behaviour * of map changes. * * Sample usage: * * * * $map_manager = new NetworkMap( * [ * 'nodes' => vmware_get_nodes($show_vms, $show_ds, $show_esx), * 'pure' => 1, * 'use_tooltipster' => 1, * 'tooltip_params' => [ * 'page' => 'operation/agentes/ver_agente', * 'get_agent_json' => 1, * ], * ] * ); * * $map_manager->printMap(); * * */ class NetworkMap { /** * Target map Id, from tmap. If the maps is being simulated * then the idMap value will be uniqid. * * @var integer */ public $idMap; /** * Content of tmap. Map definition. If the map is being simulated * then defaults to constructor received parameters. * * @var array */ public $map; /** * Data origin, network. * * @var string */ public $network; /** * Data origin, group id. * * @var integer */ public $idGroup; /** * Data origin, Discovery task. * * @var integer */ public $idTask; /** * Graph definition. Previously was 'nodes_and_relationships' * Is the data format before be translated to JS variables. * * @var array */ public $graph; /** * Dot string with graph definition. * Its contents will be send to graphviz to calculate node positions. * * @var string */ public $dotGraph; /** * Node list processed by NetworkMap class. * * @var array */ public $nodes; /** * Node list RAW. * A simple list of nodes, could content information of agents, modules... * Is the 'raw' information. * * @var array */ public $rawNodes; /** * Useful to translate id_node to id_agent or id_module. * Maps built nodes to original node information (agents, modules). * * @var array */ public $nodeMapping; /** * Relationship map. * Each element contents: * id_parent * id_child * parent_type * child_type * id_parent_source_data (from $this->nodes) * id_child_source_data (from $this->nodes) * * @var array */ public $relations; /** * Private nodes converted to JS. * * @var array */ private $nodesJS; /** * Private relations converted to JS. * * @var array */ private $relationsJS; /** * Include a Pandora (or vendor) node or not. * * @var integer */ public $noPandoraNode; /** * Use tooltipster interface instead of standard. * * @var boolean */ public $useTooltipster; /** * Options used in AJAX call while using tooltipster. * * @var array */ public $tooltipParams; /** * Defines if map is widget or not for JS * * @var boolean; */ public $widget; /** * Shows the map using 100% of height and width if is a widget. * * @var boolean */ public $fullSize; /** * Defines a custom method to parse Graphviz output and generate Graph. * Function pointer. * * @var string */ public $customParser; /** * Defines arguments to be passed to $customParser. * If is not defined, default arguments will be used. * * @var array */ public $customParserArgs; /** * If using a custom parser, fallback to default parser while * found exceptions. * * @var boolean */ public $fallbackDefaultParser; /** * Array of map options. Because how is built, the structure matches * with tmap definition, where map_filter is the json-extracted data. * Duplicate options appears since tmap stores information in different * ways (simplifies process). * If an idMap is defined, map is loaded into this structure and used along * the class. * generation_method * simple * font_size * nooverlap * z_dash * ranksep * center * regen * pure * show_snmp_modules * cut_names * relative * text_filter * dont_show_subgroups * strict_user * size_canvas * old_mode * map_filter (array) * dont_show_subgroups * node_radius * x_offs * y_offs * z_dash * node_sep * rank_sep * mindist * kval * * @var array */ public $mapOptions; /** * Filter (command) to use to calculate node positions. * * @var string */ private $filter; /** * Do not show the popup window. * * @var integer */ private $noPopUp; /** * Base constructor. * * @param mixed $options Could define in array as: * id_map => target discovery task id. * id_group => target group. * network => target CIDR. * graph => target graph (already built). * nodes => target agents or nodes. * relations => target array of relationships. * mode => simple (0) or advanced (1). * map_options => Map options. * * @return object New networkmap manager. */ public function __construct($options=false) { global $config; // Default mapOptions values. // Defines the command to generate positions. $this->mapOptions['generation_method'] = LAYOUT_SPRING1; // Use fixed positions defined (X,Y) per node. $this->mapOptions['fixed_positions'] = 0; $this->mapOptions['width'] = $config['networkmap_max_width']; $this->mapOptions['height'] = $config['networkmap_max_width']; $this->mapOptions['simple'] = 0; $this->mapOptions['font_size'] = 20; $this->mapOptions['nooverlap'] = 1; $this->mapOptions['z_dash'] = 0.5; $this->mapOptions['center'] = 0; $this->mapOptions['regen'] = 0; $this->mapOptions['pure'] = 0; $this->mapOptions['show_snmp_modules'] = false; $this->mapOptions['cut_names'] = false; $this->mapOptions['relative'] = true; $this->mapOptions['text_filter'] = ''; $this->mapOptions['dont_show_subgroups'] = false; $this->mapOptions['strict_user'] = false; $this->mapOptions['size_canvas'] = 0; $this->mapOptions['old_mode'] = false; $this->mapOptions['map_filter'] = [ 'dont_show_subgroups' => 0, 'node_radius' => 40, 'x_offs' => 0, 'y_offs' => 0, 'z_dash' => 0.5, 'node_sep' => 5, 'rank_sep' => 5, 'mindist' => 1, 'kval' => 0.1, ]; if (is_array($options)) { // Previously nodes_and_relations. if (isset($options['graph'])) { $this->graph = $options['graph']; } // String dotmap. if (isset($options['dot_graph'])) { $this->dotGraph = $options['dot_graph']; } // Array of nodes, agents, virtual, etc. if (isset($options['nodes'])) { $this->rawNodes = $options['nodes']; } // Array of relations. if (isset($options['relations'])) { $this->relations = $options['relations']; } // User interface type. Simple or advanced. if (isset($options['mode'])) { $this->mode = $options['mode']; } // Show interface elements or dashboard style. if (isset($options['pure'])) { $this->mapOptions['pure'] = $options['pure']; } if (isset($options['no_pandora_node'])) { $this->noPandoraNode = $options['no_pandora_node']; } if (isset($options['no_popup'])) { $this->noPopUp = $options['no_popup']; } // Initialize as widget? if (isset($options['widget'])) { $this->fullSize = (bool) $options['widget']; $this->widget = true; } else { $this->fullSize = true; $this->widget = false; } // Use a custom parser. if (isset($options['custom_parser'])) { $this->customParser = $options['custom_parser']; } // Custom parser arguments. if (isset($options['custom_parser_args'])) { if (is_array($options['custom_parser_args'])) { foreach ($options['custom_parser_args'] as $k) { $this->customParserArgs[] = $k; } } else { $this->customParserArgs = $options['custom_parser_args']; } } // Fallback to default parser. if (isset($options['fallback_to_default_parser'])) { $this->fallbackDefaultParser = $options['fallback_to_default_parser']; } if (isset($options['use_tooltipster'])) { $this->useTooltipster = $options['use_tooltipster']; } if (is_array($options['tooltip_params'])) { foreach ($options['tooltip_params'] as $k => $v) { $this->tooltipParams[$k] = $v; } } // Map options, check default values above. // This is only used while generating new maps using // (generateDotGraph). if (isset($options['map_options']) && is_array($options['map_options']) ) { foreach ($options['map_options'] as $k => $v) { if ($k == 'map_filter' && is_array($v)) { foreach ($v as $kc => $vc) { $this->mapOptions['map_filter'][$kc] = $vc; } } else { $this->mapOptions[$k] = $v; } } } // Load from tmap. if ($options['id_map']) { $this->idMap = $options['id_map']; // Update nodes and relations. $this->loadMap(); if (empty($this->nodes) && empty($this->relations) ) { $this->createMap(); } } else { // Generate from group, task or network. if ($options['id_group']) { $this->idGroup = $options['id_group']; } if ($options['id_task']) { $this->idTask = $options['id_task']; } if ($options['network']) { $this->network = $options['network']; } $this->createMap(); } } return $this; } /** * Creates a new map based on a target. * * Target is specified from constructor arguments. * options: * - id_task => create a map from task. * - id_group => create a map from group. * - network => create a map from network. * * @return void */ public function createMap() { global $config; // If exists, load from DB. if ($this->idMap) { $this->loadMap(); return; } // Simulated map. $this->idMap = uniqid(); // No tmap definition. Paint data. if ($this->idTask) { $recon_task = db_get_row_filter( 'trecon_task', ['id_rt' => $this->idTask] ); $this->network = $recon_task['subnet']; } // Simulate map entry. $this->map = [ 'id' => $this->idMap, '__simulated' => 1, 'background' => '', 'background_options' => 0, 'source_period' => 60, 'filter' => $this->mapOptions['map_filter'], 'width' => $config['networkmap_max_width'], 'height' => $config['networkmap_max_width'], 'center_x' => 0, 'center_y' => 0, ]; if (isset($this->mapOptions['generation_method']) === false) { $this->mapOptions['generation_method'] = LAYOUT_SPRING1; } // Load filter. $this->loadFilter(); // Will be stored in $this->graph. $this->generateNetworkMap(); } /** * Update filter and layout based on generation_method selected. * * @return boolean True or false. */ private function loadFilter() { if (is_array($this->mapOptions) === false) { return false; } switch ($this->mapOptions['generation_method']) { case LAYOUT_CIRCULAR: $this->filter = 'circo'; $this->mapOptions['layout'] = 'circular'; break; case LAYOUT_FLAT: $this->filter = 'dot'; $this->mapOptions['layout'] = 'flat'; break; case LAYOUT_RADIAL: $this->filter = 'twopi'; $this->mapOptions['layout'] = 'radial'; break; case LAYOUT_SPRING1: default: $this->filter = 'neato'; $this->mapOptions['layout'] = 'spring1'; break; case LAYOUT_SPRING2: $this->filter = 'fdp'; $this->mapOptions['layout'] = 'spring2'; break; } return true; } /** * Loads a map from a target map ID. * * @return void. */ public function loadMap() { if ($this->map) { // Already loaded. return; } if ($this->idMap) { $this->map = db_get_row('tmap', 'id', $this->idMap); $this->mapOptions['map_filter'] = json_decode( $this->map['filter'], true ); foreach ($this->map as $k => $v) { $this->mapOptions[$k] = $v; } // Load filter. $this->loadFilter(); // Retrieve data origin. $this->network = null; $this->idTask = null; $this->idGroup = $this->map['id_group']; switch ($this->map['source']) { case SOURCE_TASK: $this->idTask = $this->map['source_data']; break; case SOURCE_NETWORK: $this->network = $this->map['source_data']; break; case SOURCE_GROUP: // Already load. default: // Ignore. break; } if ($this->idTask) { $recon_task = db_get_row_filter( 'trecon_task', ['id_rt' => $this->idTask] ); $this->network = $recon_task['subnet']; } // Retrieve or update nodes and relations. $this->getNodes(); $this->getRelations(); // Nodes and relations will be stored in $this->graph. $this->loadGraph(); } } /** * Retrieves node information using id_node as mapping instead element id. * * @param integer $id_node Target node. * @param string $field Field to retrieve, if null, all are return. * * @return mixed Array (node data) or null if error. */ public function getNodeData(int $id_node, $field=null) { if (is_array($this->nodes) === false || is_array($this->nodeMapping) === false ) { return null; } if (is_array($this->nodes[$this->nodeMapping[$id_node]]) === true) { if (isset($field) === false) { return $this->nodes[$this->nodeMapping[$id_node]]; } else { return $this->nodes[$this->nodeMapping[$id_node]][$field]; } } else { return null; } } /** * Set nodes. * * @param array $nodes Nodes definition. * * @return void */ public function setNodes($nodes) { $this->nodes = $nodes; } /** * Return nodes of current map. * * @return array Nodes. */ public function getNodes() { if ($this->nodes) { return $this->nodes; } if ($this->idMap !== false) { $this->nodes = get_nodes_from_db($this->idMap); } return $this->nodes; } /** * Set relations. * * @param array $relations Relations definition. * * @return void */ public function setRelations($relations) { $this->relations = $relations; } /** * Return relations of current map. * * @return array Relations. */ public function getRelations() { if ($this->relations) { return $this->relations; } if ($this->idMap !== false) { $this->relations = get_relations_from_db($this->idMap); } return $this->relations; } /** * Search for nodes in current map definition. * * @return array Nodes detected, internal variable NOT updated. */ public function calculateNodes() { global $config; // Calculate. // Search. if ($this->idTask) { // Network map, based on discovery task. enterprise_hook('get_discovery_agents', [$this->idTask]); } if ($this->network) { // Network map, based on direct network. $nodes = networkmap_get_nodes_from_ip_mask( $this->network, false, ' ' ); } else if ($this->mapOptions['map_filter']['empty_map']) { // Empty map returns no data. $nodes = []; } else { if ($this->mapOptions['map_filter']['dont_show_subgroups'] === 'true' || $this->mapOptions['map_filter']['dont_show_subgroups'] == 1 ) { // Show only current selected group. $filter['id_grupo'] = explode(',', $this->idGroup); } else { // Show current group and children. foreach (explode(',', $this->idGroup) as $key => $group) { $childrens = groups_get_children($group, null, true); if (!empty($childrens)) { $childrens = array_keys($childrens); if (empty($filter['id_grupo']) === false) { $filter['id_grupo'] = array_merge($filter['id_grupo'], $childrens); } else { $filter['id_grupo'] = $childrens; } } else { $filter['id_grupo'][] = $group; } } } // Group map. $nodes = agents_get_agents( $filter, ['*'], 'AR', [ 'field' => 'id_parent', 'order' => 'ASC', ] ); if (is_array($nodes)) { // Remap ids. $nodes = array_reduce( $nodes, function ($carry, $item) { $carry[$item['id_agente']] = $item; return $carry; } ); } else { $nodes = []; } } return $nodes; } /** * Search for relations for a given node in current map definition. * Use id_parent in custom node definition to create an edge between * two nodes. * * Representation is to => from because from could be equal in multiple * edges but no to (1 origin, multiple targets). * * @param array $id_source Id for source data, agent, module or custom. * * @return array Relations found for given node. */ public function calculateRelations( $id_source ) { // Calculate. $node = $this->nodes[$id_source]; if (is_array($node) === false) { return false; } $relations = []; $i = 0; $from_type = NODE_AGENT; $to_type = NODE_AGENT; switch ($node['node_type']) { case NODE_AGENT: // Search for agent parent and module relationships. $module_relations = modules_get_relations( ['id_agent' => $node['id_agente']] ); if ($module_relations !== false) { // Module relation exist. foreach ($module_relations as $mod_rel) { // Check if target referenced agent is defined in // current map. $agent_a = modules_get_agentmodule_agent( $mod_rel['module_a'] ); $module_a = $mod_rel['module_a']; $agent_b = modules_get_agentmodule_agent( $mod_rel['module_b'] ); $module_b = $mod_rel['module_b']; // Calculate target. $module_to = $module_a; $agent_to = $agent_a; $module_from = $module_b; $agent_from = $agent_b; // Module relations does not have from and to, // If current agent_a is current node, reverse relation. if ($agent_a == $node['id_agente']) { $module_to = $module_b; $agent_to = $agent_b; $module_from = $module_a; $agent_from = $agent_a; } $target_node = $this->nodes[NODE_AGENT.'_'.$agent_to]; if (isset($target_node) === false) { // Agent is not present in this map. continue; } $rel = []; // Node reference (child). $rel['id_child'] = $node['id_node']; $rel['child_type'] = NODE_MODULE; $rel['id_child_source_data'] = $module_from; $rel['id_child_agent'] = $agent_from; // Node reference (parent). $rel['id_parent'] = $target_node['id_node']; $rel['parent_type'] = NODE_MODULE; $rel['id_parent_source_data'] = $module_to; $rel['id_parent_agent'] = $agent_to; // Store relation. $relations[] = $rel; } } // Add also parent relationship. $parent_id = NODE_AGENT.'_'.$node['id_parent']; if ((int) $node['id_parent'] > 0) { $parent_node = $this->nodes[$parent_id]['id_node']; } // Store relationship. if (is_integer($parent_node) && $node['id_parent'] > 0) { $rel = []; // Node reference (parent). $rel['id_parent'] = $parent_node; $rel['parent_type'] = NODE_AGENT; $rel['id_parent_source_data'] = $node['id_parent']; // Node reference (child). $rel['id_child'] = $node['id_node']; $rel['child_type'] = NODE_AGENT; $rel['id_child_source_data'] = $node['id_agente']; // Store relation. $relations[] = $rel; } break; case NODE_MODULE: // Search for module relationships. $module_relations = modules_get_relations( ['id_module' => $node['id_agente_modulo']] ); if ($module_relations !== false) { // Module relation exist. foreach ($module_relations as $mod_rel) { // Check if target referenced agent is defined in // current map. $agent_a = modules_get_agentmodule_agent( $mod_rel['module_a'] ); $module_a = $mod_rel['module_a']; $agent_b = modules_get_agentmodule_agent( $mod_rel['module_b'] ); $module_b = $mod_rel['module_b']; // Calculate target. $module_to = $module_a; $agent_to = $agent_a; $module_from = $module_b; $agent_from = $agent_b; // Module relations does not have from and to, // If current agent_a is current node, reverse relation. if ($agent_a == $node['id_agente']) { $module_to = $module_b; $agent_to = $agent_b; $module_from = $module_a; $agent_from = $agent_a; } $target_node = $this->nodes[NODE_AGENT.'_'.$agent_to]; if (isset($target_node) === false) { // Agent is not present in this map. continue; } $rel = []; // Node reference (child). $rel['id_child'] = $node['id_node']; $rel['child_type'] = NODE_MODULE; $rel['id_child_source_data'] = $module_from; $rel['id_child_agent'] = $agent_from; // Node reference (parent). $rel['id_parent'] = $target_node['id_node']; $rel['parent_type'] = NODE_MODULE; $rel['id_parent_source_data'] = $module_to; $rel['id_parent_agent'] = $agent_to; // Store relation. $relations[] = $rel; } } break; case NODE_GENERIC: // Handmade ones. // Add also parent relationship. if (isset($node['id_parent'])) { $parent_id = NODE_AGENT.'_'.$node['id_parent']; $parent_node = $this->nodes[$parent_id]['id_node']; if ($parent_node === null) { $parent_id = NODE_MODULE.'_'.$node['id_parent']; $parent_node = $this->nodes[$parent_id]['id_node']; } if ($parent_node === null) { $parent_id = NODE_GENERIC.'_'.$node['id_parent']; $parent_node = $this->nodes[$parent_id]['id_node']; } // Store relationship. if ($parent_node !== null) { $relations[] = [ 'id_parent' => $parent_node, 'parent_type' => NODE_GENERIC, 'id_child' => $node['id_node'], 'child_type' => NODE_GENERIC, ]; } } break; case NODE_PANDORA: default: // Ignore. break; } // Others. return $relations; } /** * Generates or loads nodes&relations array from DB. * Load, calculates statuses and leave the structure in $this->graph. * * * Structure generated: * Nodes: * id_map. * id. * id_agent. * id_module. * type. * x. * y. * width. * height. * text. * source_data. * style (json). * * Relations: * id_map. * id_parent. * parent_type. * id_parent_source_data. * id_child. * child_type. * id_child_source_data. * id_parent_agent. * id_child_agent. * * @return void */ public function loadGraph() { $nodes = $this->nodes; $relations = $this->relations; // Generate if there's no data in DB about nodes or relations. if (empty($nodes) === true && empty($relations) === true) { $this->generateNetworkMap(); return; } $graph = networkmap_load_map($this); if (empty($graph) === true) { $this->generateNetworkMap(); return; } $this->graph = $graph; } /** * Generates a graph definition (header only) for dot graph. * * @return string Dot graph header. */ public function openDotFile() { global $config; $overlap = 'compress'; $map_filter = $this->mapOptions['map_filter']; $nooverlap = $this->mapOptions['nooverlap']; $zoom = $this->mapOptions['zoom']; $layout = $this->mapOptions['layout']; if (isset($this->mapOptions['width']) && isset($this->mapOptions['height']) ) { $size_x = ($this->mapOptions['width'] / 100); $size_y = ($this->mapOptions['height'] / 100); } else if (isset($config['networkmap_max_width'])) { $size_x = ($config['networkmap_max_width'] / 100); $size_y = ($size_x * 0.8); } else { $size_x = 8; $size_y = 5.4; $size = ''; } if ($zoom > 0) { $size_x *= $zoom; $size_y *= $zoom; } $size = $size_x.','.$size_y; if ($this->mapOptions['size_canvas'] !== null) { $size = ($this->mapOptions['size_canvas']['x'] / 100); $size .= ','.($this->mapOptions['size_canvas']['y'] / 100); } // Graphviz custom values. if (isset($map_filter['node_sep'])) { $node_sep = $map_filter['node_sep']; } else { $node_sep = 0.1; } if (isset($map_filter['rank_sep'])) { $rank_sep = $map_filter['rank_sep']; } else { if ($layout == 'radial') { $rank_sep = 1.0; } else { $rank_sep = 0.5; } } if (isset($map_filter['mindist'])) { $mindist = $map_filter['mindist']; } else { $mindist = 1.0; } if (isset($map_filter['kval'])) { $kval = $map_filter['kval']; } else { $kval = 0.1; } // BEWARE: graphwiz DONT use single ('), you need double ("). $head = 'graph networkmap { dpi=100; bgcolor="transparent"; labeljust=l; margin=0; pad="0.75,0.75";'; if ($nooverlap != '') { $head .= 'overlap=scale;'; $head .= 'outputorder=first;'; } if ($layout == 'flat' || $layout == 'spring1' || $layout == 'spring2' ) { if ($nooverlap != '') { $head .= 'overlap="scalexy";'; } if ($layout == 'spring1' || $layout == 'spring2') { $head .= 'sep="'.$node_sep.'";'; } if ($layout == 'flat') { $head .= 'ranksep="'.$rank_sep.'";'; } if ($layout == 'spring2') { $head .= 'K="'.$kval.'";'; } } if ($layout == 'radial') { $head .= 'ranksep="'.$rank_sep.'";'; } if ($layout == 'circular') { $head .= 'mindist="'.$mindist.'";'; } $head .= 'ratio="fill";'; $head .= 'root=0;'; $head .= 'nodesep="'.$node_sep.'";'; $head .= 'size="'.$size.'";'; $head .= "\n"; return $head; } /** * Creates a node in dot format. * Requirements: * id_node * id_source * status => defines 'color' * label * image * url * * @param array $data Node definition. * * @return string Dot node. */ public function createDotNode($data) { global $config; if (is_array($data) === false) { return ''; } if ($this->mapOptions['fixed_positions']) { // Ignore. return; } $dot_str = ''; // Color is being printed by D3, not graphviz. // Used only for positioning. $color = COL_NORMAL; $label = $data['label']; $url = 'none'; $parent = $data['parent']; $font_size = $this->mapOptions['font_size']; if (isset($data['radius'])) { $radius = $data['radius']; } else { $radius = $this->mapOptions['map_filter']['node_radius']; } $radius /= GRAPHVIZ_CONVERSION_FACTOR; if (is_array($label)) { $label = join('', $label); } if (strlen($label) > 16) { $label = ui_print_truncate_text($label, 16, false, true, false); } // If radius is 0, set to 1 instead. if ($radius <= 0) { $radius = 1; } // Simple node always. This kind of node is used only to // retrieve X,Y positions from graphviz no for personalization. $dot_str = $data['id_node'].' [ parent="'.$data['id_parent'].'"'; $dot_str .= ', color="'.$color.'", fontsize='.$font_size; $dot_str .= ', shape="doublecircle"'.$data['url_node_link']; $dot_str .= ', style="filled", fixedsize=true, width='.$radius; $dot_str .= ', height='.$radius.', label="'.$label.'"]'."\n"; return $dot_str; } /** * Avoid multiple connections between two nodes if any of them does not * add more information. Prioritize. * * For instance, if we have module - module relationship and agent - agent * discard agent - agent relationship (module - module apports more * information). * * @return void */ public function cleanGraphRelations() { global $config; $relations = $this->graph['relations']; $cleaned = []; $rel_map = []; /* * Relation map: * id_child.'_'.id_parent => [ * 'priority' (0,1) * 'relation_index' * ] */ if (is_array($relations)) { foreach ($relations as $index => $rel) { /* * AA, AM and MM links management * Priority: * 1 -> MM (module - module) * 1 -> AM (agent - module) * 0 -> AA (agent - agent) */ $id_parent = $rel['id_parent']; $id_child = $rel['id_child']; $rel_type = $rel['child_type'].'_'.$rel['parent_type']; $valid = 0; $key = -1; if ($rel['parent_type'] == NODE_MODULE && $rel['child_type'] == NODE_MODULE ) { // Keep std references. $ref_id_parent = $id_parent; $ref_id_child = $id_child; // Module information available. $id_parent = $rel['id_parent_source_data']; $id_child = $rel['id_child_source_data']; $priority = 1; $valid = 1; if (is_array($rel_map[$id_child.'_'.$id_parent])) { // Already defined. $key = $id_child.'_'.$id_parent; $data = $rel_map[$id_child.'_'.$id_parent]; if ($priority > $data['priority']) { unset($rel[$data['index']]); } else { $valid = 0; } } if (is_array($rel_map[$id_parent.'_'.$id_child])) { // Already defined. $key = $id_parent.'_'.$id_child; $data = $rel_map[$id_parent.'_'.$id_child]; if ($priority > $data['priority']) { unset($rel[$data['index']]); } else { $valid = 0; } } if ($valid == 1) { $rel_map[$id_parent.'_'.$id_child] = [ 'index' => $index, 'priority' => $priority, ]; // Keep node reference mapping - low precedence relationship. $rel_map[$ref_id_parent.'_'.$ref_id_child] = [ 'index' => $index, 'priority' => $priority, ]; } } else if ($rel['parent_type'] == NODE_AGENT && $rel['child_type'] == NODE_AGENT ) { // Module information not available. $priority = 0; $valid = 1; if (is_array($rel_map[$id_child.'_'.$id_parent])) { // Already defined. $key = $id_child.'_'.$id_parent; $data = $rel_map[$id_child.'_'.$id_parent]; if ($priority > $data['priority']) { unset($rel[$data['index']]); } else { $valid = 0; } } if (is_array($rel_map[$id_parent.'_'.$id_child])) { // Already defined. $key = $id_parent.'_'.$id_child; $data = $rel_map[$id_parent.'_'.$id_child]; if ($priority > $data['priority']) { unset($rel[$data['index']]); } else { $valid = 0; } } if ($valid == 1) { $rel_map[$id_parent.'_'.$id_child] = [ 'index' => $index, 'priority' => $priority, ]; } } else if ($rel['parent_type'] == NODE_MODULE && $rel['child_type'] == NODE_AGENT ) { // Module information not available. $priority = 1; $valid = 1; } else if ($rel['parent_type'] == NODE_AGENT && $rel['child_type'] == NODE_MODULE ) { // Module information not available. $priority = 1; $valid = 1; } else { // Pandora & generic links are always accepted. $valid = 1; } if ($valid === 1) { if ($rel['id_parent'] != $rel['id_child']) { $cleaned[] = $rel; } } } } else { return; } $this->graph['relations'] = $cleaned; } /** * Internal method to allow developer to compare status from * different origins by checking a value. * * Greater value implies more critical. * * @param integer $status Status. * * @return integer Criticity value. */ private static function getStatusNumeric($status) { if (isset($status) === false) { return NO_CRIT; } switch ($status) { case AGENT_MODULE_STATUS_NORMAL: case AGENT_STATUS_NORMAL: return CRIT_1; case AGENT_MODULE_STATUS_NOT_INIT: case AGENT_STATUS_NOT_INIT: return CRIT_0; case AGENT_MODULE_STATUS_CRITICAL_BAD: case AGENT_STATUS_CRITICAL: return CRIT_4; case AGENT_MODULE_STATUS_WARNING: case AGENT_STATUS_WARNING: return CRIT_3; case AGENT_MODULE_STATUS_CRITICAL_ALERT: case AGENT_MODULE_STATUS_WARNING_ALERT: case AGENT_STATUS_ALERT_FIRED: return CRIT_5; case AGENT_MODULE_STATUS_UNKNOWN: case AGENT_STATUS_UNKNOWN: return CRIT_2; default: // Ignored. break; } return NO_CRIT; } /** * Returns worst status from two received. * Agent and module statuses should be identical, unless little differences. * * @param integer $status_a Status A. * @param integer $status_b Status B. * * @return integer Status A or status B, the worstest one. */ public static function getWorstStatus($status_a, $status_b) { // Case agent statuses. $a = self::getStatusNumeric($status_a); $b = self::getStatusNumeric($status_b); return ($a > $b) ? $status_a : $status_b; } /** * Returns target color to be used based on the status received. * * @param integer $status Source information. * @param boolean $force_module It's a module. * * @return string HTML tag for color. */ public static function getColorByStatus($status, ?bool $force_module=false) { include_once __DIR__.'/../functions_modules.php'; return modules_get_color_status($status, $force_module); } /** * Translates a standard node into a JS node with following attributes: * * @param array $nodes Input array (standard nodes structure). * id_map. * id_db. * type. * source_data. * x. * y. * z. * state. * deleted. * style. * shape. * image. * label. * id_agent. * id_networkmap. * * @return array Object ready to be dump to JS. * * Output array (translated): * id. * id_db. * type. * id_agent. * id_module. * fixed. * x. * y. * px. * py. * z. * state. * deleted. * image_url. * image_width. * image_height. * raw_text. * text. * shape. * color. * map_id. * networkmap_id. */ public function nodesToJS($nodes) { global $config; $return = []; $count_item_holding_area = 0; foreach ($nodes as $node) { $item = []; $item['id'] = $node['id']; if ($node['deleted']) { // Skip deleted nodes. continue; } // Id titem. if (isset($this->map['__simulated']) === false) { $item['id_db'] = $node['id_db']; } else { $item['id_db'] = (int) $node['id']; } // Get source data. $source_data = $this->getNodeData($node['id']); if (is_array($node['style']) === false) { $node['style'] = json_decode($node['style'], true); // Add styles. if (isset($source_data['style']) === true && is_array($source_data['style']) === true ) { $node['style'] = array_merge( $node['style'], $source_data['style'] ); } } // Propagate styles. foreach ($node['style'] as $k => $v) { $item[$k] = $v; } $item['type'] = $node['type']; $item['fixed'] = true; $item['x'] = (int) $node['x']; $item['y'] = (int) $node['y']; $item['z'] = (int) $node['z']; // X,Y aliases for D3. $item['px'] = $item['x']; $item['py'] = $item['y']; // Status represents the status of the node (critical, warning...). // State represents state of node in map (in holding_area or not). $item['state'] = $node['state']; $item['deleted'] = $node['deleted']; // Node color. $item['color'] = self::getColorByStatus($source_data['status']); switch ($node['type']) { case NODE_AGENT: $item['id_agent'] = $node['source_data']; break; case NODE_MODULE: $item['id_module'] = $node['source_data']; $item['color'] = self::getColorByStatus( $source_data['status'], true ); break; case NODE_PANDORA: $item['color'] = COL_IGNORED; $node['style']['image'] = ui_get_logo_to_center_networkmap(); break; case NODE_GENERIC: default: foreach ($source_data as $k => $v) { $node[$k] = $v; $item[$k] = $v; } $item['id_agent'] = $node['id_agente']; if (!empty($node['text'])) { $node['style']['label'] = $node['text']; } else { $node['style']['label'] = $node['name']; } if (isset($source_data['color'])) { $item['color'] = $source_data['color']; } else { if (empty($node['style']['id_networkmap']) === false) { $status_aux = get_status_color_networkmap_fictional_point($node['style']['id_networkmap']); $item['color'] = $status_aux; } else { $item['color'] = self::getColorByStatus( $node['status'], (bool) $node['id_module'] ); } } break; } // Calculate values. // 40 => DEFAULT NODE RADIUS. // 30 => alignment factor. $holding_area_max_y = ($this->mapOptions['height'] + 30 + $this->mapOptions['map_filter']['node_radius'] * 2 - $this->mapOptions['map_filter']['holding_area'][1] + 10 * $this->mapOptions['map_filter']['node_radius']); // Update position if node must be stored in holding_area. if ($item['state'] == 'holding_area') { $holding_area_x = ($this->mapOptions['width'] + 30 + $this->mapOptions['map_filter']['node_radius'] * 2 - $this->mapOptions['map_filter']['holding_area'][0] + ($count_item_holding_area % 11) * $this->mapOptions['map_filter']['node_radius']); $holding_area_y = ($this->mapOptions['height'] + 30 + $this->mapOptions['map_filter']['node_radius'] * 2 - $this->mapOptions['map_filter']['holding_area'][1] + (int) (($count_item_holding_area / 11)) * $this->mapOptions['map_filter']['node_radius']); // Keep holding area nodes in holding area. if ($holding_area_max_y <= $holding_area_y) { $holding_area_y = $holding_area_max_y; } $item['x'] = $holding_area_x; $item['y'] = $holding_area_y; // Increment for the next node in holding area. $count_item_holding_area++; } // Node image. $item['image_url'] = ''; $item['image_width'] = 0; $item['image_height'] = 0; if (empty($node['style']['image']) === false) { if (strpos($node['style']['image'], '.svg') === false) { $node['style']['image'] = os_transform_url_icon( $node['style']['image'] ); } $item['image_url'] = ui_get_full_url( $node['style']['image'], false, false, false ); $image_size = getimagesize( $config['homedir'].'/'.$node['style']['image'] ); $item['image_width'] = (int) $image_size[0]; $item['image_height'] = (int) $image_size[1]; } $item['raw_text'] = $node['style']['label']; $item['text'] = io_safe_output($node['style']['label']); $item['shape'] = $node['style']['shape']; $item['map_id'] = $node['id_map']; if (!isset($node['style']['id_networkmap']) || $node['style']['id_networkmap'] == '' || $node['style']['id_networkmap'] == 0 ) { $item['networkmap_id'] = 0; } else { $item['networkmap_id'] = $node['style']['id_networkmap']; } // XXX: Compatibility with Tooltipster - Simple map controller. if ($this->useTooltipster) { $item['label'] = $item['text']; $item['image'] = $item['image_url']; $item['image_height'] = 52; $item['image_width'] = 52; } $return[] = $item; } return $return; } /** * Transforms an edge relationship into a JS array to be dumped. * Sets fields like status, link color and updates some internal identifiers * used by JS frontend. * * @param array $edges Edges information in array of following items. * * * Input structure: * id_map. * id_parent. * parent_type. * id_parent_source_data. * id_child. * child_type. * id_child_source_data. * id_parent_agent. * id_child_agent. * * @return array Edge translated to JS object. * * * Output structure: * arrow_start. * arrow_end. * status_start. * status_end. * id_module_start. * id_agent_start. * id_module_end. * id_agent_end. * link_color. * target. * source. * deleted. * target_id_db. * source_id_db. * text_start. * text_end. */ public function edgeToJS($edges) { $return = []; // JS edge pseudo identificator. $i = 0; foreach ($edges as $rel) { $item = []; // Simulated index. $item['id_db'] = $i; $item['deleted'] = 0; // Else load. if (isset($this->map['__simulated']) === false) { $item['id_db'] = $rel['id_db']; $item['deleted'] = $rel['deleted']; $item['target_id_db'] = $this->getNodeData( $rel['id_parent'], 'id_db' ); $item['source_id_db'] = $this->getNodeData( $rel['id_child'], 'id_db' ); } if ($item['deleted']) { // Relation is deleted. Avoid. continue; } // Set relationship as 'agent' by default. // Generic and Pandora nodes simulates agent relationships. $item['arrow_start'] = 'agent'; $item['arrow_end'] = 'agent'; $item['source'] = $rel['id_parent']; $item['target'] = $rel['id_child']; $item['id_agent_start'] = $rel['id_child_agent']; $item['id_agent_end'] = $rel['id_parent_agent']; if ($rel['parent_type'] == NODE_MODULE) { $item['arrow_start'] = 'module'; $item['id_module_start'] = $rel['id_parent_source_data']; $item['status_start'] = modules_get_agentmodule_status( $item['id_module_start'] ); // Extract interface name to be placed on edge. $text = modules_get_agentmodule_name( (int) $item['id_module_start'] ); if (preg_match( '/(.+)_ifOperStatus$/', (string) $text, $matches ) ) { if ($matches[1]) { $item['text_start'] = io_safe_output($matches[1]); } } } if ($rel['child_type'] == NODE_MODULE) { $item['arrow_end'] = 'module'; $item['id_module_end'] = $rel['id_child_source_data']; $item['status_end'] = modules_get_agentmodule_status( $item['id_module_end'] ); // Extract interface name to be placed on edge. $text = modules_get_agentmodule_name( (int) $item['id_module_end'] ); if (preg_match( '/(.+)_ifOperStatus$/', (string) $text, $matches ) ) { if ($matches[1]) { $item['text_end'] = io_safe_output($matches[1]); } } } if (isset($rel['text_start']) && !empty($rel['text_start'])) { // Direct text_start definition. $item['text_start'] = $rel['text_start']; } if (isset($rel['text_end']) && !empty($rel['text_end'])) { // Direct text_end definition. $item['text_end'] = $rel['text_end']; } if (isset($rel['link_color']) && !empty($rel['link_color'])) { // Direct color definition. $item['link_color'] = $rel['link_color']; } else { // Use worst case to set link color. $item['link_color'] = self::getColorByStatus( self::getWorstStatus( $item['status_start'], $item['status_end'] ) ); } // XXX: Compatibility with Tooltipster - Simple map controller. if ($this->useTooltipster) { $item['orig'] = $rel['id_parent']; $item['dest'] = $rel['id_child']; } // Set direct values. $item['id'] = $i++; $return[] = $item; } return $return; } /** * Creates an edge in dot format. * Requirements: * from * to * * @param array $data Edge content. * * @return string Dot code for given edge. */ public function createDotEdge($data) { if (is_array($data) === false) { return ''; } if (!isset($data['from']) || !isset($data['to'])) { return ''; } $edge = "\n".$data['from'].' -- '.$data['to']; $edge .= '[len='.$this->mapOptions['map_filter']['node_sep']; $edge .= ', color="#BDBDBD", headclip=false, tailclip=false,'; $edge .= ' edgeURL=""];'."\n"; return $edge; } /** * Returns dot file end string. * * @return string Dot file end string. */ public function closeDotFile() { return '}'; } /** * Generate a graphviz string structure to be used later. * * Usage: * To create a new handmade graph: * Define node struture * key => node source data (agent/module row or custom) * * Minimum required fields in array: * label * status * id * * @param array $nodes Generate dotgraph using defined nodes. * * @return void */ public function generateDotGraph($nodes=false) { if (!isset($this->dotGraph)) { // Generate dot file. $this->nodes = []; $edges = []; $graph = ''; if ($nodes === false) { if (isset($this->rawNodes)) { $nodes = $this->rawNodes; } else { // Search for nodes. $nodes = $this->calculateNodes(); } } // Search for relations. // Build dot structure. // Open Graph. $graph = $this->openDotFile(); if (!$this->noPandoraNode) { // Create empty pandora node to link orphans. $this->nodes[0] = [ 'label' => get_product_name(), 'id_node' => 0, 'id_agente' => 0, 'id_agente_modulo' => 0, 'node_type' => NODE_PANDORA, ]; $this->nodeMapping[0] = 0; $graph .= $this->createDotNode( $this->nodes[0] ); $i = 1; } else { $i = 0; } // Create dot nodes. $orphans = []; foreach ($nodes as $k => $node) { if ((isset($node['type']) && $node['type'] == NODE_AGENT || isset($node['type']) && $node['type'] == NODE_MODULE) || (isset($node['type']) === false && isset($node['id_agente']) === true && $node['id_agente'] > 0) ) { // Origin is agent or module. if (isset($node['type']) && $node['type'] == NODE_MODULE || (isset($node['type']) === false && isset($node['id_agente_modulo']) === true && $node['id_agente_modulo'] > 0) ) { $k = NODE_MODULE.'_'.$k; // Origin is module. $id_source = $node['id_agente_modulo']; $label = io_safe_output($node['nombre']); $status = modules_get_agentmodule_status($node); $this->nodes[$k]['node_type'] = NODE_MODULE; $url = 'index.php?sec=estado&sec2=operation/agentes/ver_agente&id_agente='.$node['id_agente']; $url_tooltip = 'ajax.php?page=operation/agentes/ver_agente&get_agentmodule_status_tooltip=1&id_module='.$node['id_agente_modulo']; } else { // Origin is agent. $k = NODE_AGENT.'_'.$k; $id_source = $node['id_agente']; $label = io_safe_output($node['alias']); $status = agents_get_status_from_counts($node); $this->nodes[$k]['node_type'] = NODE_AGENT; $url = 'index.php?sec=estado&sec2=operation/agentes/ver_agente&id_agente='.$node['id_agente']; $url_tooltip = 'ajax.php?page=operation/agentes/ver_agente&get_agent_status_tooltip=1&id_agent='.$node['id_agente']; } } else { // Handmade node. // Store user node definitions. $k = NODE_GENERIC.'_'.$k; $id_source = $node['id']; $label = $node['label']; $status = $node['status']; $this->nodes[$k]['node_type'] = NODE_GENERIC; // In handmade nodes, edges are defined by using id_parent // Referencing target parent 'id'. $this->nodes[$k]['id_parent'] = $node['id_parent']; $this->nodes[$k]['width'] = $node['width']; $this->nodes[$k]['height'] = $node['height']; $this->nodes[$k]['id_source'] = $node['id_source']; $this->nodes[$k]['shape'] = $node['shape']; $url = $this->node['url']; $url_tooltip = $this->node['url_tooltip']; } $this->nodes[$k]['url'] = $url; $this->nodes[$k]['url_tooltip'] = $url_tooltip; // Fullfill data. // If url is defined in node will be overwritten. foreach ($node as $key => $value) { $this->nodes[$k][$key] = $value; } $graph .= $this->createDotNode( [ 'id_node' => $i, 'id_source' => $id_source, 'label' => $label, 'image' => null, 'radius' => max( $node['width'], $node['height'] ), ] ); // Keep reverse reference. $this->nodeMapping[$i] = $k; $this->nodes[$k]['id_source_data'] = $id_source; $this->nodes[$k]['id_node'] = $i; $this->nodes[$k]['status'] = $status; // Increase for next node. $i++; } // Search for relations. foreach ($this->nodes as $k => $item) { $target = $this->calculateRelations($k); // Adopt all orphan nodes but pandora one. if (empty($target) === true) { if (isset($this->noPandoraNode) === false || $this->noPandoraNode == false ) { if ($item['id_node'] != 0) { $rel = []; $rel['id_parent'] = 0; $rel['id_child'] = $item['id_node']; $rel['parent_type'] = NODE_PANDORA; $rel['child_type'] = $item['node_type']; $rel['id_child_source_data'] = $item['id_source_data']; $orphans[] = $rel; } } } else { // Flattern edges. foreach ($target as $rel) { $edges[] = $rel; } } } if (is_array($edges)) { $array_aux = $edges; $target_aux = $edges; foreach ($edges as $key => $rel) { foreach ($array_aux as $key2 => $rel2) { if ($key2 <= $key) { continue; } if ($rel['child_type'] == 1 && $rel['parent_type'] == 1 && $rel2['child_type'] == 1 && $rel2['parent_type'] == 1 ) { if ($rel['id_parent'] == $rel2['id_parent'] && $rel['id_child'] == $rel2['id_child']) { if ($rel['id_parent_source_data'] == $rel2['id_parent_source_data']) { if (modules_get_agentmodule_type($rel['id_child_source_data']) === 6) { unset($target_aux[$key]); } else if (modules_get_agentmodule_type($rel2['id_child_source_data']) === 6) { unset($target_aux[$key2]); } } else if ($rel['id_child_source_data'] == $rel2['id_child_source_data']) { if (modules_get_agentmodule_type($rel['id_parent_source_data']) === 6) { unset($target_aux[$key]); } else if (modules_get_agentmodule_type($rel2['id_parent_source_data']) === 6) { unset($target_aux[$key2]); } } } else if ($rel['id_parent'] == $rel2['id_child'] && $rel['id_child'] == $rel2['id_parent']) { if ($rel['id_parent_source_data'] == $rel2['id_child_source_data'] && $rel['id_child_source_data'] == $rel2['id_parent_source_data'] ) { unset($target_aux[$key2]); continue; } if ($rel['id_parent_source_data'] == $rel2['id_child_source_data']) { if (modules_get_agentmodule_type($rel['id_child_source_data']) === 6) { unset($target_aux[$key]); } else if (modules_get_agentmodule_type($rel2['id_parent_source_data']) === 6) { unset($target_aux[$key2]); } } else if ($rel['id_child_source_data'] == $rel2['id_parent_source_data']) { if (modules_get_agentmodule_type($rel['id_parent_source_data']) === 6) { unset($target_aux[$key]); } else if (modules_get_agentmodule_type($rel2['id_child_source_data']) === 6) { unset($target_aux[$key2]); } } } } } } $edges = []; foreach ($target_aux as $key => $value) { $edges[] = $value; } foreach ($edges as $rel) { $graph .= $this->createDotEdge( [ 'to' => $rel['id_child'], 'from' => $rel['id_parent'], ] ); } } else { $edges = []; } if (isset($this->noPandoraNode) === false || $this->noPandoraNode == false ) { // Add missed edges. foreach ($orphans as $rel) { $graph .= $this->createDotEdge( [ 'from' => $rel['id_child'], 'to' => $rel['id_parent'], ] ); } // Store relationships. $this->relations = array_merge($edges, $orphans); } else { if (empty($this->relations) === true && empty($this->$edges) === false) { $this->relations = $edges; } } // Close dot file. $graph .= $this->closeDotFile(); $this->dotGraph = $graph; } } /** * Extracts node coordinates and relationships built by graphviz. * * @param string $graphviz_file Graphviz output file path. * * @return mixed Nodes and relations if success. False if not. */ private function parseGraphvizMapFile($graphviz_file) { global $config; if (isset($graphviz_file) === false || is_file($graphviz_file) === false ) { return false; } $content = file($graphviz_file); $nodes = []; $relations = []; foreach ($content as $key => $line) { // Reduce blank spaces. $line = preg_replace('/\ +/', ' ', $line); if (preg_match('/^graph.*$/', $line) != 0) { // Graph definition. $fields = explode(' ', $line); $this->map['width'] = ($fields[2] * GRAPHVIZ_CONVERSION_FACTOR); $this->map['height'] = ($fields[3] * GRAPHVIZ_CONVERSION_FACTOR); if ($this->map['width'] > $config['networkmap_max_width']) { $this->map['width'] = $config['networkmap_max_width']; } if ($this->map['height'] > $config['networkmap_max_width']) { $this->map['height'] = $config['networkmap_max_width']; } } else if (preg_match('/^node.*$/', $line) != 0) { // Node. $fields = explode(' ', $line); $id = $fields[1]; $nodes[$id]['x'] = ($fields[2] * GRAPHVIZ_CONVERSION_FACTOR); $nodes[$id]['y'] = ($fields[3] * GRAPHVIZ_CONVERSION_FACTOR); } else if (preg_match('/^edge.*$/', $line) != 0 && empty($this->relations) === true ) { // Edge. // This is really not needed, because is already defined // in $this->relations. Only for debug purposes. $fields = explode(' ', $line); if (strpos($fields[1], 'transp_') !== false || strpos($fields[2], 'transp_') !== false ) { // Skip transparent nodes relationships. continue; } $relations[] = [ 'id_parent' => $fields[1], 'parent_type' => NODE_GENERIC, 'id_parent_source_data' => $fields[3], 'id_child' => $fields[2], 'child_type' => NODE_GENERIC, 'id_child_source_data' => null, ]; } } // Use current relationship definitions (if exists). if (empty($this->relations) === false) { $relations = $this->relations; } return [ 'nodes' => $nodes, 'relations' => $relations, ]; } /** * Calculates X,Y positions foreach element defined in dotGraph. * * @return array Structure parsed. */ public function calculateCoords() { global $config; switch (PHP_OS) { case 'WIN32': case 'WINNT': case 'Windows': $filename_dot = sys_get_temp_dir()."\\networkmap_".$this->filter; break; default: $filename_dot = sys_get_temp_dir().'/networkmap_'.$this->filter; break; } if ($this->mapOptions['simple']) { $filename_dot .= '_simple'; } if ($this->mapOptions['nooverlap']) { $filename_dot .= '_nooverlap'; } $filename_dot .= uniqid().'_'.$this->idMap.'.dot'; file_put_contents($filename_dot, $this->dotGraph); $plain_file = 'plain'.uniqid().'.txt'; switch (PHP_OS) { case 'WIN32': case 'WINNT': case 'Windows': $filename_plain = sys_get_temp_dir().'\\'.$plain_file; $cmd = io_safe_output( $config['graphviz_bin_dir'].'\\'.$this->filter.'.exe -Tplain -o '.$filename_plain.' '.$filename_dot ); break; default: $filename_plain = sys_get_temp_dir().'/'.$plain_file; $cmd = $this->filter.' -Tplain -o '.$filename_plain.' '.$filename_dot; break; } $retval = 0; $r = system($cmd, $retval); if ($retval != 0) { ui_print_error_message( __('Failed to generate dotmap, please select different layout schema') ); return []; } unlink($filename_dot); if (empty($this->customParser) === false && function_exists($this->customParser) ) { try { if (empty($this->customParserArgs)) { $graph = call_user_func( $this->customParser, $filename_plain, $this->dotGraph ); } else { $graph = call_user_func( $this->customParser, $filename_plain, $this->dotGraph, $this->customParserArgs ); } } catch (Exception $e) { // If developer is using a custom method to parse graphviz // results, but want to handle using default parser // or custom based on data, it is possible to launch // exceptions to control internal flow. if ($this->fallbackDefaultParser === true) { $graph = $this->parseGraphvizMapFile( $filename_plain ); } else { ui_print_error_message($e->getMessage()); $graph = []; } } } else { $graph = $this->parseGraphvizMapFile( $filename_plain ); } if (isset($this->mapOptions['map_filter']['node_separation']) === true) { foreach ($graph['nodes'] as $key => $value) { $graph['nodes'][$key]['x'] *= $this->mapOptions['map_filter']['node_separation']; $graph['nodes'][$key]['y'] *= $this->mapOptions['map_filter']['node_separation']; } } unlink($filename_plain); /* * Graphviz section ends here. */ return $graph; } /** * Creates an empty dot graph (with only base node) * * @return void */ public function generateEmptyDotGraph() { // Create an empty map dot structure. $graph = $this->openDotFile(); $this->nodes[0] = [ 'label' => get_product_name(), 'id_node' => 0, 'id_agente' => 0, 'id_agente_modulo' => 0, 'node_type' => NODE_PANDORA, ]; $this->nodeMapping[0] = 0; $graph .= $this->createDotNode( $this->nodes[0] ); $graph .= $this->closeDotFile(); $this->dotGraph = $graph; } /** * Generates a nodes - relationships array using graphviz dot * schema and stores nodes&relations into $this->graph. * * @return void */ public function generateNetworkMap() { global $config; include_once 'include/functions_os.php'; $map_filter = $this->mapOptions['map_filter']; /* * Let graphviz place the nodes. */ if ($map_filter['empty_map']) { $this->generateEmptyDotGraph(); } else if (!isset($this->dotGraph)) { $this->generateDotGraph(); } /* * Calculate X,Y positions. */ if (!$this->mapOptions['fixed_positions']) { $graph = $this->calculateCoords(); } else { // Set by user. $graph['nodes'] = $this->rawNodes; $graph['relations'] = $this->relations; $this->map['width'] = $this->mapOptions['width']; $this->map['height'] = $this->mapOptions['height']; } if (is_string($this->map['filter']) === true) { $this->map['filter'] = json_decode($this->map['filter'], true); if (json_last_error() !== JSON_ERROR_NONE) { $this->map['filter'] = []; } } $this->map['filter']['z_dash'] = $this->mapOptions['z_dash']; if (is_array($graph) === true) { $nodes = $graph['nodes']; $relations = $graph['relations']; } else { ui_print_error_message( __('Failed to retrieve graph data.') ); return; } /* * Calculate references. */ $index = 0; $node_center = []; $graph = []; $graph['nodes'] = []; // Prepare graph nodes. foreach ($nodes as $id => $coords) { $node_tmp['id_map'] = $this->idMap; $node_tmp['id'] = $id; $source = $this->getNodeData($id); $node_tmp['id_agent'] = $source['id_agente']; $node_tmp['id_module'] = $source['id_agente_modulo']; $node_tmp['type'] = $source['node_type']; $node_tmp['x'] = $coords['x']; $node_tmp['y'] = $coords['y']; $node_tmp['width'] = $this->mapOptions['map_filter']['node_radius']; $node_tmp['height'] = $this->mapOptions['map_filter']['node_radius']; if (isset($source['width'])) { $node_tmp['width'] = $source['width']; } if (isset($source['height'])) { $node_tmp['height'] = $source['height']; } switch ($node_tmp['type']) { case NODE_AGENT: $node_tmp['source_data'] = $source['id_agente']; $node_tmp['text'] = $source['alias']; $node_tmp['image'] = ui_print_os_icon( $source['id_os'], false, true, true, false, true, true ); break; case NODE_MODULE: $node_tmp['source_data'] = $source['id_agente_modulo']; $node_tmp['text'] = $source['nombre']; $node_tmp['image'] = ui_print_moduletype_icon( $this->getNodeData($id, 'id_tipo_modulo'), true, true, false, true ); break; case NODE_PANDORA: $node_tmp['text'] = $source['label']; $node_tmp['id_agent'] = $source['id_agente']; $node_tmp['id_module'] = $source['id_agente_modulo']; $node_tmp['source_data'] = 0; $node_center['x'] = ($coords['x'] - MAP_X_CORRECTION); $node_center['y'] = ($coords['y'] - MAP_Y_CORRECTION); break; case NODE_GENERIC: default: $node_tmp['text'] = $source['label']; $node_tmp['id_agent'] = $source['id_agente']; $node_tmp['id_module'] = $source['id_agente_modulo']; $node_tmp['source_data'] = $source['id_source']; $node_tmp['image'] = $source['image']; break; } $style = []; $style['shape'] = $source['shape']; if (isset($style['shape']) === false) { $style['shape'] = 'circle'; } $style['image'] = $node_tmp['image']; $style['width'] = $node_tmp['width']; $style['height'] = $node_tmp['height']; $style['radius'] = max( $style['width'], $style['height'] ); $style['label'] = $node_tmp['text']; $node_tmp['style'] = json_encode($style); $graph['nodes'][$index] = $node_tmp; $index++; } // Prepare graph edges and clean double references. $graph['relations'] = []; $parents = []; foreach ($relations as $rel) { $tmp = [ 'id_map' => $this->idMap, 'id_parent' => $rel['id_parent'], 'parent_type' => $rel['parent_type'], 'id_parent_source_data' => $rel['id_parent_source_data'], 'id_child' => $rel['id_child'], 'child_type' => $rel['child_type'], 'id_child_source_data' => $rel['id_child_source_data'], 'id_parent_agent' => $rel['id_parent_agent'], 'id_child_agent' => $rel['id_child_agent'], 'link_color' => $rel['link_color'], 'text_start' => $rel['text_start'], 'text_end' => $rel['text_end'], ]; $found = 0; if (isset($tmp['id_parent_source_data'])) { // Avoid [child - parent] : [parent - child] relation duplicates. if (is_array($parents[$tmp['id_parent_source_data']])) { foreach ($parents[$tmp['id_parent_source_data']] as $k) { if ($k === $tmp['id_child_source_data']) { $found = 1; break; } } } else { $parents[$tmp['id_parent_source_data']] = []; } } if ($found == 0) { $parents[$tmp['id_child_source_data']][] = $tmp['id_parent_source_data']; $graph['relations'][] = $tmp; } } // Prioritize relations between same nodes. $this->cleanGraphRelations(); // Save data. if ($this->idMap > 0 && (isset($this->map['__simulated']) === false)) { $graph = save_generate_nodes($this->idMap, $graph); db_process_sql_update( 'tmap', [ 'width' => $this->map['width'], 'height' => $this->map['height'], 'center_x' => $this->map['center_x'], 'center_y' => $this->map['center_y'], ], ['id' => $this->idMap] ); } else { $this->map['center_x'] = $node_center['x']; $this->map['center_y'] = $node_center['y']; if (!isset($this->map['center_x']) && !isset($this->map['center_y']) ) { $this->map['center_x'] = ($nodes[0]['x'] - MAP_X_CORRECTION); $this->map['center_y'] = ($nodes[0]['y'] - MAP_Y_CORRECTION); } } $this->graph = $graph; } /** * Regenerates a nodes - relationships array using graphviz dot * schema and stores nodes&relations into $this->graph. * * @return object */ public function recalculateCoords() { global $config; include_once 'include/functions_os.php'; $map_filter = $this->mapOptions['map_filter']; /* * Let graphviz place the nodes. */ if ($map_filter['empty_map']) { $this->generateEmptyDotGraph(); } else if (!isset($this->dotGraph)) { $this->generateDotGraph(); } $graph = $this->calculateCoords(); if (is_array($graph) === true) { $nodes = $graph['nodes']; } else { return []; } $nodes_aux = []; // Prepare graph nodes. foreach ($nodes as $id => $coords) { $node_tmp['id'] = $id; $source = $this->getNodeData($id); $node_tmp['type'] = $source['node_type']; $node_tmp['x'] = $coords['x']; $node_tmp['y'] = $coords['y']; switch ($node_tmp['type']) { case NODE_AGENT: $node_tmp['source_data'] = $source['id_agente']; break; case NODE_MODULE: $node_tmp['source_data'] = $source['id_agente_modulo']; break; case NODE_PANDORA: $node_tmp['source_data'] = 0; $node_center['x'] = ($coords['x'] - MAP_X_CORRECTION); $node_center['y'] = ($coords['y'] - MAP_Y_CORRECTION); break; case NODE_GENERIC: default: $node_tmp['source_data'] = $source['id_source']; break; } $nodes_aux[$index] = $node_tmp; $index++; } return $nodes_aux; } /** * Transform node information into JS data. * * @return string HTML code with JS data. */ public function loadMapData() { global $config; $networkmap = $this->map; // ACL. $networkmap_write = check_acl( $config['id_user'], $networkmap['id_group'], 'MW' ); if (isset($networkmap['__simulated']) === false) { if ($this->widget) { $networkmap['filter'] = $this->mapOptions; } else if (is_string($networkmap['filter']) === true) { $networkmap['filter'] = json_decode( $networkmap['filter'], true ); } $networkmap['filter']['holding_area'] = [ 500, 500, ]; $holding_area_title = __('Holding Area'); } else { $holding_area_title = ''; $networkmap['filter']['holding_area'] = [ 0, 0, ]; } // Prioritize relations between same nodes. $this->cleanGraphRelations(); // Print some params to handle it in js. html_print_input_hidden('widget', $this->widget); html_print_input_hidden('product_name', get_product_name()); html_print_input_hidden('center_logo', ui_get_full_url(ui_get_logo_to_center_networkmap())); $output = ''; return $output; } /** * Generates a simple interface to interact with nodes. * * @return string HTML code for simple interface. */ public function loadSimpleInterface() { $output = ''; $output .= ''; return $output; } /** * Show an advanced interface to manage dialogs. * * @return string HTML code with dialogs. */ public function loadAdvancedInterface() { $list_networkmaps = get_networkmaps($this->idMap); if (empty($list_networkmaps) === true) { $list_networkmaps = []; } $id = 'dialog_node_edit'; $output = ''; $output .= ''; $output .= ''; return $output; } /** * Loads advanced map controller (JS). * * @param boolean $return Dumps to output if false. * * @return string HTML code for advanced controller. */ public function loadController(?bool $return=true) { if (isset($this->mapOptions['refresh_time']) === false) { $this->mapOptions['refresh_time'] = 0; } $output = ''; if ($this->useTooltipster ) { $output .= ''; } else { // Generate JS for advanced controller. $output .= ' '; } if ($return === false) { echo $output; } return $output; } /** * Load networkmap HTML skel and JS requires. * * @return string HTML code for skel. */ public function loadMapSkel() { global $config; if (isset($this->useTooltipster) === true && (bool) $this->useTooltipster === true ) { $output = ''; $output .= ''; $output .= ''; $output .= ''; $output .= ''; $output .= ''."\n"; ui_require_css_file('jquery.contextMenu', 'include/styles/js/'); $output .= ''; $output .= '
fullSize) { $output .= ' width:100%'; $output .= ' ;height: 100%">'; $output .= ''; } else { $output .= ' width:'.$this->mapOptions['width'].'px'; $output .= ' ;height:'.$this->mapOptions['height'].'px">'; $output .= ''; } $output .= ''; $output .= '
'; } else { // Load default interface. ui_require_css_file('networkmap'); ui_require_css_file('jquery.contextMenu', 'include/styles/js/'); $output = ''; $minimap_display = ''; if ($this->mapOptions['pure']) { $minimap_display = 'none'; } $networkmap = $this->map; if (is_array($networkmap['filter']) === false) { $networkmap['filter'] = json_decode($networkmap['filter'], true); } $networkmap['filter']['l2_network_interfaces'] = 1; $output .= ''; if (isset($this->map['__simulated']) === false) { // Load context menu if manageable networkmap. $output .= ''; } $output .= ''; // Open networkconsole_id div. $output .= '
fullSize) { $output .= ' class="networkconsole">'; } else { $output .= ' style="width: '.$this->mapOptions['width'].'px; height: '.$this->mapOptions['height'].'px;position: relative; overflow: hidden; background: #FAFAFA">'; } $output .= ''; $output .= '
'; $output .= ''; $output .= html_print_image('/images/minimap_open_arrow.png', true, ['id' => 'arrow_minimap_'.$networkmap['id']]); $output .= '
'; $output .= '
'; $output .= html_print_image('/images/disable.svg', true, ['id' => 'image_hide_show_labels', 'class' => 'main_menu_icon invert_filter']); $output .= '
'; $output .= '
'image_hide_show_labels']); $output .= '
'; // Close networkconsole_id div. $output .= "\n"; } return $output; } /** * Print all components required to visualizate a network map. * * @param boolean $return Return as string or not. * * @return string HTML code. */ public function printMap($return=false, $ignore_acl=false) { global $config; $networkmap = $this->map; if ($ignore_acl === false) { // ACL. $networkmap_read = check_acl( $config['id_user'], $networkmap['id_group'], 'MR' ); $networkmap_write = check_acl( $config['id_user'], $networkmap['id_group'], 'MW' ); $networkmap_manage = check_acl( $config['id_user'], $networkmap['id_group'], 'MM' ); if (!$networkmap_read && !$networkmap_write && !$networkmap_manage ) { db_pandora_audit( AUDIT_LOG_ACL_VIOLATION, 'Trying to access networkmap' ); include 'general/noaccess.php'; return ''; } } $user_readonly = !$networkmap_write && !$networkmap_manage; if (isset($this->idMap) && isset($this->map['__simulated']) === false ) { $output .= $this->loadMapSkel(); $output .= $this->loadMapData(); $output .= $this->loadController(); if (!$this->noPopUp) { $output .= $this->loadAdvancedInterface(); } } else { // Simulated, no tmap entries. $output .= $this->loadMapSkel(); $output .= $this->loadMapData(); $output .= $this->loadController(); if (!$this->noPopUp) { $output .= $this->loadSimpleInterface(); } } $output .= ' '; if ($return === false) { echo $output; } return $output; } }