300 lines
6.3 KiB
C
300 lines
6.3 KiB
C
/*
|
|
*
|
|
* This program is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU Lesser General Public
|
|
* License as published by the Free Software Foundation; either
|
|
* version 2 of the License, or (at your option) version 3.
|
|
*
|
|
* 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
|
|
* Lesser General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU Lesser General Public
|
|
* License along with the program; if not, see <http://www.gnu.org/licenses/>
|
|
*
|
|
*
|
|
* Authors:
|
|
* Srinivasa Ragavan <sragavan@gnome.org>
|
|
*
|
|
* Copyright (C) 1999-2008 Novell, Inc. (www.novell.com)
|
|
*
|
|
*/
|
|
|
|
#ifdef HAVE_CONFIG_H
|
|
#include <config.h>
|
|
#endif
|
|
|
|
#include <stdio.h>
|
|
#include <string.h>
|
|
#include <stdlib.h>
|
|
|
|
#include <glib.h>
|
|
#include <glib/gi18n.h>
|
|
#include <glib/gstdio.h>
|
|
#include "e-logger.h"
|
|
#include "e-mktemp.h"
|
|
|
|
/* 5 Minutes */
|
|
#define TIMEOUT_INTERVAL 300
|
|
|
|
#define E_LOGGER_GET_PRIVATE(obj) \
|
|
(G_TYPE_INSTANCE_GET_PRIVATE \
|
|
((obj), E_TYPE_LOGGER, ELoggerPrivate))
|
|
|
|
struct _ELoggerPrivate {
|
|
gchar *name;
|
|
gchar *logfile;
|
|
FILE *fp;
|
|
|
|
guint timer;
|
|
};
|
|
|
|
enum {
|
|
PROP_0,
|
|
PROP_NAME
|
|
};
|
|
|
|
static gpointer parent_class;
|
|
|
|
static gboolean
|
|
logger_flush (ELogger *logger)
|
|
{
|
|
if (logger->priv->fp)
|
|
fflush (logger->priv->fp);
|
|
logger->priv->timer = 0;
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
static void
|
|
logger_set_dirty (ELogger *logger)
|
|
{
|
|
if (logger->priv->timer)
|
|
return;
|
|
|
|
logger->priv->timer = g_timeout_add (
|
|
TIMEOUT_INTERVAL, (GSourceFunc) logger_flush, logger);
|
|
}
|
|
|
|
static void
|
|
logger_set_name (ELogger *logger,
|
|
const gchar *name)
|
|
{
|
|
gchar *temp;
|
|
|
|
g_return_if_fail (logger->priv->name == NULL);
|
|
|
|
temp = g_strdup_printf ("%s.log.XXXXXX", name);
|
|
|
|
logger->priv->name = g_strdup (name);
|
|
logger->priv->logfile = e_mktemp (temp);
|
|
logger->priv->fp = g_fopen (logger->priv->logfile, "w");
|
|
logger->priv->timer = 0;
|
|
|
|
if (!logger->priv->fp)
|
|
g_warning ("%s: Failed to open log file '%s' for writing.", G_STRFUNC, logger->priv->logfile ? logger->priv->logfile : "[null]");
|
|
|
|
g_free (temp);
|
|
}
|
|
|
|
static void
|
|
logger_set_property (GObject *object,
|
|
guint property_id,
|
|
const GValue *value,
|
|
GParamSpec *pspec)
|
|
{
|
|
switch (property_id) {
|
|
case PROP_NAME:
|
|
logger_set_name (
|
|
E_LOGGER (object),
|
|
g_value_get_string (value));
|
|
return;
|
|
}
|
|
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, property_id, pspec);
|
|
}
|
|
|
|
static void
|
|
logger_get_property (GObject *object,
|
|
guint property_id,
|
|
GValue *value,
|
|
GParamSpec *pspec)
|
|
{
|
|
switch (property_id) {
|
|
case PROP_NAME:
|
|
g_value_set_string (
|
|
value, e_logger_get_name (
|
|
E_LOGGER (object)));
|
|
return;
|
|
}
|
|
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, property_id, pspec);
|
|
}
|
|
|
|
static void
|
|
logger_finalize (GObject *object)
|
|
{
|
|
ELogger *logger = E_LOGGER (object);
|
|
|
|
if (logger->priv->timer)
|
|
g_source_remove (logger->priv->timer);
|
|
logger_flush (logger);
|
|
if (logger->priv->fp)
|
|
fclose (logger->priv->fp);
|
|
|
|
g_free (logger->priv->name);
|
|
g_free (logger->priv->logfile);
|
|
|
|
/* Chain up to parent's finalize() method. */
|
|
G_OBJECT_CLASS (parent_class)->finalize (object);
|
|
}
|
|
|
|
static void
|
|
logger_class_init (ELoggerClass *class)
|
|
{
|
|
GObjectClass *object_class;
|
|
|
|
parent_class = g_type_class_peek_parent (class);
|
|
g_type_class_add_private (class, sizeof (ELoggerPrivate));
|
|
|
|
object_class = G_OBJECT_CLASS (class);
|
|
object_class->set_property = logger_set_property;
|
|
object_class->get_property = logger_get_property;
|
|
object_class->finalize = logger_finalize;
|
|
|
|
g_object_class_install_property (
|
|
object_class,
|
|
PROP_NAME,
|
|
g_param_spec_string (
|
|
"name",
|
|
_("Name"),
|
|
_("Name of the logger"),
|
|
"anonymous",
|
|
G_PARAM_READWRITE |
|
|
G_PARAM_CONSTRUCT_ONLY));
|
|
}
|
|
|
|
static void
|
|
logger_init (ELogger *logger)
|
|
{
|
|
logger->priv = E_LOGGER_GET_PRIVATE (logger);
|
|
}
|
|
|
|
GType
|
|
e_logger_get_type (void)
|
|
{
|
|
static GType type = 0;
|
|
|
|
if (G_UNLIKELY (type == 0)) {
|
|
static const GTypeInfo type_info = {
|
|
sizeof (ELoggerClass),
|
|
(GBaseInitFunc) NULL,
|
|
(GBaseFinalizeFunc) NULL,
|
|
(GClassInitFunc) logger_class_init,
|
|
(GClassFinalizeFunc) NULL,
|
|
NULL, /* class_data */
|
|
sizeof (ELogger),
|
|
0, /* n_preallocs */
|
|
(GInstanceInitFunc) logger_init,
|
|
NULL /* value_table */
|
|
};
|
|
|
|
type = g_type_register_static (
|
|
G_TYPE_OBJECT, "ELogger", &type_info, 0);
|
|
}
|
|
|
|
return type;
|
|
}
|
|
|
|
ELogger *
|
|
e_logger_new (const gchar *name)
|
|
{
|
|
g_return_val_if_fail (name != NULL, NULL);
|
|
|
|
return g_object_new (E_TYPE_LOGGER, "name", name, NULL);
|
|
}
|
|
|
|
const gchar *
|
|
e_logger_get_name (ELogger *logger)
|
|
{
|
|
g_return_val_if_fail (E_IS_LOGGER (logger), NULL);
|
|
|
|
return logger->priv->name;
|
|
}
|
|
|
|
void
|
|
e_logger_log (ELogger *logger,
|
|
ELogLevel level,
|
|
gchar *primary,
|
|
gchar *secondary)
|
|
{
|
|
time_t t = time (NULL);
|
|
|
|
g_return_if_fail (E_LOGGER (logger));
|
|
g_return_if_fail (primary != NULL);
|
|
g_return_if_fail (secondary != NULL);
|
|
|
|
if (!logger->priv->fp)
|
|
return;
|
|
|
|
fprintf (logger->priv->fp, "%d:%ld:%s\n", level, t, primary);
|
|
fprintf (logger->priv->fp, "%d:%ld:%s\n", level, t, secondary);
|
|
logger_set_dirty (logger);
|
|
}
|
|
|
|
void
|
|
e_logger_get_logs (ELogger *logger,
|
|
ELogFunction func,
|
|
gpointer user_data)
|
|
{
|
|
FILE *fp;
|
|
gchar buf[250];
|
|
|
|
g_return_if_fail (E_LOGGER (logger));
|
|
g_return_if_fail (func != NULL);
|
|
|
|
/* Flush everything before we get the logs */
|
|
if (logger->priv->fp)
|
|
fflush (logger->priv->fp);
|
|
fp = g_fopen (logger->priv->logfile, "r");
|
|
|
|
if (!fp) {
|
|
g_warning ("%s: Cannot open log file '%s' for reading! No flush yet?\n", G_STRFUNC, logger->priv->logfile ? logger->priv->logfile : "[null]");
|
|
return;
|
|
}
|
|
|
|
while (!feof (fp)) {
|
|
gchar *tmp;
|
|
gsize len;
|
|
|
|
tmp = fgets (buf, sizeof (buf), fp);
|
|
if (!tmp)
|
|
break;
|
|
|
|
len = strlen (tmp);
|
|
if (len > 0 && tmp [len - 1] != '\n' && !feof (fp)) {
|
|
/* there are more characters on a row than 249, so read them all */
|
|
GString *str = g_string_sized_new (1024);
|
|
|
|
g_string_append (str, tmp);
|
|
|
|
while (!feof (fp) && len > 0 && tmp [len - 1] != '\n') {
|
|
tmp = fgets (buf, sizeof (buf), fp);
|
|
if (!tmp)
|
|
break;
|
|
|
|
len = strlen (tmp);
|
|
g_string_append (str, tmp);
|
|
}
|
|
|
|
func (str->str, user_data);
|
|
|
|
g_string_free (str, TRUE);
|
|
} else
|
|
func (tmp, user_data);
|
|
}
|
|
|
|
fclose (fp);
|
|
}
|