mirror of
git://sourceware.org/git/lvm2.git
synced 2025-01-03 05:18:29 +03:00
ccd693d065
When processing LVs for a command we stored '*object_id' & '*group_id' as printable string that was however only used with json reporting. Refactor code so we simply store there 'struct id*' that is just converted into printable string when json reporting is really used. Also check for 'sigint()' right before loop processing begins which is primary purpose of this test.
935 lines
24 KiB
C
935 lines
24 KiB
C
/*
|
|
* Copyright (C) 2001-2004 Sistina Software, Inc. All rights reserved.
|
|
* Copyright (C) 2004-2007 Red Hat, Inc. All rights reserved.
|
|
*
|
|
* This file is part of LVM2.
|
|
*
|
|
* This copyrighted material is made available to anyone wishing to use,
|
|
* modify, copy, or redistribute it subject to the terms and conditions
|
|
* of the GNU Lesser General Public License v.2.1.
|
|
*
|
|
* 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., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
|
|
*/
|
|
|
|
#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 char _log_file_path[PATH_MAX];
|
|
|
|
static int _syslog = 0;
|
|
static int _log_to_file = 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 = 0;
|
|
static int _log_suppress = 0;
|
|
static char _msg_prefix[30] = " ";
|
|
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;
|
|
|
|
static int _lvm_errno = 0;
|
|
static int _store_errmsg = 0;
|
|
static char *_lvm_errmsg = NULL;
|
|
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(*stream, new_stream);
|
|
|
|
*stream = new_stream;
|
|
return 1;
|
|
}
|
|
|
|
void init_log_fn(lvm2_log_fn_t log_fn)
|
|
{
|
|
_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[] = DEFAULT_PROC_DIR "/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 up to 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 *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 (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;
|
|
}
|
|
|
|
_log_to_file = 1;
|
|
}
|
|
|
|
/*
|
|
* Unlink the log file depending 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)
|
|
{
|
|
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) && (errno != ENOENT))
|
|
log_sys_debug("unlink", _log_file_path);
|
|
_log_file_path[0] = '\0';
|
|
}
|
|
}
|
|
|
|
void init_log_while_suspended(int log_while_suspended)
|
|
{
|
|
_log_while_suspended = log_while_suspended;
|
|
}
|
|
|
|
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;
|
|
}
|
|
|
|
int log_suppress(int suppress)
|
|
{
|
|
int old_suppress = _log_suppress;
|
|
|
|
_log_suppress = suppress;
|
|
|
|
return old_suppress;
|
|
}
|
|
|
|
void fin_log(void)
|
|
{
|
|
if (_log_to_file) {
|
|
if (dm_fclose(_log_file)) {
|
|
if (errno)
|
|
fprintf(err_stream, "failed to write log file: %s\n",
|
|
strerror(errno));
|
|
else
|
|
fprintf(err_stream, "failed to write log file\n");
|
|
|
|
}
|
|
_log_to_file = 0;
|
|
}
|
|
}
|
|
|
|
void fin_syslog(void)
|
|
{
|
|
if (_syslog)
|
|
closelog();
|
|
_syslog = 0;
|
|
}
|
|
|
|
void init_msg_prefix(const char *prefix)
|
|
{
|
|
if (prefix)
|
|
/* Cut away too long prefix */
|
|
dm_strncpy(_msg_prefix, prefix, sizeof(_msg_prefix));
|
|
}
|
|
|
|
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_config = fatal;
|
|
}
|
|
|
|
void reset_lvm_errno(int store_errmsg)
|
|
{
|
|
_lvm_errno = 0;
|
|
|
|
if (_lvm_errmsg) {
|
|
free(_lvm_errmsg);
|
|
_lvm_errmsg = NULL;
|
|
_lvm_errmsg_size = _lvm_errmsg_len = 0;
|
|
}
|
|
|
|
_store_errmsg = store_errmsg;
|
|
}
|
|
|
|
int stored_errno(void)
|
|
{
|
|
return _lvm_errno;
|
|
}
|
|
|
|
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) {
|
|
if (_duplicated) {
|
|
dm_hash_destroy(_duplicated);
|
|
_duplicated = NULL;
|
|
}
|
|
}
|
|
|
|
static const char *_get_log_level_name(int use_stderr, int level)
|
|
{
|
|
static const char _log_level_names[][8] = {
|
|
"", /* 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][16] = {
|
|
[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][8] = {
|
|
[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, ".%06d ", (int)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], message[4096];
|
|
char time_prefix[32] = "";
|
|
const char *command_prefix = NULL;
|
|
int n;
|
|
const char *trformat; /* Translated format string */
|
|
char *newbuf;
|
|
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;
|
|
}
|
|
|
|
/* 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 (level <= _LOG_ERR)
|
|
init_error_message_produced(1);
|
|
|
|
trformat = _(format);
|
|
|
|
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_copy(ap, orig_ap);
|
|
/* coverity[format_string_injection] our code expectes this behavior. */
|
|
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(err_stream, _("vsnprintf failed: skipping external "
|
|
"logging function"));
|
|
goto log_it;
|
|
}
|
|
}
|
|
|
|
/* FIXME Avoid pointless use of message buffer when it'll never be read! */
|
|
if (_store_errmsg && (level <= _LOG_ERR) &&
|
|
_lvm_errmsg_len < MAX_ERRMSG_LEN) {
|
|
msglen = strlen(message);
|
|
if ((_lvm_errmsg_len + msglen + 1) >= _lvm_errmsg_size) {
|
|
_lvm_errmsg_size = 2 * (_lvm_errmsg_len + msglen + 1);
|
|
if ((newbuf = realloc(_lvm_errmsg,
|
|
_lvm_errmsg_size)))
|
|
_lvm_errmsg = newbuf;
|
|
else
|
|
_lvm_errmsg_size = _lvm_errmsg_len;
|
|
}
|
|
if (_lvm_errmsg &&
|
|
(_lvm_errmsg_len + msglen + 2) < _lvm_errmsg_size) {
|
|
/* prepend '\n' and copy with '\0' but do not count in */
|
|
if (_lvm_errmsg_len)
|
|
_lvm_errmsg[_lvm_errmsg_len++] = '\n';
|
|
memcpy(_lvm_errmsg + _lvm_errmsg_len, message, msglen + 1);
|
|
_lvm_errmsg_len += msglen;
|
|
}
|
|
}
|
|
|
|
if (log_once) {
|
|
if (!_duplicated)
|
|
_duplicated = dm_hash_create(117);
|
|
if (_duplicated) {
|
|
if (dm_hash_lookup(_duplicated, message))
|
|
level = _LOG_NOTICE;
|
|
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)
|
|
abort();
|
|
return;
|
|
}
|
|
|
|
log_it:
|
|
|
|
#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;
|
|
}
|
|
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';
|
|
}
|
|
|
|
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 */;
|
|
}
|
|
|
|
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:
|
|
/* 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 ((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())) {
|
|
|
|
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);
|
|
|
|
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_copy(ap, orig_ap);
|
|
vsyslog(level, trformat, ap);
|
|
va_end(ap);
|
|
}
|
|
|
|
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 struct id *id)
|
|
{
|
|
_log_report.object_group = group;
|
|
_log_report.object_group_id = id;
|
|
}
|
|
|
|
void log_set_report_object_name_and_id(const char *name, const struct id *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_command_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=["daemon_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;
|
|
}
|