pacemaker 2.1.5-a3f44794f94
Scalable High-Availability cluster resource manager
output_log.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>
12
13#include <ctype.h>
14#include <stdarg.h>
15#include <stdlib.h>
16#include <stdio.h>
17
18GOptionEntry pcmk__log_output_entries[] = {
19 { NULL }
20};
21
22typedef struct private_data_s {
23 /* gathered in log_begin_list */
24 GQueue/*<char*>*/ *prefixes;
25 int log_level;
27
28static void
29log_subprocess_output(pcmk__output_t *out, int exit_status,
30 const char *proc_stdout, const char *proc_stderr) {
31 /* This function intentionally left blank */
32}
33
34static void
35log_free_priv(pcmk__output_t *out) {
36 private_data_t *priv = NULL;
37
38 if (out == NULL || out->priv == NULL) {
39 return;
40 }
41
42 priv = out->priv;
43
44 g_queue_free(priv->prefixes);
45 free(priv);
46 out->priv = NULL;
47}
48
49static bool
50log_init(pcmk__output_t *out) {
51 private_data_t *priv = NULL;
52
53 CRM_ASSERT(out != NULL);
54
55 /* If log_init was previously called on this output struct, just return. */
56 if (out->priv != NULL) {
57 return true;
58 }
59
60 out->priv = calloc(1, sizeof(private_data_t));
61 if (out->priv == NULL) {
62 return false;
63 }
64
65 priv = out->priv;
66
67 priv->prefixes = g_queue_new();
68 priv->log_level = LOG_INFO;
69
70 return true;
71}
72
73static void
74log_finish(pcmk__output_t *out, crm_exit_t exit_status, bool print, void **copy_dest) {
75 /* This function intentionally left blank */
76}
77
78static void
79log_reset(pcmk__output_t *out) {
80 CRM_ASSERT(out != NULL);
81
82 out->dest = freopen(NULL, "w", out->dest);
83 CRM_ASSERT(out->dest != NULL);
84
85 log_free_priv(out);
86 log_init(out);
87}
88
89static void
90log_version(pcmk__output_t *out, bool extended) {
91 private_data_t *priv = NULL;
92
93 CRM_ASSERT(out != NULL && out->priv != NULL);
94 priv = out->priv;
95
96 if (extended) {
97 do_crm_log(priv->log_level, "Pacemaker %s (Build: %s): %s",
99 } else {
100 do_crm_log(priv->log_level, "Pacemaker %s", PACEMAKER_VERSION);
101 do_crm_log(priv->log_level, "Written by Andrew Beekhof and"
102 "the Pacemaker project contributors");
103 }
104}
105
106G_GNUC_PRINTF(2, 3)
107static void
108log_err(pcmk__output_t *out, const char *format, ...) {
109 va_list ap;
110 char* buffer = NULL;
111 int len = 0;
112
113 CRM_ASSERT(out != NULL);
114
115 va_start(ap, format);
116 /* Informational output does not get indented, to separate it from other
117 * potentially indented list output.
118 */
119 len = vasprintf(&buffer, format, ap);
120 CRM_ASSERT(len >= 0);
121 va_end(ap);
122
123 crm_err("%s", buffer);
124
125 free(buffer);
126}
127
128static void
129log_output_xml(pcmk__output_t *out, const char *name, const char *buf) {
130 xmlNodePtr node = NULL;
131 private_data_t *priv = NULL;
132
133 CRM_ASSERT(out != NULL && out->priv != NULL);
134 priv = out->priv;
135
136 node = create_xml_node(NULL, name);
137 xmlNodeSetContent(node, (pcmkXmlStr) buf);
138 do_crm_log_xml(priv->log_level, name, node);
139 free(node);
140}
141
142G_GNUC_PRINTF(4, 5)
143static void
144log_begin_list(pcmk__output_t *out, const char *singular_noun, const char *plural_noun,
145 const char *format, ...) {
146 int len = 0;
147 va_list ap;
148 char* buffer = NULL;
149 private_data_t *priv = NULL;
150
151 CRM_ASSERT(out != NULL && out->priv != NULL);
152 priv = out->priv;
153
154 va_start(ap, format);
155 len = vasprintf(&buffer, format, ap);
156 CRM_ASSERT(len >= 0);
157 va_end(ap);
158
159 /* Don't skip empty prefixes,
160 * otherwise there will be mismatch
161 * in the log_end_list */
162 if(strcmp(buffer, "") == 0) {
163 /* nothing */
164 }
165
166 g_queue_push_tail(priv->prefixes, buffer);
167}
168
169G_GNUC_PRINTF(3, 4)
170static void
171log_list_item(pcmk__output_t *out, const char *name, const char *format, ...) {
172 int len = 0;
173 va_list ap;
174 private_data_t *priv = NULL;
175 char prefix[LINE_MAX] = { 0 };
176 int offset = 0;
177 char* buffer = NULL;
178
179 CRM_ASSERT(out != NULL && out->priv != NULL);
180 priv = out->priv;
181
182 for (GList* gIter = priv->prefixes->head; gIter; gIter = gIter->next) {
183 if (strcmp(prefix, "") != 0) {
184 offset += snprintf(prefix + offset, LINE_MAX - offset, ": %s", (char *)gIter->data);
185 } else {
186 offset = snprintf(prefix, LINE_MAX, "%s", (char *)gIter->data);
187 }
188 }
189
190 va_start(ap, format);
191 len = vasprintf(&buffer, format, ap);
192 CRM_ASSERT(len >= 0);
193 va_end(ap);
194
195 if (strcmp(buffer, "") != 0) { /* We don't want empty messages */
196 if ((name != NULL) && (strcmp(name, "") != 0)) {
197 if (strcmp(prefix, "") != 0) {
198 do_crm_log(priv->log_level, "%s: %s: %s", prefix, name, buffer);
199 } else {
200 do_crm_log(priv->log_level, "%s: %s", name, buffer);
201 }
202 } else {
203 if (strcmp(prefix, "") != 0) {
204 do_crm_log(priv->log_level, "%s: %s", prefix, buffer);
205 } else {
206 do_crm_log(priv->log_level, "%s", buffer);
207 }
208 }
209 }
210 free(buffer);
211}
212
213static void
214log_end_list(pcmk__output_t *out) {
215 private_data_t *priv = NULL;
216
217 CRM_ASSERT(out != NULL && out->priv != NULL);
218 priv = out->priv;
219
220 if (priv->prefixes == NULL) {
221 return;
222 }
223 CRM_ASSERT(priv->prefixes->tail != NULL);
224
225 free((char *)priv->prefixes->tail->data);
226 g_queue_pop_tail(priv->prefixes);
227}
228
229G_GNUC_PRINTF(2, 3)
230static int
231log_info(pcmk__output_t *out, const char *format, ...) {
232 private_data_t *priv = NULL;
233 int len = 0;
234 va_list ap;
235 char* buffer = NULL;
236
237 CRM_ASSERT(out != NULL && out->priv != NULL);
238 priv = out->priv;
239
240 va_start(ap, format);
241 len = vasprintf(&buffer, format, ap);
242 CRM_ASSERT(len >= 0);
243 va_end(ap);
244
245 do_crm_log(priv->log_level, "%s", buffer);
246
247 free(buffer);
248 return pcmk_rc_ok;
249}
250
251static bool
252log_is_quiet(pcmk__output_t *out) {
253 return false;
254}
255
256static void
257log_spacer(pcmk__output_t *out) {
258 /* This function intentionally left blank */
259}
260
261static void
262log_progress(pcmk__output_t *out, bool end) {
263 /* This function intentionally left blank */
264}
265
266static void
267log_prompt(const char *prompt, bool echo, char **dest) {
268 /* This function intentionally left blank */
269}
270
273 pcmk__output_t *retval = calloc(1, sizeof(pcmk__output_t));
274
275 if (retval == NULL) {
276 return NULL;
277 }
278
279 retval->fmt_name = "log";
280 retval->request = pcmk__quote_cmdline(argv);
281
282 retval->init = log_init;
283 retval->free_priv = log_free_priv;
284 retval->finish = log_finish;
285 retval->reset = log_reset;
286
288 retval->message = pcmk__call_message;
289
290 retval->subprocess_output = log_subprocess_output;
291 retval->version = log_version;
292 retval->info = log_info;
293 retval->err = log_err;
294 retval->output_xml = log_output_xml;
295
296 retval->begin_list = log_begin_list;
297 retval->list_item = log_list_item;
298 retval->end_list = log_end_list;
299
300 retval->is_quiet = log_is_quiet;
301 retval->spacer = log_spacer;
302 retval->progress = log_progress;
303 retval->prompt = log_prompt;
304
305 return retval;
306}
307
308void
310 private_data_t *priv = NULL;
311
312 CRM_ASSERT(out != NULL && out->priv != NULL);
313 CRM_CHECK(pcmk__str_eq(out->fmt_name, "log", pcmk__str_none), return);
314
315 priv = out->priv;
316 priv->log_level = log_level;
317}
const char * name
Definition: cib.c:24
gchar * pcmk__quote_cmdline(gchar **argv)
Definition: cmdline.c:165
#define PACEMAKER_VERSION
Definition: config.h:502
#define CRM_FEATURES
Definition: config.h:33
#define BUILD_VERSION
Definition: config.h:8
#define do_crm_log_xml(level, text, xml)
Log XML line-by-line in a formatted fashion.
Definition: logging.h:251
#define do_crm_log(level, fmt, args...)
Log a message.
Definition: logging.h:168
#define CRM_CHECK(expr, failure_action)
Definition: logging.h:227
#define crm_err(fmt, args...)
Definition: logging.h:359
struct private_data_s private_data_t
int pcmk__call_message(pcmk__output_t *out, const char *message_id,...)
Definition: output.c:131
void pcmk__register_message(pcmk__output_t *out, const char *message_id, pcmk__message_fn_t fn)
Definition: output.c:153
GOptionEntry pcmk__log_output_entries[]
Definition: output_log.c:18
void pcmk__output_set_log_level(pcmk__output_t *out, int log_level)
Definition: output_log.c:309
pcmk__output_t * pcmk__mk_log_output(char **argv)
Definition: output_log.c:272
struct private_data_s private_data_t
#define CRM_ASSERT(expr)
Definition: results.h:42
@ pcmk_rc_ok
Definition: results.h:148
enum crm_exit_e crm_exit_t
@ 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.
int(*) void(*) void(* output_xml)(pcmk__output_t *out, const char *name, const char *buf)
void(* register_message)(pcmk__output_t *out, const char *message_id, pcmk__message_fn_t fn)
void(*) void(* list_item)(pcmk__output_t *out, const char *name, const char *format,...) G_GNUC_PRINTF(3
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(* 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.
const xmlChar * pcmkXmlStr
Definition: xml.h:50
xmlNode * create_xml_node(xmlNode *parent, const char *name)
Definition: xml.c:749