mirror of
				https://github.com/postgres/postgres.git
				synced 2025-11-03 09:13:20 +03:00 
			
		
		
		
	Backend support for streaming base backups
Add BASE_BACKUP command to walsender, allowing it to stream a base backup to the client (in tar format). The syntax is still far from ideal, that will be fixed in the switch to use a proper grammar for walsender. No client included yet, will come as a separate commit. Magnus Hagander and Heikki Linnakangas
This commit is contained in:
		@@ -1458,6 +1458,98 @@ The commands accepted in walsender mode are:
 | 
				
			|||||||
     </para>
 | 
					     </para>
 | 
				
			||||||
    </listitem>
 | 
					    </listitem>
 | 
				
			||||||
  </varlistentry>
 | 
					  </varlistentry>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  <varlistentry>
 | 
				
			||||||
 | 
					    <term>BASE_BACKUP <replaceable>options</><literal>;</><replaceable>label</></term>
 | 
				
			||||||
 | 
					    <listitem>
 | 
				
			||||||
 | 
					     <para>
 | 
				
			||||||
 | 
					      Instructs the server to start streaming a base backup.
 | 
				
			||||||
 | 
					      The system will automatically be put in backup mode with the label
 | 
				
			||||||
 | 
					      specified in <replaceable>label</> before the backup is started, and
 | 
				
			||||||
 | 
					      taken out of it when the backup is complete. The following options
 | 
				
			||||||
 | 
					      are accepted:
 | 
				
			||||||
 | 
					      <variablelist>
 | 
				
			||||||
 | 
					       <varlistentry>
 | 
				
			||||||
 | 
					        <term><literal>PROGRESS</></term>
 | 
				
			||||||
 | 
					        <listitem>
 | 
				
			||||||
 | 
					         <para>
 | 
				
			||||||
 | 
					          Request information required to generate a progress report. This will
 | 
				
			||||||
 | 
					          send back an approximate size in the header of each tablespace, which
 | 
				
			||||||
 | 
					          can be used to calculate how far along the stream is done. This is
 | 
				
			||||||
 | 
					          calculated by enumerating all the file sizes once before the transfer
 | 
				
			||||||
 | 
					          is even started, and may as such have a negative impact on the
 | 
				
			||||||
 | 
					          performance - in particular it may take longer before the first data
 | 
				
			||||||
 | 
					          is streamed. Since the database files can change during the backup,
 | 
				
			||||||
 | 
					          the size is only approximate and may both grow and shrink between
 | 
				
			||||||
 | 
					          the time of approximation and the sending of the actual files.
 | 
				
			||||||
 | 
					         </para>
 | 
				
			||||||
 | 
					        </listitem>
 | 
				
			||||||
 | 
					       </varlistentry>
 | 
				
			||||||
 | 
					      </variablelist>
 | 
				
			||||||
 | 
					     </para>
 | 
				
			||||||
 | 
					     <para>
 | 
				
			||||||
 | 
					      When the backup is started, the server will first send a header in
 | 
				
			||||||
 | 
					      ordinary result set format, followed by one or more CopyResponse
 | 
				
			||||||
 | 
					      results, one for PGDATA and one for each additional tablespace other
 | 
				
			||||||
 | 
					      than <literal>pg_default</> and <literal>pg_global</>. The data in
 | 
				
			||||||
 | 
					      the CopyResponse results will be a tar format (using ustar00
 | 
				
			||||||
 | 
					      extensions) dump of the tablespace contents.
 | 
				
			||||||
 | 
					     </para>
 | 
				
			||||||
 | 
					     <para>
 | 
				
			||||||
 | 
					      The header is an ordinary resultset with one row for each tablespace.
 | 
				
			||||||
 | 
					      The fields in this row are:
 | 
				
			||||||
 | 
					      <variablelist>
 | 
				
			||||||
 | 
					       <varlistentry>
 | 
				
			||||||
 | 
					        <term>spcoid</term>
 | 
				
			||||||
 | 
					        <listitem>
 | 
				
			||||||
 | 
					         <para>
 | 
				
			||||||
 | 
					          The oid of the tablespace, or <literal>NULL</> if it's the base
 | 
				
			||||||
 | 
					          directory.
 | 
				
			||||||
 | 
					         </para>
 | 
				
			||||||
 | 
					        </listitem>
 | 
				
			||||||
 | 
					       </varlistentry>
 | 
				
			||||||
 | 
					       <varlistentry>
 | 
				
			||||||
 | 
					        <term>spclocation</term>
 | 
				
			||||||
 | 
					        <listitem>
 | 
				
			||||||
 | 
					         <para>
 | 
				
			||||||
 | 
					          The full path of the tablespace directory, or <literal>NULL</>
 | 
				
			||||||
 | 
					          if it's the base directory.
 | 
				
			||||||
 | 
					         </para>
 | 
				
			||||||
 | 
					        </listitem>
 | 
				
			||||||
 | 
					       </varlistentry>
 | 
				
			||||||
 | 
					       <varlistentry>
 | 
				
			||||||
 | 
					        <term>size</term>
 | 
				
			||||||
 | 
					        <listitem>
 | 
				
			||||||
 | 
					         <para>
 | 
				
			||||||
 | 
					          The approximate size of the tablespace, if progress report has
 | 
				
			||||||
 | 
					          been requested; otherwise it's <literal>NULL</>.
 | 
				
			||||||
 | 
					         </para>
 | 
				
			||||||
 | 
					        </listitem>
 | 
				
			||||||
 | 
					       </varlistentry>
 | 
				
			||||||
 | 
					      </variablelist>
 | 
				
			||||||
 | 
					     </para>
 | 
				
			||||||
 | 
					     <para>
 | 
				
			||||||
 | 
					      The tar archive for the data directory and each tablespace will contain
 | 
				
			||||||
 | 
					      all files in the directories, regardless of whether they are
 | 
				
			||||||
 | 
					      <productname>PostgreSQL</> files or other files added to the same
 | 
				
			||||||
 | 
					      directory. The only excluded files are:
 | 
				
			||||||
 | 
					      <itemizedlist spacing="compact" mark="bullet">
 | 
				
			||||||
 | 
					       <listitem>
 | 
				
			||||||
 | 
					        <para>
 | 
				
			||||||
 | 
					         <filename>postmaster.pid</>
 | 
				
			||||||
 | 
					        </para>
 | 
				
			||||||
 | 
					       </listitem>
 | 
				
			||||||
 | 
					       <listitem>
 | 
				
			||||||
 | 
					        <para>
 | 
				
			||||||
 | 
					         <filename>pg_xlog</> (including subdirectories)
 | 
				
			||||||
 | 
					        </para>
 | 
				
			||||||
 | 
					       </listitem>
 | 
				
			||||||
 | 
					      </itemizedlist>
 | 
				
			||||||
 | 
					      Owner, group and file mode are set if the underlying filesystem on
 | 
				
			||||||
 | 
					      the server supports it.
 | 
				
			||||||
 | 
					     </para>
 | 
				
			||||||
 | 
					    </listitem>
 | 
				
			||||||
 | 
					  </varlistentry>
 | 
				
			||||||
</variablelist>
 | 
					</variablelist>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
</para>
 | 
					</para>
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -12,6 +12,6 @@ subdir = src/backend/replication
 | 
				
			|||||||
top_builddir = ../../..
 | 
					top_builddir = ../../..
 | 
				
			||||||
include $(top_builddir)/src/Makefile.global
 | 
					include $(top_builddir)/src/Makefile.global
 | 
				
			||||||
 | 
					
 | 
				
			||||||
OBJS = walsender.o walreceiverfuncs.o walreceiver.o
 | 
					OBJS = walsender.o walreceiverfuncs.o walreceiver.o basebackup.o
 | 
				
			||||||
 | 
					
 | 
				
			||||||
include $(top_srcdir)/src/backend/common.mk
 | 
					include $(top_srcdir)/src/backend/common.mk
 | 
				
			||||||
 
 | 
				
			|||||||
							
								
								
									
										555
									
								
								src/backend/replication/basebackup.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										555
									
								
								src/backend/replication/basebackup.c
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,555 @@
 | 
				
			|||||||
 | 
					/*-------------------------------------------------------------------------
 | 
				
			||||||
 | 
					 *
 | 
				
			||||||
 | 
					 * basebackup.c
 | 
				
			||||||
 | 
					 *	  code for taking a base backup and streaming it to a standby
 | 
				
			||||||
 | 
					 *
 | 
				
			||||||
 | 
					 * Portions Copyright (c) 2010-2011, PostgreSQL Global Development Group
 | 
				
			||||||
 | 
					 *
 | 
				
			||||||
 | 
					 * IDENTIFICATION
 | 
				
			||||||
 | 
					 *	  src/backend/replication/basebackup.c
 | 
				
			||||||
 | 
					 *
 | 
				
			||||||
 | 
					 *-------------------------------------------------------------------------
 | 
				
			||||||
 | 
					 */
 | 
				
			||||||
 | 
					#include "postgres.h"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					#include <sys/types.h>
 | 
				
			||||||
 | 
					#include <sys/stat.h>
 | 
				
			||||||
 | 
					#include <unistd.h>
 | 
				
			||||||
 | 
					#include <time.h>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					#include "access/xlog_internal.h"		/* for pg_start/stop_backup */
 | 
				
			||||||
 | 
					#include "catalog/pg_type.h"
 | 
				
			||||||
 | 
					#include "lib/stringinfo.h"
 | 
				
			||||||
 | 
					#include "libpq/libpq.h"
 | 
				
			||||||
 | 
					#include "libpq/pqformat.h"
 | 
				
			||||||
 | 
					#include "nodes/pg_list.h"
 | 
				
			||||||
 | 
					#include "replication/basebackup.h"
 | 
				
			||||||
 | 
					#include "storage/fd.h"
 | 
				
			||||||
 | 
					#include "storage/ipc.h"
 | 
				
			||||||
 | 
					#include "utils/builtins.h"
 | 
				
			||||||
 | 
					#include "utils/elog.h"
 | 
				
			||||||
 | 
					#include "utils/memutils.h"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					static int64 sendDir(char *path, int basepathlen, bool sizeonly);
 | 
				
			||||||
 | 
					static void sendFile(char *path, int basepathlen, struct stat * statbuf);
 | 
				
			||||||
 | 
					static void _tarWriteHeader(char *filename, char *linktarget,
 | 
				
			||||||
 | 
									struct stat * statbuf);
 | 
				
			||||||
 | 
					static void send_int8_string(StringInfoData *buf, int64 intval);
 | 
				
			||||||
 | 
					static void SendBackupHeader(List *tablespaces);
 | 
				
			||||||
 | 
					static void SendBackupDirectory(char *location, char *spcoid);
 | 
				
			||||||
 | 
					static void base_backup_cleanup(int code, Datum arg);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					typedef struct
 | 
				
			||||||
 | 
					{
 | 
				
			||||||
 | 
						char	   *oid;
 | 
				
			||||||
 | 
						char	   *path;
 | 
				
			||||||
 | 
						int64		size;
 | 
				
			||||||
 | 
					}	tablespaceinfo;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					/*
 | 
				
			||||||
 | 
					 * Called when ERROR or FATAL happens in SendBaseBackup() after
 | 
				
			||||||
 | 
					 * we have started the backup - make sure we end it!
 | 
				
			||||||
 | 
					 */
 | 
				
			||||||
 | 
					static void
 | 
				
			||||||
 | 
					base_backup_cleanup(int code, Datum arg)
 | 
				
			||||||
 | 
					{
 | 
				
			||||||
 | 
						do_pg_abort_backup();
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					/*
 | 
				
			||||||
 | 
					 * SendBaseBackup() - send a complete base backup.
 | 
				
			||||||
 | 
					 *
 | 
				
			||||||
 | 
					 * The function will take care of running pg_start_backup() and
 | 
				
			||||||
 | 
					 * pg_stop_backup() for the user.
 | 
				
			||||||
 | 
					 */
 | 
				
			||||||
 | 
					void
 | 
				
			||||||
 | 
					SendBaseBackup(const char *options)
 | 
				
			||||||
 | 
					{
 | 
				
			||||||
 | 
						DIR		   *dir;
 | 
				
			||||||
 | 
						struct dirent *de;
 | 
				
			||||||
 | 
						char	   *backup_label = strchr(options, ';');
 | 
				
			||||||
 | 
						bool		progress = false;
 | 
				
			||||||
 | 
						List	   *tablespaces = NIL;
 | 
				
			||||||
 | 
						tablespaceinfo *ti;
 | 
				
			||||||
 | 
						MemoryContext backup_context;
 | 
				
			||||||
 | 
						MemoryContext old_context;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						backup_context = AllocSetContextCreate(CurrentMemoryContext,
 | 
				
			||||||
 | 
															   "Streaming base backup context",
 | 
				
			||||||
 | 
															   ALLOCSET_DEFAULT_MINSIZE,
 | 
				
			||||||
 | 
															   ALLOCSET_DEFAULT_INITSIZE,
 | 
				
			||||||
 | 
															   ALLOCSET_DEFAULT_MAXSIZE);
 | 
				
			||||||
 | 
						old_context = MemoryContextSwitchTo(backup_context);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						if (backup_label == NULL)
 | 
				
			||||||
 | 
							ereport(FATAL,
 | 
				
			||||||
 | 
									(errcode(ERRCODE_PROTOCOL_VIOLATION),
 | 
				
			||||||
 | 
									 errmsg("invalid base backup options: %s", options)));
 | 
				
			||||||
 | 
						backup_label++;				/* Walk past the semicolon */
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						/* Currently the only option string supported is PROGRESS */
 | 
				
			||||||
 | 
						if (strncmp(options, "PROGRESS", 8) == 0)
 | 
				
			||||||
 | 
							progress = true;
 | 
				
			||||||
 | 
						else if (options[0] != ';')
 | 
				
			||||||
 | 
							ereport(FATAL,
 | 
				
			||||||
 | 
									(errcode(ERRCODE_PROTOCOL_VIOLATION),
 | 
				
			||||||
 | 
									 errmsg("invalid base backup options: %s", options)));
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						/* Make sure we can open the directory with tablespaces in it */
 | 
				
			||||||
 | 
						dir = AllocateDir("pg_tblspc");
 | 
				
			||||||
 | 
						if (!dir)
 | 
				
			||||||
 | 
							ereport(ERROR,
 | 
				
			||||||
 | 
									(errmsg("unable to open directory pg_tblspc: %m")));
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						/* Add a node for the base directory */
 | 
				
			||||||
 | 
						ti = palloc0(sizeof(tablespaceinfo));
 | 
				
			||||||
 | 
						ti->size = progress ? sendDir(".", 1, true) : -1;
 | 
				
			||||||
 | 
						tablespaces = lappend(tablespaces, ti);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						/* Collect information about all tablespaces */
 | 
				
			||||||
 | 
						while ((de = ReadDir(dir, "pg_tblspc")) != NULL)
 | 
				
			||||||
 | 
						{
 | 
				
			||||||
 | 
							char		fullpath[MAXPGPATH];
 | 
				
			||||||
 | 
							char		linkpath[MAXPGPATH];
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							/* 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);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							MemSet(linkpath, 0, sizeof(linkpath));
 | 
				
			||||||
 | 
							if (readlink(fullpath, linkpath, sizeof(linkpath) - 1) == -1)
 | 
				
			||||||
 | 
							{
 | 
				
			||||||
 | 
								ereport(WARNING,
 | 
				
			||||||
 | 
									  (errmsg("unable to read symbolic link %s: %m", fullpath)));
 | 
				
			||||||
 | 
								continue;
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							ti = palloc(sizeof(tablespaceinfo));
 | 
				
			||||||
 | 
							ti->oid = pstrdup(de->d_name);
 | 
				
			||||||
 | 
							ti->path = pstrdup(linkpath);
 | 
				
			||||||
 | 
							ti->size = progress ? sendDir(linkpath, strlen(linkpath), true) : -1;
 | 
				
			||||||
 | 
							tablespaces = lappend(tablespaces, ti);
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						FreeDir(dir);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						do_pg_start_backup(backup_label, true);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						PG_ENSURE_ERROR_CLEANUP(base_backup_cleanup, (Datum) 0);
 | 
				
			||||||
 | 
						{
 | 
				
			||||||
 | 
							ListCell   *lc;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							/* Send tablespace header */
 | 
				
			||||||
 | 
							SendBackupHeader(tablespaces);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							/* Send off our tablespaces one by one */
 | 
				
			||||||
 | 
							foreach(lc, tablespaces)
 | 
				
			||||||
 | 
							{
 | 
				
			||||||
 | 
								ti = (tablespaceinfo *) lfirst(lc);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
								SendBackupDirectory(ti->path, ti->oid);
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						PG_END_ENSURE_ERROR_CLEANUP(base_backup_cleanup, (Datum) 0);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						do_pg_stop_backup();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						MemoryContextSwitchTo(old_context);
 | 
				
			||||||
 | 
						MemoryContextDelete(backup_context);
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					static void
 | 
				
			||||||
 | 
					send_int8_string(StringInfoData *buf, int64 intval)
 | 
				
			||||||
 | 
					{
 | 
				
			||||||
 | 
						char		is[32];
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						sprintf(is, INT64_FORMAT, intval);
 | 
				
			||||||
 | 
						pq_sendint(buf, strlen(is), 4);
 | 
				
			||||||
 | 
						pq_sendbytes(buf, is, strlen(is));
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					static void
 | 
				
			||||||
 | 
					SendBackupHeader(List *tablespaces)
 | 
				
			||||||
 | 
					{
 | 
				
			||||||
 | 
						StringInfoData buf;
 | 
				
			||||||
 | 
						ListCell   *lc;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						/* Construct and send the directory information */
 | 
				
			||||||
 | 
						pq_beginmessage(&buf, 'T'); /* RowDescription */
 | 
				
			||||||
 | 
						pq_sendint(&buf, 3, 2);		/* 3 fields */
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						/* First field - spcoid */
 | 
				
			||||||
 | 
						pq_sendstring(&buf, "spcoid");
 | 
				
			||||||
 | 
						pq_sendint(&buf, 0, 4);		/* table oid */
 | 
				
			||||||
 | 
						pq_sendint(&buf, 0, 2);		/* attnum */
 | 
				
			||||||
 | 
						pq_sendint(&buf, OIDOID, 4);	/* type oid */
 | 
				
			||||||
 | 
						pq_sendint(&buf, 4, 2);		/* typlen */
 | 
				
			||||||
 | 
						pq_sendint(&buf, 0, 4);		/* typmod */
 | 
				
			||||||
 | 
						pq_sendint(&buf, 0, 2);		/* format code */
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						/* Second field - spcpath */
 | 
				
			||||||
 | 
						pq_sendstring(&buf, "spclocation");
 | 
				
			||||||
 | 
						pq_sendint(&buf, 0, 4);
 | 
				
			||||||
 | 
						pq_sendint(&buf, 0, 2);
 | 
				
			||||||
 | 
						pq_sendint(&buf, TEXTOID, 4);
 | 
				
			||||||
 | 
						pq_sendint(&buf, -1, 2);
 | 
				
			||||||
 | 
						pq_sendint(&buf, 0, 4);
 | 
				
			||||||
 | 
						pq_sendint(&buf, 0, 2);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						/* Third field - size */
 | 
				
			||||||
 | 
						pq_sendstring(&buf, "size");
 | 
				
			||||||
 | 
						pq_sendint(&buf, 0, 4);
 | 
				
			||||||
 | 
						pq_sendint(&buf, 0, 2);
 | 
				
			||||||
 | 
						pq_sendint(&buf, INT8OID, 4);
 | 
				
			||||||
 | 
						pq_sendint(&buf, 8, 2);
 | 
				
			||||||
 | 
						pq_sendint(&buf, 0, 4);
 | 
				
			||||||
 | 
						pq_sendint(&buf, 0, 2);
 | 
				
			||||||
 | 
						pq_endmessage(&buf);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						foreach(lc, tablespaces)
 | 
				
			||||||
 | 
						{
 | 
				
			||||||
 | 
							tablespaceinfo *ti = lfirst(lc);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							/* Send one datarow message */
 | 
				
			||||||
 | 
							pq_beginmessage(&buf, 'D');
 | 
				
			||||||
 | 
							pq_sendint(&buf, 3, 2); /* number of columns */
 | 
				
			||||||
 | 
							if (ti->path == NULL)
 | 
				
			||||||
 | 
							{
 | 
				
			||||||
 | 
								pq_sendint(&buf, -1, 4);	/* Length = -1 ==> NULL */
 | 
				
			||||||
 | 
								pq_sendint(&buf, -1, 4);
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							else
 | 
				
			||||||
 | 
							{
 | 
				
			||||||
 | 
								pq_sendint(&buf, strlen(ti->oid), 4);		/* length */
 | 
				
			||||||
 | 
								pq_sendbytes(&buf, ti->oid, strlen(ti->oid));
 | 
				
			||||||
 | 
								pq_sendint(&buf, strlen(ti->path), 4);		/* length */
 | 
				
			||||||
 | 
								pq_sendbytes(&buf, ti->path, strlen(ti->path));
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							if (ti->size >= 0)
 | 
				
			||||||
 | 
								send_int8_string(&buf, ti->size / 1024);
 | 
				
			||||||
 | 
							else
 | 
				
			||||||
 | 
								pq_sendint(&buf, -1, 4);	/* NULL */
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							pq_endmessage(&buf);
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						/* Send a CommandComplete message */
 | 
				
			||||||
 | 
						pq_puttextmessage('C', "SELECT");
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					static void
 | 
				
			||||||
 | 
					SendBackupDirectory(char *location, char *spcoid)
 | 
				
			||||||
 | 
					{
 | 
				
			||||||
 | 
						StringInfoData buf;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						/* Send CopyOutResponse message */
 | 
				
			||||||
 | 
						pq_beginmessage(&buf, 'H');
 | 
				
			||||||
 | 
						pq_sendbyte(&buf, 0);		/* overall format */
 | 
				
			||||||
 | 
						pq_sendint(&buf, 0, 2);		/* natts */
 | 
				
			||||||
 | 
						pq_endmessage(&buf);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						/* tar up the data directory if NULL, otherwise the tablespace */
 | 
				
			||||||
 | 
						sendDir(location == NULL ? "." : location,
 | 
				
			||||||
 | 
								location == NULL ? 1 : strlen(location),
 | 
				
			||||||
 | 
								false);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						/* Send CopyDone message */
 | 
				
			||||||
 | 
						pq_putemptymessage('c');
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					static int64
 | 
				
			||||||
 | 
					sendDir(char *path, int basepathlen, bool sizeonly)
 | 
				
			||||||
 | 
					{
 | 
				
			||||||
 | 
						DIR		   *dir;
 | 
				
			||||||
 | 
						struct dirent *de;
 | 
				
			||||||
 | 
						char		pathbuf[MAXPGPATH];
 | 
				
			||||||
 | 
						struct stat statbuf;
 | 
				
			||||||
 | 
						int64		size = 0;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						dir = AllocateDir(path);
 | 
				
			||||||
 | 
						while ((de = ReadDir(dir, path)) != NULL)
 | 
				
			||||||
 | 
						{
 | 
				
			||||||
 | 
							/* Skip special stuff */
 | 
				
			||||||
 | 
							if (strcmp(de->d_name, ".") == 0 || strcmp(de->d_name, "..") == 0)
 | 
				
			||||||
 | 
								continue;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							snprintf(pathbuf, MAXPGPATH, "%s/%s", path, de->d_name);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							/* Skip postmaster.pid in the data directory */
 | 
				
			||||||
 | 
							if (strcmp(pathbuf, "./postmaster.pid") == 0)
 | 
				
			||||||
 | 
								continue;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							if (lstat(pathbuf, &statbuf) != 0)
 | 
				
			||||||
 | 
							{
 | 
				
			||||||
 | 
								if (errno != ENOENT)
 | 
				
			||||||
 | 
									ereport(ERROR,
 | 
				
			||||||
 | 
											(errcode(errcode_for_file_access()),
 | 
				
			||||||
 | 
											 errmsg("could not stat file or directory \"%s\": %m",
 | 
				
			||||||
 | 
													pathbuf)));
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
								/* If the file went away while scanning, it's no error. */
 | 
				
			||||||
 | 
								continue;
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							/*
 | 
				
			||||||
 | 
							 * We can skip pg_xlog, the WAL segments need to be fetched from the
 | 
				
			||||||
 | 
							 * WAL archive anyway. But include it as an empty directory anyway, so
 | 
				
			||||||
 | 
							 * we get permissions right.
 | 
				
			||||||
 | 
							 */
 | 
				
			||||||
 | 
							if (strcmp(pathbuf, "./pg_xlog") == 0)
 | 
				
			||||||
 | 
							{
 | 
				
			||||||
 | 
								if (!sizeonly)
 | 
				
			||||||
 | 
									_tarWriteHeader(pathbuf + basepathlen + 1, NULL, &statbuf);
 | 
				
			||||||
 | 
								size += 512;		/* Size of the header just added */
 | 
				
			||||||
 | 
								continue;			/* don't recurse into pg_xlog */
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					#ifndef WIN32
 | 
				
			||||||
 | 
							if (S_ISLNK(statbuf.st_mode) && strcmp(path, "./pg_tblspc") == 0)
 | 
				
			||||||
 | 
					#else
 | 
				
			||||||
 | 
							if (pgwin32_is_junction(pathbuf) && strcmp(path, "./pg_tblspc") == 0)
 | 
				
			||||||
 | 
					#endif
 | 
				
			||||||
 | 
							{
 | 
				
			||||||
 | 
								/* Allow symbolic links in pg_tblspc */
 | 
				
			||||||
 | 
								char		linkpath[MAXPGPATH];
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
								MemSet(linkpath, 0, sizeof(linkpath));
 | 
				
			||||||
 | 
								if (readlink(pathbuf, linkpath, sizeof(linkpath) - 1) == -1)
 | 
				
			||||||
 | 
									ereport(ERROR,
 | 
				
			||||||
 | 
											(errcode(errcode_for_file_access()),
 | 
				
			||||||
 | 
											 errmsg("could not read symbolic link \"%s\": %m",
 | 
				
			||||||
 | 
													pathbuf)));
 | 
				
			||||||
 | 
								if (!sizeonly)
 | 
				
			||||||
 | 
									_tarWriteHeader(pathbuf + basepathlen + 1, linkpath, &statbuf);
 | 
				
			||||||
 | 
								size += 512;		/* Size of the header just added */
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							else if (S_ISDIR(statbuf.st_mode))
 | 
				
			||||||
 | 
							{
 | 
				
			||||||
 | 
								/*
 | 
				
			||||||
 | 
								 * Store a directory entry in the tar file so we can get the
 | 
				
			||||||
 | 
								 * permissions right.
 | 
				
			||||||
 | 
								 */
 | 
				
			||||||
 | 
								if (!sizeonly)
 | 
				
			||||||
 | 
									_tarWriteHeader(pathbuf + basepathlen + 1, NULL, &statbuf);
 | 
				
			||||||
 | 
								size += 512;		/* Size of the header just added */
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
								/* call ourselves recursively for a directory */
 | 
				
			||||||
 | 
								size += sendDir(pathbuf, basepathlen, sizeonly);
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							else if (S_ISREG(statbuf.st_mode))
 | 
				
			||||||
 | 
							{
 | 
				
			||||||
 | 
								/* Add size, rounded up to 512byte block */
 | 
				
			||||||
 | 
								size += ((statbuf.st_size + 511) & ~511);
 | 
				
			||||||
 | 
								if (!sizeonly)
 | 
				
			||||||
 | 
									sendFile(pathbuf, basepathlen, &statbuf);
 | 
				
			||||||
 | 
								size += 512;		/* Size of the header of the file */
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							else
 | 
				
			||||||
 | 
								ereport(WARNING,
 | 
				
			||||||
 | 
										(errmsg("skipping special file \"%s\"", pathbuf)));
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						FreeDir(dir);
 | 
				
			||||||
 | 
						return size;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					/*****
 | 
				
			||||||
 | 
					 * Functions for handling tar file format
 | 
				
			||||||
 | 
					 *
 | 
				
			||||||
 | 
					 * Copied from pg_dump, but modified to work with libpq for sending
 | 
				
			||||||
 | 
					 */
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					/*
 | 
				
			||||||
 | 
					 * Utility routine to print possibly larger than 32 bit integers in a
 | 
				
			||||||
 | 
					 * portable fashion.  Filled with zeros.
 | 
				
			||||||
 | 
					 */
 | 
				
			||||||
 | 
					static void
 | 
				
			||||||
 | 
					print_val(char *s, uint64 val, unsigned int base, size_t len)
 | 
				
			||||||
 | 
					{
 | 
				
			||||||
 | 
						int			i;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						for (i = len; i > 0; i--)
 | 
				
			||||||
 | 
						{
 | 
				
			||||||
 | 
							int			digit = val % base;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							s[i - 1] = '0' + digit;
 | 
				
			||||||
 | 
							val = val / base;
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					/*
 | 
				
			||||||
 | 
					 * Maximum file size for a tar member: The limit inherent in the
 | 
				
			||||||
 | 
					 * format is 2^33-1 bytes (nearly 8 GB).  But we don't want to exceed
 | 
				
			||||||
 | 
					 * what we can represent in pgoff_t.
 | 
				
			||||||
 | 
					 */
 | 
				
			||||||
 | 
					#define MAX_TAR_MEMBER_FILELEN (((int64) 1 << Min(33, sizeof(pgoff_t)*8 - 1)) - 1)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					static int
 | 
				
			||||||
 | 
					_tarChecksum(char *header)
 | 
				
			||||||
 | 
					{
 | 
				
			||||||
 | 
						int			i,
 | 
				
			||||||
 | 
									sum;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						sum = 0;
 | 
				
			||||||
 | 
						for (i = 0; i < 512; i++)
 | 
				
			||||||
 | 
							if (i < 148 || i >= 156)
 | 
				
			||||||
 | 
								sum += 0xFF & header[i];
 | 
				
			||||||
 | 
						return sum + 256;			/* Assume 8 blanks in checksum field */
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					/* Given the member, write the TAR header & send the file */
 | 
				
			||||||
 | 
					static void
 | 
				
			||||||
 | 
					sendFile(char *filename, int basepathlen, struct stat * statbuf)
 | 
				
			||||||
 | 
					{
 | 
				
			||||||
 | 
						FILE	   *fp;
 | 
				
			||||||
 | 
						char		buf[32768];
 | 
				
			||||||
 | 
						size_t		cnt;
 | 
				
			||||||
 | 
						pgoff_t		len = 0;
 | 
				
			||||||
 | 
						size_t		pad;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						fp = AllocateFile(filename, "rb");
 | 
				
			||||||
 | 
						if (fp == NULL)
 | 
				
			||||||
 | 
							ereport(ERROR,
 | 
				
			||||||
 | 
									(errcode(errcode_for_file_access()),
 | 
				
			||||||
 | 
									 errmsg("could not open file \"%s\": %m", filename)));
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						/*
 | 
				
			||||||
 | 
						 * Some compilers will throw a warning knowing this test can never be true
 | 
				
			||||||
 | 
						 * because pgoff_t can't exceed the compared maximum on their platform.
 | 
				
			||||||
 | 
						 */
 | 
				
			||||||
 | 
						if (statbuf->st_size > MAX_TAR_MEMBER_FILELEN)
 | 
				
			||||||
 | 
							ereport(ERROR,
 | 
				
			||||||
 | 
									(errmsg("archive member \"%s\" too large for tar format",
 | 
				
			||||||
 | 
											filename)));
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						_tarWriteHeader(filename + basepathlen + 1, NULL, statbuf);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						while ((cnt = fread(buf, 1, Min(sizeof(buf), statbuf->st_size - len), fp)) > 0)
 | 
				
			||||||
 | 
						{
 | 
				
			||||||
 | 
							/* Send the chunk as a CopyData message */
 | 
				
			||||||
 | 
							pq_putmessage('d', buf, cnt);
 | 
				
			||||||
 | 
							len += cnt;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							if (len >= statbuf->st_size)
 | 
				
			||||||
 | 
							{
 | 
				
			||||||
 | 
								/*
 | 
				
			||||||
 | 
								 * Reached end of file. The file could be longer, if it was
 | 
				
			||||||
 | 
								 * extended while we were sending it, but for a base backup we can
 | 
				
			||||||
 | 
								 * ignore such extended data. It will be restored from WAL.
 | 
				
			||||||
 | 
								 */
 | 
				
			||||||
 | 
								break;
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						/* If the file was truncated while we were sending it, pad it with zeros */
 | 
				
			||||||
 | 
						if (len < statbuf->st_size)
 | 
				
			||||||
 | 
						{
 | 
				
			||||||
 | 
							MemSet(buf, 0, sizeof(buf));
 | 
				
			||||||
 | 
							while (len < statbuf->st_size)
 | 
				
			||||||
 | 
							{
 | 
				
			||||||
 | 
								cnt = Min(sizeof(buf), statbuf->st_size - len);
 | 
				
			||||||
 | 
								pq_putmessage('d', buf, cnt);
 | 
				
			||||||
 | 
								len += cnt;
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						/* Pad to 512 byte boundary, per tar format requirements */
 | 
				
			||||||
 | 
						pad = ((len + 511) & ~511) - len;
 | 
				
			||||||
 | 
						if (pad > 0)
 | 
				
			||||||
 | 
						{
 | 
				
			||||||
 | 
							MemSet(buf, 0, pad);
 | 
				
			||||||
 | 
							pq_putmessage('d', buf, pad);
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						FreeFile(fp);
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					static void
 | 
				
			||||||
 | 
					_tarWriteHeader(char *filename, char *linktarget, struct stat * statbuf)
 | 
				
			||||||
 | 
					{
 | 
				
			||||||
 | 
						char		h[512];
 | 
				
			||||||
 | 
						int			lastSum = 0;
 | 
				
			||||||
 | 
						int			sum;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						memset(h, 0, sizeof(h));
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						/* Name 100 */
 | 
				
			||||||
 | 
						sprintf(&h[0], "%.99s", filename);
 | 
				
			||||||
 | 
						if (linktarget != NULL || S_ISDIR(statbuf->st_mode))
 | 
				
			||||||
 | 
						{
 | 
				
			||||||
 | 
							/*
 | 
				
			||||||
 | 
							 * We only support symbolic links to directories, and this is
 | 
				
			||||||
 | 
							 * indicated in the tar format by adding a slash at the end of the
 | 
				
			||||||
 | 
							 * name, the same as for regular directories.
 | 
				
			||||||
 | 
							 */
 | 
				
			||||||
 | 
							h[strlen(filename)] = '/';
 | 
				
			||||||
 | 
							h[strlen(filename) + 1] = '\0';
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						/* Mode 8 */
 | 
				
			||||||
 | 
						sprintf(&h[100], "%07o ", statbuf->st_mode);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						/* User ID 8 */
 | 
				
			||||||
 | 
						sprintf(&h[108], "%07o ", statbuf->st_uid);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						/* Group 8 */
 | 
				
			||||||
 | 
						sprintf(&h[117], "%07o ", statbuf->st_gid);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						/* File size 12 - 11 digits, 1 space, no NUL */
 | 
				
			||||||
 | 
						if (linktarget != NULL || S_ISDIR(statbuf->st_mode))
 | 
				
			||||||
 | 
							/* Symbolic link or directory has size zero */
 | 
				
			||||||
 | 
							print_val(&h[124], 0, 8, 11);
 | 
				
			||||||
 | 
						else
 | 
				
			||||||
 | 
							print_val(&h[124], statbuf->st_size, 8, 11);
 | 
				
			||||||
 | 
						sprintf(&h[135], " ");
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						/* Mod Time 12 */
 | 
				
			||||||
 | 
						sprintf(&h[136], "%011o ", (int) statbuf->st_mtime);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						/* Checksum 8 */
 | 
				
			||||||
 | 
						sprintf(&h[148], "%06o ", lastSum);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						if (linktarget != NULL)
 | 
				
			||||||
 | 
						{
 | 
				
			||||||
 | 
							/* Type - Symbolic link */
 | 
				
			||||||
 | 
							sprintf(&h[156], "2");
 | 
				
			||||||
 | 
							strcpy(&h[157], linktarget);
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						else if (S_ISDIR(statbuf->st_mode))
 | 
				
			||||||
 | 
							/* Type - directory */
 | 
				
			||||||
 | 
							sprintf(&h[156], "5");
 | 
				
			||||||
 | 
						else
 | 
				
			||||||
 | 
							/* Type - regular file */
 | 
				
			||||||
 | 
							sprintf(&h[156], "0");
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						/* Link tag 100 (NULL) */
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						/* Magic 6 + Version 2 */
 | 
				
			||||||
 | 
						sprintf(&h[257], "ustar00");
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						/* User 32 */
 | 
				
			||||||
 | 
						/* XXX: Do we need to care about setting correct username? */
 | 
				
			||||||
 | 
						sprintf(&h[265], "%.31s", "postgres");
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						/* Group 32 */
 | 
				
			||||||
 | 
						/* XXX: Do we need to care about setting correct group name? */
 | 
				
			||||||
 | 
						sprintf(&h[297], "%.31s", "postgres");
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						/* Maj Dev 8 */
 | 
				
			||||||
 | 
						sprintf(&h[329], "%6o ", 0);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						/* Min Dev 8 */
 | 
				
			||||||
 | 
						sprintf(&h[337], "%6o ", 0);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						while ((sum = _tarChecksum(h)) != lastSum)
 | 
				
			||||||
 | 
						{
 | 
				
			||||||
 | 
							sprintf(&h[148], "%06o ", sum);
 | 
				
			||||||
 | 
							lastSum = sum;
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						pq_putmessage('d', h, 512);
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
@@ -44,6 +44,7 @@
 | 
				
			|||||||
#include "libpq/pqformat.h"
 | 
					#include "libpq/pqformat.h"
 | 
				
			||||||
#include "libpq/pqsignal.h"
 | 
					#include "libpq/pqsignal.h"
 | 
				
			||||||
#include "miscadmin.h"
 | 
					#include "miscadmin.h"
 | 
				
			||||||
 | 
					#include "replication/basebackup.h"
 | 
				
			||||||
#include "replication/walprotocol.h"
 | 
					#include "replication/walprotocol.h"
 | 
				
			||||||
#include "replication/walsender.h"
 | 
					#include "replication/walsender.h"
 | 
				
			||||||
#include "storage/fd.h"
 | 
					#include "storage/fd.h"
 | 
				
			||||||
@@ -54,6 +55,7 @@
 | 
				
			|||||||
#include "utils/guc.h"
 | 
					#include "utils/guc.h"
 | 
				
			||||||
#include "utils/memutils.h"
 | 
					#include "utils/memutils.h"
 | 
				
			||||||
#include "utils/ps_status.h"
 | 
					#include "utils/ps_status.h"
 | 
				
			||||||
 | 
					#include "utils/resowner.h"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
/* Array of WalSnds in shared memory */
 | 
					/* Array of WalSnds in shared memory */
 | 
				
			||||||
@@ -136,6 +138,9 @@ WalSenderMain(void)
 | 
				
			|||||||
										   ALLOCSET_DEFAULT_MAXSIZE);
 | 
															   ALLOCSET_DEFAULT_MAXSIZE);
 | 
				
			||||||
	MemoryContextSwitchTo(walsnd_context);
 | 
						MemoryContextSwitchTo(walsnd_context);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						/* Set up resource owner */
 | 
				
			||||||
 | 
						CurrentResourceOwner = ResourceOwnerCreate(NULL, "walsender top-level resource owner");
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	/* Unblock signals (they were blocked when the postmaster forked us) */
 | 
						/* Unblock signals (they were blocked when the postmaster forked us) */
 | 
				
			||||||
	PG_SETMASK(&UnBlockSig);
 | 
						PG_SETMASK(&UnBlockSig);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -305,6 +310,15 @@ WalSndHandshake(void)
 | 
				
			|||||||
						/* break out of the loop */
 | 
											/* break out of the loop */
 | 
				
			||||||
						replication_started = true;
 | 
											replication_started = true;
 | 
				
			||||||
					}
 | 
										}
 | 
				
			||||||
 | 
										else if (strncmp(query_string, "BASE_BACKUP ", 12) == 0)
 | 
				
			||||||
 | 
										{
 | 
				
			||||||
 | 
											/* Command is BASE_BACKUP <options>;<label> */
 | 
				
			||||||
 | 
											SendBaseBackup(query_string + strlen("BASE_BACKUP "));
 | 
				
			||||||
 | 
											/* Send CommandComplete and ReadyForQuery messages */
 | 
				
			||||||
 | 
											EndCommand("SELECT", DestRemote);
 | 
				
			||||||
 | 
											ReadyForQuery(DestRemote);
 | 
				
			||||||
 | 
											/* ReadyForQuery did pq_flush for us */
 | 
				
			||||||
 | 
										}
 | 
				
			||||||
					else
 | 
										else
 | 
				
			||||||
					{
 | 
										{
 | 
				
			||||||
						ereport(FATAL,
 | 
											ereport(FATAL,
 | 
				
			||||||
 
 | 
				
			|||||||
							
								
								
									
										17
									
								
								src/include/replication/basebackup.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										17
									
								
								src/include/replication/basebackup.h
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,17 @@
 | 
				
			|||||||
 | 
					/*-------------------------------------------------------------------------
 | 
				
			||||||
 | 
					 *
 | 
				
			||||||
 | 
					 * basebackup.h
 | 
				
			||||||
 | 
					 *	  Exports from replication/basebackup.c.
 | 
				
			||||||
 | 
					 *
 | 
				
			||||||
 | 
					 * Portions Copyright (c) 2010-2010, PostgreSQL Global Development Group
 | 
				
			||||||
 | 
					 *
 | 
				
			||||||
 | 
					 * src/include/replication/walsender.h
 | 
				
			||||||
 | 
					 *
 | 
				
			||||||
 | 
					 *-------------------------------------------------------------------------
 | 
				
			||||||
 | 
					 */
 | 
				
			||||||
 | 
					#ifndef _BASEBACKUP_H
 | 
				
			||||||
 | 
					#define _BASEBACKUP_H
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					extern void SendBaseBackup(const char *options);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					#endif   /* _BASEBACKUP_H */
 | 
				
			||||||
		Reference in New Issue
	
	Block a user