* [pve-devel] [PATCH storage/manager v4] allow down/upload & import of images in the web UI
@ 2025-03-26 15:26 Dominik Csapak
2025-03-26 15:26 ` [pve-devel] [PATCH storage v4 1/1] import: allow upload of guest images files into import storage Dominik Csapak
` (5 more replies)
0 siblings, 6 replies; 7+ messages in thread
From: Dominik Csapak @ 2025-03-26 15:26 UTC (permalink / raw)
To: pve-devel
most of the building blocks are already there:
* we can have image files in an import storage
* we can import image files via the api from such a storage
this series fills in the missing bits & pieces:
* allow uploading/downloading image files into an import storage via the webgui
* adding the possibility to select such a file when creating a vm/disk
One minor "ugliness": when switching between import on/off, the target
storage selector resets. This is normally intended by the component,
since it's most often only disabled when it's still visible, except here
in this case.
If this is a blocker, I can of course add an option to the selector
to not do this here, but IMHO this is a rather odd use case anyway,
so I opted for not handling that explicitely.
changes from v3:
* allow raw (.raw, .img) and vmdk files too
* automatically append '.raw' for '.img' files
changes from v2:
* fix correctly unset 'import-from' in wizard when going to summary,
back to disk, unselecting import, then going forward to the summary
again
* fixed an issue with the file selector being mistakenly disabled
changes from v1:
* fixed an issue where the file selector would be hidden but invalid
pve-storage:
Dominik Csapak (1):
import: allow upload of guest images files into import storage
src/PVE/API2/Storage/Status.pm | 17 ++++++++++++++++-
src/PVE/Storage.pm | 3 ++-
2 files changed, 18 insertions(+), 2 deletions(-)
pve-manager:
Dominik Csapak (3):
ui: storage content: allow upload of guest images for import type
ui: form: file selector: allow optional filter
ui: qemu hd edit: allow importing a disk from the import storage
www/manager6/form/FileSelector.js | 10 +++
www/manager6/qemu/HDEdit.js | 71 ++++++++++++++++++++-
www/manager6/window/DownloadUrlToStorage.js | 4 ++
www/manager6/window/UploadToStorage.js | 7 +-
4 files changed, 89 insertions(+), 3 deletions(-)
--
2.39.5
_______________________________________________
pve-devel mailing list
pve-devel@lists.proxmox.com
https://lists.proxmox.com/cgi-bin/mailman/listinfo/pve-devel
^ permalink raw reply [flat|nested] 7+ messages in thread
* [pve-devel] [PATCH storage v4 1/1] import: allow upload of guest images files into import storage
2025-03-26 15:26 [pve-devel] [PATCH storage/manager v4] allow down/upload & import of images in the web UI Dominik Csapak
@ 2025-03-26 15:26 ` Dominik Csapak
2025-03-26 15:26 ` [pve-devel] [PATCH manager v4 1/3] ui: storage content: allow upload of guest images for import type Dominik Csapak
` (4 subsequent siblings)
5 siblings, 0 replies; 7+ messages in thread
From: Dominik Csapak @ 2025-03-26 15:26 UTC (permalink / raw)
To: pve-devel
so users can upload qcow2/raw/vmdk files directly in the ui
Signed-off-by: Dominik Csapak <d.csapak@proxmox.com>
---
changes from v3:
* also allow vmdk and raw
src/PVE/API2/Storage/Status.pm | 17 ++++++++++++++++-
src/PVE/Storage.pm | 3 ++-
2 files changed, 18 insertions(+), 2 deletions(-)
diff --git a/src/PVE/API2/Storage/Status.pm b/src/PVE/API2/Storage/Status.pm
index c854b53..b23d283 100644
--- a/src/PVE/API2/Storage/Status.pm
+++ b/src/PVE/API2/Storage/Status.pm
@@ -456,6 +456,7 @@ __PACKAGE__->register_method ({
my $path;
my $isOva = 0;
+ my $imageFormat;
if ($content eq 'iso') {
if ($filename !~ m![^/]+$PVE::Storage::ISO_EXT_RE_0$!) {
@@ -472,7 +473,12 @@ __PACKAGE__->register_method ({
raise_param_exc({ filename => "invalid filename or wrong extension" });
}
- $isOva = 1;
+ if ($filename =~ m/\.ova$/) {
+ $isOva = 1;
+ } elsif ($filename =~ m/${PVE::Storage::UPLOAD_IMPORT_IMAGE_EXT_RE_1}$/) {
+ $imageFormat = $1;
+ }
+
$path = PVE::Storage::get_import_dir($cfg, $storage);
} else {
raise_param_exc({ content => "upload content type '$content' not allowed" });
@@ -543,6 +549,9 @@ __PACKAGE__->register_method ({
if ($isOva) {
assert_ova_contents($tmpfilename);
+ } elsif (defined($imageFormat)) {
+ # checks untrusted image
+ PVE::Storage::file_size_info($tmpfilename, 10, $imageFormat, 1);
}
};
if (my $err = $@) {
@@ -667,6 +676,7 @@ __PACKAGE__->register_method({
my $path;
my $isOva = 0;
+ my $imageFormat;
if ($content eq 'iso') {
if ($filename !~ m![^/]+$PVE::Storage::ISO_EXT_RE_0$!) {
@@ -685,6 +695,8 @@ __PACKAGE__->register_method({
if ($filename =~ m/\.ova$/) {
$isOva = 1;
+ } elsif ($filename =~ m/${PVE::Storage::UPLOAD_IMPORT_IMAGE_EXT_RE_1}$/) {
+ $imageFormat = $1;
}
$path = PVE::Storage::get_import_dir($cfg, $storage);
@@ -717,6 +729,9 @@ __PACKAGE__->register_method({
if ($isOva) {
assert_ova_contents($tmp_path);
+ } elsif (defined($imageFormat)) {
+ # checks untrusted image
+ PVE::Storage::file_size_info($tmp_path, 10, $imageFormat, 1);
}
};
diff --git a/src/PVE/Storage.pm b/src/PVE/Storage.pm
index c5d4ff8..09d9883 100755
--- a/src/PVE/Storage.pm
+++ b/src/PVE/Storage.pm
@@ -116,7 +116,8 @@ our $BACKUP_EXT_RE_2 = qr/\.(tgz|(?:tar|vma)(?:\.(${\PVE::Storage::Plugin::COMPR
our $IMPORT_EXT_RE_1 = qr/\.(ova|ovf|qcow2|raw|vmdk)/;
-our $UPLOAD_IMPORT_EXT_RE_1 = qr/\.(ova)/;
+our $UPLOAD_IMPORT_EXT_RE_1 = qr/\.(ova|qcow2|raw|vmdk)/;
+our $UPLOAD_IMPORT_IMAGE_EXT_RE_1 = qr/\.(qcow2|raw|vmdk)/;
our $SAFE_CHAR_CLASS_RE = qr/[a-zA-Z0-9\-\.\+\=\_]/;
our $SAFE_CHAR_WITH_WHITESPACE_CLASS_RE = qr/[ a-zA-Z0-9\-\.\+\=\_]/;
--
2.39.5
_______________________________________________
pve-devel mailing list
pve-devel@lists.proxmox.com
https://lists.proxmox.com/cgi-bin/mailman/listinfo/pve-devel
^ permalink raw reply [flat|nested] 7+ messages in thread
* [pve-devel] [PATCH manager v4 1/3] ui: storage content: allow upload of guest images for import type
2025-03-26 15:26 [pve-devel] [PATCH storage/manager v4] allow down/upload & import of images in the web UI Dominik Csapak
2025-03-26 15:26 ` [pve-devel] [PATCH storage v4 1/1] import: allow upload of guest images files into import storage Dominik Csapak
@ 2025-03-26 15:26 ` Dominik Csapak
2025-03-26 15:26 ` [pve-devel] [PATCH manager v4 2/3] ui: form: file selector: allow optional filter Dominik Csapak
` (3 subsequent siblings)
5 siblings, 0 replies; 7+ messages in thread
From: Dominik Csapak @ 2025-03-26 15:26 UTC (permalink / raw)
To: pve-devel
by allowing 'qcow2', 'vmdk', 'raw' and 'img' files to upload.
Since we expect raw images to have a '.raw' extension, append that for
'.img' files.
partially fixes #2424
Signed-off-by: Dominik Csapak <d.csapak@proxmox.com>
---
changes from v3:
* also allow vmdk/raw
* rename '.img' files to '.raw'
www/manager6/window/DownloadUrlToStorage.js | 4 ++++
www/manager6/window/UploadToStorage.js | 7 +++++--
2 files changed, 9 insertions(+), 2 deletions(-)
diff --git a/www/manager6/window/DownloadUrlToStorage.js b/www/manager6/window/DownloadUrlToStorage.js
index 3ac80ac9..aabc9d3c 100644
--- a/www/manager6/window/DownloadUrlToStorage.js
+++ b/www/manager6/window/DownloadUrlToStorage.js
@@ -89,6 +89,10 @@ Ext.define('PVE.window.DownloadUrlToStorage', {
filename = matches[1];
compression = matches[2].toLowerCase();
}
+ } else if (view.content === 'import') {
+ if (filename.endsWith('.img')) {
+ filename += '.raw';
+ }
}
view.setValues({
diff --git a/www/manager6/window/UploadToStorage.js b/www/manager6/window/UploadToStorage.js
index cdf548a8..0dad6333 100644
--- a/www/manager6/window/UploadToStorage.js
+++ b/www/manager6/window/UploadToStorage.js
@@ -9,7 +9,7 @@ Ext.define('PVE.window.UploadToStorage', {
title: gettext('Upload'),
acceptedExtensions: {
- 'import': ['.ova'],
+ 'import': ['.ova', '.qcow2', '.raw', '.vmdk', '.img'],
iso: ['.img', '.iso'],
vztmpl: ['.tar.gz', '.tar.xz', '.tar.zst'],
},
@@ -135,7 +135,10 @@ Ext.define('PVE.window.UploadToStorage', {
fileChange: function(input) {
const vm = this.getViewModel();
- const name = input.value.replace(/^.*(\/|\\)/, '');
+ let name = input.value.replace(/^.*(\/|\\)/, '');
+ if (name.endsWith('.img')) {
+ name += '.raw';
+ }
const fileInput = input.fileInputEl.dom;
vm.set('filename', name);
vm.set('size', (fileInput.files[0] && Proxmox.Utils.format_size(fileInput.files[0].size)) || '-');
--
2.39.5
_______________________________________________
pve-devel mailing list
pve-devel@lists.proxmox.com
https://lists.proxmox.com/cgi-bin/mailman/listinfo/pve-devel
^ permalink raw reply [flat|nested] 7+ messages in thread
* [pve-devel] [PATCH manager v4 2/3] ui: form: file selector: allow optional filter
2025-03-26 15:26 [pve-devel] [PATCH storage/manager v4] allow down/upload & import of images in the web UI Dominik Csapak
2025-03-26 15:26 ` [pve-devel] [PATCH storage v4 1/1] import: allow upload of guest images files into import storage Dominik Csapak
2025-03-26 15:26 ` [pve-devel] [PATCH manager v4 1/3] ui: storage content: allow upload of guest images for import type Dominik Csapak
@ 2025-03-26 15:26 ` Dominik Csapak
2025-03-26 15:26 ` [pve-devel] [PATCH manager v4 3/3] ui: qemu hd edit: allow importing a disk from the import storage Dominik Csapak
` (2 subsequent siblings)
5 siblings, 0 replies; 7+ messages in thread
From: Dominik Csapak @ 2025-03-26 15:26 UTC (permalink / raw)
To: pve-devel
this sometimes comes in handy when we only want to show specific files.
Signed-off-by: Dominik Csapak <d.csapak@proxmox.com>
---
no changes from v3
www/manager6/form/FileSelector.js | 10 ++++++++++
1 file changed, 10 insertions(+)
diff --git a/www/manager6/form/FileSelector.js b/www/manager6/form/FileSelector.js
index ef2bedf9..9db20711 100644
--- a/www/manager6/form/FileSelector.js
+++ b/www/manager6/form/FileSelector.js
@@ -43,6 +43,13 @@ Ext.define('PVE.form.FileSelector', {
url: url,
});
+ if (Ext.isFunction(me.filter)) {
+ me.store.clearFilter();
+ me.store.addFilter([me.filter]);
+ } else {
+ me.store.clearFilter();
+ }
+
me.store.removeAll();
me.store.load();
},
@@ -60,6 +67,9 @@ Ext.define('PVE.form.FileSelector', {
valueField: 'volid',
displayField: 'text',
+ // An optional filter function
+ filter: undefined,
+
listConfig: {
width: 600,
columns: [
--
2.39.5
_______________________________________________
pve-devel mailing list
pve-devel@lists.proxmox.com
https://lists.proxmox.com/cgi-bin/mailman/listinfo/pve-devel
^ permalink raw reply [flat|nested] 7+ messages in thread
* [pve-devel] [PATCH manager v4 3/3] ui: qemu hd edit: allow importing a disk from the import storage
2025-03-26 15:26 [pve-devel] [PATCH storage/manager v4] allow down/upload & import of images in the web UI Dominik Csapak
` (2 preceding siblings ...)
2025-03-26 15:26 ` [pve-devel] [PATCH manager v4 2/3] ui: form: file selector: allow optional filter Dominik Csapak
@ 2025-03-26 15:26 ` Dominik Csapak
2025-03-31 14:06 ` [pve-devel] [PATCH storage/manager v4] allow down/upload & import of images in the web UI Filip Schauer
2025-04-01 8:24 ` Dominik Csapak
5 siblings, 0 replies; 7+ messages in thread
From: Dominik Csapak @ 2025-03-26 15:26 UTC (permalink / raw)
To: pve-devel
adds a checkbox 'import image' above the storage selector which:
* hides the original storage selector
* shows a 'source storage' selector
* shows a 'import file' selector
* shows a 'target storage' selector
Since the wizard and the hd edit share this panel, this also works in
the wizard.
Signed-off-by: Dominik Csapak <d.csapak@proxmox.com>
---
changes from v3:
* also allow importing raw and vmdk (by changing the filter)
www/manager6/qemu/HDEdit.js | 71 ++++++++++++++++++++++++++++++++++++-
1 file changed, 70 insertions(+), 1 deletion(-)
diff --git a/www/manager6/qemu/HDEdit.js b/www/manager6/qemu/HDEdit.js
index b78647ec..d6357a91 100644
--- a/www/manager6/qemu/HDEdit.js
+++ b/www/manager6/qemu/HDEdit.js
@@ -78,11 +78,14 @@ Ext.define('PVE.qemu.HDInputPanel', {
if (values.hdimage) {
me.drive.file = values.hdimage;
} else {
- me.drive.file = values.hdstorage + ":" + values.disksize;
+ let disksize = values['import-from'] ? 0 : values.disksize;
+ me.drive.file = `${values.hdstorage}:${disksize}`;
+ PVE.Utils.propertyStringSet(me.drive, values['import-from'], 'import-from');
}
me.drive.format = values.diskformat;
}
+
PVE.Utils.propertyStringSet(me.drive, !values.backup, 'backup', '0');
PVE.Utils.propertyStringSet(me.drive, values.noreplicate, 'replicate', 'no');
PVE.Utils.propertyStringSet(me.drive, values.discard, 'discard', 'on');
@@ -168,6 +171,11 @@ Ext.define('PVE.qemu.HDInputPanel', {
var me = this;
me.down('#hdstorage').setNodename(nodename);
me.down('#hdimage').setStorage(undefined, nodename);
+
+ me.lookup('new-disk').setNodename(nodename);
+ me.lookup('import-source').setNodename(nodename);
+ me.lookup('import-source-file').setNodename(nodename);
+ me.lookup('import-target').setNodename(nodename);
},
hasAdvanced: true,
@@ -221,12 +229,73 @@ Ext.define('PVE.qemu.HDInputPanel', {
column1.push(me.unusedDisks);
} else if (me.isCreate) {
column1.push({
+ xtype: 'proxmoxcheckbox',
+ isFormField: false,
+ fieldLabel: gettext("Import Image"),
+ listeners: {
+ change: function(_cb, value) {
+ me.lookup('new-disk').setVisible(!value);
+ me.lookup('new-disk').setDisabled(!!value);
+
+ let importSource = me.lookup('import-source');
+ importSource.setVisible(!!value);
+ importSource.setDisabled(!value);
+ me.lookup('import-source-file').setVisible(!!value);
+ me.lookup('import-source-file').setDisabled(!value || !importSource.getValue());
+
+ me.lookup('import-target').setVisible(!!value);
+ me.lookup('import-target').setDisabled(!value);
+ },
+ },
+ });
+ column1.push({
+ reference: 'new-disk',
xtype: 'pveDiskStorageSelector',
storageContent: 'images',
name: 'disk',
nodename: me.nodename,
autoSelect: me.insideWizard,
});
+ column1.push({
+ xtype: 'pveStorageSelector',
+ reference: 'import-source',
+ fieldLabel: gettext('Import Storage'),
+ name: 'import-source-storage',
+ hidden: true,
+ disabled: true,
+ storageContent: 'import',
+ nodename: me.nodename,
+ autoSelect: me.insideWizard,
+ listeners: {
+ change: function(selector, storage) {
+ me.lookup('import-source-file').setStorage(storage);
+ me.lookup('import-source-file').setDisabled(!storage || selector.isDisabled());
+ },
+ },
+ });
+ column1.push({
+ xtype: 'pveFileSelector',
+ reference: 'import-source-file',
+ fieldLabel: gettext("Select Image"),
+ hidden: true,
+ disabled: true,
+ storageContent: 'import',
+ name: 'import-from',
+ filter: (rec) => ['qcow2', 'vmdk', 'raw'].indexOf(rec?.data?.format) !== -1,
+ nodename: me.nodename,
+ });
+ column1.push({
+ xtype: 'pveDiskStorageSelector',
+ reference: 'import-target',
+ storageLabel: gettext('Target Storage'),
+ hidden: true,
+ disabled: true,
+ hideSize: true,
+ storageContent: 'images',
+ name: 'target',
+ nodename: me.nodename,
+ autoSelect: me.insideWizard,
+ });
} else {
column1.push({
xtype: 'textfield',
--
2.39.5
_______________________________________________
pve-devel mailing list
pve-devel@lists.proxmox.com
https://lists.proxmox.com/cgi-bin/mailman/listinfo/pve-devel
^ permalink raw reply [flat|nested] 7+ messages in thread
* Re: [pve-devel] [PATCH storage/manager v4] allow down/upload & import of images in the web UI
2025-03-26 15:26 [pve-devel] [PATCH storage/manager v4] allow down/upload & import of images in the web UI Dominik Csapak
` (3 preceding siblings ...)
2025-03-26 15:26 ` [pve-devel] [PATCH manager v4 3/3] ui: qemu hd edit: allow importing a disk from the import storage Dominik Csapak
@ 2025-03-31 14:06 ` Filip Schauer
2025-04-01 8:24 ` Dominik Csapak
5 siblings, 0 replies; 7+ messages in thread
From: Filip Schauer @ 2025-03-31 14:06 UTC (permalink / raw)
To: pve-devel
Tested with (.img.raw/.raw), .qcow2, .vmdk and different combinations of
target storages/formats.
Everything worked fine except for one thing:
The upload dialog allows the user to manually change the file name. When
changing the file extension to ".img". Clicking "Upload" is allowed but
after the upload completes, the following error is thrown:
```
Error 400: Parameter verification failed.
filename: invalid filename or wrong extension
```
On 26/03/2025 16:26, Dominik Csapak wrote:
> most of the building blocks are already there:
> * we can have image files in an import storage
> * we can import image files via the api from such a storage
>
> this series fills in the missing bits & pieces:
> * allow uploading/downloading image files into an import storage via the webgui
> * adding the possibility to select such a file when creating a vm/disk
>
> One minor "ugliness": when switching between import on/off, the target
> storage selector resets. This is normally intended by the component,
> since it's most often only disabled when it's still visible, except here
> in this case.
>
> If this is a blocker, I can of course add an option to the selector
> to not do this here, but IMHO this is a rather odd use case anyway,
> so I opted for not handling that explicitely.
>
> changes from v3:
> * allow raw (.raw, .img) and vmdk files too
> * automatically append '.raw' for '.img' files
>
> changes from v2:
> * fix correctly unset 'import-from' in wizard when going to summary,
> back to disk, unselecting import, then going forward to the summary
> again
> * fixed an issue with the file selector being mistakenly disabled
>
> changes from v1:
> * fixed an issue where the file selector would be hidden but invalid
>
> pve-storage:
>
> Dominik Csapak (1):
> import: allow upload of guest images files into import storage
>
> src/PVE/API2/Storage/Status.pm | 17 ++++++++++++++++-
> src/PVE/Storage.pm | 3 ++-
> 2 files changed, 18 insertions(+), 2 deletions(-)
>
> pve-manager:
>
> Dominik Csapak (3):
> ui: storage content: allow upload of guest images for import type
> ui: form: file selector: allow optional filter
> ui: qemu hd edit: allow importing a disk from the import storage
>
> www/manager6/form/FileSelector.js | 10 +++
> www/manager6/qemu/HDEdit.js | 71 ++++++++++++++++++++-
> www/manager6/window/DownloadUrlToStorage.js | 4 ++
> www/manager6/window/UploadToStorage.js | 7 +-
> 4 files changed, 89 insertions(+), 3 deletions(-)
>
_______________________________________________
pve-devel mailing list
pve-devel@lists.proxmox.com
https://lists.proxmox.com/cgi-bin/mailman/listinfo/pve-devel
^ permalink raw reply [flat|nested] 7+ messages in thread
* Re: [pve-devel] [PATCH storage/manager v4] allow down/upload & import of images in the web UI
2025-03-26 15:26 [pve-devel] [PATCH storage/manager v4] allow down/upload & import of images in the web UI Dominik Csapak
` (4 preceding siblings ...)
2025-03-31 14:06 ` [pve-devel] [PATCH storage/manager v4] allow down/upload & import of images in the web UI Filip Schauer
@ 2025-04-01 8:24 ` Dominik Csapak
5 siblings, 0 replies; 7+ messages in thread
From: Dominik Csapak @ 2025-04-01 8:24 UTC (permalink / raw)
To: pve-devel
sent a v5:
https://lore.proxmox.com/pve-devel/20250401082318.1312661-1-d.csapak@proxmox.com/
_______________________________________________
pve-devel mailing list
pve-devel@lists.proxmox.com
https://lists.proxmox.com/cgi-bin/mailman/listinfo/pve-devel
^ permalink raw reply [flat|nested] 7+ messages in thread
end of thread, other threads:[~2025-04-01 8:25 UTC | newest]
Thread overview: 7+ messages (download: mbox.gz / follow: Atom feed)
-- links below jump to the message on this page --
2025-03-26 15:26 [pve-devel] [PATCH storage/manager v4] allow down/upload & import of images in the web UI Dominik Csapak
2025-03-26 15:26 ` [pve-devel] [PATCH storage v4 1/1] import: allow upload of guest images files into import storage Dominik Csapak
2025-03-26 15:26 ` [pve-devel] [PATCH manager v4 1/3] ui: storage content: allow upload of guest images for import type Dominik Csapak
2025-03-26 15:26 ` [pve-devel] [PATCH manager v4 2/3] ui: form: file selector: allow optional filter Dominik Csapak
2025-03-26 15:26 ` [pve-devel] [PATCH manager v4 3/3] ui: qemu hd edit: allow importing a disk from the import storage Dominik Csapak
2025-03-31 14:06 ` [pve-devel] [PATCH storage/manager v4] allow down/upload & import of images in the web UI Filip Schauer
2025-04-01 8:24 ` Dominik Csapak
This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox
Service provided by Proxmox Server Solutions GmbH | Privacy | Legal