detection into separate functions so they are reusable and easier modifiable.
Signed-off-by: Alwin Antreich <a.antre...@proxmox.com> --- test/Makefile | 5 ++- PVE/Storage.pm | 79 ++++++++++++++++++++++++++++----------- test/run_parser_tests.pl | 12 ++++++ test/test_archive_info.pm | 54 ++++++++++++++++++++++++++ 4 files changed, 128 insertions(+), 22 deletions(-) create mode 100755 test/run_parser_tests.pl create mode 100644 test/test_archive_info.pm diff --git a/test/Makefile b/test/Makefile index 833a597..838449f 100644 --- a/test/Makefile +++ b/test/Makefile @@ -1,6 +1,6 @@ all: test -test: test_zfspoolplugin test_disklist test_bwlimit +test: test_zfspoolplugin test_disklist test_bwlimit test_parsers test_zfspoolplugin: run_test_zfspoolplugin.pl ./run_test_zfspoolplugin.pl @@ -10,3 +10,6 @@ test_disklist: run_disk_tests.pl test_bwlimit: run_bwlimit_tests.pl ./run_bwlimit_tests.pl + +test_parsers: run_parser_tests.pl + ./run_parser_tests.pl diff --git a/PVE/Storage.pm b/PVE/Storage.pm index 60b8310..0bbd168 100755 --- a/PVE/Storage.pm +++ b/PVE/Storage.pm @@ -1284,6 +1284,53 @@ sub foreach_volid { } } +sub decompressor_info { + my ($format, $comp) = @_; + + if ($format eq 'tgz' && !defined($comp)) { + ($format, $comp) = ('tar', 'gz'); + } + + my $decompressor = { + tar => { + gz => ['tar', '-z'], + lzo => ['tar', '--lzop'], + }, + vma => { + gz => ['zcat'], + lzo => ['lzop', '-d', '-c'], + }, + }; + + die "ERROR: archive format not defined\n" + if !defined($decompressor->{$format}); + + my $decomp = $decompressor->{$format}->{$comp} if $comp; + + my $info = { + format => $format, + compression => $comp, + decompressor => $decomp, + }; + + return $info; +} + +sub archive_info { + my ($archive) = shift; + my $info; + + my $volid = basename($archive); + if ($volid =~ /vzdump-(lxc|openvz|qemu)-\d+-(?:\d{4})_(?:\d{2})_(?:\d{2})-(?:\d{2})_(?:\d{2})_(?:\d{2})\.(tgz$|tar|vma)(?:\.(gz|lzo))?$/) { + $info = decompressor_info($2, $3); + $info->{type} = $1; + } else { + die "ERROR: couldn't determine format and compression type\n"; + } + + return $info; +} + sub extract_vzdump_config_tar { my ($archive, $conf_re) = @_; @@ -1329,16 +1376,12 @@ sub extract_vzdump_config_vma { }; + my $info = archive_info($archive); + $comp //= $info->{compression}; + my $decompressor = $info->{decompressor}; + if ($comp) { - my $uncomp; - if ($comp eq 'gz') { - $uncomp = ["zcat", $archive]; - } elsif ($comp eq 'lzo') { - $uncomp = ["lzop", "-d", "-c", $archive]; - } else { - die "unknown compression method '$comp'\n"; - } - $cmd = [$uncomp, ["vma", "config", "-"]]; + $cmd = [ [@$decompressor, $archive], ["vma", "config", "-"] ]; # in some cases, lzop/zcat exits with 1 when its stdout pipe is # closed early by vma, detect this and ignore the exit code later @@ -1388,20 +1431,14 @@ sub extract_vzdump_config { } my $archive = abs_filesystem_path($cfg, $volid); + my $info = archive_info($archive); + my $format = $info->{format}; + my $comp = $info->{compression}; + my $type = $info->{type}; - if ($volid =~ /vzdump-(lxc|openvz)-\d+-(\d{4})_(\d{2})_(\d{2})-(\d{2})_(\d{2})_(\d{2})\.(tgz|(tar(\.(gz|lzo))?))$/) { + if ($type eq 'lxc' || $type eq 'openvz') { return extract_vzdump_config_tar($archive, qr!^(\./etc/vzdump/(pct|vps)\.conf)$!); - } elsif ($volid =~ /vzdump-qemu-\d+-(\d{4})_(\d{2})_(\d{2})-(\d{2})_(\d{2})_(\d{2})\.(tgz|((tar|vma)(\.(gz|lzo))?))$/) { - my $format; - my $comp; - if ($7 eq 'tgz') { - $format = 'tar'; - $comp = 'gz'; - } else { - $format = $9; - $comp = $11 if defined($11); - } - + } elsif ($type eq 'qemu') { if ($format eq 'tar') { return extract_vzdump_config_tar($archive, qr!\(\./qemu-server\.conf\)!); } else { diff --git a/test/run_parser_tests.pl b/test/run_parser_tests.pl new file mode 100755 index 0000000..042112c --- /dev/null +++ b/test/run_parser_tests.pl @@ -0,0 +1,12 @@ +#!/usr/bin/perl + +use strict; +use warnings; + +use TAP::Harness; + +my $harness = TAP::Harness->new( { verbosity => -1 }); +my $res = $harness->runtests("test_archive_info.pm"); + +exit -1 if !$res || $res->{failed} || $res->{parse_errors}; + diff --git a/test/test_archive_info.pm b/test/test_archive_info.pm new file mode 100644 index 0000000..464cc89 --- /dev/null +++ b/test/test_archive_info.pm @@ -0,0 +1,54 @@ +package PVE::Storage::TestArchiveInfo; + +use strict; +use warnings; + +use lib qw(..); + +use PVE::Storage; +use Test::More; + +my @tests = ( + # backup archives + [ 'backup/vzdump-qemu-16110-2020_03_30-21_12_40.vma', { 'type' => 'qemu', 'format' => 'vma', 'decompressor' => undef, 'compression' => undef }, 'Backup archive, vma' ], + [ 'backup/vzdump-qemu-16110-2020_03_30-21_12_40.vma.gz', { 'type' => 'qemu', 'format' => 'vma', 'decompressor' => ['zcat'], 'compression' => 'gz' }, 'Backup archive, vma, gz' ], + [ 'backup/vzdump-qemu-16110-2020_03_30-21_12_40.vma.lzo', { 'type' => 'qemu', 'format' => 'vma', 'decompressor' => ['lzop', '-d', '-c'], 'compression' => 'lzo' }, 'Backup archive, vma, lzo' ], + + [ 'backup/vzdump-lxc-16112-2020_03_30-21_39_30.tar', { 'type' => 'lxc', 'format' => 'tar', 'decompressor' => undef, 'compression' => undef }, 'Backup archive, lxc' ], + [ 'backup/vzdump-lxc-16112-2020_03_30-21_39_30.tar.gz', { 'type' => 'lxc', 'format' => 'tar', 'decompressor' => ['tar', '-z'], 'compression' => 'gz' }, 'Backup archive, lxc, gz' ], + [ 'backup/vzdump-lxc-16112-2020_03_30-21_39_30.tgz', { 'type' => 'lxc', 'format' => 'tar', 'decompressor' => ['tar', '-z'], 'compression' => 'gz' }, 'Backup archive, lxc, tgz' ], + [ 'backup/vzdump-lxc-16112-2020_03_30-21_39_30.tar.lzo', { 'type' => 'lxc', 'format' => 'tar', 'decompressor' => ['tar', '--lzop'], 'compression' => 'lzo' }, 'Backup archive, lxc, lzo' ], + + [ 'backup/vzdump-openvz-16112-2020_03_30-21_39_30.tar', { 'type' => 'openvz', 'format' => 'tar', 'decompressor' => undef, 'compression' => undef }, 'Backup archive, openvz' ], + [ 'backup/vzdump-openvz-16112-2020_03_30-21_39_30.tar.gz', { 'type' => 'openvz', 'format' => 'tar', 'decompressor' => ['tar', '-z'], 'compression' => 'gz' }, 'Backup archive, openvz, gz' ], + [ 'backup/vzdump-openvz-16112-2020_03_30-21_39_30.tgz', { 'type' => 'openvz', 'format' => 'tar', 'decompressor' => ['tar', '-z'], 'compression' => 'gz' }, 'Backup archive, openvz, tgz' ], + [ 'backup/vzdump-openvz-16112-2020_03_30-21_39_30.tar.lzo', { 'type' => 'openvz', 'format' => 'tar', 'decompressor' => ['tar', '--lzop'], 'compression' => 'lzo' }, 'Backup archive, openvz, lzo' ], + + # failed matches + [ 'backup/vzdump-none-16112-2020_03_30-21_39_30.tar', "ERROR: couldn't determine format and compression type\n", 'Failed match: Backup archive, no virtualization type' ], + [ 'backup/vzdump-lxc-16112-2020_03_30-21_39_30', "ERROR: couldn't determine format and compression type\n", 'Failed match: Backup archive, lxc, no ending' ], + [ 'backup/vzdump-lxc-16112-2020_03_30-21_39_30.zip', "ERROR: couldn't determine format and compression type\n", 'Failed match: Backup archive, lxc, zip' ], + [ 'backup/vzdump-openvz-16112-2020_03_30-21_39_30.tgz.lzo', "ERROR: couldn't determine format and compression type\n", 'Failed match: Backup archive, openvz, tgz.lzo' ], + [ 'backup/vzdump-openvz-16112-2020_03_30-21_39_30.tar.bz2', "ERROR: couldn't determine format and compression type\n", 'Failed match: Backup archive, openvz, bz2' ], + [ 'backup/vzdump-openvz-16112-2020_03_30-21_39_30.zip.gz', "ERROR: couldn't determine format and compression type\n", 'Failed match: Backup archive, openvz, zip.gz' ], + [ 'backup/vzdump-qemu-16110-2020_03_30-21_12_40.vma.xz', "ERROR: couldn't determine format and compression type\n", 'Failed match: Backup archive, vma, xz' ], + [ 'backup/vzdump-qemu-16110-2020_03_30-21_12_40.vms.gz', "ERROR: couldn't determine format and compression type\n", 'Failed match: Backup archive, vms, gz' ], +); + + + + +plan tests => scalar @tests; + +foreach my $t (@tests) { + my ($archive, $expected, $description) = @$t; + my $got; + eval { $got = PVE::Storage::archive_info($archive) }; + $got = $@ if $@; + + is_deeply($got, $expected, $description) || diag(explain($got)); +} + +done_testing(); + +1; -- 2.20.1 _______________________________________________ pve-devel mailing list pve-devel@pve.proxmox.com https://pve.proxmox.com/cgi-bin/mailman/listinfo/pve-devel