root/lib/common/output.c

/* [previous][next][first][last][top][bottom][index][help] */

DEFINITIONS

This source file includes following definitions.
  1. pcmk__output_formatters
  2. pcmk__output_free
  3. pcmk__bare_output_new
  4. pcmk__output_new
  5. pcmk__register_format
  6. pcmk__register_formats
  7. pcmk__unregister_formats
  8. pcmk__call_message
  9. pcmk__register_message
  10. pcmk__register_messages
  11. pcmk__output_and_clear_error
  12. pcmk__xml_output_new
  13. pcmk__xml_output_finish
  14. pcmk__log_output_new
  15. pcmk__text_output_new

   1 /*
   2  * Copyright 2019-2024 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 <crm/common/util.h>
  13 #include <crm/common/xml.h>
  14 #include <libxml/tree.h>
  15 
  16 #include "crmcommon_private.h"
  17 
  18 static GHashTable *formatters = NULL;
  19 
  20 #if defined(PCMK__UNIT_TESTING)
  21 // LCOV_EXCL_START
  22 GHashTable *
  23 pcmk__output_formatters(void) {
     /* [previous][next][first][last][top][bottom][index][help] */
  24     return formatters;
  25 }
  26 // LCOV_EXCL_STOP
  27 #endif
  28 
  29 void
  30 pcmk__output_free(pcmk__output_t *out) {
     /* [previous][next][first][last][top][bottom][index][help] */
  31     if (out == NULL) {
  32         return;
  33     }
  34 
  35     out->free_priv(out);
  36 
  37     if (out->messages != NULL) {
  38         g_hash_table_destroy(out->messages);
  39     }
  40 
  41     g_free(out->request);
  42     free(out);
  43 }
  44 
  45 /*!
  46  * \internal
  47  * \brief Create a new \p pcmk__output_t structure
  48  *
  49  * This function does not register any message functions with the newly created
  50  * object.
  51  *
  52  * \param[in,out] out       Where to store the new output object
  53  * \param[in]     fmt_name  How to format output
  54  * \param[in]     filename  Where to write formatted output. This can be a
  55  *                          filename (the file will be overwritten if it already
  56  *                          exists), or \p NULL or \p "-" for stdout. For no
  57  *                          output, pass a filename of \p "/dev/null".
  58  * \param[in]     argv      List of command line arguments
  59  *
  60  * \return Standard Pacemaker return code
  61  */
  62 int
  63 pcmk__bare_output_new(pcmk__output_t **out, const char *fmt_name,
     /* [previous][next][first][last][top][bottom][index][help] */
  64                       const char *filename, char **argv)
  65 {
  66     pcmk__output_factory_t create = NULL;
  67 
  68     pcmk__assert((formatters != NULL) && (out != NULL));
  69 
  70     /* If no name was given, just try "text".  It's up to each tool to register
  71      * what it supports so this also may not be valid.
  72      */
  73     if (fmt_name == NULL) {
  74         create = g_hash_table_lookup(formatters, "text");
  75     } else {
  76         create = g_hash_table_lookup(formatters, fmt_name);
  77     }
  78 
  79     if (create == NULL) {
  80         return pcmk_rc_unknown_format;
  81     }
  82 
  83     *out = create(argv);
  84     if (*out == NULL) {
  85         return ENOMEM;
  86     }
  87 
  88     if (pcmk__str_eq(filename, "-", pcmk__str_null_matches)) {
  89         (*out)->dest = stdout;
  90     } else {
  91         (*out)->dest = fopen(filename, "w");
  92         if ((*out)->dest == NULL) {
  93             pcmk__output_free(*out);
  94             *out = NULL;
  95             return errno;
  96         }
  97     }
  98 
  99     (*out)->quiet = false;
 100     (*out)->messages = pcmk__strkey_table(free, NULL);
 101 
 102     if ((*out)->init(*out) == false) {
 103         pcmk__output_free(*out);
 104         return ENOMEM;
 105     }
 106 
 107     setenv("OCF_OUTPUT_FORMAT", (*out)->fmt_name, 1);
 108 
 109     return pcmk_rc_ok;
 110 }
 111 
 112 int
 113 pcmk__output_new(pcmk__output_t **out, const char *fmt_name,
     /* [previous][next][first][last][top][bottom][index][help] */
 114                  const char *filename, char **argv)
 115 {
 116     int rc = pcmk__bare_output_new(out, fmt_name, filename, argv);
 117 
 118     if (rc == pcmk_rc_ok) {
 119         // Register libcrmcommon messages
 120         pcmk__register_option_messages(*out);
 121         pcmk__register_patchset_messages(*out);
 122     }
 123     return rc;
 124 }
 125 
 126 int
 127 pcmk__register_format(GOptionGroup *group, const char *name,
     /* [previous][next][first][last][top][bottom][index][help] */
 128                       pcmk__output_factory_t create,
 129                       const GOptionEntry *options)
 130 {
 131     char *name_copy = NULL;
 132 
 133     pcmk__assert((create != NULL) && !pcmk__str_empty(name));
 134 
 135     name_copy = strdup(name);
 136     if (name_copy == NULL) {
 137         return ENOMEM;
 138     }
 139 
 140     if (formatters == NULL) {
 141         formatters = pcmk__strkey_table(free, NULL);
 142     }
 143 
 144     if (options != NULL && group != NULL) {
 145         g_option_group_add_entries(group, options);
 146     }
 147 
 148     g_hash_table_insert(formatters, name_copy, create);
 149     return pcmk_rc_ok;
 150 }
 151 
 152 void
 153 pcmk__register_formats(GOptionGroup *group,
     /* [previous][next][first][last][top][bottom][index][help] */
 154                        const pcmk__supported_format_t *formats)
 155 {
 156     if (formats == NULL) {
 157         return;
 158     }
 159     for (const pcmk__supported_format_t *entry = formats; entry->name != NULL;
 160          entry++) {
 161         pcmk__register_format(group, entry->name, entry->create, entry->options);
 162     }
 163 }
 164 
 165 void
 166 pcmk__unregister_formats(void) {
     /* [previous][next][first][last][top][bottom][index][help] */
 167     if (formatters != NULL) {
 168         g_hash_table_destroy(formatters);
 169         formatters = NULL;
 170     }
 171 }
 172 
 173 int
 174 pcmk__call_message(pcmk__output_t *out, const char *message_id, ...) {
     /* [previous][next][first][last][top][bottom][index][help] */
 175     va_list args;
 176     int rc = pcmk_rc_ok;
 177     pcmk__message_fn_t fn;
 178 
 179     pcmk__assert((out != NULL) && !pcmk__str_empty(message_id));
 180 
 181     fn = g_hash_table_lookup(out->messages, message_id);
 182     if (fn == NULL) {
 183         crm_debug("Called unknown output message '%s' for format '%s'",
 184                   message_id, out->fmt_name);
 185         return EINVAL;
 186     }
 187 
 188     va_start(args, message_id);
 189     rc = fn(out, args);
 190     va_end(args);
 191 
 192     return rc;
 193 }
 194 
 195 void
 196 pcmk__register_message(pcmk__output_t *out, const char *message_id,
     /* [previous][next][first][last][top][bottom][index][help] */
 197                        pcmk__message_fn_t fn)
 198 {
 199     pcmk__assert((out != NULL) && !pcmk__str_empty(message_id) && (fn != NULL));
 200     g_hash_table_replace(out->messages, pcmk__str_copy(message_id), fn);
 201 }
 202 
 203 void
 204 pcmk__register_messages(pcmk__output_t *out, const pcmk__message_entry_t *table)
     /* [previous][next][first][last][top][bottom][index][help] */
 205 {
 206     for (const pcmk__message_entry_t *entry = table; entry->message_id != NULL;
 207          entry++) {
 208         if (pcmk__strcase_any_of(entry->fmt_name, "default", out->fmt_name, NULL)) {
 209             pcmk__register_message(out, entry->message_id, entry->fn);
 210         }
 211     }
 212 }
 213 
 214 void
 215 pcmk__output_and_clear_error(GError **error, pcmk__output_t *out)
     /* [previous][next][first][last][top][bottom][index][help] */
 216 {
 217     if (error == NULL || *error == NULL) {
 218         return;
 219     }
 220 
 221     if (out != NULL) {
 222         out->err(out, "%s: %s", g_get_prgname(), (*error)->message);
 223     } else {
 224         fprintf(stderr, "%s: %s\n", g_get_prgname(), (*error)->message);
 225     }
 226 
 227     g_clear_error(error);
 228 }
 229 
 230 /*!
 231  * \internal
 232  * \brief Create an XML-only output object
 233  *
 234  * Create an output object that supports only the XML format, and free
 235  * existing XML if supplied (particularly useful for libpacemaker public API
 236  * functions that want to free any previous result supplied by the caller).
 237  *
 238  * \param[out]     out  Where to put newly created output object
 239  * \param[in,out]  xml  If \c *xml is non-NULL, this will be freed
 240  *
 241  * \return Standard Pacemaker return code
 242  */
 243 int
 244 pcmk__xml_output_new(pcmk__output_t **out, xmlNodePtr *xml) {
     /* [previous][next][first][last][top][bottom][index][help] */
 245     pcmk__supported_format_t xml_format[] = {
 246         PCMK__SUPPORTED_FORMAT_XML,
 247         { NULL, NULL, NULL }
 248     };
 249 
 250     if (xml == NULL) {
 251         return EINVAL;
 252     }
 253 
 254     if (*xml != NULL) {
 255         free_xml(*xml);
 256         *xml = NULL;
 257     }
 258     pcmk__register_formats(NULL, xml_format);
 259     return pcmk__output_new(out, "xml", NULL, NULL);
 260 }
 261 
 262 /*!
 263  * \internal
 264  * \brief  Finish and free an XML-only output object
 265  *
 266  * \param[in,out] out         Output object to free
 267  * \param[in]     exit_status The exit value of the whole program
 268  * \param[out]    xml         If not NULL, where to store XML output
 269  */
 270 void
 271 pcmk__xml_output_finish(pcmk__output_t *out, crm_exit_t exit_status,
     /* [previous][next][first][last][top][bottom][index][help] */
 272                         xmlNodePtr *xml)
 273 {
 274     if (out == NULL) {
 275         return;
 276     }
 277 
 278     out->finish(out, exit_status, FALSE, (void **) xml);
 279     pcmk__output_free(out);
 280 }
 281 
 282 /*!
 283  * \internal
 284  * \brief Create a new output object using the "log" format
 285  *
 286  * \param[out] out  Where to store newly allocated output object
 287  *
 288  * \return Standard Pacemaker return code
 289  */
 290 int
 291 pcmk__log_output_new(pcmk__output_t **out)
     /* [previous][next][first][last][top][bottom][index][help] */
 292 {
 293     int rc = pcmk_rc_ok;
 294     const char* argv[] = { "", NULL };
 295     pcmk__supported_format_t formats[] = {
 296         PCMK__SUPPORTED_FORMAT_LOG,
 297         { NULL, NULL, NULL }
 298     };
 299 
 300     pcmk__register_formats(NULL, formats);
 301     rc = pcmk__output_new(out, "log", NULL, (char **) argv);
 302     if ((rc != pcmk_rc_ok) || (*out == NULL)) {
 303         crm_err("Can't log certain messages due to internal error: %s",
 304                 pcmk_rc_str(rc));
 305         return rc;
 306     }
 307     return pcmk_rc_ok;
 308 }
 309 
 310 /*!
 311  * \internal
 312  * \brief Create a new output object using the "text" format
 313  *
 314  * \param[out] out       Where to store newly allocated output object
 315  * \param[in]  filename  Name of output destination file
 316  *
 317  * \return Standard Pacemaker return code
 318  */
 319 int
 320 pcmk__text_output_new(pcmk__output_t **out, const char *filename)
     /* [previous][next][first][last][top][bottom][index][help] */
 321 {
 322     int rc = pcmk_rc_ok;
 323     const char* argv[] = { "", NULL };
 324     pcmk__supported_format_t formats[] = {
 325         PCMK__SUPPORTED_FORMAT_TEXT,
 326         { NULL, NULL, NULL }
 327     };
 328 
 329     pcmk__register_formats(NULL, formats);
 330     rc = pcmk__output_new(out, "text", filename, (char **) argv);
 331     if ((rc != pcmk_rc_ok) || (*out == NULL)) {
 332         crm_err("Can't create text output object to internal error: %s",
 333                 pcmk_rc_str(rc));
 334         return rc;
 335     }
 336     return pcmk_rc_ok;
 337 }

/* [previous][next][first][last][top][bottom][index][help] */