/*
* libxl_logger.c: libxl logger implementation
*
* Copyright (c) 2016 SUSE LINUX Products GmbH, Nuernberg, Germany.
*
* This library is free software; you can redistribute it and/or
* modify it under the terms of the GNU Lesser General Public
* License as published by the Free Software Foundation; either
* version 2.1 of the License, or (at your option) any later version.
*
* This library is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public
* License along with this library. If not, see
* .
*
* Authors:
* Cédric Bosdonnat
*/
#include
#include
#include "internal.h"
#include "libxl_logger.h"
#include "util/viralloc.h"
#include "util/virerror.h"
#include "util/virfile.h"
#include "util/virhash.h"
#include "util/virstring.h"
#include "util/virtime.h"
#define VIR_FROM_THIS VIR_FROM_LIBXL
VIR_LOG_INIT("libxl.libxl_logger");
typedef struct xentoollog_logger_libvirt xentoollog_logger_libvirt;
struct xentoollog_logger_libvirt {
xentoollog_logger vtable;
xentoollog_level minLevel;
const char *logDir;
/* map storing the opened fds: "domid" -> FILE* */
virHashTablePtr files;
FILE *defaultLogFile;
};
static void
libxlLoggerFileFree(void *payload, const void *key ATTRIBUTE_UNUSED)
{
FILE *file = payload;
VIR_FORCE_FCLOSE(file);
file = NULL;
}
ATTRIBUTE_FMT_PRINTF(5, 0) static void
libvirt_vmessage(xentoollog_logger *logger_in,
xentoollog_level level,
int errnoval,
const char *context,
const char *format,
va_list args)
{
xentoollog_logger_libvirt *lg = (xentoollog_logger_libvirt *)logger_in;
FILE *logFile = lg->defaultLogFile;
char timestamp[VIR_TIME_STRING_BUFLEN];
char *message = NULL;
char *start, *end;
char ebuf[1024];
VIR_DEBUG("libvirt_vmessage: context='%s' format='%s'", context, format);
if (level < lg->minLevel)
return;
if (virVasprintf(&message, format, args) < 0)
return;
/* Should we print to a domain-specific log file? */
if ((start = strstr(message, ": Domain ")) &&
(end = strstr(start + 9, ":"))) {
FILE *domainLogFile;
VIR_DEBUG("Found domain log message");
start = start + 9;
*end = '\0';
domainLogFile = virHashLookup(lg->files, start);
if (domainLogFile)
logFile = domainLogFile;
*end = ':';
}
/* Do the actual print to the log file */
if (virTimeStringNowRaw(timestamp) < 0)
timestamp[0] = '\0';
fprintf(logFile, "%s: ", timestamp);
if (context)
fprintf(logFile, "%s: ", context);
fprintf(logFile, "%s", message);
if (errnoval >= 0)
fprintf(logFile, ": %s", virStrerror(errnoval, ebuf, sizeof(ebuf)));
fputc('\n', logFile);
fflush(logFile);
VIR_FREE(message);
}
static void
libvirt_progress(xentoollog_logger *logger_in ATTRIBUTE_UNUSED,
const char *context ATTRIBUTE_UNUSED,
const char *doingwhat ATTRIBUTE_UNUSED,
int percent ATTRIBUTE_UNUSED,
unsigned long done ATTRIBUTE_UNUSED,
unsigned long total ATTRIBUTE_UNUSED)
{
/* This function purposedly does nothing: it's no logging info */
}
static void
libvirt_destroy(xentoollog_logger *logger_in)
{
xentoollog_logger_libvirt *lg = (xentoollog_logger_libvirt*)logger_in;
VIR_FREE(lg);
}
libxlLoggerPtr
libxlLoggerNew(const char *logDir, virLogPriority minLevel)
{
xentoollog_logger_libvirt logger;
libxlLoggerPtr logger_out = NULL;
char *path = NULL;
switch (minLevel) {
case VIR_LOG_DEBUG:
logger.minLevel = XTL_DEBUG;
break;
case VIR_LOG_INFO:
logger.minLevel = XTL_INFO;
break;
case VIR_LOG_WARN:
logger.minLevel = XTL_WARN;
break;
case VIR_LOG_ERROR:
logger.minLevel = XTL_ERROR;
break;
}
logger.logDir = logDir;
if ((logger.files = virHashCreate(3, libxlLoggerFileFree)) == NULL)
return NULL;
if (virAsprintf(&path, "%s/libxl-driver.log", logDir) < 0)
goto error;
if ((logger.defaultLogFile = fopen(path, "a")) == NULL)
goto error;
logger_out = XTL_NEW_LOGGER(libvirt, logger);
cleanup:
VIR_FREE(path);
return logger_out;
error:
virHashFree(logger.files);
goto cleanup;
}
void
libxlLoggerFree(libxlLoggerPtr logger)
{
xentoollog_logger *xtl_logger = (xentoollog_logger*)logger;
if (logger->defaultLogFile)
VIR_FORCE_FCLOSE(logger->defaultLogFile);
virHashFree(logger->files);
xtl_logger_destroy(xtl_logger);
}
void
libxlLoggerOpenFile(libxlLoggerPtr logger,
int id,
const char *name,
const char *domain_config)
{
char *path = NULL;
FILE *logFile = NULL;
char *domidstr = NULL;
char ebuf[1024];
if (virAsprintf(&path, "%s/%s.log", logger->logDir, name) < 0 ||
virAsprintf(&domidstr, "%d", id) < 0)
goto cleanup;
if (!(logFile = fopen(path, "a"))) {
VIR_WARN("Failed to open log file %s: %s",
path, virStrerror(errno, ebuf, sizeof(ebuf)));
goto cleanup;
}
ignore_value(virHashAddEntry(logger->files, domidstr, logFile));
/* domain_config is non NULL only when starting a new domain */
if (domain_config) {
fprintf(logFile, "Domain start: %s\n", domain_config);
fflush(logFile);
}
cleanup:
VIR_FREE(path);
VIR_FREE(domidstr);
}
void
libxlLoggerCloseFile(libxlLoggerPtr logger, int id)
{
char *domidstr = NULL;
if (virAsprintf(&domidstr, "%d", id) < 0)
return;
ignore_value(virHashRemoveEntry(logger->files, domidstr));
VIR_FREE(domidstr);
}