On Mon, Nov 21, 2022 at 03:29:57PM +0600, Oleg Vasilev wrote:
Before, logs from deleted machines have been piling up, since there were no garbadge collection mechanism. Now virtlogd can be configured to periodically scan log folder for orphan logs with no recent modfications and delete it. Signed-off-by: Oleg Vasilev <oleg.vasilev@xxxxxxxxxxxxx> --- src/logging/log_daemon_config.c | 9 +++ src/logging/log_daemon_config.h | 3 + src/logging/log_handler.c | 108 +++++++++++++++++++++++++++++++ src/logging/test_virtlogd.aug.in | 2 + src/logging/virtlogd.aug | 2 + src/logging/virtlogd.conf | 7 ++ 6 files changed, 131 insertions(+) diff --git a/src/logging/log_daemon_config.c b/src/logging/log_daemon_config.c index 4436745488..248bd927d3 100644 --- a/src/logging/log_daemon_config.c +++ b/src/logging/log_daemon_config.c @@ -28,6 +28,7 @@ #include "virutil.h" #define VIR_FROM_THIS VIR_FROM_CONF +#define DEFAULT_LOG_ROOT LOCALSTATEDIR "/log/libvirt/" VIR_LOG_INIT("logging.log_daemon_config"); @@ -60,6 +61,7 @@ virLogDaemonConfigNew(bool privileged G_GNUC_UNUSED) data->admin_max_clients = 5000; data->max_size = 1024 * 1024 * 2; data->max_backups = 3; + data->max_age_days = 0; return data; } @@ -72,6 +74,7 @@ virLogDaemonConfigFree(virLogDaemonConfig *data) g_free(data->log_filters); g_free(data->log_outputs); + g_free(data->log_root); g_free(data); } @@ -94,6 +97,12 @@ virLogDaemonConfigLoadOptions(virLogDaemonConfig *data, return -1; if (virConfGetValueSizeT(conf, "max_backups", &data->max_backups) < 0) return -1; + if (virConfGetValueSizeT(conf, "max_age_days", &data->max_age_days) < 0) + return -1; + if (virConfGetValueString(conf, "log_root", &data->log_root) < 0) + return -1; + if (!data->log_root) + data->log_root = g_strdup(DEFAULT_LOG_ROOT); return 0; } diff --git a/src/logging/log_daemon_config.h b/src/logging/log_daemon_config.h index 2ab0f67c96..43922feedf 100644 --- a/src/logging/log_daemon_config.h +++ b/src/logging/log_daemon_config.h @@ -33,6 +33,9 @@ struct _virLogDaemonConfig { size_t max_backups; size_t max_size; + + char *log_root; + size_t max_age_days; }; diff --git a/src/logging/log_handler.c b/src/logging/log_handler.c index 7342404b00..2690d7519b 100644 --- a/src/logging/log_handler.c +++ b/src/logging/log_handler.c @@ -33,6 +33,7 @@ #include <unistd.h> #include <fcntl.h> #include <poll.h> +#include <fnmatch.h> #include "configmake.h" @@ -42,6 +43,10 @@ VIR_LOG_INIT("logging.log_handler"); #define DEFAULT_MODE 0600 +/* Cleanup log root (/var/log/libvirt) and all subfolders (e.g. /var/log/libvirt/qemu) */ +#define CLEANUP_OBSOLETE_LOG_DEPTH 1 +#define CLEANUP_OBSOLETE_LOG_TIMEOUT_MS (24 * 3600 * 1000) /* One day */ + typedef struct _virLogHandlerLogFile virLogHandlerLogFile; struct _virLogHandlerLogFile { virRotatingFileWriter *file; @@ -60,6 +65,8 @@ struct _virLogHandler { bool privileged; virLogDaemonConfig *config; + int cleanup_log_timer; + virLogHandlerLogFile **files; size_t nfiles; @@ -81,6 +88,93 @@ virLogHandlerOnceInit(void) VIR_ONCE_GLOBAL_INIT(virLogHandler); +static void +virLogHandlerMaybeCleanupObsoleteLog(virLogHandler *handler, + const char* path) { + size_t i; + bool remove = true; + + if (fnmatch("*.log*", path, 0)) + return; + + virObjectLock(handler);
If you use a lock guard: VIR_LOCK_GUARD lock = virObjectLockGuard(handler); then ...
+ for (i = 0; i < handler->nfiles; i++) { + virLogHandlerLogFile *file = handler->files[i]; + if (STRPREFIX(path, virRotatingFileWriterGetPath(file->file))) { + remove = false; + break; + }
... this condition body can just be replaced with `return`, and
+ } + virObjectUnlock(handler); + + if (!remove) + return; +
the unlocking and this above condition removed.
+ if (unlink(path) < 0) { + virReportSystemError(errno, _("Unable to delete %s"), path); + } +} + +static void +virLogHandlerCleanupObsoleteLogsFolder(virLogHandler *handler, + time_t oldest_to_keep, + const char *path, + int depth_left) +{ + DIR *dir; + struct dirent *entry; + char *newpath; + struct stat sb; + + if (virDirOpenIfExists(&dir, path) < 0) + return; + + while (virDirRead(dir, &entry, path) > 0) { + if (STREQ(entry->d_name, ".")) + continue; + if (STREQ(entry->d_name, "..")) + continue; + + newpath = g_strdup_printf("%s/%s", path, entry->d_name); +
If you make this newpath a `g_autofree char *` in the while loop then you can replace all `goto next` with `continue` and remove the label and VIR_FREE() call.
+ if (stat(newpath, &sb) < 0) { + virReportSystemError(errno, _("Unable to stat %s"), newpath); + goto next; + } + + if (S_ISDIR(sb.st_mode)) { + if (depth_left > 0) + virLogHandlerCleanupObsoleteLogsFolder(handler, oldest_to_keep, newpath, depth_left - 1); + goto next; + } + + if (!S_ISREG(sb.st_mode)) { + goto next; + } + + if (sb.st_mtim.tv_sec > oldest_to_keep) { + goto next; + } + + virLogHandlerMaybeCleanupObsoleteLog(handler, newpath); + + next: + VIR_FREE(newpath); + } + + virDirClose(dir); +} + +static void +virLogHandlerCleanupObsoleteLogs(int timer G_GNUC_UNUSED, void* opaque) +{ + virLogHandler *handler = opaque; + time_t oldest_to_keep = time(NULL) - 3600 * 24 * handler->config->max_age_days; + const char *log_root = handler->config->log_root; + + virLogHandlerCleanupObsoleteLogsFolder(handler, oldest_to_keep, log_root, CLEANUP_OBSOLETE_LOG_DEPTH); +} + static void virLogHandlerLogFileFree(virLogHandlerLogFile *file) @@ -201,7 +295,19 @@ virLogHandlerNew(bool privileged, handler->inhibitor = inhibitor; handler->opaque = opaque; + if (config->max_age_days > 0) { + handler->cleanup_log_timer = virEventAddTimeout(CLEANUP_OBSOLETE_LOG_TIMEOUT_MS, + virLogHandlerCleanupObsoleteLogs, + handler, NULL); + if (handler->cleanup_log_timer < 0) + goto error; + } + return handler; + + error: + virObjectUnref(handler); + return NULL; } @@ -344,6 +450,8 @@ virLogHandlerDispose(void *obj) virLogHandlerLogFileFree(handler->files[i]); } g_free(handler->files); + if (handler->cleanup_log_timer != 0) + virEventRemoveTimeout(handler->cleanup_log_timer); } diff --git a/src/logging/test_virtlogd.aug.in b/src/logging/test_virtlogd.aug.in index cd5b0d91f8..8dfad39506 100644 --- a/src/logging/test_virtlogd.aug.in +++ b/src/logging/test_virtlogd.aug.in @@ -9,3 +9,5 @@ module Test_virtlogd = { "admin_max_clients" = "5" } { "max_size" = "2097152" } { "max_backups" = "3" } + { "max_age_days" = "0" } + { "log_root" = "/var/log/libvirt" } diff --git a/src/logging/virtlogd.aug b/src/logging/virtlogd.aug index 0f1b290c72..bdf61dea6e 100644 --- a/src/logging/virtlogd.aug +++ b/src/logging/virtlogd.aug @@ -31,6 +31,8 @@ module Virtlogd = | int_entry "admin_max_clients" | int_entry "max_size" | int_entry "max_backups" + | int_entry "max_age_days" + | str_entry "log_root" (* Each entry in the config is one of the following three ... *) let entry = logging_entry diff --git a/src/logging/virtlogd.conf b/src/logging/virtlogd.conf index c53a1112bd..d88ce31327 100644 --- a/src/logging/virtlogd.conf +++ b/src/logging/virtlogd.conf @@ -101,3 +101,10 @@ # Maximum number of backup files to keep. Defaults to 3, # not including the primary active file #max_backups = 3 + +# Maximum age for log files to live after the last modification. +# Defaults to 0, which means "forever". +#max_age_days = 0 + +# Root of all logs managed by virtlogd. Used to GC logs from obsolete machines. +#log_root = "/var/log/libvirt"
One thing I am afraid of is that if there is some other log that is unused for max_age_days, but is not managed by virlogd, then it will get removed as well, but it might not be what users want. Thankfully this is an opt-in, but I would sleep more calmly if there was a warning with explanation of what this might do outside of its scope. One more thing is that if larger logs are rotated, then this would remove the old ones even if they are kept as number of backups. That seems like something that should be avoided (probably not even configurable).
-- 2.38.1
Attachment:
signature.asc
Description: PGP signature