18 #include <sys/types.h> 29 struct stonith_action_s {
39 void (*fork_cb) (
int pid,
void *user_data);
44 time_t initial_start_time;
46 int remaining_timeout;
79 if (
action->result.action_stderr != NULL) {
89 append_config_arg(gpointer key, gpointer value, gpointer user_data)
102 crm_trace(
"Passing %s=%s with fence action",
103 (
const char *) key, (
const char *) (value? value :
""));
123 make_args(
const char *agent,
const char *
action,
const char *
target,
124 uint32_t target_nodeid, GHashTable *device_args,
125 GHashTable *port_map,
const char *host_arg)
127 GHashTable *arg_list = NULL;
128 const char *value = NULL;
138 snprintf(buffer,
sizeof(buffer),
"pcmk_%s_action",
action);
139 value = g_hash_table_lookup(device_args, buffer);
141 crm_debug(
"Substituting '%s' for fence action %s targeting %s",
153 if ((
target != NULL) && (device_args != NULL)) {
154 const char *param = NULL;
162 if (target_nodeid != 0) {
166 crm_info(
"Passing '%s' as nodeid with fence action '%s' targeting %s",
168 g_hash_table_insert(arg_list, strdup(
"nodeid"), nodeid);
181 value = g_hash_table_lookup(device_args, param);
182 if (pcmk__str_eq(value,
"dynamic",
187 const char *alias = NULL;
190 alias = g_hash_table_lookup(port_map,
target);
195 crm_debug(
"Passing %s='%s' with fence action %s targeting %s",
203 g_hash_table_foreach(device_args, append_config_arg, arg_list);
221 g_hash_table_destroy(
action->args);
246 #define FAILURE_MAX_RETRIES 2 265 const char *
target, uint32_t target_nodeid,
266 int timeout_sec, GHashTable *device_args,
267 GHashTable *port_map,
const char *host_arg)
271 action->args = make_args(agent, action_name,
target, target_nodeid,
272 device_args, port_map, host_arg);
273 crm_debug(
"Preparing '%s' action targeting %s using agent %s",
274 action_name, pcmk__s(
target,
"no node"), agent);
275 action->agent = strdup(agent);
276 action->action = strdup(action_name);
277 action->timeout =
action->remaining_timeout = timeout_sec;
281 "Initialization bug in fencing library");
285 const char *value = NULL;
287 snprintf(buffer,
sizeof(buffer),
"pcmk_%s_retries", action_name);
288 value = g_hash_table_lookup(device_args, buffer);
291 action->max_retries = atoi(value);
301 int diff = time(NULL) -
action->initial_start_time;
304 crm_info(
"Attempted to execute agent %s (%s) the maximum number of times (%d) allowed",
306 action->remaining_timeout = 0;
308 && (diff < (
action->timeout * 0.7))) {
313 action->remaining_timeout = 0;
315 return action->remaining_timeout ? TRUE : FALSE;
329 if (pcmk__result_ok(
result)) {
435 const char *exit_reason = NULL;
436 const char *action_stdout = NULL;
495 const char *exit_reason = NULL;
496 char *action_stdout = NULL;
506 &execution_status) < 0)) {
514 if ((rc ==
pcmk_ok) || (rc == -EINPROGRESS)) {
522 exit_reason =
"Fencer reply contained neither a full result " 523 "nor a legacy return code (bug?)";
535 set_result_from_svc_action(
action, svc_action);
536 svc_action->
params = NULL;
539 if (!pcmk__result_ok(&(
action->result))
540 && update_remaining_timeout(
action)) {
542 int rc = internal_stonith_action_execute(
action);
552 action->svc_action = NULL;
562 action->svc_action = svc_action;
571 crm_trace(
"Child process %d performing action '%s' successfully forked",
581 static int stonith_sequence = 0;
587 || (
action->agent == NULL)) {
594 action->initial_start_time = time(NULL);
599 crm_info(
"Attempt %d to execute %s (%s). remaining timeout is %d",
610 set_result_from_svc_action(
action, svc_action);
621 svc_action->
sequence = stonith_sequence++;
624 svc_action->
flags = pcmk__set_flags_as(__func__, __LINE__,
626 svc_action->
id, svc_action->
flags,
628 "SVC_ACTION_NON_BLOCKED");
639 &stonith_action_async_done,
640 &stonith_action_async_forked));
650 set_result_from_svc_action(
action, svc_action);
651 svc_action->
params = NULL;
669 void (*done) (
int pid,
672 void (*fork_cb) (
int pid,
void *user_data))
678 action->userdata = userdata;
680 action->fork_cb = fork_cb;
683 return internal_stonith_action_execute(
action);
703 rc = internal_stonith_action_execute(
action);
704 }
while ((rc !=
pcmk_ok) && update_remaining_timeout(
action));
int rc
Exit status of action (set by library upon completion)
#define CRM_CHECK(expr, failure_action)
struct stonith_action_s stonith_action_t
const char * pcmk_strerror(int rc)
No connection to executor.
void services_action_free(svc_action_t *op)
#define PCMK__FENCE_BINDIR
int pcmk_rc2legacy(int rc)
char * standard
Resource standard for resource actions, otherwise NULL.
void stonith__xe_get_result(const xmlNode *xml, pcmk__action_result_t *result)
#define PCMK__XA_ST_OUTPUT
#define crm_log_output(level, prefix, output)
void void pcmk__set_result_output(pcmk__action_result_t *result, char *out, char *err)
bool pcmk_stonith_param(const char *param)
Check whether a given stonith parameter is handled by Pacemaker.
#define PCMK_XA_EXIT_REASON
enum pcmk_exec_status execution_status
svc_action_t * services_action_create_generic(const char *exec, const char *args[])
Request execution of an arbitrary command.
int stonith__legacy2status(int rc)
gboolean services_action_async_fork_notify(svc_action_t *op, void(*action_callback)(svc_action_t *), void(*action_fork_callback)(svc_action_t *))
Run an action asynchronously, with callback after process is forked.
Necessary CIB secrets are unavailable.
#define CRM_LOG_ASSERT(expr)
stonith_action_t * stonith__action_create(const char *agent, const char *action_name, const char *target, uint32_t target_nodeid, int timeout_sec, GHashTable *device_args, GHashTable *port_map, const char *host_arg)
int timeout
Action timeout (in milliseconds)
Action did not complete in time.
xmlNode * get_xpath_object(const char *xpath, xmlNode *xml_obj, int error_level)
No fence device is configured for target.
const char * crm_xml_add(xmlNode *node, const char *name, const char *value)
Create an XML attribute with specified name and value.
#define FAILURE_MAX_RETRIES
enum svc_action_flags flags
Flag group of enum svc_action_flags.
char * services__grab_stderr(svc_action_t *action)
#define crm_debug(fmt, args...)
const char * services__exit_reason(const svc_action_t *action)
Used only to initialize variables.
gboolean services_action_sync(svc_action_t *op)
char * crm_element_value_copy(const xmlNode *data, const char *name)
Retrieve a copy of the value of an XML attribute.
const char * crm_element_value(const xmlNode *data, const char *name)
Retrieve the value of an XML attribute.
#define crm_trace(fmt, args...)
#define PCMK_RESOURCE_CLASS_STONITH
Object for executing external actions.
char * services__grab_stdout(svc_action_t *action)
char * agent
Resource agent name for resource actions, otherwise NULL.
Wrappers for and extensions to libxml2.
Action completed, result is known.
Execution failed, do not retry anywhere.
#define STONITH_ATTR_ACTION_OP
int stonith__execute_async(stonith_action_t *action, void *userdata, void(*done)(int pid, const pcmk__action_result_t *result, void *user_data), void(*fork_cb)(int pid, void *user_data))
Requested item has expired.
#define PCMK_XA_CRM_FEATURE_SET
#define PCMK_STONITH_HOST_ARGUMENT
Agent does not implement requested action.
GHashTable * pcmk__strkey_table(GDestroyNotify key_destroy_func, GDestroyNotify value_destroy_func)
pcmk__action_result_t result
pcmk__action_result_t * stonith__action_result(stonith_action_t *action)
int stonith__result2rc(const pcmk__action_result_t *result)
int status
Execution status (enum pcmk_exec_status set by library)
#define PCMK__XA_OP_STATUS
int crm_element_value_int(const xmlNode *data, const char *name, int *dest)
Retrieve the integer value of an XML attribute.
Agent or dependency not available locally.
int stonith__execute(stonith_action_t *action)
xmlNode * stonith__find_xe_with_result(xmlNode *xml)
void * cb_data
For caller's use (not used by library)
void pcmk__set_result(pcmk__action_result_t *result, int exit_status, enum pcmk_exec_status exec_status, const char *exit_reason)
pcmk_exec_status
Execution status.
Parameter invalid (in local context)
Action cannot be attempted (e.g. shutdown)
#define pcmk__assert_alloc(nmemb, size)
void pcmk__reset_result(pcmk__action_result_t *result)
void stonith__destroy_action(stonith_action_t *action)
Execution failed, may be retried.
#define crm_info(fmt, args...)
void pcmk__insert_dup(GHashTable *table, const char *name, const char *value)
void stonith__xe_set_result(xmlNode *xml, const pcmk__action_result_t *result)
char * crm_strdup_printf(char const *format,...) G_GNUC_PRINTF(1
const char * crm_xml_add_int(xmlNode *node, const char *name, int value)
Create an XML attribute with specified name and integer value.