[Web] Fix symbol options encoding in rspamd item view

[Web] Feature: Allow to view mailq item content via postcat
[Web] Fix Rspamd not drawing
master
andryyy 2020-05-19 20:20:05 +02:00
parent 3ccc9c07c1
commit 21a153c2a7
9 changed files with 153 additions and 94 deletions

View File

@ -1,16 +0,0 @@
<?php
session_start();
require_once $_SERVER['DOCUMENT_ROOT'] . '/inc/prerequisites.inc.php';
header('Content-Type: text/plain');
if (!isset($_SESSION['mailcow_cc_role']) || $_SESSION['mailcow_cc_role'] != 'admin') {
exit();
}
$docker_return = docker('post', 'postfix-mailcow', 'exec', array('cmd' => 'mailq'));
if (isset($docker_return['type']['danger'])) {
echo "Cannot load mail queue: " . $docker_return['msg'];
}
else {
echo $docker_return;
}
?>

View File

@ -10,64 +10,98 @@ function mailq($_action, $_data = null) {
} }
function process_mailq_output($returned_output, $_action, $_data) { function process_mailq_output($returned_output, $_action, $_data) {
if ($returned_output !== NULL) { if ($returned_output !== NULL) {
if (isset($returned_output['type']) && $returned_output['type'] == 'danger') { if ($_action == 'cat') {
$_SESSION['return'][] = array( logger(array('return' => array(
'type' => 'danger', array(
'log' => array('mailq', $_action, $_data), 'type' => 'success',
'msg' => 'Error: ' . $returned_output['msg'] 'log' => array(__FUNCTION__, $_action, $_data),
); 'msg' => 'queue_cat_success'
)
)));
return $returned_output;
} }
if (isset($returned_output['type']) && $returned_output['type'] == 'success') { else {
$_SESSION['return'][] = array( if (isset($returned_output['type']) && $returned_output['type'] == 'danger') {
'type' => 'success', $_SESSION['return'][] = array(
'log' => array('mailq', $_action, $_data), 'type' => 'danger',
'msg' => 'queue_command_success' 'log' => array(__FUNCTION__, $_action, $_data),
); 'msg' => 'Error: ' . $returned_output['msg']
);
}
if (isset($returned_output['type']) && $returned_output['type'] == 'success') {
$_SESSION['return'][] = array(
'type' => 'success',
'log' => array(__FUNCTION__, $_action, $_data),
'msg' => 'queue_command_success'
);
}
} }
} }
else { else {
$_SESSION['return'][] = array( $_SESSION['return'][] = array(
'type' => 'danger', 'type' => 'danger',
'log' => array('mailq', $_action, $_data), 'log' => array(__FUNCTION__, $_action, $_data),
'msg' => 'unknown' 'msg' => 'unknown'
); );
} }
} }
global $lang; global $lang;
switch ($_action) { if ($_action == 'get') {
case 'get': $mailq_lines = docker('post', 'postfix-mailcow', 'exec', array('cmd' => 'mailq', 'task' => 'list'));
$mailq_lines = docker('post', 'postfix-mailcow', 'exec', array('cmd' => 'mailq', 'task' => 'list')); $lines = 0;
$lines = 0; // Hard limit to 10000 items
// Hard limit to 10000 items foreach (preg_split("/((\r?\n)|(\r\n?))/", $mailq_lines) as $mailq_item) if ($lines++ < 10000) {
foreach (preg_split("/((\r?\n)|(\r\n?))/", $mailq_lines) as $mailq_item) if ($lines++ < 10000) { if (empty($mailq_item) || $mailq_item == '1') {
if (empty($mailq_item) || $mailq_item == '1') { continue;
continue; }
} $mq_line = json_decode($mailq_item, true);
$mq_line = json_decode($mailq_item, true); if ($mq_line !== NULL) {
if ($mq_line !== NULL) { $rcpts = array();
$rcpts = array(); foreach ($mq_line['recipients'] as $rcpt) {
foreach ($mq_line['recipients'] as $rcpt) { if (isset($rcpt['delay_reason'])) {
if (isset($rcpt['delay_reason'])) { $rcpts[] = $rcpt['address'] . ' (' . $rcpt['delay_reason'] . ')';
$rcpts[] = $rcpt['address'] . ' (' . $rcpt['delay_reason'] . ')';
}
else {
$rcpts[] = $rcpt['address'];
}
} }
if (!empty($rcpts)) { else {
$mq_line['recipients'] = $rcpts; $rcpts[] = $rcpt['address'];
} }
$line[] = $mq_line;
} }
if (!empty($rcpts)) {
$mq_line['recipients'] = $rcpts;
}
$line[] = $mq_line;
} }
if (!isset($line) || empty($line)) { }
return '{}'; if (!isset($line) || empty($line)) {
} return '{}';
else { }
return json_encode($line); else {
} return json_encode($line);
break; }
case 'delete': }
elseif ($_action == 'delete') {
if (!is_array($_data['qid'])) {
$qids = array();
$qids[] = $_data['qid'];
}
else {
$qids = $_data['qid'];
}
$docker_return = docker('post', 'postfix-mailcow', 'exec', array('cmd' => 'mailq', 'task' => 'delete', 'items' => $qids));
process_mailq_output(json_decode($docker_return, true), $_action, $_data);
}
elseif ($_action == 'cat') {
if (!is_array($_data['qid'])) {
$qids = array();
$qids[] = $_data['qid'];
}
else {
$qids = $_data['qid'];
}
$docker_return = docker('post', 'postfix-mailcow', 'exec', array('cmd' => 'mailq', 'task' => 'cat', 'items' => $qids));
return process_mailq_output($docker_return, $_action, $_data);
}
elseif ($_action == 'edit') {
if (in_array($_data['action'], array('hold', 'unhold', 'deliver'))) {
if (!is_array($_data['qid'])) { if (!is_array($_data['qid'])) {
$qids = array(); $qids = array();
$qids[] = $_data['qid']; $qids[] = $_data['qid'];
@ -75,30 +109,14 @@ function mailq($_action, $_data = null) {
else { else {
$qids = $_data['qid']; $qids = $_data['qid'];
} }
$docker_return = docker('post', 'postfix-mailcow', 'exec', array('cmd' => 'mailq', 'task' => 'delete', 'items' => $qids)); if (!empty($qids)) {
process_mailq_output(json_decode($docker_return, true), $_action, $_data); $docker_return = docker('post', 'postfix-mailcow', 'exec', array('cmd' => 'mailq', 'task' => $_data['action'], 'items' => $qids));
break;
case 'edit':
if (in_array($_data['action'], array('hold', 'unhold', 'deliver'))) {
if (!is_array($_data['qid'])) {
$qids = array();
$qids[] = $_data['qid'];
}
else {
$qids = $_data['qid'];
}
if (!empty($qids)) {
$docker_return = docker('post', 'postfix-mailcow', 'exec', array('cmd' => 'mailq', 'task' => $_data['action'], 'items' => $qids));
process_mailq_output(json_decode($docker_return, true), $_action, $_data);
}
}
if (in_array($_data['action'], array('flush', 'super_delete'))) {
$docker_return = docker('post', 'postfix-mailcow', 'exec', array('cmd' => 'mailq', 'task' => $_data['action']));
process_mailq_output(json_decode($docker_return, true), $_action, $_data); process_mailq_output(json_decode($docker_return, true), $_action, $_data);
} }
break; }
case 'get': if (in_array($_data['action'], array('flush', 'super_delete'))) {
// todo: move get from json_api here $docker_return = docker('post', 'postfix-mailcow', 'exec', array('cmd' => 'mailq', 'task' => $_data['action']));
break; process_mailq_output(json_decode($docker_return, true), $_action, $_data);
}
} }
} }

View File

@ -248,6 +248,7 @@ jQuery(function($){
}}, }},
{"name":"sender","title":lang.sender, "type": "text","breakpoints":"xs sm"}, {"name":"sender","title":lang.sender, "type": "text","breakpoints":"xs sm"},
{"name":"recipients","title":lang.recipients, "type": "text","style":{"word-break":"break-all","min-width":"300px"},"breakpoints":"xs sm md"}, {"name":"recipients","title":lang.recipients, "type": "text","style":{"word-break":"break-all","min-width":"300px"},"breakpoints":"xs sm md"},
{"name":"action","filterable": false,"sortable": false,"style":{"text-align":"right","maxWidth":"220px","width":"220px"},"type":"html","title":lang.action,"breakpoints":"xs sm md"}
], ],
"rows": $.ajax({ "rows": $.ajax({
dataType: 'json', dataType: 'json',
@ -301,6 +302,9 @@ jQuery(function($){
return escapeHtml(i); return escapeHtml(i);
}); });
item.recipients = rcpts.join('<hr style="margin:1px!important">'); item.recipients = rcpts.join('<hr style="margin:1px!important">');
item.action = '<div class="btn-group">' +
'<a href="#" data-toggle="modal" data-target="#showQueuedMsg" data-queue-id="' + encodeURI(item.queue_id) + '" class="btn btn-xs btn-default">' + lang.queue_show_message + '</a>' +
'</div>';
}); });
} else if (table == 'forwardinghoststable') { } else if (table == 'forwardinghoststable') {
$.each(data, function (i, item) { $.each(data, function (i, item) {
@ -413,6 +417,22 @@ jQuery(function($){
$('#transport_type').val(button.data('transport-type')); $('#transport_type').val(button.data('transport-type'));
} }
}) })
// Queue item
$('#showQueuedMsg').on('show.bs.modal', function (e) {
$('#queue_msg_content').text("Loading...");
button = $(e.relatedTarget)
if (button != null) {
$('#queue_id').text(button.data('queue-id'));
}
$.ajax({
type: 'GET',
url: '/api/v1/get/postcat/' + button.data('queue-id'),
dataType: 'text',
complete: function (data) {
$('#queue_msg_content').text(data.responseText);
}
});
})
$('#test_transport').on('click', function (e) { $('#test_transport').on('click', function (e) {
e.preventDefault(); e.preventDefault();
prev = $('#test_transport').text(); prev = $('#test_transport').text();

View File

@ -404,7 +404,7 @@ jQuery(function($){
function rspamd_pie_graph() { function rspamd_pie_graph() {
$.ajax({ $.ajax({
url: '/api/v1/get/rspamd/actions', url: '/api/v1/get/rspamd/actions',
async: false, async: true,
success: function(data){ success: function(data){
var total = 0; var total = 0;
@ -438,17 +438,25 @@ jQuery(function($){
} }
} }
}; };
var chartcanvas = document.getElementById('rspamd_donut'); var chartcanvas = document.getElementById('rspamd_donut');
Chart.plugins.register('ChartDataLabels'); Chart.plugins.register('ChartDataLabels');
if(typeof chart == 'undefined') {
var chart = new Chart(chartcanvas.getContext("2d"), { chart = new Chart(chartcanvas.getContext("2d"), {
plugins: [ChartDataLabels], plugins: [ChartDataLabels],
type: 'doughnut', type: 'doughnut',
data: graphdata, data: graphdata,
options: options options: options
}); });
}
else {
chart.destroy();
chart = new Chart(chartcanvas.getContext("2d"), {
plugins: [ChartDataLabels],
type: 'doughnut',
data: graphdata,
options: options
});
}
} }
}); });
} }
@ -534,7 +542,7 @@ jQuery(function($){
} }
var str = '<strong>' + key + '</strong> ' + sym.score_formatted; var str = '<strong>' + key + '</strong> ' + sym.score_formatted;
if (sym.options) { if (sym.options) {
str += ' [' + sym.options.join(", ") + "]"; str += ' [' + escapeHtml(sym.options.join(", ")) + "]";
} }
return str return str
}).join('<br>\n'); }).join('<br>\n');
@ -703,7 +711,7 @@ jQuery(function($){
draw_rspamd_history(); draw_rspamd_history();
$('a[data-toggle="tab"]').on('shown.bs.tab', function (e) { $('a[data-toggle="tab"]').on('shown.bs.tab', function (e) {
var target = $(e.target).attr("href"); var target = $(e.target).attr("href");
if ((target == '#tab-rspamd-history')) { if (target == '#tab-rspamd-history') {
rspamd_pie_graph(); rspamd_pie_graph();
} }
}); });

View File

@ -375,6 +375,15 @@ if (isset($_SESSION['mailcow_cc_role']) || isset($_SESSION['pending_mailcow_cc_u
break; break;
} }
break; break;
case "postcat":
switch ($object) {
default:
$data = mailq('cat', array('qid' => $object));
echo $data;
break;
}
break;
case "global_filters": case "global_filters":
$global_filters = mailbox('get', 'global_filter_details'); $global_filters = mailbox('get', 'global_filter_details');

View File

@ -231,6 +231,7 @@
"queue_manager": "Queue Manager", "queue_manager": "Queue Manager",
"queue_unban": "Entsperren einreihen", "queue_unban": "Entsperren einreihen",
"queue_unhold_mail": "Freigeben", "queue_unhold_mail": "Freigeben",
"queue_show_message": "Nachricht anzeigen",
"quota_notification_html": "Benachrichtigungs-E-Mail Inhalt:<br><small>Leer lassen, um Standard-Template wiederherzustellen.</small>", "quota_notification_html": "Benachrichtigungs-E-Mail Inhalt:<br><small>Leer lassen, um Standard-Template wiederherzustellen.</small>",
"quota_notification_sender": "Benachrichtigungs-E-Mail Absender", "quota_notification_sender": "Benachrichtigungs-E-Mail Absender",
"quota_notification_subject": "Benachrichtigungs-E-Mail Betreff", "quota_notification_subject": "Benachrichtigungs-E-Mail Betreff",
@ -707,6 +708,7 @@
"tls_map_policy": "Richtlinie", "tls_map_policy": "Richtlinie",
"tls_policy_maps": "TLS-Richtlinien", "tls_policy_maps": "TLS-Richtlinien",
"tls_policy_maps_info": "Nachstehende Richtlinien erzwingen TLS-Transportregeln unabhängig von TLS-Richtlinieneinstellungen eines Benutzers.<br>\r\n Für weitere Informationen zur Syntax sollte <a href=\"http://www.postfix.org/postconf.5.html#smtp_tls_policy_maps\" target=\"_blank\">die \"smtp_tls_policy_maps\" Dokumentation</a> konsultiert werden.", "tls_policy_maps_info": "Nachstehende Richtlinien erzwingen TLS-Transportregeln unabhängig von TLS-Richtlinieneinstellungen eines Benutzers.<br>\r\n Für weitere Informationen zur Syntax sollte <a href=\"http://www.postfix.org/postconf.5.html#smtp_tls_policy_maps\" target=\"_blank\">die \"smtp_tls_policy_maps\" Dokumentation</a> konsultiert werden.",
"tls_policy_maps_enforced_tls": "Die Richtlinien überschreiben auch das Verhalten für Mailbox-Benutzer, die für ausgehende Verbindungen TLS erzwingen. Ist keine Policy nachstehend konfiguriert, richtet sich der Standard für diese Benutzer sich nach den Werten <code>smtp_tls_mandatory_protocols</code> und <code>smtp_tls_mandatory_ciphers</code>.",
"tls_policy_maps_long": "Ausgehende TLS-Richtlinien", "tls_policy_maps_long": "Ausgehende TLS-Richtlinien",
"toggle_all": "Alle", "toggle_all": "Alle",
"username": "Benutzername", "username": "Benutzername",

View File

@ -230,6 +230,7 @@
"queue_manager": "Queue manager", "queue_manager": "Queue manager",
"queue_unban": "queue unban", "queue_unban": "queue unban",
"queue_unhold_mail": "Unhold", "queue_unhold_mail": "Unhold",
"queue_show_message": "Show message",
"quota_notification_html": "Notification email template:<br><small>Leave empty to restore default template.</small>", "quota_notification_html": "Notification email template:<br><small>Leave empty to restore default template.</small>",
"quota_notification_sender": "Notification email sender", "quota_notification_sender": "Notification email sender",
"quota_notification_subject": "Notification email subject", "quota_notification_subject": "Notification email subject",
@ -706,6 +707,7 @@
"tls_map_policy": "Policy", "tls_map_policy": "Policy",
"tls_policy_maps": "TLS policy maps", "tls_policy_maps": "TLS policy maps",
"tls_policy_maps_info": "This policy map overrides outgoing TLS transport rules independently of a users TLS policy settings.<br>\r\n Please check <a href=\"http://www.postfix.org/postconf.5.html#smtp_tls_policy_maps\" target=\"_blank\">the \"smtp_tls_policy_maps\" docs</a> for further information.", "tls_policy_maps_info": "This policy map overrides outgoing TLS transport rules independently of a users TLS policy settings.<br>\r\n Please check <a href=\"http://www.postfix.org/postconf.5.html#smtp_tls_policy_maps\" target=\"_blank\">the \"smtp_tls_policy_maps\" docs</a> for further information.",
"tls_policy_maps_enforced_tls": "These policies will also override the behaviour for mailbox users that enforce outgoing TLS connections. If no policy exists below, these users will apply the default values specified as <code>smtp_tls_mandatory_protocols</code> and <code>smtp_tls_mandatory_ciphers</code>.",
"tls_policy_maps_long": "Outgoing TLS policy map overrides", "tls_policy_maps_long": "Outgoing TLS policy map overrides",
"toggle_all": "Toggle all", "toggle_all": "Toggle all",
"username": "Username", "username": "Username",

View File

@ -498,6 +498,7 @@ $_SESSION['return_to'] = $_SERVER['REQUEST_URI'];
</div> </div>
</div> </div>
<p style="margin:10px" class="help-block"><?=$lang['mailbox']['tls_policy_maps_info'];?></p> <p style="margin:10px" class="help-block"><?=$lang['mailbox']['tls_policy_maps_info'];?></p>
<p style="margin:10px" class="help-block"><?=$lang['mailbox']['tls_policy_maps_enforced_tls'];?></p>
<!-- <div class="mass-actions-mailbox" data-actions-header="true"></div> --> <!-- <div class="mass-actions-mailbox" data-actions-header="true"></div> -->
<div class="table-responsive"> <div class="table-responsive">
<table class="table table-striped" id="tls_policy_table"></table> <table class="table table-striped" id="tls_policy_table"></table>

View File

@ -206,6 +206,21 @@ if (!isset($_SESSION['mailcow_cc_role'])) {
</div> </div>
</div> </div>
</div><!-- test transport modal --> </div><!-- test transport modal -->
<!-- show queue item modal -->
<div class="modal fade" id="showQueuedMsg" tabindex="-1" role="dialog" aria-hidden="true">
<div class="modal-dialog modal-lg">
<div class="modal-content">
<div class="modal-header">
<button type="button" class="close" data-dismiss="modal"><span aria-hidden="true">×</span></button>
<h3 class="modal-title"><span class="glyphicon glyphicon-hourglass" style="font-size:18px"></span> ID <span id="queue_id"></span></h3>
</div>
<div class="modal-body">
<pre id="queue_msg_content">
</pre>
</div>
</div>
</div>
</div><!-- show queue item modal -->
<!-- priv key modal --> <!-- priv key modal -->
<div class="modal fade" id="showDKIMprivKey" tabindex="-1" role="dialog" aria-hidden="true"> <div class="modal fade" id="showDKIMprivKey" tabindex="-1" role="dialog" aria-hidden="true">
<div class="modal-dialog"> <div class="modal-dialog">