45#define COLOR_RESET "\033[0m"
46#define COLOR_RED "\033[31m"
47#define COLOR_GREEN "\033[32m"
48#define COLOR_YELLOW "\033[33m"
49#define COLOR_BLUE "\033[34m"
50#define COLOR_MAGENTA "\033[35m"
51#define COLOR_CYAN "\033[36m"
52#define COLOR_WHITE "\033[37m"
100 openlog(identity, LOG_CONS | LOG_PID | LOG_NDELAY, LOG_LOCAL7);
135 HANDLE hOut = GetStdHandle(STD_OUTPUT_HANDLE);
136 if (hOut != INVALID_HANDLE_VALUE) {
138 if (GetConsoleMode(hOut, &dwMode)) {
139 dwMode |= ENABLE_VIRTUAL_TERMINAL_PROCESSING;
140 if (SetConsoleMode(hOut, dwMode)) {
148 if (isatty(fileno(stdout))) {
175 int fd = open(file, O_CREAT | O_APPEND | O_WRONLY, 0600);
184 n_log(
LOG_ERR,
"fdopen returned an invalid file descriptor pointer for %s:%d",
_str(file), fd);
211 va_copy(argcopy, pargs);
212 retval = vsnprintf(NULL, 0, format, argcopy);
226int vasprintf(
char** strp,
const char* fmt, va_list ap) {
231 Malloc(str,
char, (
size_t)len + 1 +
sizeof(
void*));
234 int r = vsnprintf(str, (
size_t)(len + 1), fmt, ap);
236 return free(str), -1;
267void _n_log(
int level,
const char* file,
const char* func,
int line,
const char* format, ...) {
282 char* syslogbuffer = NULL;
283 char* eventbuffer = NULL;
284 char* syslogsafebuffer = NULL;
287 const char* name =
"NULL";
292 const char* color =
"";
293 const char* color_reset =
"";
329 va_start(args, format);
330 if (
vasprintf(&syslogbuffer, format, args) == -1) {
335 syslogsafebuffer = syslogbuffer ? syslogbuffer :
"<log: vasprintf failed>";
342 for (
char* p = syslogsafebuffer; *p; p++) {
343 if (*p ==
'"' || *p ==
'&' || *p ==
'|' || *p ==
'<' || *p ==
'>' || *p ==
'^') *p =
'_';
346 int snprintf_ret = snprintf(NULL, 0,
"start /B EventCreate /t %s /id 666 /l APPLICATION /so %s /d \"%s\" > NUL 2>&1",
prioritynames[level].w_name, name, syslogsafebuffer);
347 if (snprintf_ret > 0) {
348 needed = (
unsigned long long)snprintf_ret;
349 Malloc(eventbuffer,
char, needed + 4);
351 snprintf(eventbuffer, needed + 4,
"start /B EventCreate /t %s /id 666 /l APPLICATION /so %s /d \"%s\" > NUL 2>&1",
prioritynames[level].w_name, name, syslogsafebuffer);
356 syslog(level,
"%s->%s:%d %s",
_str(file),
_str(func), line, syslogsafebuffer);
362 fprintf(out,
"%s%s:%jd:%s->%s:%d ", color,
prioritynames[level].c_name, (intmax_t)time(NULL),
_str(file),
_str(func), line);
364 va_start(args, format);
365 vfprintf(out, format, args);
368 fprintf(out,
"%s\n", color_reset);
393 n_log(
LOG_ERR,
"Log struct already allocated without file");
397 if (!pathname || !opt) {
408 pthread_mutex_init(&(*log)->LOG_MUTEX, NULL);
411 if (strcmp(opt,
"a") == 0 || strcmp(opt,
"a+") == 0) {
412 flags = O_CREAT | O_APPEND | O_WRONLY;
413 }
else if (strcmp(opt,
"w") == 0 || strcmp(opt,
"w+") == 0) {
414 flags = O_CREAT | O_TRUNC | O_WRONLY;
417 pthread_mutex_destroy(&(*log)->LOG_MUTEX);
423 int fd = open(pathname, flags, 0600);
425 n_log(
LOG_ERR,
"Failed to open '%s' with 0600 permissions",
_str(pathname));
426 pthread_mutex_destroy(&(*log)->LOG_MUTEX);
432 (*log)->
file = fdopen(fd, opt);
436 pthread_mutex_destroy(&(*log)->LOG_MUTEX);
461 vsnprintf(str,
sizeof(str), pat, arg);
466 fprintf(log->
file,
"%s", str);
#define FreeNoLog(__ptr)
Free Handler without log.
#define Malloc(__ptr, __struct, __size)
Malloc Handler to get errors and set to 0.
#define __n_assert(__ptr, __ret)
macro to assert things
#define _str(__PTR)
define true
#define Free(__ptr)
Free Handler to get errors.
pthread_mutex_t LOG_MUTEX
mutex for thread-safe writting
#define LOG_ALERT
action must be taken immediately
#define LOG_SYSJRNL
to sysjrnl
FILE * get_log_file(void)
return the current log_file
int write_safe_log(TS_LOG *log, char *pat,...)
write to a thread-safe logging file
int open_safe_logging(TS_LOG **log, char *pathname, char *opt)
Open a thread-safe logging file.
#define LOG_EMERG
system is unusable
#define n_log(__LEVEL__,...)
Logging function wrapper to get line and func.
#define LOG_FILE
internal, logging to file
void close_sysjrnl(void)
Close syslog connection or clean internals for event log.
#define LOG_DEBUG
debug-level messages
#define LOG_ERR
error conditions
char * open_sysjrnl(const char *identity)
Open connection to syslog or create internals for event log.
#define LOG_STDERR
internal, default LOG_TYPE
#define LOG_CRIT
critical conditions
int close_safe_logging(TS_LOG *log)
close a thread-safe logging file
int set_log_file(char *file)
Set the logging to a file instead of stderr.
void set_log_level(const int log_level)
Set the global log level value ( static int LOG_LEVEL )
#define LOG_NOTICE
normal but significant condition
#define LOG_WARNING
warning conditions
#define LOG_NULL
no log output
#define LOG_INFO
informational
void _n_log(int level, const char *file, const char *func, int line, const char *format,...)
Logging function.
int get_log_level(void)
Get the global log level value.
ThreadSafe LOGging structure.
Common headers and low-level functions & define.
static LOG_LEVELS prioritynames[]
array of log levels
static char * proc_name
static proc name, for windows event log
static int LOG_LEVEL
static global maximum wanted log level value
static int LOG_TYPE
static global logging type ( STDERR, FILE, SYSJRNL )
int vasprintf(char **strp, const char *fmt, va_list ap)
snprintf from a va_list, helper for asprintf
int asprintf(char *strp[], const char *fmt,...)
snprintf from a va_list
static int terminal_support_colors
static FILE * log_file
static FILE handling if logging to file is enabled
char * c_name
string of log type
int _vscprintf_so(const char *format, va_list pargs)
compute the size of a string made with format 'fmt' and arguments in the va_list 'ap',...
internal struct to handle log types