dfshealth.js 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427
  1. /**
  2. * Licensed to the Apache Software Foundation (ASF) under one
  3. * or more contributor license agreements. See the NOTICE file
  4. * distributed with this work for additional information
  5. * regarding copyright ownership. The ASF licenses this file
  6. * to you under the Apache License, Version 2.0 (the
  7. * "License"); you may not use this file except in compliance
  8. * with the License. You may obtain a copy of the License at
  9. *
  10. * http://www.apache.org/licenses/LICENSE-2.0
  11. *
  12. * Unless required by applicable law or agreed to in writing, software
  13. * distributed under the License is distributed on an "AS IS" BASIS,
  14. * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  15. * See the License for the specific language governing permissions and
  16. * limitations under the License.
  17. */
  18. (function () {
  19. "use strict";
  20. dust.loadSource(dust.compile($('#tmpl-dfshealth').html(), 'dfshealth'));
  21. dust.loadSource(dust.compile($('#tmpl-startup-progress').html(), 'startup-progress'));
  22. dust.loadSource(dust.compile($('#tmpl-datanode').html(), 'datanode-info'));
  23. dust.loadSource(dust.compile($('#tmpl-datanode-volume-failures').html(), 'datanode-volume-failures'));
  24. dust.loadSource(dust.compile($('#tmpl-snapshot').html(), 'snapshot-info'));
  25. $.fn.dataTable.ext.order['ng-value'] = function (settings, col)
  26. {
  27. return this.api().column(col, {order:'index'} ).nodes().map(function (td, i) {
  28. return $(td).attr('ng-value');
  29. });
  30. };
  31. function load_overview() {
  32. var BEANS = [
  33. {"name": "nn", "url": "/jmx?qry=Hadoop:service=NameNode,name=NameNodeInfo"},
  34. {"name": "nnstat", "url": "/jmx?qry=Hadoop:service=NameNode,name=NameNodeStatus"},
  35. {"name": "fs", "url": "/jmx?qry=Hadoop:service=NameNode,name=FSNamesystemState"},
  36. {"name": "fsn", "url": "/jmx?qry=Hadoop:service=NameNode,name=FSNamesystem"},
  37. {"name": "replicastat", "url": "/jmx?qry=Hadoop:service=NameNode,name=ReplicatedBlocksState"},
  38. {"name": "ecstat", "url": "/jmx?qry=Hadoop:service=NameNode,name=ECBlockGroupsState"},
  39. {"name": "blockstats", "url": "/jmx?qry=Hadoop:service=NameNode,name=BlockStats"},
  40. {"name": "mem", "url": "/jmx?qry=java.lang:type=Memory"}
  41. ];
  42. var HELPERS = {
  43. 'helper_fs_max_objects': function (chunk, ctx, bodies, params) {
  44. var o = ctx.current();
  45. if (o.MaxObjects > 0) {
  46. chunk.write('(' + Math.round((o.FilesTotal + o.BlockTotal) / o.MaxObjects * 100) * 100 + ')%');
  47. }
  48. },
  49. 'helper_dir_status': function (chunk, ctx, bodies, params) {
  50. var j = ctx.current();
  51. for (var i in j) {
  52. chunk.write('<tr><td>' + i + '</td><td>' + j[i] + '</td><td>' + params.type + '</td></tr>');
  53. }
  54. },
  55. 'helper_date_tostring' : function (chunk, ctx, bodies, params) {
  56. var value = dust.helpers.tap(params.value, chunk, ctx);
  57. return chunk.write('' + moment(Number(value)).format('ddd MMM DD HH:mm:ss ZZ YYYY'));
  58. }
  59. };
  60. var data = {};
  61. $.ajax({'url': '/conf', 'dataType': 'xml', 'async': false}).done(
  62. function(d) {
  63. var $xml = $(d);
  64. var namespace, nnId;
  65. $xml.find('property').each(function(idx,v) {
  66. if ($(v).find('name').text() === 'dfs.nameservice.id') {
  67. namespace = $(v).find('value').text();
  68. }
  69. if ($(v).find('name').text() === 'dfs.ha.namenode.id') {
  70. nnId = $(v).find('value').text();
  71. }
  72. });
  73. if (namespace && nnId) {
  74. data['HAInfo'] = {"Namespace": namespace, "NamenodeID": nnId};
  75. }
  76. });
  77. // Workarounds for the fact that JMXJsonServlet returns non-standard JSON strings
  78. function workaround(nn) {
  79. nn.JournalTransactionInfo = JSON.parse(nn.JournalTransactionInfo);
  80. nn.NameJournalStatus = JSON.parse(nn.NameJournalStatus);
  81. nn.NameDirStatuses = JSON.parse(nn.NameDirStatuses);
  82. nn.NodeUsage = JSON.parse(nn.NodeUsage);
  83. nn.CorruptFiles = JSON.parse(nn.CorruptFiles);
  84. return nn;
  85. }
  86. load_json(
  87. BEANS,
  88. guard_with_startup_progress(function(d) {
  89. for (var k in d) {
  90. data[k] = k === 'nn' ? workaround(d[k].beans[0]) : d[k].beans[0];
  91. }
  92. var blockstats = data['blockstats'];
  93. for (var k in blockstats.StorageTypeStats) {
  94. var b = blockstats.StorageTypeStats[k].value;
  95. b.capacityUsedPercentage = b.capacityUsed * 100.0 / b.capacityTotal;
  96. b.capacityRemainingPercentage = b.capacityRemaining * 100.0 / b.capacityTotal;
  97. }
  98. data.fs.ObjectsTotal = data.fs.FilesTotal + data.fs.BlocksTotal;
  99. render();
  100. }),
  101. function (url, jqxhr, text, err) {
  102. show_err_msg('<p>Failed to retrieve data from ' + url + ', cause: ' + err + '</p>');
  103. });
  104. function render() {
  105. var base = dust.makeBase(HELPERS);
  106. dust.render('dfshealth', base.push(data), function(err, out) {
  107. $('#tab-overview').html(out);
  108. $('#ui-tabs a[href="#tab-overview"]').tab('show');
  109. });
  110. }
  111. }
  112. function show_err_msg(msg) {
  113. $('#alert-panel-body').html(msg);
  114. $('#alert-panel').show();
  115. }
  116. function ajax_error_handler(url, jqxhr, text, err) {
  117. show_err_msg('<p>Failed to retrieve data from ' + url + ', cause: ' + err + '</p>');
  118. }
  119. function guard_with_startup_progress(fn) {
  120. return function() {
  121. try {
  122. fn.apply(this, arguments);
  123. } catch (err) {
  124. if (err instanceof TypeError) {
  125. show_err_msg('NameNode is still loading. Redirecting to the Startup Progress page.');
  126. load_startup_progress();
  127. }
  128. }
  129. };
  130. }
  131. function load_startup_progress() {
  132. function workaround(r) {
  133. function rename_property(o, s, d) {
  134. if (o[s] !== undefined) {
  135. o[d] = o[s];
  136. delete o[s];
  137. }
  138. }
  139. r.percentComplete *= 100;
  140. $.each(r.phases, function (idx, p) {
  141. p.percentComplete *= 100;
  142. $.each(p.steps, function (idx2, s) {
  143. s.percentComplete *= 100;
  144. // dust.js is confused by these optional keys in nested
  145. // structure, rename them
  146. rename_property(s, "desc", "stepDesc");
  147. rename_property(s, "file", "stepFile");
  148. rename_property(s, "size", "stepSize");
  149. });
  150. });
  151. return r;
  152. }
  153. $.get('/startupProgress', function (resp) {
  154. var data = workaround(resp);
  155. dust.render('startup-progress', data, function(err, out) {
  156. $('#tab-startup-progress').html(out);
  157. $('#ui-tabs a[href="#tab-startup-progress"]').tab('show');
  158. });
  159. }).fail(ajax_error_handler);
  160. }
  161. function load_datanode_info() {
  162. var HELPERS = {
  163. 'helper_relative_time' : function (chunk, ctx, bodies, params) {
  164. var value = dust.helpers.tap(params.value, chunk, ctx);
  165. return chunk.write(moment().subtract(Number(value), 'seconds').format('ddd MMM DD HH:mm:ss ZZ YYYY'));
  166. },
  167. 'helper_usage_bar' : function (chunk, ctx, bodies, params) {
  168. var value = dust.helpers.tap(params.value, chunk, ctx);
  169. var v = Number(value);
  170. var r = null;
  171. if (v < 70) {
  172. r = 'progress-bar-success';
  173. } else if (v < 85) {
  174. r = 'progress-bar-warning';
  175. } else {
  176. r = "progress-bar-danger";
  177. }
  178. return chunk.write(r);
  179. },
  180. };
  181. function workaround(r) {
  182. function node_map_to_array(nodes) {
  183. var res = [];
  184. for (var n in nodes) {
  185. var p = nodes[n];
  186. p.name = n;
  187. res.push(p);
  188. }
  189. return res;
  190. }
  191. function augment_live_nodes(nodes) {
  192. for (var i = 0, e = nodes.length; i < e; ++i) {
  193. var n = nodes[i];
  194. n.usedPercentage = Math.round((n.used + n.nonDfsUsedSpace) * 1.0 / n.capacity * 100);
  195. var port = n.infoAddr.split(":")[1];
  196. var securePort = n.infoSecureAddr.split(":")[1];
  197. var dnHost = n.name.split(":")[0];
  198. n.dnWebAddress = "http://" + dnHost + ":" + port;
  199. if (securePort != 0) {
  200. n.dnWebAddress = "https://" + dnHost + ":" + securePort;
  201. }
  202. if (n.adminState === "In Service") {
  203. n.state = "alive";
  204. } else if (nodes[i].adminState === "Decommission In Progress") {
  205. n.state = "decommissioning";
  206. } else if (nodes[i].adminState === "Decommissioned") {
  207. n.state = "decommissioned";
  208. } else if (nodes[i].adminState === "Entering Maintenance") {
  209. n.state = "entering-maintenance";
  210. } else if (nodes[i].adminState === "In Maintenance") {
  211. n.state = "in-maintenance";
  212. }
  213. }
  214. }
  215. function augment_dead_nodes(nodes) {
  216. for (var i = 0, e = nodes.length; i < e; ++i) {
  217. if (nodes[i].adminState === "Decommissioned") {
  218. nodes[i].state = "down-decommissioned";
  219. } else if (nodes[i].adminState === "In Maintenance") {
  220. nodes[i].state = "down-maintenance";
  221. } else {
  222. nodes[i].state = "down";
  223. }
  224. }
  225. }
  226. r.LiveNodes = node_map_to_array(JSON.parse(r.LiveNodes));
  227. augment_live_nodes(r.LiveNodes);
  228. r.DeadNodes = node_map_to_array(JSON.parse(r.DeadNodes));
  229. augment_dead_nodes(r.DeadNodes);
  230. r.DecomNodes = node_map_to_array(JSON.parse(r.DecomNodes));
  231. r.EnteringMaintenanceNodes = node_map_to_array(JSON.parse(r.EnteringMaintenanceNodes));
  232. return r;
  233. }
  234. function renderHistogram(dnData) {
  235. var data = dnData.LiveNodes.map(function(dn) {
  236. return (dn.usedSpace / dn.capacity) * 100.0;
  237. });
  238. var formatCount = d3.format(",.0f");
  239. var widthCap = $("div.container").width();
  240. var heightCap = 150;
  241. var margin = {top: 10, right: 60, bottom: 30, left: 30},
  242. width = widthCap * 0.9,
  243. height = heightCap - margin.top - margin.bottom;
  244. var x = d3.scaleLinear()
  245. .domain([0.0, 100.0])
  246. .range([0, width]);
  247. var bins = d3.histogram()
  248. .domain(x.domain())
  249. .thresholds(x.ticks(20))
  250. (data);
  251. var y = d3.scaleLinear()
  252. .domain([0, d3.max(bins, function(d) { return d.length; })])
  253. .range([height, 0]);
  254. var svg = d3.select("#datanode-usage-histogram").append("svg")
  255. .attr("width", width + 50.0)
  256. .attr("height", height + margin.top + margin.bottom)
  257. .append("g")
  258. .attr("transform", "translate(" + margin.left + "," + margin.top + ")");
  259. svg.append("text")
  260. .attr("x", (width / 2))
  261. .attr("y", heightCap - 6 - (margin.top / 2))
  262. .attr("text-anchor", "middle")
  263. .style("font-size", "15px")
  264. .text("Disk usage of each DataNode (%)");
  265. var bar = svg.selectAll(".bar")
  266. .data(bins)
  267. .enter().append("g")
  268. .attr("class", "bar")
  269. .attr("transform", function(d) { return "translate(" + x(d.x0) + "," + y(d.length) + ")"; });
  270. bar.append("rect")
  271. .attr("x", 1)
  272. .attr("width", x(bins[0].x1) - x(bins[0].x0) - 1)
  273. .attr("height", function(d) { return height - y(d.length); });
  274. bar.append("text")
  275. .attr("dy", ".75em")
  276. .attr("y", 6)
  277. .attr("x", (x(bins[0].x1) - x(bins[0].x0)) / 2)
  278. .attr("text-anchor", "middle")
  279. .text(function(d) { return formatCount(d.length); });
  280. svg.append("g")
  281. .attr("class", "axis axis--x")
  282. .attr("transform", "translate(0," + height + ")")
  283. .call(d3.axisBottom(x));
  284. }
  285. $.get(
  286. '/jmx?qry=Hadoop:service=NameNode,name=NameNodeInfo',
  287. guard_with_startup_progress(function (resp) {
  288. var data = workaround(resp.beans[0]);
  289. var base = dust.makeBase(HELPERS);
  290. dust.render('datanode-info', base.push(data), function(err, out) {
  291. $('#tab-datanode').html(out);
  292. $('#table-datanodes').dataTable( {
  293. 'lengthMenu': [ [25, 50, 100, -1], [25, 50, 100, "All"] ],
  294. 'columns': [
  295. { 'orderDataType': 'ng-value', 'searchable': true , "defaultContent": "" },
  296. { 'orderDataType': 'ng-value', 'searchable': true , "defaultContent": ""},
  297. { 'orderDataType': 'ng-value', 'type': 'num' , "defaultContent": 0},
  298. { 'orderDataType': 'ng-value', 'type': 'num' , "defaultContent": 0},
  299. { 'orderDataType': 'ng-value', 'type': 'num' , "defaultContent": 0},
  300. { 'type': 'num' , "defaultContent": 0},
  301. { 'orderDataType': 'ng-value', 'type': 'num' , "defaultContent": 0},
  302. { 'type': 'string' , "defaultContent": ""}
  303. ]});
  304. renderHistogram(data);
  305. $('#ui-tabs a[href="#tab-datanode"]').tab('show');
  306. });
  307. })).fail(ajax_error_handler);
  308. }
  309. function load_datanode_volume_failures() {
  310. var HELPERS = {
  311. 'helper_date_tostring' : function (chunk, ctx, bodies, params) {
  312. var value = dust.helpers.tap(params.value, chunk, ctx);
  313. return chunk.write('' + moment(Number(value)).format('ddd MMM DD HH:mm:ss ZZ YYYY'));
  314. }
  315. };
  316. function workaround(r) {
  317. function node_map_to_array(nodes) {
  318. var res = [];
  319. for (var n in nodes) {
  320. var p = nodes[n];
  321. // Filter the display to only datanodes with volume failures.
  322. if (p.volfails > 0) {
  323. p.name = n;
  324. res.push(p);
  325. }
  326. }
  327. return res;
  328. }
  329. r.LiveNodes = node_map_to_array(JSON.parse(r.LiveNodes));
  330. return r;
  331. }
  332. $.get(
  333. '/jmx?qry=Hadoop:service=NameNode,name=NameNodeInfo',
  334. guard_with_startup_progress(function (resp) {
  335. var data = workaround(resp.beans[0]);
  336. var base = dust.makeBase(HELPERS);
  337. dust.render('datanode-volume-failures', base.push(data), function(err, out) {
  338. $('#tab-datanode-volume-failures').html(out);
  339. $('#ui-tabs a[href="#tab-datanode-volume-failures"]').tab('show');
  340. });
  341. })).fail(ajax_error_handler);
  342. }
  343. function load_snapshot_info() {
  344. $.get(
  345. '/jmx?qry=Hadoop:service=NameNode,name=SnapshotInfo',
  346. guard_with_startup_progress(function (resp) {
  347. dust.render('snapshot-info', resp.beans[0], function(err, out) {
  348. $('#tab-snapshot').html(out);
  349. $('#ui-tabs a[href="#tab-snapshot"]').tab('show');
  350. });
  351. })).fail(ajax_error_handler);
  352. }
  353. function load_page() {
  354. var hash = window.location.hash;
  355. switch(hash) {
  356. case "#tab-datanode":
  357. load_datanode_info();
  358. break;
  359. case "#tab-datanode-volume-failures":
  360. load_datanode_volume_failures();
  361. break;
  362. case "#tab-snapshot":
  363. load_snapshot_info();
  364. break;
  365. case "#tab-startup-progress":
  366. load_startup_progress();
  367. break;
  368. case "#tab-overview":
  369. load_overview();
  370. break;
  371. default:
  372. window.location.hash = "tab-overview";
  373. break;
  374. }
  375. }
  376. load_page();
  377. $(window).bind('hashchange', function () {
  378. load_page();
  379. });
  380. })();