mirror of
https://github.com/pygos/init.git
synced 2024-11-05 12:17:10 +01:00
29e4fc5607
- Add more fine grained control over how log rotation is supposed to behave - Add command line option processing to usyslogd - Expose log rotation control via command line switches - Add default values to usyslogd service for pygos use case Signed-off-by: David Oberhollenzer <david.oberhollenzer@tele2.at>
294 lines
5.8 KiB
C
294 lines
5.8 KiB
C
/* 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 <https://www.gnu.org/licenses/>.
|
|
*/
|
|
#include <sys/types.h>
|
|
#include <sys/stat.h>
|
|
#include <stdlib.h>
|
|
#include <unistd.h>
|
|
#include <string.h>
|
|
#include <stdio.h>
|
|
#include <fcntl.h>
|
|
#include <errno.h>
|
|
|
|
#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;
|
|
size_t size;
|
|
int fd;
|
|
char filename[];
|
|
} logfile_t;
|
|
|
|
|
|
typedef struct {
|
|
log_backend_t base;
|
|
logfile_t *list;
|
|
size_t maxsize;
|
|
int flags;
|
|
} log_backend_file_t;
|
|
|
|
|
|
static int logfile_open(logfile_t *file)
|
|
{
|
|
struct stat sb;
|
|
|
|
file->fd = open(file->filename, O_WRONLY | O_CREAT, 0640);
|
|
if (file->fd < 0) {
|
|
perror(file->filename);
|
|
return -1;
|
|
}
|
|
|
|
if (lseek(file->fd, 0, SEEK_END))
|
|
goto fail;
|
|
|
|
if (fstat(file->fd, &sb))
|
|
goto fail;
|
|
|
|
file->size = sb.st_size;
|
|
return 0;
|
|
fail:
|
|
close(file->fd);
|
|
file->fd = -1;
|
|
return -1;
|
|
}
|
|
|
|
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);
|
|
|
|
if (logfile_open(file)) {
|
|
free(file);
|
|
return NULL;
|
|
}
|
|
|
|
return file;
|
|
}
|
|
|
|
static int logfile_write(logfile_t *file, const syslog_msg_t *msg)
|
|
{
|
|
const char *lvl_str;
|
|
char timebuf[32];
|
|
struct tm tm;
|
|
int ret;
|
|
|
|
if (file->fd < 0 && logfile_open(file) != 0)
|
|
return -1;
|
|
|
|
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);
|
|
|
|
ret = dprintf(file->fd, "[%s][%s][%u] %s", timebuf, lvl_str, msg->pid,
|
|
msg->message);
|
|
|
|
fsync(file->fd);
|
|
|
|
if (ret > 0)
|
|
file->size += ret;
|
|
return 0;
|
|
}
|
|
|
|
static int logfile_rotate(logfile_t *f, int flags)
|
|
{
|
|
char timebuf[32];
|
|
char *filename;
|
|
struct tm tm;
|
|
time_t now;
|
|
|
|
if (flags & LOG_ROTATE_OVERWRITE) {
|
|
strcpy(timebuf, "1");
|
|
} else {
|
|
now = time(NULL);
|
|
gmtime_r(&now, &tm);
|
|
strftime(timebuf, sizeof(timebuf), "%FT%T", &tm);
|
|
}
|
|
|
|
filename = alloca(strlen(f->filename) + strlen(timebuf) + 2);
|
|
sprintf(filename, "%s.%s", f->filename, timebuf);
|
|
|
|
if (rename(f->filename, filename)) {
|
|
perror(filename);
|
|
return -1;
|
|
}
|
|
|
|
close(f->fd);
|
|
logfile_open(f);
|
|
return 0;
|
|
}
|
|
|
|
/*****************************************************************************/
|
|
|
|
static int file_backend_init(log_backend_t *backend, int flags,
|
|
size_t sizelimit)
|
|
{
|
|
log_backend_file_t *log = (log_backend_file_t *)backend;
|
|
|
|
if (mkdir(SYSLOG_PATH, 0755)) {
|
|
if (errno != EEXIST) {
|
|
perror("mkdir " SYSLOG_PATH);
|
|
return -1;
|
|
}
|
|
}
|
|
|
|
if (chdir(SYSLOG_PATH)) {
|
|
perror("cd " SYSLOG_PATH);
|
|
return -1;
|
|
}
|
|
|
|
log->flags = flags;
|
|
log->maxsize = sizelimit;
|
|
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;
|
|
}
|
|
|
|
if (logfile_write(f, msg))
|
|
return -1;
|
|
|
|
if ((log->flags & LOG_ROTATE_SIZE_LIMIT) && f->size >= log->maxsize)
|
|
logfile_rotate(f, log->flags);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static void file_backend_rotate(log_backend_t *backend)
|
|
{
|
|
log_backend_file_t *log = (log_backend_file_t *)backend;
|
|
logfile_t *f;
|
|
|
|
for (f = log->list; f != NULL; f = f->next)
|
|
logfile_rotate(f, log->flags);
|
|
}
|
|
|
|
log_backend_file_t filebackend = {
|
|
.base = {
|
|
.init = file_backend_init,
|
|
.cleanup = file_backend_cleanup,
|
|
.write = file_backend_write,
|
|
.rotate = file_backend_rotate,
|
|
},
|
|
.list = NULL,
|
|
};
|
|
|
|
log_backend_t *logmgr = (log_backend_t *)&filebackend;
|