]> git.proxmox.com Git - pve-manager.git/blame - www/manager6/window/Migrate.js
ui: pci map edit: make top fields more clear
[pve-manager.git] / www / manager6 / window / Migrate.js
CommitLineData
04c32bed
DM
1Ext.define('PVE.window.Migrate', {
2 extend: 'Ext.window.Window',
3
9706707d
TM
4 vmtype: undefined,
5 nodename: undefined,
6 vmid: undefined,
8b4987e5 7 maxHeight: 450,
9706707d
TM
8
9 viewModel: {
10 data: {
11 vmid: undefined,
12 nodename: undefined,
13 vmtype: undefined,
14 running: false,
15 qemu: {
16 onlineHelp: 'qm_migration',
f6710aac 17 commonName: 'VM',
9706707d
TM
18 },
19 lxc: {
20 onlineHelp: 'pct_migration',
f6710aac 21 commonName: 'CT',
9706707d
TM
22 },
23 migration: {
24 possible: true,
25 preconditions: [],
26 'with-local-disks': 0,
27 mode: undefined,
4ddaf290
TM
28 allowedNodes: undefined,
29 overwriteLocalResourceCheck: false,
f6710aac
TL
30 hasLocalResources: false,
31 },
9706707d
TM
32
33 },
34
35 formulas: {
36 setMigrationMode: function(get) {
8058410f 37 if (get('running')) {
9706707d
TM
38 if (get('vmtype') === 'qemu') {
39 return gettext('Online');
40 } else {
41 return gettext('Restart Mode');
42 }
43 } else {
44 return gettext('Offline');
45 }
46 },
47 setStorageselectorHidden: function(get) {
48 if (get('migration.with-local-disks') && get('running')) {
49 return false;
50 } else {
51 return true;
52 }
4ddaf290
TM
53 },
54 setLocalResourceCheckboxHidden: function(get) {
55 if (get('running') || !get('migration.hasLocalResources') ||
56 Proxmox.UserName !== 'root@pam') {
57 return true;
58 } else {
59 return false;
60 }
f6710aac
TL
61 },
62 },
2df7ea9f 63 },
2df7ea9f
EK
64
65 controller: {
66 xclass: 'Ext.app.ViewController',
67 control: {
68 'panel[reference=formPanel]': {
69 validityChange: function(panel, isValid) {
9706707d
TM
70 this.getViewModel().set('migration.possible', isValid);
71 this.checkMigratePreconditions();
f6710aac
TL
72 },
73 },
9706707d
TM
74 },
75
76 init: function(view) {
77 var me = this,
78 vm = view.getViewModel();
79
80 if (!view.nodename) {
81 throw "missing custom view config: nodename";
82 }
83 vm.set('nodename', view.nodename);
84
85 if (!view.vmid) {
86 throw "missing custom view config: vmid";
87 }
88 vm.set('vmid', view.vmid);
89
90 if (!view.vmtype) {
91 throw "missing custom view config: vmtype";
92 }
93 vm.set('vmtype', view.vmtype);
94
9706707d 95 view.setTitle(
f6710aac 96 Ext.String.format('{0} {1} {2}', gettext('Migrate'), vm.get(view.vmtype).commonName, view.vmid),
9706707d
TM
97 );
98 me.lookup('proxmoxHelpButton').setHelpConfig({
f6710aac 99 onlineHelp: vm.get(view.vmtype).onlineHelp,
9706707d 100 });
9706707d 101 me.lookup('formPanel').isValid();
9706707d
TM
102 },
103
8058410f 104 onTargetChange: function(nodeSelector) {
d8d4c14b 105 // Always display the storages of the currently seleceted migration target
9706707d
TM
106 this.lookup('pveDiskStorageSelector').setNodename(nodeSelector.value);
107 this.checkMigratePreconditions();
108 },
109
110 startMigration: function() {
111 var me = this,
112 view = me.getView(),
113 vm = me.getViewModel();
114
115 var values = me.lookup('formPanel').getValues();
116 var params = {
f6710aac 117 target: values.target,
9706707d
TM
118 };
119
120 if (vm.get('migration.mode')) {
121 params[vm.get('migration.mode')] = 1;
122 }
123 if (vm.get('migration.with-local-disks')) {
124 params['with-local-disks'] = 1;
125 }
11c17a01
FE
126 //offline migration to a different storage currently might fail at a late stage
127 //(i.e. after some disks have been moved), so don't expose it yet in the GUI
128 if (vm.get('migration.with-local-disks') && vm.get('running') && values.targetstorage) {
9706707d
TM
129 params.targetstorage = values.targetstorage;
130 }
131
4ddaf290 132 if (vm.get('migration.overwriteLocalResourceCheck')) {
399ffa76 133 params.force = 1;
4ddaf290
TM
134 }
135
9706707d
TM
136 Proxmox.Utils.API2Request({
137 params: params,
138 url: '/nodes/' + vm.get('nodename') + '/' + vm.get('vmtype') + '/' + vm.get('vmid') + '/migrate',
139 waitMsgTarget: view,
140 method: 'POST',
141 failure: function(response, opts) {
142 Ext.Msg.alert(gettext('Error'), response.htmlStatus);
143 },
144 success: function(response, options) {
145 var upid = response.result.data;
146 var extraTitle = Ext.String.format(' ({0} ---> {1})', vm.get('nodename'), params.target);
147
148 Ext.create('Proxmox.window.TaskViewer', {
149 upid: upid,
f6710aac 150 extraTitle: extraTitle,
9706707d
TM
151 }).show();
152
153 view.close();
f6710aac 154 },
9706707d 155 });
9706707d
TM
156 },
157
4ddaf290 158 checkMigratePreconditions: function(resetMigrationPossible) {
9706707d
TM
159 var me = this,
160 vm = me.getViewModel();
161
9706707d
TM
162 var vmrec = PVE.data.ResourceStore.findRecord('vmid', vm.get('vmid'),
163 0, false, false, true);
164 if (vmrec && vmrec.data && vmrec.data.running) {
165 vm.set('running', true);
166 }
167
168 if (vm.get('vmtype') === 'qemu') {
4ddaf290 169 me.checkQemuPreconditions(resetMigrationPossible);
9706707d 170 } else {
4ddaf290 171 me.checkLxcPreconditions(resetMigrationPossible);
9706707d
TM
172 }
173 me.lookup('pveNodeSelector').disallowedNodes = [vm.get('nodename')];
174
175 // Only allow nodes where the local storage is available in case of offline migration
176 // where storage migration is not possible
177 me.lookup('pveNodeSelector').allowedNodes = vm.get('migration.allowedNodes');
178
179 me.lookup('formPanel').isValid();
9706707d
TM
180 },
181
d8d4c14b
TL
182 checkQemuPreconditions: async function(resetMigrationPossible) {
183 let me = this,
9706707d
TM
184 vm = me.getViewModel(),
185 migrateStats;
186
187 if (vm.get('running')) {
188 vm.set('migration.mode', 'online');
189 }
190
d8d4c14b 191 try {
685b52f5
TL
192 if (me.fetchingNodeMigrateInfo && me.fetchingNodeMigrateInfo === vm.get('nodename')) {
193 return;
194 }
195 me.fetchingNodeMigrateInfo = vm.get('nodename');
d8d4c14b
TL
196 let { result } = await Proxmox.Async.api2({
197 url: `/nodes/${vm.get('nodename')}/${vm.get('vmtype')}/${vm.get('vmid')}/migrate`,
198 method: 'GET',
199 });
200 migrateStats = result.data;
685b52f5 201 me.fetchingNodeMigrateInfo = false;
d8d4c14b
TL
202 } catch (error) {
203 Ext.Msg.alert(gettext('Error'), error.htmlStatus);
204 return;
205 }
2df7ea9f 206
d8d4c14b
TL
207 if (migrateStats.running) {
208 vm.set('running', true);
209 }
210 // Get migration object from viewmodel to prevent to many bind callbacks
211 let migration = vm.get('migration');
212 if (resetMigrationPossible) {
213 migration.possible = true;
214 }
215 migration.preconditions = [];
216
217 if (migrateStats.allowed_nodes) {
218 migration.allowedNodes = migrateStats.allowed_nodes;
219 let target = me.lookup('pveNodeSelector').value;
220 if (target.length && !migrateStats.allowed_nodes.includes(target)) {
221 let disallowed = migrateStats.not_allowed_nodes[target];
02f14aa8
DC
222 if (disallowed.unavailable_storages !== undefined) {
223 let missingStorages = disallowed.unavailable_storages.join(', ');
d8d4c14b 224
02f14aa8
DC
225 migration.possible = false;
226 migration.preconditions.push({
227 text: 'Storage (' + missingStorages + ') not available on selected target. ' +
228 'Start VM to use live storage migration or select other target node',
229 severity: 'error',
230 });
231 }
232
233 if (disallowed['unavailable-resources'] !== undefined) {
234 let unavailableResources = disallowed['unavailable-resources'].join(', ');
235
236 migration.possible = false;
237 migration.preconditions.push({
238 text: 'Mapped Resources (' + unavailableResources + ') not available on selected target. ',
239 severity: 'error',
240 });
241 }
d8d4c14b
TL
242 }
243 }
244
02f14aa8
DC
245 let blockingResources = [];
246 let mappedResources = migrateStats['mapped-resources'] ?? [];
247
248 for (const res of migrateStats.local_resources) {
249 if (mappedResources.indexOf(res) === -1) {
250 blockingResources.push(res);
251 }
252 }
253
254 if (blockingResources.length) {
d8d4c14b
TL
255 migration.hasLocalResources = true;
256 if (!migration.overwriteLocalResourceCheck || vm.get('running')) {
257 migration.possible = false;
258 migration.preconditions.push({
259 text: Ext.String.format('Can\'t migrate VM with local resources: {0}',
02f14aa8 260 blockingResources.join(', ')),
d8d4c14b
TL
261 severity: 'error',
262 });
263 } else {
264 migration.preconditions.push({
265 text: Ext.String.format('Migrate VM with local resources: {0}. ' +
266 'This might fail if resources aren\'t available on the target node.',
02f14aa8 267 blockingResources.join(', ')),
d8d4c14b
TL
268 severity: 'warning',
269 });
270 }
271 }
9706707d 272
02f14aa8
DC
273 if (mappedResources && mappedResources.length) {
274 if (vm.get('running')) {
275 migration.possible = false;
276 migration.preconditions.push({
277 text: Ext.String.format('Can\'t migrate running VM with mapped resources: {0}',
278 mappedResources.join(', ')),
279 severity: 'error',
280 });
281 }
282 }
283
d8d4c14b
TL
284 if (migrateStats.local_disks.length) {
285 migrateStats.local_disks.forEach(function(disk) {
286 if (disk.cdrom && disk.cdrom === 1) {
4d66fc7f 287 if (!disk.volid.includes('vm-' + vm.get('vmid') + '-cloudinit')) {
4ddaf290
TM
288 migration.possible = false;
289 migration.preconditions.push({
d8d4c14b 290 text: "Can't migrate VM with local CD/DVD",
f6710aac 291 severity: 'error',
4ddaf290 292 });
4ddaf290 293 }
d8d4c14b
TL
294 } else {
295 let size = disk.size ? '(' + Proxmox.Utils.render_size(disk.size) + ')' : '';
296 migration['with-local-disks'] = 1;
297 migration.preconditions.push({
298 text: Ext.String.format('Migration with local disk might take long: {0} {1}', disk.volid, size),
299 severity: 'warning',
9706707d 300 });
9706707d 301 }
d8d4c14b
TL
302 });
303 }
9706707d 304
d8d4c14b 305 vm.set('migration', migration);
9706707d 306 },
4ddaf290 307 checkLxcPreconditions: function(resetMigrationPossible) {
d8d4c14b 308 let vm = this.getViewModel();
9706707d
TM
309 if (vm.get('running')) {
310 vm.set('migration.mode', 'restart');
04c32bed 311 }
f6710aac 312 },
2df7ea9f 313 },
04c32bed 314
3188b0b7 315 width: 600,
2df7ea9f 316 modal: true,
9706707d
TM
317 layout: {
318 type: 'vbox',
f6710aac 319 align: 'stretch',
9706707d 320 },
2df7ea9f 321 border: false,
2df7ea9f
EK
322 items: [
323 {
324 xtype: 'form',
325 reference: 'formPanel',
04c32bed
DM
326 bodyPadding: 10,
327 border: false,
d27a44a6 328 layout: 'hbox',
04c32bed
DM
329 items: [
330 {
9706707d 331 xtype: 'container',
d27a44a6 332 flex: 1,
9706707d 333 items: [{
090dff2f
TL
334 xtype: 'displayfield',
335 name: 'source',
336 fieldLabel: gettext('Source node'),
337 bind: {
f6710aac
TL
338 value: '{nodename}',
339 },
9706707d
TM
340 },
341 {
342 xtype: 'displayfield',
343 reference: 'migrationMode',
344 fieldLabel: gettext('Mode'),
345 bind: {
f6710aac
TL
346 value: '{setMigrationMode}',
347 },
348 }],
04c32bed
DM
349 },
350 {
9706707d 351 xtype: 'container',
d27a44a6 352 flex: 1,
090dff2f
TL
353 items: [{
354 xtype: 'pveNodeSelector',
355 reference: 'pveNodeSelector',
356 name: 'target',
357 fieldLabel: gettext('Target node'),
358 allowBlank: false,
359 disallowedNodes: undefined,
360 onlineValidator: true,
361 listeners: {
f6710aac
TL
362 change: 'onTargetChange',
363 },
090dff2f
TL
364 },
365 {
366 xtype: 'pveStorageSelector',
367 reference: 'pveDiskStorageSelector',
368 name: 'targetstorage',
369 fieldLabel: gettext('Target storage'),
370 storageContent: 'images',
11c17a01
FE
371 allowBlank: true,
372 autoSelect: false,
cd338fe1 373 emptyText: gettext('Current layout'),
090dff2f 374 bind: {
f6710aac
TL
375 hidden: '{setStorageselectorHidden}',
376 },
4ddaf290
TM
377 },
378 {
379 xtype: 'proxmoxcheckbox',
380 name: 'overwriteLocalResourceCheck',
381 fieldLabel: gettext('Force'),
382 autoEl: {
383 tag: 'div',
f6710aac 384 'data-qtip': 'Overwrite local resources unavailable check',
4ddaf290
TM
385 },
386 bind: {
387 hidden: '{setLocalResourceCheckboxHidden}',
f6710aac 388 value: '{migration.overwriteLocalResourceCheck}',
4ddaf290
TM
389 },
390 listeners: {
d8d4c14b
TL
391 change: {
392 fn: 'checkMigratePreconditions',
393 extraArg: true,
394 },
f6710aac
TL
395 },
396 }],
397 },
398 ],
9706707d
TM
399 },
400 {
401 xtype: 'gridpanel',
402 reference: 'preconditionGrid',
b4533390 403 selectable: false,
9706707d 404 flex: 1,
b4533390
TL
405 columns: [{
406 text: '',
407 dataIndex: 'severity',
408 renderer: function(v) {
409 switch (v) {
410 case 'warning':
411 return '<i class="fa fa-exclamation-triangle warning"></i> ';
412 case 'error':
413 return '<i class="fa fa-times critical"></i>';
414 default:
415 return v;
416 }
417 },
f6710aac 418 width: 35,
b4533390
TL
419 },
420 {
421 text: 'Info',
422 dataIndex: 'text',
423 cellWrap: true,
f6710aac 424 flex: 1,
b4533390 425 }],
9706707d
TM
426 bind: {
427 hidden: '{!migration.preconditions.length}',
428 store: {
cd338fe1
TL
429 fields: ['severity', 'text'],
430 data: '{migration.preconditions}',
431 sorters: 'text',
f6710aac
TL
432 },
433 },
434 },
9706707d 435
2df7ea9f
EK
436 ],
437 buttons: [
438 {
672a6270
TL
439 xtype: 'proxmoxHelpButton',
440 reference: 'proxmoxHelpButton',
2df7ea9f
EK
441 onlineHelp: 'pct_migration',
442 listenToGlobalEvent: false,
f6710aac 443 hidden: false,
2df7ea9f
EK
444 },
445 '->',
446 {
447 xtype: 'button',
448 reference: 'submitButton',
9706707d
TM
449 text: gettext('Migrate'),
450 handler: 'startMigration',
451 bind: {
f6710aac
TL
452 disabled: '{!migration.possible}',
453 },
454 },
455 ],
3188b0b7 456});