pacemaker 2.1.5-a3f44794f94
Scalable High-Availability cluster resource manager
nvpair.c
Go to the documentation of this file.
1/*
2 * Copyright 2004-2022 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
12#include <stdio.h>
13#include <sys/types.h>
14#include <string.h>
15#include <ctype.h>
16#include <glib.h>
17#include <libxml/tree.h>
18
19#include <crm/crm.h>
20#include <crm/msg_xml.h>
21#include <crm/common/xml.h>
23#include "crmcommon_private.h"
24
25/*
26 * This file isolates handling of three types of name/value pairs:
27 *
28 * - pcmk_nvpair_t data type
29 * - XML attributes (<TAG ... NAME=VALUE ...>)
30 * - XML nvpair elements (<nvpair id=ID name=NAME value=VALUE>)
31 */
32
33// pcmk_nvpair_t handling
34
46static pcmk_nvpair_t *
47pcmk__new_nvpair(const char *name, const char *value)
48{
49 pcmk_nvpair_t *nvpair = NULL;
50
52
53 nvpair = calloc(1, sizeof(pcmk_nvpair_t));
54 CRM_ASSERT(nvpair);
55
56 pcmk__str_update(&nvpair->name, name);
57 pcmk__str_update(&nvpair->value, value);
58 return nvpair;
59}
60
67static void
68pcmk__free_nvpair(gpointer data)
69{
70 if (data) {
71 pcmk_nvpair_t *nvpair = data;
72
73 free(nvpair->name);
74 free(nvpair->value);
75 free(nvpair);
76 }
77}
78
90GSList *
91pcmk_prepend_nvpair(GSList *nvpairs, const char *name, const char *value)
92{
93 return g_slist_prepend(nvpairs, pcmk__new_nvpair(name, value));
94}
95
101void
102pcmk_free_nvpairs(GSList *nvpairs)
103{
104 g_slist_free_full(nvpairs, pcmk__free_nvpair);
105}
106
116static gint
117pcmk__compare_nvpair(gconstpointer a, gconstpointer b)
118{
119 int rc = 0;
120 const pcmk_nvpair_t *pair_a = a;
121 const pcmk_nvpair_t *pair_b = b;
122
123 CRM_ASSERT(a != NULL);
124 CRM_ASSERT(pair_a->name != NULL);
125
126 CRM_ASSERT(b != NULL);
127 CRM_ASSERT(pair_b->name != NULL);
128
129 rc = strcmp(pair_a->name, pair_b->name);
130 if (rc < 0) {
131 return -1;
132 } else if (rc > 0) {
133 return 1;
134 }
135 return 0;
136}
137
145GSList *
146pcmk_sort_nvpairs(GSList *list)
147{
148 return g_slist_sort(list, pcmk__compare_nvpair);
149}
150
160GSList *
161pcmk_xml_attrs2nvpairs(const xmlNode *xml)
162{
163 GSList *result = NULL;
164
165 for (xmlAttrPtr iter = pcmk__xe_first_attr(xml); iter != NULL;
166 iter = iter->next) {
167
169 (const char *) iter->name,
170 (const char *) pcmk__xml_attr_value(iter));
171 }
172 return result;
173}
174
185static void
186pcmk__nvpair_add_xml_attr(gpointer data, gpointer user_data)
187{
188 pcmk_nvpair_t *pair = data;
189 xmlNode *parent = user_data;
190
191 crm_xml_add(parent, pair->name, pair->value);
192}
193
200void
201pcmk_nvpairs2xml_attrs(GSList *list, xmlNode *xml)
202{
203 g_slist_foreach(list, pcmk__nvpair_add_xml_attr, xml);
204}
205
206// convenience function for name=value strings
207
220int
221pcmk__scan_nvpair(const char *input, char **name, char **value)
222{
223#ifdef HAVE_SSCANF_M
224 *name = NULL;
225 *value = NULL;
226 if (sscanf(input, "%m[^=]=%m[^\n]", name, value) <= 0) {
227 return -pcmk_err_bad_nvpair;
228 }
229#else
230 char *sep = NULL;
231 *name = NULL;
232 *value = NULL;
233
234 sep = strstr(optarg, "=");
235 if (sep == NULL) {
236 return -pcmk_err_bad_nvpair;
237 }
238
239 *name = strndup(input, sep-input);
240
241 if (*name == NULL) {
242 return -ENOMEM;
243 }
244
245 /* If the last char in optarg is =, the user gave no
246 * value for the option. Leave it as NULL.
247 */
248 if (*(sep+1) != '\0') {
249 *value = strdup(sep+1);
250
251 if (*value == NULL) {
252 return -ENOMEM;
253 }
254 }
255#endif
256
257 if (*name != NULL && *value != NULL) {
258 return 2;
259 } else if (*name != NULL || *value != NULL) {
260 return 1;
261 } else {
262 return -pcmk_err_bad_nvpair;
263 }
264}
265
283char *
284pcmk__format_nvpair(const char *name, const char *value, const char *units)
285{
286 return crm_strdup_printf("%s=\"%s%s\"", name, value, units ? units : "");
287}
288
302char *
303pcmk__format_named_time(const char *name, time_t epoch_time)
304{
305 const char *now_str = pcmk__epoch2str(&epoch_time);
306
307 return crm_strdup_printf("%s=\"%s\"", name, now_str ? now_str : "");
308}
309
310// XML attribute handling
311
322const char *
323crm_xml_add(xmlNode *node, const char *name, const char *value)
324{
325 bool dirty = FALSE;
326 xmlAttr *attr = NULL;
327
328 CRM_CHECK(node != NULL, return NULL);
329 CRM_CHECK(name != NULL, return NULL);
330
331 if (value == NULL) {
332 return NULL;
333 }
334
335 if (pcmk__tracking_xml_changes(node, FALSE)) {
336 const char *old = crm_element_value(node, name);
337
338 if (old == NULL || value == NULL || strcmp(old, value) != 0) {
339 dirty = TRUE;
340 }
341 }
342
343 if (dirty && (pcmk__check_acl(node, name, pcmk__xf_acl_create) == FALSE)) {
344 crm_trace("Cannot add %s=%s to %s", name, value, node->name);
345 return NULL;
346 }
347
348 attr = xmlSetProp(node, (pcmkXmlStr) name, (pcmkXmlStr) value);
349 if (dirty) {
351 }
352
353 CRM_CHECK(attr && attr->children && attr->children->content, return NULL);
354 return (char *)attr->children->content;
355}
356
367const char *
368crm_xml_replace(xmlNode *node, const char *name, const char *value)
369{
370 bool dirty = FALSE;
371 xmlAttr *attr = NULL;
372 const char *old_value = NULL;
373
374 CRM_CHECK(node != NULL, return NULL);
375 CRM_CHECK(name != NULL && name[0] != 0, return NULL);
376
377 old_value = crm_element_value(node, name);
378
379 /* Could be re-setting the same value */
380 CRM_CHECK(old_value != value, return value);
381
382 if (pcmk__check_acl(node, name, pcmk__xf_acl_write) == FALSE) {
383 /* Create a fake object linked to doc->_private instead? */
384 crm_trace("Cannot replace %s=%s to %s", name, value, node->name);
385 return NULL;
386
387 } else if (old_value && !value) {
388 xml_remove_prop(node, name);
389 return NULL;
390 }
391
392 if (pcmk__tracking_xml_changes(node, FALSE)) {
393 if (!old_value || !value || !strcmp(old_value, value)) {
394 dirty = TRUE;
395 }
396 }
397
398 attr = xmlSetProp(node, (pcmkXmlStr) name, (pcmkXmlStr) value);
399 if (dirty) {
401 }
402 CRM_CHECK(attr && attr->children && attr->children->content, return NULL);
403 return (char *) attr->children->content;
404}
405
418const char *
419crm_xml_add_int(xmlNode *node, const char *name, int value)
420{
421 char *number = pcmk__itoa(value);
422 const char *added = crm_xml_add(node, name, number);
423
424 free(number);
425 return added;
426}
427
440const char *
441crm_xml_add_ms(xmlNode *node, const char *name, guint ms)
442{
443 char *number = crm_strdup_printf("%u", ms);
444 const char *added = crm_xml_add(node, name, number);
445
446 free(number);
447 return added;
448}
449
450// Maximum size of null-terminated string representation of 64-bit integer
451// -9223372036854775808
452#define LLSTRSIZE 21
453
468const char *
469crm_xml_add_ll(xmlNode *xml, const char *name, long long value)
470{
471 char s[LLSTRSIZE] = { '\0', };
472
473 if (snprintf(s, LLSTRSIZE, "%lld", (long long) value) == LLSTRSIZE) {
474 return NULL;
475 }
476 return crm_xml_add(xml, name, s);
477}
478
492const char *
493crm_xml_add_timeval(xmlNode *xml, const char *name_sec, const char *name_usec,
494 const struct timeval *value)
495{
496 const char *added = NULL;
497
498 if (xml && name_sec && value) {
499 added = crm_xml_add_ll(xml, name_sec, (long long) value->tv_sec);
500 if (added && name_usec) {
501 // Any error is ignored (we successfully added seconds)
502 crm_xml_add_ll(xml, name_usec, (long long) value->tv_usec);
503 }
504 }
505 return added;
506}
507
516const char *
517crm_element_value(const xmlNode *data, const char *name)
518{
519 xmlAttr *attr = NULL;
520
521 if (data == NULL) {
522 crm_err("Couldn't find %s in NULL", name ? name : "<null>");
523 CRM_LOG_ASSERT(data != NULL);
524 return NULL;
525
526 } else if (name == NULL) {
527 crm_err("Couldn't find NULL in %s", crm_element_name(data));
528 return NULL;
529 }
530
531 /* The first argument to xmlHasProp() has always been const,
532 * but libxml2 <2.9.2 didn't declare that, so cast it
533 */
534 attr = xmlHasProp((xmlNode *) data, (pcmkXmlStr) name);
535 if (!attr || !attr->children) {
536 return NULL;
537 }
538 return (const char *) attr->children->content;
539}
540
552int
553crm_element_value_int(const xmlNode *data, const char *name, int *dest)
554{
555 const char *value = NULL;
556
557 CRM_CHECK(dest != NULL, return -1);
558 value = crm_element_value(data, name);
559 if (value) {
560 long long value_ll;
561
562 if ((pcmk__scan_ll(value, &value_ll, 0LL) != pcmk_rc_ok)
563 || (value_ll < INT_MIN) || (value_ll > INT_MAX)) {
565 } else {
566 *dest = (int) value_ll;
567 return 0;
568 }
569 }
570 return -1;
571}
572
584int
585crm_element_value_ll(const xmlNode *data, const char *name, long long *dest)
586{
587 const char *value = NULL;
588
589 CRM_CHECK(dest != NULL, return -1);
590 value = crm_element_value(data, name);
591 if ((value != NULL)
592 && (pcmk__scan_ll(value, dest, PCMK__PARSE_INT_DEFAULT) == pcmk_rc_ok)) {
593 return 0;
594 }
595 return -1;
596}
597
609int
610crm_element_value_ms(const xmlNode *data, const char *name, guint *dest)
611{
612 const char *value = NULL;
613 long long value_ll;
614
615 CRM_CHECK(dest != NULL, return -1);
616 *dest = 0;
617 value = crm_element_value(data, name);
618 if ((pcmk__scan_ll(value, &value_ll, 0LL) != pcmk_rc_ok)
619 || (value_ll < 0) || (value_ll > G_MAXUINT)) {
620 return -1;
621 }
622 *dest = (guint) value_ll;
623 return pcmk_ok;
624}
625
637int
638crm_element_value_epoch(const xmlNode *xml, const char *name, time_t *dest)
639{
640 long long value_ll = 0;
641
642 if (crm_element_value_ll(xml, name, &value_ll) < 0) {
643 return -1;
644 }
645
646 /* Unfortunately, we can't do any bounds checking, since time_t has neither
647 * standardized bounds nor constants defined for them.
648 */
649 *dest = (time_t) value_ll;
650 return pcmk_ok;
651}
652
666int
667crm_element_value_timeval(const xmlNode *xml, const char *name_sec,
668 const char *name_usec, struct timeval *dest)
669{
670 long long value_i = 0;
671
672 CRM_CHECK(dest != NULL, return -EINVAL);
673 dest->tv_sec = 0;
674 dest->tv_usec = 0;
675
676 if (xml == NULL) {
677 return pcmk_ok;
678 }
679
680 /* Unfortunately, we can't do any bounds checking, since there are no
681 * constants provided for the bounds of time_t and suseconds_t, and
682 * calculating them isn't worth the effort. If there are XML values
683 * beyond the native sizes, there will probably be worse problems anyway.
684 */
685
686 // Parse seconds
687 errno = 0;
688 if (crm_element_value_ll(xml, name_sec, &value_i) < 0) {
689 return -errno;
690 }
691 dest->tv_sec = (time_t) value_i;
692
693 // Parse microseconds
694 if (crm_element_value_ll(xml, name_usec, &value_i) < 0) {
695 return -errno;
696 }
697 dest->tv_usec = (suseconds_t) value_i;
698
699 return pcmk_ok;
700}
701
713char *
714crm_element_value_copy(const xmlNode *data, const char *name)
715{
716 char *value_copy = NULL;
717
719 return value_copy;
720}
721
735void
736hash2smartfield(gpointer key, gpointer value, gpointer user_data)
737{
738 const char *name = key;
739 const char *s_value = value;
740
741 xmlNode *xml_node = user_data;
742
743 if (isdigit(name[0])) {
744 xmlNode *tmp = create_xml_node(xml_node, XML_TAG_PARAM);
745
747 crm_xml_add(tmp, XML_NVPAIR_ATTR_VALUE, s_value);
748
749 } else if (crm_element_value(xml_node, name) == NULL) {
750 crm_xml_add(xml_node, name, s_value);
751 crm_trace("dumped: %s=%s", name, s_value);
752
753 } else {
754 crm_trace("duplicate: %s=%s", name, s_value);
755 }
756}
757
769void
770hash2field(gpointer key, gpointer value, gpointer user_data)
771{
772 const char *name = key;
773 const char *s_value = value;
774
775 xmlNode *xml_node = user_data;
776
777 if (crm_element_value(xml_node, name) == NULL) {
778 crm_xml_add(xml_node, name, s_value);
779
780 } else {
781 crm_trace("duplicate: %s=%s", name, s_value);
782 }
783}
784
797void
798hash2metafield(gpointer key, gpointer value, gpointer user_data)
799{
800 char *crm_name = NULL;
801
802 if (key == NULL || value == NULL) {
803 return;
804 }
805
806 /* Filter out cluster-generated attributes that contain a '#' or ':'
807 * (like fail-count and last-failure).
808 */
809 for (crm_name = key; *crm_name; ++crm_name) {
810 if ((*crm_name == '#') || (*crm_name == ':')) {
811 return;
812 }
813 }
814
815 crm_name = crm_meta_name(key);
816 hash2field(crm_name, value, user_data);
817 free(crm_name);
818}
819
820// nvpair handling
821
832xmlNode *
833crm_create_nvpair_xml(xmlNode *parent, const char *id, const char *name,
834 const char *value)
835{
836 xmlNode *nvp;
837
838 /* id can be NULL so we auto-generate one, and name can be NULL if this
839 * will be used to delete a name/value pair by ID, but both can't be NULL
840 */
841 CRM_CHECK(id || name, return NULL);
842
844 CRM_CHECK(nvp, return NULL);
845
846 if (id) {
847 crm_xml_add(nvp, XML_ATTR_ID, id);
848 } else {
849 const char *parent_id = ID(parent);
850
851 crm_xml_set_id(nvp, "%s-%s",
852 (parent_id? parent_id : XML_CIB_TAG_NVPAIR), name);
853 }
856 return nvp;
857}
858
870void
871hash2nvpair(gpointer key, gpointer value, gpointer user_data)
872{
873 const char *name = key;
874 const char *s_value = value;
875 xmlNode *xml_node = user_data;
876
877 crm_create_nvpair_xml(xml_node, name, name, s_value);
878 crm_trace("dumped: name=%s value=%s", name, s_value);
879}
880
895GHashTable *
896xml2list(const xmlNode *parent)
897{
898 xmlNode *child = NULL;
899 xmlAttrPtr pIter = NULL;
900 xmlNode *nvpair_list = NULL;
901 GHashTable *nvpair_hash = pcmk__strkey_table(free, free);
902
903 CRM_CHECK(parent != NULL, return nvpair_hash);
904
905 nvpair_list = find_xml_node(parent, XML_TAG_ATTRS, FALSE);
906 if (nvpair_list == NULL) {
907 crm_trace("No attributes in %s", crm_element_name(parent));
908 crm_log_xml_trace(parent, "No attributes for resource op");
909 }
910
911 crm_log_xml_trace(nvpair_list, "Unpacking");
912
913 for (pIter = pcmk__xe_first_attr(nvpair_list); pIter != NULL;
914 pIter = pIter->next) {
915
916 const char *p_name = (const char *)pIter->name;
917 const char *p_value = pcmk__xml_attr_value(pIter);
918
919 crm_trace("Added %s=%s", p_name, p_value);
920
921 g_hash_table_insert(nvpair_hash, strdup(p_name), strdup(p_value));
922 }
923
924 for (child = pcmk__xml_first_child(nvpair_list); child != NULL;
925 child = pcmk__xml_next(child)) {
926
927 if (strcmp((const char *)child->name, XML_TAG_PARAM) == 0) {
928 const char *key = crm_element_value(child, XML_NVPAIR_ATTR_NAME);
929 const char *value = crm_element_value(child, XML_NVPAIR_ATTR_VALUE);
930
931 crm_trace("Added %s=%s", key, value);
932 if (key != NULL && value != NULL) {
933 g_hash_table_insert(nvpair_hash, strdup(key), strdup(value));
934 }
935 }
936 }
937
938 return nvpair_hash;
939}
940
941void
942pcmk__xe_set_bool_attr(xmlNodePtr node, const char *name, bool value)
943{
945}
946
947int
948pcmk__xe_get_bool_attr(const xmlNode *node, const char *name, bool *value)
949{
950 const char *xml_value = NULL;
951 int ret, rc;
952
953 if (node == NULL) {
954 return ENODATA;
955 } else if (name == NULL || value == NULL) {
956 return EINVAL;
957 }
958
959 xml_value = crm_element_value(node, name);
960
961 if (xml_value == NULL) {
962 return ENODATA;
963 }
964
965 rc = crm_str_to_boolean(xml_value, &ret);
966 if (rc == 1) {
967 *value = ret;
968 return pcmk_rc_ok;
969 } else {
971 }
972}
973
974bool
975pcmk__xe_attr_is_true(const xmlNode *node, const char *name)
976{
977 bool value = false;
978 int rc;
979
980 rc = pcmk__xe_get_bool_attr(node, name, &value);
981 return rc == pcmk_rc_ok && value == true;
982}
983
984// Deprecated functions kept only for backward API compatibility
985// LCOV_EXCL_START
986
988
989int
990pcmk_scan_nvpair(const char *input, char **name, char **value)
991{
992 return pcmk__scan_nvpair(input, name, value);
993}
994
995char *
996pcmk_format_nvpair(const char *name, const char *value,
997 const char *units)
998{
999 return pcmk__format_nvpair(name, value, units);
1000}
1001
1002char *
1003pcmk_format_named_time(const char *name, time_t epoch_time)
1004{
1005 return pcmk__format_named_time(name, epoch_time);
1006}
1007
1008// LCOV_EXCL_STOP
1009// End deprecated API
bool pcmk__check_acl(xmlNode *xml, const char *name, enum xml_private_flags mode)
Definition: acl.c:655
const char * parent
Definition: cib.c:25
const char * name
Definition: cib.c:24
char * crm_meta_name(const char *field)
Definition: utils.c:468
char * crm_strdup_printf(char const *format,...) G_GNUC_PRINTF(1
int crm_str_to_boolean(const char *s, int *ret)
Definition: strings.c:427
Deprecated Pacemaker utilities.
char data[0]
Definition: cpg.c:10
A dumping ground.
G_GNUC_INTERNAL bool pcmk__tracking_xml_changes(xmlNode *xml, bool lazy)
Definition: xml.c:49
G_GNUC_INTERNAL void pcmk__mark_xml_attr_dirty(xmlAttr *a)
Definition: xml.c:163
const char * pcmk__epoch2str(const time_t *when)
Definition: iso8601.c:1730
#define CRM_LOG_ASSERT(expr)
Definition: logging.h:211
#define CRM_CHECK(expr, failure_action)
Definition: logging.h:227
#define crm_err(fmt, args...)
Definition: logging.h:359
#define crm_log_xml_trace(xml, text)
Definition: logging.h:373
#define crm_trace(fmt, args...)
Definition: logging.h:365
#define ID(x)
Definition: msg_xml.h:468
#define XML_BOOLEAN_TRUE
Definition: msg_xml.h:146
#define XML_NVPAIR_ATTR_VALUE
Definition: msg_xml.h:392
#define XML_ATTR_ID
Definition: msg_xml.h:134
#define XML_TAG_ATTRS
Definition: msg_xml.h:211
#define XML_BOOLEAN_FALSE
Definition: msg_xml.h:147
#define XML_NVPAIR_ATTR_NAME
Definition: msg_xml.h:391
#define XML_CIB_TAG_NVPAIR
Definition: msg_xml.h:206
#define XML_TAG_PARAM
Definition: msg_xml.h:213
xmlNode * input
int pcmk__xe_get_bool_attr(const xmlNode *node, const char *name, bool *value)
Definition: nvpair.c:948
GHashTable * xml2list(const xmlNode *parent)
Retrieve XML attributes as a hash table.
Definition: nvpair.c:896
const char * crm_xml_add_ll(xmlNode *xml, const char *name, long long value)
Create an XML attribute with specified name and long long int value.
Definition: nvpair.c:469
int crm_element_value_timeval(const xmlNode *xml, const char *name_sec, const char *name_usec, struct timeval *dest)
Retrieve the value of XML second/microsecond attributes as time.
Definition: nvpair.c:667
const char * crm_element_value(const xmlNode *data, const char *name)
Retrieve the value of an XML attribute.
Definition: nvpair.c:517
const char * crm_xml_add_timeval(xmlNode *xml, const char *name_sec, const char *name_usec, const struct timeval *value)
Create XML attributes for seconds and microseconds.
Definition: nvpair.c:493
void hash2nvpair(gpointer key, gpointer value, gpointer user_data)
Add XML nvpair element based on hash table entry.
Definition: nvpair.c:871
char * pcmk_format_nvpair(const char *name, const char *value, const char *units)
Definition: nvpair.c:996
int crm_element_value_int(const xmlNode *data, const char *name, int *dest)
Retrieve the integer value of an XML attribute.
Definition: nvpair.c:553
GSList * pcmk_prepend_nvpair(GSList *nvpairs, const char *name, const char *value)
Prepend a name/value pair to a list.
Definition: nvpair.c:91
xmlNode * crm_create_nvpair_xml(xmlNode *parent, const char *id, const char *name, const char *value)
Create an XML name/value pair.
Definition: nvpair.c:833
void hash2metafield(gpointer key, gpointer value, gpointer user_data)
Set XML attribute based on hash table entry, as meta-attribute name.
Definition: nvpair.c:798
bool pcmk__xe_attr_is_true(const xmlNode *node, const char *name)
Definition: nvpair.c:975
GSList * pcmk_sort_nvpairs(GSList *list)
Sort a list of name/value pairs.
Definition: nvpair.c:146
void hash2field(gpointer key, gpointer value, gpointer user_data)
Set XML attribute based on hash table entry.
Definition: nvpair.c:770
const char * crm_xml_add_int(xmlNode *node, const char *name, int value)
Create an XML attribute with specified name and integer value.
Definition: nvpair.c:419
const char * crm_xml_replace(xmlNode *node, const char *name, const char *value)
Replace an XML attribute with specified name and (possibly NULL) value.
Definition: nvpair.c:368
int pcmk_scan_nvpair(const char *input, char **name, char **value)
Definition: nvpair.c:990
int pcmk__scan_nvpair(const char *input, char **name, char **value)
Definition: nvpair.c:221
char * pcmk__format_nvpair(const char *name, const char *value, const char *units)
Definition: nvpair.c:284
void pcmk_nvpairs2xml_attrs(GSList *list, xmlNode *xml)
Add XML attributes based on a list of name/value pairs.
Definition: nvpair.c:201
void pcmk_free_nvpairs(GSList *nvpairs)
Free a list of name/value pairs.
Definition: nvpair.c:102
int crm_element_value_ms(const xmlNode *data, const char *name, guint *dest)
Retrieve the millisecond value of an XML attribute.
Definition: nvpair.c:610
char * crm_element_value_copy(const xmlNode *data, const char *name)
Retrieve a copy of the value of an XML attribute.
Definition: nvpair.c:714
void hash2smartfield(gpointer key, gpointer value, gpointer user_data)
Add hash table entry to XML as (possibly legacy) name/value.
Definition: nvpair.c:736
int crm_element_value_ll(const xmlNode *data, const char *name, long long *dest)
Retrieve the long long integer value of an XML attribute.
Definition: nvpair.c:585
char * pcmk__format_named_time(const char *name, time_t epoch_time)
Definition: nvpair.c:303
void pcmk__xe_set_bool_attr(xmlNodePtr node, const char *name, bool value)
Definition: nvpair.c:942
char * pcmk_format_named_time(const char *name, time_t epoch_time)
Definition: nvpair.c:1003
GSList * pcmk_xml_attrs2nvpairs(const xmlNode *xml)
Create a list of name/value pairs from an XML node's attributes.
Definition: nvpair.c:161
int crm_element_value_epoch(const xmlNode *xml, const char *name, time_t *dest)
Retrieve the seconds-since-epoch value of an XML attribute.
Definition: nvpair.c:638
#define LLSTRSIZE
Definition: nvpair.c:452
const char * crm_xml_add(xmlNode *node, const char *name, const char *value)
Create an XML attribute with specified name and value.
Definition: nvpair.c:323
const char * crm_xml_add_ms(xmlNode *node, const char *name, guint ms)
Create an XML attribute with specified name and unsigned value.
Definition: nvpair.c:441
pcmk__action_result_t result
Definition: pcmk_fence.c:35
#define ENODATA
Definition: portability.h:145
char * strndup(const char *str, size_t len)
#define CRM_ASSERT(expr)
Definition: results.h:42
@ pcmk_rc_ok
Definition: results.h:148
@ pcmk_rc_unknown_format
Definition: results.h:142
#define pcmk_err_bad_nvpair
Definition: results.h:91
#define pcmk_ok
Definition: results.h:68
GHashTable * pcmk__strkey_table(GDestroyNotify key_destroy_func, GDestroyNotify value_destroy_func)
Definition: strings.c:611
int pcmk__scan_ll(const char *text, long long *result, long long default_value)
Definition: strings.c:97
void pcmk__str_update(char **str, const char *value)
Definition: strings.c:1190
#define PCMK__PARSE_INT_DEFAULT
char * value
Definition: nvpair.h:31
char * name
Definition: nvpair.h:30
Wrappers for and extensions to libxml2.
const xmlChar * pcmkXmlStr
Definition: xml.h:50
void crm_xml_set_id(xmlNode *xml, const char *format,...) G_GNUC_PRINTF(2
xmlNode * find_xml_node(const xmlNode *root, const char *search_path, gboolean must_find)
Definition: xml.c:470
xmlNode * create_xml_node(xmlNode *parent, const char *name)
Definition: xml.c:749
void xml_remove_prop(xmlNode *obj, const char *name)
Definition: xml.c:2145
@ pcmk__xf_acl_create
Definition: xml_internal.h:337
@ pcmk__xf_acl_write
Definition: xml_internal.h:334