public inbox for pve-devel@lists.proxmox.com
 help / color / mirror / Atom feed
From: "Max R. Carrara" <m.carrara@proxmox.com>
To: pve-devel@lists.proxmox.com
Subject: [PATCH pve-storage v1 24/54] tree-wide: replace usages of VZTMPL_EXT_RE_1 with parsing functions
Date: Wed, 22 Apr 2026 13:12:50 +0200	[thread overview]
Message-ID: <20260422111322.257380-25-m.carrara@proxmox.com> (raw)
In-Reply-To: <20260422111322.257380-1-m.carrara@proxmox.com>

Add support for the 'vztmpl' volume type in
`PVE::Storage::Common::Parse`.

Also add corresponding test cases.

Use the module's parsing functions to replace usages of the
`PVE::Storage::VZTMPL_EXT_RE_1` regex across the repository.

As done with the `ISO_EXT_RE_0` regex, keep the `VZTMPL_EXT_RE_1`
regex around for now, since it's public and therefore also part of the
storage API. On an APIVER + APIAGE bump, this regex should be marked
for removal.

Signed-off-by: Max R. Carrara <m.carrara@proxmox.com>
---
 src/PVE/API2/Storage/Status.pm              |   6 +-
 src/PVE/Storage.pm                          |   4 +-
 src/PVE/Storage/Common/Parse.pm             |  23 +++++
 src/PVE/Storage/Common/test/parser_tests.pl | 109 ++++++++++++++++++++
 src/PVE/Storage/Plugin.pm                   |  16 +--
 5 files changed, 147 insertions(+), 11 deletions(-)

diff --git a/src/PVE/API2/Storage/Status.pm b/src/PVE/API2/Storage/Status.pm
index cf60c148..03929d26 100644
--- a/src/PVE/API2/Storage/Status.pm
+++ b/src/PVE/API2/Storage/Status.pm
@@ -85,11 +85,13 @@ my sub parse_transferred_file_path_extension : prototype($$) {
     }

     if ($vtype eq 'vztmpl') {
-        if ($path !~ m![^/]+$PVE::Storage::VZTMPL_EXT_RE_1$!) {
+        my $parts = parse_path_as_volname_parts($path, $vtype);
+
+        if (!defined($parts)) {
             raise_param_exc({ filename => "wrong file extension" });
         }

-        my $ext = $1;
+        my $ext = $parts->{ext};
         return $ext;
     }

diff --git a/src/PVE/Storage.pm b/src/PVE/Storage.pm
index d7d683ad..d5b0b637 100755
--- a/src/PVE/Storage.pm
+++ b/src/PVE/Storage.pm
@@ -761,9 +761,7 @@ sub path_to_volume_id {
         }

         if ($vtype eq 'vztmpl') {
-            return if $filename !~ m!/([^/]+$VZTMPL_EXT_RE_1)$!;
-            my $name = $1;
-            return "$sid:vztmpl/$name";
+            return parse_path_as_volid($sid, $scfg, $path, $vtype);
         }

         if ($vtype eq 'backup') {
diff --git a/src/PVE/Storage/Common/Parse.pm b/src/PVE/Storage/Common/Parse.pm
index bcc6f9fc..b18ef576 100644
--- a/src/PVE/Storage/Common/Parse.pm
+++ b/src/PVE/Storage/Common/Parse.pm
@@ -33,6 +33,14 @@ primarily related to.

 =cut

+my @VZTMPL_COMPRESSION_EXTENSIONS = ('gz', 'xz', 'zst', 'bz2');
+
+my sub join_to_re_alternations(@list) {
+    return join('|', map { quotemeta } @list);
+}
+
+my $RE_VZTMPL_COMPRESSION_EXTENSIONS = join_to_re_alternations(@VZTMPL_COMPRESSION_EXTENSIONS);
+
 my $RE_PARENT_DIR = quotemeta('..');
 my $RE_CONTAINS_PARENT_DIR = qr!
     ( ^$RE_PARENT_DIR/ )  #  ../ --> Beginning of path
@@ -48,12 +56,27 @@ my $RE_ISO_FILE_PATH = qr!
     )
 !xn;

+my $RE_VZTMPL_FILE_PATH = qr!
+    (?<path>
+        (?<file>
+            [^/]+
+            \.
+            (?<ext>
+                (?<ext_archive> (?i: tar) )
+                ( \. (?<ext_compression> (?i: $RE_VZTMPL_COMPRESSION_EXTENSIONS)) )?
+            )
+        )
+    )
+!xn;
+
 my $RE_FILE_PATH_FOR_VTYPE = {
     iso => qr/^$RE_ISO_FILE_PATH$/,
+    vztmpl => qr/^$RE_VZTMPL_FILE_PATH$/,
 };

 my $RE_VOLNAME_FOR_VTYPE = {
     iso => qr/^$RE_ISO_FILE_PATH$/,
+    vztmpl => qr/^$RE_VZTMPL_FILE_PATH$/,
 };

 my sub contains_parent_dir($path) {
diff --git a/src/PVE/Storage/Common/test/parser_tests.pl b/src/PVE/Storage/Common/test/parser_tests.pl
index 0167ca74..31575b66 100755
--- a/src/PVE/Storage/Common/test/parser_tests.pl
+++ b/src/PVE/Storage/Common/test/parser_tests.pl
@@ -86,12 +86,121 @@ my $volname_cases_iso_invalid = [
     },
 ];

+my $volname_cases_vztmpl_valid = [
+    # plain paths
+    {
+        path => 'archlinux-base_20190924-1_amd64.tar',
+        expected => {
+            file => 'archlinux-base_20190924-1_amd64.tar',
+            ext => 'tar',
+            'ext-archive' => 'tar',
+            'disk-path' => 'archlinux-base_20190924-1_amd64.tar',
+            path => 'archlinux-base_20190924-1_amd64.tar',
+            vtype => 'vztmpl',
+            volname => 'vztmpl/archlinux-base_20190924-1_amd64.tar',
+        },
+    },
+    {
+        path => 'archlinux-base_20190924-1_amd64.tar.gz',
+        expected => {
+            file => 'archlinux-base_20190924-1_amd64.tar.gz',
+            ext => 'tar.gz',
+            'ext-archive' => 'tar',
+            'ext-compression' => 'gz',
+            'disk-path' => 'archlinux-base_20190924-1_amd64.tar.gz',
+            path => 'archlinux-base_20190924-1_amd64.tar.gz',
+            vtype => 'vztmpl',
+            volname => 'vztmpl/archlinux-base_20190924-1_amd64.tar.gz',
+        },
+    },
+    {
+        path => 'alpine-3.10-default_20190626_amd64.tar.xz',
+        expected => {
+            file => 'alpine-3.10-default_20190626_amd64.tar.xz',
+            ext => 'tar.xz',
+            'ext-archive' => 'tar',
+            'ext-compression' => 'xz',
+            'disk-path' => 'alpine-3.10-default_20190626_amd64.tar.xz',
+            path => 'alpine-3.10-default_20190626_amd64.tar.xz',
+            vtype => 'vztmpl',
+            volname => 'vztmpl/alpine-3.10-default_20190626_amd64.tar.xz',
+        },
+    },
+    {
+        path => 'debian-10.0-standard_10.0-1_amd64.tar.zst',
+        expected => {
+            file => 'debian-10.0-standard_10.0-1_amd64.tar.zst',
+            ext => 'tar.zst',
+            'ext-archive' => 'tar',
+            'ext-compression' => 'zst',
+            'disk-path' => 'debian-10.0-standard_10.0-1_amd64.tar.zst',
+            path => 'debian-10.0-standard_10.0-1_amd64.tar.zst',
+            vtype => 'vztmpl',
+            volname => 'vztmpl/debian-10.0-standard_10.0-1_amd64.tar.zst',
+        },
+    },
+    {
+        path => 'debian-11.0-standard_11.0-1_amd64.tar.bz2',
+        expected => {
+            file => 'debian-11.0-standard_11.0-1_amd64.tar.bz2',
+            ext => 'tar.bz2',
+            'ext-archive' => 'tar',
+            'ext-compression' => 'bz2',
+            'disk-path' => 'debian-11.0-standard_11.0-1_amd64.tar.bz2',
+            path => 'debian-11.0-standard_11.0-1_amd64.tar.bz2',
+            vtype => 'vztmpl',
+            volname => 'vztmpl/debian-11.0-standard_11.0-1_amd64.tar.bz2',
+        },
+    },
+
+    # case-insensitive file extensions
+    {
+        path => 'ARCHLINUX-BASE 20190924-1 AMD64.TAR.GZ',
+        expected => {
+            file => 'ARCHLINUX-BASE 20190924-1 AMD64.TAR.GZ',
+            ext => 'TAR.GZ',
+            'ext-archive' => 'TAR',
+            'ext-compression' => 'GZ',
+            'disk-path' => 'ARCHLINUX-BASE 20190924-1 AMD64.TAR.GZ',
+            path => 'ARCHLINUX-BASE 20190924-1 AMD64.TAR.GZ',
+            vtype => 'vztmpl',
+            volname => 'vztmpl/ARCHLINUX-BASE 20190924-1 AMD64.TAR.GZ',
+        },
+    },
+    {
+        path => 'Alpine 3.10 default_20190626_amd64.Tar.xZ',
+        expected => {
+            file => 'Alpine 3.10 default_20190626_amd64.Tar.xZ',
+            ext => 'Tar.xZ',
+            'ext-archive' => 'Tar',
+            'ext-compression' => 'xZ',
+            'disk-path' => 'Alpine 3.10 default_20190626_amd64.Tar.xZ',
+            path => 'Alpine 3.10 default_20190626_amd64.Tar.xZ',
+            vtype => 'vztmpl',
+            volname => 'vztmpl/Alpine 3.10 default_20190626_amd64.Tar.xZ',
+        },
+    },
+];
+
+my $volname_cases_vztmpl_invalid = [
+    {
+        description => "Invalid file extension (vztmpl)",
+        args => {
+            path => 'archlinux-base_20190924-1_amd64.zip',
+            vtype => 'vztmpl',
+        },
+        expected => undef,
+    },
+];
+
 my $cases_valid_all = [
     $volname_cases_iso_valid,
+    $volname_cases_vztmpl_valid,
 ];

 my $cases_invalid_all = [
     $volname_cases_iso_invalid,
+    $volname_cases_vztmpl_invalid,
 ];

 {
diff --git a/src/PVE/Storage/Plugin.pm b/src/PVE/Storage/Plugin.pm
index 590ba3e0..6bfa5c11 100644
--- a/src/PVE/Storage/Plugin.pm
+++ b/src/PVE/Storage/Plugin.pm
@@ -826,10 +826,10 @@ sub parse_volname {
         if ($vtype eq 'iso') {
             return ($vtype, $volume_path, undef, undef, undef, undef, 'raw');
         }
-    }

-    if ($volname =~ m!^vztmpl/([^/]+$PVE::Storage::VZTMPL_EXT_RE_1)$!) {
-        return ('vztmpl', $1, undef, undef, undef, undef, 'raw');
+        if ($vtype eq 'vztmpl') {
+            return ($vtype, $volume_path, undef, undef, undef, undef, 'raw');
+        }
     }

     if ($volname =~ m!^backup/([^/]+$PVE::Storage::BACKUP_EXT_RE_2)$!) {
@@ -1708,11 +1708,15 @@ my sub get_subdir_files {
         }

         if ($vtype eq 'vztmpl') {
-            return if $filename !~ m!/([^/]+$PVE::Storage::VZTMPL_EXT_RE_1)$!;
+            my $parts = parse_path_as_volid_parts($storeid, $scfg, $path, $vtype);
+            return if !defined($parts);
+
+            my ($ext, $ext_compression) = $parts->@{qw(ext ext-compression)};
+            my $format = $ext eq 'tar' ? $ext : ('t' . $ext_compression);

             return {
-                volid => "$storeid:vztmpl/$1",
-                format => $2 eq 'tar' ? $2 : "t$2",
+                volid => $parts->{volid},
+                format => $format,
             };
         }

--
2.47.3





  parent reply	other threads:[~2026-04-22 11:18 UTC|newest]

Thread overview: 55+ messages / expand[flat|nested]  mbox.gz  Atom feed  top
2026-04-22 11:12 [PATCH pve-storage, pve-manager v1 00/54] Fix #2884: Implement Subdirectory Scanning for Dir-Based Storage Types Max R. Carrara
2026-04-22 11:12 ` [PATCH pve-storage v1 01/54] test: plugin tests: run tests with at most 4 jobs Max R. Carrara
2026-04-22 11:12 ` [PATCH pve-storage v1 02/54] plugin, common: remove superfluous use of =pod command paragraph Max R. Carrara
2026-04-22 11:12 ` [PATCH pve-storage v1 03/54] common: add POD headings for groups of helpers Max R. Carrara
2026-04-22 11:12 ` [PATCH pve-storage v1 04/54] common: use Exporter module for PVE::Storage::Common Max R. Carrara
2026-04-22 11:12 ` [PATCH pve-storage v1 05/54] plugin: make get_subdir_files a proper subroutine and update style Max R. Carrara
2026-04-22 11:12 ` [PATCH pve-storage v1 06/54] plugin api: replace helpers w/ standalone subs, bump API version & age Max R. Carrara
2026-04-22 11:12 ` [PATCH pve-storage v1 07/54] common: prevent autovivification in plugin_get_vtype_subdir helper Max R. Carrara
2026-04-22 11:12 ` [PATCH pve-storage v1 08/54] plugin: break up needless if-elsif chain into separate if-blocks Max R. Carrara
2026-04-22 11:12 ` [PATCH pve-storage v1 09/54] plugin: adapt get_subdir_files helper of list_volumes API method Max R. Carrara
2026-04-22 11:12 ` [PATCH pve-storage v1 10/54] plugin: update code style of list_volumes plugin " Max R. Carrara
2026-04-22 11:12 ` [PATCH pve-storage v1 11/54] plugin: use closure for obtaining raw volume data in list_volumes Max R. Carrara
2026-04-22 11:12 ` [PATCH pve-storage v1 12/54] plugin: use closure for inner loop logic " Max R. Carrara
2026-04-22 11:12 ` [PATCH pve-storage v1 13/54] storage: update code style in function path_to_volume_id Max R. Carrara
2026-04-22 11:12 ` [PATCH pve-storage v1 14/54] storage: break up needless if-elsif chain in path_to_volume_id Max R. Carrara
2026-04-22 11:12 ` [PATCH pve-storage v1 15/54] storage: heave vtype file path parsing logic inside loop into helper Max R. Carrara
2026-04-22 11:12 ` [PATCH pve-storage v1 16/54] storage: clean up code that was moved into helper in path_to_volume_id Max R. Carrara
2026-04-22 11:12 ` [PATCH pve-storage v1 17/54] api: status: move content type assert for up-/downloads into helper Max R. Carrara
2026-04-22 11:12 ` [PATCH pve-storage v1 18/54] api: status: use helper from common module to get content directory Max R. Carrara
2026-04-22 11:12 ` [PATCH pve-storage v1 19/54] api: status: move up-/download file path parsing code into helper Max R. Carrara
2026-04-22 11:12 ` [PATCH pve-storage v1 20/54] api: status: simplify file content assertion logic for up-/download Max R. Carrara
2026-04-22 11:12 ` [PATCH pve-storage v1 21/54] test: guest import: add tests for PVE::GuestImport Max R. Carrara
2026-04-22 11:12 ` [PATCH pve-storage v1 22/54] tree-wide: introduce parsing module and replace usages of ISO_EXT_RE_0 Max R. Carrara
2026-04-22 11:12 ` [PATCH pve-storage v1 23/54] common: test: set up parser testing code, add tests for 'iso' vtype Max R. Carrara
2026-04-22 11:12 ` Max R. Carrara [this message]
2026-04-22 11:12 ` [PATCH pve-storage v1 25/54] tree-wide: replace usages of BACKUP_EXT_RE_2 with parsing functions Max R. Carrara
2026-04-22 11:12 ` [PATCH pve-storage v1 26/54] tree-wide: replace usages of inline regexes for snippets with parsers Max R. Carrara
2026-04-22 11:12 ` [PATCH pve-storage v1 27/54] tree-wide: partially replace usages of regexes for 'import' vtype Max R. Carrara
2026-04-22 11:12 ` [PATCH pve-storage v1 28/54] tree-wide: replace remaining " Max R. Carrara
2026-04-22 11:12 ` [PATCH pve-storage v1 29/54] plugin: simplify recently refactored logic in parse_volname method Max R. Carrara
2026-04-22 11:12 ` [PATCH pve-storage v1 30/54] plugin: simplify recently refactored logic in get_subdir_files helper Max R. Carrara
2026-04-22 11:12 ` [PATCH pve-storage v1 31/54] storage: simplify recently refactored logic in path_to_volume_id sub Max R. Carrara
2026-04-22 11:12 ` [PATCH pve-storage v1 32/54] api: status: simplify recently added parsing helper for file transfers Max R. Carrara
2026-04-22 11:12 ` [PATCH pve-storage v1 33/54] plugin: use parsing helper in parse_volume_id sub Max R. Carrara
2026-04-22 11:13 ` [PATCH pve-storage v1 34/54] test: list volumes: reorganize and modernize test running code Max R. Carrara
2026-04-22 11:13 ` [PATCH pve-storage v1 35/54] test: list volumes: fix broken test checking for vmlist modifications Max R. Carrara
2026-04-22 11:13 ` [PATCH pve-storage v1 36/54] test: list volumes: introduce new format for test cases Max R. Carrara
2026-04-22 11:13 ` [PATCH pve-storage v1 37/54] test: list volumes: remove legacy code and migrate cases to new format Max R. Carrara
2026-04-22 11:13 ` [PATCH pve-storage v1 38/54] test: list volumes: document behavior wrt. undeclared content types Max R. Carrara
2026-04-22 11:13 ` [PATCH pve-storage v1 39/54] plugin: correct comment in get_subdir_files helper Max R. Carrara
2026-04-22 11:13 ` [PATCH pve-storage v1 40/54] test: parse volname: modernize code Max R. Carrara
2026-04-22 11:13 ` [PATCH pve-storage v1 41/54] test: parse volname: adapt tests regarding 'import' volume type Max R. Carrara
2026-04-22 11:13 ` [PATCH pve-storage v1 42/54] test: parse volname: move VM disk test creation into separate block Max R. Carrara
2026-04-22 11:13 ` [PATCH pve-storage v1 43/54] test: parse volname: move backup file test creation into sep. block Max R. Carrara
2026-04-22 11:13 ` [PATCH pve-storage v1 44/54] test: parse volname: parameterize test case creation for some vtypes Max R. Carrara
2026-04-22 11:13 ` [PATCH pve-storage v1 45/54] test: volume id: modernize code Max R. Carrara
2026-04-22 11:13 ` [PATCH pve-storage v1 46/54] test: volume id: rename 'volname' test case parameter to 'file' Max R. Carrara
2026-04-22 11:13 ` [PATCH pve-storage v1 47/54] test: filesystem path: modernize code Max R. Carrara
2026-04-22 11:13 ` [PATCH pve-storage v1 48/54] fix #2884: implement nested subdir scanning and support 'iso' vtype Max R. Carrara
2026-04-22 11:13 ` [PATCH pve-storage v1 49/54] fix #2884: support nested subdir scanning for 'vztmpl' volume type Max R. Carrara
2026-04-22 11:13 ` [PATCH pve-storage v1 50/54] fix #2884: support nested subdir scanning for 'snippets' vtype Max R. Carrara
2026-04-22 11:13 ` [PATCH pve-storage v1 51/54] test: add more tests for 'import' vtype & guard against nested subdirs Max R. Carrara
2026-04-22 11:13 ` [PATCH pve-storage v1 52/54] test: add tests guarding against subdir scanning for vtypes Max R. Carrara
2026-04-22 11:13 ` [PATCH pve-storage v1 53/54] storage api: mark old public regexes for removal, bump APIVER & APIAGE Max R. Carrara
2026-04-22 11:13 ` [PATCH pve-manager v1 54/54] fix #2884: ui: storage: add field for 'max-scan-depth' property Max R. Carrara

Reply instructions:

You may reply publicly to this message via plain-text email
using any one of the following methods:

* Save the following mbox file, import it into your mail client,
  and reply-to-all from there: mbox

  Avoid top-posting and favor interleaved quoting:
  https://en.wikipedia.org/wiki/Posting_style#Interleaved_style

* Reply using the --to, --cc, and --in-reply-to
  switches of git-send-email(1):

  git send-email \
    --in-reply-to=20260422111322.257380-25-m.carrara@proxmox.com \
    --to=m.carrara@proxmox.com \
    --cc=pve-devel@lists.proxmox.com \
    /path/to/YOUR_REPLY

  https://kernel.org/pub/software/scm/git/docs/git-send-email.html

* If your mail client supports setting the In-Reply-To header
  via mailto: links, try the mailto: link
Be sure your reply has a Subject: header at the top and a blank line before the message body.
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