Commit 72d422a5 authored by Andrew Dunstan's avatar Andrew Dunstan

Map basebackup tablespaces using a tablespace_map file

Windows can't reliably restore symbolic links from a tar format, so
instead during backup start we create a tablespace_map file, which is
used by the restoring postgres to create the correct links in pg_tblspc.
The backup protocol also now has an option to request this file to be
included in the backup stream, and this is used by pg_basebackup when
operating in tar mode.

This is done on all platforms, not just Windows.

This means that pg_basebackup will not not work in tar mode against 9.4
and older servers, as this protocol option isn't implemented there.

Amit Kapila, reviewed by Dilip Kumar, with a little editing from me.
parent d02f1647
...@@ -836,8 +836,11 @@ SELECT pg_start_backup('label'); ...@@ -836,8 +836,11 @@ SELECT pg_start_backup('label');
<function>pg_start_backup</> creates a <firstterm>backup label</> file, <function>pg_start_backup</> creates a <firstterm>backup label</> file,
called <filename>backup_label</>, in the cluster directory with called <filename>backup_label</>, in the cluster directory with
information about your backup, including the start time and label information about your backup, including the start time and label
string. The file is critical to the integrity of the backup, should string. The function also creates a <firstterm>tablespace map</> file,
you need to restore from it. called <filename>tablespace_map</>, in the cluster directory with
information about tablespace symbolic links in <filename>pg_tblspc/</>
if one or more such link is present. Both files are critical to the
integrity of the backup, should you need to restore from it.
</para> </para>
<para> <para>
...@@ -965,17 +968,20 @@ SELECT pg_stop_backup(); ...@@ -965,17 +968,20 @@ SELECT pg_stop_backup();
<para> <para>
It's also worth noting that the <function>pg_start_backup</> function It's also worth noting that the <function>pg_start_backup</> function
makes a file named <filename>backup_label</> in the database cluster makes files named <filename>backup_label</> and
directory, which is removed by <function>pg_stop_backup</>. <filename>tablesapce_map</> in the database cluster directory,
This file will of course be archived as a part of your backup dump file. which are removed by <function>pg_stop_backup</>. These files will of
The backup label file includes the label string you gave to course be archived as a part of your backup dump file. The backup label
<function>pg_start_backup</>, as well as the time at which file includes the label string you gave to <function>pg_start_backup</>,
<function>pg_start_backup</> was run, and the name of the starting WAL as well as the time at which <function>pg_start_backup</> was run, and
file. In case of confusion it is therefore possible to look inside a the name of the starting WAL file. In case of confusion it is therefore
backup dump file and determine exactly which backup session the dump file possible to look inside a backup dump file and determine exactly which
came from. However, this file is not merely for your information; its backup session the dump file came from. The tablespace map file includes
presence and contents are critical to the proper operation of the system's the symbolic link names as they exist in the directory
recovery process. <filename>pg_tblspc/</> and the full path of each symbolic link.
These files are not merely for your information; their presence and
contents are critical to the proper operation of the system's recovery
process.
</para> </para>
<para> <para>
......
...@@ -16591,11 +16591,12 @@ SELECT set_config('log_statement_stats', 'off', false); ...@@ -16591,11 +16591,12 @@ SELECT set_config('log_statement_stats', 'off', false);
<function>pg_start_backup</> accepts an <function>pg_start_backup</> accepts an
arbitrary user-defined label for the backup. (Typically this would be arbitrary user-defined label for the backup. (Typically this would be
the name under which the backup dump file will be stored.) The function the name under which the backup dump file will be stored.) The function
writes a backup label file (<filename>backup_label</>) into the writes a backup label file (<filename>backup_label</>) and, if there
database cluster's data directory, performs a checkpoint, are any links in the <filename>pg_tblspc/</> directory, a tablespace map
and then returns the backup's starting transaction log location as text. file (<filename>tablespace_map</>) into the database cluster's data
The user can ignore this result value, but it is directory, performs a checkpoint, and then returns the backup's starting
provided in case it is useful. transaction log location as text. The user can ignore this result value,
but it is provided in case it is useful.
<programlisting> <programlisting>
postgres=# select pg_start_backup('label_goes_here'); postgres=# select pg_start_backup('label_goes_here');
pg_start_backup pg_start_backup
...@@ -16610,7 +16611,8 @@ postgres=# select pg_start_backup('label_goes_here'); ...@@ -16610,7 +16611,8 @@ postgres=# select pg_start_backup('label_goes_here');
</para> </para>
<para> <para>
<function>pg_stop_backup</> removes the label file created by <function>pg_stop_backup</> removes the label file and, if it exists,
the <filename>tablespace_map</> file created by
<function>pg_start_backup</>, and creates a backup history file in <function>pg_start_backup</>, and creates a backup history file in
the transaction log archive area. The history file includes the label given to the transaction log archive area. The history file includes the label given to
<function>pg_start_backup</>, the starting and ending transaction log locations for <function>pg_start_backup</>, the starting and ending transaction log locations for
......
...@@ -1882,7 +1882,7 @@ The commands accepted in walsender mode are: ...@@ -1882,7 +1882,7 @@ The commands accepted in walsender mode are:
</varlistentry> </varlistentry>
<varlistentry> <varlistentry>
<term>BASE_BACKUP [<literal>LABEL</literal> <replaceable>'label'</replaceable>] [<literal>PROGRESS</literal>] [<literal>FAST</literal>] [<literal>WAL</literal>] [<literal>NOWAIT</literal>] [<literal>MAX_RATE</literal> <replaceable>rate</replaceable>] <term>BASE_BACKUP [<literal>LABEL</literal> <replaceable>'label'</replaceable>] [<literal>PROGRESS</literal>] [<literal>FAST</literal>] [<literal>WAL</literal>] [<literal>NOWAIT</literal>] [<literal>MAX_RATE</literal> <replaceable>rate</replaceable>] [<literal>TABLESPACE_MAP</literal>]
<indexterm><primary>BASE_BACKUP</primary></indexterm> <indexterm><primary>BASE_BACKUP</primary></indexterm>
</term> </term>
<listitem> <listitem>
...@@ -1968,6 +1968,19 @@ The commands accepted in walsender mode are: ...@@ -1968,6 +1968,19 @@ The commands accepted in walsender mode are:
</para> </para>
</listitem> </listitem>
</varlistentry> </varlistentry>
<varlistentry>
<term><literal>TABLESPACE_MAP</literal></term>
<listitem>
<para>
Include information about symbolic links present in the directory
<filename>pg_tblspc</filename> in a file named
<filename>tablespace_map</filename>. The tablespace map file includes
each symbolic link name as it exists in the directory
<filename>pg_tblspc/</> and the full path of that symbolic link.
</para>
</listitem>
</varlistentry>
</variablelist> </variablelist>
</para> </para>
<para> <para>
......
...@@ -587,11 +587,23 @@ PostgreSQL documentation ...@@ -587,11 +587,23 @@ PostgreSQL documentation
tablespaces. tablespaces.
</para> </para>
<para>
When tar format mode is used, it is the user's responsibility to unpack each
tar file before starting postgres. If there are additional tablespaces, the
tar files for them need to be unpacked in the correct locations. In this
case the symbolic links for those tablespaces will be created by Postgres
according to the contents of the <filename>tablespace_map</> file that is
included in the <filename>base.tar</> file.
</para>
<para> <para>
<application>pg_basebackup</application> works with servers of the same <application>pg_basebackup</application> works with servers of the same
or an older major version, down to 9.1. However, WAL streaming mode (-X or an older major version, down to 9.1. However, WAL streaming mode (-X
stream) only works with server version 9.3 and later. stream) only works with server version 9.3 and later, and tar format mode
(--format=tar) of the current version only works with server version 9.5
or later.
</para> </para>
</refsect1> </refsect1>
<refsect1> <refsect1>
......
...@@ -42,6 +42,7 @@ ...@@ -42,6 +42,7 @@
#include "pgstat.h" #include "pgstat.h"
#include "postmaster/bgwriter.h" #include "postmaster/bgwriter.h"
#include "postmaster/startup.h" #include "postmaster/startup.h"
#include "replication/basebackup.h"
#include "replication/logical.h" #include "replication/logical.h"
#include "replication/slot.h" #include "replication/slot.h"
#include "replication/origin.h" #include "replication/origin.h"
...@@ -824,6 +825,8 @@ static void xlog_outdesc(StringInfo buf, XLogReaderState *record); ...@@ -824,6 +825,8 @@ static void xlog_outdesc(StringInfo buf, XLogReaderState *record);
static void pg_start_backup_callback(int code, Datum arg); static void pg_start_backup_callback(int code, Datum arg);
static bool read_backup_label(XLogRecPtr *checkPointLoc, static bool read_backup_label(XLogRecPtr *checkPointLoc,
bool *backupEndRequired, bool *backupFromStandby); bool *backupEndRequired, bool *backupFromStandby);
static bool read_tablespace_map(List **tablespaces);
static void rm_redo_error_callback(void *arg); static void rm_redo_error_callback(void *arg);
static int get_sync_bit(int method); static int get_sync_bit(int method);
...@@ -5917,6 +5920,7 @@ StartupXLOG(void) ...@@ -5917,6 +5920,7 @@ StartupXLOG(void)
bool wasShutdown; bool wasShutdown;
bool reachedStopPoint = false; bool reachedStopPoint = false;
bool haveBackupLabel = false; bool haveBackupLabel = false;
bool haveTblspcMap = false;
XLogRecPtr RecPtr, XLogRecPtr RecPtr,
checkPointLoc, checkPointLoc,
EndOfLog; EndOfLog;
...@@ -6001,16 +6005,6 @@ StartupXLOG(void) ...@@ -6001,16 +6005,6 @@ StartupXLOG(void)
*/ */
ValidateXLOGDirectoryStructure(); ValidateXLOGDirectoryStructure();
/*
* Clear out any old relcache cache files. This is *necessary* if we do
* any WAL replay, since that would probably result in the cache files
* being out of sync with database reality. In theory we could leave them
* in place if the database had been cleanly shut down, but it seems
* safest to just remove them always and let them be rebuilt during the
* first backend startup.
*/
RelationCacheInitFileRemove();
/* /*
* Initialize on the assumption we want to recover to the latest timeline * Initialize on the assumption we want to recover to the latest timeline
* that's active according to pg_control. * that's active according to pg_control.
...@@ -6080,6 +6074,8 @@ StartupXLOG(void) ...@@ -6080,6 +6074,8 @@ StartupXLOG(void)
if (read_backup_label(&checkPointLoc, &backupEndRequired, if (read_backup_label(&checkPointLoc, &backupEndRequired,
&backupFromStandby)) &backupFromStandby))
{ {
List *tablespaces = NIL;
/* /*
* Archive recovery was requested, and thanks to the backup label * Archive recovery was requested, and thanks to the backup label
* file, we know how far we need to replay to reach consistency. Enter * file, we know how far we need to replay to reach consistency. Enter
...@@ -6124,6 +6120,59 @@ StartupXLOG(void) ...@@ -6124,6 +6120,59 @@ StartupXLOG(void)
errhint("If you are not restoring from a backup, try removing the file \"%s/backup_label\".", DataDir))); errhint("If you are not restoring from a backup, try removing the file \"%s/backup_label\".", DataDir)));
wasShutdown = false; /* keep compiler quiet */ wasShutdown = false; /* keep compiler quiet */
} }
/* read the tablespace_map file if present and create symlinks. */
if (read_tablespace_map(&tablespaces))
{
ListCell *lc;
struct stat st;
foreach(lc, tablespaces)
{
tablespaceinfo *ti = lfirst(lc);
char *linkloc;
linkloc = psprintf("pg_tblspc/%s", ti->oid);
/*
* Remove the existing symlink if any and Create the symlink
* under PGDATA. We need to use rmtree instead of rmdir as
* the link location might contain directories or files
* corresponding to the actual path. Some tar utilities do
* things that way while extracting symlinks.
*/
if (lstat(linkloc, &st) == 0 && S_ISDIR(st.st_mode))
{
if (!rmtree(linkloc,true))
ereport(ERROR,
(errcode_for_file_access(),
errmsg("could not remove directory \"%s\": %m",
linkloc)));
}
else
{
if (unlink(linkloc) < 0 && errno != ENOENT)
ereport(ERROR,
(errcode_for_file_access(),
errmsg("could not remove symbolic link \"%s\": %m",
linkloc)));
}
if (symlink(ti->path, linkloc) < 0)
ereport(ERROR,
(errcode_for_file_access(),
errmsg("could not create symbolic link \"%s\": %m",
linkloc)));
pfree(ti->oid);
pfree(ti->path);
pfree(ti);
}
/* set flag to delete it later */
haveTblspcMap = true;
}
/* set flag to delete it later */ /* set flag to delete it later */
haveBackupLabel = true; haveBackupLabel = true;
} }
...@@ -6197,6 +6246,20 @@ StartupXLOG(void) ...@@ -6197,6 +6246,20 @@ StartupXLOG(void)
wasShutdown = (record->xl_info == XLOG_CHECKPOINT_SHUTDOWN); wasShutdown = (record->xl_info == XLOG_CHECKPOINT_SHUTDOWN);
} }
/*
* Clear out any old relcache cache files. This is *necessary* if we do
* any WAL replay, since that would probably result in the cache files
* being out of sync with database reality. In theory we could leave them
* in place if the database had been cleanly shut down, but it seems
* safest to just remove them always and let them be rebuilt during the
* first backend startup. These files needs to be removed from all
* directories including pg_tblspc, however the symlinks are created
* only after reading tablesapce_map file in case of archive recovery
* from backup, so needs to clear old relcache files here after creating
* symlinks.
*/
RelationCacheInitFileRemove();
/* /*
* If the location of the checkpoint record is not on the expected * If the location of the checkpoint record is not on the expected
* timeline in the history of the requested timeline, we cannot proceed: * timeline in the history of the requested timeline, we cannot proceed:
...@@ -6466,6 +6529,23 @@ StartupXLOG(void) ...@@ -6466,6 +6529,23 @@ StartupXLOG(void)
BACKUP_LABEL_FILE, BACKUP_LABEL_OLD))); BACKUP_LABEL_FILE, BACKUP_LABEL_OLD)));
} }
/*
* If there was a tablespace_map file, it's done its job and the
* symlinks have been created. We must get rid of the map file
* so that if we crash during recovery, we don't create symlinks
* again. It seems prudent though to just rename the file out of
* the way rather than delete it completely.
*/
if (haveTblspcMap)
{
unlink(TABLESPACE_MAP_OLD);
if (rename(TABLESPACE_MAP, TABLESPACE_MAP_OLD) != 0)
ereport(FATAL,
(errcode_for_file_access(),
errmsg("could not rename file \"%s\" to \"%s\": %m",
TABLESPACE_MAP, TABLESPACE_MAP_OLD)));
}
/* Check that the GUCs used to generate the WAL allow recovery */ /* Check that the GUCs used to generate the WAL allow recovery */
CheckRequiredParameterValues(); CheckRequiredParameterValues();
...@@ -9610,16 +9690,27 @@ XLogFileNameP(TimeLineID tli, XLogSegNo segno) ...@@ -9610,16 +9690,27 @@ XLogFileNameP(TimeLineID tli, XLogSegNo segno)
* *
* There are two kind of backups: exclusive and non-exclusive. An exclusive * There are two kind of backups: exclusive and non-exclusive. An exclusive
* backup is started with pg_start_backup(), and there can be only one active * backup is started with pg_start_backup(), and there can be only one active
* at a time. The backup label file of an exclusive backup is written to * at a time. The backup and tablespace map files of an exclusive backup are
* $PGDATA/backup_label, and it is removed by pg_stop_backup(). * written to $PGDATA/backup_label and $PGDATA/tablespace_map, and they are
* removed by pg_stop_backup().
* *
* A non-exclusive backup is used for the streaming base backups (see * A non-exclusive backup is used for the streaming base backups (see
* src/backend/replication/basebackup.c). The difference to exclusive backups * src/backend/replication/basebackup.c). The difference to exclusive backups
* is that the backup label file is not written to disk. Instead, its would-be * is that the backup label and tablespace map files are not written to disk.
* contents are returned in *labelfile, and the caller is responsible for * Instead, there would-be contents are returned in *labelfile and *tblspcmapfile,
* including it in the backup archive as 'backup_label'. There can be many * and the caller is responsible for including them in the backup archive as
* non-exclusive backups active at the same time, and they don't conflict * 'backup_label' and 'tablespace_map'. There can be many non-exclusive backups
* with an exclusive backup either. * active at the same time, and they don't conflict with an exclusive backup
* either.
*
* tblspcmapfile is required mainly for tar format in windows as native windows
* utilities are not able to create symlinks while extracting files from tar.
* However for consistency, the same is used for all platforms.
*
* needtblspcmapfile is true for the cases (exclusive backup and for
* non-exclusive backup only when tar format is used for taking backup)
* when backup needs to generate tablespace_map file, it is used to
* embed escape character before newline character in tablespace path.
* *
* Returns the minimum WAL position that must be present to restore from this * Returns the minimum WAL position that must be present to restore from this
* backup, and the corresponding timeline ID in *starttli_p. * backup, and the corresponding timeline ID in *starttli_p.
...@@ -9632,7 +9723,9 @@ XLogFileNameP(TimeLineID tli, XLogSegNo segno) ...@@ -9632,7 +9723,9 @@ XLogFileNameP(TimeLineID tli, XLogSegNo segno)
*/ */
XLogRecPtr XLogRecPtr
do_pg_start_backup(const char *backupidstr, bool fast, TimeLineID *starttli_p, do_pg_start_backup(const char *backupidstr, bool fast, TimeLineID *starttli_p,
char **labelfile) char **labelfile, DIR *tblspcdir, List **tablespaces,
char **tblspcmapfile, bool infotbssize,
bool needtblspcmapfile)
{ {
bool exclusive = (labelfile == NULL); bool exclusive = (labelfile == NULL);
bool backup_started_in_recovery = false; bool backup_started_in_recovery = false;
...@@ -9646,6 +9739,7 @@ do_pg_start_backup(const char *backupidstr, bool fast, TimeLineID *starttli_p, ...@@ -9646,6 +9739,7 @@ do_pg_start_backup(const char *backupidstr, bool fast, TimeLineID *starttli_p,
struct stat stat_buf; struct stat stat_buf;
FILE *fp; FILE *fp;
StringInfoData labelfbuf; StringInfoData labelfbuf;
StringInfoData tblspc_mapfbuf;
backup_started_in_recovery = RecoveryInProgress(); backup_started_in_recovery = RecoveryInProgress();
...@@ -9717,6 +9811,9 @@ do_pg_start_backup(const char *backupidstr, bool fast, TimeLineID *starttli_p, ...@@ -9717,6 +9811,9 @@ do_pg_start_backup(const char *backupidstr, bool fast, TimeLineID *starttli_p,
PG_ENSURE_ERROR_CLEANUP(pg_start_backup_callback, (Datum) BoolGetDatum(exclusive)); PG_ENSURE_ERROR_CLEANUP(pg_start_backup_callback, (Datum) BoolGetDatum(exclusive));
{ {
bool gotUniqueStartpoint = false; bool gotUniqueStartpoint = false;
struct dirent *de;
tablespaceinfo *ti;
int datadirpathlen;
/* /*
* Force an XLOG file switch before the checkpoint, to ensure that the * Force an XLOG file switch before the checkpoint, to ensure that the
...@@ -9836,6 +9933,98 @@ do_pg_start_backup(const char *backupidstr, bool fast, TimeLineID *starttli_p, ...@@ -9836,6 +9933,98 @@ do_pg_start_backup(const char *backupidstr, bool fast, TimeLineID *starttli_p,
XLByteToSeg(startpoint, _logSegNo); XLByteToSeg(startpoint, _logSegNo);
XLogFileName(xlogfilename, ThisTimeLineID, _logSegNo); XLogFileName(xlogfilename, ThisTimeLineID, _logSegNo);
/*
* Construct tablespace_map file
*/
initStringInfo(&tblspc_mapfbuf);
datadirpathlen = strlen(DataDir);
/* Collect information about all tablespaces */
while ((de = ReadDir(tblspcdir, "pg_tblspc")) != NULL)
{
char fullpath[MAXPGPATH];
char linkpath[MAXPGPATH];
char *relpath = NULL;
int rllen;
StringInfoData buflinkpath;
char *s = linkpath;
/* Skip special stuff */
if (strcmp(de->d_name, ".") == 0 || strcmp(de->d_name, "..") == 0)
continue;
snprintf(fullpath, sizeof(fullpath), "pg_tblspc/%s", de->d_name);
#if defined(HAVE_READLINK) || defined(WIN32)
rllen = readlink(fullpath, linkpath, sizeof(linkpath));
if (rllen < 0)
{
ereport(WARNING,
(errmsg("could not read symbolic link \"%s\": %m",
fullpath)));
continue;
}
else if (rllen >= sizeof(linkpath))
{
ereport(WARNING,
(errmsg("symbolic link \"%s\" target is too long",
fullpath)));
continue;
}
linkpath[rllen] = '\0';
/*
* Add the escape character '\\' before newline in a string
* to ensure that we can distinguish between the newline in
* the tablespace path and end of line while reading
* tablespace_map file during archive recovery.
*/
initStringInfo(&buflinkpath);
while (*s)
{
if ((*s == '\n' || *s == '\r') && needtblspcmapfile)
appendStringInfoChar(&buflinkpath, '\\');
appendStringInfoChar(&buflinkpath, *s++);
}
/*
* Relpath holds the relative path of the tablespace directory
* when it's located within PGDATA, or NULL if it's located
* elsewhere.
*/
if (rllen > datadirpathlen &&
strncmp(linkpath, DataDir, datadirpathlen) == 0 &&
IS_DIR_SEP(linkpath[datadirpathlen]))
relpath = linkpath + datadirpathlen + 1;
ti = palloc(sizeof(tablespaceinfo));
ti->oid = pstrdup(de->d_name);
ti->path = pstrdup(buflinkpath.data);
ti->rpath = relpath ? pstrdup(relpath) : NULL;
ti->size = infotbssize ? sendTablespace(fullpath, true) : -1;
if(tablespaces)
*tablespaces = lappend(*tablespaces, ti);
appendStringInfo(&tblspc_mapfbuf, "%s %s\n", ti->oid, ti->path);
pfree(buflinkpath.data);
#else
/*
* If the platform does not have symbolic links, it should not be
* possible to have tablespaces - clearly somebody else created
* them. Warn about it and ignore.
*/
ereport(WARNING,
(errcode(ERRCODE_FEATURE_NOT_SUPPORTED),
errmsg("tablespaces are not supported on this platform")));
#endif
}
/* /*
* Construct backup label file * Construct backup label file
*/ */
...@@ -9899,9 +10088,51 @@ do_pg_start_backup(const char *backupidstr, bool fast, TimeLineID *starttli_p, ...@@ -9899,9 +10088,51 @@ do_pg_start_backup(const char *backupidstr, bool fast, TimeLineID *starttli_p,
errmsg("could not write file \"%s\": %m", errmsg("could not write file \"%s\": %m",
BACKUP_LABEL_FILE))); BACKUP_LABEL_FILE)));
pfree(labelfbuf.data); pfree(labelfbuf.data);
/* Write backup tablespace_map file. */
if (tblspc_mapfbuf.len > 0)
{
if (stat(TABLESPACE_MAP, &stat_buf) != 0)
{
if (errno != ENOENT)
ereport(ERROR,
(errcode_for_file_access(),
errmsg("could not stat file \"%s\": %m",
TABLESPACE_MAP)));
}
else
ereport(ERROR,
(errcode(ERRCODE_OBJECT_NOT_IN_PREREQUISITE_STATE),
errmsg("a backup is already in progress"),
errhint("If you're sure there is no backup in progress, remove file \"%s\" and try again.",
TABLESPACE_MAP)));
fp = AllocateFile(TABLESPACE_MAP, "w");
if (!fp)
ereport(ERROR,
(errcode_for_file_access(),
errmsg("could not create file \"%s\": %m",
TABLESPACE_MAP)));
if (fwrite(tblspc_mapfbuf.data, tblspc_mapfbuf.len, 1, fp) != 1 ||
fflush(fp) != 0 ||
pg_fsync(fileno(fp)) != 0 ||
ferror(fp) ||
FreeFile(fp))
ereport(ERROR,
(errcode_for_file_access(),
errmsg("could not write file \"%s\": %m",
TABLESPACE_MAP)));
}
pfree(tblspc_mapfbuf.data);
} }
else else
{
*labelfile = labelfbuf.data; *labelfile = labelfbuf.data;
if (tblspc_mapfbuf.len > 0)
*tblspcmapfile = tblspc_mapfbuf.data;
}
} }
PG_END_ENSURE_ERROR_CLEANUP(pg_start_backup_callback, (Datum) BoolGetDatum(exclusive)); PG_END_ENSURE_ERROR_CLEANUP(pg_start_backup_callback, (Datum) BoolGetDatum(exclusive));
...@@ -10072,6 +10303,12 @@ do_pg_stop_backup(char *labelfile, bool waitforarchive, TimeLineID *stoptli_p) ...@@ -10072,6 +10303,12 @@ do_pg_stop_backup(char *labelfile, bool waitforarchive, TimeLineID *stoptli_p)
(errcode_for_file_access(), (errcode_for_file_access(),
errmsg("could not remove file \"%s\": %m", errmsg("could not remove file \"%s\": %m",
BACKUP_LABEL_FILE))); BACKUP_LABEL_FILE)));
/*
* Remove tablespace_map file if present, it is created
* only if there are tablespaces.
*/
unlink(TABLESPACE_MAP);
} }
/* /*
...@@ -10471,6 +10708,86 @@ read_backup_label(XLogRecPtr *checkPointLoc, bool *backupEndRequired, ...@@ -10471,6 +10708,86 @@ read_backup_label(XLogRecPtr *checkPointLoc, bool *backupEndRequired,
return true; return true;
} }
/*
* read_tablespace_map: check to see if a tablespace_map file is present
*
* If we see a tablespace_map file during recovery, we assume that we are
* recovering from a backup dump file, and we therefore need to create symlinks
* as per the information present in tablespace_map file.
*
* Returns TRUE if a tablespace_map file was found (and fills the link
* information for all the tablespace links present in file); returns FALSE
* if not.
*/
static bool
read_tablespace_map(List **tablespaces)
{
tablespaceinfo *ti;
FILE *lfp;
char tbsoid[MAXPGPATH];
char *tbslinkpath;
char str[MAXPGPATH];
int ch, prev_ch = -1,
i = 0, n;
/*
* See if tablespace_map file is present
*/
lfp = AllocateFile(TABLESPACE_MAP, "r");
if (!lfp)
{
if (errno != ENOENT)
ereport(FATAL,
(errcode_for_file_access(),
errmsg("could not read file \"%s\": %m",
TABLESPACE_MAP)));
return false; /* it's not there, all is fine */
}
/*
* Read and parse the link name and path lines from tablespace_map file
* (this code is pretty crude, but we are not expecting any variability
* in the file format). While taking backup we embed escape character
* '\\' before newline in tablespace path, so that during reading of
* tablespace_map file, we could distinguish newline in tablespace path
* and end of line. Now while reading tablespace_map file, remove the
* escape character that has been added in tablespace path during backup.
*/
while ((ch = fgetc(lfp)) != EOF)
{
if ((ch == '\n' || ch == '\r') && prev_ch != '\\')
{
str[i] = '\0';
if (sscanf(str, "%s %n", tbsoid, &n) != 1)
ereport(FATAL,
(errcode(ERRCODE_OBJECT_NOT_IN_PREREQUISITE_STATE),
errmsg("invalid data in file \"%s\"", TABLESPACE_MAP)));
tbslinkpath = str + n;
i = 0;
ti = palloc(sizeof(tablespaceinfo));
ti->oid = pstrdup(tbsoid);
ti->path = pstrdup(tbslinkpath);
*tablespaces = lappend(*tablespaces, ti);
continue;
}
else if ((ch == '\n' || ch == '\r') && prev_ch == '\\')
str[i-1] = ch;
else
str[i++] = ch;
prev_ch = ch;
}
if (ferror(lfp) || FreeFile(lfp))
ereport(FATAL,
(errcode_for_file_access(),
errmsg("could not read file \"%s\": %m",
TABLESPACE_MAP)));
return true;
}
/* /*
* Error context callback for errors occurring during rm_redo(). * Error context callback for errors occurring during rm_redo().
*/ */
...@@ -10502,11 +10819,16 @@ BackupInProgress(void) ...@@ -10502,11 +10819,16 @@ BackupInProgress(void)
} }
/* /*
* CancelBackup: rename the "backup_label" file to cancel backup mode * CancelBackup: rename the "backup_label" and "tablespace_map"
* files to cancel backup mode
* *
* If the "backup_label" file exists, it will be renamed to "backup_label.old". * If the "backup_label" file exists, it will be renamed to "backup_label.old".
* Note that this will render an online backup in progress useless. * Similarly, if the "tablespace_map" file exists, it will be renamed to
* To correctly finish an online backup, pg_stop_backup must be called. * "tablespace_map.old".
*
* Note that this will render an online backup in progress
* useless. To correctly finish an online backup, pg_stop_backup must be
* called.
*/ */
void void
CancelBackup(void) CancelBackup(void)
...@@ -10535,6 +10857,29 @@ CancelBackup(void) ...@@ -10535,6 +10857,29 @@ CancelBackup(void)
errdetail("Could not rename \"%s\" to \"%s\": %m.", errdetail("Could not rename \"%s\" to \"%s\": %m.",
BACKUP_LABEL_FILE, BACKUP_LABEL_OLD))); BACKUP_LABEL_FILE, BACKUP_LABEL_OLD)));
} }
/* if the tablespace_map file is not there, return */
if (stat(TABLESPACE_MAP, &stat_buf) < 0)
return;
/* remove leftover file from previously canceled backup if it exists */
unlink(TABLESPACE_MAP_OLD);
if (rename(TABLESPACE_MAP, TABLESPACE_MAP_OLD) == 0)
{
ereport(LOG,
(errmsg("online backup mode canceled"),
errdetail("\"%s\" was renamed to \"%s\".",
TABLESPACE_MAP, TABLESPACE_MAP_OLD)));
}
else
{
ereport(WARNING,
(errcode_for_file_access(),
errmsg("online backup mode was not canceled"),
errdetail("Could not rename \"%s\" to \"%s\": %m.",
TABLESPACE_MAP, TABLESPACE_MAP_OLD)));
}
} }
/* /*
......
...@@ -51,6 +51,7 @@ pg_start_backup(PG_FUNCTION_ARGS) ...@@ -51,6 +51,7 @@ pg_start_backup(PG_FUNCTION_ARGS)
bool fast = PG_GETARG_BOOL(1); bool fast = PG_GETARG_BOOL(1);
char *backupidstr; char *backupidstr;
XLogRecPtr startpoint; XLogRecPtr startpoint;
DIR *dir;
backupidstr = text_to_cstring(backupid); backupidstr = text_to_cstring(backupid);
...@@ -59,7 +60,16 @@ pg_start_backup(PG_FUNCTION_ARGS) ...@@ -59,7 +60,16 @@ pg_start_backup(PG_FUNCTION_ARGS)
(errcode(ERRCODE_INSUFFICIENT_PRIVILEGE), (errcode(ERRCODE_INSUFFICIENT_PRIVILEGE),
errmsg("must be superuser or replication role to run a backup"))); errmsg("must be superuser or replication role to run a backup")));
startpoint = do_pg_start_backup(backupidstr, fast, NULL, NULL); /* Make sure we can open the directory with tablespaces in it */
dir = AllocateDir("pg_tblspc");
if (!dir)
ereport(ERROR,
(errmsg("could not open directory \"%s\": %m", "pg_tblspc")));
startpoint = do_pg_start_backup(backupidstr, fast, NULL, NULL,
dir, NULL, NULL, false, true);
FreeDir(dir);
PG_RETURN_LSN(startpoint); PG_RETURN_LSN(startpoint);
} }
......
...@@ -46,11 +46,12 @@ typedef struct ...@@ -46,11 +46,12 @@ typedef struct
bool nowait; bool nowait;
bool includewal; bool includewal;
uint32 maxrate; uint32 maxrate;
bool sendtblspcmapfile;
} basebackup_options; } basebackup_options;
static int64 sendDir(char *path, int basepathlen, bool sizeonly, List *tablespaces); static int64 sendDir(char *path, int basepathlen, bool sizeonly,
static int64 sendTablespace(char *path, bool sizeonly); List *tablespaces, bool sendtblspclinks);
static bool sendFile(char *readfilename, char *tarfilename, static bool sendFile(char *readfilename, char *tarfilename,
struct stat * statbuf, bool missing_ok); struct stat * statbuf, bool missing_ok);
static void sendFileWithContent(const char *filename, const char *content); static void sendFileWithContent(const char *filename, const char *content);
...@@ -93,15 +94,6 @@ static int64 elapsed_min_unit; ...@@ -93,15 +94,6 @@ static int64 elapsed_min_unit;
/* The last check of the transfer rate. */ /* The last check of the transfer rate. */
static int64 throttled_last; static int64 throttled_last;
typedef struct
{
char *oid;
char *path;
char *rpath; /* relative path within PGDATA, or NULL */
int64 size;
} tablespaceinfo;
/* /*
* Called when ERROR or FATAL happens in perform_base_backup() after * Called when ERROR or FATAL happens in perform_base_backup() after
* we have started the backup - make sure we end it! * we have started the backup - make sure we end it!
...@@ -126,14 +118,18 @@ perform_base_backup(basebackup_options *opt, DIR *tblspcdir) ...@@ -126,14 +118,18 @@ perform_base_backup(basebackup_options *opt, DIR *tblspcdir)
XLogRecPtr endptr; XLogRecPtr endptr;
TimeLineID endtli; TimeLineID endtli;
char *labelfile; char *labelfile;
char *tblspc_map_file = NULL;
int datadirpathlen; int datadirpathlen;
List *tablespaces = NIL;
datadirpathlen = strlen(DataDir); datadirpathlen = strlen(DataDir);
backup_started_in_recovery = RecoveryInProgress(); backup_started_in_recovery = RecoveryInProgress();
startptr = do_pg_start_backup(opt->label, opt->fastcheckpoint, &starttli, startptr = do_pg_start_backup(opt->label, opt->fastcheckpoint, &starttli,
&labelfile); &labelfile, tblspcdir, &tablespaces,
&tblspc_map_file,
opt->progress, opt->sendtblspcmapfile);
/* /*
* Once do_pg_start_backup has been called, ensure that any failure causes * Once do_pg_start_backup has been called, ensure that any failure causes
* us to abort the backup so we don't "leak" a backup counter. For this reason, * us to abort the backup so we don't "leak" a backup counter. For this reason,
...@@ -143,9 +139,7 @@ perform_base_backup(basebackup_options *opt, DIR *tblspcdir) ...@@ -143,9 +139,7 @@ perform_base_backup(basebackup_options *opt, DIR *tblspcdir)
PG_ENSURE_ERROR_CLEANUP(base_backup_cleanup, (Datum) 0); PG_ENSURE_ERROR_CLEANUP(base_backup_cleanup, (Datum) 0);
{ {
List *tablespaces = NIL;
ListCell *lc; ListCell *lc;
struct dirent *de;
tablespaceinfo *ti; tablespaceinfo *ti;
SendXlogRecPtrResult(startptr, starttli); SendXlogRecPtrResult(startptr, starttli);
...@@ -162,70 +156,9 @@ perform_base_backup(basebackup_options *opt, DIR *tblspcdir) ...@@ -162,70 +156,9 @@ perform_base_backup(basebackup_options *opt, DIR *tblspcdir)
else else
statrelpath = pgstat_stat_directory; statrelpath = pgstat_stat_directory;
/* Collect information about all tablespaces */
while ((de = ReadDir(tblspcdir, "pg_tblspc")) != NULL)
{
char fullpath[MAXPGPATH];
char linkpath[MAXPGPATH];
char *relpath = NULL;
int rllen;
/* Skip special stuff */
if (strcmp(de->d_name, ".") == 0 || strcmp(de->d_name, "..") == 0)
continue;
snprintf(fullpath, sizeof(fullpath), "pg_tblspc/%s", de->d_name);
#if defined(HAVE_READLINK) || defined(WIN32)
rllen = readlink(fullpath, linkpath, sizeof(linkpath));
if (rllen < 0)
{
ereport(WARNING,
(errmsg("could not read symbolic link \"%s\": %m",
fullpath)));
continue;
}
else if (rllen >= sizeof(linkpath))
{
ereport(WARNING,
(errmsg("symbolic link \"%s\" target is too long",
fullpath)));
continue;
}
linkpath[rllen] = '\0';
/*
* Relpath holds the relative path of the tablespace directory
* when it's located within PGDATA, or NULL if it's located
* elsewhere.
*/
if (rllen > datadirpathlen &&
strncmp(linkpath, DataDir, datadirpathlen) == 0 &&
IS_DIR_SEP(linkpath[datadirpathlen]))
relpath = linkpath + datadirpathlen + 1;
ti = palloc(sizeof(tablespaceinfo));
ti->oid = pstrdup(de->d_name);
ti->path = pstrdup(linkpath);
ti->rpath = relpath ? pstrdup(relpath) : NULL;
ti->size = opt->progress ? sendTablespace(fullpath, true) : -1;
tablespaces = lappend(tablespaces, ti);
#else
/*
* If the platform does not have symbolic links, it should not be
* possible to have tablespaces - clearly somebody else created
* them. Warn about it and ignore.
*/
ereport(WARNING,
(errcode(ERRCODE_FEATURE_NOT_SUPPORTED),
errmsg("tablespaces are not supported on this platform")));
#endif
}
/* Add a node for the base directory at the end */ /* Add a node for the base directory at the end */
ti = palloc0(sizeof(tablespaceinfo)); ti = palloc0(sizeof(tablespaceinfo));
ti->size = opt->progress ? sendDir(".", 1, true, tablespaces) : -1; ti->size = opt->progress ? sendDir(".", 1, true, tablespaces, true) : -1;
tablespaces = lappend(tablespaces, ti); tablespaces = lappend(tablespaces, ti);
/* Send tablespace header */ /* Send tablespace header */
...@@ -274,8 +207,17 @@ perform_base_backup(basebackup_options *opt, DIR *tblspcdir) ...@@ -274,8 +207,17 @@ perform_base_backup(basebackup_options *opt, DIR *tblspcdir)
/* In the main tar, include the backup_label first... */ /* In the main tar, include the backup_label first... */
sendFileWithContent(BACKUP_LABEL_FILE, labelfile); sendFileWithContent(BACKUP_LABEL_FILE, labelfile);
/* ... then the bulk of the files ... */ /*
sendDir(".", 1, false, tablespaces); * Send tablespace_map file if required and then the bulk of
* the files.
*/
if (tblspc_map_file && opt->sendtblspcmapfile)
{
sendFileWithContent(TABLESPACE_MAP, tblspc_map_file);
sendDir(".", 1, false, tablespaces, false);
}
else
sendDir(".", 1, false, tablespaces, true);
/* ... and pg_control after everything else. */ /* ... and pg_control after everything else. */
if (lstat(XLOG_CONTROL_FILE, &statbuf) != 0) if (lstat(XLOG_CONTROL_FILE, &statbuf) != 0)
...@@ -567,6 +509,7 @@ parse_basebackup_options(List *options, basebackup_options *opt) ...@@ -567,6 +509,7 @@ parse_basebackup_options(List *options, basebackup_options *opt)
bool o_nowait = false; bool o_nowait = false;
bool o_wal = false; bool o_wal = false;
bool o_maxrate = false; bool o_maxrate = false;
bool o_tablespace_map = false;
MemSet(opt, 0, sizeof(*opt)); MemSet(opt, 0, sizeof(*opt));
foreach(lopt, options) foreach(lopt, options)
...@@ -637,6 +580,15 @@ parse_basebackup_options(List *options, basebackup_options *opt) ...@@ -637,6 +580,15 @@ parse_basebackup_options(List *options, basebackup_options *opt)
opt->maxrate = (uint32) maxrate; opt->maxrate = (uint32) maxrate;
o_maxrate = true; o_maxrate = true;
} }
else if (strcmp(defel->defname, "tablespace_map") == 0)
{
if (o_tablespace_map)
ereport(ERROR,
(errcode(ERRCODE_SYNTAX_ERROR),
errmsg("duplicate option \"%s\"", defel->defname)));
opt->sendtblspcmapfile = true;
o_tablespace_map = true;
}
else else
elog(ERROR, "option \"%s\" not recognized", elog(ERROR, "option \"%s\" not recognized",
defel->defname); defel->defname);
...@@ -865,7 +817,7 @@ sendFileWithContent(const char *filename, const char *content) ...@@ -865,7 +817,7 @@ sendFileWithContent(const char *filename, const char *content)
* *
* Only used to send auxiliary tablespaces, not PGDATA. * Only used to send auxiliary tablespaces, not PGDATA.
*/ */
static int64 int64
sendTablespace(char *path, bool sizeonly) sendTablespace(char *path, bool sizeonly)
{ {
int64 size; int64 size;
...@@ -899,7 +851,7 @@ sendTablespace(char *path, bool sizeonly) ...@@ -899,7 +851,7 @@ sendTablespace(char *path, bool sizeonly)
size = 512; /* Size of the header just added */ size = 512; /* Size of the header just added */
/* Send all the files in the tablespace version directory */ /* Send all the files in the tablespace version directory */
size += sendDir(pathbuf, strlen(path), sizeonly, NIL); size += sendDir(pathbuf, strlen(path), sizeonly, NIL, true);
return size; return size;
} }
...@@ -911,9 +863,14 @@ sendTablespace(char *path, bool sizeonly) ...@@ -911,9 +863,14 @@ sendTablespace(char *path, bool sizeonly)
* *
* Omit any directory in the tablespaces list, to avoid backing up * Omit any directory in the tablespaces list, to avoid backing up
* tablespaces twice when they were created inside PGDATA. * tablespaces twice when they were created inside PGDATA.
*
* If sendtblspclinks is true, we need to include symlink
* information in the tar file. If not, we can skip that
* as it will be sent separately in the tablespace_map file.
*/ */
static int64 static int64
sendDir(char *path, int basepathlen, bool sizeonly, List *tablespaces) sendDir(char *path, int basepathlen, bool sizeonly, List *tablespaces,
bool sendtblspclinks)
{ {
DIR *dir; DIR *dir;
struct dirent *de; struct dirent *de;
...@@ -941,13 +898,17 @@ sendDir(char *path, int basepathlen, bool sizeonly, List *tablespaces) ...@@ -941,13 +898,17 @@ sendDir(char *path, int basepathlen, bool sizeonly, List *tablespaces)
continue; continue;
/* /*
* If there's a backup_label file, it belongs to a backup started by * If there's a backup_label or tablespace_map file, it belongs to a
* the user with pg_start_backup(). It is *not* correct for this * backup started by the user with pg_start_backup(). It is *not*
* backup, our backup_label is injected into the tar separately. * correct for this backup, our backup_label/tablespace_map is injected
* into the tar separately.
*/ */
if (strcmp(de->d_name, BACKUP_LABEL_FILE) == 0) if (strcmp(de->d_name, BACKUP_LABEL_FILE) == 0)
continue; continue;
if (strcmp(de->d_name, TABLESPACE_MAP) == 0)
continue;
/* /*
* Check if the postmaster has signaled us to exit, and abort with an * Check if the postmaster has signaled us to exit, and abort with an
* error in that case. The error handler further up will call * error in that case. The error handler further up will call
...@@ -1120,8 +1081,15 @@ sendDir(char *path, int basepathlen, bool sizeonly, List *tablespaces) ...@@ -1120,8 +1081,15 @@ sendDir(char *path, int basepathlen, bool sizeonly, List *tablespaces)
break; break;
} }
} }
/*
* skip sending directories inside pg_tblspc, if not required.
*/
if (strcmp(pathbuf, "./pg_tblspc") == 0 && !sendtblspclinks)
skip_this_dir = true;
if (!skip_this_dir) if (!skip_this_dir)
size += sendDir(pathbuf, basepathlen, sizeonly, tablespaces); size += sendDir(pathbuf, basepathlen, sizeonly, tablespaces, sendtblspclinks);
} }
else if (S_ISREG(statbuf.st_mode)) else if (S_ISREG(statbuf.st_mode))
{ {
......
...@@ -71,13 +71,16 @@ Node *replication_parse_result; ...@@ -71,13 +71,16 @@ Node *replication_parse_result;
%token K_NOWAIT %token K_NOWAIT
%token K_MAX_RATE %token K_MAX_RATE
%token K_WAL %token K_WAL
%token K_TABLESPACE_MAP
%token K_TIMELINE %token K_TIMELINE
%token K_PHYSICAL %token K_PHYSICAL
%token K_LOGICAL %token K_LOGICAL
%token K_SLOT %token K_SLOT
%type <node> command %type <node> command
%type <node> base_backup start_replication start_logical_replication create_replication_slot drop_replication_slot identify_system timeline_history %type <node> base_backup start_replication start_logical_replication
create_replication_slot drop_replication_slot identify_system
timeline_history
%type <list> base_backup_opt_list %type <list> base_backup_opt_list
%type <defelt> base_backup_opt %type <defelt> base_backup_opt
%type <uintval> opt_timeline %type <uintval> opt_timeline
...@@ -119,12 +122,14 @@ identify_system: ...@@ -119,12 +122,14 @@ identify_system:
; ;
/* /*
* BASE_BACKUP [LABEL '<label>'] [PROGRESS] [FAST] [WAL] [NOWAIT] [MAX_RATE %d] * BASE_BACKUP [LABEL '<label>'] [PROGRESS] [FAST] [WAL] [NOWAIT]
* [MAX_RATE %d] [TABLESPACE_MAP]
*/ */
base_backup: base_backup:
K_BASE_BACKUP base_backup_opt_list K_BASE_BACKUP base_backup_opt_list
{ {
BaseBackupCmd *cmd = (BaseBackupCmd *) makeNode(BaseBackupCmd); BaseBackupCmd *cmd =
(BaseBackupCmd *) makeNode(BaseBackupCmd);
cmd->options = $2; cmd->options = $2;
$$ = (Node *) cmd; $$ = (Node *) cmd;
} }
...@@ -168,6 +173,11 @@ base_backup_opt: ...@@ -168,6 +173,11 @@ base_backup_opt:
$$ = makeDefElem("max_rate", $$ = makeDefElem("max_rate",
(Node *)makeInteger($2)); (Node *)makeInteger($2));
} }
| K_TABLESPACE_MAP
{
$$ = makeDefElem("tablespace_map",
(Node *)makeInteger(TRUE));
}
; ;
create_replication_slot: create_replication_slot:
......
...@@ -88,6 +88,7 @@ NOWAIT { return K_NOWAIT; } ...@@ -88,6 +88,7 @@ NOWAIT { return K_NOWAIT; }
PROGRESS { return K_PROGRESS; } PROGRESS { return K_PROGRESS; }
MAX_RATE { return K_MAX_RATE; } MAX_RATE { return K_MAX_RATE; }
WAL { return K_WAL; } WAL { return K_WAL; }
TABLESPACE_MAP { return K_TABLESPACE_MAP; }
TIMELINE { return K_TIMELINE; } TIMELINE { return K_TIMELINE; }
START_REPLICATION { return K_START_REPLICATION; } START_REPLICATION { return K_START_REPLICATION; }
CREATE_REPLICATION_SLOT { return K_CREATE_REPLICATION_SLOT; } CREATE_REPLICATION_SLOT { return K_CREATE_REPLICATION_SLOT; }
......
...@@ -1652,13 +1652,14 @@ BaseBackup(void) ...@@ -1652,13 +1652,14 @@ BaseBackup(void)
maxrate_clause = psprintf("MAX_RATE %u", maxrate); maxrate_clause = psprintf("MAX_RATE %u", maxrate);
basebkp = basebkp =
psprintf("BASE_BACKUP LABEL '%s' %s %s %s %s %s", psprintf("BASE_BACKUP LABEL '%s' %s %s %s %s %s %s",
escaped_label, escaped_label,
showprogress ? "PROGRESS" : "", showprogress ? "PROGRESS" : "",
includewal && !streamwal ? "WAL" : "", includewal && !streamwal ? "WAL" : "",
fastcheckpoint ? "FAST" : "", fastcheckpoint ? "FAST" : "",
includewal ? "NOWAIT" : "", includewal ? "NOWAIT" : "",
maxrate_clause ? maxrate_clause : ""); maxrate_clause ? maxrate_clause : "",
format == 't' ? "TABLESPACE_MAP": "");
if (PQsendQuery(conn, basebkp) == 0) if (PQsendQuery(conn, basebkp) == 0)
{ {
......
...@@ -17,6 +17,8 @@ ...@@ -17,6 +17,8 @@
#include "access/xlogreader.h" #include "access/xlogreader.h"
#include "datatype/timestamp.h" #include "datatype/timestamp.h"
#include "lib/stringinfo.h" #include "lib/stringinfo.h"
#include "nodes/pg_list.h"
#include "storage/fd.h"
/* Sync methods */ /* Sync methods */
...@@ -258,7 +260,9 @@ extern void assign_checkpoint_completion_target(double newval, void *extra); ...@@ -258,7 +260,9 @@ extern void assign_checkpoint_completion_target(double newval, void *extra);
* Starting/stopping a base backup * Starting/stopping a base backup
*/ */
extern XLogRecPtr do_pg_start_backup(const char *backupidstr, bool fast, extern XLogRecPtr do_pg_start_backup(const char *backupidstr, bool fast,
TimeLineID *starttli_p, char **labelfile); TimeLineID *starttli_p, char **labelfile, DIR *tblspcdir,
List **tablespaces, char **tblspcmapfile, bool infotbssize,
bool needtblspcmapfile);
extern XLogRecPtr do_pg_stop_backup(char *labelfile, bool waitforarchive, extern XLogRecPtr do_pg_stop_backup(char *labelfile, bool waitforarchive,
TimeLineID *stoptli_p); TimeLineID *stoptli_p);
extern void do_pg_abort_backup(void); extern void do_pg_abort_backup(void);
...@@ -267,4 +271,7 @@ extern void do_pg_abort_backup(void); ...@@ -267,4 +271,7 @@ extern void do_pg_abort_backup(void);
#define BACKUP_LABEL_FILE "backup_label" #define BACKUP_LABEL_FILE "backup_label"
#define BACKUP_LABEL_OLD "backup_label.old" #define BACKUP_LABEL_OLD "backup_label.old"
#define TABLESPACE_MAP "tablespace_map"
#define TABLESPACE_MAP_OLD "tablespace_map.old"
#endif /* XLOG_H */ #endif /* XLOG_H */
...@@ -21,6 +21,16 @@ ...@@ -21,6 +21,16 @@
#define MAX_RATE_UPPER 1048576 #define MAX_RATE_UPPER 1048576
typedef struct
{
char *oid;
char *path;
char *rpath; /* relative path within PGDATA, or NULL */
int64 size;
} tablespaceinfo;
extern void SendBaseBackup(BaseBackupCmd *cmd); extern void SendBaseBackup(BaseBackupCmd *cmd);
extern int64 sendTablespace(char *path, bool sizeonly);
#endif /* _BASEBACKUP_H */ #endif /* _BASEBACKUP_H */
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment