2013-01-24 09:35:48 +03:00
|
|
|
/*-------------------------------------------------------------------------
|
|
|
|
*
|
|
|
|
* util.c: log messages to log file or stderr, and misc code.
|
|
|
|
*
|
|
|
|
* Copyright (c) 2009-2011, NIPPON TELEGRAPH AND TELEPHONE CORPORATION
|
|
|
|
*
|
|
|
|
*-------------------------------------------------------------------------
|
|
|
|
*/
|
|
|
|
|
2014-01-27 06:02:56 +03:00
|
|
|
#include "pg_arman.h"
|
2013-01-24 09:35:48 +03:00
|
|
|
|
|
|
|
#include <time.h>
|
|
|
|
|
2016-01-15 16:47:38 +02:00
|
|
|
#include "storage/bufpage.h"
|
2016-01-15 07:09:31 +02:00
|
|
|
|
|
|
|
static void
|
|
|
|
checkControlFile(ControlFileData *ControlFile)
|
|
|
|
{
|
|
|
|
pg_crc32c crc;
|
|
|
|
|
|
|
|
/* Calculate CRC */
|
|
|
|
INIT_CRC32C(crc);
|
|
|
|
COMP_CRC32C(crc, (char *) ControlFile, offsetof(ControlFileData, crc));
|
|
|
|
FIN_CRC32C(crc);
|
|
|
|
|
|
|
|
/* Then compare it */
|
|
|
|
if (!EQ_CRC32C(crc, ControlFile->crc))
|
2016-01-19 05:41:30 +02:00
|
|
|
elog(ERROR, "Calculated CRC checksum does not match value stored in file.\n"
|
2016-01-15 07:09:31 +02:00
|
|
|
"Either the file is corrupt, or it has a different layout than this program\n"
|
2016-01-15 16:47:38 +02:00
|
|
|
"is expecting. The results below are untrustworthy.");
|
2016-01-15 07:09:31 +02:00
|
|
|
|
|
|
|
if (ControlFile->pg_control_version % 65536 == 0 && ControlFile->pg_control_version / 65536 != 0)
|
2016-01-19 05:41:30 +02:00
|
|
|
elog(ERROR, "possible byte ordering mismatch\n"
|
2016-01-15 07:09:31 +02:00
|
|
|
"The byte ordering used to store the pg_control file might not match the one\n"
|
|
|
|
"used by this program. In that case the results below would be incorrect, and\n"
|
2016-01-15 16:47:38 +02:00
|
|
|
"the PostgreSQL installation would be incompatible with this data directory.");
|
2016-01-15 07:09:31 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Verify control file contents in the buffer src, and copy it to *ControlFile.
|
|
|
|
*/
|
|
|
|
static void
|
|
|
|
digestControlFile(ControlFileData *ControlFile, char *src, size_t size)
|
|
|
|
{
|
|
|
|
if (size != PG_CONTROL_SIZE)
|
2016-01-19 05:41:30 +02:00
|
|
|
elog(ERROR, "unexpected control file size %d, expected %d",
|
2016-01-15 07:09:31 +02:00
|
|
|
(int) size, PG_CONTROL_SIZE);
|
|
|
|
|
|
|
|
memcpy(ControlFile, src, sizeof(ControlFileData));
|
|
|
|
|
|
|
|
/* Additional checks on control file */
|
|
|
|
checkControlFile(ControlFile);
|
|
|
|
}
|
|
|
|
|
2016-01-15 16:47:38 +02:00
|
|
|
void
|
|
|
|
sanityChecks(void)
|
|
|
|
{
|
|
|
|
ControlFileData ControlFile;
|
|
|
|
char *buffer;
|
|
|
|
size_t size;
|
|
|
|
|
|
|
|
/* First fetch file... */
|
2016-03-02 19:50:33 +02:00
|
|
|
buffer = slurpFile(pgdata, "global/pg_control", &size, false);
|
2016-01-15 16:47:38 +02:00
|
|
|
digestControlFile(&ControlFile, buffer, size);
|
|
|
|
pg_free(buffer);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Node work is done on need to use checksums or hint bit wal-logging
|
|
|
|
* this to prevent from data corruption that could occur because of
|
|
|
|
* hint bits.
|
|
|
|
*/
|
|
|
|
if (ControlFile.data_checksum_version != PG_DATA_CHECKSUM_VERSION &&
|
|
|
|
!ControlFile.wal_log_hints)
|
2016-01-19 05:41:30 +02:00
|
|
|
elog(ERROR,
|
2016-01-15 16:47:38 +02:00
|
|
|
"target master need to use either data checksums or \"wal_log_hints = on\".");
|
|
|
|
}
|
|
|
|
|
2016-03-10 16:40:33 +02:00
|
|
|
XLogRecPtr
|
|
|
|
get_last_ptrack_lsn(void)
|
|
|
|
{
|
|
|
|
char *buffer;
|
|
|
|
size_t size;
|
|
|
|
XLogRecPtr lsn;
|
|
|
|
|
|
|
|
buffer = slurpFile(pgdata, "global/ptrack_control", &size, false);
|
|
|
|
if (buffer == NULL)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
lsn = *(XLogRecPtr *)buffer;
|
|
|
|
return lsn;
|
|
|
|
}
|
|
|
|
|
2016-01-15 07:09:31 +02:00
|
|
|
/*
|
|
|
|
* Utility shared by backup and restore to fetch the current timeline
|
|
|
|
* used by a node.
|
|
|
|
*/
|
|
|
|
TimeLineID
|
2016-03-02 19:50:33 +02:00
|
|
|
get_current_timeline(bool safe)
|
2016-01-15 07:09:31 +02:00
|
|
|
{
|
|
|
|
ControlFileData ControlFile;
|
|
|
|
char *buffer;
|
|
|
|
size_t size;
|
|
|
|
|
|
|
|
/* First fetch file... */
|
2016-03-02 19:50:33 +02:00
|
|
|
buffer = slurpFile(pgdata, "global/pg_control", &size, safe);
|
|
|
|
if (buffer == NULL)
|
|
|
|
return 0;
|
2016-01-15 07:09:31 +02:00
|
|
|
digestControlFile(&ControlFile, buffer, size);
|
|
|
|
pg_free(buffer);
|
|
|
|
|
|
|
|
return ControlFile.checkPointCopy.ThisTimeLineID;
|
|
|
|
}
|
|
|
|
|
2013-01-24 09:35:48 +03:00
|
|
|
/*
|
|
|
|
* Convert time_t value to ISO-8601 format string
|
|
|
|
*/
|
|
|
|
void
|
|
|
|
time2iso(char *buf, size_t len, time_t time)
|
|
|
|
{
|
|
|
|
struct tm *tm = localtime(&time);
|
|
|
|
|
|
|
|
strftime(buf, len, "%Y-%m-%d %H:%M:%S", tm);
|
|
|
|
}
|
|
|
|
|
|
|
|
const char *
|
|
|
|
status2str(BackupStatus status)
|
|
|
|
{
|
|
|
|
static const char *statusName[] =
|
|
|
|
{
|
|
|
|
"UNKNOWN",
|
|
|
|
"OK",
|
|
|
|
"RUNNING",
|
|
|
|
"ERROR",
|
|
|
|
"DELETING",
|
|
|
|
"DELETED",
|
|
|
|
"DONE",
|
|
|
|
"CORRUPT"
|
|
|
|
};
|
|
|
|
|
|
|
|
if (status < BACKUP_STATUS_INVALID || BACKUP_STATUS_CORRUPT < status)
|
|
|
|
return "UNKNOWN";
|
|
|
|
|
|
|
|
return statusName[status];
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
remove_trailing_space(char *buf, int comment_mark)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
char *last_char = NULL;
|
|
|
|
|
|
|
|
for (i = 0; buf[i]; i++)
|
|
|
|
{
|
|
|
|
if (buf[i] == comment_mark || buf[i] == '\n' || buf[i] == '\r')
|
|
|
|
{
|
|
|
|
buf[i] = '\0';
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
for (i = 0; buf[i]; i++)
|
|
|
|
{
|
|
|
|
if (!isspace(buf[i]))
|
|
|
|
last_char = buf + i;
|
|
|
|
}
|
|
|
|
if (last_char != NULL)
|
|
|
|
*(last_char + 1) = '\0';
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
remove_not_digit(char *buf, size_t len, const char *str)
|
|
|
|
{
|
|
|
|
int i, j;
|
|
|
|
|
|
|
|
for (i = 0, j = 0; str[i] && j < len; i++)
|
|
|
|
{
|
|
|
|
if (!isdigit(str[i]))
|
|
|
|
continue;
|
|
|
|
buf[j++] = str[i];
|
|
|
|
}
|
|
|
|
buf[j] = '\0';
|
|
|
|
}
|