summaryrefslogtreecommitdiff
path: root/lib/log/log.c
diff options
context:
space:
mode:
Diffstat (limited to 'lib/log/log.c')
-rw-r--r--lib/log/log.c868
1 files changed, 698 insertions, 170 deletions
diff --git a/lib/log/log.c b/lib/log/log.c
index d50cb4a..6dc2696 100644
--- a/lib/log/log.c
+++ b/lib/log/log.c
@@ -10,30 +10,41 @@
*
* You should have received a copy of the GNU Lesser General Public License
* along with this program; if not, write to the Free Software Foundation,
- * Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
+ * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
*/
-#include "lib.h"
-#include "device.h"
-#include "memlock.h"
-#include "defaults.h"
+#include "lib/misc/lib.h"
+#include "lib/device/device.h"
+#include "lib/mm/memlock.h"
+#include "lib/config/defaults.h"
+#include "lib/report/report.h"
+#include "lib/misc/lvm-file.h"
+#include <stdio.h>
#include <stdarg.h>
#include <syslog.h>
+#include <ctype.h>
+#include <time.h>
+
+#ifdef SYSTEMD_JOURNAL_SUPPORT
+#include <systemd/sd-journal.h>
+#endif
static FILE *_log_file;
-static struct device _log_dev;
-static struct str_list _log_dev_alias;
+static char _log_file_path[PATH_MAX];
static int _syslog = 0;
static int _log_to_file = 0;
-static int _log_direct = 0;
+static uint64_t _log_file_max_lines = 0;
+static uint64_t _log_file_lines = 0;
static int _log_while_suspended = 0;
-static int _indent = 1;
+static int _indent = 0;
static int _log_suppress = 0;
static char _msg_prefix[30] = " ";
-static int _already_logging = 0;
-static int _abort_on_internal_errors = 0;
+static int _abort_on_internal_errors_config = 0;
+static uint32_t _debug_file_fields;
+static uint32_t _debug_output_fields;
+static uint32_t _log_journal = 0;
static lvm2_log_fn_t _lvm2_log_fn = NULL;
@@ -44,19 +55,256 @@ static size_t _lvm_errmsg_size = 0;
static size_t _lvm_errmsg_len = 0;
#define MAX_ERRMSG_LEN (512 * 1024) /* Max size of error buffer 512KB */
+static log_report_t _log_report = {
+ .report = NULL,
+ .context = LOG_REPORT_CONTEXT_NULL,
+ .object_type = LOG_REPORT_OBJECT_TYPE_NULL,
+ .object_id = NULL,
+ .object_name = NULL,
+ .object_group = NULL
+};
+
+#define LOG_STREAM_BUFFER_SIZE 4096
+
+struct log_stream_item {
+ FILE *stream;
+ char *buffer;
+};
+
+static struct log_stream {
+ struct log_stream_item out;
+ struct log_stream_item err;
+ struct log_stream_item report;
+} _log_stream = {{NULL, NULL},
+ {NULL, NULL},
+ {NULL, NULL}};
+
+#define out_stream (_log_stream.out.stream ? : stdout)
+#define err_stream (_log_stream.err.stream ? : stderr)
+#define report_stream (_log_stream.report.stream ? : stdout)
+
+static int _set_custom_log_stream(struct log_stream_item *stream_item, int custom_fd)
+{
+ FILE *final_stream = NULL;
+ int flags;
+ int r = 1;
+
+ if (custom_fd < 0)
+ goto out;
+
+ if (is_valid_fd(custom_fd)) {
+ if ((flags = fcntl(custom_fd, F_GETFL)) > 0) {
+ if ((flags & O_ACCMODE) == O_RDONLY) {
+ log_error("File descriptor %d already open in read-only "
+ "mode, expected write-only or read-write mode.",
+ (int) custom_fd);
+ r = 0;
+ goto out;
+ }
+ }
+
+ if (custom_fd == STDIN_FILENO) {
+ log_error("Can't set standard input for log output.");
+ r = 0;
+ goto out;
+ }
+
+ if (custom_fd == STDOUT_FILENO) {
+ final_stream = stdout;
+ goto out;
+ }
+
+ if (custom_fd == STDERR_FILENO) {
+ final_stream = stderr;
+ goto out;
+ }
+ }
+
+ if (!(final_stream = fdopen(custom_fd, "w"))) {
+ log_error("Failed to open stream for file descriptor %d.",
+ (int) custom_fd);
+ r = 0;
+ goto out;
+ }
+
+ if (!(stream_item->buffer = malloc(LOG_STREAM_BUFFER_SIZE))) {
+ log_error("Failed to allocate buffer for stream on file "
+ "descriptor %d.", (int) custom_fd);
+ } else {
+ if (setvbuf(final_stream, stream_item->buffer, _IOLBF, LOG_STREAM_BUFFER_SIZE)) {
+ log_sys_error("setvbuf", "");
+ free(stream_item->buffer);
+ stream_item->buffer = NULL;
+ }
+ }
+out:
+ stream_item->stream = final_stream;
+ return r;
+}
+
+int init_custom_log_streams(struct custom_fds *custom_fds)
+{
+ return _set_custom_log_stream(&_log_stream.out, custom_fds->out) &&
+ _set_custom_log_stream(&_log_stream.err, custom_fds->err) &&
+ _set_custom_log_stream(&_log_stream.report, custom_fds->report);
+}
+
+static void _check_and_replace_standard_log_streams(FILE *old_stream, FILE *new_stream)
+{
+ if (_log_stream.out.stream == old_stream)
+ _log_stream.out.stream = new_stream;
+
+ if (_log_stream.err.stream == old_stream)
+ _log_stream.err.stream = new_stream;
+
+ if (_log_stream.report.stream == old_stream)
+ _log_stream.report.stream = new_stream;
+}
+
+/*
+ * Close and reopen standard stream on file descriptor fd.
+ */
+int reopen_standard_stream(FILE **stream, const char *mode)
+{
+ int fd, fd_copy, new_fd;
+ const char *name;
+ FILE *old_stream = *stream;
+ FILE *new_stream;
+
+ if (old_stream == stdin) {
+ fd = STDIN_FILENO;
+ name = "stdin";
+ } else if (old_stream == stdout) {
+ fd = STDOUT_FILENO;
+ name = "stdout";
+ } else if (old_stream == stderr) {
+ fd = STDERR_FILENO;
+ name = "stderr";
+ } else {
+ log_error(INTERNAL_ERROR "reopen_standard_stream called on non-standard stream");
+ return 0;
+ }
+
+ if ((fd_copy = dup(fd)) < 0) {
+ log_sys_error("dup", name);
+ return 0;
+ }
+
+ if (fclose(old_stream))
+ log_sys_error("fclose", name);
+
+ if ((new_fd = dup2(fd_copy, fd)) < 0)
+ log_sys_error("dup2", name);
+ else if (new_fd != fd)
+ log_error("dup2(%d, %d) returned %d", fd_copy, fd, new_fd);
+
+ if (close(fd_copy) < 0)
+ log_sys_error("close", name);
+
+ if (!(new_stream = fdopen(fd, mode))) {
+ log_sys_error("fdopen", name);
+ return 0;
+ }
+
+ _check_and_replace_standard_log_streams(old_stream, new_stream);
+
+ *stream = new_stream;
+ return 1;
+}
+
void init_log_fn(lvm2_log_fn_t log_fn)
{
- if (log_fn)
- _lvm2_log_fn = log_fn;
- else
- _lvm2_log_fn = NULL;
+ _lvm2_log_fn = log_fn;
}
+/* Read /proc/self/stat to extract pid and starttime */
+static int _get_pid_starttime(int *pid, unsigned long long *starttime)
+{
+ static const char statfile[] = "/proc/self/stat";
+ char buf[1024];
+ char *p;
+ int fd;
+ int e;
+
+ if ((fd = open(statfile, O_RDONLY)) == -1) {
+ log_sys_debug("open", statfile);
+ return 0;
+ }
+
+ if ((e = read(fd, buf, sizeof(buf) - 1)) <= 0)
+ log_sys_debug("read", statfile);
+
+ if (!close(fd))
+ log_sys_debug("close", statfile);
+
+ if (e <= 0)
+ return 0;
+
+ buf[e] = '\0';
+ if ((sscanf(buf, "%d ", pid) == 1) &&
+ /* Jump past COMM, don't use scanf with '%s' since COMM may contain a space. */
+ (p = strrchr(buf, ')')) &&
+ (sscanf(++p, " %*c %*d %*d %*d %*d " /* tty_nr */
+ "%*d %*u %*u %*u %*u " /* mjflt */
+ "%*u %*u %*u %*d %*d " /* cstim */
+ "%*d %*d %*d %*d " /* itrealvalue */
+ "%llu", starttime) == 1))
+ return 1;
+
+ log_debug("Cannot parse content of %s.", statfile);
+
+ return 0;
+}
+
+/*
+ * Support envvar LVM_LOG_FILE_EPOCH and allow to attach
+ * extra keyword (consist of upto 32 alpha chars) to
+ * opened log file. After this 'epoch' word pid and starttime
+ * (in kernel units, read from /proc/self/stat)
+ * is automatically attached.
+ * If command/daemon forks multiple times, it could create multiple
+ * log files ensuring, there are no overwrites.
+ */
void init_log_file(const char *log_file, int append)
{
- const char *open_mode = append ? "a" : "w";
+ const char *env;
+ int pid = 0;
+ unsigned long long starttime = 0;
+ int i = 0;
+
+ _log_file_path[0] = '\0';
+ if ((env = getenv("LVM_LOG_FILE_EPOCH"))) {
+ while (isalpha(env[i]) && i < 32) /* Up to 32 alphas */
+ i++;
+ if (env[i]) {
+ if (i)
+ log_warn("WARNING: Ignoring invalid LVM_LOG_FILE_EPOCH envvar \"%s\".", env);
+ goto no_epoch;
+ }
+
+ if (!_get_pid_starttime(&pid, &starttime))
+ log_debug("Failed to obtain pid and starttime.");
- if (!(_log_file = fopen(log_file, open_mode))) {
+ if (dm_snprintf(_log_file_path, sizeof(_log_file_path),
+ "%s_%s_%d_%llu", log_file, env, pid, starttime) < 0) {
+ log_warn("WARNING: Debug log file path is too long for epoch.");
+ _log_file_path[0] = '\0';
+ } else {
+ log_file = _log_file_path;
+ append = 1; /* force */
+ }
+
+ if ((env = getenv("LVM_LOG_FILE_MAX_LINES"))) {
+ if (sscanf(env, FMTu64, &_log_file_max_lines) != 1) {
+ log_warn("WARNING: Ignoring invalid LVM_LOG_MAX_LINES envvar \"%s\".", env);
+ _log_file_max_lines = 0;
+ }
+ _log_file_lines = 0;
+ }
+ }
+
+no_epoch:
+ if (!(_log_file = fopen(log_file, append ? "a" : "w"))) {
log_sys_error("fopen", log_file);
return;
}
@@ -64,15 +312,29 @@ void init_log_file(const char *log_file, int append)
_log_to_file = 1;
}
-void init_log_direct(const char *log_file, int append)
+/*
+ * Unlink the log file depeding on command's return value
+ *
+ * When envvar LVM_EXPECTED_EXIT_STATUS is set, compare
+ * resulting status with this string.
+ *
+ * It's possible to specify 2 variants - having it equal to
+ * a single number or having it different from a single number.
+ *
+ * i.e. LVM_EXPECTED_EXIT_STATUS=">1" # delete when ret > 1.
+ */
+void unlink_log_file(int ret)
{
- int open_flags = append ? 0 : O_TRUNC;
-
- dev_create_file(log_file, &_log_dev, &_log_dev_alias, 1);
- if (!dev_open_flags(&_log_dev, O_RDWR | O_CREAT | open_flags, 1, 0))
- return;
-
- _log_direct = 1;
+ const char *env;
+
+ if (_log_file_path[0] &&
+ (env = getenv("LVM_EXPECTED_EXIT_STATUS")) &&
+ ((env[0] == '>' && ret > atoi(env + 1)) ||
+ (atoi(env) == ret))) {
+ if (unlink(_log_file_path))
+ log_sys_error("unlink", _log_file_path);
+ _log_file_path[0] = '\0';
+ }
}
void init_log_while_suspended(int log_while_suspended)
@@ -80,8 +342,16 @@ void init_log_while_suspended(int log_while_suspended)
_log_while_suspended = log_while_suspended;
}
-void init_syslog(int facility)
+void init_syslog(int enable, int facility)
{
+ if (!enable) {
+ _syslog = 0;
+ return;
+ }
+
+ if (getenv("LVM_SUPPRESS_SYSLOG"))
+ return;
+
openlog("lvm", LOG_PID, facility);
_syslog = 1;
}
@@ -95,29 +365,15 @@ int log_suppress(int suppress)
return old_suppress;
}
-void release_log_memory(void)
-{
- if (!_log_direct)
- return;
-
- dm_free((char *) _log_dev_alias.str);
- _log_dev_alias.str = "activate_log file";
-}
-
void fin_log(void)
{
- if (_log_direct) {
- (void) dev_close(&_log_dev);
- _log_direct = 0;
- }
-
if (_log_to_file) {
if (dm_fclose(_log_file)) {
if (errno)
- fprintf(stderr, "failed to write log file: %s\n",
+ fprintf(err_stream, "failed to write log file: %s\n",
strerror(errno));
else
- fprintf(stderr, "failed to write log file\n");
+ fprintf(err_stream, "failed to write log file\n");
}
_log_to_file = 0;
@@ -133,8 +389,9 @@ void fin_syslog(void)
void init_msg_prefix(const char *prefix)
{
- strncpy(_msg_prefix, prefix, sizeof(_msg_prefix) - 1);
- _msg_prefix[sizeof(_msg_prefix) - 1] = '\0';
+ if (prefix)
+ /* Cut away too long prefix */
+ (void) dm_strncpy(_msg_prefix, prefix, sizeof(_msg_prefix));
}
void init_indent(int indent)
@@ -142,9 +399,10 @@ void init_indent(int indent)
_indent = indent;
}
+/* If present, environment setting will override this. */
void init_abort_on_internal_errors(int fatal)
{
- _abort_on_internal_errors = fatal;
+ _abort_on_internal_errors_config = fatal;
}
void reset_lvm_errno(int store_errmsg)
@@ -152,7 +410,7 @@ void reset_lvm_errno(int store_errmsg)
_lvm_errno = 0;
if (_lvm_errmsg) {
- dm_free(_lvm_errmsg);
+ free(_lvm_errmsg);
_lvm_errmsg = NULL;
_lvm_errmsg_size = _lvm_errmsg_len = 0;
}
@@ -170,6 +428,13 @@ const char *stored_errmsg(void)
return _lvm_errmsg ? : "";
}
+const char *stored_errmsg_with_clear(void)
+{
+ const char *rc = strdup(stored_errmsg());
+ reset_lvm_errno(1);
+ return rc;
+}
+
static struct dm_hash_table *_duplicated = NULL;
void reset_log_duplicated(void) {
@@ -179,57 +444,155 @@ void reset_log_duplicated(void) {
}
}
-void print_log(int level, const char *file, int line, int dm_errno,
- const char *format, ...)
+static const char *_get_log_level_name(int use_stderr, int level)
+{
+ static const char *log_level_names[] = {"", /* unassigned */
+ "", /* unassigned */
+ "fatal", /* _LOG_FATAL */
+ "error", /* _LOG_ERROR */
+ "warn", /* _LOG_WARN */
+ "notice",/* _LOG_NOTICE */
+ "info", /* _LOG_INFO */
+ "debug" /* _LOG_DEBUG */
+ };
+ if (level == _LOG_WARN && !use_stderr)
+ return "print";
+
+ return log_level_names[level];
+}
+
+const char *log_get_report_context_name(log_report_context_t context)
+{
+ static const char *log_context_names[LOG_REPORT_CONTEXT_COUNT] = {[LOG_REPORT_CONTEXT_NULL] = "",
+ [LOG_REPORT_CONTEXT_SHELL] = "shell",
+ [LOG_REPORT_CONTEXT_PROCESSING] = "processing"};
+ return log_context_names[context];
+}
+
+
+const char *log_get_report_object_type_name(log_report_object_type_t object_type)
+{
+ static const char *log_object_type_names[LOG_REPORT_OBJECT_TYPE_COUNT] = {[LOG_REPORT_OBJECT_TYPE_NULL] = "",
+ [LOG_REPORT_OBJECT_TYPE_PRE_CMD] = "pre-cmd",
+ [LOG_REPORT_OBJECT_TYPE_CMD] = "cmd",
+ [LOG_REPORT_OBJECT_TYPE_ORPHAN] = "orphan",
+ [LOG_REPORT_OBJECT_TYPE_PV] = "pv",
+ [LOG_REPORT_OBJECT_TYPE_LABEL] = "label",
+ [LOG_REPORT_OBJECT_TYPE_VG] = "vg",
+ [LOG_REPORT_OBJECT_TYPE_LV] = "lv"};
+ return log_object_type_names[object_type];
+}
+
+void init_debug_file_fields(uint32_t debug_fields)
+{
+ _debug_file_fields = debug_fields;
+}
+
+void init_debug_output_fields(uint32_t debug_fields)
+{
+ _debug_output_fields = debug_fields;
+}
+
+void init_log_journal(uint32_t fields)
+{
+ _log_journal = fields;
+}
+
+static void _set_time_prefix(char *prefix, int buflen)
+{
+
+ struct timespec ts;
+ struct tm time_info;
+ int len;
+
+ if (clock_gettime(CLOCK_REALTIME, &ts) < 0)
+ goto fail;
+
+ if (!localtime_r(&ts.tv_sec, &time_info))
+ goto fail;
+
+ len = strftime(prefix, buflen, "%H:%M:%S", &time_info);
+ if (!len)
+ goto fail;
+
+ len = dm_snprintf(prefix + len, buflen - len, ".%06ld ", ts.tv_nsec/1000);
+ if (len < 0)
+ goto fail;
+
+ return;
+
+fail:
+ *prefix = '\0';
+}
+
+__attribute__ ((format(printf, 5, 0)))
+static void _vprint_log(int level, const char *file, int line, int dm_errno_or_class,
+ const char *format, va_list orig_ap)
{
va_list ap;
- char buf[1024], locn[4096];
- int bufused, n;
- const char *message;
+ char buf[1024], message[4096];
+ char time_prefix[32] = "";
+ const char *command_prefix = NULL;
+ int n;
const char *trformat; /* Translated format string */
char *newbuf;
- int use_stderr = level & _LOG_STDERR;
- int log_once = level & _LOG_ONCE;
+ int use_stderr = log_stderr(level);
+ int log_once = log_once(level);
+ int log_bypass_report = log_bypass_report(level);
int fatal_internal_error = 0;
size_t msglen;
+ const char *indent_spaces = "";
+ FILE *stream;
+ static int _abort_on_internal_errors_env_present = -1;
+ static int _abort_on_internal_errors_env = 0;
+ char *env_str;
+ struct dm_report *orig_report;
+ int logged_via_report = 0;
+
+ level = log_level(level);
+
+ if (_abort_on_internal_errors_env_present < 0) {
+ if ((env_str = getenv("DM_ABORT_ON_INTERNAL_ERRORS"))) {
+ _abort_on_internal_errors_env_present = 1;
+ /* Set when env DM_ABORT_ON_INTERNAL_ERRORS is not "0" */
+ _abort_on_internal_errors_env = strcmp(env_str, "0");
+ } else
+ _abort_on_internal_errors_env_present = 0;
+ }
- level &= ~(_LOG_STDERR|_LOG_ONCE);
-
- if (_abort_on_internal_errors &&
- !strncmp(format, INTERNAL_ERROR,
- strlen(INTERNAL_ERROR))) {
+ /* Use value from environment if present, otherwise use value from config. */
+ if (((_abort_on_internal_errors_env_present && _abort_on_internal_errors_env) ||
+ (!_abort_on_internal_errors_env_present && _abort_on_internal_errors_config)) &&
+ !strncmp(format, INTERNAL_ERROR, sizeof(INTERNAL_ERROR) - 1)) {
fatal_internal_error = 1;
/* Internal errors triggering abort cannot be suppressed. */
_log_suppress = 0;
level = _LOG_FATAL;
}
- if (_log_suppress == 2)
- return;
-
if (level <= _LOG_ERR)
init_error_message_produced(1);
trformat = _(format);
- if (dm_errno && !_lvm_errno)
- _lvm_errno = dm_errno;
+ if (level < _LOG_DEBUG && dm_errno_or_class && !_lvm_errno)
+ _lvm_errno = dm_errno_or_class;
if (_lvm2_log_fn ||
(_store_errmsg && (level <= _LOG_ERR)) ||
+ (_log_report.report && !log_bypass_report && (use_stderr || (level <=_LOG_WARN))) ||
log_once) {
- va_start(ap, format);
- n = vsnprintf(locn, sizeof(locn) - 1, trformat, ap);
+ va_copy(ap, orig_ap);
+ n = vsnprintf(message, sizeof(message), trformat, ap);
va_end(ap);
+ /* When newer glibc returns >= sizeof(locn), we will just log what
+ * has fit into buffer, it's '\0' terminated string */
if (n < 0) {
- fprintf(stderr, _("vsnprintf failed: skipping external "
- "logging function"));
+ fprintf(err_stream, _("vsnprintf failed: skipping external "
+ "logging function"));
goto log_it;
}
-
- locn[sizeof(locn) - 1] = '\0';
- message = locn;
}
/* FIXME Avoid pointless use of message buffer when it'll never be read! */
@@ -238,7 +601,7 @@ void print_log(int level, const char *file, int line, int dm_errno,
msglen = strlen(message);
if ((_lvm_errmsg_len + msglen + 1) >= _lvm_errmsg_size) {
_lvm_errmsg_size = 2 * (_lvm_errmsg_len + msglen + 1);
- if ((newbuf = dm_realloc(_lvm_errmsg,
+ if ((newbuf = realloc(_lvm_errmsg,
_lvm_errmsg_size)))
_lvm_errmsg = newbuf;
else
@@ -256,14 +619,31 @@ void print_log(int level, const char *file, int line, int dm_errno,
if (log_once) {
if (!_duplicated)
- _duplicated = dm_hash_create(128);
+ _duplicated = dm_hash_create(117);
if (_duplicated) {
if (dm_hash_lookup(_duplicated, message))
level = _LOG_NOTICE;
- (void) dm_hash_insert(_duplicated, message, (void*)1);
+ else
+ (void) dm_hash_insert(_duplicated, message, (void*)1);
}
}
+ if (_log_report.report && !log_bypass_report && (use_stderr || (level <= _LOG_WARN))) {
+ orig_report = _log_report.report;
+ _log_report.report = NULL;
+ if (!report_cmdlog(orig_report, _get_log_level_name(use_stderr, level),
+ log_get_report_context_name(_log_report.context),
+ log_get_report_object_type_name(_log_report.object_type),
+ _log_report.object_name, _log_report.object_id,
+ _log_report.object_group, _log_report.object_group_id,
+ message, _lvm_errno, 0))
+ fprintf(err_stream, _("failed to report cmdstatus"));
+ else
+ logged_via_report = 1;
+
+ _log_report.report = orig_report;
+ }
+
if (_lvm2_log_fn) {
_lvm2_log_fn(level, file, line, 0, message);
if (fatal_internal_error)
@@ -272,126 +652,274 @@ void print_log(int level, const char *file, int line, int dm_errno,
}
log_it:
- if (!_log_suppress) {
- if (verbose_level() > _LOG_DEBUG)
- (void) dm_snprintf(locn, sizeof(locn), "#%s:%d ",
- file, line);
- else
- locn[0] = '\0';
- va_start(ap, format);
- switch (level) {
- case _LOG_DEBUG:
- if (!strcmp("<backtrace>", format) &&
- verbose_level() <= _LOG_DEBUG)
- break;
- if (verbose_level() >= _LOG_DEBUG) {
- fprintf(stderr, "%s%s%s", locn, log_command_name(),
- _msg_prefix);
- if (_indent)
- fprintf(stderr, " ");
- vfprintf(stderr, trformat, ap);
- fputc('\n', stderr);
- }
- break;
-
- case _LOG_INFO:
- if (verbose_level() >= _LOG_INFO) {
- fprintf(stderr, "%s%s%s", locn, log_command_name(),
- _msg_prefix);
- if (_indent)
- fprintf(stderr, " ");
- vfprintf(stderr, trformat, ap);
- fputc('\n', stderr);
+#ifdef SYSTEMD_JOURNAL_SUPPORT
+ if (_log_journal) {
+ int to_journal = 0;
+
+ /* By default the visible command output is _LOG_WARN or less. */
+
+ if (_log_journal & LOG_JOURNAL_DEBUG)
+ to_journal = 1;
+ if ((_log_journal & LOG_JOURNAL_OUTPUT) && (log_level(level) <= _LOG_WARN))
+ to_journal = 1;
+
+ if (to_journal) {
+ int prio;
+ switch (log_level(level)) {
+ case _LOG_ERR: prio = LOG_ERR; break;
+ case _LOG_WARN: prio = LOG_WARNING; break;
+ case _LOG_INFO: prio = LOG_INFO; break;
+ case _LOG_NOTICE: prio = LOG_NOTICE; break;
+ case _LOG_DEBUG: prio = LOG_DEBUG; break;
+ default: prio = LOG_INFO;
}
- break;
- case _LOG_NOTICE:
- if (verbose_level() >= _LOG_NOTICE) {
- fprintf(stderr, "%s%s%s", locn, log_command_name(),
- _msg_prefix);
- if (_indent)
- fprintf(stderr, " ");
- vfprintf(stderr, trformat, ap);
- fputc('\n', stderr);
- }
- break;
- case _LOG_WARN:
- if (verbose_level() >= _LOG_WARN) {
- fprintf(use_stderr ? stderr : stdout, "%s%s",
- log_command_name(), _msg_prefix);
- vfprintf(use_stderr ? stderr : stdout, trformat, ap);
- fputc('\n', use_stderr ? stderr : stdout);
+ va_copy(ap, orig_ap);
+ sd_journal_printv(prio, trformat, ap);
+ va_end(ap);
+ }
+ }
+#endif
+
+ if (!logged_via_report && ((verbose_level() >= level) && !_log_suppress)) {
+ if (verbose_level() > _LOG_DEBUG) {
+ memset(buf, 0, sizeof(buf));
+
+ if (!_debug_output_fields || (_debug_output_fields & LOG_DEBUG_FIELD_TIME)) {
+ if (!time_prefix[0])
+ _set_time_prefix(time_prefix, sizeof(time_prefix));
+ else
+ time_prefix[0] = '\0';
}
- break;
- case _LOG_ERR:
- if (verbose_level() >= _LOG_ERR) {
- fprintf(stderr, "%s%s%s", locn, log_command_name(),
- _msg_prefix);
- vfprintf(stderr, trformat, ap);
- fputc('\n', stderr);
+
+ if (!_debug_output_fields || (_debug_output_fields & LOG_DEBUG_FIELD_COMMAND))
+ command_prefix = log_command_file();
+ else
+ command_prefix = NULL;
+
+ if (!_debug_output_fields || (_debug_output_fields & LOG_DEBUG_FIELD_FILELINE))
+ (void) dm_snprintf(buf, sizeof(buf), "%s%s %s:%d",
+ time_prefix, command_prefix ?: "", file, line);
+ else
+ (void) dm_snprintf(buf, sizeof(buf), "%s%s",
+ time_prefix, command_prefix ?: "");
+ } else {
+ memset(buf, 0, sizeof(buf));
+
+ /* without -vvvv, command[pid] is controlled by config settings */
+
+ (void) dm_snprintf(buf, sizeof(buf), "%s", log_command_info());
+ }
+
+ if (_indent)
+ switch (level) {
+ case _LOG_NOTICE: indent_spaces = " "; break;
+ case _LOG_INFO: indent_spaces = " "; break;
+ case _LOG_DEBUG: indent_spaces = " "; break;
+ default: /* nothing to do */;
}
- break;
- case _LOG_FATAL:
+
+ va_copy(ap, orig_ap);
+ switch (level) {
+ case _LOG_DEBUG:
+ if (verbose_level() < _LOG_DEBUG)
+ break;
+ if (!debug_class_is_logged(dm_errno_or_class))
+ break;
+ if ((verbose_level() == level) &&
+ (strcmp("<backtrace>", format) == 0))
+ break;
+ /* fall through */
default:
- if (verbose_level() >= _LOG_FATAL) {
- fprintf(stderr, "%s%s%s", locn, log_command_name(),
- _msg_prefix);
- vfprintf(stderr, trformat, ap);
- fputc('\n', stderr);
- }
- break;
+ /* Typically only log_warn goes to out_stream */
+ stream = (use_stderr || (level != _LOG_WARN)) ? err_stream : out_stream;
+ if (stream == err_stream)
+ fflush(out_stream);
+ fprintf(stream, "%s%s%s", buf, _msg_prefix, indent_spaces);
+ vfprintf(stream, trformat, ap);
+ fputc('\n', stream);
}
va_end(ap);
}
- if (fatal_internal_error)
- abort();
-
- if (level > debug_level())
+ if ((level > debug_level()) ||
+ (level >= _LOG_DEBUG && !debug_class_is_logged(dm_errno_or_class))) {
+ if (fatal_internal_error)
+ abort();
return;
+ }
if (_log_to_file && (_log_while_suspended || !critical_section())) {
- fprintf(_log_file, "%s:%d %s%s", file, line, log_command_name(),
- _msg_prefix);
- va_start(ap, format);
+ if (!_debug_file_fields || (_debug_file_fields & LOG_DEBUG_FIELD_TIME)) {
+ if (!time_prefix[0])
+ _set_time_prefix(time_prefix, sizeof(time_prefix));
+ else
+ time_prefix[0] = '\0';
+ }
+
+ if (!_debug_file_fields || (_debug_file_fields & LOG_DEBUG_FIELD_COMMAND))
+ command_prefix = log_command_file();
+ else
+ command_prefix = NULL;
+
+ if (!_debug_file_fields || (_debug_file_fields & LOG_DEBUG_FIELD_FILELINE))
+ fprintf(_log_file, "%s%s %s:%d%s", time_prefix, command_prefix ?: "", file, line, _msg_prefix);
+ else
+ fprintf(_log_file, "%s%s %s", time_prefix, command_prefix ?: "", _msg_prefix);
+
+ va_copy(ap, orig_ap);
vfprintf(_log_file, trformat, ap);
va_end(ap);
- fprintf(_log_file, "\n");
+ if (_log_file_max_lines && ++_log_file_lines >= _log_file_max_lines) {
+ fprintf(_log_file, "\n%s:%d %sAborting. Command has reached limit "
+ "for logged lines (LVM_LOG_FILE_MAX_LINES=" FMTu64 ").",
+ file, line, _msg_prefix,
+ _log_file_max_lines);
+ fatal_internal_error = 1;
+ }
+
+ fputc('\n', _log_file);
fflush(_log_file);
}
if (_syslog && (_log_while_suspended || !critical_section())) {
- va_start(ap, format);
+ va_copy(ap, orig_ap);
vsyslog(level, trformat, ap);
va_end(ap);
}
- /* FIXME This code is unfinished - pre-extend & condense. */
- if (!_already_logging && _log_direct && critical_section()) {
- _already_logging = 1;
- memset(&buf, ' ', sizeof(buf));
- bufused = 0;
- if ((n = dm_snprintf(buf, sizeof(buf) - 1,
- "%s:%d %s%s", file, line, log_command_name(),
- _msg_prefix)) == -1)
- goto done;
-
- bufused += n;
-
- va_start(ap, format);
- n = vsnprintf(buf + bufused - 1, sizeof(buf) - bufused - 1,
- trformat, ap);
- va_end(ap);
- bufused += n;
-
- buf[bufused - 1] = '\n';
- done:
- buf[bufused] = '\n';
- buf[sizeof(buf) - 1] = '\n';
- /* FIXME real size bufused */
- dev_append(&_log_dev, sizeof(buf), buf);
- _already_logging = 0;
+ if (fatal_internal_error)
+ abort();
+}
+
+void print_log(int level, const char *file, int line, int dm_errno_or_class,
+ const char *format, ...)
+{
+ va_list ap;
+
+ va_start(ap, format);
+ _vprint_log(level, file, line, dm_errno_or_class, format, ap);
+ va_end(ap);
+}
+
+void print_log_libdm(int level, const char *file, int line, int dm_errno_or_class,
+ const char *format, ...)
+{
+ FILE *orig_out_stream = out_stream;
+ va_list ap;
+
+ /*
+ * Bypass report if printing output from libdm and if we have
+ * LOG_WARN level and it's not going to stderr (so we're
+ * printing common message that is not an error/warning).
+ */
+ if (!log_stderr(level) &&
+ (log_level(level) == _LOG_WARN))
+ level |= _LOG_BYPASS_REPORT;
+
+ _log_stream.out.stream = report_stream;
+
+ va_start(ap, format);
+ _vprint_log(level, file, line, dm_errno_or_class, format, ap);
+ va_end(ap);
+
+ _log_stream.out.stream = orig_out_stream;
+}
+
+log_report_t log_get_report_state(void)
+{
+ return _log_report;
+}
+
+void log_restore_report_state(log_report_t log_report)
+{
+ _log_report = log_report;
+}
+
+void log_set_report(struct dm_report *report)
+{
+ _log_report.report = report;
+}
+
+void log_set_report_context(log_report_context_t context)
+{
+ _log_report.context = context;
+}
+
+void log_set_report_object_type(log_report_object_type_t object_type)
+{
+ _log_report.object_type = object_type;
+}
+
+void log_set_report_object_group_and_group_id(const char *group, const char *id)
+{
+ _log_report.object_group = group;
+ _log_report.object_group_id = id;
+}
+
+void log_set_report_object_name_and_id(const char *name, const char *id)
+{
+ _log_report.object_name = name;
+ _log_report.object_id = id;
+}
+
+/*
+ * TODO: log/journal=["daemon_command"]
+ * daemon_command: record commands that are run by an lvm daemon.
+ * (i.e. not commands run directly by a user.)
+ * For this we need to be able to clearly identify when a command is
+ * being run by dmeventd/lvmpolld/lvmdbusd.
+ *
+ * TODO: log/journal_commmand_names=["lvcreate","lvconvert"]
+ * This would restrict log/journal=["command"] to the listed command names.
+ * Also allow "!command" to exclude a command, e.g. ["!pvs"]
+ *
+ * TODO: log/journal_daemon_command_names=["lvcreate","lvconvert"]
+ * This would restrict log/journal=["dameon_command"] to the listed command names.
+ *
+ * TODO: log/journal_daemon_names=["dmeventd"]
+ * This would restrict log/journal=["daemon_command"] to commands run by
+ * the named daemon.
+ *
+ * TODO: log/command_to_file=<path> would write this info to the file.
+ *
+ * TODO: log/debug_to_file=<path> would write full debugging to the file.
+ * (the same effect as log/file=<path> log/level=7)
+ */
+
+void log_command(const char *cmd_line, const char *cmd_name, const char *cmd_id)
+{
+#ifdef SYSTEMD_JOURNAL_SUPPORT
+ if (_log_journal & LOG_JOURNAL_COMMAND) {
+
+ /*
+ * TODO: DAEMON=dmeventd|lvmpolld|lvmdbusd,
+ * Could we include caller info such as libblkid, udev rule, etc?
+ * Does systemd already record the caller for us?
+ */
+
+ /* The command line, pid, and other things are automatically included. */
+
+ sd_journal_send("MESSAGE=lvm command %s", cmd_name,
+ "MESSAGE_ID=3ca432788c374e4ba684b834188eca36",
+ "LVM_CMD_NAME=%s", cmd_name,
+ "LVM_CMD_ID=%s", cmd_id,
+ "PRIORITY=%i", LOG_INFO,
+ NULL);
}
+#endif
+}
+
+uint32_t log_journal_str_to_val(const char *str)
+{
+ if (!strcasecmp(str, "command"))
+ return LOG_JOURNAL_COMMAND;
+ if (!strcasecmp(str, "output"))
+ return LOG_JOURNAL_OUTPUT;
+ if (!strcasecmp(str, "debug"))
+ return LOG_JOURNAL_DEBUG;
+ log_warn("WARNING: Ignoring unrecognized journal value.");
+ return 0;
}