user.js 20 KB

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