pacemaker 3.0.1-16e74fc4da
Scalable High-Availability cluster resource manager
Loading...
Searching...
No Matches
output_log.c
Go to the documentation of this file.
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>
12
13#include <ctype.h>
14#include <stdarg.h>
15#include <stdint.h>
16#include <stdlib.h>
17#include <stdio.h>
18
19typedef struct private_data_s {
20 /* gathered in log_begin_list */
21 GQueue/*<char*>*/ *prefixes;
22 uint8_t log_level;
23 const char *function;
24 const char *file;
25 uint32_t line;
26 uint32_t tags;
28
37#define logger(priv, fmt, args...) do { \
38 qb_log_from_external_source(pcmk__s((priv)->function, __func__), \
39 pcmk__s((priv)->file, __FILE__), fmt, (priv)->log_level, \
40 (((priv)->line == 0)? __LINE__ : (priv)->line), (priv)->tags, \
41 ##args); \
42 } while (0);
43
53#define logger_va(priv, level, fmt, ap) do { \
54 qb_log_from_external_source_va(pcmk__s((priv)->function, __func__), \
55 pcmk__s((priv)->file, __FILE__), fmt, level, \
56 (((priv)->line == 0)? __LINE__ : (priv)->line), (priv)->tags, \
57 ap); \
58 } while (0);
59
60static void
61log_subprocess_output(pcmk__output_t *out, int exit_status,
62 const char *proc_stdout, const char *proc_stderr) {
63 /* This function intentionally left blank */
64}
65
66static void
67log_free_priv(pcmk__output_t *out) {
68 private_data_t *priv = NULL;
69
70 if (out == NULL || out->priv == NULL) {
71 return;
72 }
73
74 priv = out->priv;
75
76 g_queue_free(priv->prefixes);
77 free(priv);
78 out->priv = NULL;
79}
80
81static bool
82log_init(pcmk__output_t *out) {
83 private_data_t *priv = NULL;
84
85 pcmk__assert(out != NULL);
86
87 /* If log_init was previously called on this output struct, just return. */
88 if (out->priv != NULL) {
89 return true;
90 }
91
92 out->priv = calloc(1, sizeof(private_data_t));
93 if (out->priv == NULL) {
94 return false;
95 }
96
97 priv = out->priv;
98
99 priv->prefixes = g_queue_new();
100 priv->log_level = LOG_INFO;
101
102 return true;
103}
104
105static void
106log_finish(pcmk__output_t *out, crm_exit_t exit_status, bool print, void **copy_dest) {
107 /* This function intentionally left blank */
108}
109
110static void
111log_reset(pcmk__output_t *out) {
112 pcmk__assert(out != NULL);
113
114 out->dest = freopen(NULL, "w", out->dest);
115 pcmk__assert(out->dest != NULL);
116
117 log_free_priv(out);
118 log_init(out);
119}
120
121static void
122log_version(pcmk__output_t *out, bool extended) {
123 private_data_t *priv = NULL;
124
125 pcmk__assert((out != NULL) && (out->priv != NULL));
126 priv = out->priv;
127
128 if (extended) {
129 logger(priv, "Pacemaker %s (Build: %s): %s",
131 } else {
132 logger(priv, "Pacemaker " PACEMAKER_VERSION);
133 logger(priv, "Written by Andrew Beekhof and "
134 "the Pacemaker project contributors");
135 }
136}
137
138G_GNUC_PRINTF(2, 3)
139static void
140log_err(pcmk__output_t *out, const char *format, ...)
141{
142 va_list ap;
143 private_data_t *priv = NULL;
144
145 pcmk__assert((out != NULL) && (out->priv != NULL));
146 priv = out->priv;
147
148 /* Error output does not get indented, to separate it from other
149 * potentially indented list output.
150 */
151 va_start(ap, format);
152 logger_va(priv, LOG_ERR, format, ap);
153 va_end(ap);
154}
155
156static void
157log_output_xml(pcmk__output_t *out, const char *name, const char *buf) {
158 xmlNodePtr node = NULL;
159 private_data_t *priv = NULL;
160
161 pcmk__assert((out != NULL) && (out->priv != NULL));
162 priv = out->priv;
163
164 node = pcmk__xe_create(NULL, name);
165 pcmk__xe_set_content(node, "%s", buf);
166 do_crm_log_xml(priv->log_level, name, node);
167 free(node);
168}
169
170G_GNUC_PRINTF(4, 5)
171static void
172log_begin_list(pcmk__output_t *out, const char *singular_noun, const char *plural_noun,
173 const char *format, ...) {
174 int len = 0;
175 va_list ap;
176 char* buffer = NULL;
177 private_data_t *priv = NULL;
178
179 pcmk__assert((out != NULL) && (out->priv != NULL));
180 priv = out->priv;
181
182 va_start(ap, format);
183 len = vasprintf(&buffer, format, ap);
184 pcmk__assert(len >= 0);
185 va_end(ap);
186
187 /* Don't skip empty prefixes,
188 * otherwise there will be mismatch
189 * in the log_end_list */
190 if(strcmp(buffer, "") == 0) {
191 /* nothing */
192 }
193
194 g_queue_push_tail(priv->prefixes, buffer);
195}
196
197G_GNUC_PRINTF(3, 4)
198static void
199log_list_item(pcmk__output_t *out, const char *name, const char *format, ...) {
200 int len = 0;
201 va_list ap;
202 private_data_t *priv = NULL;
203 char prefix[LINE_MAX] = { 0 };
204 int offset = 0;
205 char* buffer = NULL;
206
207 pcmk__assert((out != NULL) && (out->priv != NULL));
208 priv = out->priv;
209
210 for (GList* gIter = priv->prefixes->head; gIter; gIter = gIter->next) {
211 if (strcmp(prefix, "") != 0) {
212 offset += snprintf(prefix + offset, LINE_MAX - offset, ": %s", (char *)gIter->data);
213 } else {
214 offset = snprintf(prefix, LINE_MAX, "%s", (char *)gIter->data);
215 }
216 }
217
218 va_start(ap, format);
219 len = vasprintf(&buffer, format, ap);
220 pcmk__assert(len >= 0);
221 va_end(ap);
222
223 if (strcmp(buffer, "") != 0) { /* We don't want empty messages */
224 if ((name != NULL) && (strcmp(name, "") != 0)) {
225 if (strcmp(prefix, "") != 0) {
226 logger(priv, "%s: %s: %s", prefix, name, buffer);
227 } else {
228 logger(priv, "%s: %s", name, buffer);
229 }
230 } else {
231 if (strcmp(prefix, "") != 0) {
232 logger(priv, "%s: %s", prefix, buffer);
233 } else {
234 logger(priv, "%s", buffer);
235 }
236 }
237 }
238 free(buffer);
239}
240
241static void
242log_end_list(pcmk__output_t *out) {
243 private_data_t *priv = NULL;
244
245 pcmk__assert((out != NULL) && (out->priv != NULL));
246 priv = out->priv;
247
248 if (priv->prefixes == NULL) {
249 return;
250 }
251 pcmk__assert(priv->prefixes->tail != NULL);
252
253 free((char *)priv->prefixes->tail->data);
254 g_queue_pop_tail(priv->prefixes);
255}
256
257G_GNUC_PRINTF(2, 3)
258static int
259log_info(pcmk__output_t *out, const char *format, ...)
260{
261 va_list ap;
262 private_data_t *priv = NULL;
263
264 pcmk__assert((out != NULL) && (out->priv != NULL));
265 priv = out->priv;
266
267 /* Informational output does not get indented, to separate it from other
268 * potentially indented list output.
269 */
270 va_start(ap, format);
271 logger_va(priv, priv->log_level, format, ap);
272 va_end(ap);
273
274 return pcmk_rc_ok;
275}
276
277G_GNUC_PRINTF(2, 3)
278static int
279log_transient(pcmk__output_t *out, const char *format, ...)
280{
281 va_list ap;
282 private_data_t *priv = NULL;
283
284 pcmk__assert((out != NULL) && (out->priv != NULL));
285 priv = out->priv;
286
287 va_start(ap, format);
288 logger_va(priv, QB_MAX(priv->log_level, LOG_DEBUG), format, ap);
289 va_end(ap);
290
291 return pcmk_rc_ok;
292}
293
294static bool
295log_is_quiet(pcmk__output_t *out) {
296 return false;
297}
298
299static void
300log_spacer(pcmk__output_t *out) {
301 /* This function intentionally left blank */
302}
303
304static void
305log_progress(pcmk__output_t *out, bool end) {
306 /* This function intentionally left blank */
307}
308
309static void
310log_prompt(const char *prompt, bool echo, char **dest) {
311 /* This function intentionally left blank */
312}
313
316 pcmk__output_t *retval = calloc(1, sizeof(pcmk__output_t));
317
318 if (retval == NULL) {
319 return NULL;
320 }
321
322 retval->fmt_name = "log";
323 retval->request = pcmk__quote_cmdline(argv);
324
325 retval->init = log_init;
326 retval->free_priv = log_free_priv;
327 retval->finish = log_finish;
328 retval->reset = log_reset;
329
331 retval->message = pcmk__call_message;
332
333 retval->subprocess_output = log_subprocess_output;
334 retval->version = log_version;
335 retval->info = log_info;
336 retval->transient = log_transient;
337 retval->err = log_err;
338 retval->output_xml = log_output_xml;
339
340 retval->begin_list = log_begin_list;
341 retval->list_item = log_list_item;
342 retval->end_list = log_end_list;
343
344 retval->is_quiet = log_is_quiet;
345 retval->spacer = log_spacer;
346 retval->progress = log_progress;
347 retval->prompt = log_prompt;
348
349 return retval;
350}
351
362uint8_t
364{
365 pcmk__assert(out != NULL);
366
367 if (pcmk__str_eq(out->fmt_name, "log", pcmk__str_none)) {
368 private_data_t *priv = out->priv;
369
370 pcmk__assert(priv != NULL);
371 return priv->log_level;
372 }
373 return 0;
374}
375
389void
391{
392 pcmk__assert(out != NULL);
393
394 if (pcmk__str_eq(out->fmt_name, "log", pcmk__str_none)) {
395 private_data_t *priv = out->priv;
396
397 pcmk__assert(priv != NULL);
398 priv->log_level = log_level;
399 }
400}
401
418void
420 const char *function, uint32_t line, uint32_t tags)
421{
422 pcmk__assert(out != NULL);
423
424 if (pcmk__str_eq(out->fmt_name, "log", pcmk__str_none)) {
425 private_data_t *priv = out->priv;
426
427 pcmk__assert(priv != NULL);
428 priv->file = file;
429 priv->function = function;
430 priv->line = line;
431 priv->tags = tags;
432 }
433}
const char * name
Definition cib.c:26
gchar * pcmk__quote_cmdline(gchar **argv)
Definition cmdline.c:163
#define PACEMAKER_VERSION
Definition config.h:430
#define CRM_FEATURES
Definition config.h:30
#define BUILD_VERSION
Definition config.h:5
#define do_crm_log_xml(level, text, xml)
Log XML line-by-line in a formatted fashion.
Definition logging.h:236
struct private_data_s private_data_t
int pcmk__call_message(pcmk__output_t *out, const char *message_id,...)
Definition output.c:176
void pcmk__register_message(pcmk__output_t *out, const char *message_id, pcmk__message_fn_t fn)
Definition output.c:198
#define logger(priv, fmt, args...)
Definition output_log.c:37
void pcmk__output_set_log_level(pcmk__output_t *out, uint8_t log_level)
Definition output_log.c:390
pcmk__output_t * pcmk__mk_log_output(char **argv)
Definition output_log.c:315
struct private_data_s private_data_t
#define logger_va(priv, level, fmt, ap)
Definition output_log.c:53
void pcmk__output_set_log_filter(pcmk__output_t *out, const char *file, const char *function, uint32_t line, uint32_t tags)
Definition output_log.c:419
uint8_t pcmk__output_get_log_level(const pcmk__output_t *out)
Definition output_log.c:363
@ pcmk_rc_ok
Definition results.h:159
enum crm_exit_e crm_exit_t
Exit status codes for tools and daemons.
#define pcmk__assert(expr)
@ pcmk__str_none
This structure contains everything that makes up a single output formatter.
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 void void(* output_xml)(pcmk__output_t *out, const char *name, const char *buf)
int int void(* err)(pcmk__output_t *out, const char *format,...) G_GNUC_PRINTF(2
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)
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 void(* list_item)(pcmk__output_t *out, const char *name, const char *format,...) G_GNUC_PRINTF(3
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
int int(* transient)(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.
xmlNode * pcmk__xe_create(xmlNode *parent, const char *name)
void pcmk__xe_set_content(xmlNode *node, const char *format,...) G_GNUC_PRINTF(2