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 .= '