0361e95683
git-svn-id: https://svn.code.sf.net/p/xcat/code/xcat-core/trunk@11804 8638fb3e-16cb-4fca-ae20-7b5d299a9bcd
1104 lines
30 KiB
JavaScript
1104 lines
30 KiB
JavaScript
/**
|
|
* Global variables
|
|
*/
|
|
|
|
// Save grid summary data, update every one minute
|
|
var gridData;
|
|
|
|
// Save nodes path, used for getting detail from rrd file
|
|
var nodePath = new Object();
|
|
|
|
// Save nodes current status,
|
|
// unknown = -2, error = -1, warning = 0 ,normal = 1 are used for sorting
|
|
var nodeStatus = new Object();
|
|
|
|
// Update timer
|
|
var gangliaTimer;
|
|
|
|
// Global frame hash
|
|
var framehash;
|
|
// Global cec hash
|
|
var cechash;
|
|
// Global blade hash
|
|
var bladehash;
|
|
// Global x rack hash
|
|
var rackhash;
|
|
// Global other type node hash
|
|
var otherhash;
|
|
|
|
/**
|
|
* Load Ganglia monitoring tool
|
|
*
|
|
* @return Nothing
|
|
*/
|
|
function loadGangliaMon() {
|
|
$('#gangliamon').append(createInfoBar('Checking RPMs'));
|
|
|
|
// Get groups and set cookie
|
|
if (!$.cookie('groups')) {
|
|
$.ajax( {
|
|
url : 'lib/cmd.php',
|
|
dataType : 'json',
|
|
data : {
|
|
cmd : 'extnoderange',
|
|
tgt : '/.*',
|
|
args : 'subgroups',
|
|
msg : ''
|
|
},
|
|
|
|
success : setGroupsCookies
|
|
});
|
|
}
|
|
|
|
// Check whether Ganglia RPMs are installed on the xCAT MN
|
|
$.ajax({
|
|
url : 'lib/systemcmd.php',
|
|
dataType : 'json',
|
|
data : {
|
|
cmd : 'rpm -q rrdtool ganglia-gmetad ganglia-gmond'
|
|
},
|
|
|
|
success : checkGangliaRPMs
|
|
});
|
|
return;
|
|
}
|
|
|
|
/**
|
|
* Check whether Ganglia RPMs are installed
|
|
*
|
|
* @param data
|
|
* Data returned from HTTP request
|
|
* @return Nothing
|
|
*/
|
|
function checkGangliaRPMs(data) {
|
|
var gangliaTab = $('#gangliamon');
|
|
gangliaTab.empty();
|
|
|
|
// Get the list of Ganglia RPMs installed
|
|
var status = data.rsp.split(/\n/);
|
|
var gangliaRPMs = ["rrdtool", "ganglia-gmetad", "ganglia-gmond"];
|
|
var warningMsg = 'Before continuing, please install the following packages: ';
|
|
var missingRPMs = false;
|
|
for (var i in status) {
|
|
if (status[i].indexOf("not installed") > -1) {
|
|
warningMsg += gangliaRPMs[i] + ' ';
|
|
missingRPMs = true;
|
|
}
|
|
}
|
|
|
|
// Append Ganglia PDF
|
|
if (missingRPMs) {
|
|
var warningBar = createWarnBar(warningMsg);
|
|
warningBar.css('margin-bottom', '10px');
|
|
warningBar.prependTo(gangliaTab);
|
|
} else {
|
|
gangliaTab.append(createInfoBar('Checking running status'));
|
|
|
|
// Check if ganglia is running on the xCAT MN
|
|
$.ajax( {
|
|
url : 'lib/cmd.php',
|
|
dataType : 'json',
|
|
data : {
|
|
cmd : 'monls',
|
|
tgt : '',
|
|
args : 'gangliamon',
|
|
msg : ''
|
|
},
|
|
|
|
success : checkGangliaRunning
|
|
});
|
|
}
|
|
|
|
return;
|
|
}
|
|
|
|
/**
|
|
* Check whether Ganglia is running
|
|
*
|
|
* @param data
|
|
* Data returned from HTTP request
|
|
* @return Nothing
|
|
*/
|
|
function checkGangliaRunning(data) {
|
|
var gangliaTab = $('#gangliamon');
|
|
gangliaTab.empty();
|
|
|
|
// If Ganglia is not started
|
|
if (data.rsp[0].indexOf("not-monitored") > -1) {
|
|
// Create link to start Ganglia
|
|
var startLnk = $('<a href="#">Click here</a>');
|
|
startLnk.css({
|
|
'color' : 'blue',
|
|
'text-decoration' : 'none'
|
|
});
|
|
startLnk.click(function() {
|
|
// Turn on Ganglia for all nodes
|
|
monitorNode('', 'on');
|
|
});
|
|
|
|
// Create warning bar
|
|
var warningBar = $('<div class="ui-state-error ui-corner-all"></div>');
|
|
var msg = $('<p></p>').css({
|
|
'display': 'inline-block',
|
|
'width': '90%'
|
|
});
|
|
var icon = $('<span class="ui-icon ui-icon-alert"></span>').css({
|
|
'display': 'inline-block',
|
|
'margin': '10px 5px'
|
|
});
|
|
warningBar.append(icon);
|
|
msg.append('Please start Ganglia on xCAT. ');
|
|
msg.append(startLnk);
|
|
msg.append(' to start Ganglia.');
|
|
warningBar.append(msg);
|
|
warningBar.css('margin-bottom', '10px');
|
|
|
|
// If there are any warning messages, append this warning after it
|
|
var curWarnings = $('#gangliamon').find('.ui-state-error');
|
|
|
|
if (curWarnings.length) {
|
|
curWarnings.after(warningBar);
|
|
} else {
|
|
warningBar.prependTo(gangliaTab);
|
|
}
|
|
|
|
return;
|
|
}
|
|
|
|
// Legend for node status
|
|
var legend = '<table style="float:right"><tr>' +
|
|
'<td style="background:#66CD00;width:16px;padding:0px;"> </td><td style="padding:0px;border:0px">Normal</td>' +
|
|
'<td style="background:#FFD700;width:16px;padding:0px;"> </td><td style="padding:0px;">Heavy Load</td>' +
|
|
'<td style="background:#FF3030;width:16px;padding:0px;"> </td><td style="padding:0px;">Error</td>' +
|
|
'<td style="background:#8B8B7A;width:16px;padding:0px;"> </td><td style="padding:0px;">Unknown</td>' +
|
|
'</tr></table>';
|
|
|
|
// Gganglia grid overview
|
|
var showStr = '<div><h3 style="display:inline;">Grid Overview</h3>' +
|
|
'<sup id="hidesup" style="cursor: pointer;color:blue;float:right">[Hide]</sup></div><hr>' +
|
|
'<div id="gangliaGridSummary"></div>' +
|
|
'<div><h3 style="display:inline;">Current Nodes Status</h3>' + legend + '</div><hr>' +
|
|
'<div id="zoomDiv" style="padding:0px 0px 12px 0px;"><span name="ALL">All Nodes</span></div>' +
|
|
'<div id="gangliaNodes"></div>';
|
|
gangliaTab.append(showStr);
|
|
|
|
// Get summary data and draw on page
|
|
$('#gangliaGridSummary').append('Getting grid summary data <img src="images/loader.gif"></img>');
|
|
sendGridSummaryAjax();
|
|
|
|
// Get all nodes location data which can support the zoom monitor
|
|
$('#gangliaNodes').append('Getting all nodes status <img src="images/loader.gif"></img>');
|
|
sendLocationAjax();
|
|
|
|
// Bind the hide/show button event
|
|
$('#gangliamon #hidesup').bind('click', function(){
|
|
if ('[Hide]' == $(this).text()) {
|
|
$(this).html('[Show]');
|
|
} else {
|
|
$(this).html('[Hide]');
|
|
}
|
|
|
|
$('#gangliaGridSummary').toggle();
|
|
});
|
|
}
|
|
|
|
/**
|
|
* Send AJAX request to get all nodes parent and position to create hardware hierarchy hash
|
|
*
|
|
* @return Nothing
|
|
*/
|
|
function sendLocationAjax() {
|
|
$.ajax({
|
|
url : 'lib/cmd.php',
|
|
dataType : 'json',
|
|
data : {
|
|
cmd : 'webrun',
|
|
tgt : '',
|
|
args : 'graph',
|
|
msg : ''
|
|
},
|
|
|
|
success: function(data){
|
|
if (!data.rsp[0]) {
|
|
return;
|
|
}
|
|
|
|
extractLocationlData(data.rsp[0]);
|
|
// Get nodes current status and draw on the page
|
|
sendNodeCurrentAjax();
|
|
|
|
// Start the timer to update page per minute
|
|
gangliaTimer = window.setTimeout('updateGangliaPage()', 60000);
|
|
}
|
|
});
|
|
}
|
|
|
|
/**
|
|
* Extract the location query data and saved in global variable
|
|
*
|
|
* @return Nothing
|
|
*/
|
|
function extractLocationlData(locationData) {
|
|
framehash = new Object();
|
|
cechash = new Object();
|
|
bladehash = new Object();
|
|
rackhash = new Object();
|
|
|
|
// Linux nodes which has no parent
|
|
linuxArray = new Array();
|
|
|
|
// Other unknown nodes only have one parent, use number 1 as there parent
|
|
otherhash = new Object();
|
|
otherhash[1] = new Array();
|
|
|
|
var allnodearray = locationData.split(';');
|
|
var temparray;
|
|
var parent = '';
|
|
var name = '';
|
|
for (var i in allnodearray) {
|
|
temparray = allnodearray[i].split(':');
|
|
name = temparray[0];
|
|
|
|
// If there is not parent (or mpa, or rack) information
|
|
parent = temparray[2];
|
|
if (!parent) {
|
|
// Go to next node
|
|
otherhash[1].push(name);
|
|
continue;
|
|
}
|
|
|
|
switch (temparray[1].toLowerCase()) {
|
|
case 'blade': {
|
|
if (!bladehash[parent]) {
|
|
bladehash[parent] = new Array();
|
|
}
|
|
|
|
bladehash[parent].push(name);
|
|
}
|
|
break;
|
|
|
|
case 'systemx': {
|
|
if (!rackhash[parent]) {
|
|
rackhash[parent] = new Array();
|
|
}
|
|
|
|
rackhash[parent].push(name);
|
|
}
|
|
break;
|
|
|
|
case 'frame': {
|
|
if (!framehash[name]) {
|
|
framehash[name] = new Array();
|
|
}
|
|
}
|
|
break;
|
|
|
|
case 'cec': {
|
|
if (!framehash[parent]) {
|
|
framehash[parent] = new Array();
|
|
}
|
|
|
|
framehash[parent].push(name);
|
|
}
|
|
break;
|
|
|
|
case 'lpar':
|
|
case 'lpar,osi':
|
|
case 'osi,lpar': {
|
|
if (!cechash[parent]) {
|
|
cechash[parent] = new Array();
|
|
}
|
|
|
|
cechash[parent].push(name);
|
|
}
|
|
|
|
break;
|
|
default: {
|
|
otherhash[1].push(name);
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Send AJAX request to get grid summary information
|
|
*
|
|
* @return Nothing
|
|
*/
|
|
function sendGridSummaryAjax() {
|
|
// Get the summary data
|
|
$.ajax({
|
|
url : 'lib/cmd.php',
|
|
dataType : 'json',
|
|
data : {
|
|
cmd : 'webrun',
|
|
tgt : '',
|
|
args : 'gangliashow;_grid_;hour;_summary_',
|
|
msg : ''
|
|
},
|
|
|
|
success: function(data) {
|
|
createGridSummaryData(data.rsp[0]);
|
|
drawGridSummary();
|
|
}
|
|
});
|
|
}
|
|
|
|
/**
|
|
* Send AJAX request to get nodes current load information
|
|
*
|
|
* @return Nothing
|
|
*/
|
|
function sendNodeCurrentAjax() {
|
|
|
|
// Get all nodes current status
|
|
$.ajax({
|
|
url : 'lib/cmd.php',
|
|
dataType : 'json',
|
|
data : {
|
|
cmd : 'webrun',
|
|
tgt : '',
|
|
args : 'gangliacurrent;node;',
|
|
msg : ''
|
|
},
|
|
|
|
success: function(data){
|
|
createNodeStatusData(data.rsp[0]);
|
|
drawGangliaNodesArea($('#gangliaorder').val());
|
|
}
|
|
});
|
|
}
|
|
|
|
/**
|
|
* Send AJAX request to get grid current summary information to update the page
|
|
*
|
|
* @return Nothing
|
|
*/
|
|
function sendGridCurrentAjax(){
|
|
// Get the summary data
|
|
$.ajax({
|
|
url : 'lib/cmd.php',
|
|
dataType : 'json',
|
|
data : {
|
|
cmd : 'webrun',
|
|
tgt : '',
|
|
args : 'gangliacurrent;grid',
|
|
msg : ''
|
|
},
|
|
|
|
success: function(data){
|
|
updateGridSummaryData(data.rsp[0]);
|
|
drawGridSummary();
|
|
}
|
|
});
|
|
}
|
|
|
|
/**
|
|
* Save the grid summary data to local global variable
|
|
*
|
|
* @param data structure
|
|
* metric1:time11,val11,time12,val12....;metric2:time21,val21,time22,val22,...;....
|
|
* @return Nothing
|
|
*/
|
|
function createGridSummaryData(summaryString){
|
|
// Empty the global data
|
|
gridData = new Object();
|
|
|
|
var metricArray = summaryString.split(';');
|
|
var metricname = '';
|
|
var valueArray = '';
|
|
var position = 0;
|
|
var tempLength = 0;
|
|
for (var index = 0; index < metricArray.length; index++) {
|
|
position = metricArray[index].indexOf(':');
|
|
|
|
// Get the metric name and init its global array to save timestamp and value pair
|
|
metricname = metricArray[index].substr(0, position);
|
|
gridData[metricname] = new Array();
|
|
valueArray = metricArray[index].substr(position + 1).split(',');
|
|
tempLength = valueArray.length;
|
|
|
|
// Save timestamp and value into global array
|
|
for (var i = 0; i < tempLength; i++) {
|
|
gridData[metricname].push(Number(valueArray[i]));
|
|
}
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Update the grid summary data to local global variable
|
|
*
|
|
* @param data structure
|
|
* metric1:time11,val11;metric2:time21,val21,time22;....
|
|
* @return Nothing
|
|
*/
|
|
function updateGridSummaryData(currentString){
|
|
var metricArray = currentString.split(';');
|
|
var metricname = '';
|
|
var position = 0;
|
|
var tempLength = 0;
|
|
var index = 0;
|
|
var tempArray;
|
|
|
|
tempLength = metricArray.length;
|
|
for (index = 0; index < tempLength; index++) {
|
|
position = metricArray[index].indexOf(':');
|
|
metricname = metricArray[index].substr(0, position);
|
|
tempArray = metricArray[index].substr(position + 1).split(',');
|
|
if (gridData[metricname]) {
|
|
gridData[metricname].shift();
|
|
gridData[metricname].shift();
|
|
gridData[metricname].push(Number(tempArray[0]));
|
|
gridData[metricname].push(Number(tempArray[1]));
|
|
}
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Draw the grid summay area by global data
|
|
*
|
|
* @return Nothing
|
|
*/
|
|
function drawGridSummary() {
|
|
var gridDrawArea = $('#gangliaGridSummary');
|
|
var showStr = '';
|
|
var tempStr = $('#gangliamon').attr('class');
|
|
|
|
// jqflot only draws on the visible area
|
|
// If the tab is hide, return directly
|
|
if (tempStr.indexOf('hide') != -1) {
|
|
return;
|
|
}
|
|
|
|
if ($('#gangliamon #hidesup').text() == '[Show]') {
|
|
return;
|
|
}
|
|
|
|
gridDrawArea.empty();
|
|
showStr = '<table style="border-style:none;"><tr><td style="padding:0;border-style:none;"><div id="gangliasummaryload" class="monitorsumdiv"></div></td>' +
|
|
'<td style="padding:0;border-style:none;"><div id="gangliasummarycpu" class="monitorsumdiv"></div></td>' +
|
|
'<td style="padding:0;border-style:none;"><div id="gangliasummarymem" class="monitorsumdiv"></div></td></tr>' +
|
|
'<tr><td style="padding:0;border-style:none;"><div id="gangliasummarydisk" class="monitorsumdiv"></div></td>' +
|
|
'<td style="padding:0;border-style:none;"><div id="gangliasummarynetwork" class="monitorsumdiv"></div></td>' +
|
|
'<td style="padding:0;border-style:none;"></td></tr></table>';
|
|
gridDrawArea.append(showStr);
|
|
drawLoadFlot('gangliasummaryload', 'Grid', gridData['load_one'], gridData['cpu_num']);
|
|
drawCpuFlot('gangliasummarycpu', 'Grid', gridData['cpu_idle']);
|
|
drawMemFlot('gangliasummarymem', 'Grid', gridData['mem_free'], gridData['mem_total']);
|
|
drawDiskFlot('gangliasummarydisk', 'Grid', gridData['disk_free'], gridData['disk_total']);
|
|
drawNetworkFlot('gangliasummarynetwork', 'Grid', gridData['bytes_in'], gridData['bytes_out']);
|
|
}
|
|
|
|
/**
|
|
* Draw the load flot by data(summary data, or one node's data)
|
|
*
|
|
* @param areaid
|
|
* Which DIV draw this flot
|
|
* @param loadpair
|
|
* The load timestamp and value pair
|
|
* @param cpupair
|
|
* The CPU number and value pair
|
|
* @return Nothing
|
|
*/
|
|
function drawLoadFlot(areaid, titleprefix, loadpair, cpupair) {
|
|
var load = new Array();
|
|
var cpunum = new Array();
|
|
var index = 0;
|
|
var yaxismax = 0;
|
|
var interval = 1;
|
|
|
|
$('#' + areaid).empty();
|
|
// Parse load pair, the timestamp must mutiply 1000, javascript time stamp is millisecond
|
|
for (index = 0; index < loadpair.length; index += 2) {
|
|
load.push([loadpair[index] * 1000, loadpair[index + 1]]);
|
|
if (loadpair[index + 1] > yaxismax) {
|
|
yaxismax = loadpair[index + 1];
|
|
}
|
|
}
|
|
|
|
// Parse cpu pair
|
|
for (index = 0; index < cpupair.length; index += 2) {
|
|
cpunum.push([cpupair[index] * 1000, cpupair[index + 1]]);
|
|
if (cpupair[index + 1] > yaxismax) {
|
|
yaxismax = cpupair[index + 1];
|
|
}
|
|
}
|
|
|
|
interval = parseInt(yaxismax / 3);
|
|
if (interval < 1) {
|
|
interval = 1;
|
|
}
|
|
|
|
$.jqplot(areaid, [load, cpunum], {
|
|
title: titleprefix + ' Loads/Procs Last Hour',
|
|
axes:{
|
|
xaxis:{
|
|
renderer : $.jqplot.DateAxisRenderer,
|
|
numberTicks: 4,
|
|
tickOptions : {
|
|
formatString : '%R',
|
|
show : true
|
|
}
|
|
},
|
|
yaxis: {
|
|
min : 0,
|
|
tickInterval : interval
|
|
}
|
|
},
|
|
legend : {
|
|
show: true,
|
|
location: 'nw'
|
|
},
|
|
series:[{label:'Load'}, {label: 'CPU Number'}],
|
|
seriesDefaults : {showMarker: false}
|
|
});
|
|
}
|
|
|
|
/**
|
|
* Draw the CPU usage flot by data(maybe summary data, or one node's data)
|
|
*
|
|
* @param areaid
|
|
* Which DIV draw this flot
|
|
* @param titleprefix
|
|
* Title used name
|
|
* @param cpupair
|
|
* The CPU timestamp and value pair
|
|
* @return Nothing
|
|
*/
|
|
function drawCpuFlot(areaid, titleprefix, cpupair) {
|
|
var cpu = new Array();
|
|
var index = 0;
|
|
|
|
$('#' + areaid).empty();
|
|
|
|
// Time stamp should be mutiplied by 1000
|
|
// We get the CPU idle from server
|
|
for (index = 0; index < cpupair.length; index +=2) {
|
|
cpu.push([(cpupair[index] * 1000), (100 - cpupair[index + 1])]);
|
|
}
|
|
|
|
$.jqplot(areaid, [cpu],{
|
|
title: titleprefix + ' Cpu Use Last Hour',
|
|
axes:{
|
|
xaxis:{
|
|
renderer : $.jqplot.DateAxisRenderer,
|
|
numberTicks: 4,
|
|
tickOptions : {
|
|
formatString : '%R',
|
|
show : true
|
|
}
|
|
},
|
|
yaxis: {
|
|
min : 0,
|
|
max : 100,
|
|
tickOptions:{formatString : '%d\%'}
|
|
}
|
|
},
|
|
seriesDefaults : {showMarker: false}
|
|
});
|
|
}
|
|
|
|
/**
|
|
* Draw the memory usage flot by data(summary data, or one node's data)
|
|
*
|
|
* @param areaid
|
|
* Which DIV draw this flot
|
|
* @param titleprefix
|
|
* Title used name
|
|
* @param cpupair
|
|
* The CPU timestamp and value pair
|
|
* @return Nothing
|
|
*/
|
|
function drawMemFlot(areaid, titleprefix, freepair, totalpair){
|
|
var use = new Array();
|
|
var total = new Array();
|
|
var tempsize = 0;
|
|
var index = 0;
|
|
|
|
$('#' + areaid).empty();
|
|
if (freepair.length < totalpair.length) {
|
|
tempsize = freepair.length;
|
|
} else {
|
|
tempsize = freepair.length;
|
|
}
|
|
|
|
for (index = 0; index < tempsize; index += 2) {
|
|
var temptotal = totalpair[index + 1];
|
|
var tempuse = temptotal - freepair[index + 1];
|
|
temptotal = temptotal / 1000000;
|
|
tempuse = tempuse / 1000000;
|
|
total.push([totalpair[index] * 1000, temptotal]);
|
|
use.push([freepair[index] * 1000, tempuse]);
|
|
}
|
|
|
|
$.jqplot(areaid, [use, total], {
|
|
title: titleprefix + ' Memory Use Last Hour',
|
|
axes:{
|
|
xaxis:{
|
|
renderer : $.jqplot.DateAxisRenderer,
|
|
numberTicks: 4,
|
|
tickOptions : {
|
|
formatString : '%R',
|
|
show : true
|
|
}
|
|
},
|
|
yaxis: {
|
|
min : 0,
|
|
tickOptions:{formatString : '%.2fG'}
|
|
}
|
|
},
|
|
legend : {
|
|
show: true,
|
|
location: 'nw'
|
|
},
|
|
series:[{label:'Used'}, {label: 'Total'}],
|
|
seriesDefaults : {showMarker: false}
|
|
});
|
|
}
|
|
|
|
/**
|
|
* Draw the disk usage flot by data(summary data, or one node's data)
|
|
*
|
|
* @param areaid
|
|
* Which div draw this flot
|
|
* @param titleprefix
|
|
* Title used name
|
|
* @param freepair
|
|
* The free disk number, Ganglia only logs the free data
|
|
* @param totalpair
|
|
* The total disk number
|
|
* @return Nothing
|
|
*/
|
|
function drawDiskFlot(areaid, titleprefix, freepair, totalpair) {
|
|
var use = new Array();
|
|
var total = new Array();
|
|
var tempsize = 0;
|
|
var index = 0;
|
|
|
|
$('#' + areaid).empty();
|
|
if (freepair.length < totalpair.length) {
|
|
tempsize = freepair.length;
|
|
} else{
|
|
tempsize = freepair.length;
|
|
}
|
|
|
|
for (index = 0; index < tempsize; index += 2) {
|
|
var temptotal = totalpair[index + 1];
|
|
var tempuse = temptotal - freepair[index + 1];
|
|
total.push([totalpair[index] * 1000, temptotal]);
|
|
use.push([freepair[index] * 1000, tempuse]);
|
|
}
|
|
|
|
$.jqplot(areaid, [use, total], {
|
|
title: titleprefix + ' Disk Use Last Hour',
|
|
axes:{
|
|
xaxis:{
|
|
renderer : $.jqplot.DateAxisRenderer,
|
|
numberTicks: 4,
|
|
tickOptions : {
|
|
formatString : '%R',
|
|
show : true
|
|
}
|
|
},
|
|
yaxis: {
|
|
min : 0,
|
|
tickOptions:{formatString : '%.2fG'}
|
|
}
|
|
},
|
|
legend : {
|
|
show: true,
|
|
location: 'nw'
|
|
},
|
|
series:[{label:'Used'}, {label: 'Total'}],
|
|
seriesDefaults : {showMarker: false}
|
|
});
|
|
}
|
|
|
|
/**
|
|
* Draw the network load flot by data(summary data, or one node's data)
|
|
*
|
|
* @param areaid
|
|
* Which div draw this flot
|
|
* @param titleprefix
|
|
* Title used name
|
|
* @param inpair
|
|
* The timestamp and value pair for download
|
|
* @param outpair
|
|
* The timestamp and value pair for upload
|
|
* @return Nothing
|
|
*/
|
|
function drawNetworkFlot(areaid, titleprefix, inpair, outpair) {
|
|
var inArray = new Array();
|
|
var outArray = new Array();
|
|
var index = 0;
|
|
var maxvalue = 0;
|
|
var unitname = 'B';
|
|
var divisor = 1;
|
|
|
|
for (index = 0; index < inpair.length; index += 2) {
|
|
if (inpair[index + 1] > maxvalue) {
|
|
maxvalue = inpair[index + 1];
|
|
}
|
|
}
|
|
|
|
for (index = 0; index < outpair.length; index += 2) {
|
|
if (outpair[index + 1] > maxvalue) {
|
|
maxvalue = outpair[index + 1];
|
|
}
|
|
}
|
|
|
|
if (maxvalue > 3000000) {
|
|
divisor = 1000000;
|
|
unitname = 'GB';
|
|
} else if (maxvalue >= 3000) {
|
|
divisor = 1000;
|
|
unitname = 'MB';
|
|
} else {
|
|
//do nothing
|
|
}
|
|
|
|
for (index = 0; index < inpair.length; index += 2) {
|
|
inArray.push([(inpair[index] * 1000), (inpair[index + 1] / divisor)]);
|
|
}
|
|
|
|
for (index = 0; index < outpair.length; index += 2) {
|
|
outArray.push([(outpair[index] * 1000), (outpair[index + 1] / divisor)]);
|
|
}
|
|
|
|
$.jqplot(areaid, [inArray, outArray], {
|
|
title: titleprefix + ' Network Last Hour',
|
|
axes:{
|
|
xaxis:{
|
|
renderer : $.jqplot.DateAxisRenderer,
|
|
numberTicks: 4,
|
|
tickOptions : {
|
|
formatString : '%R',
|
|
show : true
|
|
}
|
|
},
|
|
yaxis: {
|
|
min : 0,
|
|
tickOptions:{formatString : '%d' + unitname}
|
|
}
|
|
},
|
|
legend : {
|
|
show: true,
|
|
location: 'nw'
|
|
},
|
|
series:[{label:'In'}, {label: 'Out'}],
|
|
seriesDefaults : {showMarker: false}
|
|
});
|
|
}
|
|
|
|
/**
|
|
* Create node status data
|
|
*
|
|
* @param nodesStatus
|
|
* Node status
|
|
* @return Nothing
|
|
*/
|
|
function createNodeStatusData(nodesStatus) {
|
|
var nodesArray = nodesStatus.split(';');
|
|
var position = 0;
|
|
var nodename = '';
|
|
var index = 0;
|
|
var tempArray;
|
|
var tempStr = '';
|
|
|
|
for (index in nodePath) {
|
|
delete(nodePath[index]);
|
|
}
|
|
|
|
for (index in nodeStatus) {
|
|
delete(nodeStatus[index]);
|
|
}
|
|
|
|
for (index = 0; index < nodesArray.length; index++) {
|
|
tempStr = nodesArray[index];
|
|
position = tempStr.indexOf(':');
|
|
nodename = tempStr.substring(0, position);
|
|
tempArray = tempStr.substring(position + 1).split(',');
|
|
nodeStatus[nodename] = tempArray[0];
|
|
if (('WARNING' == tempArray[0]) || ('NORMAL' == tempArray[0])){
|
|
nodePath[nodename] = tempArray[1];
|
|
}
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Draw nodes current status, there are four type:
|
|
* a. unknown(gray): cannot find save data for this node
|
|
* b. error(red): got status sometime earlier, but cannot get now
|
|
* c. warning(orange): heavy load
|
|
* d. normal(green): normal load
|
|
*
|
|
* @return Nothing
|
|
*/
|
|
function drawGangliaNodesArea() {
|
|
var position = 0;
|
|
|
|
// Find out the last child's type and name
|
|
var currentobj = $('#zoomDiv span:last');
|
|
var type = currentobj.attr('name').toLowerCase();
|
|
var name = currentobj.text();
|
|
position = name.indexOf('(');
|
|
|
|
if (position > -1) {
|
|
name = name.substr(3, position - 3);
|
|
}
|
|
$('#gangliaNodes').empty();
|
|
|
|
switch (type) {
|
|
// Draw the node current status
|
|
case 'blade':
|
|
case 'cec':
|
|
case 'rack':
|
|
case 'other': {
|
|
drawGangliaNodesAreaPic(type, name);
|
|
}
|
|
break;
|
|
|
|
// Draw a summary table
|
|
case 'all':
|
|
case 'frame': {
|
|
drawGangliaNodesAreaTable(type, name);
|
|
}
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
|
|
function drawGangliaNodesAreaPic(type, name) {
|
|
var index = 0;
|
|
var arraypoint = '';
|
|
var templength = 0;
|
|
var showStr = '';
|
|
var nodename = '';
|
|
|
|
switch(type) {
|
|
case 'blade': {
|
|
arraypoint = bladehash[name];
|
|
}
|
|
break;
|
|
case 'cec': {
|
|
arraypoint = cechash[name];
|
|
}
|
|
break;
|
|
case 'rack': {
|
|
arraypoint = rackhash[name];
|
|
}
|
|
break;
|
|
case 'other': {
|
|
arraypoint = otherhash[1];
|
|
}
|
|
default:
|
|
break;
|
|
}
|
|
$('#gangliaNodes').html('<ul style="margin:0px;padding:0px;"></ul>');
|
|
|
|
templength = arraypoint.length;
|
|
|
|
for (index = 0; index < templength; index++) {
|
|
nodename = arraypoint[index];
|
|
switch (nodeStatus[nodename]) {
|
|
case 'ERROR':
|
|
showStr = '<li class="monitorerror ui-corner-all monitornodeli" title="' + nodename + '"></li>';
|
|
break;
|
|
case 'WARNING':
|
|
showStr = '<li class="mornitorwarning ui-corner-all monitornodeli" title="' + nodename + '"></li>';
|
|
break;
|
|
case 'NORMAL':
|
|
showStr = '<li class="monitornormal ui-corner-all monitornodeli" title="' + nodename + '"></li>';
|
|
break;
|
|
default:
|
|
showStr = '<li class="monitorunknown ui-corner-all monitornodeli" title="' + nodename + '"></li>';
|
|
break;
|
|
}
|
|
$('#gangliaNodes ul').append(showStr);
|
|
}
|
|
|
|
// Bind all normal and warning nodes click event
|
|
$('.monitornormal,.monitorwarning').bind('click', function() {
|
|
var nodename = $(this).attr('title');
|
|
window.open('ganglianode.php?n=' + nodename + '&p=' + nodePath[nodename],
|
|
'nodedetail','height=430,width=950,scrollbars=yes,status =no');
|
|
});
|
|
}
|
|
|
|
function drawGangliaNodesAreaTable(type, name) {
|
|
var table = $('<table></table>');
|
|
var row = '';
|
|
var usedCec = new Object();
|
|
|
|
var header = $('<thead class="ui-widget-header"> <th>Name</th><th>Type</th><th>Normal</th><th>Heavy Load</th><th>Error</th><th>Unknown</th> </thead>');
|
|
table.append(header);
|
|
|
|
if (type == 'all') {
|
|
for (var i in framehash) {
|
|
var framename = i;
|
|
row = '<tr><td><a href="#" onclick="addZoomDiv(this)" name="frame">' + framename + '</a></td><td>Frame</td>' +
|
|
monitorStatAgg('frame', framehash[i]) + '</tr>';
|
|
table.append(row);
|
|
for(var j in framehash[i]){
|
|
usedCec[framehash[i][j]] = 1;
|
|
}
|
|
}
|
|
|
|
for (var i in cechash) {
|
|
if (usedCec[i]) {
|
|
continue;
|
|
}
|
|
var cecname = i;
|
|
row = '<tr><td><a href="#" onclick="addZoomDiv(this)" name="cec">' + cecname + '</a></td><td>CEC</td>' +
|
|
monitorStatAgg('cec', cechash[i]) + '</tr>';
|
|
table.append(row);
|
|
}
|
|
|
|
for (var i in bladehash) {
|
|
var bladename = i;
|
|
row = '<tr><td><a href="#" onclick="addZoomDiv(this)" name="blade">' + bladename + '</a></td><td>Blade</td>' +
|
|
monitorStatAgg('blade', bladehash[i]) + '</tr>';
|
|
table.append(row);
|
|
}
|
|
|
|
for (var i in rackhash) {
|
|
var rackname = i;
|
|
row = '<tr><td><a href="#" onclick="addZoomDiv(this)" name="rack">' + rackname + '</a></td><td>Rack</td>' +
|
|
monitorStatAgg('rack', rackhash[i]) + '</tr>';
|
|
table.append(row);
|
|
}
|
|
|
|
if (otherhash[1].length > 0) {
|
|
row = '<tr><td><a href="#" onclick="addZoomDiv(this)" name="other">Other</a></td><td>Other</td>' +
|
|
monitorStatAgg('other', otherhash[1]) + '</tr>';
|
|
table.append(row);
|
|
}
|
|
} else {
|
|
for (var i in framehash[name]) {
|
|
var cecname = framehash[name][i];
|
|
row = '<tr><td><a href="#" onclick="addZoomDiv(this)" name="cec">' + cecname + '</a></td>' +
|
|
'<td>CEC</td>' + monitorStatAgg('cec', cechash[cecname]) + '</tr>';
|
|
table.append(row);
|
|
}
|
|
}
|
|
|
|
$('#gangliaNodes').append(table);
|
|
}
|
|
|
|
/**
|
|
* Update all tab per minute.
|
|
*
|
|
* @return Nothing
|
|
*/
|
|
function monitorStatAgg(type, inputarray) {
|
|
var normalnum = 0;
|
|
var warningnum = 0;
|
|
var errornum = 0;
|
|
var nuknownnum = 0;
|
|
var tempArray;
|
|
var tempname;
|
|
|
|
switch (type) {
|
|
case 'blade':
|
|
case 'cec':
|
|
case 'rack':
|
|
case 'other': {
|
|
tempArray = inputarray;
|
|
}
|
|
break;
|
|
case 'frame': {
|
|
tempArray = new Array();
|
|
for (var i in inputarray){
|
|
tempname = inputarray[i];
|
|
for (var j in cechash[tempname]) {
|
|
tempArray.push(cechash[tempname][j]);
|
|
}
|
|
}
|
|
}
|
|
break;
|
|
default:
|
|
return;
|
|
break;
|
|
}
|
|
|
|
for (var i in tempArray) {
|
|
tempname = tempArray[i];
|
|
switch(nodeStatus[tempname]) {
|
|
case 'NORMAL':
|
|
normalnum++;
|
|
break;
|
|
case 'WARNING':
|
|
warningnum++;
|
|
break;
|
|
case 'ERROR':
|
|
errornum++;
|
|
break;
|
|
default:
|
|
nuknownnum++;
|
|
break;
|
|
}
|
|
}
|
|
|
|
normalnum = normalnum?normalnum:'-';
|
|
warningnum = warningnum?warningnum:'-';
|
|
errornum = errornum?errornum:'-';
|
|
nuknownnum = nuknownnum?nuknownnum:'-';
|
|
|
|
return ('<td>' + normalnum + '</td><td>' + warningnum + '</td><td>' + errornum + '</td><td>' + nuknownnum + '</td>');
|
|
}
|
|
|
|
/**
|
|
* Update all tab per minute.
|
|
*
|
|
* @return Nothing
|
|
*/
|
|
function updateGangliaPage() {
|
|
if ($('#gangliaNodes').size() < 1) {
|
|
return;
|
|
}
|
|
|
|
sendGridCurrentAjax();
|
|
sendNodeCurrentAjax();
|
|
|
|
gangliaTimer = window.setTimeout('updateGangliaPage()', 60000);
|
|
}
|
|
|
|
/**
|
|
* change the zoom area when click the zoom button
|
|
*
|
|
* @return Nothing
|
|
*/
|
|
function updateZoom(obj) {
|
|
var type=$(obj).attr('name');
|
|
while ($('#zoomDiv span:last').attr('name') != type) {
|
|
$('#zoomDiv span:last').remove();
|
|
}
|
|
$(obj).removeClass('monitorzoomlinkli');
|
|
$(obj).unbind('click');
|
|
|
|
drawGangliaNodesArea();
|
|
}
|
|
|
|
/**
|
|
* add the zoom level when click the group link in the summary table
|
|
*
|
|
* @return Nothing
|
|
*/
|
|
function addZoomDiv(obj) {
|
|
var name = $(obj).text();
|
|
var type = $(obj).attr('name');
|
|
|
|
var lastzoomobj = $('#zoomDiv span:last');
|
|
lastzoomobj.addClass('monitorzoomlink');
|
|
lastzoomobj.bind('click', function() {
|
|
updateZoom(this);
|
|
});
|
|
|
|
var newcontent = ' > ' + name + '(' + type.toUpperCase() + ')';
|
|
var newli = '<span name="' + type + '">' + newcontent + '</span>';
|
|
$('#zoomDiv').append(newli);
|
|
|
|
drawGangliaNodesArea();
|
|
} |