1 /* -*- mode: C; c-file-style: "gnu"; indent-tabs-mode: nil; -*- */
2 /* config-loader-libxml.c libxml2 XML loader
3 *
4 * Copyright (C) 2003 Red Hat, Inc.
5 *
6 * Licensed under the Academic Free License version 2.1
7 *
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.
12 *
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.
17 *
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., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
21 *
22 */
23
24 #include <config.h>
25 #include "config-parser.h"
26 #include <dbus/dbus-internals.h>
27 #include <libxml/xmlreader.h>
28 #include <libxml/parser.h>
29 #include <libxml/globals.h>
30 #include <libxml/xmlmemory.h>
31 #ifdef HAVE_ERRNO_H
32 #include <errno.h>
33 #endif
34 #include <string.h>
35
36 /* About the error handling:
37 * - setup a "structured" error handler that catches structural
38 * errors and some oom errors
39 * - assume that a libxml function returning an error code means
40 * out-of-memory
41 */
42 #define _DBUS_MAYBE_SET_OOM(e) (dbus_error_is_set(e) ? (void)0 : _DBUS_SET_OOM(e))
43
44
45 static dbus_bool_t
xml_text_start_element(BusConfigParser * parser,xmlTextReader * reader,DBusError * error)46 xml_text_start_element (BusConfigParser *parser,
47 xmlTextReader *reader,
48 DBusError *error)
49 {
50 const char *name;
51 int n_attributes;
52 const char **attribute_names, **attribute_values;
53 dbus_bool_t ret;
54 int i, status, is_empty;
55
56 _DBUS_ASSERT_ERROR_IS_CLEAR (error);
57
58 ret = FALSE;
59 attribute_names = NULL;
60 attribute_values = NULL;
61
62 name = xmlTextReaderConstName (reader);
63 n_attributes = xmlTextReaderAttributeCount (reader);
64 is_empty = xmlTextReaderIsEmptyElement (reader);
65
66 if (name == NULL || n_attributes < 0 || is_empty == -1)
67 {
68 _DBUS_MAYBE_SET_OOM (error);
69 goto out;
70 }
71
72 attribute_names = dbus_new0 (const char *, n_attributes + 1);
73 attribute_values = dbus_new0 (const char *, n_attributes + 1);
74 if (attribute_names == NULL || attribute_values == NULL)
75 {
76 _DBUS_SET_OOM (error);
77 goto out;
78 }
79 i = 0;
80 while ((status = xmlTextReaderMoveToNextAttribute (reader)) == 1)
81 {
82 _dbus_assert (i < n_attributes);
83 attribute_names[i] = xmlTextReaderConstName (reader);
84 attribute_values[i] = xmlTextReaderConstValue (reader);
85 if (attribute_names[i] == NULL || attribute_values[i] == NULL)
86 {
87 _DBUS_MAYBE_SET_OOM (error);
88 goto out;
89 }
90 i++;
91 }
92 if (status == -1)
93 {
94 _DBUS_MAYBE_SET_OOM (error);
95 goto out;
96 }
97 _dbus_assert (i == n_attributes);
98
99 ret = bus_config_parser_start_element (parser, name,
100 attribute_names, attribute_values,
101 error);
102 if (ret && is_empty == 1)
103 ret = bus_config_parser_end_element (parser, name, error);
104
105 out:
106 dbus_free (attribute_names);
107 dbus_free (attribute_values);
108
109 return ret;
110 }
111
xml_shut_up(void * ctx,const char * msg,...)112 static void xml_shut_up (void *ctx, const char *msg, ...)
113 {
114 return;
115 }
116
117 static void
xml_text_reader_error(void * arg,xmlErrorPtr xml_error)118 xml_text_reader_error (void *arg, xmlErrorPtr xml_error)
119 {
120 DBusError *error = arg;
121
122 #if 0
123 _dbus_verbose ("XML_ERROR level=%d, domain=%d, code=%d, msg=%s\n",
124 xml_error->level, xml_error->domain,
125 xml_error->code, xml_error->message);
126 #endif
127
128 if (!dbus_error_is_set (error))
129 {
130 if (xml_error->code == XML_ERR_NO_MEMORY)
131 _DBUS_SET_OOM (error);
132 else if (xml_error->level == XML_ERR_ERROR ||
133 xml_error->level == XML_ERR_FATAL)
134 dbus_set_error (error, DBUS_ERROR_FAILED,
135 "Error loading config file: '%s'",
136 xml_error->message);
137 }
138 }
139
140
141 BusConfigParser*
bus_config_load(const DBusString * file,dbus_bool_t is_toplevel,const BusConfigParser * parent,DBusError * error)142 bus_config_load (const DBusString *file,
143 dbus_bool_t is_toplevel,
144 const BusConfigParser *parent,
145 DBusError *error)
146
147 {
148 xmlTextReader *reader;
149 BusConfigParser *parser;
150 DBusString dirname, data;
151 DBusError tmp_error;
152 int ret;
153
154 _DBUS_ASSERT_ERROR_IS_CLEAR (error);
155
156 parser = NULL;
157 reader = NULL;
158
159 if (!_dbus_string_init (&dirname))
160 {
161 _DBUS_SET_OOM (error);
162 return NULL;
163 }
164
165 if (!_dbus_string_init (&data))
166 {
167 _DBUS_SET_OOM (error);
168 _dbus_string_free (&dirname);
169 return NULL;
170 }
171
172 if (is_toplevel)
173 {
174 /* xmlMemSetup only fails if one of the functions is NULL */
175 xmlMemSetup (dbus_free,
176 dbus_malloc,
177 dbus_realloc,
178 _dbus_strdup);
179 xmlInitParser ();
180 xmlSetGenericErrorFunc (NULL, xml_shut_up);
181 }
182
183 if (!_dbus_string_get_dirname (file, &dirname))
184 {
185 _DBUS_SET_OOM (error);
186 goto failed;
187 }
188
189 parser = bus_config_parser_new (&dirname, is_toplevel, parent);
190 if (parser == NULL)
191 {
192 _DBUS_SET_OOM (error);
193 goto failed;
194 }
195
196 if (!_dbus_file_get_contents (&data, file, error))
197 goto failed;
198
199 reader = xmlReaderForMemory (_dbus_string_get_const_data (&data),
200 _dbus_string_get_length (&data),
201 NULL, NULL, 0);
202 if (reader == NULL)
203 {
204 _DBUS_SET_OOM (error);
205 goto failed;
206 }
207
208 xmlTextReaderSetParserProp (reader, XML_PARSER_SUBST_ENTITIES, 1);
209
210 dbus_error_init (&tmp_error);
211 xmlTextReaderSetStructuredErrorHandler (reader, xml_text_reader_error, &tmp_error);
212
213 while ((ret = xmlTextReaderRead (reader)) == 1)
214 {
215 int type;
216
217 if (dbus_error_is_set (&tmp_error))
218 goto reader_out;
219
220 type = xmlTextReaderNodeType (reader);
221 if (type == -1)
222 {
223 _DBUS_MAYBE_SET_OOM (&tmp_error);
224 goto reader_out;
225 }
226
227 switch ((xmlReaderTypes) type) {
228 case XML_READER_TYPE_ELEMENT:
229 xml_text_start_element (parser, reader, &tmp_error);
230 break;
231
232 case XML_READER_TYPE_TEXT:
233 case XML_READER_TYPE_CDATA:
234 {
235 DBusString content;
236 const char *value;
237 value = xmlTextReaderConstValue (reader);
238 if (value != NULL)
239 {
240 _dbus_string_init_const (&content, value);
241 bus_config_parser_content (parser, &content, &tmp_error);
242 }
243 else
244 _DBUS_MAYBE_SET_OOM (&tmp_error);
245 break;
246 }
247
248 case XML_READER_TYPE_DOCUMENT_TYPE:
249 {
250 const char *name;
251 name = xmlTextReaderConstName (reader);
252 if (name != NULL)
253 bus_config_parser_check_doctype (parser, name, &tmp_error);
254 else
255 _DBUS_MAYBE_SET_OOM (&tmp_error);
256 break;
257 }
258
259 case XML_READER_TYPE_END_ELEMENT:
260 {
261 const char *name;
262 name = xmlTextReaderConstName (reader);
263 if (name != NULL)
264 bus_config_parser_end_element (parser, name, &tmp_error);
265 else
266 _DBUS_MAYBE_SET_OOM (&tmp_error);
267 break;
268 }
269
270 case XML_READER_TYPE_DOCUMENT:
271 case XML_READER_TYPE_DOCUMENT_FRAGMENT:
272 case XML_READER_TYPE_PROCESSING_INSTRUCTION:
273 case XML_READER_TYPE_COMMENT:
274 case XML_READER_TYPE_ENTITY:
275 case XML_READER_TYPE_NOTATION:
276 case XML_READER_TYPE_WHITESPACE:
277 case XML_READER_TYPE_SIGNIFICANT_WHITESPACE:
278 case XML_READER_TYPE_END_ENTITY:
279 case XML_READER_TYPE_XML_DECLARATION:
280 /* nothing to do, just read on */
281 break;
282
283 case XML_READER_TYPE_NONE:
284 case XML_READER_TYPE_ATTRIBUTE:
285 case XML_READER_TYPE_ENTITY_REFERENCE:
286 _dbus_assert_not_reached ("unexpected nodes in XML");
287 }
288
289 if (dbus_error_is_set (&tmp_error))
290 goto reader_out;
291 }
292
293 if (ret == -1)
294 _DBUS_MAYBE_SET_OOM (&tmp_error);
295
296 reader_out:
297 xmlFreeTextReader (reader);
298 reader = NULL;
299 if (dbus_error_is_set (&tmp_error))
300 {
301 dbus_move_error (&tmp_error, error);
302 goto failed;
303 }
304
305 if (!bus_config_parser_finished (parser, error))
306 goto failed;
307 _dbus_string_free (&dirname);
308 _dbus_string_free (&data);
309 if (is_toplevel)
310 xmlCleanupParser();
311 _DBUS_ASSERT_ERROR_IS_CLEAR (error);
312 return parser;
313
314 failed:
315 _DBUS_ASSERT_ERROR_IS_SET (error);
316 _dbus_string_free (&dirname);
317 _dbus_string_free (&data);
318 if (is_toplevel)
319 xmlCleanupParser();
320 if (parser)
321 bus_config_parser_unref (parser);
322 _dbus_assert (reader == NULL); /* must go to reader_out first */
323 return NULL;
324 }
325