2013-01-24 06:35:48 +00:00
|
|
|
/*-------------------------------------------------------------------------
|
|
|
|
*
|
2014-01-27 12:02:56 +09:00
|
|
|
* pg_arman.h: Backup/Recovery manager for PostgreSQL.
|
2013-01-24 06:35:48 +00:00
|
|
|
*
|
2013-09-09 09:00:13 +00:00
|
|
|
* Copyright (c) 2009-2013, NIPPON TELEGRAPH AND TELEPHONE CORPORATION
|
2013-01-24 06:35:48 +00:00
|
|
|
*
|
|
|
|
*-------------------------------------------------------------------------
|
|
|
|
*/
|
|
|
|
#ifndef PG_RMAN_H
|
|
|
|
#define PG_RMAN_H
|
|
|
|
|
|
|
|
#include "postgres_fe.h"
|
|
|
|
|
|
|
|
#include <limits.h>
|
|
|
|
#include "libpq-fe.h"
|
|
|
|
|
|
|
|
#include "pgut/pgut.h"
|
|
|
|
#include "access/xlogdefs.h"
|
2013-12-12 21:39:40 +09:00
|
|
|
#include "access/xlog_internal.h"
|
2016-01-15 23:47:38 +09:00
|
|
|
#include "catalog/pg_control.h"
|
2013-01-24 06:35:48 +00:00
|
|
|
#include "utils/pg_crc.h"
|
|
|
|
#include "parray.h"
|
2016-01-15 23:47:38 +09:00
|
|
|
#include "datapagemap.h"
|
2016-02-27 21:07:55 +03:00
|
|
|
#include "storage/bufpage.h"
|
|
|
|
#include "storage/block.h"
|
2016-07-07 18:38:56 +03:00
|
|
|
#include "storage/checksum.h"
|
2013-01-24 06:35:48 +00:00
|
|
|
|
2013-12-12 22:20:08 +09:00
|
|
|
/* Query to fetch current transaction ID */
|
2013-01-24 06:35:48 +00:00
|
|
|
#define TXID_CURRENT_SQL "SELECT txid_current();"
|
2016-09-29 17:33:21 +03:00
|
|
|
#define TXID_CURRENT_IF_SQL "SELECT txid_snapshot_xmax(txid_current_snapshot());"
|
2013-01-24 06:35:48 +00:00
|
|
|
|
|
|
|
/* Directory/File names */
|
|
|
|
#define DATABASE_DIR "database"
|
|
|
|
#define RESTORE_WORK_DIR "backup"
|
2013-09-09 09:00:13 +00:00
|
|
|
#define PG_XLOG_DIR "pg_xlog"
|
2013-01-24 06:35:48 +00:00
|
|
|
#define PG_TBLSPC_DIR "pg_tblspc"
|
|
|
|
#define BACKUP_INI_FILE "backup.ini"
|
2014-01-27 12:02:56 +09:00
|
|
|
#define PG_RMAN_INI_FILE "pg_arman.ini"
|
2013-01-24 06:35:48 +00:00
|
|
|
#define MKDIRS_SH_FILE "mkdirs.sh"
|
|
|
|
#define DATABASE_FILE_LIST "file_database.txt"
|
2013-09-09 09:00:13 +00:00
|
|
|
#define PG_BACKUP_LABEL_FILE "backup_label"
|
|
|
|
#define PG_BLACK_LIST "black_list"
|
2013-01-24 06:35:48 +00:00
|
|
|
|
|
|
|
/* Direcotry/File permission */
|
|
|
|
#define DIR_PERMISSION (0700)
|
|
|
|
#define FILE_PERMISSION (0600)
|
|
|
|
|
|
|
|
/* backup mode file */
|
|
|
|
typedef struct pgFile
|
|
|
|
{
|
|
|
|
time_t mtime; /* time of last modification */
|
|
|
|
mode_t mode; /* protection (file type and permission) */
|
|
|
|
size_t size; /* size of the file */
|
|
|
|
size_t read_size; /* size of the portion read (if only some pages are
|
|
|
|
backed up partially, it's different from size) */
|
|
|
|
size_t write_size; /* size of the backed-up file. BYTES_INVALID means
|
|
|
|
that the file existed but was not backed up
|
|
|
|
because not modified since last backup. */
|
|
|
|
pg_crc32 crc; /* CRC value of the file, regular file only */
|
2016-09-02 20:38:39 +03:00
|
|
|
char *linked; /* path of the linked file */
|
2013-01-24 06:35:48 +00:00
|
|
|
bool is_datafile; /* true if the file is PostgreSQL data file */
|
2016-02-27 21:07:55 +03:00
|
|
|
char *path; /* path of the file */
|
|
|
|
char *ptrack_path;
|
2016-05-01 22:05:18 +03:00
|
|
|
int segno; /* Segment number for ptrack */
|
2016-09-02 20:38:39 +03:00
|
|
|
volatile uint32 lock;
|
2016-01-15 23:47:38 +09:00
|
|
|
datapagemap_t pagemap;
|
2013-01-24 06:35:48 +00:00
|
|
|
} pgFile;
|
|
|
|
|
|
|
|
#define IsValidTime(tm) \
|
|
|
|
((tm.tm_sec >= 0 && tm.tm_sec <= 60) && /* range check for tm_sec (0-60) */ \
|
|
|
|
(tm.tm_min >= 0 && tm.tm_min <= 59) && /* range check for tm_min (0-59) */ \
|
|
|
|
(tm.tm_hour >= 0 && tm.tm_hour <= 23) && /* range check for tm_hour(0-23) */ \
|
|
|
|
(tm.tm_mday >= 1 && tm.tm_mday <= 31) && /* range check for tm_mday(1-31) */ \
|
|
|
|
(tm.tm_mon >= 0 && tm.tm_mon <= 11) && /* range check for tm_mon (0-23) */ \
|
|
|
|
(tm.tm_year + 1900 >= 1900)) /* range check for tm_year(70-) */
|
|
|
|
|
2016-05-01 22:05:18 +03:00
|
|
|
/* Effective data size */
|
|
|
|
#define MAPSIZE (BLCKSZ - MAXALIGN(SizeOfPageHeaderData))
|
|
|
|
|
2013-01-24 06:35:48 +00:00
|
|
|
/* Backup status */
|
|
|
|
/* XXX re-order ? */
|
|
|
|
typedef enum BackupStatus
|
|
|
|
{
|
|
|
|
BACKUP_STATUS_INVALID, /* the pgBackup is invalid */
|
|
|
|
BACKUP_STATUS_OK, /* completed backup */
|
|
|
|
BACKUP_STATUS_RUNNING, /* running backup */
|
|
|
|
BACKUP_STATUS_ERROR, /* aborted because of unexpected error */
|
|
|
|
BACKUP_STATUS_DELETING, /* data files are being deleted */
|
|
|
|
BACKUP_STATUS_DELETED, /* data files have been deleted */
|
|
|
|
BACKUP_STATUS_DONE, /* completed but not validated yet */
|
|
|
|
BACKUP_STATUS_CORRUPT /* files are corrupted, not available */
|
|
|
|
} BackupStatus;
|
|
|
|
|
|
|
|
typedef enum BackupMode
|
|
|
|
{
|
|
|
|
BACKUP_MODE_INVALID,
|
2014-01-30 15:58:55 +09:00
|
|
|
BACKUP_MODE_DIFF_PAGE, /* differential page backup */
|
2016-02-27 21:07:55 +03:00
|
|
|
BACKUP_MODE_DIFF_PTRACK, /* differential page backup with ptrack system*/
|
2013-01-24 06:35:48 +00:00
|
|
|
BACKUP_MODE_FULL /* full backup */
|
|
|
|
} BackupMode;
|
|
|
|
|
|
|
|
/*
|
2014-01-27 12:02:56 +09:00
|
|
|
* pg_arman takes backup into the directroy $BACKUP_PATH/<date>/<time>.
|
2013-01-24 06:35:48 +00:00
|
|
|
*
|
|
|
|
* status == -1 indicates the pgBackup is invalid.
|
|
|
|
*/
|
|
|
|
typedef struct pgBackup
|
|
|
|
{
|
|
|
|
/* Backup Level */
|
|
|
|
BackupMode backup_mode;
|
|
|
|
|
|
|
|
/* Status - one of BACKUP_STATUS_xxx */
|
|
|
|
BackupStatus status;
|
|
|
|
|
|
|
|
/* Timestamp, etc. */
|
|
|
|
TimeLineID tli;
|
|
|
|
XLogRecPtr start_lsn;
|
|
|
|
XLogRecPtr stop_lsn;
|
|
|
|
time_t start_time;
|
|
|
|
time_t end_time;
|
|
|
|
time_t recovery_time;
|
|
|
|
uint32 recovery_xid;
|
|
|
|
|
2014-01-10 04:11:27 +09:00
|
|
|
/* Different sizes (-1 means nothing was backed up) */
|
|
|
|
/*
|
|
|
|
* Amount of raw data. For a full backup, this is the total amount of
|
2014-01-30 15:58:55 +09:00
|
|
|
* data while for a differential backup this is just the difference
|
2014-01-10 04:11:27 +09:00
|
|
|
* of data taken.
|
|
|
|
*/
|
|
|
|
int64 data_bytes;
|
2013-01-24 06:35:48 +00:00
|
|
|
|
|
|
|
/* data/wal block size for compatibility check */
|
|
|
|
uint32 block_size;
|
|
|
|
uint32 wal_block_size;
|
2016-07-07 18:38:56 +03:00
|
|
|
uint32 checksum_version;
|
2016-11-01 17:25:46 +03:00
|
|
|
bool stream;
|
2013-01-24 06:35:48 +00:00
|
|
|
} pgBackup;
|
|
|
|
|
2013-09-09 09:00:13 +00:00
|
|
|
typedef struct pgBackupOption
|
|
|
|
{
|
|
|
|
bool smooth_checkpoint;
|
|
|
|
int keep_data_generations;
|
|
|
|
int keep_data_days;
|
|
|
|
} pgBackupOption;
|
|
|
|
|
|
|
|
|
2013-01-24 06:35:48 +00:00
|
|
|
/* special values of pgBackup */
|
|
|
|
#define KEEP_INFINITE (INT_MAX)
|
|
|
|
#define BYTES_INVALID (-1)
|
|
|
|
|
|
|
|
typedef struct pgTimeLine
|
|
|
|
{
|
|
|
|
TimeLineID tli;
|
|
|
|
XLogRecPtr end;
|
|
|
|
} pgTimeLine;
|
|
|
|
|
|
|
|
typedef struct pgRecoveryTarget
|
|
|
|
{
|
|
|
|
bool time_specified;
|
|
|
|
time_t recovery_target_time;
|
|
|
|
bool xid_specified;
|
|
|
|
unsigned int recovery_target_xid;
|
|
|
|
bool recovery_target_inclusive;
|
|
|
|
} pgRecoveryTarget;
|
|
|
|
|
2016-02-27 21:07:55 +03:00
|
|
|
typedef union DataPage
|
|
|
|
{
|
|
|
|
PageHeaderData page_data;
|
|
|
|
char data[BLCKSZ];
|
|
|
|
} DataPage;
|
2013-01-24 06:35:48 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* return pointer that exceeds the length of prefix from character string.
|
|
|
|
* ex. str="/xxx/yyy/zzz", prefix="/xxx/yyy", return="zzz".
|
|
|
|
*/
|
|
|
|
#define JoinPathEnd(str, prefix) \
|
|
|
|
((strlen(str) <= strlen(prefix)) ? "" : str + strlen(prefix) + 1)
|
|
|
|
|
2013-12-13 03:55:39 +09:00
|
|
|
/*
|
|
|
|
* Return timeline, xlog ID and record offset from an LSN of the type
|
|
|
|
* 0/B000188, usual result from pg_stop_backup() and friends.
|
|
|
|
*/
|
|
|
|
#define XLogDataFromLSN(data, xlogid, xrecoff) \
|
|
|
|
sscanf(data, "%X/%X", xlogid, xrecoff)
|
|
|
|
|
2013-01-24 06:35:48 +00:00
|
|
|
/* path configuration */
|
|
|
|
extern char *backup_path;
|
|
|
|
extern char *pgdata;
|
2016-10-18 20:22:53 +03:00
|
|
|
extern char arclog_path[MAXPGPATH];
|
2013-01-24 06:35:48 +00:00
|
|
|
|
|
|
|
/* common configuration */
|
|
|
|
extern bool check;
|
|
|
|
|
|
|
|
/* current settings */
|
|
|
|
extern pgBackup current;
|
|
|
|
|
|
|
|
/* exclude directory list for $PGDATA file listing */
|
|
|
|
extern const char *pgdata_exclude[];
|
|
|
|
|
2016-01-15 23:47:38 +09:00
|
|
|
/* backup file list from non-snapshot */
|
|
|
|
extern parray *backup_files_list;
|
|
|
|
|
2016-02-29 20:23:48 +03:00
|
|
|
extern int num_threads;
|
2016-05-26 15:56:32 +03:00
|
|
|
extern bool stream_wal;
|
2016-09-29 17:33:21 +03:00
|
|
|
extern bool from_replica;
|
2016-09-02 20:16:00 +03:00
|
|
|
extern bool progress;
|
2016-02-29 20:23:48 +03:00
|
|
|
|
2013-01-24 06:35:48 +00:00
|
|
|
/* in backup.c */
|
2013-09-09 09:00:13 +00:00
|
|
|
extern int do_backup(pgBackupOption bkupopt);
|
2013-12-16 00:50:36 +09:00
|
|
|
extern BackupMode parse_backup_mode(const char *value);
|
2013-12-12 22:20:08 +09:00
|
|
|
extern void check_server_version(void);
|
2013-09-09 09:00:13 +00:00
|
|
|
extern bool fileExists(const char *path);
|
2016-01-15 23:47:38 +09:00
|
|
|
extern void process_block_change(ForkNumber forknum, RelFileNode rnode,
|
|
|
|
BlockNumber blkno);
|
2013-01-24 06:35:48 +00:00
|
|
|
|
|
|
|
/* in restore.c */
|
|
|
|
extern int do_restore(const char *target_time,
|
|
|
|
const char *target_xid,
|
|
|
|
const char *target_inclusive,
|
2014-01-24 22:36:31 +09:00
|
|
|
TimeLineID target_tli);
|
2013-01-24 06:35:48 +00:00
|
|
|
|
|
|
|
/* in init.c */
|
|
|
|
extern int do_init(void);
|
|
|
|
|
|
|
|
/* in show.c */
|
2016-10-31 18:19:11 +03:00
|
|
|
extern int do_show(time_t backup_id, bool show_all);
|
2013-01-24 06:35:48 +00:00
|
|
|
|
|
|
|
/* in delete.c */
|
2016-10-31 18:19:11 +03:00
|
|
|
extern int do_delete(time_t backup_id);
|
2013-01-24 06:35:48 +00:00
|
|
|
extern void pgBackupDelete(int keep_generations, int keep_days);
|
|
|
|
|
2013-12-13 03:55:39 +09:00
|
|
|
/* in fetch.c */
|
|
|
|
extern char *slurpFile(const char *datadir,
|
|
|
|
const char *path,
|
2016-03-02 20:50:33 +03:00
|
|
|
size_t *filesize,
|
|
|
|
bool safe);
|
2013-12-13 03:55:39 +09:00
|
|
|
|
2013-01-24 06:35:48 +00:00
|
|
|
/* in validate.c */
|
2016-10-31 18:19:11 +03:00
|
|
|
extern int do_validate(time_t backup_id);
|
2013-12-16 00:30:49 +09:00
|
|
|
extern void pgBackupValidate(pgBackup *backup,
|
|
|
|
bool size_only,
|
|
|
|
bool for_get_timeline);
|
2013-01-24 06:35:48 +00:00
|
|
|
|
|
|
|
/* in catalog.c */
|
|
|
|
extern pgBackup *catalog_get_backup(time_t timestamp);
|
2016-10-31 18:19:11 +03:00
|
|
|
extern parray *catalog_get_backup_list(time_t backup_id);
|
2013-12-25 05:27:25 +09:00
|
|
|
extern pgBackup *catalog_get_last_data_backup(parray *backup_list,
|
|
|
|
TimeLineID tli);
|
2013-01-24 06:35:48 +00:00
|
|
|
|
|
|
|
extern int catalog_lock(void);
|
|
|
|
extern void catalog_unlock(void);
|
|
|
|
|
|
|
|
extern void catalog_init_config(pgBackup *backup);
|
|
|
|
|
|
|
|
extern void pgBackupWriteConfigSection(FILE *out, pgBackup *backup);
|
|
|
|
extern void pgBackupWriteResultSection(FILE *out, pgBackup *backup);
|
|
|
|
extern void pgBackupWriteIni(pgBackup *backup);
|
|
|
|
extern void pgBackupGetPath(const pgBackup *backup, char *path, size_t len, const char *subdir);
|
|
|
|
extern int pgBackupCreateDir(pgBackup *backup);
|
|
|
|
extern void pgBackupFree(void *backup);
|
|
|
|
extern int pgBackupCompareId(const void *f1, const void *f2);
|
|
|
|
extern int pgBackupCompareIdDesc(const void *f1, const void *f2);
|
|
|
|
|
|
|
|
/* in dir.c */
|
|
|
|
extern void dir_list_file(parray *files, const char *root, const char *exclude[], bool omit_symlink, bool add_root);
|
2013-09-09 09:00:13 +00:00
|
|
|
extern void dir_list_file_internal(parray *files, const char *root, const char *exclude[],
|
|
|
|
bool omit_symlink, bool add_root, parray *black_list);
|
2013-01-24 06:35:48 +00:00
|
|
|
extern void dir_print_mkdirs_sh(FILE *out, const parray *files, const char *root);
|
|
|
|
extern void dir_print_file_list(FILE *out, const parray *files, const char *root, const char *prefix);
|
|
|
|
extern parray *dir_read_file_list(const char *root, const char *file_txt);
|
|
|
|
|
|
|
|
extern int dir_create_dir(const char *path, mode_t mode);
|
|
|
|
extern void dir_copy_files(const char *from_root, const char *to_root);
|
|
|
|
|
2016-09-29 19:44:59 +03:00
|
|
|
extern pgFile *pgFileNew(const char *path, bool omit_symlink);
|
2013-01-24 06:35:48 +00:00
|
|
|
extern void pgFileDelete(pgFile *file);
|
|
|
|
extern void pgFileFree(void *file);
|
|
|
|
extern pg_crc32 pgFileGetCRC(pgFile *file);
|
|
|
|
extern int pgFileComparePath(const void *f1, const void *f2);
|
|
|
|
extern int pgFileComparePathDesc(const void *f1, const void *f2);
|
2016-09-02 19:31:49 +03:00
|
|
|
extern int pgFileCompareSize(const void *f1, const void *f2);
|
2013-01-24 06:35:48 +00:00
|
|
|
extern int pgFileCompareMtime(const void *f1, const void *f2);
|
|
|
|
extern int pgFileCompareMtimeDesc(const void *f1, const void *f2);
|
|
|
|
|
|
|
|
/* in data.c */
|
|
|
|
extern bool backup_data_file(const char *from_root, const char *to_root,
|
2016-01-14 15:19:38 +09:00
|
|
|
pgFile *file, const XLogRecPtr *lsn);
|
2013-01-24 06:35:48 +00:00
|
|
|
extern void restore_data_file(const char *from_root, const char *to_root,
|
2016-07-07 18:38:56 +03:00
|
|
|
pgFile *file, pgBackup *backup);
|
2013-01-24 06:35:48 +00:00
|
|
|
extern bool copy_file(const char *from_root, const char *to_root,
|
2016-01-14 15:19:38 +09:00
|
|
|
pgFile *file);
|
2013-01-24 06:35:48 +00:00
|
|
|
|
2016-05-26 15:56:32 +03:00
|
|
|
extern bool calc_file(pgFile *file);
|
|
|
|
|
2016-01-15 23:47:38 +09:00
|
|
|
/* parsexlog.c */
|
|
|
|
extern void extractPageMap(const char *datadir, XLogRecPtr startpoint,
|
|
|
|
TimeLineID tli, XLogRecPtr endpoint);
|
|
|
|
|
2013-01-24 06:35:48 +00:00
|
|
|
/* in util.c */
|
2016-03-02 20:50:33 +03:00
|
|
|
extern TimeLineID get_current_timeline(bool safe);
|
2016-01-15 23:47:38 +09:00
|
|
|
extern void sanityChecks(void);
|
2013-01-24 06:35:48 +00:00
|
|
|
extern void time2iso(char *buf, size_t len, time_t time);
|
|
|
|
extern const char *status2str(BackupStatus status);
|
|
|
|
extern void remove_trailing_space(char *buf, int comment_mark);
|
|
|
|
extern void remove_not_digit(char *buf, size_t len, const char *str);
|
2016-03-10 17:40:33 +03:00
|
|
|
extern XLogRecPtr get_last_ptrack_lsn(void);
|
2016-07-07 18:38:56 +03:00
|
|
|
extern uint32 get_data_checksum_version(bool safe);
|
2016-10-31 18:19:11 +03:00
|
|
|
extern char *base36enc(long unsigned int value);
|
|
|
|
extern long unsigned int base36dec(const char *text);
|
2013-01-24 06:35:48 +00:00
|
|
|
|
2013-12-16 00:36:50 +09:00
|
|
|
/* in status.c */
|
2013-01-24 06:35:48 +00:00
|
|
|
extern bool is_pg_running(void);
|
|
|
|
|
|
|
|
#endif /* PG_RMAN_H */
|