1
0
mirror of https://github.com/rkd77/elinks.git synced 2025-01-03 14:57:44 -05:00
elinks/src/bookmarks/backend/xbel.c
Petr Baudis 0f6d4310ad Initial commit of the HEAD branch of the ELinks CVS repository, as of
Thu Sep 15 15:57:07 CEST 2005. The previous history can be added to this
by grafting.
2005-09-15 15:58:31 +02:00

550 lines
12 KiB
C

/* Internal bookmarks XBEL bookmarks basic support */
/* $Id: xbel.c,v 1.53 2005/04/29 07:40:51 zas Exp $ */
/*
* TODO: Decent XML output.
* TODO: Validation of the document (with librxp?). An invalid document can
* crash elinks.
* TODO: Support all the XBEL elements. */
#ifdef HAVE_CONFIG_H
#include "config.h"
#endif /* HAVE_CONFIG_H */
#include <ctype.h>
#include <expat.h>
#include <stdio.h>
#include <stdlib.h>
#ifdef HAVE_UNISTD_H
#include <unistd.h>
#endif
#include "elinks.h"
#include "bfu/dialog.h"
#include "bookmarks/bookmarks.h"
#include "bookmarks/backend/common.h"
#include "bookmarks/backend/xbel.h"
#include "intl/charsets.h"
#include "intl/gettext/libintl.h"
#include "util/conv.h"
#include "util/lists.h"
#include "util/string.h"
#define BOOKMARKS_XBEL_FILENAME "bookmarks.xbel"
/* Elements' attributes */
struct attributes {
LIST_HEAD(struct attributes);
unsigned char *name;
};
/* Prototypes */
static void on_element_open(void *data, const char *name, const char **attr);
static void on_element_close(void *data, const char *name);
static void on_text(void *data, const XML_Char *text, int len);
static struct tree_node *new_node(struct tree_node *parent);
static void free_node(struct tree_node *node);
static void free_xbeltree(struct tree_node *node);
static struct tree_node *get_child(struct tree_node *node, unsigned char *name);
static unsigned char *get_attribute_value(struct attributes *attr,
unsigned char *name);
static void read_bookmarks_xbel(FILE *f);
static unsigned char * filename_bookmarks_xbel(int writing);
static int xbeltree_to_bookmarks_list(struct tree_node *root,
struct bookmark *current_parent);
static void write_bookmarks_list(struct secure_save_info *ssi,
struct list_head *bookmarks_list,
int n, int folder_state);
static void write_bookmarks_xbel(struct secure_save_info *ssi,
struct list_head *bookmarks_list);
/* Element */
struct tree_node {
unsigned char *name; /* Name of the element */
unsigned char *text; /* Text inside the element */
struct attributes *attrs; /* Attributes of the element */
struct tree_node *parent;
struct tree_node *children;
struct tree_node *prev;
struct tree_node *next;
};
static struct tree_node *root_node = NULL;
static struct tree_node *current_node = NULL;
/* This is 1 so that we won't fail miserably if we read bookmarks in a
* different format. */
static int readok = 1;
static void
read_bookmarks_xbel(FILE *f)
{
unsigned char in_buffer[BUFSIZ];
XML_Parser p;
int done = 0;
int err = 0;
readok = 0;
p = XML_ParserCreate(NULL);
if (!p) {
ERROR(gettext("read_bookmarks_xbel(): Error in XML_ParserCreate()"));
return;
}
XML_SetElementHandler(p, on_element_open, on_element_close);
XML_SetCharacterDataHandler(p, on_text);
while (!done && !err) {
size_t len = fread(in_buffer, 1, BUFSIZ, f);
if (ferror(f)) {
ERROR(gettext("read_bookmarks_xbel(): Error reading %s"),
filename_bookmarks_xbel(0));
err = 1;
} else {
done = feof(f);
if (!err && !XML_Parse(p, in_buffer, len, done)) {
usrerror(gettext("Parse error while processing "
"XBEL bookmarks in %s at line %d "
"column %d:\n%s"),
filename_bookmarks_xbel(0),
XML_GetCurrentLineNumber(p),
XML_GetCurrentColumnNumber(p),
XML_ErrorString(XML_GetErrorCode(p)));
err = 1;
}
}
}
if (!err) readok = xbeltree_to_bookmarks_list(root_node->children, NULL); /* Top node is xbel */
XML_ParserFree(p);
free_xbeltree(root_node);
}
static void
write_bookmarks_xbel(struct secure_save_info *ssi,
struct list_head *bookmarks_list)
{
int folder_state = get_opt_bool("bookmarks.folder_state");
/* We check for readok in filename_bookmarks_xbel(). */
secure_fputs(ssi,
"<?xml version=\"1.0\"?>\n"
"<!DOCTYPE xbel PUBLIC \"+//IDN python.org//DTD XML "
"Bookmark Exchange Language 1.0//EN//XML\"\n"
" "
"\"http://www.python.org/topics/xml/dtds/xbel-1.0.dtd\">\n\n"
"<xbel>\n\n\n");
write_bookmarks_list(ssi, bookmarks_list, 0, folder_state);
secure_fputs(ssi, "\n</xbel>\n");
}
static unsigned char *
filename_bookmarks_xbel(int writing)
{
if (writing && !readok) return NULL;
return BOOKMARKS_XBEL_FILENAME;
}
static void
indentation(struct secure_save_info *ssi, int num)
{
int i;
for (i = 0; i < num; i++)
secure_fputs(ssi, " ");
}
/* FIXME This is totally broken, we should use the Unicode value in
* numeric entities.
* Additionally it is slow, not elegant, incomplete and
* if you pay enough attention you can smell the unmistakable
* odor of doom coming from it. --fabio */
static void
print_xml_entities(struct secure_save_info *ssi, const unsigned char *str)
{
#define accept_char(x) (isident((x)) || (x) == ' ' || (x) == '.' \
|| (x) == ':' || (x) == ';' \
|| (x) == '/' || (x) == '(' \
|| (x) == ')' || (x) == '}' \
|| (x) == '{' || (x) == '%' \
|| (x) == '+')
static int cp = 0;
if (!cp) get_cp_index("us-ascii");
for (; *str; str++) {
if (accept_char(*str))
secure_fputc(ssi, *str);
else {
if (isascii(*str)) {
secure_fprintf(ssi, "&#%i;", (int) *str);
}
else {
unsigned char *s = u2cp_no_nbsp(*str, cp);
if (s) print_xml_entities(ssi, s);
}
}
}
#undef accept_char
}
static void
write_bookmarks_list(struct secure_save_info *ssi,
struct list_head *bookmarks_list,
int n, int folder_state)
{
struct bookmark *bm;
foreach (bm, *bookmarks_list) {
indentation(ssi, n + 1);
if (bm->box_item->type == BI_FOLDER) {
int expanded = folder_state && bm->box_item->expanded;
secure_fputs(ssi, "<folder folded=\"");
secure_fputs(ssi, expanded ? "no" : "yes");
secure_fputs(ssi, "\">\n");
indentation(ssi, n + 2);
secure_fputs(ssi, "<title>");
print_xml_entities(ssi, bm->title);
secure_fputs(ssi, "</title>\n");
if (!list_empty(bm->child))
write_bookmarks_list(ssi, &bm->child, n + 2, folder_state);
indentation(ssi, n + 1);
secure_fputs(ssi, "</folder>\n\n");
} else if (bm->box_item->type == BI_LEAF) {
secure_fputs(ssi, "<bookmark href=\"");
print_xml_entities(ssi, bm->url);
secure_fputs(ssi, "\">\n");
indentation(ssi, n + 2);
secure_fputs(ssi, "<title>");
print_xml_entities(ssi, bm->title);
secure_fputs(ssi, "</title>\n");
indentation(ssi, n + 1);
secure_fputs(ssi, "</bookmark>\n\n");
} else if (bm->box_item->type == BI_SEPARATOR) {
secure_fputs(ssi, "<separator/>\n\n");
}
}
}
static void
on_element_open(void *data, const char *name, const char **attr)
{
struct attributes *attribute;
struct tree_node *node;
node = new_node(current_node);
if (!node) return;
if (root_node) {
if (current_node->children) {
struct tree_node *tmp;
tmp = current_node->children;
current_node->children = node;
current_node->children->next = tmp;
current_node->children->prev = NULL;
}
else current_node->children = node;
}
else root_node = node;
current_node = node;
current_node->name = stracpy((unsigned char *) name);
if (!current_node->name) {
mem_free(current_node);
return;
}
while (*attr) {
unsigned char *tmp = stracpy((unsigned char *) *attr);
if (!tmp) {
free_node(current_node);
return;
}
attribute = mem_calloc(1, sizeof(*attribute));
if (!attribute) {
mem_free(tmp);
free_node(current_node);
return;
}
attribute->name = tmp;
add_to_list(*current_node->attrs, attribute);
++attr;
}
}
static void
on_element_close(void *data, const char *name)
{
current_node = current_node->parent;
}
static unsigned char *
delete_whites(unsigned char *s)
{
unsigned char *r;
int count = 0, c = 0, i;
int len = strlen(s);
r = mem_alloc(len + 1);
if (!r) return NULL;
for (i = 0; i < len; i++) {
if (isspace(s[i])) {
if (count == 1) continue;
else count = 1;
}
else count = 0;
if (s[i] == '\n' || s[i] == '\t')
r[c++] = ' ';
else r[c++] = s[i];
}
r[c] = '\0';
/* XXX This should never return NULL, right? wrong! --fabio */
/* r = mem_realloc(r, strlen(r + 1)); */
return r;
}
static void
on_text(void *data, const XML_Char *text, int len)
{
char *tmp;
int len2 = 0;
if (len) {
len2 = current_node->text ? strlen(current_node->text) : 0;
tmp = mem_realloc(current_node->text, (size_t) (len + 1 + len2));
/* Out of memory */
if (!tmp) return;
strncpy(tmp + len2, text, len);
tmp[len + len2] = '\0';
current_node->text = delete_whites(tmp);
mem_free(tmp);
}
}
/* xbel_tree_to_bookmarks_list: returns 0 on fail,
* 1 on success */
static int
xbeltree_to_bookmarks_list(struct tree_node *node,
struct bookmark *current_parent)
{
struct bookmark *tmp;
struct tree_node *title;
static struct bookmark *lastbm;
while (node) {
if (!strcmp(node->name, "bookmark")) {
unsigned char *href;
title = get_child(node, "title");
href = get_attribute_value(node->attrs, "href");
tmp = add_bookmark(current_parent, 0,
/* The <title> element is optional */
title ? title->text
: (unsigned char *) gettext("No title"),
/* XXX: The href attribute isn't optional but
* we don't validate the source XML yet, so
* we can't always assume a non NULL value for
* get_attribute_value() */
href ? href
: (unsigned char *) gettext("No URL"));
/* Out of memory */
if (!tmp) return 0;
tmp->root = current_parent;
lastbm = tmp;
} else if (!strcmp(node->name, "folder")) {
unsigned char *folded;
title = get_child(node, "title");
tmp = add_bookmark(current_parent, 0,
title ? title->text
: (unsigned char *) gettext("No title"),
NULL);
/* Out of memory */
if (!tmp) return 0;
folded = get_attribute_value(node->attrs, "folded");
if (folded && !strcmp(folded, "no"))
tmp->box_item->expanded = 1;
lastbm = tmp;
} else if (!strcmp(node->name, "separator")) {
tmp = add_bookmark(current_parent, 0, "-", "");
/* Out of memory */
if (!tmp) return 0;
tmp->root = current_parent;
lastbm = tmp;
}
if (node->children) {
int ret;
/* If this node is a <folder> element, current parent
* changes */
ret = (!strcmp(node->name, "folder") ?
xbeltree_to_bookmarks_list(node->children,
lastbm) :
xbeltree_to_bookmarks_list(node->children,
current_parent));
/* Out of memory */
if (!ret) return 0;
}
node = node->next;
}
/* Success */
return 1;
}
static void
free_xbeltree(struct tree_node *node)
{
struct tree_node *next_node;
while (node) {
if (node->children)
free_xbeltree(node->children);
next_node = node->next;
free_node(node);
node = next_node;
}
}
static struct tree_node *
get_child(struct tree_node *node, unsigned char *name)
{
struct tree_node *ret;
if (!node) return NULL;
ret = node->children;
while (ret) {
if (!strcmp(name, ret->name)) {
return ret;
}
ret = ret->next;
}
return NULL;
}
static unsigned char *
get_attribute_value(struct attributes *attr, unsigned char *name)
{
struct attributes *attribute;
foreachback (attribute, *attr) {
if (!strcmp(attribute->name, name)) {
return attribute->prev->name;
}
}
return NULL;
}
static struct tree_node *
new_node(struct tree_node *parent)
{
struct tree_node *node;
node = mem_calloc(1, sizeof(*node));
if (!node) return NULL;
node->parent = parent ? parent : node;
node->attrs = mem_calloc(1, sizeof(*node->attrs));
if (!node->attrs) {
mem_free(node);
return NULL;
}
init_list(*node->attrs);
return node;
}
static void
free_node(struct tree_node *node)
{
struct attributes *attribute;
if (node->attrs) {
foreachback (attribute, *node->attrs)
mem_free_if(attribute->name);
free_list(*(struct list_head *) node->attrs); /* Don't free list during traversal */
mem_free(node->attrs);
}
mem_free_if(node->name);
mem_free_if(node->text);
mem_free(node);
}
/* Read and write functions for the XBEL backend */
struct bookmarks_backend xbel_bookmarks_backend = {
filename_bookmarks_xbel,
read_bookmarks_xbel,
write_bookmarks_xbel,
};