2014-11-08 13:27:43 +01:00
|
|
|
/* $OpenBSD$ */
|
2007-07-09 21:04:12 +02:00
|
|
|
|
|
|
|
/*
|
2016-01-19 16:59:12 +01:00
|
|
|
* Copyright (c) 2007 Nicholas Marriott <nicholas.marriott@gmail.com>
|
2007-07-09 21:04:12 +02:00
|
|
|
*
|
|
|
|
* Permission to use, copy, modify, and distribute this software for any
|
|
|
|
* purpose with or without fee is hereby granted, provided that the above
|
|
|
|
* copyright notice and this permission notice appear in all copies.
|
|
|
|
*
|
|
|
|
* THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
|
|
|
|
* WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
|
|
|
|
* MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
|
|
|
|
* ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
|
|
|
|
* WHATSOEVER RESULTING FROM LOSS OF MIND, USE, DATA OR PROFITS, WHETHER
|
|
|
|
* IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING
|
|
|
|
* OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <sys/types.h>
|
|
|
|
|
|
|
|
#include <errno.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
2015-11-24 22:19:46 +01:00
|
|
|
#include <unistd.h>
|
2007-07-09 21:04:12 +02:00
|
|
|
|
|
|
|
#include "tmux.h"
|
|
|
|
|
2015-11-24 22:19:46 +01:00
|
|
|
static FILE *log_file;
|
|
|
|
static int log_level;
|
2007-07-09 21:04:12 +02:00
|
|
|
|
2015-11-24 22:19:46 +01:00
|
|
|
static void log_event_cb(int, const char *);
|
|
|
|
static void log_vwrite(const char *, va_list);
|
2007-07-26 01:13:18 +02:00
|
|
|
|
2019-11-03 11:11:37 +01:00
|
|
|
static int is_log_stdout(void)
|
|
|
|
{
|
|
|
|
return fileno(log_file) <= 2;
|
|
|
|
}
|
|
|
|
|
2012-03-18 02:58:09 +01:00
|
|
|
/* Log callback for libevent. */
|
2015-11-24 22:19:46 +01:00
|
|
|
static void
|
2015-11-18 15:27:44 +01:00
|
|
|
log_event_cb(__unused int severity, const char *msg)
|
2012-03-18 02:58:09 +01:00
|
|
|
{
|
2014-03-07 16:49:09 +01:00
|
|
|
log_debug("%s", msg);
|
2012-03-18 02:58:09 +01:00
|
|
|
}
|
|
|
|
|
2015-11-24 22:19:46 +01:00
|
|
|
/* Increment log level. */
|
|
|
|
void
|
|
|
|
log_add_level(void)
|
|
|
|
{
|
|
|
|
log_level++;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Get log level. */
|
|
|
|
int
|
|
|
|
log_get_level(void)
|
|
|
|
{
|
|
|
|
return (log_level);
|
|
|
|
}
|
|
|
|
|
2019-11-03 11:11:37 +01:00
|
|
|
void
|
|
|
|
log_open_fp(FILE *f)
|
|
|
|
{
|
2020-03-11 16:30:33 +01:00
|
|
|
if (log_file == f)
|
|
|
|
return;
|
|
|
|
|
2019-11-03 11:11:37 +01:00
|
|
|
if (log_file != NULL && !is_log_stdout())
|
|
|
|
fclose(log_file);
|
|
|
|
|
|
|
|
log_file = f;
|
|
|
|
|
|
|
|
setvbuf(log_file, NULL, _IOLBF, 0);
|
|
|
|
event_set_log_callback(log_event_cb);
|
|
|
|
}
|
|
|
|
|
2008-10-09 07:31:04 +02:00
|
|
|
/* Open logging to file. */
|
2008-08-08 19:35:42 +02:00
|
|
|
void
|
2015-11-24 22:19:46 +01:00
|
|
|
log_open(const char *name)
|
2008-08-08 19:35:42 +02:00
|
|
|
{
|
2015-11-24 22:19:46 +01:00
|
|
|
char *path;
|
|
|
|
|
|
|
|
if (log_level == 0)
|
|
|
|
return;
|
|
|
|
|
2015-12-23 11:57:43 +01:00
|
|
|
xasprintf(&path, "tmate-%s-%ld.log", name, (long)getpid());
|
2019-11-03 11:11:37 +01:00
|
|
|
FILE *f = fopen(path, "w");
|
2015-11-24 22:19:46 +01:00
|
|
|
free(path);
|
2019-11-03 11:11:37 +01:00
|
|
|
if (f)
|
|
|
|
log_open_fp(f);
|
2007-07-09 21:04:12 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Close logging. */
|
|
|
|
void
|
|
|
|
log_close(void)
|
|
|
|
{
|
2019-11-03 11:11:37 +01:00
|
|
|
if (log_file != NULL && !is_log_stdout())
|
2008-08-08 19:35:42 +02:00
|
|
|
fclose(log_file);
|
2014-03-07 17:05:29 +01:00
|
|
|
log_file = NULL;
|
2007-07-09 21:04:12 +02:00
|
|
|
|
2012-03-18 02:58:09 +01:00
|
|
|
event_set_log_callback(NULL);
|
2007-07-09 21:04:12 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Write a log message. */
|
2019-11-07 15:08:17 +01:00
|
|
|
__attribute__((__format__(__printf__, 1, 0)))
|
2015-11-24 22:19:46 +01:00
|
|
|
static void
|
2012-05-30 15:42:57 +02:00
|
|
|
log_vwrite(const char *msg, va_list ap)
|
2007-07-09 21:04:12 +02:00
|
|
|
{
|
2015-09-01 21:14:43 +02:00
|
|
|
char *fmt, *out;
|
2015-08-29 02:24:44 +02:00
|
|
|
struct timeval tv;
|
2012-05-30 15:42:57 +02:00
|
|
|
|
2012-05-30 17:01:57 +02:00
|
|
|
if (log_file == NULL)
|
|
|
|
return;
|
|
|
|
|
2015-09-01 21:14:43 +02:00
|
|
|
if (vasprintf(&fmt, msg, ap) == -1)
|
2012-05-30 15:42:57 +02:00
|
|
|
exit(1);
|
2015-09-01 21:14:43 +02:00
|
|
|
if (stravis(&out, fmt, VIS_OCTAL|VIS_CSTYLE|VIS_TAB|VIS_NL) == -1)
|
|
|
|
exit(1);
|
|
|
|
|
|
|
|
gettimeofday(&tv, NULL);
|
2019-11-03 11:11:37 +01:00
|
|
|
|
|
|
|
if (is_log_stdout()) {
|
|
|
|
if (fprintf(log_file, "%s\n", out) == -1)
|
|
|
|
exit(1);
|
|
|
|
} else {
|
|
|
|
if (fprintf(log_file, "%lld.%06d %s\n", (long long)tv.tv_sec,
|
|
|
|
(int)tv.tv_usec, out) == -1)
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
|
2012-05-30 15:42:57 +02:00
|
|
|
fflush(log_file);
|
2015-09-01 21:14:43 +02:00
|
|
|
|
|
|
|
free(out);
|
2012-05-30 15:42:57 +02:00
|
|
|
free(fmt);
|
2007-07-09 21:04:12 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Log a debug message. */
|
2014-10-21 01:57:13 +02:00
|
|
|
void
|
2019-11-03 11:11:37 +01:00
|
|
|
log_emit(int level, const char *msg, ...)
|
2007-07-09 21:04:12 +02:00
|
|
|
{
|
|
|
|
va_list ap;
|
|
|
|
|
2019-11-03 11:11:37 +01:00
|
|
|
if (log_level < level)
|
|
|
|
return;
|
|
|
|
|
2014-03-07 17:05:29 +01:00
|
|
|
va_start(ap, msg);
|
|
|
|
log_vwrite(msg, ap);
|
|
|
|
va_end(ap);
|
2007-07-09 21:04:12 +02:00
|
|
|
}
|
|
|
|
|
2014-03-07 17:05:29 +01:00
|
|
|
/* Log a critical error with error string and die. */
|
2019-11-07 15:08:17 +01:00
|
|
|
__attribute__((__format__(__printf__, 1, 0)))
|
2014-10-21 01:57:13 +02:00
|
|
|
__dead void
|
2015-11-18 14:06:54 +01:00
|
|
|
fatal(const char *msg, ...)
|
2007-07-09 21:04:12 +02:00
|
|
|
{
|
2014-03-07 17:05:29 +01:00
|
|
|
char *fmt;
|
|
|
|
va_list ap;
|
2007-07-09 21:04:12 +02:00
|
|
|
|
|
|
|
va_start(ap, msg);
|
2014-03-07 17:05:29 +01:00
|
|
|
if (asprintf(&fmt, "fatal: %s: %s", msg, strerror(errno)) == -1)
|
|
|
|
exit(1);
|
2019-11-07 15:08:17 +01:00
|
|
|
msg = fmt;
|
|
|
|
log_vwrite(msg, ap);
|
2014-03-07 17:05:29 +01:00
|
|
|
exit(1);
|
2007-07-09 21:04:12 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Log a critical error and die. */
|
2019-11-07 15:08:17 +01:00
|
|
|
__attribute__((__format__(__printf__, 1, 0)))
|
2014-10-21 01:57:13 +02:00
|
|
|
__dead void
|
2015-11-18 14:06:54 +01:00
|
|
|
fatalx(const char *msg, ...)
|
2007-07-09 21:04:12 +02:00
|
|
|
{
|
2014-03-07 17:05:29 +01:00
|
|
|
char *fmt;
|
|
|
|
va_list ap;
|
2007-07-09 21:04:12 +02:00
|
|
|
|
|
|
|
va_start(ap, msg);
|
2014-03-07 17:05:29 +01:00
|
|
|
if (asprintf(&fmt, "fatal: %s", msg) == -1)
|
|
|
|
exit(1);
|
2019-11-07 15:08:17 +01:00
|
|
|
msg = fmt;
|
|
|
|
log_vwrite(msg, ap);
|
2014-03-07 17:05:29 +01:00
|
|
|
exit(1);
|
2007-07-09 21:04:12 +02:00
|
|
|
}
|