Ext.define('PVE.grid.BackupView', { extend: 'Ext.grid.GridPanel', alias: ['widget.pveBackupView'], onlineHelp: 'chapter_vzdump', stateful: true, stateId: 'grid-guest-backup', initComponent : function() { var me = this; var nodename = me.pveSelNode.data.node; if (!nodename) { throw "no node name specified"; } var vmid = me.pveSelNode.data.vmid; if (!vmid) { throw "no VM ID specified"; } var vmtype = me.pveSelNode.data.type; if (!vmtype) { throw "no VM type specified"; } var vmtypeFilter; if (vmtype === 'lxc' || vmtype === 'openvz') { vmtypeFilter = function(item) { return PVE.Utils.volume_is_lxc_backup(item.data.volid, item.data.format); }; } else if (vmtype === 'qemu') { vmtypeFilter = function(item) { return PVE.Utils.volume_is_qemu_backup(item.data.volid, item.data.format); }; } else { throw "unsupported VM type '" + vmtype + "'"; } var searchFilter = { property: 'volid', value: '', anyMatch: true, caseSensitive: false }; var vmidFilter = { property: 'vmid', value: vmid, exactMatch: true, }; me.store = Ext.create('Ext.data.Store', { model: 'pve-storage-content', sorters: { property: 'volid', order: 'DESC' }, filters: [ vmtypeFilter, searchFilter, vmidFilter, ] }); let updateFilter = function() { me.store.filter([ vmtypeFilter, searchFilter, vmidFilter, ]); }; var reload = Ext.Function.createBuffered(function() { if (me.store) { me.store.load(); } }, 100); var setStorage = function(storage) { var url = '/api2/json/nodes/' + nodename + '/storage/' + storage + '/content'; url += '?content=backup'; me.store.setProxy({ type: 'proxmox', url: url }); reload(); }; var storagesel = Ext.create('PVE.form.StorageSelector', { nodename: nodename, fieldLabel: gettext('Storage'), labelAlign: 'right', storageContent: 'backup', allowBlank: false, listeners: { change: function(f, value) { let storage = f.getStore().findRecord('storage', value); if (storage) { let isPBS = storage.data.type === 'pbs'; me.getColumns().forEach((column) => { if (column.dataIndex === 'verification') { column.setHidden(!isPBS); } }); } setStorage(value); } } }); var storagefilter = Ext.create('Ext.form.field.Text', { fieldLabel: gettext('Search'), labelWidth: 50, labelAlign: 'right', enableKeyEvents: true, value: searchFilter.value, listeners: { buffer: 500, keyup: function(field) { me.store.clearFilter(true); searchFilter.value = field.getValue(); updateFilter(); } } }); var vmidfilterCB = Ext.create('Ext.form.field.Checkbox', { boxLabel: gettext('Filter VMID'), value: '1', listeners: { change: function(cb, value) { vmidFilter.value = value ? vmid : ''; vmidFilter.exactMatch = !!value; updateFilter(); }, }, }); var sm = Ext.create('Ext.selection.RowModel', {}); var backup_btn = Ext.create('Ext.button.Button', { text: gettext('Backup now'), handler: function() { var win = Ext.create('PVE.window.Backup', { nodename: nodename, vmid: vmid, vmtype: vmtype, storage: storagesel.getValue(), listeners : { close: function() { reload(); } } }); win.show(); } }); var restore_btn = Ext.create('Proxmox.button.Button', { text: gettext('Restore'), disabled: true, selModel: sm, enableFn: function(rec) { return !!rec; }, handler: function(b, e, rec) { let win = Ext.create('PVE.window.Restore', { nodename: nodename, vmid: vmid, volid: rec.data.volid, volidText: PVE.Utils.render_storage_content(rec.data.volid, {}, rec), vmtype: vmtype }); win.show(); win.on('destroy', reload); } }); var delete_btn = Ext.create('Proxmox.button.StdRemoveButton', { selModel: sm, dangerous: true, delay: 5, confirmMsg: function(rec) { var msg = Ext.String.format(gettext('Are you sure you want to remove entry {0}'), "'" + rec.data.volid + "'"); msg += " " + gettext('This will permanently erase all data.'); return msg; }, getUrl: function(rec) { var storage = storagesel.getValue(); return '/nodes/' + nodename + '/storage/' + storage + '/content/' + rec.data.volid; }, callback: function() { reload(); } }); var config_btn = Ext.create('Proxmox.button.Button', { text: gettext('Show Configuration'), disabled: true, selModel: sm, enableFn: function(rec) { return !!rec; }, handler: function(b, e, rec) { var storage = storagesel.getValue(); if (!storage) { return; } var win = Ext.create('PVE.window.BackupConfig', { volume: rec.data.volid, pveSelNode: me.pveSelNode }); win.show(); } }); Ext.apply(me, { selModel: sm, tbar: { overflowHandler: 'scroller', items: [ backup_btn, restore_btn, delete_btn,config_btn, '->', storagesel, '-', vmidfilterCB, storagefilter ], }, columns: [ { header: gettext('Name'), flex: 2, sortable: true, renderer: PVE.Utils.render_storage_content, dataIndex: 'volid' }, { header: gettext('Comment'), dataIndex: 'comment', flex: 1, renderer: Ext.htmlEncode, }, { header: gettext('Date'), width: 150, dataIndex: 'vdate' }, { header: gettext('Format'), width: 100, dataIndex: 'format' }, { header: gettext('Size'), width: 100, renderer: Proxmox.Utils.format_size, dataIndex: 'size' }, { header: gettext('VMID'), dataIndex: 'vmid', hidden: true, }, { header: gettext('Verify State'), dataIndex: 'verification', renderer: function(v) { let i = (cls, txt) => ` ${txt}`; if (v === undefined || v === null) { return i('question-circle-o warning', gettext('None')); } let tip = "" let txt = gettext('Failed'); let iconCls = 'times critical'; if (v.state === 'ok') { txt = gettext('OK'); iconCls = 'check good'; let now = Date.now() / 1000; let task = Proxmox.Utils.parse_task_upid(v.upid); let verify_time = Proxmox.Utils.render_timestamp(task.starttime); tip = `Last verify task started on ${verify_time}`; if (now - v.starttime > 30 * 24 * 60 * 60) { tip = `Last verify task over 30 days ago: ${verify_time}`; iconCls = 'check warning'; } } return ` ${i(iconCls, txt)} `; } } ] }); me.callParent(); } });