mirror of
https://github.com/EnterpriseDB/repmgr.git
synced 2026-03-26 00:26:30 +00:00
Get data directory for server commands if needed
Also add configuration file option "pgdata" for hard-coding the node's data directory - if the "repmgr" DB user isn't a superuser or doesn't have permission to extract the data directory, we'll need another way of finding out.
This commit is contained in:
@@ -210,6 +210,7 @@ _parse_config(t_configuration_options *options, ItemList *error_list, ItemList *
|
|||||||
memset(options->node_name, 0, sizeof(options->node_name));
|
memset(options->node_name, 0, sizeof(options->node_name));
|
||||||
memset(options->conninfo, 0, sizeof(options->conninfo));
|
memset(options->conninfo, 0, sizeof(options->conninfo));
|
||||||
memset(options->pg_bindir, 0, sizeof(options->pg_bindir));
|
memset(options->pg_bindir, 0, sizeof(options->pg_bindir));
|
||||||
|
memset(options->pgdata, 0, sizeof(options->pgdata));
|
||||||
options->replication_type = REPLICATION_TYPE_PHYSICAL;
|
options->replication_type = REPLICATION_TYPE_PHYSICAL;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
@@ -350,7 +351,10 @@ _parse_config(t_configuration_options *options, ItemList *error_list, ItemList *
|
|||||||
_( "value for \"replication_user\" must contain fewer than " STR(NAMEDATALEN) " characters"));
|
_( "value for \"replication_user\" must contain fewer than " STR(NAMEDATALEN) " characters"));
|
||||||
}
|
}
|
||||||
else if (strcmp(name, "pg_bindir") == 0)
|
else if (strcmp(name, "pg_bindir") == 0)
|
||||||
strncpy(options->pg_bindir, value, MAXLEN);
|
strncpy(options->pg_bindir, value, MAXPGPATH);
|
||||||
|
else if (strcmp(name, "pgdata") == 0)
|
||||||
|
strncpy(options->pgdata, value, MAXPGPATH);
|
||||||
|
|
||||||
else if (strcmp(name, "replication_type") == 0)
|
else if (strcmp(name, "replication_type") == 0)
|
||||||
{
|
{
|
||||||
if (strcmp(value, "physical") == 0)
|
if (strcmp(value, "physical") == 0)
|
||||||
|
|||||||
@@ -54,7 +54,8 @@ typedef struct
|
|||||||
char node_name[MAXLEN];
|
char node_name[MAXLEN];
|
||||||
char conninfo[MAXLEN];
|
char conninfo[MAXLEN];
|
||||||
char replication_user[NAMEDATALEN];
|
char replication_user[NAMEDATALEN];
|
||||||
char pg_bindir[MAXLEN];
|
char pg_bindir[MAXPGPATH];
|
||||||
|
char pgdata[MAXPGPATH];
|
||||||
int replication_type;
|
int replication_type;
|
||||||
|
|
||||||
/* log settings */
|
/* log settings */
|
||||||
@@ -118,7 +119,7 @@ typedef struct
|
|||||||
|
|
||||||
#define T_CONFIGURATION_OPTIONS_INITIALIZER { \
|
#define T_CONFIGURATION_OPTIONS_INITIALIZER { \
|
||||||
/* node information */ \
|
/* node information */ \
|
||||||
UNKNOWN_NODE_ID, "", "", "", "", REPLICATION_TYPE_PHYSICAL, \
|
UNKNOWN_NODE_ID, "", "", "", "", "", REPLICATION_TYPE_PHYSICAL, \
|
||||||
/* log settings */ \
|
/* log settings */ \
|
||||||
"", "", "", DEFAULT_LOG_STATUS_INTERVAL, \
|
"", "", "", DEFAULT_LOG_STATUS_INTERVAL, \
|
||||||
/* standby clone settings */ \
|
/* standby clone settings */ \
|
||||||
|
|||||||
@@ -19,7 +19,7 @@ static void format_archive_dir(char *archive_dir);
|
|||||||
static t_server_action parse_server_action(const char *action);
|
static t_server_action parse_server_action(const char *action);
|
||||||
|
|
||||||
static void _do_node_service_check(void);
|
static void _do_node_service_check(void);
|
||||||
static void _do_node_service_list(void);
|
static void _do_node_service_list(t_server_action action);
|
||||||
|
|
||||||
void
|
void
|
||||||
do_node_status(void)
|
do_node_status(void)
|
||||||
@@ -292,13 +292,15 @@ do_node_check(void)
|
|||||||
void
|
void
|
||||||
do_node_service(void)
|
do_node_service(void)
|
||||||
{
|
{
|
||||||
t_server_action action = parse_server_action(runtime_options.action);
|
t_server_action action;
|
||||||
|
|
||||||
|
action = parse_server_action(runtime_options.action);
|
||||||
|
|
||||||
if (action == ACTION_UNKNOWN)
|
if (action == ACTION_UNKNOWN)
|
||||||
{
|
{
|
||||||
log_error(_("unknown value \"%s\" provided for parameter --action"),
|
log_error(_("unknown value \"%s\" provided for parameter --action"),
|
||||||
runtime_options.action);
|
runtime_options.action);
|
||||||
log_hint(_("valid values are \"start\", \"stop\" and \"restart\""));
|
log_hint(_("valid values are \"start\", \"stop\", \"restart\", \"reload\" and \"promote\""));
|
||||||
exit(ERR_BAD_CONFIG);
|
exit(ERR_BAD_CONFIG);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -312,10 +314,19 @@ do_node_service(void)
|
|||||||
|
|
||||||
if (runtime_options.list == true)
|
if (runtime_options.list == true)
|
||||||
{
|
{
|
||||||
return _do_node_service_list();
|
return _do_node_service_list(action);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
// do we need data directory?
|
||||||
|
// - service command defined for action ? -> no
|
||||||
|
// -> yes
|
||||||
|
// - pgdata defined in config? OK
|
||||||
|
//
|
||||||
|
// - connection available?
|
||||||
|
// -> get data dir OK (superuser connection issue)
|
||||||
|
|
||||||
|
|
||||||
// perform action...
|
// perform action...
|
||||||
// --dry-run: print only
|
// --dry-run: print only
|
||||||
}
|
}
|
||||||
@@ -328,12 +339,42 @@ _do_node_service_check(void)
|
|||||||
|
|
||||||
|
|
||||||
static void
|
static void
|
||||||
_do_node_service_list(void)
|
_do_node_service_list(t_server_action action)
|
||||||
{
|
{
|
||||||
char command[MAXLEN] = "";
|
char command[MAXLEN] = "";
|
||||||
|
|
||||||
char *data_dir = runtime_options.data_dir;
|
char data_dir[MAXPGPATH] = "";
|
||||||
|
|
||||||
|
bool data_dir_required = false;
|
||||||
|
|
||||||
|
/* do we need to provide a data directory for any of the actions? */
|
||||||
|
if (data_dir_required_for_action(ACTION_START))
|
||||||
|
data_dir_required = true;
|
||||||
|
|
||||||
|
if (data_dir_required_for_action(ACTION_STOP))
|
||||||
|
data_dir_required = true;
|
||||||
|
|
||||||
|
if (data_dir_required_for_action(ACTION_RESTART))
|
||||||
|
data_dir_required = true;
|
||||||
|
|
||||||
|
if (data_dir_required_for_action(ACTION_RELOAD))
|
||||||
|
data_dir_required = true;
|
||||||
|
|
||||||
|
if (data_dir_required_for_action(ACTION_PROMOTE))
|
||||||
|
data_dir_required = true;
|
||||||
|
|
||||||
|
if (data_dir_required == true)
|
||||||
|
{
|
||||||
|
get_node_data_directory(data_dir);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* show command for specific action only */
|
||||||
|
if (action != ACTION_NONE)
|
||||||
|
{
|
||||||
|
get_server_action(action, command, data_dir);
|
||||||
|
printf("%s\n", command);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
puts(_("Following commands would be executed for each action:"));
|
puts(_("Following commands would be executed for each action:"));
|
||||||
puts("");
|
puts("");
|
||||||
|
|||||||
@@ -1282,7 +1282,7 @@ do_standby_follow(void)
|
|||||||
* If --wait provided, loop for up `primary_response_timeout`
|
* If --wait provided, loop for up `primary_response_timeout`
|
||||||
* seconds before giving up
|
* seconds before giving up
|
||||||
*/
|
*/
|
||||||
|
// XXX ??? primary_follow_timeout
|
||||||
for (timer = 0; timer < config_file_options.primary_follow_timeout; timer++)
|
for (timer = 0; timer < config_file_options.primary_follow_timeout; timer++)
|
||||||
{
|
{
|
||||||
primary_conn = get_primary_connection_quiet(local_conn,
|
primary_conn = get_primary_connection_quiet(local_conn,
|
||||||
@@ -1327,8 +1327,6 @@ do_standby_follow(void)
|
|||||||
exit(ERR_BAD_CONFIG);
|
exit(ERR_BAD_CONFIG);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* If 9.4 or later, and replication slots in use, we'll need to create a
|
* If 9.4 or later, and replication slots in use, we'll need to create a
|
||||||
* slot on the new primary
|
* slot on the new primary
|
||||||
|
|||||||
@@ -169,6 +169,6 @@ extern bool remote_command(const char *host, const char *user, const char *comma
|
|||||||
|
|
||||||
/* server control functions */
|
/* server control functions */
|
||||||
extern void get_server_action(t_server_action action, char *script, char *data_dir);
|
extern void get_server_action(t_server_action action, char *script, char *data_dir);
|
||||||
|
extern bool data_dir_required_for_action(t_server_action action);
|
||||||
|
extern void get_node_data_directory(char *data_dir_buf);
|
||||||
#endif
|
#endif
|
||||||
|
|||||||
@@ -1185,6 +1185,7 @@ check_cli_parameters(const int action)
|
|||||||
{
|
{
|
||||||
case STANDBY_CLONE:
|
case STANDBY_CLONE:
|
||||||
case STANDBY_FOLLOW:
|
case STANDBY_FOLLOW:
|
||||||
|
case NODE_SERVICE:
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
item_list_append_format(&cli_warnings,
|
item_list_append_format(&cli_warnings,
|
||||||
@@ -1372,7 +1373,7 @@ action_name(const int action)
|
|||||||
case NODE_RESTORE_CONFIG:
|
case NODE_RESTORE_CONFIG:
|
||||||
return "NODE RESTORE-CONFIG";
|
return "NODE RESTORE-CONFIG";
|
||||||
case NODE_SERVICE:
|
case NODE_SERVICE:
|
||||||
return "NODE_SERVICE";
|
return "NODE SERVICE";
|
||||||
|
|
||||||
case CLUSTER_SHOW:
|
case CLUSTER_SHOW:
|
||||||
return "CLUSTER SHOW";
|
return "CLUSTER SHOW";
|
||||||
@@ -2565,7 +2566,7 @@ get_server_action(t_server_action action, char *script, char *data_dir)
|
|||||||
{
|
{
|
||||||
PQExpBufferData command;
|
PQExpBufferData command;
|
||||||
|
|
||||||
if (data_dir == NULL)
|
if (data_dir == NULL || data_dir[0] == '\0')
|
||||||
data_dir = "(none provided)";
|
data_dir = "(none provided)";
|
||||||
|
|
||||||
switch(action)
|
switch(action)
|
||||||
@@ -2673,7 +2674,8 @@ get_server_action(t_server_action action, char *script, char *data_dir)
|
|||||||
{
|
{
|
||||||
if (config_file_options.service_reload_command[0] != '\0')
|
if (config_file_options.service_reload_command[0] != '\0')
|
||||||
{
|
{
|
||||||
maxlen_snprintf(script, "%s", config_file_options.service_reload_command);
|
maxlen_snprintf(script, "%s",
|
||||||
|
config_file_options.service_reload_command);
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
@@ -2704,7 +2706,8 @@ get_server_action(t_server_action action, char *script, char *data_dir)
|
|||||||
{
|
{
|
||||||
if (config_file_options.service_promote_command[0] != '\0')
|
if (config_file_options.service_promote_command[0] != '\0')
|
||||||
{
|
{
|
||||||
maxlen_snprintf(script, "%s", config_file_options.service_promote_command);
|
maxlen_snprintf(script, "%s",
|
||||||
|
config_file_options.service_promote_command);
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
@@ -2737,3 +2740,88 @@ get_server_action(t_server_action action, char *script, char *data_dir)
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
bool
|
||||||
|
data_dir_required_for_action(t_server_action action)
|
||||||
|
{
|
||||||
|
switch(action)
|
||||||
|
{
|
||||||
|
case ACTION_NONE:
|
||||||
|
return false;
|
||||||
|
|
||||||
|
case ACTION_START:
|
||||||
|
if (config_file_options.service_start_command[0] != '\0')
|
||||||
|
{
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
|
||||||
|
case ACTION_STOP:
|
||||||
|
if (config_file_options.service_stop_command[0] != '\0')
|
||||||
|
{
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
|
||||||
|
case ACTION_RESTART:
|
||||||
|
if (config_file_options.service_restart_command[0] != '\0')
|
||||||
|
{
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
|
||||||
|
case ACTION_RELOAD:
|
||||||
|
if (config_file_options.service_reload_command[0] != '\0')
|
||||||
|
{
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
|
||||||
|
case ACTION_PROMOTE:
|
||||||
|
if (config_file_options.service_promote_command[0] != '\0')
|
||||||
|
{
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
|
||||||
|
default:
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
// optionally pass conn?
|
||||||
|
void
|
||||||
|
get_node_data_directory(char *data_dir_buf)
|
||||||
|
{
|
||||||
|
PGconn *conn = NULL;
|
||||||
|
bool success = false;
|
||||||
|
|
||||||
|
if (config_file_options.pgdata[0] != '\0')
|
||||||
|
{
|
||||||
|
strncpy(data_dir_buf, config_file_options.pgdata, MAXPGPATH);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (strlen(config_file_options.conninfo))
|
||||||
|
conn = establish_db_connection(config_file_options.conninfo, true);
|
||||||
|
else
|
||||||
|
conn = establish_db_connection_by_params(&source_conninfo, true);
|
||||||
|
|
||||||
|
success = get_pg_setting(conn, "data_directory", data_dir_buf);
|
||||||
|
|
||||||
|
PQfinish(conn);
|
||||||
|
|
||||||
|
if (success == true)
|
||||||
|
return;
|
||||||
|
|
||||||
|
if (runtime_options.data_dir[0] != '\0')
|
||||||
|
{
|
||||||
|
strncpy(data_dir_buf, runtime_options.data_dir, MAXPGPATH);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|||||||
@@ -40,6 +40,12 @@
|
|||||||
|
|
||||||
# Optional configuration items
|
# Optional configuration items
|
||||||
# =============================================================================
|
# =============================================================================
|
||||||
|
#------------------------------------------------------------------------------
|
||||||
|
# General settings
|
||||||
|
#------------------------------------------------------------------------------
|
||||||
|
|
||||||
|
#pgdata # The node's data directory; usually only required
|
||||||
|
# if the repmgr user is not a superuser
|
||||||
|
|
||||||
#------------------------------------------------------------------------------
|
#------------------------------------------------------------------------------
|
||||||
# Replication settings
|
# Replication settings
|
||||||
|
|||||||
Reference in New Issue
Block a user