rubyfox-server 2.17.3.2 → 2.19.2.0

Sign up to get free protection for your applications and to get access to all the features.
Files changed (160) hide show
  1. checksums.yaml +4 -4
  2. data/lib/rubyfox/server/data/config/admin/descriptors/config_room.txt +10 -1
  3. data/lib/rubyfox/server/data/config/admin/descriptors/config_server.txt +90 -20
  4. data/lib/rubyfox/server/data/config/admin/descriptors/config_zone.txt +9 -0
  5. data/lib/rubyfox/server/data/config/admin/descriptors/runtime_room.txt +11 -0
  6. data/lib/rubyfox/server/data/config/admin/descriptors/runtime_user.txt +3 -3
  7. data/lib/rubyfox/server/data/config/core.xml +4 -4
  8. data/lib/rubyfox/server/data/config/default.words.txt +11 -0
  9. data/lib/rubyfox/server/data/config/log4j.properties +1 -2
  10. data/lib/rubyfox/server/data/config/server.xml +1 -1
  11. data/lib/rubyfox/server/data/data/GeoLite2-Country.mmdb +0 -0
  12. data/lib/rubyfox/server/data/data/bannedusers/users.bin +0 -0
  13. data/lib/rubyfox/server/data/lib/apache-tomcat/bin/bootstrap.jar +0 -0
  14. data/lib/rubyfox/server/data/lib/apache-tomcat/bin/catalina-tasks.xml +39 -39
  15. data/lib/rubyfox/server/data/lib/apache-tomcat/bin/catalina.sh +0 -0
  16. data/lib/rubyfox/server/data/lib/apache-tomcat/bin/ciphers.sh +0 -0
  17. data/lib/rubyfox/server/data/lib/apache-tomcat/bin/commons-daemon-native.tar.gz +0 -0
  18. data/lib/rubyfox/server/data/lib/apache-tomcat/bin/commons-daemon.jar +0 -0
  19. data/lib/rubyfox/server/data/lib/apache-tomcat/bin/configtest.sh +0 -0
  20. data/lib/rubyfox/server/data/lib/apache-tomcat/bin/daemon.sh +0 -0
  21. data/lib/rubyfox/server/data/lib/apache-tomcat/bin/digest.sh +0 -0
  22. data/lib/rubyfox/server/data/lib/apache-tomcat/bin/makebase.sh +0 -0
  23. data/lib/rubyfox/server/data/lib/apache-tomcat/bin/setclasspath.sh +0 -0
  24. data/lib/rubyfox/server/data/lib/apache-tomcat/bin/shutdown.sh +0 -0
  25. data/lib/rubyfox/server/data/lib/apache-tomcat/bin/startup.sh +0 -0
  26. data/lib/rubyfox/server/data/lib/apache-tomcat/bin/tomcat-juli.jar +0 -0
  27. data/lib/rubyfox/server/data/lib/apache-tomcat/bin/tomcat-native.tar.gz +0 -0
  28. data/lib/rubyfox/server/data/lib/apache-tomcat/bin/tool-wrapper.sh +0 -0
  29. data/lib/rubyfox/server/data/lib/apache-tomcat/bin/version.sh +0 -0
  30. data/lib/rubyfox/server/data/lib/apache-tomcat/conf/Catalina/localhost/rewrite.config +1 -1
  31. data/lib/rubyfox/server/data/lib/apache-tomcat/conf/catalina.policy +263 -263
  32. data/lib/rubyfox/server/data/lib/apache-tomcat/conf/catalina.properties +209 -207
  33. data/lib/rubyfox/server/data/lib/apache-tomcat/conf/context.xml +31 -31
  34. data/lib/rubyfox/server/data/lib/apache-tomcat/conf/jaspic-providers.xml +23 -23
  35. data/lib/rubyfox/server/data/lib/apache-tomcat/conf/jaspic-providers.xsd +52 -52
  36. data/lib/rubyfox/server/data/lib/apache-tomcat/conf/keystore.jks +0 -0
  37. data/lib/rubyfox/server/data/lib/apache-tomcat/conf/server.xml +177 -161
  38. data/lib/rubyfox/server/data/lib/apache-tomcat/conf/tomcat-users.xml +18 -7
  39. data/lib/rubyfox/server/data/lib/apache-tomcat/conf/tomcat-users.xsd +59 -59
  40. data/lib/rubyfox/server/data/lib/apache-tomcat/conf/web.xml +4740 -4737
  41. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/annotations-api.jar +0 -0
  42. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/catalina-ant.jar +0 -0
  43. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/catalina-ha.jar +0 -0
  44. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/catalina-ssi.jar +0 -0
  45. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/catalina-storeconfig.jar +0 -0
  46. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/catalina-tribes.jar +0 -0
  47. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/catalina.jar +0 -0
  48. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/el-api.jar +0 -0
  49. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/jasper-el.jar +0 -0
  50. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/jasper.jar +0 -0
  51. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/jaspic-api.jar +0 -0
  52. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/jsp-api.jar +0 -0
  53. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/servlet-api.jar +0 -0
  54. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/sfs2x-ws-helper.jar +0 -0
  55. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/tomcat-api.jar +0 -0
  56. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/tomcat-coyote.jar +0 -0
  57. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/tomcat-dbcp.jar +0 -0
  58. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/tomcat-i18n-cs.jar +0 -0
  59. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/tomcat-i18n-de.jar +0 -0
  60. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/tomcat-i18n-es.jar +0 -0
  61. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/tomcat-i18n-fr.jar +0 -0
  62. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/tomcat-i18n-ja.jar +0 -0
  63. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/tomcat-i18n-ko.jar +0 -0
  64. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/tomcat-i18n-pt-BR.jar +0 -0
  65. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/tomcat-i18n-ru.jar +0 -0
  66. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/tomcat-i18n-zh-CN.jar +0 -0
  67. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/tomcat-jdbc.jar +0 -0
  68. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/tomcat-jni.jar +0 -0
  69. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/tomcat-util-scan.jar +0 -0
  70. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/tomcat-util.jar +0 -0
  71. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/tomcat-websocket.jar +0 -0
  72. data/lib/rubyfox/server/data/lib/apache-tomcat/lib/websocket-api.jar +0 -0
  73. data/lib/rubyfox/server/data/lib/javax.activation-1.2.0.jar +0 -0
  74. data/lib/rubyfox/server/data/lib/javax.mail.jar +0 -0
  75. data/lib/rubyfox/server/data/lib/js/JSApi.js +2 -1
  76. data/lib/rubyfox/server/data/lib/js/LibApi.js +181 -48
  77. data/lib/rubyfox/server/data/lib/sfs2x-admin.jar +0 -0
  78. data/lib/rubyfox/server/data/lib/sfs2x-cluster.jar +0 -0
  79. data/lib/rubyfox/server/data/lib/sfs2x-core.jar +0 -0
  80. data/lib/rubyfox/server/data/lib/sfs2x.jar +0 -0
  81. data/lib/rubyfox/server/data/sfs2x-service +26 -30
  82. data/lib/rubyfox/server/data/www/BlueBox.war +0 -0
  83. data/lib/rubyfox/server/data/www/HelloServlet/WEB-INF/web.xml +1 -3
  84. data/lib/rubyfox/server/data/www/ROOT/_css_/default.css +14 -6
  85. data/lib/rubyfox/server/data/www/ROOT/admin/assets/css/style.css +44 -2
  86. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/core/application.bundle.js +98 -61
  87. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/core/modules/endors~mod-0~mod-1~mod-11~mod-12~mod-17~mod-6~mod-7~mod-8~mod-9.bundle.js +17357 -0
  88. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/core/modules/module-0.bundle.js +4 -4
  89. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/core/modules/module-1.bundle.js +3 -3
  90. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/core/modules/module-10.bundle.js +101 -66
  91. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/core/modules/module-11.bundle.js +544 -8
  92. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/core/modules/module-12.bundle.js +915 -1480
  93. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/core/modules/module-12~module-15~module-16~module-4.bundle.js +2665 -0
  94. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/core/modules/module-13.bundle.js +606 -3093
  95. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/core/modules/module-13~module-16~module-17~module-4.bundle.js +2665 -0
  96. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/core/modules/module-14.bundle.js +764 -0
  97. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/core/modules/module-15.bundle.js +71 -0
  98. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/core/modules/module-16.bundle.js +1787 -0
  99. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/core/modules/module-17.bundle.js +3383 -0
  100. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/core/modules/module-4.bundle.js +121 -1009
  101. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/core/modules/module-5.bundle.js +1214 -1744
  102. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/core/modules/module-6.bundle.js +398 -666
  103. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/core/modules/module-7.bundle.js +717 -192
  104. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/core/modules/module-8.bundle.js +2117 -665
  105. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/core/modules/module-9.bundle.js +613 -690
  106. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/core/modules/vendors~mod-0~mod-1~mod-10~mod-11~mod-16~mod-5~mod-6~mod-7~mod-8.bundle.js +17357 -0
  107. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/core/modules/vendors~mod-0~mod-1~mod-11~mod-12~mod-17~mod-5~mod-6~mod-7~mod-8~mod-9.bundle.js +17357 -0
  108. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/core/modules/{vendors~module-0~module-1~module-13~module-4~module-5~module-7~module-8.bundle.js → vendors~mod-0~mod-1~mod-11~mod-12~mod-17~mod-5~mod-7~mod-8~mod-9.bundle.js} +2 -2
  109. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/core/modules/vendors~module-12.bundle.js +807 -0
  110. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/core/modules/vendors~module-13.bundle.js +807 -0
  111. data/lib/rubyfox/server/data/www/ROOT/admin/modules/cluster-configurator.html +32 -0
  112. data/lib/rubyfox/server/data/www/ROOT/admin/modules/cluster-monitor.html +185 -0
  113. data/lib/rubyfox/server/data/www/ROOT/admin/modules/cluster-updater.html +47 -0
  114. data/lib/rubyfox/server/data/www/ROOT/admin/modules/extension-deployer.html +84 -0
  115. data/lib/rubyfox/server/data/www/ROOT/admin/modules/zone-monitor.html +15 -8
  116. data/lib/rubyfox/server/data/www/ROOT/index.html +13 -23
  117. data/lib/rubyfox/server/data/www/host-manager/META-INF/context.xml +2 -2
  118. data/lib/rubyfox/server/data/www/host-manager/WEB-INF/jsp/404.jsp +2 -2
  119. data/lib/rubyfox/server/data/www/host-manager/{manager.xml → WEB-INF/manager.xml} +5 -1
  120. data/lib/rubyfox/server/data/www/host-manager/WEB-INF/web.xml +17 -0
  121. data/lib/rubyfox/server/data/www/host-manager/css/manager.css +141 -0
  122. data/lib/rubyfox/server/data/www/host-manager/images/tomcat.svg +967 -0
  123. data/lib/rubyfox/server/data/www/manager/META-INF/context.xml +2 -0
  124. data/lib/rubyfox/server/data/www/manager/WEB-INF/jsp/connectorCerts.jsp +1 -1
  125. data/lib/rubyfox/server/data/www/manager/WEB-INF/jsp/connectorCiphers.jsp +1 -1
  126. data/lib/rubyfox/server/data/www/manager/WEB-INF/jsp/connectorTrustedCerts.jsp +1 -1
  127. data/lib/rubyfox/server/data/www/manager/WEB-INF/jsp/sessionDetail.jsp +3 -3
  128. data/lib/rubyfox/server/data/www/manager/WEB-INF/jsp/sessionsList.jsp +1 -1
  129. data/lib/rubyfox/server/data/www/manager/WEB-INF/web.xml +17 -0
  130. data/lib/rubyfox/server/data/www/manager/css/manager.css +141 -0
  131. data/lib/rubyfox/server/data/www/manager/images/tomcat.svg +967 -0
  132. data/lib/rubyfox/server/data/www/manager/xform.xsl +74 -59
  133. data/lib/rubyfox/server/version.rb +1 -1
  134. metadata +30 -31
  135. data/lib/rubyfox/server/data/config/admin/icons/Analytics.png +0 -0
  136. data/lib/rubyfox/server/data/config/admin/icons/BanManager.png +0 -0
  137. data/lib/rubyfox/server/data/config/admin/icons/BlueBoxMonitor.png +0 -0
  138. data/lib/rubyfox/server/data/config/admin/icons/Console.png +0 -0
  139. data/lib/rubyfox/server/data/config/admin/icons/Dashboard.png +0 -0
  140. data/lib/rubyfox/server/data/config/admin/icons/ExtensionManager.png +0 -0
  141. data/lib/rubyfox/server/data/config/admin/icons/LicenseManager.png +0 -0
  142. data/lib/rubyfox/server/data/config/admin/icons/LogViewer.png +0 -0
  143. data/lib/rubyfox/server/data/config/admin/icons/ServerConfigurator.png +0 -0
  144. data/lib/rubyfox/server/data/config/admin/icons/ServletManager.png +0 -0
  145. data/lib/rubyfox/server/data/config/admin/icons/ZoneConfigurator.png +0 -0
  146. data/lib/rubyfox/server/data/config/admin/icons/ZoneMonitor.png +0 -0
  147. data/lib/rubyfox/server/data/lib/BlueBox.war +0 -0
  148. data/lib/rubyfox/server/data/lib/apache-tomcat/LICENSE +0 -1061
  149. data/lib/rubyfox/server/data/lib/apache-tomcat/NOTICE +0 -68
  150. data/lib/rubyfox/server/data/lib/apache-tomcat/README.md +0 -81
  151. data/lib/rubyfox/server/data/lib/apache-tomcat/RELEASE-NOTES +0 -174
  152. data/lib/rubyfox/server/data/lib/imap.jar +0 -0
  153. data/lib/rubyfox/server/data/lib/mailapi.jar +0 -0
  154. data/lib/rubyfox/server/data/lib/pop3.jar +0 -0
  155. data/lib/rubyfox/server/data/lib/smtp.jar +0 -0
  156. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/core/modules/module-12~module-13~module-9.bundle.js +0 -2634
  157. data/lib/rubyfox/server/data/www/ROOT/admin/assets/js/core/modules/vendors~module-9.bundle.js +0 -807
  158. data/lib/rubyfox/server/data/www/host-manager/images/tomcat.gif +0 -0
  159. data/lib/rubyfox/server/data/www/manager/images/tomcat.gif +0 -0
  160. /data/lib/rubyfox/server/data/data/buddylists/{BasicExamples/.keep → .keep} +0 -0
@@ -1,23 +1,6 @@
1
1
  /*! (c) gotoAndPlay | All rights reserved */
2
2
  (window["webpackJsonpapplication"] = window["webpackJsonpapplication"] || []).push([["module-8"],{
3
3
 
4
- /***/ "./node_modules/file-saver/dist/FileSaver.min.js":
5
- /*!*******************************************************!*\
6
- !*** ./node_modules/file-saver/dist/FileSaver.min.js ***!
7
- \*******************************************************/
8
- /*! no static exports found */
9
- /***/ (function(module, exports, __webpack_require__) {
10
-
11
- /* WEBPACK VAR INJECTION */(function(global) {var __WEBPACK_AMD_DEFINE_FACTORY__, __WEBPACK_AMD_DEFINE_ARRAY__, __WEBPACK_AMD_DEFINE_RESULT__;(function(a,b){if(true)!(__WEBPACK_AMD_DEFINE_ARRAY__ = [], __WEBPACK_AMD_DEFINE_FACTORY__ = (b),
12
- __WEBPACK_AMD_DEFINE_RESULT__ = (typeof __WEBPACK_AMD_DEFINE_FACTORY__ === 'function' ?
13
- (__WEBPACK_AMD_DEFINE_FACTORY__.apply(exports, __WEBPACK_AMD_DEFINE_ARRAY__)) : __WEBPACK_AMD_DEFINE_FACTORY__),
14
- __WEBPACK_AMD_DEFINE_RESULT__ !== undefined && (module.exports = __WEBPACK_AMD_DEFINE_RESULT__));else {}})(this,function(){"use strict";function b(a,b){return"undefined"==typeof b?b={autoBom:!1}:"object"!=typeof b&&(console.warn("Deprecated: Expected third argument to be a object"),b={autoBom:!b}),b.autoBom&&/^\s*(?:text\/\S*|application\/xml|\S*\/\S*\+xml)\s*;.*charset\s*=\s*utf-8/i.test(a.type)?new Blob(["\uFEFF",a],{type:a.type}):a}function c(b,c,d){var e=new XMLHttpRequest;e.open("GET",b),e.responseType="blob",e.onload=function(){a(e.response,c,d)},e.onerror=function(){console.error("could not download file")},e.send()}function d(a){var b=new XMLHttpRequest;b.open("HEAD",a,!1);try{b.send()}catch(a){}return 200<=b.status&&299>=b.status}function e(a){try{a.dispatchEvent(new MouseEvent("click"))}catch(c){var b=document.createEvent("MouseEvents");b.initMouseEvent("click",!0,!0,window,0,0,0,80,20,!1,!1,!1,!1,0,null),a.dispatchEvent(b)}}var f="object"==typeof window&&window.window===window?window:"object"==typeof self&&self.self===self?self:"object"==typeof global&&global.global===global?global:void 0,a=f.saveAs||("object"!=typeof window||window!==f?function(){}:"download"in HTMLAnchorElement.prototype?function(b,g,h){var i=f.URL||f.webkitURL,j=document.createElement("a");g=g||b.name||"download",j.download=g,j.rel="noopener","string"==typeof b?(j.href=b,j.origin===location.origin?e(j):d(j.href)?c(b,g,h):e(j,j.target="_blank")):(j.href=i.createObjectURL(b),setTimeout(function(){i.revokeObjectURL(j.href)},4E4),setTimeout(function(){e(j)},0))}:"msSaveOrOpenBlob"in navigator?function(f,g,h){if(g=g||f.name||"download","string"!=typeof f)navigator.msSaveOrOpenBlob(b(f,h),g);else if(d(f))c(f,g,h);else{var i=document.createElement("a");i.href=f,i.target="_blank",setTimeout(function(){e(i)})}}:function(a,b,d,e){if(e=e||open("","_blank"),e&&(e.document.title=e.document.body.innerText="downloading..."),"string"==typeof a)return c(a,b,d);var g="application/octet-stream"===a.type,h=/constructor/i.test(f.HTMLElement)||f.safari,i=/CriOS\/[\d]+/.test(navigator.userAgent);if((i||g&&h)&&"object"==typeof FileReader){var j=new FileReader;j.onloadend=function(){var a=j.result;a=i?a:a.replace(/^data:[^;]*;/,"data:attachment/file;"),e?e.location.href=a:location=a,e=null},j.readAsDataURL(a)}else{var k=f.URL||f.webkitURL,l=k.createObjectURL(a);e?e.location=l:location.href=l,e=null,setTimeout(function(){k.revokeObjectURL(l)},4E4)}});f.saveAs=a.saveAs=a, true&&(module.exports=a)});
15
-
16
- //# sourceMappingURL=FileSaver.min.js.map
17
- /* WEBPACK VAR INJECTION */}.call(this, __webpack_require__(/*! ./../../webpack/buildin/global.js */ "./node_modules/webpack/buildin/global.js")))
18
-
19
- /***/ }),
20
-
21
4
  /***/ "./node_modules/moment/locale sync recursive ^\\.\\/.*$":
22
5
  /*!**************************************************!*\
23
6
  !*** ./node_modules/moment/locale sync ^\.\/.*$ ***!
@@ -304,906 +287,2375 @@ webpackContext.id = "./node_modules/moment/locale sync recursive ^\\.\\/.*$";
304
287
 
305
288
  /***/ }),
306
289
 
307
- /***/ "./node_modules/webpack/buildin/global.js":
308
- /*!***********************************!*\
309
- !*** (webpack)/buildin/global.js ***!
310
- \***********************************/
311
- /*! no static exports found */
312
- /***/ (function(module, exports) {
290
+ /***/ "./src/components/charts/base-chart.js":
291
+ /*!*********************************************!*\
292
+ !*** ./src/components/charts/base-chart.js ***!
293
+ \*********************************************/
294
+ /*! exports provided: BaseChart */
295
+ /***/ (function(module, __webpack_exports__, __webpack_require__) {
296
+
297
+ "use strict";
298
+ __webpack_require__.r(__webpack_exports__);
299
+ /* WEBPACK VAR INJECTION */(function($) {/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "BaseChart", function() { return BaseChart; });
300
+ class BaseChart extends HTMLElement
301
+ {
302
+ constructor(maxValues)
303
+ {
304
+ super();
313
305
 
314
- var g;
306
+ this._maxValues = maxValues;
315
307
 
316
- // This works in non-strict mode
317
- g = (function() {
318
- return this;
319
- })();
308
+ // Initialize
309
+ this._initialize();
310
+ }
320
311
 
321
- try {
322
- // This works if eval is allowed (see CSP)
323
- g = g || new Function("return this")();
324
- } catch (e) {
325
- // This works if the window reference is available
326
- if (typeof window === "object") g = window;
327
- }
312
+ // Can be overridden
313
+ _initialize()
314
+ {
315
+ // Create chart html
316
+ let chartHtml = $('<div>');
317
+ $(this).append(chartHtml);
328
318
 
329
- // g can still be undefined, but nothing to do about it...
330
- // We return undefined, instead of nothing here, so it's
331
- // easier to handle this case. if(!global) { ...}
319
+ this._chartWidget = this._initChartWidget(chartHtml);
332
320
 
333
- module.exports = g;
321
+ // Reset data source
322
+ this.reset();
323
+ }
334
324
 
325
+ // MUST BE OVERRIDDEN
326
+ _initChartWidget()
327
+ {
328
+ throw new Error('Chart widget must be initialized!')
329
+ }
335
330
 
336
- /***/ }),
331
+ set paused(value)
332
+ {
333
+ this.isPaused = value;
334
+ this._onPauseChange();
335
+ }
337
336
 
338
- /***/ "./src/data/runtime-log-entry.js":
339
- /*!***************************************!*\
340
- !*** ./src/data/runtime-log-entry.js ***!
341
- \***************************************/
342
- /*! exports provided: RuntimeLogEntry */
343
- /***/ (function(module, __webpack_exports__, __webpack_require__) {
337
+ get paused()
338
+ {
339
+ return this.isPaused;
340
+ }
344
341
 
345
- "use strict";
346
- __webpack_require__.r(__webpack_exports__);
347
- /* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "RuntimeLogEntry", function() { return RuntimeLogEntry; });
348
- class RuntimeLogEntry
349
- {
350
- constructor(separator)
342
+ reset()
343
+ {
344
+ this._chartWidget.setDataSource([]);
345
+ }
346
+
347
+ redraw()
351
348
  {
352
- this.sep = separator
349
+ this._chartWidget.redraw();
353
350
  }
354
351
 
355
- static fromArray(separator, logEntryData)
352
+ getLastEntry()
356
353
  {
357
- let rle = new RuntimeLogEntry(separator);
354
+ let ds;
358
355
 
359
- rle.date = logEntryData[0];
360
- rle.time = logEntryData[1];
361
- rle.dateTime = rle._getDateTime();
362
- rle.level = logEntryData[2].trim();
363
- rle.thread = logEntryData[3];
364
- rle.clazz = logEntryData[4];
365
- rle.message = logEntryData[6];
356
+ if (this.isPaused)
357
+ ds = this._tempDS;
358
+ else
359
+ ds = this._dataSource;
360
+
361
+ if (ds != null && ds.total() > 0)
362
+ return ds.at(ds.total() - 1);
363
+ else
364
+ return null;
365
+ }
366
+
367
+ _addEntry(newEntry)
368
+ {
369
+ if (this.isPaused)
370
+ {
371
+ if (this._tempDS == null)
372
+ this._tempDS = new kendo.data.DataSource();
373
+
374
+ // Store new value in a temporary array
375
+ this._tempDS.add(newEntry);
376
+ }
377
+ else
378
+ {
379
+ let ds = this._dataSource;
380
+
381
+ if (this._tempDS != null)
382
+ {
383
+ // Concatenate temporary entries to actual data source
384
+ let newArr = $.merge(ds.data(), this._tempDS.data());
385
+
386
+ // Remove values exceeding MAX_VALUES
387
+ if (newArr.length > this._maxValues)
388
+ newArr.splice(0, newArr.length - this._maxValues)
389
+
390
+ // Create new datasource
391
+ let newDS = new kendo.data.DataSource({data: newArr})
392
+
393
+ // Assign new data source to chart
394
+ this._chartWidget.setDataSource(newDS);
366
395
 
367
- return rle;
396
+ // Remove temporary array
397
+ this._tempDS = null;
398
+ }
399
+
400
+ // Add new entry
401
+ ds.add(newEntry);
402
+
403
+ // Remove first item if needed
404
+ if (ds.total() > this._maxValues)
405
+ ds.remove(ds.at(0));
406
+ }
368
407
  }
369
408
 
370
- _getDateTime()
409
+ get _dataSource()
371
410
  {
372
- return this.date + '\n' + this.time;
411
+ return this._chartWidget.dataSource;
412
+ }
413
+
414
+ _onPauseChange()
415
+ {
416
+ // Nothing to do, should be overridden
373
417
  }
374
418
  }
375
419
 
420
+ // DEFINE COMPONENT
421
+ if (!window.customElements.get('base-chart'))
422
+ window.customElements.define('base-chart', BaseChart);
423
+
424
+ /* WEBPACK VAR INJECTION */}.call(this, __webpack_require__(/*! jquery */ "jquery")))
376
425
 
377
426
  /***/ }),
378
427
 
379
- /***/ "./src/modules/log-viewer.js":
380
- /*!***********************************!*\
381
- !*** ./src/modules/log-viewer.js ***!
382
- \***********************************/
383
- /*! exports provided: default */
428
+ /***/ "./src/components/charts/chart-utils.js":
429
+ /*!**********************************************!*\
430
+ !*** ./src/components/charts/chart-utils.js ***!
431
+ \**********************************************/
432
+ /*! exports provided: getFormattedDateTime, getBasicSharedTemplate, getMemorySharedTemplate, getMemoryValueAxisLabelTemplate, getNetworkSharedTemplate, getNetworkValueAxisLabelTemplate, getNetworkValueAxisLabelTemplateB, getUserCountSharedTemplate, getTimeRangeTooltipTemplate, getWRBytesLabelTemplate, getBasicTemplate, getMemoryTemplate, getNetworkTemplate */
384
433
  /***/ (function(module, __webpack_exports__, __webpack_require__) {
385
434
 
386
435
  "use strict";
387
436
  __webpack_require__.r(__webpack_exports__);
388
- /* WEBPACK VAR INJECTION */(function($) {/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "default", function() { return LogViewer; });
389
- /* harmony import */ var _base_module__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ./base-module */ "./src/modules/base-module.js");
390
- /* harmony import */ var _data_runtime_log_entry__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ../data/runtime-log-entry */ "./src/data/runtime-log-entry.js");
391
- /* harmony import */ var file_saver__WEBPACK_IMPORTED_MODULE_2__ = __webpack_require__(/*! file-saver */ "./node_modules/file-saver/dist/FileSaver.min.js");
392
- /* harmony import */ var file_saver__WEBPACK_IMPORTED_MODULE_2___default = /*#__PURE__*/__webpack_require__.n(file_saver__WEBPACK_IMPORTED_MODULE_2__);
393
- /* harmony import */ var moment__WEBPACK_IMPORTED_MODULE_3__ = __webpack_require__(/*! moment */ "./node_modules/moment/moment.js");
394
- /* harmony import */ var moment__WEBPACK_IMPORTED_MODULE_3___default = /*#__PURE__*/__webpack_require__.n(moment__WEBPACK_IMPORTED_MODULE_3__);
395
- /* harmony import */ var _utils_utilities__WEBPACK_IMPORTED_MODULE_4__ = __webpack_require__(/*! ../utils/utilities */ "./src/utils/utilities.js");
437
+ /* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "getFormattedDateTime", function() { return getFormattedDateTime; });
438
+ /* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "getBasicSharedTemplate", function() { return getBasicSharedTemplate; });
439
+ /* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "getMemorySharedTemplate", function() { return getMemorySharedTemplate; });
440
+ /* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "getMemoryValueAxisLabelTemplate", function() { return getMemoryValueAxisLabelTemplate; });
441
+ /* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "getNetworkSharedTemplate", function() { return getNetworkSharedTemplate; });
442
+ /* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "getNetworkValueAxisLabelTemplate", function() { return getNetworkValueAxisLabelTemplate; });
443
+ /* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "getNetworkValueAxisLabelTemplateB", function() { return getNetworkValueAxisLabelTemplateB; });
444
+ /* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "getUserCountSharedTemplate", function() { return getUserCountSharedTemplate; });
445
+ /* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "getTimeRangeTooltipTemplate", function() { return getTimeRangeTooltipTemplate; });
446
+ /* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "getWRBytesLabelTemplate", function() { return getWRBytesLabelTemplate; });
447
+ /* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "getBasicTemplate", function() { return getBasicTemplate; });
448
+ /* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "getMemoryTemplate", function() { return getMemoryTemplate; });
449
+ /* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "getNetworkTemplate", function() { return getNetworkTemplate; });
450
+ /* harmony import */ var _utils_utilities__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ../../utils/utilities */ "./src/utils/utilities.js");
451
+
452
+
453
+ function getFormattedDateTime(date, format = 'dd MMM HH:mm:ss')
454
+ {
455
+ return kendo.toString(date, format);
456
+ }
457
+
458
+ function getBasicSharedTemplate(valueFormat = '{0}') {
459
+ return kendo.template(`
460
+ <div class="chart-tooltip-category"><strong>#: category #</strong></div>
461
+ # for (var i = 0; i < points.length; i++) { #
462
+ <div>
463
+ <i class="fas fa-square" style="color: #: points[i].color # "></i>
464
+ #: points[i].series.name# : #: kendo.format('${valueFormat}', points[i].value) #
465
+ </div>
466
+ # } #
467
+ `);
468
+ }
396
469
 
470
+ function getMemorySharedTemplate() {
471
+ return function(dataItem) {
472
+ dataItem.bytesToSize = _utils_utilities__WEBPACK_IMPORTED_MODULE_0__["bytesToSize"]; // Pass bytesToSize utility function to template
473
+ return kendo.template(`
474
+ <div class="chart-tooltip-category"><strong>#: category #</strong></div>
475
+ # for (var i = 0; i < points.length; i++) { #
476
+ <div>
477
+ <i class="fas fa-square" style="color: #: points[i].color # "></i>
478
+ #: points[i].series.name# : #: bytesToSize(points[i].value) #
479
+ </div>
480
+ # } #
481
+ `)(dataItem);
482
+ };
483
+ }
397
484
 
485
+ function getMemoryValueAxisLabelTemplate() {
486
+ return function(dataItem) {
487
+ dataItem.bytesToSize = _utils_utilities__WEBPACK_IMPORTED_MODULE_0__["bytesToSize"]; // Pass bytesToSize utility function to template
488
+ return kendo.template(`
489
+ #: bytesToSize(value) #
490
+ `)(dataItem);
491
+ };
492
+ }
398
493
 
494
+ function getNetworkSharedTemplate(categoryFormat = 'dd MMM HH:mm:ss') {
495
+ return function(dataItem) {
496
+ dataItem.kBytesToSize = _utils_utilities__WEBPACK_IMPORTED_MODULE_0__["kBytesToSize"]; // Pass kBytesToSize utility function to template
497
+ return kendo.template(`
498
+ <div class="chart-tooltip-category"><strong>#: kendo.toString(category, '${categoryFormat}') #</strong></div>
499
+ # for (var i = 0; i < points.length; i++) { #
500
+ <div>
501
+ <i class="fas fa-square" style="color: #: points[i].color # "></i>
502
+ #: points[i].series.name# : #: kBytesToSize(points[i].value, 1, '', '/s') #
503
+ </div>
504
+ # } #
505
+ `)(dataItem);
506
+ };
507
+ }
399
508
 
509
+ function getNetworkValueAxisLabelTemplate() {
510
+ return function(dataItem) {
511
+ dataItem.kBytesToSize = _utils_utilities__WEBPACK_IMPORTED_MODULE_0__["kBytesToSize"]; // Pass kBytesToSize utility function to template
512
+ return kendo.template(`
513
+ #: kBytesToSize(value, 1, '', '/s') #
514
+ `)(dataItem);
515
+ };
516
+ }
400
517
 
518
+ function getNetworkValueAxisLabelTemplateB() {
519
+ return function(dataItem) {
520
+ dataItem.bytesToSize = _utils_utilities__WEBPACK_IMPORTED_MODULE_0__["bytesToSize"]; // Pass bytesToSize utility function to template
521
+ return kendo.template(`
522
+ #: bytesToSize(value, 1, '', '/s') #
523
+ `)(dataItem);
524
+ };
525
+ }
401
526
 
402
- class LogViewer extends _base_module__WEBPACK_IMPORTED_MODULE_0__["BaseModule"]
403
- {
404
- constructor()
405
- {
406
- super('logViewer');
527
+ function getUserCountSharedTemplate(categoryFormat = 'dd MMM HH:mm:ss') {
528
+ return function(dataItem) {
529
+ return kendo.template(`
530
+ <div class="chart-tooltip-category"><strong>#: kendo.toString(category, '${categoryFormat}') #</strong></div>
531
+ # for (var i = 0; i < points.length; i++) { #
532
+ <div>
533
+ <i class="fas fa-square" style="color: #: points[i].color # "></i>
534
+ #: points[i].series.name# : #: points[i].value #
535
+ </div>
536
+ # } #
537
+ `)(dataItem);
538
+ };
539
+ }
407
540
 
408
- this.COMMANDS_PREFIX = 'logViewer';
409
- this.HELP_URL = '/admintool-LogViewer';
541
+ function getTimeRangeTooltipTemplate() {
542
+ return function(dataItem) {
543
+ dataItem.getTooltipString = function(v1, v2) {
544
+ if (v1 == v2)
545
+ {
546
+ if (v2 == -24)
547
+ v2 = -23;
410
548
 
411
- this.ANY_LEVEL = '[any]';
412
- this.LEVELS = ['TRACE','DEBUG','INFO','WARN','ERROR'];
413
- this.ANY_CLASS = '[any]';
549
+ v1 = v2 - 1;
550
+ }
414
551
 
415
- this.LOG_DATA_LABEL = 'lines';
552
+ let start = Math.abs(v1);
553
+ let end = Math.abs(v2);
416
554
 
417
- this.BOOT_LOG_BACKUP_ID = 'SFS2X_BootLog';
418
- this.RUNTIME_LOG_BACKUP_ID = 'SFS2X_RuntimeLog';
419
- this.FULL_BACKUP_ID = 'SFS2X_Logs';
555
+ return `From ${start} hour${start != 1 ? 's' : ''} ago to ${end > 0 ? ('to ' + end + ' hour' + (end != 1 ? 's' : '') + ' ago') : 'now'}`;
556
+ }; // Pass utility function to template
557
+ return kendo.template(`
558
+ #: getTooltipString(selectionStart, selectionEnd) #
559
+ `)(dataItem);
560
+ };
561
+ }
420
562
 
421
- // Outgoing requests
422
- this.REQ_GET_RUNTIME_LOG = 'getRunLog';
423
- this.REQ_GET_BOOT_LOG = 'getBootLog';
424
- this.REQ_GET_BACKUPS_STATUS = 'getBakStatus';
425
- this.REQ_BACKUP_BOOT_LOG = 'bakBootLog';
426
- this.REQ_BACKUP_RUNTIME_LOG = 'bakRunLog';
427
- this.REQ_BACKUP_FULL_LOGS = 'bakFullLogs';
428
- this.REQ_DELETE_BACKUP = 'delBackup';
563
+ function getWRBytesLabelTemplate() {
564
+ return function(dataItem) {
565
+ return kendo.template(`
566
+ #: value # #: dataItem.unit #
567
+ `)(dataItem);
568
+ };
569
+ }
429
570
 
430
- // Incoming responses
431
- this.RESP_INIT_ERROR = 'initErr';
432
- this.RESP_RUNTIME_LOG_ERROR = 'runLogErr';
433
- this.RESP_RUNTIME_LOG_INVALID = 'runLogInv';
434
- this.RESP_RUNTIME_LOG = 'runLog';
435
- this.RESP_BOOT_LOG_ERROR = 'bootLogErr';
436
- this.RESP_BOOT_LOG = 'bootLog';
437
- this.RESP_BACKUPS_STATUS = 'bakStatus';
438
- this.RESP_DELETE_BACKUP_FAILED = 'delBakFail';
439
- this.RESP_BACKUP_ERROR = 'bakError';
440
- this.RESP_BACKUP_WARNING = 'bakWarn';
441
- }
571
+ function getBasicTemplate(valueFormat = '{0}') {
572
+ return kendo.template(`
573
+ <div>
574
+ <span class="text-nowrap">#: kendo.toString(new Date(value.x), "dd/MM/yy HH:mm") #</span>
575
+ <br>
576
+ <strong class="text-nowrap">#: series.name #: #: kendo.format('${valueFormat}', value.y) # </strong>
577
+ </div>
578
+ `);
579
+ }
442
580
 
443
- //------------------------------------
444
- // COMMON MODULE INTERFACE METHODS
445
- // This members are used by the main controller
446
- // to communicate with the module's controller.
447
- //------------------------------------
581
+ function getMemoryTemplate() {
582
+ return function(dataItem) {
583
+ dataItem.bytesToSize = _utils_utilities__WEBPACK_IMPORTED_MODULE_0__["bytesToSize"]; // Pass bytesToSize utility function to template
584
+ return kendo.template(`
585
+ <div>
586
+ <span class="text-nowrap">#: kendo.toString(new Date(value.x), "dd/MM/yy HH:mm") #</span>
587
+ <br>
588
+ <strong class="text-nowrap">#: series.name #: #: bytesToSize(value.y) # </strong>
589
+ </div>
590
+ `)(dataItem);
591
+ };
592
+ }
448
593
 
449
- initialize(idData, shellController)
450
- {
451
- // Call super method
452
- super.initialize(idData, shellController);
594
+ function getNetworkTemplate() {
595
+ return function(dataItem) {
596
+ dataItem.bytesToSize = _utils_utilities__WEBPACK_IMPORTED_MODULE_0__["bytesToSize"]; // Pass bytesToSize utility function to template
597
+ return kendo.template(`
598
+ <div>
599
+ <span class="text-nowrap">#: kendo.toString(new Date(value.x), "dd/MM/yy HH:mm") #</span>
600
+ <br>
601
+ <strong class="text-nowrap">#: series.name #: #: bytesToSize(value.y, 1, '', '/s') # </strong>
602
+ </div>
603
+ `)(dataItem);
604
+ };
605
+ }
453
606
 
454
- // Initialize scrolling tabs
455
- $('#lgv-tabNavigator > #tabs').scrollingTabs({
456
- bootstrapVersion: 4,
457
- scrollToTabEdge: true,
458
- enableSwiping: true,
459
- disableScrollArrowsOnFullyScrolled: true,
460
- cssClassLeftArrow: 'fa fa-chevron-left',
461
- cssClassRightArrow: 'fa fa-chevron-right'
462
- });
463
607
 
464
- // Add listener to tab shown event
465
- $('a[data-toggle="tab"]').on('shown.bs.tab', $.proxy(this._onTabShown, this));
608
+ /***/ }),
466
609
 
467
- // Initialize log lines dropdown
468
- this._logLinesDD = $('#lgv-logLinesDD').kendoDropDownList({
469
- valueTemplate: '<span class="text-muted pr-1">Log entries:</span><span>#:data.text#</span>',
470
- }).data('kendoDropDownList');
610
+ /***/ "./src/components/charts/cpu-usage-chart.js":
611
+ /*!**************************************************!*\
612
+ !*** ./src/components/charts/cpu-usage-chart.js ***!
613
+ \**************************************************/
614
+ /*! exports provided: CpuUsageChart */
615
+ /***/ (function(module, __webpack_exports__, __webpack_require__) {
471
616
 
472
- // Initialize load button
473
- $('#lgv-loadBt').on('click', $.proxy(this._onRuntimeLogLoadBtClick, this));
474
-
475
- // Initialize progress bars
476
- $('.progress-bar').kendoProgressBar({
477
- min: 0,
478
- max: 100,
479
- value: false,
480
- type: 'value',
481
- animation: {
482
- duration: 400
483
- }
484
- });
485
-
486
- // Initialize level filter dropdown
487
- this._levelFilterDD = $('#lgv-levelDD').kendoDropDownList({
488
- dataSource: [this.ANY_LEVEL].concat(this.LEVELS),
489
- change: $.proxy(this._onFilterChange, this)
490
- }).data('kendoDropDownList');
617
+ "use strict";
618
+ __webpack_require__.r(__webpack_exports__);
619
+ /* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "CpuUsageChart", function() { return CpuUsageChart; });
620
+ /* harmony import */ var _base_chart__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ./base-chart */ "./src/components/charts/base-chart.js");
621
+ /* harmony import */ var _chart_utils__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ./chart-utils */ "./src/components/charts/chart-utils.js");
491
622
 
492
- // Initialize class filter dropdown
493
- this._classFilterDD = $('#lgv-classDD').kendoDropDownList({
494
- change: $.proxy(this._onFilterChange, this)
495
- }).data('kendoDropDownList');
496
623
 
497
- // Initialize message filter input
498
- $('#lgv-messageIn').on('input', $.proxy(this._onFilterChange, this));
499
624
 
500
- // Initialize clear button
501
- $('#lgv-clearFilterBt').on('click', $.proxy(this._onClearFilterClick, this));
502
625
 
503
- // Initialize export button
504
- $('#lgv-exportRuntimeLogBt').on('click', $.proxy(this._onExportRuntimeLogBtClick, this));
626
+ class CpuUsageChart extends _base_chart__WEBPACK_IMPORTED_MODULE_0__["BaseChart"]
627
+ {
628
+ constructor()
629
+ {
630
+ super(120);
631
+ }
505
632
 
506
- // Initialize runtime log grid
507
- this._runtimeLogGrid = $('#lgv-runtimeLogGrid').kendoGrid({
508
- scrollable: true,
509
- sortable: false,
510
- //resizable: true,
511
- selectable: false,
512
- columns:
513
- [
633
+ _initChartWidget(chartHtml)
634
+ {
635
+ return chartHtml.kendoChart({
636
+ transitions: false,
637
+ chartArea: {
638
+ height: 120
639
+ },
640
+ legend: {
641
+ visible: false
642
+ },
643
+ seriesDefaults: {
644
+ type: 'area',
645
+ labels: {
646
+ visible: false,
647
+ format: '{0}%',
648
+ background: 'transparent'
649
+ }
650
+ },
651
+ series: [
514
652
  {
515
- field: 'dateTime',
516
- width: 150,
517
- title: 'Date/Time',
518
- },
519
- {
520
- field: 'level',
521
- width: 100,
522
- title: 'Level',
523
- },
524
- {
525
- field: 'thread',
526
- width: 150,
527
- title: 'Thread',
528
- },
529
- {
530
- field: 'clazz',
531
- width: 250,
532
- title: 'Class',
533
- },
653
+ name: 'System',
654
+ field: 'systemLoad',
655
+ categoryField: 'time',
656
+ color: this.systemColor,
657
+ line: {
658
+ width: 2
659
+ }
660
+ },
534
661
  {
535
- field: 'message',
536
- width: 1000,
537
- title: 'Message',
538
- },
662
+ name: 'Process',
663
+ field: 'processLoad',
664
+ categoryField: 'time',
665
+ color: this.processColor,
666
+ line: {
667
+ width: 2
668
+ }
669
+ }
539
670
  ],
540
- noRecords: {
541
- template: 'No log entries to display.'
542
- },
543
- dataSource: []
544
- }).data('kendoGrid');
545
-
546
- // Initialize boot log view buttons
547
- $('#lgv-exportBootLogBt').on('click', $.proxy(this._onExportBootLogBtClick, this));
548
- $('#lgv-switchBootLogColorBt').on('click', $.proxy(this._onSwitchBootLogColorBtClick, this));
549
-
550
- // Initialize generate backup buttons
551
- $('#lgv-bootLogBackupCard .backup-button').on('click', $.proxy(this._onBootLogGenerateBtClick, this));
552
- $('#lgv-runtimeLogBackupCard .backup-button').on('click', $.proxy(this._onRuntimeLogGenerateBtClick, this));
553
- $('#lgv-fullLogBackupCard .backup-button').on('click', $.proxy(this._onFullLogsGenerateBtClick, this));
554
-
555
- this._initBackupCard('#lgv-bootLogBackupCard');
556
- this._initBackupCard('#lgv-runtimeLogBackupCard');
557
- this._initBackupCard('#lgv-fullLogBackupCard');
558
-
559
- // Initialize download grid
560
- this._downloadGrid = $('#lgv-downloadGrid').kendoGrid({
561
- scrollable: true,
562
- sortable: true,
563
- //resizable: true,
564
- selectable: 'row',
565
- columns:
566
- [
567
- {
568
- field: 'date',
569
- width: 100,
570
- title: 'Date',
571
- },
572
- {
573
- field: 'time',
574
- width: 100,
575
- title: 'Time',
671
+ valueAxis: {
672
+ labels: {
673
+ format: '{0}%'
576
674
  },
577
- {
578
- field: 'name',
579
- width: 300,
580
- title: 'Filename',
675
+ line: {
676
+ visible: false
581
677
  },
582
- {
583
- field: 'size',
584
- width: 100,
585
- title: 'Size',
678
+ min: 0,
679
+ max: 100
680
+ },
681
+ categoryAxis: {
682
+ majorGridLines: {
683
+ visible: false
586
684
  },
587
- ],
588
- noRecords: {
589
- template: 'No backups available.'
590
- },
591
- change: $.proxy(this._onDownloadGridSelectionChange, this),
592
- dataSource: []
593
- }).data('kendoGrid');
685
+ majorTicks: {
686
+ visible: false
687
+ },
688
+ labels: {
689
+ visible: false
690
+ }
691
+ },
692
+ tooltip: {
693
+ shared: true,
694
+ visible: true,
695
+ sharedTemplate: Object(_chart_utils__WEBPACK_IMPORTED_MODULE_1__["getBasicSharedTemplate"])('{0}%'),
696
+ background: '#e4e4e4'
697
+ }
698
+ }).data('kendoChart');
699
+ }
594
700
 
595
- // Initialize delete button
596
- $('#lgv-deleteBt').on('click', $.proxy(this._onDeleteBtClick, this));
701
+ get systemColor()
702
+ {
703
+ return '#3399FF';
597
704
  }
598
705
 
599
- destroy()
706
+ get processColor()
600
707
  {
601
- // Call super method
602
- super.destroy();
708
+ return '#0033CC';
709
+ }
603
710
 
604
- // Remove listener to tab shown event
605
- $('a[data-toggle="tab"]').off('shown.bs.tab');
711
+ addEntry(processLoad, systemLoad)
712
+ {
713
+ let newEntry = {
714
+ time: Object(_chart_utils__WEBPACK_IMPORTED_MODULE_1__["getFormattedDateTime"])(new Date()),
715
+ processLoad: processLoad,
716
+ systemLoad: systemLoad
717
+ };
718
+
719
+ // Add entry to chart
720
+ super._addEntry(newEntry);
721
+ }
722
+ }
606
723
 
607
- // Destroy scrolling tabs
608
- $('#lgv-tabNavigator #tabs').scrollingTabs('destroy');
724
+ // DEFINE COMPONENT
725
+ if (!window.customElements.get('cpu-usage-chart'))
726
+ window.customElements.define('cpu-usage-chart', CpuUsageChart);
609
727
 
610
- // Remove click listeners
611
- $('#lgv-loadBt').off('click');
612
- $('#lgv-exportBootLogBt').off('click');
613
- $('#lgv-switchBootLogColorBt').off('click');
614
- $('#lgv-bootLogBackupCard .backup-button').off('click');
615
- $('#lgv-runtimeLogBackupCard .backup-button').off('click');
616
- $('#lgv-clearFilterBt').off('click');
617
- $('#lgv-deleteBt').off('click');
618
728
 
619
- $('#lgv-messageIn').off('input');
620
- }
729
+ /***/ }),
621
730
 
622
- onExtensionCommand(command, data)
623
- {
624
- // Error during initialization (unable to access log4j configuration file)
625
- if (command == this.RESP_INIT_ERROR)
626
- {
627
- const error = data.getUtfString('error');
731
+ /***/ "./src/components/charts/memory-chart.js":
732
+ /*!***********************************************!*\
733
+ !*** ./src/components/charts/memory-chart.js ***!
734
+ \***********************************************/
735
+ /*! exports provided: MemoryChart */
736
+ /***/ (function(module, __webpack_exports__, __webpack_require__) {
628
737
 
629
- // Show an alert
630
- this.shellCtrl.showSimpleAlert(error, true);
738
+ "use strict";
739
+ __webpack_require__.r(__webpack_exports__);
740
+ /* WEBPACK VAR INJECTION */(function($) {/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "MemoryChart", function() { return MemoryChart; });
741
+ /* harmony import */ var _base_chart__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ./base-chart */ "./src/components/charts/base-chart.js");
742
+ /* harmony import */ var _chart_utils__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ./chart-utils */ "./src/components/charts/chart-utils.js");
631
743
 
632
- // Set all tabs to show errors
633
- this._switchBootViewStack('lgv-bootLogErrorView');
634
- this._switchRuntimeViewStack('lgv-runtimeLogErrorView');
635
- this._switchDownloadViewStack('lgv-downloadErrorView');
636
744
 
637
- // Disable runtime log controls
638
- this._enableRuntimeLogControls(false);
639
745
 
640
- this._initFailed = true;
641
- }
642
746
 
643
- // Error responses
644
- else if (command == this.RESP_BOOT_LOG_ERROR || command == this.RESP_RUNTIME_LOG_ERROR)
645
- {
646
- const error = data.getUtfString('error');
747
+ class MemoryChart extends _base_chart__WEBPACK_IMPORTED_MODULE_0__["BaseChart"]
748
+ {
749
+ constructor()
750
+ {
751
+ super(300);
647
752
 
648
- // Show an alert
649
- this.shellCtrl.showSimpleAlert(error, true);
753
+ // Append text to display elapsed time
754
+ this._elapsedTxt = $('<div>', {class: 'text-center', text: '---'});
755
+ $(this).append(this._elapsedTxt);
756
+ }
650
757
 
651
- if (command == this.RESP_BOOT_LOG_ERROR)
758
+ _initChartWidget(chartHtml)
759
+ {
760
+ return chartHtml.kendoChart({
761
+ transitions: false,
762
+ chartArea: {
763
+ height: 240
764
+ },
765
+ legend: {
766
+ visible: false
767
+ },
768
+ seriesDefaults: {
769
+ type: 'area',
770
+ labels: {
771
+ visible: false,
772
+ format: '{0}',
773
+ background: 'transparent'
774
+ },
775
+ },
776
+ series: [{
777
+ name: 'Allocated',
778
+ field: 'allocMem',
779
+ categoryField: 'time',
780
+ color: this.allocatedColor,
781
+ line: {
782
+ width: 2
783
+ }
784
+ },
652
785
  {
653
- // Set tab to show error
654
- this._switchBootViewStack('lgv-bootLogErrorView');
655
-
656
- // Disable boot log backup generation
657
- this._bootLogBackupUnavailable = true;
658
- this._disableBackupInterface(false);
786
+ name: 'Used',
787
+ field: 'usedMem',
788
+ categoryField: 'time',
789
+ color: this.usedColor,
790
+ line: {
791
+ width: 2
792
+ }
793
+ }],
794
+ valueAxis: {
795
+ labels: {
796
+ format: '{0}',
797
+ template: Object(_chart_utils__WEBPACK_IMPORTED_MODULE_1__["getMemoryValueAxisLabelTemplate"])()
798
+ },
799
+ line: {
800
+ visible: false
801
+ },
802
+ min: 0
803
+ },
804
+ categoryAxis: {
805
+ majorGridLines: {
806
+ visible: false
807
+ },
808
+ majorTicks: {
809
+ visible: false
810
+ },
811
+ labels: {
812
+ visible: false
813
+ }
814
+ },
815
+ tooltip: {
816
+ shared: true,
817
+ visible: true,
818
+ sharedTemplate: Object(_chart_utils__WEBPACK_IMPORTED_MODULE_1__["getMemorySharedTemplate"])(),
819
+ background: '#e4e4e4'
820
+ }
821
+ }).data('kendoChart');
822
+ }
659
823
 
660
- this._bootLogRequested = true;
661
- }
824
+ get allocatedColor()
825
+ {
826
+ return '#3399FF';
827
+ }
662
828
 
663
- if (command == this.RESP_RUNTIME_LOG_ERROR)
664
- {
665
- // Disable controls
666
- this._enableRuntimeLogControls(false);
829
+ get usedColor()
830
+ {
831
+ return '#0033CC';
832
+ }
667
833
 
668
- // Remove loading bar (runtime log)
669
- this._switchRuntimeViewStack('lgv-runtimeLogErrorView');
834
+ addEntry(allocMem, usedMem)
835
+ {
836
+ let now = new Date();
670
837
 
671
- // Disable runtime log backup download
672
- this._runtimeLogBackupUnavailable = true;
673
- this._disableBackupInterface(false);
674
- }
675
- }
838
+ let newEntry = {
839
+ time: Object(_chart_utils__WEBPACK_IMPORTED_MODULE_1__["getFormattedDateTime"])(now),
840
+ timestamp: now.getTime(),
841
+ allocMem: allocMem,
842
+ usedMem: usedMem
843
+ };
676
844
 
677
- // Modified conversion pattern in the log4j properties file: unable to parse the log
678
- else if (command == this.RESP_RUNTIME_LOG_INVALID)
679
- {
680
- // Disable controls
681
- this._enableRuntimeLogControls(false);
845
+ // Add entry to chart
846
+ super._addEntry(newEntry);
682
847
 
683
- // Fill in error message
684
- $('#lgv-convPattName').text(data.getUtfString('param'));
685
- $('#lgv-convPattVal').text(data.getUtfString('value'));
848
+ // Update x axis
849
+ if (!this.isPaused)
850
+ this._updateElapsedText();
851
+ }
686
852
 
687
- // Remove loading bar (runtime log)
688
- this._switchRuntimeViewStack('lgv-invConvPattView');
689
- }
853
+ _updateElapsedText()
854
+ {
855
+ let ds = this._dataSource;
690
856
 
691
- // Runtime log received
692
- else if (command == this.RESP_RUNTIME_LOG)
857
+ if (ds.total() > 1)
693
858
  {
694
- let classes = [];
695
- classes.push(this.ANY_CLASS);
696
-
697
- let logEntries = data.getUtfStringArray(this.LOG_DATA_LABEL);
698
- let separator = data.getUtfString('sep');
699
- let columns = data.getInt('cols');
700
- let dsArr = [];
859
+ // Get axis limits
860
+ const chartMinimum = ds.at(0).timestamp;
861
+ const chartMaximum = ds.at(ds.total() - 1).timestamp;
701
862
 
702
- this._totalRuntimeLogEntries = logEntries.length;
863
+ // Set elapsed text
864
+ let seconds = Math.round( (chartMaximum - chartMinimum) / 1000 );
865
+ let title;
703
866
 
704
- // Parse log entries
705
- // We can't use the split method because there could be instances of the separator in the log message too
706
- for (let e = 0; e < logEntries.length; e++)
867
+ if (seconds < 60)
868
+ title = 'Last ' + (seconds != 1 ? seconds + ' seconds' : 'second');
869
+ else
707
870
  {
708
- const logEntry = logEntries[e];
709
-
710
- let logEntryData = [];
711
- let startIndex = 0;
712
-
713
- for (let c = 0; c < columns - 1; c++)
714
- {
715
- let endIndex = logEntry.indexOf(separator, startIndex);
716
- logEntryData.push(logEntry.substring(startIndex, endIndex));
717
- startIndex = endIndex + separator.length;
718
- }
719
-
720
- if (startIndex < logEntry.length)
721
- logEntryData.push(logEntry.substring(startIndex));
722
-
723
- // Fill datagrid's dataprovider
724
- let rle = _data_runtime_log_entry__WEBPACK_IMPORTED_MODULE_1__["RuntimeLogEntry"].fromArray(separator, logEntryData);
725
- dsArr.push(rle);
726
-
727
- // Add class to filtering dropdown
728
- if (classes.indexOf(rle.clazz) < 0)
729
- classes.push(rle.clazz);
871
+ const minutes = Math.floor(seconds / 60);
872
+ title = 'Last ' + (minutes != 1 ? minutes + ' minutes' : 'minute');
730
873
  }
731
874
 
732
- // Show classes list
733
- classes.sort(function (a, b) {
734
- return a.localeCompare(b);
735
- });
736
-
737
- this._classFilterDD.setDataSource(classes);
738
- this._classFilterDD.select(0);
875
+ this._elapsedTxt.text(title);
876
+ }
877
+ }
878
+ }
739
879
 
740
- // Assign data source to grid
741
- let ds = new kendo.data.DataSource({
742
- data: dsArr
743
- })
880
+ // DEFINE COMPONENT
881
+ if (!window.customElements.get('memory-chart'))
882
+ window.customElements.define('memory-chart', MemoryChart);
744
883
 
745
- this._setRuntimeLogDataSource(ds);
884
+ /* WEBPACK VAR INJECTION */}.call(this, __webpack_require__(/*! jquery */ "jquery")))
746
885
 
747
- // Re-enable log loading controls
748
- this._enableRuntimeLogControls(true);
886
+ /***/ }),
749
887
 
750
- // Remove loading bar
751
- this._switchRuntimeViewStack('lgv-runtimeLogView');
752
- }
888
+ /***/ "./src/components/charts/network-24h-chart.js":
889
+ /*!****************************************************!*\
890
+ !*** ./src/components/charts/network-24h-chart.js ***!
891
+ \****************************************************/
892
+ /*! exports provided: Network24hChart */
893
+ /***/ (function(module, __webpack_exports__, __webpack_require__) {
753
894
 
754
- // Boot log received
755
- else if (command == this.RESP_BOOT_LOG)
756
- {
757
- const bootLogEntries = data.getSFSArray(this.LOG_DATA_LABEL);
758
- let text = '';
895
+ "use strict";
896
+ __webpack_require__.r(__webpack_exports__);
897
+ /* WEBPACK VAR INJECTION */(function($) {/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "Network24hChart", function() { return Network24hChart; });
898
+ /* harmony import */ var _chart_utils__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ./chart-utils */ "./src/components/charts/chart-utils.js");
899
+ /* harmony import */ var moment__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! moment */ "./node_modules/moment/moment.js");
900
+ /* harmony import */ var moment__WEBPACK_IMPORTED_MODULE_1___default = /*#__PURE__*/__webpack_require__.n(moment__WEBPACK_IMPORTED_MODULE_1__);
759
901
 
760
- for (let i = 0; i < bootLogEntries.size(); i++)
761
- text += bootLogEntries.getUtfString(i) + '\n';
762
902
 
763
- $('#lgv-bootLogText').text(text);
764
903
 
765
- // Remove loading bar
766
- this._switchBootViewStack('lgv-bootLogView');
767
- }
904
+ class Network24hChart extends HTMLElement
905
+ {
906
+ constructor()
907
+ {
908
+ super();
768
909
 
769
- // Logs backups status received
770
- else if (command == this.RESP_BACKUPS_STATUS)
771
- {
772
- // Show/hide operation in progress message
773
- this._disableBackupInterface(data.getBool('running'), data.getUtfString('type'));
910
+ this._lastTotalOut = 0;
911
+ this._lastTotalIn = 0;
912
+ this._lastDate = new Date();
913
+ this._samplingRateSeconds = 0;
774
914
 
775
- // Backup files list
776
- if (data.containsKey('files'))
777
- {
778
- let files = data.getSFSArray('files');
915
+ // Create chart html
916
+ let chartHtml = $('<div>');
917
+ $(this).append(chartHtml);
779
918
 
780
- let lastBootLogBackupFound = false;
781
- let lastRuntimeLogBackupFound = false;
782
- let lastFullBackupFound = false;
919
+ // Initialize chart
920
+ this._chartWidget = this._initChartWidget(chartHtml);
921
+ }
783
922
 
784
- let backupsList = [];
923
+ _initChartWidget(chartHtml)
924
+ {
925
+ return chartHtml.kendoChart({
926
+ transitions: false,
927
+ chartArea: {
928
+ height: 270,
929
+ background: 'transparent'
930
+ },
931
+ legend: {
932
+ visible: false
933
+ },
934
+ seriesDefaults: {
935
+ type: 'area',
936
+ labels: {
937
+ visible: false,
938
+ format: '{0}',
939
+ background: 'transparent'
940
+ },
941
+ },
942
+ series: [{
943
+ name: 'Outgoing',
944
+ field: 'outgoing',
945
+ categoryField: 'time',
946
+ color: '#CC0000',
947
+ line: {
948
+ width: 2
949
+ }
950
+ },
951
+ {
952
+ name: 'Incoming',
953
+ field: 'incoming',
954
+ categoryField: 'time',
955
+ color: '#009900',
956
+ line: {
957
+ width: 2
958
+ }
959
+ }],
960
+ valueAxis: {
961
+ labels: {
962
+ format: '{0}',
963
+ template: Object(_chart_utils__WEBPACK_IMPORTED_MODULE_0__["getNetworkValueAxisLabelTemplate"])()
964
+ },
965
+ line: {
966
+ visible: false
967
+ },
968
+ min: 0
969
+ },
970
+ categoryAxis: {
971
+ majorGridLines: {
972
+ visible: true,
973
+ step: 60
974
+ },
975
+ majorTicks: {
976
+ visible: true,
977
+ step: 60
978
+ },
979
+ labels: {
980
+ visible: true,
981
+ step: 60
982
+ },
983
+ baseUnit: 'minutes',
984
+ type: 'date',
985
+ },
986
+ tooltip: {
987
+ shared: true,
988
+ visible: true,
989
+ sharedTemplate: Object(_chart_utils__WEBPACK_IMPORTED_MODULE_0__["getNetworkSharedTemplate"])('dd MMM HH:mm'),
990
+ background: '#e4e4e4'
991
+ }
992
+ }).data('kendoChart');
993
+ }
785
994
 
786
- const webServerProtocol = (data.containsKey('protocol') ? data.getUtfString('protocol') : 'http') + '://';
787
- const webServerPort = (data.containsKey('port') ? ':' + data.getInt('port') : '');
995
+ redraw()
996
+ {
997
+ this._chartWidget.redraw();
998
+ }
788
999
 
789
- let totalSize = 0;
1000
+ set range(values)
1001
+ {
1002
+ this._range = values;
790
1003
 
791
- for (let f = 0; f < files.size(); f++)
792
- {
793
- const file = files.getSFSObject(f);
1004
+ this._updateXAxisLimits();
1005
+ }
794
1006
 
795
- const filePath = file.getUtfString('path');
1007
+ get range()
1008
+ {
1009
+ return this._range;
1010
+ }
796
1011
 
797
- const fileObj = {};
798
- fileObj.path = filePath;
799
- fileObj.url = webServerProtocol + this.smartFox.config.host + webServerPort + '/' + filePath;
800
- fileObj.name = filePath.substr(filePath.lastIndexOf('/') + 1);
801
- fileObj.date = file.getUtfString('date');
802
- fileObj.time = file.getUtfString('time');
803
- fileObj.size = Object(_utils_utilities__WEBPACK_IMPORTED_MODULE_4__["bytesToSize"])(file.getLong('size'), 2);
1012
+ addHistoryEntries(outgoingData, incomingData, samplingRateSeconds, shiftSeconds)
1013
+ {
1014
+ this._samplingRateSeconds = samplingRateSeconds;
804
1015
 
805
- totalSize += file.getLong('size');
1016
+ let now = new Date();
806
1017
 
807
- // Check if this is a boot log backup
808
- if (!lastBootLogBackupFound)
809
- {
810
- if (fileObj.name.startsWith(this.BOOT_LOG_BACKUP_ID))
811
- {
812
- this._fillBackupCard('#lgv-bootLogBackupCard', fileObj);
1018
+ // NOTE: outgoingData and incominData are supposed to have the same size!!!
813
1019
 
814
- lastBootLogBackupFound = true;
815
- }
816
- }
1020
+ if (outgoingData.length > 0)
1021
+ {
1022
+ let start = moment__WEBPACK_IMPORTED_MODULE_1__(now);
1023
+ start.subtract((samplingRateSeconds * outgoingData.length) + shiftSeconds, 's')
817
1024
 
818
- // Check if this is a runtime log backup
819
- if (!lastRuntimeLogBackupFound)
820
- {
821
- if (fileObj.name.startsWith(this.RUNTIME_LOG_BACKUP_ID))
822
- {
823
- this._fillBackupCard('#lgv-runtimeLogBackupCard', fileObj);
1025
+ this._lastTotalOut = outgoingData[0];
1026
+ this._lastTotalIn = incomingData[0];
1027
+ this._lastDate = start.toDate();
824
1028
 
825
- lastRuntimeLogBackupFound = true;
826
- }
827
- }
1029
+ if (outgoingData.length > 1)
1030
+ {
1031
+ let dataSource = new kendo.data.DataSource();
828
1032
 
829
- // Check if this is a full backup
830
- if (!lastFullBackupFound)
831
- {
832
- if (fileObj.name.startsWith(this.FULL_BACKUP_ID))
833
- {
834
- this._fillBackupCard('#lgv-fullLogBackupCard', fileObj);
1033
+ for (let i = 1; i < outgoingData.length; i++)
1034
+ {
1035
+ const totalOut = outgoingData[i];
1036
+ const totalIn = incomingData[i];
835
1037
 
836
- lastFullBackupFound = true;
837
- }
838
- }
1038
+ let date = start.clone();
1039
+ date.add(i * samplingRateSeconds, 's');
839
1040
 
840
- // Populate logs list
841
- backupsList.push(fileObj);
1041
+ this.addEntry(totalOut, totalIn, dataSource, date.toDate(), false);
842
1042
  }
843
1043
 
844
- // Show total backups size
845
- $('#lgv-logSizeLb').html(`Total size: <strong>${Object(_utils_utilities__WEBPACK_IMPORTED_MODULE_4__["bytesToSize"])(totalSize, 2, 'KB')}</strong>`);
1044
+ // Assign datasource to chart
1045
+ this._chartWidget.setDataSource(dataSource);
846
1046
 
847
- // Assign data source to grid
848
- this._setDownloadGridDataSource(backupsList);
849
- this._onDownloadGridSelectionChange();
1047
+ // Update axis
1048
+ this._updateXAxisLimits();
1049
+ }
1050
+ }
1051
+ else
1052
+ {
1053
+ this._lastTotalOut = 0;
1054
+ this._lastTotalIn = 0;
1055
+ this._lastDate = now;
1056
+ }
1057
+ }
850
1058
 
851
- // Hide links to latest files if not available
852
- if (!lastBootLogBackupFound)
853
- this._fillBackupCard('#lgv-bootLogBackupCard', null);
1059
+ addEntry(totalOut, totalIn, dataSource = null, date = null, updateAxis = true)
1060
+ {
1061
+ if (date == null)
1062
+ date = new Date();
854
1063
 
855
- if (!lastRuntimeLogBackupFound)
856
- this._fillBackupCard('#lgv-runtimeLogBackupCard', null);
1064
+ const elapsedSecs = Math.round((date.getTime() - this._lastDate.getTime()) / 1000);
857
1065
 
858
- if (!lastFullBackupFound)
859
- this._fillBackupCard('#lgv-fullLogBackupCard', null);
1066
+ // The 24h chart is updated at the same sampling rate used by the server to save the traffic history
1067
+ if (elapsedSecs >= this._samplingRateSeconds)
1068
+ {
1069
+ let newEntry = {
1070
+ time: date,
1071
+ outgoing: Math.round((totalOut - this._lastTotalOut) / elapsedSecs),
1072
+ incoming: Math.round((totalIn - this._lastTotalIn) / elapsedSecs)
1073
+ };
860
1074
 
861
- if (data.containsKey('message'))
862
- {
863
- // Display notification
864
- this.shellCtrl.showNotification(`Log backup warning`, data.getUtfString('message'));
865
- }
866
- }
1075
+ // Add entry to data source
1076
+ if (dataSource == null)
1077
+ dataSource = this._dataSource;
867
1078
 
868
- // Set download view to main
869
- this._switchDownloadViewStack('lgv-downloadView');
870
- }
1079
+ dataSource.add(newEntry);
871
1080
 
872
- // Logs backup deletion failed
873
- else if (command == this.RESP_DELETE_BACKUP_FAILED)
874
- {
875
- const error = data.getUtfString('error');
1081
+ this._lastTotalOut = totalOut;
1082
+ this._lastTotalIn = totalIn;
1083
+ this._lastDate = date;
876
1084
 
877
- // Show an alert
878
- this.shellCtrl.showSimpleAlert(error, true);
1085
+ if (updateAxis)
1086
+ this._updateXAxisLimits();
879
1087
  }
1088
+ }
880
1089
 
881
- // A blocking error occurred during backup operation
882
- else if (command == this.RESP_BACKUP_ERROR)
883
- {
884
- const error = data.getUtfString('error');
885
-
886
- // Show an alert
887
- this.shellCtrl.showSimpleAlert(error, true);
888
- }
1090
+ _updateXAxisLimits()
1091
+ {
1092
+ let ds = this._dataSource;
889
1093
 
890
- // An non-blocking error occurred during backup operation
891
- else if (command == this.RESP_BACKUP_WARNING)
1094
+ if (ds.total() > 0)
892
1095
  {
893
- let warn = data.getUtfString('warn');
1096
+ const chartMinimum = moment__WEBPACK_IMPORTED_MODULE_1__(this._lastDate).add(this._range[0], 'h');
1097
+ const chartMaximum = moment__WEBPACK_IMPORTED_MODULE_1__(this._lastDate).add(this._range[1], 'h');
1098
+
1099
+ this._chartWidget.options.categoryAxis.min = chartMinimum.toDate();
1100
+ this._chartWidget.options.categoryAxis.max = chartMaximum.toDate();
894
1101
 
895
- // Display notification
896
- this.shellCtrl.showNotification(`Log backup warning`, warn);
1102
+ this._chartWidget.redraw();
897
1103
  }
898
1104
  }
899
1105
 
900
- //---------------------------------
901
- // UI EVENT LISTENERS
902
- //---------------------------------
1106
+ get _dataSource()
1107
+ {
1108
+ return this._chartWidget.dataSource;
1109
+ }
1110
+ }
1111
+
1112
+ // DEFINE COMPONENT
1113
+ if (!window.customElements.get('network-24h-chart'))
1114
+ window.customElements.define('network-24h-chart', Network24hChart);
1115
+
1116
+ /* WEBPACK VAR INJECTION */}.call(this, __webpack_require__(/*! jquery */ "jquery")))
1117
+
1118
+ /***/ }),
1119
+
1120
+ /***/ "./src/components/charts/network-realtime-chart.js":
1121
+ /*!*********************************************************!*\
1122
+ !*** ./src/components/charts/network-realtime-chart.js ***!
1123
+ \*********************************************************/
1124
+ /*! exports provided: NetworkRealtimeChart */
1125
+ /***/ (function(module, __webpack_exports__, __webpack_require__) {
903
1126
 
904
- _onTabShown(e)
1127
+ "use strict";
1128
+ __webpack_require__.r(__webpack_exports__);
1129
+ /* WEBPACK VAR INJECTION */(function($) {/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "NetworkRealtimeChart", function() { return NetworkRealtimeChart; });
1130
+ /* harmony import */ var _base_chart__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ./base-chart */ "./src/components/charts/base-chart.js");
1131
+ /* harmony import */ var _chart_utils__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ./chart-utils */ "./src/components/charts/chart-utils.js");
1132
+
1133
+
1134
+
1135
+
1136
+ class NetworkRealtimeChart extends _base_chart__WEBPACK_IMPORTED_MODULE_0__["BaseChart"]
1137
+ {
1138
+ constructor()
905
1139
  {
906
- if (!this._initFailed)
907
- {
908
- // If boot log view was displayed...
909
- if (e.target.id == 'lgv-bootLog-tab')
1140
+ super(300);
1141
+
1142
+ this._lastTotalKbOut = 0;
1143
+ this._lastTotalKbIn = 0;
1144
+ this._lastTime = 0;
1145
+
1146
+ // Append text to display elapsed time
1147
+ this._elapsedTxt = $('<div>', {class: 'text-center', text: '---'});
1148
+ $(this).append(this._elapsedTxt);
1149
+ }
1150
+
1151
+ _initChartWidget(chartHtml)
1152
+ {
1153
+ return chartHtml.kendoChart({
1154
+ transitions: false,
1155
+ chartArea: {
1156
+ height: 270,
1157
+ background: 'transparent'
1158
+ },
1159
+ legend: {
1160
+ visible: false
1161
+ },
1162
+ seriesDefaults: {
1163
+ type: 'area',
1164
+ labels: {
1165
+ visible: false,
1166
+ format: '{0}',
1167
+ background: 'transparent'
1168
+ },
1169
+ },
1170
+ series: [{
1171
+ name: 'Outgoing',
1172
+ field: 'outgoing',
1173
+ categoryField: 'time',
1174
+ color: this.outgoingColor,
1175
+ line: {
1176
+ width: 2
1177
+ }
1178
+ },
910
1179
  {
911
- // Load boot log the first time the tab is selected
912
- if (!this._bootLogRequested)
913
- {
914
- this.sendExtensionRequest(this.REQ_GET_BOOT_LOG);
915
- this._bootLogRequested = true;
1180
+ name: 'Incoming',
1181
+ field: 'incoming',
1182
+ categoryField: 'time',
1183
+ color: this.incomingColor,
1184
+ line: {
1185
+ width: 2
916
1186
  }
917
- }
1187
+ }],
1188
+ valueAxis: {
1189
+ labels: {
1190
+ format: '{0}',
1191
+ template: Object(_chart_utils__WEBPACK_IMPORTED_MODULE_1__["getNetworkValueAxisLabelTemplate"])()
1192
+ },
1193
+ line: {
1194
+ visible: false
1195
+ },
1196
+ min: 0,
1197
+ },
1198
+ categoryAxis: {
1199
+ majorGridLines: {
1200
+ visible: false
1201
+ },
1202
+ majorTicks: {
1203
+ visible: false
1204
+ },
1205
+ labels: {
1206
+ visible: false
1207
+ }
1208
+ },
1209
+ tooltip: {
1210
+ shared: true,
1211
+ visible: true,
1212
+ sharedTemplate: Object(_chart_utils__WEBPACK_IMPORTED_MODULE_1__["getNetworkSharedTemplate"])(),
1213
+ background: '#e4e4e4'
1214
+ }
1215
+ }).data('kendoChart');
1216
+ }
1217
+
1218
+ get outgoingColor()
1219
+ {
1220
+ return '#CC0000';
1221
+ }
1222
+
1223
+ get incomingColor()
1224
+ {
1225
+ return '#009900';
1226
+ }
1227
+
1228
+ addEntry(totalOutKb, totalInKb)
1229
+ {
1230
+ let now = new Date()
1231
+ let time = now.getTime();
1232
+
1233
+ // The first value sent by the server is not plotted: it would be a worng value because
1234
+ // we have to calculate it as the difference between two total values, but we don't have the previous one
1235
+ if (this._lastTime > 0)
1236
+ {
1237
+ let elapsedSecs = (time - this._lastTime) / 1000;
1238
+
1239
+ const valueOut = Math.round((totalOutKb - this._lastTotalKbOut) / elapsedSecs);
1240
+ const valueIn = Math.round((totalInKb - this._lastTotalKbIn) / elapsedSecs);
1241
+
1242
+ let newEntry = {
1243
+ time: Object(_chart_utils__WEBPACK_IMPORTED_MODULE_1__["getFormattedDateTime"])(now),
1244
+ timestamp: time,
1245
+ outgoing: valueOut,
1246
+ incoming: valueIn
1247
+ };
1248
+
1249
+ // Add entry to chart
1250
+ super._addEntry(newEntry);
1251
+
1252
+ // Update x axis
1253
+ if (!this.isPaused)
1254
+ this._updateElapsedText();
1255
+ }
918
1256
 
919
- // If backup&dowload view was displayed...
920
- else if (e.target.id == 'lgv-logsDownload-tab')
1257
+ this._lastTotalKbOut = totalOutKb;
1258
+ this._lastTotalKbIn = totalInKb;
1259
+ this._lastTime = time;
1260
+ }
1261
+
1262
+ _updateElapsedText()
1263
+ {
1264
+ let ds = this._dataSource;
1265
+
1266
+ if (ds.total() > 1)
1267
+ {
1268
+ // Get axis limits
1269
+ const chartMinimum = ds.at(0).timestamp;
1270
+ const chartMaximum = ds.at(ds.total() - 1).timestamp;
1271
+
1272
+ // Set elapsed text
1273
+ let title;
1274
+ let seconds = Math.round( (chartMaximum - chartMinimum) / 1000 );
1275
+
1276
+ if (seconds < 60)
1277
+ title = 'Last ' + (seconds != 1 ? seconds + ' seconds' : 'second');
1278
+ else
921
1279
  {
922
- // Request logs backup status the first time the tab is selected
923
- if (!this._backupStatusRequested)
924
- {
925
- this.sendExtensionRequest(this.REQ_GET_BACKUPS_STATUS);
926
- this._backupStatusRequested = true;
927
- }
1280
+ const minutes = Math.floor(seconds / 60);
1281
+ title = 'Last ' + (minutes != 1 ? minutes + ' minutes' : 'minute');
928
1282
  }
1283
+
1284
+ this._elapsedTxt.text(title);
929
1285
  }
930
1286
  }
1287
+ }
1288
+
1289
+ // DEFINE COMPONENT
1290
+ if (!window.customElements.get('network-realtime-chart'))
1291
+ window.customElements.define('network-realtime-chart', NetworkRealtimeChart);
1292
+
1293
+ /* WEBPACK VAR INJECTION */}.call(this, __webpack_require__(/*! jquery */ "jquery")))
1294
+
1295
+ /***/ }),
1296
+
1297
+ /***/ "./src/components/charts/system-queue-chart.js":
1298
+ /*!*****************************************************!*\
1299
+ !*** ./src/components/charts/system-queue-chart.js ***!
1300
+ \*****************************************************/
1301
+ /*! exports provided: SystemQueueChart */
1302
+ /***/ (function(module, __webpack_exports__, __webpack_require__) {
1303
+
1304
+ "use strict";
1305
+ __webpack_require__.r(__webpack_exports__);
1306
+ /* WEBPACK VAR INJECTION */(function($) {/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "SystemQueueChart", function() { return SystemQueueChart; });
1307
+ /* harmony import */ var _base_chart__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ./base-chart */ "./src/components/charts/base-chart.js");
1308
+ /* harmony import */ var _chart_utils__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ./chart-utils */ "./src/components/charts/chart-utils.js");
1309
+
1310
+
1311
+
1312
+
1313
+ class SystemQueueChart extends _base_chart__WEBPACK_IMPORTED_MODULE_0__["BaseChart"]
1314
+ {
1315
+ constructor()
1316
+ {
1317
+ super(240);
1318
+
1319
+ this._averageLoad = 0;
1320
+ }
931
1321
 
932
- _onRuntimeLogLoadBtClick()
1322
+ _initialize()
933
1323
  {
934
- // Request log
935
- this._loadRuntimeLog();
1324
+ const chartHeight = 240;
1325
+
1326
+ // Append container for chart & gauge
1327
+ let container = $('<div>', {class: 'd-flex flex-row'});
1328
+ $(this).append(container);
1329
+
1330
+ // CHART
1331
+
1332
+ let chartContainer = $('<div>', {class: 'flex-grow-1'});
1333
+ container.append(chartContainer);
1334
+
1335
+ // Create chart html
1336
+ let chartHtml = $('<div>');
1337
+ chartContainer.append(chartHtml);
1338
+
1339
+ // Initialize chart
1340
+ this._chartWidget = this._initChartWidget(chartHtml, chartHeight);
1341
+
1342
+ // Append text to display elapsed time
1343
+ this._elapsedTxt = $('<div>', {class: 'text-center', text: '---'});
1344
+ chartContainer.append(this._elapsedTxt);
1345
+
1346
+ // GAUGE
1347
+
1348
+ let gaugeContainer = $('<div>');
1349
+ container.append(gaugeContainer);
1350
+
1351
+ // Create gauge html
1352
+ let gaugeHtml = $('<div>');
1353
+ gaugeContainer.append(gaugeHtml);
1354
+
1355
+ // Initialize gauge
1356
+ this._gaugeWidget = this._initGaugeWidget(gaugeHtml, chartHeight);
1357
+
1358
+ // Append text to display load
1359
+ this._loadTxt = $('<div>', {class: 'text-center font-weight-bold', text: '---'});
1360
+ gaugeContainer.append(this._loadTxt);
1361
+
1362
+ //------------------------------
1363
+
1364
+ // Reset data source
1365
+ this.reset();
936
1366
  }
937
1367
 
938
- _onExportBootLogBtClick()
1368
+ _initChartWidget(chartHtml, height)
939
1369
  {
940
- // Export log to file
941
- this._exportLog($('#lgv-bootLogText').text(), this.BOOT_LOG_BACKUP_ID);
1370
+ return chartHtml.kendoChart({
1371
+ transitions: false,
1372
+ chartArea: {
1373
+ height: height,
1374
+ background: 'transparent'
1375
+ },
1376
+ legend: {
1377
+ visible: false
1378
+ },
1379
+ seriesDefaults: {
1380
+ type: 'area',
1381
+ labels: {
1382
+ visible: false,
1383
+ format: '{0}',
1384
+ background: 'transparent'
1385
+ }
1386
+ },
1387
+ series: [
1388
+ {
1389
+ name: 'Messages',
1390
+ field: 'load',
1391
+ categoryField: 'time',
1392
+ color: '#3399ff',
1393
+ line: {
1394
+ width: 2
1395
+ }
1396
+ }
1397
+ ],
1398
+ valueAxis: {
1399
+ title: {
1400
+ text: 'Messages'
1401
+ },
1402
+ labels: {
1403
+ format: '{0}'
1404
+ },
1405
+ line: {
1406
+ visible: false
1407
+ },
1408
+ min: 0
1409
+ },
1410
+ categoryAxis: {
1411
+ majorGridLines: {
1412
+ visible: false
1413
+ },
1414
+ majorTicks: {
1415
+ visible: false
1416
+ },
1417
+ labels: {
1418
+ visible: false
1419
+ }
1420
+ },
1421
+ tooltip: {
1422
+ shared: true,
1423
+ visible: true,
1424
+ sharedTemplate: Object(_chart_utils__WEBPACK_IMPORTED_MODULE_1__["getBasicSharedTemplate"])(),
1425
+ background: '#e4e4e4'
1426
+ }
1427
+ }).data('kendoChart');
942
1428
  }
943
1429
 
944
- _onSwitchBootLogColorBtClick()
1430
+ _initGaugeWidget(gaugeHtml, height)
945
1431
  {
946
- if ($('#lgv-bootLogText').hasClass('invert'))
947
- $('#lgv-bootLogText').removeClass('invert');
1432
+ return gaugeHtml.kendoLinearGauge({
1433
+ pointer: {
1434
+ value: 0,
1435
+ shape: 'arrow',
1436
+ size: 10,
1437
+ color: '#212529'
1438
+ },
1439
+ gaugeArea: {
1440
+ height: height
1441
+ },
1442
+ scale: {
1443
+ majorUnit: 10,
1444
+ minorUnit: 1,
1445
+ min: 0,
1446
+ max: 100,
1447
+ ranges: [
1448
+ {
1449
+ from: 0,
1450
+ to: 50,
1451
+ color: '#00cc00',
1452
+ opacity: .5
1453
+ }, {
1454
+ from: 50,
1455
+ to: 75,
1456
+ color: '#ffe106',
1457
+ opacity: .5
1458
+ }, {
1459
+ from: 75,
1460
+ to: 100,
1461
+ color: '#ed1601',
1462
+ opacity: .5
1463
+ }
1464
+ ],
1465
+ majorTicks: {
1466
+ visible: true,
1467
+ size: 10
1468
+ },
1469
+ minorTicks: {
1470
+ visible: false
1471
+ },
1472
+ labels: {
1473
+ visible: false
1474
+ },
1475
+ line: {
1476
+ visible: false
1477
+ },
1478
+ rangeSize: 10
1479
+ }
1480
+ }).data('kendoLinearGauge');
1481
+ }
1482
+
1483
+ addEntry(load, queueMsgLimit)
1484
+ {
1485
+ // CHART
1486
+
1487
+ let now = new Date();
1488
+
1489
+ let newEntry = {
1490
+ time: Object(_chart_utils__WEBPACK_IMPORTED_MODULE_1__["getFormattedDateTime"])(now),
1491
+ timestamp: now.getTime(),
1492
+ load: load
1493
+ };
1494
+
1495
+ // Add entry to chart
1496
+ super._addEntry(newEntry);
1497
+
1498
+ // Update x axis
1499
+ if (!this.isPaused)
1500
+ this._updateElapsedText();
1501
+
1502
+ // GAUGE
1503
+
1504
+ // Evaluate average
1505
+ let numValues = this._dataSource.total();
1506
+
1507
+ if (numValues > 0)
1508
+ {
1509
+ let total = 0;
1510
+ for (let i = 0; i < numValues; i++)
1511
+ total += this._dataSource.at(i).load;
1512
+
1513
+ this._averageLoad = Math.round(total / numValues);
1514
+ }
948
1515
  else
949
- $('#lgv-bootLogText').addClass('invert');
1516
+ this._averageLoad = 0;
1517
+
1518
+ const avgLoadPerc = Math.round(this._averageLoad / queueMsgLimit * 1000) / 10;
1519
+
1520
+ // Set gauge value
1521
+ this._gaugeWidget.value(avgLoadPerc);
1522
+
1523
+ // Set load text
1524
+ this._loadTxt.text(avgLoadPerc + '%')
950
1525
  }
951
1526
 
952
- _onBootLogGenerateBtClick()
1527
+ _updateElapsedText()
953
1528
  {
954
- // Show/hide operation in progress message
955
- this._disableBackupInterface(true, this.BOOT_LOG_BACKUP_ID);
1529
+ let ds = this._dataSource;
1530
+
1531
+ if (ds.total() > 1)
1532
+ {
1533
+ // Get axis limits
1534
+ const chartMinimum = ds.at(0).timestamp;
1535
+ const chartMaximum = ds.at(ds.total() - 1).timestamp;
1536
+
1537
+ // Set elapsed text
1538
+ let seconds = Math.round( (chartMaximum - chartMinimum) / 1000 );
1539
+ let title;
1540
+
1541
+ if (seconds < 60)
1542
+ title = 'Last ' + (seconds != 1 ? seconds + ' seconds' : 'second');
1543
+ else
1544
+ {
1545
+ const minutes = Math.floor(seconds / 60);
1546
+ title = 'Last ' + (minutes != 1 ? minutes + ' minutes' : 'minute');
1547
+ }
956
1548
 
957
- // Request backup generation
958
- this.sendExtensionRequest(this.REQ_BACKUP_BOOT_LOG);
1549
+ this._elapsedTxt.text(title);
1550
+ }
959
1551
  }
960
1552
 
961
- _onRuntimeLogGenerateBtClick()
1553
+ get load()
962
1554
  {
963
- // Show/hide operation in progress message
964
- this._disableBackupInterface(true, this.RUNTIME_LOG_BACKUP_ID);
1555
+ return this._averageLoad;
1556
+ }
1557
+ }
1558
+
1559
+ // DEFINE COMPONENT
1560
+ if (!window.customElements.get('system-queue-chart'))
1561
+ window.customElements.define('system-queue-chart', SystemQueueChart);
1562
+
1563
+ /* WEBPACK VAR INJECTION */}.call(this, __webpack_require__(/*! jquery */ "jquery")))
1564
+
1565
+ /***/ }),
1566
+
1567
+ /***/ "./src/components/charts/system-workload-chart.js":
1568
+ /*!********************************************************!*\
1569
+ !*** ./src/components/charts/system-workload-chart.js ***!
1570
+ \********************************************************/
1571
+ /*! exports provided: SystemWorkloadChart */
1572
+ /***/ (function(module, __webpack_exports__, __webpack_require__) {
1573
+
1574
+ "use strict";
1575
+ __webpack_require__.r(__webpack_exports__);
1576
+ /* WEBPACK VAR INJECTION */(function($) {/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "SystemWorkloadChart", function() { return SystemWorkloadChart; });
1577
+ /* harmony import */ var _base_chart__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ./base-chart */ "./src/components/charts/base-chart.js");
1578
+ /* harmony import */ var _chart_utils__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ./chart-utils */ "./src/components/charts/chart-utils.js");
965
1579
 
966
- // Request backup generation
967
- this.sendExtensionRequest(this.REQ_BACKUP_RUNTIME_LOG);
1580
+
1581
+
1582
+
1583
+ class SystemWorkloadChart extends _base_chart__WEBPACK_IMPORTED_MODULE_0__["BaseChart"]
1584
+ {
1585
+ constructor()
1586
+ {
1587
+ super(240);
968
1588
  }
969
1589
 
970
- _onFullLogsGenerateBtClick()
1590
+ _initialize()
971
1591
  {
972
- // Show/hide operation in progress message
973
- this._disableBackupInterface(true, this.FULL_BACKUP_ID);
1592
+ const chartHeight = 240;
1593
+
1594
+ // Append container for chart & gauge
1595
+ let container = $('<div>', {class: 'd-flex flex-row'});
1596
+ $(this).append(container);
1597
+
1598
+ // CHART
1599
+
1600
+ let chartContainer = $('<div>', {class: 'flex-grow-1'});
1601
+ container.append(chartContainer);
1602
+
1603
+ // Create chart html
1604
+ let chartHtml = $('<div>');
1605
+ chartContainer.append(chartHtml);
1606
+
1607
+ // Initialize chart
1608
+ this._chartWidget = this._initChartWidget(chartHtml, chartHeight);
1609
+
1610
+ // Append text to display elapsed time
1611
+ this._elapsedTxt = $('<div>', {class: 'text-center', text: '---'});
1612
+ chartContainer.append(this._elapsedTxt);
1613
+
1614
+ // GAUGE
974
1615
 
975
- // Request backup generation
976
- this.sendExtensionRequest(this.REQ_BACKUP_FULL_LOGS);
1616
+ let gaugeContainer = $('<div>');
1617
+ container.append(gaugeContainer);
1618
+
1619
+ // Create gauge html
1620
+ let gaugeHtml = $('<div>');
1621
+ gaugeContainer.append(gaugeHtml);
1622
+
1623
+ // Initialize gauge
1624
+ this._gaugeWidget = this._initGaugeWidget(gaugeHtml, chartHeight);
1625
+
1626
+ // Append text to display load
1627
+ this._loadTxt = $('<div>', {class: 'text-center font-weight-bold', text: '---'});
1628
+ gaugeContainer.append(this._loadTxt);
1629
+
1630
+ //------------------------------
1631
+
1632
+ // Reset data source
1633
+ this.reset();
977
1634
  }
978
1635
 
979
- _onFilterChange()
1636
+ _initChartWidget(chartHtml, height)
980
1637
  {
981
- // Set filters
982
- this._setRuntimeLogDataSource(this._runtimeLogGrid.dataSource);
1638
+ return chartHtml.kendoChart({
1639
+ transitions: false,
1640
+ chartArea: {
1641
+ height: height,
1642
+ background: 'transparent'
1643
+ },
1644
+ legend: {
1645
+ visible: false
1646
+ },
1647
+ seriesDefaults: {
1648
+ type: 'area',
1649
+ labels: {
1650
+ visible: false,
1651
+ format: '{0}%',
1652
+ background: 'transparent'
1653
+ }
1654
+ },
1655
+ series: [
1656
+ {
1657
+ name: 'Workload',
1658
+ field: 'load',
1659
+ categoryField: 'time',
1660
+ color: '#3399ff',
1661
+ line: {
1662
+ width: 2
1663
+ }
1664
+ }
1665
+ ],
1666
+ valueAxis: {
1667
+ labels: {
1668
+ format: '{0}%'
1669
+ },
1670
+ line: {
1671
+ visible: false
1672
+ },
1673
+ min: 0,
1674
+ max: 100
1675
+ },
1676
+ categoryAxis: {
1677
+ majorGridLines: {
1678
+ visible: false
1679
+ },
1680
+ majorTicks: {
1681
+ visible: false
1682
+ },
1683
+ labels: {
1684
+ visible: false
1685
+ }
1686
+ },
1687
+ tooltip: {
1688
+ shared: true,
1689
+ visible: true,
1690
+ sharedTemplate: Object(_chart_utils__WEBPACK_IMPORTED_MODULE_1__["getBasicSharedTemplate"])('{0}%'),
1691
+ background: '#e4e4e4'
1692
+ }
1693
+ }).data('kendoChart');
983
1694
  }
984
1695
 
985
- _onClearFilterClick()
1696
+ _initGaugeWidget(gaugeHtml, height)
986
1697
  {
987
- this._clearRuntimeLogFilters();
988
- this._setRuntimeLogDataSource(this._runtimeLogGrid.dataSource);
1698
+ return gaugeHtml.kendoLinearGauge({
1699
+ pointer: {
1700
+ value: 0,
1701
+ shape: 'arrow',
1702
+ size: 10,
1703
+ color: '#212529'
1704
+ },
1705
+ gaugeArea: {
1706
+ height: height
1707
+ },
1708
+ scale: {
1709
+ majorUnit: 10,
1710
+ minorUnit: 1,
1711
+ max: 100,
1712
+ min: 0,
1713
+ ranges: [
1714
+ {
1715
+ from: 0,
1716
+ to: 50,
1717
+ color: '#00cc00',
1718
+ opacity: .5
1719
+ }, {
1720
+ from: 50,
1721
+ to: 75,
1722
+ color: '#ffe106',
1723
+ opacity: .5
1724
+ }, {
1725
+ from: 75,
1726
+ to: 100,
1727
+ color: '#ed1601',
1728
+ opacity: .5
1729
+ }
1730
+ ],
1731
+ majorTicks: {
1732
+ visible: true,
1733
+ size: 10
1734
+ },
1735
+ minorTicks: {
1736
+ visible: false
1737
+ },
1738
+ labels: {
1739
+ visible: false
1740
+ },
1741
+ line: {
1742
+ visible: false
1743
+ },
1744
+ rangeSize: 10
1745
+ }
1746
+ }).data('kendoLinearGauge');
989
1747
  }
990
1748
 
991
- _onExportRuntimeLogBtClick()
1749
+ addEntry(load)
992
1750
  {
993
- let log = '';
994
- const entries = this._runtimeLogGrid.dataSource.view();
1751
+ // CHART
1752
+
1753
+ let now = new Date();
1754
+
1755
+ let newEntry = {
1756
+ time: Object(_chart_utils__WEBPACK_IMPORTED_MODULE_1__["getFormattedDateTime"])(now),
1757
+ timestamp: now.getTime(),
1758
+ load: load
1759
+ };
995
1760
 
996
- for (let i = 0; i < entries.length; i++)
1761
+ // Add entry to chart
1762
+ super._addEntry(newEntry);
1763
+
1764
+ // Update x axis
1765
+ if (!this.isPaused)
1766
+ this._updateElapsedText();
1767
+
1768
+ // GAUGE
1769
+
1770
+ // Evaluate average
1771
+ let numValues = this._dataSource.total();
1772
+ let averageLoad = 0;
1773
+
1774
+ if (numValues > 0)
997
1775
  {
998
- const item = entries[i];
999
- log += [item.date, item.time, item.level, item.thread, item.clazz, item.message].join(item.sep) + '\n';
1776
+ let total = 0;
1777
+ for (let i = 0; i < numValues; i++)
1778
+ total += this._dataSource.at(i).load;
1779
+
1780
+ averageLoad = Math.round(total / numValues);
1000
1781
  }
1001
1782
 
1002
- // Export log to file
1003
- this._exportLog(log, this.RUNTIME_LOG_BACKUP_ID);
1783
+ // Set gauge value
1784
+ this._gaugeWidget.value(averageLoad);
1785
+
1786
+ // Set load text
1787
+ this._loadTxt.text(averageLoad + '%')
1004
1788
  }
1005
1789
 
1006
- _onDownloadGridSelectionChange()
1790
+ _updateElapsedText()
1007
1791
  {
1008
- // Enable/disable buttons
1009
- const selectedRows = this._downloadGrid.select();
1010
- $('#lgv-downloadBt').attr('disabled', selectedRows.length == 0);
1011
- $('#lgv-deleteBt').attr('disabled', selectedRows.length == 0);
1792
+ let ds = this._dataSource;
1012
1793
 
1013
- if (selectedRows.length > 0)
1794
+ if (ds.total() > 1)
1014
1795
  {
1015
- let dataItem = this._downloadGrid.dataItem(selectedRows[0]);
1016
- $('#lgv-downloadBt').attr('href', dataItem.url);
1796
+ // Get axis limits
1797
+ const chartMinimum = ds.at(0).timestamp;
1798
+ const chartMaximum = ds.at(ds.total() - 1).timestamp;
1799
+
1800
+ // Set elapsed text
1801
+ let seconds = Math.round( (chartMaximum - chartMinimum) / 1000 );
1802
+ let title;
1803
+
1804
+ if (seconds < 60)
1805
+ title = 'Last ' + (seconds != 1 ? seconds + ' seconds' : 'second');
1806
+ else
1807
+ {
1808
+ const minutes = Math.floor(seconds / 60);
1809
+ title = 'Last ' + (minutes != 1 ? minutes + ' minutes' : 'minute');
1810
+ }
1811
+
1812
+ this._elapsedTxt.text(title);
1017
1813
  }
1018
- else
1019
- $('#lgv-downloadBt').attr('href', '#');
1020
1814
  }
1021
1815
 
1022
- _onDeleteBtClick()
1816
+ get load()
1023
1817
  {
1024
- let selectedRows = this._downloadGrid.select();
1818
+ return this._averageLoad;
1819
+ }
1820
+ }
1025
1821
 
1026
- if (selectedRows.length > 0)
1027
- {
1028
- let dataItem = this._downloadGrid.dataItem(selectedRows[0]);
1822
+ // DEFINE COMPONENT
1823
+ if (!window.customElements.get('system-workload-chart'))
1824
+ window.customElements.define('system-workload-chart', SystemWorkloadChart);
1029
1825
 
1030
- // Request backup deletion
1031
- let params = new SFS2X.SFSObject();
1032
- params.putUtfString('file', dataItem.name);
1826
+ /* WEBPACK VAR INJECTION */}.call(this, __webpack_require__(/*! jquery */ "jquery")))
1033
1827
 
1034
- this.sendExtensionRequest(this.REQ_DELETE_BACKUP, params);
1035
- }
1828
+ /***/ }),
1829
+
1830
+ /***/ "./src/components/charts/thread-count-chart.js":
1831
+ /*!*****************************************************!*\
1832
+ !*** ./src/components/charts/thread-count-chart.js ***!
1833
+ \*****************************************************/
1834
+ /*! exports provided: ThreadCountChart */
1835
+ /***/ (function(module, __webpack_exports__, __webpack_require__) {
1836
+
1837
+ "use strict";
1838
+ __webpack_require__.r(__webpack_exports__);
1839
+ /* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "ThreadCountChart", function() { return ThreadCountChart; });
1840
+ /* harmony import */ var _base_chart__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ./base-chart */ "./src/components/charts/base-chart.js");
1841
+ /* harmony import */ var _chart_utils__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ./chart-utils */ "./src/components/charts/chart-utils.js");
1842
+
1843
+
1844
+
1845
+
1846
+ class ThreadCountChart extends _base_chart__WEBPACK_IMPORTED_MODULE_0__["BaseChart"]
1847
+ {
1848
+ constructor()
1849
+ {
1850
+ super(120);
1851
+ }
1852
+
1853
+ _initChartWidget(chartHtml)
1854
+ {
1855
+ return chartHtml.kendoChart({
1856
+ transitions: false,
1857
+ chartArea: {
1858
+ height: 120,
1859
+ background: 'transparent'
1860
+ },
1861
+ legend: {
1862
+ visible: false
1863
+ },
1864
+ seriesDefaults: {
1865
+ type: 'area',
1866
+ labels: {
1867
+ visible: false,
1868
+ format: '{0}',
1869
+ background: 'transparent'
1870
+ },
1871
+ },
1872
+ series: [
1873
+ {
1874
+ name: 'Threads',
1875
+ field: 'threads',
1876
+ categoryField: 'time',
1877
+ color: this.threadsColor,
1878
+ line: {
1879
+ width: 2
1880
+ }
1881
+ }
1882
+ ],
1883
+ valueAxis: {
1884
+ min: 0,
1885
+ labels: {
1886
+ format: '{0}'
1887
+ },
1888
+ line: {
1889
+ visible: false
1890
+ },
1891
+ },
1892
+ categoryAxis: {
1893
+ majorGridLines: {
1894
+ visible: false
1895
+ },
1896
+ majorTicks: {
1897
+ visible: false
1898
+ },
1899
+ labels: {
1900
+ visible: false
1901
+ }
1902
+ },
1903
+ tooltip: {
1904
+ shared: true,
1905
+ visible: true,
1906
+ sharedTemplate: Object(_chart_utils__WEBPACK_IMPORTED_MODULE_1__["getBasicSharedTemplate"])(),
1907
+ background: '#e4e4e4'
1908
+ }
1909
+ }).data('kendoChart');
1910
+ }
1911
+
1912
+ get threadsColor()
1913
+ {
1914
+ return '#3399FF';
1915
+ }
1916
+
1917
+ addEntry(threads)
1918
+ {
1919
+ let newEntry = {
1920
+ time: Object(_chart_utils__WEBPACK_IMPORTED_MODULE_1__["getFormattedDateTime"])(new Date()),
1921
+ threads: threads
1922
+ };
1923
+
1924
+ // Add entry to chart
1925
+ super._addEntry(newEntry);
1926
+ }
1927
+ }
1928
+
1929
+ // DEFINE COMPONENT
1930
+ if (!window.customElements.get('thread-count-chart'))
1931
+ window.customElements.define('thread-count-chart', ThreadCountChart);
1932
+
1933
+
1934
+ /***/ }),
1935
+
1936
+ /***/ "./src/modules/dashboard.js":
1937
+ /*!**********************************!*\
1938
+ !*** ./src/modules/dashboard.js ***!
1939
+ \**********************************/
1940
+ /*! exports provided: default */
1941
+ /***/ (function(module, __webpack_exports__, __webpack_require__) {
1942
+
1943
+ "use strict";
1944
+ __webpack_require__.r(__webpack_exports__);
1945
+ /* WEBPACK VAR INJECTION */(function($) {/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "default", function() { return Dashboard; });
1946
+ /* harmony import */ var _base_module__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ./base-module */ "./src/modules/base-module.js");
1947
+ /* harmony import */ var _utils_utilities__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ../utils/utilities */ "./src/utils/utilities.js");
1948
+ /* harmony import */ var _components_charts_chart_utils__WEBPACK_IMPORTED_MODULE_2__ = __webpack_require__(/*! ../components/charts/chart-utils */ "./src/components/charts/chart-utils.js");
1949
+ /* harmony import */ var _components_charts_cpu_usage_chart__WEBPACK_IMPORTED_MODULE_3__ = __webpack_require__(/*! ../components/charts/cpu-usage-chart */ "./src/components/charts/cpu-usage-chart.js");
1950
+ /* harmony import */ var _components_charts_thread_count_chart__WEBPACK_IMPORTED_MODULE_4__ = __webpack_require__(/*! ../components/charts/thread-count-chart */ "./src/components/charts/thread-count-chart.js");
1951
+ /* harmony import */ var _components_charts_memory_chart__WEBPACK_IMPORTED_MODULE_5__ = __webpack_require__(/*! ../components/charts/memory-chart */ "./src/components/charts/memory-chart.js");
1952
+ /* harmony import */ var _components_charts_network_realtime_chart__WEBPACK_IMPORTED_MODULE_6__ = __webpack_require__(/*! ../components/charts/network-realtime-chart */ "./src/components/charts/network-realtime-chart.js");
1953
+ /* harmony import */ var _components_charts_network_24h_chart__WEBPACK_IMPORTED_MODULE_7__ = __webpack_require__(/*! ../components/charts/network-24h-chart */ "./src/components/charts/network-24h-chart.js");
1954
+ /* harmony import */ var _components_charts_system_workload_chart__WEBPACK_IMPORTED_MODULE_8__ = __webpack_require__(/*! ../components/charts/system-workload-chart */ "./src/components/charts/system-workload-chart.js");
1955
+ /* harmony import */ var _components_charts_system_queue_chart__WEBPACK_IMPORTED_MODULE_9__ = __webpack_require__(/*! ../components/charts/system-queue-chart */ "./src/components/charts/system-queue-chart.js");
1956
+
1957
+
1958
+
1959
+
1960
+
1961
+
1962
+
1963
+
1964
+
1965
+
1966
+
1967
+ class Dashboard extends _base_module__WEBPACK_IMPORTED_MODULE_0__["BaseModule"]
1968
+ {
1969
+ constructor()
1970
+ {
1971
+ super('dashboard');
1972
+
1973
+ // Outgoing requests
1974
+ this.REQ_INIT = 'init';
1975
+ this.REQ_GET_DATA = 'getData';
1976
+ this.REQ_DUMP_THREADS = 'getThDump';
1977
+
1978
+ // Incoming responses
1979
+ this.RESP_INIT = 'init';
1980
+ this.RESP_DATA = 'data';
1981
+ this.RESP_THREAD_DUMP = 'thDump';
1982
+
1983
+ this._tcpOutThreads = 0;
1984
+ this._tcpOutLoad = 0;
1985
+ this._udpOutThreads = 0;
1986
+ this._udpOutLoad = 0;
1036
1987
  }
1037
1988
 
1038
1989
  //------------------------------------
1039
- // PRIVATE METHODS
1990
+ // COMMON MODULE INTERFACE METHODS
1991
+ // This members are used by the main controller
1992
+ // to communicate with the module's controller.
1040
1993
  //------------------------------------
1041
1994
 
1042
- _switchRuntimeViewStack(viewId)
1995
+ initialize(idData, shellController)
1043
1996
  {
1044
- document.getElementById('lgv-runtime-viewstack').selectedElement = document.getElementById(viewId);
1997
+ // Call super method
1998
+ super.initialize(idData, shellController);
1999
+
2000
+ // Initialize scrolling tabs
2001
+ $('#dsh-tabNavigator > #tabs').scrollingTabs({
2002
+ bootstrapVersion: 4,
2003
+ scrollToTabEdge: true,
2004
+ enableSwiping: true,
2005
+ disableScrollArrowsOnFullyScrolled: true,
2006
+ cssClassLeftArrow: 'fa fa-chevron-left',
2007
+ cssClassRightArrow: 'fa fa-chevron-right'
2008
+ });
2009
+
2010
+ // Initialize interval dropdown
2011
+ this._intervalDropDown = $('#dsh-intervalDD').kendoDropDownList({
2012
+ valueTemplate: '<span class="text-muted pr-1">Interval:</span><span>#:data.text#</span>',
2013
+ change: $.proxy(this._onUpdateIntervalChange, this)
2014
+ }).data('kendoDropDownList');
2015
+
2016
+ // Initialize threads grid
2017
+ this._threadsGrid = $('#dsh-threadsGrid').kendoGrid({
2018
+ dataSource: this._getThreadsGridDataSource(),
2019
+ scrollable: true,
2020
+ sortable: true,
2021
+ filterable: false,
2022
+ resizable: true,
2023
+ selectable: false,
2024
+ columns: [
2025
+ {
2026
+ field: 'id',
2027
+ title: 'Id',
2028
+ width: 60
2029
+ },
2030
+ {
2031
+ field: 'name',
2032
+ title: 'Name',
2033
+ width: '65%'
2034
+ },
2035
+ {
2036
+ field: 'cpuTime',
2037
+ title: 'CPU Time',
2038
+ format: '{0}%',
2039
+ width: '35%'
2040
+ },
2041
+ ],
2042
+ noRecords: {
2043
+ template: 'No threads.'
2044
+ },
2045
+ height: 'calc(240px - .75rem + 1rem)'
2046
+ }).data('kendoGrid');
2047
+
2048
+ // Add click listener to threads dump button
2049
+ $('#dsh-dumpThreadsBt').on('click', $.proxy(this._onDumpThreadsClick, this));
2050
+
2051
+ // Configure thread dump panel
2052
+ $('#dsh-threadDumpModal').modal({
2053
+ backdrop: 'static',
2054
+ keyboard: false,
2055
+ show: false,
2056
+ });
2057
+
2058
+ // Add listener for threads dump panel hide
2059
+ $('#dsh-threadDumpModal').on('hidden.bs.modal', $.proxy(this._onThreadDumpModalHidden, this));
2060
+
2061
+ // Add click listener to threads dump copy button
2062
+ $('#dsh-copyThreadDumpBt').on('click', $.proxy(this._onCopyThreadDumpClick, this));
2063
+
2064
+ // Initialize time range slider
2065
+ this._timeSlider = $('#dsh-timeSlider').kendoRangeSlider({
2066
+ min: -24,
2067
+ max: 0,
2068
+ smallStep: 1,
2069
+ largeStep: 0,
2070
+ change: $.proxy(this._onTimeRangeChange, this),
2071
+ tooltip: {
2072
+ template: Object(_components_charts_chart_utils__WEBPACK_IMPORTED_MODULE_2__["getTimeRangeTooltipTemplate"])()
2073
+ },
2074
+ leftDragHandleTitle: 'Drag',
2075
+ rightDragHandleTitle: 'Drag',
2076
+ }).data('kendoRangeSlider');
2077
+
2078
+ // Add click listener to reset charts button
2079
+ $('#dsh-resetBt').on('click', $.proxy(this._onResetChartsClick, this));
2080
+
2081
+ // Add click listener to play & pause chart update buttons
2082
+ $('#dsh-pauseBt').on('click', $.proxy(this._onPauseUpdateClick, this));
2083
+ $('#dsh-playBt').on('click', $.proxy(this._onPlayUpdateClick, this));
2084
+
2085
+ // Set reference to charts
2086
+ this._cpuUsageChart = document.getElementById('dsh-cpuUsageChart');
2087
+ this._threadCountChart = document.getElementById('dsh-threadCountChart');
2088
+ this._memoryChart = document.getElementById('dsh-memoryChart');
2089
+ this._networkRTChart = document.getElementById('dsh-networkRTChart');
2090
+ this._network24hChart = document.getElementById('dsh-network24hChart');
2091
+ this._sysWorkloadChart = document.getElementById('dsh-systemWorkloadChart');
2092
+ this._outTcpMsgQueueChart = document.getElementById('dsh-outgoingTcpMessagesQueueChart');
2093
+ this._outUdpMsgQueueChart = document.getElementById('dsh-outgoingUdpMessagesQueueChart');
2094
+ this._sysQueueChart = document.getElementById('dsh-systemQueueChart');
2095
+ this._extQueueChart = document.getElementById('dsh-extensionQueueChart');
2096
+
2097
+ // Set colors in chart legends
2098
+ $('#dsh-systemLoadColor').css('color', this._cpuUsageChart.systemColor);
2099
+ $('#dsh-processLoadColor').css('color', this._cpuUsageChart.processColor);
2100
+ $('#dsh-allocMemColor').css('color', this._memoryChart.allocatedColor);
2101
+ $('#dsh-usedMemColor').css('color', this._memoryChart.usedColor);
2102
+ $('#dsh-incomingNetColor').css('color', this._networkRTChart.incomingColor);
2103
+ $('#dsh-outgoingNetColor').css('color', this._networkRTChart.outgoingColor);
2104
+
2105
+ // Reset time range on 24h network chart
2106
+ this._onTimeRangeChange();
2107
+
2108
+ // Add listener to redraw all charts on main tab change
2109
+ // (to work around an issue with the chart resizing to default width when pane is hidden)
2110
+ $('a[data-toggle="tab"]').on('shown.bs.tab', $.proxy(this._onWindowResize, this));
2111
+
2112
+ // Add listener to redraw network charts on network tab change
2113
+ // (to work around an issue with the chart resizing to default width when pane is hidden)
2114
+ $('#dsh-networkPills a[data-toggle="pill"]').on('shown.bs.tab', $.proxy(this._redrawNetworkCharts, this));
2115
+
2116
+ // Add listener to redraw outgoing messages queue charts on tab change
2117
+ // (to work around an issue with the chart resizing to default width when pane is hidden)
2118
+ $('#dsh-msgQueuePills a[data-toggle="pill"]').on('shown.bs.tab', $.proxy(this._redrawOutMsgCharts, this));
2119
+
2120
+ // Add listener to redraw all charts in case of window resize
2121
+ $(window).on('resize', $.proxy(this._onWindowResize, this));
2122
+ this._onWindowResize(); // Also do it immediately
2123
+
2124
+ // Add listener to show help tooltips in queues status tab
2125
+ $('#dsh-queues').kendoTooltip({
2126
+ filter: 'i[title].help',
2127
+ position: 'right',
2128
+ width: '250px',
2129
+ content: function(e) {
2130
+ return `<div class="help-tooltip">${e.target.data('title')}</div>`;
2131
+ }
2132
+ });
2133
+
2134
+ // Add listener to regenerate link to open web server external manager console
2135
+ $('#dsh-useHttpsCheck').on('change', $.proxy(this._onUseHttpsForWsManagerChange, this));
2136
+
2137
+ // Send initialization request
2138
+ this.sendExtensionRequest(this.REQ_INIT);
1045
2139
  }
1046
2140
 
1047
- _switchBootViewStack(viewId)
2141
+ destroy()
1048
2142
  {
1049
- document.getElementById('lgv-boot-viewstack').selectedElement = document.getElementById(viewId);
2143
+ // Call super method
2144
+ super.destroy();
2145
+
2146
+ // Destroy scrolling tabs
2147
+ $('#dsh-tabNavigator #tabs').scrollingTabs('destroy');
2148
+
2149
+ // Remove other listeners
2150
+ $('#dsh-resetBt').off('click');
2151
+ $('#dsh-pauseBt').off('click');
2152
+ $('#dsh-playBt').off('click');
2153
+ $('#dsh-dumpThreadsBt').off('click');
2154
+ $('a[data-toggle="tab"]').off('shown.bs.tab');
2155
+ $('a[data-toggle="pill"]').off('shown.bs.tab');
2156
+ $(window).off('resize');
2157
+ $('#dsh-useHttpsCheck').off('change');
2158
+
2159
+ // Remove thread dump panel listener
2160
+ $('#dsh-threadDumpModal').off('hidden.bs.modal');
2161
+
2162
+ // Clear request scheduling
2163
+ clearTimeout(this._requestTimer);
1050
2164
  }
1051
2165
 
1052
- _switchDownloadViewStack(viewId)
2166
+ onExtensionCommand(command, data)
1053
2167
  {
1054
- document.getElementById('lgv-download-viewstack').selectedElement = document.getElementById(viewId);
2168
+ // Initialization data received
2169
+ if (command == this.RESP_INIT)
2170
+ {
2171
+ // Server sends flags indicating if web server's internal manager is accessible and how
2172
+ this._wsManagerConfig = {
2173
+ hasAccess: data.getBool('hasWsManager'),
2174
+ httpEnabled: data.getBool('httpEnabled'),
2175
+ httpsEnabled: data.getBool('httpsEnabled'),
2176
+ httpPort: data.getInt('httpPort'),
2177
+ httpsPort: data.getInt('httpsPort'),
2178
+ }
2179
+
2180
+ // Enable access to web server manager console
2181
+ this._enableWebServerManagerAccess();
2182
+
2183
+ // Request data to server
2184
+ this._requestData(true);
2185
+ }
2186
+
2187
+ // Data received
2188
+ else if (command == this.RESP_DATA)
2189
+ {
2190
+ // CPU LOAD CHART
2191
+
2192
+ this._cpuUsageChart.addEntry(data.getDouble('pCpu'), data.getDouble('sCpu'));
2193
+ $('#dsh-systemLoadLabel').text(data.getDouble('sCpu'));
2194
+ $('#dsh-processLoadLabel').text(data.getDouble('pCpu'));
2195
+
2196
+ // As system cpu load is not always available (for example on MacOS X), let's hide it in case it isn't
2197
+ if (data.getDouble('sCpu') < 0)
2198
+ $('#dsh-systemLoadLegend').hide();
2199
+
2200
+ // THREADS COUNT CHART
2201
+
2202
+ const threadCount = data.getSFSArray('thList').size();
2203
+ this._threadCountChart.addEntry(threadCount);
2204
+ $('#dsh-threadsCountLabel').text(threadCount);
2205
+
2206
+ // MEMORY USAGE CHART
2207
+ // NOTE: memory data is sent as Bytes by the server
2208
+
2209
+ const freeMem = data.getLong('freeMem');
2210
+ const maxMem = data.getLong('maxMem');
2211
+ const totalMem = data.getLong('totalMem');
2212
+ const usedMem = totalMem - freeMem;
2213
+
2214
+ this._memoryChart.addEntry(totalMem, usedMem);
2215
+ $('#dsh-maxMemoryLabel').text(this._getMemObject(maxMem).label);
2216
+ $('#dsh-freeMemoryLabel').text(this._getMemObject(freeMem).label);
2217
+
2218
+ // THREADS LIST
2219
+
2220
+ const totalThreadCpuTime = data.getLong('thCpu');
2221
+ let threads = [];
2222
+
2223
+ for (let i = 0; i < data.getSFSArray('thList').size(); i++)
2224
+ {
2225
+ let threadData = data.getSFSArray('thList').getSFSObject(i);
2226
+
2227
+ threads.push({
2228
+ id: threadData.getLong('id'),
2229
+ name: threadData.getUtfString('name'),
2230
+ cpuTime: Math.round(threadData.getLong('cpu') / totalThreadCpuTime * 1000) / 10
2231
+ });
2232
+ }
2233
+
2234
+ // Substitute grid's data source, retaining current sorting
2235
+ let currSort = this._threadsGrid.dataSource.sort();
2236
+ this._threadsGrid.setDataSource(this._getThreadsGridDataSource(threads, currSort));
2237
+
2238
+ // NETWORK TRAFFIC
2239
+ // NOTE: network traffic data is sent as KBytes by the server
2240
+
2241
+ if (data.containsKey('ntOutHistory'))
2242
+ {
2243
+ const outgoingData = data.getLongArray('ntOutHistory');
2244
+ const incomingData = data.getLongArray('ntInHistory');
2245
+ const samplingRateMins = data.getInt('ntRate');
2246
+ const shiftSeconds = data.getInt('ntShift');
2247
+
2248
+ this._network24hChart.addHistoryEntries(outgoingData, incomingData, samplingRateMins * 60, shiftSeconds);
2249
+ }
2250
+
2251
+ // Add current values
2252
+ this._network24hChart.addEntry(data.getLong('ntOut'), data.getLong('ntIn'));
2253
+ this._networkRTChart.addEntry(data.getLong('ntOut'), data.getLong('ntIn'));
2254
+
2255
+ // TRAFFIC DETAILS
2256
+
2257
+ // Sessions
2258
+ $('#dsh-detSessionsCurrent').text('Current: ' + (data.getInt('sSocket') + data.getInt('sHttp')));
2259
+ $('#dsh-detSessionsMax').text('Max: ' + data.getInt('ccsMax'));
2260
+ $('#dsh-detSessionsSocket').text('Socket: ' + data.getInt('sSocket'));
2261
+ $('#dsh-detSessionsHttp').text('Http: ' + data.getInt('sHttp'));
2262
+
2263
+ // Users
2264
+ $('#dsh-detUsersCurrent').text('Current: ' + (data.getInt('uSocket') + data.getInt('uHttp') + data.getInt('uNpc')));
2265
+ $('#dsh-detUsersMax').text('Max: ' + data.getInt('ccuMax'));
2266
+ $('#dsh-detUsersSocket').text('Socket: ' + data.getInt('uSocket'));
2267
+ $('#dsh-detUsersHttp').text('Http: ' + data.getInt('uHttp'));
2268
+ $('#dsh-detUsersNpc').text('Npc: ' + data.getInt('uNpc'));
2269
+
2270
+ // Rooms
2271
+ $('#dsh-detRooms').text(data.getInt('rTotal'));
2272
+ $('#dsh-detRoomsRegular').text('Regular: ' + (data.getInt('rTotal') - data.getInt('rGame')));
2273
+ $('#dsh-detRoomsGame').text('Game: ' + data.getInt('rGame'));
2274
+
2275
+ // Total data transfer
2276
+ $('#dsh-detDataTransferOut').text('Out: ' + Object(_utils_utilities__WEBPACK_IMPORTED_MODULE_1__["kBytesToSize"])(data.getLong('ntOut')));
2277
+ $('#dsh-detDataTransferIn').text('In: ' + Object(_utils_utilities__WEBPACK_IMPORTED_MODULE_1__["kBytesToSize"])(data.getLong('ntIn')));
2278
+
2279
+ // Current data transfer rate
2280
+ let obj = this._networkRTChart.getLastEntry();
2281
+
2282
+ if (obj == null)
2283
+ obj = {outgoing: 0, incoming: 0};
2284
+
2285
+ $('#dsh-detTransferRateOut').text('Out: ' + Object(_utils_utilities__WEBPACK_IMPORTED_MODULE_1__["kBytesToSize"])(obj.outgoing, 2, 'KB', '/s'));
2286
+ $('#dsh-detTransferRateIn').text('In: ' + Object(_utils_utilities__WEBPACK_IMPORTED_MODULE_1__["kBytesToSize"])(obj.incoming, 2, 'KB', '/s'));
2287
+
2288
+ // Average data transfer rate over whole uptime
2289
+ if (this._uptimeSecs > 0)
2290
+ {
2291
+ const avgOut = Math.round(data.getLong('ntOut') / this._uptimeSecs * 100) / 100;
2292
+ const avgIn = Math.round(data.getLong('ntIn') / this._uptimeSecs * 100) / 100;
2293
+
2294
+ $('#dsh-detAvgTransferRateOut').text('Out: ' + Object(_utils_utilities__WEBPACK_IMPORTED_MODULE_1__["kBytesToSize"])(avgOut, 2, 'KB', '/s'));
2295
+ $('#dsh-detAvgTransferRateIn').text('In: ' + Object(_utils_utilities__WEBPACK_IMPORTED_MODULE_1__["kBytesToSize"])(avgIn, 2, 'KB', '/s'));
2296
+ }
2297
+
2298
+ // Dropped packets
2299
+ const totalPacketsOut = data.getLong('pktOut');
2300
+
2301
+ const totalDroppedTcpPacketsOut = data.getLong('pktOutDrop');
2302
+ $('#dsh-detDroppedTcpPacketsOut').text(`Out TCP: ${totalDroppedTcpPacketsOut} (${Math.round(totalDroppedTcpPacketsOut / totalPacketsOut * 100)}%)`);
2303
+
2304
+ const totalDroppedUdpPacketsOut = data.getLong('pktOutUdpDrop');
2305
+ $('#dsh-detDroppedUdpPacketsOut').text(`Out UDP: ${totalDroppedUdpPacketsOut} (${Math.round(totalDroppedUdpPacketsOut / totalPacketsOut * 100)}%)`);
2306
+
2307
+ const totalPacketsIn = data.getLong('pktIn');
2308
+ const totalDroppedPacketsIn = data.getLong('pktInDrop');
2309
+ $('#dsh-detDroppedPacketsIn').text(`In: ${totalDroppedPacketsIn} (${Math.round(totalDroppedPacketsIn / totalPacketsIn * 100)}%)`);
2310
+
2311
+ // SYSTEM STATUS
2312
+
2313
+ // Outgoing TCP messages queue
2314
+ const outMsgQueue = data.getInt('outMsgQueue');
2315
+ const outMsgQueueMax = data.getInt('outMsgQueueMax');
2316
+ this._outTcpMsgQueueChart.addEntry(outMsgQueue, outMsgQueueMax);
2317
+ this._tcpOutThreads = data.getInt('outMsgTh');
2318
+ this._tcpOutLoad = this._outTcpMsgQueueChart.load;
2319
+
2320
+ // Outgoing UDP messages queue
2321
+ const outUdpMsgQueue = data.getInt('outUdpMsgQueue');
2322
+ const outUdpMsgQueueMax = data.getInt('outUdpMsgQueueMax');
2323
+ this._outUdpMsgQueueChart.addEntry(outUdpMsgQueue, outUdpMsgQueueMax);
2324
+ this._udpOutThreads = data.getInt('outUdpMsgTh');
2325
+ this._udpOutLoad = this._outUdpMsgQueueChart.load;
2326
+
2327
+ this._updateOutMessagesQueueStatus();
2328
+
2329
+ // System Controller requests queue
2330
+ const sysCtrlQueue = data.getInt('sysCtrlQueue');
2331
+ const sysCtrlQueueMax = data.getInt('sysCtrlQueueMax');
2332
+ this._sysQueueChart.addEntry(sysCtrlQueue, sysCtrlQueueMax);
2333
+ $('#dsh-systemQueueThreads').text(this._getQueueThreadsLabel(data.getInt('sysCtrlTh')));
2334
+ $('#dsh-systemQueueLoad').text(this._getQueueLoadLabel(this._sysQueueChart.load));
2335
+
2336
+ // Extension Controller requests queue
2337
+ const extCtrlQueue = data.getInt('extCtrlQueue');
2338
+ const extCtrlQueueMax = data.getInt('extCtrlQueueMax');
2339
+ this._extQueueChart.addEntry(extCtrlQueue, extCtrlQueueMax);
2340
+ $('#dsh-extensionQueueThreads').text(this._getQueueThreadsLabel(data.getInt('extCtrlTh')));
2341
+ $('#dsh-extensionQueueLoad').text(this._getQueueLoadLabel(this._sysQueueChart.load));
2342
+
2343
+ // Evaluate overall system workload
2344
+ let sysWorkload = Math.round((outMsgQueue / outMsgQueueMax +
2345
+ outUdpMsgQueue / outUdpMsgQueueMax +
2346
+ sysCtrlQueue / sysCtrlQueueMax +
2347
+ extCtrlQueue / extCtrlQueueMax) * 100 / 4);
2348
+ if (sysWorkload > 100)
2349
+ sysWorkload = 100;
2350
+
2351
+ this._sysWorkloadChart.addEntry(sysWorkload);
2352
+ }
2353
+
2354
+ // Thread dump received
2355
+ else if (command == this.RESP_THREAD_DUMP)
2356
+ {
2357
+ // Enable button
2358
+ $('#dsh-dumpThreadsBt').attr('disabled', false);
2359
+
2360
+ // Set result
2361
+ $('#dsh-threadDumpResult').text(data.getText('dump'));
2362
+
2363
+ // Display modal
2364
+ $('#dsh-threadDumpModal').modal('show');
2365
+ }
1055
2366
  }
1056
2367
 
1057
- _enableRuntimeLogControls(enable)
2368
+ onUptimeUpdated(values)
1058
2369
  {
1059
- $('#lgv-loadBt').attr('disabled', !enable);
1060
- $('#lgv-exportRuntimeLogBt').attr('disabled', !enable);
1061
- $('#lgv-filterBt').attr('disabled', !enable);
2370
+ this._uptimeSecs = values[3];
2371
+ this._uptimeSecs += (values[2] * 60);
2372
+ this._uptimeSecs += (values[1] * 60 * 60);
2373
+ this._uptimeSecs += (values[0] * 24 * 60 * 60);
2374
+
2375
+ $('#dsh-uptimeDays').text( (values[0] <= 9 ? '00' : (values[0] <= 99 ? '0' : '')) + values[0] );
2376
+ $('#dsh-uptimeHours').text( (values[1] <= 9 ? '0' : '') + values[1] );
2377
+ $('#dsh-uptimeMinutes').text( (values[2] <= 9 ? '0' : '') + values[2] );
2378
+ $('#dsh-uptimeSeconds').text( (values[3] <= 9 ? '0' : '') + values[3] );
1062
2379
  }
1063
2380
 
1064
- _loadRuntimeLog()
2381
+ //---------------------------------
2382
+ // UI EVENT LISTENERS
2383
+ //---------------------------------
2384
+
2385
+ _onUpdateIntervalChange()
1065
2386
  {
1066
- // Disable controls to load log, so that impatient users can't send multiple requests
1067
- // (it will be enabled again when a response is received)
1068
- this._enableRuntimeLogControls(false);
2387
+ // Request data to server
2388
+ this._requestData();
2389
+ }
1069
2390
 
1070
- // Clear filters
1071
- this._clearRuntimeLogFilters();
2391
+ _onPauseUpdateClick()
2392
+ {
2393
+ this._setChartUpdatePaused(true);
1072
2394
 
1073
- // Show loading bar
1074
- this._switchRuntimeViewStack('lgv-runtimeLogLoadingView');
2395
+ // Hide pause button and show play button
2396
+ $('#dsh-pauseBt').addClass('hidden');
2397
+ $('#dsh-playBt').removeClass('hidden');
2398
+ }
1075
2399
 
1076
- // Send request
1077
- // (the number of lines to be retrieved is sent)
1078
- let params = new SFS2X.SFSObject();
1079
- params.putInt('numEntries', Number(this._logLinesDD.value()));
2400
+ _onPlayUpdateClick()
2401
+ {
2402
+ this._setChartUpdatePaused(false);
1080
2403
 
1081
- this.sendExtensionRequest(this.REQ_GET_RUNTIME_LOG, params);
2404
+ // Hide play button and show pause button
2405
+ $('#dsh-pauseBt').removeClass('hidden');
2406
+ $('#dsh-playBt').addClass('hidden');
1082
2407
  }
1083
2408
 
1084
- _clearRuntimeLogFilters()
2409
+ _onResetChartsClick()
1085
2410
  {
1086
- this._levelFilterDD.select(0);
1087
- this._classFilterDD.select(0);
1088
- $('#lgv-messageIn').val('');
2411
+ // Reactivate realtime charts update
2412
+ this._onPlayUpdateClick();
2413
+
2414
+ // Reset realtime charts
2415
+ this._cpuUsageChart.reset();
2416
+ this._threadCountChart.reset();
2417
+ this._memoryChart.reset();
2418
+ this._networkRTChart.reset();
2419
+ this._sysWorkloadChart.reset();
2420
+ this._outTcpMsgQueueChart.reset();
2421
+ this._outUdpMsgQueueChart.reset();
2422
+ this._sysQueueChart.reset();
2423
+ this._extQueueChart.reset();
1089
2424
  }
1090
2425
 
1091
- _exportLog(log, name)
2426
+ _redrawNetworkCharts()
1092
2427
  {
1093
- let blob = new Blob([log], {type: "text/plain;charset=utf-8"});
1094
- let date = moment__WEBPACK_IMPORTED_MODULE_3__().format('YYYYMMDD_HHmmss');
2428
+ // Redraw charts
2429
+ this._networkRTChart.redraw();
2430
+ this._network24hChart.redraw();
1095
2431
 
1096
- file_saver__WEBPACK_IMPORTED_MODULE_2__["saveAs"](blob, `${name}_${date}.log`);
2432
+ // Redraw time range slider
2433
+ this._timeSlider.resize();
1097
2434
  }
1098
2435
 
1099
- _setRuntimeLogDataSource(ds)
2436
+ _redrawOutMsgCharts()
1100
2437
  {
1101
- // Read current horizontal scroll value
1102
- const scrollLeft = $('#lgv-runtimeLogGrid .k-grid-content', this._runtimeLogGrid.wrapper).scrollLeft();
2438
+ // Redraw charts
2439
+ this._outTcpMsgQueueChart.redraw();
2440
+ this._outUdpMsgQueueChart.redraw();
1103
2441
 
1104
- // Assign data source to grid
1105
- this._runtimeLogGrid.setDataSource(ds);
2442
+ this._updateOutMessagesQueueStatus();
2443
+ }
1106
2444
 
1107
- // Set filters
1108
- this._setFilters(ds);
2445
+ _onWindowResize()
2446
+ {
2447
+ // Redraw all global charts
2448
+ this._cpuUsageChart.redraw();
2449
+ this._threadCountChart.redraw();
2450
+ this._memoryChart.redraw();
2451
+ this._redrawNetworkCharts();
2452
+
2453
+ // Redraw all queue charts
2454
+ this._sysWorkloadChart.redraw();
2455
+ this._sysQueueChart.redraw();
2456
+ this._extQueueChart.redraw();
2457
+ this._redrawOutMsgCharts();
2458
+ }
1109
2459
 
1110
- // Set horizontal scroll
1111
- $('#lgv-runtimeLogGrid .k-grid-content', this._runtimeLogGrid.wrapper).scrollLeft(scrollLeft);
2460
+ _onTimeRangeChange()
2461
+ {
2462
+ let values = this._timeSlider.value();
1112
2463
 
1113
- // Update counter
1114
- $('#lgv-runtimeLogEntriesLb').text(`Log entries: ${this._totalRuntimeLogEntries} (${ds.total()} displayed)`);
2464
+ if (values[0] == values[1])
2465
+ {
2466
+ if (values[1] == -24)
2467
+ values[1] = -23;
2468
+
2469
+ values[0] = values[1] - 1;
2470
+
2471
+ // Reset the time range slider value (we need to use setTimeout
2472
+ // bacause doing it in the change event listener doesn't redraw the slider)
2473
+ setTimeout($.proxy( function(values) { this._timeSlider.value(values); }, this), 10, values);
2474
+ }
2475
+
2476
+ this._network24hChart.range = values;
1115
2477
  }
1116
2478
 
1117
- _setFilters(ds)
2479
+ _onUseHttpsForWsManagerChange()
1118
2480
  {
1119
- let filters = [];
2481
+ const wsMan = this._wsManagerConfig;
1120
2482
 
1121
- // Level filtering
1122
- if (this._levelFilterDD.select() > 0)
1123
- filters.push({
1124
- field: 'level', operator: 'eq', value: this._levelFilterDD.value()
1125
- });
2483
+ let url = ($('#dsh-useHttpsCheck').prop('checked') ? 'https' : 'http') + '://';
2484
+ url += this.smartFox.config.host;
2485
+ url += ':' + ($('#dsh-useHttpsCheck').prop('checked') ? wsMan.httpsPort : wsMan.httpPort);
2486
+ url += '/manager/status';
1126
2487
 
1127
- // Class filtering
1128
- if (this._classFilterDD.select() > 0)
1129
- filters.push({
1130
- field: 'clazz', operator: 'eq', value: this._classFilterDD.value()
1131
- });
2488
+ $('#dsh-toWsManagerBt').attr('href', url);
2489
+ }
1132
2490
 
1133
- // Message filtering
1134
- if ($('#lgv-messageIn').val() != '')
1135
- filters.push({
1136
- field: 'message', operator: 'contains', value: $('#lgv-messageIn').val()
1137
- });
2491
+ _onDumpThreadsClick()
2492
+ {
2493
+ // Disable button
2494
+ $('#dsh-dumpThreadsBt').attr('disabled', true);
1138
2495
 
1139
- // Set filters
1140
- ds.filter(filters);
2496
+ // Send request to extension
2497
+ this.sendExtensionRequest(this.REQ_DUMP_THREADS);
1141
2498
  }
1142
2499
 
1143
- _disableBackupInterface(disable, backupId = null)
2500
+ _onCopyThreadDumpClick()
2501
+ {
2502
+ $('#dsh-threadDumpResult').attr('contenteditable', true);
2503
+ $('#dsh-threadDumpResult').attr('spellcheck', false);
2504
+ $('#dsh-threadDumpResult').focus();
2505
+ document.execCommand('selectAll');
2506
+ document.execCommand('copy');
2507
+ window.getSelection().removeAllRanges();
2508
+ $('#dsh-threadDumpResult').blur();
2509
+ $('#dsh-threadDumpResult').attr('contenteditable', false);
2510
+ }
2511
+
2512
+ _onThreadDumpModalHidden()
2513
+ {
2514
+ // Clear modal content
2515
+ $('#dsh-threadDumpResult').text('');
2516
+ }
2517
+
2518
+ //------------------------------------
2519
+ // PRIVATE METHODS
2520
+ //------------------------------------
2521
+
2522
+ _enableWebServerManagerAccess()
1144
2523
  {
1145
- if (disable)
2524
+ const wsMan = this._wsManagerConfig;
2525
+
2526
+ if (wsMan.hasAccess && (wsMan.httpEnabled || wsMan.httpsEnabled))
1146
2527
  {
1147
- // Show proper progress bar
1148
- if (backupId == this.BOOT_LOG_BACKUP_ID)
1149
- $('#lgv-bootLogBackupCard .progress-bar').show();
1150
- else if (backupId == this.RUNTIME_LOG_BACKUP_ID)
1151
- $('#lgv-runtimeLogBackupCard .progress-bar').show();
1152
- else if (backupId == this.FULL_BACKUP_ID)
1153
- $('#lgv-fullLogBackupCard .progress-bar').show();
1154
-
1155
- // Disable buttons
1156
- $('#lgv-bootLogBackupCard .backup-button').attr('disabled', true);
1157
- $('#lgv-runtimeLogBackupCard .backup-button').attr('disabled', true);
1158
- $('#lgv-fullLogBackupCard .backup-button').attr('disabled', true);
2528
+ // Hide alert
2529
+ $('#dsh-wsManagerAlert').hide();
2530
+
2531
+ // Enable button
2532
+ $('#dsh-toWsManagerBt').attr('disabled', false);
2533
+
2534
+ // Set checkbox
2535
+ $('#dsh-useHttpsCheck').prop('checked', wsMan.httpsEnabled);
2536
+ $('#dsh-useHttpsCheck').attr('disabled', !wsMan.httpEnabled || !wsMan.httpsEnabled);
2537
+
2538
+ // Generate link
2539
+ this._onUseHttpsForWsManagerChange();
1159
2540
  }
1160
2541
  else
1161
2542
  {
1162
- // Hide all progress bar
1163
- $('.card-body .progress-bar').hide();
2543
+ // Show alert
2544
+ $('#dsh-wsManagerAlert').show();
1164
2545
 
1165
- // Enable buttons
1166
- $('#lgv-fullLogBackupCard .backup-button').attr('disabled', false);
2546
+ if (!wsMan.hasAccess)
2547
+ $('#dsh-wsManagerAlert').html('Access to Tomcat Manager console requires at least one user to be defined in <em>Web Server</em> tab of <strong>Server Configurator</strong> module.');
2548
+ else
2549
+ $('#dsh-wsManagerAlert').html('Access to Tomcat Manager console requires either HTTP or HTTPS to be enabled in <em>Web Server</em> tab of <strong>Server Configurator</strong> module.');
1167
2550
 
1168
- if (!this._bootLogBackupUnavailable)
1169
- $('#lgv-bootLogBackupCard .backup-button').attr('disabled', false);
2551
+ // Disable button
2552
+ $('#dsh-toWsManagerBt').attr('disabled', true);
1170
2553
 
1171
- if (!this._runtimeLogBackupUnavailable)
1172
- $('#lgv-runtimeLogBackupCard .backup-button').attr('disabled', false);
2554
+ // Disable checkbox
2555
+ $('#dsh-useHttpsCheck').attr('checked', wsMan.httpsEnabled);
2556
+ $('#dsh-useHttpsCheck').attr('disabled', true);
1173
2557
  }
1174
2558
  }
1175
2559
 
1176
- _initBackupCard(idSelector)
2560
+ /**
2561
+ * Build the polling request to be sent to the server.
2562
+ */
2563
+ _requestData(getHistory = false)
1177
2564
  {
1178
- $(idSelector + ' .backup-details').hide();
1179
- $(idSelector + ' .progress-bar').hide();
2565
+ // Clear previous request scheduling
2566
+ clearTimeout(this._requestTimer);
2567
+
2568
+ // Check if connection is still available
2569
+ if (this.smartFox.isConnected)
2570
+ {
2571
+ // Build request parameters
2572
+ let params = new SFS2X.SFSObject();
2573
+
2574
+ if (getHistory)
2575
+ params.putBool('history', true);
2576
+
2577
+ // Send request to extension
2578
+ this.sendExtensionRequest(this.REQ_GET_DATA, params);
2579
+
2580
+ // Schedule next request
2581
+ this._requestTimer = setTimeout($.proxy(this._requestData, this), Number(this._intervalDropDown.value()) * 1000);
2582
+ }
1180
2583
  }
1181
2584
 
1182
- _fillBackupCard(idSelector, detailsObj = null)
2585
+ _getMemObject(bytes)
1183
2586
  {
1184
- if (detailsObj == null)
1185
- $(idSelector + ' .backup-details').hide();
1186
- else
1187
- {
1188
- $(idSelector + ' .backup-details').show();
2587
+ return {
2588
+ data: bytes,
2589
+ label: Object(_utils_utilities__WEBPACK_IMPORTED_MODULE_1__["bytesToSize"])(bytes)
2590
+ };
2591
+ }
2592
+
2593
+ _setChartUpdatePaused(paused)
2594
+ {
2595
+ // Pause/reactivate realtime charts
2596
+ this._cpuUsageChart.paused = paused;
2597
+ this._threadCountChart.paused = paused;
2598
+ this._memoryChart.paused = paused;
2599
+ this._networkRTChart.paused = paused;
2600
+ this._sysWorkloadChart.paused = paused;
2601
+ this._outTcpMsgQueueChart.paused = paused;
2602
+ this._sysQueueChart.paused = paused;
2603
+ this._extQueueChart.paused = paused;
2604
+ }
1189
2605
 
1190
- $(idSelector + ' .backup-link').attr('href', detailsObj.url);
1191
- $(idSelector + ' .backup-date').text(detailsObj.date);
1192
- $(idSelector + ' .backup-time').text(detailsObj.time);
1193
- $(idSelector + ' .backup-size').text(detailsObj.size);
2606
+ _getThreadsGridDataSource(items, sort)
2607
+ {
2608
+ if (!items)
2609
+ items = '';
2610
+
2611
+ if (!sort)
2612
+ {
2613
+ sort = {
2614
+ field: 'cpuTime',
2615
+ dir: 'desc'
2616
+ };
1194
2617
  }
2618
+
2619
+ return new kendo.data.DataSource({
2620
+ schema: {
2621
+ model: {
2622
+ id: 'id'
2623
+ }
2624
+ },
2625
+ sort: sort,
2626
+ data: items
2627
+ });
2628
+ }
2629
+
2630
+ _getQueueThreadsLabel(threadsNum)
2631
+ {
2632
+ return `Threads: ${threadsNum}`;
2633
+ }
2634
+
2635
+ _getQueueLoadLabel(load)
2636
+ {
2637
+ return `Avg. messages: ${load}`;
1195
2638
  }
1196
2639
 
1197
- _setDownloadGridDataSource(ds)
2640
+ _updateOutMessagesQueueStatus()
1198
2641
  {
1199
- // Read current horizontal scroll value
1200
- const scrollLeft = $('#lgv-downloadGrid .k-grid-content', this._downloadGrid.wrapper).scrollLeft();
2642
+ let threadsStr = this._getQueueThreadsLabel('--');
2643
+ let loadStr = this._getQueueLoadLabel('--');
2644
+
2645
+ if ($('#dsh-tcpPill').hasClass('active'))
2646
+ {
2647
+ threadsStr = this._getQueueThreadsLabel(this._tcpOutThreads);
2648
+ loadStr = this._getQueueLoadLabel(this._tcpOutLoad);
2649
+ }
1201
2650
 
1202
- // Assign data source to grid
1203
- this._downloadGrid.setDataSource(ds);
2651
+ if ($('#dsh-udpPill').hasClass('active'))
2652
+ {
2653
+ threadsStr = this._getQueueThreadsLabel(this._udpOutThreads);
2654
+ loadStr = this._getQueueLoadLabel(this._udpOutLoad);
2655
+ }
1204
2656
 
1205
- // Set horizontal scroll
1206
- $('#lgv-downloadGrid .k-grid-content', this._downloadGrid.wrapper).scrollLeft(scrollLeft);
2657
+ $('#dsh-outgoingMessagesQueueThreads').text(threadsStr);
2658
+ $('#dsh-outgoingMessagesQueueLoad').text(loadStr);
1207
2659
  }
1208
2660
 
1209
2661
  //---------------------------------
@@ -1218,4 +2670,4 @@ class LogViewer extends _base_module__WEBPACK_IMPORTED_MODULE_0__["BaseModule"]
1218
2670
  /***/ })
1219
2671
 
1220
2672
  }]);
1221
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"assets/js/core/modules/module-8.bundle.js","sources":["webpack://application/./node_modules/file-saver/dist/FileSaver.min.js","webpack://application/(webpack)/buildin/global.js","webpack://application/./src/data/runtime-log-entry.js","webpack://application/./src/modules/log-viewer.js"],"sourcesContent":["(function(a,b){if(\"function\"==typeof define&&define.amd)define([],b);else if(\"undefined\"!=typeof exports)b();else{b(),a.FileSaver={exports:{}}.exports}})(this,function(){\"use strict\";function b(a,b){return\"undefined\"==typeof b?b={autoBom:!1}:\"object\"!=typeof b&&(console.warn(\"Deprecated: Expected third argument to be a object\"),b={autoBom:!b}),b.autoBom&&/^\\s*(?:text\\/\\S*|application\\/xml|\\S*\\/\\S*\\+xml)\\s*;.*charset\\s*=\\s*utf-8/i.test(a.type)?new Blob([\"\\uFEFF\",a],{type:a.type}):a}function c(b,c,d){var e=new XMLHttpRequest;e.open(\"GET\",b),e.responseType=\"blob\",e.onload=function(){a(e.response,c,d)},e.onerror=function(){console.error(\"could not download file\")},e.send()}function d(a){var b=new XMLHttpRequest;b.open(\"HEAD\",a,!1);try{b.send()}catch(a){}return 200<=b.status&&299>=b.status}function e(a){try{a.dispatchEvent(new MouseEvent(\"click\"))}catch(c){var b=document.createEvent(\"MouseEvents\");b.initMouseEvent(\"click\",!0,!0,window,0,0,0,80,20,!1,!1,!1,!1,0,null),a.dispatchEvent(b)}}var f=\"object\"==typeof window&&window.window===window?window:\"object\"==typeof self&&self.self===self?self:\"object\"==typeof global&&global.global===global?global:void 0,a=f.saveAs||(\"object\"!=typeof window||window!==f?function(){}:\"download\"in HTMLAnchorElement.prototype?function(b,g,h){var i=f.URL||f.webkitURL,j=document.createElement(\"a\");g=g||b.name||\"download\",j.download=g,j.rel=\"noopener\",\"string\"==typeof b?(j.href=b,j.origin===location.origin?e(j):d(j.href)?c(b,g,h):e(j,j.target=\"_blank\")):(j.href=i.createObjectURL(b),setTimeout(function(){i.revokeObjectURL(j.href)},4E4),setTimeout(function(){e(j)},0))}:\"msSaveOrOpenBlob\"in navigator?function(f,g,h){if(g=g||f.name||\"download\",\"string\"!=typeof f)navigator.msSaveOrOpenBlob(b(f,h),g);else if(d(f))c(f,g,h);else{var i=document.createElement(\"a\");i.href=f,i.target=\"_blank\",setTimeout(function(){e(i)})}}:function(a,b,d,e){if(e=e||open(\"\",\"_blank\"),e&&(e.document.title=e.document.body.innerText=\"downloading...\"),\"string\"==typeof a)return c(a,b,d);var g=\"application/octet-stream\"===a.type,h=/constructor/i.test(f.HTMLElement)||f.safari,i=/CriOS\\/[\\d]+/.test(navigator.userAgent);if((i||g&&h)&&\"object\"==typeof FileReader){var j=new FileReader;j.onloadend=function(){var a=j.result;a=i?a:a.replace(/^data:[^;]*;/,\"data:attachment/file;\"),e?e.location.href=a:location=a,e=null},j.readAsDataURL(a)}else{var k=f.URL||f.webkitURL,l=k.createObjectURL(a);e?e.location=l:location.href=l,e=null,setTimeout(function(){k.revokeObjectURL(l)},4E4)}});f.saveAs=a.saveAs=a,\"undefined\"!=typeof module&&(module.exports=a)});\n\n//# sourceMappingURL=FileSaver.min.js.map","var g;\n\n// This works in non-strict mode\ng = (function() {\n\treturn this;\n})();\n\ntry {\n\t// This works if eval is allowed (see CSP)\n\tg = g || new Function(\"return this\")();\n} catch (e) {\n\t// This works if the window reference is available\n\tif (typeof window === \"object\") g = window;\n}\n\n// g can still be undefined, but nothing to do about it...\n// We return undefined, instead of nothing here, so it's\n// easier to handle this case. if(!global) { ...}\n\nmodule.exports = g;\n","export class RuntimeLogEntry\n{\n\tconstructor(separator)\n\t{\n\t\tthis.sep = separator\n\t}\n\n\tstatic fromArray(separator, logEntryData)\n\t{\n\t\tlet rle = new RuntimeLogEntry(separator);\n\n\t\trle.date = logEntryData[0];\n\t\trle.time = logEntryData[1];\n\t\trle.dateTime = rle._getDateTime();\n\t\trle.level = logEntryData[2].trim();\n\t\trle.thread = logEntryData[3];\n\t\trle.clazz = logEntryData[4];\n\t\trle.message = logEntryData[6];\n\n\t\treturn rle;\n\t}\n\n\t_getDateTime()\n\t{\n\t\treturn this.date + '\\n' + this.time;\n\t}\n}\n","import {BaseModule} from './base-module';\nimport {RuntimeLogEntry} from '../data/runtime-log-entry';\nimport * as FileSaver from 'file-saver';\nimport * as moment from 'moment';\nimport {bytesToSize} from '../utils/utilities';\n\nexport default class LogViewer extends BaseModule\n{\n\tconstructor()\n\t{\n\t    super('logViewer');\n\n\t\tthis.COMMANDS_PREFIX = 'logViewer';\n\t\tthis.HELP_URL = '/admintool-LogViewer';\n\n\t\tthis.ANY_LEVEL = '[any]';\n\t\tthis.LEVELS = ['TRACE','DEBUG','INFO','WARN','ERROR'];\n\t\tthis.ANY_CLASS = '[any]';\n\n\t\tthis.LOG_DATA_LABEL = 'lines';\n\n\t\tthis.BOOT_LOG_BACKUP_ID = 'SFS2X_BootLog';\n\t\tthis.RUNTIME_LOG_BACKUP_ID = 'SFS2X_RuntimeLog';\n\t\tthis.FULL_BACKUP_ID = 'SFS2X_Logs';\n\n\t\t// Outgoing requests\n\t\tthis.REQ_GET_RUNTIME_LOG = 'getRunLog';\n\t\tthis.REQ_GET_BOOT_LOG = 'getBootLog';\n\t\tthis.REQ_GET_BACKUPS_STATUS = 'getBakStatus';\n\t\tthis.REQ_BACKUP_BOOT_LOG = 'bakBootLog';\n\t\tthis.REQ_BACKUP_RUNTIME_LOG = 'bakRunLog';\n\t\tthis.REQ_BACKUP_FULL_LOGS = 'bakFullLogs';\n\t\tthis.REQ_DELETE_BACKUP = 'delBackup';\n\n\t\t// Incoming responses\n\t\tthis.RESP_INIT_ERROR = 'initErr';\n\t\tthis.RESP_RUNTIME_LOG_ERROR = 'runLogErr';\n\t\tthis.RESP_RUNTIME_LOG_INVALID = 'runLogInv';\n\t\tthis.RESP_RUNTIME_LOG = 'runLog';\n\t\tthis.RESP_BOOT_LOG_ERROR = 'bootLogErr';\n\t\tthis.RESP_BOOT_LOG = 'bootLog';\n\t\tthis.RESP_BACKUPS_STATUS = 'bakStatus';\n\t\tthis.RESP_DELETE_BACKUP_FAILED = 'delBakFail';\n\t\tthis.RESP_BACKUP_ERROR = 'bakError';\n\t\tthis.RESP_BACKUP_WARNING = 'bakWarn';\n\t}\n\n\t//------------------------------------\n\t// COMMON MODULE INTERFACE METHODS\n\t// This members are used by the main controller\n\t// to communicate with the module's controller.\n\t//------------------------------------\n\n\tinitialize(idData, shellController)\n\t{\n\t\t// Call super method\n\t\tsuper.initialize(idData, shellController);\n\n\t\t// Initialize scrolling tabs\n\t\t$('#lgv-tabNavigator > #tabs').scrollingTabs({\n\t\t\tbootstrapVersion: 4,\n\t\t\tscrollToTabEdge: true,\n\t\t\tenableSwiping: true,\n\t\t\tdisableScrollArrowsOnFullyScrolled: true,\n\t\t\tcssClassLeftArrow: 'fa fa-chevron-left',\n\t\t\tcssClassRightArrow: 'fa fa-chevron-right'\n\t\t});\n\n\t\t// Add listener to tab shown event\n\t\t$('a[data-toggle=\"tab\"]').on('shown.bs.tab', $.proxy(this._onTabShown, this));\n\n\t\t// Initialize log lines dropdown\n\t\tthis._logLinesDD = $('#lgv-logLinesDD').kendoDropDownList({\n\t\t\tvalueTemplate: '<span class=\"text-muted pr-1\">Log entries:</span><span>#:data.text#</span>',\n\t\t}).data('kendoDropDownList');\n\n\t\t// Initialize load button\n\t\t$('#lgv-loadBt').on('click', $.proxy(this._onRuntimeLogLoadBtClick, this));\n\n\t\t// Initialize progress bars\n\t\t$('.progress-bar').kendoProgressBar({\n\t\t\tmin: 0,\n            max: 100,\n\t\t\tvalue: false,\n            type: 'value',\n            animation: {\n                duration: 400\n            }\n        });\n\n\t\t// Initialize level filter dropdown\n\t\tthis._levelFilterDD = $('#lgv-levelDD').kendoDropDownList({\n\t\t\tdataSource: [this.ANY_LEVEL].concat(this.LEVELS),\n\t\t\tchange: $.proxy(this._onFilterChange, this)\n\t\t}).data('kendoDropDownList');\n\n\t\t// Initialize class filter dropdown\n\t\tthis._classFilterDD = $('#lgv-classDD').kendoDropDownList({\n\t\t\tchange: $.proxy(this._onFilterChange, this)\n\t\t}).data('kendoDropDownList');\n\n\t\t// Initialize message filter input\n\t\t$('#lgv-messageIn').on('input', $.proxy(this._onFilterChange, this));\n\n\t\t// Initialize clear button\n\t\t$('#lgv-clearFilterBt').on('click', $.proxy(this._onClearFilterClick, this));\n\n\t\t// Initialize export button\n\t\t$('#lgv-exportRuntimeLogBt').on('click', $.proxy(this._onExportRuntimeLogBtClick, this));\n\n\t\t// Initialize runtime log grid\n\t\tthis._runtimeLogGrid = $('#lgv-runtimeLogGrid').kendoGrid({\n\t\t\tscrollable: true,\n            sortable: false,\n\t\t\t//resizable: true,\n\t\t\tselectable: false,\n            columns:\n            [\n\t\t\t\t{\n\t                field: 'dateTime',\n\t                width: 150,\n\t\t\t\t\ttitle: 'Date/Time',\n\t            },\n\t            {\n\t                field: 'level',\n\t                width: 100,\n\t                title: 'Level',\n\t            },\n\t            {\n\t                field: 'thread',\n\t                width: 150,\n\t                title: 'Thread',\n\t            },\n\t            {\n\t                field: 'clazz',\n\t                width: 250,\n\t                title: 'Class',\n\t            },\n\t\t\t\t{\n\t                field: 'message',\n\t                width: 1000,\n\t\t\t\t\ttitle: 'Message',\n\t            },\n\t\t\t],\n\t\t\tnoRecords: {\n\t\t\t\ttemplate: 'No log entries to display.'\n\t\t\t},\n\t\t\tdataSource: []\n        }).data('kendoGrid');\n\n\t\t// Initialize boot log view buttons\n\t\t$('#lgv-exportBootLogBt').on('click', $.proxy(this._onExportBootLogBtClick, this));\n\t\t$('#lgv-switchBootLogColorBt').on('click', $.proxy(this._onSwitchBootLogColorBtClick, this));\n\n\t\t// Initialize generate backup buttons\n\t\t$('#lgv-bootLogBackupCard .backup-button').on('click', $.proxy(this._onBootLogGenerateBtClick, this));\n\t\t$('#lgv-runtimeLogBackupCard .backup-button').on('click', $.proxy(this._onRuntimeLogGenerateBtClick, this));\n\t\t$('#lgv-fullLogBackupCard .backup-button').on('click', $.proxy(this._onFullLogsGenerateBtClick, this));\n\n\t\tthis._initBackupCard('#lgv-bootLogBackupCard');\n\t\tthis._initBackupCard('#lgv-runtimeLogBackupCard');\n\t\tthis._initBackupCard('#lgv-fullLogBackupCard');\n\n\t\t// Initialize download grid\n\t\tthis._downloadGrid = $('#lgv-downloadGrid').kendoGrid({\n\t\t\tscrollable: true,\n            sortable: true,\n\t\t\t//resizable: true,\n\t\t\tselectable: 'row',\n            columns:\n            [\n\t\t\t\t{\n\t                field: 'date',\n\t                width: 100,\n\t\t\t\t\ttitle: 'Date',\n\t            },\n\t            {\n\t                field: 'time',\n\t                width: 100,\n\t                title: 'Time',\n\t            },\n\t            {\n\t                field: 'name',\n\t                width: 300,\n\t                title: 'Filename',\n\t            },\n\t            {\n\t                field: 'size',\n\t                width: 100,\n\t                title: 'Size',\n\t            },\n\t\t\t],\n\t\t\tnoRecords: {\n\t\t\t\ttemplate: 'No backups available.'\n\t\t\t},\n\t\t\tchange: $.proxy(this._onDownloadGridSelectionChange, this),\n\t\t\tdataSource: []\n        }).data('kendoGrid');\n\n\t\t// Initialize delete button\n\t\t$('#lgv-deleteBt').on('click', $.proxy(this._onDeleteBtClick, this));\n\t}\n\n\tdestroy()\n\t{\n\t\t// Call super method\n\t\tsuper.destroy();\n\n\t\t// Remove listener to tab shown event\n\t\t$('a[data-toggle=\"tab\"]').off('shown.bs.tab');\n\n\t\t// Destroy scrolling tabs\n\t\t$('#lgv-tabNavigator #tabs').scrollingTabs('destroy');\n\n\t\t// Remove click listeners\n\t\t$('#lgv-loadBt').off('click');\n\t\t$('#lgv-exportBootLogBt').off('click');\n\t\t$('#lgv-switchBootLogColorBt').off('click');\n\t\t$('#lgv-bootLogBackupCard .backup-button').off('click');\n\t\t$('#lgv-runtimeLogBackupCard .backup-button').off('click');\n\t\t$('#lgv-clearFilterBt').off('click');\n\t\t$('#lgv-deleteBt').off('click');\n\n\t\t$('#lgv-messageIn').off('input');\n\t}\n\n\tonExtensionCommand(command, data)\n\t{\n\t\t// Error during initialization (unable to access log4j configuration file)\n\t\tif (command == this.RESP_INIT_ERROR)\n\t\t{\n\t\t\tconst error = data.getUtfString('error');\n\n\t\t\t// Show an alert\n\t\t\tthis.shellCtrl.showSimpleAlert(error, true);\n\n\t\t\t// Set all tabs to show errors\n\t\t\tthis._switchBootViewStack('lgv-bootLogErrorView');\n\t\t\tthis._switchRuntimeViewStack('lgv-runtimeLogErrorView');\n\t\t\tthis._switchDownloadViewStack('lgv-downloadErrorView');\n\n\t\t\t// Disable runtime log controls\n\t\t\tthis._enableRuntimeLogControls(false);\n\n\t\t\tthis._initFailed = true;\n\t\t}\n\n\t\t// Error responses\n\t\telse if (command == this.RESP_BOOT_LOG_ERROR || command == this.RESP_RUNTIME_LOG_ERROR)\n\t\t{\n\t\t\tconst error = data.getUtfString('error');\n\n\t\t\t// Show an alert\n\t\t\tthis.shellCtrl.showSimpleAlert(error, true);\n\n\t\t\tif (command == this.RESP_BOOT_LOG_ERROR)\n\t\t\t{\n\t\t\t\t// Set tab to show error\n\t\t\t\tthis._switchBootViewStack('lgv-bootLogErrorView');\n\n\t\t\t\t// Disable boot log backup generation\n\t\t\t\tthis._bootLogBackupUnavailable = true;\n\t\t\t\tthis._disableBackupInterface(false);\n\n\t\t\t\tthis._bootLogRequested = true;\n\t\t\t}\n\n\t\t\tif (command == this.RESP_RUNTIME_LOG_ERROR)\n\t\t\t{\n\t\t\t\t// Disable controls\n\t\t\t\tthis._enableRuntimeLogControls(false);\n\n\t\t\t\t// Remove loading bar (runtime log)\n\t\t\t\tthis._switchRuntimeViewStack('lgv-runtimeLogErrorView');\n\n\t\t\t\t// Disable runtime log backup download\n\t\t\t\tthis._runtimeLogBackupUnavailable = true;\n\t\t\t\tthis._disableBackupInterface(false);\n\t\t\t}\n\t\t}\n\n\t\t// Modified conversion pattern in the log4j properties file: unable to parse the log\n\t\telse if (command == this.RESP_RUNTIME_LOG_INVALID)\n\t\t{\n\t\t\t// Disable controls\n\t\t\tthis._enableRuntimeLogControls(false);\n\n\t\t\t// Fill in error message\n\t\t\t$('#lgv-convPattName').text(data.getUtfString('param'));\n\t\t\t$('#lgv-convPattVal').text(data.getUtfString('value'));\n\n\t\t\t// Remove loading bar (runtime log)\n\t\t\tthis._switchRuntimeViewStack('lgv-invConvPattView');\n\t\t}\n\n\t\t// Runtime log received\n\t\telse if (command == this.RESP_RUNTIME_LOG)\n\t\t{\n\t\t\tlet classes = [];\n\t\t\tclasses.push(this.ANY_CLASS);\n\n\t\t\tlet logEntries = data.getUtfStringArray(this.LOG_DATA_LABEL);\n\t\t\tlet separator = data.getUtfString('sep');\n\t\t\tlet columns = data.getInt('cols');\n\t\t\tlet dsArr = [];\n\n\t\t\tthis._totalRuntimeLogEntries = logEntries.length;\n\n\t\t\t// Parse log entries\n\t\t\t// We can't use the split method because there could be instances of the separator in the log message too\n\t\t\tfor (let e = 0; e < logEntries.length; e++)\n\t\t\t{\n\t\t\t\tconst logEntry = logEntries[e];\n\n\t\t\t\tlet logEntryData = [];\n\t\t\t\tlet startIndex = 0;\n\n\t\t\t\tfor (let c = 0; c < columns - 1; c++)\n\t\t\t\t{\n\t\t\t\t\tlet endIndex = logEntry.indexOf(separator, startIndex);\n\t\t\t\t\tlogEntryData.push(logEntry.substring(startIndex, endIndex));\n\t\t\t\t\tstartIndex = endIndex + separator.length;\n\t\t\t\t}\n\n\t\t\t\tif (startIndex < logEntry.length)\n\t\t\t\t\tlogEntryData.push(logEntry.substring(startIndex));\n\n\t\t\t\t// Fill datagrid's dataprovider\n\t\t\t\tlet rle = RuntimeLogEntry.fromArray(separator, logEntryData);\n\t\t\t\tdsArr.push(rle);\n\n\t\t\t\t// Add class to filtering dropdown\n\t\t\t\tif (classes.indexOf(rle.clazz) < 0)\n\t\t\t\t\tclasses.push(rle.clazz);\n\t\t\t}\n\n\t\t\t// Show classes list\n\t\t\tclasses.sort(function (a, b) {\n\t\t\t\treturn a.localeCompare(b);\n\t\t\t});\n\n\t\t\tthis._classFilterDD.setDataSource(classes);\n\t\t\tthis._classFilterDD.select(0);\n\n\t\t\t// Assign data source to grid\n\t\t\tlet ds = new kendo.data.DataSource({\n\t\t\t\tdata: dsArr\n\t\t\t})\n\n\t\t\tthis._setRuntimeLogDataSource(ds);\n\n\t\t\t// Re-enable log loading controls\n\t\t\tthis._enableRuntimeLogControls(true);\n\n\t\t\t// Remove loading bar\n\t\t\tthis._switchRuntimeViewStack('lgv-runtimeLogView');\n\t\t}\n\n\t\t// Boot log received\n\t\telse if (command == this.RESP_BOOT_LOG)\n\t\t{\n\t\t\tconst bootLogEntries = data.getSFSArray(this.LOG_DATA_LABEL);\n\t\t\tlet text = '';\n\n\t\t\tfor (let i = 0; i < bootLogEntries.size(); i++)\n\t\t\t\ttext += bootLogEntries.getUtfString(i) + '\\n';\n\n\t\t\t$('#lgv-bootLogText').text(text);\n\n\t\t\t// Remove loading bar\n\t\t\tthis._switchBootViewStack('lgv-bootLogView');\n\t\t}\n\n\t\t// Logs backups status received\n\t\telse if (command == this.RESP_BACKUPS_STATUS)\n\t\t{\n\t\t\t// Show/hide operation in progress message\n\t\t\tthis._disableBackupInterface(data.getBool('running'), data.getUtfString('type'));\n\n\t\t\t// Backup files list\n\t\t\tif (data.containsKey('files'))\n\t\t\t{\n\t\t\t\tlet files = data.getSFSArray('files');\n\n\t\t\t\tlet lastBootLogBackupFound = false;\n\t\t\t\tlet lastRuntimeLogBackupFound = false;\n\t\t\t\tlet lastFullBackupFound = false;\n\n\t\t\t\tlet backupsList = [];\n\n\t\t\t\tconst webServerProtocol = (data.containsKey('protocol') ? data.getUtfString('protocol') : 'http') + '://';\n\t\t\t\tconst webServerPort = (data.containsKey('port') ? ':' + data.getInt('port') : '');\n\n\t\t\t\tlet totalSize = 0;\n\n\t\t\t\tfor (let f = 0; f < files.size(); f++)\n\t\t\t\t{\n\t\t\t\t\tconst file = files.getSFSObject(f);\n\n\t\t\t\t\tconst filePath = file.getUtfString('path');\n\n\t\t\t\t\tconst fileObj = {};\n\t\t\t\t\tfileObj.path = filePath;\n\t\t\t\t\tfileObj.url = webServerProtocol + this.smartFox.config.host + webServerPort + '/' + filePath;\n\t\t\t\t\tfileObj.name = filePath.substr(filePath.lastIndexOf('/') + 1);\n\t\t\t\t\tfileObj.date = file.getUtfString('date');\n\t\t\t\t\tfileObj.time = file.getUtfString('time');\n\t\t\t\t\tfileObj.size = bytesToSize(file.getLong('size'), 2);\n\n\t\t\t\t\ttotalSize += file.getLong('size');\n\n\t\t\t\t\t// Check if this is a boot log backup\n\t\t\t\t\tif (!lastBootLogBackupFound)\n\t\t\t\t\t{\n\t\t\t\t\t\tif (fileObj.name.startsWith(this.BOOT_LOG_BACKUP_ID))\n\t\t\t\t\t\t{\n\t\t\t\t\t\t\tthis._fillBackupCard('#lgv-bootLogBackupCard', fileObj);\n\n\t\t\t\t\t\t\tlastBootLogBackupFound = true;\n\t\t\t\t\t\t}\n\t\t\t\t\t}\n\n\t\t\t\t\t// Check if this is a runtime log backup\n\t\t\t\t\tif (!lastRuntimeLogBackupFound)\n\t\t\t\t\t{\n\t\t\t\t\t\tif (fileObj.name.startsWith(this.RUNTIME_LOG_BACKUP_ID))\n\t\t\t\t\t\t{\n\t\t\t\t\t\t\tthis._fillBackupCard('#lgv-runtimeLogBackupCard', fileObj);\n\n\t\t\t\t\t\t\tlastRuntimeLogBackupFound = true;\n\t\t\t\t\t\t}\n\t\t\t\t\t}\n\n\t\t\t\t\t// Check if this is a full backup\n\t\t\t\t\tif (!lastFullBackupFound)\n\t\t\t\t\t{\n\t\t\t\t\t\tif (fileObj.name.startsWith(this.FULL_BACKUP_ID))\n\t\t\t\t\t\t{\n\t\t\t\t\t\t\tthis._fillBackupCard('#lgv-fullLogBackupCard', fileObj);\n\n\t\t\t\t\t\t\tlastFullBackupFound = true;\n\t\t\t\t\t\t}\n\t\t\t\t\t}\n\n\t\t\t\t\t// Populate logs list\n\t\t\t\t\tbackupsList.push(fileObj);\n\t\t\t\t}\n\n\t\t\t\t// Show total backups size\n\t\t\t\t$('#lgv-logSizeLb').html(`Total size: <strong>${bytesToSize(totalSize, 2, 'KB')}</strong>`);\n\n\t\t   \t\t// Assign data source to grid\n\t\t\t\tthis._setDownloadGridDataSource(backupsList);\n\t\t\t\tthis._onDownloadGridSelectionChange();\n\n\t\t\t\t// Hide links to latest files if not available\n\t\t\t\tif (!lastBootLogBackupFound)\n\t\t\t\t\tthis._fillBackupCard('#lgv-bootLogBackupCard', null);\n\n\t\t\t\tif (!lastRuntimeLogBackupFound)\n\t\t\t\t\tthis._fillBackupCard('#lgv-runtimeLogBackupCard', null);\n\n\t\t\t\tif (!lastFullBackupFound)\n\t\t\t\t\tthis._fillBackupCard('#lgv-fullLogBackupCard', null);\n\n\t\t\t\tif (data.containsKey('message'))\n\t\t\t\t{\n\t\t\t\t\t// Display notification\n\t\t\t\t\tthis.shellCtrl.showNotification(`Log backup warning`, data.getUtfString('message'));\n\t\t\t\t}\n\t\t\t}\n\n\t\t\t// Set download view to main\n\t\t\tthis._switchDownloadViewStack('lgv-downloadView');\n\t\t}\n\n\t\t// Logs backup deletion failed\n\t\telse if (command == this.RESP_DELETE_BACKUP_FAILED)\n\t\t{\n\t\t\tconst error = data.getUtfString('error');\n\n\t\t\t// Show an alert\n\t\t\tthis.shellCtrl.showSimpleAlert(error, true);\n\t\t}\n\n\t\t// A blocking error occurred during backup operation\n\t\telse if (command == this.RESP_BACKUP_ERROR)\n\t\t{\n\t\t\tconst error = data.getUtfString('error');\n\n\t\t\t// Show an alert\n\t\t\tthis.shellCtrl.showSimpleAlert(error, true);\n\t\t}\n\n\t\t// An non-blocking error occurred during backup operation\n\t\telse if (command == this.RESP_BACKUP_WARNING)\n\t\t{\n\t\t\tlet warn = data.getUtfString('warn');\n\n\t\t\t// Display notification\n\t\t\tthis.shellCtrl.showNotification(`Log backup warning`, warn);\n\t\t}\n\t}\n\n\t//---------------------------------\n\t// UI EVENT LISTENERS\n\t//---------------------------------\n\n\t_onTabShown(e)\n\t{\n\t\tif (!this._initFailed)\n\t\t{\n\t\t\t// If boot log view was displayed...\n\t\t\tif (e.target.id == 'lgv-bootLog-tab')\n\t\t\t{\n\t\t\t\t// Load boot log the first time the tab is selected\n\t\t\t\tif (!this._bootLogRequested)\n\t\t\t\t{\n\t\t\t\t\tthis.sendExtensionRequest(this.REQ_GET_BOOT_LOG);\n\t\t\t\t\tthis._bootLogRequested = true;\n\t\t\t\t}\n\t\t\t}\n\n\t\t\t// If backup&dowload view was displayed...\n\t\t\telse if (e.target.id == 'lgv-logsDownload-tab')\n\t\t\t{\n\t\t\t\t// Request logs backup status the first time the tab is selected\n\t\t\t\tif (!this._backupStatusRequested)\n\t\t\t\t{\n\t\t\t\t\tthis.sendExtensionRequest(this.REQ_GET_BACKUPS_STATUS);\n\t\t\t\t\tthis._backupStatusRequested = true;\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\t}\n\n\t_onRuntimeLogLoadBtClick()\n\t{\n\t\t// Request log\n\t\tthis._loadRuntimeLog();\n\t}\n\n\t_onExportBootLogBtClick()\n\t{\n\t\t// Export log to file\n\t\tthis._exportLog($('#lgv-bootLogText').text(), this.BOOT_LOG_BACKUP_ID);\n\t}\n\n\t_onSwitchBootLogColorBtClick()\n\t{\n\t\tif ($('#lgv-bootLogText').hasClass('invert'))\n\t\t\t$('#lgv-bootLogText').removeClass('invert');\n\t\telse\n\t\t\t$('#lgv-bootLogText').addClass('invert');\n\t}\n\n\t_onBootLogGenerateBtClick()\n\t{\n\t\t// Show/hide operation in progress message\n\t\tthis._disableBackupInterface(true, this.BOOT_LOG_BACKUP_ID);\n\n\t\t// Request backup generation\n\t\tthis.sendExtensionRequest(this.REQ_BACKUP_BOOT_LOG);\n\t}\n\n\t_onRuntimeLogGenerateBtClick()\n\t{\n\t\t// Show/hide operation in progress message\n\t\tthis._disableBackupInterface(true, this.RUNTIME_LOG_BACKUP_ID);\n\n\t\t// Request backup generation\n\t\tthis.sendExtensionRequest(this.REQ_BACKUP_RUNTIME_LOG);\n\t}\n\n\t_onFullLogsGenerateBtClick()\n\t{\n\t\t// Show/hide operation in progress message\n\t\tthis._disableBackupInterface(true, this.FULL_BACKUP_ID);\n\n\t\t// Request backup generation\n\t\tthis.sendExtensionRequest(this.REQ_BACKUP_FULL_LOGS);\n\t}\n\n\t_onFilterChange()\n\t{\n\t\t// Set filters\n\t\tthis._setRuntimeLogDataSource(this._runtimeLogGrid.dataSource);\n\t}\n\n\t_onClearFilterClick()\n\t{\n\t\tthis._clearRuntimeLogFilters();\n\t\tthis._setRuntimeLogDataSource(this._runtimeLogGrid.dataSource);\n\t}\n\n\t_onExportRuntimeLogBtClick()\n\t{\n\t\tlet log = '';\n\t\tconst entries = this._runtimeLogGrid.dataSource.view();\n\n\t\tfor (let i = 0; i < entries.length; i++)\n\t\t{\n\t\t\tconst item = entries[i];\n\t\t\tlog += [item.date, item.time, item.level, item.thread, item.clazz, item.message].join(item.sep) + '\\n';\n\t\t}\n\n\t\t// Export log to file\n\t\tthis._exportLog(log, this.RUNTIME_LOG_BACKUP_ID);\n\t}\n\n\t_onDownloadGridSelectionChange()\n\t{\n\t\t// Enable/disable buttons\n\t\tconst selectedRows = this._downloadGrid.select();\n\t\t$('#lgv-downloadBt').attr('disabled', selectedRows.length == 0);\n\t\t$('#lgv-deleteBt').attr('disabled', selectedRows.length == 0);\n\n\t\tif (selectedRows.length > 0)\n\t\t{\n\t\t\tlet dataItem = this._downloadGrid.dataItem(selectedRows[0]);\n\t\t\t$('#lgv-downloadBt').attr('href', dataItem.url);\n\t\t}\n\t\telse\n\t\t\t$('#lgv-downloadBt').attr('href', '#');\n\t}\n\n\t_onDeleteBtClick()\n\t{\n\t\tlet selectedRows = this._downloadGrid.select();\n\n\t\tif (selectedRows.length > 0)\n\t\t{\n\t\t\tlet dataItem = this._downloadGrid.dataItem(selectedRows[0]);\n\n\t\t\t// Request backup deletion\n\t\t\tlet params = new SFS2X.SFSObject();\n\t\t\tparams.putUtfString('file', dataItem.name);\n\n\t\t\tthis.sendExtensionRequest(this.REQ_DELETE_BACKUP, params);\n\t\t}\n\t}\n\n\t//------------------------------------\n\t// PRIVATE METHODS\n\t//------------------------------------\n\n\t_switchRuntimeViewStack(viewId)\n\t{\n\t\tdocument.getElementById('lgv-runtime-viewstack').selectedElement = document.getElementById(viewId);\n\t}\n\n\t_switchBootViewStack(viewId)\n\t{\n\t\tdocument.getElementById('lgv-boot-viewstack').selectedElement = document.getElementById(viewId);\n\t}\n\n\t_switchDownloadViewStack(viewId)\n\t{\n\t\tdocument.getElementById('lgv-download-viewstack').selectedElement = document.getElementById(viewId);\n\t}\n\n\t_enableRuntimeLogControls(enable)\n\t{\n\t\t$('#lgv-loadBt').attr('disabled', !enable);\n\t\t$('#lgv-exportRuntimeLogBt').attr('disabled', !enable);\n\t\t$('#lgv-filterBt').attr('disabled', !enable);\n\t}\n\n\t_loadRuntimeLog()\n\t{\n\t\t// Disable controls to load log, so that impatient users can't send multiple requests\n\t\t// (it will be enabled again when a response is received)\n\t\tthis._enableRuntimeLogControls(false);\n\n\t\t// Clear filters\n\t\tthis._clearRuntimeLogFilters();\n\n\t\t// Show loading bar\n\t\tthis._switchRuntimeViewStack('lgv-runtimeLogLoadingView');\n\n\t\t// Send request\n\t\t// (the number of lines to be retrieved is sent)\n\t\tlet params = new SFS2X.SFSObject();\n\t\tparams.putInt('numEntries', Number(this._logLinesDD.value()));\n\n\t\tthis.sendExtensionRequest(this.REQ_GET_RUNTIME_LOG, params);\n\t}\n\n\t_clearRuntimeLogFilters()\n\t{\n\t\tthis._levelFilterDD.select(0);\n\t\tthis._classFilterDD.select(0);\n\t\t$('#lgv-messageIn').val('');\n\t}\n\n\t_exportLog(log, name)\n\t{\n\t\tlet blob = new Blob([log], {type: \"text/plain;charset=utf-8\"});\n\t\tlet date = moment().format('YYYYMMDD_HHmmss');\n\n\t\tFileSaver.saveAs(blob, `${name}_${date}.log`);\n\t}\n\n\t_setRuntimeLogDataSource(ds)\n\t{\n\t\t// Read current horizontal scroll value\n\t\tconst scrollLeft = $('#lgv-runtimeLogGrid .k-grid-content', this._runtimeLogGrid.wrapper).scrollLeft();\n\n\t\t// Assign data source to grid\n\t\tthis._runtimeLogGrid.setDataSource(ds);\n\n\t\t// Set filters\n\t\tthis._setFilters(ds);\n\n\t\t// Set horizontal scroll\n\t\t$('#lgv-runtimeLogGrid .k-grid-content', this._runtimeLogGrid.wrapper).scrollLeft(scrollLeft);\n\n\t\t// Update counter\n\t\t$('#lgv-runtimeLogEntriesLb').text(`Log entries: ${this._totalRuntimeLogEntries} (${ds.total()} displayed)`);\n\t}\n\n\t_setFilters(ds)\n\t{\n\t\tlet filters = [];\n\n\t\t// Level filtering\n\t\tif (this._levelFilterDD.select() > 0)\n\t\t\tfilters.push({\n\t\t\t\tfield: 'level', operator: 'eq', value: this._levelFilterDD.value()\n\t\t\t});\n\n\t\t// Class filtering\n\t\tif (this._classFilterDD.select() > 0)\n\t\t\tfilters.push({\n\t\t\t\tfield: 'clazz', operator: 'eq', value: this._classFilterDD.value()\n\t\t\t});\n\n\t\t// Message filtering\n\t\tif ($('#lgv-messageIn').val() != '')\n\t\t\tfilters.push({\n\t\t\t\tfield: 'message', operator: 'contains', value: $('#lgv-messageIn').val()\n\t\t\t});\n\n\t\t// Set filters\n\t\tds.filter(filters);\n\t}\n\n\t_disableBackupInterface(disable, backupId = null)\n\t{\n\t\tif (disable)\n\t\t{\n\t\t\t// Show proper progress bar\n\t\t\tif (backupId == this.BOOT_LOG_BACKUP_ID)\n\t\t\t\t$('#lgv-bootLogBackupCard .progress-bar').show();\n\t\t\telse if (backupId == this.RUNTIME_LOG_BACKUP_ID)\n\t\t\t\t$('#lgv-runtimeLogBackupCard .progress-bar').show();\n\t\t\telse if (backupId == this.FULL_BACKUP_ID)\n\t\t\t\t$('#lgv-fullLogBackupCard .progress-bar').show();\n\n\t\t\t// Disable buttons\n\t\t\t$('#lgv-bootLogBackupCard .backup-button').attr('disabled', true);\n\t\t\t$('#lgv-runtimeLogBackupCard .backup-button').attr('disabled', true);\n\t\t\t$('#lgv-fullLogBackupCard .backup-button').attr('disabled', true);\n\t\t}\n\t\telse\n\t\t{\n\t\t\t// Hide all progress bar\n\t\t\t$('.card-body .progress-bar').hide();\n\n\t\t\t// Enable buttons\n\t\t\t$('#lgv-fullLogBackupCard .backup-button').attr('disabled', false);\n\n\t\t\tif (!this._bootLogBackupUnavailable)\n\t\t\t\t$('#lgv-bootLogBackupCard .backup-button').attr('disabled', false);\n\n\t\t\tif (!this._runtimeLogBackupUnavailable)\n\t\t\t\t$('#lgv-runtimeLogBackupCard .backup-button').attr('disabled', false);\n\t\t}\n\t}\n\n\t_initBackupCard(idSelector)\n\t{\n\t\t$(idSelector + ' .backup-details').hide();\n\t\t$(idSelector + ' .progress-bar').hide();\n\t}\n\n\t_fillBackupCard(idSelector, detailsObj = null)\n\t{\n\t\tif (detailsObj == null)\n\t\t\t$(idSelector + ' .backup-details').hide();\n\t\telse\n\t\t{\n\t\t\t$(idSelector + ' .backup-details').show();\n\n\t\t\t$(idSelector + ' .backup-link').attr('href', detailsObj.url);\n\t\t\t$(idSelector + ' .backup-date').text(detailsObj.date);\n\t\t\t$(idSelector + ' .backup-time').text(detailsObj.time);\n\t\t\t$(idSelector + ' .backup-size').text(detailsObj.size);\n\t\t}\n\t}\n\n\t_setDownloadGridDataSource(ds)\n\t{\n\t\t// Read current horizontal scroll value\n\t   const scrollLeft = $('#lgv-downloadGrid .k-grid-content', this._downloadGrid.wrapper).scrollLeft();\n\n\t   // Assign data source to grid\n\t   this._downloadGrid.setDataSource(ds);\n\n\t   // Set horizontal scroll\n\t   $('#lgv-downloadGrid .k-grid-content', this._downloadGrid.wrapper).scrollLeft(scrollLeft);\n\t}\n\n\t//---------------------------------\n\t// PRIVATE GETTERS\n\t//---------------------------------\n\n\n}\n"],"mappings":";;;;;;;;;;AAAA;AAAA;AAAA;AAAA;AACA;AACA;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;ACFA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;ACnBA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;AC1BA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;A","sourceRoot":""}
2673
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"assets/js/core/modules/module-8.bundle.js","sources":["webpack://application/./src/components/charts/base-chart.js","webpack://application/./src/components/charts/chart-utils.js","webpack://application/./src/components/charts/cpu-usage-chart.js","webpack://application/./src/components/charts/memory-chart.js","webpack://application/./src/components/charts/network-24h-chart.js","webpack://application/./src/components/charts/network-realtime-chart.js","webpack://application/./src/components/charts/system-queue-chart.js","webpack://application/./src/components/charts/system-workload-chart.js","webpack://application/./src/components/charts/thread-count-chart.js","webpack://application/./src/modules/dashboard.js"],"sourcesContent":["export class BaseChart extends HTMLElement\n{\n\tconstructor(maxValues)\n\t{\n\t    super();\n\n\t\tthis._maxValues = maxValues;\n\n\t\t// Initialize\n\t\tthis._initialize();\n\t}\n\n\t// Can be overridden\n\t_initialize()\n\t{\n\t\t// Create chart html\n\t\tlet chartHtml = $('<div>');\n\t\t$(this).append(chartHtml);\n\n\t\tthis._chartWidget = this._initChartWidget(chartHtml);\n\n\t\t// Reset data source\n\t\tthis.reset();\n\t}\n\n\t// MUST BE OVERRIDDEN\n\t_initChartWidget()\n\t{\n\t\tthrow new Error('Chart widget must be initialized!')\n\t}\n\n\tset paused(value)\n\t{\n\t\tthis.isPaused = value;\n\t\tthis._onPauseChange();\n\t}\n\n\tget paused()\n\t{\n\t\treturn this.isPaused;\n\t}\n\n\treset()\n\t{\n\t\tthis._chartWidget.setDataSource([]);\n\t}\n\n\tredraw()\n\t{\n\t\tthis._chartWidget.redraw();\n\t}\n\n\tgetLastEntry()\n\t{\n\t\tlet ds;\n\n\t\tif (this.isPaused)\n\t\t\tds = this._tempDS;\n\t\telse\n\t\t\tds = this._dataSource;\n\n\t\tif (ds != null && ds.total() > 0)\n\t\t\treturn ds.at(ds.total() - 1);\n\t\telse\n\t\t\treturn null;\n\t}\n\n\t_addEntry(newEntry)\n\t{\n\t\tif (this.isPaused)\n\t\t{\n\t\t\tif (this._tempDS == null)\n\t\t\t\tthis._tempDS = new kendo.data.DataSource();\n\n\t\t\t// Store new value in a temporary array\n\t\t\tthis._tempDS.add(newEntry);\n\t\t}\n\t\telse\n\t\t{\n\t\t\tlet ds = this._dataSource;\n\n\t\t\tif (this._tempDS != null)\n\t\t\t{\n\t\t\t\t// Concatenate temporary entries to actual data source\n\t\t\t\tlet newArr = $.merge(ds.data(), this._tempDS.data());\n\n\t\t\t\t// Remove values exceeding MAX_VALUES\n\t\t\t\tif (newArr.length > this._maxValues)\n\t\t\t\t\tnewArr.splice(0, newArr.length - this._maxValues)\n\n\t\t\t\t// Create new datasource\n\t\t\t\tlet newDS = new kendo.data.DataSource({data: newArr})\n\n\t\t\t\t// Assign new data source to chart\n\t\t\t\tthis._chartWidget.setDataSource(newDS);\n\n\t\t\t\t// Remove temporary array\n\t\t\t\tthis._tempDS = null;\n\t\t\t}\n\n\t\t\t// Add new entry\n\t\t\tds.add(newEntry);\n\n\t\t\t// Remove first item if needed\n\t\t\tif (ds.total() > this._maxValues)\n\t\t\t\tds.remove(ds.at(0));\n\t\t}\n\t}\n\n\tget _dataSource()\n\t{\n\t\treturn this._chartWidget.dataSource;\n\t}\n\n\t_onPauseChange()\n\t{\n\t\t// Nothing to do, should be overridden\n\t}\n}\n\n// DEFINE COMPONENT\nif (!window.customElements.get('base-chart'))\n\twindow.customElements.define('base-chart', BaseChart);\n","import {bytesToSize, kBytesToSize} from '../../utils/utilities';\n\nexport function getFormattedDateTime(date, format = 'dd MMM HH:mm:ss')\n{\n\treturn kendo.toString(date, format);\n}\n\nexport function getBasicSharedTemplate(valueFormat = '{0}') {\n\treturn kendo.template(`\n\t\t<div class=\"chart-tooltip-category\"><strong>#: category #</strong></div>\n\t\t# for (var i = 0; i < points.length; i++) { #\n\t\t<div>\n\t\t\t<i class=\"fas fa-square\" style=\"color: #: points[i].color # \"></i>\n\t\t\t#: points[i].series.name# : #: kendo.format('${valueFormat}', points[i].value) #\n\t\t</div>\n\t\t# } #\n\t`);\n}\n\nexport function getMemorySharedTemplate() {\n\treturn function(dataItem) {\n\t\tdataItem.bytesToSize = bytesToSize; // Pass bytesToSize utility function to template\n\t\treturn kendo.template(`\n\t\t\t<div class=\"chart-tooltip-category\"><strong>#: category #</strong></div>\n\t\t\t# for (var i = 0; i < points.length; i++) { #\n\t\t\t<div>\n\t\t\t\t<i class=\"fas fa-square\" style=\"color: #: points[i].color # \"></i>\n\t\t\t\t#: points[i].series.name# : #: bytesToSize(points[i].value) #\n\t\t\t</div>\n\t\t\t# } #\n\t\t`)(dataItem);\n\t};\n}\n\nexport function getMemoryValueAxisLabelTemplate() {\n\treturn function(dataItem) {\n\t\tdataItem.bytesToSize = bytesToSize; // Pass bytesToSize utility function to template\n\t\treturn kendo.template(`\n\t\t\t#: bytesToSize(value) #\n\t\t`)(dataItem);\n\t};\n}\n\nexport function getNetworkSharedTemplate(categoryFormat = 'dd MMM HH:mm:ss') {\n\treturn function(dataItem) {\n\t\tdataItem.kBytesToSize = kBytesToSize; // Pass kBytesToSize utility function to template\n\t\treturn kendo.template(`\n\t\t\t<div class=\"chart-tooltip-category\"><strong>#: kendo.toString(category, '${categoryFormat}') #</strong></div>\n\t\t\t# for (var i = 0; i < points.length; i++) { #\n\t\t\t<div>\n\t\t\t\t<i class=\"fas fa-square\" style=\"color: #: points[i].color # \"></i>\n\t\t\t\t#: points[i].series.name# : #: kBytesToSize(points[i].value, 1, '', '/s') #\n\t\t\t</div>\n\t\t\t# } #\n\t\t`)(dataItem);\n\t};\n}\n\nexport function getNetworkValueAxisLabelTemplate() {\n\treturn function(dataItem) {\n\t\tdataItem.kBytesToSize = kBytesToSize; // Pass kBytesToSize utility function to template\n\t\treturn kendo.template(`\n\t\t\t#: kBytesToSize(value, 1, '', '/s') #\n\t\t`)(dataItem);\n\t};\n}\n\nexport function getNetworkValueAxisLabelTemplateB() {\n\treturn function(dataItem) {\n\t\tdataItem.bytesToSize = bytesToSize; // Pass bytesToSize utility function to template\n\t\treturn kendo.template(`\n\t\t\t#: bytesToSize(value, 1, '', '/s') #\n\t\t`)(dataItem);\n\t};\n}\n\nexport function getUserCountSharedTemplate(categoryFormat = 'dd MMM HH:mm:ss') {\n\treturn function(dataItem) {\n\t\treturn kendo.template(`\n\t\t\t<div class=\"chart-tooltip-category\"><strong>#: kendo.toString(category, '${categoryFormat}') #</strong></div>\n\t\t\t# for (var i = 0; i < points.length; i++) { #\n\t\t\t<div>\n\t\t\t\t<i class=\"fas fa-square\" style=\"color: #: points[i].color # \"></i>\n\t\t\t\t#: points[i].series.name# : #: points[i].value #\n\t\t\t</div>\n\t\t\t# } #\n\t\t`)(dataItem);\n\t};\n}\n\nexport function getTimeRangeTooltipTemplate() {\n\treturn function(dataItem) {\n\t\tdataItem.getTooltipString = function(v1, v2) {\n\t\t\tif (v1 == v2)\n\t\t\t{\n\t\t\t\tif (v2 == -24)\n\t\t\t\t\tv2 = -23;\n\n\t\t\t\tv1 = v2 - 1;\n\t\t\t}\n\n\t\t\tlet start = Math.abs(v1);\n\t\t\tlet end = Math.abs(v2);\n\n\t\t\treturn `From ${start} hour${start != 1 ? 's' : ''} ago to ${end > 0 ? ('to ' + end + ' hour' + (end != 1 ? 's' : '') + ' ago') : 'now'}`;\n\t\t}; // Pass utility function to template\n\t\treturn kendo.template(`\n\t\t\t#: getTooltipString(selectionStart, selectionEnd) #\n\t\t`)(dataItem);\n\t};\n}\n\nexport function getWRBytesLabelTemplate() {\n\treturn function(dataItem) {\n\t\treturn kendo.template(`\n\t\t\t#: value # #: dataItem.unit #\n\t\t`)(dataItem);\n\t};\n}\n\nexport function getBasicTemplate(valueFormat = '{0}') {\n\treturn kendo.template(`\n\t\t<div>\n\t\t\t<span class=\"text-nowrap\">#: kendo.toString(new Date(value.x), \"dd/MM/yy HH:mm\") #</span>\n\t\t\t<br>\n\t\t\t<strong class=\"text-nowrap\">#: series.name #: #: kendo.format('${valueFormat}', value.y) # </strong>\n\t\t</div>\n\t`);\n}\n\nexport function getMemoryTemplate() {\n\treturn function(dataItem) {\n\t\tdataItem.bytesToSize = bytesToSize; // Pass bytesToSize utility function to template\n\t\treturn kendo.template(`\n\t\t\t<div>\n\t\t\t\t<span class=\"text-nowrap\">#: kendo.toString(new Date(value.x), \"dd/MM/yy HH:mm\") #</span>\n\t\t\t\t<br>\n\t\t\t\t<strong class=\"text-nowrap\">#: series.name #: #: bytesToSize(value.y) # </strong>\n\t\t\t</div>\n\t\t`)(dataItem);\n\t};\n}\n\nexport function getNetworkTemplate() {\n\treturn function(dataItem) {\n\t\tdataItem.bytesToSize = bytesToSize; // Pass bytesToSize utility function to template\n\t\treturn kendo.template(`\n\t\t\t<div>\n\t\t\t\t<span class=\"text-nowrap\">#: kendo.toString(new Date(value.x), \"dd/MM/yy HH:mm\") #</span>\n\t\t\t\t<br>\n\t\t\t\t<strong class=\"text-nowrap\">#: series.name #: #: bytesToSize(value.y, 1, '', '/s') # </strong>\n\t\t\t</div>\n\t\t`)(dataItem);\n\t};\n}\n","import {BaseChart} from './base-chart';\n\nimport {getBasicSharedTemplate, getFormattedDateTime} from './chart-utils';\n\nexport class CpuUsageChart extends BaseChart\n{\n\tconstructor()\n\t{\n\t    super(120);\n\t}\n\n\t_initChartWidget(chartHtml)\n\t{\n\t\treturn chartHtml.kendoChart({\n\t\t\ttransitions: false,\n\t\t\tchartArea: {\n\t\t\t\theight: 120\n\t\t\t},\n\t        legend: {\n\t            visible: false\n\t        },\n\t        seriesDefaults: {\n\t            type: 'area',\n\t            labels: {\n\t                visible: false,\n\t                format: '{0}%',\n\t                background: 'transparent'\n\t            }\n\t        },\n\t        series: [\n\t\t\t\t{\n\t\t\t\t\tname: 'System',\n\t\t            field: 'systemLoad',\n\t\t            categoryField: 'time',\n\t\t\t\t\tcolor: this.systemColor,\n\t\t\t\t\tline: {\n\t\t\t\t\t\twidth: 2\n\t\t\t\t\t}\n\t\t        },\n\t\t\t\t{\n\t\t\t\t\tname: 'Process',\n\t\t\t\t\tfield: 'processLoad',\n\t\t\t\t\tcategoryField: 'time',\n\t\t\t\t\tcolor: this.processColor,\n\t\t\t\t\tline: {\n\t\t\t\t\t\twidth: 2\n\t\t\t\t\t}\n\t\t\t\t}\n\t\t\t],\n\t        valueAxis: {\n\t            labels: {\n\t                format: '{0}%'\n\t            },\n\t            line: {\n\t                visible: false\n\t            },\n\t\t\t\tmin: 0,\n\t    \t\tmax: 100\n\t        },\n\t        categoryAxis: {\n\t            majorGridLines: {\n\t                visible: false\n\t            },\n\t\t\t\tmajorTicks: {\n\t\t\t\t\tvisible: false\n\t\t\t\t},\n\t\t\t\tlabels: {\n\t\t\t\t\tvisible: false\n\t\t\t\t}\n\t        },\n\t\t\ttooltip: {\n\t\t\t\tshared: true,\n\t            visible: true,\n\t\t\t\tsharedTemplate: getBasicSharedTemplate('{0}%'),\n\t\t\t\tbackground: '#e4e4e4'\n\t        }\n\t    }).data('kendoChart');\n\t}\n\n\tget systemColor()\n\t{\n\t\treturn '#3399FF';\n\t}\n\n\tget processColor()\n\t{\n\t\treturn '#0033CC';\n\t}\n\n\taddEntry(processLoad, systemLoad)\n\t{\n\t\tlet newEntry = {\n\t\t\ttime: getFormattedDateTime(new Date()),\n\t\t\tprocessLoad: processLoad,\n\t\t\tsystemLoad: systemLoad\n\t\t};\n\n\t\t// Add entry to chart\n\t\tsuper._addEntry(newEntry);\n\t}\n}\n\n// DEFINE COMPONENT\nif (!window.customElements.get('cpu-usage-chart'))\n\twindow.customElements.define('cpu-usage-chart', CpuUsageChart);\n","import {BaseChart} from './base-chart';\n\nimport {getMemorySharedTemplate, getMemoryValueAxisLabelTemplate, getFormattedDateTime} from './chart-utils';\n\nexport class MemoryChart extends BaseChart\n{\n\tconstructor()\n\t{\n\t    super(300);\n\n\t\t// Append text to display elapsed time\n\t\tthis._elapsedTxt = $('<div>', {class: 'text-center', text: '---'});\n\t\t$(this).append(this._elapsedTxt);\n\t}\n\n\t_initChartWidget(chartHtml)\n\t{\n\t\treturn chartHtml.kendoChart({\n\t\t\ttransitions: false,\n\t\t\tchartArea: {\n\t\t\t\theight: 240\n\t\t\t},\n\t        legend: {\n\t            visible: false\n\t        },\n\t        seriesDefaults: {\n\t            type: 'area',\n\t            labels: {\n\t                visible: false,\n\t                format: '{0}',\n\t                background: 'transparent'\n\t            },\n\t        },\n\t        series: [{\n\t\t\t\tname: 'Allocated',\n\t            field: 'allocMem',\n\t            categoryField: 'time',\n\t\t\t\tcolor: this.allocatedColor,\n\t\t\t\tline: {\n\t\t\t\t\twidth: 2\n\t\t\t\t}\n\t        },\n\t\t\t{\n\t\t\t\tname: 'Used',\n\t\t\t\tfield: 'usedMem',\n\t\t\t\tcategoryField: 'time',\n\t\t\t\tcolor: this.usedColor,\n\t\t\t\tline: {\n\t\t\t\t\twidth: 2\n\t\t\t\t}\n\t\t\t}],\n\t        valueAxis: {\n\t            labels: {\n\t                format: '{0}',\n      \t\t\t\ttemplate: getMemoryValueAxisLabelTemplate()\n\t            },\n\t            line: {\n\t                visible: false\n\t            },\n\t\t\t\tmin: 0\n\t        },\n\t        categoryAxis: {\n\t\t\t\tmajorGridLines: {\n\t                visible: false\n\t            },\n\t\t\t\tmajorTicks: {\n\t\t\t\t\tvisible: false\n\t\t\t\t},\n\t\t\t\tlabels: {\n\t\t\t\t\tvisible: false\n\t\t\t\t}\n\t        },\n\t        tooltip: {\n\t\t\t\tshared: true,\n\t            visible: true,\n\t\t\t\tsharedTemplate: getMemorySharedTemplate(),\n\t\t\t\tbackground: '#e4e4e4'\n\t        }\n\t    }).data('kendoChart');\n\t}\n\n\tget allocatedColor()\n\t{\n\t\treturn '#3399FF';\n\t}\n\n\tget usedColor()\n\t{\n\t\treturn '#0033CC';\n\t}\n\n\taddEntry(allocMem, usedMem)\n\t{\n\t\tlet now = new Date();\n\n\t\tlet newEntry = {\n\t\t\ttime: getFormattedDateTime(now),\n\t\t\ttimestamp: now.getTime(),\n\t\t\tallocMem: allocMem,\n\t\t\tusedMem: usedMem\n\t\t};\n\n\t\t// Add entry to chart\n\t\tsuper._addEntry(newEntry);\n\n\t\t// Update x axis\n\t\tif (!this.isPaused)\n\t\t\tthis._updateElapsedText();\n\t}\n\n\t_updateElapsedText()\n\t{\n\t\tlet ds = this._dataSource;\n\n\t\tif (ds.total() > 1)\n\t\t{\n\t\t\t// Get axis limits\n\t\t\tconst chartMinimum = ds.at(0).timestamp;\n\t\t\tconst chartMaximum = ds.at(ds.total() - 1).timestamp;\n\n\t\t\t// Set elapsed text\n\t\t\tlet seconds = Math.round( (chartMaximum - chartMinimum) / 1000 );\n\t\t\tlet title;\n\n\t\t\tif (seconds < 60)\n\t\t\t\ttitle = 'Last ' + (seconds != 1 ? seconds + ' seconds' : 'second');\n\t\t\telse\n\t\t\t{\n\t\t\t\tconst minutes = Math.floor(seconds / 60);\n\t\t\t\ttitle = 'Last ' + (minutes != 1 ? minutes + ' minutes' : 'minute');\n\t\t\t}\n\n\t\t\tthis._elapsedTxt.text(title);\n\t\t}\n\t}\n}\n\n// DEFINE COMPONENT\nif (!window.customElements.get('memory-chart'))\n\twindow.customElements.define('memory-chart', MemoryChart);\n","import {getNetworkSharedTemplate, getNetworkValueAxisLabelTemplate} from './chart-utils';\nimport * as moment from 'moment';\n\nexport class Network24hChart extends HTMLElement\n{\n\tconstructor()\n\t{\n\t    super();\n\n\t\tthis._lastTotalOut = 0;\n\t\tthis._lastTotalIn = 0;\n\t\tthis._lastDate = new Date();\n\t\tthis._samplingRateSeconds = 0;\n\n\t\t// Create chart html\n\t\tlet chartHtml = $('<div>');\n\t\t$(this).append(chartHtml);\n\n\t\t// Initialize chart\n\t\tthis._chartWidget = this._initChartWidget(chartHtml);\n\t}\n\n\t_initChartWidget(chartHtml)\n\t{\n\t\treturn chartHtml.kendoChart({\n\t\t\ttransitions: false,\n\t\t\tchartArea: {\n\t\t\t\theight: 270,\n\t\t\t\tbackground: 'transparent'\n\t\t\t},\n\t        legend: {\n\t            visible: false\n\t        },\n\t        seriesDefaults: {\n\t            type: 'area',\n\t            labels: {\n\t                visible: false,\n\t                format: '{0}',\n\t                background: 'transparent'\n\t            },\n\t        },\n\t        series: [{\n\t\t\t\tname: 'Outgoing',\n\t            field: 'outgoing',\n\t            categoryField: 'time',\n\t\t\t\tcolor: '#CC0000',\n\t\t\t\tline: {\n\t\t\t\t\twidth: 2\n\t\t\t\t}\n\t        },\n\t\t\t{\n\t\t\t\tname: 'Incoming',\n\t\t\t\tfield: 'incoming',\n\t\t\t\tcategoryField: 'time',\n\t\t\t\tcolor: '#009900',\n\t\t\t\tline: {\n\t\t\t\t\twidth: 2\n\t\t\t\t}\n\t\t\t}],\n\t        valueAxis: {\n\t            labels: {\n\t                format: '{0}',\n      \t\t\t\ttemplate: getNetworkValueAxisLabelTemplate()\n\t            },\n\t            line: {\n\t                visible: false\n\t            },\n\t\t\t\tmin: 0\n\t        },\n\t        categoryAxis: {\n\t\t\t\tmajorGridLines: {\n\t                visible: true,\n\t\t\t\t\tstep: 60\n\t            },\n\t\t\t\tmajorTicks: {\n\t\t\t\t\tvisible: true,\n\t\t\t\t\tstep: 60\n\t\t\t\t},\n\t\t\t\tlabels: {\n\t\t\t\t\tvisible: true,\n\t\t\t\t\tstep: 60\n\t\t\t\t},\n\t\t\t\tbaseUnit: 'minutes',\n\t\t\t\ttype: 'date',\n\t        },\n\t        tooltip: {\n\t\t\t\tshared: true,\n\t            visible: true,\n\t\t\t\tsharedTemplate: getNetworkSharedTemplate('dd MMM HH:mm'),\n\t\t\t\tbackground: '#e4e4e4'\n\t        }\n\t    }).data('kendoChart');\n\t}\n\n\tredraw()\n\t{\n\t\tthis._chartWidget.redraw();\n\t}\n\n\tset range(values)\n\t{\n\t\tthis._range = values;\n\n\t\tthis._updateXAxisLimits();\n\t}\n\n\tget range()\n\t{\n\t\treturn this._range;\n\t}\n\n\taddHistoryEntries(outgoingData, incomingData, samplingRateSeconds, shiftSeconds)\n\t{\n\t\tthis._samplingRateSeconds = samplingRateSeconds;\n\n\t\tlet now = new Date();\n\n\t\t// NOTE: outgoingData and incominData are supposed to have the same size!!!\n\n\t\tif (outgoingData.length > 0)\n\t\t{\n\t\t\tlet start = moment(now);\n\t\t\tstart.subtract((samplingRateSeconds * outgoingData.length) + shiftSeconds, 's')\n\n\t\t\tthis._lastTotalOut = outgoingData[0];\n\t\t\tthis._lastTotalIn = incomingData[0];\n\t\t\tthis._lastDate = start.toDate();\n\n\t\t\tif (outgoingData.length > 1)\n\t\t\t{\n\t\t\t\tlet dataSource = new kendo.data.DataSource();\n\n\t\t\t\tfor (let i = 1; i < outgoingData.length; i++)\n\t\t\t\t{\n\t\t\t\t\tconst totalOut = outgoingData[i];\n\t\t\t\t\tconst totalIn = incomingData[i];\n\n\t\t\t\t\tlet date = start.clone();\n\t\t\t\t\tdate.add(i * samplingRateSeconds, 's');\n\n\t\t\t\t\tthis.addEntry(totalOut, totalIn, dataSource, date.toDate(), false);\n\t\t\t\t}\n\n\t\t\t\t// Assign datasource to chart\n\t\t\t\tthis._chartWidget.setDataSource(dataSource);\n\n\t\t\t\t// Update axis\n\t\t\t\tthis._updateXAxisLimits();\n\t\t\t}\n\t\t}\n\t\telse\n\t\t{\n\t\t\tthis._lastTotalOut = 0;\n\t\t\tthis._lastTotalIn = 0;\n\t\t\tthis._lastDate = now;\n\t\t}\n\t}\n\n\taddEntry(totalOut, totalIn, dataSource = null, date = null, updateAxis = true)\n\t{\n\t\tif (date == null)\n\t\t\tdate = new Date();\n\n\t\tconst elapsedSecs = Math.round((date.getTime() - this._lastDate.getTime()) / 1000);\n\n\t\t// The 24h chart is updated at the same sampling rate used by the server to save the traffic history\n\t\tif (elapsedSecs >= this._samplingRateSeconds)\n\t\t{\n\t\t\tlet newEntry = {\n\t\t\t\ttime: date,\n\t\t\t\toutgoing: Math.round((totalOut - this._lastTotalOut) / elapsedSecs),\n\t\t\t\tincoming: Math.round((totalIn - this._lastTotalIn) / elapsedSecs)\n\t\t\t};\n\n\t\t\t// Add entry to data source\n\t\t\tif (dataSource == null)\n\t\t\t\tdataSource = this._dataSource;\n\n\t\t\tdataSource.add(newEntry);\n\n\t\t\tthis._lastTotalOut = totalOut;\n\t\t\tthis._lastTotalIn = totalIn;\n\t\t\tthis._lastDate = date;\n\n\t\t\tif (updateAxis)\n\t\t\t\tthis._updateXAxisLimits();\n\t\t}\n\t}\n\n\t_updateXAxisLimits()\n\t{\n\t\tlet ds = this._dataSource;\n\n\t\tif (ds.total() > 0)\n\t\t{\n\t\t\tconst chartMinimum = moment(this._lastDate).add(this._range[0], 'h');\n\t\t\tconst chartMaximum = moment(this._lastDate).add(this._range[1], 'h');\n\n\t\t\tthis._chartWidget.options.categoryAxis.min = chartMinimum.toDate();\n\t\t\tthis._chartWidget.options.categoryAxis.max = chartMaximum.toDate();\n\n\t\t\tthis._chartWidget.redraw();\n\t\t}\n\t}\n\n\tget _dataSource()\n\t{\n\t\treturn this._chartWidget.dataSource;\n\t}\n}\n\n// DEFINE COMPONENT\nif (!window.customElements.get('network-24h-chart'))\n\twindow.customElements.define('network-24h-chart', Network24hChart);\n","import {BaseChart} from './base-chart';\n\nimport {getNetworkSharedTemplate, getNetworkValueAxisLabelTemplate, getFormattedDateTime} from './chart-utils';\n\nexport class NetworkRealtimeChart extends BaseChart\n{\n\tconstructor()\n\t{\n\t    super(300);\n\n\t\tthis._lastTotalKbOut = 0;\n\t\tthis._lastTotalKbIn = 0;\n\t\tthis._lastTime = 0;\n\n\t\t// Append text to display elapsed time\n\t\tthis._elapsedTxt = $('<div>', {class: 'text-center', text: '---'});\n\t\t$(this).append(this._elapsedTxt);\n\t}\n\n\t_initChartWidget(chartHtml)\n\t{\n\t\treturn chartHtml.kendoChart({\n\t\t\ttransitions: false,\n\t\t\tchartArea: {\n\t\t\t\theight: 270,\n\t\t\t\tbackground: 'transparent'\n\t\t\t},\n\t        legend: {\n\t            visible: false\n\t        },\n\t        seriesDefaults: {\n\t            type: 'area',\n\t            labels: {\n\t                visible: false,\n\t                format: '{0}',\n\t                background: 'transparent'\n\t            },\n\t        },\n\t        series: [{\n\t\t\t\tname: 'Outgoing',\n\t            field: 'outgoing',\n\t            categoryField: 'time',\n\t\t\t\tcolor: this.outgoingColor,\n\t\t\t\tline: {\n\t\t\t\t\twidth: 2\n\t\t\t\t}\n\t        },\n\t\t\t{\n\t\t\t\tname: 'Incoming',\n\t\t\t\tfield: 'incoming',\n\t\t\t\tcategoryField: 'time',\n\t\t\t\tcolor: this.incomingColor,\n\t\t\t\tline: {\n\t\t\t\t\twidth: 2\n\t\t\t\t}\n\t\t\t}],\n\t        valueAxis: {\n\t            labels: {\n\t                format: '{0}',\n      \t\t\t\ttemplate: getNetworkValueAxisLabelTemplate()\n\t            },\n\t            line: {\n\t                visible: false\n\t            },\n\t\t\t\tmin: 0,\n\t        },\n\t        categoryAxis: {\n\t\t\t\tmajorGridLines: {\n\t                visible: false\n\t            },\n\t\t\t\tmajorTicks: {\n\t\t\t\t\tvisible: false\n\t\t\t\t},\n\t\t\t\tlabels: {\n\t\t\t\t\tvisible: false\n\t\t\t\t}\n\t        },\n\t        tooltip: {\n\t\t\t\tshared: true,\n\t            visible: true,\n\t\t\t\tsharedTemplate: getNetworkSharedTemplate(),\n\t\t\t\tbackground: '#e4e4e4'\n\t        }\n\t    }).data('kendoChart');\n\t}\n\n\tget outgoingColor()\n\t{\n\t\treturn '#CC0000';\n\t}\n\n\tget incomingColor()\n\t{\n\t\treturn '#009900';\n\t}\n\n\taddEntry(totalOutKb, totalInKb)\n\t{\n\t\tlet now = new Date()\n\t\tlet time = now.getTime();\n\n\t\t// The first value sent by the server is not plotted: it would be a worng value because\n\t\t// we have to calculate it as the difference between two total values, but we don't have the previous one\n\t\tif (this._lastTime > 0)\n\t\t{\n\t\t\tlet elapsedSecs = (time - this._lastTime) / 1000;\n\n\t\t\tconst valueOut = Math.round((totalOutKb - this._lastTotalKbOut) / elapsedSecs);\n\t\t\tconst valueIn = Math.round((totalInKb - this._lastTotalKbIn) / elapsedSecs);\n\n\t\t\tlet newEntry = {\n\t\t\t\ttime: getFormattedDateTime(now),\n\t\t\t\ttimestamp: time,\n\t\t\t\toutgoing: valueOut,\n\t\t\t\tincoming: valueIn\n\t\t\t};\n\n\t\t\t// Add entry to chart\n\t\t\tsuper._addEntry(newEntry);\n\n\t\t\t// Update x axis\n\t\t\tif (!this.isPaused)\n\t\t\t\tthis._updateElapsedText();\n\t\t}\n\n\t\tthis._lastTotalKbOut = totalOutKb;\n\t\tthis._lastTotalKbIn = totalInKb;\n\t\tthis._lastTime = time;\n\t}\n\n\t_updateElapsedText()\n\t{\n\t\tlet ds = this._dataSource;\n\n\t\tif (ds.total() > 1)\n\t\t{\n\t\t\t// Get axis limits\n\t\t\tconst chartMinimum = ds.at(0).timestamp;\n\t\t\tconst chartMaximum = ds.at(ds.total() - 1).timestamp;\n\n\t\t\t// Set elapsed text\n\t\t\tlet title;\n\t\t\tlet seconds = Math.round( (chartMaximum - chartMinimum) / 1000 );\n\n\t\t\tif (seconds < 60)\n\t\t\t\ttitle = 'Last ' + (seconds != 1 ? seconds + ' seconds' : 'second');\n\t\t\telse\n\t\t\t{\n\t\t\t\tconst minutes = Math.floor(seconds / 60);\n\t\t\t\ttitle = 'Last ' + (minutes != 1 ? minutes + ' minutes' : 'minute');\n\t\t\t}\n\n\t\t\tthis._elapsedTxt.text(title);\n\t\t}\n\t}\n}\n\n// DEFINE COMPONENT\nif (!window.customElements.get('network-realtime-chart'))\n\twindow.customElements.define('network-realtime-chart', NetworkRealtimeChart);\n","import {BaseChart} from './base-chart';\n\nimport {getBasicSharedTemplate, getFormattedDateTime} from './chart-utils';\n\nexport class SystemQueueChart extends BaseChart\n{\n\tconstructor()\n\t{\n\t    super(240);\n\n\t\tthis._averageLoad = 0;\n\t}\n\n\t_initialize()\n\t{\n\t\tconst chartHeight = 240;\n\n\t\t// Append container for chart & gauge\n\t\tlet container = $('<div>', {class: 'd-flex flex-row'});\n\t\t$(this).append(container);\n\n\t\t// CHART\n\n\t\tlet chartContainer = $('<div>', {class: 'flex-grow-1'});\n\t\tcontainer.append(chartContainer);\n\n\t\t// Create chart html\n\t\tlet chartHtml = $('<div>');\n\t\tchartContainer.append(chartHtml);\n\n\t\t// Initialize chart\n\t\tthis._chartWidget = this._initChartWidget(chartHtml, chartHeight);\n\n\t\t// Append text to display elapsed time\n\t\tthis._elapsedTxt = $('<div>', {class: 'text-center', text: '---'});\n\t\tchartContainer.append(this._elapsedTxt);\n\n\t\t// GAUGE\n\n\t\tlet gaugeContainer = $('<div>');\n\t\tcontainer.append(gaugeContainer);\n\n\t\t// Create gauge html\n\t\tlet gaugeHtml = $('<div>');\n\t\tgaugeContainer.append(gaugeHtml);\n\n\t\t// Initialize gauge\n\t\tthis._gaugeWidget = this._initGaugeWidget(gaugeHtml, chartHeight);\n\n\t\t// Append text to display load\n\t\tthis._loadTxt = $('<div>', {class: 'text-center font-weight-bold', text: '---'});\n\t\tgaugeContainer.append(this._loadTxt);\n\n\t\t//------------------------------\n\n\t\t// Reset data source\n\t\tthis.reset();\n\t}\n\n\t_initChartWidget(chartHtml, height)\n\t{\n\t\treturn chartHtml.kendoChart({\n\t\t\ttransitions: false,\n\t\t\tchartArea: {\n\t\t\t\theight: height,\n\t\t\t\tbackground: 'transparent'\n\t\t\t},\n\t        legend: {\n\t            visible: false\n\t        },\n\t        seriesDefaults: {\n\t            type: 'area',\n\t            labels: {\n\t                visible: false,\n\t                format: '{0}',\n\t                background: 'transparent'\n\t            }\n\t        },\n\t        series: [\n\t\t\t\t{\n\t\t\t\t\tname: 'Messages',\n\t\t            field: 'load',\n\t\t            categoryField: 'time',\n\t\t\t\t\tcolor: '#3399ff',\n\t\t\t\t\tline: {\n\t\t\t\t\t\twidth: 2\n\t\t\t\t\t}\n\t\t        }\n\t\t\t],\n\t        valueAxis: {\n\t\t\t\ttitle: {\n\t\t\t\t\ttext: 'Messages'\n\t\t\t\t},\n\t            labels: {\n\t                format: '{0}'\n\t            },\n\t            line: {\n\t                visible: false\n\t            },\n\t\t\t\tmin: 0\n\t        },\n\t        categoryAxis: {\n\t            majorGridLines: {\n\t                visible: false\n\t            },\n\t\t\t\tmajorTicks: {\n\t\t\t\t\tvisible: false\n\t\t\t\t},\n\t\t\t\tlabels: {\n\t\t\t\t\tvisible: false\n\t\t\t\t}\n\t        },\n\t\t\ttooltip: {\n\t\t\t\tshared: true,\n\t            visible: true,\n\t\t\t\tsharedTemplate: getBasicSharedTemplate(),\n\t\t\t\tbackground: '#e4e4e4'\n\t        }\n\t    }).data('kendoChart');\n\t}\n\n\t_initGaugeWidget(gaugeHtml, height)\n\t{\n\t\treturn gaugeHtml.kendoLinearGauge({\n\t\t\tpointer: {\n\t            value: 0,\n\t            shape: 'arrow',\n\t          \tsize: 10,\n\t\t\t\tcolor: '#212529'\n\t        },\n\t\t\tgaugeArea: {\n\t\t\t\theight: height\n\t\t\t},\n\t        scale: {\n\t            majorUnit: 10,\n\t            minorUnit: 1,\n\t\t\t\tmin: 0,\n\t\t\t\tmax: 100,\n\t            ranges: [\n\t                {\n\t\t\t\t\t\tfrom: 0,\n\t                    to: 50,\n\t                    color: '#00cc00',\n\t                  \topacity: .5\n\t                }, {\n\t                    from: 50,\n\t                    to: 75,\n\t                    color: '#ffe106',\n\t                  \topacity: .5\n\t                }, {\n\t                    from: 75,\n\t                    to: 100,\n\t                    color: '#ed1601',\n\t                  \topacity: .5\n\t                }\n\t            ],\n\t          \tmajorTicks: {\n\t              visible: true,\n\t              size: 10\n\t            },\n\t          \tminorTicks: {\n\t              visible: false\n\t            },\n\t          \tlabels: {\n\t              visible: false\n\t            },\n\t          \tline: {\n\t              visible: false\n\t            },\n\t          \trangeSize: 10\n\t        }\n\t    }).data('kendoLinearGauge');\n\t}\n\n\taddEntry(load, queueMsgLimit)\n\t{\n\t\t// CHART\n\n\t\tlet now = new Date();\n\n\t\tlet newEntry = {\n\t\t\ttime: getFormattedDateTime(now),\n\t\t\ttimestamp: now.getTime(),\n\t\t\tload: load\n\t\t};\n\n\t\t// Add entry to chart\n\t\tsuper._addEntry(newEntry);\n\n\t\t// Update x axis\n\t\tif (!this.isPaused)\n\t\t\tthis._updateElapsedText();\n\n\t\t// GAUGE\n\n\t\t// Evaluate average\n\t\tlet numValues = this._dataSource.total();\n\n\t\tif (numValues > 0)\n\t\t{\n\t\t\tlet total = 0;\n\t\t\tfor (let i = 0; i < numValues; i++)\n\t\t\t\ttotal += this._dataSource.at(i).load;\n\n\t\t\tthis._averageLoad = Math.round(total / numValues);\n\t\t}\n\t\telse\n\t\t\tthis._averageLoad = 0;\n\n\t\tconst avgLoadPerc = Math.round(this._averageLoad / queueMsgLimit * 1000) / 10;\n\n\t\t// Set gauge value\n\t\tthis._gaugeWidget.value(avgLoadPerc);\n\n\t\t// Set load text\n\t\tthis._loadTxt.text(avgLoadPerc + '%')\n\t}\n\n\t_updateElapsedText()\n\t{\n\t\tlet ds = this._dataSource;\n\n\t\tif (ds.total() > 1)\n\t\t{\n\t\t\t// Get axis limits\n\t\t\tconst chartMinimum = ds.at(0).timestamp;\n\t\t\tconst chartMaximum = ds.at(ds.total() - 1).timestamp;\n\n\t\t\t// Set elapsed text\n\t\t\tlet seconds = Math.round( (chartMaximum - chartMinimum) / 1000 );\n\t\t\tlet title;\n\n\t\t\tif (seconds < 60)\n\t\t\t\ttitle = 'Last ' + (seconds != 1 ? seconds + ' seconds' : 'second');\n\t\t\telse\n\t\t\t{\n\t\t\t\tconst minutes = Math.floor(seconds / 60);\n\t\t\t\ttitle = 'Last ' + (minutes != 1 ? minutes + ' minutes' : 'minute');\n\t\t\t}\n\n\t\t\tthis._elapsedTxt.text(title);\n\t\t}\n\t}\n\n\tget load()\n\t{\n\t\treturn this._averageLoad;\n\t}\n}\n\n// DEFINE COMPONENT\nif (!window.customElements.get('system-queue-chart'))\n\twindow.customElements.define('system-queue-chart', SystemQueueChart);\n","import {BaseChart} from './base-chart';\n\nimport {getBasicSharedTemplate, getFormattedDateTime} from './chart-utils';\n\nexport class SystemWorkloadChart extends BaseChart\n{\n\tconstructor()\n\t{\n\t    super(240);\n\t}\n\n\t_initialize()\n\t{\n\t\tconst chartHeight = 240;\n\n\t\t// Append container for chart & gauge\n\t\tlet container = $('<div>', {class: 'd-flex flex-row'});\n\t\t$(this).append(container);\n\n\t\t// CHART\n\n\t\tlet chartContainer = $('<div>', {class: 'flex-grow-1'});\n\t\tcontainer.append(chartContainer);\n\n\t\t// Create chart html\n\t\tlet chartHtml = $('<div>');\n\t\tchartContainer.append(chartHtml);\n\n\t\t// Initialize chart\n\t\tthis._chartWidget = this._initChartWidget(chartHtml, chartHeight);\n\n\t\t// Append text to display elapsed time\n\t\tthis._elapsedTxt = $('<div>', {class: 'text-center', text: '---'});\n\t\tchartContainer.append(this._elapsedTxt);\n\n\t\t// GAUGE\n\n\t\tlet gaugeContainer = $('<div>');\n\t\tcontainer.append(gaugeContainer);\n\n\t\t// Create gauge html\n\t\tlet gaugeHtml = $('<div>');\n\t\tgaugeContainer.append(gaugeHtml);\n\n\t\t// Initialize gauge\n\t\tthis._gaugeWidget = this._initGaugeWidget(gaugeHtml, chartHeight);\n\n\t\t// Append text to display load\n\t\tthis._loadTxt = $('<div>', {class: 'text-center font-weight-bold', text: '---'});\n\t\tgaugeContainer.append(this._loadTxt);\n\n\t\t//------------------------------\n\n\t\t// Reset data source\n\t\tthis.reset();\n\t}\n\n\t_initChartWidget(chartHtml, height)\n\t{\n\t\treturn chartHtml.kendoChart({\n\t\t\ttransitions: false,\n\t\t\tchartArea: {\n\t\t\t\theight: height,\n\t\t\t\tbackground: 'transparent'\n\t\t\t},\n\t        legend: {\n\t            visible: false\n\t        },\n\t        seriesDefaults: {\n\t            type: 'area',\n\t            labels: {\n\t                visible: false,\n\t                format: '{0}%',\n\t                background: 'transparent'\n\t            }\n\t        },\n\t        series: [\n\t\t\t\t{\n\t\t\t\t\tname: 'Workload',\n\t\t            field: 'load',\n\t\t            categoryField: 'time',\n\t\t\t\t\tcolor: '#3399ff',\n\t\t\t\t\tline: {\n\t\t\t\t\t\twidth: 2\n\t\t\t\t\t}\n\t\t        }\n\t\t\t],\n\t        valueAxis: {\n\t            labels: {\n\t                format: '{0}%'\n\t            },\n\t            line: {\n\t                visible: false\n\t            },\n\t\t\t\tmin: 0,\n\t    \t\tmax: 100\n\t        },\n\t        categoryAxis: {\n\t            majorGridLines: {\n\t                visible: false\n\t            },\n\t\t\t\tmajorTicks: {\n\t\t\t\t\tvisible: false\n\t\t\t\t},\n\t\t\t\tlabels: {\n\t\t\t\t\tvisible: false\n\t\t\t\t}\n\t        },\n\t\t\ttooltip: {\n\t\t\t\tshared: true,\n\t            visible: true,\n\t\t\t\tsharedTemplate: getBasicSharedTemplate('{0}%'),\n\t\t\t\tbackground: '#e4e4e4'\n\t        }\n\t    }).data('kendoChart');\n\t}\n\n\t_initGaugeWidget(gaugeHtml, height)\n\t{\n\t\treturn gaugeHtml.kendoLinearGauge({\n\t\t\tpointer: {\n\t            value: 0,\n\t            shape: 'arrow',\n\t          \tsize: 10,\n\t\t\t\tcolor: '#212529'\n\t        },\n\t\t\tgaugeArea: {\n\t\t\t\theight: height\n\t\t\t},\n\t        scale: {\n\t            majorUnit: 10,\n\t            minorUnit: 1,\n\t            max: 100,\n\t\t\t\tmin: 0,\n\t            ranges: [\n\t                {\n\t                    from: 0,\n\t                    to: 50,\n\t                    color: '#00cc00',\n\t                  \topacity: .5\n\t                }, {\n\t                    from: 50,\n\t                    to: 75,\n\t                    color: '#ffe106',\n\t                  \topacity: .5\n\t                }, {\n\t                    from: 75,\n\t                    to: 100,\n\t                    color: '#ed1601',\n\t                  \topacity: .5\n\t                }\n\t            ],\n\t          \tmajorTicks: {\n\t              visible: true,\n\t              size: 10\n\t            },\n\t          \tminorTicks: {\n\t              visible: false\n\t            },\n\t          \tlabels: {\n\t              visible: false\n\t            },\n\t          \tline: {\n\t              visible: false\n\t            },\n\t          \trangeSize: 10\n\t        }\n\t    }).data('kendoLinearGauge');\n\t}\n\n\taddEntry(load)\n\t{\n\t\t// CHART\n\n\t\tlet now = new Date();\n\n\t\tlet newEntry = {\n\t\t\ttime: getFormattedDateTime(now),\n\t\t\ttimestamp: now.getTime(),\n\t\t\tload: load\n\t\t};\n\n\t\t// Add entry to chart\n\t\tsuper._addEntry(newEntry);\n\n\t\t// Update x axis\n\t\tif (!this.isPaused)\n\t\t\tthis._updateElapsedText();\n\n\t\t// GAUGE\n\n\t\t// Evaluate average\n\t\tlet numValues = this._dataSource.total();\n\t\tlet averageLoad = 0;\n\n\t\tif (numValues > 0)\n\t\t{\n\t\t\tlet total = 0;\n\t\t\tfor (let i = 0; i < numValues; i++)\n\t\t\t\ttotal += this._dataSource.at(i).load;\n\n\t\t\taverageLoad = Math.round(total / numValues);\n\t\t}\n\n\t\t// Set gauge value\n\t\tthis._gaugeWidget.value(averageLoad);\n\n\t\t// Set load text\n\t\tthis._loadTxt.text(averageLoad + '%')\n\t}\n\n\t_updateElapsedText()\n\t{\n\t\tlet ds = this._dataSource;\n\n\t\tif (ds.total() > 1)\n\t\t{\n\t\t\t// Get axis limits\n\t\t\tconst chartMinimum = ds.at(0).timestamp;\n\t\t\tconst chartMaximum = ds.at(ds.total() - 1).timestamp;\n\n\t\t\t// Set elapsed text\n\t\t\tlet seconds = Math.round( (chartMaximum - chartMinimum) / 1000 );\n\t\t\tlet title;\n\n\t\t\tif (seconds < 60)\n\t\t\t\ttitle = 'Last ' + (seconds != 1 ? seconds + ' seconds' : 'second');\n\t\t\telse\n\t\t\t{\n\t\t\t\tconst minutes = Math.floor(seconds / 60);\n\t\t\t\ttitle = 'Last ' + (minutes != 1 ? minutes + ' minutes' : 'minute');\n\t\t\t}\n\n\t\t\tthis._elapsedTxt.text(title);\n\t\t}\n\t}\n\n\tget load()\n\t{\n\t\treturn this._averageLoad;\n\t}\n}\n\n// DEFINE COMPONENT\nif (!window.customElements.get('system-workload-chart'))\n\twindow.customElements.define('system-workload-chart', SystemWorkloadChart);\n","import {BaseChart} from './base-chart';\n\nimport {getBasicSharedTemplate, getFormattedDateTime} from './chart-utils';\n\nexport class ThreadCountChart extends BaseChart\n{\n\tconstructor()\n\t{\n\t    super(120);\n\t}\n\n\t_initChartWidget(chartHtml)\n\t{\n\t\treturn chartHtml.kendoChart({\n\t\t\ttransitions: false,\n\t\t\tchartArea: {\n\t\t\t\theight: 120,\n\t\t\t\tbackground: 'transparent'\n\t\t\t},\n\t        legend: {\n\t            visible: false\n\t        },\n\t        seriesDefaults: {\n\t            type: 'area',\n\t            labels: {\n\t                visible: false,\n\t                format: '{0}',\n\t                background: 'transparent'\n\t            },\n\t        },\n\t        series: [\n\t\t\t\t{\n\t\t\t\t\tname: 'Threads',\n\t\t            field: 'threads',\n\t\t            categoryField: 'time',\n\t\t\t\t\tcolor: this.threadsColor,\n\t\t\t\t\tline: {\n\t\t\t\t\t\twidth: 2\n\t\t\t\t\t}\n\t\t        }\n\t\t\t],\n\t        valueAxis: {\n\t\t\t\tmin: 0,\n\t            labels: {\n\t                format: '{0}'\n\t            },\n\t            line: {\n\t                visible: false\n\t            },\n\t        },\n\t        categoryAxis: {\n\t            majorGridLines: {\n\t                visible: false\n\t            },\n\t\t\t\tmajorTicks: {\n\t\t\t\t\tvisible: false\n\t\t\t\t},\n\t\t\t\tlabels: {\n\t\t\t\t\tvisible: false\n\t\t\t\t}\n\t        },\n\t\t\ttooltip: {\n\t\t\t\tshared: true,\n\t            visible: true,\n\t\t\t\tsharedTemplate: getBasicSharedTemplate(),\n\t\t\t\tbackground: '#e4e4e4'\n\t        }\n\t    }).data('kendoChart');\n\t}\n\n\tget threadsColor()\n\t{\n\t\treturn '#3399FF';\n\t}\n\n\taddEntry(threads)\n\t{\n\t\tlet newEntry = {\n\t\t\ttime: getFormattedDateTime(new Date()),\n\t\t\tthreads: threads\n\t\t};\n\n\t\t// Add entry to chart\n\t\tsuper._addEntry(newEntry);\n\t}\n}\n\n// DEFINE COMPONENT\nif (!window.customElements.get('thread-count-chart'))\n\twindow.customElements.define('thread-count-chart', ThreadCountChart);\n","import {BaseModule} from './base-module';\nimport {bytesToSize, kBytesToSize} from '../utils/utilities';\nimport {getTimeRangeTooltipTemplate} from '../components/charts/chart-utils';\nimport {CpuUsageChart} from '../components/charts/cpu-usage-chart';\nimport {ThreadCountChart} from '../components/charts/thread-count-chart';\nimport {MemoryChart} from '../components/charts/memory-chart';\nimport {NetworkRealtimeChart} from '../components/charts/network-realtime-chart';\nimport {Network24hChart} from '../components/charts/network-24h-chart';\nimport {SystemWorkloadChart} from '../components/charts/system-workload-chart';\nimport {SystemQueueChart} from '../components/charts/system-queue-chart';\n\nexport default class Dashboard extends BaseModule\n{\n\tconstructor()\n\t{\n\t    super('dashboard');\n\n\t\t// Outgoing requests\n\t\tthis.REQ_INIT = 'init';\n\t\tthis.REQ_GET_DATA = 'getData';\n\t\tthis.REQ_DUMP_THREADS = 'getThDump';\n\n\t\t// Incoming responses\n\t\tthis.RESP_INIT = 'init';\n\t\tthis.RESP_DATA = 'data';\n\t\tthis.RESP_THREAD_DUMP = 'thDump';\n\n\t\tthis._tcpOutThreads = 0;\n\t\tthis._tcpOutLoad = 0;\n\t\tthis._udpOutThreads = 0;\n\t\tthis._udpOutLoad = 0;\n\t}\n\n\t//------------------------------------\n\t// COMMON MODULE INTERFACE METHODS\n\t// This members are used by the main controller\n\t// to communicate with the module's controller.\n\t//------------------------------------\n\n\tinitialize(idData, shellController)\n\t{\n\t\t// Call super method\n\t\tsuper.initialize(idData, shellController);\n\n\t\t// Initialize scrolling tabs\n\t\t$('#dsh-tabNavigator > #tabs').scrollingTabs({\n\t\t\tbootstrapVersion: 4,\n\t\t\tscrollToTabEdge: true,\n\t\t\tenableSwiping: true,\n\t\t\tdisableScrollArrowsOnFullyScrolled: true,\n\t\t\tcssClassLeftArrow: 'fa fa-chevron-left',\n\t\t\tcssClassRightArrow: 'fa fa-chevron-right'\n\t\t});\n\n\t\t// Initialize interval dropdown\n\t\tthis._intervalDropDown = $('#dsh-intervalDD').kendoDropDownList({\n\t\t\tvalueTemplate: '<span class=\"text-muted pr-1\">Interval:</span><span>#:data.text#</span>',\n\t\t\tchange: $.proxy(this._onUpdateIntervalChange, this)\n\t\t}).data('kendoDropDownList');\n\n\t\t// Initialize threads grid\n\t\tthis._threadsGrid = $('#dsh-threadsGrid').kendoGrid({\n\t\t\tdataSource: this._getThreadsGridDataSource(),\n\t\t\tscrollable: true,\n            sortable: true,\n            filterable: false,\n\t\t\tresizable: true,\n\t\t\tselectable: false,\n\t\t\tcolumns: [\n\t\t\t\t{\n\t\t\t\t\tfield: 'id',\n\t\t\t\t\ttitle: 'Id',\n\t\t\t\t\twidth: 60\n\t\t\t\t},\n\t\t\t\t{\n\t\t\t\t\tfield: 'name',\n\t\t\t\t\ttitle: 'Name',\n\t\t\t\t\twidth: '65%'\n\t\t\t\t},\n\t\t\t\t{\n\t\t\t\t\tfield: 'cpuTime',\n\t\t\t\t\ttitle: 'CPU Time',\n\t\t\t\t\tformat: '{0}%',\n\t\t\t\t\twidth: '35%'\n\t\t\t\t},\n\t\t\t],\n\t\t\tnoRecords: {\n\t\t\t\ttemplate: 'No threads.'\n\t\t\t},\n\t\t\theight: 'calc(240px - .75rem + 1rem)'\n\t\t}).data('kendoGrid');\n\n\t\t// Add click listener to threads dump button\n\t\t$('#dsh-dumpThreadsBt').on('click', $.proxy(this._onDumpThreadsClick, this));\n\n\t\t// Configure thread dump panel\n\t\t$('#dsh-threadDumpModal').modal({\n\t\t\tbackdrop: 'static',\n\t\t\tkeyboard: false,\n\t\t\tshow: false,\n\t\t});\n\n\t\t// Add listener for threads dump panel hide\n\t\t$('#dsh-threadDumpModal').on('hidden.bs.modal', $.proxy(this._onThreadDumpModalHidden, this));\n\n\t\t// Add click listener to threads dump copy button\n\t\t$('#dsh-copyThreadDumpBt').on('click', $.proxy(this._onCopyThreadDumpClick, this));\n\n\t\t// Initialize time range slider\n\t\tthis._timeSlider = $('#dsh-timeSlider').kendoRangeSlider({\n\t\t\tmin: -24,\n            max: 0,\n            smallStep: 1,\n            largeStep: 0,\n\t\t\tchange: $.proxy(this._onTimeRangeChange, this),\n\t\t\ttooltip: {\n\t\t\t\ttemplate: getTimeRangeTooltipTemplate()\n\t\t\t},\n\t\t\tleftDragHandleTitle: 'Drag',\n\t\t\trightDragHandleTitle: 'Drag',\n\t\t}).data('kendoRangeSlider');\n\n\t\t// Add click listener to reset charts button\n\t\t$('#dsh-resetBt').on('click', $.proxy(this._onResetChartsClick, this));\n\n\t\t// Add click listener to play & pause chart update buttons\n\t\t$('#dsh-pauseBt').on('click', $.proxy(this._onPauseUpdateClick, this));\n\t\t$('#dsh-playBt').on('click', $.proxy(this._onPlayUpdateClick, this));\n\n\t\t// Set reference to charts\n\t\tthis._cpuUsageChart = document.getElementById('dsh-cpuUsageChart');\n\t\tthis._threadCountChart = document.getElementById('dsh-threadCountChart');\n\t\tthis._memoryChart = document.getElementById('dsh-memoryChart');\n\t\tthis._networkRTChart = document.getElementById('dsh-networkRTChart');\n\t\tthis._network24hChart = document.getElementById('dsh-network24hChart');\n\t\tthis._sysWorkloadChart = document.getElementById('dsh-systemWorkloadChart');\n\t\tthis._outTcpMsgQueueChart = document.getElementById('dsh-outgoingTcpMessagesQueueChart');\n\t\tthis._outUdpMsgQueueChart = document.getElementById('dsh-outgoingUdpMessagesQueueChart');\n\t\tthis._sysQueueChart = document.getElementById('dsh-systemQueueChart');\n\t\tthis._extQueueChart = document.getElementById('dsh-extensionQueueChart');\n\n\t\t// Set colors in chart legends\n\t\t$('#dsh-systemLoadColor').css('color', this._cpuUsageChart.systemColor);\n\t\t$('#dsh-processLoadColor').css('color', this._cpuUsageChart.processColor);\n\t\t$('#dsh-allocMemColor').css('color', this._memoryChart.allocatedColor);\n\t\t$('#dsh-usedMemColor').css('color', this._memoryChart.usedColor);\n\t\t$('#dsh-incomingNetColor').css('color', this._networkRTChart.incomingColor);\n\t\t$('#dsh-outgoingNetColor').css('color', this._networkRTChart.outgoingColor);\n\n\t\t// Reset time range on 24h network chart\n\t\tthis._onTimeRangeChange();\n\n\t\t// Add listener to redraw all charts on main tab change\n\t\t// (to work around an issue with the chart resizing to default width when pane is hidden)\n\t\t$('a[data-toggle=\"tab\"]').on('shown.bs.tab', $.proxy(this._onWindowResize, this));\n\n\t\t// Add listener to redraw network charts on network tab change\n\t\t// (to work around an issue with the chart resizing to default width when pane is hidden)\n\t\t$('#dsh-networkPills a[data-toggle=\"pill\"]').on('shown.bs.tab', $.proxy(this._redrawNetworkCharts, this));\n\n\t\t// Add listener to redraw outgoing messages queue charts on tab change\n\t\t// (to work around an issue with the chart resizing to default width when pane is hidden)\n\t\t$('#dsh-msgQueuePills a[data-toggle=\"pill\"]').on('shown.bs.tab', $.proxy(this._redrawOutMsgCharts, this));\n\n\t\t// Add listener to redraw all charts in case of window resize\n\t\t$(window).on('resize', $.proxy(this._onWindowResize, this));\n\t\tthis._onWindowResize(); // Also do it immediately\n\n\t\t// Add listener to show help tooltips in queues status tab\n\t\t$('#dsh-queues').kendoTooltip({\n\t\t\tfilter: 'i[title].help',\n\t\t\tposition: 'right',\n\t\t\twidth: '250px',\n\t\t\tcontent: function(e) {\n\t\t\t\treturn `<div class=\"help-tooltip\">${e.target.data('title')}</div>`;\n\t\t\t}\n\t\t});\n\n\t\t// Add listener to regenerate link to open web server external manager console\n\t\t$('#dsh-useHttpsCheck').on('change', $.proxy(this._onUseHttpsForWsManagerChange, this));\n\n\t\t// Send initialization request\n\t\tthis.sendExtensionRequest(this.REQ_INIT);\n\t}\n\n\tdestroy()\n\t{\n\t\t// Call super method\n\t\tsuper.destroy();\n\n\t\t// Destroy scrolling tabs\n\t\t$('#dsh-tabNavigator #tabs').scrollingTabs('destroy');\n\n\t\t// Remove other listeners\n\t\t$('#dsh-resetBt').off('click');\n\t\t$('#dsh-pauseBt').off('click');\n\t\t$('#dsh-playBt').off('click');\n\t\t$('#dsh-dumpThreadsBt').off('click');\n\t\t$('a[data-toggle=\"tab\"]').off('shown.bs.tab');\n\t\t$('a[data-toggle=\"pill\"]').off('shown.bs.tab');\n\t\t$(window).off('resize');\n\t\t$('#dsh-useHttpsCheck').off('change');\n\n\t\t// Remove thread dump panel listener\n\t\t$('#dsh-threadDumpModal').off('hidden.bs.modal');\n\n\t\t// Clear request scheduling\n\t\tclearTimeout(this._requestTimer);\n\t}\n\n\tonExtensionCommand(command, data)\n\t{\n\t\t// Initialization data received\n\t\tif (command == this.RESP_INIT)\n\t\t{\n\t\t\t// Server sends flags indicating if web server's internal manager is accessible and how\n\t\t\tthis._wsManagerConfig = {\n\t\t\t\thasAccess: data.getBool('hasWsManager'),\n\t\t\t\thttpEnabled: data.getBool('httpEnabled'),\n\t\t\t\thttpsEnabled: data.getBool('httpsEnabled'),\n\t\t\t\thttpPort: data.getInt('httpPort'),\n\t\t\t\thttpsPort: data.getInt('httpsPort'),\n\t\t\t}\n\n\t\t\t// Enable access to web server manager console\n\t\t\tthis._enableWebServerManagerAccess();\n\n\t\t\t// Request data to server\n\t\t\tthis._requestData(true);\n\t\t}\n\n\t\t// Data received\n\t\telse if (command == this.RESP_DATA)\n\t\t{\n\t\t\t// CPU LOAD CHART\n\n\t\t\tthis._cpuUsageChart.addEntry(data.getDouble('pCpu'), data.getDouble('sCpu'));\n\t\t\t$('#dsh-systemLoadLabel').text(data.getDouble('sCpu'));\n\t\t\t$('#dsh-processLoadLabel').text(data.getDouble('pCpu'));\n\n\t\t\t// As system cpu load is not always available (for example on MacOS X), let's hide it in case it isn't\n\t\t\tif (data.getDouble('sCpu') < 0)\n\t\t\t\t$('#dsh-systemLoadLegend').hide();\n\n\t\t\t// THREADS COUNT CHART\n\n\t\t\tconst threadCount = data.getSFSArray('thList').size();\n\t\t\tthis._threadCountChart.addEntry(threadCount);\n\t\t\t$('#dsh-threadsCountLabel').text(threadCount);\n\n\t\t\t// MEMORY USAGE CHART\n\t\t\t// NOTE: memory data is sent as Bytes by the server\n\n\t\t\tconst freeMem = data.getLong('freeMem');\n\t\t\tconst maxMem = data.getLong('maxMem');\n\t\t\tconst totalMem = data.getLong('totalMem');\n\t\t\tconst usedMem = totalMem - freeMem;\n\n\t\t\tthis._memoryChart.addEntry(totalMem, usedMem);\n\t\t\t$('#dsh-maxMemoryLabel').text(this._getMemObject(maxMem).label);\n\t\t\t$('#dsh-freeMemoryLabel').text(this._getMemObject(freeMem).label);\n\n\t\t\t// THREADS LIST\n\n\t\t\tconst totalThreadCpuTime = data.getLong('thCpu');\n\t\t\tlet threads = [];\n\n\t\t\tfor (let i = 0; i < data.getSFSArray('thList').size(); i++)\n\t\t\t{\n\t\t\t\tlet threadData = data.getSFSArray('thList').getSFSObject(i);\n\n\t\t\t\tthreads.push({\n\t\t\t\t\tid: threadData.getLong('id'),\n\t\t\t\t\tname: threadData.getUtfString('name'),\n\t\t\t\t\tcpuTime: Math.round(threadData.getLong('cpu') / totalThreadCpuTime * 1000) / 10\n\t\t\t\t});\n\t\t\t}\n\n\t\t\t// Substitute grid's data source, retaining current sorting\n\t\t\tlet currSort = this._threadsGrid.dataSource.sort();\n\t\t\tthis._threadsGrid.setDataSource(this._getThreadsGridDataSource(threads, currSort));\n\n\t\t\t// NETWORK TRAFFIC\n\t\t\t// NOTE: network traffic data is sent as KBytes by the server\n\n\t\t\tif (data.containsKey('ntOutHistory'))\n\t\t\t{\n\t\t\t\tconst outgoingData = data.getLongArray('ntOutHistory');\n\t\t\t\tconst incomingData = data.getLongArray('ntInHistory');\n\t\t\t\tconst samplingRateMins = data.getInt('ntRate');\n\t\t\t\tconst shiftSeconds = data.getInt('ntShift');\n\n\t\t\t\tthis._network24hChart.addHistoryEntries(outgoingData, incomingData, samplingRateMins * 60, shiftSeconds);\n\t\t\t}\n\n\t\t\t// Add current values\n\t\t\tthis._network24hChart.addEntry(data.getLong('ntOut'), data.getLong('ntIn'));\n\t\t\tthis._networkRTChart.addEntry(data.getLong('ntOut'), data.getLong('ntIn'));\n\n\t\t\t// TRAFFIC DETAILS\n\n\t\t\t// Sessions\n\t\t\t$('#dsh-detSessionsCurrent').text('Current: ' + (data.getInt('sSocket') + data.getInt('sHttp')));\n\t\t\t$('#dsh-detSessionsMax').text('Max: ' + data.getInt('ccsMax'));\n\t\t\t$('#dsh-detSessionsSocket').text('Socket: ' + data.getInt('sSocket'));\n\t\t\t$('#dsh-detSessionsHttp').text('Http: ' + data.getInt('sHttp'));\n\n\t\t\t// Users\n\t\t\t$('#dsh-detUsersCurrent').text('Current: ' + (data.getInt('uSocket') + data.getInt('uHttp') + data.getInt('uNpc')));\n\t\t\t$('#dsh-detUsersMax').text('Max: ' + data.getInt('ccuMax'));\n\t\t\t$('#dsh-detUsersSocket').text('Socket: ' + data.getInt('uSocket'));\n\t\t\t$('#dsh-detUsersHttp').text('Http: ' + data.getInt('uHttp'));\n\t\t\t$('#dsh-detUsersNpc').text('Npc: ' + data.getInt('uNpc'));\n\n\t\t\t// Rooms\n\t\t\t$('#dsh-detRooms').text(data.getInt('rTotal'));\n\t\t\t$('#dsh-detRoomsRegular').text('Regular: ' + (data.getInt('rTotal') - data.getInt('rGame')));\n\t\t\t$('#dsh-detRoomsGame').text('Game: ' + data.getInt('rGame'));\n\n\t\t\t// Total data transfer\n\t\t\t$('#dsh-detDataTransferOut').text('Out: ' + kBytesToSize(data.getLong('ntOut')));\n\t\t\t$('#dsh-detDataTransferIn').text('In: ' + kBytesToSize(data.getLong('ntIn')));\n\n\t\t\t// Current data transfer rate\n\t\t\tlet obj = this._networkRTChart.getLastEntry();\n\n\t\t\tif (obj == null)\n\t\t\t\tobj = {outgoing: 0, incoming: 0};\n\n\t\t\t$('#dsh-detTransferRateOut').text('Out: ' + kBytesToSize(obj.outgoing, 2, 'KB', '/s'));\n\t\t\t$('#dsh-detTransferRateIn').text('In: ' + kBytesToSize(obj.incoming, 2, 'KB', '/s'));\n\n\t\t\t// Average data transfer rate over whole uptime\n\t\t\tif (this._uptimeSecs > 0)\n\t\t\t{\n\t\t\t\tconst avgOut = Math.round(data.getLong('ntOut') / this._uptimeSecs * 100) / 100;\n\t\t\t\tconst avgIn = Math.round(data.getLong('ntIn') / this._uptimeSecs * 100) / 100;\n\n\t\t\t\t$('#dsh-detAvgTransferRateOut').text('Out: ' + kBytesToSize(avgOut, 2, 'KB', '/s'));\n\t\t\t\t$('#dsh-detAvgTransferRateIn').text('In: ' + kBytesToSize(avgIn, 2, 'KB', '/s'));\n\t\t\t}\n\n\t\t\t// Dropped packets\n\t\t\tconst totalPacketsOut = data.getLong('pktOut');\n\n\t\t\tconst totalDroppedTcpPacketsOut = data.getLong('pktOutDrop');\n\t\t\t$('#dsh-detDroppedTcpPacketsOut').text(`Out TCP: ${totalDroppedTcpPacketsOut} (${Math.round(totalDroppedTcpPacketsOut / totalPacketsOut * 100)}%)`);\n\n\t\t\tconst totalDroppedUdpPacketsOut = data.getLong('pktOutUdpDrop');\n\t\t\t$('#dsh-detDroppedUdpPacketsOut').text(`Out UDP: ${totalDroppedUdpPacketsOut} (${Math.round(totalDroppedUdpPacketsOut / totalPacketsOut * 100)}%)`);\n\n\t\t\tconst totalPacketsIn = data.getLong('pktIn');\n\t\t\tconst totalDroppedPacketsIn = data.getLong('pktInDrop');\n\t\t\t$('#dsh-detDroppedPacketsIn').text(`In: ${totalDroppedPacketsIn} (${Math.round(totalDroppedPacketsIn / totalPacketsIn * 100)}%)`);\n\n\t\t\t// SYSTEM STATUS\n\n\t\t\t// Outgoing TCP messages queue\n\t\t\tconst outMsgQueue = data.getInt('outMsgQueue');\n\t\t\tconst outMsgQueueMax = data.getInt('outMsgQueueMax');\n\t\t\tthis._outTcpMsgQueueChart.addEntry(outMsgQueue, outMsgQueueMax);\n\t\t\tthis._tcpOutThreads = data.getInt('outMsgTh');\n\t\t\tthis._tcpOutLoad = this._outTcpMsgQueueChart.load;\n\n\t\t\t// Outgoing UDP messages queue\n\t\t\tconst outUdpMsgQueue = data.getInt('outUdpMsgQueue');\n\t\t\tconst outUdpMsgQueueMax = data.getInt('outUdpMsgQueueMax');\n\t\t\tthis._outUdpMsgQueueChart.addEntry(outUdpMsgQueue, outUdpMsgQueueMax);\n\t\t\tthis._udpOutThreads = data.getInt('outUdpMsgTh');\n\t\t\tthis._udpOutLoad = this._outUdpMsgQueueChart.load;\n\n\t\t\tthis._updateOutMessagesQueueStatus();\n\n\t\t\t// System Controller requests queue\n\t\t\tconst sysCtrlQueue = data.getInt('sysCtrlQueue');\n\t\t\tconst sysCtrlQueueMax = data.getInt('sysCtrlQueueMax');\n\t\t\tthis._sysQueueChart.addEntry(sysCtrlQueue, sysCtrlQueueMax);\n\t\t\t$('#dsh-systemQueueThreads').text(this._getQueueThreadsLabel(data.getInt('sysCtrlTh')));\n\t\t\t$('#dsh-systemQueueLoad').text(this._getQueueLoadLabel(this._sysQueueChart.load));\n\n\t\t\t// Extension Controller requests queue\n\t\t\tconst extCtrlQueue = data.getInt('extCtrlQueue');\n\t\t\tconst extCtrlQueueMax = data.getInt('extCtrlQueueMax');\n\t\t\tthis._extQueueChart.addEntry(extCtrlQueue, extCtrlQueueMax);\n\t\t\t$('#dsh-extensionQueueThreads').text(this._getQueueThreadsLabel(data.getInt('extCtrlTh')));\n\t\t\t$('#dsh-extensionQueueLoad').text(this._getQueueLoadLabel(this._sysQueueChart.load));\n\n\t\t\t// Evaluate overall system workload\n\t\t\tlet sysWorkload = Math.round((outMsgQueue / outMsgQueueMax +\n\t\t\t\t\t\t\t\t\t\t  outUdpMsgQueue / outUdpMsgQueueMax +\n\t\t\t\t \t\t\t\t\t\t  sysCtrlQueue / sysCtrlQueueMax +\n\t\t\t\t \t\t\t\t\t  \t  extCtrlQueue / extCtrlQueueMax) * 100 / 4);\n\t\t\tif (sysWorkload > 100)\n\t\t\t\tsysWorkload = 100;\n\n\t\t\tthis._sysWorkloadChart.addEntry(sysWorkload);\n\t\t}\n\n\t\t// Thread dump received\n\t\telse if (command == this.RESP_THREAD_DUMP)\n\t\t{\n\t\t\t// Enable button\n\t\t\t$('#dsh-dumpThreadsBt').attr('disabled', false);\n\n\t\t\t// Set result\n\t\t\t$('#dsh-threadDumpResult').text(data.getText('dump'));\n\n\t\t\t// Display modal\n\t\t\t$('#dsh-threadDumpModal').modal('show');\n\t\t}\n\t}\n\n\tonUptimeUpdated(values)\n\t{\n\t\tthis._uptimeSecs = values[3];\n\t\tthis._uptimeSecs += (values[2] * 60);\n\t\tthis._uptimeSecs += (values[1] * 60 * 60);\n\t\tthis._uptimeSecs += (values[0] * 24 * 60 * 60);\n\n\t\t$('#dsh-uptimeDays').text( (values[0] <= 9 ? '00' : (values[0] <= 99 ? '0' : '')) + values[0] );\n\t\t$('#dsh-uptimeHours').text( (values[1] <= 9 ? '0' : '') + values[1] );\n\t\t$('#dsh-uptimeMinutes').text( (values[2] <= 9 ? '0' : '') + values[2] );\n\t\t$('#dsh-uptimeSeconds').text( (values[3] <= 9 ? '0' : '') + values[3] );\n\t}\n\n\t//---------------------------------\n\t// UI EVENT LISTENERS\n\t//---------------------------------\n\n\t_onUpdateIntervalChange()\n\t{\n\t\t// Request data to server\n\t\tthis._requestData();\n\t}\n\n\t_onPauseUpdateClick()\n\t{\n\t\tthis._setChartUpdatePaused(true);\n\n\t\t// Hide pause button and show play button\n\t\t$('#dsh-pauseBt').addClass('hidden');\n\t\t$('#dsh-playBt').removeClass('hidden');\n\t}\n\n\t_onPlayUpdateClick()\n\t{\n\t\tthis._setChartUpdatePaused(false);\n\n\t\t// Hide play button and show pause button\n\t\t$('#dsh-pauseBt').removeClass('hidden');\n\t\t$('#dsh-playBt').addClass('hidden');\n\t}\n\n\t_onResetChartsClick()\n\t{\n\t\t// Reactivate realtime charts update\n\t\tthis._onPlayUpdateClick();\n\n\t\t// Reset realtime charts\n\t\tthis._cpuUsageChart.reset();\n\t\tthis._threadCountChart.reset();\n\t\tthis._memoryChart.reset();\n\t\tthis._networkRTChart.reset();\n\t\tthis._sysWorkloadChart.reset();\n\t\tthis._outTcpMsgQueueChart.reset();\n\t\tthis._outUdpMsgQueueChart.reset();\n\t\tthis._sysQueueChart.reset();\n\t\tthis._extQueueChart.reset();\n\t}\n\n\t_redrawNetworkCharts()\n\t{\n\t\t// Redraw charts\n\t\tthis._networkRTChart.redraw();\n\t\tthis._network24hChart.redraw();\n\n\t\t// Redraw time range slider\n\t\tthis._timeSlider.resize();\n\t}\n\n\t_redrawOutMsgCharts()\n\t{\n\t\t// Redraw charts\n\t\tthis._outTcpMsgQueueChart.redraw();\n\t\tthis._outUdpMsgQueueChart.redraw();\n\n\t\tthis._updateOutMessagesQueueStatus();\n\t}\n\n\t_onWindowResize()\n\t{\n\t\t// Redraw all global charts\n\t\tthis._cpuUsageChart.redraw();\n\t\tthis._threadCountChart.redraw();\n\t\tthis._memoryChart.redraw();\n\t\tthis._redrawNetworkCharts();\n\n\t\t// Redraw all queue charts\n\t\tthis._sysWorkloadChart.redraw();\n\t\tthis._sysQueueChart.redraw();\n\t\tthis._extQueueChart.redraw();\n\t\tthis._redrawOutMsgCharts();\n\t}\n\n\t_onTimeRangeChange()\n\t{\n\t\tlet values = this._timeSlider.value();\n\n\t\tif (values[0] == values[1])\n\t\t{\n\t\t\tif (values[1] == -24)\n\t\t\t\tvalues[1] = -23;\n\n\t\t\tvalues[0] = values[1] - 1;\n\n\t\t\t// Reset the time range slider value (we need to use setTimeout\n\t\t\t// bacause doing it in the change event listener doesn't redraw the slider)\n\t\t\tsetTimeout($.proxy( function(values) { this._timeSlider.value(values); }, this), 10, values);\n\t\t}\n\n\t\tthis._network24hChart.range = values;\n\t}\n\n\t_onUseHttpsForWsManagerChange()\n\t{\n\t\tconst wsMan = this._wsManagerConfig;\n\n\t\tlet url = ($('#dsh-useHttpsCheck').prop('checked') ? 'https' : 'http') + '://';\n\t\turl += this.smartFox.config.host;\n\t\turl += ':' + ($('#dsh-useHttpsCheck').prop('checked') ? wsMan.httpsPort : wsMan.httpPort);\n\t\turl += '/manager/status';\n\n\t\t$('#dsh-toWsManagerBt').attr('href', url);\n\t}\n\n\t_onDumpThreadsClick()\n\t{\n\t\t// Disable button\n\t\t$('#dsh-dumpThreadsBt').attr('disabled', true);\n\n\t\t// Send request to extension\n\t\tthis.sendExtensionRequest(this.REQ_DUMP_THREADS);\n\t}\n\n\t_onCopyThreadDumpClick()\n\t{\n\t\t$('#dsh-threadDumpResult').attr('contenteditable', true);\n\t\t$('#dsh-threadDumpResult').attr('spellcheck', false);\n\t\t$('#dsh-threadDumpResult').focus();\n\t\tdocument.execCommand('selectAll');\n\t\tdocument.execCommand('copy');\n\t\twindow.getSelection().removeAllRanges();\n\t\t$('#dsh-threadDumpResult').blur();\n\t\t$('#dsh-threadDumpResult').attr('contenteditable', false);\n\t}\n\n\t_onThreadDumpModalHidden()\n\t{\n\t\t// Clear modal content\n\t\t$('#dsh-threadDumpResult').text('');\n\t}\n\n\t//------------------------------------\n\t// PRIVATE METHODS\n\t//------------------------------------\n\n\t_enableWebServerManagerAccess()\n\t{\n\t\tconst wsMan = this._wsManagerConfig;\n\n\t\tif (wsMan.hasAccess && (wsMan.httpEnabled || wsMan.httpsEnabled))\n\t\t{\n\t\t\t// Hide alert\n\t\t\t$('#dsh-wsManagerAlert').hide();\n\n\t\t\t// Enable button\n\t\t\t$('#dsh-toWsManagerBt').attr('disabled', false);\n\n\t\t\t// Set checkbox\n\t\t\t$('#dsh-useHttpsCheck').prop('checked', wsMan.httpsEnabled);\n\t\t\t$('#dsh-useHttpsCheck').attr('disabled', !wsMan.httpEnabled || !wsMan.httpsEnabled);\n\n\t\t\t// Generate link\n\t\t\tthis._onUseHttpsForWsManagerChange();\n\t\t}\n\t\telse\n\t\t{\n\t\t\t// Show alert\n\t\t\t$('#dsh-wsManagerAlert').show();\n\n\t\t\tif (!wsMan.hasAccess)\n\t\t\t\t$('#dsh-wsManagerAlert').html('Access to Tomcat Manager console requires at least one user to be defined in <em>Web Server</em> tab of <strong>Server Configurator</strong> module.');\n\t\t\telse\n\t\t\t\t$('#dsh-wsManagerAlert').html('Access to Tomcat Manager console requires either HTTP or HTTPS to be enabled in <em>Web Server</em> tab of <strong>Server Configurator</strong> module.');\n\n\t\t\t// Disable button\n\t\t\t$('#dsh-toWsManagerBt').attr('disabled', true);\n\n\t\t\t// Disable checkbox\n\t\t\t$('#dsh-useHttpsCheck').attr('checked', wsMan.httpsEnabled);\n\t\t\t$('#dsh-useHttpsCheck').attr('disabled', true);\n\t\t}\n\t}\n\n\t/**\n\t * Build the polling request to be sent to the server.\n\t */\n\t_requestData(getHistory = false)\n\t{\n\t\t// Clear previous request scheduling\n\t\tclearTimeout(this._requestTimer);\n\n\t\t// Check if connection is still available\n\t\tif (this.smartFox.isConnected)\n\t\t{\n\t\t\t// Build request parameters\n\t\t\tlet params = new SFS2X.SFSObject();\n\n\t\t\tif (getHistory)\n\t\t\t\tparams.putBool('history', true);\n\n\t\t\t// Send request to extension\n\t\t\tthis.sendExtensionRequest(this.REQ_GET_DATA, params);\n\n\t\t\t// Schedule next request\n\t\t\tthis._requestTimer = setTimeout($.proxy(this._requestData, this), Number(this._intervalDropDown.value()) * 1000);\n\t\t}\n\t}\n\n\t_getMemObject(bytes)\n\t{\n\t\treturn {\n\t\t\tdata: bytes,\n\t\t\tlabel: bytesToSize(bytes)\n\t\t};\n\t}\n\n\t_setChartUpdatePaused(paused)\n\t{\n\t\t// Pause/reactivate realtime charts\n\t\tthis._cpuUsageChart.paused = paused;\n\t\tthis._threadCountChart.paused = paused;\n\t\tthis._memoryChart.paused = paused;\n\t\tthis._networkRTChart.paused = paused;\n\t\tthis._sysWorkloadChart.paused = paused;\n\t\tthis._outTcpMsgQueueChart.paused = paused;\n\t\tthis._sysQueueChart.paused = paused;\n\t\tthis._extQueueChart.paused = paused;\n\t}\n\n\t_getThreadsGridDataSource(items, sort)\n\t{\n\t\tif (!items)\n\t\t\titems = '';\n\n\t\tif (!sort)\n\t\t{\n\t\t\tsort = {\n\t\t\t\tfield: 'cpuTime',\n\t\t\t\tdir: 'desc'\n\t\t\t};\n\t\t}\n\n\t\treturn new kendo.data.DataSource({\n\t\t\tschema: {\n\t\t\t\tmodel: {\n\t\t\t\t\tid: 'id'\n\t\t\t\t}\n\t\t\t},\n\t\t\tsort: sort,\n\t\t\tdata: items\n\t\t});\n\t}\n\n\t_getQueueThreadsLabel(threadsNum)\n\t{\n\t\treturn `Threads: ${threadsNum}`;\n\t}\n\n\t_getQueueLoadLabel(load)\n\t{\n\t\treturn `Avg. messages: ${load}`;\n\t}\n\n\t_updateOutMessagesQueueStatus()\n\t{\n\t\tlet threadsStr = this._getQueueThreadsLabel('--');\n\t\tlet loadStr = this._getQueueLoadLabel('--');\n\n\t\tif ($('#dsh-tcpPill').hasClass('active'))\n\t\t{\n\t\t\tthreadsStr = this._getQueueThreadsLabel(this._tcpOutThreads);\n\t\t\tloadStr = this._getQueueLoadLabel(this._tcpOutLoad);\n\t\t}\n\n\t\tif ($('#dsh-udpPill').hasClass('active'))\n\t\t{\n\t\t\tthreadsStr = this._getQueueThreadsLabel(this._udpOutThreads);\n\t\t\tloadStr = this._getQueueLoadLabel(this._udpOutLoad);\n\t\t}\n\n\t\t$('#dsh-outgoingMessagesQueueThreads').text(threadsStr);\n\t\t$('#dsh-outgoingMessagesQueueLoad').text(loadStr);\n\t}\n\n\t//---------------------------------\n\t// PRIVATE GETTERS\n\t//---------------------------------\n\n\n}\n"],"mappings":";;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;AC1HA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;AC1JA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;ACxGA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;AC3IA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;ACrNA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;AC/JA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;AC5PA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;ACrPA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;ACzFA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;A","sourceRoot":""}