pacemaker  2.1.2-ada5c3b36
Scalable High-Availability cluster resource manager
alerts.c
Go to the documentation of this file.
1 /*
2  * Copyright 2015-2021 the Pacemaker project contributors
3  *
4  * The version control history for this file may have further details.
5  *
6  * This source code is licensed under the GNU Lesser General Public License
7  * version 2.1 or later (LGPLv2.1+) WITHOUT ANY WARRANTY.
8  */
9 
10 #include <crm_internal.h>
11 #include <crm/crm.h>
12 #include <crm/lrmd.h>
13 #include <crm/msg_xml.h>
16 #include <crm/cib/internal.h> /* for F_CIB_UPDATE_RESULT */
17 
18 /*
19  * to allow script compatibility we can have more than one
20  * set of environment variables
21  */
23 {
25  "CRM_notify_recipient", "CRM_alert_recipient", NULL
26  },
28  "CRM_notify_node", "CRM_alert_node", NULL
29  },
31  "CRM_notify_nodeid", "CRM_alert_nodeid", NULL
32  },
34  "CRM_notify_rsc", "CRM_alert_rsc", NULL
35  },
37  "CRM_notify_task", "CRM_alert_task", NULL
38  },
40  "CRM_notify_interval", "CRM_alert_interval", NULL
41  },
43  "CRM_notify_desc", "CRM_alert_desc", NULL
44  },
46  "CRM_notify_status", "CRM_alert_status", NULL
47  },
49  "CRM_notify_target_rc", "CRM_alert_target_rc", NULL
50  },
51  [PCMK__alert_key_rc] = {
52  "CRM_notify_rc", "CRM_alert_rc", NULL
53  },
55  "CRM_notify_kind", "CRM_alert_kind", NULL
56  },
58  "CRM_notify_version", "CRM_alert_version", NULL
59  },
61  "CRM_notify_node_sequence", PCMK__ALERT_NODE_SEQUENCE, NULL
62  },
64  "CRM_notify_timestamp", "CRM_alert_timestamp", NULL
65  },
67  "CRM_notify_attribute_name", "CRM_alert_attribute_name", NULL
68  },
70  "CRM_notify_attribute_value", "CRM_alert_attribute_value", NULL
71  },
73  "CRM_notify_timestamp_epoch", "CRM_alert_timestamp_epoch", NULL
74  },
76  "CRM_notify_timestamp_usec", "CRM_alert_timestamp_usec", NULL
77  },
79  "CRM_notify_exec_time", "CRM_alert_exec_time", NULL
80  }
81 };
82 
95 pcmk__alert_new(const char *id, const char *path)
96 {
97  pcmk__alert_t *entry = calloc(1, sizeof(pcmk__alert_t));
98 
99  CRM_ASSERT(entry && id && path);
100  entry->id = strdup(id);
101  entry->path = strdup(path);
103  entry->flags = pcmk__alert_default;
104  return entry;
105 }
106 
107 void
109 {
110  if (entry) {
111  free(entry->id);
112  free(entry->path);
113  free(entry->tstamp_format);
114  free(entry->recipient);
115 
116  g_strfreev(entry->select_attribute_name);
117  if (entry->envvars) {
118  g_hash_table_destroy(entry->envvars);
119  }
120  free(entry);
121  }
122 }
123 
134 {
135  pcmk__alert_t *new_entry = pcmk__alert_new(entry->id, entry->path);
136 
137  new_entry->timeout = entry->timeout;
138  new_entry->flags = entry->flags;
139  new_entry->envvars = pcmk__str_table_dup(entry->envvars);
140  if (entry->tstamp_format) {
141  new_entry->tstamp_format = strdup(entry->tstamp_format);
142  }
143  if (entry->recipient) {
144  new_entry->recipient = strdup(entry->recipient);
145  }
146  if (entry->select_attribute_name) {
147  new_entry->select_attribute_name = g_strdupv(entry->select_attribute_name);
148  }
149  return new_entry;
150 }
151 
152 void
154  const char *value)
155 {
156  for (const char **key = pcmk__alert_keys[name]; *key; key++) {
157  crm_trace("Inserting alert key %s = '%s'", *key, value);
158  if (value) {
159  g_hash_table_insert(table, strdup(*key), strdup(value));
160  } else {
161  g_hash_table_remove(table, *key);
162  }
163  }
164 }
165 
166 void
168  int value)
169 {
170  for (const char **key = pcmk__alert_keys[name]; *key; key++) {
171  crm_trace("Inserting alert key %s = %d", *key, value);
172  g_hash_table_insert(table, strdup(*key), pcmk__itoa(value));
173  }
174 }
175 
176 #define XPATH_PATCHSET1_DIFF "//" F_CIB_UPDATE_RESULT "//" XML_TAG_DIFF_ADDED
177 
178 #define XPATH_PATCHSET1_CRMCONFIG XPATH_PATCHSET1_DIFF "//" XML_CIB_TAG_CRMCONFIG
179 #define XPATH_PATCHSET1_ALERTS XPATH_PATCHSET1_DIFF "//" XML_CIB_TAG_ALERTS
180 
181 #define XPATH_PATCHSET1_EITHER \
182  XPATH_PATCHSET1_CRMCONFIG " | " XPATH_PATCHSET1_ALERTS
183 
184 #define XPATH_CONFIG "/" XML_TAG_CIB "/" XML_CIB_TAG_CONFIGURATION
185 
186 #define XPATH_CRMCONFIG XPATH_CONFIG "/" XML_CIB_TAG_CRMCONFIG "/"
187 #define XPATH_ALERTS XPATH_CONFIG "/" XML_CIB_TAG_ALERTS
188 
198 bool
199 pcmk__alert_in_patchset(xmlNode *msg, bool config)
200 {
201  int rc = -1;
202  int format= 1;
203  xmlNode *patchset = get_message_xml(msg, F_CIB_UPDATE_RESULT);
204  xmlNode *change = NULL;
205  xmlXPathObject *xpathObj = NULL;
206 
207  CRM_CHECK(msg != NULL, return FALSE);
208 
210  if (rc < pcmk_ok) {
211  crm_trace("Ignore failed CIB update: %s (%d)", pcmk_strerror(rc), rc);
212  return FALSE;
213  }
214 
215  crm_element_value_int(patchset, "format", &format);
216  if (format == 1) {
217  const char *diff = (config? XPATH_PATCHSET1_EITHER : XPATH_PATCHSET1_ALERTS);
218 
219  if ((xpathObj = xpath_search(msg, diff)) != NULL) {
220  freeXpathObject(xpathObj);
221  return TRUE;
222  }
223  } else if (format == 2) {
224  for (change = pcmk__xml_first_child(patchset); change != NULL;
225  change = pcmk__xml_next(change)) {
226  const char *xpath = crm_element_value(change, XML_DIFF_PATH);
227 
228  if (xpath == NULL) {
229  continue;
230  }
231 
232  if ((!config || !strstr(xpath, XPATH_CRMCONFIG))
233  && !strstr(xpath, XPATH_ALERTS)) {
234 
235  /* this is not a change to an existing section ... */
236 
237  xmlNode *section = NULL;
238  const char *name = NULL;
239 
240  if ((strcmp(xpath, XPATH_CONFIG) != 0) ||
241  ((section = pcmk__xml_first_child(change)) == NULL) ||
242  ((name = crm_element_name(section)) == NULL) ||
243  (strcmp(name, XML_CIB_TAG_ALERTS) != 0)) {
244 
245  /* ... nor is it a newly added alerts section */
246  continue;
247  }
248  }
249 
250  return TRUE;
251  }
252 
253  } else {
254  crm_warn("Unknown patch format: %d", format);
255  }
256  return FALSE;
257 }
#define CRM_CHECK(expr, failure_action)
Definition: logging.h:225
const char * pcmk__alert_keys[PCMK__ALERT_INTERNAL_KEY_MAX][3]
Definition: alerts.c:22
A dumping ground.
xmlNode * get_message_xml(xmlNode *msg, const char *field)
Definition: messages.c:154
const char * pcmk_strerror(int rc)
Definition: results.c:58
void pcmk__free_alert(pcmk__alert_t *entry)
Definition: alerts.c:108
#define XPATH_ALERTS
Definition: alerts.c:187
#define PCMK__ALERT_INTERNAL_KEY_MAX
char ** select_attribute_name
char * tstamp_format
Resource agent executor.
#define PCMK__ALERT_NODE_SEQUENCE
pcmk__alert_keys_e
int crm_element_value_int(const xmlNode *data, const char *name, int *dest)
Retrieve the integer value of an XML attribute.
Definition: nvpair.c:565
#define XPATH_CRMCONFIG
Definition: alerts.c:186
#define XPATH_PATCHSET1_EITHER
Definition: alerts.c:181
#define crm_warn(fmt, args...)
Definition: logging.h:358
#define XPATH_PATCHSET1_ALERTS
Definition: alerts.c:179
int rc
Definition: pcmk_fence.c:35
#define F_CIB_RC
Definition: internal.h:41
const char * crm_element_value(const xmlNode *data, const char *name)
Retrieve the value of an XML attribute.
Definition: nvpair.c:529
void pcmk__add_alert_key(GHashTable *table, enum pcmk__alert_keys_e name, const char *value)
Definition: alerts.c:153
#define XPATH_CONFIG
Definition: alerts.c:184
#define crm_trace(fmt, args...)
Definition: logging.h:363
#define F_CIB_UPDATE_RESULT
Definition: internal.h:52
void pcmk__add_alert_key_int(GHashTable *table, enum pcmk__alert_keys_e name, int value)
Definition: alerts.c:167
GHashTable * envvars
pcmk__alert_t * pcmk__dup_alert(pcmk__alert_t *entry)
Definition: alerts.c:133
#define XML_DIFF_PATH
Definition: msg_xml.h:451
#define CRM_ASSERT(expr)
Definition: results.h:42
xmlXPathObjectPtr xpath_search(xmlNode *xml_top, const char *path)
Definition: xpath.c:139
GHashTable * pcmk__str_table_dup(GHashTable *old_table)
Definition: strings.c:675
#define pcmk_ok
Definition: results.h:68
bool pcmk__alert_in_patchset(xmlNode *msg, bool config)
Definition: alerts.c:199
#define XML_CIB_TAG_ALERTS
Definition: msg_xml.h:188
char * name
Definition: pcmk_fence.c:31
void freeXpathObject(xmlXPathObjectPtr xpathObj)
Definition: xpath.c:39
pcmk__alert_t * pcmk__alert_new(const char *id, const char *path)
Create a new alert entry structure.
Definition: alerts.c:95
#define PCMK__ALERT_DEFAULT_TIMEOUT_MS