mirror of
				https://github.com/xcat2/xcat-core.git
				synced 2025-10-26 08:55:24 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			346 lines
		
	
	
		
			9.5 KiB
		
	
	
	
		
			Perl
		
	
	
	
	
	
			
		
		
	
	
			346 lines
		
	
	
		
			9.5 KiB
		
	
	
	
		
			Perl
		
	
	
	
	
	
| #!/usr/bin/perl
 | |
| # IBM(c) 2007 EPL license http://www.eclipse.org/legal/epl-v10.html
 | |
| #(C)IBM Corp
 | |
| 
 | |
| #
 | |
| 
 | |
| BEGIN
 | |
| {
 | |
|     $::XCATROOT =
 | |
|       $ENV{'XCATROOT'} ? $ENV{'XCATROOT'}
 | |
|       : -d '/opt/xcat' ? '/opt/xcat'
 | |
|       :                  '/usr';
 | |
| }
 | |
| use lib "$::XCATROOT/lib/perl";
 | |
| use Getopt::Long;
 | |
| use xCAT::MsgUtils;
 | |
| use xCAT::Utils;
 | |
| use strict;
 | |
| 
 | |
| #-----------------------------------------------------------------------------
 | |
| 
 | |
| =head1  restorexCATdb 
 | |
| 
 | |
| 
 | |
| 
 | |
|  restorexCATdb  -p <directory containing db restore .csv files> 
 | |
|  restorexCATdb -b -p <directory binary database backup> 
 | |
| 
 | |
| 
 | |
| =cut
 | |
| 
 | |
| #-----------------------------------------------------------------------------
 | |
| # Main
 | |
| my $rc = 0;
 | |
| 
 | |
| &parse_args;
 | |
| if ($::BINARY) {    # not using xCAT to dump, using the database utility
 | |
|     my $DBname = xCAT::Utils->get_DBName;
 | |
|     if ($DBname eq "DB2") {
 | |
|         $rc = &DB2_binrestore;
 | |
|         if ($rc == 0) {
 | |
|             xCAT::MsgUtils->message("I", "Restore Complete. You can now restart the daemons.");
 | |
|         } else {
 | |
|             xCAT::MsgUtils->message("E", "Restore Failed.");
 | |
|         }
 | |
|         exit $rc;
 | |
|     } else {
 | |
|         if ($DBname eq "PG") {
 | |
|             $rc = &PG_binrestore;
 | |
|             if ($rc == 0) {
 | |
|                 xCAT::MsgUtils->message("I", "Restore Complete.");
 | |
|             } else {
 | |
|                 xCAT::MsgUtils->message("I", "Restore Failed.");
 | |
|             }
 | |
|             exit $rc;
 | |
|         } else {
 | |
|             xCAT::MsgUtils->message("E",
 | |
|                 "Binary dump (-b) is not supported for $DBname");
 | |
|             exit 1;
 | |
|         }
 | |
|     }
 | |
| }
 | |
| 
 | |
| # read all the *.csv files from the input directory and restore the database
 | |
| opendir DIRPATH, $::PATH;
 | |
| if ($? != 0)
 | |
| {
 | |
|     my $msg = " Unable to read directory $::PATH \n";
 | |
|     xCAT::MsgUtils->message("E", $msg);
 | |
|     exit 1;
 | |
| }
 | |
| 
 | |
| # read tables to skip during restore from site.skiptables attribute
 | |
| my @output2 = xCAT::TableUtils->get_site_attribute('skiptables');
 | |
| my @skiptbls;
 | |
| if (@output2) {
 | |
|     @skiptbls = split(/\,/, $output2[0]);
 | |
| }
 | |
| 
 | |
| my @files = readdir(DIRPATH);
 | |
| foreach my $table (@files)
 | |
| {
 | |
|     if ($table ne '.' and $table ne '..')
 | |
|     {
 | |
|         my $tablename = $::PATH;
 | |
|         $tablename .= "/";
 | |
|         $tablename .= $table;
 | |
|         if ((!$::ALL) && (($table =~ /^eventlog/) || ($table =~ /^auditlog/))) {
 | |
| 
 | |
|             if ($::VERBOSEREST) {
 | |
|                 xCAT::MsgUtils->message("I", "Skipping $table\n");
 | |
|             }
 | |
|             next;
 | |
|         }
 | |
| 
 | |
|         # skip teal tables
 | |
|         if ($table =~ /^x_teal/) {
 | |
|             if ($::VERBOSEREST) {
 | |
|                 xCAT::MsgUtils->message("I", "Skipping $table\n");
 | |
|             }
 | |
|             next;
 | |
|         }
 | |
| 
 | |
|         # skip ISNM tables except isnm_config
 | |
|         if ($table =~ /^isnm_perf/) {
 | |
|             if ($::DUMPVERBOSE) {
 | |
|                 xCAT::MsgUtils->message("I", "Skipping $table\n");
 | |
|             }
 | |
|             next;
 | |
|         }
 | |
| 
 | |
|         # skip and table in the site.skiptables attribute
 | |
|         if (@skiptbls) {
 | |
|             my ($tmptable, $suffix) = split(/\./, $table);
 | |
|             chomp $tmptable;
 | |
|             if (grep(/^$tmptable$/, @skiptbls)) {
 | |
|                 if ($::VERBOSEREST) {
 | |
|                     xCAT::MsgUtils->message("I", "Skipping $tmptable\n");
 | |
|                 }
 | |
|                 next;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         my $cmd = "tabrestore $tablename";
 | |
|         my @errout = xCAT::Utils->runcmd($cmd, 0);
 | |
|         if ($::RUNCMD_RC != 0)
 | |
|         {    # error
 | |
|             xCAT::MsgUtils->message("E", "Error running $cmd, @errout\n");
 | |
|         } else {
 | |
|             if ($::VERBOSEREST) {
 | |
|                 xCAT::MsgUtils->message("I", "Restoring $table.\n");
 | |
|             }
 | |
|         }
 | |
|     }
 | |
| }
 | |
| closedir DIRPATH;
 | |
| xCAT::MsgUtils->message("I", "Restore of Database Complete.");
 | |
| exit $rc;
 | |
| 
 | |
| #-----------------------------------------------------------------------------
 | |
| 
 | |
| =head3 parse_args
 | |
|   
 | |
|   Parses for  input
 | |
| 
 | |
| =cut
 | |
| 
 | |
| #-----------------------------------------------------------------------------
 | |
| sub parse_args
 | |
| {
 | |
|     my $msg;
 | |
|     my $usagemsg =
 | |
| " restorexCATdb  -h \n restorexCATdb -v \n restorexCATdb [-a] [-V] <-p> [path to restore .csv files]\n restorexCATdb -b [-V] <-t> [timestamp on restore backup file]  <-p> [path to restore backup file]";
 | |
|     Getopt::Long::Configure("posix_default");
 | |
|     Getopt::Long::Configure("no_gnu_compat");
 | |
|     Getopt::Long::Configure("bundling");
 | |
|     if (
 | |
|         !GetOptions(
 | |
|             'a|all'         => \$::ALL,
 | |
|             'b|binary'      => \$::BINARY,
 | |
|             'p|path=s'      => \$::PATH,
 | |
|             't|timestamp=s' => \$::TIMESTAMP,
 | |
|             'h|help'        => \$::HELP,
 | |
|             'V|verbose'     => \$::VERBOSEREST,
 | |
|             'v|version'     => \$::VERSION
 | |
| 
 | |
|         )
 | |
|       )
 | |
|     {
 | |
|         xCAT::MsgUtils->message("E", $usagemsg);
 | |
|         exit 1;
 | |
|     }
 | |
|     if ($::HELP)
 | |
|     {
 | |
|         xCAT::MsgUtils->message("I", $usagemsg);
 | |
|         exit 0;
 | |
|     }
 | |
|     if ($::VERSION)
 | |
|     {
 | |
|         my $version = xCAT::Utils->Version();
 | |
|         xCAT::MsgUtils->message("N", $version);
 | |
| 
 | |
|         exit 0;
 | |
|     }
 | |
|     if (!($::PATH))
 | |
|     {
 | |
|         my $msg = " -p with path to database files is required \n";
 | |
|         xCAT::MsgUtils->message("E", $msg);
 | |
|         exit 1;
 | |
|     }
 | |
|     if (!(-e $::PATH))
 | |
|     {
 | |
|         my $msg = " Input path must exist \n";
 | |
|         xCAT::MsgUtils->message("E", $msg);
 | |
|         exit 1;
 | |
|     }
 | |
| 
 | |
| }
 | |
| 
 | |
| #-----------------------------------------------------------------------------
 | |
| 
 | |
| =head3 DB2_binrestore
 | |
| 
 | |
|     Uses the DB2 Database supplied restore utility to restore the database
 | |
| 
 | |
| =cut
 | |
| 
 | |
| #-----------------------------------------------------------------------------
 | |
| sub DB2_binrestore
 | |
| {
 | |
|     my $msg;
 | |
|     my $rc = 0;
 | |
|     if (!($::TIMESTAMP)) {
 | |
|         xCAT::MsgUtils->message("E", "To restore from a binary backup, the timestamp of the backup must be provided with the -t option.");
 | |
|         return 1;
 | |
|     }
 | |
| 
 | |
|     # check to see if  xcatd running.   Cannot restore if it is
 | |
|     my $cmd = "ps -ef | grep xcatd";
 | |
|     my @output = xCAT::Utils->runcmd($cmd, -1);
 | |
|     if ($::RUNCMD_RC == 0) {    # found something
 | |
|         if (grep(/xcatd: SSL listener/, @output)) {    # xcatd running
 | |
|             xCAT::MsgUtils->message("E", "xcatd is still accessing the database. All applications accessing the DB2 database must be stopped before you can restore.");
 | |
|             return 1;
 | |
| 
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     # get password from cfgloc
 | |
| 
 | |
|     my $cmd = "cat /etc/xcat/cfgloc";
 | |
|     my $info = xCAT::Utils->runcmd($cmd, -1);
 | |
|     if ($::RUNCMD_RC != 0) {
 | |
|         xCAT::MsgUtils->message("E", "Cannot read /etc/xcat/cfgloc.");
 | |
|         return 1;
 | |
|     }
 | |
|     chomp($info);
 | |
|     my ($db, $inst, $pw) = split(/\|/, $info);
 | |
| 
 | |
|     # Hopefully nothing else is accessing the database, so try restore
 | |
|     $cmd = "db2 restore db xcatdb user xcatdb using $pw from $::PATH taken at $::TIMESTAMP";
 | |
|     $rc = &rundb2cmd($cmd);    # must su to xcatdb
 | |
|     if ($rc != 0)
 | |
|     {
 | |
|         xCAT::MsgUtils->message("E", " $cmd error.");
 | |
|         return 1;
 | |
|     }
 | |
| 
 | |
|     #  Now roll forward any tranactions since the backup
 | |
|     $cmd = "db2 rollforward database xcatdb to end of logs and complete";
 | |
|     $rc = &rundb2cmd($cmd);    # must su to xcatdb
 | |
|     if ($rc != 0)
 | |
|     {
 | |
|         xCAT::MsgUtils->message("E", " $cmd error.");
 | |
|         return 1;
 | |
|     }
 | |
|     return 0;
 | |
| }
 | |
| 
 | |
| #-----------------------------------------------------------------------------
 | |
| 
 | |
| =head3 PG_binrestore 
 | |
|   
 | |
|     Uses the PostgreSQL Database supplied backup utility to restore the database
 | |
| 
 | |
| =cut
 | |
| 
 | |
| #-----------------------------------------------------------------------------
 | |
| sub PG_binrestore
 | |
| {
 | |
|     my $msg;
 | |
|     my $rc       = 0;
 | |
|     my $pgcmddir = "/usr/bin";
 | |
| 
 | |
|     # get path to Postgresql commands if running 9.X version
 | |
|     my $cmd = "rpm -qa | grep postgresql";
 | |
|     my @output = xCAT::Utils->runcmd($cmd, 0);
 | |
|     if ($::RUNCMD_RC != 0)
 | |
|     {
 | |
|         my $message =
 | |
| "\nPostgreSQL is not installed.  If on AIX, it should be first obtained from the xcat dependency tarballs and installed before running this command.\n If on Linux, install from the OS CDs.";
 | |
|         xCAT::MsgUtils->message("E", " $cmd failed. $message");
 | |
|         exit(1);
 | |
|     }
 | |
| 
 | |
|     # check if 9.X release,  setup different
 | |
|     if (grep(/postgresql9/, @output)) {    # postgresql 9.x
 | |
|             # figure out which 9.x release  and build path
 | |
|             # for example 9.x release /usr/pgsql-9.x/bin
 | |
|         my @parseout = split(/\-/,          $output[0]);
 | |
|         my @ptflevel = split("postgresql9", $parseout[0]);
 | |
|         my $postgres9 = @ptflevel[1];    # set it to the PTF level
 | |
|         $pgcmddir = "/usr/pgsql-9.$postgres9/bin";    # pg cmds location
 | |
|     }
 | |
| 
 | |
|     # Get database, admin from cfgloc file
 | |
|     my $cmd = "cat /etc/xcat/cfgloc";
 | |
|     my $info = xCAT::Utils->runcmd($cmd, -1);
 | |
|     if ($::RUNCMD_RC != 0) {
 | |
|         xCAT::MsgUtils->message("E", "Cannot read /etc/xcat/cfgloc.");
 | |
|         return 1;
 | |
|     }
 | |
|     chomp($info);
 | |
|     my ($db, $admin, $pw) = split(/\|/, $info);
 | |
|     my ($info1,  $info2) = split(/=/, $db);
 | |
|     my ($dbname, $host)  = split(/;/, $info2);
 | |
| 
 | |
|     # restore from backup file
 | |
|     my $cmd = "$pgcmddir/pg_restore  $::PATH -U postgres -d $dbname -c ";
 | |
|     my $info = xCAT::Utils->runcmd($cmd, -1);
 | |
|     if ($::RUNCMD_RC != 0) {
 | |
|         xCAT::MsgUtils->message("E", "$cmd failed");
 | |
|         return 1;
 | |
|     }
 | |
|     return 0;
 | |
| 
 | |
| }
 | |
| 
 | |
| #-----------------------------------------------------------------------------
 | |
| 
 | |
| =head3   rundb2cmd
 | |
| 
 | |
| 
 | |
|   Run a commmand as the xcatdb instance id
 | |
|   Input: command
 | |
| 
 | |
| =cut
 | |
| 
 | |
| #-----------------------------------------------------------------------------
 | |
| sub rundb2cmd
 | |
| {
 | |
|     my $orgcmd = shift;
 | |
|     my $rc     = 0;
 | |
|     my $cmd    = "\'";
 | |
|     $cmd .= $orgcmd;
 | |
|     $cmd .= ' 2>&1';
 | |
|     $cmd .= "\'";
 | |
|     system("su - xcatdb -c $cmd");
 | |
|     if ($? > 0)    # error
 | |
|     {
 | |
|         $rc = $? >> 8;
 | |
|     }
 | |
|     return ($rc);
 | |
| }
 | |
| 
 |