user.js 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418
  1. $(document).ready(function() {
  2. // Spam score slider
  3. var spam_slider = $('#spam_score')[0];
  4. noUiSlider.create(spam_slider, {
  5. start: user_spam_score,
  6. connect: [true, true, true],
  7. range: {
  8. 'min': [0], //stepsize is 50.000
  9. '50%': [10],
  10. '70%': [20, 5],
  11. '80%': [50, 10],
  12. '90%': [100, 100],
  13. '95%': [1000, 1000],
  14. 'max': [5000]
  15. },
  16. });
  17. var connect = spam_slider.querySelectorAll('.noUi-connect');
  18. var classes = ['c-1-color', 'c-2-color', 'c-3-color'];
  19. for (var i = 0; i < connect.length; i++) {
  20. connect[i].classList.add(classes[i]);
  21. }
  22. spam_slider.noUiSlider.on('update', function (values, handle) {
  23. $('.spam-ham-score').text('< ' + Math.round(values[0] * 10) / 10);
  24. $('.spam-spam-score').text(Math.round(values[0] * 10) / 10 + ' - ' + Math.round(values[1] * 10) / 10);
  25. $('.spam-reject-score').text('> ' + Math.round(values[1] * 10) / 10);
  26. $('#spam_score_value').val((Math.round(values[0] * 10) / 10) + ',' + (Math.round(values[1] * 10) / 10));
  27. });
  28. // syncjobLogModal
  29. $('#syncjobLogModal').on('show.bs.modal', function(e) {
  30. var syncjob_id = $(e.relatedTarget).data('syncjob-id');
  31. $.ajax({
  32. url: '/inc/ajax/syncjob_logs.php',
  33. data: { id: syncjob_id },
  34. dataType: 'text',
  35. success: function(data){
  36. $(e.currentTarget).find('#logText').text(data);
  37. },
  38. error: function(xhr, status, error) {
  39. $(e.currentTarget).find('#logText').text(xhr.responseText);
  40. }
  41. });
  42. });
  43. $(".arrow-toggle").on('click', function(e) { e.preventDefault(); $(this).find('.arrow').toggleClass("animation"); });
  44. $("#pushover_delete").click(function() { return confirm(lang.delete_ays); });
  45. });
  46. jQuery(function($){
  47. // http://stackoverflow.com/questions/24816/escaping-html-strings-with-jquery
  48. var entityMap = {
  49. '&': '&amp;',
  50. '<': '&lt;',
  51. '>': '&gt;',
  52. '"': '&quot;',
  53. "'": '&#39;',
  54. '/': '&#x2F;',
  55. '`': '&#x60;',
  56. '=': '&#x3D;'
  57. };
  58. function escapeHtml(string) {
  59. return String(string).replace(/[&<>"'`=\/]/g, function (s) {
  60. return entityMap[s];
  61. });
  62. }
  63. // http://stackoverflow.com/questions/46155/validate-email-address-in-javascript
  64. function validateEmail(email) {
  65. 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,}))$/;
  66. return re.test(email);
  67. }
  68. function unix_time_format(tm) {
  69. var date = new Date(tm ? tm * 1000 : 0);
  70. return date.toLocaleDateString(undefined, {year: "numeric", month: "2-digit", day: "2-digit", hour: "2-digit", minute: "2-digit", second: "2-digit"});
  71. }
  72. acl_data = JSON.parse(acl);
  73. $('.clear-last-logins').on('click', function () {
  74. if (confirm(lang.delete_ays)) {
  75. last_logins('reset');
  76. }
  77. })
  78. function last_logins(action) {
  79. if (action == 'get') {
  80. $.ajax({
  81. dataType: 'json',
  82. url: '/api/v1/get/last-login/' + encodeURIComponent(mailcow_cc_username),
  83. jsonp: false,
  84. error: function () {
  85. console.log('error reading last logins');
  86. },
  87. success: function (data) {
  88. $('.last-login').html();
  89. if (data.ui.time) {
  90. $('.last-login').html('<i class="bi bi-person-fill"></i> ' + lang.last_ui_login + ': ' + unix_time_format(data.ui.time));
  91. } else {
  92. $('.last-login').text(lang.no_last_login);
  93. }
  94. if (data.sasl) {
  95. $('.last-login').append('<ul class="list-group">');
  96. $.each(data.sasl, function (i, item) {
  97. var datetime = new Date(item.datetime.replace(/-/g, "/"));
  98. var local_datetime = datetime.toLocaleDateString(undefined, {year: "numeric", month: "2-digit", day: "2-digit", hour: "2-digit", minute: "2-digit", second: "2-digit"});
  99. if (item.service == "smtp") { service = '<div class="label label-default">' + item.service.toUpperCase() + '<i class="bi bi-chevron-compact-right"></i></div>'; }
  100. else if (item.service == "imap") { service = '<div class="label label-default"><i class="bi bi-chevron-compact-left"></i> ' + item.service.toUpperCase() + '</div>'; }
  101. else { service = '<div class="label label-default">' + item.service.toUpperCase() + '</div>'; }
  102. if (item.real_rip.startsWith("Web")) {
  103. real_rip = item.real_rip;
  104. } else {
  105. real_rip = '<a href="https://bgp.he.net/ip/' + item.real_rip + '" target="_blank">' + item.real_rip + '</a>';
  106. }
  107. $('.last-login').append('<li class="list-group-item">' +
  108. local_datetime + ' ' + service + ' ' + lang.from + ' ' +
  109. real_rip +
  110. '</li>');
  111. })
  112. $('.last-login').append('</ul>');
  113. }
  114. }
  115. })
  116. } else if (action == 'reset') {
  117. $.ajax({
  118. dataType: 'json',
  119. url: '/api/v1/get/reset-last-login/' + encodeURIComponent(mailcow_cc_username),
  120. jsonp: false,
  121. error: function () {
  122. console.log('cannot reset last logins');
  123. },
  124. success: function (data) {
  125. last_logins('get');
  126. }
  127. })
  128. }
  129. }
  130. function draw_tla_table() {
  131. ft_tla_table = FooTable.init('#tla_table', {
  132. "columns": [
  133. {"name":"chkbox","title":"","style":{"maxWidth":"40px","width":"40px","text-align":"center"},"filterable": false,"sortable": false,"type":"html"},
  134. {"name":"address","title":lang.alias},
  135. {"name":"validity","formatter":function unix_time_format(tm) { var date = new Date(tm ? tm * 1000 : 0); return date.toLocaleDateString(undefined, {year: "numeric", month: "2-digit", day: "2-digit", hour: "2-digit", minute: "2-digit", second: "2-digit"});},"title":lang.alias_valid_until,"style":{"width":"170px"}},
  136. {"sorted": true,"sortValue": function(value){res = new Date(value);return res.getTime();},"direction":"DESC","name":"created","formatter":function date_format(datetime) { var date = new Date(datetime.replace(/-/g, "/")); return date.toLocaleDateString(undefined, {year: "numeric", month: "2-digit", day: "2-digit", hour: "2-digit", minute: "2-digit", second: "2-digit"});},"title":lang.created_on,"style":{"width":"170px"}},
  137. {"name":"action","filterable": false,"sortable": false,"style":{"text-align":"right","maxWidth":"180px","width":"180px"},"type":"html","title":lang.action,"breakpoints":"xs sm"}
  138. ],
  139. "empty": lang.empty,
  140. "rows": $.ajax({
  141. dataType: 'json',
  142. url: '/api/v1/get/time_limited_aliases',
  143. jsonp: false,
  144. error: function () {
  145. console.log('Cannot draw tla table');
  146. },
  147. success: function (data) {
  148. $.each(data, function (i, item) {
  149. if (acl_data.spam_alias === 1) {
  150. item.action = '<div class="btn-group">' +
  151. '<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>' +
  152. '</div>';
  153. item.chkbox = '<input type="checkbox" data-id="tla" name="multi_select" value="' + encodeURIComponent(item.address) + '" />';
  154. item.address = escapeHtml(item.address);
  155. }
  156. else {
  157. item.chkbox = '<input type="checkbox" disabled />';
  158. item.action = '<span>-</span>';
  159. }
  160. });
  161. }
  162. }),
  163. "paging": {
  164. "enabled": true,
  165. "limit": 5,
  166. "size": pagination_size
  167. },
  168. "state": {"enabled": true},
  169. "sorting": {
  170. "enabled": true
  171. },
  172. "toggleSelector": "table tbody span.footable-toggle"
  173. });
  174. }
  175. function draw_sync_job_table() {
  176. ft_syncjob_table = FooTable.init('#sync_job_table', {
  177. "columns": [
  178. {"name":"chkbox","title":"","style":{"maxWidth":"60px","width":"60px","text-align":"center"},"filterable": false,"sortable": false,"type":"html"},
  179. {"sorted": true,"name":"id","title":"ID","style":{"maxWidth":"60px","width":"60px","text-align":"center"}},
  180. {"name":"server_w_port","title":"Server"},
  181. {"name":"enc1","title":lang.encryption,"breakpoints":"all"},
  182. {"name":"user1","title":lang.username},
  183. {"name":"exclude","title":lang.excludes,"breakpoints":"all"},
  184. {"name":"mins_interval","title":lang.interval + " (min)","breakpoints":"all"},
  185. {"name":"last_run","title":lang.last_run,"breakpoints":"all"},
  186. {"name":"log","title":"Log"},
  187. {"name":"active","filterable": false,"style":{"maxWidth":"70px","width":"70px"},"title":lang.active,"formatter": function(value){return 1==value?'<i class="bi bi-check-lg"></i>':0==value&&'<i class="bi bi-x-lg"></i>';}},
  188. {"name":"is_running","filterable": false,"style":{"maxWidth":"120px","width":"100px"},"title":lang.status},
  189. {"name":"action","filterable": false,"sortable": false,"style":{"text-align":"right","min-width":"260px","width":"260px"},"type":"html","title":lang.action,"breakpoints":"xs sm"}
  190. ],
  191. "empty": lang.empty,
  192. "rows": $.ajax({
  193. dataType: 'json',
  194. url: '/api/v1/get/syncjobs/' + encodeURIComponent(mailcow_cc_username) + '/no_log',
  195. jsonp: false,
  196. error: function () {
  197. console.log('Cannot draw sync job table');
  198. },
  199. success: function (data) {
  200. $.each(data, function (i, item) {
  201. item.user1 = escapeHtml(item.user1);
  202. item.log = '<a href="#syncjobLogModal" data-toggle="modal" data-syncjob-id="' + item.id + '">Open logs</a>'
  203. if (!item.exclude > 0) {
  204. item.exclude = '-';
  205. } else {
  206. item.exclude = '<code>' + escapeHtml(item.exclude) + '</code>';
  207. }
  208. item.server_w_port = escapeHtml(item.user1 + '@' + item.host1 + ':' + item.port1);
  209. if (acl_data.syncjobs === 1) {
  210. item.action = '<div class="btn-group">' +
  211. '<a href="/edit/syncjob/' + item.id + '" class="btn btn-xs btn-default"><i class="bi bi-pencil-fill"></i> ' + lang.edit + '</a>' +
  212. '<a href="#" data-action="delete_selected" data-id="single-syncjob" data-api-url="delete/syncjob" data-item="' + item.id + '" class="btn btn-xs btn-danger"><i class="bi bi-trash"></i> ' + lang.remove + '</a>' +
  213. '</div>';
  214. item.chkbox = '<input type="checkbox" data-id="syncjob" name="multi_select" value="' + item.id + '" />';
  215. }
  216. else {
  217. item.action = '<span>-</span>';
  218. item.chkbox = '<input type="checkbox" disabled />';
  219. }
  220. if (item.is_running == 1) {
  221. item.is_running = '<span id="active-script" class="label label-success">' + lang.running + '</span>';
  222. } else {
  223. item.is_running = '<span id="inactive-script" class="label label-warning">' + lang.waiting + '</span>';
  224. }
  225. if (!item.last_run > 0) {
  226. item.last_run = lang.waiting;
  227. }
  228. });
  229. }
  230. }),
  231. "paging": {
  232. "enabled": true,
  233. "limit": 5,
  234. "size": pagination_size
  235. },
  236. "state": {"enabled": true},
  237. "sorting": {
  238. "enabled": true
  239. },
  240. "toggleSelector": "table tbody span.footable-toggle"
  241. });
  242. }
  243. function draw_app_passwd_table() {
  244. ft_apppasswd_table = FooTable.init('#app_passwd_table', {
  245. "columns": [
  246. {"name":"chkbox","title":"","style":{"maxWidth":"60px","width":"60px","text-align":"center"},"filterable": false,"sortable": false,"type":"html"},
  247. {"sorted": true,"name":"id","title":"ID","style":{"maxWidth":"60px","width":"60px","text-align":"center"}},
  248. {"name":"name","title":lang.app_name},
  249. {"name":"active","filterable": false,"style":{"maxWidth":"70px","width":"70px"},"title":lang.active,"formatter": function(value){return 1==value?'<i class="bi bi-check-lg"></i>':0==value&&'<i class="bi bi-x-lg"></i>';}},
  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. "empty": lang.empty,
  253. "rows": $.ajax({
  254. dataType: 'json',
  255. url: '/api/v1/get/app-passwd/all',
  256. jsonp: false,
  257. error: function () {
  258. console.log('Cannot draw app passwd table');
  259. },
  260. success: function (data) {
  261. $.each(data, function (i, item) {
  262. item.name = escapeHtml(item.name);
  263. if (acl_data.app_passwds === 1) {
  264. item.action = '<div class="btn-group">' +
  265. '<a href="/edit/app-passwd/' + item.id + '" class="btn btn-xs btn-default"><i class="bi bi-pencil-fill"></i> ' + lang.edit + '</a>' +
  266. '<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-danger"><i class="bi bi-trash"></i> ' + lang.remove + '</a>' +
  267. '</div>';
  268. item.chkbox = '<input type="checkbox" data-id="apppasswd" name="multi_select" value="' + item.id + '" />';
  269. }
  270. else {
  271. item.action = '<span>-</span>';
  272. item.chkbox = '<input type="checkbox" disabled />';
  273. }
  274. });
  275. }
  276. }),
  277. "paging": {
  278. "enabled": true,
  279. "limit": 5,
  280. "size": pagination_size
  281. },
  282. "state": {"enabled": true},
  283. "sorting": {
  284. "enabled": true
  285. },
  286. "toggleSelector": "table tbody span.footable-toggle"
  287. });
  288. }
  289. function draw_wl_policy_mailbox_table() {
  290. ft_wl_policy_mailbox_table = FooTable.init('#wl_policy_mailbox_table', {
  291. "columns": [
  292. {"name":"chkbox","title":"","style":{"maxWidth":"40px","width":"40px","text-align":"center"},"filterable": false,"sortable": false,"type":"html"},
  293. {"name":"prefid","style":{"maxWidth":"40px","width":"40px"},"title":"ID","filterable": false,"sortable": false},
  294. {"sorted": true,"name":"value","title":lang.spamfilter_table_rule},
  295. {"name":"object","title":"Scope"}
  296. ],
  297. "empty": lang.empty,
  298. "rows": $.ajax({
  299. dataType: 'json',
  300. url: '/api/v1/get/policy_wl_mailbox',
  301. jsonp: false,
  302. error: function () {
  303. console.log('Cannot draw mailbox policy wl table');
  304. },
  305. success: function (data) {
  306. $.each(data, function (i, item) {
  307. if (validateEmail(item.object)) {
  308. item.chkbox = '<input type="checkbox" data-id="policy_wl_mailbox" name="multi_select" value="' + item.prefid + '" />';
  309. }
  310. else {
  311. item.chkbox = '<input type="checkbox" disabled title="' + lang.spamfilter_table_domain_policy + '" />';
  312. }
  313. if (acl_data.spam_policy === 0) {
  314. item.chkbox = '<input type="checkbox" disabled />';
  315. }
  316. });
  317. }
  318. }),
  319. "state": {"enabled": true},
  320. "paging": {
  321. "enabled": true,
  322. "limit": 5,
  323. "size": pagination_size
  324. },
  325. "sorting": {
  326. "enabled": true
  327. }
  328. });
  329. }
  330. function draw_bl_policy_mailbox_table() {
  331. ft_bl_policy_mailbox_table = FooTable.init('#bl_policy_mailbox_table', {
  332. "columns": [
  333. {"name":"chkbox","title":"","style":{"maxWidth":"40px","width":"40px","text-align":"center"},"filterable": false,"sortable": false,"type":"html"},
  334. {"name":"prefid","style":{"maxWidth":"40px","width":"40px"},"title":"ID","filterable": false,"sortable": false},
  335. {"sorted": true,"name":"value","title":lang.spamfilter_table_rule},
  336. {"name":"object","title":"Scope"}
  337. ],
  338. "empty": lang.empty,
  339. "rows": $.ajax({
  340. dataType: 'json',
  341. url: '/api/v1/get/policy_bl_mailbox',
  342. jsonp: false,
  343. error: function () {
  344. console.log('Cannot draw mailbox policy bl table');
  345. },
  346. success: function (data) {
  347. $.each(data, function (i, item) {
  348. if (validateEmail(item.object)) {
  349. item.chkbox = '<input type="checkbox" data-id="policy_bl_mailbox" name="multi_select" value="' + item.prefid + '" />';
  350. }
  351. else {
  352. item.chkbox = '<input type="checkbox" disabled tooltip="' + lang.spamfilter_table_domain_policy + '" />';
  353. }
  354. if (acl_data.spam_policy === 0) {
  355. item.chkbox = '<input type="checkbox" disabled />';
  356. }
  357. });
  358. }
  359. }),
  360. "paging": {
  361. "enabled": true,
  362. "limit": 5,
  363. "size": pagination_size
  364. },
  365. "state": {"enabled": true},
  366. "sorting": {
  367. "enabled": true
  368. }
  369. });
  370. }
  371. $('body').on('click', 'span.footable-toggle', function () {
  372. event.stopPropagation();
  373. })
  374. draw_sync_job_table();
  375. draw_app_passwd_table();
  376. draw_tla_table();
  377. draw_wl_policy_mailbox_table();
  378. draw_bl_policy_mailbox_table();
  379. last_logins('get');
  380. // FIDO2 friendly name modal
  381. $('#fido2ChangeFn').on('show.bs.modal', function (e) {
  382. rename_link = $(e.relatedTarget)
  383. if (rename_link != null) {
  384. $('#fido2_cid').val(rename_link.data('cid'));
  385. $('#fido2_subject_desc').text(Base64.decode(rename_link.data('subject')));
  386. }
  387. })
  388. // Sieve data modal
  389. $('#userFilterModal').on('show.bs.modal', function(e) {
  390. $('#user_sieve_filter').text(lang.loading);
  391. $.ajax({
  392. dataType: 'json',
  393. url: '/api/v1/get/active-user-sieve/' + encodeURIComponent(mailcow_cc_username),
  394. jsonp: false,
  395. error: function () {
  396. console.log('Cannot get active sieve script');
  397. },
  398. complete: function (data) {
  399. if (data.responseText == '{}') {
  400. $('#user_sieve_filter').text(lang.no_active_filter);
  401. } else {
  402. $('#user_sieve_filter').text(JSON.parse(data.responseText));
  403. }
  404. }
  405. })
  406. });
  407. $('#userFilterModal').on('hidden.bs.modal', function () {
  408. $('#user_sieve_filter').text(lang.loading);
  409. });
  410. });