]>
git.proxmox.com Git - pve-manager.git/blob - www/manager6/window/GuestImport.js
1 Ext
.define('PVE.window.GuestImport', {
2 extend
: 'Proxmox.window.Edit', // fixme: Proxmox.window.Edit?
3 alias
: 'widget.pveGuestImportWindow',
5 title
: gettext('Import Guest'),
7 submitUrl: function() {
9 return `/nodes/${me.nodename}/qemu`;
14 submitText
: gettext('Import'),
18 loadUrl: function(_url
, { storage
, nodename
, volumeName
}) {
19 let args
= Ext
.Object
.toQueryString({ volume
: volumeName
});
20 return `/nodes/${nodename}/storage/${storage}/import-metadata?${args}`;
24 xclass
: 'Ext.app.ViewController',
26 setNodename: function(_column
, widget
) {
28 let view
= me
.getView();
29 widget
.setNodename(view
.nodename
);
32 diskStorageChange: function(storageSelector
, value
) {
35 let grid
= me
.lookup('diskGrid');
36 let rec
= storageSelector
.getWidgetRecord();
37 let validFormats
= storageSelector
.store
.getById(value
)?.data
.format
;
38 grid
.query('pveDiskFormatSelector').some((selector
) => {
39 if (selector
.getWidgetRecord().data
.id
!== rec
.data
.id
) {
43 if (validFormats
?.[0]?.qcow2
) {
44 selector
.setDisabled(false);
45 selector
.setValue('qcow2');
47 selector
.setValue('raw');
48 selector
.setDisabled(true);
55 isoStorageChange: function(storageSelector
, value
) {
58 let grid
= me
.lookup('cdGrid');
59 let rec
= storageSelector
.getWidgetRecord();
60 grid
.query('pveFileSelector').some((selector
) => {
61 if (selector
.getWidgetRecord().data
.id
!== rec
.data
.id
) {
65 selector
.setStorage(value
);
67 selector
.setValue('');
74 onOSBaseChange: function(_field
, value
) {
76 let ostype
= me
.lookup('ostype');
77 let store
= ostype
.getStore();
78 store
.setData(PVE
.Utils
.kvm_ostypes
[value
]);
79 let old_val
= ostype
.getValue();
80 if (old_val
&& store
.find('val', old_val
) !== -1) {
81 ostype
.setValue(old_val
);
83 ostype
.setValue(store
.getAt(0));
87 calculateConfig: function() {
89 let inputPanel
= me
.lookup('mainInputPanel');
90 let summaryGrid
= me
.lookup('summaryGrid');
91 let values
= inputPanel
.getValues();
92 summaryGrid
.getStore().setData(Object
.entries(values
).map(([key
, value
]) => ({ key
, value
})));
95 calculateAdditionalCDIdx: function() {
98 let maxIde
= me
.getMaxControllerId('ide');
99 let maxSata
= me
.getMaxControllerId('sata');
100 // only ide0 and ide2 can be used reliably for isos (e.g. for q35)
107 if (maxSata
< PVE
.Utils
.diskControllerMaxIDs
.sata
- 1) {
108 return `sata${maxSata+1}`;
114 // assume assigned sata disks indices are continuous, so without holes
115 getMaxControllerId: function(controller
) {
117 let view
= me
.getView();
122 let max
= view
[`max${controller}`];
123 if (max
!== undefined) {
128 for (const key
of Object
.keys(me
.getView().vmConfig
)) {
129 if (!key
.toLowerCase().startsWith(controller
)) {
132 let idx
= parseInt(key
.slice(controller
.length
), 10);
137 me
.lookup('diskGrid').getStore().each(rec
=> {
138 if (!rec
.data
.id
.toLowerCase().startsWith(controller
)) {
141 let idx
= parseInt(rec
.data
.id
.slice(controller
.length
), 10);
146 me
.lookup('cdGrid').getStore().each(rec
=> {
147 if (!rec
.data
.id
.toLowerCase().startsWith(controller
) || rec
.data
.hidden
) {
150 let idx
= parseInt(rec
.data
.id
.slice(controller
.length
), 10);
156 view
[`max${controller}`] = max
;
160 mapDisk: function(value
, metaData
) {
162 let mapSata
= me
.lookup('mapSata');
163 if (mapSata
.isDisabled() || !mapSata
.getValue()) {
166 if (!value
.toLowerCase().startsWith('scsi')) {
169 let offset
= parseInt(value
.slice(4), 10);
170 let newIdx
= offset
+ me
.getMaxControllerId('sata') + 1;
171 if (me
.getViewModel().get('isWindows') && me
.getView().additionalCdIdx
?.startsWith('sata')) {
172 // additionalCdIdx takes the highest sata port
175 if (newIdx
>= PVE
.Utils
.diskControllerMaxIDs
.sata
) {
177 if (metaData
!== undefined) {
178 // we're in the renderer so put a warning here
179 let warning
= gettext('Too many disks, could not map to SATA.');
180 prefix
= `<i data-qtip="${warning}" class="fa fa-exclamation-triangle warning"></i> `;
182 return `${prefix}${value}`;
184 return `sata${newIdx}`;
187 refreshGrids: function() {
188 this.lookup('diskGrid').reconfigure();
189 this.lookup('cdGrid').reconfigure();
190 this.lookup('netGrid').reconfigure();
193 onOSTypeChange: function(_cb
, value
) {
198 let store
= me
.lookup('cdGrid').getStore();
199 let collection
= store
.getData().getSource() ?? store
.getData();
200 let rec
= collection
.find('autogenerated', true);
202 let isWindows
= (value
?? '').startsWith('w');
204 rec
.set('hidden', !isWindows
);
207 let prepareVirtio
= me
.lookup('mapSata').getValue();
208 let defaultScsiHw
= me
.getView().vmConfig
.scsihw
?? '__default__';
209 me
.lookup('scsihw').setValue(prepareVirtio
&& isWindows
? 'virtio-scsi-single' : defaultScsiHw
);
214 onPrepareVirtioChange: function(_cb
, value
) {
217 let scsihw
= me
.lookup('scsihw');
218 scsihw
.suspendEvents();
219 scsihw
.setValue(value
? 'virtio-scsi-single' : me
.getView().vmConfig
.scsihw
);
220 scsihw
.resumeEvents();
225 onScsiHwChange: function(_field
, value
) {
227 me
.getView().vmConfig
.scsihw
= value
;
230 onUniqueMACChange: function(_cb
, value
) {
233 me
.getViewModel().set('uniqueMACAdresses', value
);
235 me
.lookup('netGrid').reconfigure();
238 renderMacAddress: function(value
, metaData
, record
, rowIndex
, colIndex
, store
, view
) {
240 let vm
= me
.getViewModel();
242 return !vm
.get('uniqueMACAdresses') && value
? value
: 'auto';
247 // update records from widgetcolumns
248 change: function(widget
, value
) {
249 let rec
= widget
.getWidgetRecord();
250 rec
.set(widget
.name
, value
);
254 'grid[reference=diskGrid] pveStorageSelector': {
255 change
: 'diskStorageChange',
257 'grid[reference=cdGrid] pveStorageSelector': {
258 change
: 'isoStorageChange',
260 'field[name=osbase]': {
261 change
: 'onOSBaseChange',
263 'panel[reference=summaryTab]': {
264 activate
: 'calculateConfig',
266 'proxmoxcheckbox[reference=mapSata]': {
267 change
: 'onPrepareVirtioChange',
269 'combobox[name=ostype]': {
270 change
: 'onOSTypeChange',
272 'pveScsiHwSelector': {
273 change
: 'onScsiHwChange',
275 'proxmoxcheckbox[name=uniqueMACs]': {
276 change
: 'onUniqueMACChange',
288 uniqueMACAdresses
: false,
293 totalCoreCount
: get => get('socketCount') * get('coreCount'),
294 hideWarnings
: get => get('warnings').length
=== 0,
295 warningsText
: get => '<ul style="margin: 0; padding-left: 20px;">'
296 + get('warnings').map(w
=> `<li>${w}</li>`).join('') + '</ul>',
297 liveImportNote
: get => !get('liveImport') ? ''
298 : gettext('Note: If anything goes wrong during the live-import, new data written by the VM may be lost.'),
299 isWindows
: get => (get('os') ?? '').startsWith('w'),
314 title
: gettext('General'),
316 reference
: 'mainInputPanel',
317 onGetValues: function(values
) {
319 let grid
= me
.up('pveGuestImportWindow');
320 let vm
= grid
.getViewModel();
322 // from pveDiskStorageSelector
323 let defaultStorage
= values
.hdstorage
;
324 let defaultFormat
= values
.diskformat
;
325 delete values
.hdstorage
;
326 delete values
.diskformat
;
328 let defaultBridge
= values
.defaultBridge
;
329 delete values
.defaultBridge
;
331 let config
= Ext
.apply(grid
.vmConfig
, values
);
334 config
.scsi0
= config
.scsi0
.replace('local:0,', 'local:0,format=qcow2,');
337 let parsedBoot
= PVE
.Parser
.parsePropertyString(config
.boot
?? '');
338 if (parsedBoot
.order
) {
339 parsedBoot
.order
= parsedBoot
.order
.split(';');
342 grid
.lookup('diskGrid').getStore().each((rec
) => {
343 if (!rec
.data
.enable
) {
346 let id
= grid
.getController().mapDisk(rec
.data
.id
);
347 if (id
!== rec
.data
.id
&& parsedBoot
?.order
) {
348 let idx
= parsedBoot
.order
.indexOf(rec
.data
.id
);
350 parsedBoot
.order
[idx
] = id
;
360 data
.file
= defaultStorage
;
361 data
.format
= defaultFormat
;
363 data
.file
+= ':0'; // for our special api format
364 if (id
=== 'efidisk0') {
365 delete data
['import-from'];
367 config
[id
] = PVE
.Parser
.printQemuDrive(data
);
370 if (parsedBoot
.order
) {
371 parsedBoot
.order
= parsedBoot
.order
.join(';');
373 config
.boot
= PVE
.Parser
.printPropertyString(parsedBoot
);
375 grid
.lookup('netGrid').getStore().each((rec
) => {
376 if (!rec
.data
.enable
) {
379 let id
= rec
.data
.id
;
386 data
.bridge
= defaultBridge
;
388 if (vm
.get('uniqueMACAdresses')) {
389 data
.macaddr
= undefined;
391 config
[id
] = PVE
.Parser
.printQemuNetwork(data
);
394 grid
.lookup('cdGrid').getStore().each((rec
) => {
395 if (!rec
.data
.enable
) {
398 let id
= rec
.data
.id
;
401 file
: rec
.data
.file
? rec
.data
.file
: 'none',
403 config
[id
] = PVE
.Parser
.printPropertyString(cd
);
406 config
.scsihw
= grid
.lookup('scsihw').getValue();
408 if (grid
.lookup('liveimport').getValue()) {
409 config
['live-restore'] = 1;
412 // remove __default__ values
413 for (const [key
, value
] of Object
.entries(config
)) {
414 if (value
=== '__default__') {
424 xtype
: 'pveGuestIDSelector',
428 loadNextFreeID
: true,
431 xtype
: 'proxmoxintegerfield',
432 fieldLabel
: gettext('Sockets'),
434 reference
: 'socketsField',
440 value
: '{socketCount}',
444 xtype
: 'proxmoxintegerfield',
445 fieldLabel
: gettext('Cores'),
447 reference
: 'coresField',
453 value
: '{coreCount}',
457 xtype
: 'pveMemoryField',
458 fieldLabel
: gettext('Memory') + ' (MiB)',
460 reference
: 'memoryField',
464 { xtype
: 'displayfield' }, // spacer
465 { xtype
: 'displayfield' }, // spacer
467 xtype
: 'pveDiskStorageSelector',
468 reference
: 'defaultStorage',
469 storageLabel
: gettext('Default Storage'),
470 storageContent
: 'images',
473 name
: 'defaultStorage',
480 fieldLabel
: gettext('Name'),
483 reference
: 'nameField',
487 xtype
: 'CPUModelSelector',
489 reference
: 'cputype',
490 value
: 'x86-64-v2-AES',
491 fieldLabel
: gettext('CPU Type'),
494 xtype
: 'displayfield',
495 fieldLabel
: gettext('Total cores'),
499 value
: '{totalCoreCount}',
506 fieldLabel
: gettext('OS Type'),
510 store
: Object
.keys(PVE
.Utils
.kvm_ostypes
),
516 fieldLabel
: gettext('Version'),
522 displayField
: 'desc',
527 fields
: ['desc', 'val'],
528 data
: PVE
.Utils
.kvm_ostypes
.Linux
,
531 { xtype
: 'displayfield' }, // spacer
533 xtype
: 'PVE.form.BridgeSelector',
534 reference
: 'defaultBridge',
535 name
: 'defaultBridge',
537 fieldLabel
: gettext('Default Bridge'),
543 xtype
: 'proxmoxcheckbox',
544 fieldLabel
: gettext('Live Import'),
545 reference
: 'liveimport',
547 boxLabelCls
: 'pmx-hint black x-form-cb-label',
549 value
: '{liveImport}',
550 boxLabel
: '{liveImportNote}',
554 xtype
: 'displayfield',
555 fieldLabel
: gettext('Warnings'),
559 hidden
: '{hideWarnings}',
563 xtype
: 'displayfield',
564 reference
: 'warningText',
568 hidden
: '{hideWarnings}',
569 value
: '{warningsText}',
575 title
: gettext('Advanced'),
578 // the first inputpanel handles all values, so prevent value leakage here
579 onGetValues
: () => ({}),
583 xtype
: 'displayfield',
584 fieldLabel
: gettext('Disks'),
589 reference
: 'diskGrid',
600 xtype
: 'checkcolumn',
601 header
: gettext('Use'),
605 checkchange: function(_column
, _rowIndex
, _checked
, record
) {
611 text
: gettext('Disk'),
616 text
: gettext('Source'),
617 dataIndex
: 'import-from',
619 renderer: function(value
) {
620 return value
.replace(/^.*\//, '');
624 text
: gettext('Size'),
626 renderer
: (value
) => {
627 if (Ext
.isNumeric(value
)) {
628 return Proxmox
.Utils
.render_size(value
);
630 return value
?? Proxmox
.Utils
.unknownText
;
634 text
: gettext('Storage'),
636 xtype
: 'widgetcolumn',
639 xtype
: 'pveStorageSelector',
643 emptyText
: gettext('From Default'),
645 storageContent
: 'images',
647 onWidgetAttach
: 'setNodename',
650 text
: gettext('Format'),
652 xtype
: 'widgetcolumn',
655 xtype
: 'pveDiskFormatSelector',
659 matchFieldWidth
: false,
668 xtype
: 'proxmoxcheckbox',
669 fieldLabel
: gettext('Prepare for VirtIO-SCSI'),
671 reference
: 'mapSata',
676 disabled
: '{!isWindows}',
680 'data-qtip': gettext('Maps SCSI disks to SATA and changes the SCSI Controller. Useful for a quicker switch to VirtIO-SCSI attached disks'),
687 xtype
: 'pveScsiHwSelector',
690 value
: '__default__',
692 fieldLabel
: gettext('SCSI Controller'),
698 xtype
: 'displayfield',
699 fieldLabel
: gettext('CD/DVD Drives'),
714 return !rec
.data
.hidden
;
720 xtype
: 'checkcolumn',
721 header
: gettext('Use'),
725 checkchange: function(_column
, _rowIndex
, _checked
, record
) {
731 text
: gettext('Slot'),
736 text
: gettext('Storage'),
737 xtype
: 'widgetcolumn',
740 xtype
: 'pveStorageSelector',
744 emptyText
: Proxmox
.Utils
.noneText
,
745 storageContent
: 'iso',
747 onWidgetAttach
: 'setNodename',
750 text
: gettext('ISO'),
752 xtype
: 'widgetcolumn',
755 xtype
: 'pveFileSelector',
759 emptyText
: Proxmox
.Utils
.noneText
,
760 storageContent
: 'iso',
762 onWidgetAttach
: 'setNodename',
767 xtype
: 'displayfield',
768 fieldLabel
: gettext('Network Interfaces'),
778 reference
: 'netGrid',
787 xtype
: 'checkcolumn',
788 header
: gettext('Use'),
792 checkchange: function(_column
, _rowIndex
, _checked
, record
) {
802 text
: gettext('MAC address'),
804 dataIndex
: 'macaddr',
805 renderer
: 'renderMacAddress',
808 text
: gettext('Model'),
811 xtype
: 'widgetcolumn',
813 xtype
: 'pveNetworkCardSelector',
820 text
: gettext('Bridge'),
822 xtype
: 'widgetcolumn',
825 xtype
: 'PVE.form.BridgeSelector',
830 emptyText
: gettext('From Default'),
832 onWidgetAttach
: 'setNodename',
837 xtype
: 'proxmoxcheckbox',
839 boxLabel
: gettext('Unique MAC addresses'),
840 uncheckedValue
: false,
846 title
: gettext('Resulting Config'),
847 reference
: 'summaryTab',
851 reference
: 'summaryGrid',
862 { header
: 'Key', width
: 150, dataIndex
: 'key' },
863 { header
: 'Value', flex
: 1, dataIndex
: 'value' },
872 initComponent: function() {
875 if (!me
.volumeName
) {
876 throw "no volumeName given";
880 throw "no storage given";
884 throw "no nodename given";
889 me
.setTitle(Ext
.String
.format(gettext('Import Guest - {0}'), `${me.storage}:${me.volumeName}`));
891 me
.lookup('defaultStorage').setNodename(me
.nodename
);
892 me
.lookup('defaultBridge').setNodename(me
.nodename
);
894 let renderWarning
= w
=> {
895 const warningsCatalogue
= {
896 'cdrom-image-ignored': gettext("CD-ROM images cannot get imported, if required you can reconfigure the '{0}' drive in the 'Advanced' tab."),
897 'nvme-unsupported': gettext("NVMe disks are currently not supported, '{0}' will get attaced as SCSI"),
898 'ovmf-with-lsi-unsupported': gettext("OVMF is built without LSI drivers, scsi hardware was set to '{1}'"),
899 'serial-port-socket-only': gettext("Serial socket '{0}' will be mapped to a socket"),
900 'guest-is-running': gettext('Virtual guest seems to be running on source host. Import might fail or have inconsistent state!'),
901 'efi-state-lost': Ext
.String
.format(
902 gettext('EFI state cannot be imported, you may need to reconfigure the boot order (see {0})'),
903 '<a href="https://pve.proxmox.com/wiki/OVMF/UEFI_Boot_Entries">OVMF/UEFI Boot Entries</a>',
906 let message
= warningsCatalogue
[w
.type
];
907 if (!w
.type
|| !message
) {
908 return w
.message
?? w
.type
?? gettext('Unknown warning');
910 return Ext
.String
.format(message
, w
.key
?? 'unknown', w
.value
?? 'unknown');
914 success: function(response
) {
915 let data
= response
.result
.data
;
916 me
.vmConfig
= data
['create-args'];
919 for (const [id
, value
] of Object
.entries(data
.disks
?? {})) {
920 let volid
= Ext
.htmlEncode('<none>');
922 if (Ext
.isObject(value
)) {
930 'import-from': volid
,
936 for (const [id
, parsed
] of Object
.entries(data
.net
?? {})) {
938 parsed
.enable
= true;
943 for (const [id
, value
] of Object
.entries(me
.vmConfig
)) {
944 if (!Ext
.isString(value
) || !value
.match(/media=cdrom/)) {
952 delete me
.vmConfig
[id
];
955 me
.lookup('diskGrid').getStore().setData(disks
);
956 me
.lookup('netGrid').getStore().setData(nets
);
957 me
.lookup('cdGrid').getStore().setData(cdroms
);
959 let additionalCdIdx
= me
.getController().calculateAdditionalCDIdx();
960 if (additionalCdIdx
=== '') {
961 me
.getViewModel().set('maxCdDrives', true);
962 } else if (cdroms
.length
=== 0) {
963 me
.additionalCdIdx
= additionalCdIdx
;
964 me
.lookup('cdGrid').getStore().add({
966 hidden
: !(me
.vmConfig
.ostype
?? '').startsWith('w'),
972 me
.getViewModel().set('warnings', data
.warnings
.map(w
=> renderWarning(w
)));
974 let osinfo
= PVE
.Utils
.get_kvm_osinfo(me
.vmConfig
.ostype
?? '');
975 let mapSata
= (me
.vmConfig
.ostype
?? '').startsWith('w') && (me
.vmConfig
.bios
?? '').indexOf('ovmf') !== -1;
983 me
.lookup('mapSata').setValue(mapSata
);