340 lines
9.7 KiB
C++
340 lines
9.7 KiB
C++
/**
|
|
* Copyright (c) 2021 OceanBase
|
|
* OceanBase CE is licensed under Mulan PubL v2.
|
|
* You can use this software according to the terms and conditions of the Mulan PubL v2.
|
|
* You may obtain a copy of Mulan PubL v2 at:
|
|
* http://license.coscl.org.cn/MulanPubL-2.0
|
|
* THIS SOFTWARE IS PROVIDED ON AN "AS IS" BASIS, WITHOUT WARRANTIES OF ANY KIND,
|
|
* EITHER EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO NON-INFRINGEMENT,
|
|
* MERCHANTABILITY OR FIT FOR A PARTICULAR PURPOSE.
|
|
* See the Mulan PubL v2 for more details.
|
|
*/
|
|
#define USING_LOG_PREFIX SHARE
|
|
#ifndef OCEANBASE_UNIT_TEST_XML_UTILS_H_
|
|
#define OCEANBASE_UNIT_TEST_XML_UTILS_H_
|
|
|
|
#include "lib/xml/ob_xml_parser.h"
|
|
#include "lib/xml/ob_xml_util.h"
|
|
#include "lib/oblog/ob_log.h"
|
|
namespace oceanbase {
|
|
namespace common {
|
|
|
|
namespace ObXmlTestUtils {
|
|
|
|
class ObXmlNodeVisitor {
|
|
public:
|
|
virtual int visit(ObXmlNode* node);
|
|
virtual int visit_xml_document(ObXmlDocument* node) = 0;
|
|
virtual int visit_xml_content(ObXmlDocument* node) = 0;
|
|
virtual int visit_xml_element(ObXmlElement* node) = 0;
|
|
virtual int visit_xml_attribute(ObXmlAttribute* node) = 0;
|
|
virtual int visit_xml_text(ObXmlText* node) = 0;
|
|
virtual int visit_xml_comment(ObXmlText* node) = 0;
|
|
virtual int visit_xml_cdata(ObXmlText* node) = 0;
|
|
virtual int visit_xml_processing_instruction(ObXmlAttribute* node) = 0;
|
|
};
|
|
|
|
int ObXmlNodeVisitor::visit(ObXmlNode* node) {
|
|
INIT_SUCC(ret);
|
|
if (OB_ISNULL(node)) {
|
|
ret = OB_ERR_UNEXPECTED;
|
|
LOG_WARN("node is null", K(ret));
|
|
} else {
|
|
switch (node->type())
|
|
{
|
|
case ObMulModeNodeType::M_DOCUMENT:
|
|
if (OB_FAIL(visit_xml_document(ObXmlUtil::xml_node_cast<ObXmlDocument>(node, node->type())))) {
|
|
LOG_WARN("visit document failed", K(ret), K(node->type()));
|
|
}
|
|
break;
|
|
case ObMulModeNodeType::M_CONTENT:
|
|
if (OB_FAIL(visit_xml_content(ObXmlUtil::xml_node_cast<ObXmlDocument>(node, node->type())))) {
|
|
LOG_WARN("visit document failed", K(ret), K(node->type()));
|
|
}
|
|
break;
|
|
case ObMulModeNodeType::M_ELEMENT:
|
|
if (OB_FAIL(visit_xml_element(ObXmlUtil::xml_node_cast<ObXmlElement>(node, node->type())))) {
|
|
LOG_WARN("visit element failed", K(ret), K(node->type()));
|
|
}
|
|
break;
|
|
case ObMulModeNodeType::M_ATTRIBUTE:
|
|
case ObMulModeNodeType::M_NAMESPACE:
|
|
if (OB_FAIL(visit_xml_attribute(ObXmlUtil::xml_node_cast<ObXmlAttribute>(node, node->type())))) {
|
|
LOG_WARN("visit attribute failed", K(ret), K(node->type()));
|
|
}
|
|
break;
|
|
case ObMulModeNodeType::M_TEXT:
|
|
if (OB_FAIL(visit_xml_text(ObXmlUtil::xml_node_cast<ObXmlText>(node, node->type())))) {
|
|
LOG_WARN("visit text failed", K(ret), K(node->type()));
|
|
}
|
|
break;
|
|
case ObMulModeNodeType::M_CDATA:
|
|
if (OB_FAIL(visit_xml_cdata(ObXmlUtil::xml_node_cast<ObXmlText>(node, node->type())))) {
|
|
LOG_WARN("visit cdata failed", K(ret), K(node->type()));
|
|
}
|
|
break;
|
|
case ObMulModeNodeType::M_COMMENT:
|
|
if (OB_FAIL(visit_xml_comment(ObXmlUtil::xml_node_cast<ObXmlText>(node, node->type())))) {
|
|
LOG_WARN("visit comment failed", K(ret), K(node->type()));
|
|
}
|
|
break;
|
|
case ObMulModeNodeType::M_INSTRUCT:
|
|
if (OB_FAIL(visit_xml_processing_instruction(ObXmlUtil::xml_node_cast<ObXmlAttribute>(node, node->type())))) {
|
|
LOG_WARN("visit processing instruction failed", K(ret), K(node->type()));
|
|
}
|
|
break;
|
|
default:
|
|
ret = OB_ERR_UNEXPECTED;
|
|
LOG_WARN("node type incorrect", K(ret), K(node->type()));
|
|
break;
|
|
}
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
class ObXmlTreeTextWriter : public ObXmlNodeVisitor{
|
|
public:
|
|
ObXmlTreeTextWriter(ObIAllocator* allocator) : buffer_(allocator) {}
|
|
virtual ~ObXmlTreeTextWriter() {}
|
|
|
|
virtual int visit_xml_document(ObXmlDocument* node);
|
|
virtual int visit_xml_content(ObXmlDocument* node);
|
|
virtual int visit_xml_element(ObXmlElement* node);
|
|
virtual int visit_xml_attribute(ObXmlAttribute* node);
|
|
virtual int visit_xml_text(ObXmlText* node);
|
|
virtual int visit_xml_comment(ObXmlText* node);
|
|
virtual int visit_xml_cdata(ObXmlText* node);
|
|
virtual int visit_xml_processing_instruction(ObXmlAttribute* node);
|
|
|
|
int append_qname(const ObString& prefix, const ObString& localname) {
|
|
INIT_SUCC(ret);
|
|
if (!prefix.empty()) {
|
|
buffer_.append(prefix);
|
|
buffer_.append(":");
|
|
}
|
|
if (!localname.empty()) {
|
|
buffer_.append(localname);
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
void reuse() {
|
|
buffer_.reuse();
|
|
}
|
|
|
|
ObString get_xml_text() {
|
|
return ObString(buffer_.length(), buffer_.ptr());
|
|
}
|
|
|
|
private:
|
|
ObStringBuffer buffer_;
|
|
};
|
|
|
|
int ObXmlTreeTextWriter::visit_xml_document(ObXmlDocument* node) {
|
|
INIT_SUCC(ret);
|
|
if(OB_ISNULL(node)) {
|
|
ret = OB_ERR_UNEXPECTED;
|
|
LOG_WARN("null node", K(ret));
|
|
} else {
|
|
if (node->has_xml_decl()) {
|
|
buffer_.append("<?xml");
|
|
if (!node->get_version().empty()) {
|
|
buffer_.append(" version=\"");
|
|
buffer_.append(node->get_version());
|
|
buffer_.append("\"");
|
|
}
|
|
if (! node->get_encoding().empty() || node->get_encoding_flag()) {
|
|
buffer_.append(" encoding=\"");
|
|
buffer_.append(node->get_encoding());
|
|
buffer_.append("\"");
|
|
}
|
|
|
|
switch(node->get_standalone()) {
|
|
case OB_XML_STANDALONE_NO:
|
|
buffer_.append(" standalone=\"no\"");
|
|
break;
|
|
case OB_XML_STANDALONE_YES:
|
|
buffer_.append(" standalone=\"yes\"");
|
|
break;
|
|
}
|
|
buffer_.append("?>\n");
|
|
}
|
|
|
|
for (int i = 0; OB_SUCC(ret) && i < node->size(); ++i) {
|
|
ObXmlNode* child = node->at(i);
|
|
if (OB_ISNULL(child)) {
|
|
ret = OB_ERR_UNEXPECTED;
|
|
LOG_WARN("get child failed", K(ret), K(i));
|
|
} else if (OB_FAIL(visit(child))) {
|
|
LOG_WARN("visit child failed", K(ret), K(i));
|
|
} else {
|
|
buffer_.append("\n");
|
|
}
|
|
}
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
int ObXmlTreeTextWriter::visit_xml_content(ObXmlDocument* node) {
|
|
INIT_SUCC(ret);
|
|
if(OB_ISNULL(node)) {
|
|
ret = OB_ERR_UNEXPECTED;
|
|
LOG_WARN("null node", K(ret));
|
|
} else {
|
|
for (int i = 0; OB_SUCC(ret) && i < node->size(); ++i) {
|
|
ObXmlNode* child = node->at(i);
|
|
if (OB_ISNULL(child)) {
|
|
ret = OB_ERR_UNEXPECTED;
|
|
LOG_WARN("get child failed", K(ret), K(i));
|
|
} else if (OB_FAIL(visit(child))) {
|
|
LOG_WARN("visit child failed", K(ret), K(i));
|
|
}
|
|
}
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
int ObXmlTreeTextWriter::visit_xml_element(ObXmlElement* node) {
|
|
INIT_SUCC(ret);
|
|
if(OB_ISNULL(node)) {
|
|
ret = OB_ERR_UNEXPECTED;
|
|
LOG_WARN("null node", K(ret));
|
|
} else {
|
|
buffer_.append("<");
|
|
append_qname(node->get_prefix(), node->get_key());
|
|
|
|
for (int i = 0; OB_SUCC(ret) && i < node->attribute_size(); ++i) {
|
|
ObXmlAttribute* attr = NULL;
|
|
if (OB_FAIL(node->get_attribute(attr, i))) {
|
|
LOG_WARN("get attr failed", K(ret), K(i));
|
|
} else if (OB_ISNULL(attr)) {
|
|
ret = OB_ERR_UNEXPECTED;
|
|
LOG_WARN("fail to get attr", K(ret));
|
|
} else {
|
|
buffer_.append(" ");
|
|
if (OB_FAIL(visit(attr))) {
|
|
LOG_WARN("visit attr failed", K(ret), K(i));
|
|
}
|
|
}
|
|
}
|
|
if (OB_SUCC(ret)) {
|
|
if (node->is_empty()) {
|
|
buffer_.append("/>");
|
|
} else {
|
|
buffer_.append(">");
|
|
for (int i = 0; OB_SUCC(ret) && i < node->size(); ++i) {
|
|
ObXmlNode* child = node->at(i);
|
|
if (OB_ISNULL(child)) {
|
|
ret = OB_ERR_UNEXPECTED;
|
|
LOG_WARN("get child failed", K(ret), K(i));
|
|
} else if (OB_FAIL(visit(child))) {
|
|
LOG_WARN("visit child failed", K(ret), K(i));
|
|
}
|
|
}
|
|
}
|
|
}
|
|
if (OB_SUCC(ret) && !node->is_empty()) {
|
|
buffer_.append("</");
|
|
append_qname(node->get_prefix(), node->get_key());
|
|
buffer_.append(">");
|
|
}
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
int ObXmlTreeTextWriter::visit_xml_attribute(ObXmlAttribute* node) {
|
|
INIT_SUCC(ret);
|
|
if(OB_ISNULL(node)) {
|
|
ret = OB_ERR_UNEXPECTED;
|
|
LOG_WARN("null node", K(ret));
|
|
} else {
|
|
ObString attr_value;
|
|
append_qname(node->get_prefix(), node->get_key());
|
|
buffer_.append("=\"");
|
|
node->get_value(attr_value);
|
|
buffer_.append(attr_value);
|
|
buffer_.append("\"");
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
int ObXmlTreeTextWriter::visit_xml_text(ObXmlText* node) {
|
|
INIT_SUCC(ret);
|
|
if(OB_ISNULL(node)) {
|
|
ret = OB_ERR_UNEXPECTED;
|
|
LOG_WARN("null node", K(ret));
|
|
} else {
|
|
ObString value;
|
|
node->get_value(value);
|
|
buffer_.append(value);
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
int ObXmlTreeTextWriter::visit_xml_cdata(ObXmlText* node) {
|
|
INIT_SUCC(ret);
|
|
if(OB_ISNULL(node)) {
|
|
ret = OB_ERR_UNEXPECTED;
|
|
LOG_WARN("null node", K(ret));
|
|
} else {
|
|
ObString value;
|
|
buffer_.append("<![CDATA[");
|
|
node->get_value(value);
|
|
buffer_.append(value);
|
|
buffer_.append("]]>");
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
int ObXmlTreeTextWriter::visit_xml_comment(ObXmlText* node) {
|
|
INIT_SUCC(ret);
|
|
if(OB_ISNULL(node)) {
|
|
ret = OB_ERR_UNEXPECTED;
|
|
LOG_WARN("null node", K(ret));
|
|
} else {
|
|
ObString value;
|
|
buffer_.append("<!--");
|
|
node->get_value(value);
|
|
buffer_.append(value);
|
|
buffer_.append("-->");
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
int ObXmlTreeTextWriter::visit_xml_processing_instruction(ObXmlAttribute* node) {
|
|
INIT_SUCC(ret);
|
|
if(OB_ISNULL(node)) {
|
|
ret = OB_ERR_UNEXPECTED;
|
|
LOG_WARN("null node", K(ret));
|
|
} else {
|
|
ObString value;
|
|
buffer_.append("<?");
|
|
buffer_.append(node->get_key());
|
|
node->get_value(value);
|
|
if (!value.empty()) {
|
|
buffer_.append(" ");
|
|
buffer_.append(value);
|
|
}
|
|
buffer_.append("?>");
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
|
|
class ObXmlWriterUtils {
|
|
public:
|
|
static ObString to_xml_text(ObIAllocator* allocator, ObXmlNode& node) {
|
|
ObXmlTreeTextWriter writer(allocator);
|
|
writer.visit(&node);
|
|
return writer.get_xml_text();
|
|
}
|
|
};
|
|
|
|
};
|
|
|
|
|
|
}//end namespace share
|
|
}//end namespace oceanbase
|
|
|
|
#endif // OCEANBASE_UNIT_TEST_XML_UTILS_H_
|