oceanbase/unittest/share/test_xml_utils.h

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_