Update contrib.
1 /* -*- mode: C; c-file-style: "gnu" -*- */
2 /* config-loader-libxml.c libxml2 XML loader
4 * Copyright (C) 2003 Red Hat, Inc.
5 * Portion Copyright © 2008 Nokia Corporation and/or its subsidiary(-ies). All rights reserved.
6 * Licensed under the Academic Free License version 2.1
8 * This program is free software; you can redistribute it and/or modify
9 * it under the terms of the GNU General Public License as published by
10 * the Free Software Foundation; either version 2 of the License, or
11 * (at your option) any later version.
13 * This program is distributed in the hope that it will be useful,
14 * but WITHOUT ANY WARRANTY; without even the implied warranty of
15 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
16 * GNU General Public License for more details.
18 * You should have received a copy of the GNU General Public License
19 * along with this program; if not, write to the Free Software
20 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
24 #include "config-parser.h"
26 #include <dbus/dbus-internals.h>
28 #include "dbus-internals.h"
29 #endif //__SYMBIAN32__
31 #include <libxml2_xmlreader.h>
34 #include <libxml/xmlreader.h>
35 #include <libxml/parser.h>
36 #include <libxml/globals.h>
37 #include <libxml/xmlmemory.h>
42 /* About the error handling:
43 * - setup a "structured" error handler that catches structural
44 * errors and some oom errors
45 * - assume that a libxml function returning an error code means
48 #define _DBUS_MAYBE_SET_OOM(e) (dbus_error_is_set(e) ? (void)0 : _DBUS_SET_OOM(e))
52 xml_text_start_element (BusConfigParser *parser,
53 xmlTextReader *reader,
58 const char **attribute_names, **attribute_values;
60 int i, status, is_empty;
62 _DBUS_ASSERT_ERROR_IS_CLEAR (error);
65 attribute_names = NULL;
66 attribute_values = NULL;
69 name = (const char *) xmlTextReaderConstName (reader);
71 name = xmlTextReaderConstName (reader);
73 n_attributes = xmlTextReaderAttributeCount (reader);
74 is_empty = xmlTextReaderIsEmptyElement (reader);
76 if (name == NULL || n_attributes < 0 || is_empty == -1)
78 _DBUS_MAYBE_SET_OOM (error);
82 attribute_names = dbus_new0 (const char *, n_attributes + 1);
83 attribute_values = dbus_new0 (const char *, n_attributes + 1);
84 if (attribute_names == NULL || attribute_values == NULL)
86 _DBUS_SET_OOM (error);
90 while ((status = xmlTextReaderMoveToNextAttribute (reader)) == 1)
92 _dbus_assert (i < n_attributes);
94 attribute_names[i] = (const char *) xmlTextReaderConstName (reader);
95 attribute_values[i] = (const char *) xmlTextReaderConstValue (reader);
97 attribute_names[i] = xmlTextReaderConstName (reader);
98 attribute_values[i] = (xmlTextReaderConstValue (reader);
100 if (attribute_names[i] == NULL || attribute_values[i] == NULL)
102 _DBUS_MAYBE_SET_OOM (error);
109 _DBUS_MAYBE_SET_OOM (error);
112 _dbus_assert (i == n_attributes);
114 ret = bus_config_parser_start_element (parser, name,
115 attribute_names, attribute_values,
117 if (ret && is_empty == 1)
118 ret = bus_config_parser_end_element (parser, name, error);
121 dbus_free (attribute_names);
122 dbus_free (attribute_values);
127 static void xml_shut_up (void *ctx, const char *msg, ...)
133 xml_text_reader_error (void *arg, xmlErrorPtr xml_error)
135 DBusError *error = arg;
138 _dbus_verbose ("XML_ERROR level=%d, domain=%d, code=%d, msg=%s\n",
139 xml_error->level, xml_error->domain,
140 xml_error->code, xml_error->message);
143 if (!dbus_error_is_set (error))
145 if (xml_error->code == XML_ERR_NO_MEMORY)
146 _DBUS_SET_OOM (error);
147 else if (xml_error->level == XML_ERR_ERROR ||
148 xml_error->level == XML_ERR_FATAL)
149 dbus_set_error (error, DBUS_ERROR_FAILED,
150 "Error loading config file: '%s'",
157 bus_config_load (const DBusString *file,
158 dbus_bool_t is_toplevel,
159 const BusConfigParser *parent,
163 xmlTextReader *reader;
164 BusConfigParser *parser;
165 DBusString dirname, data;
169 _DBUS_ASSERT_ERROR_IS_CLEAR (error);
174 if (!_dbus_string_init (&dirname))
176 _DBUS_SET_OOM (error);
180 if (!_dbus_string_init (&data))
182 _DBUS_SET_OOM (error);
183 _dbus_string_free (&dirname);
189 /* xmlMemSetup only fails if one of the functions is NULL */
191 xmlMemSetup (dbus_free,
197 xmlSetGenericErrorFunc (NULL, xml_shut_up);
200 if (!_dbus_string_get_dirname (file, &dirname))
202 _DBUS_SET_OOM (error);
206 parser = bus_config_parser_new (&dirname, is_toplevel, parent);
209 _DBUS_SET_OOM (error);
213 if (!_dbus_file_get_contents (&data, file, error))
216 reader = xmlReaderForMemory (_dbus_string_get_const_data (&data),
217 _dbus_string_get_length (&data),
221 _DBUS_SET_OOM (error);
225 xmlTextReaderSetParserProp (reader, XML_PARSER_SUBST_ENTITIES, 1);
227 dbus_error_init (&tmp_error);
228 xmlTextReaderSetStructuredErrorHandler (reader, xml_text_reader_error, &tmp_error);
230 while ((ret = xmlTextReaderRead (reader)) == 1)
234 if (dbus_error_is_set (&tmp_error))
237 type = xmlTextReaderNodeType (reader);
240 _DBUS_MAYBE_SET_OOM (&tmp_error);
244 switch ((xmlReaderTypes) type) {
245 case XML_READER_TYPE_ELEMENT:
246 xml_text_start_element (parser, reader, &tmp_error);
249 case XML_READER_TYPE_TEXT:
250 case XML_READER_TYPE_CDATA:
255 value = (const char *) xmlTextReaderConstValue (reader);
257 value = xmlTextReaderConstValue (reader);
261 _dbus_string_init_const (&content, value);
262 bus_config_parser_content (parser, &content, &tmp_error);
265 _DBUS_MAYBE_SET_OOM (&tmp_error);
269 case XML_READER_TYPE_DOCUMENT_TYPE:
273 name = (const char *) xmlTextReaderConstName (reader);
275 name = xmlTextReaderConstName (reader);
278 bus_config_parser_check_doctype (parser, name, &tmp_error);
280 _DBUS_MAYBE_SET_OOM (&tmp_error);
284 case XML_READER_TYPE_END_ELEMENT:
288 name = (const char *) xmlTextReaderConstName (reader);
290 name = xmlTextReaderConstName (reader);
293 bus_config_parser_end_element (parser, name, &tmp_error);
295 _DBUS_MAYBE_SET_OOM (&tmp_error);
299 case XML_READER_TYPE_DOCUMENT:
300 case XML_READER_TYPE_DOCUMENT_FRAGMENT:
301 case XML_READER_TYPE_PROCESSING_INSTRUCTION:
302 case XML_READER_TYPE_COMMENT:
303 case XML_READER_TYPE_ENTITY:
304 case XML_READER_TYPE_NOTATION:
305 case XML_READER_TYPE_WHITESPACE:
306 case XML_READER_TYPE_SIGNIFICANT_WHITESPACE:
307 case XML_READER_TYPE_END_ENTITY:
308 case XML_READER_TYPE_XML_DECLARATION:
309 /* nothing to do, just read on */
312 case XML_READER_TYPE_NONE:
313 case XML_READER_TYPE_ATTRIBUTE:
314 case XML_READER_TYPE_ENTITY_REFERENCE:
315 _dbus_assert_not_reached ("unexpected nodes in XML");
318 if (dbus_error_is_set (&tmp_error))
323 _DBUS_MAYBE_SET_OOM (&tmp_error);
326 xmlFreeTextReader (reader);
328 if (dbus_error_is_set (&tmp_error))
330 dbus_move_error (&tmp_error, error);
334 if (!bus_config_parser_finished (parser, error))
336 _dbus_string_free (&dirname);
337 _dbus_string_free (&data);
340 _DBUS_ASSERT_ERROR_IS_CLEAR (error);
344 _DBUS_ASSERT_ERROR_IS_SET (error);
345 _dbus_string_free (&dirname);
346 _dbus_string_free (&data);
350 bus_config_parser_unref (parser);
351 _dbus_assert (reader == NULL); /* must go to reader_out first */