1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141
|
[/
Documentation for libzeep version 5.0
Copyright Maarten L. Hekkelman, 2020
Distributed under the Boost Software License, Version 1.0.
(See accompanying file LICENSE_1_0.txt or copy at
http://www.boost.org/LICENSE_1_0.txt)
]
[chapter XML
[quickbook 1.7]
[id libzeep.xml]
]
[section:descr Introduction]
The core of this library is a validating XML parser with DTD processing and all. On top of this are implemented an API for manipulating XML data in a DOM like fashion and a serialization API. As a bonus there's also an XPath implementation, albeit this is limited to XPath 1.0.
[endsect:descr]
[section:api The DOM API]
libzeep uses a modern C++ way of accessing and manipulating data. To give an idea have a look at the following code.
[import ../examples/synopsis-xml.cpp]
[synopsis_xml_main]
[heading XML nodes]
The class __node__ is the base class for all classes in the DOM API. The class is not copy constructable and subclasses use move semantics to offer a simple API while still being memory and performance efficient. Nodes can have siblings and a parent but no children.
The class __element__ is the main class, it implements a full XML node with child nodes and attributes. The children are stored as a linked list and same goes for the attributes.
The class __text__ contains the text between XML elements. A __cdata__ class is derived from __text__ and other possible child nodes for an XML element are __processing_instruction__ and __comment__.
XML elements also contain attributes, stored in the __attribute__ class. Namespace information is stored in these attributes as well. Attributes support structured binding, so the following works:
zeep::xml::attribute a("x", "1");
auto& [name, value] = a; // name == "x", value == "1"
[endsect:api]
[section:xml_io Input and output]
The class __document__ derives from __element__ can load from and write to files.
[heading streaming I/O]
You can use std::iostream to read and write __document__ objects. Reading is as simple as:
zeep::xml::document doc;
std::cin >> doc;
Writing is just as simple. A warning though, round trip fidelity is not guaranteed. There are a few issues with that. First of all, the default is to replace CDATA sections in a file with their content. If this is not the desired behaviour you can call [memberref zeep::xml::document::set_preserve_cdata `set_preserve_cdata(true)`].
Another issue is that text nodes containing only white space are present in documents read from disk while these are absent by default in documents created on the fly. When writing out XML using `iostream` you can specify to wrap and indent a document. But if the document was read in, the result will have extraneous spacing.
Specifying indentation is BTW done like this:
std::cout << std::setw(2) << doc;
That will indent with two spaces for each level.
[heading validation]
This will not validate the XML using the DTD by default. If you do want to validate and process the DTD, you have to specify where to find this DTD and other external entities. You can either use [memberref zeep::xml::document::set_base_dir `set_base_dir`] or you can specify an entity_loader using [memberref zeep::xml::document::set_entity_loader `set_entity_loader`]
As an example, take the following DTD file[teletype]
<!ELEMENT foo (bar)>
<!ELEMENT bar (#PCDATA)>
<!ENTITY hello "Hello, world!">
And an XML document containing[teletype]
<?xml version="1.0" encoding="UTF-8" ?>
<!DOCTYPE foo SYSTEM "sample.dtd">
<foo>
<bar>&hello;</bar>
</foo>
When we want to see the `&hello;` entity replaced with `'Hello, world!'` as specified in the DTD, we need to provide a way to load this DTD. To do this, look at the following code. Of course, in this example a simple call to [memberref zeep::xml::document::set_base_dir `set_base_dir`] would have been sufficient.
[import ../examples/validating-xml-sample.cpp]
[xml_validation_sample]
[endsect:xml_io]
[section:ser Serialization]
An alternative way to read/write XML files is using serialization. To do this, we first construct a structure called Person. We add a templated function to this struct just like in `boost::serialize` and then we can read the file.
[import ../examples/serialize-xml.cpp]
[synopsis_xml_serialize]
[heading attributes]
Suppose you want to serialize a value into a XML attribute, you would have to replace `zeep::make_nvp` with `zeep::make_attribute_nvp`.
[heading custom types]
What happens during serialization is deconstruction of structured data types into parts that can be converted into text strings. For this final conversion there are __value_serializer__ helper classes. __value_serializer__ is a template and specializations for the default types are given in `<zeep/value_serializer.hpp>`. You can create your own specializations for this class for custom data types, look at the one for `boost::posix_time::ptime` for inspiration.
[heading enums]
For conversion of enum's you can use the __value_serializer__ specialization for enums: [c++]
enum class MyEnum { FOO, BAR };
__value_serializer__<MyEnum>::instance()
("foo", MyEnum::FOO)
("bar", MyEnum::BAR);
There's also a new interface, somewhat more intuitive from a modern C++ viewpoint:
[import ../examples/synopsis-json.cpp]
[enum_support]
[endsect:ser]
[section:xpath XPath 1.0]
Libzeep comes with a [@http://www.w3.org/TR/xpath/ XPath 1.0] implementation. You can use this to locate elements in a DOM tree easily. For a complete description of the XPath specification you should read the documentation at e.g. [@http://www.w3.org/TR/xpath/] or [@https://www.w3schools.com/xml/xpath_intro.asp].
The way it works in libzeep is that you can call `find()` on an __element__ object and it will return a `zeep::xml::element_set` object which is actually a `std::list` of __element__ pointers of the elements that conform to the specification in XPath passed as parameter to `find()`. An alternative method `find_first()` can be used to return only the first element.
An example where we look for the first person in our test file with the lastname Jones:
__element__* jones = doc.child()->find_first("//person[lastname='Jones']");
[heading variables]
XPath constructs can reference variables. As an example, suppose you need to find nodes in a special XML Namespace but you do not want to find out what the prefix of this Namespace is, you could do something like this:
[import ../examples/xpath-sample.cpp]
[xpath_example]
[note Please note that the evaluation of an XPath returns pointers to XML nodes. Of course these are only valid as long as you do not modify the the document in which they are contained.]
[endsect]
|