Blame src/ibusobservedpath.c

Packit Service 1d8f1c
/* -*- mode: C; c-basic-offset: 4; indent-tabs-mode: nil; -*- */
Packit Service 1d8f1c
/* vim:set et sts=4: */
Packit Service 1d8f1c
/* ibus - The Input IBus
Packit Service 1d8f1c
 * Copyright (C) 2008-2015 Peng Huang <shawn.p.huang@gmail.com>
Packit Service 1d8f1c
 * Copyright (C) 2008-2015 Red Hat, Inc.
Packit Service 1d8f1c
 *
Packit Service 1d8f1c
 * This library is free software; you can redistribute it and/or
Packit Service 1d8f1c
 * modify it under the terms of the GNU Lesser General Public
Packit Service 1d8f1c
 * License as published by the Free Software Foundation; either
Packit Service 1d8f1c
 * version 2.1 of the License, or (at your option) any later version.
Packit Service 1d8f1c
 *
Packit Service 1d8f1c
 * This library is distributed in the hope that it will be useful,
Packit Service 1d8f1c
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
Packit Service 1d8f1c
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
Packit Service 1d8f1c
 * Lesser General Public License for more details.
Packit Service 1d8f1c
 *
Packit Service 1d8f1c
 * You should have received a copy of the GNU Lesser General Public
Packit Service 1d8f1c
 * License along with this library; if not, write to the Free Software
Packit Service 1d8f1c
 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301
Packit Service 1d8f1c
 * USA
Packit Service 1d8f1c
 */
Packit Service 1d8f1c
#include <glib/gstdio.h>
Packit Service 1d8f1c
#include <stdlib.h>
Packit Service 1d8f1c
#include "ibusinternal.h"
Packit Service 1d8f1c
#include "ibusobservedpath.h"
Packit Service 1d8f1c
Packit Service 1d8f1c
Packit Service 1d8f1c
enum {
Packit Service 1d8f1c
    LAST_SIGNAL,
Packit Service 1d8f1c
};
Packit Service 1d8f1c
Packit Service 1d8f1c
Packit Service 1d8f1c
/* IBusObservedPathPriv */
Packit Service 1d8f1c
struct _IBusObservedPathPrivate {
Packit Service 1d8f1c
    gpointer pad;
Packit Service 1d8f1c
};
Packit Service 1d8f1c
typedef struct _IBusObservedPathPrivate IBusObservedPathPrivate;
Packit Service 1d8f1c
Packit Service 1d8f1c
#define IBUS_OBSERVED_PATH_GET_PRIVATE(o)  \
Packit Service 1d8f1c
   (G_TYPE_INSTANCE_GET_PRIVATE ((o), IBUS_TYPE_OBSERVED_PATH, IBusObservedPathPrivate))
Packit Service 1d8f1c
Packit Service 1d8f1c
// static guint            _signals[LAST_SIGNAL] = { 0 };
Packit Service 1d8f1c
Packit Service 1d8f1c
/* functions prototype */
Packit Service 1d8f1c
static void      ibus_observed_path_destroy         (IBusObservedPath       *path);
Packit Service 1d8f1c
static gboolean  ibus_observed_path_serialize       (IBusObservedPath       *path,
Packit Service 1d8f1c
                                                     GVariantBuilder        *builder);
Packit Service 1d8f1c
static gint      ibus_observed_path_deserialize     (IBusObservedPath       *path,
Packit Service 1d8f1c
                                                     GVariant               *variant);
Packit Service 1d8f1c
static gboolean  ibus_observed_path_copy            (IBusObservedPath       *dest,
Packit Service 1d8f1c
                                                     const IBusObservedPath *src);
Packit Service 1d8f1c
static gboolean  ibus_observed_path_parse_xml_node  (IBusObservedPath       *path,
Packit Service 1d8f1c
                                                     XMLNode                *node);
Packit Service 1d8f1c
Packit Service 1d8f1c
G_DEFINE_TYPE (IBusObservedPath, ibus_observed_path, IBUS_TYPE_SERIALIZABLE)
Packit Service 1d8f1c
Packit Service 1d8f1c
static void
Packit Service 1d8f1c
ibus_observed_path_class_init (IBusObservedPathClass *class)
Packit Service 1d8f1c
{
Packit Service 1d8f1c
    IBusObjectClass *object_class = IBUS_OBJECT_CLASS (class);
Packit Service 1d8f1c
    IBusSerializableClass *serializable_class = IBUS_SERIALIZABLE_CLASS (class);
Packit Service 1d8f1c
Packit Service 1d8f1c
    // g_type_class_add_private (class, sizeof (IBusObservedPathPrivate));
Packit Service 1d8f1c
Packit Service 1d8f1c
    object_class->destroy = (IBusObjectDestroyFunc) ibus_observed_path_destroy;
Packit Service 1d8f1c
Packit Service 1d8f1c
    serializable_class->serialize   = (IBusSerializableSerializeFunc) ibus_observed_path_serialize;
Packit Service 1d8f1c
    serializable_class->deserialize = (IBusSerializableDeserializeFunc) ibus_observed_path_deserialize;
Packit Service 1d8f1c
    serializable_class->copy        = (IBusSerializableCopyFunc) ibus_observed_path_copy;
Packit Service 1d8f1c
}
Packit Service 1d8f1c
Packit Service 1d8f1c
Packit Service 1d8f1c
static void
Packit Service 1d8f1c
ibus_observed_path_init (IBusObservedPath *path)
Packit Service 1d8f1c
{
Packit Service 1d8f1c
}
Packit Service 1d8f1c
Packit Service 1d8f1c
static void
Packit Service 1d8f1c
ibus_observed_path_destroy (IBusObservedPath *path)
Packit Service 1d8f1c
{
Packit Service 1d8f1c
    g_free (path->path);
Packit Service 1d8f1c
    IBUS_OBJECT_CLASS (ibus_observed_path_parent_class)->destroy (IBUS_OBJECT (path));
Packit Service 1d8f1c
}
Packit Service 1d8f1c
Packit Service 1d8f1c
static gboolean
Packit Service 1d8f1c
ibus_observed_path_serialize (IBusObservedPath *path,
Packit Service 1d8f1c
                              GVariantBuilder  *builder)
Packit Service 1d8f1c
{
Packit Service 1d8f1c
    gboolean retval;
Packit Service 1d8f1c
Packit Service 1d8f1c
    retval = IBUS_SERIALIZABLE_CLASS (ibus_observed_path_parent_class)->
Packit Service 1d8f1c
            serialize ((IBusSerializable *)path, builder);
Packit Service 1d8f1c
    g_return_val_if_fail (retval, FALSE);
Packit Service 1d8f1c
Packit Service 1d8f1c
    g_variant_builder_add (builder, "s", path->path);
Packit Service 1d8f1c
    g_variant_builder_add (builder, "x", path->mtime);
Packit Service 1d8f1c
Packit Service 1d8f1c
    return TRUE;
Packit Service 1d8f1c
}
Packit Service 1d8f1c
Packit Service 1d8f1c
static gint
Packit Service 1d8f1c
ibus_observed_path_deserialize (IBusObservedPath *path,
Packit Service 1d8f1c
                                GVariant         *variant)
Packit Service 1d8f1c
{
Packit Service 1d8f1c
    gint retval;
Packit Service 1d8f1c
Packit Service 1d8f1c
    retval = IBUS_SERIALIZABLE_CLASS (ibus_observed_path_parent_class)->
Packit Service 1d8f1c
            deserialize ((IBusSerializable *)path, variant);
Packit Service 1d8f1c
    g_return_val_if_fail (retval, 0);
Packit Service 1d8f1c
Packit Service 1d8f1c
    ibus_g_variant_get_child_string (variant, retval++, &path->path);
Packit Service 1d8f1c
    g_variant_get_child (variant, retval++, "x", &path->mtime);
Packit Service 1d8f1c
Packit Service 1d8f1c
    return retval;
Packit Service 1d8f1c
}
Packit Service 1d8f1c
Packit Service 1d8f1c
static gboolean
Packit Service 1d8f1c
ibus_observed_path_copy (IBusObservedPath       *dest,
Packit Service 1d8f1c
                         const IBusObservedPath *src)
Packit Service 1d8f1c
{
Packit Service 1d8f1c
    gboolean retval;
Packit Service 1d8f1c
Packit Service 1d8f1c
    retval = IBUS_SERIALIZABLE_CLASS (ibus_observed_path_parent_class)->copy ((IBusSerializable *)dest, (IBusSerializable *)src);
Packit Service 1d8f1c
    g_return_val_if_fail (retval, FALSE);
Packit Service 1d8f1c
Packit Service 1d8f1c
    dest->path = g_strdup (src->path);
Packit Service 1d8f1c
    dest->mtime = src->mtime;
Packit Service 1d8f1c
Packit Service 1d8f1c
    return TRUE;
Packit Service 1d8f1c
}
Packit Service 1d8f1c
Packit Service 1d8f1c
#define g_string_append_indent(string, indent)  \
Packit Service 1d8f1c
    {                                           \
Packit Service 1d8f1c
        gint i;                                 \
Packit Service 1d8f1c
        for (i = 0; i < (indent); i++) {        \
Packit Service 1d8f1c
            g_string_append (string, "    ");   \
Packit Service 1d8f1c
        }                                       \
Packit Service 1d8f1c
    }
Packit Service 1d8f1c
Packit Service 1d8f1c
void
Packit Service 1d8f1c
ibus_observed_path_output (IBusObservedPath *path,
Packit Service 1d8f1c
                          GString         *output,
Packit Service 1d8f1c
                          gint             indent)
Packit Service 1d8f1c
{
Packit Service 1d8f1c
    g_assert (IBUS_IS_OBSERVED_PATH (path));
Packit Service 1d8f1c
    g_assert (output);
Packit Service 1d8f1c
Packit Service 1d8f1c
    g_string_append_indent (output, indent);
Packit Service 1d8f1c
    g_string_append_printf (output, "<path mtime=\"%ld\" >%s</path>\n",
Packit Service 1d8f1c
                                    path->mtime,
Packit Service 1d8f1c
                                    path->path);
Packit Service 1d8f1c
}
Packit Service 1d8f1c
Packit Service 1d8f1c
gboolean
Packit Service 1d8f1c
ibus_observed_path_check_modification (IBusObservedPath *path)
Packit Service 1d8f1c
{
Packit Service 1d8f1c
    gchar *real_path = NULL;
Packit Service 1d8f1c
    struct stat buf;
Packit Service 1d8f1c
Packit Service 1d8f1c
    g_assert (IBUS_IS_OBSERVED_PATH (path));
Packit Service 1d8f1c
Packit Service 1d8f1c
    if (path->path[0] == '~') {
Packit Service 1d8f1c
        const gchar *homedir = g_get_home_dir ();
Packit Service 1d8f1c
        real_path = g_build_filename (homedir, path->path + 2, NULL);
Packit Service 1d8f1c
    }
Packit Service 1d8f1c
    else {
Packit Service 1d8f1c
        real_path = g_strdup (path->path);
Packit Service 1d8f1c
    }
Packit Service 1d8f1c
Packit Service 1d8f1c
    if (g_stat (real_path, &buf) != 0) {
Packit Service 1d8f1c
        buf.st_mtime = 0;
Packit Service 1d8f1c
    }
Packit Service 1d8f1c
Packit Service 1d8f1c
    g_free (real_path);
Packit Service 1d8f1c
Packit Service 1d8f1c
    if (path->mtime == buf.st_mtime)
Packit Service 1d8f1c
        return FALSE;
Packit Service 1d8f1c
    return TRUE;
Packit Service 1d8f1c
}
Packit Service 1d8f1c
Packit Service 1d8f1c
static void
Packit Service 1d8f1c
ibus_observed_path_fill_stat (IBusObservedPath *path)
Packit Service 1d8f1c
{
Packit Service 1d8f1c
    g_assert (IBUS_IS_OBSERVED_PATH (path));
Packit Service 1d8f1c
Packit Service 1d8f1c
    struct stat buf;
Packit Service 1d8f1c
Packit Service 1d8f1c
    if (g_stat (path->path, &buf) == 0) {
Packit Service 1d8f1c
        path->is_exist = 1;
Packit Service 1d8f1c
        if (S_ISDIR (buf.st_mode)) {
Packit Service 1d8f1c
            path->is_dir = 1;
Packit Service 1d8f1c
        }
Packit Service 1d8f1c
        path->mtime = buf.st_mtime;
Packit Service 1d8f1c
    }
Packit Service 1d8f1c
    else {
Packit Service 1d8f1c
        path->is_dir = 0;
Packit Service 1d8f1c
        path->is_exist = 0;
Packit Service 1d8f1c
        path->mtime = 0;
Packit Service 1d8f1c
    }
Packit Service 1d8f1c
}
Packit Service 1d8f1c
Packit Service 1d8f1c
GList *
Packit Service 1d8f1c
ibus_observed_path_traverse (IBusObservedPath *path,
Packit Service 1d8f1c
                             gboolean          dir_only)
Packit Service 1d8f1c
{
Packit Service 1d8f1c
    g_assert (IBUS_IS_OBSERVED_PATH (path));
Packit Service 1d8f1c
Packit Service 1d8f1c
    GDir *dir;
Packit Service 1d8f1c
    const gchar *name;
Packit Service 1d8f1c
    GList *paths = NULL;
Packit Service 1d8f1c
Packit Service 1d8f1c
    dir = g_dir_open (path->path, 0, NULL);
Packit Service 1d8f1c
Packit Service 1d8f1c
    if (dir == NULL)
Packit Service 1d8f1c
        return NULL;
Packit Service 1d8f1c
Packit Service 1d8f1c
    while ((name = g_dir_read_name (dir)) != NULL) {
Packit Service 1d8f1c
        IBusObservedPath *sub;
Packit Service 1d8f1c
Packit Service 1d8f1c
        sub = g_object_new (IBUS_TYPE_OBSERVED_PATH, NULL);
Packit Service 1d8f1c
        g_object_ref_sink (sub);
Packit Service 1d8f1c
        sub->path = g_build_filename (path->path, name, NULL);
Packit Service 1d8f1c
Packit Service 1d8f1c
        ibus_observed_path_fill_stat (sub);
Packit Service 1d8f1c
        if (sub->is_exist && sub->is_dir) {
Packit Service 1d8f1c
            paths = g_list_append (paths, sub);
Packit Service 1d8f1c
            paths = g_list_concat (paths,
Packit Service 1d8f1c
                                   ibus_observed_path_traverse (sub, dir_only));
Packit Service 1d8f1c
        } else if (!dir_only) {
Packit Service 1d8f1c
            paths = g_list_append (paths, sub);
Packit Service 1d8f1c
        }
Packit Service 1d8f1c
    }
Packit Service 1d8f1c
    g_dir_close (dir);
Packit Service 1d8f1c
Packit Service 1d8f1c
    return paths;
Packit Service 1d8f1c
}
Packit Service 1d8f1c
Packit Service 1d8f1c
static gboolean
Packit Service 1d8f1c
ibus_observed_path_parse_xml_node (IBusObservedPath *path,
Packit Service 1d8f1c
                                   XMLNode          *node)
Packit Service 1d8f1c
{
Packit Service 1d8f1c
    g_assert (IBUS_IS_OBSERVED_PATH (path));
Packit Service 1d8f1c
    g_assert (node);
Packit Service 1d8f1c
Packit Service 1d8f1c
    if (G_UNLIKELY (g_strcmp0 (node->name, "path") != 0)) {
Packit Service 1d8f1c
        return FALSE;
Packit Service 1d8f1c
    }
Packit Service 1d8f1c
Packit Service 1d8f1c
    if (node->text[0] == '~' && node->text[1] != G_DIR_SEPARATOR) {
Packit Service 1d8f1c
        g_warning ("invalide path \"%s\"", node->text);
Packit Service 1d8f1c
        return FALSE;
Packit Service 1d8f1c
    }
Packit Service 1d8f1c
Packit Service 1d8f1c
    path->path = g_strdup (node->text);
Packit Service 1d8f1c
Packit Service 1d8f1c
    gchar **attr;
Packit Service 1d8f1c
    for (attr = node->attributes; attr[0]; attr += 2) {
Packit Service 1d8f1c
        if (g_strcmp0 (*attr, "mtime") == 0) {
Packit Service 1d8f1c
            path->mtime = atol (attr[1]);
Packit Service 1d8f1c
            continue;
Packit Service 1d8f1c
        }
Packit Service 1d8f1c
        g_warning ("Unkonwn attribute %s", attr[0]);
Packit Service 1d8f1c
    }
Packit Service 1d8f1c
Packit Service 1d8f1c
    return TRUE;
Packit Service 1d8f1c
}
Packit Service 1d8f1c
Packit Service 1d8f1c
IBusObservedPath *
Packit Service 1d8f1c
ibus_observed_path_new_from_xml_node (XMLNode *node,
Packit Service 1d8f1c
                                     gboolean fill_stat)
Packit Service 1d8f1c
{
Packit Service 1d8f1c
    g_assert (node);
Packit Service 1d8f1c
Packit Service 1d8f1c
    IBusObservedPath *path;
Packit Service 1d8f1c
Packit Service 1d8f1c
    path = (IBusObservedPath *) g_object_new (IBUS_TYPE_OBSERVED_PATH, NULL);
Packit Service 1d8f1c
Packit Service 1d8f1c
    if (!ibus_observed_path_parse_xml_node (path, node)) {
Packit Service 1d8f1c
        g_object_unref (path);
Packit Service 1d8f1c
        path = NULL;
Packit Service 1d8f1c
    }
Packit Service 1d8f1c
    else if (fill_stat) {
Packit Service 1d8f1c
        ibus_observed_path_fill_stat (path);
Packit Service 1d8f1c
    }
Packit Service 1d8f1c
Packit Service 1d8f1c
    return path;
Packit Service 1d8f1c
}
Packit Service 1d8f1c
Packit Service 1d8f1c
IBusObservedPath *
Packit Service 1d8f1c
ibus_observed_path_new (const gchar *path,
Packit Service 1d8f1c
                        gboolean     fill_stat)
Packit Service 1d8f1c
{
Packit Service 1d8f1c
    g_assert (path);
Packit Service 1d8f1c
Packit Service 1d8f1c
    IBusObservedPath *op;
Packit Service 1d8f1c
Packit Service 1d8f1c
    op = (IBusObservedPath *) g_object_new (IBUS_TYPE_OBSERVED_PATH, NULL);
Packit Service 1d8f1c
    op->path = g_strdup (path);
Packit Service 1d8f1c
Packit Service 1d8f1c
    if (fill_stat) {
Packit Service 1d8f1c
        ibus_observed_path_fill_stat (op);
Packit Service 1d8f1c
    }
Packit Service 1d8f1c
Packit Service 1d8f1c
    return op;
Packit Service 1d8f1c
}
Packit Service 1d8f1c