blob: e54099ebfa67caa12abc843ddd599e43490a25fa (
plain)
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
|
/**
* @file
* @author 2012-2013 Stefan Radomski (stefan.radomski@cs.tu-darmstadt.de)
* @copyright Simplified BSD
*
* @cond
* This program is free software: you can redistribute it and/or modify
* it under the terms of the FreeBSD license as published by the FreeBSD
* project.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
*
* You should have received a copy of the FreeBSD license along with this
* program. If not, see <http://www.opensource.org/licenses/bsd-license>.
* @endcond
*/
#include <uscxml/Common.h>
#include "uscxml/DOMUtils.h"
#include <uscxml/Convenience.h>
#include <glog/logging.h>
#include <SAX/helpers/InputSourceResolver.hpp>
namespace uscxml {
std::string DOMUtils::xPathForNode(const Arabica::DOM::Node<std::string>& node) {
std::string xPath;
if (!node || node.getNodeType() != Arabica::DOM::Node_base::ELEMENT_NODE)
return xPath;
Arabica::DOM::Node<std::string> curr = node;
while(curr) {
switch (curr.getNodeType()) {
case Arabica::DOM::Node_base::ELEMENT_NODE: {
if (HAS_ATTR(curr, "id")) {
// we assume ids to be unique and return immediately
xPath.insert(0, "//" + TAGNAME(curr) + "[@id=\"" + ATTR(curr, "id") + "\"]");
return xPath;
} else {
// check previous siblings to count our index
Arabica::DOM::Node<std::string> sibling = curr.getPreviousSibling();
int index = 1;
while(sibling) {
if (sibling.getNodeType() == Arabica::DOM::Node_base::ELEMENT_NODE) {
if (iequals(TAGNAME(sibling), TAGNAME(curr))) {
index++;
}
}
sibling = sibling.getPreviousSibling();
}
xPath.insert(0, "/" + TAGNAME(curr) + "[" + toStr(index) + "]");
}
break;
}
case Arabica::DOM::Node_base::DOCUMENT_NODE:
return xPath;
default:
LOG(ERROR) << "Only nodes of type element supported for now";
return "";
break;
}
curr = curr.getParentNode();
}
return xPath;
}
NameSpacingParser NameSpacingParser::fromXML(const std::string& xml) {
std::stringstream* ss = new std::stringstream();
(*ss) << xml;
// we need an auto_ptr for arabica to assume ownership
std::auto_ptr<std::istream> ssPtr(ss);
Arabica::SAX::InputSource<std::string> inputSource;
inputSource.setByteStream(ssPtr);
return fromInputSource(inputSource);
}
NameSpacingParser NameSpacingParser::fromInputSource(Arabica::SAX::InputSource<std::string>& source) {
NameSpacingParser parser;
if(!parser.parse(source) || !parser.getDocument().hasChildNodes()) {
if(parser._errorHandler.errorsReported()) {
LOG(ERROR) << "could not parse input:";
LOG(ERROR) << parser._errorHandler.errors() << std::endl;
} else {
Arabica::SAX::InputSourceResolver resolver(source, Arabica::default_string_adaptor<std::string>());
if (!resolver.resolve()) {
LOG(ERROR) << source.getSystemId() << ": no such file";
}
}
}
return parser;
}
NameSpacingParser::NameSpacingParser() {
setErrorHandler(_errorHandler);
}
void NameSpacingParser::startPrefixMapping(const std::string& prefix, const std::string& uri) {
nameSpace.insert(std::make_pair(uri, prefix));
}
}
|