admin.js 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504
  1. jQuery(function($){
  2. // http://stackoverflow.com/questions/24816/escaping-html-strings-with-jquery
  3. var entityMap = {
  4. '&': '&',
  5. '<': '&lt;',
  6. '>': '&gt;',
  7. '"': '&quot;',
  8. "'": '&#39;',
  9. '/': '&#x2F;',
  10. '`': '&#x60;',
  11. '=': '&#x3D;'
  12. };
  13. function escapeHtml(string) {
  14. return String(string).replace(/[&<>"'`=\/]/g, function (s) {
  15. return entityMap[s];
  16. });
  17. }
  18. function humanFileSize(bytes) {
  19. if(Math.abs(bytes) < 1024) {
  20. return bytes + ' B';
  21. }
  22. var units = ['KiB','MiB','GiB','TiB','PiB','EiB','ZiB','YiB'];
  23. var u = -1;
  24. do {
  25. bytes /= 1024;
  26. ++u;
  27. } while(Math.abs(bytes) >= 1024 && u < units.length - 1);
  28. return bytes.toFixed(1)+' '+units[u];
  29. }
  30. $("#refresh_postfix_log").on('click', function(e) {
  31. e.preventDefault();
  32. draw_postfix_logs();
  33. });
  34. $("#refresh_dovecot_log").on('click', function(e) {
  35. e.preventDefault();
  36. draw_dovecot_logs();
  37. });
  38. $("#refresh_sogo_log").on('click', function(e) {
  39. e.preventDefault();
  40. draw_sogo_logs();
  41. });
  42. $("#refresh_fail2ban_log").on('click', function(e) {
  43. e.preventDefault();
  44. draw_fail2ban_logs();
  45. });
  46. $("#refresh_rspamd_history").on('click', function(e) {
  47. e.preventDefault();
  48. draw_rspamd_history();
  49. });
  50. function draw_postfix_logs() {
  51. ft_postfix_logs = FooTable.init('#postfix_log', {
  52. "columns": [
  53. {"name":"time","formatter":function unix_time_format(tm) { var date = new Date(tm ? tm * 1000 : 0); return date.toLocaleString();},"title":lang.time,"style":{"width":"170px"}},
  54. {"name":"priority","title":lang.priority,"style":{"width":"80px"}},
  55. {"name":"message","title":lang.message},
  56. ],
  57. "rows": $.ajax({
  58. dataType: 'json',
  59. url: '/api/v1/get/logs/postfix/1000',
  60. jsonp: false,
  61. error: function () {
  62. console.log('Cannot draw postfix log table');
  63. },
  64. success: function (data) {
  65. $.each(data, function (i, item) {
  66. item.message = escapeHtml(item.message);
  67. var danger_class = ["emerg", "alert", "crit", "err"];
  68. var warning_class = ["warning", "warn"];
  69. var info_class = ["notice", "info", "debug"];
  70. if (jQuery.inArray(item.priority, danger_class) !== -1) {
  71. item.priority = '<span class="label label-danger">' + item.priority + '</span>';
  72. }
  73. else if (jQuery.inArray(item.priority, warning_class) !== -1) {
  74. item.priority = '<span class="label label-warning">' + item.priority + '</span>';
  75. }
  76. else if (jQuery.inArray(item.priority, info_class) !== -1) {
  77. item.priority = '<span class="label label-info">' + item.priority + '</span>';
  78. }
  79. });
  80. }
  81. }),
  82. "empty": lang.empty,
  83. "paging": {
  84. "enabled": true,
  85. "limit": 5,
  86. "size": log_pagination_size
  87. },
  88. "filtering": {
  89. "enabled": true,
  90. "position": "left",
  91. "placeholder": lang.filter_table
  92. },
  93. "sorting": {
  94. "enabled": true
  95. }
  96. });
  97. }
  98. function draw_fail2ban_logs() {
  99. ft_fail2ban_logs = FooTable.init('#fail2ban_log', {
  100. "columns": [
  101. {"name":"time","formatter":function unix_time_format(tm) { var date = new Date(tm ? tm * 1000 : 0); return date.toLocaleString();},"title":lang.time,"style":{"width":"170px"}},
  102. {"name":"priority","title":lang.priority,"style":{"width":"80px"}},
  103. {"name":"message","title":lang.message},
  104. ],
  105. "rows": $.ajax({
  106. dataType: 'json',
  107. url: '/api/v1/get/logs/fail2ban/1000',
  108. jsonp: false,
  109. error: function () {
  110. console.log('Cannot draw fail2ban log table');
  111. },
  112. success: function (data) {
  113. $.each(data, function (i, item) {
  114. var danger_class = ["emerg", "alert", "crit", "err"];
  115. var warning_class = ["warning", "warn"];
  116. var info_class = ["notice", "info", "debug"];
  117. item.message = escapeHtml(item.message);
  118. if (jQuery.inArray(item.priority, danger_class) !== -1) {
  119. item.priority = '<span class="label label-danger">' + item.priority + '</span>';
  120. }
  121. else if (jQuery.inArray(item.priority, warning_class) !== -1) {
  122. item.priority = '<span class="label label-warning">' + item.priority + '</span>';
  123. }
  124. else if (jQuery.inArray(item.priority, info_class) !== -1) {
  125. item.priority = '<span class="label label-info">' + item.priority + '</span>';
  126. }
  127. });
  128. }
  129. }),
  130. "empty": lang.empty,
  131. "paging": {
  132. "enabled": true,
  133. "limit": 5,
  134. "size": log_pagination_size
  135. },
  136. "filtering": {
  137. "enabled": true,
  138. "position": "left",
  139. "placeholder": lang.filter_table
  140. },
  141. "sorting": {
  142. "enabled": true
  143. }
  144. });
  145. }
  146. function draw_sogo_logs() {
  147. ft_sogo_logs = FooTable.init('#sogo_log', {
  148. "columns": [
  149. {"name":"time","formatter":function unix_time_format(tm) { var date = new Date(tm ? tm * 1000 : 0); return date.toLocaleString();},"title":lang.time,"style":{"width":"170px"}},
  150. {"name":"priority","title":lang.priority,"style":{"width":"80px"}},
  151. {"name":"message","title":lang.message},
  152. ],
  153. "rows": $.ajax({
  154. dataType: 'json',
  155. url: '/api/v1/get/logs/sogo/1000',
  156. jsonp: false,
  157. error: function () {
  158. console.log('Cannot draw sogo log table');
  159. },
  160. success: function (data) {
  161. $.each(data, function (i, item) {
  162. var danger_class = ["emerg", "alert", "crit", "err"];
  163. var warning_class = ["warning", "warn"];
  164. var info_class = ["notice", "info", "debug"];
  165. item.message = escapeHtml(item.message);
  166. if (jQuery.inArray(item.priority, danger_class) !== -1) {
  167. item.priority = '<span class="label label-danger">' + item.priority + '</span>';
  168. }
  169. else if (jQuery.inArray(item.priority, warning_class) !== -1) {
  170. item.priority = '<span class="label label-warning">' + item.priority + '</span>';
  171. }
  172. else if (jQuery.inArray(item.priority, info_class) !== -1) {
  173. item.priority = '<span class="label label-info">' + item.priority + '</span>';
  174. }
  175. });
  176. }
  177. }),
  178. "empty": lang.empty,
  179. "paging": {
  180. "enabled": true,
  181. "limit": 5,
  182. "size": log_pagination_size
  183. },
  184. "filtering": {
  185. "enabled": true,
  186. "position": "left",
  187. "placeholder": lang.filter_table
  188. },
  189. "sorting": {
  190. "enabled": true
  191. }
  192. });
  193. }
  194. function draw_dovecot_logs() {
  195. ft_postfix_logs = FooTable.init('#dovecot_log', {
  196. "columns": [
  197. {"name":"time","formatter":function unix_time_format(tm) { var date = new Date(tm ? tm * 1000 : 0); return date.toLocaleString();},"title":lang.time,"style":{"width":"170px"}},
  198. {"name":"priority","title":lang.priority,"style":{"width":"80px"}},
  199. {"name":"message","title":lang.message},
  200. ],
  201. "rows": $.ajax({
  202. dataType: 'json',
  203. url: '/api/v1/get/logs/dovecot/1000',
  204. jsonp: false,
  205. error: function () {
  206. console.log('Cannot draw dovecot log table');
  207. },
  208. success: function (data) {
  209. $.each(data, function (i, item) {
  210. var danger_class = ["emerg", "alert", "crit", "err"];
  211. var warning_class = ["warning", "warn"];
  212. var info_class = ["notice", "info", "debug"];
  213. item.message = escapeHtml(item.message);
  214. if (jQuery.inArray(item.priority, danger_class) !== -1) {
  215. item.priority = '<span class="label label-danger">' + item.priority + '</span>';
  216. }
  217. else if (jQuery.inArray(item.priority, warning_class) !== -1) {
  218. item.priority = '<span class="label label-warning">' + item.priority + '</span>';
  219. }
  220. else if (jQuery.inArray(item.priority, info_class) !== -1) {
  221. item.priority = '<span class="label label-info">' + item.priority + '</span>';
  222. }
  223. });
  224. }
  225. }),
  226. "empty": lang.empty,
  227. "paging": {
  228. "enabled": true,
  229. "limit": 5,
  230. "size": log_pagination_size
  231. },
  232. "filtering": {
  233. "enabled": true,
  234. "position": "left",
  235. "placeholder": lang.filter_table
  236. },
  237. "sorting": {
  238. "enabled": true
  239. }
  240. });
  241. }
  242. function draw_domain_admins() {
  243. ft_domainadmins = FooTable.init('#domainadminstable', {
  244. "columns": [
  245. {"name":"chkbox","title":"","style":{"maxWidth":"40px","width":"40px"},"filterable": false,"sortable": false,"type":"html"},
  246. {"sorted": true,"name":"username","title":lang.username,"style":{"width":"250px"}},
  247. {"name":"selected_domains","title":lang.admin_domains,"breakpoints":"xs sm"},
  248. {"name":"tfa_active","title":"TFA", "filterable": false,"style":{"maxWidth":"80px","width":"80px"}},
  249. {"name":"active","filterable": false,"style":{"maxWidth":"80px","width":"80px"},"title":lang.active},
  250. {"name":"action","filterable": false,"sortable": false,"style":{"text-align":"right","maxWidth":"180px","width":"180px"},"type":"html","title":lang.action,"breakpoints":"xs sm"}
  251. ],
  252. "rows": $.ajax({
  253. dataType: 'json',
  254. url: '/api/v1/get/domain-admin/all',
  255. jsonp: false,
  256. error: function () {
  257. console.log('Cannot draw domain admin table');
  258. },
  259. success: function (data) {
  260. $.each(data, function (i, item) {
  261. item.chkbox = '<input type="checkbox" data-id="domain_admins" name="multi_select" value="' + item.username + '" />';
  262. item.action = '<div class="btn-group">' +
  263. '<a href="/edit.php?domainadmin=' + encodeURI(item.username) + '" class="btn btn-xs btn-default"><span class="glyphicon glyphicon-pencil"></span> ' + lang.edit + '</a>' +
  264. '<a href="#" id="delete_selected" data-id="single-domain-admin" data-api-url="delete/domain-admin" data-item="' + encodeURI(item.username) + '" class="btn btn-xs btn-danger"><span class="glyphicon glyphicon-trash"></span> ' + lang.remove + '</a>' +
  265. '</div>';
  266. });
  267. }
  268. }),
  269. "empty": lang.empty,
  270. "paging": {
  271. "enabled": true,
  272. "limit": 5,
  273. "size": log_pagination_size
  274. },
  275. "filtering": {
  276. "enabled": true,
  277. "position": "left",
  278. "placeholder": lang.filter_table
  279. },
  280. "sorting": {
  281. "enabled": true
  282. }
  283. });
  284. }
  285. function draw_fwd_hosts() {
  286. ft_forwardinghoststable = FooTable.init('#forwardinghoststable', {
  287. "columns": [
  288. {"name":"chkbox","title":"","style":{"maxWidth":"40px","width":"40px"},"filterable": false,"sortable": false,"type":"html"},
  289. {"name":"host","type":"text","title":lang.host,"style":{"width":"250px"}},
  290. {"name":"source","title":lang.source,"breakpoints":"xs sm"},
  291. {"name":"keep_spam","title":lang.spamfilter, "type": "text","style":{"maxWidth":"80px","width":"80px"}},
  292. {"name":"action","filterable": false,"sortable": false,"style":{"text-align":"right","maxWidth":"180px","width":"180px"},"type":"html","title":lang.action,"breakpoints":"xs sm"}
  293. ],
  294. "rows": $.ajax({
  295. dataType: 'json',
  296. url: '/api/v1/get/fwdhost/all',
  297. jsonp: false,
  298. error: function () {
  299. console.log('Cannot draw forwarding hosts table');
  300. },
  301. success: function (data) {
  302. $.each(data, function (i, item) {
  303. item.action = '<div class="btn-group">' +
  304. '<a href="#" id="delete_selected" data-id="single-fwdhost" data-api-url="delete/fwdhost" data-item="' + encodeURI(item.host) + '" class="btn btn-xs btn-danger"><span class="glyphicon glyphicon-trash"></span> ' + lang.remove + '</a>' +
  305. '</div>';
  306. if (item.keep_spam == "yes") {
  307. item.keep_spam = lang.no;
  308. }
  309. else {
  310. item.keep_spam = lang.yes;
  311. }
  312. item.chkbox = '<input type="checkbox" data-id="fwdhosts" name="multi_select" value="' + item.host + '" />';
  313. });
  314. }
  315. }),
  316. "empty": lang.empty,
  317. "paging": {
  318. "enabled": true,
  319. "limit": 5,
  320. "size": log_pagination_size
  321. },
  322. "sorting": {
  323. "enabled": true
  324. }
  325. });
  326. }
  327. function draw_rspamd_history() {
  328. ft_postfix_logs = FooTable.init('#rspamd_history', {
  329. "columns": [{
  330. "name":"unix_time",
  331. "formatter":function unix_time_format(tm) { var date = new Date(tm ? tm * 1000 : 0); return date.toLocaleString();},
  332. "title":lang.time,
  333. "style":{
  334. "width":"170px"
  335. }
  336. }, {
  337. "name": "ip",
  338. "title": "IP address",
  339. "breakpoints": "all",
  340. "style": {
  341. "minWidth": 88
  342. }
  343. }, {
  344. "name": "sender_mime",
  345. "title": "From",
  346. "breakpoints": "xs sm md",
  347. "style": {
  348. "minWidth": 100
  349. }
  350. }, {
  351. "name": "rcpt_mime",
  352. "title": "To",
  353. "breakpoints": "xs sm md",
  354. "style": {
  355. "minWidth": 100
  356. }
  357. }, {
  358. "name": "subject",
  359. "title": "Subject",
  360. "breakpoints": "all",
  361. "style": {
  362. "word-break": "break-all",
  363. "minWidth": 150
  364. }
  365. }, {
  366. "name": "action",
  367. "title": "Action",
  368. "style": {
  369. "minwidth": 82
  370. }
  371. }, {
  372. "name": "score",
  373. "title": "Score",
  374. "style": {
  375. "maxWidth": 110
  376. },
  377. }, {
  378. "name": "symbols",
  379. "title": "Symbols",
  380. "breakpoints": "all",
  381. }, {
  382. "name": "size",
  383. "title": "Msg size",
  384. "breakpoints": "all",
  385. "style": {
  386. "minwidth": 50,
  387. },
  388. "formatter": function(value) { return humanFileSize(value); }
  389. }, {
  390. "name": "scan_time",
  391. "title": "Scan time",
  392. "breakpoints": "all",
  393. "style": {
  394. "maxWidth": 72
  395. },
  396. }, {
  397. "name": "message-id",
  398. "title": "ID",
  399. "breakpoints": "all",
  400. "style": {
  401. "minWidth": 130,
  402. "overflow": "hidden",
  403. "textOverflow": "ellipsis",
  404. "wordBreak": "break-all",
  405. "whiteSpace": "normal"
  406. }
  407. }, {
  408. "name": "user",
  409. "title": "Authenticated user",
  410. "breakpoints": "xs sm md",
  411. "style": {
  412. "minWidth": 100
  413. }
  414. }],
  415. "rows": $.ajax({
  416. dataType: 'json',
  417. url: '/api/v1/get/logs/rspamd-history',
  418. jsonp: false,
  419. error: function () {
  420. console.log('Cannot draw rspamd history table');
  421. },
  422. success: function (data) {
  423. $.each(data, function (i, item) {
  424. item.rcpt_mime = item.rcpt_mime.join(",&#8203;");
  425. Object.keys(item.symbols).map(function(key) {
  426. var sym = item.symbols[key];
  427. if (sym.score <= 0) {
  428. sym.score_formatted = '(<span class="text-success"><b>' + sym.score + '</b></span>)'
  429. }
  430. else {
  431. sym.score_formatted = '(<span class="text-danger"><b>' + sym.score + '</b></span>)'
  432. }
  433. var str = '<strong>' + key + '</strong> ' + sym.score_formatted;
  434. if (sym.options) {
  435. str += ' [' + sym.options.join(",") + "]";
  436. }
  437. item.symbols[key].str = str;
  438. });
  439. item.symbols = Object.keys(item.symbols).
  440. map(function(key) {
  441. return item.symbols[key];
  442. }).sort(function(e1, e2) {
  443. return Math.abs(e1.score) < Math.abs(e2.score);
  444. }).map(function(e) {
  445. return e.str;
  446. }).join("<br>\n");
  447. var scan_time = item.time_real.toFixed(3) + ' / ' + item.time_virtual.toFixed(3);
  448. item.scan_time = {
  449. "options": {
  450. "sortValue": item.time_real
  451. },
  452. "value": scan_time
  453. };
  454. if (item.action === 'clean' || item.action === 'no action') {
  455. item.action = "<div class='label label-success'>" + item.action + "</div>";
  456. } else if (item.action === 'rewrite subject' || item.action === 'add header' || item.action === 'probable spam') {
  457. item.action = "<div class='label label-warning'>" + item.action + "</div>";
  458. } else if (item.action === 'spam' || item.action === 'reject') {
  459. item.action = "<div class='label label-danger'>" + item.action + "</div>";
  460. } else {
  461. item.action = "<div class='label label-info'>" + item.action + "</div>";
  462. }
  463. var score_content;
  464. if (item.score < item.required_score) {
  465. score_content = "[ <span class='text-success'>" + item.score.toFixed(2) + " / " + item.required_score + "</span> ]";
  466. } else {
  467. score_content = "[ <span class='text-danger'>" + item.score.toFixed(2) + " / " + item.required_score + "</span> ]";
  468. }
  469. item.score = {
  470. "options": {
  471. "sortValue": item.score
  472. },
  473. "value": score_content
  474. };
  475. if (item.user == null) {
  476. item.user = "none";
  477. }
  478. });
  479. }
  480. }),
  481. "empty": lang.empty,
  482. "paging": {
  483. "enabled": true,
  484. "limit": 5,
  485. "size": log_pagination_size
  486. },
  487. "filtering": {
  488. "enabled": true,
  489. "position": "left",
  490. "placeholder": lang.filter_table
  491. },
  492. "sorting": {
  493. "enabled": true
  494. }
  495. });
  496. }
  497. draw_postfix_logs();
  498. draw_dovecot_logs();
  499. draw_sogo_logs();
  500. draw_fail2ban_logs();
  501. draw_domain_admins();
  502. draw_fwd_hosts();
  503. draw_rspamd_history();
  504. });