
On 30.11.2022 16:49, Martin Kletzander wrote:
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@virtuozzo.com> --- 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
...
+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.
Right, thanks, not yet used to the autoptr functionality in C.
+ 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); +} +
...
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 scop
Yes, this is a valid concern. Warning here in config file would be helpful, maybe also warn on stdout during startup if this is enabled?
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).
I think I will delete all log files (rotated and fresh) from one domain at the same moment, namely, when the fresh part becomes older than max_age_days. Let me post a v2. Thanks, Oleg
-- 2.38.1