2 #============================================================= -*-perl-*-
4 # configure.pl: Configuration and installation program for BackupPC
8 # This script should be run as root:
12 # The installation steps are described as the script runs.
15 # Craig Barratt <cbarratt@users.sourceforge.net>
18 # Copyright (C) 2001-2003 Craig Barratt
20 # This program is free software; you can redistribute it and/or modify
21 # it under the terms of the GNU General Public License as published by
22 # the Free Software Foundation; either version 2 of the License, or
23 # (at your option) any later version.
25 # This program is distributed in the hope that it will be useful,
26 # but WITHOUT ANY WARRANTY; without even the implied warranty of
27 # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
28 # GNU General Public License for more details.
30 # You should have received a copy of the GNU General Public License
31 # along with this program; if not, write to the Free Software
32 # Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
34 #========================================================================
36 # Version 2.0.0_CVS, released 18 Jan 2003.
38 # See http://backuppc.sourceforge.net.
40 #========================================================================
43 use vars qw(%Conf %OrigConf);
46 my @Packages = qw(ExtUtils::MakeMaker File::Path File::Spec File::Copy
47 DirHandle Digest::MD5 Data::Dumper Getopt::Std
48 BackupPC::Lib BackupPC::FileZIO);
50 foreach my $pkg ( @Packages ) {
55 BackupPC needs the package $pkg. Please install $pkg
56 before installing BackupPC.
64 This configure script should be run as root, rather than uid $<.
65 Provided uid $< has sufficient permissions to create the data and
66 install directories, then it should be ok to proceed. Otherwise,
67 please quit and restart as root.
70 exit unless prompt("--> Do you want to continue?", "y") =~ /y/i;
75 Is this a new installation or upgrade for BackupPC? If this is
76 an upgrade please tell me the full path of the existing BackupPC
77 configuration file (eg: /xxxx/conf/config.pl). Otherwise, just
83 # Check if this is an upgrade, in which case read the existing
84 # config file to get all the defaults.
88 $ConfigPath = prompt("--> Full path to existing conf/config.pl",
90 last if ( $ConfigPath eq ""
91 || ($ConfigPath =~ /^\// && -r $ConfigPath && -w $ConfigPath) );
92 my $problem = "is not an absolute path";
93 $problem = "is not writable" if ( !-w $ConfigPath );
94 $problem = "is not readable" if ( !-r $ConfigPath );
95 $problem = "doesn't exist" if ( !-f $ConfigPath );
96 print("The file '$ConfigPath' $problem.\n");
99 if ( $ConfigPath ne "" && -r $ConfigPath ) {
100 (my $topDir = $ConfigPath) =~ s{/[^/]+/[^/]+$}{};
101 die("BackupPC::Lib->new failed\n")
102 if ( !($bpc = BackupPC::Lib->new($topDir, ".", 1)) );
103 %Conf = $bpc->Conf();
105 $Conf{TopDir} = $topDir;
106 my $err = $bpc->ServerConnect($Conf{ServerHost}, $Conf{ServerPort}, 1);
110 BackupPC is running on $Conf{ServerHost}. You need to stop BackupPC
111 before you can upgrade the code. Depending upon your installation,
112 you could run "/etc/init.d/backuppc stop".
120 # These are the programs whose paths we need to find
124 'gtar/tar' => "TarClientPath",
125 smbclient => "SmbClientPath",
126 nmblookup => "NmbLookupPath",
127 rsync => "RsyncClientPath",
130 'ssh/ssh2' => "SshPath",
131 sendmail => "SendmailPath",
132 hostname => "HostnamePath",
135 foreach my $prog ( sort(keys(%Programs)) ) {
137 foreach my $subProg ( split(/\//, $prog) ) {
138 $path ||= FindProgram("$ENV{PATH}:/bin:/usr/bin:/sbin:/usr/sbin",
141 $Conf{$Programs{$prog}} ||= $path;
147 I found the following locations for these programs:
150 foreach my $prog ( sort(keys(%Programs)) ) {
151 printf(" %-11s => %s\n", $prog, $Conf{$Programs{$prog}});
154 last if (prompt('--> Are these paths correct?', 'y') =~ /^y/i);
155 foreach my $prog ( sort(keys(%Programs)) ) {
156 $Conf{$Programs{$prog}} = prompt("--> $prog path",
157 $Conf{$Programs{$prog}});
161 my $Perl56 = system($Conf{PerlPath}
162 . q{ -e 'exit($^V && $^V ge v5.6.0 ? 1 : 0);'});
167 BackupPC needs perl version 5.6.0 or later. $Conf{PerlPath} appears
168 to be an older version. Please upgrade to a newer version of perl
169 and re-run this configure script.
177 Please tell me the hostname of the machine that BackupPC will run on.
180 chomp($Conf{ServerHost} = `$Conf{HostnamePath}`)
181 if ( defined($Conf{HostnamePath}) && !defined($Conf{ServerHost}) );
182 $Conf{ServerHost} = prompt("--> BackupPC will run on host", $Conf{ServerHost});
186 BackupPC should run as a dedicated user with limited privileges. You
187 need to create a user. This user will need read/write permission on
188 the main data directory and read/execute permission on the install
189 directory (these directories will be setup shortly).
191 The primary group for this user should also be chosen carefully.
192 By default the install directories will have group write permission.
193 The data directories and files will have group read permission but
197 my($name, $passwd, $Uid, $Gid);
199 $Conf{BackupPCUser} = prompt("--> BackupPC should run as user",
200 $Conf{BackupPCUser} || "backuppc");
201 ($name, $passwd, $Uid, $Gid) = getpwnam($Conf{BackupPCUser});
202 last if ( $name ne "" );
205 getpwnam() says that user $Conf{BackupPCUser} doesn't exist. Please check the
206 name and verify that this user is in the passwd file.
213 Please specify an install directory for BackupPC. This is where the
214 BackupPC scripts, library and documentation will be installed.
219 $Conf{InstallDir} = prompt("--> Install directory (full path)",
221 last if ( $Conf{InstallDir} =~ /^\// );
226 Please specify a data directory for BackupPC. This is where the
227 configuration files, LOG files and all the PC backups are stored.
228 This file system needs to be big enough to accommodate all the
229 PCs you expect to backup (eg: at least 1-2GB per machine).
234 $Conf{TopDir} = prompt("--> Data directory (full path)", $Conf{TopDir});
235 last if ( $Conf{TopDir} =~ /^\// );
238 if ( !defined($Conf{CompressLevel}) ) {
239 $Conf{CompressLevel} = BackupPC::FileZIO->compOk ? 3 : 0;
240 if ( $ConfigPath eq "" && $Conf{CompressLevel} ) {
243 BackupPC can compress pool files, providing around a 40% reduction in pool
244 size (your mileage may vary). Specify the compression level (0 turns
245 off compression, and 1 to 9 represent good/fastest to best/slowest).
246 The recommended values are 0 (off) or 3 (reasonable compression and speed).
247 Increasing the compression level to 5 will use around 20% more cpu time
248 and give perhaps 2-3% more compression.
251 } elsif ( $ConfigPath eq "" ) {
254 BackupPC can compress pool files, but it needs the Compress::Zlib
255 package installed (see www.cpan.org). Compression will provide around a
256 40% reduction in pool size, at the expense of cpu time. You can leave
257 compression off and run BackupPC without compression, in which case you
258 should leave the compression level at 0 (which means off). You could
259 install Compress::Zlib and turn compression on later, but read the
260 documentation first about how to do this. Or the better choice is
261 to quit, install Compress::Zlib, and re-run configure.pl.
264 } elsif ( $Conf{CompressLevel} ) {
265 $Conf{CompressLevel} = 0;
268 BackupPC now supports pool file compression. Since you are upgrading
269 BackupPC you probably have existing uncompressed backups. You have
270 several choices if you want to turn on compression. You can run
271 the script BackupPC_compressPool to convert everything to compressed
272 form. Or you can simply turn on compression, so that new backups
273 will be compressed. This will increase the pool storage requirement,
274 since both uncompressed and compressed copies of files will be stored.
275 But eventually the old uncompressed backups will expire, recovering
276 the pool storage. Please see the documentation for more details.
278 If you are not sure what to do, leave the Compression Level at 0,
279 which disables compression. You can always read the documentation
280 and turn it on later.
284 $Conf{CompressLevel} = 0;
287 BackupPC now supports pool file compression, but it needs the
288 Compress::Zlib module (see www.cpan.org). For now, leave
289 the compression level set at 0 to disable compression. If you
290 want you can install Compress::Zlib and turn compression on.
291 Please see the documentation for more details about converting
292 old backups to compressed form.
298 = prompt("--> Compression level", $Conf{CompressLevel});
299 last if ( $Conf{CompressLevel} =~ /^\d+$/ );
305 BackupPC has a powerful CGI perl interface that runs under Apache.
306 A single executable needs to be installed in a cgi-bin directory.
307 This executable needs to run as set-uid $Conf{BackupPCUser}, or
308 it can be run under mod_perl with Apache running as user $Conf{BackupPCUser}.
310 Leave this path empty if you don't want to install the CGI interface.
315 $Conf{CgiDir} = prompt("--> CGI bin directory (full path)", $Conf{CgiDir});
316 last if ( $Conf{CgiDir} =~ /^\// || $Conf{CgiDir} eq "" );
319 if ( $Conf{CgiDir} ne "" ) {
323 BackupPC's CGI script needs to display various GIF images that
324 should be stored where Apache can serve them. They should be
325 placed somewher under Apache's DocumentRoot. BackupPC also
326 needs to know the URL to access these images. Example:
328 Apache image directory: /usr/local/apache/htdocs/BackupPC
329 URL for image directory: /BackupPC
331 The URL for the image directory should start with a slash.
335 $Conf{CgiImageDir} = prompt("--> Apache image directory (full path)",
337 last if ( $Conf{CgiImageDir} =~ /^\// );
340 $Conf{CgiImageDirURL} = prompt("--> URL for image directory (omit http://host; starts with '/')",
341 $Conf{CgiImageDirURL});
342 last if ( $Conf{CgiImageDirURL} =~ /^\// );
350 - install the binaries, lib and docs in $Conf{InstallDir},
351 - create the data directory $Conf{TopDir},
352 - create/update the config.pl file $Conf{TopDir}/conf,
353 - optionally install the cgi-bin interface.
357 exit unless prompt("--> Do you want to continue?", "y") =~ /y/i;
360 # Create install directories
362 foreach my $dir ( qw(bin lib/BackupPC/Xfer lib/BackupPC/Zip
363 lib/BackupPC/Lang doc) ) {
364 next if ( -d "$Conf{InstallDir}/$dir" );
365 mkpath("$Conf{InstallDir}/$dir", 0, 0775);
366 if ( !-d "$Conf{InstallDir}/$dir"
367 || !chown($Uid, $Gid, "$Conf{InstallDir}/$dir") ) {
368 die("Failed to create or chown $Conf{InstallDir}/$dir\n");
370 print("Created $Conf{InstallDir}/$dir\n");
375 # Create CGI image directory
377 foreach my $dir ( ($Conf{CgiImageDir}) ) {
378 next if ( $dir eq "" || -d $dir );
379 mkpath($dir, 0, 0775);
380 if ( !-d $dir || !chown($Uid, $Gid, $dir) ) {
381 die("Failed to create or chown $dir");
383 print("Created $dir\n");
388 # Create $TopDir's top-level directories
390 foreach my $dir ( qw(. conf pool cpool pc trash log) ) {
391 mkpath("$Conf{TopDir}/$dir", 0, 0750) if ( !-d "$Conf{TopDir}/$dir" );
392 if ( !-d "$Conf{TopDir}/$dir"
393 || !chown($Uid, $Gid, "$Conf{TopDir}/$dir") ) {
394 die("Failed to create or chown $Conf{TopDir}/$dir\n");
396 print("Created $Conf{TopDir}/$dir\n");
400 printf("Installing binaries in $Conf{InstallDir}/bin\n");
401 foreach my $prog ( qw(BackupPC BackupPC_dump BackupPC_link BackupPC_nightly
402 BackupPC_sendEmail BackupPC_tarCreate BackupPC_trashClean
403 BackupPC_tarExtract BackupPC_compressPool BackupPC_zcat
404 BackupPC_restore BackupPC_serverMesg BackupPC_zipCreate ) ) {
405 InstallFile("bin/$prog", "$Conf{InstallDir}/bin/$prog", 0555);
409 # Remove unused binaries from older versions
411 unlink("$Conf{InstallDir}/bin/BackupPC_queueAll");
413 printf("Installing library in $Conf{InstallDir}/lib\n");
414 foreach my $lib ( qw(BackupPC/Lib.pm BackupPC/FileZIO.pm BackupPC/Attrib.pm
415 BackupPC/PoolWrite.pm BackupPC/View.pm BackupPC/Xfer/Tar.pm
416 BackupPC/Xfer/Smb.pm BackupPC/Xfer/Rsync.pm
417 BackupPC/Xfer/RsyncFileIO.pm BackupPC/Zip/FileMember.pm
418 BackupPC/Lang/en.pm BackupPC/Lang/fr.pm BackupPC/Lang/es.pm
421 InstallFile("lib/$lib", "$Conf{InstallDir}/lib/$lib", 0444);
424 if ( $Conf{CgiImageDir} ne "" ) {
425 printf("Installing images in $Conf{CgiImageDir}\n");
426 foreach my $img ( <images/*> ) {
427 (my $destImg = $img) =~ s{^images/}{};
428 InstallFile($img, "$Conf{CgiImageDir}/$destImg", 0444, 1);
432 printf("Making init.d scripts\n");
433 foreach my $init ( qw(gentoo-backuppc gentoo-backuppc.conf linux-backuppc
434 solaris-backuppc debian-backuppc suse-backuppc) ) {
435 InstallFile("init.d/src/$init", "init.d/$init", 0444);
438 printf("Installing docs in $Conf{InstallDir}/doc\n");
439 foreach my $doc ( qw(BackupPC.pod BackupPC.html) ) {
440 InstallFile("doc/$doc", "$Conf{InstallDir}/doc/$doc", 0444);
443 printf("Installing config.pl and hosts in $Conf{TopDir}/conf\n");
444 InstallFile("conf/hosts", "$Conf{TopDir}/conf/hosts", 0644)
445 if ( !-f "$Conf{TopDir}/conf/hosts" );
448 # Now do the config file. If there is an existing config file we
449 # merge in the new config file, adding any new configuration
450 # parameters and deleting ones that are no longer needed.
452 my $dest = "$Conf{TopDir}/conf/config.pl";
453 my ($newConf, $newVars) = ConfigParse("conf/config.pl");
454 my ($oldConf, $oldVars);
456 ($oldConf, $oldVars) = ConfigParse($dest);
457 $newConf = ConfigMerge($oldConf, $oldVars, $newConf, $newVars);
459 $Conf{EMailFromUserName} ||= $Conf{BackupPCUser};
460 $Conf{EMailAdminUserName} ||= $Conf{BackupPCUser};
463 # Update various config parameters
467 # Guess $Conf{CgiURL}
469 if ( !defined($Conf{CgiURL}) ) {
470 if ( $Conf{CgiDir} =~ m{cgi-bin(/.*)} ) {
471 $Conf{CgiURL} = "'http://$Conf{ServerHost}/cgi-bin$1/BackupPC_Admin'";
473 $Conf{CgiURL} = "'http://$Conf{ServerHost}/cgi-bin/BackupPC_Admin'";
478 # The smbclient commands have moved from hard-coded to the config file.
479 # $Conf{SmbClientArgs} no longer exists, so merge it into the new
480 # commands if it still exists.
482 if ( defined($Conf{SmbClientArgs}) ) {
483 if ( $Conf{SmbClientArgs} ne "" ) {
484 foreach my $param ( qw(SmbClientRestoreCmd SmbClientFullCmd
485 SmbClientIncrCmd) ) {
486 $newConf->[$newVars->{$param}]{text}
487 =~ s/(-E\s+-N)/$1 $Conf{SmbClientArgs}/;
490 delete($Conf{SmbClientArgs});
494 # IncrFill should now be off
499 # Figure out sensible arguments for the ping command
501 if ( defined($Conf{PingArgs}) ) {
502 $Conf{PingCmd} = '$pingPath ' . $Conf{PingArgs};
503 } elsif ( !defined($Conf{PingCmd}) ) {
504 if ( $^O eq "solaris" || $^O eq "sunos" ) {
505 $Conf{PingCmd} = '$pingPath -s $host 56 1';
506 } elsif ( ($^O eq "linux" || $^O eq "openbsd" || $^O eq "netbsd")
507 && !system("$Conf{PingClientPath} -c 1 -w 3 localhost") ) {
508 $Conf{PingCmd} = '$pingPath -c 1 -w 3 $host';
510 $Conf{PingCmd} = '$pingPath -c 1 $host';
512 delete($Conf{PingArgs});
516 # Figure out sensible arguments for the df command
518 if ( !defined($Conf{DfCmd}) ) {
519 if ( $^O eq "solaris" || $^O eq "sunos" ) {
520 $Conf{DfCmd} = '$dfPath -k $topDir';
525 # $Conf{SmbClientTimeout} is now $Conf{ClientTimeout}
527 if ( defined($Conf{SmbClientTimeout}) ) {
528 $Conf{ClientTimeout} = $Conf{SmbClientTimeout};
529 delete($Conf{SmbClientTimeout});
532 my $confCopy = "$dest.pre-__VERSION__";
533 if ( -f $dest && !-f $confCopy ) {
535 # Make copy of config file, preserving ownership and modes
537 printf("Making backup copy of $dest -> $confCopy\n");
538 my @stat = stat($dest);
542 die("can't copy($dest, $confCopy)\n") unless copy($dest, $confCopy);
543 die("can't chown $uid, $gid $confCopy\n")
544 unless chown($uid, $gid, $confCopy);
545 die("can't chmod $mode $confCopy\n") unless chmod($mode, $confCopy);
547 open(OUT, ">$dest") || die("can't open $dest for writing\n");
549 foreach my $var ( @$newConf ) {
550 if ( length($blockComment)
551 && substr($var->{text}, 0, length($blockComment)) eq $blockComment ) {
552 $var->{text} = substr($var->{text}, length($blockComment));
553 $blockComment = undef;
555 $blockComment = $1 if ( $var->{text} =~ /^([\s\n]*#{70}.*#{70}[\s\n]+)/s );
556 $var->{text} =~ s/^\s*\$Conf\{(.*?)\}(\s*=\s*['"]?)(.*?)(['"]?\s*;)/
557 defined($Conf{$1}) && ref($Conf{$1}) eq ""
558 && $Conf{$1} ne $OrigConf{$1}
559 ? "\$Conf{$1}$2$Conf{$1}$4"
560 : "\$Conf{$1}$2$3$4"/emg;
561 print OUT $var->{text};
564 if ( !defined($oldConf) ) {
565 die("can't chmod 0640 mode $dest\n") unless chmod(0640, $dest);
566 die("can't chown $Uid, $Gid $dest\n") unless chown($Uid, $Gid, $dest);
569 if ( $Conf{CgiDir} ne "" ) {
570 printf("Installing cgi script BackupPC_Admin in $Conf{CgiDir}\n");
571 mkpath("$Conf{CgiDir}", 0, 0755);
572 InstallFile("cgi-bin/BackupPC_Admin", "$Conf{CgiDir}/BackupPC_Admin",
578 Ok, it looks like we are finished. There are several more things you
581 - Browse through the config file, $Conf{TopDir}/conf/config.pl,
582 and make sure all the settings are correct. In particular, you
583 will need to set the smb share password and user name, backup
584 policies and check the email message headers and bodies.
586 - Edit the list of hosts to backup in $Conf{TopDir}/conf/hosts.
588 - Read the documentation in $Conf{InstallDir}/doc/BackupPC.html.
589 Please pay special attention to the security section.
591 - Verify that the CGI script BackupPC_Admin runs correctly. You might
592 need to change the permissions or group ownership of BackupPC_Admin.
594 - BackupPC should be ready to start. Don't forget to run it
595 as user $Conf{BackupPCUser}! The installation also contains an
596 init.d/backuppc script that can be copied to /etc/init.d
597 so that BackupPC can auto-start on boot. See init.d/README.
602 if ( $ENV{LANG} =~ /utf/i && $^V ge v5.8.0 ) {
605 WARNING: Your LANG environment variable is set to $ENV{LANG}, which
606 doesn't behave well with this version of perl. Please set the
607 LANG environment variable to en_US before running BackupPC.
609 On RH-8 this setting is in the file /etc/sysconfig/i18n, or you
610 could set it in BackupPC's init.d script.
616 ###########################################################################
618 ###########################################################################
622 my($prog, $dest, $mode, $binary) = @_;
624 my($uid, $gid) = ($Uid, $Gid);
628 # preserve ownership and modes of files that already exist
630 my @stat = stat($dest);
635 unlink($dest) if ( -f $dest );
637 die("can't copy($prog, $dest)\n") unless copy($prog, $dest);
639 open(PROG, $prog) || die("can't open $prog for reading\n");
640 open(OUT, ">$dest") || die("can't open $dest for writing\n");
642 s/__INSTALLDIR__/$Conf{InstallDir}/g;
643 s/__TOPDIR__/$Conf{TopDir}/g;
644 s/__BACKUPPCUSER__/$Conf{BackupPCUser}/g;
645 s/__CGIDIR__/$Conf{CgiDir}/g;
646 if ( $first && /^#.*bin\/perl/ ) {
649 # perl56 and later is taint ok
651 print OUT "#!$Conf{PerlPath} -T\n";
654 # prior to perl56, File::Find fails taint checks,
655 # so we run without -T. It's still safe.
657 print OUT "#!$Conf{PerlPath}\n";
667 die("can't chown $uid, $gid $dest") unless chown($uid, $gid, $dest);
668 die("can't chmod $mode $dest") unless chmod($mode, $dest);
673 my($path, $prog) = @_;
674 foreach my $dir ( split(/:/, $path) ) {
675 my $file = File::Spec->catfile($dir, $prog);
676 return $file if ( -x $file );
683 open(C, $file) || die("can't open $file");
684 my($out, @conf, $var);
693 $allVars->{$var} = @conf if ( defined($var) );
703 } elsif ( /^\s*\$Conf\{([^}]*)/ ) {
705 if ( defined($var) ) {
706 $allVars->{$var} = @conf if ( defined($var) );
721 $allVars->{$var} = @conf if ( defined($var) );
728 return (\@conf, $allVars);
733 my($old, $oldVars, $new, $newVars) = @_;
738 # Find which config parameters are not needed any longer
740 foreach my $var ( @$old ) {
741 next if ( !defined($var->{var}) || defined($newVars->{$var->{var}}) );
742 #print(STDERR "Deleting old config parameter $var->{var}\n");
746 # Find which config parameters are new
748 foreach my $var ( @$new ) {
749 next if ( !defined($var->{var}) );
750 if ( defined($oldVars->{$var->{var}}) ) {
751 $posn = $oldVars->{$var->{var}};
753 #print(STDERR "New config parameter $var->{var}: $var->{text}\n");
754 push(@{$old->[$posn]{new}}, $var);
758 # Create merged config file
760 foreach my $var ( @$old ) {
761 next if ( $var->{delete} );
763 foreach my $new ( @{$var->{new}} ) {