]> git.proxmox.com Git - pmg-gui.git/blame - js/SpamQuarantine.js
fix #4137: display receiver in attachment/virus quarantine
[pmg-gui.git] / js / SpamQuarantine.js
CommitLineData
d9c3552a
DM
1Ext.define('pmg-spam-archive', {
2 extend: 'Ext.data.Model',
3 fields: [
4 { type: 'number', name: 'spamavg' },
5 { type: 'integer', name: 'count' },
c87d46fb 6 { type: 'date', dateFormat: 'timestamp', name: 'day' },
d9c3552a
DM
7 ],
8 proxy: {
9 type: 'proxmox',
c87d46fb 10 url: "/api2/json/quarantine/spam",
d9c3552a 11 },
c87d46fb 12 idProperty: 'day',
d9c3552a
DM
13});
14
d9c3552a
DM
15Ext.define('pmg-spam-list', {
16 extend: 'Ext.data.Model',
c87d46fb 17 fields: ['id', 'envelope_sender', 'from', 'sender', 'receiver', 'subject',
af5aba0c 18 { type: 'number', name: 'spamlevel' },
d9c3552a 19 { type: 'integer', name: 'bytes' },
c96a22cb
DC
20 { type: 'date', dateFormat: 'timestamp', name: 'time' },
21 {
22 type: 'string',
23 name: 'day',
24 convert: function(v, rec) {
25 return Ext.Date.format(rec.get('time'), 'Y-m-d');
c87d46fb
TL
26 }, depends: ['time'],
27 },
d9c3552a
DM
28 ],
29 proxy: {
30 type: 'proxmox',
c87d46fb 31 url: "/api2/json/quarantine/spam",
d9c3552a 32 },
c87d46fb 33 idProperty: 'id',
d9c3552a
DM
34});
35
0277bfeb 36Ext.define('PMG.SpamQuarantine', {
d9c3552a 37 extend: 'Ext.container.Container',
0277bfeb
DM
38 xtype: 'pmgSpamQuarantine',
39
0277bfeb 40 border: false,
ea07c9aa 41 layout: { type: 'border' },
d9c3552a 42
0277bfeb
DM
43 defaults: { border: false },
44
207471c0
DC
45 // from mail link
46 cselect: undefined,
47
41a290e3
TL
48 viewModel: {
49 parent: null,
50 data: {
51 mailid: '',
52 },
53 formulas: {
54 downloadMailURL: get => '/api2/json/quarantine/download?mailid=' + encodeURIComponent(get('mailid')),
55 },
56 },
d9c3552a
DM
57 controller: {
58
2d1f7824 59 xclass: 'Ext.app.ViewController',
d9c3552a 60
2d1f7824 61 updatePreview: function(raw, rec) {
d9c3552a
DM
62 var preview = this.lookupReference('preview');
63
c87d46fb 64 if (!rec || !rec.data || !rec.data.id) {
d9c3552a 65 preview.update('');
2d1f7824 66 preview.setDisabled(true);
c96a22cb
DC
67 return;
68 }
69
a2293354
SI
70 let url = `/api2/htmlmail/quarantine/content?id=${rec.data.id}`;
71 if (raw) {
72 url += '&raw=1';
73 }
2d1f7824 74 preview.setDisabled(false);
d42f85b8
DC
75 this.lookupReference('raw').setDisabled(false);
76 this.lookupReference('spam').setDisabled(false);
7ad0de10 77 this.lookupReference('download').setDisabled(false);
c96a22cb
DC
78 preview.update("<iframe frameborder=0 width=100% height=100% sandbox='allow-same-origin' src='" + url +"'></iframe>");
79 },
80
b9a5e707 81 multiSelect: function(selection) {
d42f85b8
DC
82 var preview = this.lookupReference('preview');
83 var raw = this.lookupReference('raw');
84 var spam = this.lookupReference('spam');
85 var spaminfo = this.lookupReference('spaminfo');
ac6c1fb8 86 var mailinfo = this.lookupReference('mailinfo');
7ad0de10 87 var download = this.lookupReference('download');
d42f85b8
DC
88
89 preview.setDisabled(false);
b9a5e707 90 preview.update(`<h3 style="padding-left:5px;">${gettext('Multiple E-Mails selected')} (${selection.length})</h3>`);
d42f85b8
DC
91 raw.setDisabled(true);
92 spam.setDisabled(true);
93 spam.setPressed(false);
94 spaminfo.setVisible(false);
ac6c1fb8 95 mailinfo.setVisible(false);
7ad0de10 96 download.setDisabled(true);
d42f85b8
DC
97 },
98
2d1f7824 99 toggleRaw: function(button) {
c96a22cb 100 var me = this;
4935e8fb 101 var list = me.lookupReference('list');
b76ce2c5 102 var rec = list.selModel.getSelection()[0];
4935e8fb 103 me.lookupReference('mailinfo').setVisible(me.raw);
2d1f7824 104 me.raw = !me.raw;
b76ce2c5 105 me.updatePreview(me.raw, rec);
c96a22cb
DC
106 },
107
2d1f7824 108 btnHandler: function(button, e) {
7f0619ff
DC
109 var me = this;
110 var action = button.reference;
2d1f7824
DC
111 var list = this.lookupReference('list');
112 var selected = list.getSelection();
7f0619ff
DC
113 me.doAction(action, selected);
114 },
115
116 doAction: function(action, selected) {
2d1f7824 117 if (!selected.length) {
d9c3552a
DM
118 return;
119 }
120
7f0619ff 121 var list = this.lookupReference('list');
c96a22cb 122
d42f85b8 123 if (selected.length > 1) {
b61e38d7 124 let idlist = selected.map(item => item.data.id);
d42f85b8
DC
125 Ext.Msg.confirm(
126 gettext('Confirm'),
127 Ext.String.format(
128 gettext("Action '{0}' for '{1}' items"),
c87d46fb 129 action, selected.length,
d42f85b8 130 ),
3168b7f7 131 async function(button) {
d42f85b8
DC
132 if (button !== 'yes') {
133 return;
134 }
135
3168b7f7
TL
136 list.mask(gettext('Processing...'), 'x-mask-loading');
137
138 const sliceSize = 2500, maxInFlight = 2;
139 let batches = [], batchCount = Math.ceil(selected.length / sliceSize);
140 for (let i = 0; i * sliceSize < selected.length; i++) {
141 let sliceStart = i * sliceSize;
142 let sliceEnd = Math.min(sliceStart + sliceSize, selected.length);
143 batches.push(
144 PMG.Async.doQAction(
145 action,
146 idlist.slice(sliceStart, sliceEnd),
147 i + 1,
148 batchCount,
149 ),
150 );
151 if (batches.length >= maxInFlight) {
152 await Promise.allSettled(batches); // eslint-disable-line no-await-in-loop
153 batches = [];
154 }
155 }
156 await Promise.allSettled(batches); // await possible remaining ones
157 list.unmask();
158 // below can be slow, we could remove directly from the in-memory store, but
159 // with lots of elements and some failures we could be quite out of sync?
160 list.getController().load();
c87d46fb 161 },
d42f85b8
DC
162 );
163 return;
164 }
165
aac17b9b 166 PMG.Utils.doQuarantineAction(action, selected[0].data.id, function() {
980e88a9
TL
167 let listController = list.getController();
168 listController.allowPositionSave = false;
9aed379a
TL
169 // success -> remove directly to avoid slow store reload for a single-element action
170 list.getStore().remove(selected[0]);
980e88a9
TL
171 listController.restoreSavedSelection();
172 listController.allowPositionSave = true;
2d1f7824 173 });
c96a22cb
DC
174 },
175
2d1f7824
DC
176 onSelectMail: function() {
177 var me = this;
178 var list = this.lookupReference('list');
d42f85b8
DC
179 var selection = list.selModel.getSelection();
180 if (selection.length > 1) {
b9a5e707 181 me.multiSelect(selection);
d42f85b8
DC
182 return;
183 }
2d1f7824 184
38771d94
DC
185 var rec = selection[0] || {};
186
187 me.getViewModel().set('mailid', rec.data ? rec.data.id : '');
2d1f7824
DC
188 me.updatePreview(me.raw || false, rec);
189 me.lookupReference('spaminfo').setID(rec);
72d8403c 190 me.lookupReference('mailinfo').setVisible(!!rec.data && !me.raw);
ac6c1fb8 191 me.lookupReference('mailinfo').update(rec.data);
d9c3552a 192 },
c96a22cb 193
2d1f7824
DC
194 toggleSpamInfo: function(btn) {
195 var grid = this.lookupReference('spaminfo');
196 grid.setVisible(!grid.isVisible());
8e89b895
DC
197 },
198
c627f092
DC
199 openContextMenu: function(table, record, tr, index, event) {
200 event.stopEvent();
0affcba5
TL
201 let me = this;
202 let list = me.lookup('list');
203 Ext.create('PMG.menu.SpamContextMenu', {
204 callback: action => me.doAction(action, list.getSelection()),
205 }).showAt(event.getXY());
c627f092
DC
206 },
207
c87d46fb 208 keyPress: function(table, record, item, index, event) {
0b9c0528
DC
209 var me = this;
210 var list = me.lookup('list');
211 var key = event.getKey();
212 var action = '';
c87d46fb 213 switch (key) {
0b9c0528
DC
214 case event.DELETE:
215 case 127:
216 action = 'delete';
217 break;
218 case Ext.event.Event.D:
219 case Ext.event.Event.D + 32:
220 action = 'deliver';
221 break;
222 case Ext.event.Event.W:
223 case Ext.event.Event.W + 32:
224 action = 'whitelist';
225 break;
226 case Ext.event.Event.B:
227 case Ext.event.Event.B + 32:
228 action = 'blacklist';
229 break;
230 }
231
232 if (action !== '') {
233 me.doAction(action, list.getSelection());
234 }
235 },
236
207471c0
DC
237 init: function(view) {
238 this.lookup('list').cselect = view.cselect;
239 },
240
c96a22cb 241 control: {
c96a22cb 242 'button[reference=raw]': {
c87d46fb 243 click: 'toggleRaw',
c96a22cb 244 },
2d1f7824 245 'button[reference=spam]': {
c87d46fb 246 click: 'toggleSpamInfo',
c96a22cb 247 },
2d1f7824 248 'pmgQuarantineList': {
c627f092 249 selectionChange: 'onSelectMail',
0b9c0528 250 itemkeypress: 'keyPress',
c87d46fb
TL
251 rowcontextmenu: 'openContextMenu',
252 },
253 },
d9c3552a
DM
254 },
255
0277bfeb
DM
256 items: [
257 {
56b4528f 258 title: gettext('Spam Quarantine'),
2d1f7824 259 xtype: 'pmgQuarantineList',
62651172 260 selModel: 'checkboxmodel',
2d1f7824
DC
261 emailSelection: true,
262 reference: 'list',
d9c3552a 263 region: 'west',
c96a22cb 264 width: 500,
d9c3552a 265 split: true,
c96a22cb 266 collapsible: false,
2d1f7824
DC
267 store: {
268 model: 'pmg-spam-list',
269 groupField: 'day',
270 groupDir: 'DESC',
271 sorters: [{
272 property: 'time',
c87d46fb
TL
273 direction: 'DESC',
274 }],
2d1f7824
DC
275 },
276
277 columns: [
278 {
279 header: gettext('Sender/Subject'),
280 dataIndex: 'subject',
281 renderer: PMG.Utils.sender_renderer,
c87d46fb 282 flex: 1,
2d1f7824
DC
283 },
284 {
285 header: gettext('Score'),
286 dataIndex: 'spamlevel',
287 align: 'right',
c87d46fb 288 width: 70,
2d1f7824
DC
289 },
290 {
291 header: gettext('Size') + ' (KB)',
292 renderer: function(v) { return Ext.Number.toFixed(v/1024, 0); },
293 dataIndex: 'bytes',
294 align: 'right',
c87d46fb 295 width: 90,
2d1f7824
DC
296 },
297 {
298 header: gettext('Date'),
299 dataIndex: 'day',
c87d46fb 300 hidden: true,
2d1f7824
DC
301 },
302 {
303 xtype: 'datecolumn',
304 header: gettext('Time'),
305 dataIndex: 'time',
c87d46fb
TL
306 format: 'H:i:s',
307 },
308 ],
0277bfeb
DM
309 },
310 {
c96a22cb 311 title: gettext('Selected Mail'),
ea07c9aa 312 border: false,
d9c3552a 313 region: 'center',
60a9d6fd 314 layout: 'fit',
c96a22cb
DC
315 split: true,
316 reference: 'preview',
317 disabled: true,
2d1f7824
DC
318 dockedItems: [
319 {
320 xtype: 'toolbar',
321 dock: 'top',
322 items: [
323 {
324 xtype: 'button',
325 reference: 'raw',
326 text: gettext('Toggle Raw'),
327 enableToggle: true,
c87d46fb 328 iconCls: 'fa fa-file-code-o',
2d1f7824
DC
329 },
330 {
331 xtype: 'button',
332 reference: 'spam',
333 text: gettext('Toggle Spam Info'),
334 enableToggle: true,
c87d46fb 335 iconCls: 'fa fa-bullhorn',
2d1f7824
DC
336 },
337 '->',
7ad0de10
DC
338 {
339 xtype: 'button',
340 reference: 'download',
341 text: gettext('Download'),
07a9b445
TL
342 setDownload: function(id) {
343 this.el.dom.download = id + ".eml";
344 },
345 bind: {
346 href: '{downloadMailURL}',
347 download: '{mailid}',
348 },
c87d46fb 349 iconCls: 'fa fa-download',
7ad0de10
DC
350 },
351 '-',
2d1f7824
DC
352 {
353 reference: 'whitelist',
354 text: gettext('Whitelist'),
355 iconCls: 'fa fa-check',
c87d46fb 356 handler: 'btnHandler',
2d1f7824
DC
357 },
358 {
359 reference: 'blacklist',
360 text: gettext('Blacklist'),
361 iconCls: 'fa fa-times',
c87d46fb 362 handler: 'btnHandler',
2d1f7824
DC
363 },
364 {
365 reference: 'deliver',
366 text: gettext('Deliver'),
367 iconCls: 'fa fa-paper-plane-o',
c87d46fb 368 handler: 'btnHandler',
2d1f7824
DC
369 },
370 {
371 reference: 'delete',
372 text: gettext('Delete'),
373 iconCls: 'fa fa-trash-o',
c87d46fb
TL
374 handler: 'btnHandler',
375 },
376 ],
2d1f7824
DC
377 },
378 {
379 xtype: 'pmgSpamInfoGrid',
f34abaca 380 border: false,
c87d46fb 381 reference: 'spaminfo',
ac6c1fb8
DC
382 },
383 {
384 xtype: 'pmgMailInfo',
385 hidden: true,
386 reference: 'mailinfo',
387 },
c87d46fb
TL
388 ],
389 },
390 ],
0277bfeb 391});