user.js 21 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565
  1. // Base64 functions
  2. var Base64={_keyStr:"ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/=",encode:function(r){var t,e,o,a,h,n,c,d="",C=0;for(r=Base64._utf8_encode(r);C<r.length;)a=(t=r.charCodeAt(C++))>>2,h=(3&t)<<4|(e=r.charCodeAt(C++))>>4,n=(15&e)<<2|(o=r.charCodeAt(C++))>>6,c=63&o,isNaN(e)?n=c=64:isNaN(o)&&(c=64),d=d+this._keyStr.charAt(a)+this._keyStr.charAt(h)+this._keyStr.charAt(n)+this._keyStr.charAt(c);return d},decode:function(r){var t,e,o,a,h,n,c="",d=0;for(r=r.replace(/[^A-Za-z0-9\+\/\=]/g,"");d<r.length;)t=this._keyStr.indexOf(r.charAt(d++))<<2|(a=this._keyStr.indexOf(r.charAt(d++)))>>4,e=(15&a)<<4|(h=this._keyStr.indexOf(r.charAt(d++)))>>2,o=(3&h)<<6|(n=this._keyStr.indexOf(r.charAt(d++))),c+=String.fromCharCode(t),64!=h&&(c+=String.fromCharCode(e)),64!=n&&(c+=String.fromCharCode(o));return c=Base64._utf8_decode(c)},_utf8_encode:function(r){r=r.replace(/\r\n/g,"\n");for(var t="",e=0;e<r.length;e++){var o=r.charCodeAt(e);o<128?t+=String.fromCharCode(o):o>127&&o<2048?(t+=String.fromCharCode(o>>6|192),t+=String.fromCharCode(63&o|128)):(t+=String.fromCharCode(o>>12|224),t+=String.fromCharCode(o>>6&63|128),t+=String.fromCharCode(63&o|128))}return t},_utf8_decode:function(r){for(var t="",e=0,o=c1=c2=0;e<r.length;)(o=r.charCodeAt(e))<128?(t+=String.fromCharCode(o),e++):o>191&&o<224?(c2=r.charCodeAt(e+1),t+=String.fromCharCode((31&o)<<6|63&c2),e+=2):(c2=r.charCodeAt(e+1),c3=r.charCodeAt(e+2),t+=String.fromCharCode((15&o)<<12|(63&c2)<<6|63&c3),e+=3);return t}};
  3. $(document).ready(function() {
  4. // Spam score slider
  5. var spam_slider = $('#spam_score')[0];
  6. if (typeof spam_slider !== 'undefined') {
  7. noUiSlider.create(spam_slider, {
  8. start: user_spam_score,
  9. connect: [true, true, true],
  10. range: {
  11. 'min': [0], //stepsize is 50.000
  12. '50%': [10],
  13. '70%': [20, 5],
  14. '80%': [50, 10],
  15. '90%': [100, 100],
  16. '95%': [1000, 1000],
  17. 'max': [5000]
  18. },
  19. });
  20. var connect = spam_slider.querySelectorAll('.noUi-connect');
  21. var classes = ['c-1-color', 'c-2-color', 'c-3-color'];
  22. for (var i = 0; i < connect.length; i++) {
  23. connect[i].classList.add(classes[i]);
  24. }
  25. spam_slider.noUiSlider.on('update', function (values, handle) {
  26. $('.spam-ham-score').text('< ' + Math.round(values[0] * 10) / 10);
  27. $('.spam-spam-score').text(Math.round(values[0] * 10) / 10 + ' - ' + Math.round(values[1] * 10) / 10);
  28. $('.spam-reject-score').text('> ' + Math.round(values[1] * 10) / 10);
  29. $('#spam_score_value').val((Math.round(values[0] * 10) / 10) + ',' + (Math.round(values[1] * 10) / 10));
  30. });
  31. }
  32. // syncjobLogModal
  33. $('#syncjobLogModal').on('show.bs.modal', function(e) {
  34. var syncjob_id = $(e.relatedTarget).data('syncjob-id');
  35. $.ajax({
  36. url: '/inc/ajax/syncjob_logs.php',
  37. data: { id: syncjob_id },
  38. dataType: 'text',
  39. success: function(data){
  40. $(e.currentTarget).find('#logText').text(data);
  41. },
  42. error: function(xhr, status, error) {
  43. $(e.currentTarget).find('#logText').text(xhr.responseText);
  44. }
  45. });
  46. });
  47. $(".arrow-toggle").on('click', function(e) { e.preventDefault(); $(this).find('.arrow').toggleClass("animation"); });
  48. $("#pushover_delete").click(function() { return confirm(lang.delete_ays); });
  49. });
  50. jQuery(function($){
  51. // http://stackoverflow.com/questions/24816/escaping-html-strings-with-jquery
  52. var entityMap = {
  53. '&': '&amp;',
  54. '<': '&lt;',
  55. '>': '&gt;',
  56. '"': '&quot;',
  57. "'": '&#39;',
  58. '/': '&#x2F;',
  59. '`': '&#x60;',
  60. '=': '&#x3D;'
  61. };
  62. function escapeHtml(string) {
  63. return String(string).replace(/[&<>"'`=\/]/g, function (s) {
  64. return entityMap[s];
  65. });
  66. }
  67. // http://stackoverflow.com/questions/46155/validate-email-address-in-javascript
  68. function validateEmail(email) {
  69. var re = /^(([^<>()[\]\\.,;:\s@\"]+(\.[^<>()[\]\\.,;:\s@\"]+)*)|(\".+\"))@((\[[0-9]{1,3}\.[0-9]{1,3}\.[0-9]{1,3}\.[0-9]{1,3}\])|(([a-zA-Z\-0-9]+\.)+[a-zA-Z]{2,}))$/;
  70. return re.test(email);
  71. }
  72. function unix_time_format(tm) {
  73. var date = new Date(tm ? tm * 1000 : 0);
  74. return date.toLocaleDateString(undefined, {year: "numeric", month: "2-digit", day: "2-digit", hour: "2-digit", minute: "2-digit", second: "2-digit"});
  75. }
  76. acl_data = JSON.parse(acl);
  77. $('.clear-last-logins').on('click', function () {if (confirm(lang.delete_ays)) {last_logins('reset');}})
  78. $(".login-history").on('click', function(e) {e.preventDefault(); last_logins('get', $(this).data('days'));$(this).addClass('active').siblings().removeClass('active');});
  79. function last_logins(action, days = 7) {
  80. if (action == 'get') {
  81. $('.last-login').html('<i class="bi bi-hourglass"></i>' + lang.waiting);
  82. $.ajax({
  83. dataType: 'json',
  84. url: '/api/v1/get/last-login/' + encodeURIComponent(mailcow_cc_username) + '/' + days,
  85. jsonp: false,
  86. error: function () {
  87. console.log('error reading last logins');
  88. },
  89. success: function (data) {
  90. $('.last-login').html();
  91. if (data.ui.time) {
  92. $('.last-login').html('<i class="bi bi-person-fill"></i> ' + lang.last_ui_login + ': ' + unix_time_format(data.ui.time));
  93. } else {
  94. $('.last-login').text(lang.no_last_login);
  95. }
  96. if (data.sasl) {
  97. $('.last-login').append('<ul class="list-group">');
  98. $.each(data.sasl, function (i, item) {
  99. var datetime = new Date(item.datetime.replace(/-/g, "/"));
  100. var local_datetime = datetime.toLocaleDateString(undefined, {year: "numeric", month: "2-digit", day: "2-digit", hour: "2-digit", minute: "2-digit", second: "2-digit"});
  101. var service = '<div class="badge fs-5 bg-secondary">' + item.service.toUpperCase() + '</div>';
  102. var app_password = item.app_password ? ' <a href="/edit/app-passwd/' + item.app_password + '"><i class="bi bi-app-indicator"></i> ' + escapeHtml(item.app_password_name || "App") + '</a>' : '';
  103. var real_rip = item.real_rip.startsWith("Web") ? item.real_rip : '<a href="https://bgp.he.net/ip/' + item.real_rip + '" target="_blank">' + item.real_rip + "</a>";
  104. var ip_location = item.location ? ' <span class="flag-icon flag-icon-' + item.location.toLowerCase() + '"></span>' : '';
  105. var ip_data = real_rip + ip_location + app_password;
  106. $(".last-login").append('<li class="list-group-item">' + local_datetime + " " + service + " " + lang.from + " " + ip_data + "</li>");
  107. })
  108. $('.last-login').append('</ul>');
  109. }
  110. }
  111. })
  112. } else if (action == 'reset') {
  113. $.ajax({
  114. dataType: 'json',
  115. url: '/api/v1/get/reset-last-login/' + encodeURIComponent(mailcow_cc_username),
  116. jsonp: false,
  117. error: function () {
  118. console.log('cannot reset last logins');
  119. },
  120. success: function (data) {
  121. last_logins('get');
  122. }
  123. })
  124. }
  125. }
  126. function draw_tla_table() {
  127. $('#tla_table').DataTable({
  128. processing: true,
  129. serverSide: false,
  130. language: lang_datatables,
  131. ajax: {
  132. type: "GET",
  133. url: "/api/v1/get/time_limited_aliases",
  134. dataSrc: function(data){
  135. console.log(data);
  136. $.each(data, function (i, item) {
  137. if (acl_data.spam_alias === 1) {
  138. item.action = '<div class="btn-group">' +
  139. '<a href="#" data-action="delete_selected" data-id="single-tla" data-api-url="delete/time_limited_alias" data-item="' + encodeURIComponent(item.address) + '" class="btn btn-xs btn-danger"><i class="bi bi-trash"></i> ' + lang.remove + '</a>' +
  140. '</div>';
  141. item.chkbox = '<input type="checkbox" data-id="tla" name="multi_select" value="' + encodeURIComponent(item.address) + '" />';
  142. item.address = escapeHtml(item.address);
  143. }
  144. else {
  145. item.chkbox = '<input type="checkbox" disabled />';
  146. item.action = '<span>-</span>';
  147. }
  148. });
  149. return data;
  150. }
  151. },
  152. columns: [
  153. {
  154. // placeholder, so checkbox will not block child row toggle
  155. title: '',
  156. data: null,
  157. searchable: false,
  158. orderable: false,
  159. defaultContent: ''
  160. },
  161. {
  162. title: '',
  163. data: 'chkbox',
  164. searchable: false,
  165. orderable: false,
  166. defaultContent: ''
  167. },
  168. {
  169. title: lang.alias,
  170. data: 'address'
  171. },
  172. {
  173. title: lang.alias_valid_until,
  174. data: 'validity',
  175. render: function (data, type) {
  176. var date = new Date(data ? data * 1000 : 0);
  177. return date.toLocaleDateString(undefined, {year: "numeric", month: "2-digit", day: "2-digit", hour: "2-digit", minute: "2-digit", second: "2-digit"});
  178. }
  179. },
  180. {
  181. title: lang.action,
  182. data: 'action',
  183. className: 'text-md-end d-md-block dt-sm-head-hidden dt-body-right'
  184. }
  185. ]
  186. });
  187. }
  188. function draw_sync_job_table() {
  189. $('#sync_job_table').DataTable({
  190. processing: true,
  191. serverSide: false,
  192. language: lang_datatables,
  193. ajax: {
  194. type: "GET",
  195. url: '/api/v1/get/syncjobs/' + encodeURIComponent(mailcow_cc_username) + '/no_log',
  196. dataSrc: function(data){
  197. console.log(data);
  198. $.each(data, function (i, item) {
  199. item.user1 = escapeHtml(item.user1);
  200. item.log = '<a href="#syncjobLogModal" data-bs-toggle="modal" data-syncjob-id="' + item.id + '">' + lang.open_logs + '</a>'
  201. if (!item.exclude > 0) {
  202. item.exclude = '-';
  203. } else {
  204. item.exclude = '<code>' + escapeHtml(item.exclude) + '</code>';
  205. }
  206. item.server_w_port = escapeHtml(item.user1 + '@' + item.host1 + ':' + item.port1);
  207. if (acl_data.syncjobs === 1) {
  208. item.action = '<div class="btn-group">' +
  209. '<a href="/edit/syncjob/' + item.id + '" class="btn btn-xs btn-xs-half btn-secondary"><i class="bi bi-pencil-fill"></i> ' + lang.edit + '</a>' +
  210. '<a href="#" data-action="delete_selected" data-id="single-syncjob" data-api-url="delete/syncjob" data-item="' + item.id + '" class="btn btn-xs btn-xs-half btn-danger"><i class="bi bi-trash"></i> ' + lang.remove + '</a>' +
  211. '</div>';
  212. item.chkbox = '<input type="checkbox" data-id="syncjob" name="multi_select" value="' + item.id + '" />';
  213. }
  214. else {
  215. item.action = '<span>-</span>';
  216. item.chkbox = '<input type="checkbox" disabled />';
  217. }
  218. if (item.is_running == 1) {
  219. item.is_running = '<span id="active-script" class="badge fs-5 bg-success">' + lang.running + '</span>';
  220. } else {
  221. item.is_running = '<span id="inactive-script" class="badge fs-5 bg-warning">' + lang.waiting + '</span>';
  222. }
  223. if (!item.last_run > 0) {
  224. item.last_run = lang.waiting;
  225. }
  226. if (item.success == null) {
  227. item.success = '-';
  228. item.exit_status = '';
  229. } else {
  230. item.success = '<i class="text-' + (item.success == 1 ? 'success' : 'danger') + ' bi bi-' + (item.success == 1 ? 'check-lg' : 'x-lg') + '"></i>';
  231. }
  232. if (lang['syncjob_'+item.exit_status]) {
  233. item.exit_status = lang['syncjob_'+item.exit_status];
  234. } else if (item.success != '-') {
  235. item.exit_status = lang.syncjob_check_log;
  236. }
  237. item.exit_status = item.success + ' ' + item.exit_status;
  238. });
  239. return data;
  240. }
  241. },
  242. columns: [
  243. {
  244. // placeholder, so checkbox will not block child row toggle
  245. title: '',
  246. data: null,
  247. searchable: false,
  248. orderable: false,
  249. defaultContent: ''
  250. },
  251. {
  252. title: '',
  253. data: 'chkbox',
  254. searchable: false,
  255. orderable: false,
  256. defaultContent: ''
  257. },
  258. {
  259. title: 'ID',
  260. data: 'id'
  261. },
  262. {
  263. title: 'Server',
  264. data: 'server_w_port'
  265. },
  266. {
  267. title: lang.encryption,
  268. data: 'enc1'
  269. },
  270. {
  271. title: lang.username,
  272. data: 'user1'
  273. },
  274. {
  275. title: lang.excludes,
  276. data: 'exclude'
  277. },
  278. {
  279. title: lang.interval + " (min)",
  280. data: 'mins_interval'
  281. },
  282. {
  283. title: lang.last_run,
  284. data: 'last_run'
  285. },
  286. {
  287. title: lang.syncjob_last_run_result,
  288. data: 'exit_status'
  289. },
  290. {
  291. title: 'Log',
  292. data: 'log'
  293. },
  294. {
  295. title: lang.active,
  296. data: 'active',
  297. render: function (data, type) {
  298. return 1==data?'<i class="bi bi-check-lg"></i>':0==data&&'<i class="bi bi-x-lg"></i>'
  299. }
  300. },
  301. {
  302. title: lang.status,
  303. data: 'is_running'
  304. },
  305. {
  306. title: lang.action,
  307. data: 'action',
  308. className: 'text-md-end d-md-block dt-sm-head-hidden dt-body-right'
  309. }
  310. ]
  311. });
  312. }
  313. function draw_app_passwd_table() {
  314. $('#app_passwd_table').DataTable({
  315. processing: true,
  316. serverSide: false,
  317. language: lang_datatables,
  318. ajax: {
  319. type: "GET",
  320. url: '/api/v1/get/app-passwd/all',
  321. dataSrc: function(data){
  322. console.log(data);
  323. $.each(data, function (i, item) {
  324. item.name = escapeHtml(item.name)
  325. item.protocols = []
  326. if (item.imap_access == 1) { item.protocols.push("<code>IMAP</code>"); }
  327. if (item.smtp_access == 1) { item.protocols.push("<code>SMTP</code>"); }
  328. if (item.eas_access == 1) { item.protocols.push("<code>EAS/ActiveSync</code>"); }
  329. if (item.dav_access == 1) { item.protocols.push("<code>DAV</code>"); }
  330. if (item.pop3_access == 1) { item.protocols.push("<code>POP3</code>"); }
  331. if (item.sieve_access == 1) { item.protocols.push("<code>Sieve</code>"); }
  332. item.protocols = item.protocols.join(" ")
  333. if (acl_data.app_passwds === 1) {
  334. item.action = '<div class="btn-group">' +
  335. '<a href="/edit/app-passwd/' + item.id + '" class="btn btn-xs btn-xs-half btn-secondary"><i class="bi bi-pencil-fill"></i> ' + lang.edit + '</a>' +
  336. '<a href="#" data-action="delete_selected" data-id="single-apppasswd" data-api-url="delete/app-passwd" data-item="' + item.id + '" class="btn btn-xs btn-xs-half btn-danger"><i class="bi bi-trash"></i> ' + lang.remove + '</a>' +
  337. '</div>';
  338. item.chkbox = '<input type="checkbox" data-id="apppasswd" name="multi_select" value="' + item.id + '" />';
  339. }
  340. else {
  341. item.action = '<span>-</span>';
  342. item.chkbox = '<input type="checkbox" disabled />';
  343. }
  344. });
  345. return data;
  346. }
  347. },
  348. columns: [
  349. {
  350. // placeholder, so checkbox will not block child row toggle
  351. title: '',
  352. data: null,
  353. searchable: false,
  354. orderable: false,
  355. defaultContent: ''
  356. },
  357. {
  358. title: '',
  359. data: 'chkbox',
  360. searchable: false,
  361. orderable: false,
  362. defaultContent: ''
  363. },
  364. {
  365. title: 'ID',
  366. data: 'id'
  367. },
  368. {
  369. title: lang.app_name,
  370. data: 'name'
  371. },
  372. {
  373. title: lang.allowed_protocols,
  374. data: 'protocols'
  375. },
  376. {
  377. title: lang.active,
  378. data: 'active',
  379. render: function (data, type) {
  380. return 1==data?'<i class="bi bi-check-lg"></i>':0==data&&'<i class="bi bi-x-lg"></i>'
  381. }
  382. },
  383. {
  384. title: lang.action,
  385. data: 'action',
  386. className: 'text-md-end d-md-block dt-sm-head-hidden dt-body-right'
  387. }
  388. ]
  389. });
  390. }
  391. function draw_wl_policy_mailbox_table() {
  392. $('#wl_policy_mailbox_table').DataTable({
  393. processing: true,
  394. serverSide: false,
  395. language: lang_datatables,
  396. ajax: {
  397. type: "GET",
  398. url: '/api/v1/get/policy_wl_mailbox',
  399. dataSrc: function(data){
  400. console.log(data);
  401. $.each(data, function (i, item) {
  402. if (validateEmail(item.object)) {
  403. item.chkbox = '<input type="checkbox" data-id="policy_wl_mailbox" name="multi_select" value="' + item.prefid + '" />';
  404. }
  405. else {
  406. item.chkbox = '<input type="checkbox" disabled title="' + lang.spamfilter_table_domain_policy + '" />';
  407. }
  408. if (acl_data.spam_policy === 0) {
  409. item.chkbox = '<input type="checkbox" disabled />';
  410. }
  411. });
  412. return data;
  413. }
  414. },
  415. columns: [
  416. {
  417. // placeholder, so checkbox will not block child row toggle
  418. title: '',
  419. data: null,
  420. searchable: false,
  421. orderable: false,
  422. defaultContent: ''
  423. },
  424. {
  425. title: '',
  426. data: 'chkbox',
  427. searchable: false,
  428. orderable: false,
  429. defaultContent: ''
  430. },
  431. {
  432. title: 'ID',
  433. data: 'prefid'
  434. },
  435. {
  436. title: lang.spamfilter_table_rule,
  437. data: 'name'
  438. },
  439. {
  440. title:'Scope',
  441. data: 'object'
  442. }
  443. ]
  444. });
  445. }
  446. function draw_bl_policy_mailbox_table() {
  447. $('#bl_policy_mailbox_table').DataTable({
  448. processing: true,
  449. serverSide: false,
  450. language: lang_datatables,
  451. ajax: {
  452. type: "GET",
  453. url: '/api/v1/get/policy_bl_mailbox',
  454. dataSrc: function(data){
  455. console.log(data);
  456. $.each(data, function (i, item) {
  457. if (validateEmail(item.object)) {
  458. item.chkbox = '<input type="checkbox" data-id="policy_bl_mailbox" name="multi_select" value="' + item.prefid + '" />';
  459. }
  460. else {
  461. item.chkbox = '<input type="checkbox" disabled tooltip="' + lang.spamfilter_table_domain_policy + '" />';
  462. }
  463. if (acl_data.spam_policy === 0) {
  464. item.chkbox = '<input type="checkbox" disabled />';
  465. }
  466. });
  467. return data;
  468. }
  469. },
  470. columns: [
  471. {
  472. // placeholder, so checkbox will not block child row toggle
  473. title: '',
  474. data: null,
  475. searchable: false,
  476. orderable: false,
  477. defaultContent: ''
  478. },
  479. {
  480. title: '',
  481. data: 'chkbox',
  482. searchable: false,
  483. orderable: false,
  484. defaultContent: ''
  485. },
  486. {
  487. title: 'ID',
  488. data: 'prefid'
  489. },
  490. {
  491. title: lang.spamfilter_table_rule,
  492. data: 'name'
  493. },
  494. {
  495. title:'Scope',
  496. data: 'object'
  497. }
  498. ]
  499. });
  500. }
  501. // FIDO2 friendly name modal
  502. $('#fido2ChangeFn').on('show.bs.modal', function (e) {
  503. rename_link = $(e.relatedTarget)
  504. if (rename_link != null) {
  505. $('#fido2_cid').val(rename_link.data('cid'));
  506. $('#fido2_subject_desc').text(Base64.decode(rename_link.data('subject')));
  507. }
  508. })
  509. // Sieve data modal
  510. $('#userFilterModal').on('show.bs.modal', function(e) {
  511. $('#user_sieve_filter').text(lang.loading);
  512. $.ajax({
  513. dataType: 'json',
  514. url: '/api/v1/get/active-user-sieve/' + encodeURIComponent(mailcow_cc_username),
  515. jsonp: false,
  516. error: function () {
  517. console.log('Cannot get active sieve script');
  518. },
  519. complete: function (data) {
  520. if (data.responseText == '{}') {
  521. $('#user_sieve_filter').text(lang.no_active_filter);
  522. } else {
  523. $('#user_sieve_filter').text(JSON.parse(data.responseText));
  524. }
  525. }
  526. })
  527. });
  528. $('#userFilterModal').on('hidden.bs.modal', function () {
  529. $('#user_sieve_filter').text(lang.loading);
  530. });
  531. // detect element visibility changes
  532. function onVisible(element, callback) {
  533. $(element).ready(function() {
  534. element_object = document.querySelector(element)
  535. new IntersectionObserver((entries, observer) => {
  536. entries.forEach(entry => {
  537. if(entry.intersectionRatio > 0) {
  538. callback(element_object);
  539. observer.disconnect();
  540. }
  541. });
  542. }).observe(element_object);
  543. });
  544. }
  545. // Load only if the tab is visible
  546. onVisible("[id^=tla_table]", () => draw_tla_table());
  547. onVisible("[id^=bl_policy_mailbox_table]", () => draw_bl_policy_mailbox_table());
  548. onVisible("[id^=wl_policy_mailbox_table]", () => draw_wl_policy_mailbox_table());
  549. onVisible("[id^=sync_job_table]", () => draw_sync_job_table());
  550. onVisible("[id^=app_passwd_table]", () => draw_app_passwd_table());
  551. last_logins('get');
  552. });