]> granicus.if.org Git - postgresql/commitdiff
Add a hook for processing messages due to be sent to the server log.
authorTom Lane <tgl@sss.pgh.pa.us>
Tue, 6 Mar 2012 20:35:41 +0000 (15:35 -0500)
committerTom Lane <tgl@sss.pgh.pa.us>
Tue, 6 Mar 2012 20:35:41 +0000 (15:35 -0500)
Use-cases for this include custom log filtering rules and custom log
message transmission mechanisms (for instance, lossy log message
collection, which has been discussed several times recently).

As is our common practice for hooks, there's no regression test nor
user-facing documentation for this, though the author did exhibit a
sample module using the hook.

Martin Pihlak, reviewed by Marti Raudsepp

src/backend/utils/error/elog.c
src/include/utils/elog.h

index 470081a18009e5cedbd9e5999496ed4e8f29d43f..239ac19882d6b020c6eb71a57de1c048d01ebd30 100644 (file)
@@ -95,6 +95,15 @@ sigjmp_buf *PG_exception_stack = NULL;
 
 extern bool redirection_done;
 
+/*
+ * Hook for intercepting messages before they are sent to the server log.
+ * Note that the hook will not get called for messages that are suppressed
+ * by log_min_messages.  Also note that logging hooks implemented in preload
+ * libraries will miss any log messages that are generated before the
+ * library is loaded.
+ */
+emit_log_hook_type emit_log_hook = NULL;
+
 /* GUC parameters */
 int                    Log_error_verbosity = PGERROR_VERBOSE;
 char      *Log_line_prefix = NULL;             /* format for extra log line info */
@@ -1276,6 +1285,23 @@ EmitErrorReport(void)
        CHECK_STACK_DEPTH();
        oldcontext = MemoryContextSwitchTo(ErrorContext);
 
+       /*
+        * Call hook before sending message to log.  The hook function is allowed
+        * to turn off edata->output_to_server, so we must recheck that afterward.
+        * Making any other change in the content of edata is not considered
+        * supported.
+        *
+        * Note: the reason why the hook can only turn off output_to_server, and
+        * not turn it on, is that it'd be unreliable: we will never get here at
+        * all if errstart() deems the message uninteresting.  A hook that could
+        * make decisions in that direction would have to hook into errstart(),
+        * where it would have much less information available.  emit_log_hook is
+        * intended for custom log filtering and custom log message transmission
+        * mechanisms.
+        */
+       if (edata->output_to_server && emit_log_hook)
+               (*emit_log_hook) (edata);
+
        /* Send to server log, if enabled */
        if (edata->output_to_server)
                send_message_to_server_log(edata);
index fbc08df7dec42c9eac0ad9b9c0dfe30439f8464e..7b5bcfae6fdb95fa6b4d7021c5eb2f55ce03b646 100644 (file)
@@ -334,6 +334,10 @@ extern void FlushErrorState(void);
 extern void ReThrowError(ErrorData *edata);
 extern void pg_re_throw(void) __attribute__((noreturn));
 
+/* Hook for intercepting messages before they are sent to the server log */
+typedef void (*emit_log_hook_type) (ErrorData *edata);
+extern PGDLLIMPORT emit_log_hook_type emit_log_hook;
+
 
 /* GUC-configurable parameters */