Status: " + + textStatus + + "
" + + "Error: " + + errorThrown + + "
" + + XMLHttpRequest.responseText + + "
" + ); } }); } @@ -499,6 +513,7 @@ function checkSNMPVersion() { function snmpBrowserWindow() { // Keep elements in the form and the SNMP browser synced $("#text-target_ip").val($("#text-ip_target").val()); + $("#target_port").val($("#text-tcp_port").val()); $("#text-community").val($("#text-snmp_community").val()); $("#snmp_browser_version").val($("#snmp_version").val()); $("#text-snmp3_browser_auth_user").val($("#text-snmp3_auth_user").val()); @@ -525,7 +540,7 @@ function snmpBrowserWindow() { opacity: 0.5, background: "black" }, - width: 920, + width: 1000, height: 500 }); } @@ -561,6 +576,7 @@ function snmp_browser_create_modules(module_target, return_post = true) { .get(); var target_ip = $("#text-target_ip").val(); + var target_port = $("#target_port").val(); var community = $("#text-community").val(); var snmp_version = $("#snmp_browser_version").val(); var snmp3_auth_user = $("#text-snmp3_browser_auth_user").val(); @@ -590,6 +606,7 @@ function snmp_browser_create_modules(module_target, return_post = true) { var snmp_conf = {}; snmp_conf["target_ip"] = target_ip; + snmp_conf["target_port"] = target_port; snmp_conf["community"] = community; snmp_conf["oids"] = oids; snmp_conf["snmp_browser_version"] = snmp_version; diff --git a/pandora_console/include/javascript/pandora_ui.js b/pandora_console/include/javascript/pandora_ui.js index 35d46f79f0..880fd77b99 100644 --- a/pandora_console/include/javascript/pandora_ui.js +++ b/pandora_console/include/javascript/pandora_ui.js @@ -87,7 +87,12 @@ function load_modal(settings) { div.id = "div-modal-" + uniq; div.style.display = "none"; - document.getElementById("main").append(div); + if (document.getElementById("main") == null) { + // MC env. + document.getElementById("page").append(div); + } else { + document.getElementById("main").append(div); + } var id_modal_target = "#div-modal-" + uniq; @@ -173,138 +178,148 @@ function load_modal(settings) { } if (settings.modal.ok != undefined) { + var btnClickHandler = function(d) { + if (AJAX_RUNNING) return; + if (settings.onsubmit != undefined) { + if (settings.onsubmit.preaction != undefined) { + settings.onsubmit.preaction(); + } + AJAX_RUNNING = 1; + if (settings.onsubmit.dataType == undefined) { + settings.onsubmit.dataType = "html"; + } + + var formdata = new FormData(); + if (settings.extradata) { + settings.extradata.forEach(function(item) { + if (item.value != undefined) formdata.append(item.name, item.value); + }); + } + formdata.append("page", settings.onsubmit.page); + formdata.append("method", settings.onsubmit.method); + + var flagError = false; + if (Array.isArray(settings.form) === false) { + $("#" + settings.form + " :input").each(function() { + if (this.checkValidity() === false) { + $(this).attr("title", this.validationMessage); + $(this).tooltip({ + tooltipClass: "uitooltip", + position: { + my: "right bottom", + at: "right top", + using: function(position, feedback) { + $(this).css(position); + $("" + + // item.props.labelStart + + // "
" + + // item.props.labelEnd + + // "" + // }); } }); // VC Item moved. @@ -203,7 +234,7 @@ function createVisualConsole( y: e.newPosition.y, type: e.item.props.type }; - if (e.item.props.type === 13) { + if (e.item.props.type === 13 || e.item.props.type === 21) { var startIsLeft = e.item.props.startPosition.x - e.item.props.endPosition.x <= 0; var startIsTop = @@ -279,6 +310,7 @@ function createVisualConsole( endX: e.endPosition.x, endY: e.endPosition.y }; + var taskId = "visual-console-item-update-" + id; // Persist the new position. @@ -290,18 +322,14 @@ function createVisualConsole( id, data, function(error, data) { - // if (!error && !data) return; - if (error || !data) { - console.log( - "[ERROR]", - "[VISUAL-CONSOLE-CLIENT]", - "[API]", - error ? error.message : "Invalid response" - ); + if (!error && !data) return; - // TODO: Move the element to its initial position. + try { + var decoded_data = JSON.parse(data); + visualConsole.updateElement(decoded_data); + } catch (error) { + console.error(error); } - done(); } ); @@ -427,6 +455,7 @@ function createVisualConsole( }, createItem: function(typeString) { var type; + console.log(typeString); switch (typeString) { case "STATIC_GRAPH": type = 0; @@ -479,6 +508,9 @@ function createVisualConsole( case "COLOR_CLOUD": type = 20; break; + case "NETWORK_LINK": + type = 21; + break; default: type = 0; } @@ -565,7 +597,7 @@ function createVisualConsole( item.setMeta({ isUpdating: false }); var itemRetrieved = item.props; - if (itemRetrieved["type"] == 13) { + if (itemRetrieved["type"] == 13 || itemRetrieved["type"] == 21) { var startIsLeft = itemRetrieved["startPosition"]["x"] - itemRetrieved["endPosition"]["x"] <= @@ -1179,6 +1211,9 @@ function createOrUpdateVisualConsoleItem( case 20: nameType = "Color Cloud"; break; + case 21: + nameType = "Network Link"; + break; default: nameType = "Static graph"; @@ -1259,7 +1294,8 @@ function createOrUpdateVisualConsoleItem( tinyMCE != undefined && tinyMCE.editors.length > 0 && item.itemProps.type != 12 && - item.itemProps.type != 13 + item.itemProps.type != 13 && + item.itemProps.type != 21 ) { // Content tiny. var label = tinyMCE.activeEditor.getContent(); diff --git a/pandora_console/include/javascript/tree/TreeController.js b/pandora_console/include/javascript/tree/TreeController.js index 17f0539848..ee8de7a5d9 100644 --- a/pandora_console/include/javascript/tree/TreeController.js +++ b/pandora_console/include/javascript/tree/TreeController.js @@ -1073,7 +1073,25 @@ var TreeController = { $node.append($group); } - _.each(data.tree, function(element) { + // Get the main values of the tree. + var rawTree = Object.values(data.tree); + // Sorting tree by description (services.treeview_services.php). + rawTree.sort(function(a, b) { + // Only the services are ordered since only they have the elementDescription property. + if (a.elementDescription && b.elementDescription) { + var x = a.elementDescription.toLowerCase(); + var y = b.elementDescription.toLowerCase(); + if (x < y) { + return -1; + } + if (x > y) { + return 1; + } + } + return 0; + }); + + _.each(rawTree, function(element) { element.jqObject = _processNode($group, element); }); diff --git a/pandora_console/include/lib/Agent.php b/pandora_console/include/lib/Agent.php index f5535e72d2..3d97b86c00 100644 --- a/pandora_console/include/lib/Agent.php +++ b/pandora_console/include/lib/Agent.php @@ -419,13 +419,126 @@ class Agent extends Entity } + /** + * Return a list of interfaces. + * + * @param array $filter Filter interfaces by name in array. + * + * @return array Of interfaces and modules PandoraFMS\Modules. + */ + public function getInterfaces(array $filter=[]) + { + $modules = $this->searchModules( + ['nombre' => '%ifOperStatus%'] + ); + + $interfaces = []; + foreach ($modules as $module) { + $matches = []; + if (preg_match( + '/^(.*?)_ifOperStatus$/', + $module->name(), + $matches + ) > 0 + ) { + $interface = $matches[1]; + } + + if (empty($interface) === true) { + continue; + } + + if (empty($filter) === false + && in_array($interface, $filter) !== true + ) { + continue; + } + + $name_filters = [ + 'ifOperStatus' => ['nombre' => $interface.'_ifOperStatus'], + 'ifInOctets' => ['nombre' => $interface.'_ifInOctets'], + 'ifOutOctets' => ['nombre' => $interface.'_ifOutOctets'], + 'ifHCInOctets' => ['nombre' => $interface.'_ifHCInOctets'], + 'ifHCOutOctets' => ['nombre' => $interface.'_ifHCOutOctets'], + ]; + + $ifOperStatus = $this->searchModules( + $name_filters['ifOperStatus'] + ); + $ifInOctets = $this->searchModules( + $name_filters['ifInOctets'] + ); + $ifOutOctets = $this->searchModules( + $name_filters['ifOutOctets'] + ); + $ifHCInOctets = $this->searchModules( + $name_filters['ifHCInOctets'] + ); + $ifHCOutOctets = $this->searchModules( + $name_filters['ifHCOutOctets'] + ); + + $interfaces[$interface] = [ + 'ifOperStatus' => array_shift($ifOperStatus), + 'ifInOctets' => array_shift($ifInOctets), + 'ifOutOctets' => array_shift($ifOutOctets), + 'ifHCInOctets' => array_shift($ifHCInOctets), + 'ifHCOutOctets' => array_shift($ifHCOutOctets), + ]; + } + + return $interfaces; + } + + + /** + * Retrieves status, in and out modules from given interface name. + * + * @param string $interface Interface name. + * + * @return array|null With status, in and out modules. Null if no iface. + */ + public function getInterfaceMetrics(string $interface):?array + { + $modules = $this->getInterfaces([$interface]); + if (empty($modules) === true) { + return null; + } + + $modules = $modules[$interface]; + + $in = null; + $out = null; + $status = $modules['ifOperStatus']; + + if (empty($modules['ifHCInOctets']) === false) { + $in = $modules['ifHCInOctets']; + } else if (empty($modules['ifInOctets']) === false) { + $in = $modules['ifInOctets']; + } + + if (empty($modules['ifHCOutOctets']) === false) { + $out = $modules['ifHCOutOctets']; + } else if (empty($modules['ifOutOctets']) === false) { + $out = $modules['ifOutOctets']; + } + + return [ + 'in' => $in, + 'out' => $out, + 'status' => $status, + ]; + + } + + /** * Search for modules into this agent. * * @param array $filter Filters. * @param integer $limit Limit search results. * - * @return PandoraFMS\Module Module found. + * @return array Of PandoraFMS\Module Modules found. */ public function searchModules(array $filter, int $limit=0) { @@ -452,7 +565,12 @@ class Agent extends Entity return $results; } else { // Search in db. - return Module::search($filter, $limit); + $return = Module::search($filter, $limit); + if (is_array($return) === false) { + return []; + } + + return $return; } } diff --git a/pandora_console/include/lib/Dashboard/Manager.php b/pandora_console/include/lib/Dashboard/Manager.php index eebfc3233c..891ef524df 100644 --- a/pandora_console/include/lib/Dashboard/Manager.php +++ b/pandora_console/include/lib/Dashboard/Manager.php @@ -1008,6 +1008,7 @@ class Manager 'dashboardName' => $this->dashboardFields['name'], 'hash' => self::generatePublicHash(), 'publicLink' => $this->publicLink, + 'dashboardGroup' => $this->dashboardFields['id_group'], ] ); } else { @@ -1025,6 +1026,7 @@ class Manager 'cells' => $this->cells, 'cellModeSlides' => $this->cellModeSlides, 'cellId' => ($this->cellId === 0) ? $this->cells[0]['id'] : $this->cellId, + 'dashboardGroup' => $this->dashboardFields['id_group'], ] ); } diff --git a/pandora_console/include/lib/Dashboard/Widget.php b/pandora_console/include/lib/Dashboard/Widget.php index bfc640eaf0..7168a3afe9 100644 --- a/pandora_console/include/lib/Dashboard/Widget.php +++ b/pandora_console/include/lib/Dashboard/Widget.php @@ -571,9 +571,9 @@ class Widget } if ($this->width === 0) { - $width = (((int) $this->position['width'] / 12 * $gridWidth) - 50); + $width = (((int) $this->position['width'] / 12 * $gridWidth) - 60); } else { - $width = (((int) $this->width / 12 * $gridWidth) - 50); + $width = (((int) $this->width / 12 * $gridWidth) - 60); } if ($this->height === 0) { diff --git a/pandora_console/include/lib/Dashboard/Widgets/agent_module.php b/pandora_console/include/lib/Dashboard/Widgets/agent_module.php index 6bb1bf4e37..7920be767f 100644 --- a/pandora_console/include/lib/Dashboard/Widgets/agent_module.php +++ b/pandora_console/include/lib/Dashboard/Widgets/agent_module.php @@ -246,18 +246,26 @@ class AgentModuleWidget extends Widget 'label' => __('Filter modules'), ]; + $return_all_group = false; + + if (users_can_manage_group_all('RM') || $this->values['mGroup'] == 0) { + $return_all_group = true; + } + $inputs[] = [ 'class' => 'flex flex-row', 'id' => 'select_multiple_modules_filtered', 'arguments' => [ - 'type' => 'select_multiple_modules_filtered', - 'uniqId' => $this->cellId, - 'mGroup' => $this->values['mGroup'], - 'mRecursion' => $this->values['mRecursion'], - 'mModuleGroup' => $this->values['mModuleGroup'], - 'mAgents' => $this->values['mAgents'], - 'mShowCommonModules' => $this->values['mShowCommonModules'], - 'mModules' => $this->values['mModules'], + 'type' => 'select_multiple_modules_filtered', + 'uniqId' => $this->cellId, + 'mGroup' => $this->values['mGroup'], + 'mRecursion' => $this->values['mRecursion'], + 'mModuleGroup' => $this->values['mModuleGroup'], + 'mAgents' => $this->values['mAgents'], + 'mShowCommonModules' => $this->values['mShowCommonModules'], + 'mModules' => $this->values['mModules'], + 'mShowSelectedOtherGroups' => true, + 'mReturnAllGroup' => $return_all_group, ], ]; @@ -326,10 +334,6 @@ class AgentModuleWidget extends Widget } foreach ($agents as $agent) { - if (users_access_to_agent($agent['id_agente']) === false) { - continue; - } - $row = []; $row['agent_status'] = agents_get_status( $agent['id_agente'], @@ -337,10 +341,18 @@ class AgentModuleWidget extends Widget ); $row['agent_name'] = $agent['nombre']; $row['agent_alias'] = $agent['alias']; - $agent_modules = agents_get_modules( + + $sql = sprintf( + 'SELECT id_agente_modulo, nombre + FROM tagente_modulo + WHERE id_agente = %d', $agent['id_agente'] ); + $agent_modules = db_get_all_rows_sql($sql); + + $agent_modules = array_combine(array_column($agent_modules, 'id_agente_modulo'), array_column($agent_modules, 'nombre')); + $row['modules'] = []; foreach ($modules_by_name as $module) { $row['modules'][$module['name']] = null; diff --git a/pandora_console/include/lib/Dashboard/Widgets/alerts_fired.php b/pandora_console/include/lib/Dashboard/Widgets/alerts_fired.php index 2a248d7306..fdb5c3648a 100755 --- a/pandora_console/include/lib/Dashboard/Widgets/alerts_fired.php +++ b/pandora_console/include/lib/Dashboard/Widgets/alerts_fired.php @@ -215,13 +215,19 @@ class AlertsFiredWidget extends Widget // Retrieve global - common inputs. $inputs = parent::getFormInputs(); + $return_all_group = false; + + if (users_can_manage_group_all('RM') || $values['groupId'] == 0) { + $return_all_group = true; + } + // Groups. $inputs[] = [ 'label' => __('Group'), 'arguments' => [ 'type' => 'select_groups', 'name' => 'groupId', - 'returnAllGroup' => true, + 'returnAllGroup' => $return_all_group, 'privilege' => 'AR', 'selected' => $values['groupId'], 'return' => true, diff --git a/pandora_console/include/lib/Dashboard/Widgets/custom_graph.php b/pandora_console/include/lib/Dashboard/Widgets/custom_graph.php index 7ab6e7c0fe..e883dd0ec2 100644 --- a/pandora_console/include/lib/Dashboard/Widgets/custom_graph.php +++ b/pandora_console/include/lib/Dashboard/Widgets/custom_graph.php @@ -243,8 +243,23 @@ class CustomGraphWidget extends Widget $values['showLegend'] = 1; } + $return_all_group = false; + + if (users_can_manage_group_all('RM')) { + $return_all_group = true; + } + // Custom graph. - $fields = \custom_graphs_get_user(); + $fields = \custom_graphs_get_user(0, false, $return_all_group); + + // If currently selected graph is not included in fields array (it belongs to a group over which user has no permissions), then add it to fields array. + // This is aimed to avoid overriding this value when a user with narrower permissions edits widget configuration. + if ($values['id_graph'] !== null && !array_key_exists($values['id_graph'], $fields)) { + $selected_graph = db_get_row('tgraph', 'id_graph', $values['id_graph']); + + $fields[$values['id_graph']] = $selected_graph; + } + $inputs[] = [ 'label' => __('Graph'), 'arguments' => [ @@ -351,11 +366,10 @@ class CustomGraphWidget extends Widget ); $hackLegendHight = (30 * count($sources)); - if ($hackLegendHight < ($size['height'] - 10 - $hackLegendHight)) { - $height = ($size['height'] - 10 - $hackLegendHight); + if ($hackLegendHight > ($size['height'] - 10 - $hackLegendHight)) { + $height = ($size['height'] - $hackLegendHight); } else { $height = ($size['height'] - 10); - $this->values['showLegend'] = 0; } } else { $height = ($size['height'] - 10); @@ -381,6 +395,11 @@ class CustomGraphWidget extends Widget break; } + // Not posible height < 0. + if ($height <= 0) { + $height = 10; + } + $params = [ 'period' => $this->values['period'], 'width' => ($size['width'] - 10), @@ -392,6 +411,7 @@ class CustomGraphWidget extends Widget 'menu' => false, 'show_legend' => $this->values['showLegend'], 'vconsole' => true, + 'dashboard' => true, ]; $params_combined = [ diff --git a/pandora_console/include/lib/Dashboard/Widgets/events_list.php b/pandora_console/include/lib/Dashboard/Widgets/events_list.php index ad19a48d43..5cce5ae1d7 100644 --- a/pandora_console/include/lib/Dashboard/Widgets/events_list.php +++ b/pandora_console/include/lib/Dashboard/Widgets/events_list.php @@ -364,6 +364,14 @@ class EventsListWidget extends Widget ], ]; + $return_all_group = false; + $selected_groups_array = explode(',', $values['groupId'][0]); + + if (users_can_manage_group_all('RM') || ($selected_groups_array[0] !== '' && in_array(0, $selected_groups_array) === true)) { + // Return all group if user has permissions or it is a currently selected group. + $return_all_group = true; + } + // Groups. $inputs[] = [ 'label' => __('Groups'), @@ -372,9 +380,10 @@ class EventsListWidget extends Widget 'name' => 'groupId[]', 'returnAllGroup' => true, 'privilege' => 'AR', - 'selected' => explode(',', $values['groupId'][0]), + 'selected' => $selected_groups_array, 'return' => true, 'multiple' => true, + 'returnAllGroup' => $return_all_group, ], ]; @@ -429,7 +438,14 @@ class EventsListWidget extends Widget global $config; $output = ''; - $user_groups = \users_get_groups(); + + $return_all_group = false; + + if (users_can_manage_group_all('RM')) { + $return_all_group = true; + } + + $user_groups = \users_get_groups(false, 'AR', $return_all_group); ui_require_css_file('events', 'include/styles/', true); ui_require_css_file('tables', 'include/styles/', true); @@ -442,14 +458,6 @@ class EventsListWidget extends Widget return $output; } - foreach ($this->values['groupId'] as $id_group) { - // Sanity check for user access. - if (isset($user_groups[$id_group]) === false) { - $output .= __('You must select some group'); - return; - } - } - $useTags = \tags_has_user_acl_tags($config['id_user']); if ($useTags) { if (empty($this->values['tagsId']) === true) { @@ -465,9 +473,20 @@ class EventsListWidget extends Widget $filter = []; // Group all. if (in_array(0, $this->values['groupId'])) { - $filter['id_grupo'] = array_keys(users_get_groups()); + $filter['id_grupo'] = array_keys($user_groups); } else { - $filter['id_grupo'] = $this->values['groupId']; + $filter['id_grupo'] = array_intersect($this->values['groupId'], array_keys($user_groups)); + } + + if (empty($filter['id_grupo'])) { + $output .= '
Et)if(f>Ot-Et)a.moveTo(h*_t(d),h*vt(d)),a.arc(0,0,h,d,m,!_),p>Et&&(a.moveTo(p*_t(m),p*vt(m)),a.arc(0,0,p,m,d,_));else{var y,b,v=d,g=m,E=d,x=m,w=f,O=f,k=o.apply(this,arguments)/2,M=k>Et&&(i?+i.apply(this,arguments):gt(p*p+h*h)),A=bt(mt(h-p)/2,+n.apply(this,arguments)),j=A,T=A;if(M>Et){var C=kt(M/p*vt(k)),P=kt(M/h*vt(k));(w-=2*C)>Et?(E+=C*=_?1:-1,x-=C):(w=0,E=x=(d+m)/2),(O-=2*P)>Et?(v+=P*=_?1:-1,g-=P):(O=0,v=g=(d+m)/2)}var S=h*_t(v),I=h*vt(v),L=p*_t(x),R=p*vt(x);if(A>Et){var N,D=h*_t(g),z=h*vt(g),B=p*_t(E),H=p*vt(E);if(f
\");\n labelStartWidth = 0;\n lines.forEach(l => {\n if (l.length > labelStartWidth) {\n labelStartWidth = l.length * fontsize;\n }\n });\n if (labelStartHeight <= 0) {\n labelStartHeight = lines.length * fontheight;\n }\n }\n\n if (labelEndWidth <= 0) {\n let lines = labelEnd.split(\"
\");\n labelEndWidth = 0;\n lines.forEach(l => {\n if (l.length > labelEndWidth) {\n labelEndWidth = l.length * fontsize;\n }\n });\n if (labelEndHeight <= 0) {\n labelEndHeight = lines.length * fontheight;\n }\n }\n\n if (x1 < x2) {\n // x1 on left of x2.\n x1 += adjustment;\n x2 -= adjustment + labelEndWidth;\n }\n\n if (x1 > x2) {\n // x1 on right of x2.\n x1 -= adjustment + labelStartWidth;\n x2 += adjustment;\n }\n\n if (y1 < y2) {\n // y1 on y2.\n y1 += adjustment;\n y2 -= adjustment + labelEndHeight;\n }\n\n if (y1 > y2) {\n // y1 under y2.\n y1 -= adjustment + labelStartHeight;\n y2 += adjustment;\n }\n\n if (typeof color == \"undefined\") {\n color = \"#000\";\n }\n\n // Clean.\n if (element.parentElement !== null) {\n const labels = element.parentElement.getElementsByClassName(\n \"vc-item-nl-label\"\n );\n while (labels.length > 0) {\n const label = labels.item(0);\n if (label) label.remove();\n }\n\n const arrows = element.parentElement.getElementsByClassName(\n \"vc-item-nl-arrow\"\n );\n while (arrows.length > 0) {\n const arrow = arrows.item(0);\n if (arrow) arrow.remove();\n }\n }\n\n let arrowSize = lineWidth * 2;\n\n let arrowPosX = lineX1 + (lineX2 - lineX1) / 2 - arrowSize;\n let arrowPosY = lineY1 + (lineY2 - lineY1) / 2 - arrowSize;\n\n let arrowStart: HTMLElement = document.createElement(\"div\");\n arrowStart.classList.add(\"vc-item-nl-arrow\");\n arrowStart.style.position = \"absolute\";\n arrowStart.style.border = `${arrowSize}px solid transparent`;\n arrowStart.style.borderBottom = `${arrowSize}px solid ${color}`;\n arrowStart.style.left = `${arrowPosX}px`;\n arrowStart.style.top = `${arrowPosY}px`;\n arrowStart.style.transform = `rotate(${90 + g}deg)`;\n\n let arrowEnd: HTMLElement = document.createElement(\"div\");\n arrowEnd.classList.add(\"vc-item-nl-arrow\");\n arrowEnd.style.position = \"absolute\";\n arrowEnd.style.border = `${arrowSize}px solid transparent`;\n arrowEnd.style.borderBottom = `${arrowSize}px solid ${color}`;\n arrowEnd.style.left = `${arrowPosX}px`;\n arrowEnd.style.top = `${arrowPosY}px`;\n arrowEnd.style.transform = `rotate(${270 + g}deg)`;\n\n if (element.parentElement !== null) {\n element.parentElement.appendChild(arrowStart);\n element.parentElement.appendChild(arrowEnd);\n }\n\n if (labelStart != \"\") {\n let htmlLabelStart: HTMLElement = document.createElement(\"div\");\n\n try {\n htmlLabelStart.innerHTML = labelStart;\n htmlLabelStart.style.position = \"absolute\";\n htmlLabelStart.style.left = `${x1}px`;\n htmlLabelStart.style.top = `${y1}px`;\n htmlLabelStart.style.width = `${labelStartWidth}px`;\n htmlLabelStart.style.border = `2px solid ${color}`;\n\n htmlLabelStart.classList.add(\"vc-item-nl-label\", \"label-start\");\n } catch (error) {\n console.error(error);\n }\n\n if (element.parentElement !== null) {\n element.parentElement.appendChild(htmlLabelStart);\n }\n }\n\n if (labelEnd != \"\") {\n let htmlLabelEnd: HTMLElement = document.createElement(\"div\");\n\n try {\n htmlLabelEnd.innerHTML = labelEnd;\n htmlLabelEnd.style.position = \"absolute\";\n htmlLabelEnd.style.left = `${x2}px`;\n htmlLabelEnd.style.top = `${y2}px`;\n htmlLabelEnd.style.width = `${labelEndWidth}px`;\n htmlLabelEnd.style.border = `2px solid ${color}`;\n\n htmlLabelEnd.classList.add(\"vc-item-nl-label\", \"label-end\");\n } catch (error) {\n console.error(error);\n }\n\n if (element.parentElement !== null) {\n element.parentElement.appendChild(htmlLabelEnd);\n }\n }\n }\n}\n","import { LinkedVisualConsoleProps, AnyObject } from \"../lib/types\";\nimport {\n linkedVCPropsDecoder,\n parseIntOr,\n notEmptyStringOr,\n stringIsEmpty,\n decodeBase64,\n parseBoolean,\n t\n} from \"../lib\";\nimport Item, { ItemProps, itemBasePropsDecoder, ItemType } from \"../Item\";\n\nexport type GroupProps = {\n type: ItemType.GROUP_ITEM;\n groupId: number;\n imageSrc: string | null; // URL?\n statusImageSrc: string | null;\n showStatistics: boolean;\n html?: string | null;\n} & ItemProps &\n LinkedVisualConsoleProps;\n\nfunction extractHtml(data: AnyObject): string | null {\n if (!stringIsEmpty(data.html)) return data.html;\n if (!stringIsEmpty(data.encodedHtml)) return decodeBase64(data.encodedHtml);\n return null;\n}\n\n/**\n * Build a valid typed object from a raw object.\n * This will allow us to ensure the type safety.\n *\n * @param data Raw object.\n * @return An object representing the group props.\n * @throws Will throw a TypeError if some property\n * is missing from the raw object or have an invalid type.\n */\nexport function groupPropsDecoder(data: AnyObject): GroupProps | never {\n if (\n (typeof data.imageSrc !== \"string\" || data.imageSrc.length === 0) &&\n data.encodedHtml === null\n ) {\n throw new TypeError(\"invalid image src.\");\n }\n if (parseIntOr(data.groupId, null) === null) {\n throw new TypeError(\"invalid group Id.\");\n }\n\n const showStatistics = parseBoolean(data.showStatistics);\n const html = showStatistics ? extractHtml(data) : null;\n\n return {\n ...itemBasePropsDecoder(data), // Object spread. It will merge the properties of the two objects.\n type: ItemType.GROUP_ITEM,\n groupId: parseInt(data.groupId),\n imageSrc: notEmptyStringOr(data.imageSrc, null),\n statusImageSrc: notEmptyStringOr(data.statusImageSrc, null),\n showStatistics,\n html,\n ...linkedVCPropsDecoder(data) // Object spread. It will merge the properties of the two objects.\n };\n}\nexport default class Group extends Item