pacemaker 2.1.6-6fdc9deea29
Scalable High-Availability cluster resource manager
Loading...
Searching...
No Matches
output_xml.c
Go to the documentation of this file.
1/*
2 * Copyright 2019-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 <ctype.h>
13#include <stdarg.h>
14#include <stdlib.h>
15#include <stdio.h>
16#include <glib.h>
17
19#include <crm/common/xml.h>
20
21static gboolean legacy_xml = FALSE;
22static gboolean simple_list = FALSE;
23static gboolean substitute = FALSE;
24
25GOptionEntry pcmk__xml_output_entries[] = {
26 { "xml-legacy", 0, G_OPTION_FLAG_HIDDEN, G_OPTION_ARG_NONE, &legacy_xml,
27 NULL,
28 NULL },
29 { "xml-simple-list", 0, G_OPTION_FLAG_HIDDEN, G_OPTION_ARG_NONE, &simple_list,
30 NULL,
31 NULL },
32 { "xml-substitute", 0, G_OPTION_FLAG_HIDDEN, G_OPTION_ARG_NONE, &substitute,
33 NULL,
34 NULL },
35
36 { NULL }
37};
38
39typedef struct subst_s {
40 const char *from;
41 const char *to;
43
44static subst_t substitutions[] = {
45 { "Active Resources", "resources" },
46 { "Allocation Scores", "allocations" },
47 { "Allocation Scores and Utilization Information", "allocations_utilizations" },
48 { "Cluster Summary", "summary" },
49 { "Current cluster status", "cluster_status" },
50 { "Executing Cluster Transition", "transition" },
51 { "Failed Resource Actions", "failures" },
52 { "Fencing History", "fence_history" },
53 { "Full List of Resources", "resources" },
54 { "Inactive Resources", "resources" },
55 { "Migration Summary", "node_history" },
56 { "Negative Location Constraints", "bans" },
57 { "Node Attributes", "node_attributes" },
58 { "Operations", "node_history" },
59 { "Resource Config", "resource_config" },
60 { "Resource Operations", "operations" },
61 { "Revised Cluster Status", "revised_cluster_status" },
62 { "Transition Summary", "actions" },
63 { "Utilization Information", "utilizations" },
64
65 { NULL, NULL }
66};
67
68/* The first several elements of this struct must be the same as the first
69 * several elements of private_data_s in lib/common/output_html.c. That
70 * struct gets passed to a bunch of the pcmk__output_xml_* functions which
71 * assume an XML private_data_s. Keeping them laid out the same means this
72 * still works.
73 */
74typedef struct private_data_s {
75 /* Begin members that must match the HTML version */
76 xmlNode *root;
77 GQueue *parent_q;
78 GSList *errors;
79 /* End members that must match the HTML version */
80 bool legacy_xml;
82
83static void
84xml_free_priv(pcmk__output_t *out) {
85 private_data_t *priv = NULL;
86
87 if (out == NULL || out->priv == NULL) {
88 return;
89 }
90
91 priv = out->priv;
92
93 free_xml(priv->root);
94 g_queue_free(priv->parent_q);
95 g_slist_free(priv->errors);
96 free(priv);
97 out->priv = NULL;
98}
99
100static bool
101xml_init(pcmk__output_t *out) {
102 private_data_t *priv = NULL;
103
104 CRM_ASSERT(out != NULL);
105
106 /* If xml_init was previously called on this output struct, just return. */
107 if (out->priv != NULL) {
108 return true;
109 } else {
110 out->priv = calloc(1, sizeof(private_data_t));
111 if (out->priv == NULL) {
112 return false;
113 }
114
115 priv = out->priv;
116 }
117
118 if (legacy_xml) {
119 priv->root = create_xml_node(NULL, "crm_mon");
120 crm_xml_add(priv->root, "version", PACEMAKER_VERSION);
121 } else {
122 priv->root = create_xml_node(NULL, "pacemaker-result");
123 crm_xml_add(priv->root, "api-version", PCMK__API_VERSION);
124
125 if (out->request != NULL) {
126 crm_xml_add(priv->root, "request", out->request);
127 }
128 }
129
130 priv->parent_q = g_queue_new();
131 priv->errors = NULL;
132 g_queue_push_tail(priv->parent_q, priv->root);
133
134 /* Copy this from the file-level variable. This means that it is only settable
135 * as a command line option, and that pcmk__output_new must be called after all
136 * command line processing is completed.
137 */
138 priv->legacy_xml = legacy_xml;
139
140 return true;
141}
142
143static void
144add_error_node(gpointer data, gpointer user_data) {
145 char *str = (char *) data;
146 xmlNodePtr node = (xmlNodePtr) user_data;
147 pcmk_create_xml_text_node(node, "error", str);
148}
149
150static void
151xml_finish(pcmk__output_t *out, crm_exit_t exit_status, bool print, void **copy_dest) {
152 private_data_t *priv = NULL;
153 xmlNodePtr node;
154
155 CRM_ASSERT(out != NULL);
156 priv = out->priv;
157
158 /* If root is NULL, xml_init failed and we are being called from pcmk__output_free
159 * in the pcmk__output_new path.
160 */
161 if (priv == NULL || priv->root == NULL) {
162 return;
163 }
164
165 if (legacy_xml) {
166 GSList *node = priv->errors;
167
168 if (exit_status != CRM_EX_OK) {
169 fprintf(stderr, "%s\n", crm_exit_str(exit_status));
170 }
171
172 while (node != NULL) {
173 fprintf(stderr, "%s\n", (char *) node->data);
174 node = node->next;
175 }
176 } else {
177 char *rc_as_str = pcmk__itoa(exit_status);
178
179 node = create_xml_node(priv->root, "status");
180 pcmk__xe_set_props(node, "code", rc_as_str,
181 "message", crm_exit_str(exit_status),
182 NULL);
183
184 if (g_slist_length(priv->errors) > 0) {
185 xmlNodePtr errors_node = create_xml_node(node, "errors");
186 g_slist_foreach(priv->errors, add_error_node, (gpointer) errors_node);
187 }
188
189 free(rc_as_str);
190 }
191
192 if (print) {
193 char *buf = dump_xml_formatted_with_text(priv->root);
194 fprintf(out->dest, "%s", buf);
195 fflush(out->dest);
196 free(buf);
197 }
198
199 if (copy_dest != NULL) {
200 *copy_dest = copy_xml(priv->root);
201 }
202}
203
204static void
205xml_reset(pcmk__output_t *out) {
206 CRM_ASSERT(out != NULL);
207
208 out->dest = freopen(NULL, "w", out->dest);
209 CRM_ASSERT(out->dest != NULL);
210
211 xml_free_priv(out);
212 xml_init(out);
213}
214
215static void
216xml_subprocess_output(pcmk__output_t *out, int exit_status,
217 const char *proc_stdout, const char *proc_stderr) {
218 xmlNodePtr node, child_node;
219 char *rc_as_str = NULL;
220
221 CRM_ASSERT(out != NULL);
222
223 rc_as_str = pcmk__itoa(exit_status);
224
225 node = pcmk__output_xml_create_parent(out, "command",
226 "code", rc_as_str,
227 NULL);
228
229 if (proc_stdout != NULL) {
230 child_node = pcmk_create_xml_text_node(node, "output", proc_stdout);
231 crm_xml_add(child_node, "source", "stdout");
232 }
233
234 if (proc_stderr != NULL) {
235 child_node = pcmk_create_xml_text_node(node, "output", proc_stderr);
236 crm_xml_add(child_node, "source", "stderr");
237 }
238
239 free(rc_as_str);
240}
241
242static void
243xml_version(pcmk__output_t *out, bool extended) {
244 CRM_ASSERT(out != NULL);
245
246 pcmk__output_create_xml_node(out, "version",
247 "program", "Pacemaker",
248 "version", PACEMAKER_VERSION,
249 "author", "Andrew Beekhof and the "
250 "Pacemaker project contributors",
251 "build", BUILD_VERSION,
252 "features", CRM_FEATURES,
253 NULL);
254}
255
256G_GNUC_PRINTF(2, 3)
257static void
258xml_err(pcmk__output_t *out, const char *format, ...) {
259 private_data_t *priv = NULL;
260 int len = 0;
261 char *buf = NULL;
262 va_list ap;
263
264 CRM_ASSERT(out != NULL && out->priv != NULL);
265 priv = out->priv;
266
267 va_start(ap, format);
268 len = vasprintf(&buf, format, ap);
269 CRM_ASSERT(len > 0);
270 va_end(ap);
271
272 priv->errors = g_slist_append(priv->errors, buf);
273}
274
275G_GNUC_PRINTF(2, 3)
276static int
277xml_info(pcmk__output_t *out, const char *format, ...) {
278 return pcmk_rc_no_output;
279}
280
281static void
282xml_output_xml(pcmk__output_t *out, const char *name, const char *buf) {
283 xmlNodePtr parent = NULL;
284 xmlNodePtr cdata_node = NULL;
285
286 CRM_ASSERT(out != NULL);
287
289 cdata_node = xmlNewCDataBlock(getDocPtr(parent), (pcmkXmlStr) buf, strlen(buf));
290 xmlAddChild(parent, cdata_node);
291}
292
293G_GNUC_PRINTF(4, 5)
294static void
295xml_begin_list(pcmk__output_t *out, const char *singular_noun, const char *plural_noun,
296 const char *format, ...) {
297 va_list ap;
298 char *name = NULL;
299 char *buf = NULL;
300 int len;
301
302 CRM_ASSERT(out != NULL);
303
304 va_start(ap, format);
305 len = vasprintf(&buf, format, ap);
306 CRM_ASSERT(len >= 0);
307 va_end(ap);
308
309 if (substitute) {
310 for (subst_t *s = substitutions; s->from != NULL; s++) {
311 if (!strcmp(s->from, buf)) {
312 name = g_strdup(s->to);
313 break;
314 }
315 }
316 }
317
318 if (name == NULL) {
319 name = g_ascii_strdown(buf, -1);
320 }
321
322 if (legacy_xml || simple_list) {
324 } else {
326 "name", name,
327 NULL);
328 }
329
330 g_free(name);
331 free(buf);
332}
333
334G_GNUC_PRINTF(3, 4)
335static void
336xml_list_item(pcmk__output_t *out, const char *name, const char *format, ...) {
337 xmlNodePtr item_node = NULL;
338 va_list ap;
339 char *buf = NULL;
340 int len;
341
342 CRM_ASSERT(out != NULL);
343
344 va_start(ap, format);
345 len = vasprintf(&buf, format, ap);
346 CRM_ASSERT(len >= 0);
347 va_end(ap);
348
349 item_node = pcmk__output_create_xml_text_node(out, "item", buf);
350
351 if (name != NULL) {
352 crm_xml_add(item_node, "name", name);
353 }
354
355 free(buf);
356}
357
358static void
359xml_increment_list(pcmk__output_t *out) {
360 /* This function intentially left blank */
361}
362
363static void
364xml_end_list(pcmk__output_t *out) {
365 private_data_t *priv = NULL;
366
367 CRM_ASSERT(out != NULL && out->priv != NULL);
368 priv = out->priv;
369
370 if (priv->legacy_xml || simple_list) {
371 g_queue_pop_tail(priv->parent_q);
372 } else {
373 char *buf = NULL;
374 xmlNodePtr node;
375
376 node = g_queue_pop_tail(priv->parent_q);
377 buf = crm_strdup_printf("%lu", xmlChildElementCount(node));
378 crm_xml_add(node, "count", buf);
379 free(buf);
380 }
381}
382
383static bool
384xml_is_quiet(pcmk__output_t *out) {
385 return false;
386}
387
388static void
389xml_spacer(pcmk__output_t *out) {
390 /* This function intentionally left blank */
391}
392
393static void
394xml_progress(pcmk__output_t *out, bool end) {
395 /* This function intentionally left blank */
396}
397
400 pcmk__output_t *retval = calloc(1, sizeof(pcmk__output_t));
401
402 if (retval == NULL) {
403 return NULL;
404 }
405
406 retval->fmt_name = "xml";
407 retval->request = pcmk__quote_cmdline(argv);
408
409 retval->init = xml_init;
410 retval->free_priv = xml_free_priv;
411 retval->finish = xml_finish;
412 retval->reset = xml_reset;
413
415 retval->message = pcmk__call_message;
416
417 retval->subprocess_output = xml_subprocess_output;
418 retval->version = xml_version;
419 retval->info = xml_info;
420 retval->transient = xml_info;
421 retval->err = xml_err;
422 retval->output_xml = xml_output_xml;
423
424 retval->begin_list = xml_begin_list;
425 retval->list_item = xml_list_item;
426 retval->increment_list = xml_increment_list;
427 retval->end_list = xml_end_list;
428
429 retval->is_quiet = xml_is_quiet;
430 retval->spacer = xml_spacer;
431 retval->progress = xml_progress;
432 retval->prompt = pcmk__text_prompt;
433
434 return retval;
435}
436
437xmlNodePtr
439 va_list args;
440 xmlNodePtr node = NULL;
441
442 CRM_ASSERT(out != NULL);
443 CRM_CHECK(pcmk__str_any_of(out->fmt_name, "xml", "html", NULL), return NULL);
444
445 node = pcmk__output_create_xml_node(out, name, NULL);
446
447 va_start(args, name);
448 pcmk__xe_set_propv(node, args);
449 va_end(args);
450
452 return node;
453}
454
455void
457 private_data_t *priv = NULL;
458 xmlNodePtr parent = NULL;
459
460 CRM_ASSERT(out != NULL && out->priv != NULL);
461 CRM_ASSERT(node != NULL);
462 CRM_CHECK(pcmk__str_any_of(out->fmt_name, "xml", "html", NULL), return);
463
464 priv = out->priv;
465 parent = g_queue_peek_tail(priv->parent_q);
466
467 // Shouldn't happen unless the caller popped priv->root
468 CRM_CHECK(parent != NULL, return);
469
470 add_node_copy(parent, node);
471}
472
473xmlNodePtr
475 xmlNodePtr node = NULL;
476 private_data_t *priv = NULL;
477 va_list args;
478
479 CRM_ASSERT(out != NULL && out->priv != NULL);
480 CRM_CHECK(pcmk__str_any_of(out->fmt_name, "xml", "html", NULL), return NULL);
481
482 priv = out->priv;
483
484 node = create_xml_node(g_queue_peek_tail(priv->parent_q), name);
485 va_start(args, name);
486 pcmk__xe_set_propv(node, args);
487 va_end(args);
488
489 return node;
490}
491
492xmlNodePtr
493pcmk__output_create_xml_text_node(pcmk__output_t *out, const char *name, const char *content) {
494 xmlNodePtr node = NULL;
495
496 CRM_ASSERT(out != NULL);
497 CRM_CHECK(pcmk__str_any_of(out->fmt_name, "xml", "html", NULL), return NULL);
498
499 node = pcmk__output_create_xml_node(out, name, NULL);
500 xmlNodeSetContent(node, (pcmkXmlStr) content);
501 return node;
502}
503
504void
506 private_data_t *priv = NULL;
507
508 CRM_ASSERT(out != NULL && out->priv != NULL);
509 CRM_ASSERT(parent != NULL);
510 CRM_CHECK(pcmk__str_any_of(out->fmt_name, "xml", "html", NULL), return);
511
512 priv = out->priv;
513
514 g_queue_push_tail(priv->parent_q, parent);
515}
516
517void
519 private_data_t *priv = NULL;
520
521 CRM_ASSERT(out != NULL && out->priv != NULL);
522 CRM_CHECK(pcmk__str_any_of(out->fmt_name, "xml", "html", NULL), return);
523
524 priv = out->priv;
525
526 CRM_ASSERT(g_queue_get_length(priv->parent_q) > 0);
527 g_queue_pop_tail(priv->parent_q);
528}
529
530xmlNodePtr
532 private_data_t *priv = NULL;
533
534 CRM_ASSERT(out != NULL && out->priv != NULL);
535 CRM_CHECK(pcmk__str_any_of(out->fmt_name, "xml", "html", NULL), return NULL);
536
537 priv = out->priv;
538
539 /* If queue is empty NULL will be returned */
540 return g_queue_peek_tail(priv->parent_q);
541}
const char * parent
Definition cib.c:25
const char * name
Definition cib.c:24
gchar * pcmk__quote_cmdline(gchar **argv)
Definition cmdline.c:163
char * crm_strdup_printf(char const *format,...) G_GNUC_PRINTF(1
#define PACEMAKER_VERSION
Definition config.h:499
#define CRM_FEATURES
Definition config.h:33
#define PCMK__API_VERSION
Definition config.h:529
#define BUILD_VERSION
Definition config.h:8
char data[0]
Definition cpg.c:10
#define CRM_CHECK(expr, failure_action)
Definition logging.h:235
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:302
struct private_data_s private_data_t
void void void void void pcmk__text_prompt(const char *prompt, bool echo, char **dest)
int pcmk__call_message(pcmk__output_t *out, const char *message_id,...)
Definition output.c:166
void pcmk__register_message(pcmk__output_t *out, const char *message_id, pcmk__message_fn_t fn)
Definition output.c:188
GOptionEntry pcmk__xml_output_entries[]
Definition output_xml.c:25
void pcmk__output_xml_add_node_copy(pcmk__output_t *out, xmlNodePtr node)
Definition output_xml.c:456
void pcmk__output_xml_pop_parent(pcmk__output_t *out)
Definition output_xml.c:518
void pcmk__output_xml_push_parent(pcmk__output_t *out, xmlNodePtr parent)
Definition output_xml.c:505
xmlNodePtr pcmk__output_xml_peek_parent(pcmk__output_t *out)
Definition output_xml.c:531
xmlNodePtr pcmk__output_create_xml_node(pcmk__output_t *out, const char *name,...)
Definition output_xml.c:474
xmlNodePtr pcmk__output_xml_create_parent(pcmk__output_t *out, const char *name,...)
Definition output_xml.c:438
struct subst_s subst_t
struct private_data_s private_data_t
pcmk__output_t * pcmk__mk_xml_output(char **argv)
Definition output_xml.c:399
xmlNodePtr pcmk__output_create_xml_text_node(pcmk__output_t *out, const char *name, const char *content)
Definition output_xml.c:493
#define CRM_ASSERT(expr)
Definition results.h:42
@ CRM_EX_OK
Success.
Definition results.h:237
@ pcmk_rc_no_output
Definition results.h:121
const char * crm_exit_str(crm_exit_t exit_code)
Definition results.c:627
enum crm_exit_e crm_exit_t
bool pcmk__str_any_of(const char *s,...) G_GNUC_NULL_TERMINATED
Definition strings.c:957
This structure contains everything that makes up a single output formatter.
void(*) void(*) void(* increment_list)(pcmk__output_t *out)
void(* end_list)(pcmk__output_t *out)
void(* version)(pcmk__output_t *out, bool extended)
int(* message)(pcmk__output_t *out, const char *message_id,...)
bool(* is_quiet)(pcmk__output_t *out)
const char * fmt_name
The name of this output formatter.
FILE * dest
Where output should be written.
void(* register_message)(pcmk__output_t *out, const char *message_id, pcmk__message_fn_t fn)
int(*) int(* transient)(pcmk__output_t *out, const char *format,...) G_GNUC_PRINTF(2
void(*) void(* list_item)(pcmk__output_t *out, const char *name, const char *format,...) G_GNUC_PRINTF(3
void(* finish)(pcmk__output_t *out, crm_exit_t exit_status, bool print, void **copy_dest)
void(* prompt)(const char *prompt, bool echo, char **dest)
int(*) int(*) void(*) void(* output_xml)(pcmk__output_t *out, const char *name, const char *buf)
void(* subprocess_output)(pcmk__output_t *out, int exit_status, const char *proc_stdout, const char *proc_stderr)
void(* begin_list)(pcmk__output_t *out, const char *singular_noun, const char *plural_noun, const char *format,...) G_GNUC_PRINTF(4
bool(* init)(pcmk__output_t *out)
void * priv
Implementation-specific private data.
void(* spacer)(pcmk__output_t *out)
void(* progress)(pcmk__output_t *out, bool end)
void(* reset)(pcmk__output_t *out)
int(* info)(pcmk__output_t *out, const char *format,...) G_GNUC_PRINTF(2
void(* free_priv)(pcmk__output_t *out)
gchar * request
A copy of the request that generated this output.
int(*) int(*) void(* err)(pcmk__output_t *out, const char *format,...) G_GNUC_PRINTF(2
Wrappers for and extensions to libxml2.
xmlNode * pcmk_create_xml_text_node(xmlNode *parent, const char *name, const char *content)
Definition xml.c:702
const xmlChar * pcmkXmlStr
Definition xml.h:50
char * dump_xml_formatted_with_text(xmlNode *msg)
Definition xml.c:1686
xmlDoc * getDocPtr(xmlNode *node)
Definition xml.c:647
void free_xml(xmlNode *child)
Definition xml.c:813
xmlNode * add_node_copy(xmlNode *new_parent, xmlNode *xml_node)
Definition xml.c:663
xmlNode * copy_xml(xmlNode *src_node)
Definition xml.c:819
xmlNode * create_xml_node(xmlNode *parent, const char *name)
Definition xml.c:677
void pcmk__xe_set_propv(xmlNodePtr node, va_list pairs)
Definition xml.c:2677
void pcmk__xe_set_props(xmlNodePtr node,...) G_GNUC_NULL_TERMINATED
Definition xml.c:2695