diff options
Diffstat (limited to 'e-util/e-plugin.c')
-rw-r--r-- | e-util/e-plugin.c | 1283 |
1 files changed, 0 insertions, 1283 deletions
diff --git a/e-util/e-plugin.c b/e-util/e-plugin.c deleted file mode 100644 index a60aa7bafd..0000000000 --- a/e-util/e-plugin.c +++ /dev/null @@ -1,1283 +0,0 @@ - -#ifdef HAVE_CONFIG_H -#include <config.h> -#endif - -#include <sys/types.h> -#include <dirent.h> -#include <string.h> - -#include <glib/gi18n.h> -#include <gconf/gconf-client.h> - -#include "e-plugin.h" -#include "libedataserver/e-msgport.h" - -/* plugin debug */ -#define pd(x) -/* plugin hook debug */ -#define phd(x) - -/* -<camel-plugin - class="org.gnome.camel.plugin.provider:1.0" - id="org.gnome.camel.provider.imap:1.0" - type="shlib" - location="/opt/gnome2/lib/camel/1.0/libcamelimap.so" - factory="camel_imap_provider_new"> - <name>imap</name> - <description>IMAP4 and IMAP4v1 mail store</description> - <class-data class="org.gnome.camel.plugin.provider:1.0" - protocol="imap" - domain="mail" - flags="remote,source,storage,ssl"/> -</camel-plugin> - -<camel-plugin - class="org.gnome.camel.plugin.sasl:1.0" - id="org.gnome.camel.sasl.plain:1.0" - type="shlib" - location="/opt/gnome2/lib/camel/1.0/libcamelsasl.so" - factory="camel_sasl_plain_new"> - <name>PLAIN</name> - <description>SASL PLAIN authentication mechanism</description> -</camel-plugin> -*/ - -/* EPlugin stuff */ -static GObjectClass *ep_parent_class; - -/* global table of plugin types by pluginclass.type */ -static GHashTable *ep_types; -/* plugin load path */ -static GSList *ep_path; -/* global table of plugins by plugin.id */ -static GHashTable *ep_plugins; -/* a table of GSLists of plugins by hook class for hooks not loadable yet */ -static GHashTable *ep_plugins_pending_hooks; -/* list of all cached xml docs:struct _plugin_doc's */ -static EDList ep_plugin_docs = E_DLIST_INITIALISER(ep_plugin_docs); -/* gconf client */ -static GConfClient *ep_gconf; -/* the list of disabled plugins from gconf */ -static GSList *ep_disabled; - -/* EPluginHook stuff */ -static void *eph_parent_class; -/* All classes which implement EPluginHooks, by class.id */ -static GHashTable *eph_types; - -struct _plugin_doc { - struct _plugin_doc *next; - struct _plugin_doc *prev; - - char *filename; - xmlDocPtr doc; - - GSList *plugin_hooks; /* EPlugin objects with pending hooks */ - GSList *plugins; /* xmlNodePtr's of plugins with unknown type (mono,etc) */ -}; - -static gboolean -ep_check_enabled(const char *id) -{ - GSList *l = ep_disabled; - - for (;l;l = g_slist_next(l)) - if (!strcmp((char *)l->data, id)) - return FALSE; - - return TRUE; -} - -static void -ep_set_enabled(const char *id, int state) -{ - if ((state == 0) == ep_check_enabled(id) == 0) - return; - - if (state) { - GSList *l = ep_disabled; - - while (l) { - GSList *n = l->next; - - if (!strcmp((char *)l->data, id)) { - g_free(l->data); - ep_disabled = g_slist_remove_link(ep_disabled, l); - } - l = n; - } - } else { - ep_disabled = g_slist_prepend(ep_disabled, g_strdup(id)); - } - - gconf_client_set_list(ep_gconf, "/apps/evolution/eplugin/disabled", GCONF_VALUE_STRING, ep_disabled, NULL); -} - -static int -ep_construct(EPlugin *ep, xmlNodePtr root) -{ - xmlNodePtr node; - int res = -1; - char *localedir; - - ep->domain = e_plugin_xml_prop(root, "domain"); - if (ep->domain - && (localedir = e_plugin_xml_prop(root, "localedir"))) { - bindtextdomain(ep->domain, localedir); - g_free(localedir); - } - - ep->name = e_plugin_xml_prop_domain(root, "name", ep->domain); - - pd(printf("creating plugin '%s' '%s'\n", ep->name?ep->name:"un-named", ep->id)); - - node = root->children; - while (node) { - if (strcmp(node->name, "hook") == 0) { - struct _EPluginHook *hook; - EPluginHookClass *type; - char *class = e_plugin_xml_prop(node, "class"); - - if (class == NULL) { - g_warning("Plugin '%s' load failed in '%s', missing class property for hook", ep->id, ep->path); - goto fail; - } - - if (ep->enabled - && eph_types != NULL - && (type = g_hash_table_lookup(eph_types, class)) != NULL) { - g_free(class); - hook = g_object_new(G_OBJECT_CLASS_TYPE(type), NULL); - res = type->construct(hook, ep, node); - if (res == -1) { - g_warning("Plugin '%s' failed to load hook", ep->name); - g_object_unref(hook); - goto fail; - } else { - ep->hooks = g_slist_append(ep->hooks, hook); - } - } else { - GSList *l; - char *oldclass; - - if (ep_plugins_pending_hooks == NULL) - ep_plugins_pending_hooks = g_hash_table_new(g_str_hash, g_str_equal); - if (!g_hash_table_lookup_extended(ep_plugins_pending_hooks, class, (void **)&oldclass, (void **)&l)) { - oldclass = class; - l = NULL; - } - l = g_slist_prepend(l, ep); - g_hash_table_insert(ep_plugins_pending_hooks, oldclass, l); - ep->hooks_pending = g_slist_prepend(ep->hooks_pending, node); - } - } else if (strcmp(node->name, "description") == 0) { - ep->description = e_plugin_xml_content_domain(node, ep->domain); - } else if (strcmp(node->name, "author") == 0) { - char *name = e_plugin_xml_prop(node, "name"); - char *email = e_plugin_xml_prop(node, "email"); - - if (name || email) { - EPluginAuthor *epa = g_malloc0(sizeof(*epa)); - - epa->name = name; - epa->email = email; - ep->authors = g_slist_append(ep->authors, epa); - } - } - node = node->next; - } - res = 0; -fail: - return res; -} - -static void -ep_enable(EPlugin *ep, int state) -{ - GSList *l; - - ep->enabled = state; - for (l=ep->hooks;l;l = g_slist_next(l)) - e_plugin_hook_enable((EPluginHook *)l->data, state); - - ep_set_enabled(ep->id, state); -} - -static void -ep_finalise(GObject *o) -{ - EPlugin *ep = (EPlugin *)o; - - g_free(ep->id); - g_free(ep->description); - g_free(ep->name); - g_free(ep->domain); - g_slist_free(ep->hooks_pending); - - g_slist_foreach(ep->hooks, (GFunc)g_object_unref, NULL); - g_slist_free(ep->hooks); - - ((GObjectClass *)ep_parent_class)->finalize(o); -} - -static void -ep_init(GObject *o) -{ - EPlugin *ep = (EPlugin *)o; - - ep->enabled = TRUE; -} - -static void -ep_class_init(EPluginClass *klass) -{ - ((GObjectClass *)klass)->finalize = ep_finalise; - klass->construct = ep_construct; - klass->enable = ep_enable; -} - -/** - * e_plugin_get_type: - * - * Standard GObject type function. This is only an abstract class, so - * you can only use this to subclass EPlugin. - * - * Return value: The type. - **/ -GType -e_plugin_get_type(void) -{ - static GType type = 0; - - if (!type) { - char *path, *col, *p; - - static const GTypeInfo info = { - sizeof(EPluginClass), NULL, NULL, (GClassInitFunc)ep_class_init, NULL, NULL, - sizeof(EPlugin), 0, (GInstanceInitFunc)ep_init, - }; - - ep_parent_class = g_type_class_ref(G_TYPE_OBJECT); - type = g_type_register_static(G_TYPE_OBJECT, "EPlugin", &info, 0); - - /* Add paths in the environment variable or default global and user specific paths */ - path = g_strdup(getenv("EVOLUTION_PLUGIN_PATH")); - if (path == NULL) { - /* Add the global path */ - e_plugin_add_load_path(EVOLUTION_PLUGINDIR); - - path = g_build_filename(g_get_home_dir(), ".eplugins", NULL); - } - - p = path; - while ((col = strchr(p, ':'))) { - *col++ = 0; - e_plugin_add_load_path(p); - p = col; - } - e_plugin_add_load_path(p); - g_free(path); - } - - return type; -} - -static EPlugin * -ep_load_plugin(xmlNodePtr root, struct _plugin_doc *pdoc) -{ - char *prop, *id; - EPluginClass *klass; - EPlugin *ep; - - id = e_plugin_xml_prop(root, "id"); - if (id == NULL) { - g_warning("Invalid e-plugin entry in '%s': no id", pdoc->filename); - return NULL; - } - - if (g_hash_table_lookup(ep_plugins, id)) { - g_warning("Plugin '%s' already defined", id); - g_free(id); - return NULL; - } - - prop = xmlGetProp(root, "type"); - if (prop == NULL) { - g_free(id); - g_warning("Invalid e-plugin entry in '%s': no type", pdoc->filename); - return NULL; - } - - /* If we can't find a plugin, add it to a pending list which is checked when a new type is registered */ - klass = g_hash_table_lookup(ep_types, prop); - if (klass == NULL) { - pd(printf("Delaying loading of plugin '%s' unknown type '%s'\n", id, prop)); - g_free(id); - xmlFree(prop); - pdoc->plugins = g_slist_prepend(pdoc->plugins, root); - return NULL; - } - xmlFree(prop); - - ep = g_object_new(G_TYPE_FROM_CLASS(klass), NULL); - ep->id = id; - ep->path = g_strdup(pdoc->filename); - ep->enabled = ep_check_enabled(id); - if (e_plugin_construct(ep, root) == -1) - e_plugin_enable(ep, FALSE); - g_hash_table_insert(ep_plugins, ep->id, ep); - - return ep; -} - -static int -ep_load(const char *filename) -{ - xmlDocPtr doc; - xmlNodePtr root; - int res = -1; - EPlugin *ep; - int cache = FALSE; - struct _plugin_doc *pdoc; - - doc = xmlParseFile(filename); - if (doc == NULL) { - return -1; - } - - root = xmlDocGetRootElement(doc); - if (strcmp(root->name, "e-plugin-list") != 0) { - g_warning("No <e-plugin-list> root element: %s", filename); - xmlFreeDoc(doc); - return -1; - } - - pdoc = g_malloc0(sizeof(*pdoc)); - pdoc->doc = doc; - pdoc->filename = g_strdup(filename); - - for (root = root->children; root ; root = root->next) { - if (strcmp(root->name, "e-plugin") == 0) { - ep = ep_load_plugin(root, pdoc); - if (ep) { - pdoc->plugin_hooks = g_slist_prepend(pdoc->plugin_hooks, ep); - cache |= (ep->hooks_pending != NULL); - } - cache |= pdoc->plugins != NULL; - } - } - - res = 0; - - if (cache) { - pd(printf("Caching plugin description '%s' for unknown future hooks\n", filename)); - e_dlist_addtail(&ep_plugin_docs, (EDListNode *)pdoc); - } else { - pd(printf("freeing plugin description '%s', nothing uses it\n", filename)); - xmlFreeDoc(pdoc->doc); - g_free(pdoc->filename); - g_free(pdoc); - } - - return res; -} - -/* This loads a hook that was pending on a given plugin but the type wasn't registered yet */ -/* This works in conjunction with ep_construct and e_plugin_hook_register_type to make sure - everything works nicely together. Apparently. */ -static int -ep_load_pending(EPlugin *ep, EPluginHookClass *type) -{ - int res = 0; - GSList *l, *p; - - phd(printf("New hook type registered '%s', loading pending hooks on plugin '%s'\n", type->id, ep->id)); - - l = ep->hooks_pending; - p = NULL; - while (l) { - GSList *n = l->next; - xmlNodePtr node = l->data; - char *class = xmlGetProp(node, "class"); - EPluginHook *hook; - - phd(printf(" checking pending hook '%s'\n", class?class:"<unknown>")); - - if (class) { - if (strcmp(class, type->id) == 0) { - hook = g_object_new(G_OBJECT_CLASS_TYPE(type), NULL); - res = type->construct(hook, ep, node); - if (res == -1) { - g_warning("Plugin '%s' failed to load hook '%s'", ep->name, type->id); - g_object_unref(hook); - } else { - ep->hooks = g_slist_append(ep->hooks, hook); - } - - if (p) - p->next = n; - else - ep->hooks_pending = n; - g_slist_free_1(l); - l = p; - } - - xmlFree(class); - } - - p = l; - l = n; - } - - return res; -} - -/** - * e_plugin_add_load_path: - * @path: The path to add to search for plugins. - * - * Add a path to be searched when e_plugin_load_plugins() is called. - * By default the system plugin directory and ~/.eplugins is used as - * the search path unless overriden by the environmental variable - * %EVOLUTION_PLUGIN_PATH. - * - * %EVOLUTION_PLUGIN_PATH is a : separated list of paths to search for - * plugin definitions in order. - * - * Plugin definitions are XML files ending in the extension ".eplug". - **/ -void -e_plugin_add_load_path(const char *path) -{ - ep_path = g_slist_append(ep_path, g_strdup(path)); -} - -/** - * e_plugin_load_plugins: - * - * Scan the search path, looking for plugin definitions, and load them - * into memory. - * - * Return value: Returns -1 if an error occured. - **/ -int -e_plugin_load_plugins(void) -{ - GSList *l; - - if (ep_types == NULL) { - g_warning("no plugin types defined"); - return 0; - } - - for (l = ep_path;l;l = g_slist_next(l)) { - DIR *dir; - struct dirent *d; - char *path = l->data; - - pd(printf("scanning plugin dir '%s'\n", path)); - - dir = opendir(path); - if (dir == NULL) { - /*g_warning("Could not find plugin path: %s", path);*/ - continue; - } - - while ( (d = readdir(dir)) ) { - if (strlen(d->d_name) > 6 - && !strcmp(d->d_name + strlen(d->d_name) - 6, ".eplug")) { - char * name = g_build_filename(path, d->d_name, NULL); - - ep_load(name); - g_free(name); - } - } - - closedir(dir); - } - - return 0; -} - -/** - * e_plugin_register_type: - * @type: The GObject type of the plugin loader. - * - * Register a new plugin type with the plugin system. Each type must - * subclass EPlugin and must override the type member of the - * EPluginClass with a unique name. - **/ -void -e_plugin_register_type(GType type) -{ - EPluginClass *klass; - struct _plugin_doc *pdoc, *ndoc; - - if (ep_types == NULL) { - ep_types = g_hash_table_new(g_str_hash, g_str_equal); - ep_plugins = g_hash_table_new(g_str_hash, g_str_equal); - /* TODO: notify listening */ - ep_gconf = gconf_client_get_default(); - ep_disabled = gconf_client_get_list(ep_gconf, "/apps/evolution/eplugin/disabled", GCONF_VALUE_STRING, NULL); - } - - klass = g_type_class_ref(type); - - pd(printf("register plugin type '%s'\n", klass->type)); - - g_hash_table_insert(ep_types, (void *)klass->type, klass); - - /* check for pending plugins */ - pdoc = (struct _plugin_doc *)ep_plugin_docs.head; - ndoc = pdoc->next; - while (ndoc) { - if (pdoc->plugins) { - GSList *l, *add = NULL; - - for (l=pdoc->plugins;l;l=g_slist_next(l)) { - xmlNodePtr root = l->data; - char *type; - - type = xmlGetProp(root, "type"); - if (!strcmp(type, klass->type)) - add = g_slist_append(add, l->data); - xmlFree(type); - } - - for (l=add;l;l=g_slist_next(l)) { - xmlNodePtr root = l->data; - EPlugin *ep; - - pdoc->plugins = g_slist_remove(pdoc->plugins, root); - ep = ep_load_plugin(root, pdoc); - if (ep) - pdoc->plugin_hooks = g_slist_prepend(pdoc->plugin_hooks, ep); - /* TODO: garbage collect plugin doc? */ - } - - g_slist_free(add); - } - - pdoc = ndoc; - ndoc = ndoc->next; - } -} - -static void -ep_list_plugin(void *key, void *val, void *dat) -{ - GSList **l = (GSList **)dat; - - *l = g_slist_prepend(*l, g_object_ref(val)); -} - -/** - * e_plugin_list_plugins: List all plugins. - * - * Static class method to retrieve a list of all current plugins. They - * are listed in no particular order. - * - * Return value: A GSList of all plugins, they must be - * g_object_unref'd and the list freed. - **/ -GSList * -e_plugin_list_plugins(void) -{ - GSList *l = NULL; - - if (ep_plugins) - g_hash_table_foreach(ep_plugins, ep_list_plugin, &l); - - return l; -} - -/** - * e_plugin_construct: - * @ep: An EPlugin derived object. - * @root: The XML root node of the sub-tree containing the plugin - * definition. - * - * Helper to invoke the construct virtual method. - * - * Return value: The return from the construct virtual method. - **/ -int -e_plugin_construct(EPlugin *ep, xmlNodePtr root) -{ - return ((EPluginClass *)G_OBJECT_GET_CLASS(ep))->construct(ep, root); -} - -/** - * e_plugin_invoke: - * @ep: - * @name: The name of the function to invoke. The format of this name - * will depend on the EPlugin type and its language conventions. - * @data: The argument to the function. Its actual type depends on - * the hook on which the function resides. It is up to the called - * function to get this right. - * - * Helper to invoke the invoke virtual method. - * - * Return value: The return of the plugin invocation. - **/ -void * -e_plugin_invoke(EPlugin *ep, const char *name, void *data) -{ - if (!ep->enabled) { - g_warning("Invoking method on disabled plugin, ignored"); - return NULL; - } - - return ((EPluginClass *)G_OBJECT_GET_CLASS(ep))->invoke(ep, name, data); -} - -/** - * e_plugin_enable: - * @ep: - * @state: - * - * Set the enable state of a plugin. - * - * THIS IS NOT FULLY IMPLEMENTED YET - **/ -void -e_plugin_enable(EPlugin *ep, int state) -{ - if ((ep->enabled == 0) == (state == 0)) - return; - - ((EPluginClass *)G_OBJECT_GET_CLASS(ep))->enable(ep, state); -} - -/** - * e_plugin_xml_prop: - * @node: An XML node. - * @id: The name of the property to retrieve. - * - * A static helper function to look up a property on an XML node, and - * ensure it is allocated in GLib system memory. If GLib isn't using - * the system malloc then it must copy the property value. - * - * Return value: The property, allocated in GLib memory, or NULL if no - * such property exists. - **/ -char * -e_plugin_xml_prop(xmlNodePtr node, const char *id) -{ - char *p = xmlGetProp(node, id); - - if (g_mem_is_system_malloc()) { - return p; - } else { - char * out = g_strdup(p); - - if (p) - xmlFree(p); - return out; - } -} - -/** - * e_plugin_xml_prop_domain: - * @node: An XML node. - * @id: The name of the property to retrieve. - * @domain: The translation domain for this string. - * - * A static helper function to look up a property on an XML node, and - * translate it based on @domain. - * - * Return value: The property, allocated in GLib memory, or NULL if no - * such property exists. - **/ -char * -e_plugin_xml_prop_domain(xmlNodePtr node, const char *id, const char *domain) -{ - char *p, *out; - - p = xmlGetProp(node, id); - if (p == NULL) - return NULL; - - out = g_strdup(dgettext(domain, p)); - xmlFree(p); - - return out; -} - -/** - * e_plugin_xml_int: - * @node: An XML node. - * @id: The name of the property to retrieve. - * @def: A default value if the property doesn't exist. Can be used - * to determine if the property isn't set. - * - * A static helper function to look up a property on an XML node as an - * integer. If the property doesn't exist, then @def is returned as a - * default value instead. - * - * Return value: The value if set, or @def if not. - **/ -int -e_plugin_xml_int(xmlNodePtr node, const char *id, int def) -{ - char *p = xmlGetProp(node, id); - - if (p) - return atoi(p); - else - return def; -} - -/** - * e_plugin_xml_content: - * @node: - * - * A static helper function to retrieve the entire textual content of - * an XML node, and ensure it is allocated in GLib system memory. If - * GLib isn't using the system malloc them it must copy the content. - * - * Return value: The node content, allocated in GLib memory. - **/ -char * -e_plugin_xml_content(xmlNodePtr node) -{ - char *p = xmlNodeGetContent(node); - - if (g_mem_is_system_malloc()) { - return p; - } else { - char * out = g_strdup(p); - - if (p) - xmlFree(p); - return out; - } -} - -/** - * e_plugin_xml_content_domain: - * @node: - * @domain: - * - * A static helper function to retrieve the entire textual content of - * an XML node, and ensure it is allocated in GLib system memory. If - * GLib isn't using the system malloc them it must copy the content. - * - * Return value: The node content, allocated in GLib memory. - **/ -char * -e_plugin_xml_content_domain(xmlNodePtr node, const char *domain) -{ - char *p, *out; - - p = xmlNodeGetContent(node); - if (p == NULL) - return NULL; - - out = g_strdup(dgettext(domain, p)); - xmlFree(p); - - return out; -} - -/* ********************************************************************** */ -static void *epl_parent_class; - -/* this looks weird, but it saves a lot of typing */ -#define epl ((EPluginLib *)ep) - -/* TODO: - We need some way to manage lifecycle. - We need some way to manage state. - - Maybe just the g module init method will do, or we could add - another which returns context. - - There is also the question of per-instance context, e.g. for config - pages. -*/ - -static int -epl_loadmodule(EPlugin *ep) -{ - if (epl->module == NULL) { - EPluginLibEnableFunc enable; - - if ((epl->module = g_module_open(epl->location, 0)) == NULL) { - g_warning("can't load plugin '%s'", g_module_error()); - return -1; - } - - if (g_module_symbol(epl->module, "e_plugin_lib_enable", (void *)&enable)) { - if (enable(epl, TRUE) != 0) { - ep->enabled = FALSE; - g_module_close(epl->module); - epl->module = NULL; - return -1; - } - } - } - - return 0; -} - -static void * -epl_invoke(EPlugin *ep, const char *name, void *data) -{ - EPluginLibFunc cb; - - if (!ep->enabled) { - g_warning("trying to invoke '%s' on disabled plugin '%s'", name, ep->id); - return NULL; - } - - if (epl_loadmodule(ep) != 0) - return NULL; - - if (!g_module_symbol(epl->module, name, (void *)&cb)) { - g_warning("Cannot resolve symbol '%s' in plugin '%s' (not exported?)", name, epl->location); - return NULL; - } - - return cb(epl, data); -} - -static int -epl_construct(EPlugin *ep, xmlNodePtr root) -{ - if (((EPluginClass *)epl_parent_class)->construct(ep, root) == -1) - return -1; - - epl->location = e_plugin_xml_prop(root, "location"); - - if (epl->location == NULL) { - g_warning("Library plugin '%s' has no location", ep->id); - return -1; - } - - /* If we're enabled, check for the load-on-startup property */ - if (ep->enabled) { - xmlChar *tmp; - - tmp = xmlGetProp(root, "load-on-startup"); - if (tmp) { - xmlFree(tmp); - if (epl_loadmodule(ep) != 0) - return -1; - } - } - - return 0; -} - -static void -epl_enable(EPlugin *ep, int state) -{ - EPluginLibEnableFunc enable; - - ((EPluginClass *)epl_parent_class)->enable(ep, state); - - /* if we're disabling and it isn't loaded, nothing to do */ - if (!state && epl->module == NULL) - return; - - /* this will noop if we're disabling since we tested it above */ - if (epl_loadmodule(ep) != 0) - return; - - if (g_module_symbol(epl->module, "e_plugin_lib_enable", (void *)&enable)) { - if (enable(epl, state) != 0) - return; - } -#if 0 - if (!state) { - g_module_close(epl->module); - epl->module = NULL; - } -#endif -} - -static void -epl_finalise(GObject *o) -{ - EPlugin *ep = (EPlugin *)o; - - g_free(epl->location); - - if (epl->module) - g_module_close(epl->module); - - ((GObjectClass *)epl_parent_class)->finalize(o); -} - -static void -epl_class_init(EPluginClass *klass) -{ - ((GObjectClass *)klass)->finalize = epl_finalise; - klass->construct = epl_construct; - klass->invoke = epl_invoke; - klass->enable = epl_enable; - klass->type = "shlib"; -} - -/** - * e_plugin_lib_get_type: - * - * Standard GObject function to retrieve the EPluginLib type. Use to - * register the type with the plugin system if you want to use shared - * library plugins. - * - * Return value: The EPluginLib type. - **/ -GType -e_plugin_lib_get_type(void) -{ - static GType type = 0; - - if (!type) { - static const GTypeInfo info = { - sizeof(EPluginLibClass), NULL, NULL, (GClassInitFunc) epl_class_init, NULL, NULL, - sizeof(EPluginLib), 0, (GInstanceInitFunc) NULL, - }; - - epl_parent_class = g_type_class_ref(e_plugin_get_type()); - type = g_type_register_static(e_plugin_get_type(), "EPluginLib", &info, 0); - } - - return type; -} - -/* ********************************************************************** */ - -static int -eph_construct(EPluginHook *eph, EPlugin *ep, xmlNodePtr root) -{ - eph->plugin = ep; - - return 0; -} - -static void -eph_enable(EPluginHook *eph, int state) -{ - /* NOOP */ -} - -static void -eph_finalise(GObject *o) -{ - ((GObjectClass *)eph_parent_class)->finalize((GObject *)o); -} - -static void -eph_class_init(EPluginHookClass *klass) -{ - ((GObjectClass *)klass)->finalize = eph_finalise; - klass->construct = eph_construct; - klass->enable = eph_enable; -} - -/** - * e_plugin_hook_get_type: - * - * Standard GObject function to retrieve the EPluginHook type. Since - * EPluginHook is an abstract class, this is only used to subclass it. - * - * Return value: The EPluginHook type. - **/ -GType -e_plugin_hook_get_type(void) -{ - static GType type = 0; - - if (!type) { - static const GTypeInfo info = { - sizeof(EPluginHookClass), NULL, NULL, (GClassInitFunc) eph_class_init, NULL, NULL, - sizeof(EPluginHook), 0, (GInstanceInitFunc) NULL, - }; - - eph_parent_class = g_type_class_ref(G_TYPE_OBJECT); - type = g_type_register_static(G_TYPE_OBJECT, "EPluginHook", &info, 0); - } - - return type; -} - -/** - * e_plugin_hook_enable: Set hook enabled state. - * @eph: - * @state: - * - * Set the enabled state of the plugin hook. This is called by the - * plugin code. - * - * THIS IS NOT FULY IMEPLEMENTED YET - **/ -void -e_plugin_hook_enable(EPluginHook *eph, int state) -{ - ((EPluginHookClass *)G_OBJECT_GET_CLASS(eph))->enable(eph, state); -} - -/** - * e_plugin_hook_register_type: - * @type: - * - * Register a new plugin hook type with the plugin system. Each type - * must subclass EPluginHook and must override the id member of the - * EPluginHookClass with a unique identification string. - **/ -void -e_plugin_hook_register_type(GType type) -{ - EPluginHookClass *klass, *oldklass; - GSList *l, *plugins; - char *class; - - if (eph_types == NULL) - eph_types = g_hash_table_new(g_str_hash, g_str_equal); - - klass = g_type_class_ref(type); - - oldklass = g_hash_table_lookup(eph_types, (void *)klass->id); - if (oldklass == klass) { - g_type_class_unref(klass); - return; - } else if (oldklass != NULL) { - g_warning("Trying to re-register hook type '%s'", klass->id); - return; - } - - phd(printf("register plugin hook type '%s'\n", klass->id)); - g_hash_table_insert(eph_types, (void *)klass->id, klass); - - /* if we've already loaded a plugin that needed this hook but it didn't exist, re-load it now */ - - if (ep_plugins_pending_hooks - && g_hash_table_lookup_extended(ep_plugins_pending_hooks, klass->id, (void **)&class, (void **)&plugins)) { - struct _plugin_doc *pdoc, *ndoc; - - g_hash_table_remove(ep_plugins_pending_hooks, class); - g_free(class); - for (l = plugins; l; l = g_slist_next(l)) { - EPlugin *ep = l->data; - - ep_load_pending(ep, klass); - } - g_slist_free(plugins); - - /* See if we can now garbage collect the xml definition since its been fully loaded */ - - /* This is all because libxml doesn't refcount! */ - - pdoc = (struct _plugin_doc *)ep_plugin_docs.head; - ndoc = pdoc->next; - while (ndoc) { - if (pdoc->doc) { - int cache = pdoc->plugins != NULL; - - for (l=pdoc->plugin_hooks;!cache && l;l=g_slist_next(l)) - cache |= (((EPlugin *)l->data)->hooks_pending != NULL); - - if (!cache) { - pd(printf("Gargabe collecting plugin description '%s'\n", pdoc->filename)); - e_dlist_remove((EDListNode *)pdoc); - xmlFreeDoc(pdoc->doc); - g_free(pdoc->filename); - g_free(pdoc); - } - } - - pdoc = ndoc; - ndoc = ndoc->next; - } - } -} - -/** - * e_plugin_hook_mask: - * @root: An XML node. - * @map: A zero-fill terminated array of EPluginHookTargeKeys used to - * map a string with a bit value. - * @prop: The property name. - * - * This is a static helper function which looks up a property @prop on - * the XML node @root, and then uses the @map table to convert it into - * a bitmask. The property value is a comma separated list of - * enumeration strings which are indexed into the @map table. - * - * Return value: A bitmask representing the inclusive-or of all of the - * integer values of the corresponding string id's stored in the @map. - **/ -guint32 -e_plugin_hook_mask(xmlNodePtr root, const struct _EPluginHookTargetKey *map, const char *prop) -{ - char *val, *p, *start, c; - guint32 mask = 0; - - val = xmlGetProp(root, prop); - if (val == NULL) - return 0; - - p = val; - do { - start = p; - while (*p && *p != ',') - p++; - c = *p; - *p = 0; - if (start != p) { - int i; - - for (i=0;map[i].key;i++) { - if (!strcmp(map[i].key, start)) { - mask |= map[i].value; - break; - } - } - } - *p++ = c; - } while (c); - - xmlFree(val); - - return mask; -} - -/** - * e_plugin_hook_id: - * @root: - * @map: - * @prop: - * - * This is a static helper function which looks up a property @prop on - * the XML node @root, and then uses the @map table to convert it into - * an integer. - * - * This is used as a helper wherever you need to represent an - * enumerated value in the XML. - * - * Return value: If the @prop value is in @map, then the corresponding - * integer value, if not, then ~0. - **/ -guint32 -e_plugin_hook_id(xmlNodePtr root, const struct _EPluginHookTargetKey *map, const char *prop) -{ - char *val; - int i; - - val = xmlGetProp(root, prop); - if (val == NULL) - return ~0; - - for (i=0;map[i].key;i++) { - if (!strcmp(map[i].key, val)) { - xmlFree(val); - return map[i].value; - } - } - - xmlFree(val); - - return ~0; -} - -/* ********************************************************************** */ -/* Plugin plugin */ - -static void *epth_parent_class; -#define epth ((EPluginTypeHook *)eph) - -static int -epth_load_plugin(void *d) -{ - EPluginHook *eph = d; - GType type; - - epth->idle = 0; - - type = GPOINTER_TO_UINT(e_plugin_invoke(eph->plugin, epth->get_type, eph->plugin)); - if (type != 0) - e_plugin_register_type(type); - - return FALSE; -} - -static int -epth_construct(EPluginHook *eph, EPlugin *ep, xmlNodePtr root) -{ - xmlNodePtr node; - - phd(printf("loading plugin hook\n")); - - if (((EPluginHookClass *)epth_parent_class)->construct(eph, ep, root) == -1) - return -1; - - node = root->children; - while (node) { - if (strcmp(node->name, "plugin-type") == 0) { - epth->get_type = e_plugin_xml_prop(node, "get-type"); - /* We need to run this in an idle handler, - * since at this point the parent EPlugin wont - * be fully initialised ... darn */ - if (epth->get_type) - epth->idle = g_idle_add(epth_load_plugin, epth); - else - g_warning("Plugin type plugin missing get-type callback"); - } - node = node->next; - } - - eph->plugin = ep; - - return 0; -} - -static void -epth_finalise(GObject *o) -{ - EPluginHook *eph = (EPluginHook *)o; - - if (epth->idle != 0) - g_source_remove(epth->idle); - - g_free(epth->get_type); - - ((GObjectClass *)eph_parent_class)->finalize((GObject *)o); -} - -static void -epth_class_init(EPluginHookClass *klass) -{ - ((GObjectClass *)klass)->finalize = epth_finalise; - klass->construct = epth_construct; - - klass->id = "org.gnome.evolution.plugin.type:1.0"; -} - -/** - * e_plugin_type_hook_get_type: - * - * Get the type for the plugin plugin hook. - * - * Return value: The type of the plugin type hook. - **/ -GType -e_plugin_type_hook_get_type(void) -{ - static GType type = 0; - - if (!type) { - static const GTypeInfo info = { - sizeof(EPluginTypeHookClass), NULL, NULL, (GClassInitFunc) epth_class_init, NULL, NULL, - sizeof(EPluginTypeHook), 0, (GInstanceInitFunc) NULL, - }; - - epth_parent_class = g_type_class_ref(e_plugin_hook_get_type()); - type = g_type_register_static(e_plugin_hook_get_type(), "EPluginTypeHook", &info, 0); - } - - return type; -} |