mirror of
https://github.com/postgrespro/pg_probackup.git
synced 2024-11-25 09:01:48 +02:00
217 lines
4.8 KiB
C
217 lines
4.8 KiB
C
/*-------------------------------------------------------------------------
|
|
*
|
|
* show.c: show backup catalog.
|
|
*
|
|
* Copyright (c) 2009-2011, NIPPON TELEGRAPH AND TELEPHONE CORPORATION
|
|
*
|
|
*-------------------------------------------------------------------------
|
|
*/
|
|
|
|
#include "pg_probackup.h"
|
|
|
|
static void show_backup_list(FILE *out, parray *backup_list);
|
|
static void show_backup_detail(FILE *out, pgBackup *backup);
|
|
|
|
/*
|
|
* Show backup catalog information.
|
|
* If range is { 0, 0 }, show list of all backup, otherwise show detail of the
|
|
* backup indicated by id.
|
|
*/
|
|
int
|
|
do_show(time_t backup_id)
|
|
{
|
|
/*
|
|
* Safety check for archive folder, this is necessary to fetch
|
|
* the parent TLI from history field generated by server after
|
|
* child timeline is chosen.
|
|
*/
|
|
if (backup_id != 0)
|
|
{
|
|
pgBackup *backup;
|
|
|
|
backup = catalog_get_backup(backup_id);
|
|
if (backup == NULL)
|
|
{
|
|
char timestamp[100];
|
|
time2iso(timestamp, lengthof(timestamp), backup_id);
|
|
elog(INFO, "backup taken at \"%s\" does not exist.",
|
|
timestamp);
|
|
/* This is not error case */
|
|
return 0;
|
|
}
|
|
show_backup_detail(stdout, backup);
|
|
|
|
/* cleanup */
|
|
pgBackupFree(backup);
|
|
}
|
|
else
|
|
{
|
|
parray *backup_list;
|
|
|
|
backup_list = catalog_get_backup_list(backup_id);
|
|
if (backup_list == NULL)
|
|
elog(ERROR, "can't process any more.");
|
|
|
|
show_backup_list(stdout, backup_list);
|
|
|
|
/* cleanup */
|
|
parray_walk(backup_list, pgBackupFree);
|
|
parray_free(backup_list);
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static void
|
|
pretty_size(int64 size, char *buf, size_t len)
|
|
{
|
|
int exp = 0;
|
|
|
|
/* minus means the size is invalid */
|
|
if (size < 0)
|
|
{
|
|
strncpy(buf, "----", len);
|
|
return;
|
|
}
|
|
|
|
/* determine postfix */
|
|
while (size > 9999)
|
|
{
|
|
++exp;
|
|
size /= 1000;
|
|
}
|
|
|
|
switch (exp)
|
|
{
|
|
case 0:
|
|
snprintf(buf, len, INT64_FORMAT "B", size);
|
|
break;
|
|
case 1:
|
|
snprintf(buf, len, INT64_FORMAT "kB", size);
|
|
break;
|
|
case 2:
|
|
snprintf(buf, len, INT64_FORMAT "MB", size);
|
|
break;
|
|
case 3:
|
|
snprintf(buf, len, INT64_FORMAT "GB", size);
|
|
break;
|
|
case 4:
|
|
snprintf(buf, len, INT64_FORMAT "TB", size);
|
|
break;
|
|
case 5:
|
|
snprintf(buf, len, INT64_FORMAT "PB", size);
|
|
break;
|
|
default:
|
|
strncpy(buf, "***", len);
|
|
break;
|
|
}
|
|
}
|
|
|
|
static TimeLineID
|
|
get_parent_tli(TimeLineID child_tli)
|
|
{
|
|
TimeLineID result = 0;
|
|
char path[MAXPGPATH];
|
|
char fline[MAXPGPATH];
|
|
FILE *fd;
|
|
|
|
/* Search history file in archives */
|
|
snprintf(path, lengthof(path), "%s/%08X.history", arclog_path,
|
|
child_tli);
|
|
fd = fopen(path, "rt");
|
|
if (fd == NULL)
|
|
{
|
|
if (errno != ENOENT)
|
|
elog(ERROR, "could not open file \"%s\": %s", path,
|
|
strerror(errno));
|
|
|
|
return 0;
|
|
}
|
|
|
|
/*
|
|
* Parse the file...
|
|
*/
|
|
while (fgets(fline, sizeof(fline), fd) != NULL)
|
|
{
|
|
/* skip leading whitespace and check for # comment */
|
|
char *ptr;
|
|
char *endptr;
|
|
|
|
for (ptr = fline; *ptr; ptr++)
|
|
{
|
|
if (!IsSpace(*ptr))
|
|
break;
|
|
}
|
|
if (*ptr == '\0' || *ptr == '#')
|
|
continue;
|
|
|
|
/* expect a numeric timeline ID as first field of line */
|
|
result = (TimeLineID) strtoul(ptr, &endptr, 0);
|
|
if (endptr == ptr)
|
|
elog(ERROR,
|
|
"syntax error(timeline ID) in history file: %s",
|
|
fline);
|
|
}
|
|
|
|
fclose(fd);
|
|
|
|
/* TLI of the last line is parent TLI */
|
|
return result;
|
|
}
|
|
|
|
static void
|
|
show_backup_list(FILE *out, parray *backup_list)
|
|
{
|
|
int i;
|
|
|
|
/* show header */
|
|
fputs("=========================================================================================\n", out);
|
|
fputs("ID Recovery time Mode Current/Parent TLI Time Data Status \n", out);
|
|
fputs("=========================================================================================\n", out);
|
|
|
|
for (i = 0; i < parray_num(backup_list); i++)
|
|
{
|
|
pgBackup *backup;
|
|
const char *modes[] = { "", "PAGE", "PTRACK", "FULL", "PAGE+STREAM", "PTRACK+STERAM", "FULL+STREAM"};
|
|
TimeLineID parent_tli;
|
|
char timestamp[20];
|
|
char duration[20] = "----";
|
|
char data_bytes_str[10] = "----";
|
|
|
|
backup = parray_get(backup_list, i);
|
|
|
|
time2iso(timestamp, lengthof(timestamp), backup->recovery_time);
|
|
if (backup->end_time != (time_t) 0)
|
|
snprintf(duration, lengthof(duration), "%lum",
|
|
(backup->end_time - backup->start_time) / 60);
|
|
|
|
/*
|
|
* Calculate Data field, in the case of full backup this shows the
|
|
* total amount of data. For an differential backup, this size is only
|
|
* the difference of data accumulated.
|
|
*/
|
|
pretty_size(backup->data_bytes, data_bytes_str,
|
|
lengthof(data_bytes_str));
|
|
|
|
/* Get parent timeline before printing */
|
|
parent_tli = get_parent_tli(backup->tli);
|
|
|
|
fprintf(out, "%-8s %-19s %-12s %2d /%2d %5s %6s %s \n",
|
|
base36enc(backup->start_time),
|
|
timestamp,
|
|
modes[backup->backup_mode*(backup->stream+1)],
|
|
backup->tli,
|
|
parent_tli,
|
|
duration,
|
|
data_bytes_str,
|
|
status2str(backup->status));
|
|
}
|
|
}
|
|
|
|
static void
|
|
show_backup_detail(FILE *out, pgBackup *backup)
|
|
{
|
|
pgBackupWriteConfigSection(out, backup);
|
|
pgBackupWriteResultSection(out, backup);
|
|
}
|