2014-03-06 03:53:13 +03:00
|
|
|
####################################################################################################################################
|
|
|
|
# DB MODULE
|
|
|
|
####################################################################################################################################
|
2016-04-14 15:30:54 +02:00
|
|
|
package pgBackRest::Db;
|
2014-03-06 03:53:13 +03:00
|
|
|
|
|
|
|
use strict;
|
2015-03-03 07:57:20 +02:00
|
|
|
use warnings FATAL => qw(all);
|
|
|
|
use Carp qw(confess);
|
2019-07-25 23:05:39 +02:00
|
|
|
use English '-no_match_vars';
|
2014-07-28 01:13:23 +03:00
|
|
|
|
2015-06-14 00:25:49 +02:00
|
|
|
use Exporter qw(import);
|
2015-08-08 23:11:20 +02:00
|
|
|
our @EXPORT = qw();
|
2015-06-14 00:25:49 +02:00
|
|
|
use Fcntl qw(O_RDONLY);
|
2015-06-30 04:07:42 +02:00
|
|
|
use File::Basename qw(dirname);
|
2019-07-25 23:05:39 +02:00
|
|
|
use JSON::PP;
|
2014-03-06 03:53:13 +03:00
|
|
|
|
2016-08-12 04:35:24 +02:00
|
|
|
use pgBackRest::DbVersion;
|
2016-04-14 15:30:54 +02:00
|
|
|
use pgBackRest::Common::Exception;
|
|
|
|
use pgBackRest::Common::Log;
|
|
|
|
use pgBackRest::Common::String;
|
|
|
|
use pgBackRest::Common::Wait;
|
|
|
|
use pgBackRest::Config::Config;
|
2016-05-11 15:21:39 +02:00
|
|
|
use pgBackRest::Manifest;
|
2017-06-09 23:51:41 +02:00
|
|
|
use pgBackRest::Protocol::Storage::Helper;
|
|
|
|
use pgBackRest::Version;
|
2014-03-06 03:53:13 +03:00
|
|
|
|
2015-06-14 00:25:49 +02:00
|
|
|
####################################################################################################################################
|
2016-05-14 16:33:12 +02:00
|
|
|
# Backup advisory lock
|
2015-06-14 00:25:49 +02:00
|
|
|
####################################################################################################################################
|
2016-05-14 16:33:12 +02:00
|
|
|
use constant DB_BACKUP_ADVISORY_LOCK => '12340078987004321';
|
|
|
|
push @EXPORT, qw(DB_BACKUP_ADVISORY_LOCK);
|
2015-03-12 18:15:19 +02:00
|
|
|
|
2015-08-08 23:11:20 +02:00
|
|
|
####################################################################################################################################
|
2016-05-14 16:33:12 +02:00
|
|
|
# Map the control and catalog versions to PostgreSQL version.
|
|
|
|
#
|
|
|
|
# The control version can be found in src/include/catalog/pg_control.h and may not change with every version of PostgreSQL but is
|
|
|
|
# still checked to detect custom builds which change the structure. The catalog version can be found in
|
|
|
|
# src/include/catalog/catversion.h and should change with every release.
|
2015-08-08 23:11:20 +02:00
|
|
|
####################################################################################################################################
|
2016-05-14 16:33:12 +02:00
|
|
|
my $oPgControlVersionHash =
|
|
|
|
{
|
|
|
|
# iControlVersion => {iCatalogVersion => strDbVersion}
|
2017-06-27 21:58:02 +02:00
|
|
|
833 => {200711281 => PG_VERSION_83},
|
2016-05-14 16:33:12 +02:00
|
|
|
843 => {200904091 => PG_VERSION_84},
|
|
|
|
903 =>
|
|
|
|
{
|
|
|
|
201008051 => PG_VERSION_90,
|
|
|
|
201105231 => PG_VERSION_91,
|
|
|
|
},
|
|
|
|
922 => {201204301 => PG_VERSION_92},
|
|
|
|
937 => {201306121 => PG_VERSION_93},
|
|
|
|
942 =>
|
|
|
|
{
|
|
|
|
201409291 => PG_VERSION_94,
|
|
|
|
201510051 => PG_VERSION_95,
|
2016-07-26 22:39:01 +02:00
|
|
|
},
|
|
|
|
960 =>
|
|
|
|
{
|
2016-09-04 00:34:38 +02:00
|
|
|
201608131 => PG_VERSION_96,
|
2016-05-14 16:33:12 +02:00
|
|
|
},
|
2017-09-01 18:29:34 +02:00
|
|
|
1002 =>
|
|
|
|
{
|
|
|
|
201707211 => PG_VERSION_10,
|
|
|
|
},
|
2018-06-05 14:59:17 +02:00
|
|
|
1100 =>
|
|
|
|
{
|
2018-09-21 19:25:27 +02:00
|
|
|
201809051 => PG_VERSION_11,
|
2018-06-05 14:59:17 +02:00
|
|
|
},
|
2019-10-01 19:20:43 +02:00
|
|
|
1201 =>
|
|
|
|
{
|
|
|
|
201909212 => PG_VERSION_12,
|
|
|
|
},
|
2016-05-14 16:33:12 +02:00
|
|
|
};
|
2015-08-08 23:11:20 +02:00
|
|
|
|
2014-03-06 03:53:13 +03:00
|
|
|
####################################################################################################################################
|
|
|
|
# CONSTRUCTOR
|
|
|
|
####################################################################################################################################
|
2014-10-10 23:03:33 +03:00
|
|
|
sub new
|
2014-03-06 03:53:13 +03:00
|
|
|
{
|
2015-04-01 21:58:33 +02:00
|
|
|
my $class = shift; # Class name
|
2014-10-10 23:03:33 +03:00
|
|
|
|
|
|
|
# Create the class hash
|
|
|
|
my $self = {};
|
|
|
|
bless $self, $class;
|
|
|
|
|
2015-08-29 20:20:46 +02:00
|
|
|
# Assign function parameters, defaults, and log debug info
|
2016-08-24 18:39:27 +02:00
|
|
|
(
|
|
|
|
my $strOperation,
|
|
|
|
$self->{iRemoteIdx},
|
|
|
|
) =
|
|
|
|
logDebugParam
|
|
|
|
(
|
|
|
|
__PACKAGE__ . '->new', \@_,
|
|
|
|
{name => 'iRemoteIdx', required => false},
|
|
|
|
);
|
|
|
|
|
|
|
|
if (defined($self->{iRemoteIdx}))
|
|
|
|
{
|
2018-02-04 01:27:38 +02:00
|
|
|
$self->{strDbPath} = cfgOption(cfgOptionIdFromIndex(CFGOPT_PG_PATH, $self->{iRemoteIdx}));
|
2016-08-24 18:39:27 +02:00
|
|
|
}
|
2015-08-29 20:20:46 +02:00
|
|
|
|
|
|
|
# Return from function and log return values if any
|
|
|
|
return logDebugReturn
|
|
|
|
(
|
|
|
|
$strOperation,
|
|
|
|
{name => 'self', value => $self}
|
|
|
|
);
|
2014-03-06 03:53:13 +03:00
|
|
|
}
|
|
|
|
|
2015-08-05 18:32:12 +02:00
|
|
|
####################################################################################################################################
|
|
|
|
# DESTRUCTOR
|
|
|
|
####################################################################################################################################
|
|
|
|
sub DESTROY
|
|
|
|
{
|
|
|
|
my $self = shift;
|
|
|
|
|
2015-08-29 20:20:46 +02:00
|
|
|
# Assign function parameters, defaults, and log debug info
|
2016-08-11 23:32:28 +02:00
|
|
|
my ($strOperation) = logDebugParam(__PACKAGE__ . '->DESTROY');
|
2015-08-29 20:20:46 +02:00
|
|
|
|
2019-07-25 23:05:39 +02:00
|
|
|
if (defined($self->{oDb}))
|
2015-08-05 18:32:12 +02:00
|
|
|
{
|
2019-07-25 23:05:39 +02:00
|
|
|
$self->{oDb}->close();
|
|
|
|
undef($self->{oDb});
|
2015-08-05 18:32:12 +02:00
|
|
|
}
|
2015-08-29 20:20:46 +02:00
|
|
|
|
|
|
|
# Return from function and log return values if any
|
2016-08-11 23:32:28 +02:00
|
|
|
return logDebugReturn($strOperation);
|
2015-08-05 18:32:12 +02:00
|
|
|
}
|
|
|
|
|
2014-03-06 03:53:13 +03:00
|
|
|
####################################################################################################################################
|
2016-08-15 23:44:51 +02:00
|
|
|
# connect
|
2014-03-06 03:53:13 +03:00
|
|
|
####################################################################################################################################
|
2016-08-15 23:44:51 +02:00
|
|
|
sub connect
|
2014-03-06 03:53:13 +03:00
|
|
|
{
|
|
|
|
my $self = shift;
|
2015-06-30 04:07:42 +02:00
|
|
|
|
2015-08-29 20:20:46 +02:00
|
|
|
# Assign function parameters, defaults, and log debug info
|
|
|
|
my
|
|
|
|
(
|
|
|
|
$strOperation,
|
2016-08-15 23:44:51 +02:00
|
|
|
$bWarnOnError,
|
2015-08-29 20:20:46 +02:00
|
|
|
) =
|
|
|
|
logDebugParam
|
|
|
|
(
|
2016-08-15 23:44:51 +02:00
|
|
|
__PACKAGE__ . '::connect', \@_,
|
|
|
|
{name => 'bWarnOnError', default => false},
|
2015-08-29 20:20:46 +02:00
|
|
|
);
|
2014-03-06 03:53:13 +03:00
|
|
|
|
2016-08-15 23:44:51 +02:00
|
|
|
# Only connect if not already connected
|
|
|
|
my $bResult = true;
|
2014-03-06 03:53:13 +03:00
|
|
|
|
2019-12-14 00:55:41 +02:00
|
|
|
if (!defined($self->{oDb}))
|
2014-03-06 03:53:13 +03:00
|
|
|
{
|
2019-12-14 00:55:41 +02:00
|
|
|
$self->{oDb} = new pgBackRest::LibC::PgClient(
|
|
|
|
cfgOption(cfgOptionIdFromIndex(CFGOPT_PG_SOCKET_PATH, $self->{iRemoteIdx}), false),
|
|
|
|
cfgOption(cfgOptionIdFromIndex(CFGOPT_PG_PORT, $self->{iRemoteIdx})), 'postgres',
|
|
|
|
cfgOption(CFGOPT_DB_TIMEOUT) * 1000);
|
2015-08-06 04:05:45 +02:00
|
|
|
|
2019-12-14 00:55:41 +02:00
|
|
|
if ($bWarnOnError)
|
|
|
|
{
|
|
|
|
eval
|
2015-08-06 04:05:45 +02:00
|
|
|
{
|
2019-12-14 00:55:41 +02:00
|
|
|
$self->{oDb}->open();
|
|
|
|
return true;
|
2016-08-15 23:44:51 +02:00
|
|
|
}
|
2019-12-14 00:55:41 +02:00
|
|
|
or do
|
2019-07-25 23:05:39 +02:00
|
|
|
{
|
2019-12-14 00:55:41 +02:00
|
|
|
&log(WARN, exceptionMessage($EVAL_ERROR));
|
|
|
|
$bResult = false;
|
|
|
|
|
|
|
|
undef($self->{oDb});
|
2019-07-25 23:05:39 +02:00
|
|
|
}
|
2019-12-14 00:55:41 +02:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
$self->{oDb}->open();
|
|
|
|
}
|
|
|
|
|
|
|
|
if (defined($self->{oDb}))
|
|
|
|
{
|
|
|
|
my ($fDbVersion) = $self->versionGet();
|
2019-07-25 23:05:39 +02:00
|
|
|
|
2019-12-14 00:55:41 +02:00
|
|
|
if ($fDbVersion >= PG_VERSION_APPLICATION_NAME)
|
2016-10-06 03:15:10 +02:00
|
|
|
{
|
2019-12-14 00:55:41 +02:00
|
|
|
# Set application name for monitoring and debugging
|
2019-12-30 06:22:18 +02:00
|
|
|
$self->{oDb}->query("set application_name = '" . PROJECT_NAME . ' [' . cfgCommandName(cfgCommandGet()) . "]'");
|
2019-12-14 00:55:41 +02:00
|
|
|
|
|
|
|
# Clear search path to prevent possible function overrides
|
|
|
|
$self->{oDb}->query("set search_path = 'pg_catalog'");
|
2016-10-06 03:15:10 +02:00
|
|
|
}
|
2016-08-15 23:44:51 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
# Return from function and log return values if any
|
|
|
|
return logDebugReturn
|
|
|
|
(
|
|
|
|
$strOperation,
|
|
|
|
{name => 'bResult', value => $bResult}
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
####################################################################################################################################
|
|
|
|
# executeSql
|
|
|
|
####################################################################################################################################
|
|
|
|
sub executeSql
|
|
|
|
{
|
|
|
|
my $self = shift;
|
|
|
|
|
|
|
|
# Assign function parameters, defaults, and log debug info
|
|
|
|
my
|
|
|
|
(
|
|
|
|
$strOperation,
|
|
|
|
$strSql,
|
|
|
|
$bIgnoreError,
|
|
|
|
$bResult,
|
|
|
|
) =
|
|
|
|
logDebugParam
|
|
|
|
(
|
|
|
|
__PACKAGE__ . '::executeSql', \@_,
|
|
|
|
{name => 'strSql'},
|
|
|
|
{name => 'bIgnoreError', default => false},
|
|
|
|
{name => 'bResult', default => true},
|
|
|
|
);
|
|
|
|
|
|
|
|
# Get the user-defined command for psql
|
2016-12-04 00:34:51 +02:00
|
|
|
my @stryResult;
|
2016-08-15 23:44:51 +02:00
|
|
|
|
2019-12-14 00:55:41 +02:00
|
|
|
$self->connect();
|
|
|
|
my $strResult = $self->{oDb}->query($strSql);
|
2015-08-05 18:32:12 +02:00
|
|
|
|
2019-12-14 00:55:41 +02:00
|
|
|
if (defined($strResult))
|
|
|
|
{
|
|
|
|
@stryResult = @{JSON::PP->new()->allow_nonref()->decode($strResult)};
|
2014-03-06 03:53:13 +03:00
|
|
|
}
|
|
|
|
|
2015-08-29 20:20:46 +02:00
|
|
|
# Return from function and log return values if any
|
|
|
|
return logDebugReturn
|
|
|
|
(
|
|
|
|
$strOperation,
|
2016-12-04 00:34:51 +02:00
|
|
|
{name => 'stryResult', value => \@stryResult, ref => true}
|
2015-08-29 20:20:46 +02:00
|
|
|
);
|
2014-03-06 03:53:13 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
####################################################################################################################################
|
2015-08-08 23:11:20 +02:00
|
|
|
# executeSqlRow
|
|
|
|
####################################################################################################################################
|
|
|
|
sub executeSqlRow
|
|
|
|
{
|
|
|
|
my $self = shift;
|
|
|
|
|
2015-08-29 20:20:46 +02:00
|
|
|
# Assign function parameters, defaults, and log debug info
|
|
|
|
my
|
|
|
|
(
|
|
|
|
$strOperation,
|
|
|
|
$strSql
|
|
|
|
) =
|
|
|
|
logDebugParam
|
|
|
|
(
|
2016-08-11 23:32:28 +02:00
|
|
|
__PACKAGE__ . '->executeSqlRow', \@_,
|
2016-12-04 00:34:51 +02:00
|
|
|
{name => 'strSql'}
|
2015-08-29 20:20:46 +02:00
|
|
|
);
|
|
|
|
|
|
|
|
# Return from function and log return values if any
|
|
|
|
return logDebugReturn
|
|
|
|
(
|
|
|
|
$strOperation,
|
2016-12-04 00:34:51 +02:00
|
|
|
{name => 'stryResult', value => @{$self->executeSql($strSql)}[0]}
|
2015-08-29 20:20:46 +02:00
|
|
|
);
|
2015-08-08 23:11:20 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
####################################################################################################################################
|
|
|
|
# executeSqlOne
|
2014-03-06 03:53:13 +03:00
|
|
|
####################################################################################################################################
|
2015-08-08 23:11:20 +02:00
|
|
|
sub executeSqlOne
|
2014-03-06 03:53:13 +03:00
|
|
|
{
|
|
|
|
my $self = shift;
|
2015-08-08 23:11:20 +02:00
|
|
|
|
2015-08-29 20:20:46 +02:00
|
|
|
# Assign function parameters, defaults, and log debug info
|
|
|
|
my
|
|
|
|
(
|
|
|
|
$strOperation,
|
|
|
|
$strSql
|
|
|
|
) =
|
|
|
|
logDebugParam
|
|
|
|
(
|
2016-08-11 23:32:28 +02:00
|
|
|
__PACKAGE__ . '->executeSqlOne', \@_,
|
2016-12-04 00:34:51 +02:00
|
|
|
{name => 'strSql'}
|
2015-08-29 20:20:46 +02:00
|
|
|
);
|
|
|
|
|
|
|
|
# Return from function and log return values if any
|
|
|
|
return logDebugReturn
|
|
|
|
(
|
|
|
|
$strOperation,
|
2016-12-04 00:34:51 +02:00
|
|
|
{name => 'strResult', value => @{@{$self->executeSql($strSql)}[0]}[0]}
|
2015-08-29 20:20:46 +02:00
|
|
|
);
|
2015-08-08 23:11:20 +02:00
|
|
|
}
|
|
|
|
|
2015-06-14 00:25:49 +02:00
|
|
|
####################################################################################################################################
|
|
|
|
# info
|
|
|
|
####################################################################################################################################
|
|
|
|
sub info
|
|
|
|
{
|
|
|
|
my $self = shift;
|
|
|
|
|
2015-08-29 20:20:46 +02:00
|
|
|
# Assign function parameters, defaults, and log debug info
|
|
|
|
my
|
|
|
|
(
|
|
|
|
$strOperation,
|
|
|
|
$strDbPath
|
|
|
|
) =
|
|
|
|
logDebugParam
|
|
|
|
(
|
2016-08-11 23:32:28 +02:00
|
|
|
__PACKAGE__ . '->info', \@_,
|
2016-08-24 18:39:27 +02:00
|
|
|
{name => 'strDbPath', default => $self->{strDbPath}}
|
2015-08-29 20:20:46 +02:00
|
|
|
);
|
2015-06-14 00:25:49 +02:00
|
|
|
|
2016-05-14 16:33:12 +02:00
|
|
|
# Get info if it is not cached
|
|
|
|
#-------------------------------------------------------------------------------------------------------------------------------
|
|
|
|
if (!defined($self->{info}{$strDbPath}))
|
2015-08-06 04:05:45 +02:00
|
|
|
{
|
2019-12-14 00:55:41 +02:00
|
|
|
# Open the control file and read system id and versions
|
|
|
|
#-----------------------------------------------------------------------------------------------------------------------
|
|
|
|
my $strControlFile = "${strDbPath}/" . DB_FILE_PGCONTROL;
|
|
|
|
my $hFile;
|
|
|
|
my $tBlock;
|
2015-06-14 00:25:49 +02:00
|
|
|
|
2019-12-14 00:55:41 +02:00
|
|
|
sysopen($hFile, $strControlFile, O_RDONLY)
|
|
|
|
or confess &log(ERROR, "unable to open ${strControlFile}", ERROR_FILE_OPEN);
|
2015-06-14 00:25:49 +02:00
|
|
|
|
2019-12-14 00:55:41 +02:00
|
|
|
# Read system identifier
|
|
|
|
sysread($hFile, $tBlock, 8) == 8
|
|
|
|
or confess &log(ERROR, "unable to read database system identifier");
|
2015-06-14 00:25:49 +02:00
|
|
|
|
2019-12-14 00:55:41 +02:00
|
|
|
$self->{info}{$strDbPath}{ullDbSysId} = unpack('Q', $tBlock);
|
2015-06-14 00:25:49 +02:00
|
|
|
|
2019-12-14 00:55:41 +02:00
|
|
|
# Read control version
|
|
|
|
sysread($hFile, $tBlock, 4) == 4
|
|
|
|
or confess &log(ERROR, "unable to read control version");
|
2015-06-14 00:25:49 +02:00
|
|
|
|
2019-12-14 00:55:41 +02:00
|
|
|
$self->{info}{$strDbPath}{iDbControlVersion} = unpack('L', $tBlock);
|
2015-06-14 00:25:49 +02:00
|
|
|
|
2019-12-14 00:55:41 +02:00
|
|
|
# Read catalog version
|
|
|
|
sysread($hFile, $tBlock, 4) == 4
|
|
|
|
or confess &log(ERROR, "unable to read catalog version");
|
2015-06-14 00:25:49 +02:00
|
|
|
|
2019-12-14 00:55:41 +02:00
|
|
|
$self->{info}{$strDbPath}{iDbCatalogVersion} = unpack('L', $tBlock);
|
2015-06-14 00:25:49 +02:00
|
|
|
|
2019-12-14 00:55:41 +02:00
|
|
|
# Close the control file
|
|
|
|
close($hFile);
|
2015-06-14 00:25:49 +02:00
|
|
|
|
2019-12-14 00:55:41 +02:00
|
|
|
# Get PostgreSQL version
|
|
|
|
$self->{info}{$strDbPath}{strDbVersion} =
|
|
|
|
$oPgControlVersionHash->{$self->{info}{$strDbPath}{iDbControlVersion}}
|
|
|
|
{$self->{info}{$strDbPath}{iDbCatalogVersion}};
|
2015-06-14 00:25:49 +02:00
|
|
|
|
2019-12-14 00:55:41 +02:00
|
|
|
if (!defined($self->{info}{$strDbPath}{strDbVersion}))
|
|
|
|
{
|
|
|
|
confess &log(
|
|
|
|
ERROR,
|
|
|
|
'unexpected control version = ' . $self->{info}{$strDbPath}{iDbControlVersion} .
|
|
|
|
' and catalog version = ' . $self->{info}{$strDbPath}{iDbCatalogVersion} . "\n" .
|
|
|
|
'HINT: is this version of PostgreSQL supported?',
|
|
|
|
ERROR_VERSION_NOT_SUPPORTED);
|
2015-06-14 00:25:49 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-08-29 20:20:46 +02:00
|
|
|
# Return from function and log return values if any
|
|
|
|
return logDebugReturn
|
|
|
|
(
|
|
|
|
$strOperation,
|
2016-05-14 16:33:12 +02:00
|
|
|
{name => 'strDbVersion', value => $self->{info}{$strDbPath}{strDbVersion}},
|
2016-12-04 00:34:51 +02:00
|
|
|
{name => 'iDbControlVersion', value => $self->{info}{$strDbPath}{iDbControlVersion}},
|
|
|
|
{name => 'iDbCatalogVersion', value => $self->{info}{$strDbPath}{iDbCatalogVersion}},
|
2016-05-14 16:33:12 +02:00
|
|
|
{name => 'ullDbSysId', value => $self->{info}{$strDbPath}{ullDbSysId}}
|
2015-08-29 20:20:46 +02:00
|
|
|
);
|
2015-06-14 00:25:49 +02:00
|
|
|
}
|
|
|
|
|
2014-03-06 03:53:13 +03:00
|
|
|
####################################################################################################################################
|
2015-06-30 04:07:42 +02:00
|
|
|
# versionGet
|
2014-03-06 03:53:13 +03:00
|
|
|
####################################################################################################################################
|
2015-06-30 04:07:42 +02:00
|
|
|
sub versionGet
|
2014-03-06 03:53:13 +03:00
|
|
|
{
|
|
|
|
my $self = shift;
|
2014-06-04 05:02:56 +03:00
|
|
|
|
2015-08-29 20:20:46 +02:00
|
|
|
# Assign function parameters, defaults, and log debug info
|
2016-08-11 23:32:28 +02:00
|
|
|
my ($strOperation) = logDebugParam(__PACKAGE__ . '->versionGet');
|
2015-08-29 20:20:46 +02:00
|
|
|
|
2015-08-06 04:05:45 +02:00
|
|
|
# Get data from the cache if possible
|
2016-05-14 16:33:12 +02:00
|
|
|
if (defined($self->{strDbVersion}) && defined($self->{strDbPath}))
|
2014-03-06 03:53:13 +03:00
|
|
|
{
|
2016-05-14 16:33:12 +02:00
|
|
|
return $self->{strDbVersion}, $self->{strDbPath};
|
2014-03-06 03:53:13 +03:00
|
|
|
}
|
|
|
|
|
2018-02-04 01:27:38 +02:00
|
|
|
# Get version and pg-path from
|
2019-02-05 18:55:07 +02:00
|
|
|
my ($strVersionNum, $strDbPath) =
|
2017-09-01 18:29:34 +02:00
|
|
|
$self->executeSqlRow(
|
|
|
|
"select (select setting from pg_settings where name = 'server_version_num'), " .
|
|
|
|
" (select setting from pg_settings where name = 'data_directory')");
|
|
|
|
|
|
|
|
# Get first part of the major version - for 10 and above there will only be one part
|
|
|
|
$self->{strDbVersion} = substr($strVersionNum, 0, length($strVersionNum) - 4);
|
|
|
|
|
|
|
|
# Now retrieve the second part of the major version for versions less than 10
|
|
|
|
if ($self->{strDbVersion} < PG_VERSION_10)
|
|
|
|
{
|
|
|
|
$self->{strDbVersion} .= qw{.} . int(substr($strVersionNum, 1, 2));
|
|
|
|
}
|
2014-03-06 03:53:13 +03:00
|
|
|
|
2017-09-01 18:29:34 +02:00
|
|
|
# Check that the version is supported
|
2015-08-29 20:20:46 +02:00
|
|
|
my @stryVersionSupport = versionSupport();
|
2015-04-03 04:07:23 +02:00
|
|
|
|
2016-05-14 16:33:12 +02:00
|
|
|
if ($self->{strDbVersion} < $stryVersionSupport[0])
|
2015-04-03 04:07:23 +02:00
|
|
|
{
|
2016-05-14 16:33:12 +02:00
|
|
|
confess &log(ERROR, 'unsupported Postgres version' . $self->{strDbVersion}, ERROR_VERSION_NOT_SUPPORTED);
|
2015-04-03 04:07:23 +02:00
|
|
|
}
|
|
|
|
|
2015-08-29 20:20:46 +02:00
|
|
|
# Return from function and log return values if any
|
|
|
|
return logDebugReturn
|
|
|
|
(
|
|
|
|
$strOperation,
|
2016-05-14 16:33:12 +02:00
|
|
|
{name => 'strDbVersion', value => $self->{strDbVersion}},
|
2019-02-05 18:55:07 +02:00
|
|
|
{name => 'strDbPath', value => $strDbPath}
|
2015-08-29 20:20:46 +02:00
|
|
|
);
|
2014-03-06 03:53:13 +03:00
|
|
|
}
|
|
|
|
|
2016-06-12 15:13:46 +02:00
|
|
|
####################################################################################################################################
|
|
|
|
# configValidate
|
|
|
|
#
|
|
|
|
# Validate the database configuration and archiving.
|
|
|
|
####################################################################################################################################
|
|
|
|
sub configValidate
|
|
|
|
{
|
|
|
|
my $self = shift;
|
|
|
|
|
|
|
|
# Assign function parameters, defaults, and log debug info
|
|
|
|
my
|
|
|
|
(
|
|
|
|
$strOperation,
|
|
|
|
) =
|
|
|
|
logDebugParam
|
|
|
|
(
|
|
|
|
__PACKAGE__ . '->configValidate', \@_,
|
|
|
|
);
|
|
|
|
|
|
|
|
# Get the version from the control file
|
2016-11-30 21:15:11 +02:00
|
|
|
my ($strDbVersion) = $self->info();
|
2016-06-12 15:13:46 +02:00
|
|
|
|
|
|
|
# Get version and db path from the database
|
|
|
|
my ($fCompareDbVersion, $strCompareDbPath) = $self->versionGet();
|
|
|
|
|
2018-02-04 01:27:38 +02:00
|
|
|
# Error if the version from the control file and the configured pg-path do not match the values obtained from the database
|
2016-11-30 21:15:11 +02:00
|
|
|
if (!($strDbVersion == $fCompareDbVersion && $self->{strDbPath} eq $strCompareDbPath))
|
2016-06-12 15:13:46 +02:00
|
|
|
{
|
|
|
|
confess &log(ERROR,
|
2018-02-04 01:27:38 +02:00
|
|
|
"version '${fCompareDbVersion}' and path '${strCompareDbPath}' queried from cluster do not match version" .
|
|
|
|
" '${strDbVersion}' and " . cfgOptionName(CFGOPT_PG_PATH) . " '$self->{strDbPath}' read from" .
|
|
|
|
" '$self->{strDbPath}/" . DB_FILE_PGCONTROL . "'\n" .
|
|
|
|
"HINT: the " . cfgOptionName(CFGOPT_PG_PATH) . " and " . cfgOptionName(CFGOPT_PG_PORT) .
|
2019-09-14 18:21:08 +02:00
|
|
|
" settings likely reference different clusters.",
|
2016-11-30 21:15:11 +02:00
|
|
|
ERROR_DB_MISMATCH);
|
2016-06-12 15:13:46 +02:00
|
|
|
}
|
|
|
|
|
2016-10-01 19:44:45 +02:00
|
|
|
# If cluster is not a standby and archive checking is enabled, then perform various validations
|
2017-08-25 22:47:47 +02:00
|
|
|
if (!$self->isStandby() && cfgOptionValid(CFGOPT_ARCHIVE_CHECK) && cfgOption(CFGOPT_ARCHIVE_CHECK))
|
2016-06-12 15:13:46 +02:00
|
|
|
{
|
2016-12-04 00:34:51 +02:00
|
|
|
my $strArchiveMode = $self->executeSqlOne('show archive_mode');
|
|
|
|
|
2016-09-29 01:45:33 +02:00
|
|
|
# Error if archive_mode = off since pg_start_backup () will fail
|
2016-12-04 00:34:51 +02:00
|
|
|
if ($strArchiveMode eq 'off')
|
2016-09-29 01:45:33 +02:00
|
|
|
{
|
|
|
|
confess &log(ERROR, 'archive_mode must be enabled', ERROR_ARCHIVE_DISABLED);
|
|
|
|
}
|
2016-06-12 15:13:46 +02:00
|
|
|
|
2016-09-29 01:45:33 +02:00
|
|
|
# Error if archive_mode = always (support has not been added yet)
|
2016-12-04 00:34:51 +02:00
|
|
|
if ($strArchiveMode eq 'always')
|
2016-09-29 01:45:33 +02:00
|
|
|
{
|
|
|
|
confess &log(ERROR, "archive_mode=always not supported", ERROR_FEATURE_NOT_SUPPORTED);
|
|
|
|
}
|
|
|
|
|
|
|
|
# Check if archive_command is set
|
2016-12-04 00:34:51 +02:00
|
|
|
my $strArchiveCommand = $self->executeSqlOne('show archive_command');
|
2016-07-02 15:02:55 +02:00
|
|
|
|
2018-11-25 02:05:03 +02:00
|
|
|
if (index($strArchiveCommand, PROJECT_EXE) == -1)
|
2016-07-02 15:02:55 +02:00
|
|
|
{
|
|
|
|
confess &log(ERROR,
|
2016-09-29 01:45:33 +02:00
|
|
|
'archive_command ' . (defined($strArchiveCommand) ? "'${strArchiveCommand}'" : '[null]') . ' must contain \'' .
|
2018-11-25 02:05:03 +02:00
|
|
|
PROJECT_EXE . '\'', ERROR_ARCHIVE_COMMAND_INVALID);
|
2016-07-02 15:02:55 +02:00
|
|
|
}
|
2016-06-12 15:13:46 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
return logDebugReturn
|
|
|
|
(
|
|
|
|
$strOperation
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
####################################################################################################################################
|
2017-09-01 18:29:34 +02:00
|
|
|
# walId
|
|
|
|
#
|
|
|
|
# Returns 'wal' or 'xlog' depending on the version of PostgreSQL.
|
|
|
|
####################################################################################################################################
|
|
|
|
sub walId
|
|
|
|
{
|
|
|
|
my $self = shift;
|
|
|
|
|
|
|
|
return $self->{strDbVersion} >= PG_VERSION_10 ? 'wal' : 'xlog';
|
|
|
|
}
|
|
|
|
|
2016-10-01 19:44:45 +02:00
|
|
|
####################################################################################################################################
|
|
|
|
# isStandby
|
|
|
|
#
|
|
|
|
# Determines if a database is a standby by testing if it is in recovery mode.
|
|
|
|
####################################################################################################################################
|
|
|
|
sub isStandby
|
|
|
|
{
|
|
|
|
my $self = shift;
|
|
|
|
|
|
|
|
# Assign function parameters, defaults, and log debug info
|
|
|
|
my ($strOperation) = logDebugParam(__PACKAGE__ . '->isStandby');
|
|
|
|
|
|
|
|
if (!defined($self->{bStandby}))
|
|
|
|
{
|
|
|
|
my ($strDbVersion) = $self->versionGet();
|
|
|
|
|
|
|
|
if ($strDbVersion <= PG_VERSION_90)
|
|
|
|
{
|
|
|
|
$self->{bStandby} = false;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
$self->{bStandby} = $self->executeSqlOne('select pg_is_in_recovery()') ? true : false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
# Return from function and log return values if any
|
|
|
|
return logDebugReturn
|
|
|
|
(
|
|
|
|
$strOperation,
|
|
|
|
{name => 'bStandby', value => $self->{bStandby}}
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2016-11-30 21:15:11 +02:00
|
|
|
####################################################################################################################################
|
|
|
|
# dbObjectGet
|
|
|
|
#
|
|
|
|
# Gets the database objects(s) and indexes. The databases required for the backup type must be online. A connection to the available
|
|
|
|
# databases will be established to determine which is the master and which, if any, is the standby. If there is a master and a
|
|
|
|
# standby to which a connection can be established, it returns both, else just the master.
|
|
|
|
####################################################################################################################################
|
|
|
|
sub dbObjectGet
|
|
|
|
{
|
|
|
|
# Assign function parameters, defaults, and log debug info
|
2017-09-01 01:15:44 +02:00
|
|
|
my (
|
|
|
|
$strOperation,
|
|
|
|
$bMasterOnly,
|
|
|
|
) =
|
|
|
|
logDebugParam
|
|
|
|
(
|
|
|
|
__PACKAGE__ . '::dbObjectGet', \@_,
|
|
|
|
{name => 'bMasterOnly', optional => true, default => false},
|
|
|
|
);
|
2016-11-30 21:15:11 +02:00
|
|
|
|
|
|
|
my $iStandbyIdx = undef;
|
|
|
|
my $iMasterRemoteIdx = 1;
|
|
|
|
my $oDbMaster = undef;
|
|
|
|
my $oDbStandby = undef;
|
|
|
|
|
|
|
|
# Only iterate databases if online and more than one is defined. It might be better to check the version of each database but
|
|
|
|
# this is simple and works.
|
2017-09-01 01:15:44 +02:00
|
|
|
if (!$bMasterOnly && cfgOptionTest(CFGOPT_ONLINE) && cfgOption(CFGOPT_ONLINE) && multipleDb())
|
2016-11-30 21:15:11 +02:00
|
|
|
{
|
2018-02-04 01:27:38 +02:00
|
|
|
for (my $iRemoteIdx = 1; $iRemoteIdx <= cfgOptionIndexTotal(CFGOPT_PG_HOST); $iRemoteIdx++)
|
2016-11-30 21:15:11 +02:00
|
|
|
{
|
|
|
|
# Make sure a db is defined for this index
|
2018-02-04 01:27:38 +02:00
|
|
|
if (cfgOptionTest(cfgOptionIdFromIndex(CFGOPT_PG_PATH, $iRemoteIdx)) ||
|
|
|
|
cfgOptionTest(cfgOptionIdFromIndex(CFGOPT_PG_HOST, $iRemoteIdx)))
|
2016-11-30 21:15:11 +02:00
|
|
|
{
|
|
|
|
# Create the db object
|
2017-09-01 01:15:44 +02:00
|
|
|
my $oDb;
|
|
|
|
|
|
|
|
logWarnOnErrorEnable();
|
|
|
|
eval
|
|
|
|
{
|
|
|
|
$oDb = new pgBackRest::Db($iRemoteIdx);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
or do {};
|
|
|
|
|
|
|
|
logWarnOnErrorDisable();
|
2016-11-30 21:15:11 +02:00
|
|
|
my $bAssigned = false;
|
|
|
|
|
2017-09-01 01:15:44 +02:00
|
|
|
if (defined($oDb))
|
2016-11-30 21:15:11 +02:00
|
|
|
{
|
2017-09-01 01:15:44 +02:00
|
|
|
# If able to connect then test if the database is a master or a standby. It's OK if some databases cannot be
|
|
|
|
# reached as long as the databases required for the backup type are present.
|
|
|
|
if ($oDb->connect(true))
|
2016-11-30 21:15:11 +02:00
|
|
|
{
|
2017-09-01 01:15:44 +02:00
|
|
|
# If this db is a standby
|
|
|
|
if ($oDb->isStandby())
|
2016-11-30 21:15:11 +02:00
|
|
|
{
|
2017-09-01 01:15:44 +02:00
|
|
|
# If standby backup is requested then use the first standby found
|
|
|
|
if (cfgOption(CFGOPT_BACKUP_STANDBY) && !defined($oDbStandby))
|
|
|
|
{
|
|
|
|
$oDbStandby = $oDb;
|
|
|
|
$iStandbyIdx = $iRemoteIdx;
|
|
|
|
$bAssigned = true;
|
|
|
|
}
|
2016-11-30 21:15:11 +02:00
|
|
|
}
|
2017-09-01 01:15:44 +02:00
|
|
|
# Else this db is a master
|
|
|
|
else
|
2016-11-30 21:15:11 +02:00
|
|
|
{
|
2017-09-01 01:15:44 +02:00
|
|
|
# Error if more than one master is found
|
|
|
|
if (defined($oDbMaster))
|
|
|
|
{
|
|
|
|
confess &log(ERROR, 'more than one master database found');
|
|
|
|
}
|
|
|
|
|
|
|
|
$oDbMaster = $oDb;
|
|
|
|
$iMasterRemoteIdx = $iRemoteIdx;
|
|
|
|
$bAssigned = true;
|
2016-11-30 21:15:11 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-09-01 01:15:44 +02:00
|
|
|
# Make sure a standby database is defined when backup from standby option is set
|
2017-08-25 22:47:47 +02:00
|
|
|
if (cfgOption(CFGOPT_BACKUP_STANDBY) && !defined($oDbStandby))
|
2016-11-30 21:15:11 +02:00
|
|
|
{
|
2017-09-01 01:15:44 +02:00
|
|
|
# Throw an error that is distinct from connecting to the master for testing purposes
|
|
|
|
confess &log(ERROR, 'unable to find standby database - cannot proceed', ERROR_HOST_CONNECT);
|
2016-11-30 21:15:11 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
# A master database is always required
|
|
|
|
if (!defined($oDbMaster))
|
|
|
|
{
|
2017-09-01 01:15:44 +02:00
|
|
|
# Throw an error that is distinct from connecting to a standy for testing purposes
|
|
|
|
confess &log(ERROR, 'unable to find master database - cannot proceed', ERROR_DB_CONNECT);
|
2016-11-30 21:15:11 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
# If master db is not already defined then set to default
|
|
|
|
if (!defined($oDbMaster))
|
|
|
|
{
|
|
|
|
$oDbMaster = new pgBackRest::Db($iMasterRemoteIdx);
|
|
|
|
}
|
|
|
|
|
|
|
|
# Return from function and log return values if any
|
|
|
|
return logDebugReturn
|
|
|
|
(
|
|
|
|
$strOperation,
|
|
|
|
{name => 'oDbMaster', value => $oDbMaster},
|
|
|
|
{name => 'iDbMasterIdx', value => $iMasterRemoteIdx},
|
|
|
|
{name => 'oDbStandby', value => $oDbStandby},
|
|
|
|
{name => 'iDbStandbyIdx', value => $iStandbyIdx},
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
push @EXPORT, qw(dbObjectGet);
|
|
|
|
|
|
|
|
####################################################################################################################################
|
|
|
|
# dbMasterGet
|
|
|
|
#
|
|
|
|
# Usually only the master database is required so this function makes getting it simple. If in offline mode (which is true for a
|
|
|
|
# lot of archive operations) then the database returned is simply the first configured.
|
|
|
|
####################################################################################################################################
|
|
|
|
sub dbMasterGet
|
|
|
|
{
|
|
|
|
# Assign function parameters, defaults, and log debug info
|
|
|
|
my ($strOperation) = logDebugParam(__PACKAGE__ . '::dbMasterGet');
|
|
|
|
|
2017-09-01 01:15:44 +02:00
|
|
|
my ($oDbMaster) = dbObjectGet({bMasterOnly => true});
|
2016-11-30 21:15:11 +02:00
|
|
|
|
|
|
|
# Return from function and log return values if any
|
|
|
|
return logDebugReturn
|
|
|
|
(
|
|
|
|
$strOperation,
|
|
|
|
{name => 'oDbMaster', value => $oDbMaster, trace => true},
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
push @EXPORT, qw(dbMasterGet);
|
|
|
|
|
2017-09-01 01:15:44 +02:00
|
|
|
####################################################################################################################################
|
|
|
|
# multipleDb
|
|
|
|
#
|
|
|
|
# Helper function to determine if there is more than one database defined.
|
|
|
|
####################################################################################################################################
|
|
|
|
sub multipleDb
|
|
|
|
{
|
2018-02-04 01:27:38 +02:00
|
|
|
for (my $iDbPathIdx = 2; $iDbPathIdx <= cfgOptionIndexTotal(CFGOPT_PG_PATH); $iDbPathIdx++)
|
2017-09-01 01:15:44 +02:00
|
|
|
{
|
|
|
|
# If an index exists above 1 then return true
|
2018-02-04 01:27:38 +02:00
|
|
|
if (cfgOptionTest(cfgOptionIdFromIndex(CFGOPT_PG_PATH, $iDbPathIdx)))
|
2017-09-01 01:15:44 +02:00
|
|
|
{
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2014-10-10 23:03:33 +03:00
|
|
|
1;
|