user.js 20 KB

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