blob: cb6a3c07d268e442e1c136d575e439e686008a33 [file] [log] [blame]
Adam Lesinski6f6ceb72014-11-14 14:48:12 -08001/*
2 * Copyright (C) 2015 The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16
17#include <iostream>
18#include <string>
19
20#include "SourceXmlPullParser.h"
21#include "Util.h"
22
23namespace aapt {
24
25constexpr char kXmlNamespaceSep = 1;
26
27SourceXmlPullParser::SourceXmlPullParser(std::istream& in) : mIn(in), mEmpty(), mDepth(0) {
28 mParser = XML_ParserCreateNS(nullptr, kXmlNamespaceSep);
29 XML_SetUserData(mParser, this);
30 XML_SetElementHandler(mParser, startElementHandler, endElementHandler);
31 XML_SetNamespaceDeclHandler(mParser, startNamespaceHandler, endNamespaceHandler);
32 XML_SetCharacterDataHandler(mParser, characterDataHandler);
33 XML_SetCommentHandler(mParser, commentDataHandler);
34 mEventQueue.push(EventData{ Event::kStartDocument, 0, mDepth++ });
35}
36
37SourceXmlPullParser::~SourceXmlPullParser() {
38 XML_ParserFree(mParser);
39}
40
41SourceXmlPullParser::Event SourceXmlPullParser::next() {
42 const Event currentEvent = getEvent();
43 if (currentEvent == Event::kBadDocument || currentEvent == Event::kEndDocument) {
44 return currentEvent;
45 }
46
47 mEventQueue.pop();
48 while (mEventQueue.empty()) {
49 mIn.read(mBuffer, sizeof(mBuffer) / sizeof(*mBuffer));
50
51 const bool done = mIn.eof();
52 if (mIn.bad() && !done) {
53 mLastError = strerror(errno);
54 mEventQueue.push(EventData{ Event::kBadDocument });
55 continue;
56 }
57
58 if (XML_Parse(mParser, mBuffer, mIn.gcount(), done) == XML_STATUS_ERROR) {
59 mLastError = XML_ErrorString(XML_GetErrorCode(mParser));
60 mEventQueue.push(EventData{ Event::kBadDocument });
61 continue;
62 }
63
64 if (done) {
65 mEventQueue.push(EventData{ Event::kEndDocument, 0, 0 });
66 }
67 }
68
69 return getEvent();
70}
71
72SourceXmlPullParser::Event SourceXmlPullParser::getEvent() const {
73 return mEventQueue.front().event;
74}
75
76const std::string& SourceXmlPullParser::getLastError() const {
77 return mLastError;
78}
79
80const std::u16string& SourceXmlPullParser::getComment() const {
81 return mEventQueue.front().comment;
82}
83
84size_t SourceXmlPullParser::getLineNumber() const {
85 return mEventQueue.front().lineNumber;
86}
87
88size_t SourceXmlPullParser::getDepth() const {
89 return mEventQueue.front().depth;
90}
91
92const std::u16string& SourceXmlPullParser::getText() const {
93 if (getEvent() != Event::kText) {
94 return mEmpty;
95 }
96 return mEventQueue.front().data1;
97}
98
99const std::u16string& SourceXmlPullParser::getNamespacePrefix() const {
100 const Event currentEvent = getEvent();
101 if (currentEvent != Event::kStartNamespace && currentEvent != Event::kEndNamespace) {
102 return mEmpty;
103 }
104 return mEventQueue.front().data1;
105}
106
107const std::u16string& SourceXmlPullParser::getNamespaceUri() const {
108 const Event currentEvent = getEvent();
109 if (currentEvent != Event::kStartNamespace && currentEvent != Event::kEndNamespace) {
110 return mEmpty;
111 }
112 return mEventQueue.front().data2;
113}
114
115const std::u16string& SourceXmlPullParser::getElementNamespace() const {
116 const Event currentEvent = getEvent();
117 if (currentEvent != Event::kStartElement && currentEvent != Event::kEndElement) {
118 return mEmpty;
119 }
120 return mEventQueue.front().data1;
121}
122
123const std::u16string& SourceXmlPullParser::getElementName() const {
124 const Event currentEvent = getEvent();
125 if (currentEvent != Event::kStartElement && currentEvent != Event::kEndElement) {
126 return mEmpty;
127 }
128 return mEventQueue.front().data2;
129}
130
131XmlPullParser::const_iterator SourceXmlPullParser::beginAttributes() const {
132 return mEventQueue.front().attributes.begin();
133}
134
135XmlPullParser::const_iterator SourceXmlPullParser::endAttributes() const {
136 return mEventQueue.front().attributes.end();
137}
138
139size_t SourceXmlPullParser::getAttributeCount() const {
140 if (getEvent() != Event::kStartElement) {
141 return 0;
142 }
143 return mEventQueue.front().attributes.size();
144}
145
146/**
147 * Extracts the namespace and name of an expanded element or attribute name.
148 */
149static void splitName(const char* name, std::u16string& outNs, std::u16string& outName) {
150 const char* p = name;
151 while (*p != 0 && *p != kXmlNamespaceSep) {
152 p++;
153 }
154
155 if (*p == 0) {
156 outNs = std::u16string();
157 outName = util::utf8ToUtf16(name);
158 } else {
159 outNs = util::utf8ToUtf16(StringPiece(name, (p - name)));
160 outName = util::utf8ToUtf16(p + 1);
161 }
162}
163
164void XMLCALL SourceXmlPullParser::startNamespaceHandler(void* userData, const char* prefix,
165 const char* uri) {
166 SourceXmlPullParser* parser = reinterpret_cast<SourceXmlPullParser*>(userData);
167 std::u16string namespaceUri = uri != nullptr ? util::utf8ToUtf16(uri) : std::u16string();
168 parser->mNamespaceUris.push(namespaceUri);
169 parser->mEventQueue.push(EventData{
170 Event::kStartNamespace,
171 XML_GetCurrentLineNumber(parser->mParser),
172 parser->mDepth++,
173 prefix != nullptr ? util::utf8ToUtf16(prefix) : std::u16string(),
174 namespaceUri
175 });
176}
177
178void XMLCALL SourceXmlPullParser::startElementHandler(void* userData, const char* name,
179 const char** attrs) {
180 SourceXmlPullParser* parser = reinterpret_cast<SourceXmlPullParser*>(userData);
181
182 EventData data = {
183 Event::kStartElement, XML_GetCurrentLineNumber(parser->mParser), parser->mDepth++
184 };
185 splitName(name, data.data1, data.data2);
186
187 while (*attrs) {
188 Attribute attribute;
189 splitName(*attrs++, attribute.namespaceUri, attribute.name);
190 attribute.value = util::utf8ToUtf16(*attrs++);
191
192 // Insert in sorted order.
193 auto iter = std::lower_bound(data.attributes.begin(), data.attributes.end(), attribute);
194 data.attributes.insert(iter, std::move(attribute));
195 }
196
197 // Move the structure into the queue (no copy).
198 parser->mEventQueue.push(std::move(data));
199}
200
201void XMLCALL SourceXmlPullParser::characterDataHandler(void* userData, const char* s, int len) {
202 SourceXmlPullParser* parser = reinterpret_cast<SourceXmlPullParser*>(userData);
203
204 parser->mEventQueue.push(EventData{
205 Event::kText,
206 XML_GetCurrentLineNumber(parser->mParser),
207 parser->mDepth,
208 util::utf8ToUtf16(StringPiece(s, len))
209 });
210}
211
212void XMLCALL SourceXmlPullParser::endElementHandler(void* userData, const char* name) {
213 SourceXmlPullParser* parser = reinterpret_cast<SourceXmlPullParser*>(userData);
214
215 EventData data = {
216 Event::kEndElement, XML_GetCurrentLineNumber(parser->mParser), --(parser->mDepth)
217 };
218 splitName(name, data.data1, data.data2);
219
220 // Move the data into the queue (no copy).
221 parser->mEventQueue.push(std::move(data));
222}
223
224void XMLCALL SourceXmlPullParser::endNamespaceHandler(void* userData, const char* prefix) {
225 SourceXmlPullParser* parser = reinterpret_cast<SourceXmlPullParser*>(userData);
226
227 parser->mEventQueue.push(EventData{
228 Event::kEndNamespace,
229 XML_GetCurrentLineNumber(parser->mParser),
230 --(parser->mDepth),
231 prefix != nullptr ? util::utf8ToUtf16(prefix) : std::u16string(),
232 parser->mNamespaceUris.top()
233 });
234 parser->mNamespaceUris.pop();
235}
236
237void XMLCALL SourceXmlPullParser::commentDataHandler(void* userData, const char* comment) {
238 SourceXmlPullParser* parser = reinterpret_cast<SourceXmlPullParser*>(userData);
239
240 parser->mEventQueue.push(EventData{
241 Event::kComment,
242 XML_GetCurrentLineNumber(parser->mParser),
243 parser->mDepth,
244 util::utf8ToUtf16(comment)
245 });
246}
247
248} // namespace aapt