1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
|
#ifndef LIB_EVENT_H
#define LIB_EVENT_H
/* event.h name is probably a bit too generic, so lets avoid using it. */
#include <sys/time.h>
/* Field name for the reason_code string list. */
#define EVENT_REASON_CODE "reason_code"
struct event;
struct event_log_params;
/* Hierarchical category of events. Each event can belong to multiple
categories. For example [ lib-storage/maildir, syscall/io ]. The categories
are expected to live as long as they're used in events. */
struct event_category {
struct event_category *parent;
const char *name;
/* non-NULL if this category has been registered
Do NOT dereference outside of event code in src/lib.
At any point in time it is safe to (1) check the pointer for
NULL/non-NULL to determine if this particular category instance
has been registered, and (2) compare two categories' internal
pointers to determine if they represent the same category. */
void *internal;
};
enum event_field_value_type {
EVENT_FIELD_VALUE_TYPE_STR,
EVENT_FIELD_VALUE_TYPE_INTMAX,
EVENT_FIELD_VALUE_TYPE_TIMEVAL,
EVENT_FIELD_VALUE_TYPE_STRLIST,
};
struct event_field {
const char *key;
enum event_field_value_type value_type;
struct {
const char *str;
intmax_t intmax;
struct timeval timeval;
ARRAY_TYPE(const_string) strlist;
} value;
};
struct event_add_field {
const char *key;
/* The first non-0/NULL value is used. */
const char *value;
intmax_t value_intmax;
struct timeval value_timeval;
};
struct event_passthrough {
/* wrappers to event_set_*() and event_add_*() for passthrough events,
so these can be chained like:
event_create_passthrough(parent)->name("name")->...->event() */
struct event_passthrough *
(*append_log_prefix)(const char *prefix);
struct event_passthrough *
(*replace_log_prefix)(const char *prefix);
struct event_passthrough *
(*set_name)(const char *name);
struct event_passthrough *
(*set_source)(const char *filename,
unsigned int linenum, bool literal_fname);
struct event_passthrough *
(*set_always_log_source)(void);
struct event_passthrough *
(*add_categories)(struct event_category *const *categories);
struct event_passthrough *
(*add_category)(struct event_category *category);
struct event_passthrough *
(*add_fields)(const struct event_add_field *fields);
struct event_passthrough *
(*add_str)(const char *key, const char *value);
struct event_passthrough *
(*add_int)(const char *key, intmax_t num);
struct event_passthrough *
(*add_int_nonzero)(const char *key, intmax_t num);
struct event_passthrough *
(*add_timeval)(const char *key, const struct timeval *tv);
struct event_passthrough *
(*inc_int)(const char *key, intmax_t num);
struct event_passthrough *
(*strlist_append)(const char *key, const char *value);
struct event_passthrough *
(*strlist_replace)(const char *key, const char *const *value,
unsigned int count);
struct event_passthrough *
(*clear_field)(const char *key);
struct event *(*event)(void);
};
typedef const char *
event_log_prefix_callback_t(void *context);
typedef const char *
event_log_message_callback_t(void *context, enum log_type log_type,
const char *message);
/* Returns TRUE if the event has all the categories that the "other" event has
(and maybe more). */
bool event_has_all_categories(struct event *event, const struct event *other);
/* Returns TRUE if the event has all the fields that the "other" event has
(and maybe more). Only the fields in the events themselves are checked.
Parent events' fields are not checked. */
bool event_has_all_fields(struct event *event, const struct event *other);
/* Returns the source event duplicated into a new event. Event pointers are
dropped. */
struct event *event_dup(const struct event *source);
/* Returns a flattened version of the source event.
Both categories and fields will be flattened.
A new reference to the source event is returned if no flattening was
needed. Event pointers are dropped if a new event was created. */
struct event *event_flatten(struct event *src);
/* Returns a minimized version of the source event.
Remove parents with no fields or categories, attempt to flatten fields
and categories to avoid sending one-off parent events. (There is a more
detailed description in a comment above the function implementation.)
A new reference to the source event is returned if no simplification
occured. Event pointers are dropped if a new event was created. */
struct event *event_minimize(struct event *src);
/* Copy all categories from source to dest.
Only the categories in source event itself are copied.
Parent events' categories aren't copied. */
void event_copy_categories(struct event *to, struct event *from);
/* Copy all fields from source to dest.
Only the fields in source event itself are copied.
Parent events' fields aren't copied. */
void event_copy_fields(struct event *to, struct event *from);
/* Create a new empty event under the parent event, or NULL for root event. */
struct event *event_create(struct event *parent, const char *source_filename,
unsigned int source_linenum);
#define event_create(parent) \
event_create((parent), __FILE__, __LINE__)
/* This is a temporary "passthrough" event. Its main purpose is to make it
easier to create temporary events as part of the event parameter in
e_error(), e_warning(), e_info() or e_debug(). These passthrough events are
automatically freed when the e_*() call is finished. Because this makes the
freeing less obvious, it should be avoided outside e_*()'s event parameter.
The passthrough events also change the API to be more convenient towards
being used in a parameter. Instead of having to use e.g.
event_add_str(event_set_name(event_create(parent), "name"), "key", "value")
the event_passthrough API can be a bit more readable as:
event_create_passthrough(parent)->set_name("name")->
add_str("key", "value")->event(). The passthrough event is converted to
a normal event at the end with the event() call. Note that this API works
by modifying the last created passthrough event, so it's not possible to
have multiple passthrough events created in parallel. */
struct event_passthrough *
event_create_passthrough(struct event *parent, const char *source_filename,
unsigned int source_linenum);
#define event_create_passthrough(parent) \
event_create_passthrough((parent), __FILE__, __LINE__)
/* Reference the event. Returns the event parameter. */
struct event *event_ref(struct event *event);
/* Unreference the event. If the reference count drops to 0, the event is
freed. The current global event's refcount must not drop to 0. */
void event_unref(struct event **event);
/* Set the event to be the global event and push it at the top of the global
event stack. Returns the event parameter. The event must be explicitly
popped before it's freed.
The global event acts as the root event for all the events while they are
being emitted. The global events don't permanently affect the event
hierarchy. The global events are typically used to add extra fields to all
emitted events while some specific work is running.
For example the global event can be "IMAP command SELECT", which can be used
for filtering events that happen while the SELECT command is being executed.
However, for the created struct mailbox the parent event should be the
mail_user, not the SELECT command. (If the mailbox used SELECT command as
the parent event, then any future event emitted via the mailbox event would
show SELECT command as the parent, even after SELECT had already finished.)
The global event works the same as if all the events' roots were instead
pointing to the global event. Global events don't affect log prefixes.
If ioloop contexts are used, the global events will automatically follow the
contexts. Any global events pushed while running in a context are popped
out when the context is deactivated, and pushed back when context is
activated again.
The created global events should use event_get_global() as their parent
event. Only the last pushed global event is used. */
struct event *event_push_global(struct event *event);
/* Pop the current global event and set the global event to the next one at
the top of the stack. Assert-crash if the current global event isn't the
given event parameter. Returns the next (now activated) global event in the
stack, or NULL if the stack is now empty. */
struct event *event_pop_global(struct event *event);
/* Returns the current global event. */
struct event *event_get_global(void);
/* Shortcut to create and push a global event and set its reason_code field. */
struct event_reason *
event_reason_begin(const char *reason_code, const char *source_filename,
unsigned int source_linenum);
#define event_reason_begin(reason_code) \
event_reason_begin(reason_code, __FILE__, __LINE__)
/* Finish the reason event. It pops the global event, which means it must be
at the top of the stack. */
void event_reason_end(struct event_reason **reason);
/* Generate a reason code as <module>:<name>. This function does some
sanity checks and conversions to make sure the reason codes are reasonable:
- Assert-crash if module has space, '-', ':' or uppercase characters.
- Assert-crash if module is empty
- Convert name to lowercase.
- Replace all space and '-' in name with '_'.
- Assert-crash if name has ':'
- assert-crash if name is empty
*/
const char *event_reason_code(const char *module, const char *name);
/* Same as event_reason_code(), but concatenate name_prefix and name.
The name_prefix must not contain spaces, '-', ':' or uppercase characters. */
const char *event_reason_code_prefix(const char *module,
const char *name_prefix, const char *name);
/* Set the appended log prefix string for this event. All the parent events'
log prefixes will be concatenated together when logging. The log type
text (e.g. "Info: ") will be inserted before appended log prefixes (but
after replaced log prefix).
Clears log_prefix callback.
*/
struct event *
event_set_append_log_prefix(struct event *event, const char *prefix);
/* Replace the full log prefix string for this event. The parent events' log
prefixes won't be used. Also, any parent event's message amendment callback
is not used.
Clears log_prefix callback.
*/
struct event *event_replace_log_prefix(struct event *event, const char *prefix);
/* Drop count prefixes from parents when this event is used for logging. This
does not affect the parent events. This only counts actual prefixes and not
parents. If the count is higher than the actual number of prefixes added by
parents, all will be dropped. */
struct event *
event_drop_parent_log_prefixes(struct event *event, unsigned int count);
/* Sets event prefix callback, sets log_prefix empty */
struct event *
event_set_log_prefix_callback(struct event *event, bool replace,
event_log_prefix_callback_t *callback,
void *context);
#define event_set_log_prefix_callback(event, replace, callback, context) \
event_set_log_prefix_callback(event, replace, \
(event_log_prefix_callback_t*)callback, TRUE ? context : \
CALLBACK_TYPECHECK(callback, const char *(*)(typeof(context))))
/* Sets event message amendment callback */
struct event *
event_set_log_message_callback(struct event *event,
event_log_message_callback_t *callback,
void *context);
#define event_set_log_message_callback(event, callback, context) \
event_set_log_message_callback(event, \
(event_log_message_callback_t*)callback, TRUE ? context : \
CALLBACK_TYPECHECK(callback, \
const char *(*)(typeof(context), enum log_type, \
const char *)))
/* Unsets the event message amendment callback. */
void event_unset_log_message_callback(struct event *event,
event_log_message_callback_t *callback,
void *context);
#define event_unset_log_message_callback(event, callback, context) \
event_unset_log_message_callback(event, \
(event_log_message_callback_t*)callback, context)
/* Set the event's name. The name is specific to a single sending of an event,
and it'll be automatically cleared once the event is sent. This should
typically be used only in a parameter to e_debug(), etc. */
struct event *
event_set_name(struct event *event, const char *name);
/* Set the source filename:linenum to the event. If literal_fname==TRUE,
it's assumed that __FILE__ has been used and the pointer is stored directly,
otherwise the filename is strdup()ed. */
struct event *
event_set_source(struct event *event, const char *filename,
unsigned int linenum, bool literal_fname);
/* Always include the source path:line in the log replies. This is
especially useful when logging about unexpected syscall failures, because
it allow quickly finding which of the otherwise identical syscalls in the
code generated the error. */
struct event *event_set_always_log_source(struct event *event);
/* Set minimum normal log level for the event. By default events with INFO
level and higher are logged. This can be used to easily hide even the INFO
log lines unless some verbose-setting is enabled.
Note that this functionality is mostly independent of debug logging.
Don't use this to enable debug log - use event_set_forced_debug() instead. */
struct event *event_set_min_log_level(struct event *event, enum log_type level);
enum log_type event_get_min_log_level(const struct event *event);
/* Add an internal pointer to an event. It can be looked up only with
event_get_ptr(). The keys are in their own namespace and won't conflict
with event fields. The pointers are specific to this specific event only -
they will be dropped from any duplicated/flattened/minimized events. */
struct event *event_set_ptr(struct event *event, const char *key, void *value);
/* Return a pointer set with event_set_ptr(), or NULL if it doesn't exist.
The pointer is looked up only from the event itself, not its parents. */
void *event_get_ptr(const struct event *event, const char *key);
/* Add NULL-terminated list of categories to the event. The categories pointer
doesn't need to stay valid afterwards, but the event_category structs
themselves must be. Returns the event parameter. */
struct event *
event_add_categories(struct event *event,
struct event_category *const *categories);
/* Add a single category to the event. */
struct event *
event_add_category(struct event *event, struct event_category *category);
/* Add key=value field to the event. If a key already exists, it's replaced.
Child events automatically inherit key=values from their parents at the
time the event is sent. So changing a key in parent will change the values
in the child events as well, unless the key has been overwritten in the
child event. Setting the value to "" is the same as event_field_clear().
Returns the event parameter. */
struct event *
event_add_str(struct event *event, const char *key, const char *value);
struct event *
event_add_int(struct event *event, const char *key, intmax_t num);
/* Adds int value to event if it is non-zero */
struct event *
event_add_int_nonzero(struct event *event, const char *key, intmax_t num);
/* Increase the key's value. If it's not set or isn't an integer type,
initialize the value to num. */
struct event *
event_inc_int(struct event *event, const char *key, intmax_t num);
struct event *
event_add_timeval(struct event *event, const char *key,
const struct timeval *tv);
/* Append new value to list. If the key is not a list, it will
be cleared first. NULL values are ignored. Duplicate values are ignored. */
struct event *
event_strlist_append(struct event *event, const char *key, const char *value);
/* Replace value with this strlist. */
struct event *
event_strlist_replace(struct event *event, const char *key,
const char *const *value, unsigned int count);
/* Copy the string list from src and its parents to dest. This can be especially
useful to copy the current global events' reason_codes to a more permanent
(e.g. async) event that can exist after the global events are popped out. */
struct event *
event_strlist_copy_recursive(struct event *dest, const struct event *src,
const char *key);
/* Same as event_add_str/int(), but do it via event_field struct. The fields
terminates with key=NULL. Returns the event parameter. */
struct event *
event_add_fields(struct event *event, const struct event_add_field *fields);
/* Mark a field as nonexistent. If a parent event has the field set, this
allows removing it from the child event. Using an event filter with e.g.
"key=*" won't match this field anymore, although it's still visible in
event_find_field*() and event_get_fields(). This is the same as using
event_add_str() with value="". */
void event_field_clear(struct event *event, const char *key);
/* Returns the parent event, or NULL if it doesn't exist. */
struct event *event_get_parent(const struct event *event);
/* Get the event's creation time. */
void event_get_create_time(const struct event *event, struct timeval *tv_r);
/* Get the time when the event was last sent. Returns TRUE if time was
returned, FALSE if event has never been sent. */
bool event_get_last_send_time(const struct event *event, struct timeval *tv_r);
/* Get the event duration field in microseconds. This is calculated from
the event's last sent time. */
void event_get_last_duration(const struct event *event,
uintmax_t *duration_usecs_r);
/* Returns field for a given key, or NULL if it doesn't exist. */
struct event_field *
event_find_field_nonrecursive(const struct event *event, const char *key);
/* Returns field for a given key, or NULL if it doesn't exist. If the key
isn't found from the event itself, find it from parent events, including
from the global event. */
const struct event_field *
event_find_field_recursive(const struct event *event, const char *key);
/* Same as event_find_field(), but return the value converted to a string.
If the field isn't stored as a string, the result is allocated from
data stack. */
const char *
event_find_field_recursive_str(const struct event *event, const char *key);
/* Returns all key=value fields that the event has.
Parent events' fields aren't returned. */
const struct event_field *
event_get_fields(const struct event *event, unsigned int *count_r);
/* Return all categories that the event has.
Parent events' categories aren't returned. */
struct event_category *const *
event_get_categories(const struct event *event, unsigned int *count_r);
/* Export the event into a tabescaped string, so its fields are separated
with TABs and there are no NUL, CR or LF characters. */
void event_export(const struct event *event, string_t *dest);
/* Import event. The string is expected to be generated by event_export().
All the used categories must already be registered.
Returns TRUE on success, FALSE on invalid string. */
bool event_import(struct event *event, const char *str, const char **error_r);
/* Same as event_import(), but string is already split into an array
of strings via *_strsplit_tabescaped(). */
bool event_import_unescaped(struct event *event, const char *const *args,
const char **error_r);
/* The event wasn't sent after all - free everything related to it.
Most importantly this frees any passthrough events. Typically this shouldn't
need to be called. */
void event_send_abort(struct event *event);
/* Enable "user_cpu_usecs" event field to event by getting current resource
usage which will be used in consequent event_send() to calculate
cpu time. This function can be called multiple times to update the current
resource usage.
The "user_cpu_usecs" field is automatically inherited by passthrough events,
but not full events.
*/
void event_enable_user_cpu_usecs(struct event *event);
void lib_event_init(void);
void lib_event_deinit(void);
#endif
|