git subrepo commit (merge) mailcow/src/mailcow-dockerized

subrepo: subdir:   "mailcow/src/mailcow-dockerized"
  merged:   "02ae5285"
upstream: origin:   "https://github.com/mailcow/mailcow-dockerized.git"
  branch:   "master"
  commit:   "649a5c01"
git-subrepo: version:  "0.4.3"
  origin:   "???"
  commit:   "???"
Change-Id: I870ad468fba026cc5abf3c5699ed1e12ff28b32b
diff --git a/mailcow/src/mailcow-dockerized/data/web/js/site/user.js b/mailcow/src/mailcow-dockerized/data/web/js/site/user.js
index 0fdec8f..83aedba 100644
--- a/mailcow/src/mailcow-dockerized/data/web/js/site/user.js
+++ b/mailcow/src/mailcow-dockerized/data/web/js/site/user.js
@@ -1,4 +1,35 @@
+// Base64 functions

+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}};

 $(document).ready(function() {

+  // Spam score slider

+  var spam_slider = $('#spam_score')[0];

+  if (typeof spam_slider !== 'undefined') {

+    noUiSlider.create(spam_slider, {

+      start: user_spam_score,

+      connect: [true, true, true],

+      range: {

+        'min': [0], //stepsize is 50.000

+        '50%': [10],

+        '70%': [20, 5],

+        '80%': [50, 10],

+        '90%': [100, 100],

+        '95%': [1000, 1000],

+        'max': [5000]

+      },

+    });

+    var connect = spam_slider.querySelectorAll('.noUi-connect');

+    var classes = ['c-1-color', 'c-2-color', 'c-3-color'];

+    for (var i = 0; i < connect.length; i++) {

+      connect[i].classList.add(classes[i]);

+    }

+    spam_slider.noUiSlider.on('update', function (values, handle) {

+      $('.spam-ham-score').text('< ' + Math.round(values[0] * 10) / 10);

+      $('.spam-spam-score').text(Math.round(values[0] * 10) / 10 + ' - ' + Math.round(values[1] * 10) / 10);

+      $('.spam-reject-score').text('> ' + Math.round(values[1] * 10) / 10);

+      $('#spam_score_value').val((Math.round(values[0] * 10) / 10) + ',' + (Math.round(values[1] * 10) / 10));

+    });

+  }

+  // syncjobLogModal

   $('#syncjobLogModal').on('show.bs.modal', function(e) {

     var syncjob_id = $(e.relatedTarget).data('syncjob-id');

     $.ajax({

@@ -15,6 +46,7 @@
   });

   $(".arrow-toggle").on('click', function(e) { e.preventDefault(); $(this).find('.arrow').toggleClass("animation"); });

   $("#pushover_delete").click(function() { return confirm(lang.delete_ays); });

+

 });

 jQuery(function($){

   // http://stackoverflow.com/questions/24816/escaping-html-strings-with-jquery

@@ -43,15 +75,64 @@
     return date.toLocaleDateString(undefined, {year: "numeric", month: "2-digit", day: "2-digit", hour: "2-digit", minute: "2-digit", second: "2-digit"});

   }

   acl_data = JSON.parse(acl);

-  var last_login = $('.last_login_date').data('time');

-  $('.last_login_date').text(unix_time_format(last_login));

+

+  $('.clear-last-logins').on('click', function () {if (confirm(lang.delete_ays)) {last_logins('reset');}})

+  $(".login-history").on('click', function(e) {e.preventDefault(); last_logins('get', $(this).data('days'));$(this).addClass('active').siblings().removeClass('active');});

+

+  function last_logins(action, days = 1) {

+    if (action == 'get') {

+      $('.last-login').html('<i class="bi bi-hourglass"></i>' +  lang.waiting);

+      $.ajax({

+        dataType: 'json',

+        url: '/api/v1/get/last-login/' + encodeURIComponent(mailcow_cc_username) + '/' + days,

+        jsonp: false,

+        error: function () {

+          console.log('error reading last logins');

+        },

+        success: function (data) {

+          $('.last-login').html();

+          if (data.ui.time) {

+            $('.last-login').html('<i class="bi bi-person-fill"></i> ' + lang.last_ui_login + ': ' + unix_time_format(data.ui.time));

+          } else {

+            $('.last-login').text(lang.no_last_login);

+          }

+          if (data.sasl) {

+            $('.last-login').append('<ul class="list-group">');

+            $.each(data.sasl, function (i, item) {

+              var datetime = new Date(item.datetime.replace(/-/g, "/"));

+              var local_datetime = datetime.toLocaleDateString(undefined, {year: "numeric", month: "2-digit", day: "2-digit", hour: "2-digit", minute: "2-digit", second: "2-digit"});

+              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 = "";

+              "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>";

+              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>";

+              ip_data = real_rip + ip_location + app_password;

+              $(".last-login").append('<li class="list-group-item">' + local_datetime + " " + service + " " + lang.from + " " + ip_data + "</li>");

+            })

+            $('.last-login').append('</ul>');

+          }

+        }

+      })

+    } else if (action == 'reset') {

+      $.ajax({

+        dataType: 'json',

+        url: '/api/v1/get/reset-last-login/' + encodeURIComponent(mailcow_cc_username),

+        jsonp: false,

+        error: function () {

+          console.log('cannot reset last logins');

+        },

+        success: function (data) {

+          last_logins('get');

+        }

+      })

+    }

+  }

 

   function draw_tla_table() {

     ft_tla_table = FooTable.init('#tla_table', {

       "columns": [

-        {"name":"chkbox","title":"","style":{"maxWidth":"40px","width":"40px","text-align":"center"},"filterable": false,"sortable": false,"type":"html"},

-        {"sorted": true,"name":"address","title":lang.alias},

+        {"name":"chkbox","title":"","style":{"min-width":"60px","width":"60px","text-align":"center"},"filterable": false,"sortable": false,"type":"html"},

+        {"name":"address","title":lang.alias},

         {"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"}},

+        {"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"}},

         {"name":"action","filterable": false,"sortable": false,"style":{"text-align":"right","maxWidth":"180px","width":"180px"},"type":"html","title":lang.action,"breakpoints":"xs sm"}

       ],

       "empty": lang.empty,

@@ -65,8 +146,8 @@
         success: function (data) {

           $.each(data, function (i, item) {

             if (acl_data.spam_alias === 1) {

-              item.action = '<div class="btn-group">' +

-                '<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"><span class="glyphicon glyphicon-trash"></span> ' + lang.remove + '</a>' +

+              item.action = '<div class="btn-group footable-actions">' +

+                '<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>' +

                 '</div>';

               item.chkbox = '<input type="checkbox" data-id="tla" name="multi_select" value="' + encodeURIComponent(item.address) + '" />';

               item.address = escapeHtml(item.address);

@@ -93,18 +174,18 @@
   function draw_sync_job_table() {

     ft_syncjob_table = FooTable.init('#sync_job_table', {

       "columns": [

-        {"name":"chkbox","title":"","style":{"maxWidth":"60px","width":"60px","text-align":"center"},"filterable": false,"sortable": false,"type":"html"},

+        {"name":"chkbox","title":"","style":{"min-width":"60px","width":"60px","text-align":"center"},"filterable": false,"sortable": false,"type":"html"},

         {"sorted": true,"name":"id","title":"ID","style":{"maxWidth":"60px","width":"60px","text-align":"center"}},

         {"name":"server_w_port","title":"Server"},

-        {"name":"enc1","title":lang.encryption,"breakpoints":"xs sm"},

+        {"name":"enc1","title":lang.encryption,"breakpoints":"all"},

         {"name":"user1","title":lang.username},

         {"name":"exclude","title":lang.excludes,"breakpoints":"all"},

         {"name":"mins_interval","title":lang.interval + " (min)","breakpoints":"all"},

         {"name":"last_run","title":lang.last_run,"breakpoints":"all"},

         {"name":"log","title":"Log"},

-        {"name":"active","filterable": false,"style":{"maxWidth":"70px","width":"70px"},"title":lang.active,"formatter": function(value){return 1==value?'&#10003;':0==value&&'&#10005;';}},

+        {"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>';}},

         {"name":"is_running","filterable": false,"style":{"maxWidth":"120px","width":"100px"},"title":lang.status},

-        {"name":"action","filterable": false,"sortable": false,"style":{"text-align":"right","maxWidth":"180px","width":"180px"},"type":"html","title":lang.action,"breakpoints":"xs sm"}

+        {"name":"action","filterable": false,"sortable": false,"style":{"text-align":"right","min-width":"260px","width":"260px"},"type":"html","title":lang.action,"breakpoints":"xs sm"}

       ],

       "empty": lang.empty,

       "rows": $.ajax({

@@ -117,7 +198,7 @@
         success: function (data) {

           $.each(data, function (i, item) {

             item.user1 = escapeHtml(item.user1);

-            item.log = '<a href="#syncjobLogModal" data-toggle="modal" data-syncjob-id="' + item.id + '">Open logs</a>'

+            item.log = '<a href="#syncjobLogModal" data-toggle="modal" data-syncjob-id="' + item.id + '">' + lang.open_logs + '</a>'

             if (!item.exclude > 0) {

               item.exclude = '-';

             } else {

@@ -125,9 +206,9 @@
             }

             item.server_w_port = escapeHtml(item.user1 + '@' + item.host1 + ':' + item.port1);

             if (acl_data.syncjobs === 1) {

-              item.action = '<div class="btn-group">' +

-                '<a href="/edit/syncjob/' + item.id + '" class="btn btn-xs btn-default"><span class="glyphicon glyphicon-pencil"></span> ' + lang.edit + '</a>' +

-                '<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"><span class="glyphicon glyphicon-trash"></span> ' + lang.remove + '</a>' +

+              item.action = '<div class="btn-group footable-actions">' +

+                '<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>' +

+                '<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>' +

                 '</div>';

               item.chkbox = '<input type="checkbox" data-id="syncjob" name="multi_select" value="' + item.id + '" />';

             }

@@ -164,7 +245,7 @@
         {"name":"chkbox","title":"","style":{"maxWidth":"60px","width":"60px","text-align":"center"},"filterable": false,"sortable": false,"type":"html"},

         {"sorted": true,"name":"id","title":"ID","style":{"maxWidth":"60px","width":"60px","text-align":"center"}},

         {"name":"name","title":lang.app_name},

-        {"name":"active","filterable": false,"style":{"maxWidth":"70px","width":"70px"},"title":lang.active,"formatter": function(value){return 1==value?'&#10003;':0==value&&'&#10005;';}},

+        {"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>';}},

         {"name":"action","filterable": false,"sortable": false,"style":{"text-align":"right","maxWidth":"180px","width":"180px"},"type":"html","title":lang.action,"breakpoints":"xs sm"}

       ],

       "empty": lang.empty,

@@ -177,10 +258,11 @@
         },

         success: function (data) {

           $.each(data, function (i, item) {

+            item.name = escapeHtml(item.name);

             if (acl_data.app_passwds === 1) {

-              item.action = '<div class="btn-group">' +

-                '<a href="/edit/app-passwd/' + item.id + '" class="btn btn-xs btn-default"><span class="glyphicon glyphicon-pencil"></span> ' + lang.edit + '</a>' +

-                '<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"><span class="glyphicon glyphicon-trash"></span> ' + lang.remove + '</a>' +

+              item.action = '<div class="btn-group footable-actions">' +

+                '<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>' +

+                '<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>' +

                 '</div>';

               item.chkbox = '<input type="checkbox" data-id="apppasswd" name="multi_select" value="' + item.id + '" />';

             }

@@ -295,6 +377,7 @@
   draw_tla_table();

   draw_wl_policy_mailbox_table();

   draw_bl_policy_mailbox_table();

+  last_logins('get');

 

   // FIDO2 friendly name modal

   $('#fido2ChangeFn').on('show.bs.modal', function (e) {

@@ -327,4 +410,4 @@
   $('#userFilterModal').on('hidden.bs.modal', function () {

     $('#user_sieve_filter').text(lang.loading);

   });

-});
\ No newline at end of file
+});