aboutsummaryrefslogtreecommitdiffstats
path: root/src/ephy-extensions-manager.c
diff options
context:
space:
mode:
Diffstat (limited to 'src/ephy-extensions-manager.c')
-rw-r--r--src/ephy-extensions-manager.c430
1 files changed, 73 insertions, 357 deletions
diff --git a/src/ephy-extensions-manager.c b/src/ephy-extensions-manager.c
index 5eaf35801..e7589be6d 100644
--- a/src/ephy-extensions-manager.c
+++ b/src/ephy-extensions-manager.c
@@ -37,6 +37,12 @@
#include <libxml/tree.h>
#include <libxml/xmlreader.h>
+#include <libxml/globals.h>
+#include <libxml/tree.h>
+#include <libxml/xmlwriter.h>
+#include <libxslt/xslt.h>
+#include <libxslt/transform.h>
+#include <libxslt/xsltutils.h>
#include <libgnomevfs/gnome-vfs-ops.h>
#include <libgnomevfs/gnome-vfs-utils.h>
@@ -55,6 +61,8 @@
#define CONF_LOADED_EXTENSIONS "/apps/epiphany/general/active_extensions"
#define DOT_INI ".ephy-extension"
+#define ENABLE_LEGACY_FORMAT
+
#define EPHY_EXTENSIONS_MANAGER_GET_PRIVATE(object)(G_TYPE_INSTANCE_GET_PRIVATE ((object), EPHY_TYPE_EXTENSIONS_MANAGER, EphyExtensionsManagerPrivate))
struct _EphyExtensionsManagerPrivate
@@ -67,6 +75,10 @@ struct _EphyExtensionsManagerPrivate
GList *dir_monitors;
GList *windows;
guint active_extensions_notifier_id;
+
+#ifdef ENABLE_LEGACY_FORMAT
+ xsltStylesheetPtr xml2ini_xsl;
+#endif
};
typedef struct
@@ -291,359 +303,6 @@ find_extension_info (const ExtensionInfo *info,
return strcmp (info->info.identifier, identifier);
}
-typedef struct
-{
- xmlChar *string;
- guint match;
-} LocalisedString;
-
-static const char * const *languages = NULL;
-
-static void
-assign_localised_string (xmlTextReaderPtr reader,
- LocalisedString *string)
-{
- const xmlChar *value, *lang;
- guint i;
-
- lang = xmlTextReaderConstXmlLang (reader);
- value = xmlTextReaderConstValue (reader);
-
- if (G_UNLIKELY (lang == NULL))
- {
- /* languages always has "C" in it, so make sure we get a match */
- lang = (const xmlChar *) "C";
- }
-
- for (i = 0; languages[i] != NULL && i < string->match; i++)
- {
- if (lang != NULL &&
- strcmp ((const char *) lang, languages[i]) == 0)
- {
- xmlFree (string->string);
- string->string = xmlStrdup (value);
- string->match = i;
- break;
- }
- }
-}
-
-typedef enum
-{
- STATE_START,
- STATE_STOP,
- STATE_ERROR,
- STATE_EXTENSION,
- STATE_NAME,
- STATE_DESCRIPTION,
- STATE_VERSION,
- STATE_AUTHOR,
- STATE_URL,
- STATE_GETTEXT_DOMAIN,
- STATE_LOCALE_DIRECTORY,
- STATE_LOADER,
- STATE_LOADER_ATTRIBUTE,
- STATE_LOAD_DEFERRED,
-} ParserState;
-
-static void
-ephy_extensions_manager_load_xml_string (EphyExtensionsManager *manager,
- const char *identifier,
- /* const */ char *xml)
-{
- xmlDocPtr doc;
- xmlTextReaderPtr reader;
- ParserState state = STATE_START;
- GQuark attr_quark = 0;
- EphyExtensionInfo *einfo;
- ExtensionInfo *info;
- int ret;
- LocalisedString description = { NULL, G_MAXUINT };
- LocalisedString name = { NULL, G_MAXUINT };
-
- LOG ("Loading XML description file for '%s'", identifier);
-
- if (g_list_find_custom (manager->priv->data, identifier,
- (GCompareFunc) find_extension_info) != NULL)
- {
- g_warning ("Extension description for '%s' already read!",
- identifier);
- return;
- }
-
- /* FIXME: Ideally we'd put the schema validator in the reader. libxml2
- * doesn't seem to support that at this point in time, so we've got to
- * put the schema validation on the Doc Tree and then pass that to the
- * reader. (maybe switch to RelaxNG?)
- */
- doc = xmlParseDoc ((xmlChar *) xml);
-
- if (doc == NULL)
- {
- g_warning ("Couldn't read '%s' data\n", identifier);
- return;
- }
-
- /* Now parse it */
- reader = xmlReaderWalker (doc);
- g_return_if_fail (reader != NULL);
-
- info = g_new0 (ExtensionInfo, 1);
- einfo = (EphyExtensionInfo *) info;
- einfo->identifier = g_strdup (identifier);
- g_datalist_init (&info->loader_attributes);
-
- ret = xmlTextReaderRead (reader);
-
- while (ret == 1)
- {
- const xmlChar *tag;
- xmlReaderTypes type;
-
- tag = xmlTextReaderConstName (reader);
- type = xmlTextReaderNodeType (reader);
-
- if (state == STATE_LOADER &&
- type == XML_READER_TYPE_ELEMENT &&
- xmlStrEqual (tag, (const xmlChar *) "attribute"))
- {
- xmlChar *name;
-
- state = STATE_LOADER_ATTRIBUTE;
-
- name = xmlTextReaderGetAttribute (reader, (const xmlChar *) "name");
- attr_quark = g_quark_from_string ((const char *) name);
- xmlFree (name);
- }
- else if (state == STATE_EXTENSION &&
- type == XML_READER_TYPE_ELEMENT &&
- xmlStrEqual (tag, (const xmlChar *) "author"))
- {
- state = STATE_AUTHOR;
- }
- else if (state == STATE_EXTENSION &&
- type == XML_READER_TYPE_ELEMENT &&
- xmlStrEqual (tag, (const xmlChar *) "description"))
- {
- state = STATE_DESCRIPTION;
- }
- else if (state == STATE_EXTENSION &&
- type == XML_READER_TYPE_ELEMENT &&
- xmlStrEqual (tag, (const xmlChar *) "gettext-domain"))
- {
- state = STATE_GETTEXT_DOMAIN;
- }
- else if (state == STATE_EXTENSION &&
- type == XML_READER_TYPE_ELEMENT &&
- xmlStrEqual (tag, (const xmlChar *) "load-deferred"))
- {
- state = STATE_LOAD_DEFERRED;
- }
- else if (state == STATE_EXTENSION &&
- type == XML_READER_TYPE_ELEMENT &&
- xmlStrEqual (tag, (const xmlChar *) "locale-directory"))
- {
- state = STATE_LOCALE_DIRECTORY;
- }
- else if (state == STATE_EXTENSION &&
- type == XML_READER_TYPE_ELEMENT &&
- xmlStrEqual (tag, (const xmlChar *) "name"))
- {
- state = STATE_NAME;
- }
- else if (state == STATE_EXTENSION &&
- type == XML_READER_TYPE_ELEMENT &&
- xmlStrEqual (tag, (const xmlChar *) "url"))
- {
- state = STATE_URL;
- }
- else if (state == STATE_EXTENSION &&
- type == XML_READER_TYPE_ELEMENT &&
- xmlStrEqual (tag, (const xmlChar *) "version"))
- {
- state = STATE_VERSION;
- }
- else if (state == STATE_EXTENSION &&
- type == XML_READER_TYPE_ELEMENT &&
- xmlStrEqual (tag, (const xmlChar *) "loader"))
- {
- xmlChar * attr;
-
- state = STATE_LOADER;
-
- attr = xmlTextReaderGetAttribute
- (reader, (const xmlChar *) "type");
-
- info->loader_type = g_strdup ((char*)attr);
-
- xmlFree (attr);
- }
- else if (state == STATE_LOADER_ATTRIBUTE &&
- type == XML_READER_TYPE_TEXT &&
- attr_quark != 0)
- {
- const xmlChar *value;
-
- value = xmlTextReaderConstValue (reader);
-
- g_datalist_id_set_data_full (&info->loader_attributes,
- attr_quark,
- g_strdup ((char*)value),
- (GDestroyNotify) g_free);
- attr_quark = 0;
- }
- else if (state == STATE_LOADER_ATTRIBUTE &&
- type == XML_READER_TYPE_END_ELEMENT &&
- xmlStrEqual (tag, (const xmlChar *) "attribute"))
- {
- state = STATE_LOADER;
- }
- else if (state == STATE_AUTHOR &&
- type == XML_READER_TYPE_TEXT)
- {
- const xmlChar *attr;
-
- attr = xmlTextReaderConstValue (reader);
-
- einfo->authors = g_list_prepend
- (einfo->authors, g_strdup ((char*)attr));
- }
- else if (state == STATE_DESCRIPTION &&
- type == XML_READER_TYPE_TEXT)
- {
- assign_localised_string (reader, &description);
- }
- else if (type == XML_READER_TYPE_TEXT &&
- (state == STATE_GETTEXT_DOMAIN ||
- state == STATE_LOAD_DEFERRED ||
- state == STATE_LOCALE_DIRECTORY))
- {
- /* not supported anymore */
- }
- else if (state == STATE_NAME &&
- type == XML_READER_TYPE_TEXT)
- {
- assign_localised_string (reader, &name);
- }
- else if (state == STATE_VERSION &&
- type == XML_READER_TYPE_TEXT)
- {
- info->version = (guint) strtol ((const char *) xmlTextReaderConstValue (reader), NULL, 10);
- }
- else if (state == STATE_URL &&
- type == XML_READER_TYPE_TEXT)
- {
- const xmlChar *attr;
-
- attr = xmlTextReaderConstValue (reader);
-
- einfo->url = g_strdup ((char*)attr);
- }
- else if (state == STATE_AUTHOR &&
- type == XML_READER_TYPE_END_ELEMENT &&
- xmlStrEqual (tag, (const xmlChar *) "author"))
- {
- state = STATE_EXTENSION;
- }
- else if (state == STATE_DESCRIPTION &&
- type == XML_READER_TYPE_END_ELEMENT &&
- xmlStrEqual (tag, (const xmlChar *) "description"))
- {
- state = STATE_EXTENSION;
- }
- else if (state == STATE_GETTEXT_DOMAIN &&
- type == XML_READER_TYPE_END_ELEMENT &&
- xmlStrEqual (tag, (const xmlChar *) "gettext-domain"))
- {
- state = STATE_EXTENSION;
- }
- else if (state == STATE_LOCALE_DIRECTORY &&
- type == XML_READER_TYPE_END_ELEMENT &&
- xmlStrEqual (tag, (const xmlChar *) "locale-directory"))
- {
- state = STATE_EXTENSION;
- }
- else if (state == STATE_LOADER &&
- type == XML_READER_TYPE_END_ELEMENT &&
- xmlStrEqual (tag, (const xmlChar *) "loader"))
- {
- state = STATE_EXTENSION;
- }
- else if (state == STATE_NAME &&
- type == XML_READER_TYPE_END_ELEMENT &&
- xmlStrEqual (tag, (const xmlChar *) "name"))
- {
- state = STATE_EXTENSION;
- }
- else if (state == STATE_URL &&
- type == XML_READER_TYPE_END_ELEMENT &&
- xmlStrEqual (tag, (const xmlChar *) "url"))
- {
- state = STATE_EXTENSION;
- }
- else if (state == STATE_VERSION &&
- type == XML_READER_TYPE_END_ELEMENT &&
- xmlStrEqual (tag, (const xmlChar *) "version"))
- {
- state = STATE_EXTENSION;
- }
- else if (type == XML_READER_TYPE_SIGNIFICANT_WHITESPACE ||
- type == XML_READER_TYPE_WHITESPACE ||
- type == XML_READER_TYPE_TEXT)
- {
- /* eat it */
- }
- else if (state == STATE_START &&
- type == XML_READER_TYPE_ELEMENT &&
- xmlStrEqual (tag, (const xmlChar *) "extension"))
- {
- state = STATE_EXTENSION;
- }
- else if (state == STATE_EXTENSION &&
- type == XML_READER_TYPE_END_ELEMENT &&
- xmlStrEqual (tag, (const xmlChar *) "extension"))
- {
- state = STATE_STOP;
- }
- else
- {
- const xmlChar *content;
-
- content = xmlTextReaderConstValue (reader);
- g_warning ("tag '%s' of type %d in state %d with content '%s' was unexpected!",
- tag, type, state, content ? (char *) content : "(null)");
-
- state = STATE_ERROR;
- break;
- }
-
- ret = xmlTextReaderRead (reader);
- }
-
- xmlFreeTextReader (reader);
- xmlFreeDoc (doc);
-
- /* assign localised strings */
- einfo->description = (char *)description.string;
- einfo->name = (char *)name.string;
-
- /* Reverse authors list */
- einfo->authors = g_list_reverse (einfo->authors);
-
- /* sanity check */
- if (ret < 0 || state != STATE_STOP ||
- einfo->name == NULL || einfo->description == NULL ||
- info->loader_type == NULL || info->loader_type[0] == '\0')
- {
- free_extension_info (info);
- return;
- }
-
- manager->priv->data = g_list_prepend (manager->priv->data, info);
-
- g_signal_emit (manager, signals[ADDED], 0, info);
-}
-
static void
ephy_extensions_manager_load_ini_string (EphyExtensionsManager *manager,
const char * identifier,
@@ -770,6 +429,57 @@ ephy_extensions_manager_load_ini_string (EphyExtensionsManager *manager,
g_signal_emit (manager, signals[ADDED], 0, info);
}
+#ifdef ENABLE_LEGACY_FORMAT
+
+static void
+ephy_extensions_manager_load_xml_string (EphyExtensionsManager *manager,
+ const char *identifier,
+ /* const */ char *xml)
+{
+ EphyExtensionsManagerPrivate *priv = manager->priv;
+ xmlDocPtr doc, res;
+ const xmlChar *xsl_file;
+ xmlChar *output = NULL;
+ int outlen = -1, ret = -1;
+
+ START_PROFILER ("Transforming .xml -> " DOT_INI)
+
+ doc = xmlParseMemory (xml, strlen (xml));
+ if (!doc) goto out;
+
+ if (priv->xml2ini_xsl == NULL)
+ {
+ xsl_file = (const xmlChar *) ephy_file ("ephy-xml2ini.xsl");
+ if (!xsl_file) return;
+
+ priv->xml2ini_xsl = xsltParseStylesheetFile (xsl_file);
+ if (priv->xml2ini_xsl == NULL)
+ {
+ g_warning ("Couldn't parse the XSL to transform .xml extension descriptions!\n");
+ goto out;
+ }
+ }
+
+ res = xsltApplyStylesheet (priv->xml2ini_xsl, doc, NULL);
+ if (!res) goto out;
+
+ ret = xsltSaveResultToString (&output, &outlen, res, priv->xml2ini_xsl);
+
+ xmlFreeDoc (res);
+ xmlFreeDoc (doc);
+
+out:
+ STOP_PROFILER ("Transforming .xml -> " DOT_INI)
+
+ if (ret >= 0 && output != NULL && outlen > -1)
+ {
+ ephy_extensions_manager_load_ini_string (manager, identifier, (char*) output);
+ xmlFree (output);
+ }
+}
+
+#endif /* ENABLE_LEGACY_FORMAT */
+
static char *
path_to_identifier (const char *path)
{
@@ -815,12 +525,14 @@ ephy_extensions_manager_load_file (EphyExtensionsManager *manager,
{
ephy_extensions_manager_load_ini_string (manager, identifier,
contents);
- }
+ }
+#ifdef ENABLE_LEGACY_FORMAT
else if (g_str_has_suffix (path, ".xml"))
{
ephy_extensions_manager_load_xml_string (manager, identifier,
contents);
}
+#endif
g_free (identifier);
g_free (contents);
@@ -1344,6 +1056,13 @@ ephy_extensions_manager_finalize (GObject *object)
LOG ("EphyExtensionsManager finalising");
+#ifdef ENABLE_LEGACY_FORMAT
+ if (priv->xml2ini_xsl != NULL)
+ {
+ xsltFreeStylesheet (priv->xml2ini_xsl);
+ }
+#endif
+
eel_gconf_notification_remove (manager->priv->active_extensions_notifier_id);
g_list_foreach (priv->dir_monitors, (GFunc) gnome_vfs_monitor_cancel, NULL);
@@ -1504,7 +1223,4 @@ ephy_extensions_manager_class_init (EphyExtensionsManagerClass *class)
G_TYPE_POINTER);
g_type_class_add_private (object_class, sizeof (EphyExtensionsManagerPrivate));
-
- languages = g_get_language_names ();
- g_assert (languages != NULL);
}