]>
Commit | Line | Data |
---|---|---|
c7ee0c11 | 1 | Ext.define('PVE.lxc.RessourceView', { |
9fd8d73e | 2 | extend: 'Proxmox.grid.PendingObjectGrid', |
c7ee0c11 DM |
3 | alias: ['widget.pveLxcRessourceView'], |
4 | ||
ba93a9c6 DC |
5 | onlineHelp: 'pct_configuration', |
6 | ||
c7ee0c11 | 7 | renderKey: function(key, metaData, rec, rowIndex, colIndex, store) { |
02ecbc98 TL |
8 | let me = this; |
9 | let rowdef = me.rows[key] || {}; | |
10 | ||
11 | let txt = rowdef.header || key; | |
12 | let icon = ''; | |
c7ee0c11 DM |
13 | |
14 | metaData.tdAttr = "valign=middle"; | |
c7ee0c11 DM |
15 | if (rowdef.tdCls) { |
16 | metaData.tdCls = rowdef.tdCls; | |
02ecbc98 TL |
17 | } else if (rowdef.iconCls) { |
18 | icon = `<i class='pve-grid-fa fa fa-fw fa-${rowdef.iconCls}'></i>`; | |
19 | metaData.tdCls += " pve-itype-fa"; | |
20 | } | |
21 | // only return icons in grid but not remove dialog | |
22 | if (rowIndex !== undefined) { | |
23 | return icon + txt; | |
24 | } else { | |
25 | return txt; | |
c7ee0c11 | 26 | } |
c7ee0c11 DM |
27 | }, |
28 | ||
8058410f | 29 | initComponent: function() { |
c7ee0c11 | 30 | var me = this; |
55ee6ba1 | 31 | let confid; |
c7ee0c11 DM |
32 | |
33 | var nodename = me.pveSelNode.data.node; | |
2a4971d8 | 34 | if (!nodename) { |
c7ee0c11 DM |
35 | throw "no node name specified"; |
36 | } | |
37 | ||
38 | var vmid = me.pveSelNode.data.vmid; | |
39 | if (!vmid) { | |
40 | throw "no VM ID specified"; | |
41 | } | |
42 | ||
43 | var caps = Ext.state.Manager.get('GuiCap'); | |
d35b5b2a | 44 | var diskCap = caps.vms['VM.Config.Disk']; |
c7ee0c11 DM |
45 | |
46 | var mpeditor = caps.vms['VM.Config.Disk'] ? 'PVE.lxc.MountPointEdit' : undefined; | |
47 | ||
5a2e333c FE |
48 | const nodeInfo = PVE.data.ResourceStore.getNodes().find(node => node.node === nodename); |
49 | let cpuEditor = { | |
50 | xtype: 'pveLxcCPUEdit', | |
51 | cgroupMode: nodeInfo['cgroup-mode'], | |
52 | }; | |
53 | ||
c7ee0c11 DM |
54 | var rows = { |
55 | memory: { | |
56 | header: gettext('Memory'), | |
57 | editor: caps.vms['VM.Config.Memory'] ? 'PVE.lxc.MemoryEdit' : undefined, | |
c7ee0c11 | 58 | defaultValue: 512, |
a8ea1b68 | 59 | tdCls: 'pmx-itype-icon-memory', |
1a2fdf62 | 60 | group: 1, |
c7ee0c11 | 61 | renderer: function(value) { |
e7ade592 | 62 | return Proxmox.Utils.format_size(value*1024*1024); |
f6710aac | 63 | }, |
c7ee0c11 DM |
64 | }, |
65 | swap: { | |
66 | header: gettext('Swap'), | |
67 | editor: caps.vms['VM.Config.Memory'] ? 'PVE.lxc.MemoryEdit' : undefined, | |
c7ee0c11 | 68 | defaultValue: 512, |
809f6b6e | 69 | iconCls: 'refresh', |
1a2fdf62 | 70 | group: 2, |
c7ee0c11 | 71 | renderer: function(value) { |
e7ade592 | 72 | return Proxmox.Utils.format_size(value*1024*1024); |
f6710aac | 73 | }, |
c7ee0c11 | 74 | }, |
92b5029f DM |
75 | cores: { |
76 | header: gettext('Cores'), | |
5a2e333c | 77 | editor: caps.vms['VM.Config.CPU'] ? cpuEditor : undefined, |
92b5029f | 78 | defaultValue: '', |
a8ea1b68 | 79 | tdCls: 'pmx-itype-icon-processor', |
1a2fdf62 | 80 | group: 3, |
92b5029f | 81 | renderer: function(value) { |
3e4b78dc DC |
82 | var cpulimit = me.getObjectValue('cpulimit'); |
83 | var cpuunits = me.getObjectValue('cpuunits'); | |
84 | var res; | |
85 | if (value) { | |
86 | res = value; | |
87 | } else { | |
88 | res = gettext('unlimited'); | |
89 | } | |
90 | ||
91 | if (cpulimit) { | |
92 | res += ' [cpulimit=' + cpulimit + ']'; | |
93 | } | |
94 | ||
95 | if (cpuunits) { | |
96 | res += ' [cpuunits=' + cpuunits + ']'; | |
97 | } | |
98 | return res; | |
f6710aac | 99 | }, |
92b5029f | 100 | }, |
c7ee0c11 DM |
101 | rootfs: { |
102 | header: gettext('Root Disk'), | |
e7ade592 | 103 | defaultValue: Proxmox.Utils.noneText, |
c7ee0c11 | 104 | editor: mpeditor, |
809f6b6e | 105 | iconCls: 'hdd-o', |
f6710aac | 106 | group: 4, |
c15e9cd5 | 107 | }, |
3e4b78dc | 108 | cpulimit: { |
f6710aac | 109 | visible: false, |
3e4b78dc DC |
110 | }, |
111 | cpuunits: { | |
f6710aac | 112 | visible: false, |
3e4b78dc | 113 | }, |
c15e9cd5 | 114 | unprivileged: { |
f6710aac TL |
115 | visible: false, |
116 | }, | |
c7ee0c11 DM |
117 | }; |
118 | ||
14a845bc DC |
119 | PVE.Utils.forEachMP(function(bus, i) { |
120 | confid = bus + i; | |
1a2fdf62 DL |
121 | var group = 5; |
122 | var header; | |
14a845bc DC |
123 | if (bus === 'mp') { |
124 | header = gettext('Mount Point') + ' (' + confid + ')'; | |
125 | } else { | |
126 | header = gettext('Unused Disk') + ' ' + i; | |
1a2fdf62 | 127 | group += 1; |
14a845bc | 128 | } |
c7ee0c11 | 129 | rows[confid] = { |
1a2fdf62 DL |
130 | group: group, |
131 | order: i, | |
c7ee0c11 DM |
132 | tdCls: 'pve-itype-icon-storage', |
133 | editor: mpeditor, | |
f6710aac | 134 | header: header, |
c7ee0c11 | 135 | }; |
14a845bc | 136 | }, true); |
c7ee0c11 | 137 | |
c7ee0c11 DM |
138 | var baseurl = 'nodes/' + nodename + '/lxc/' + vmid + '/config'; |
139 | ||
14dd743b | 140 | me.selModel = Ext.create('Ext.selection.RowModel', {}); |
c7ee0c11 DM |
141 | |
142 | var run_resize = function() { | |
14dd743b | 143 | var rec = me.selModel.getSelection()[0]; |
c7ee0c11 DM |
144 | if (!rec) { |
145 | return; | |
146 | } | |
147 | ||
148 | var win = Ext.create('PVE.window.MPResize', { | |
149 | disk: rec.data.key, | |
150 | nodename: nodename, | |
f6710aac | 151 | vmid: vmid, |
c7ee0c11 DM |
152 | }); |
153 | ||
154 | win.show(); | |
c7ee0c11 DM |
155 | }; |
156 | ||
157 | var run_remove = function(b, e, rec) { | |
e7ade592 | 158 | Proxmox.Utils.API2Request({ |
c7ee0c11 DM |
159 | url: '/api2/extjs/' + baseurl, |
160 | waitMsgTarget: me, | |
161 | method: 'PUT', | |
162 | params: { | |
f6710aac | 163 | 'delete': rec.data.key, |
c7ee0c11 | 164 | }, |
8058410f | 165 | failure: function(response, opts) { |
c7ee0c11 | 166 | Ext.Msg.alert('Error', response.htmlStatus); |
f6710aac | 167 | }, |
c7ee0c11 DM |
168 | }); |
169 | }; | |
170 | ||
a8d854af AL |
171 | let run_move = function() { |
172 | let rec = me.selModel.getSelection()[0]; | |
c7164db7 DC |
173 | if (!rec) { |
174 | return; | |
175 | } | |
176 | ||
177 | var win = Ext.create('PVE.window.HDMove', { | |
178 | disk: rec.data.key, | |
179 | nodename: nodename, | |
180 | vmid: vmid, | |
f6710aac | 181 | type: 'lxc', |
c7164db7 DC |
182 | }); |
183 | ||
184 | win.show(); | |
185 | ||
186 | win.on('destroy', me.reload, me); | |
187 | }; | |
188 | ||
a8d854af AL |
189 | let run_reassign = function() { |
190 | let rec = me.selModel.getSelection()[0]; | |
191 | if (!rec) { | |
192 | return; | |
193 | } | |
194 | ||
3bde324f | 195 | Ext.create('PVE.window.GuestDiskReassign', { |
a8d854af AL |
196 | disk: rec.data.key, |
197 | nodename: nodename, | |
198 | autoShow: true, | |
199 | vmid: vmid, | |
200 | type: 'lxc', | |
201 | listeners: { | |
202 | destroy: () => me.reload(), | |
203 | }, | |
204 | }); | |
205 | }; | |
206 | ||
5720fafa | 207 | var edit_btn = new Proxmox.button.Button({ |
c7ee0c11 | 208 | text: gettext('Edit'), |
14dd743b | 209 | selModel: me.selModel, |
c7ee0c11 DM |
210 | disabled: true, |
211 | enableFn: function(rec) { | |
212 | if (!rec) { | |
213 | return false; | |
214 | } | |
215 | var rowdef = rows[rec.data.key]; | |
216 | return !!rowdef.editor; | |
217 | }, | |
f6710aac | 218 | handler: function() { me.run_editor(); }, |
c7ee0c11 DM |
219 | }); |
220 | ||
5720fafa | 221 | var remove_btn = new Proxmox.button.Button({ |
c7ee0c11 | 222 | text: gettext('Remove'), |
986ccfe4 AL |
223 | defaultText: gettext('Remove'), |
224 | altText: gettext('Detach'), | |
14dd743b | 225 | selModel: me.selModel, |
c7ee0c11 DM |
226 | disabled: true, |
227 | dangerous: true, | |
228 | confirmMsg: function(rec) { | |
986ccfe4 AL |
229 | let warn = Ext.String.format(gettext('Are you sure you want to remove entry {0}')); |
230 | if (this.text === this.altText) { | |
231 | warn = gettext('Are you sure you want to detach entry {0}'); | |
232 | } | |
cc80f765 TL |
233 | let rendered = me.renderKey(rec.data.key, {}, rec); |
234 | let msg = Ext.String.format(warn, `'${rendered}'`); | |
986ccfe4 | 235 | |
c7ee0c11 | 236 | if (rec.data.key.match(/^unused\d+$/)) { |
16152937 | 237 | msg += " " + gettext('This will permanently erase all data.'); |
c7ee0c11 | 238 | } |
c7ee0c11 DM |
239 | return msg; |
240 | }, | |
f6710aac | 241 | handler: run_remove, |
986ccfe4 AL |
242 | listeners: { |
243 | render: function(btn) { | |
244 | // hack: calculate the max button width on first display to prevent the whole | |
245 | // toolbar to move when we switch between the "Remove" and "Detach" labels | |
246 | let def = btn.getSize().width; | |
247 | ||
248 | btn.setText(btn.altText); | |
249 | let alt = btn.getSize().width; | |
250 | ||
251 | btn.setText(btn.defaultText); | |
252 | ||
253 | let optimal = alt > def ? alt : def; | |
254 | btn.setSize({ width: optimal }); | |
255 | }, | |
256 | }, | |
c7ee0c11 DM |
257 | }); |
258 | ||
a8d854af AL |
259 | let move_menuitem = new Ext.menu.Item({ |
260 | text: gettext('Move Storage'), | |
261 | tooltip: gettext('Move volume to another storage'), | |
262 | iconCls: 'fa fa-database', | |
c7164db7 | 263 | selModel: me.selModel, |
f6710aac | 264 | handler: run_move, |
c7164db7 DC |
265 | }); |
266 | ||
a8d854af AL |
267 | let reassign_menuitem = new Ext.menu.Item({ |
268 | text: gettext('Reassign Owner'), | |
269 | tooltip: gettext('Reassign volume to another CT'), | |
270 | iconCls: 'fa fa-cube', | |
271 | handler: run_reassign, | |
272 | reference: 'reassing_item', | |
273 | }); | |
274 | ||
275 | let resize_menuitem = new Ext.menu.Item({ | |
276 | text: gettext('Resize'), | |
277 | iconCls: 'fa fa-plus', | |
278 | selModel: me.selModel, | |
279 | handler: run_resize, | |
280 | }); | |
281 | ||
282 | let volumeaction_btn = new Proxmox.button.Button({ | |
283 | text: gettext('Volume Action'), | |
284 | disabled: true, | |
285 | menu: { | |
286 | items: [ | |
287 | move_menuitem, | |
288 | reassign_menuitem, | |
289 | resize_menuitem, | |
290 | ], | |
291 | }, | |
292 | }); | |
293 | ||
716c3043 | 294 | let revert_btn = new PVE.button.PendingRevert(); |
273b5ce3 | 295 | |
716c3043 TL |
296 | let set_button_status = function() { |
297 | let rec = me.selModel.getSelection()[0]; | |
c7ee0c11 DM |
298 | |
299 | if (!rec) { | |
300 | edit_btn.disable(); | |
301 | remove_btn.disable(); | |
a8d854af | 302 | volumeaction_btn.disable(); |
273b5ce3 | 303 | revert_btn.disable(); |
c7ee0c11 DM |
304 | return; |
305 | } | |
716c3043 TL |
306 | let { key, value, 'delete': isDelete } = rec.data; |
307 | let rowdef = rows[key]; | |
c7ee0c11 | 308 | |
716c3043 | 309 | let pending = isDelete || me.hasPendingChanges(key); |
a8d854af AL |
310 | let isRootFS = key === 'rootfs'; |
311 | let isDisk = isRootFS || key.match(/^(mp|unused)\d+/); | |
716c3043 TL |
312 | let isUnusedDisk = key.match(/^unused\d+/); |
313 | let isUsedDisk = isDisk && !isUnusedDisk; | |
c7ee0c11 | 314 | |
716c3043 | 315 | let noedit = isDelete || !rowdef.editor; |
35a04562 | 316 | if (!noedit && Proxmox.UserName !== 'root@pam' && key.match(/^mp\d+$/)) { |
716c3043 | 317 | let mp = PVE.Parser.parseLxcMountPoint(value); |
c7ee0c11 DM |
318 | if (mp.type !== 'volume') { |
319 | noedit = true; | |
320 | } | |
321 | } | |
322 | edit_btn.setDisabled(noedit); | |
323 | ||
a8d854af AL |
324 | volumeaction_btn.setDisabled(!isDisk || !diskCap); |
325 | move_menuitem.setDisabled(isUnusedDisk); | |
326 | reassign_menuitem.setDisabled(isRootFS); | |
327 | resize_menuitem.setDisabled(isUnusedDisk); | |
328 | ||
329 | remove_btn.setDisabled(!isDisk || isRootFS || !diskCap || pending); | |
273b5ce3 | 330 | revert_btn.setDisabled(!pending); |
986ccfe4 AL |
331 | |
332 | remove_btn.setText(isUsedDisk ? remove_btn.altText : remove_btn.defaultText); | |
c7ee0c11 | 333 | }; |
2a4971d8 | 334 | |
716c3043 TL |
335 | let sorterFn = function(rec1, rec2) { |
336 | let v1 = rec1.data.key, v2 = rec2.data.key; | |
1a2fdf62 | 337 | |
716c3043 | 338 | let g1 = rows[v1].group || 0, g2 = rows[v2].group || 0; |
53e3ea84 | 339 | if (g1 - g2 !== 0) { |
1a2fdf62 DL |
340 | return g1 - g2; |
341 | } | |
342 | ||
716c3043 | 343 | let order1 = rows[v1].order || 0, order2 = rows[v2].order || 0; |
53e3ea84 | 344 | if (order1 - order2 !== 0) { |
1a2fdf62 DL |
345 | return order1 - order2; |
346 | } | |
347 | ||
348 | if (v1 > v2) { | |
349 | return 1; | |
350 | } else if (v1 < v2) { | |
351 | return -1; | |
352 | } else { | |
353 | return 0; | |
354 | } | |
93bd0d75 | 355 | }; |
1a2fdf62 | 356 | |
f7993618 | 357 | Ext.apply(me, { |
716c3043 | 358 | url: `/api2/json/nodes/${nodename}/lxc/${vmid}/pending`, |
14dd743b TL |
359 | selModel: me.selModel, |
360 | interval: 2000, | |
c7ee0c11 DM |
361 | cwidth1: 170, |
362 | tbar: [ | |
363 | { | |
364 | text: gettext('Add'), | |
365 | menu: new Ext.menu.Menu({ | |
366 | items: [ | |
367 | { | |
368 | text: gettext('Mount Point'), | |
809f6b6e | 369 | iconCls: 'fa fa-fw fa-hdd-o black', |
c7ee0c11 DM |
370 | disabled: !caps.vms['VM.Config.Disk'], |
371 | handler: function() { | |
163d9f17 TL |
372 | Ext.create('PVE.lxc.MountPointEdit', { |
373 | autoShow: true, | |
374 | url: `/api2/extjs/${baseurl}`, | |
c15e9cd5 | 375 | unprivileged: me.getObjectValue('unprivileged'), |
f6710aac | 376 | pveSelNode: me.pveSelNode, |
163d9f17 TL |
377 | listeners: { |
378 | destroy: () => me.reload(), | |
379 | }, | |
c7ee0c11 | 380 | }); |
f6710aac TL |
381 | }, |
382 | }, | |
383 | ], | |
384 | }), | |
c7ee0c11 DM |
385 | }, |
386 | edit_btn, | |
387 | remove_btn, | |
a8d854af | 388 | volumeaction_btn, |
f6710aac | 389 | revert_btn, |
c7ee0c11 DM |
390 | ], |
391 | rows: rows, | |
1a2fdf62 | 392 | sorterFn: sorterFn, |
14dd743b TL |
393 | editorConfig: { |
394 | pveSelNode: me.pveSelNode, | |
f6710aac | 395 | url: '/api2/extjs/' + baseurl, |
14dd743b | 396 | }, |
c7ee0c11 | 397 | listeners: { |
14dd743b | 398 | itemdblclick: me.run_editor, |
f6710aac TL |
399 | selectionchange: set_button_status, |
400 | }, | |
c7ee0c11 DM |
401 | }); |
402 | ||
403 | me.callParent(); | |
14dd743b TL |
404 | |
405 | me.on('activate', me.rstore.startUpdate); | |
406 | me.on('destroy', me.rstore.stopUpdate); | |
407 | me.on('deactivate', me.rstore.stopUpdate); | |
408 | ||
69f36699 DC |
409 | me.mon(me.getStore(), 'datachanged', function() { |
410 | set_button_status(); | |
411 | }); | |
412 | ||
14dd743b | 413 | Ext.apply(me.editorConfig, { unprivileged: me.getObjectValue('unprivileged') }); |
f6710aac | 414 | }, |
c7ee0c11 | 415 | }); |