VirtualBox

source: vbox/trunk/src/VBox/Main/xml/Settings.cpp@ 14854

Last change on this file since 14854 was 14854, checked in by vboxsync, 16 years ago

Main: rip XML classes out of settings code and move them to independent files and new vboxxml namespace to make them useful outside of the settings context

  • Property svn:eol-style set to native
  • Property svn:keywords set to Date Revision Author Id
File size: 36.3 KB
Line 
1/** @file
2 * Settings File Manipulation API.
3 */
4
5/*
6 * Copyright (C) 2007 Sun Microsystems, Inc.
7 *
8 * This file is part of VirtualBox Open Source Edition (OSE), as
9 * available from http://www.virtualbox.org. This file is free software;
10 * you can redistribute it and/or modify it under the terms of the GNU
11 * General Public License (GPL) as published by the Free Software
12 * Foundation, in version 2 as it comes in the "COPYING" file of the
13 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
14 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
15 *
16 * Please contact Sun Microsystems, Inc., 4150 Network Circle, Santa
17 * Clara, CA 95054 USA or visit http://www.sun.com if you need
18 * additional information or have any questions.
19 */
20
21#include "VBox/settings.h"
22
23#include "Logging.h"
24
25#include <iprt/err.h>
26#include <iprt/file.h>
27#include <iprt/lock.h>
28
29#include <libxml/tree.h>
30#include <libxml/parser.h>
31#include <libxml/globals.h>
32#include <libxml/xmlIO.h>
33#include <libxml/xmlsave.h>
34#include <libxml/uri.h>
35
36#include <libxml/xmlschemas.h>
37
38#include <libxslt/xsltInternals.h>
39#include <libxslt/transform.h>
40#include <libxslt/xsltutils.h>
41
42#include <string.h>
43
44
45/**
46 * Global module initialization structure.
47 *
48 * The constructor and destructor of this structure are used to perform global
49 * module initiaizaton and cleanup. Thee must be only one global variable of
50 * this structure.
51 */
52static
53class Global
54{
55public:
56
57 Global()
58 {
59 /* Check the parser version. The docs say it will kill the app if
60 * there is a serious version mismatch, but I couldn't find it in the
61 * source code (it only prints the error/warning message to the console) so
62 * let's leave it as is for informational purposes. */
63 LIBXML_TEST_VERSION
64
65 /* Init libxml */
66 xmlInitParser();
67
68 /* Save the default entity resolver before someone has replaced it */
69 xml.defaultEntityLoader = xmlGetExternalEntityLoader();
70 }
71
72 ~Global()
73 {
74 /* Shutdown libxml */
75 xmlCleanupParser();
76 }
77
78 struct
79 {
80 xmlExternalEntityLoader defaultEntityLoader;
81
82 /** Used to provide some thread safety missing in libxml2 (see e.g.
83 * XmlTreeBackend::read()) */
84 RTLockMtx lock;
85 }
86 xml;
87}
88gGlobal;
89
90
91namespace settings
92{
93
94// Helpers
95////////////////////////////////////////////////////////////////////////////////
96
97inline int sFromHex (char aChar)
98{
99 if (aChar >= '0' && aChar <= '9')
100 return aChar - '0';
101 if (aChar >= 'A' && aChar <= 'F')
102 return aChar - 'A' + 0xA;
103 if (aChar >= 'a' && aChar <= 'f')
104 return aChar - 'a' + 0xA;
105
106 throw ENoConversion (FmtStr ("'%c' (0x%02X) is not hex", aChar, aChar));
107}
108
109inline char sToHex (int aDigit)
110{
111 return (aDigit < 0xA) ? aDigit + '0' : aDigit - 0xA + 'A';
112}
113
114static char *duplicate_chars (const char *that)
115{
116 char *result = NULL;
117 if (that != NULL)
118 {
119 size_t len = strlen (that) + 1;
120 result = new char [len];
121 if (result != NULL)
122 memcpy (result, that, len);
123 }
124 return result;
125}
126
127//////////////////////////////////////////////////////////////////////////////
128// string -> type conversions
129//////////////////////////////////////////////////////////////////////////////
130
131uint64_t FromStringInteger (const char *aValue, bool aSigned,
132 int aBits, uint64_t aMin, uint64_t aMax)
133{
134 if (aValue == NULL)
135 throw ENoValue();
136
137 switch (aBits)
138 {
139 case 8:
140 case 16:
141 case 32:
142 case 64:
143 break;
144 default:
145 throw vboxxml::ENotImplemented (RT_SRC_POS);
146 }
147
148 if (aSigned)
149 {
150 int64_t result;
151 int vrc = RTStrToInt64Full (aValue, 0, &result);
152 if (RT_SUCCESS (vrc))
153 {
154 if (result >= (int64_t) aMin && result <= (int64_t) aMax)
155 return (uint64_t) result;
156 }
157 }
158 else
159 {
160 uint64_t result;
161 int vrc = RTStrToUInt64Full (aValue, 0, &result);
162 if (RT_SUCCESS (vrc))
163 {
164 if (result >= aMin && result <= aMax)
165 return result;
166 }
167 }
168
169 throw ENoConversion (FmtStr ("'%s' is not integer", aValue));
170}
171
172template<> bool FromString <bool> (const char *aValue)
173{
174 if (aValue == NULL)
175 throw ENoValue();
176
177 if (strcmp (aValue, "true") == 0 ||
178 strcmp (aValue, "1") == 0)
179 /* This contradicts the XML Schema's interpretation of boolean: */
180 //strcmp (aValue, "yes") == 0 ||
181 //strcmp (aValue, "on") == 0)
182 return true;
183 else if (strcmp (aValue, "false") == 0 ||
184 strcmp (aValue, "0") == 0)
185 /* This contradicts the XML Schema's interpretation of boolean: */
186 //strcmp (aValue, "no") == 0 ||
187 //strcmp (aValue, "off") == 0)
188 return false;
189
190 throw ENoConversion (FmtStr ("'%s' is not bool", aValue));
191}
192
193template<> RTTIMESPEC FromString <RTTIMESPEC> (const char *aValue)
194{
195 if (aValue == NULL)
196 throw ENoValue();
197
198 /* Parse ISO date (xsd:dateTime). The format is:
199 * '-'? yyyy '-' mm '-' dd 'T' hh ':' mm ':' ss ('.' s+)? (zzzzzz)?
200 * where zzzzzz is: (('+' | '-') hh ':' mm) | 'Z' */
201 uint32_t yyyy = 0;
202 uint16_t mm = 0, dd = 0, hh = 0, mi = 0, ss = 0;
203 char buf [256];
204 if (strlen (aValue) > RT_ELEMENTS (buf) - 1 ||
205 sscanf (aValue, "%d-%hu-%huT%hu:%hu:%hu%s",
206 &yyyy, &mm, &dd, &hh, &mi, &ss, buf) == 7)
207 {
208 /* currently, we accept only the UTC timezone ('Z'),
209 * ignoring fractional seconds, if present */
210 if (buf [0] == 'Z' ||
211 (buf [0] == '.' && buf [strlen (buf) - 1] == 'Z'))
212 {
213 RTTIME time = { yyyy, (uint8_t) mm, 0, 0, (uint8_t) dd,
214 (uint8_t) hh, (uint8_t) mi, (uint8_t) ss, 0,
215 RTTIME_FLAGS_TYPE_UTC };
216 if (RTTimeNormalize (&time))
217 {
218 RTTIMESPEC timeSpec;
219 if (RTTimeImplode (&timeSpec, &time))
220 return timeSpec;
221 }
222 }
223 else
224 throw ENoConversion (FmtStr ("'%s' is not UTC date", aValue));
225 }
226
227 throw ENoConversion (FmtStr ("'%s' is not ISO date", aValue));
228}
229
230stdx::char_auto_ptr FromString (const char *aValue, size_t *aLen)
231{
232 if (aValue == NULL)
233 throw ENoValue();
234
235 /* each two chars produce one byte */
236 size_t len = strlen (aValue) / 2;
237
238 /* therefore, the original length must be even */
239 if (len % 2 != 0)
240 throw ENoConversion (FmtStr ("'%.*s' is not binary data",
241 aLen, aValue));
242
243 stdx::char_auto_ptr result (new char [len]);
244
245 const char *src = aValue;
246 char *dst = result.get();
247
248 for (size_t i = 0; i < len; ++ i, ++ dst)
249 {
250 *dst = sFromHex (*src ++) << 4;
251 *dst |= sFromHex (*src ++);
252 }
253
254 if (aLen != NULL)
255 *aLen = len;
256
257 return result;
258}
259
260//////////////////////////////////////////////////////////////////////////////
261// type -> string conversions
262//////////////////////////////////////////////////////////////////////////////
263
264stdx::char_auto_ptr ToStringInteger (uint64_t aValue, unsigned int aBase,
265 bool aSigned, int aBits)
266{
267 unsigned int flags = RTSTR_F_SPECIAL;
268 if (aSigned)
269 flags |= RTSTR_F_VALSIGNED;
270
271 /* maximum is binary representation + terminator */
272 size_t len = aBits + 1;
273
274 switch (aBits)
275 {
276 case 8:
277 flags |= RTSTR_F_8BIT;
278 break;
279 case 16:
280 flags |= RTSTR_F_16BIT;
281 break;
282 case 32:
283 flags |= RTSTR_F_32BIT;
284 break;
285 case 64:
286 flags |= RTSTR_F_64BIT;
287 break;
288 default:
289 throw vboxxml::ENotImplemented (RT_SRC_POS);
290 }
291
292 stdx::char_auto_ptr result (new char [len]);
293 int vrc = RTStrFormatNumber (result.get(), aValue, aBase, 0, 0, flags);
294 if (RT_SUCCESS (vrc))
295 return result;
296
297 throw vboxxml::EIPRTFailure (vrc);
298}
299
300template<> stdx::char_auto_ptr ToString <bool> (const bool &aValue,
301 unsigned int aExtra /* = 0 */)
302{
303 /* Convert to the canonical form according to XML Schema */
304 stdx::char_auto_ptr result (duplicate_chars (aValue ? "true" : "false"));
305 return result;
306}
307
308template<> stdx::char_auto_ptr ToString <RTTIMESPEC> (const RTTIMESPEC &aValue,
309 unsigned int aExtra /* = 0 */)
310{
311 RTTIME time;
312 if (!RTTimeExplode (&time, &aValue))
313 throw ENoConversion (FmtStr ("timespec %lld ms is invalid",
314 RTTimeSpecGetMilli (&aValue)));
315
316 /* Store ISO date (xsd:dateTime). The format is:
317 * '-'? yyyy '-' mm '-' dd 'T' hh ':' mm ':' ss ('.' s+)? (zzzzzz)?
318 * where zzzzzz is: (('+' | '-') hh ':' mm) | 'Z' */
319 char buf [256];
320 RTStrPrintf (buf, sizeof (buf),
321 "%04ld-%02hd-%02hdT%02hd:%02hd:%02hdZ",
322 time.i32Year, (uint16_t) time.u8Month, (uint16_t) time.u8MonthDay,
323 (uint16_t) time.u8Hour, (uint16_t) time.u8Minute, (uint16_t) time.u8Second);
324
325 stdx::char_auto_ptr result (duplicate_chars (buf));
326 return result;
327}
328
329stdx::char_auto_ptr ToString (const char *aData, size_t aLen)
330{
331 /* each byte will produce two hex digits and there will be a null
332 * terminator */
333 stdx::char_auto_ptr result (new char [aLen * 2 + 1]);
334
335 const char *src = aData;
336 char *dst = result.get();
337
338 for (size_t i = 0; i < aLen; ++ i, ++ src)
339 {
340 *dst++ = sToHex ((*src) >> 4);
341 *dst++ = sToHex ((*src) & 0xF);
342 }
343
344 *dst = '\0';
345
346 return result;
347}
348
349//////////////////////////////////////////////////////////////////////////////
350// XmlKeyBackend Class
351//////////////////////////////////////////////////////////////////////////////
352
353class XmlKeyBackend : public Key::Backend
354{
355public:
356
357 XmlKeyBackend (xmlNodePtr aNode);
358 ~XmlKeyBackend();
359
360 const char *name() const;
361 void setName (const char *aName);
362 const char *value (const char *aName) const;
363 void setValue (const char *aName, const char *aValue);
364
365 Key::List keys (const char *aName = NULL) const;
366 Key findKey (const char *aName) const;
367
368 Key appendKey (const char *aName);
369 void zap();
370
371 void *position() const { return mNode; }
372
373private:
374
375 xmlNodePtr mNode;
376
377 xmlChar *mNodeText;
378
379 DECLARE_CLS_COPY_CTOR_ASSIGN_NOOP (XmlKeyBackend);
380
381 friend class XmlTreeBackend;
382};
383
384XmlKeyBackend::XmlKeyBackend (xmlNodePtr aNode)
385 : mNode (aNode), mNodeText (NULL)
386{
387 AssertReturnVoid (mNode);
388 AssertReturnVoid (mNode->type == XML_ELEMENT_NODE);
389}
390
391XmlKeyBackend::~XmlKeyBackend()
392{
393 xmlFree (mNodeText);
394}
395
396const char *XmlKeyBackend::name() const
397{
398 return mNode ? (char *) mNode->name : NULL;
399}
400
401void XmlKeyBackend::setName (const char *aName)
402{
403 throw vboxxml::ENotImplemented (RT_SRC_POS);
404}
405
406const char *XmlKeyBackend::value (const char *aName) const
407{
408 if (!mNode)
409 return NULL;
410
411 if (aName == NULL)
412 {
413 /* @todo xmlNodeListGetString (,,1) returns NULL for things like
414 * <Foo></Foo> and may want to return "" in this case to distinguish
415 * from <Foo/> (where NULL is pretty much expected). */
416 if (!mNodeText)
417 unconst (mNodeText) =
418 xmlNodeListGetString (mNode->doc, mNode->children, 0);
419 return (char *) mNodeText;
420 }
421
422 xmlAttrPtr attr = xmlHasProp (mNode, (const xmlChar *) aName);
423 if (!attr)
424 return NULL;
425
426 if (attr->type == XML_ATTRIBUTE_NODE)
427 {
428 /* @todo for now, we only understand the most common case: only 1 text
429 * node comprises the attribute's contents. Otherwise we'd need to
430 * return a newly allocated string buffer to the caller that
431 * concatenates all text nodes and obey him to free it or provide our
432 * own internal map of attribute=value pairs and return const pointers
433 * to values from this map. */
434 if (attr->children != NULL &&
435 attr->children->next == NULL &&
436 (attr->children->type == XML_TEXT_NODE ||
437 attr->children->type == XML_CDATA_SECTION_NODE))
438 return (char *) attr->children->content;
439 }
440 else if (attr->type == XML_ATTRIBUTE_DECL)
441 {
442 return (char *) ((xmlAttributePtr) attr)->defaultValue;
443 }
444
445 return NULL;
446}
447
448void XmlKeyBackend::setValue (const char *aName, const char *aValue)
449{
450 if (!mNode)
451 return;
452
453 if (aName == NULL)
454 {
455 xmlChar *value = (xmlChar *) aValue;
456 if (value != NULL)
457 {
458 value = xmlEncodeSpecialChars (mNode->doc, value);
459 if (value == NULL)
460 throw vboxxml::ENoMemory();
461 }
462
463 xmlNodeSetContent (mNode, value);
464
465 if (value != (xmlChar *) aValue)
466 xmlFree (value);
467
468 /* outdate the node text holder */
469 if (mNodeText != NULL)
470 {
471 xmlFree (mNodeText);
472 mNodeText = NULL;
473 }
474
475 return;
476 }
477
478 if (aValue == NULL)
479 {
480 /* remove the attribute if it exists */
481 xmlAttrPtr attr = xmlHasProp (mNode, (const xmlChar *) aName);
482 if (attr != NULL)
483 {
484 int rc = xmlRemoveProp (attr);
485 if (rc != 0)
486 throw vboxxml::EInvalidArg (RT_SRC_POS);
487 }
488 return;
489 }
490
491 xmlAttrPtr attr = xmlSetProp (mNode, (const xmlChar *) aName,
492 (const xmlChar *) aValue);
493 if (attr == NULL)
494 throw vboxxml::ENoMemory();
495}
496
497Key::List XmlKeyBackend::keys (const char *aName /* = NULL */) const
498{
499 Key::List list;
500
501 if (!mNode)
502 return list;
503
504 for (xmlNodePtr node = mNode->children; node; node = node->next)
505 {
506 if (node->type == XML_ELEMENT_NODE)
507 {
508 if (aName == NULL ||
509 strcmp (aName, (char *) node->name) == 0)
510 list.push_back (Key (new XmlKeyBackend (node)));
511 }
512 }
513
514 return list;
515}
516
517Key XmlKeyBackend::findKey (const char *aName) const
518{
519 Key key;
520
521 if (!mNode)
522 return key;
523
524 for (xmlNodePtr node = mNode->children; node; node = node->next)
525 {
526 if (node->type == XML_ELEMENT_NODE)
527 {
528 if (aName == NULL ||
529 strcmp (aName, (char *) node->name) == 0)
530 {
531 key = Key (new XmlKeyBackend (node));
532 break;
533 }
534 }
535 }
536
537 return key;
538}
539
540Key XmlKeyBackend::appendKey (const char *aName)
541{
542 if (!mNode)
543 return Key();
544
545 xmlNodePtr node = xmlNewChild (mNode, NULL, (const xmlChar *) aName, NULL);
546 if (node == NULL)
547 throw vboxxml::ENoMemory();
548
549 return Key (new XmlKeyBackend (node));
550}
551
552void XmlKeyBackend::zap()
553{
554 if (!mNode)
555 return;
556
557 xmlUnlinkNode (mNode);
558 xmlFreeNode (mNode);
559 mNode = NULL;
560}
561
562//////////////////////////////////////////////////////////////////////////////
563// XmlTreeBackend Class
564//////////////////////////////////////////////////////////////////////////////
565
566class XmlTreeBackend::XmlError : public XmlTreeBackend::Error
567{
568public:
569
570 XmlError (xmlErrorPtr aErr)
571 {
572 if (!aErr)
573 throw vboxxml::EInvalidArg (RT_SRC_POS);
574
575 char *msg = Format (aErr);
576 setWhat (msg);
577 RTStrFree (msg);
578 }
579
580 /**
581 * Composes a single message for the given error. The caller must free the
582 * returned string using RTStrFree() when no more necessary.
583 */
584 static char *Format (xmlErrorPtr aErr)
585 {
586 const char *msg = aErr->message ? aErr->message : "<none>";
587 size_t msgLen = strlen (msg);
588 /* strip spaces, trailing EOLs and dot-like char */
589 while (msgLen && strchr (" \n.?!", msg [msgLen - 1]))
590 -- msgLen;
591
592 char *finalMsg = NULL;
593 RTStrAPrintf (&finalMsg, "%.*s.\nLocation: '%s', line %d (%d), column %d",
594 msgLen, msg, aErr->file, aErr->line, aErr->int1, aErr->int2);
595
596 return finalMsg;
597 }
598};
599
600struct XmlTreeBackend::Data
601{
602 Data() : ctxt (NULL), doc (NULL)
603 , inputResolver (NULL)
604 , autoConverter (NULL), oldVersion (NULL) {}
605
606 xmlParserCtxtPtr ctxt;
607 xmlDocPtr doc;
608
609 Key root;
610
611 InputResolver *inputResolver;
612
613 AutoConverter *autoConverter;
614 char *oldVersion;
615
616 std::auto_ptr <stdx::exception_trap_base> trappedErr;
617
618 /**
619 * This is to avoid throwing exceptions while in libxml2 code and
620 * redirect them to our level instead. Also used to perform clean up
621 * by deleting the I/O stream instance and self when requested.
622 */
623 struct IOCtxt
624 {
625 IOCtxt (vboxxml::Stream *aStream, std::auto_ptr <stdx::exception_trap_base> &aErr)
626 : stream (aStream), deleteStreamOnClose (false)
627 , err (aErr) {}
628
629 template <typename T>
630 void setErr (const T& aErr) { err.reset (new stdx::exception_trap <T> (aErr)); }
631
632 void resetErr() { err.reset(); }
633
634 vboxxml::Stream *stream;
635 bool deleteStreamOnClose;
636
637 std::auto_ptr <stdx::exception_trap_base> &err;
638 };
639
640 struct InputCtxt : public IOCtxt
641 {
642 InputCtxt (vboxxml::Input *aInput, std::auto_ptr <stdx::exception_trap_base> &aErr)
643 : IOCtxt (aInput, aErr), input (aInput) {}
644
645 vboxxml::Input *input;
646 };
647
648 struct OutputCtxt : public IOCtxt
649 {
650 OutputCtxt (vboxxml::Output *aOutput, std::auto_ptr <stdx::exception_trap_base> &aErr)
651 : IOCtxt (aOutput, aErr), output (aOutput) {}
652
653 vboxxml::Output *output;
654 };
655};
656
657XmlTreeBackend::XmlTreeBackend()
658 : m (new Data())
659{
660 /* create a parser context */
661 m->ctxt = xmlNewParserCtxt();
662 if (m->ctxt == NULL)
663 throw vboxxml::ENoMemory();
664}
665
666XmlTreeBackend::~XmlTreeBackend()
667{
668 reset();
669
670 xmlFreeParserCtxt (m->ctxt);
671 m->ctxt = NULL;
672}
673
674void XmlTreeBackend::setInputResolver (InputResolver &aResolver)
675{
676 m->inputResolver = &aResolver;
677}
678
679void XmlTreeBackend::resetInputResolver()
680{
681 m->inputResolver = NULL;
682}
683
684void XmlTreeBackend::setAutoConverter (AutoConverter &aConverter)
685{
686 m->autoConverter = &aConverter;
687}
688
689void XmlTreeBackend::resetAutoConverter()
690{
691 m->autoConverter = NULL;
692}
693
694const char *XmlTreeBackend::oldVersion() const
695{
696 return m->oldVersion;
697}
698
699extern "C" xmlGenericErrorFunc xsltGenericError;
700extern "C" void *xsltGenericErrorContext;
701
702void XmlTreeBackend::rawRead (vboxxml::Input &aInput, const char *aSchema /* = NULL */,
703 int aFlags /* = 0 */)
704{
705 /* Reset error variables used to memorize exceptions while inside the
706 * libxml2 code. */
707 m->trappedErr.reset();
708
709 /* We use the global lock for the whole duration of this method to serialize
710 * access to thread-unsafe xmlGetExternalEntityLoader() and some other
711 * calls. It means that only one thread is able to parse an XML stream at a
712 * time but another choice would be to patch libxml2/libxslt which is
713 * unwanted now for several reasons. Search for "thread-safe" to find all
714 * unsafe cases. */
715 RTLock alock (gGlobal.xml.lock);
716
717 xmlExternalEntityLoader oldEntityLoader = xmlGetExternalEntityLoader();
718 sThat = this;
719 xmlSetExternalEntityLoader (ExternalEntityLoader);
720
721 xmlDocPtr doc = NULL;
722
723 try
724 {
725 /* Note: when parsing we use XML_PARSE_NOBLANKS to instruct libxml2 to
726 * remove text nodes that contain only blanks. This is important because
727 * otherwise xmlSaveDoc() won't be able to do proper indentation on
728 * output. */
729
730 /* parse the stream */
731 /* NOTE: new InputCtxt instance will be deleted when the stream is closed by
732 * the libxml2 API (e.g. when calling xmlFreeParserCtxt()) */
733 doc = xmlCtxtReadIO (m->ctxt,
734 ReadCallback, CloseCallback,
735 new Data::InputCtxt (&aInput, m->trappedErr),
736 aInput.uri(), NULL,
737 XML_PARSE_NOBLANKS);
738 if (doc == NULL)
739 {
740 /* look if there was a forwared exception from the lower level */
741 if (m->trappedErr.get() != NULL)
742 m->trappedErr->rethrow();
743
744 throw XmlError (xmlCtxtGetLastError (m->ctxt));
745 }
746
747 char *oldVersion = NULL;
748
749 /* perform automatic document transformation if necessary */
750 if (m->autoConverter != NULL &&
751 m->autoConverter->
752 needsConversion (Key (new XmlKeyBackend (xmlDocGetRootElement (doc))),
753 &oldVersion))
754 {
755 xmlDocPtr xsltDoc = NULL;
756 xsltStylesheetPtr xslt = NULL;
757 char *errorStr = NULL;
758
759 xmlGenericErrorFunc oldXsltGenericError = xsltGenericError;
760 void *oldXsltGenericErrorContext = xsltGenericErrorContext;
761
762 try
763 {
764 /* parse the XSLT template */
765 {
766 vboxxml::Input *xsltInput =
767 m->inputResolver->resolveEntity
768 (m->autoConverter->templateUri(), NULL);
769 /* NOTE: new InputCtxt instance will be deleted when the
770 * stream is closed by the libxml2 API */
771 xsltDoc = xmlCtxtReadIO (m->ctxt,
772 ReadCallback, CloseCallback,
773 new Data::InputCtxt (xsltInput, m->trappedErr),
774 m->autoConverter->templateUri(),
775 NULL, 0);
776 delete xsltInput;
777 }
778
779 if (xsltDoc == NULL)
780 {
781 /* look if there was a forwared exception from the lower level */
782 if (m->trappedErr.get() != NULL)
783 m->trappedErr->rethrow();
784
785 throw XmlError (xmlCtxtGetLastError (m->ctxt));
786 }
787
788 /* setup stylesheet compilation and transformation error
789 * reporting. Note that we could create a new transform context
790 * for doing xsltApplyStylesheetUser and use
791 * xsltSetTransformErrorFunc() on it to set a dedicated error
792 * handler but as long as we already do several non-thread-safe
793 * hacks, this is not really important. */
794
795 xsltGenericError = ValidityErrorCallback;
796 xsltGenericErrorContext = &errorStr;
797
798 xslt = xsltParseStylesheetDoc (xsltDoc);
799 if (xslt == NULL)
800 {
801 if (errorStr != NULL)
802 throw vboxxml::LogicError (errorStr);
803 /* errorStr is freed in catch(...) below */
804
805 throw vboxxml::LogicError (RT_SRC_POS);
806 }
807
808 /* repeat transformations until autoConverter is satisfied */
809 do
810 {
811 xmlDocPtr newDoc = xsltApplyStylesheet (xslt, doc, NULL);
812 if (newDoc == NULL && errorStr == NULL)
813 throw vboxxml::LogicError (RT_SRC_POS);
814
815 if (errorStr != NULL)
816 {
817 xmlFreeDoc (newDoc);
818 throw Error (errorStr);
819 /* errorStr is freed in catch(...) below */
820 }
821
822 /* replace the old document on success */
823 xmlFreeDoc (doc);
824 doc = newDoc;
825 }
826 while (m->autoConverter->
827 needsConversion (Key (new XmlKeyBackend (xmlDocGetRootElement (doc))),
828 NULL));
829
830 RTStrFree (errorStr);
831
832 /* NOTE: xsltFreeStylesheet() also fress the document
833 * passed to xsltParseStylesheetDoc(). */
834 xsltFreeStylesheet (xslt);
835
836 /* restore the previous generic error func */
837 xsltGenericError = oldXsltGenericError;
838 xsltGenericErrorContext = oldXsltGenericErrorContext;
839 }
840 catch (...)
841 {
842 RTStrFree (errorStr);
843
844 /* NOTE: xsltFreeStylesheet() also fress the document
845 * passed to xsltParseStylesheetDoc(). */
846 if (xslt != NULL)
847 xsltFreeStylesheet (xslt);
848 else if (xsltDoc != NULL)
849 xmlFreeDoc (xsltDoc);
850
851 /* restore the previous generic error func */
852 xsltGenericError = oldXsltGenericError;
853 xsltGenericErrorContext = oldXsltGenericErrorContext;
854
855 RTStrFree (oldVersion);
856
857 throw;
858 }
859 }
860
861 /* validate the document */
862 if (aSchema != NULL)
863 {
864 xmlSchemaParserCtxtPtr schemaCtxt = NULL;
865 xmlSchemaPtr schema = NULL;
866 xmlSchemaValidCtxtPtr validCtxt = NULL;
867 char *errorStr = NULL;
868
869 try
870 {
871 bool valid = false;
872
873 schemaCtxt = xmlSchemaNewParserCtxt (aSchema);
874 if (schemaCtxt == NULL)
875 throw vboxxml::LogicError (RT_SRC_POS);
876
877 /* set our error handlers */
878 xmlSchemaSetParserErrors (schemaCtxt, ValidityErrorCallback,
879 ValidityWarningCallback, &errorStr);
880 xmlSchemaSetParserStructuredErrors (schemaCtxt,
881 StructuredErrorCallback,
882 &errorStr);
883 /* load schema */
884 schema = xmlSchemaParse (schemaCtxt);
885 if (schema != NULL)
886 {
887 validCtxt = xmlSchemaNewValidCtxt (schema);
888 if (validCtxt == NULL)
889 throw vboxxml::LogicError (RT_SRC_POS);
890
891 /* instruct to create default attribute's values in the document */
892 if (aFlags & Read_AddDefaults)
893 xmlSchemaSetValidOptions (validCtxt, XML_SCHEMA_VAL_VC_I_CREATE);
894
895 /* set our error handlers */
896 xmlSchemaSetValidErrors (validCtxt, ValidityErrorCallback,
897 ValidityWarningCallback, &errorStr);
898
899 /* finally, validate */
900 valid = xmlSchemaValidateDoc (validCtxt, doc) == 0;
901 }
902
903 if (!valid)
904 {
905 /* look if there was a forwared exception from the lower level */
906 if (m->trappedErr.get() != NULL)
907 m->trappedErr->rethrow();
908
909 if (errorStr == NULL)
910 throw vboxxml::LogicError (RT_SRC_POS);
911
912 throw Error (errorStr);
913 /* errorStr is freed in catch(...) below */
914 }
915
916 RTStrFree (errorStr);
917
918 xmlSchemaFreeValidCtxt (validCtxt);
919 xmlSchemaFree (schema);
920 xmlSchemaFreeParserCtxt (schemaCtxt);
921 }
922 catch (...)
923 {
924 RTStrFree (errorStr);
925
926 if (validCtxt)
927 xmlSchemaFreeValidCtxt (validCtxt);
928 if (schema)
929 xmlSchemaFree (schema);
930 if (schemaCtxt)
931 xmlSchemaFreeParserCtxt (schemaCtxt);
932
933 RTStrFree (oldVersion);
934
935 throw;
936 }
937 }
938
939 /* reset the previous tree on success */
940 reset();
941
942 m->doc = doc;
943 /* assign the root key */
944 m->root = Key (new XmlKeyBackend (xmlDocGetRootElement (m->doc)));
945
946 /* memorize the old version string also used as a flag that
947 * the conversion has been performed (transfers ownership) */
948 m->oldVersion = oldVersion;
949
950 /* restore the previous entity resolver */
951 xmlSetExternalEntityLoader (oldEntityLoader);
952 sThat = NULL;
953 }
954 catch (...)
955 {
956 if (doc != NULL)
957 xmlFreeDoc (doc);
958
959 /* restore the previous entity resolver */
960 xmlSetExternalEntityLoader (oldEntityLoader);
961 sThat = NULL;
962
963 throw;
964 }
965}
966
967void XmlTreeBackend::rawWrite (vboxxml::Output &aOutput)
968{
969 /* reset error variables used to memorize exceptions while inside the
970 * libxml2 code */
971 m->trappedErr.reset();
972
973 /* set up an input stream for parsing the document. This will be deleted
974 * when the stream is closed by the libxml2 API (e.g. when calling
975 * xmlFreeParserCtxt()). */
976 Data::OutputCtxt *outputCtxt =
977 new Data::OutputCtxt (&aOutput, m->trappedErr);
978
979 /* serialize to the stream */
980
981 xmlIndentTreeOutput = 1;
982 xmlTreeIndentString = " ";
983 xmlSaveNoEmptyTags = 0;
984
985 xmlSaveCtxtPtr saveCtxt = xmlSaveToIO (WriteCallback, CloseCallback,
986 outputCtxt, NULL,
987 XML_SAVE_FORMAT);
988 if (saveCtxt == NULL)
989 throw vboxxml::LogicError (RT_SRC_POS);
990
991 long rc = xmlSaveDoc (saveCtxt, m->doc);
992 if (rc == -1)
993 {
994 /* look if there was a forwared exception from the lower level */
995 if (m->trappedErr.get() != NULL)
996 m->trappedErr->rethrow();
997
998 /* there must be an exception from the Output implementation,
999 * otherwise the save operation must always succeed. */
1000 throw vboxxml::LogicError (RT_SRC_POS);
1001 }
1002
1003 xmlSaveClose (saveCtxt);
1004}
1005
1006void XmlTreeBackend::reset()
1007{
1008 RTStrFree (m->oldVersion);
1009 m->oldVersion = NULL;
1010
1011 if (m->doc)
1012 {
1013 /* reset the root key's node */
1014 GetKeyBackend (m->root)->mNode = NULL;
1015 /* free the document*/
1016 xmlFreeDoc (m->doc);
1017 m->doc = NULL;
1018 }
1019}
1020
1021Key &XmlTreeBackend::rootKey() const
1022{
1023 return m->root;
1024}
1025
1026/* static */
1027int XmlTreeBackend::ReadCallback (void *aCtxt, char *aBuf, int aLen)
1028{
1029 AssertReturn (aCtxt != NULL, 0);
1030
1031 Data::InputCtxt *ctxt = static_cast <Data::InputCtxt *> (aCtxt);
1032
1033 /* To prevent throwing exceptions while inside libxml2 code, we catch
1034 * them and forward to our level using a couple of variables. */
1035 try
1036 {
1037 return ctxt->input->read (aBuf, aLen);
1038 }
1039 catch (const vboxxml::EIPRTFailure &err) { ctxt->setErr (err); }
1040 catch (const vboxxml::Error &err) { ctxt->setErr (err); }
1041 catch (const std::exception &err) { ctxt->setErr (err); }
1042 catch (...) { ctxt->setErr (vboxxml::LogicError (RT_SRC_POS)); }
1043
1044 return -1 /* failure */;
1045}
1046
1047/* static */
1048int XmlTreeBackend::WriteCallback (void *aCtxt, const char *aBuf, int aLen)
1049{
1050 AssertReturn (aCtxt != NULL, 0);
1051
1052 Data::OutputCtxt *ctxt = static_cast <Data::OutputCtxt *> (aCtxt);
1053
1054 /* To prevent throwing exceptions while inside libxml2 code, we catch
1055 * them and forward to our level using a couple of variables. */
1056 try
1057 {
1058 return ctxt->output->write (aBuf, aLen);
1059 }
1060 catch (const vboxxml::EIPRTFailure &err) { ctxt->setErr (err); }
1061 catch (const vboxxml::Error &err) { ctxt->setErr (err); }
1062 catch (const std::exception &err) { ctxt->setErr (err); }
1063 catch (...) { ctxt->setErr (vboxxml::LogicError (RT_SRC_POS)); }
1064
1065 return -1 /* failure */;
1066}
1067
1068/* static */
1069int XmlTreeBackend::CloseCallback (void *aCtxt)
1070{
1071 AssertReturn (aCtxt != NULL, 0);
1072
1073 Data::IOCtxt *ctxt = static_cast <Data::IOCtxt *> (aCtxt);
1074
1075 /* To prevent throwing exceptions while inside libxml2 code, we catch
1076 * them and forward to our level using a couple of variables. */
1077 try
1078 {
1079 /// @todo there is no explicit close semantics in Stream yet
1080#if 0
1081 ctxt->stream->close();
1082#endif
1083
1084 /* perform cleanup when necessary */
1085 if (ctxt->deleteStreamOnClose)
1086 delete ctxt->stream;
1087
1088 delete ctxt;
1089
1090 return 0 /* success */;
1091 }
1092 catch (const vboxxml::EIPRTFailure &err) { ctxt->setErr (err); }
1093 catch (const vboxxml::Error &err) { ctxt->setErr (err); }
1094 catch (const std::exception &err) { ctxt->setErr (err); }
1095 catch (...) { ctxt->setErr (vboxxml::LogicError (RT_SRC_POS)); }
1096
1097 return -1 /* failure */;
1098}
1099
1100/* static */
1101void XmlTreeBackend::ValidityErrorCallback (void *aCtxt, const char *aMsg, ...)
1102{
1103 AssertReturnVoid (aCtxt != NULL);
1104 AssertReturnVoid (aMsg != NULL);
1105
1106 char * &str = *(char * *) aCtxt;
1107
1108 char *newMsg = NULL;
1109 {
1110 va_list args;
1111 va_start (args, aMsg);
1112 RTStrAPrintfV (&newMsg, aMsg, args);
1113 va_end (args);
1114 }
1115
1116 AssertReturnVoid (newMsg != NULL);
1117
1118 /* strip spaces, trailing EOLs and dot-like char */
1119 size_t newMsgLen = strlen (newMsg);
1120 while (newMsgLen && strchr (" \n.?!", newMsg [newMsgLen - 1]))
1121 -- newMsgLen;
1122
1123 /* anything left? */
1124 if (newMsgLen > 0)
1125 {
1126 if (str == NULL)
1127 {
1128 str = newMsg;
1129 newMsg [newMsgLen] = '\0';
1130 }
1131 else
1132 {
1133 /* append to the existing string */
1134 size_t strLen = strlen (str);
1135 char *newStr = (char *) RTMemRealloc (str, strLen + 2 + newMsgLen + 1);
1136 AssertReturnVoid (newStr != NULL);
1137
1138 memcpy (newStr + strLen, ".\n", 2);
1139 memcpy (newStr + strLen + 2, newMsg, newMsgLen);
1140 newStr [strLen + 2 + newMsgLen] = '\0';
1141 str = newStr;
1142 RTStrFree (newMsg);
1143 }
1144 }
1145}
1146
1147/* static */
1148void XmlTreeBackend::ValidityWarningCallback (void *aCtxt, const char *aMsg, ...)
1149{
1150 NOREF (aCtxt);
1151 NOREF (aMsg);
1152}
1153
1154/* static */
1155void XmlTreeBackend::StructuredErrorCallback (void *aCtxt, xmlErrorPtr aErr)
1156{
1157 AssertReturnVoid (aCtxt != NULL);
1158 AssertReturnVoid (aErr != NULL);
1159
1160 char * &str = *(char * *) aCtxt;
1161
1162 char *newMsg = XmlError::Format (aErr);
1163 AssertReturnVoid (newMsg != NULL);
1164
1165 if (str == NULL)
1166 str = newMsg;
1167 else
1168 {
1169 /* append to the existing string */
1170 size_t newMsgLen = strlen (newMsg);
1171 size_t strLen = strlen (str);
1172 char *newStr = (char *) RTMemRealloc (str, strLen + newMsgLen + 2);
1173 AssertReturnVoid (newStr != NULL);
1174
1175 memcpy (newStr + strLen, ".\n", 2);
1176 memcpy (newStr + strLen + 2, newMsg, newMsgLen);
1177 str = newStr;
1178 RTStrFree (newMsg);
1179 }
1180}
1181
1182/* static */
1183XmlTreeBackend *XmlTreeBackend::sThat = NULL;
1184
1185/* static */
1186xmlParserInputPtr XmlTreeBackend::ExternalEntityLoader (const char *aURI,
1187 const char *aID,
1188 xmlParserCtxtPtr aCtxt)
1189{
1190 AssertReturn (sThat != NULL, NULL);
1191
1192 if (sThat->m->inputResolver == NULL)
1193 return gGlobal.xml.defaultEntityLoader (aURI, aID, aCtxt);
1194
1195 /* To prevent throwing exceptions while inside libxml2 code, we catch
1196 * them and forward to our level using a couple of variables. */
1197 try
1198 {
1199 vboxxml::Input *input = sThat->m->inputResolver->resolveEntity (aURI, aID);
1200 if (input == NULL)
1201 return NULL;
1202
1203 Data::InputCtxt *ctxt = new Data::InputCtxt (input, sThat->m->trappedErr);
1204 ctxt->deleteStreamOnClose = true;
1205
1206 /* create an input buffer with custom hooks */
1207 xmlParserInputBufferPtr bufPtr =
1208 xmlParserInputBufferCreateIO (ReadCallback, CloseCallback,
1209 ctxt, XML_CHAR_ENCODING_NONE);
1210 if (bufPtr)
1211 {
1212 /* create an input stream */
1213 xmlParserInputPtr inputPtr =
1214 xmlNewIOInputStream (aCtxt, bufPtr, XML_CHAR_ENCODING_NONE);
1215
1216 if (inputPtr != NULL)
1217 {
1218 /* pass over the URI to the stream struct (it's NULL by
1219 * default) */
1220 inputPtr->filename =
1221 (char *) xmlCanonicPath ((const xmlChar *) input->uri());
1222 return inputPtr;
1223 }
1224 }
1225
1226 /* either of libxml calls failed */
1227
1228 if (bufPtr)
1229 xmlFreeParserInputBuffer (bufPtr);
1230
1231 delete input;
1232 delete ctxt;
1233
1234 throw vboxxml::ENoMemory();
1235 }
1236 catch (const vboxxml::EIPRTFailure &err) { sThat->m->trappedErr.reset (stdx::new_exception_trap (err)); }
1237 catch (const vboxxml::Error &err) { sThat->m->trappedErr.reset (stdx::new_exception_trap (err)); }
1238 catch (const std::exception &err) { sThat->m->trappedErr.reset (stdx::new_exception_trap (err)); }
1239 catch (...) { sThat->m->trappedErr.reset (stdx::new_exception_trap (vboxxml::LogicError (RT_SRC_POS))); }
1240
1241 return NULL;
1242}
1243
1244} /* namespace settings */
1245
Note: See TracBrowser for help on using the repository browser.

© 2024 Oracle Support Privacy / Do Not Sell My Info Terms of Use Trademark Policy Automated Access Etiquette