diff --git a/data/web/admin.php b/data/web/admin.php
index 82d69b4c..8c5c606f 100644
--- a/data/web/admin.php
+++ b/data/web/admin.php
@@ -81,8 +81,8 @@ if (!isset($_SESSION['gal']) && $license_cache = $redis->Get('LICENSE_STATUS_CAC
           </div>
         </div>
 
-        <legend data-target="#license" class="arrow-toggle" unselectable="on" data-toggle="collapse">
-          <span style="font-size:12px" class="arrow rotate glyphicon glyphicon-menu-down"></span> <?=$lang['admin']['guid_and_license'];?>
+        <legend style="cursor:pointer;" data-target="#license" class="arrow-toggle" unselectable="on" data-toggle="collapse">
+          <span style="font-size:12px" class="arrow rotate glyphicon glyphicon-menu-up"></span> <?=$lang['admin']['guid_and_license'];?>
         </legend>
         <div id="license" class="collapse in">
         <form class="form-horizontal" autocapitalize="none" autocorrect="off" role="form" method="post">
@@ -113,53 +113,112 @@ if (!isset($_SESSION['gal']) && $license_cache = $redis->Get('LICENSE_STATUS_CAC
         </form>
         </div>
 
-        <legend data-target="#api" class="arrow-toggle" unselectable="on" data-toggle="collapse">
-          <span style="font-size:12px" class="arrow rotate glyphicon glyphicon-menu-down"></span> API
+        <legend style="margin-top:20px">
+          <span style="font-size:12px" class="arrow rotate glyphicon glyphicon-wrench"></span> API
         </legend>
         <?php
-        $api = admin_api('get');
+        $api_ro = admin_api('ro', 'get');
+        $api_rw = admin_api('rw', 'get');
         ?>
-        <div id="api" class="collapse">
-        <form class="form-horizontal" autocapitalize="none" autocorrect="off" role="form" method="post">
-          <div class="form-group">
-            <label class="control-label col-sm-3" for="allow_from"><?=$lang['admin']['api_allow_from'];?>:</label>
-            <div class="col-sm-9">
-              <textarea class="form-control" rows="5" name="allow_from" id="allow_from" <?=($api['skip_ip_check'] == 1) ? 'disabled' : null;?> required><?=htmlspecialchars($api['allow_from']);?></textarea>
+        <div class="panel-group" id="accordion">
+          <div class="panel panel-default">
+            <div class="panel-heading">
+              <h4 class="panel-title">
+                <a data-toggle="collapse" data-parent="#accordion" href="#api-ro">
+                ⇇ Read-Only Access</a>
+              </h4>
             </div>
-          </div>
-          <div class="form-group">
-            <div class="col-sm-offset-3 col-sm-9">
-              <label>
-                <input type="checkbox" id="skip_ip_check" name="skip_ip_check" <?=($api['skip_ip_check'] == 1) ? 'checked' : null;?>> <?=$lang['admin']['api_skip_ip_check'];?>
-              </label>
-            </div>
-          </div>
-          <div class="form-group">
-            <label class="control-label col-sm-3" for="admin_api_key"><?=$lang['admin']['api_key'];?>:</label>
-            <div class="col-sm-9">
-              <div class="input-group">
-                <span class="input-group-addon">Read-Write</span>
-                <input type="text" class="form-control" placeholder="-" value="<?=htmlspecialchars($api['api_key']);?>" readonly>
+            <div id="api-ro" class="panel-collapse collapse">
+              <div class="panel-body">
+                <form class="form-horizontal" autocapitalize="none" autocorrect="off" role="form" method="post">
+                  <div class="form-group">
+                    <label class="control-label col-sm-3" for="allow_from_ro"><?=$lang['admin']['api_allow_from'];?>:</label>
+                    <div class="col-sm-9">
+                      <textarea class="form-control" rows="2" name="allow_from" id="allow_from_ro" <?=($api_ro['skip_ip_check'] == 1) ? 'disabled' : null;?> required><?=htmlspecialchars($api_ro['allow_from']);?></textarea>
+                    </div>
+                  </div>
+                  <div class="form-group">
+                    <div class="col-sm-offset-3 col-sm-9">
+                      <label>
+                        <input type="checkbox" name="skip_ip_check" id="skip_ip_check_ro" <?=($api_ro['skip_ip_check'] == 1) ? 'checked' : null;?>> <?=$lang['admin']['api_skip_ip_check'];?>
+                      </label>
+                    </div>
+                  </div>
+                  <div class="form-group">
+                    <label class="control-label col-sm-3"><?=$lang['admin']['api_key'];?>:</label>
+                    <div class="col-sm-9">
+                      <pre><?=(empty(htmlspecialchars($api_ro['api_key']))) ? '-' : htmlspecialchars($api_ro['api_key']);?></pre>
+                    </div>
+                  </div>
+                  <div class="form-group">
+                    <div class="col-sm-offset-3 col-sm-9">
+                      <label>
+                        <input type="checkbox" name="active" <?=($api_ro['active'] == 1) ? 'checked' : null;?>> <?=$lang['admin']['activate_api'];?>
+                      </label>
+                    </div>
+                  </div>
+                  <div class="form-group">
+                    <div class="col-sm-offset-3 col-sm-9">
+                      <p class="help-block"><?=$lang['admin']['api_info'];?></p>
+                      <div class="btn-group">
+                        <button class="btn btn-sm btn-default" name="admin_api[ro]" type="submit" href="#"><span class="glyphicon glyphicon-check"></span> <?=$lang['admin']['save'];?></button>
+                        <button class="btn btn-sm btn-primary" name="admin_api_regen_key[ro]" type="submit" href="#"><?=$lang['admin']['regen_api_key'];?></button>
+                      </div>
+                    </div>
+                  </div>
+                </form>
               </div>
             </div>
           </div>
-          <div class="form-group">
-            <div class="col-sm-offset-3 col-sm-9">
-              <label>
-                <input type="checkbox" name="active" <?=($api['active'] == 1) ? 'checked' : null;?>> <?=$lang['admin']['activate_api'];?>
-              </label>
+          <div class="panel panel-default">
+            <div class="panel-heading">
+              <h4 class="panel-title">
+                <a data-toggle="collapse" data-parent="#accordion" href="#api-rw">
+                ⇄ Read-Write Access</a>
+              </h4>
             </div>
-          </div>
-          <div class="form-group">
-            <div class="col-sm-offset-3 col-sm-9">
-              <p class="help-block"><?=$lang['admin']['api_info'];?></p>
-              <div class="btn-group">
-                <button class="btn btn-default" name="admin_api" type="submit" href="#"><span class="glyphicon glyphicon-check"></span> <?=$lang['admin']['save'];?></button>
-                <button class="btn btn-info" name="admin_api_regen_key" type="submit" href="#"><?=$lang['admin']['regen_api_key'];?></button>
+            <div id="api-rw" class="panel-collapse collapse">
+              <div class="panel-body">
+                <form class="form-horizontal" autocapitalize="none" autocorrect="off" role="form" method="post">
+                  <div class="form-group">
+                    <label class="control-label col-sm-3" for="allow_from_rw"><?=$lang['admin']['api_allow_from'];?>:</label>
+                    <div class="col-sm-9">
+                      <textarea class="form-control" rows="2" name="allow_from" id="allow_from_rw" <?=($api_rw['skip_ip_check'] == 1) ? 'disabled' : null;?> required><?=htmlspecialchars($api_rw['allow_from']);?></textarea>
+                    </div>
+                  </div>
+                  <div class="form-group">
+                    <div class="col-sm-offset-3 col-sm-9">
+                      <label>
+                        <input type="checkbox" name="skip_ip_check" id="skip_ip_check_rw" <?=($api_rw['skip_ip_check'] == 1) ? 'checked' : null;?>> <?=$lang['admin']['api_skip_ip_check'];?>
+                      </label>
+                    </div>
+                  </div>
+                  <div class="form-group">
+                    <label class="control-label col-sm-3" for="admin_api_key"><?=$lang['admin']['api_key'];?>:</label>
+                    <div class="col-sm-9">
+                      <pre><?=(empty(htmlspecialchars($api_rw['api_key']))) ? '-' : htmlspecialchars($api_rw['api_key']);?></pre>
+                    </div>
+                  </div>
+                  <div class="form-group">
+                    <div class="col-sm-offset-3 col-sm-9">
+                      <label>
+                        <input type="checkbox" name="active" <?=($api_rw['active'] == 1) ? 'checked' : null;?>> <?=$lang['admin']['activate_api'];?>
+                      </label>
+                    </div>
+                  </div>
+                  <div class="form-group">
+                    <div class="col-sm-offset-3 col-sm-9">
+                      <p class="help-block"><?=$lang['admin']['api_info'];?></p>
+                      <div class="btn-group">
+                        <button class="btn btn-sm btn-default" name="admin_api[rw]" type="submit" href="#"><span class="glyphicon glyphicon-check"></span> <?=$lang['admin']['save'];?></button>
+                        <button class="btn btn-sm btn-primary" name="admin_api_regen_key[rw]" type="submit" href="#"><?=$lang['admin']['regen_api_key'];?></button>
+                      </div>
+                    </div>
+                  </div>
+                </form>
               </div>
             </div>
           </div>
-        </form>
         </div>
 
       </div>
diff --git a/data/web/edit.php b/data/web/edit.php
index b75a2317..3cc97916 100644
--- a/data/web/edit.php
+++ b/data/web/edit.php
@@ -564,6 +564,7 @@ if (isset($_SESSION['mailcow_cc_role'])) {
       $mailbox = html_entity_decode(rawurldecode($_GET["mailbox"]));
       $result = mailbox('get', 'mailbox_details', $mailbox);
       $rl = ratelimit('get', 'mailbox', $mailbox);
+      $pushover_data = pushover('get', $mailbox);
       $quarantine_notification = mailbox('get', 'quarantine_notification', $mailbox);
       if (!empty($result)) {
         ?>
@@ -733,6 +734,57 @@ if (isset($_SESSION['mailcow_cc_role'])) {
           </div>
         </form>
         <hr>
+        <form data-id="pushover" class="form well" method="post">
+          <input type="hidden" value="0" name="active">
+          <div class="row">
+            <div class="col-sm-1">
+              <p class="help-block"><a href="https://pushover.net" target="_blank"><img src="" class="img img-fluid"></a></p>
+            </div>
+            <div class="col-sm-10">
+              <div class="form-group">
+                <div class="row">
+                  <div class="col-sm-6">
+                    <div class="form-group">
+                      <label for="token">API Token/Key (Application)</label>
+                      <input type="text" class="form-control" name="token" maxlength="30" value="<?=$pushover_data['token'];?>" required>
+                    </div>
+                  </div>
+                  <div class="col-sm-6">
+                    <div class="form-group">
+                      <label for="key">User/Group Key</label>
+                      <input type="text" class="form-control" name="key" maxlength="30" value="<?=$pushover_data['key'];?>" required>
+                    </div>
+                  </div>
+                  <div class="col-sm-6">
+                    <div class="form-group">
+                      <label for="title"><?=$lang['admin']['pushover_title'];?></label>
+                      <input type="text" class="form-control" name="title" value="<?=$pushover_data['title'];?>" placeholder="Mail">
+                    </div>
+                  </div>
+                  <div class="col-sm-6">
+                    <div class="form-group">
+                      <label for="text"><?=$lang['admin']['pushover_text'];?></label>
+                      <input type="text" class="form-control" name="text" value="<?=$pushover_data['text'];?>" placeholder="You've got mail 📧">
+                    </div>
+                  </div>
+                  <div class="col-sm-12">
+                    <div class="checkbox">
+                    <label><input type="checkbox" value="1" name="active" <?=($pushover_data['active']=="1") ? "checked" : null;?>> <?=$lang['edit']['active'];?></label>
+                    </div>
+                  </div>
+                </div>
+              </div>
+              <hr>
+              <p><?=sprintf($lang['admin']['pushover_info'], $mailbox);?></p>
+              <div class="btn-group" data-acl="<?=$_SESSION['acl']['pushover'];?>">
+                  <a class="btn btn-sm btn-default" data-action="edit_selected" data-id="pushover" data-item="<?=htmlspecialchars($mailbox);?>" data-api-url='edit/pushover' data-api-attr='{}' href="#"><?=$lang['admin']['save'];?></a>
+                  <a class="btn btn-sm btn-default" data-action="edit_selected" data-id="pushover-test" data-item="<?=htmlspecialchars($mailbox);?>" data-api-url='edit/pushover-test' data-api-attr='{}' href="#"><span class="glyphicon glyphicon-check" aria-hidden="true"></span> <?=$lang['admin']['pushover_verify'];?></a>
+                  <a class="btn btn-sm btn-danger" data-action="edit_selected" data-id="pushover-delete" data-item="<?=htmlspecialchars($mailbox);?>" data-api-url='edit/pushover' data-api-attr='{"delete":"true"}' href="#"><span class="glyphicon glyphicon-trash" aria-hidden="true"></span> <?=$lang['admin']['remove'];?></a>
+              </div>
+            </div>
+          </div>
+        </form>
+        <hr>
         <form data-id="mboxratelimit" class="form-inline well" method="post">
           <div class="row">
             <div class="col-sm-1">
diff --git a/data/web/inc/functions.inc.php b/data/web/inc/functions.inc.php
index 69fbb3bb..74a13391 100644
--- a/data/web/inc/functions.inc.php
+++ b/data/web/inc/functions.inc.php
@@ -1143,7 +1143,7 @@ function verify_tfa_login($username, $token) {
 	}
   return false;
 }
-function admin_api($action, $data = null) {
+function admin_api($access, $action, $data = null) {
 	global $pdo;
 	global $lang;
 	if ($_SESSION['mailcow_cc_role'] != "admin") {
@@ -1154,89 +1154,177 @@ function admin_api($action, $data = null) {
 		);
 		return false;
 	}
-	switch ($action) {
-		case "edit":
-      $regen_key = $data['admin_api_regen_key'];
-      $active = (isset($data['active'])) ? 1 : 0;
-      $skip_ip_check = (isset($data['skip_ip_check'])) ? 1 : 0;
-      $allow_from = array_map('trim', preg_split( "/( |,|;|\n)/", $data['allow_from']));
-      foreach ($allow_from as $key => $val) {
-        if (empty($val)) {
-          continue;
-        }
-        if (!filter_var($val, FILTER_VALIDATE_IP)) {
-          $_SESSION['return'][] =  array(
-            'type' => 'warning',
-            'log' => array(__FUNCTION__, $data),
-            'msg' => array('ip_invalid', htmlspecialchars($allow_from[$key]))
-          );
-          unset($allow_from[$key]);
-          continue;
-        }
-      }
-      $allow_from = implode(',', array_unique(array_filter($allow_from)));
-      if (empty($allow_from) && $skip_ip_check == 0) {
-        $_SESSION['return'][] =  array(
-          'type' => 'danger',
-          'log' => array(__FUNCTION__, $data),
-          'msg' => 'ip_list_empty'
-        );
-        return false;
-      }
-      $api_key = implode('-', array(
-        strtoupper(bin2hex(random_bytes(3))),
-        strtoupper(bin2hex(random_bytes(3))),
-        strtoupper(bin2hex(random_bytes(3))),
-        strtoupper(bin2hex(random_bytes(3))),
-        strtoupper(bin2hex(random_bytes(3)))
-      ));
-      $stmt = $pdo->query("SELECT `api_key` FROM `api`");
-      $num_results = count($stmt->fetchAll(PDO::FETCH_ASSOC));
-      if (empty($num_results)) {
-        $stmt = $pdo->prepare("INSERT INTO `api` (`api_key`, `skip_ip_check`, `active`, `allow_from`)
-          VALUES (:api_key, :skip_ip_check, :active, :allow_from);");
-        $stmt->execute(array(
-          ':api_key' => $api_key,
-          ':skip_ip_check' => $skip_ip_check,
-          ':active' => $active,
-          ':allow_from' => $allow_from
-        ));
-      }
-      else {
-        if ($skip_ip_check == 0) {
-          $stmt = $pdo->prepare("UPDATE `api` SET `skip_ip_check` = :skip_ip_check, `active` = :active, `allow_from` = :allow_from ;");
-          $stmt->execute(array(
-            ':active' => $active,
-            ':skip_ip_check' => $skip_ip_check,
-            ':allow_from' => $allow_from
+	switch ($access) {
+    case "rw":
+      switch ($action) {
+        case "edit":
+          $active = (isset($data['active'])) ? 1 : 0;
+          $skip_ip_check = (isset($data['skip_ip_check'])) ? 1 : 0;
+          $allow_from = array_map('trim', preg_split( "/( |,|;|\n)/", $data['allow_from']));
+          foreach ($allow_from as $key => $val) {
+            if (empty($val)) {
+              continue;
+            }
+            if (!filter_var($val, FILTER_VALIDATE_IP)) {
+              $_SESSION['return'][] =  array(
+                'type' => 'warning',
+                'log' => array(__FUNCTION__, $data),
+                'msg' => array('ip_invalid', htmlspecialchars($allow_from[$key]))
+              );
+              unset($allow_from[$key]);
+              continue;
+            }
+          }
+          $allow_from = implode(',', array_unique(array_filter($allow_from)));
+          if (empty($allow_from) && $skip_ip_check == 0) {
+            $_SESSION['return'][] =  array(
+              'type' => 'danger',
+              'log' => array(__FUNCTION__, $data),
+              'msg' => 'ip_list_empty'
+            );
+            return false;
+          }
+          $api_key = implode('-', array(
+            strtoupper(bin2hex(random_bytes(3))),
+            strtoupper(bin2hex(random_bytes(3))),
+            strtoupper(bin2hex(random_bytes(3))),
+            strtoupper(bin2hex(random_bytes(3))),
+            strtoupper(bin2hex(random_bytes(3)))
           ));
-        }
-        else {
-          $stmt = $pdo->prepare("UPDATE `api` SET `skip_ip_check` = :skip_ip_check, `active` = :active ;");
-          $stmt->execute(array(
-            ':active' => $active,
-            ':skip_ip_check' => $skip_ip_check
+          $stmt = $pdo->query("SELECT `api_key` FROM `api` WHERE `access` = 'rw'");
+          $num_results = count($stmt->fetchAll(PDO::FETCH_ASSOC));
+          if (empty($num_results)) {
+            $stmt = $pdo->prepare("INSERT INTO `api` (`api_key`, `skip_ip_check`, `active`, `allow_from`, `access`)
+              VALUES (:api_key, :skip_ip_check, :active, :allow_from, 'rw');");
+            $stmt->execute(array(
+              ':api_key' => $api_key,
+              ':skip_ip_check' => $skip_ip_check,
+              ':active' => $active,
+              ':allow_from' => $allow_from
+            ));
+          }
+          else {
+            if ($skip_ip_check == 0) {
+              $stmt = $pdo->prepare("UPDATE `api` SET `skip_ip_check` = :skip_ip_check, `active` = :active, `allow_from` = :allow_from WHERE `access` = 'rw';");
+              $stmt->execute(array(
+                ':active' => $active,
+                ':skip_ip_check' => $skip_ip_check,
+                ':allow_from' => $allow_from
+              ));
+            }
+            else {
+              $stmt = $pdo->prepare("UPDATE `api` SET `skip_ip_check` = :skip_ip_check, `active` = :active WHERE `access` = 'rw';");
+              $stmt->execute(array(
+                ':active' => $active,
+                ':skip_ip_check' => $skip_ip_check
+              ));
+            }
+          }
+        break;
+        case "regen_key":
+          $api_key = implode('-', array(
+            strtoupper(bin2hex(random_bytes(3))),
+            strtoupper(bin2hex(random_bytes(3))),
+            strtoupper(bin2hex(random_bytes(3))),
+            strtoupper(bin2hex(random_bytes(3))),
+            strtoupper(bin2hex(random_bytes(3)))
           ));
-        }
+          $stmt = $pdo->prepare("UPDATE `api` SET `api_key` = :api_key WHERE `access` = 'rw'");
+          $stmt->execute(array(
+            ':api_key' => $api_key
+          ));
+        break;
+        case "get":
+          $stmt = $pdo->query("SELECT * FROM `api` WHERE `access` = 'rw'");
+          $apidata = $stmt->fetch(PDO::FETCH_ASSOC);
+          return $apidata;
+        break;
+      }
+    case "ro":
+      switch ($action) {
+        case "edit":
+          $active = (isset($data['active'])) ? 1 : 0;
+          $skip_ip_check = (isset($data['skip_ip_check'])) ? 1 : 0;
+          $allow_from = array_map('trim', preg_split( "/( |,|;|\n)/", $data['allow_from']));
+          foreach ($allow_from as $key => $val) {
+            if (empty($val)) {
+              continue;
+            }
+            if (!filter_var($val, FILTER_VALIDATE_IP)) {
+              $_SESSION['return'][] =  array(
+                'type' => 'warning',
+                'log' => array(__FUNCTION__, $data),
+                'msg' => array('ip_invalid', htmlspecialchars($allow_from[$key]))
+              );
+              unset($allow_from[$key]);
+              continue;
+            }
+          }
+          $allow_from = implode(',', array_unique(array_filter($allow_from)));
+          if (empty($allow_from) && $skip_ip_check == 0) {
+            $_SESSION['return'][] =  array(
+              'type' => 'danger',
+              'log' => array(__FUNCTION__, $data),
+              'msg' => 'ip_list_empty'
+            );
+            return false;
+          }
+          $api_key = implode('-', array(
+            strtoupper(bin2hex(random_bytes(3))),
+            strtoupper(bin2hex(random_bytes(3))),
+            strtoupper(bin2hex(random_bytes(3))),
+            strtoupper(bin2hex(random_bytes(3))),
+            strtoupper(bin2hex(random_bytes(3)))
+          ));
+          $stmt = $pdo->query("SELECT `api_key` FROM `api` WHERE `access` = 'ro'");
+          $num_results = count($stmt->fetchAll(PDO::FETCH_ASSOC));
+          if (empty($num_results)) {
+            $stmt = $pdo->prepare("INSERT INTO `api` (`api_key`, `skip_ip_check`, `active`, `allow_from`, `access`)
+              VALUES (:api_key, :skip_ip_check, :active, :allow_from, 'ro');");
+            $stmt->execute(array(
+              ':api_key' => $api_key,
+              ':skip_ip_check' => $skip_ip_check,
+              ':active' => $active,
+              ':allow_from' => $allow_from
+            ));
+          }
+          else {
+            if ($skip_ip_check == 0) {
+              $stmt = $pdo->prepare("UPDATE `api` SET `skip_ip_check` = :skip_ip_check, `active` = :active, `allow_from` = :allow_from WHERE `access` = 'ro';");
+              $stmt->execute(array(
+                ':active' => $active,
+                ':skip_ip_check' => $skip_ip_check,
+                ':allow_from' => $allow_from
+              ));
+            }
+            else {
+              $stmt = $pdo->prepare("UPDATE `api` SET `skip_ip_check` = :skip_ip_check, `active` = :active WHERE `access` = 'ro';");
+              $stmt->execute(array(
+                ':active' => $active,
+                ':skip_ip_check' => $skip_ip_check
+              ));
+            }
+          }
+        break;
+        case "regen_key":
+          $api_key = implode('-', array(
+            strtoupper(bin2hex(random_bytes(3))),
+            strtoupper(bin2hex(random_bytes(3))),
+            strtoupper(bin2hex(random_bytes(3))),
+            strtoupper(bin2hex(random_bytes(3))),
+            strtoupper(bin2hex(random_bytes(3)))
+          ));
+          $stmt = $pdo->prepare("UPDATE `api` SET `api_key` = :api_key WHERE `access` = 'ro'");
+          $stmt->execute(array(
+            ':api_key' => $api_key
+          ));
+        break;
+        case "get":
+          $stmt = $pdo->query("SELECT * FROM `api` WHERE `access` = 'ro'");
+          $apidata = $stmt->fetch(PDO::FETCH_ASSOC);
+          return $apidata;
+        break;
       }
-    break;
-    case "regen_key":
-      $api_key = implode('-', array(
-        strtoupper(bin2hex(random_bytes(3))),
-        strtoupper(bin2hex(random_bytes(3))),
-        strtoupper(bin2hex(random_bytes(3))),
-        strtoupper(bin2hex(random_bytes(3))),
-        strtoupper(bin2hex(random_bytes(3)))
-      ));
-      $stmt = $pdo->prepare("UPDATE `api` SET `api_key` = :api_key");
-      $stmt->execute(array(
-        ':api_key' => $api_key
-      ));
-    break;
-    case "get":
-      $stmt = $pdo->query("SELECT * FROM `api`");
-      $apidata = $stmt->fetch(PDO::FETCH_ASSOC);
-      return $apidata;
     break;
   }
 	$_SESSION['return'][] =  array(
diff --git a/data/web/inc/functions.mailbox.inc.php b/data/web/inc/functions.mailbox.inc.php
index 8238db31..977c1969 100644
--- a/data/web/inc/functions.mailbox.inc.php
+++ b/data/web/inc/functions.mailbox.inc.php
@@ -3343,6 +3343,9 @@ function mailbox($_action, $_type, $_data = null, $_extra = null) {
           $stmt = $pdo->prepare("SELECT `maxquota`, `quota` FROM  `domain` WHERE `domain` = :domain");
           $stmt->execute(array(':domain' => $row['domain']));
           $DomainQuota  = $stmt->fetch(PDO::FETCH_ASSOC);
+          $stmt = $pdo->prepare("SELECT IFNULL(COUNT(`active`), 0) AS `pushover_active` FROM `pushover` WHERE `username` = :username AND `active` = 1");
+          $stmt->execute(array(':username' => $_data));
+          $PushoverActive  = $stmt->fetch(PDO::FETCH_ASSOC);
           $stmt = $pdo->prepare("SELECT COALESCE(SUM(`quota`), 0) as `in_use` FROM `mailbox` WHERE `kind` NOT REGEXP 'location|thing|group' AND `domain` = :domain AND `username` != :username");
           $stmt->execute(array(':domain' => $row['domain'], ':username' => $_data));
           $MailboxUsage	= $stmt->fetch(PDO::FETCH_ASSOC);
@@ -3375,6 +3378,7 @@ function mailbox($_action, $_type, $_data = null, $_extra = null) {
           $mailboxdata['percent_in_use'] = ($row['quota'] == 0) ? '- ' : round((intval($row['bytes']) / intval($row['quota'])) * 100);
           $mailboxdata['messages'] = $row['messages'];
           $mailboxdata['spam_aliases'] = $SpamaliasUsage['sa_count'];
+          $mailboxdata['pushover_active'] = ($PushoverActive['pushover_active'] == 1) ? $lang['mailbox']['yes'] : $lang['mailbox']['no'];
           if ($mailboxdata['percent_in_use'] === '- ') {
             $mailboxdata['percent_class'] = "info";
           }
diff --git a/data/web/inc/functions.pushover.inc.php b/data/web/inc/functions.pushover.inc.php
new file mode 100644
index 00000000..f0cca30a
--- /dev/null
+++ b/data/web/inc/functions.pushover.inc.php
@@ -0,0 +1,175 @@
+<?php
+function pushover($_action, $_data = null) {
+	global $pdo;
+	global $lang;
+  switch ($_action) {
+    case 'edit':
+      if (!isset($_SESSION['acl']['pushover']) || $_SESSION['acl']['pushover'] != "1" ) {
+        $_SESSION['return'][] = array(
+          'type' => 'danger',
+          'log' => array(__FUNCTION__, $_action, $_data),
+          'msg' => 'access_denied'
+        );
+        return false;
+      }
+      if (!is_array($_data['username'])) {
+        $usernames = array();
+        $usernames[] = $_data['username'];
+      }
+      else {
+        $usernames = $_data['username'];
+      }
+      foreach ($usernames as $username) {
+        if (!hasMailboxObjectAccess($_SESSION['mailcow_cc_username'], $_SESSION['mailcow_cc_role'], $username)) {
+          $_SESSION['return'][] = array(
+            'type' => 'danger',
+            'log' => array(__FUNCTION__, $_action, $_data),
+            'msg' => 'access_denied'
+          );
+          continue;
+        }
+        $delete = $_data['delete'];
+        if ($delete == "true") {
+          $stmt = $pdo->prepare("DELETE FROM `pushover` WHERE `username` = :username");
+          $stmt->execute(array(
+            ':username' => $username
+          ));
+          $_SESSION['return'][] = array(
+            'type' => 'success',
+            'log' => array(__FUNCTION__, $_action, $_data, $_data),
+            'msg' => 'pushover_settings_edited'
+          );
+          continue;
+        }
+        $key = $_data['key'];
+        $token = $_data['token'];
+        if (!ctype_alnum($key) || strlen($key) != 30) {
+          $_SESSION['return'][] = array(
+            'type' => 'danger',
+            'log' => array(__FUNCTION__, $_action, $_data, $_data),
+            'msg' => 'pushover_key'
+          );
+          continue;
+        }
+        if (!ctype_alnum($token) || strlen($token) != 30) {
+          $_SESSION['return'][] = array(
+            'type' => 'danger',
+            'log' => array(__FUNCTION__, $_action, $_data, $_data),
+            'msg' => 'pushover_token'
+          );
+          continue;
+        }
+        $title = $_data['title'];
+        $text = $_data['text'];
+        $active = intval($_data['active']);
+        $stmt = $pdo->prepare("REPLACE INTO `pushover` (`username`, `key`, `token`, `title`, `text`, `active`)
+          VALUES (:username, :key, :token, :title, :text, :active)");
+        $stmt->execute(array(
+          ':username' => $username,
+          ':key' => $key,
+          ':token' => $token,
+          ':title' => $title,
+          ':text' => $text,
+          ':active' => $active
+        ));
+        $_SESSION['return'][] = array(
+          'type' => 'success',
+          'log' => array(__FUNCTION__, $_action, $_data, $_data),
+          'msg' => 'pushover_settings_edited'
+        );
+      }
+    break;
+    case 'get':
+      if (!hasMailboxObjectAccess($_SESSION['mailcow_cc_username'], $_SESSION['mailcow_cc_role'], $_data)) {
+        $_SESSION['return'][] = array(
+          'type' => 'danger',
+          'log' => array(__FUNCTION__, $_action, $_data),
+          'msg' => 'access_denied'
+        );
+        return false;
+      }
+      $stmt = $pdo->prepare("SELECT * FROM `pushover` WHERE `username` = :username");
+      $stmt->execute(array(
+        ':username' => $_data
+      ));
+      $data = $stmt->fetch(PDO::FETCH_ASSOC);
+      if (empty($data)) {
+        return false;
+      }
+      else {
+        return $data;
+      }
+    break;
+    case 'test':
+      if (!isset($_SESSION['acl']['pushover']) || $_SESSION['acl']['pushover'] != "1" ) {
+        $_SESSION['return'][] = array(
+          'type' => 'danger',
+          'log' => array(__FUNCTION__, $_action, $_data),
+          'msg' => 'access_denied'
+        );
+        return false;
+      }
+      if (!is_array($_data['username'])) {
+        $usernames = array();
+        $usernames[] = $_data['username'];
+      }
+      else {
+        $usernames = $_data['username'];
+      }
+      foreach ($usernames as $username) {
+        if (!hasMailboxObjectAccess($_SESSION['mailcow_cc_username'], $_SESSION['mailcow_cc_role'], $username)) {
+          $_SESSION['return'][] = array(
+            'type' => 'danger',
+            'log' => array(__FUNCTION__, $_action, $_data),
+            'msg' => 'access_denied'
+          );
+          continue;
+        }
+        $stmt = $pdo->prepare("SELECT * FROM `pushover`
+          WHERE `username` = :username");
+        $stmt->execute(array(
+          ':username' => $username
+        ));
+        $api_data = $stmt->fetch(PDO::FETCH_ASSOC);
+        if (!empty($api_data)) {
+          $title = (!empty($api_data['title'])) ? $api_data['title'] : 'Mail';
+          $text = (!empty($api_data['text'])) ? $api_data['text'] : 'You\'ve got mail 📧';
+          curl_setopt_array($ch = curl_init(), array(
+            CURLOPT_URL => "https://api.pushover.net/1/users/validate.json",
+            CURLOPT_POSTFIELDS => array(
+              "token" => $api_data['token'],
+              "user" => $api_data['key']
+            ),
+            CURLOPT_SAFE_UPLOAD => true,
+            CURLOPT_RETURNTRANSFER => true,
+          ));
+          $result = curl_exec($ch);
+          $httpcode = curl_getinfo($ch, CURLINFO_HTTP_CODE);
+          curl_close($ch);
+          if ($httpcode == 200) {
+            $_SESSION['return'][] = array(
+              'type' => 'success',
+              'log' => array(__FUNCTION__, $_action, $_data),
+              'msg' => sprintf('Pushover API OK (%d): %s', $httpcode, $result)
+            );
+          }
+          else {
+            $_SESSION['return'][] = array(
+              'type' => 'danger',
+              'log' => array(__FUNCTION__, $_action, $_data),
+              'msg' => sprintf('Pushover API ERR (%d): %s', $httpcode, $result)
+            );
+          }
+        }
+        else {
+          $_SESSION['return'][] = array(
+            'type' => 'danger',
+            'log' => array(__FUNCTION__, $_action, $_data),
+            'msg' => 'pushover_credentials_missing'
+          );
+          return false;
+        }
+      }
+    break;
+  }
+}
\ No newline at end of file
diff --git a/data/web/inc/init_db.inc.php b/data/web/inc/init_db.inc.php
index f122b220..fc50971f 100644
--- a/data/web/inc/init_db.inc.php
+++ b/data/web/inc/init_db.inc.php
@@ -3,7 +3,7 @@ function init_db_schema() {
   try {
     global $pdo;
 
-    $db_version = "03042020_0915";
+    $db_version = "09042020_1403";
 
     $stmt = $pdo->query("SHOW TABLES LIKE 'versions'");
     $num_results = count($stmt->fetchAll(PDO::FETCH_ASSOC));
@@ -374,6 +374,7 @@ function init_db_schema() {
           "syncjobs" => "TINYINT(1) NOT NULL DEFAULT '1'",
           "eas_reset" => "TINYINT(1) NOT NULL DEFAULT '0'",
           "sogo_profile_reset" => "TINYINT(1) NOT NULL DEFAULT '1'",
+          "pushover" => "TINYINT(1) NOT NULL DEFAULT '0'",
           // quarantine is for quarantine actions, todo: rename
           "quarantine" => "TINYINT(1) NOT NULL DEFAULT '1'",
           "quarantine_attachments" => "TINYINT(1) NOT NULL DEFAULT '1'",
@@ -534,6 +535,7 @@ function init_db_schema() {
           "sogo_access" => "TINYINT(1) NOT NULL DEFAULT '1'",
           "app_passwds" => "TINYINT(1) NOT NULL DEFAULT '1'",
           "bcc_maps" => "TINYINT(1) NOT NULL DEFAULT '1'",
+          "pushover" => "TINYINT(1) NOT NULL DEFAULT '0'",
           "filters" => "TINYINT(1) NOT NULL DEFAULT '1'",
           "ratelimit" => "TINYINT(1) NOT NULL DEFAULT '1'",
           "spam_policy" => "TINYINT(1) NOT NULL DEFAULT '1'",
@@ -830,6 +832,22 @@ function init_db_schema() {
         ),
         "attr" => "ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 ROW_FORMAT=DYNAMIC"
       ),
+      "pushover" => array(
+        "cols" => array(
+          "username" => "VARCHAR(255) NOT NULL",
+          "key" => "VARCHAR(255) NOT NULL",
+          "token" => "VARCHAR(255) NOT NULL",
+          "title" => "TEXT",
+          "text" => "TEXT",
+          "active" => "TINYINT(1) NOT NULL DEFAULT '1'"
+        ),
+        "keys" => array(
+          "primary" => array(
+            "" => array("username")
+          )
+        ),
+        "attr" => "ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 ROW_FORMAT=DYNAMIC"
+      ),
       "sogo_user_profile" => array(
         "cols" => array(
           "c_uid" => "VARCHAR(255) NOT NULL",
diff --git a/data/web/inc/prerequisites.inc.php b/data/web/inc/prerequisites.inc.php
index b61679a9..0601cf8e 100644
--- a/data/web/inc/prerequisites.inc.php
+++ b/data/web/inc/prerequisites.inc.php
@@ -230,6 +230,7 @@ require_once $_SERVER['DOCUMENT_ROOT'] . '/inc/functions.dkim.inc.php';
 require_once $_SERVER['DOCUMENT_ROOT'] . '/inc/functions.fwdhost.inc.php';
 require_once $_SERVER['DOCUMENT_ROOT'] . '/inc/functions.mailq.inc.php';
 require_once $_SERVER['DOCUMENT_ROOT'] . '/inc/functions.oauth2.inc.php';
+require_once $_SERVER['DOCUMENT_ROOT'] . '/inc/functions.pushover.inc.php';
 require_once $_SERVER['DOCUMENT_ROOT'] . '/inc/functions.ratelimit.inc.php';
 require_once $_SERVER['DOCUMENT_ROOT'] . '/inc/functions.transports.inc.php';
 require_once $_SERVER['DOCUMENT_ROOT'] . '/inc/functions.rspamd.inc.php';
diff --git a/data/web/inc/sessions.inc.php b/data/web/inc/sessions.inc.php
index e3840d2c..683ebf17 100644
--- a/data/web/inc/sessions.inc.php
+++ b/data/web/inc/sessions.inc.php
@@ -57,6 +57,12 @@ if (!empty($_SERVER['HTTP_X_API_KEY'])) {
       $_SESSION['mailcow_cc_username'] = 'API';
       $_SESSION['mailcow_cc_role'] = 'admin';
       $_SESSION['mailcow_cc_api'] = true;
+      if ($api_return['api_key'] == 'rw') {
+        $_SESSION['mailcow_cc_api_access'] = 'rw';
+      }
+      else {
+        $_SESSION['mailcow_cc_api_access'] = 'ro';
+      }
     }
     else {
       $redis->publish("F2B_CHANNEL", "mailcow UI: Invalid password for API_USER by " . $_SERVER['REMOTE_ADDR']);
diff --git a/data/web/inc/triggers.inc.php b/data/web/inc/triggers.inc.php
index 03f64f91..132e0bd2 100644
--- a/data/web/inc/triggers.inc.php
+++ b/data/web/inc/triggers.inc.php
@@ -100,17 +100,26 @@ if (isset($_SESSION['mailcow_cc_role']) && $_SESSION['mailcow_cc_role'] == "admi
 	if (isset($_POST["reset_main_logo"])) {
     customize('delete', 'main_logo');
 	}
-  // API and license cannot be controlled by API
+  // Some actions will not be available via API
 	if (isset($_POST["license_validate_now"])) {
 		license('verify');
 	}
   if (isset($_POST["admin_api"])) {
-		admin_api('edit', $_POST);
+    if (isset($_POST["admin_api"]["ro"])) {
+      admin_api('ro', 'edit', $_POST);
+    }
+    elseif (isset($_POST["admin_api"]["rw"])) {
+      admin_api('rw', 'edit', $_POST);
+    }
 	}
-	if (isset($_POST["admin_api_regen_key"])) {
-		admin_api('regen_key', $_POST);
+  if (isset($_POST["admin_api_regen_key"])) {
+    if (isset($_POST["admin_api_regen_key"]["ro"])) {
+      admin_api('ro', 'regen_key', $_POST);
+    }
+    elseif (isset($_POST["admin_api_regen_key"]["rw"])) {
+      admin_api('rw', 'regen_key', $_POST);
+    }
 	}
-  // Not available via API
 	if (isset($_POST["rspamd_ui"])) {
 		rspamd_ui('edit', $_POST);
 	}
diff --git a/data/web/js/site/admin.js b/data/web/js/site/admin.js
index 32f76888..62a918df 100644
--- a/data/web/js/site/admin.js
+++ b/data/web/js/site/admin.js
@@ -45,6 +45,24 @@ jQuery(function($){
       mailcow_alert_box('Regex OK', 'success');
       $('button[data-id="' + regex_map_id + '"]').attr({"disabled": false});
     }
+  });
+  $('.btn-api-ro').click(function() {
+    $('#api_rw').hide()
+    $('#api_ro').show()
+    $(this).addClass('active')
+    $('.btn-api-rw, .btn-api-hide').removeClass('active')
+  });
+  $('.btn-api-rw').click(function() {
+    $('#api_ro').hide()
+    $('#api_rw').show()
+    $(this).addClass('active')
+    $('.btn-api-ro, .btn-api-hide').removeClass('active')
+  });
+  $('.btn-api-hide').click(function() {
+    $('#api_ro').hide()
+    $('#api_rw').hide()
+    $(this).addClass('active')
+    $('.btn-api-ro, .btn-api-rw').removeClass('active')
   });
 	$('.textarea-code').on('keyup', function() {
     $('.submit_rspamd_regex').attr({"disabled": true});
@@ -360,13 +378,22 @@ jQuery(function($){
   draw_transport_maps();
   draw_queue();
   // API IP check toggle
-  $("#skip_ip_check").click(function( event ) {
-   $("#skip_ip_check").not(this).prop('checked', false);
-    if ($("#skip_ip_check:checked").length > 0) {
-      $('#allow_from').prop('disabled', true);
+  $("#skip_ip_check_ro").click(function( event ) {
+   $("#skip_ip_check_ro").not(this).prop('checked', false);
+    if ($("#skip_ip_check_ro:checked").length > 0) {
+      $('#allow_from_ro').prop('disabled', true);
     }
     else {
-      $("#allow_from").removeAttr('disabled');
+      $("#allow_from_ro").removeAttr('disabled');
+    }
+  });
+  $("#skip_ip_check_rw").click(function( event ) {
+   $("#skip_ip_check_rw").not(this).prop('checked', false);
+    if ($("#skip_ip_check_rw:checked").length > 0) {
+      $('#allow_from_rw').prop('disabled', true);
+    }
+    else {
+      $("#allow_from_rw").removeAttr('disabled');
     }
   });
   // Relayhost
diff --git a/data/web/json_api.php b/data/web/json_api.php
index a04cb491..8490f3e4 100644
--- a/data/web/json_api.php
+++ b/data/web/json_api.php
@@ -1,17 +1,8 @@
 <?php
 /*
-edit/alias => POST data:
-  {
-    address: {a, b, c},   (where a, b, c represent alias addresses)
-    active: 1             (0 or 1)
-  }
-
-delete/alias => POST data:
-  {
-    address: {a, b, c},   (where a, b, c represent alias addresses)
-  }
-
+   see /api
 */
+
 header('Content-Type: application/json');
 require_once $_SERVER['DOCUMENT_ROOT'] . '/inc/prerequisites.inc.php';
 error_reporting(0);
@@ -103,6 +94,14 @@ if (isset($_SESSION['mailcow_cc_role']) || isset($_SESSION['pending_mailcow_cc_u
 
     switch ($action) {
       case "add":
+        if ($_SESSION['mailcow_cc_api_access'] == 'ro' || isset($_SESSION['pending_mailcow_cc_username'])) {
+          http_response_code(403);
+          echo json_encode(array(
+              'type' => 'error',
+              'msg' => 'API read/write access denied'
+          ));
+          exit();
+        }
         function process_add_return($return) {
           $generic_failure = json_encode(array(
             'type' => 'error',
@@ -136,6 +135,7 @@ if (isset($_SESSION['mailcow_cc_role']) || isset($_SESSION['pending_mailcow_cc_u
           ));
           exit();
         }
+        
         switch ($category) {
           case "time_limited_alias":
             process_add_return(mailbox('add', 'time_limited_alias', $attr));
@@ -235,983 +235,993 @@ if (isset($_SESSION['mailcow_cc_role']) || isset($_SESSION['pending_mailcow_cc_u
           ));
           exit();
         }
-        switch ($category) {
-          case "rspamd":
-            switch ($object) {
-              case "actions":
-                $curl = curl_init();
-                curl_setopt($curl, CURLOPT_UNIX_SOCKET_PATH, '/var/lib/rspamd/rspamd.sock');
-                curl_setopt($curl, CURLOPT_URL,"http://rspamd/stat");
-                curl_setopt($curl, CURLOPT_RETURNTRANSFER, true);
-                $data = curl_exec($curl);
-                if ($data) {
-                  $return = array();
-                  $stats_array = json_decode($data, true)['actions'];
-                  $stats_array['soft reject'] = $stats_array['soft reject'] + $stats_array['greylist'];
-                  unset($stats_array['greylist']);
-                  foreach ($stats_array as $action => $count) {
-                    $return[] = array($action, $count);
-                  }
-                  echo json_encode($return, JSON_UNESCAPED_UNICODE | JSON_PRETTY_PRINT);
-                }
-                elseif (!isset($data) || empty($data)) {
-                  echo '{}';
-                }
-              break;
-            }
-          break;
-
-          case "domain":
-            switch ($object) {
-              case "all":
-                $domains = mailbox('get', 'domains');
-                if (!empty($domains)) {
-                  foreach ($domains as $domain) {
-                    if ($details = mailbox('get', 'domain_details', $domain)) {
-                      $data[] = $details;
-                    }
-                    else {
-                      continue;
-                    }
-                  }
-                  process_get_return($data);
-                }
-                else {
-                  echo '{}';
-                }
-              break;
-
-              default:
-                $data = mailbox('get', 'domain_details', $object);
-                process_get_return($data);
-              break;
-            }
-          break;
-
-          case "app-passwd":
-            switch ($object) {
-              case "all":
-                if (empty($extra)) {
-                  $app_passwds = app_passwd('get');
-                }
-                else {
-                  $app_passwds = app_passwd('get', array('username' => $extra));
-                }
-                if (!empty($app_passwds)) {
-                  foreach ($app_passwds as $app_passwd) {
-                    $details = app_passwd('details', array('id' => $app_passwd['id']));
-                    if ($details !== false) {
-                      $data[] = $details;
-                    }
-                    else {
-                      continue;
-                    }
-                  }
-                  process_get_return($data);
-                }
-                else {
-                  echo '{}';
-                }
-              break;
-
-              default:
-                $data = app_passwd('details', array('id' => $object['id']));
-                process_get_return($data);
-              break;
-            }
-          break;
-
-          case "mailq":
-            switch ($object) {
-              case "all":
-                $mailq = mailq('get');
-                if (!empty($mailq)) {
-                  echo $mailq;
-                }
-                else {
-                  echo '{}';
-                }
-              break;
-            }
-          break;
-
-          case "global_filters":
-            $global_filters = mailbox('get', 'global_filter_details');
-            switch ($object) {
-              case "all":
-                if (!empty($global_filters)) {
-                  process_get_return($global_filters);
-                }
-                else {
-                  echo '{}';
-                }
-              break;
-              case "prefilter":
-                if (!empty($global_filters['prefilter'])) {
-                  process_get_return($global_filters['prefilter']);
-                }
-                else {
-                  echo '{}';
-                }
-              break;
-              case "postfilter":
-                if (!empty($global_filters['postfilter'])) {
-                  process_get_return($global_filters['postfilter']);
-                }
-                else {
-                  echo '{}';
-                }
-              break;
-            }
-          break;
-
-          case "rl-domain":
-            switch ($object) {
-              case "all":
-                $domains = array_merge(mailbox('get', 'domains'), mailbox('get', 'alias_domains'));
-                if (!empty($domains)) {
-                  foreach ($domains as $domain) {
-                    if ($details = ratelimit('get', 'domain', $domain)) {
-                      $details['domain'] = $domain;
-                      $data[] = $details;
-                    }
-                    else {
-                      continue;
-                    }
-                  }
-                  process_get_return($data);
-                }
-                else {
-                  echo '{}';
-                }
-              break;
-
-              default:
-                $data = ratelimit('get', 'domain', $object);
-                process_get_return($data);
-              break;
-            }
-          break;
-
-          case "rl-mbox":
-            switch ($object) {
-              case "all":
-                $domains = mailbox('get', 'domains');
-                if (!empty($domains)) {
-                  foreach ($domains as $domain) {
-                    $mailboxes = mailbox('get', 'mailboxes', $domain);
-                    if (!empty($mailboxes)) {
-                      foreach ($mailboxes as $mailbox) {
-                        if ($details = ratelimit('get', 'mailbox', $mailbox)) {
-                          $details['mailbox'] = $mailbox;
-                          $data[] = $details;
-                        }
-                        else {
-                          continue;
-                        }
-                      }
-                    }
-                  }
-                  process_get_return($data);
-                }
-                else {
-                  echo '{}';
-                }
-              break;
-
-              default:
-                $data = ratelimit('get', 'mailbox', $object);
-                process_get_return($data);
-              break;
-            }
-          break;
-
-          case "relayhost":
-            switch ($object) {
-              case "all":
-                $relayhosts = relayhost('get');
-                if (!empty($relayhosts)) {
-                  foreach ($relayhosts as $relayhost) {
-                    if ($details = relayhost('details', $relayhost['id'])) {
-                      $data[] = $details;
-                    }
-                    else {
-                      continue;
-                    }
-                  }
-                  process_get_return($data);
-                }
-                else {
-                  echo '{}';
-                }
-              break;
-
-              default:
-                $data = relayhost('details', $object);
-                process_get_return($data);
-              break;
-            }
-          break;
-
-          case "transport":
-            switch ($object) {
-              case "all":
-                $transports = transport('get');
-                if (!empty($transports)) {
-                  foreach ($transports as $transport) {
-                    if ($details = transport('details', $transport['id'])) {
-                      $data[] = $details;
-                    }
-                    else {
-                      continue;
-                    }
-                  }
-                  process_get_return($data);
-                }
-                else {
-                  echo '{}';
-                }
-              break;
-
-              default:
-                $data = transport('details', $object);
-                process_get_return($data);
-              break;
-            }
-          break;
-
-          case "rsetting":
-            switch ($object) {
-              case "all":
-                $rsettings = rsettings('get');
-                if (!empty($rsettings)) {
-                  foreach ($rsettings as $rsetting) {
-                    if ($details = rsettings('details', $rsetting['id'])) {
-                      $data[] = $details;
-                    }
-                    else {
-                      continue;
-                    }
-                  }
-                  process_get_return($data);
-                }
-                else {
-                  echo '{}';
-                }
-              break;
-
-              default:
-                $data = rsetting('details', $object);
-                process_get_return($data);
-              break;
-            }
-          break;
-
-          case "oauth2-client":
-            switch ($object) {
-              case "all":
-                $clients = oauth2('get', 'clients');
-                if (!empty($clients)) {
-                  foreach ($clients as $client) {
-                    if ($details = oauth2('details', 'client', $client)) {
-                      $data[] = $details;
-                    }
-                    else {
-                      continue;
-                    }
-                  }
-                  process_get_return($data);
-                }
-                else {
-                  echo '{}';
-                }
-              break;
-
-              default:
-                $data = oauth2('details', 'client', $object);
-                process_get_return($data);
-              break;
-            }
-          break;
-
-          case "logs":
-            switch ($object) {
-              case "dovecot":
-                // 0 is first record, so empty is fine
-                if (isset($extra)) {
-                  $extra = preg_replace('/[^\d\-]/i', '', $extra);
-                  $logs = get_logs('dovecot-mailcow', $extra);
-                }
-                else {
-                  $logs = get_logs('dovecot-mailcow');
-                }
-                echo (isset($logs) && !empty($logs)) ? json_encode($logs, JSON_UNESCAPED_UNICODE | JSON_PRETTY_PRINT) : '{}';
-              break;
-              case "ratelimited":
-                // 0 is first record, so empty is fine
-                if (isset($extra)) {
-                  $extra = preg_replace('/[^\d\-]/i', '', $extra);
-                  $logs = get_logs('ratelimited', $extra);
-                }
-                else {
-                  $logs = get_logs('ratelimited');
-                }
-                echo (isset($logs) && !empty($logs)) ? json_encode($logs, JSON_UNESCAPED_UNICODE | JSON_PRETTY_PRINT) : '{}';
-              break;
-              case "netfilter":
-                // 0 is first record, so empty is fine
-                if (isset($extra)) {
-                  $extra = preg_replace('/[^\d\-]/i', '', $extra);
-                  $logs = get_logs('netfilter-mailcow', $extra);
-                }
-                else {
-                  $logs = get_logs('netfilter-mailcow');
-                }
-                echo (isset($logs) && !empty($logs)) ? json_encode($logs, JSON_UNESCAPED_UNICODE | JSON_PRETTY_PRINT) : '{}';
-              break;
-              case "postfix":
-                // 0 is first record, so empty is fine
-                if (isset($extra)) {
-                  $extra = preg_replace('/[^\d\-]/i', '', $extra);
-                  $logs = get_logs('postfix-mailcow', $extra);
-                }
-                else {
-                  $logs = get_logs('postfix-mailcow');
-                }
-                echo (isset($logs) && !empty($logs)) ? json_encode($logs, JSON_UNESCAPED_UNICODE | JSON_PRETTY_PRINT) : '{}';
-              break;
-              case "autodiscover":
-                // 0 is first record, so empty is fine
-                if (isset($extra)) {
-                  $extra = preg_replace('/[^\d\-]/i', '', $extra);
-                  $logs = get_logs('autodiscover-mailcow', $extra);
-                }
-                else {
-                  $logs = get_logs('autodiscover-mailcow');
-                }
-                echo (isset($logs) && !empty($logs)) ? json_encode($logs, JSON_UNESCAPED_UNICODE | JSON_PRETTY_PRINT) : '{}';
-              break;
-              case "sogo":
-                // 0 is first record, so empty is fine
-                if (isset($extra)) {
-                  $extra = preg_replace('/[^\d\-]/i', '', $extra);
-                  $logs = get_logs('sogo-mailcow', $extra);
-                }
-                else {
-                  $logs = get_logs('sogo-mailcow');
-                }
-                echo (isset($logs) && !empty($logs)) ? json_encode($logs, JSON_UNESCAPED_UNICODE | JSON_PRETTY_PRINT) : '{}';
-              break;
-              case "ui":
-                // 0 is first record, so empty is fine
-                if (isset($extra)) {
-                  $extra = preg_replace('/[^\d\-]/i', '', $extra);
-                  $logs = get_logs('mailcow-ui', $extra);
-                }
-                else {
-                  $logs = get_logs('mailcow-ui');
-                }
-                echo (isset($logs) && !empty($logs)) ? json_encode($logs, JSON_UNESCAPED_UNICODE | JSON_PRETTY_PRINT) : '{}';
-              break;
-              case "watchdog":
-                // 0 is first record, so empty is fine
-                if (isset($extra)) {
-                  $extra = preg_replace('/[^\d\-]/i', '', $extra);
-                  $logs = get_logs('watchdog-mailcow', $extra);
-                }
-                else {
-                  $logs = get_logs('watchdog-mailcow');
-                }
-                echo (isset($logs) && !empty($logs)) ? json_encode($logs, JSON_UNESCAPED_UNICODE | JSON_PRETTY_PRINT) : '{}';
-              break;
-              case "acme":
-                // 0 is first record, so empty is fine
-                if (isset($extra)) {
-                  $extra = preg_replace('/[^\d\-]/i', '', $extra);
-                  $logs = get_logs('acme-mailcow', $extra);
-                }
-                else {
-                  $logs = get_logs('acme-mailcow');
-                }
-                echo (isset($logs) && !empty($logs)) ? json_encode($logs, JSON_UNESCAPED_UNICODE | JSON_PRETTY_PRINT) : '{}';
-              break;
-              case "api":
-                // 0 is first record, so empty is fine
-                if (isset($extra)) {
-                  $extra = preg_replace('/[^\d\-]/i', '', $extra);
-                  $logs = get_logs('api-mailcow', $extra);
-                }
-                else {
-                  $logs = get_logs('api-mailcow');
-                }
-                echo (isset($logs) && !empty($logs)) ? json_encode($logs, JSON_UNESCAPED_UNICODE | JSON_PRETTY_PRINT) : '{}';
-              break;
-              case "rspamd-history":
-                // 0 is first record, so empty is fine
-                if (isset($extra)) {
-                  $extra = preg_replace('/[^\d\-]/i', '', $extra);
-                  $logs = get_logs('rspamd-history', $extra);
-                }
-                else {
-                  $logs = get_logs('rspamd-history');
-                }
-                echo (isset($logs) && !empty($logs)) ? json_encode($logs, JSON_UNESCAPED_UNICODE | JSON_PRETTY_PRINT) : '{}';
-              break;
-              // return no route found if no case is matched
-              default:
-                http_response_code(404);
-                echo json_encode(array(
-                  'type' => 'error',
-                  'msg' => 'route not found'
-                ));
-                exit();
-            }
-          break;
-          case "mailbox":
-            switch ($object) {
-              case "all":
-                if (empty($extra)) {
-                  $domains = mailbox('get', 'domains');
-                }
-                else {
-                  $domains = array($extra);
-                }
-                if (!empty($domains)) {
-                  foreach ($domains as $domain) {
-                    $mailboxes = mailbox('get', 'mailboxes', $domain);
-                    if (!empty($mailboxes)) {
-                      foreach ($mailboxes as $mailbox) {
-                        if ($details = mailbox('get', 'mailbox_details', $mailbox)) {
-                          $data[] = $details;
-                        }
-                        else {
-                          continue;
-                        }
-                      }
-                    }
-                  }
-                  process_get_return($data);
-                }
-                else {
-                  echo '{}';
-                }
-              break;
-
-              default:
-                $data = mailbox('get', 'mailbox_details', $object);
-                process_get_return($data);
-              break;
-            }
-          break;
-          case "syncjobs":
-            switch ($object) {
-              case "all":
-                $domains = mailbox('get', 'domains');
-                if (!empty($domains)) {
-                  foreach ($domains as $domain) {
-                    $mailboxes = mailbox('get', 'mailboxes', $domain);
-                    if (!empty($mailboxes)) {
-                      foreach ($mailboxes as $mailbox) {
-                        $syncjobs = mailbox('get', 'syncjobs', $mailbox);
-                        if (!empty($syncjobs)) {
-                          foreach ($syncjobs as $syncjob) {
-                            if (isset($extra)) {
-                              $details = mailbox('get', 'syncjob_details', $syncjob, explode(',', $extra));
-                            }
-                            else {
-                              $details = mailbox('get', 'syncjob_details', $syncjob);
-                            }
-                            if ($details) {
-                              $data[] = $details;
-                            }
-                            else {
-                              continue;
-                            }
-                          }
-                        }
-                      }
-                    }
-                  }
-                  process_get_return($data);
-                }
-                else {
-                  echo '{}';
-                }
-              break;
-
-              default:
-                $syncjobs = mailbox('get', 'syncjobs', $object);
-                if (!empty($syncjobs)) {
-                  foreach ($syncjobs as $syncjob) {
-                    if (isset($extra)) {
-                      $details = mailbox('get', 'syncjob_details', $syncjob, explode(',', $extra));
-                    }
-                    else {
-                      $details = mailbox('get', 'syncjob_details', $syncjob);
-                    }
-                    if ($details) {
-                      $data[] = $details;
-                    }
-                    else {
-                      continue;
-                    }
-                  }
-                }
-                process_get_return($data);
-              break;
-            }
-          break;
-          case "active-user-sieve":
-            if (isset($object)) {
-              $sieve_filter = mailbox('get', 'active_user_sieve', $object);
-              if (!empty($sieve_filter)) {
-                $data[] = $sieve_filter;
-              }
-            }
-            process_get_return($data);
-          break;
-          case "filters":
-            switch ($object) {
-              case "all":
-                $domains = mailbox('get', 'domains');
-                if (!empty($domains)) {
-                  foreach ($domains as $domain) {
-                    $mailboxes = mailbox('get', 'mailboxes', $domain);
-                    if (!empty($mailboxes)) {
-                      foreach ($mailboxes as $mailbox) {
-                        $filters = mailbox('get', 'filters', $mailbox);
-                        if (!empty($filters)) {
-                          foreach ($filters as $filter) {
-                            if ($details = mailbox('get', 'filter_details', $filter)) {
-                              $data[] = $details;
-                            }
-                            else {
-                              continue;
-                            }
-                          }
-                        }
-                      }
-                    }
-                  }
-                  process_get_return($data);
-                }
-                else {
-                  echo '{}';
-                }
-              break;
-
-              default:
-                $filters = mailbox('get', 'filters', $object);
-                if (!empty($filters)) {
-                  foreach ($filters as $filter) {
-                    if ($details = mailbox('get', 'filter_details', $filter)) {
-                      $data[] = $details;
-                    }
-                    else {
-                      continue;
-                    }
-                  }
-                }
-                process_get_return($data);
-              break;
-            }
-          break;
-          case "bcc":
-            switch ($object) {
-              case "all":
-                $bcc_items = bcc('get');
-                if (!empty($bcc_items)) {
-                  foreach ($bcc_items as $bcc_item) {
-                    if ($details = bcc('details', $bcc_item)) {
-                      $data[] = $details;
-                    }
-                    else {
-                      continue;
-                    }
-                  }
-                }
-                process_get_return($data);
-              break;
-              default:
-                $data = bcc('details', $object);
-                if (!empty($data)) {
-                  $data[] = $details;
-                }
-                process_get_return($data);
-              break;
-            }
-          break;
-          case "recipient_map":
-            switch ($object) {
-              case "all":
-                $recipient_map_items = recipient_map('get');
-                if (!empty($recipient_map_items)) {
-                  foreach ($recipient_map_items as $recipient_map_item) {
-                    if ($details = recipient_map('details', $recipient_map_item)) {
-                      $data[] = $details;
-                    }
-                    else {
-                      continue;
-                    }
-                  }
-                }
-                process_get_return($data);
-              break;
-              default:
-                $data = recipient_map('details', $object);
-                if (!empty($data)) {
-                  $data[] = $details;
-                }
-                process_get_return($data);
-              break;
-            }
-          break;
-          case "tls-policy-map":
-            switch ($object) {
-              case "all":
-                $tls_policy_maps_items = tls_policy_maps('get');
-                if (!empty($tls_policy_maps_items)) {
-                  foreach ($tls_policy_maps_items as $tls_policy_maps_item) {
-                    if ($details = tls_policy_maps('details', $tls_policy_maps_item)) {
-                      $data[] = $details;
-                    }
-                    else {
-                      continue;
-                    }
-                  }
-                }
-                process_get_return($data);
-              break;
-              default:
-                $data = tls_policy_maps('details', $object);
-                if (!empty($data)) {
-                  $data[] = $details;
-                }
-                process_get_return($data);
-              break;
-            }
-          break;
-          case "policy_wl_mailbox":
-            switch ($object) {
-              default:
-                $data = policy('get', 'mailbox', $object)['whitelist'];
-                process_get_return($data);
-              break;
-            }
-          break;
-          case "policy_bl_mailbox":
-            switch ($object) {
-              default:
-                $data = policy('get', 'mailbox', $object)['blacklist'];
-                process_get_return($data);
-              break;
-            }
-          break;
-          case "policy_wl_domain":
-            switch ($object) {
-              default:
-                $data = policy('get', 'domain', $object)['whitelist'];
-                process_get_return($data);
-              break;
-            }
-          break;
-          case "policy_bl_domain":
-            switch ($object) {
-              default:
-                $data = policy('get', 'domain', $object)['blacklist'];
-                process_get_return($data);
-              break;
-            }
-          break;
-          case "time_limited_aliases":
-            switch ($object) {
-              default:
-                $data = mailbox('get', 'time_limited_aliases', $object);
-                process_get_return($data);
-              break;
-            }
-          break;
-          case "fail2ban":
-            switch ($object) {
-              default:
-                $data = fail2ban('get');
-                process_get_return($data);
-              break;
-            }
-          break;
-          case "resource":
-            switch ($object) {
-              case "all":
-                $domains = mailbox('get', 'domains');
-                if (!empty($domains)) {
-                  foreach ($domains as $domain) {
-                    $resources = mailbox('get', 'resources', $domain);
-                    if (!empty($resources)) {
-                      foreach ($resources as $resource) {
-                        if ($details = mailbox('get', 'resource_details', $resource)) {
-                          $data[] = $details;
-                        }
-                        else {
-                          continue;
-                        }
-                      }
-                    }
-                  }
-                  process_get_return($data);
-                }
-                else {
-                  echo '{}';
-                }
-              break;
-              default:
-                $data = mailbox('get', 'resource_details', $object);
-                process_get_return($data);
-              break;
-            }
-          break;
-          case "fwdhost":
-            switch ($object) {
-              case "all":
-                process_get_return(fwdhost('get'));
-              break;
-              default:
-                process_get_return(fwdhost('details', $object));
-              break;
-            }
-          break;
-          case "quarantine":
-            // "all" will not print details
-            switch ($object) {
-              case "all":
-                process_get_return(quarantine('get'));
-              break;
-              default:
-                process_get_return(quarantine('details', $object));
-              break;
-            }
-          break;
-          case "alias-domain":
-            switch ($object) {
-              case "all":
-                $alias_domains = mailbox('get', 'alias_domains');
-                if (!empty($alias_domains)) {
-                  foreach ($alias_domains as $alias_domain) {
-                    if ($details = mailbox('get', 'alias_domain_details', $alias_domain)) {
-                      $data[] = $details;
-                    }
-                    else {
-                      continue;
-                    }
-                  }
-                }
-                process_get_return($data);
-              break;
-              default:
-                process_get_return(mailbox('get', 'alias_domain_details', $object));
-              break;
-            }
-          break;
-          case "alias":
-            switch ($object) {
-              case "all":
-                if (empty($extra)) {
-                  $domains = array_merge(mailbox('get', 'domains'), mailbox('get', 'alias_domains'));
-                }
-                else {
-                  $domains = array($extra);
-                }
-                if (!empty($domains)) {
-                  foreach ($domains as $domain) {
-                    $aliases = mailbox('get', 'aliases', $domain);
-                    if (!empty($aliases)) {
-                      foreach ($aliases as $alias) {
-                        if ($details = mailbox('get', 'alias_details', $alias)) {
-                          $data[] = $details;
-                        }
-                        else {
-                          continue;
-                        }
-                      }
-                    }
-                  }
-                  process_get_return($data);
-                }
-                else {
-                  echo '{}';
-                }
-              break;
-
-              default:
-                process_get_return(mailbox('get', 'alias_details', $object));
-              break;
-            }
-          break;
-          case "domain-admin":
-            switch ($object) {
-              case "all":
-                $domain_admins = domain_admin('get');
-                if (!empty($domain_admins)) {
-                  foreach ($domain_admins as $domain_admin) {
-                    if ($details = domain_admin('details', $domain_admin)) {
-                      $data[] = $details;
-                    }
-                    else {
-                      continue;
-                    }
-                  }
-                  process_get_return($data);
-                }
-                else {
-                  echo '{}';
-                }
-              break;
-
-              default:
-                process_get_return(domain_admin('details', $object));
-              break;
-            }
-          break;
-          case "admin":
-            switch ($object) {
-              case "all":
-                $admins = admin('get');
-                if (!empty($admins)) {
-                  foreach ($admins as $admin) {
-                    if ($details = admin('details', $admin)) {
-                      $data[] = $details;
-                    }
-                    else {
-                      continue;
-                    }
-                  }
-                  process_get_return($data);
-                }
-                else {
-                  echo '{}';
-                }
-              break;
-
-              default:
-                process_get_return(admin('details', $object));
-              break;
-            }
-          break;
-          case "u2f-registration":
-            header('Content-Type: application/javascript');
-            if (($_SESSION["mailcow_cc_role"] == "admin" || $_SESSION["mailcow_cc_role"] == "domainadmin") && $_SESSION["mailcow_cc_username"] == $object) {
-              list($req, $sigs) = $u2f->getRegisterData(get_u2f_registrations($object));
-              $_SESSION['regReq'] = json_encode($req);
-              $_SESSION['regSigs'] = json_encode($sigs);
-              echo 'var req = ' . json_encode($req) . ';';
-              echo 'var registeredKeys = ' . json_encode($sigs) . ';';
-              echo 'var appId = req.appId;';
-              echo 'var registerRequests = [{version: req.version, challenge: req.challenge}];';
-            }
-            else {
-              return;
-            }
-          break;
-          case "u2f-authentication":
-            header('Content-Type: application/javascript');
-            if (isset($_SESSION['pending_mailcow_cc_username']) && $_SESSION['pending_mailcow_cc_username'] == $object) {
-              $auth_data = $u2f->getAuthenticateData(get_u2f_registrations($object));
-              $challenge = $auth_data[0]->challenge;
-              $appId = $auth_data[0]->appId;
-              foreach ($auth_data as $each) {
-                $key = array(); // Empty array
-                $key['version']   = $each->version;
-                $key['keyHandle'] = $each->keyHandle;
-                $registeredKey[]  = $key;
-              }
-              $_SESSION['authReq']  = json_encode($auth_data);
-              echo 'var appId = "' . $appId . '";';
-              echo 'var challenge = ' . json_encode($challenge) . ';';
-              echo 'var registeredKeys = ' . json_encode($registeredKey) . ';';
-            }
-            else {
-              return;
-            }
-          break;
-          case "dkim":
-            switch ($object) {
-              default:
-                $data = dkim('details', $object);
-                process_get_return($data);
-                break;
-            }
-          break;
-          case "presets":
-            switch ($object) {
-              case "rspamd":
-                process_get_return(presets('get', 'rspamd'));
-              break;
-              case "sieve":
-                process_get_return(presets('get', 'sieve'));
-              break;
-            }
-          break;
-          case "status":
-            switch ($object) {
-              case "containers":
-                $containers = (docker('info'));
-                foreach ($containers as $container => $container_info) {
-                  $container . ' (' . $container_info['Config']['Image'] . ')';
-                  $containerstarttime = ($container_info['State']['StartedAt']);
-                  $containerstate = ($container_info['State']['Status']);
-                  $containerimage = ($container_info['Config']['Image']);
-                  $temp[$container] = array(
-                    'type' => 'info',
-                    'container' => $container,
-                    'state' => $containerstate,
-                    'started_at' => $containerstarttime,
-                    'image' => $containerimage
-                  );
-                }
-                echo json_encode($temp, JSON_UNESCAPED_SLASHES);
-              break;
-              case "vmail":
-                $exec_fields_vmail = array('cmd' => 'system', 'task' => 'df', 'dir' => '/var/vmail');
-                $vmail_df = explode(',', json_decode(docker('post', 'dovecot-mailcow', 'exec', $exec_fields_vmail), true));
-                $temp = array(
-                  'type' => 'info',
-                  'disk' => $vmail_df[0],
-                  'used' => $vmail_df[2],
-                  'total'=> $vmail_df[1],
-                  'used_percent' => $vmail_df[4]
-                );
-                echo json_encode($temp, JSON_UNESCAPED_SLASHES);
-            break;
-            case "solr":
-              $solr_status = solr_status();
-              $solr_size = ($solr_status['status']['dovecot-fts']['index']['size']);
-              $solr_documents = ($solr_status['status']['dovecot-fts']['index']['numDocs']);
-              if (strtolower(getenv('SKIP_SOLR')) != 'n') {
-                $solr_enabled = false;
+        if (!isset($_SESSION['pending_mailcow_cc_username'])) {
+          switch ($category) {
+            case "u2f-registration":
+              header('Content-Type: application/javascript');
+              if (($_SESSION["mailcow_cc_role"] == "admin" || $_SESSION["mailcow_cc_role"] == "domainadmin") && $_SESSION["mailcow_cc_username"] == $object) {
+                list($req, $sigs) = $u2f->getRegisterData(get_u2f_registrations($object));
+                $_SESSION['regReq'] = json_encode($req);
+                $_SESSION['regSigs'] = json_encode($sigs);
+                echo 'var req = ' . json_encode($req) . ';';
+                echo 'var registeredKeys = ' . json_encode($sigs) . ';';
+                echo 'var appId = req.appId;';
+                echo 'var registerRequests = [{version: req.version, challenge: req.challenge}];';
               }
               else {
-                $solr_enabled = true;
+                return;
+              }
+            break;
+            case "u2f-authentication":
+              header('Content-Type: application/javascript');
+              if (isset($_SESSION['pending_mailcow_cc_username']) && $_SESSION['pending_mailcow_cc_username'] == $object) {
+                $auth_data = $u2f->getAuthenticateData(get_u2f_registrations($object));
+                $challenge = $auth_data[0]->challenge;
+                $appId = $auth_data[0]->appId;
+                foreach ($auth_data as $each) {
+                  $key = array(); // Empty array
+                  $key['version']   = $each->version;
+                  $key['keyHandle'] = $each->keyHandle;
+                  $registeredKey[]  = $key;
+                }
+                $_SESSION['authReq']  = json_encode($auth_data);
+                echo 'var appId = "' . $appId . '";';
+                echo 'var challenge = ' . json_encode($challenge) . ';';
+                echo 'var registeredKeys = ' . json_encode($registeredKey) . ';';
+              }
+              else {
+                return;
+              }
+            break;
+            case "rspamd":
+              switch ($object) {
+                case "actions":
+                  $curl = curl_init();
+                  curl_setopt($curl, CURLOPT_UNIX_SOCKET_PATH, '/var/lib/rspamd/rspamd.sock');
+                  curl_setopt($curl, CURLOPT_URL,"http://rspamd/stat");
+                  curl_setopt($curl, CURLOPT_RETURNTRANSFER, true);
+                  $data = curl_exec($curl);
+                  if ($data) {
+                    $return = array();
+                    $stats_array = json_decode($data, true)['actions'];
+                    $stats_array['soft reject'] = $stats_array['soft reject'] + $stats_array['greylist'];
+                    unset($stats_array['greylist']);
+                    foreach ($stats_array as $action => $count) {
+                      $return[] = array($action, $count);
+                    }
+                    echo json_encode($return, JSON_UNESCAPED_UNICODE | JSON_PRETTY_PRINT);
+                  }
+                  elseif (!isset($data) || empty($data)) {
+                    echo '{}';
+                  }
+                break;
+              }
+            break;
+
+            case "domain":
+              switch ($object) {
+                case "all":
+                  $domains = mailbox('get', 'domains');
+                  if (!empty($domains)) {
+                    foreach ($domains as $domain) {
+                      if ($details = mailbox('get', 'domain_details', $domain)) {
+                        $data[] = $details;
+                      }
+                      else {
+                        continue;
+                      }
+                    }
+                    process_get_return($data);
+                  }
+                  else {
+                    echo '{}';
+                  }
+                break;
+
+                default:
+                  $data = mailbox('get', 'domain_details', $object);
+                  process_get_return($data);
+                break;
+              }
+            break;
+
+            case "app-passwd":
+              switch ($object) {
+                case "all":
+                  if (empty($extra)) {
+                    $app_passwds = app_passwd('get');
+                  }
+                  else {
+                    $app_passwds = app_passwd('get', array('username' => $extra));
+                  }
+                  if (!empty($app_passwds)) {
+                    foreach ($app_passwds as $app_passwd) {
+                      $details = app_passwd('details', array('id' => $app_passwd['id']));
+                      if ($details !== false) {
+                        $data[] = $details;
+                      }
+                      else {
+                        continue;
+                      }
+                    }
+                    process_get_return($data);
+                  }
+                  else {
+                    echo '{}';
+                  }
+                break;
+
+                default:
+                  $data = app_passwd('details', array('id' => $object['id']));
+                  process_get_return($data);
+                break;
+              }
+            break;
+
+            case "mailq":
+              switch ($object) {
+                case "all":
+                  $mailq = mailq('get');
+                  if (!empty($mailq)) {
+                    echo $mailq;
+                  }
+                  else {
+                    echo '{}';
+                  }
+                break;
+              }
+            break;
+
+            case "global_filters":
+              $global_filters = mailbox('get', 'global_filter_details');
+              switch ($object) {
+                case "all":
+                  if (!empty($global_filters)) {
+                    process_get_return($global_filters);
+                  }
+                  else {
+                    echo '{}';
+                  }
+                break;
+                case "prefilter":
+                  if (!empty($global_filters['prefilter'])) {
+                    process_get_return($global_filters['prefilter']);
+                  }
+                  else {
+                    echo '{}';
+                  }
+                break;
+                case "postfilter":
+                  if (!empty($global_filters['postfilter'])) {
+                    process_get_return($global_filters['postfilter']);
+                  }
+                  else {
+                    echo '{}';
+                  }
+                break;
+              }
+            break;
+
+            case "rl-domain":
+              switch ($object) {
+                case "all":
+                  $domains = array_merge(mailbox('get', 'domains'), mailbox('get', 'alias_domains'));
+                  if (!empty($domains)) {
+                    foreach ($domains as $domain) {
+                      if ($details = ratelimit('get', 'domain', $domain)) {
+                        $details['domain'] = $domain;
+                        $data[] = $details;
+                      }
+                      else {
+                        continue;
+                      }
+                    }
+                    process_get_return($data);
+                  }
+                  else {
+                    echo '{}';
+                  }
+                break;
+
+                default:
+                  $data = ratelimit('get', 'domain', $object);
+                  process_get_return($data);
+                break;
+              }
+            break;
+
+            case "rl-mbox":
+              switch ($object) {
+                case "all":
+                  $domains = mailbox('get', 'domains');
+                  if (!empty($domains)) {
+                    foreach ($domains as $domain) {
+                      $mailboxes = mailbox('get', 'mailboxes', $domain);
+                      if (!empty($mailboxes)) {
+                        foreach ($mailboxes as $mailbox) {
+                          if ($details = ratelimit('get', 'mailbox', $mailbox)) {
+                            $details['mailbox'] = $mailbox;
+                            $data[] = $details;
+                          }
+                          else {
+                            continue;
+                          }
+                        }
+                      }
+                    }
+                    process_get_return($data);
+                  }
+                  else {
+                    echo '{}';
+                  }
+                break;
+
+                default:
+                  $data = ratelimit('get', 'mailbox', $object);
+                  process_get_return($data);
+                break;
+              }
+            break;
+
+            case "relayhost":
+              switch ($object) {
+                case "all":
+                  $relayhosts = relayhost('get');
+                  if (!empty($relayhosts)) {
+                    foreach ($relayhosts as $relayhost) {
+                      if ($details = relayhost('details', $relayhost['id'])) {
+                        $data[] = $details;
+                      }
+                      else {
+                        continue;
+                      }
+                    }
+                    process_get_return($data);
+                  }
+                  else {
+                    echo '{}';
+                  }
+                break;
+
+                default:
+                  $data = relayhost('details', $object);
+                  process_get_return($data);
+                break;
+              }
+            break;
+
+            case "transport":
+              switch ($object) {
+                case "all":
+                  $transports = transport('get');
+                  if (!empty($transports)) {
+                    foreach ($transports as $transport) {
+                      if ($details = transport('details', $transport['id'])) {
+                        $data[] = $details;
+                      }
+                      else {
+                        continue;
+                      }
+                    }
+                    process_get_return($data);
+                  }
+                  else {
+                    echo '{}';
+                  }
+                break;
+
+                default:
+                  $data = transport('details', $object);
+                  process_get_return($data);
+                break;
+              }
+            break;
+
+            case "rsetting":
+              switch ($object) {
+                case "all":
+                  $rsettings = rsettings('get');
+                  if (!empty($rsettings)) {
+                    foreach ($rsettings as $rsetting) {
+                      if ($details = rsettings('details', $rsetting['id'])) {
+                        $data[] = $details;
+                      }
+                      else {
+                        continue;
+                      }
+                    }
+                    process_get_return($data);
+                  }
+                  else {
+                    echo '{}';
+                  }
+                break;
+
+                default:
+                  $data = rsetting('details', $object);
+                  process_get_return($data);
+                break;
+              }
+            break;
+
+            case "oauth2-client":
+              switch ($object) {
+                case "all":
+                  $clients = oauth2('get', 'clients');
+                  if (!empty($clients)) {
+                    foreach ($clients as $client) {
+                      if ($details = oauth2('details', 'client', $client)) {
+                        $data[] = $details;
+                      }
+                      else {
+                        continue;
+                      }
+                    }
+                    process_get_return($data);
+                  }
+                  else {
+                    echo '{}';
+                  }
+                break;
+
+                default:
+                  $data = oauth2('details', 'client', $object);
+                  process_get_return($data);
+                break;
+              }
+            break;
+
+            case "logs":
+              switch ($object) {
+                case "dovecot":
+                  // 0 is first record, so empty is fine
+                  if (isset($extra)) {
+                    $extra = preg_replace('/[^\d\-]/i', '', $extra);
+                    $logs = get_logs('dovecot-mailcow', $extra);
+                  }
+                  else {
+                    $logs = get_logs('dovecot-mailcow');
+                  }
+                  echo (isset($logs) && !empty($logs)) ? json_encode($logs, JSON_UNESCAPED_UNICODE | JSON_PRETTY_PRINT) : '{}';
+                break;
+                case "ratelimited":
+                  // 0 is first record, so empty is fine
+                  if (isset($extra)) {
+                    $extra = preg_replace('/[^\d\-]/i', '', $extra);
+                    $logs = get_logs('ratelimited', $extra);
+                  }
+                  else {
+                    $logs = get_logs('ratelimited');
+                  }
+                  echo (isset($logs) && !empty($logs)) ? json_encode($logs, JSON_UNESCAPED_UNICODE | JSON_PRETTY_PRINT) : '{}';
+                break;
+                case "netfilter":
+                  // 0 is first record, so empty is fine
+                  if (isset($extra)) {
+                    $extra = preg_replace('/[^\d\-]/i', '', $extra);
+                    $logs = get_logs('netfilter-mailcow', $extra);
+                  }
+                  else {
+                    $logs = get_logs('netfilter-mailcow');
+                  }
+                  echo (isset($logs) && !empty($logs)) ? json_encode($logs, JSON_UNESCAPED_UNICODE | JSON_PRETTY_PRINT) : '{}';
+                break;
+                case "postfix":
+                  // 0 is first record, so empty is fine
+                  if (isset($extra)) {
+                    $extra = preg_replace('/[^\d\-]/i', '', $extra);
+                    $logs = get_logs('postfix-mailcow', $extra);
+                  }
+                  else {
+                    $logs = get_logs('postfix-mailcow');
+                  }
+                  echo (isset($logs) && !empty($logs)) ? json_encode($logs, JSON_UNESCAPED_UNICODE | JSON_PRETTY_PRINT) : '{}';
+                break;
+                case "autodiscover":
+                  // 0 is first record, so empty is fine
+                  if (isset($extra)) {
+                    $extra = preg_replace('/[^\d\-]/i', '', $extra);
+                    $logs = get_logs('autodiscover-mailcow', $extra);
+                  }
+                  else {
+                    $logs = get_logs('autodiscover-mailcow');
+                  }
+                  echo (isset($logs) && !empty($logs)) ? json_encode($logs, JSON_UNESCAPED_UNICODE | JSON_PRETTY_PRINT) : '{}';
+                break;
+                case "sogo":
+                  // 0 is first record, so empty is fine
+                  if (isset($extra)) {
+                    $extra = preg_replace('/[^\d\-]/i', '', $extra);
+                    $logs = get_logs('sogo-mailcow', $extra);
+                  }
+                  else {
+                    $logs = get_logs('sogo-mailcow');
+                  }
+                  echo (isset($logs) && !empty($logs)) ? json_encode($logs, JSON_UNESCAPED_UNICODE | JSON_PRETTY_PRINT) : '{}';
+                break;
+                case "ui":
+                  // 0 is first record, so empty is fine
+                  if (isset($extra)) {
+                    $extra = preg_replace('/[^\d\-]/i', '', $extra);
+                    $logs = get_logs('mailcow-ui', $extra);
+                  }
+                  else {
+                    $logs = get_logs('mailcow-ui');
+                  }
+                  echo (isset($logs) && !empty($logs)) ? json_encode($logs, JSON_UNESCAPED_UNICODE | JSON_PRETTY_PRINT) : '{}';
+                break;
+                case "watchdog":
+                  // 0 is first record, so empty is fine
+                  if (isset($extra)) {
+                    $extra = preg_replace('/[^\d\-]/i', '', $extra);
+                    $logs = get_logs('watchdog-mailcow', $extra);
+                  }
+                  else {
+                    $logs = get_logs('watchdog-mailcow');
+                  }
+                  echo (isset($logs) && !empty($logs)) ? json_encode($logs, JSON_UNESCAPED_UNICODE | JSON_PRETTY_PRINT) : '{}';
+                break;
+                case "acme":
+                  // 0 is first record, so empty is fine
+                  if (isset($extra)) {
+                    $extra = preg_replace('/[^\d\-]/i', '', $extra);
+                    $logs = get_logs('acme-mailcow', $extra);
+                  }
+                  else {
+                    $logs = get_logs('acme-mailcow');
+                  }
+                  echo (isset($logs) && !empty($logs)) ? json_encode($logs, JSON_UNESCAPED_UNICODE | JSON_PRETTY_PRINT) : '{}';
+                break;
+                case "api":
+                  // 0 is first record, so empty is fine
+                  if (isset($extra)) {
+                    $extra = preg_replace('/[^\d\-]/i', '', $extra);
+                    $logs = get_logs('api-mailcow', $extra);
+                  }
+                  else {
+                    $logs = get_logs('api-mailcow');
+                  }
+                  echo (isset($logs) && !empty($logs)) ? json_encode($logs, JSON_UNESCAPED_UNICODE | JSON_PRETTY_PRINT) : '{}';
+                break;
+                case "rspamd-history":
+                  // 0 is first record, so empty is fine
+                  if (isset($extra)) {
+                    $extra = preg_replace('/[^\d\-]/i', '', $extra);
+                    $logs = get_logs('rspamd-history', $extra);
+                  }
+                  else {
+                    $logs = get_logs('rspamd-history');
+                  }
+                  echo (isset($logs) && !empty($logs)) ? json_encode($logs, JSON_UNESCAPED_UNICODE | JSON_PRETTY_PRINT) : '{}';
+                break;
+                // return no route found if no case is matched
+                default:
+                  http_response_code(404);
+                  echo json_encode(array(
+                    'type' => 'error',
+                    'msg' => 'route not found'
+                  ));
+                  exit();
+              }
+            break;
+            case "mailbox":
+              switch ($object) {
+                case "all":
+                  if (empty($extra)) {
+                    $domains = mailbox('get', 'domains');
+                  }
+                  else {
+                    $domains = array($extra);
+                  }
+                  if (!empty($domains)) {
+                    foreach ($domains as $domain) {
+                      $mailboxes = mailbox('get', 'mailboxes', $domain);
+                      if (!empty($mailboxes)) {
+                        foreach ($mailboxes as $mailbox) {
+                          if ($details = mailbox('get', 'mailbox_details', $mailbox)) {
+                            $data[] = $details;
+                          }
+                          else {
+                            continue;
+                          }
+                        }
+                      }
+                    }
+                    process_get_return($data);
+                  }
+                  else {
+                    echo '{}';
+                  }
+                break;
+
+                default:
+                  $data = mailbox('get', 'mailbox_details', $object);
+                  process_get_return($data);
+                break;
+              }
+            break;
+            case "syncjobs":
+              switch ($object) {
+                case "all":
+                  $domains = mailbox('get', 'domains');
+                  if (!empty($domains)) {
+                    foreach ($domains as $domain) {
+                      $mailboxes = mailbox('get', 'mailboxes', $domain);
+                      if (!empty($mailboxes)) {
+                        foreach ($mailboxes as $mailbox) {
+                          $syncjobs = mailbox('get', 'syncjobs', $mailbox);
+                          if (!empty($syncjobs)) {
+                            foreach ($syncjobs as $syncjob) {
+                              if (isset($extra)) {
+                                $details = mailbox('get', 'syncjob_details', $syncjob, explode(',', $extra));
+                              }
+                              else {
+                                $details = mailbox('get', 'syncjob_details', $syncjob);
+                              }
+                              if ($details) {
+                                $data[] = $details;
+                              }
+                              else {
+                                continue;
+                              }
+                            }
+                          }
+                        }
+                      }
+                    }
+                    process_get_return($data);
+                  }
+                  else {
+                    echo '{}';
+                  }
+                break;
+
+                default:
+                  $syncjobs = mailbox('get', 'syncjobs', $object);
+                  if (!empty($syncjobs)) {
+                    foreach ($syncjobs as $syncjob) {
+                      if (isset($extra)) {
+                        $details = mailbox('get', 'syncjob_details', $syncjob, explode(',', $extra));
+                      }
+                      else {
+                        $details = mailbox('get', 'syncjob_details', $syncjob);
+                      }
+                      if ($details) {
+                        $data[] = $details;
+                      }
+                      else {
+                        continue;
+                      }
+                    }
+                  }
+                  process_get_return($data);
+                break;
+              }
+            break;
+            case "active-user-sieve":
+              if (isset($object)) {
+                $sieve_filter = mailbox('get', 'active_user_sieve', $object);
+                if (!empty($sieve_filter)) {
+                  $data[] = $sieve_filter;
+                }
+              }
+              process_get_return($data);
+            break;
+            case "filters":
+              switch ($object) {
+                case "all":
+                  $domains = mailbox('get', 'domains');
+                  if (!empty($domains)) {
+                    foreach ($domains as $domain) {
+                      $mailboxes = mailbox('get', 'mailboxes', $domain);
+                      if (!empty($mailboxes)) {
+                        foreach ($mailboxes as $mailbox) {
+                          $filters = mailbox('get', 'filters', $mailbox);
+                          if (!empty($filters)) {
+                            foreach ($filters as $filter) {
+                              if ($details = mailbox('get', 'filter_details', $filter)) {
+                                $data[] = $details;
+                              }
+                              else {
+                                continue;
+                              }
+                            }
+                          }
+                        }
+                      }
+                    }
+                    process_get_return($data);
+                  }
+                  else {
+                    echo '{}';
+                  }
+                break;
+
+                default:
+                  $filters = mailbox('get', 'filters', $object);
+                  if (!empty($filters)) {
+                    foreach ($filters as $filter) {
+                      if ($details = mailbox('get', 'filter_details', $filter)) {
+                        $data[] = $details;
+                      }
+                      else {
+                        continue;
+                      }
+                    }
+                  }
+                  process_get_return($data);
+                break;
+              }
+            break;
+            case "bcc":
+              switch ($object) {
+                case "all":
+                  $bcc_items = bcc('get');
+                  if (!empty($bcc_items)) {
+                    foreach ($bcc_items as $bcc_item) {
+                      if ($details = bcc('details', $bcc_item)) {
+                        $data[] = $details;
+                      }
+                      else {
+                        continue;
+                      }
+                    }
+                  }
+                  process_get_return($data);
+                break;
+                default:
+                  $data = bcc('details', $object);
+                  if (!empty($data)) {
+                    $data[] = $details;
+                  }
+                  process_get_return($data);
+                break;
+              }
+            break;
+            case "recipient_map":
+              switch ($object) {
+                case "all":
+                  $recipient_map_items = recipient_map('get');
+                  if (!empty($recipient_map_items)) {
+                    foreach ($recipient_map_items as $recipient_map_item) {
+                      if ($details = recipient_map('details', $recipient_map_item)) {
+                        $data[] = $details;
+                      }
+                      else {
+                        continue;
+                      }
+                    }
+                  }
+                  process_get_return($data);
+                break;
+                default:
+                  $data = recipient_map('details', $object);
+                  if (!empty($data)) {
+                    $data[] = $details;
+                  }
+                  process_get_return($data);
+                break;
+              }
+            break;
+            case "tls-policy-map":
+              switch ($object) {
+                case "all":
+                  $tls_policy_maps_items = tls_policy_maps('get');
+                  if (!empty($tls_policy_maps_items)) {
+                    foreach ($tls_policy_maps_items as $tls_policy_maps_item) {
+                      if ($details = tls_policy_maps('details', $tls_policy_maps_item)) {
+                        $data[] = $details;
+                      }
+                      else {
+                        continue;
+                      }
+                    }
+                  }
+                  process_get_return($data);
+                break;
+                default:
+                  $data = tls_policy_maps('details', $object);
+                  if (!empty($data)) {
+                    $data[] = $details;
+                  }
+                  process_get_return($data);
+                break;
+              }
+            break;
+            case "policy_wl_mailbox":
+              switch ($object) {
+                default:
+                  $data = policy('get', 'mailbox', $object)['whitelist'];
+                  process_get_return($data);
+                break;
+              }
+            break;
+            case "policy_bl_mailbox":
+              switch ($object) {
+                default:
+                  $data = policy('get', 'mailbox', $object)['blacklist'];
+                  process_get_return($data);
+                break;
+              }
+            break;
+            case "policy_wl_domain":
+              switch ($object) {
+                default:
+                  $data = policy('get', 'domain', $object)['whitelist'];
+                  process_get_return($data);
+                break;
+              }
+            break;
+            case "policy_bl_domain":
+              switch ($object) {
+                default:
+                  $data = policy('get', 'domain', $object)['blacklist'];
+                  process_get_return($data);
+                break;
+              }
+            break;
+            case "time_limited_aliases":
+              switch ($object) {
+                default:
+                  $data = mailbox('get', 'time_limited_aliases', $object);
+                  process_get_return($data);
+                break;
+              }
+            break;
+            case "fail2ban":
+              switch ($object) {
+                default:
+                  $data = fail2ban('get');
+                  process_get_return($data);
+                break;
+              }
+            break;
+            case "resource":
+              switch ($object) {
+                case "all":
+                  $domains = mailbox('get', 'domains');
+                  if (!empty($domains)) {
+                    foreach ($domains as $domain) {
+                      $resources = mailbox('get', 'resources', $domain);
+                      if (!empty($resources)) {
+                        foreach ($resources as $resource) {
+                          if ($details = mailbox('get', 'resource_details', $resource)) {
+                            $data[] = $details;
+                          }
+                          else {
+                            continue;
+                          }
+                        }
+                      }
+                    }
+                    process_get_return($data);
+                  }
+                  else {
+                    echo '{}';
+                  }
+                break;
+                default:
+                  $data = mailbox('get', 'resource_details', $object);
+                  process_get_return($data);
+                break;
+              }
+            break;
+            case "fwdhost":
+              switch ($object) {
+                case "all":
+                  process_get_return(fwdhost('get'));
+                break;
+                default:
+                  process_get_return(fwdhost('details', $object));
+                break;
+              }
+            break;
+            case "quarantine":
+              // "all" will not print details
+              switch ($object) {
+                case "all":
+                  process_get_return(quarantine('get'));
+                break;
+                default:
+                  process_get_return(quarantine('details', $object));
+                break;
+              }
+            break;
+            case "alias-domain":
+              switch ($object) {
+                case "all":
+                  $alias_domains = mailbox('get', 'alias_domains');
+                  if (!empty($alias_domains)) {
+                    foreach ($alias_domains as $alias_domain) {
+                      if ($details = mailbox('get', 'alias_domain_details', $alias_domain)) {
+                        $data[] = $details;
+                      }
+                      else {
+                        continue;
+                      }
+                    }
+                  }
+                  process_get_return($data);
+                break;
+                default:
+                  process_get_return(mailbox('get', 'alias_domain_details', $object));
+                break;
+              }
+            break;
+            case "alias":
+              switch ($object) {
+                case "all":
+                  if (empty($extra)) {
+                    $domains = array_merge(mailbox('get', 'domains'), mailbox('get', 'alias_domains'));
+                  }
+                  else {
+                    $domains = array($extra);
+                  }
+                  if (!empty($domains)) {
+                    foreach ($domains as $domain) {
+                      $aliases = mailbox('get', 'aliases', $domain);
+                      if (!empty($aliases)) {
+                        foreach ($aliases as $alias) {
+                          if ($details = mailbox('get', 'alias_details', $alias)) {
+                            $data[] = $details;
+                          }
+                          else {
+                            continue;
+                          }
+                        }
+                      }
+                    }
+                    process_get_return($data);
+                  }
+                  else {
+                    echo '{}';
+                  }
+                break;
+
+                default:
+                  process_get_return(mailbox('get', 'alias_details', $object));
+                break;
+              }
+            break;
+            case "domain-admin":
+              switch ($object) {
+                case "all":
+                  $domain_admins = domain_admin('get');
+                  if (!empty($domain_admins)) {
+                    foreach ($domain_admins as $domain_admin) {
+                      if ($details = domain_admin('details', $domain_admin)) {
+                        $data[] = $details;
+                      }
+                      else {
+                        continue;
+                      }
+                    }
+                    process_get_return($data);
+                  }
+                  else {
+                    echo '{}';
+                  }
+                break;
+
+                default:
+                  process_get_return(domain_admin('details', $object));
+                break;
+              }
+            break;
+            case "admin":
+              switch ($object) {
+                case "all":
+                  $admins = admin('get');
+                  if (!empty($admins)) {
+                    foreach ($admins as $admin) {
+                      if ($details = admin('details', $admin)) {
+                        $data[] = $details;
+                      }
+                      else {
+                        continue;
+                      }
+                    }
+                    process_get_return($data);
+                  }
+                  else {
+                    echo '{}';
+                  }
+                break;
+
+                default:
+                  process_get_return(admin('details', $object));
+                break;
+              }
+            break;
+            case "dkim":
+              switch ($object) {
+                default:
+                  $data = dkim('details', $object);
+                  process_get_return($data);
+                  break;
+              }
+            break;
+            case "presets":
+              switch ($object) {
+                case "rspamd":
+                  process_get_return(presets('get', 'rspamd'));
+                break;
+                case "sieve":
+                  process_get_return(presets('get', 'sieve'));
+                break;
+              }
+            break;
+            case "status":
+              switch ($object) {
+                case "containers":
+                  $containers = (docker('info'));
+                  foreach ($containers as $container => $container_info) {
+                    $container . ' (' . $container_info['Config']['Image'] . ')';
+                    $containerstarttime = ($container_info['State']['StartedAt']);
+                    $containerstate = ($container_info['State']['Status']);
+                    $containerimage = ($container_info['Config']['Image']);
+                    $temp[$container] = array(
+                      'type' => 'info',
+                      'container' => $container,
+                      'state' => $containerstate,
+                      'started_at' => $containerstarttime,
+                      'image' => $containerimage
+                    );
+                  }
+                  echo json_encode($temp, JSON_UNESCAPED_SLASHES);
+                break;
+                case "vmail":
+                  $exec_fields_vmail = array('cmd' => 'system', 'task' => 'df', 'dir' => '/var/vmail');
+                  $vmail_df = explode(',', json_decode(docker('post', 'dovecot-mailcow', 'exec', $exec_fields_vmail), true));
+                  $temp = array(
+                    'type' => 'info',
+                    'disk' => $vmail_df[0],
+                    'used' => $vmail_df[2],
+                    'total'=> $vmail_df[1],
+                    'used_percent' => $vmail_df[4]
+                  );
+                  echo json_encode($temp, JSON_UNESCAPED_SLASHES);
+              break;
+              case "solr":
+                $solr_status = solr_status();
+                $solr_size = ($solr_status['status']['dovecot-fts']['index']['size']);
+                $solr_documents = ($solr_status['status']['dovecot-fts']['index']['numDocs']);
+                if (strtolower(getenv('SKIP_SOLR')) != 'n') {
+                  $solr_enabled = false;
+                }
+                else {
+                  $solr_enabled = true;
+                }
+                echo json_encode(array(
+                  'type' => 'info',
+                  'solr_enabled' => $solr_enabled,
+                  'solr_size' => $solr_size,
+                  'solr_documents' => $solr_documents
+                ));
+              break;
               }
-              echo json_encode(array(
-                'type' => 'info',
-                'solr_enabled' => $solr_enabled,
-                'solr_size' => $solr_size,
-                'solr_documents' => $solr_documents
-              ));
             break;
-            }
           break;
-        break;
-        // return no route found if no case is matched
-        default:
-          http_response_code(404);
-          echo json_encode(array(
-            'type' => 'error',
-            'msg' => 'route not found'
-          ));
-          exit();
+          // return no route found if no case is matched
+          default:
+            http_response_code(404);
+            echo json_encode(array(
+              'type' => 'error',
+              'msg' => 'route not found'
+            ));
+            exit();
+          }
         }
       break;
       case "delete":
+        if ($_SESSION['mailcow_cc_api_access'] == 'ro' || isset($_SESSION['pending_mailcow_cc_username'])) {
+          http_response_code(403);
+          echo json_encode(array(
+              'type' => 'error',
+              'msg' => 'API read/write access denied'
+          ));
+          exit();
+        }
         function process_delete_return($return) {
           $generic_failure = json_encode(array(
             'type' => 'error',
@@ -1338,6 +1348,14 @@ if (isset($_SESSION['mailcow_cc_role']) || isset($_SESSION['pending_mailcow_cc_u
         }
       break;
       case "edit":
+        if ($_SESSION['mailcow_cc_api_access'] == 'ro' || isset($_SESSION['pending_mailcow_cc_username'])) {
+          http_response_code(403);
+          echo json_encode(array(
+              'type' => 'error',
+              'msg' => 'API read/write access denied'
+          ));
+          exit();
+        }
         function process_edit_return($return) {
           $generic_failure = json_encode(array(
             'type' => 'error',
@@ -1376,6 +1394,12 @@ if (isset($_SESSION['mailcow_cc_role']) || isset($_SESSION['pending_mailcow_cc_u
           case "bcc":
             process_edit_return(bcc('edit', array_merge(array('id' => $items), $attr)));
           break;
+          case "pushover":
+            process_edit_return(pushover('edit', array_merge(array('username' => $items), $attr)));
+          break;
+          case "pushover-test":
+            process_edit_return(pushover('test', array_merge(array('username' => $items), $attr)));
+          break;
           case "oauth2-client":
             process_edit_return(oauth2('edit', 'client', array_merge(array('id' => $items), $attr)));
           break;
diff --git a/data/web/lang/lang.de.json b/data/web/lang/lang.de.json
index 0f8200ef..38d29e43 100644
--- a/data/web/lang/lang.de.json
+++ b/data/web/lang/lang.de.json
@@ -24,6 +24,9 @@
         "apps": "Apps"
     },
     "danger": {
+        "pushover_token": "Pushover Token hat das falsche Format",
+        "pushover_key": "Pushover Key hat das falsche Format",
+        "pushover_credentials_missing": "Pushover Token und/oder Key fehlen",
         "invalid_filter_type": "Ungültiger Filtertyp",
         "file_open_error": "Datei kann nicht zum Schreiben geöffnet werden",
         "transport_dest_exists": "Transport Maps Ziel \"%s\" existiert bereits",
@@ -135,6 +138,7 @@
         "extra_acl_invalid_domain": "Externe Absenderadresse \"%s\" verwendet eine ungültige Domain"
     },
     "success": {
+        "pushover_settings_edited": "Pushover Konfiguration gespeichert, bitte den Zugang im Anschluss verifizieren.",
         "global_filter_written": "Filterdatei wurde erfolreich geschrieben",
         "learned_ham": "ID %s wurde erfolreich als Ham gelernt",
         "verified_totp_login": "TOTP Anmeldung verifiziert",
@@ -224,8 +228,17 @@
         "quota_exceeded_scope": "Domain-Quota erschöpft: Es können nur noch unlimiterte Mailboxen in dieser Domain erstellt werden."
     },
     "user": {
+        "pushover_info": "Push-Benachrichtungen werden angewendet auf alle nicht-Spam Nachrichten zugestellt an <b>%s</b>, einschließlich Alias-Adressen (shared, non-shared, tagged).",
+        "verify": "Verifizieren",
+        "pushover_verify": "Verbindung verifizieren",
+        "title": "Title",
+        "pushover_title": "Notification Titel",
+        "text": "Text",
+        "pushover_text": "Notification Text ({SUBJECT} entspricht dem Mail-Betreff)",
+        "last_mail_login": "Letzter Mail-Login",
         "last_mail_login": "Letzter Mail-Login",
         "no_last_login": "Keine letzte UI Anmeldung gespeichert",
+        "save": "Änderungen speichern",
         "generate": "generieren",
         "apple_connection_profile": "Apple Verbindungsprofil",
         "apple_connection_profile_mailonly": "Dieses Verbindungsprofil beinhaltet IMAP und SMTP Konfigurationen für ein Apple Gerät.",
@@ -334,6 +347,13 @@
         "spam_score_reset": "Auf Server-Standard zurücksetzen"
     },
     "admin": {
+        "pushover_info": "Push-Benachrichtungen werden angewendet auf alle nicht-Spam Nachrichten zugestellt an <b>%s</b>, einschließlich Alias-Adressen (shared, non-shared, tagged).",
+        "verify": "Verifizieren",
+        "pushover_verify": "Verbindung verifizieren",
+        "title": "Title",
+        "pushover_title": "Notification Titel",
+        "text": "Text",
+        "pushover_text": "Notification Text ({SUBJECT} entspricht dem Mail-Betreff)",
         "spamfilter": "Spamfilter",
         "domain_s": "Domain(s)",
         "rspamd-com_settings": "Ein Name wird automatisch generiert. Beispielinhalte zur Einsicht stehen nachstehend bereit. Siehe auch <a href=\"https://rspamd.com/doc/configuration/settings.html#settings-structure\" target=\"_blank\">Rspamd docs</a>",
@@ -858,6 +878,7 @@
         "relayhost_wrapped_tls_info": "Bitte <b>keine</b> TLS-wrapped Ports verwenden (etwa SMTPS via Port 465/tcp).<br>\r\nDer Transport wird stattdessen STARTTLS anfordern, um TLS zu verwenden. TLS kann unter \"TLS Policy Maps\" erzwungen werden."
     },
     "acl": {
+        "pushover": "Pushover",
         "spam_alias": "Temporäre E-Mail Aliasse",
         "tls_policy": "Verschlüsselungsrichtlinie",
         "spam_score": "Spam-Bewertung",
diff --git a/data/web/lang/lang.en.json b/data/web/lang/lang.en.json
index 49cfa4a5..cc8d8023 100644
--- a/data/web/lang/lang.en.json
+++ b/data/web/lang/lang.en.json
@@ -30,6 +30,9 @@
         "unlimited_quota_acl": "Unlimited quota prohibited by ACL",
         "mysql_error": "MySQL error: %s",
         "redis_error": "Redis error: %s",
+        "pushover_token": "Pushover token has a wrong format",
+        "pushover_key": "Pushover key has a wrong format",
+        "pushover_credentials_missing": "Pushover token and or key missing",
         "unknown_tfa_method": "Unknown TFA method",
         "totp_verification_failed": "TOTP verification failed",
         "u2f_verification_failed": "U2F verification failed: %s",
@@ -135,6 +138,7 @@
         "extra_acl_invalid_domain": "External sender \"%s\" uses an invalid domain"
     },
     "success": {
+        "pushover_settings_edited": "Pushover settings successfully set, please verify credentials.",
         "global_filter_written": "Filter was successfully written to file",
         "learned_ham": "Successfully learned ID % as ham",
         "verified_totp_login": "Verified TOTP login",
@@ -221,11 +225,19 @@
         "hash_not_found": "Hash not found or already deleted",
         "fuzzy_learn_error": "Fuzzy hash learn error: %s",
         "ip_invalid": "Skipped invalid IP: %s",
-        "quota_exceeded_scope": "Domain quota exceeded: Only unlimited/unrated mailboxes can be created in this domain scope."
+        "quota_exceeded_scope": "Domain quota exceeded: Only unlimited mailboxes can be created in this domain scope."
     },
     "user": {
+        "pushover_info": "Push notification settings will apply to all clean (non-spam) mail delivered to <b>%s</b> including aliases (shared, non-shared, tagged).",
+        "verify": "Verify",
+        "pushover_verify": "Verify credentials",
+        "title": "Title",
+        "pushover_title": "Notification title",
+        "text": "Text",
+        "pushover_text": "Notification text ({SUBJECT} will be replaced by mail subject)",
         "no_last_login": "No last UI login information",
         "last_mail_login": "Last mail login",
+        "save": "Save changes",
         "apple_connection_profile": "Apple connection profile",
         "apple_connection_profile_mailonly": "This connection profile includes IMAP and SMTP configuration parameters for an Apple device.",
         "apple_connection_profile_complete": "This connection profile includes IMAP and SMTP parameters as well as CalDAV (calendars) and CardDAV (contacts) pathes for an Apple device.",
@@ -334,6 +346,13 @@
         "spam_score_reset": "Reset to server default"
     },
     "admin": {
+        "pushover_info": "Push notification settings will apply to all clean (non-spam) mail delivered to <b>%s</b> including aliases (shared, non-shared, tagged).",
+        "verify": "Verify",
+        "pushover_verify": "Verify credentials",
+        "title": "Title",
+        "pushover_title": "Notification title",
+        "text": "Text",
+        "pushover_text": "Notification text ({SUBJECT} will be replaced by mail subject)",
         "spamfilter": "Spam filter",
         "domain": "Domain",
         "domain_s": "Domain/s",
@@ -857,6 +876,7 @@
         "relayhost_wrapped_tls_info": "Please do <b>not</b> use TLS-wrapped ports (mostly used on port 465).<br>\r\nUse any non-wrapped port and issue STARTTLS. A TLS policy to enforce TLS can be created in \"TLS policy maps\"."
     },
     "acl": {
+        "pushover": "Pushover",
         "spam_alias": "Temporary aliases",
         "tls_policy": "TLS policy",
         "spam_score": "Spam score",
diff --git a/data/web/user.php b/data/web/user.php
index a312d5f4..f691b8bf 100644
--- a/data/web/user.php
+++ b/data/web/user.php
@@ -76,6 +76,7 @@ elseif (isset($_SESSION['mailcow_cc_role']) && $_SESSION['mailcow_cc_role'] == '
   $_SESSION['return_to'] = $_SERVER['REQUEST_URI'];
   $username = $_SESSION['mailcow_cc_username'];
   $mailboxdata = mailbox('get', 'mailbox_details', $username);
+  $pushover_data = pushover('get', $username);
 
   $clientconfigstr = "host=" . urlencode($mailcow_hostname) . "&email=" . urlencode($username) . "&name=" . urlencode($mailboxdata['name']) . "&ui=" . urlencode(strtok($_SERVER['HTTP_HOST'], ':')) . "&port=" . urlencode($autodiscover_config['caldav']['port']);
   if ($autodiscover_config['useEASforOutlook'] == 'yes')
@@ -101,6 +102,7 @@ elseif (isset($_SESSION['mailcow_cc_role']) && $_SESSION['mailcow_cc_role'] == '
     <li role="presentation"><a href="#Spamfilter" aria-controls="Spamfilter" role="tab" data-toggle="tab"><?=$lang['user']['spamfilter'];?></a></li>
     <li role="presentation"><a href="#Syncjobs" aria-controls="Syncjobs" role="tab" data-toggle="tab"><?=$lang['user']['sync_jobs'];?></a></li>
     <li role="presentation"><a href="#AppPasswds" aria-controls="AppPasswds" role="tab" data-toggle="tab"><?=$lang['user']['app_passwds'];?></a></li>
+    <li role="presentation"><a href="#Pushover" aria-controls="Pushover" role="tab" data-toggle="tab">Pushover API</a></li>
   </ul>
   <hr>
 
@@ -472,6 +474,59 @@ elseif (isset($_SESSION['mailcow_cc_role']) && $_SESSION['mailcow_cc_role'] == '
     </div>
   </div>
 
+	<div role="tabpanel" class="tab-pane" id="Pushover">
+    <form data-id="pushover" class="form well" method="post">
+      <input type="hidden" value="0" name="active">
+      <div class="row">
+        <div class="col-sm-1">
+          <p class="help-block"><a href="https://pushover.net" target="_blank"><img src="" class="img img-fluid"></a></p>
+        </div>
+        <div class="col-sm-10">
+          <div class="form-group">
+            <div class="row">
+              <div class="col-sm-6">
+                <div class="form-group">
+                  <label for="token">API Token/Key (Application)</label>
+                  <input type="text" class="form-control" name="token" maxlength="30" value="<?=$pushover_data['token'];?>" required>
+                </div>
+              </div>
+              <div class="col-sm-6">
+                <div class="form-group">
+                  <label for="key">User/Group Key</label>
+                  <input type="text" class="form-control" name="key" maxlength="30" value="<?=$pushover_data['key'];?>" required>
+                </div>
+              </div>
+              <div class="col-sm-6">
+                <div class="form-group">
+                  <label for="title"><?=$lang['user']['pushover_title'];?></label>
+                  <input type="text" class="form-control" name="title" value="<?=$pushover_data['title'];?>" placeholder="Mail">
+                </div>
+              </div>
+              <div class="col-sm-6">
+                <div class="form-group">
+                  <label for="text"><?=$lang['user']['pushover_text'];?></label>
+                  <input type="text" class="form-control" name="text" value="<?=$pushover_data['text'];?>" placeholder="You've got mail 📧">
+                </div>
+              </div>
+              <div class="col-sm-12">
+                <div class="checkbox">
+                <label><input type="checkbox" value="1" name="active" <?=($pushover_data['active']=="1") ? "checked" : null;?>> <?=$lang['edit']['active'];?></label>
+                </div>
+              </div>
+            </div>
+          </div>
+          <hr>
+          <p><?=sprintf($lang['user']['pushover_info'], $username);?></p>
+          <div class="btn-group" data-acl="<?=$_SESSION['acl']['pushover'];?>">
+              <a class="btn btn-sm btn-default" data-action="edit_selected" data-id="pushover" data-item="<?=htmlspecialchars($username);?>" data-api-url='edit/pushover' data-api-attr='{}' href="#"><?=$lang['user']['save'];?></a>
+              <a class="btn btn-sm btn-default" data-action="edit_selected" data-id="pushover-test" data-item="<?=htmlspecialchars($username);?>" data-api-url='edit/pushover-test' data-api-attr='{}' href="#"><span class="glyphicon glyphicon-check" aria-hidden="true"></span> <?=$lang['user']['pushover_verify'];?></a>
+              <a class="btn btn-sm btn-danger" data-action="edit_selected" data-id="pushover-delete" data-item="<?=htmlspecialchars($username);?>" data-api-url='edit/pushover' data-api-attr='{"delete":"true"}' href="#"><span class="glyphicon glyphicon-trash" aria-hidden="true"></span> <?=$lang['user']['remove'];?></a>
+          </div>
+        </div>
+      </div>
+    </form>
+  </div>
+
 	<div role="tabpanel" class="tab-pane" id="AppPasswds">
     <p><?=$lang['user']['app_hint'];?></p>
 		<div class="table-responsive">