Merge pull request #7089 from mailcow/fix/7039
[Web] switch from GET to POST for datatable requests
This commit is contained in:
commit
ddc2309f1e
@ -140,17 +140,32 @@ function session_check() {
|
|||||||
);
|
);
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
if (!empty($_POST)) {
|
// Check if this is a POST request (form-encoded or JSON)
|
||||||
if ($_SESSION['CSRF']['TOKEN'] != $_POST['csrf_token']) {
|
$is_post_request = !empty($_POST) || (
|
||||||
$_SESSION['return'][] = array(
|
isset($_SERVER['CONTENT_TYPE']) &&
|
||||||
'type' => 'warning',
|
strpos($_SERVER['CONTENT_TYPE'], 'application/json') !== false
|
||||||
'msg' => 'session_token'
|
);
|
||||||
);
|
|
||||||
return false;
|
if ($is_post_request) {
|
||||||
|
// Skip CSRF check for DataTables server-side processing endpoints
|
||||||
|
// These are read-only operations (equivalent to GET) authenticated by session
|
||||||
|
$is_search_endpoint = (
|
||||||
|
isset($_GET['query']) &&
|
||||||
|
preg_match('#^search/(domain|mailbox)$#', $_GET['query'])
|
||||||
|
);
|
||||||
|
|
||||||
|
if (!$is_search_endpoint && !empty($_POST)) {
|
||||||
|
if ($_SESSION['CSRF']['TOKEN'] != $_POST['csrf_token']) {
|
||||||
|
$_SESSION['return'][] = array(
|
||||||
|
'type' => 'warning',
|
||||||
|
'msg' => 'session_token'
|
||||||
|
);
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
unset($_POST['csrf_token']);
|
||||||
|
$_SESSION['CSRF']['TOKEN'] = bin2hex(random_bytes(32));
|
||||||
|
$_SESSION['CSRF']['TIME'] = time();
|
||||||
}
|
}
|
||||||
unset($_POST['csrf_token']);
|
|
||||||
$_SESSION['CSRF']['TOKEN'] = bin2hex(random_bytes(32));
|
|
||||||
$_SESSION['CSRF']['TIME'] = time();
|
|
||||||
}
|
}
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|||||||
@ -471,8 +471,13 @@ jQuery(function($){
|
|||||||
hideTableExpandCollapseBtn('#tab-domains', '#domain_table');
|
hideTableExpandCollapseBtn('#tab-domains', '#domain_table');
|
||||||
},
|
},
|
||||||
ajax: {
|
ajax: {
|
||||||
type: "GET",
|
type: "POST",
|
||||||
url: "/api/v1/get/domain/datatables",
|
url: "/api/v1/search/domain",
|
||||||
|
contentType: "application/json",
|
||||||
|
processData: false,
|
||||||
|
data: function(d) {
|
||||||
|
return JSON.stringify(d);
|
||||||
|
},
|
||||||
dataSrc: function(json){
|
dataSrc: function(json){
|
||||||
$.each(json.data, function(i, item) {
|
$.each(json.data, function(i, item) {
|
||||||
item.domain_name = escapeHtml(item.domain_name);
|
item.domain_name = escapeHtml(item.domain_name);
|
||||||
@ -898,8 +903,13 @@ jQuery(function($){
|
|||||||
hideTableExpandCollapseBtn('#tab-mailboxes', '#mailbox_table');
|
hideTableExpandCollapseBtn('#tab-mailboxes', '#mailbox_table');
|
||||||
},
|
},
|
||||||
ajax: {
|
ajax: {
|
||||||
type: "GET",
|
type: "POST",
|
||||||
url: "/api/v1/get/mailbox/datatables",
|
url: "/api/v1/search/mailbox",
|
||||||
|
contentType: "application/json",
|
||||||
|
processData: false,
|
||||||
|
data: function(d) {
|
||||||
|
return JSON.stringify(d);
|
||||||
|
},
|
||||||
dataSrc: function(json){
|
dataSrc: function(json){
|
||||||
$.each(json.data, function (i, item) {
|
$.each(json.data, function (i, item) {
|
||||||
item.quota = {
|
item.quota = {
|
||||||
|
|||||||
@ -91,6 +91,11 @@ if (isset($_GET['query'])) {
|
|||||||
if ($action == 'delete') {
|
if ($action == 'delete') {
|
||||||
$_POST['items'] = $request;
|
$_POST['items'] = $request;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// search
|
||||||
|
if ($action == 'search') {
|
||||||
|
// placeholder for search, as the request body is already decoded and available in $requestDecoded
|
||||||
|
}
|
||||||
}
|
}
|
||||||
api_log($_POST);
|
api_log($_POST);
|
||||||
|
|
||||||
@ -459,47 +464,6 @@ if (isset($_GET['query'])) {
|
|||||||
|
|
||||||
case "domain":
|
case "domain":
|
||||||
switch ($object) {
|
switch ($object) {
|
||||||
case "datatables":
|
|
||||||
$table = ['domain', 'd'];
|
|
||||||
$primaryKey = 'domain';
|
|
||||||
$columns = [
|
|
||||||
['db' => 'domain', 'dt' => 2],
|
|
||||||
['db' => 'aliases', 'dt' => 3, 'order_subquery' => "SELECT COUNT(*) FROM `alias` WHERE (`domain`= `d`.`domain` OR `domain` IN (SELECT `alias_domain` FROM `alias_domain` WHERE `target_domain` = `d`.`domain`)) AND `address` NOT IN (SELECT `username` FROM `mailbox`)"],
|
|
||||||
['db' => 'mailboxes', 'dt' => 4, 'order_subquery' => "SELECT COUNT(*) FROM `mailbox` WHERE `mailbox`.`domain` = `d`.`domain` AND (`mailbox`.`kind` = '' OR `mailbox`.`kind` = NULL)"],
|
|
||||||
['db' => 'quota', 'dt' => 5, 'order_subquery' => "SELECT COALESCE(SUM(`mailbox`.`quota`), 0) FROM `mailbox` WHERE `mailbox`.`domain` = `d`.`domain` AND (`mailbox`.`kind` = '' OR `mailbox`.`kind` = NULL)"],
|
|
||||||
['db' => 'stats', 'dt' => 6, 'dummy' => true, 'order_subquery' => "SELECT SUM(bytes) FROM `quota2` WHERE `quota2`.`username` IN (SELECT `username` FROM `mailbox` WHERE `domain` = `d`.`domain`)"],
|
|
||||||
['db' => 'defquota', 'dt' => 7],
|
|
||||||
['db' => 'maxquota', 'dt' => 8],
|
|
||||||
['db' => 'backupmx', 'dt' => 10],
|
|
||||||
['db' => 'tags', 'dt' => 14, 'dummy' => true, 'search' => ['join' => 'LEFT JOIN `tags_domain` AS `td` ON `td`.`domain` = `d`.`domain`', 'where_column' => '`td`.`tag_name`']],
|
|
||||||
['db' => 'active', 'dt' => 15],
|
|
||||||
];
|
|
||||||
|
|
||||||
require_once $_SERVER['DOCUMENT_ROOT'] . '/inc/lib/ssp.class.php';
|
|
||||||
global $pdo;
|
|
||||||
if($_SESSION['mailcow_cc_role'] === 'admin') {
|
|
||||||
$data = SSP::simple($_GET, $pdo, $table, $primaryKey, $columns);
|
|
||||||
} elseif ($_SESSION['mailcow_cc_role'] === 'domainadmin') {
|
|
||||||
$data = SSP::complex($_GET, $pdo, $table, $primaryKey, $columns,
|
|
||||||
'INNER JOIN domain_admins as da ON da.domain = d.domain',
|
|
||||||
[
|
|
||||||
'condition' => 'da.active = 1 and da.username = :username',
|
|
||||||
'bindings' => ['username' => $_SESSION['mailcow_cc_username']]
|
|
||||||
]);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!empty($data['data'])) {
|
|
||||||
$domainsData = [];
|
|
||||||
foreach ($data['data'] as $domain) {
|
|
||||||
if ($details = mailbox('get', 'domain_details', $domain[2])) {
|
|
||||||
$domainsData[] = $details;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
$data['data'] = $domainsData;
|
|
||||||
}
|
|
||||||
|
|
||||||
process_get_return($data);
|
|
||||||
break;
|
|
||||||
case "all":
|
case "all":
|
||||||
$tags = null;
|
$tags = null;
|
||||||
if (isset($_GET['tags']) && $_GET['tags'] != '')
|
if (isset($_GET['tags']) && $_GET['tags'] != '')
|
||||||
@ -999,46 +963,6 @@ if (isset($_GET['query'])) {
|
|||||||
break;
|
break;
|
||||||
case "mailbox":
|
case "mailbox":
|
||||||
switch ($object) {
|
switch ($object) {
|
||||||
case "datatables":
|
|
||||||
$table = ['mailbox', 'm'];
|
|
||||||
$primaryKey = 'username';
|
|
||||||
$columns = [
|
|
||||||
['db' => 'username', 'dt' => 2],
|
|
||||||
['db' => 'quota', 'dt' => 3],
|
|
||||||
['db' => 'last_mail_login', 'dt' => 4, 'dummy' => true, 'order_subquery' => "SELECT MAX(`datetime`) FROM `sasl_log` WHERE `service` != 'SSO' AND `username` = `m`.`username`"],
|
|
||||||
['db' => 'last_pw_change', 'dt' => 5, 'dummy' => true, 'order_subquery' => "JSON_EXTRACT(attributes, '$.passwd_update')"],
|
|
||||||
['db' => 'in_use', 'dt' => 6, 'dummy' => true, 'order_subquery' => "(SELECT SUM(bytes) FROM `quota2` WHERE `quota2`.`username` = `m`.`username`) / `m`.`quota`"],
|
|
||||||
['db' => 'name', 'dt' => 7],
|
|
||||||
['db' => 'messages', 'dt' => 20, 'dummy' => true, 'order_subquery' => "SELECT SUM(messages) FROM `quota2` WHERE `quota2`.`username` = `m`.`username`"],
|
|
||||||
['db' => 'tags', 'dt' => 23, 'dummy' => true, 'search' => ['join' => 'LEFT JOIN `tags_mailbox` AS `tm` ON `tm`.`username` = `m`.`username`', 'where_column' => '`tm`.`tag_name`']],
|
|
||||||
['db' => 'active', 'dt' => 24],
|
|
||||||
];
|
|
||||||
|
|
||||||
require_once $_SERVER['DOCUMENT_ROOT'] . '/inc/lib/ssp.class.php';
|
|
||||||
global $pdo;
|
|
||||||
if($_SESSION['mailcow_cc_role'] === 'admin') {
|
|
||||||
$data = SSP::complex($_GET, $pdo, $table, $primaryKey, $columns, null, "(`m`.`kind` = '' OR `m`.`kind` = NULL)");
|
|
||||||
} elseif ($_SESSION['mailcow_cc_role'] === 'domainadmin') {
|
|
||||||
$data = SSP::complex($_GET, $pdo, $table, $primaryKey, $columns,
|
|
||||||
'INNER JOIN domain_admins as da ON da.domain = m.domain',
|
|
||||||
[
|
|
||||||
'condition' => "(`m`.`kind` = '' OR `m`.`kind` = NULL) AND `da`.`active` = 1 AND `da`.`username` = :username",
|
|
||||||
'bindings' => ['username' => $_SESSION['mailcow_cc_username']]
|
|
||||||
]);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!empty($data['data'])) {
|
|
||||||
$mailboxData = [];
|
|
||||||
foreach ($data['data'] as $mailbox) {
|
|
||||||
if ($details = mailbox('get', 'mailbox_details', $mailbox[2])) {
|
|
||||||
$mailboxData[] = $details;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
$data['data'] = $mailboxData;
|
|
||||||
}
|
|
||||||
|
|
||||||
process_get_return($data);
|
|
||||||
break;
|
|
||||||
case "all":
|
case "all":
|
||||||
case "reduced":
|
case "reduced":
|
||||||
$tags = null;
|
$tags = null;
|
||||||
@ -1627,6 +1551,136 @@ if (isset($_GET['query'])) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
|
case "search":
|
||||||
|
function process_search_return($return) {
|
||||||
|
if ($return === false) {
|
||||||
|
echo json_encode(array(
|
||||||
|
'type' => 'error',
|
||||||
|
'msg' => 'Cannot get item'
|
||||||
|
));
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
echo json_encode($return, JSON_UNESCAPED_UNICODE | JSON_PRETTY_PRINT);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// only allow POST requests to SEARCH API endpoints
|
||||||
|
if ($_SERVER['REQUEST_METHOD'] != 'POST') {
|
||||||
|
http_response_code(405);
|
||||||
|
echo json_encode(array(
|
||||||
|
'type' => 'error',
|
||||||
|
'msg' => 'only POST method is allowed'
|
||||||
|
));
|
||||||
|
exit();
|
||||||
|
}
|
||||||
|
|
||||||
|
// Load SSP class
|
||||||
|
require_once $_SERVER['DOCUMENT_ROOT'] . '/inc/lib/ssp.class.php';
|
||||||
|
global $pdo;
|
||||||
|
|
||||||
|
switch ($category) {
|
||||||
|
case "domain":
|
||||||
|
$table = ['domain', 'd'];
|
||||||
|
$primaryKey = 'domain';
|
||||||
|
$columns = [
|
||||||
|
['db' => 'domain', 'dt' => 2],
|
||||||
|
['db' => 'aliases', 'dt' => 3, 'order_subquery' => "SELECT COUNT(*) FROM `alias` WHERE (`domain`= `d`.`domain` OR `domain` IN (SELECT `alias_domain` FROM `alias_domain` WHERE `target_domain` = `d`.`domain`)) AND `address` NOT IN (SELECT `username` FROM `mailbox`)"],
|
||||||
|
['db' => 'mailboxes', 'dt' => 4, 'order_subquery' => "SELECT COUNT(*) FROM `mailbox` WHERE `mailbox`.`domain` = `d`.`domain` AND (`mailbox`.`kind` = '' OR `mailbox`.`kind` = NULL)"],
|
||||||
|
['db' => 'quota', 'dt' => 5, 'order_subquery' => "SELECT COALESCE(SUM(`mailbox`.`quota`), 0) FROM `mailbox` WHERE `mailbox`.`domain` = `d`.`domain` AND (`mailbox`.`kind` = '' OR `mailbox`.`kind` = NULL)"],
|
||||||
|
['db' => 'stats', 'dt' => 6, 'dummy' => true, 'order_subquery' => "SELECT SUM(bytes) FROM `quota2` WHERE `quota2`.`username` IN (SELECT `username` FROM `mailbox` WHERE `domain` = `d`.`domain`)"],
|
||||||
|
['db' => 'defquota', 'dt' => 7],
|
||||||
|
['db' => 'maxquota', 'dt' => 8],
|
||||||
|
['db' => 'backupmx', 'dt' => 10],
|
||||||
|
['db' => 'tags', 'dt' => 14, 'dummy' => true, 'search' => ['join' => 'LEFT JOIN `tags_domain` AS `td` ON `td`.`domain` = `d`.`domain`', 'where_column' => '`td`.`tag_name`']],
|
||||||
|
['db' => 'active', 'dt' => 15],
|
||||||
|
];
|
||||||
|
|
||||||
|
if($_SESSION['mailcow_cc_role'] === 'admin') {
|
||||||
|
$data = SSP::simple($requestDecoded, $pdo, $table, $primaryKey, $columns);
|
||||||
|
} elseif ($_SESSION['mailcow_cc_role'] === 'domainadmin') {
|
||||||
|
$data = SSP::complex($requestDecoded, $pdo, $table, $primaryKey, $columns,
|
||||||
|
'INNER JOIN domain_admins as da ON da.domain = d.domain',
|
||||||
|
[
|
||||||
|
'condition' => 'da.active = 1 and da.username = :username',
|
||||||
|
'bindings' => ['username' => $_SESSION['mailcow_cc_username']]
|
||||||
|
]);
|
||||||
|
} else {
|
||||||
|
http_response_code(403);
|
||||||
|
echo json_encode(array(
|
||||||
|
'type' => 'error',
|
||||||
|
'msg' => 'Insufficient permissions'
|
||||||
|
));
|
||||||
|
exit();
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!empty($data['data'])) {
|
||||||
|
$domainsData = [];
|
||||||
|
foreach ($data['data'] as $domain) {
|
||||||
|
if ($details = mailbox('get', 'domain_details', $domain[2])) {
|
||||||
|
$domainsData[] = $details;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
$data['data'] = $domainsData;
|
||||||
|
}
|
||||||
|
|
||||||
|
process_search_return($data);
|
||||||
|
break;
|
||||||
|
|
||||||
|
case "mailbox":
|
||||||
|
$table = ['mailbox', 'm'];
|
||||||
|
$primaryKey = 'username';
|
||||||
|
$columns = [
|
||||||
|
['db' => 'username', 'dt' => 2],
|
||||||
|
['db' => 'quota', 'dt' => 3],
|
||||||
|
['db' => 'last_mail_login', 'dt' => 4, 'dummy' => true, 'order_subquery' => "SELECT MAX(`datetime`) FROM `sasl_log` WHERE `service` != 'SSO' AND `username` = `m`.`username`"],
|
||||||
|
['db' => 'last_pw_change', 'dt' => 5, 'dummy' => true, 'order_subquery' => "JSON_EXTRACT(attributes, '$.passwd_update')"],
|
||||||
|
['db' => 'in_use', 'dt' => 6, 'dummy' => true, 'order_subquery' => "(SELECT SUM(bytes) FROM `quota2` WHERE `quota2`.`username` = `m`.`username`) / `m`.`quota`"],
|
||||||
|
['db' => 'name', 'dt' => 7],
|
||||||
|
['db' => 'messages', 'dt' => 20, 'dummy' => true, 'order_subquery' => "SELECT SUM(messages) FROM `quota2` WHERE `quota2`.`username` = `m`.`username`"],
|
||||||
|
['db' => 'tags', 'dt' => 23, 'dummy' => true, 'search' => ['join' => 'LEFT JOIN `tags_mailbox` AS `tm` ON `tm`.`username` = `m`.`username`', 'where_column' => '`tm`.`tag_name`']],
|
||||||
|
['db' => 'active', 'dt' => 24],
|
||||||
|
];
|
||||||
|
|
||||||
|
if($_SESSION['mailcow_cc_role'] === 'admin') {
|
||||||
|
$data = SSP::complex($requestDecoded, $pdo, $table, $primaryKey, $columns, null,
|
||||||
|
"(`m`.`kind` = '' OR `m`.`kind` = NULL)");
|
||||||
|
} elseif ($_SESSION['mailcow_cc_role'] === 'domainadmin') {
|
||||||
|
$data = SSP::complex($requestDecoded, $pdo, $table, $primaryKey, $columns,
|
||||||
|
'INNER JOIN domain_admins as da ON da.domain = m.domain',
|
||||||
|
[
|
||||||
|
'condition' => "(`m`.`kind` = '' OR `m`.`kind` = NULL) AND `da`.`active` = 1 AND `da`.`username` = :username",
|
||||||
|
'bindings' => ['username' => $_SESSION['mailcow_cc_username']]
|
||||||
|
]);
|
||||||
|
} else {
|
||||||
|
http_response_code(403);
|
||||||
|
echo json_encode(array(
|
||||||
|
'type' => 'error',
|
||||||
|
'msg' => 'Insufficient permissions'
|
||||||
|
));
|
||||||
|
exit();
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!empty($data['data'])) {
|
||||||
|
$mailboxData = [];
|
||||||
|
foreach ($data['data'] as $mailbox) {
|
||||||
|
if ($details = mailbox('get', 'mailbox_details', $mailbox[2])) {
|
||||||
|
$mailboxData[] = $details;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
$data['data'] = $mailboxData;
|
||||||
|
}
|
||||||
|
|
||||||
|
process_search_return($data);
|
||||||
|
break;
|
||||||
|
|
||||||
|
default:
|
||||||
|
http_response_code(404);
|
||||||
|
echo json_encode(array(
|
||||||
|
'type' => 'error',
|
||||||
|
'msg' => 'Invalid search category'
|
||||||
|
));
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
break;
|
||||||
case "delete":
|
case "delete":
|
||||||
if ($_SESSION['mailcow_cc_api_access'] == 'ro' || isset($_SESSION['pending_mailcow_cc_username']) || !isset($_SESSION["mailcow_cc_username"])) {
|
if ($_SESSION['mailcow_cc_api_access'] == 'ro' || isset($_SESSION['pending_mailcow_cc_username']) || !isset($_SESSION["mailcow_cc_username"])) {
|
||||||
http_response_code(403);
|
http_response_code(403);
|
||||||
|
|||||||
Loading…
Reference in New Issue
Block a user