/* SPDX-License-Identifier: GPL-3.0-or-later */
/*
* Copyright (C) 2018 - David Oberhollenzer
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* This program 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 General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program. If not, see .
*/
#include
#include
#include
#include
#include
#include
#include
#include
#include "backend.h"
#include "util.h"
#define SYSLOG_PATH "/var/log"
static const enum_map_t levels[] = {
{ "emergency", 0 },
{ "alert", 1 },
{ "critical", 2 },
{ "error", 3 },
{ "warning", 4 },
{ "notice", 5 },
{ "info", 6 },
{ "debug", 7 },
{ NULL, 0 },
};
static const enum_map_t facilities[] = {
{ "kernel.log", 0 },
{ "user.log", 1 },
{ "mail.log", 2 },
{ "daemon.log", 3 },
{ "auth.log", 4 },
{ "syslog.log", 5 },
{ "lpr.log", 6 },
{ "news.log", 7 },
{ "uucp.log", 8 },
{ "clock.log", 9 },
{ "authpriv.log", 10 },
{ "ftp.log", 11 },
{ "ntp.log", 12 },
{ "audit.log", 13 },
{ "alert.log", 14 },
{ "cron.log", 15 },
{ "local0.log", 16 },
{ "local1.log", 17 },
{ "local2.log", 18 },
{ "local3.log", 19 },
{ "local4.log", 20 },
{ "local5.log", 21 },
{ "local6.log", 22 },
{ "local7.log", 23 },
{ NULL, 0 },
};
typedef struct logfile_t {
struct logfile_t *next;
int fd;
char filename[];
} logfile_t;
typedef struct {
log_backend_t base;
logfile_t *list;
} log_backend_file_t;
static logfile_t *logfile_create(const char *filename)
{
logfile_t *file = calloc(1, sizeof(*file) + strlen(filename) + 1);
if (file == NULL) {
perror("calloc");
return NULL;
}
strcpy(file->filename, filename);
file->fd = open(file->filename, O_WRONLY | O_CREAT, 0640);
if (file->fd < 0) {
perror(file->filename);
goto fail;
}
if (lseek(file->fd, 0, SEEK_END))
goto fail_fd;
return file;
fail_fd:
close(file->fd);
fail:
free(file);
return NULL;
}
static int logfile_write(logfile_t *file, const syslog_msg_t *msg)
{
const char *lvl_str;
char timebuf[32];
struct tm tm;
lvl_str = enum_to_name(levels, msg->level);
if (lvl_str == NULL)
return -1;
gmtime_r(&msg->timestamp, &tm);
strftime(timebuf, sizeof(timebuf), "%FT%T", &tm);
dprintf(file->fd, "[%s][%s][%u] %s", timebuf, lvl_str, msg->pid,
msg->message);
fsync(file->fd);
return 0;
}
/*****************************************************************************/
static int file_backend_init(log_backend_t *log)
{
(void)log;
if (mkdir(SYSLOG_PATH, 0755)) {
if (errno != EEXIST) {
perror("mkdir " SYSLOG_PATH);
return -1;
}
}
if (chdir(SYSLOG_PATH)) {
perror("cd " SYSLOG_PATH);
return -1;
}
return 0;
}
static void file_backend_cleanup(log_backend_t *backend)
{
log_backend_file_t *log = (log_backend_file_t *)backend;
logfile_t *f;
while (log->list != NULL) {
f = log->list;
log->list = f->next;
close(f->fd);
free(f);
}
}
static int file_backend_write(log_backend_t *backend, const syslog_msg_t *msg)
{
log_backend_file_t *log = (log_backend_file_t *)backend;
const char *fac_name;
char *filename;
logfile_t *f;
size_t len;
fac_name = enum_to_name(facilities, msg->facility);
if (fac_name == NULL)
return -1;
if (msg->ident) {
len = strlen(msg->ident) + 1 + strlen(fac_name) + 1;
filename = alloca(len);
sprintf(filename, "%s/%s", msg->ident, fac_name);
} else {
filename = (char *)fac_name;
}
for (f = log->list; f != NULL; f = f->next) {
if (strcmp(filename, f->filename) == 0)
break;
}
if (f == NULL) {
if (msg->ident != NULL && mkdir(msg->ident, 0750) != 0 &&
errno != EEXIST) {
perror(msg->ident);
return -1;
}
f = logfile_create(filename);
if (f == NULL)
return -1;
f->next = log->list;
log->list = f;
}
return logfile_write(f, msg);
}
log_backend_file_t filebackend = {
.base = {
.init = file_backend_init,
.cleanup = file_backend_cleanup,
.write = file_backend_write,
},
.list = NULL,
};
log_backend_t *logmgr = (log_backend_t *)&filebackend;