2013-01-24 06:35:48 +00:00
|
|
|
/*-------------------------------------------------------------------------
|
|
|
|
*
|
2013-12-12 23:54:52 +09:00
|
|
|
* status.c
|
2013-01-24 06:35:48 +00:00
|
|
|
*
|
2017-02-12 23:42:10 +03:00
|
|
|
* Portions Copyright (c) 1996-2017, PostgreSQL Global Development Group
|
2013-01-24 06:35:48 +00:00
|
|
|
*
|
2013-12-12 23:54:52 +09:00
|
|
|
* Monitor status of a PostgreSQL server.
|
2013-01-24 06:35:48 +00:00
|
|
|
*
|
|
|
|
*-------------------------------------------------------------------------
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
|
|
#include "postgres_fe.h"
|
|
|
|
|
|
|
|
#include <signal.h>
|
|
|
|
#include <sys/types.h>
|
|
|
|
#include <unistd.h>
|
|
|
|
|
2016-11-16 20:34:21 +03:00
|
|
|
#include "pg_probackup.h"
|
2013-01-24 06:35:48 +00:00
|
|
|
|
|
|
|
/* PID can be negative for standalone backend */
|
|
|
|
typedef long pgpid_t;
|
|
|
|
|
|
|
|
static pgpid_t get_pgpid(void);
|
|
|
|
static bool postmaster_is_alive(pid_t pid);
|
|
|
|
|
2013-12-12 23:54:52 +09:00
|
|
|
/*
|
|
|
|
* get_pgpid
|
|
|
|
*
|
|
|
|
* Get PID of postmaster, by scanning postmaster.pid.
|
|
|
|
*/
|
2013-01-24 06:35:48 +00:00
|
|
|
static pgpid_t
|
|
|
|
get_pgpid(void)
|
|
|
|
{
|
|
|
|
FILE *pidf;
|
|
|
|
long pid;
|
2013-12-12 23:54:52 +09:00
|
|
|
char pid_file[MAXPGPATH];
|
2013-01-24 06:35:48 +00:00
|
|
|
|
|
|
|
snprintf(pid_file, lengthof(pid_file), "%s/postmaster.pid", pgdata);
|
2013-12-12 23:54:52 +09:00
|
|
|
|
2013-01-24 06:35:48 +00:00
|
|
|
pidf = fopen(pid_file, "r");
|
|
|
|
if (pidf == NULL)
|
|
|
|
{
|
|
|
|
/* No pid file, not an error on startup */
|
|
|
|
if (errno == ENOENT)
|
|
|
|
return 0;
|
|
|
|
else
|
2013-12-12 23:54:52 +09:00
|
|
|
{
|
2016-01-19 12:41:30 +09:00
|
|
|
elog(ERROR, "could not open PID file \"%s\": %s",
|
2013-12-12 23:54:52 +09:00
|
|
|
pid_file, strerror(errno));
|
|
|
|
}
|
2013-12-13 00:29:25 +09:00
|
|
|
}
|
|
|
|
if (fscanf(pidf, "%ld", &pid) != 1)
|
|
|
|
{
|
|
|
|
/* Is the file empty? */
|
|
|
|
if (ftell(pidf) == 0 && feof(pidf))
|
2016-01-19 12:41:30 +09:00
|
|
|
elog(ERROR, "the PID file \"%s\" is empty",
|
2013-12-13 00:29:25 +09:00
|
|
|
pid_file);
|
|
|
|
else
|
2016-01-19 12:41:30 +09:00
|
|
|
elog(ERROR, "invalid data in PID file \"%s\"\n",
|
2013-12-13 00:29:25 +09:00
|
|
|
pid_file);
|
2013-01-24 06:35:48 +00:00
|
|
|
}
|
|
|
|
fclose(pidf);
|
|
|
|
return (pgpid_t) pid;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
2013-12-12 23:54:52 +09:00
|
|
|
* postmaster_is_alive
|
|
|
|
*
|
|
|
|
* Check whether postmaster is alive or not.
|
2013-01-24 06:35:48 +00:00
|
|
|
*/
|
|
|
|
static bool
|
|
|
|
postmaster_is_alive(pid_t pid)
|
|
|
|
{
|
|
|
|
/*
|
|
|
|
* Test to see if the process is still there. Note that we do not
|
|
|
|
* consider an EPERM failure to mean that the process is still there;
|
|
|
|
* EPERM must mean that the given PID belongs to some other userid, and
|
|
|
|
* considering the permissions on $PGDATA, that means it's not the
|
|
|
|
* postmaster we are after.
|
|
|
|
*
|
|
|
|
* Don't believe that our own PID or parent shell's PID is the postmaster,
|
|
|
|
* either. (Windows hasn't got getppid(), though.)
|
|
|
|
*/
|
|
|
|
if (pid == getpid())
|
|
|
|
return false;
|
|
|
|
#ifndef WIN32
|
|
|
|
if (pid == getppid())
|
|
|
|
return false;
|
|
|
|
#endif
|
|
|
|
if (kill(pid, 0) == 0)
|
|
|
|
return true;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
2013-12-12 23:54:52 +09:00
|
|
|
* is_pg_running
|
|
|
|
*
|
|
|
|
*
|
2013-01-24 06:35:48 +00:00
|
|
|
*/
|
|
|
|
bool
|
|
|
|
is_pg_running(void)
|
|
|
|
{
|
|
|
|
pgpid_t pid;
|
|
|
|
|
|
|
|
pid = get_pgpid();
|
2013-12-12 23:54:52 +09:00
|
|
|
|
|
|
|
/* 0 means no pid file */
|
|
|
|
if (pid == 0)
|
2013-01-24 06:35:48 +00:00
|
|
|
return false;
|
|
|
|
|
2013-12-12 23:54:52 +09:00
|
|
|
/* Case of a standalone backend */
|
|
|
|
if (pid < 0)
|
2013-01-24 06:35:48 +00:00
|
|
|
pid = -pid;
|
|
|
|
|
2013-12-12 23:54:52 +09:00
|
|
|
/* Check if postmaster is alive */
|
2013-01-24 06:35:48 +00:00
|
|
|
return postmaster_is_alive((pid_t) pid);
|
|
|
|
}
|