/*------------------------------------------------------------------------- * * pg_arman.c: Backup/Recovery manager for PostgreSQL. * * Copyright (c) 2009-2013, NIPPON TELEGRAPH AND TELEPHONE CORPORATION * *------------------------------------------------------------------------- */ #include "pg_arman.h" #include "streamutil.h" #include #include #include #include const char *PROGRAM_VERSION = "0.1"; const char *PROGRAM_URL = "https://github.com/stalkerg/pg_arman"; const char *PROGRAM_EMAIL = "https://github.com/stalkerg/pg_arman/issues"; /* path configuration */ char *backup_path; char *pgdata; char arclog_path[MAXPGPATH]; /* common configuration */ bool check = false; /* directory configuration */ pgBackup current; /* backup configuration */ static bool smooth_checkpoint; static int keep_data_generations = KEEP_INFINITE; static int keep_data_days = KEEP_INFINITE; int num_threads = 1; bool stream_wal = false; bool from_replica = false; static bool backup_logs = false; bool progress = false; /* restore configuration */ static char *target_time; static char *target_xid; static char *target_inclusive; static TimeLineID target_tli; /* show configuration */ static bool show_all = false; static void opt_backup_mode(pgut_option *opt, const char *arg); static pgut_option options[] = { /* directory options */ { 's', 'D', "pgdata", &pgdata, SOURCE_ENV }, { 's', 'B', "backup-path", &backup_path, SOURCE_ENV }, /* common options */ { 'b', 'c', "check", &check }, { 'i', 'j', "threads", &num_threads }, { 'b', 8, "stream", &stream_wal }, { 'b', 11, "progress", &progress }, /* backup options */ { 'b', 10, "backup-pg-log", &backup_logs }, { 'f', 'b', "backup-mode", opt_backup_mode, SOURCE_ENV }, { 'b', 'C', "smooth-checkpoint", &smooth_checkpoint, SOURCE_ENV }, { 's', 'S', "slot", &replication_slot, SOURCE_ENV }, /* options with only long name (keep-xxx) */ { 'i', 1, "keep-data-generations", &keep_data_generations, SOURCE_ENV }, { 'i', 2, "keep-data-days", &keep_data_days, SOURCE_ENV }, /* restore options */ { 's', 3, "recovery-target-time", &target_time, SOURCE_ENV }, { 's', 4, "recovery-target-xid", &target_xid, SOURCE_ENV }, { 's', 5, "recovery-target-inclusive", &target_inclusive, SOURCE_ENV }, { 'u', 6, "recovery-target-timeline", &target_tli, SOURCE_ENV }, /* catalog options */ { 'b', 'a', "show-all", &show_all }, { 0 } }; /* * Entry point of pg_arman command. */ int main(int argc, char *argv[]) { const char *cmd = NULL; const char *backup_id_string = NULL; time_t backup_id = 0; int i; /* do not buffer progress messages */ setvbuf(stdout, 0, _IONBF, 0); /* TODO: remove this */ /* initialize configuration */ catalog_init_config(¤t); /* overwrite configuration with command line arguments */ i = pgut_getopt(argc, argv, options); for (; i < argc; i++) { if (cmd == NULL) { cmd = argv[i]; if(strcmp(cmd, "show") != 0 && strcmp(cmd, "validate") != 0 && strcmp(cmd, "delete") != 0) break; } else if (backup_id_string == NULL) backup_id_string = argv[i]; else elog(ERROR, "too many arguments"); } /* command argument (backup/restore/show/...) is required. */ if (cmd == NULL) { help(false); return 1; } if (backup_id_string != NULL) backup_id = base36dec(backup_id_string); /* Read default configuration from file. */ if (backup_path) { char path[MAXPGPATH]; /* Check if backup_path is directory. */ struct stat stat_buf; int rc = stat(backup_path, &stat_buf); /* If rc == -1, there is no file or directory. So it's OK. */ if (rc != -1 && !S_ISDIR(stat_buf.st_mode)) elog(ERROR, "-B, --backup-path must be a path to directory"); join_path_components(path, backup_path, PG_RMAN_INI_FILE); pgut_readopt(path, options, ERROR); /* setup stream options */ if (pgut_dbname != NULL) dbname = pstrdup(pgut_dbname); if (host != NULL) dbhost = pstrdup(host); if (port != NULL) dbport = pstrdup(port); if (username != NULL) dbuser = pstrdup(username); } /* BACKUP_PATH is always required */ if (backup_path == NULL) elog(ERROR, "required parameter not specified: BACKUP_PATH (-B, --backup-path)"); /* path must be absolute */ if (backup_path != NULL && !is_absolute_path(backup_path)) elog(ERROR, "-B, --backup-path must be an absolute path"); if (pgdata != NULL && !is_absolute_path(pgdata)) elog(ERROR, "-D, --pgdata must be an absolute path"); join_path_components(arclog_path, backup_path, "wal"); /* setup exclusion list for file search */ for (i = 0; pgdata_exclude[i]; i++); /* find first empty slot */ pgdata_exclude[i++] = arclog_path; if(!backup_logs) pgdata_exclude[i++] = "pg_log"; /* do actual operation */ if (pg_strcasecmp(cmd, "init") == 0) return do_init(); else if (pg_strcasecmp(cmd, "backup") == 0) { pgBackupOption bkupopt; int res; bkupopt.smooth_checkpoint = smooth_checkpoint; bkupopt.keep_data_generations = keep_data_generations; bkupopt.keep_data_days = keep_data_days; /* Do the backup */ res = do_backup(bkupopt); if (res != 0) return res; do_validate(current.start_time); } else if (pg_strcasecmp(cmd, "restore") == 0) return do_restore(target_time, target_xid, target_inclusive, target_tli); else if (pg_strcasecmp(cmd, "show") == 0) return do_show(backup_id, show_all); else if (pg_strcasecmp(cmd, "validate") == 0) return do_validate(backup_id); else if (pg_strcasecmp(cmd, "delete") == 0) return do_delete(backup_id); else elog(ERROR, "invalid command \"%s\"", cmd); return 0; } void pgut_help(bool details) { printf(_("%s manage backup/recovery of PostgreSQL database.\n\n"), PROGRAM_NAME); printf(_("Usage:\n")); printf(_(" %s OPTION init\n"), PROGRAM_NAME); printf(_(" %s OPTION backup\n"), PROGRAM_NAME); printf(_(" %s OPTION restore\n"), PROGRAM_NAME); printf(_(" %s OPTION show [DATE] [DATE]\n"), PROGRAM_NAME); printf(_(" %s OPTION validate [DATE] [DATE]\n"), PROGRAM_NAME); printf(_(" %s OPTION delete DATE\n"), PROGRAM_NAME); if (!details) return; printf(_("\nCommon Options:\n")); printf(_(" -D, --pgdata=PATH location of the database storage area\n")); printf(_(" -B, --backup-path=PATH location of the backup storage area\n")); printf(_(" -c, --check show what would have been done\n")); printf(_(" -j, --threads=NUM num threads for backup and restore\n")); printf(_(" --stream use stream for save/restore WAL during backup\n")); printf(_(" --progress show progress copy files\n")); printf(_("\nBackup options:\n")); printf(_(" -b, --backup-mode=MODE full,page,ptrack\n")); printf(_(" -C, --smooth-checkpoint do smooth checkpoint before backup\n")); printf(_(" --keep-data-generations=N keep GENERATION of full data backup\n")); printf(_(" --keep-data-days=DAY keep enough data backup to recover to DAY days age\n")); printf(_(" --backup-pg-log start backup pg_log directory\n")); printf(_(" -S, --slot=SLOTNAME replication slot to use\n")); printf(_("\nRestore options:\n")); printf(_(" --recovery-target-time time stamp up to which recovery will proceed\n")); printf(_(" --recovery-target-xid transaction ID up to which recovery will proceed\n")); printf(_(" --recovery-target-inclusive whether we stop just after the recovery target\n")); printf(_(" --recovery-target-timeline recovering into a particular timeline\n")); printf(_("\nCatalog options:\n")); printf(_(" -a, --show-all show deleted backup too\n")); } static void opt_backup_mode(pgut_option *opt, const char *arg) { current.backup_mode = parse_backup_mode(arg); }