2 * Sylpheed -- a GTK+ based, lightweight, and fast e-mail client
3 * Copyright (C) 1999-2012 Hiroyuki Yamamoto and the Claws Mail team
5 * This program is free software; you can redistribute it and/or modify
6 * it under the terms of the GNU General Public License as published by
7 * the Free Software Foundation; either version 3 of the License, or
8 * (at your option) any later version.
10 * This program is distributed in the hope that it will be useful,
11 * but WITHOUT ANY WARRANTY; without even the implied warranty of
12 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
13 * GNU General Public License for more details.
15 * You should have received a copy of the GNU General Public License
16 * along with this program. If not, see <http://www.gnu.org/licenses/>.
22 #include "claws-features.h"
32 #include <glib/gi18n.h>
37 #include "file-utils.h"
39 #define FWRITE(_b,_s,_n,_f) if (claws_fwrite(_b,_s,_n,_f) != _n) { \
40 g_message("log claws_fwrite failed!\n"); \
43 #define FPUTS(_b,_f) if (claws_fputs(_b,_f) == EOF) { \
44 g_message("log claws_fputs failed!\n"); \
47 #define FFLUSH(_f) if (fflush(_f) != 0) { \
48 g_message("log fflush failed!\n"); \
52 static FILE *log_fp[LOG_INSTANCE_MAX] = {
57 static size_t log_size[LOG_INSTANCE_MAX] = {
62 static gchar *log_filename[LOG_INSTANCE_MAX] = {
68 static gboolean log_error_capability[LOG_INSTANCE_MAX] = {
73 typedef struct _LogInstanceData LogInstanceData;
75 struct _LogInstanceData {
78 int *prefs_logwin_width;
79 int *prefs_logwin_height;
82 static LogInstanceData log_instances[LOG_INSTANCE_MAX] = {
83 { LOG_APPEND_TEXT_HOOKLIST, NULL, NULL, NULL },
84 { DEBUG_FILTERING_APPEND_TEXT_HOOKLIST, NULL, NULL, NULL }
87 gboolean prefs_common_enable_log_standard(void);
88 gboolean prefs_common_enable_log_warning(void);
89 gboolean prefs_common_enable_log_error(void);
90 gboolean prefs_common_enable_log_status(void);
92 static gboolean invoke_hook_cb (gpointer data)
94 LogText *logtext = (LogText *)data;
95 hooks_invoke(get_log_hook(logtext->instance), logtext);
96 g_free(logtext->text);
101 void set_log_file(LogInstance instance, const gchar *filename)
103 gchar *fullname = NULL;
104 if (log_fp[instance])
107 if (!g_path_is_absolute(filename)) {
108 fullname = g_strconcat(get_rc_dir(), G_DIR_SEPARATOR_S,
111 fullname = g_strdup(filename);
113 /* backup old logfile if existing */
114 if (is_file_exist(fullname)) {
117 backupname = g_strconcat(fullname, ".bak", NULL);
118 claws_unlink(backupname);
119 if (g_rename(fullname, backupname) < 0)
120 FILE_OP_ERROR(fullname, "rename");
124 log_fp[instance] = claws_fopen(fullname, "wb");
125 if (!log_fp[instance]) {
126 FILE_OP_ERROR(fullname, "claws_fopen");
127 log_filename[instance] = NULL;
131 log_filename[instance] = g_strdup(fullname);
132 log_size[instance] = 0;
136 void close_log_file(LogInstance instance)
138 if (log_fp[instance]) {
139 claws_safe_fclose(log_fp[instance]);
140 log_fp[instance] = NULL;
141 log_size[instance] = 0;
142 g_free(log_filename[instance]);
143 log_filename[instance] = NULL;
147 static void rotate_log(LogInstance instance)
149 if (log_size[instance] > 10 * 1024* 1024) {
150 gchar *filename = g_strdup(log_filename[instance]);
151 debug_print("rotating %s\n", filename);
152 close_log_file(instance);
153 set_log_file(instance, filename);
158 const char *get_log_hook(LogInstance instance)
160 return log_instances[instance].hook;
163 void set_log_title(LogInstance instance, gchar *title)
165 log_instances[instance].title = title;
168 gchar *get_log_title(LogInstance instance)
170 return log_instances[instance].title;
173 void set_log_prefs(LogInstance instance, int* logwin_width, int* logwin_height)
175 log_instances[instance].prefs_logwin_width = logwin_width;
176 log_instances[instance].prefs_logwin_height = logwin_height;
179 void get_log_prefs(LogInstance instance, int** logwin_width, int** logwin_height)
182 *logwin_width = log_instances[instance].prefs_logwin_width;
184 *logwin_height = log_instances[instance].prefs_logwin_height;
187 gboolean get_log_error_capability(LogInstance instance)
189 return log_error_capability[instance];
193 void log_print(LogInstance instance, const gchar *format, ...)
196 gchar buf[BUFFSIZE + LOG_TIME_LEN];
198 LogText *logtext = g_new0(LogText, 1);
202 strftime(buf, LOG_TIME_LEN + 1, "[%H:%M:%S] ", localtime_r(&t, &buft));
204 va_start(args, format);
205 g_vsnprintf(buf + LOG_TIME_LEN, BUFFSIZE, format, args);
208 if (debug_get_mode()) g_print("%s", buf);
210 logtext->instance = instance;
211 logtext->text = g_strdup(buf);
212 logtext->type = LOG_NORMAL;
214 g_timeout_add(0, invoke_hook_cb, logtext);
216 if (log_fp[instance] && prefs_common_enable_log_standard()) {
217 FPUTS(buf, log_fp[instance])
218 log_size[instance] += strlen(buf);
219 FFLUSH(log_fp[instance])
220 rotate_log(instance);
224 void log_message(LogInstance instance, const gchar *format, ...)
227 gchar buf[BUFFSIZE + LOG_TIME_LEN];
229 LogText *logtext = g_new0(LogText, 1);
233 strftime(buf, LOG_TIME_LEN + 1, "[%H:%M:%S] ", localtime_r(&t, &buft));
235 va_start(args, format);
236 g_vsnprintf(buf + LOG_TIME_LEN, BUFFSIZE, format, args);
239 if (debug_get_mode()) g_message("%s", buf + LOG_TIME_LEN);
241 logtext->instance = instance;
242 logtext->text = g_strdup(buf + LOG_TIME_LEN);
243 logtext->type = LOG_MSG;
245 g_timeout_add(0, invoke_hook_cb, logtext);
247 if (log_fp[instance] && prefs_common_enable_log_standard()) {
248 FWRITE(buf, 1, LOG_TIME_LEN, log_fp[instance])
249 FPUTS("* message: ", log_fp[instance])
250 log_size[instance] += strlen("* message: ");
251 FPUTS(buf + LOG_TIME_LEN, log_fp[instance])
252 log_size[instance] += strlen(buf);
253 FFLUSH(log_fp[instance])
254 rotate_log(instance);
258 void log_warning(LogInstance instance, const gchar *format, ...)
261 gchar buf[BUFFSIZE + LOG_TIME_LEN];
263 LogText *logtext = g_new0(LogText, 1);
267 strftime(buf, LOG_TIME_LEN + 1, "[%H:%M:%S] ", localtime_r(&t, &buft));
269 va_start(args, format);
270 g_vsnprintf(buf + LOG_TIME_LEN, BUFFSIZE, format, args);
273 g_warning("%s", buf);
275 logtext->instance = instance;
276 logtext->text = g_strdup(buf + LOG_TIME_LEN);
277 logtext->type = LOG_WARN;
279 g_timeout_add(0, invoke_hook_cb, logtext);
281 if (log_fp[instance] && prefs_common_enable_log_warning()) {
282 FWRITE(buf, 1, LOG_TIME_LEN, log_fp[instance])
283 FPUTS("** warning: ", log_fp[instance])
284 log_size[instance] += strlen("** warning: ");
285 FPUTS(buf + LOG_TIME_LEN, log_fp[instance])
286 log_size[instance] += strlen(buf);
287 FFLUSH(log_fp[instance])
288 rotate_log(instance);
292 void log_error(LogInstance instance, const gchar *format, ...)
295 gchar buf[BUFFSIZE + LOG_TIME_LEN];
297 LogText *logtext = g_new0(LogText, 1);
301 strftime(buf, LOG_TIME_LEN + 1, "[%H:%M:%S] ", localtime_r(&t, &buft));
303 va_start(args, format);
304 g_vsnprintf(buf + LOG_TIME_LEN, BUFFSIZE, format, args);
307 g_warning("%s", buf);
309 logtext->instance = instance;
310 logtext->text = g_strdup(buf + LOG_TIME_LEN);
311 logtext->type = LOG_ERROR;
313 g_timeout_add(0, invoke_hook_cb, logtext);
315 if (log_fp[instance] && prefs_common_enable_log_error()) {
316 FWRITE(buf, 1, LOG_TIME_LEN, log_fp[instance])
317 FPUTS("*** error: ", log_fp[instance])
318 log_size[instance] += strlen("*** error: ");
319 FPUTS(buf + LOG_TIME_LEN, log_fp[instance])
320 log_size[instance] += strlen(buf);
321 FFLUSH(log_fp[instance])
322 rotate_log(instance);
326 void log_status_ok(LogInstance instance, const gchar *format, ...)
329 gchar buf[BUFFSIZE + LOG_TIME_LEN];
331 LogText *logtext = g_new0(LogText, 1);
335 strftime(buf, LOG_TIME_LEN + 1, "[%H:%M:%S] ", localtime_r(&t, &buft));
337 va_start(args, format);
338 g_vsnprintf(buf + LOG_TIME_LEN, BUFFSIZE, format, args);
341 if (debug_get_mode()) g_message("%s", buf + LOG_TIME_LEN);
343 logtext->instance = instance;
344 logtext->text = g_strdup(buf + LOG_TIME_LEN);
345 logtext->type = LOG_STATUS_OK;
347 g_timeout_add(0, invoke_hook_cb, logtext);
349 if (log_fp[instance] && prefs_common_enable_log_status()) {
350 FWRITE(buf, 1, LOG_TIME_LEN, log_fp[instance])
351 FPUTS("* OK: ", log_fp[instance])
352 log_size[instance] += strlen("* OK: ");
353 FPUTS(buf + LOG_TIME_LEN, log_fp[instance])
354 log_size[instance] += strlen(buf);
355 FFLUSH(log_fp[instance])
356 rotate_log(instance);
360 void log_status_nok(LogInstance instance, const gchar *format, ...)
363 gchar buf[BUFFSIZE + LOG_TIME_LEN];
365 LogText *logtext = g_new0(LogText, 1);
369 strftime(buf, LOG_TIME_LEN + 1, "[%H:%M:%S] ", localtime_r(&t, &buft));
371 va_start(args, format);
372 g_vsnprintf(buf + LOG_TIME_LEN, BUFFSIZE, format, args);
375 if (debug_get_mode()) g_message("%s", buf + LOG_TIME_LEN);
377 logtext->instance = instance;
378 logtext->text = g_strdup(buf + LOG_TIME_LEN);
379 logtext->type = LOG_STATUS_NOK;
381 g_timeout_add(0, invoke_hook_cb, logtext);
383 if (log_fp[instance] && prefs_common_enable_log_status()) {
384 FWRITE(buf, 1, LOG_TIME_LEN, log_fp[instance])
385 FPUTS("* NOT OK: ", log_fp[instance])
386 log_size[instance] += strlen("* NOT OK: ");
387 FPUTS(buf + LOG_TIME_LEN, log_fp[instance])
388 log_size[instance] += strlen(buf);
389 FFLUSH(log_fp[instance])
390 rotate_log(instance);
394 void log_status_skip(LogInstance instance, const gchar *format, ...)
397 gchar buf[BUFFSIZE + LOG_TIME_LEN];
399 LogText *logtext = g_new0(LogText, 1);
403 strftime(buf, LOG_TIME_LEN + 1, "[%H:%M:%S] ", localtime_r(&t, &buft));
405 va_start(args, format);
406 g_vsnprintf(buf + LOG_TIME_LEN, BUFFSIZE, format, args);
409 if (debug_get_mode()) g_message("%s", buf + LOG_TIME_LEN);
411 logtext->instance = instance;
412 logtext->text = g_strdup(buf + LOG_TIME_LEN);
413 logtext->type = LOG_STATUS_SKIP;
415 g_timeout_add(0, invoke_hook_cb, logtext);
417 if (log_fp[instance] && prefs_common_enable_log_status()) {
418 FWRITE(buf, 1, LOG_TIME_LEN, log_fp[instance])
419 FPUTS("* SKIPPED: ", log_fp[instance])
420 log_size[instance] += strlen("* SKIPPED: ");
421 FPUTS(buf + LOG_TIME_LEN, log_fp[instance])
422 log_size[instance] += strlen(buf);
423 FFLUSH(log_fp[instance])
424 rotate_log(instance);