# Copyright 1999-2014. Parallels IP Holdings GmbH. All Rights Reserved. package Storage::TarBundle; use strict; use Storage::Bundle; use AgentConfig; use vars qw|@ISA $FreeBSD|; @ISA=qw|Storage::Bundle|; $FreeBSD = isFreeBSD(); sub isFreeBSD { my $os = `uname -s`; chomp($os); return ($os eq 'FreeBSD'); } sub _init { my ($self, %options) = @_; $self->SUPER::_init(%options); $self->{follow_symlinks} = 1 if defined $options{follow_symlinks}; $self->{add_file} = []; if (exists $options{add_file}) { push @{$self->{add_file}}, $options{add_file}; } if (exists $options{include}) { $self->{include} = $self->fileListToFile($options{include}); if (not defined $self->{include}) { return undef; } } if (exists $options{exclude}) { $self->{exclude} = $self->fileListToFile($options{exclude}); if (not defined $self->{exclude}) { return undef; } } if (exists $options{no_recursion}) { $self->{no_recursion} = 1; } $self->{include_hidden_files} = 1 if defined $options{include_hidden_files}; return -d $self->{srcdir}; } sub fileListToFile { my ($self, $fileList) = @_; #FIXME: detect who is passing non-chomped filenames chomp @{$fileList}; my @list; foreach my $file ( @{$fileList} ) { if ( substr($file,0,1) eq "-" ) { push @{$self->{add_file}}, $file; } elsif ( $file ne '' ) { push @list, $file; } } if ( @list ) { my $fileName = POSIX::tmpnam(); if (open FILELIST, ">$fileName") { close FILELIST; if (defined $self->{sysuser}) { chown 0 + getpwnam($self->{sysuser}), 0 + getgrnam("root"), $fileName or die($fileName); } chmod 0600, $fileName; open FILELIST, ">$fileName"; foreach my $file ( @list ) { print FILELIST "$file\n"; } close FILELIST; return $fileName; } else { return undef; } } return; } sub commandLine { my ($self) = @_; # FIXME: check tar presence my $tar = AgentConfig::tarBin(); # Explicit -f - here because FreeBSD requires it my $cmd = "$tar -f - -c"; $cmd .= " -h" if exists $self->{follow_symlinks}; # FIXME: --ignore-failed-read $cmd .= " --no-recursion" if exists $self->{no_recursion}; if ( defined $self->{exclude} ) { $cmd .= " --anchored "; $cmd .= " -X '$self->{exclude}'"; } if ( defined $self->{include} ) { $cmd .= " -T '$self->{include}'"; } if ( @{$self->{add_file}} ) { foreach my $file ( @{$self->{add_file}} ) { $cmd .= " --add-file='$file'"; } } if ( ( !defined $self->{include} ) && ( !(@{$self->{add_file}}) ) ) { # When using .(dot) tar doesn't omit hidden files (i.e.starting with dot) # All files in archive are preceeded with './' # Tar archive contains: # ./ # ./file1 # ./.file2 # ./directory1 # ./.directory2 # ./directory1/file3 # ./directory1/.file4 # As a side effect, tar does backup owner, ownergroup and perms of top-level directory if (exists $self->{include_hidden_files}) { $cmd .= " -- ."; } # When using *(asterisk) tar does omit hidden files in top directory only # All files in archive arent' preceeded with ./ # file1 # directory1 # directory1/file3 # directory1/.file4 else { $cmd .= " -- *"; } } return $cmd; } sub cleanup { my ($self) = @_; my $exit_code = $self->SUPER::cleanup(); unlink($self->{include}) if exists $self->{include}; unlink($self->{exclude}) if exists $self->{exclude}; if (1 == $exit_code) { # According to GNU tar specification (http://www.gnu.org/software/tar/manual/tar.html) exit code 1 means that some files were changed while being archived Logging::debug("Tar bundle. Ignore files changes during archive creation: replace exit code $exit_code to 0"); $exit_code = 0; } return $exit_code; } sub filterStderr { my ($self, $stderr) = @_; if ($stderr =~ qr/^\/bin\/tar: (.+): Cannot (open|savedir): Permission denied$/m) { my @problemFiles; my @problems = split(/\n/ , $stderr); foreach my $problem (@problems) { if ($problem =~ qr/^\/bin\/tar: (.+): Cannot (open|savedir): Permission denied$/) { push @problemFiles, $self->{srcdir} . "/" . $1; } } return "For security reasons, backing up is performed on behalf of subscription's system user. This system user has no read access to:\n" . join("\n", @problemFiles) . "\nSo it was not backed up. All other data was backed up successfully. To fix this issue you may grant access read/write to the file or directory for system user \"" . $self->{sysuser} . "\" or \"apache\"."; } else { return $stderr; } } 1;