1
0
mirror of https://github.com/mail-in-a-box/mailinabox.git synced 2025-04-05 00:27:25 +00:00

Split the User Activity/IMAP connections tab into two tables to better deal with the quantity of data

This commit is contained in:
downtownallday 2021-04-12 15:07:56 -04:00
parent 212b0b74cb
commit 36d9cbb4e8
11 changed files with 248 additions and 58 deletions

View File

@ -102,6 +102,18 @@ def add_reports(app, env, authorized_personnel_only):
finally: finally:
db_conn_factory.close(conn) db_conn_factory.close(conn)
@app.route('/reports/uidata/imap-details', methods=['POST'])
@authorized_personnel_only
@json_payload
def get_imap_details(payload):
conn = db_conn_factory.connect()
try:
return jsonify(uidata.imap_details(conn, payload))
except uidata.InvalidArgsError as e:
return ('invalid request', 400)
finally:
db_conn_factory.close(conn)
@app.route('/reports/uidata/flagged-connections', methods=['POST']) @app.route('/reports/uidata/flagged-connections', methods=['POST'])
@authorized_personnel_only @authorized_personnel_only
@json_payload @json_payload

View File

@ -2,7 +2,8 @@ export default Vue.component('chart-table', {
props: { props: {
items: Array, items: Array,
fields: Array, fields: Array,
caption: String caption: String,
small: { type:Boolean, default:true }
}, },
/* <b-table-lite striped small :fields="fields_x" :items="items" caption-top><template #table-caption><span class="text-nowrap">{{caption}}</span></template></b-table>*/ /* <b-table-lite striped small :fields="fields_x" :items="items" caption-top><template #table-caption><span class="text-nowrap">{{caption}}</span></template></b-table>*/
@ -19,7 +20,7 @@ export default Vue.component('chart-table', {
var table = ce('b-table-lite', { var table = ce('b-table-lite', {
props: { props: {
'striped': true, 'striped': true,
'small': true, 'small': this.small,
'fields': this.fields_x, 'fields': this.fields_x,
'items': this.items, 'items': this.items,
'caption-top': true 'caption-top': true

View File

@ -493,6 +493,8 @@ export class BvTableField {
} }
else if (ft.type == 'number') { else if (ft.type == 'number') {
if (ft.subtype == 'plain' || if (ft.subtype == 'plain' ||
ft.subtype === null ||
ft.subtype === undefined ||
ft.subtype == 'decimal' && isNaN(ft.places) ft.subtype == 'decimal' && isNaN(ft.places)
) )
{ {

View File

@ -84,26 +84,30 @@
<b-tab> <b-tab>
<template #title> <template #title>
IMAP Connections<sup v-if="imap_details.items.length >= get_row_limit()">*</sup> ({{imap_details.items.length}}) IMAP Connections
</template> </template>
<b-table <b-table
class="sticky-table-header-0 bg-light" tbody-tr-class="cursor-pointer"
small selectable
select-mode="single"
:filter="show_only_flagged_filter" :filter="show_only_flagged_filter"
:filter-function="table_filter_cb" :filter-function="table_filter_cb"
tbody-tr-class="cursor-pointer" :items="imap_conn_summary.items"
details-td-class="cursor-default" :fields="imap_conn_summary.fields"
@row-clicked="row_clicked" @row-clicked="load_imap_details">
:items="imap_details.items" </b-table>
:fields="imap_details.fields">
<template #row-details="row"> <div v-if="imap_details" class="bg-white">
<b-card> <div class="mt-3 text-center bg-info p-1">{{imap_details._desc}} ({{imap_details.items.length}} rows<sup v-if="imap_details.items.length >= get_row_limit()">*</sup>)</div>
<div><strong>Connection disposition</strong>: {{ disposition_formatter(row.item.disposition) }}</div> <b-table
<div><strong>Connection security</strong> {{ row.item.connection_security }}</div> class="sticky-table-header-0"
<div><strong>Disconnect reason</strong> {{ row.item.disconnect_reason }}</div> small
</b-card> :items="imap_details.items"
</template> :fields="imap_details.fields">
</b-table> </b-table>
</div>
</b-tab> </b-tab>
</b-tabs> </b-tabs>

View File

@ -3,9 +3,10 @@
*/ */
import wbr_text from "./wbr-text.js"; import wbr_text from "./wbr-text.js";
import chart_table from "./chart-table.js";
import message_headers_view from "./message_headers_view.js"; import message_headers_view from "./message_headers_view.js";
import UserSettings from "./settings.js"; import UserSettings from "./settings.js";
import { MailBvTable, ConnectionDisposition } from "./charting.js"; import { BvTable, MailBvTable, ConnectionDisposition } from "./charting.js";
export default Vue.component('panel-user-activity', function(resolve, reject) { export default Vue.component('panel-user-activity', function(resolve, reject) {
@ -19,7 +20,8 @@ export default Vue.component('panel-user-activity', function(resolve, reject) {
components: { components: {
'wbr-text': wbr_text, 'wbr-text': wbr_text,
'message-headers-view': message_headers_view 'message-headers-view': message_headers_view,
'chart-table': chart_table,
}, },
data: function() { data: function() {
@ -36,6 +38,7 @@ export default Vue.component('panel-user-activity', function(resolve, reject) {
data_date_range: null, /* date range for active table data */ data_date_range: null, /* date range for active table data */
sent_mail: null, sent_mail: null,
received_mail: null, received_mail: null,
imap_conn_summary: null,
imap_details: null, imap_details: null,
lmtp_id: null, /* for message headers modal */ lmtp_id: null, /* for message headers modal */
all_users: [], all_users: [],
@ -169,11 +172,19 @@ export default Vue.component('panel-user-activity', function(resolve, reject) {
f.label = 'Envelope From (user)'; f.label = 'Envelope From (user)';
}, },
combine_imap_conn_summary_fields: function() {
// remove 'first_conn_time'
this.imap_conn_summary.combine_fields('first_connection_time');
// clear the label for the 'total' column (pct)
const f_total = this.imap_conn_summary.get_field('total');
f_total.label = '';
},
combine_imap_details_fields: function() { combine_imap_details_fields: function() {
// remove these fields // remove these fields
this.imap_details.combine_fields([ this.imap_details.combine_fields([
'disconnect_reason', 'remote_host',
'connection_security', 'disposition',
]); ]);
}, },
@ -272,16 +283,21 @@ export default Vue.component('panel-user-activity', function(resolve, reject) {
.get_field('connect_time') .get_field('connect_time')
.add_tdClass('text-nowrap'); .add_tdClass('text-nowrap');
/* setup imap_details */
this.imap_details = new MailBvTable( /* setup imap_conn_summary */
response.data.imap_details, { this.imap_conn_summary = new MailBvTable(
_showDetails: true response.data.imap_conn_summary
);
this.combine_imap_conn_summary_fields();
this.imap_conn_summary.flag_fields();
['last_connection_time','count']
.forEach(name => {
const f = this.imap_conn_summary.get_field(name);
f.add_cls('text-nowrap', 'tdClass');
}); });
this.combine_imap_details_fields();
this.imap_details /* clear imap_details */
.flag_fields() this.imap_details = null;
.get_field('connect_time')
.add_tdClass('text-nowrap');
}).catch(error => { }).catch(error => {
this.$root.handleError(error); this.$root.handleError(error);
@ -304,6 +320,36 @@ export default Vue.component('panel-user-activity', function(resolve, reject) {
// show the modal dialog // show the modal dialog
this.$refs.message_headers_modal.show(); this.$refs.message_headers_modal.show();
},
load_imap_details: function(item, index, event) {
this.$emit('loading', 1);
this.imap_details = null;
const promise = axios.post('reports/uidata/imap-details', {
row_limit: this.get_row_limit(),
user_id: this.user_id.trim(),
start_date: this.date_range[0],
end_date: this.date_range[1],
disposition: item.disposition,
remote_host: item.remote_host
}).then(response => {
this.imap_details = new MailBvTable(
response.data.imap_details
);
this.combine_imap_details_fields();
this.imap_details.get_field('connect_time')
.add_tdClass('text-nowrap');
this.imap_details.get_field('disconnect_time')
.add_tdClass('text-nowrap');
this.imap_details._desc =
`${item.remote_host}/${item.disposition}`;
}).catch(error => {
this.$root.handleError(error);
}).finally( () => {
this.$emit('loading', -1);
});
} }
} }

View File

@ -22,7 +22,7 @@ class Timeseries(object):
# parsefmt is a date parser string to be used to re-interpret # parsefmt is a date parser string to be used to re-interpret
# "bin" grouping dates (data.dates) to native dates. server # "bin" grouping dates (data.dates) to native dates. server
# always returns utc dates # always returns utc dates
parsefmt = '%Y-%m-%d %H:%M:%S' self.parsefmt = '%Y-%m-%d %H:%M:%S'
self.dates = [] # dates must be "bin" date strings self.dates = [] # dates must be "bin" date strings
self.series = [] self.series = []
@ -31,7 +31,7 @@ class Timeseries(object):
'range': [ self.start, self.end ], 'range': [ self.start, self.end ],
'range_parse_format': '%Y-%m-%d %H:%M:%S', 'range_parse_format': '%Y-%m-%d %H:%M:%S',
'binsize': self.binsize, 'binsize': self.binsize,
'date_parse_format': parsefmt, 'date_parse_format': self.parsefmt,
'y': desc, 'y': desc,
'dates': self.dates, 'dates': self.dates,
'series': self.series 'series': self.series

View File

@ -3,6 +3,7 @@ from .select_list_suggestions import select_list_suggestions
from .messages_sent import messages_sent from .messages_sent import messages_sent
from .messages_received import messages_received from .messages_received import messages_received
from .user_activity import user_activity from .user_activity import user_activity
from .imap_details import imap_details
from .remote_sender_activity import remote_sender_activity from .remote_sender_activity import remote_sender_activity
from .flagged_connections import flagged_connections from .flagged_connections import flagged_connections
from .capture_db_stats import capture_db_stats from .capture_db_stats import capture_db_stats

View File

@ -0,0 +1,25 @@
--
-- details on user imap connections
--
SELECT
connect_time,
disconnect_time,
CASE WHEN remote_host='unknown' THEN remote_ip ELSE remote_host END AS `remote_host`,
sasl_method,
disconnect_reason,
connection_security,
disposition,
in_bytes,
out_bytes
FROM
imap_connection
WHERE
sasl_username = :user_id AND
connect_time >= :start_date AND
connect_time < :end_date AND
(:remote_host IS NULL OR
remote_host = :remote_host OR remote_ip = :remote_host) AND
(:disposition IS NULL OR
disposition = :disposition)
ORDER BY
connect_time

View File

@ -0,0 +1,83 @@
from .Timeseries import Timeseries
from .exceptions import InvalidArgsError
with open(__file__.replace('.py','.1.sql')) as fp:
select_1 = fp.read()
def imap_details(conn, args):
'''
details on imap connections
'''
try:
user_id = args['user_id']
# use Timeseries to get a normalized start/end range
ts = Timeseries(
'IMAP details',
args['start_date'],
args['end_date'],
0
)
# optional
remote_host = args.get('remote_host')
disposition = args.get('disposition')
except KeyError:
raise InvalidArgsError()
# limit results
try:
limit = 'LIMIT ' + str(int(args.get('row_limit', 1000)));
except ValueError:
limit = 'LIMIT 1000'
c = conn.cursor()
imap_details = {
'start': ts.start,
'end': ts.end,
'y': 'IMAP Details',
'fields': [
'connect_time',
'disconnect_time',
'remote_host',
'sasl_method',
'disconnect_reason',
'connection_security',
'disposition',
'in_bytes',
'out_bytes'
],
'field_types': [
{ 'type':'datetime', 'format': ts.parsefmt }, # connect_time
{ 'type':'datetime', 'format': ts.parsefmt }, # disconnect_time
'text/plain', # remote_host
'text/plain', # sasl_method
'text/plain', # disconnect_reason
'text/plain', # connection_security
'text/plain', # disposition
'number/size', # in_bytes,
'number/size', # out_bytes,
],
'items': []
}
for row in c.execute(select_1 + limit, {
'user_id': user_id,
'start_date': ts.start,
'end_date': ts.end,
'remote_host': remote_host,
'disposition': disposition
}):
v = []
for key in imap_details['fields']:
v.append(row[key])
imap_details['items'].append(v)
return {
'imap_details': imap_details
}

View File

@ -1,20 +1,22 @@
-- --
-- details on user imap connections -- imap connection summary
-- --
SELECT SELECT
connect_time, count(*) as `count`,
CASE WHEN remote_host='unknown' THEN remote_ip ELSE remote_host END AS `remote_host`,
sasl_method,
disconnect_reason,
connection_security,
disposition, disposition,
in_bytes, CASE WHEN remote_host='unknown' THEN remote_ip ELSE remote_host END AS `remote_host`,
out_bytes sum(in_bytes) as `in_bytes`,
sum(out_bytes) as `out_bytes`,
min(connect_time) as `first_connection_time`,
max(connect_time) as `last_connection_time`
FROM FROM
imap_connection imap_connection
WHERE WHERE
sasl_username = :user_id AND sasl_username = :user_id AND
connect_time >= :start_date AND connect_time >= :start_date AND
connect_time < :end_date connect_time < :end_date
GROUP BY
disposition,
CASE WHEN remote_host='unknown' THEN remote_ip ELSE remote_host END
ORDER BY ORDER BY
connect_time `count` DESC, disposition

View File

@ -200,50 +200,64 @@ def user_activity(conn, args):
# #
# imap connections by user # IMAP connections by disposition, by remote host
# Disposition
# Remote host
# Count
# In bytes (sum)
# Out bytes (sum)
# % of total
# #
imap_details = { imap_conn_summary = {
'start': ts.start, 'start': ts.start,
'end': ts.end, 'end': ts.end,
'y': 'IMAP Details', 'y': 'IMAP connection summary by host and disposition',
'fields': [ 'fields': [
'connect_time', 'count',
'total',
'remote_host', 'remote_host',
'sasl_method',
'disconnect_reason',
'connection_security',
'disposition', 'disposition',
'first_connection_time',
'last_connection_time',
'in_bytes', 'in_bytes',
'out_bytes' 'out_bytes',
], ],
'field_types': [ 'field_types': [
{ 'type':'datetime', 'format': '%Y-%m-%d %H:%M:%S' },# connect_time 'number', # count
{ 'type': 'number/percent', 'places': 1 }, # total
'text/plain', # remote_host 'text/plain', # remote_host
'text/plain', # sasl_method
'text/plain', # disconnect_reason
'text/plain', # connection_security
'text/plain', # disposition 'text/plain', # disposition
{ 'type':'datetime', 'format': ts.parsefmt }, # first_conn_time
{ 'type':'datetime', 'format': ts.parsefmt }, # last_conn_time
'number/size', # in_bytes, 'number/size', # in_bytes,
'number/size', # out_bytes, 'number/size', # out_bytes,
], ],
'items': [] 'items': []
} }
count_field_idx = 0
total_field_idx = 1
total = 0
for row in c.execute(select_3 + limit, { for row in c.execute(select_3 + limit, {
'user_id': user_id, 'user_id': user_id,
'start_date': ts.start, 'start_date': ts.start,
'end_date': ts.end 'end_date': ts.end
}): }):
v = [] v = []
for key in imap_details['fields']: for key in imap_conn_summary['fields']:
v.append(row[key]) if key=='count':
imap_details['items'].append(v) total += row[key]
if key!='total':
v.append(row[key])
imap_conn_summary['items'].append(v)
for v in imap_conn_summary['items']:
v.insert(total_field_idx, v[count_field_idx] / total)
return { return {
'sent_mail': sent_mail, 'sent_mail': sent_mail,
'received_mail': received_mail, 'received_mail': received_mail,
'imap_details': imap_details 'imap_conn_summary': imap_conn_summary
} }