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->attributes_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_
 |