user.js 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409
  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 () {if (confirm(lang.delete_ays)) {last_logins('reset');}})
  74. $(".login-history").on('click', function(e) {e.preventDefault(); last_logins('get', $(this).data('days'));$(this).addClass('active').siblings().removeClass('active');});
  75. function last_logins(action, days = 7) {
  76. if (action == 'get') {
  77. $('.last-login').html('<i class="bi bi-hourglass"></i>' + lang.waiting);
  78. $.ajax({
  79. dataType: 'json',
  80. url: '/api/v1/get/last-login/' + encodeURIComponent(mailcow_cc_username) + '/' + days,
  81. jsonp: false,
  82. error: function () {
  83. console.log('error reading last logins');
  84. },
  85. success: function (data) {
  86. $('.last-login').html();
  87. if (data.ui.time) {
  88. $('.last-login').html('<i class="bi bi-person-fill"></i> ' + lang.last_ui_login + ': ' + unix_time_format(data.ui.time));
  89. } else {
  90. $('.last-login').text(lang.no_last_login);
  91. }
  92. if (data.sasl) {
  93. $('.last-login').append('<ul class="list-group">');
  94. $.each(data.sasl, function (i, item) {
  95. var datetime = new Date(item.datetime.replace(/-/g, "/"));
  96. var local_datetime = datetime.toLocaleDateString(undefined, {year: "numeric", month: "2-digit", day: "2-digit", hour: "2-digit", minute: "2-digit", second: "2-digit"});
  97. item.app_password ? app_password = ' <a href="/edit/app-passwd/' + item.app_password + '">(App)</a>' : app_password = "", item.location ? ip_location = ' <span class="flag-icon flag-icon-' + item.location.toLowerCase() + '"></span>' : ip_location = "";
  98. "smtp" == item.service ? service = '<div class="label label-default">' + item.service.toUpperCase() + '<i class="bi bi-chevron-compact-right"></i></div>' : "imap" == item.service ? service = '<div class="label label-default"><i class="bi bi-chevron-compact-left"></i> ' + item.service.toUpperCase() + "</div>" : service = '<div class="label label-default">' + item.service.toUpperCase() + "</div>";
  99. item.real_rip.startsWith("Web") ? real_rip = item.real_rip : real_rip = '<a href="https://www.virustotal.com/gui/ip-address/' + item.real_rip + '/detection" target="_blank">' + item.real_rip + "</a>";
  100. ip_data = real_rip + ip_location + app_password;
  101. $(".last-login").append('<li class="list-group-item">' + local_datetime + " " + service + " " + lang.from + " " + ip_data + "</li>");
  102. })
  103. $('.last-login').append('</ul>');
  104. }
  105. }
  106. })
  107. } else if (action == 'reset') {
  108. $.ajax({
  109. dataType: 'json',
  110. url: '/api/v1/get/reset-last-login/' + encodeURIComponent(mailcow_cc_username),
  111. jsonp: false,
  112. error: function () {
  113. console.log('cannot reset last logins');
  114. },
  115. success: function (data) {
  116. last_logins('get');
  117. }
  118. })
  119. }
  120. }
  121. function draw_tla_table() {
  122. ft_tla_table = FooTable.init('#tla_table', {
  123. "columns": [
  124. {"name":"chkbox","title":"","style":{"maxWidth":"40px","width":"40px","text-align":"center"},"filterable": false,"sortable": false,"type":"html"},
  125. {"name":"address","title":lang.alias},
  126. {"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"}},
  127. {"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"}},
  128. {"name":"action","filterable": false,"sortable": false,"style":{"text-align":"right","maxWidth":"180px","width":"180px"},"type":"html","title":lang.action,"breakpoints":"xs sm"}
  129. ],
  130. "empty": lang.empty,
  131. "rows": $.ajax({
  132. dataType: 'json',
  133. url: '/api/v1/get/time_limited_aliases',
  134. jsonp: false,
  135. error: function () {
  136. console.log('Cannot draw tla table');
  137. },
  138. success: function (data) {
  139. $.each(data, function (i, item) {
  140. if (acl_data.spam_alias === 1) {
  141. item.action = '<div class="btn-group">' +
  142. '<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>' +
  143. '</div>';
  144. item.chkbox = '<input type="checkbox" data-id="tla" name="multi_select" value="' + encodeURIComponent(item.address) + '" />';
  145. item.address = escapeHtml(item.address);
  146. }
  147. else {
  148. item.chkbox = '<input type="checkbox" disabled />';
  149. item.action = '<span>-</span>';
  150. }
  151. });
  152. }
  153. }),
  154. "paging": {
  155. "enabled": true,
  156. "limit": 5,
  157. "size": pagination_size
  158. },
  159. "state": {"enabled": true},
  160. "sorting": {
  161. "enabled": true
  162. },
  163. "toggleSelector": "table tbody span.footable-toggle"
  164. });
  165. }
  166. function draw_sync_job_table() {
  167. ft_syncjob_table = FooTable.init('#sync_job_table', {
  168. "columns": [
  169. {"name":"chkbox","title":"","style":{"maxWidth":"60px","width":"60px","text-align":"center"},"filterable": false,"sortable": false,"type":"html"},
  170. {"sorted": true,"name":"id","title":"ID","style":{"maxWidth":"60px","width":"60px","text-align":"center"}},
  171. {"name":"server_w_port","title":"Server"},
  172. {"name":"enc1","title":lang.encryption,"breakpoints":"all"},
  173. {"name":"user1","title":lang.username},
  174. {"name":"exclude","title":lang.excludes,"breakpoints":"all"},
  175. {"name":"mins_interval","title":lang.interval + " (min)","breakpoints":"all"},
  176. {"name":"last_run","title":lang.last_run,"breakpoints":"all"},
  177. {"name":"log","title":"Log"},
  178. {"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>';}},
  179. {"name":"is_running","filterable": false,"style":{"maxWidth":"120px","width":"100px"},"title":lang.status},
  180. {"name":"action","filterable": false,"sortable": false,"style":{"text-align":"right","min-width":"260px","width":"260px"},"type":"html","title":lang.action,"breakpoints":"xs sm"}
  181. ],
  182. "empty": lang.empty,
  183. "rows": $.ajax({
  184. dataType: 'json',
  185. url: '/api/v1/get/syncjobs/' + encodeURIComponent(mailcow_cc_username) + '/no_log',
  186. jsonp: false,
  187. error: function () {
  188. console.log('Cannot draw sync job table');
  189. },
  190. success: function (data) {
  191. $.each(data, function (i, item) {
  192. item.user1 = escapeHtml(item.user1);
  193. item.log = '<a href="#syncjobLogModal" data-toggle="modal" data-syncjob-id="' + item.id + '">Open logs</a>'
  194. if (!item.exclude > 0) {
  195. item.exclude = '-';
  196. } else {
  197. item.exclude = '<code>' + escapeHtml(item.exclude) + '</code>';
  198. }
  199. item.server_w_port = escapeHtml(item.user1 + '@' + item.host1 + ':' + item.port1);
  200. if (acl_data.syncjobs === 1) {
  201. item.action = '<div class="btn-group">' +
  202. '<a href="/edit/syncjob/' + item.id + '" class="btn btn-xs btn-default"><i class="bi bi-pencil-fill"></i> ' + lang.edit + '</a>' +
  203. '<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>' +
  204. '</div>';
  205. item.chkbox = '<input type="checkbox" data-id="syncjob" name="multi_select" value="' + item.id + '" />';
  206. }
  207. else {
  208. item.action = '<span>-</span>';
  209. item.chkbox = '<input type="checkbox" disabled />';
  210. }
  211. if (item.is_running == 1) {
  212. item.is_running = '<span id="active-script" class="label label-success">' + lang.running + '</span>';
  213. } else {
  214. item.is_running = '<span id="inactive-script" class="label label-warning">' + lang.waiting + '</span>';
  215. }
  216. if (!item.last_run > 0) {
  217. item.last_run = lang.waiting;
  218. }
  219. });
  220. }
  221. }),
  222. "paging": {
  223. "enabled": true,
  224. "limit": 5,
  225. "size": pagination_size
  226. },
  227. "state": {"enabled": true},
  228. "sorting": {
  229. "enabled": true
  230. },
  231. "toggleSelector": "table tbody span.footable-toggle"
  232. });
  233. }
  234. function draw_app_passwd_table() {
  235. ft_apppasswd_table = FooTable.init('#app_passwd_table', {
  236. "columns": [
  237. {"name":"chkbox","title":"","style":{"maxWidth":"60px","width":"60px","text-align":"center"},"filterable": false,"sortable": false,"type":"html"},
  238. {"sorted": true,"name":"id","title":"ID","style":{"maxWidth":"60px","width":"60px","text-align":"center"}},
  239. {"name":"name","title":lang.app_name},
  240. {"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>';}},
  241. {"name":"action","filterable": false,"sortable": false,"style":{"text-align":"right","maxWidth":"180px","width":"180px"},"type":"html","title":lang.action,"breakpoints":"xs sm"}
  242. ],
  243. "empty": lang.empty,
  244. "rows": $.ajax({
  245. dataType: 'json',
  246. url: '/api/v1/get/app-passwd/all',
  247. jsonp: false,
  248. error: function () {
  249. console.log('Cannot draw app passwd table');
  250. },
  251. success: function (data) {
  252. $.each(data, function (i, item) {
  253. item.name = escapeHtml(item.name);
  254. if (acl_data.app_passwds === 1) {
  255. item.action = '<div class="btn-group">' +
  256. '<a href="/edit/app-passwd/' + item.id + '" class="btn btn-xs btn-default"><i class="bi bi-pencil-fill"></i> ' + lang.edit + '</a>' +
  257. '<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>' +
  258. '</div>';
  259. item.chkbox = '<input type="checkbox" data-id="apppasswd" name="multi_select" value="' + item.id + '" />';
  260. }
  261. else {
  262. item.action = '<span>-</span>';
  263. item.chkbox = '<input type="checkbox" disabled />';
  264. }
  265. });
  266. }
  267. }),
  268. "paging": {
  269. "enabled": true,
  270. "limit": 5,
  271. "size": pagination_size
  272. },
  273. "state": {"enabled": true},
  274. "sorting": {
  275. "enabled": true
  276. },
  277. "toggleSelector": "table tbody span.footable-toggle"
  278. });
  279. }
  280. function draw_wl_policy_mailbox_table() {
  281. ft_wl_policy_mailbox_table = FooTable.init('#wl_policy_mailbox_table', {
  282. "columns": [
  283. {"name":"chkbox","title":"","style":{"maxWidth":"40px","width":"40px","text-align":"center"},"filterable": false,"sortable": false,"type":"html"},
  284. {"name":"prefid","style":{"maxWidth":"40px","width":"40px"},"title":"ID","filterable": false,"sortable": false},
  285. {"sorted": true,"name":"value","title":lang.spamfilter_table_rule},
  286. {"name":"object","title":"Scope"}
  287. ],
  288. "empty": lang.empty,
  289. "rows": $.ajax({
  290. dataType: 'json',
  291. url: '/api/v1/get/policy_wl_mailbox',
  292. jsonp: false,
  293. error: function () {
  294. console.log('Cannot draw mailbox policy wl table');
  295. },
  296. success: function (data) {
  297. $.each(data, function (i, item) {
  298. if (validateEmail(item.object)) {
  299. item.chkbox = '<input type="checkbox" data-id="policy_wl_mailbox" name="multi_select" value="' + item.prefid + '" />';
  300. }
  301. else {
  302. item.chkbox = '<input type="checkbox" disabled title="' + lang.spamfilter_table_domain_policy + '" />';
  303. }
  304. if (acl_data.spam_policy === 0) {
  305. item.chkbox = '<input type="checkbox" disabled />';
  306. }
  307. });
  308. }
  309. }),
  310. "state": {"enabled": true},
  311. "paging": {
  312. "enabled": true,
  313. "limit": 5,
  314. "size": pagination_size
  315. },
  316. "sorting": {
  317. "enabled": true
  318. }
  319. });
  320. }
  321. function draw_bl_policy_mailbox_table() {
  322. ft_bl_policy_mailbox_table = FooTable.init('#bl_policy_mailbox_table', {
  323. "columns": [
  324. {"name":"chkbox","title":"","style":{"maxWidth":"40px","width":"40px","text-align":"center"},"filterable": false,"sortable": false,"type":"html"},
  325. {"name":"prefid","style":{"maxWidth":"40px","width":"40px"},"title":"ID","filterable": false,"sortable": false},
  326. {"sorted": true,"name":"value","title":lang.spamfilter_table_rule},
  327. {"name":"object","title":"Scope"}
  328. ],
  329. "empty": lang.empty,
  330. "rows": $.ajax({
  331. dataType: 'json',
  332. url: '/api/v1/get/policy_bl_mailbox',
  333. jsonp: false,
  334. error: function () {
  335. console.log('Cannot draw mailbox policy bl table');
  336. },
  337. success: function (data) {
  338. $.each(data, function (i, item) {
  339. if (validateEmail(item.object)) {
  340. item.chkbox = '<input type="checkbox" data-id="policy_bl_mailbox" name="multi_select" value="' + item.prefid + '" />';
  341. }
  342. else {
  343. item.chkbox = '<input type="checkbox" disabled tooltip="' + lang.spamfilter_table_domain_policy + '" />';
  344. }
  345. if (acl_data.spam_policy === 0) {
  346. item.chkbox = '<input type="checkbox" disabled />';
  347. }
  348. });
  349. }
  350. }),
  351. "paging": {
  352. "enabled": true,
  353. "limit": 5,
  354. "size": pagination_size
  355. },
  356. "state": {"enabled": true},
  357. "sorting": {
  358. "enabled": true
  359. }
  360. });
  361. }
  362. $('body').on('click', 'span.footable-toggle', function () {
  363. event.stopPropagation();
  364. })
  365. draw_sync_job_table();
  366. draw_app_passwd_table();
  367. draw_tla_table();
  368. draw_wl_policy_mailbox_table();
  369. draw_bl_policy_mailbox_table();
  370. last_logins('get');
  371. // FIDO2 friendly name modal
  372. $('#fido2ChangeFn').on('show.bs.modal', function (e) {
  373. rename_link = $(e.relatedTarget)
  374. if (rename_link != null) {
  375. $('#fido2_cid').val(rename_link.data('cid'));
  376. $('#fido2_subject_desc').text(Base64.decode(rename_link.data('subject')));
  377. }
  378. })
  379. // Sieve data modal
  380. $('#userFilterModal').on('show.bs.modal', function(e) {
  381. $('#user_sieve_filter').text(lang.loading);
  382. $.ajax({
  383. dataType: 'json',
  384. url: '/api/v1/get/active-user-sieve/' + encodeURIComponent(mailcow_cc_username),
  385. jsonp: false,
  386. error: function () {
  387. console.log('Cannot get active sieve script');
  388. },
  389. complete: function (data) {
  390. if (data.responseText == '{}') {
  391. $('#user_sieve_filter').text(lang.no_active_filter);
  392. } else {
  393. $('#user_sieve_filter').text(JSON.parse(data.responseText));
  394. }
  395. }
  396. })
  397. });
  398. $('#userFilterModal').on('hidden.bs.modal', function () {
  399. $('#user_sieve_filter').text(lang.loading);
  400. });
  401. });