]> git.proxmox.com Git - pmg-gui.git/blame - js/SpamQuarantine.js
remove safe browsing from clamav tab
[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() {
9aed379a
TL
167 // success -> remove directly to avoid slow store reload for a single-element action
168 list.getStore().remove(selected[0]);
642aba90 169 list.getController().restoreSavedSelection();
2d1f7824 170 });
c96a22cb
DC
171 },
172
2d1f7824
DC
173 onSelectMail: function() {
174 var me = this;
175 var list = this.lookupReference('list');
d42f85b8
DC
176 var selection = list.selModel.getSelection();
177 if (selection.length > 1) {
b9a5e707 178 me.multiSelect(selection);
d42f85b8
DC
179 return;
180 }
2d1f7824 181
38771d94
DC
182 var rec = selection[0] || {};
183
184 me.getViewModel().set('mailid', rec.data ? rec.data.id : '');
2d1f7824
DC
185 me.updatePreview(me.raw || false, rec);
186 me.lookupReference('spaminfo').setID(rec);
72d8403c 187 me.lookupReference('mailinfo').setVisible(!!rec.data && !me.raw);
ac6c1fb8 188 me.lookupReference('mailinfo').update(rec.data);
d9c3552a 189 },
c96a22cb 190
2d1f7824
DC
191 toggleSpamInfo: function(btn) {
192 var grid = this.lookupReference('spaminfo');
193 grid.setVisible(!grid.isVisible());
8e89b895
DC
194 },
195
c627f092
DC
196 openContextMenu: function(table, record, tr, index, event) {
197 event.stopEvent();
0affcba5
TL
198 let me = this;
199 let list = me.lookup('list');
200 Ext.create('PMG.menu.SpamContextMenu', {
201 callback: action => me.doAction(action, list.getSelection()),
202 }).showAt(event.getXY());
c627f092
DC
203 },
204
c87d46fb 205 keyPress: function(table, record, item, index, event) {
0b9c0528
DC
206 var me = this;
207 var list = me.lookup('list');
208 var key = event.getKey();
209 var action = '';
c87d46fb 210 switch (key) {
0b9c0528
DC
211 case event.DELETE:
212 case 127:
213 action = 'delete';
214 break;
215 case Ext.event.Event.D:
216 case Ext.event.Event.D + 32:
217 action = 'deliver';
218 break;
219 case Ext.event.Event.W:
220 case Ext.event.Event.W + 32:
221 action = 'whitelist';
222 break;
223 case Ext.event.Event.B:
224 case Ext.event.Event.B + 32:
225 action = 'blacklist';
226 break;
227 }
228
229 if (action !== '') {
230 me.doAction(action, list.getSelection());
231 }
232 },
233
207471c0
DC
234 init: function(view) {
235 this.lookup('list').cselect = view.cselect;
236 },
237
c96a22cb 238 control: {
c96a22cb 239 'button[reference=raw]': {
c87d46fb 240 click: 'toggleRaw',
c96a22cb 241 },
2d1f7824 242 'button[reference=spam]': {
c87d46fb 243 click: 'toggleSpamInfo',
c96a22cb 244 },
2d1f7824 245 'pmgQuarantineList': {
c627f092 246 selectionChange: 'onSelectMail',
0b9c0528 247 itemkeypress: 'keyPress',
c87d46fb
TL
248 rowcontextmenu: 'openContextMenu',
249 },
250 },
d9c3552a
DM
251 },
252
0277bfeb
DM
253 items: [
254 {
56b4528f 255 title: gettext('Spam Quarantine'),
2d1f7824 256 xtype: 'pmgQuarantineList',
62651172 257 selModel: 'checkboxmodel',
2d1f7824
DC
258 emailSelection: true,
259 reference: 'list',
d9c3552a 260 region: 'west',
c96a22cb 261 width: 500,
d9c3552a 262 split: true,
c96a22cb 263 collapsible: false,
2d1f7824
DC
264 store: {
265 model: 'pmg-spam-list',
266 groupField: 'day',
267 groupDir: 'DESC',
268 sorters: [{
269 property: 'time',
c87d46fb
TL
270 direction: 'DESC',
271 }],
2d1f7824
DC
272 },
273
274 columns: [
275 {
276 header: gettext('Sender/Subject'),
277 dataIndex: 'subject',
278 renderer: PMG.Utils.sender_renderer,
c87d46fb 279 flex: 1,
2d1f7824
DC
280 },
281 {
282 header: gettext('Score'),
283 dataIndex: 'spamlevel',
284 align: 'right',
c87d46fb 285 width: 70,
2d1f7824
DC
286 },
287 {
288 header: gettext('Size') + ' (KB)',
289 renderer: function(v) { return Ext.Number.toFixed(v/1024, 0); },
290 dataIndex: 'bytes',
291 align: 'right',
c87d46fb 292 width: 90,
2d1f7824
DC
293 },
294 {
295 header: gettext('Date'),
296 dataIndex: 'day',
c87d46fb 297 hidden: true,
2d1f7824
DC
298 },
299 {
300 xtype: 'datecolumn',
301 header: gettext('Time'),
302 dataIndex: 'time',
c87d46fb
TL
303 format: 'H:i:s',
304 },
305 ],
0277bfeb
DM
306 },
307 {
c96a22cb 308 title: gettext('Selected Mail'),
ea07c9aa 309 border: false,
d9c3552a 310 region: 'center',
60a9d6fd 311 layout: 'fit',
c96a22cb
DC
312 split: true,
313 reference: 'preview',
314 disabled: true,
2d1f7824
DC
315 dockedItems: [
316 {
317 xtype: 'toolbar',
318 dock: 'top',
319 items: [
320 {
321 xtype: 'button',
322 reference: 'raw',
323 text: gettext('Toggle Raw'),
324 enableToggle: true,
c87d46fb 325 iconCls: 'fa fa-file-code-o',
2d1f7824
DC
326 },
327 {
328 xtype: 'button',
329 reference: 'spam',
330 text: gettext('Toggle Spam Info'),
331 enableToggle: true,
c87d46fb 332 iconCls: 'fa fa-bullhorn',
2d1f7824
DC
333 },
334 '->',
7ad0de10
DC
335 {
336 xtype: 'button',
337 reference: 'download',
338 text: gettext('Download'),
07a9b445
TL
339 setDownload: function(id) {
340 this.el.dom.download = id + ".eml";
341 },
342 bind: {
343 href: '{downloadMailURL}',
344 download: '{mailid}',
345 },
c87d46fb 346 iconCls: 'fa fa-download',
7ad0de10
DC
347 },
348 '-',
2d1f7824
DC
349 {
350 reference: 'whitelist',
351 text: gettext('Whitelist'),
352 iconCls: 'fa fa-check',
c87d46fb 353 handler: 'btnHandler',
2d1f7824
DC
354 },
355 {
356 reference: 'blacklist',
357 text: gettext('Blacklist'),
358 iconCls: 'fa fa-times',
c87d46fb 359 handler: 'btnHandler',
2d1f7824
DC
360 },
361 {
362 reference: 'deliver',
363 text: gettext('Deliver'),
364 iconCls: 'fa fa-paper-plane-o',
c87d46fb 365 handler: 'btnHandler',
2d1f7824
DC
366 },
367 {
368 reference: 'delete',
369 text: gettext('Delete'),
370 iconCls: 'fa fa-trash-o',
c87d46fb
TL
371 handler: 'btnHandler',
372 },
373 ],
2d1f7824
DC
374 },
375 {
376 xtype: 'pmgSpamInfoGrid',
f34abaca 377 border: false,
c87d46fb 378 reference: 'spaminfo',
ac6c1fb8
DC
379 },
380 {
381 xtype: 'pmgMailInfo',
382 hidden: true,
383 reference: 'mailinfo',
384 },
c87d46fb
TL
385 ],
386 },
387 ],
0277bfeb 388});