index.php 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282
  1. <!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01//EN" "http://www.w3.org/TR/html4/strict.dtd">
  2. <html>
  3. <head>
  4. <meta http-equiv="Content-Type" content="text/html; charset=utf-8">
  5. <title>TS3 channel usage</title>
  6. <meta name="viewport" content="width=device-width, initial-scale=1.0">
  7. <link href="stylesheet.css" rel="stylesheet" type="text/css">
  8. <!--[if lte IE 8]><script language="javascript" type="text/javascript" src="flot/excanvas.min.js"></script><![endif]-->
  9. <script language="javascript" type="text/javascript" src="flot/jquery.min.js"></script>
  10. <script language="javascript" type="text/javascript" src="flot/jquery.flot.min.js"></script>
  11. <script language="javascript" type="text/javascript" src="flot/jquery.flot.stack.min.js"></script>
  12. <script language="javascript" type="text/javascript" src="flot/jquery.flot.time.min.js"></script>
  13. <script language="javascript" type="text/javascript" src="flot/jquery.flot.navigate.min.js"></script>
  14. <script language="javascript" type="text/javascript" src="flot/jquery.flot.resize.min.js"></script>
  15. </head>
  16. <body>
  17. <div class="wrapper">
  18. <header class="main-head">TS3 channel usage</header>
  19. <div id="graphs"></div>
  20. <div id="settings">
  21. <input type=button onclick="updateDate(addDays(startDate, -1))" value="-1 Day">
  22. <input type="date" onchange="updateDate(this.valueAsDate)" id="date">
  23. <input type=button onclick="updateDate(addDays(startDate, 1))" value="+1 Day">
  24. </div>
  25. <div id="tree"></div>
  26. </div>
  27. <script>
  28. var startDate, endDate;
  29. var eventTimeRange = [0, 0];
  30. function updateDate(newDate) {
  31. newDate.setHours(6, 0, 0, 0);
  32. if(newDate.getTime() > eventTimeRange[1])
  33. newDate = new Date(eventTimeRange[1]);
  34. if(newDate.getTime() < eventTimeRange[0])
  35. newDate = new Date(eventTimeRange[0]);
  36. newDate.setHours(6, 0, 0, 0);
  37. startDate = newDate;
  38. endDate = addDays(startDate, 1);
  39. document.getElementById('date').valueAsDate = startDate;
  40. updatePlots();
  41. }
  42. function update(init = false) {
  43. fetch("ajax.php").then(res=>res.json()).then(function(json) {
  44. console.log(json);
  45. let channelTree = {channelId: 0, name: 'Server', id: 0};
  46. eventTimeRange[0] = json.events.reduce((min, e) => min && (min < e[1]) ? min : e[1]) * 1000 - 12*3600000;
  47. eventTimeRange[1] = json.events.reduce((max, e) => max && (max > e[1]) ? max : e[1]) * 1000 + 12*3600000;
  48. var dateInp = document.getElementById('date');
  49. dateInp.setAttribute('min', (new Date(eventTimeRange[0])).toISOString().split("T")[0]);
  50. dateInp.setAttribute('max', (new Date(eventTimeRange[1])).toISOString().split("T")[0]);
  51. if(init)
  52. updateDate(new Date());
  53. makeTree(channelTree, json.channels, json.events);
  54. showTree(channelTree, document.getElementById('tree'));
  55. if(init) {
  56. for(const channel of channelTree.children) {
  57. if(channel.activeCount == 0)
  58. continue;
  59. let inp = document.getElementById(`cb_${channel.channelId}`);
  60. inp.checked = true;
  61. plotEvents(channel, document.getElementById("graphs"), true);
  62. }
  63. }
  64. console.log(channelTree);
  65. });
  66. }
  67. update(true);
  68. setInterval(update, 30 * 1000);
  69. function makeTree(tree, channels, events) {
  70. tree.events = events.filter(e => e[0] == tree.id);
  71. if(tree.events.length > 0) {
  72. tree.events.splice(0,0,[tree.events[0][0], (new Date()).getTime()/1000, tree.events[0][2]]);
  73. }
  74. tree.children = channels.filter((ch) => ch.parentId == tree.channelId).sort((a, b) => a.position - b.position);
  75. tree.clientCount = tree.events.reduce((sum, e) => sum+e[2], 0) / (tree.events.length > 0 ? tree.events.length : 1);
  76. tree.activeCount = tree.events.length > 0 ? tree.events[0][2] : 0;
  77. for(const child of tree.children) {
  78. makeTree(child, channels.filter( ch => ch.parentId != tree.channelId), events.filter( e => e[0] != tree.id));
  79. tree.clientCount += child.clientCount;
  80. tree.activeCount += child.activeCount;
  81. }
  82. }
  83. function showTree(tree, el) {
  84. let ul = document.getElementById(`ul_${tree.channelId}`);
  85. if(!ul) {
  86. let inp = document.createElement('input');
  87. inp.type = 'checkbox';
  88. inp.id = `cb_${tree.channelId}`;
  89. inp.classList.add('checkbox');
  90. el.appendChild(inp);
  91. let cnt = document.createElement('label');
  92. cnt.classList.add("count");
  93. cnt.htmlFor = `cb_${tree.channelId}`;
  94. el.appendChild(cnt);
  95. let label = document.createElement('label');
  96. label.htmlFor = `cb_${tree.channelId}`;
  97. el.appendChild(label);
  98. ul = document.createElement('ul');
  99. ul.id = `ul_${tree.channelId}`;
  100. el.appendChild(ul);
  101. }
  102. let inp = document.getElementById(`cb_${tree.channelId}`);
  103. inp.onchange = function() {
  104. plotEvents(tree, document.getElementById("graphs"), this.checked);
  105. };
  106. if(inp.checked) {
  107. plotEvents(tree, document.getElementById("graphs"), true);
  108. }
  109. if(tree.events.length > 0)
  110. inp.nextSibling.textContent = tree.events[0][2] != 0 ? tree.events[0][2] : '';
  111. inp.nextSibling.nextSibling.innerHTML = formatHeadings(tree.name);
  112. let prevLi = {};
  113. let showSpacer = false;
  114. for(const channel of tree.children) {
  115. if(channel.clientCount == 0) {
  116. showSpacer = formatHeadings(channel.name, false).length == 1;
  117. continue;
  118. }
  119. let li = document.getElementById(`li_${channel.channelId}`);
  120. if(!li) {
  121. li = document.createElement('li');
  122. li.id = `li_${channel.channelId}`;
  123. if(prevLi.nextSibling)
  124. ul.insertBefore(li, prevLi.nextSibling);
  125. else
  126. ul.appendChild(li);
  127. }
  128. if(showSpacer) {
  129. li.classList.add("spacer");
  130. showSpacer = false;
  131. }
  132. prevLi = li;
  133. showTree(channel, li);
  134. }
  135. }
  136. function formatHeadings(str, html = true) {
  137. const res = /(?:\[(.)spacer[^\]]*\])?(.*)/.exec(str);
  138. if(!html)
  139. return res[2];
  140. switch(res[1]) {
  141. case '*':
  142. return `<div class='channel'>${res[2].repeat(50)}</div>`;
  143. case 'l':
  144. return `<div class='channel' align='left'>${res[2]}</div>`;
  145. case 'c':
  146. return `<div class='channel' align='center'>${res[2]}</div>`;
  147. case 'r':
  148. return `<div class='channel' align='right'>${res[2]}</div>`;
  149. default:
  150. return `<div class='channel'>${res[2]}</div>`;
  151. }
  152. }
  153. var plots = {};
  154. var viewMin = 0;
  155. var viewMax = 0;
  156. function plotEvents(tree, div, visible = true) {
  157. let placeholder = document.getElementById(`graph_${tree.channelId}`);
  158. if(!placeholder) {
  159. container = document.createElement("div");
  160. container.classList.add("demo-container");
  161. div.appendChild(container);
  162. placeholder = document.createElement("div");
  163. placeholder.classList.add("demo-placeholder");
  164. placeholder.id = `graph_${tree.channelId}`;
  165. container.appendChild(placeholder);
  166. }
  167. if(!document.getElementById(`cb_${tree.channelId}`).checked) {
  168. placeholder.parentNode.style.display = 'none';
  169. return;
  170. } else {
  171. placeholder.parentNode.style.display = 'block';
  172. }
  173. let series = [];
  174. function iterChilds(tree, prefix="") {
  175. const data = tree.events.map(e => [e[1]*1000, e[2]]);
  176. for(let i=0; i<data.length-2; i++) {
  177. data[i][1] = data[i+1][1];
  178. }
  179. const name = prefix + formatHeadings(tree.name, false);
  180. if(data.length > 0 && (data.length != 2 || data[0][1] != 0)) {
  181. series.push({
  182. label: name,
  183. data: data,
  184. lines: {
  185. show: true,
  186. fill: true,
  187. steps: true
  188. }
  189. });
  190. }
  191. for(const channel of tree.children) {
  192. iterChilds(channel, name + ' / ');
  193. }
  194. }
  195. iterChilds(tree);
  196. if(!plots[tree.channelId]) {
  197. plots[tree.channelId] = $.plot(placeholder, series, {
  198. xaxis: {
  199. mode: "time",
  200. timeBase: "milliseconds",
  201. timezone: "browser",
  202. min: viewMin ? viewMin : startDate.getTime(),
  203. max: viewMax ? viewMax : endDate.getTime(),
  204. zoomRange: [60000, null],
  205. panRange: eventTimeRange
  206. },
  207. yaxis: {
  208. zoomRange: false,
  209. panRange: false
  210. },
  211. zoom: {
  212. interactive: true
  213. },
  214. pan: {
  215. interactive: true
  216. },
  217. legend: {
  218. position: 'nw'
  219. }
  220. });
  221. $(placeholder).bind("plotpan plotzoom", function (event, plot) {
  222. var axes = plot.getAxes();
  223. viewMin = axes.xaxis.min;
  224. viewMax = axes.xaxis.max;
  225. for(const id in plots) {
  226. plots[id].getOptions().xaxes[0].min = axes.xaxis.min;
  227. plots[id].getOptions().xaxes[0].max = axes.xaxis.max;
  228. if(!document.getElementById(`cb_${id}`).checked)
  229. continue;
  230. plots[id].setupGrid();
  231. plots[id].draw();
  232. }
  233. });
  234. } else {
  235. plots[tree.channelId].getOptions().xaxis.panRange = eventTimeRange;
  236. plots[tree.channelId].setData(series);
  237. plots[tree.channelId].setupGrid();
  238. plots[tree.channelId].draw();
  239. }
  240. }
  241. function updatePlots() {
  242. viewMin = startDate.getTime();
  243. viewMax = endDate.getTime();
  244. for(const id in plots) {
  245. let opt = plots[id].getXAxes()[0].options;
  246. opt.min = viewMin;
  247. opt.max = viewMax;
  248. if(!document.getElementById(`cb_${id}`).checked)
  249. continue;
  250. plots[id].setupGrid();
  251. plots[id].draw();
  252. }
  253. }
  254. function addDays(date, days) {
  255. var result = new Date(date);
  256. result.setDate(result.getDate() + days);
  257. return result;
  258. }
  259. </script>
  260. </body>
  261. </html>