mirror of
				https://github.com/godotengine/godot.git
				synced 2025-10-29 12:44:31 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			592 lines
		
	
	
	
		
			13 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			592 lines
		
	
	
	
		
			13 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /*************************************************************************/
 | |
| /*  xml_parser.cpp                                                       */
 | |
| /*************************************************************************/
 | |
| /*                       This file is part of:                           */
 | |
| /*                           GODOT ENGINE                                */
 | |
| /*                    http://www.godotengine.org                         */
 | |
| /*************************************************************************/
 | |
| /* Copyright (c) 2007-2016 Juan Linietsky, Ariel Manzur.                 */
 | |
| /*                                                                       */
 | |
| /* Permission is hereby granted, free of charge, to any person obtaining */
 | |
| /* a copy of this software and associated documentation files (the       */
 | |
| /* "Software"), to deal in the Software without restriction, including   */
 | |
| /* without limitation the rights to use, copy, modify, merge, publish,   */
 | |
| /* distribute, sublicense, and/or sell copies of the Software, and to    */
 | |
| /* permit persons to whom the Software is furnished to do so, subject to */
 | |
| /* the following conditions:                                             */
 | |
| /*                                                                       */
 | |
| /* The above copyright notice and this permission notice shall be        */
 | |
| /* included in all copies or substantial portions of the Software.       */
 | |
| /*                                                                       */
 | |
| /* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,       */
 | |
| /* EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF    */
 | |
| /* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.*/
 | |
| /* IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY  */
 | |
| /* CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT,  */
 | |
| /* TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE     */
 | |
| /* SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.                */
 | |
| /*************************************************************************/
 | |
| #include "xml_parser.h"
 | |
| #include "print_string.h"
 | |
| //#define DEBUG_XML
 | |
| 
 | |
| VARIANT_ENUM_CAST(XMLParser::NodeType);
 | |
| 
 | |
| static bool _equalsn(const CharType* str1, const CharType* str2, int len) {
 | |
| 	int i;
 | |
| 	for(i=0; i < len && str1[i] && str2[i] ; ++i)
 | |
| 	     if (str1[i] != str2[i])
 | |
| 		     return false;
 | |
| 
 | |
| 	// if one (or both) of the strings was smaller then they
 | |
| 	// are only equal if they have the same lenght
 | |
| 	return (i == len) || (str1[i] == 0 && str2[i] == 0);
 | |
| }
 | |
| 
 | |
| 
 | |
| String XMLParser::_replace_special_characters(const String& origstr) {
 | |
| 
 | |
| 	int pos = origstr.find("&");
 | |
| 	int oldPos = 0;
 | |
| 
 | |
| 	if (pos == -1)
 | |
| 		return origstr;
 | |
| 
 | |
| 	String newstr;
 | |
| 
 | |
| 	while(pos != -1 && pos < origstr.length()-2) {
 | |
| 		// check if it is one of the special characters
 | |
| 
 | |
| 		int specialChar = -1;
 | |
| 		for (int i=0; i<(int)special_characters.size(); ++i)
 | |
| 		{
 | |
| 			const CharType* p = &origstr[pos]+1;
 | |
| 
 | |
| 			if (_equalsn(&special_characters[i][1], p, special_characters[i].length()-1))
 | |
| 			{
 | |
| 				specialChar = i;
 | |
| 				break;
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		if (specialChar != -1)
 | |
| 		{
 | |
| 			newstr+=(origstr.substr(oldPos, pos - oldPos));
 | |
| 			newstr+=(special_characters[specialChar][0]);
 | |
| 			pos += special_characters[specialChar].length();
 | |
| 		}
 | |
| 		else
 | |
| 		{
 | |
| 			newstr+=(origstr.substr(oldPos, pos - oldPos + 1));
 | |
| 			pos += 1;
 | |
| 		}
 | |
| 
 | |
| 		// find next &
 | |
| 		oldPos = pos;
 | |
| 		pos = origstr.find("&", pos);
 | |
| 	}
 | |
| 
 | |
| 	if (oldPos < origstr.length()-1)
 | |
| 		newstr+=(origstr.substr(oldPos, origstr.length()-oldPos));
 | |
| 
 | |
| 	return newstr;
 | |
| }
 | |
| 
 | |
| 
 | |
| static inline bool _is_white_space(char c)
 | |
| {
 | |
| 	return (c==' ' || c=='\t' || c=='\n' || c=='\r');
 | |
| }
 | |
| 
 | |
| 
 | |
| //! sets the state that text was found. Returns true if set should be set
 | |
| bool XMLParser::_set_text(char* start, char* end) {
 | |
| 	// check if text is more than 2 characters, and if not, check if there is
 | |
| 	// only white space, so that this text won't be reported
 | |
| 	if (end - start < 3)
 | |
| 	{
 | |
| 		char* p = start;
 | |
| 		for(; p != end; ++p)
 | |
| 			if (!_is_white_space(*p))
 | |
| 				break;
 | |
| 
 | |
| 		if (p == end)
 | |
| 			return false;
 | |
| 	}
 | |
| 
 | |
| 	// set current text to the parsed text, and replace xml special characters
 | |
| 	String s = String::utf8(start, (int)(end - start));
 | |
| 	node_name = _replace_special_characters(s);
 | |
| 
 | |
| 	// current XML node type is text
 | |
| 	node_type = NODE_TEXT;
 | |
| 
 | |
| 	return true;
 | |
| }
 | |
| 
 | |
| void XMLParser::_parse_closing_xml_element() {
 | |
| 	node_type = NODE_ELEMENT_END;
 | |
| 	node_empty = false;
 | |
| 	attributes.clear();
 | |
| 
 | |
| 	++P;
 | |
| 	const char* pBeginClose = P;
 | |
| 
 | |
| 	while(*P != '>')
 | |
| 		++P;
 | |
| 
 | |
| 	node_name = String::utf8(pBeginClose, (int)(P - pBeginClose));
 | |
| #ifdef DEBUG_XML
 | |
| 	print_line("XML CLOSE: "+node_name);
 | |
| #endif
 | |
| 	++P;
 | |
| }
 | |
| 
 | |
| void XMLParser::_ignore_definition() {
 | |
| 	node_type = NODE_UNKNOWN;
 | |
| 
 | |
| 	char *F=P;
 | |
| 	// move until end marked with '>' reached
 | |
| 	while(*P != '>')
 | |
| 		++P;
 | |
| 	node_name.parse_utf8(F,P-F);
 | |
| 	++P;
 | |
| }
 | |
| 
 | |
| bool XMLParser::_parse_cdata() {
 | |
| 
 | |
| 	if (*(P+1) != '[')
 | |
| 		return false;
 | |
| 
 | |
| 	node_type = NODE_CDATA;
 | |
| 
 | |
| 	// skip '<![CDATA['
 | |
| 	int count=0;
 | |
| 	while( *P && count<8 )
 | |
| 	{
 | |
| 		++P;
 | |
| 		++count;
 | |
| 	}
 | |
| 
 | |
| 	if (!*P)
 | |
| 		return true;
 | |
| 
 | |
| 	char *cDataBegin = P;
 | |
| 	char *cDataEnd = 0;
 | |
| 
 | |
| 	// find end of CDATA
 | |
| 	while(*P && !cDataEnd) {
 | |
| 		if (*P == '>' &&
 | |
| 		   (*(P-1) == ']') &&
 | |
| 		   (*(P-2) == ']'))
 | |
| 		{
 | |
| 			cDataEnd = P - 2;
 | |
| 		}
 | |
| 
 | |
| 		++P;
 | |
| 	}
 | |
| 
 | |
| 	if ( cDataEnd )
 | |
| 		node_name = String::utf8(cDataBegin, (int)(cDataEnd - cDataBegin));
 | |
| 	else
 | |
| 		node_name = "";
 | |
| #ifdef DEBUG_XML
 | |
| 	print_line("XML CDATA: "+node_name);
 | |
| #endif
 | |
| 
 | |
| 	return true;
 | |
| }
 | |
| 
 | |
| void XMLParser::_parse_comment() {
 | |
| 
 | |
| 	node_type = NODE_COMMENT;
 | |
| 	P += 1;
 | |
| 
 | |
| 	char *pCommentBegin = P;
 | |
| 
 | |
| 	int count = 1;
 | |
| 
 | |
| 	// move until end of comment reached
 | |
| 	while(count)
 | |
| 	{
 | |
| 		if (*P == '>')
 | |
| 			--count;
 | |
| 		else
 | |
| 		if (*P == '<')
 | |
| 			++count;
 | |
| 
 | |
| 		++P;
 | |
| 	}
 | |
| 
 | |
| 	P -= 3;
 | |
| 	node_name = String::utf8(pCommentBegin+2, (int)(P - pCommentBegin-2));
 | |
| 	P += 3;
 | |
| #ifdef DEBUG_XML
 | |
| 	print_line("XML COMMENT: "+node_name);
 | |
| #endif
 | |
| 
 | |
| }
 | |
| 
 | |
| void XMLParser::_parse_opening_xml_element() {
 | |
| 
 | |
| 	node_type = NODE_ELEMENT;
 | |
| 	node_empty = false;
 | |
| 	attributes.clear();
 | |
| 
 | |
| 	// find name
 | |
| 	const char* startName = P;
 | |
| 
 | |
| 	// find end of element
 | |
| 	while(*P != '>' && !_is_white_space(*P))
 | |
| 		++P;
 | |
| 
 | |
| 	const char* endName = P;
 | |
| 
 | |
| 	// find attributes
 | |
| 	while(*P != '>')
 | |
| 	{
 | |
| 		if (_is_white_space(*P))
 | |
| 			++P;
 | |
| 		else
 | |
| 		{
 | |
| 			if (*P != '/')
 | |
| 			{
 | |
| 				// we've got an attribute
 | |
| 
 | |
| 				// read the attribute names
 | |
| 				const char* attributeNameBegin = P;
 | |
| 
 | |
| 				while(!_is_white_space(*P) && *P != '=')
 | |
| 					++P;
 | |
| 
 | |
| 				const char* attributeNameEnd = P;
 | |
| 				++P;
 | |
| 
 | |
| 				// read the attribute value
 | |
| 				// check for quotes and single quotes, thx to murphy
 | |
| 				while( (*P != '\"') && (*P != '\'') && *P)
 | |
| 					++P;
 | |
| 
 | |
| 				if (!*P) // malformatted xml file
 | |
| 					return;
 | |
| 
 | |
| 				const char attributeQuoteChar = *P;
 | |
| 
 | |
| 				++P;
 | |
| 				const char* attributeValueBegin = P;
 | |
| 
 | |
| 				while(*P != attributeQuoteChar && *P)
 | |
| 					++P;
 | |
| 
 | |
| 				if (!*P) // malformatted xml file
 | |
| 					return;
 | |
| 
 | |
| 				const char* attributeValueEnd = P;
 | |
| 				++P;
 | |
| 
 | |
| 				Attribute attr;
 | |
| 				attr.name = String::utf8(attributeNameBegin,
 | |
| 					(int)(attributeNameEnd - attributeNameBegin));
 | |
| 
 | |
| 				String s =String::utf8(attributeValueBegin,
 | |
| 					(int)(attributeValueEnd - attributeValueBegin));
 | |
| 
 | |
| 				attr.value = _replace_special_characters(s);
 | |
| 				attributes.push_back(attr);
 | |
| 			}
 | |
| 			else
 | |
| 			{
 | |
| 				// tag is closed directly
 | |
| 				++P;
 | |
| 				node_empty = true;
 | |
| 				break;
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	// check if this tag is closing directly
 | |
| 	if (endName > startName && *(endName-1) == '/')
 | |
| 	{
 | |
| 		// directly closing tag
 | |
| 		node_empty = true;
 | |
| 		endName--;
 | |
| 	}
 | |
| 
 | |
| 	node_name = String::utf8(startName, (int)(endName - startName));
 | |
| #ifdef DEBUG_XML
 | |
| 	print_line("XML OPEN: "+node_name);
 | |
| #endif
 | |
| 
 | |
| 	++P;
 | |
| }
 | |
| 
 | |
| 
 | |
| void XMLParser::_parse_current_node() {
 | |
| 
 | |
| 	char* start = P;
 | |
| 	node_offset = P - data;
 | |
| 
 | |
| 	// more forward until '<' found
 | |
| 	while(*P != '<' && *P)
 | |
| 		++P;
 | |
| 
 | |
| 	if (!*P)
 | |
| 		return;
 | |
| 
 | |
| 	if (P - start > 0)
 | |
| 	{
 | |
| 		// we found some text, store it
 | |
| 		if (_set_text(start, P))
 | |
| 			return;
 | |
| 	}
 | |
| 
 | |
| 	++P;
 | |
| 
 | |
| 	// based on current token, parse and report next element
 | |
| 	switch(*P)
 | |
| 	{
 | |
| 	case '/':
 | |
| 		_parse_closing_xml_element();
 | |
| 		break;
 | |
| 	case '?':
 | |
| 		_ignore_definition();
 | |
| 		break;
 | |
| 	case '!':
 | |
| 		if (!_parse_cdata())
 | |
| 			_parse_comment();
 | |
| 		break;
 | |
| 	default:
 | |
| 		_parse_opening_xml_element();
 | |
| 		break;
 | |
| 	}
 | |
| }
 | |
| 
 | |
| 
 | |
| uint64_t XMLParser::get_node_offset() const {
 | |
| 
 | |
| 	return node_offset;
 | |
| };
 | |
| 
 | |
| Error XMLParser::seek(uint64_t p_pos) {
 | |
| 
 | |
| 	ERR_FAIL_COND_V(!data, ERR_FILE_EOF)
 | |
| 	ERR_FAIL_COND_V(p_pos >= length, ERR_FILE_EOF);
 | |
| 
 | |
| 	P = data + p_pos;
 | |
| 
 | |
| 	return read();
 | |
| };
 | |
| 
 | |
| void XMLParser::_bind_methods() {
 | |
| 
 | |
| 	ObjectTypeDB::bind_method(_MD("read"),&XMLParser::read);
 | |
| 	ObjectTypeDB::bind_method(_MD("get_node_type"),&XMLParser::get_node_type);
 | |
| 	ObjectTypeDB::bind_method(_MD("get_node_name"),&XMLParser::get_node_name);
 | |
| 	ObjectTypeDB::bind_method(_MD("get_node_data"),&XMLParser::get_node_data);
 | |
| 	ObjectTypeDB::bind_method(_MD("get_node_offset"),&XMLParser::get_node_offset);
 | |
| 	ObjectTypeDB::bind_method(_MD("get_attribute_count"),&XMLParser::get_attribute_count);
 | |
| 	ObjectTypeDB::bind_method(_MD("get_attribute_name","idx"),&XMLParser::get_attribute_name);
 | |
| 	ObjectTypeDB::bind_method(_MD("get_attribute_value","idx"),(String (XMLParser::*)(int) const) &XMLParser::get_attribute_value);
 | |
| 	ObjectTypeDB::bind_method(_MD("has_attribute","name"),&XMLParser::has_attribute);
 | |
| 	ObjectTypeDB::bind_method(_MD("get_named_attribute_value","name"), (String (XMLParser::*)(const String&) const) &XMLParser::get_attribute_value);
 | |
| 	ObjectTypeDB::bind_method(_MD("get_named_attribute_value_safe","name"), &XMLParser::get_attribute_value_safe);
 | |
| 	ObjectTypeDB::bind_method(_MD("is_empty"),&XMLParser::is_empty);
 | |
| 	ObjectTypeDB::bind_method(_MD("get_current_line"),&XMLParser::get_current_line);
 | |
| 	ObjectTypeDB::bind_method(_MD("skip_section"),&XMLParser::skip_section);
 | |
| 	ObjectTypeDB::bind_method(_MD("seek","pos"),&XMLParser::seek);
 | |
| 	ObjectTypeDB::bind_method(_MD("open","file"),&XMLParser::open);
 | |
| 	ObjectTypeDB::bind_method(_MD("open_buffer","buffer"),&XMLParser::open_buffer);
 | |
| 
 | |
| 	BIND_CONSTANT( NODE_NONE );
 | |
| 	BIND_CONSTANT( NODE_ELEMENT );
 | |
| 	BIND_CONSTANT( NODE_ELEMENT_END );
 | |
| 	BIND_CONSTANT( NODE_TEXT );
 | |
| 	BIND_CONSTANT( NODE_COMMENT );
 | |
| 	BIND_CONSTANT( NODE_CDATA );
 | |
| 	BIND_CONSTANT( NODE_UNKNOWN );
 | |
| 
 | |
| };
 | |
| 
 | |
| 
 | |
| 
 | |
| Error XMLParser::read() {
 | |
| 
 | |
| 	// if not end reached, parse the node
 | |
| 	if (P && (P - data) < length - 1 && *P != 0)
 | |
| 	{
 | |
| 		_parse_current_node();
 | |
| 		return OK;
 | |
| 	}
 | |
| 
 | |
| 	return ERR_FILE_EOF;
 | |
| }
 | |
| 
 | |
| XMLParser::NodeType XMLParser::get_node_type() {
 | |
| 
 | |
| 	return node_type;
 | |
| }
 | |
| String XMLParser::get_node_data() const {
 | |
| 
 | |
| 	ERR_FAIL_COND_V( node_type != NODE_TEXT, "");
 | |
| 	return node_name;
 | |
| }
 | |
| 
 | |
| String XMLParser::get_node_name() const {
 | |
| 	ERR_FAIL_COND_V( node_type == NODE_TEXT, "");
 | |
| 	return node_name;
 | |
| }
 | |
| int XMLParser::get_attribute_count() const {
 | |
| 
 | |
| 	return attributes.size();
 | |
| }
 | |
| String XMLParser::get_attribute_name(int p_idx) const {
 | |
| 
 | |
| 	ERR_FAIL_INDEX_V(p_idx,attributes.size(),"");
 | |
| 	return attributes[p_idx].name;
 | |
| }
 | |
| String XMLParser::get_attribute_value(int p_idx) const {
 | |
| 
 | |
| 	ERR_FAIL_INDEX_V(p_idx,attributes.size(),"");
 | |
| 	return attributes[p_idx].value;
 | |
| }
 | |
| bool XMLParser::has_attribute(const String& p_name) const {
 | |
| 
 | |
| 	for(int i=0;i<attributes.size();i++) {
 | |
| 		if (attributes[i].name==p_name)
 | |
| 			return true;
 | |
| 	}
 | |
| 
 | |
| 	return false;
 | |
| }
 | |
| String XMLParser::get_attribute_value(const String& p_name) const {
 | |
| 
 | |
| 	int idx=-1;
 | |
| 	for(int i=0;i<attributes.size();i++) {
 | |
| 		if (attributes[i].name==p_name) {
 | |
| 			idx=i;
 | |
| 			break;
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	if (idx<0) {
 | |
| 		ERR_EXPLAIN("Attribute not found: "+p_name);
 | |
| 	}
 | |
| 	ERR_FAIL_COND_V(idx<0,"");
 | |
| 	return attributes[idx].value;
 | |
| 
 | |
| }
 | |
| 
 | |
| String XMLParser::get_attribute_value_safe(const String& p_name) const {
 | |
| 
 | |
| 	int idx=-1;
 | |
| 	for(int i=0;i<attributes.size();i++) {
 | |
| 		if (attributes[i].name==p_name) {
 | |
| 			idx=i;
 | |
| 			break;
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	if (idx<0)
 | |
| 		return "";
 | |
| 	return attributes[idx].value;
 | |
| 
 | |
| }
 | |
| bool XMLParser::is_empty() const {
 | |
| 
 | |
| 	return node_empty;
 | |
| }
 | |
| 
 | |
| Error XMLParser::open_buffer(const Vector<uint8_t>& p_buffer) {
 | |
| 
 | |
| 	ERR_FAIL_COND_V(p_buffer.size()==0,ERR_INVALID_DATA);
 | |
| 
 | |
| 	length = p_buffer.size();
 | |
| 	data = memnew_arr( char, length+1);
 | |
| 	copymem(data,p_buffer.ptr(),length);
 | |
| 	data[length]=0;
 | |
| 	P=data;
 | |
| 	return OK;
 | |
| 
 | |
| }
 | |
| 
 | |
| Error XMLParser::open(const String& p_path) {
 | |
| 
 | |
| 	Error err;
 | |
| 	FileAccess * file = FileAccess::open(p_path,FileAccess::READ,&err);
 | |
| 
 | |
| 	if (err) {
 | |
| 		ERR_FAIL_COND_V(err!=OK,err);
 | |
| 	}
 | |
| 
 | |
| 	length = file->get_len();
 | |
| 	ERR_FAIL_COND_V(length<1, ERR_FILE_CORRUPT);
 | |
| 
 | |
| 	data = memnew_arr( char, length+1);
 | |
| 	file->get_buffer((uint8_t*)data,length);
 | |
| 	data[length]=0;
 | |
| 	P=data;
 | |
| 
 | |
| 	memdelete(file);
 | |
| 
 | |
| 	return OK;
 | |
| 
 | |
| }
 | |
| 
 | |
| void XMLParser::skip_section() {
 | |
| 
 | |
| 	// skip if this element is empty anyway.
 | |
| 	if (is_empty())
 | |
| 		return;
 | |
| 
 | |
| 	// read until we've reached the last element in this section
 | |
| 	int tagcount = 1;
 | |
| 
 | |
| 	while(tagcount && read()==OK)
 | |
| 	{
 | |
| 		if (get_node_type() == XMLParser::NODE_ELEMENT &&
 | |
| 			!is_empty())
 | |
| 		{
 | |
| 			++tagcount;
 | |
| 		}
 | |
| 		else
 | |
| 		if (get_node_type() == XMLParser::NODE_ELEMENT_END)
 | |
| 			--tagcount;
 | |
| 	}
 | |
| 
 | |
| }
 | |
| 
 | |
| void XMLParser::close() {
 | |
| 
 | |
| 	if (data)
 | |
| 		memdelete_arr(data);
 | |
| 	data=NULL;
 | |
| 	length=0;
 | |
| 	P=NULL;
 | |
| 	node_empty=false;
 | |
| 	node_type=NODE_NONE;
 | |
| 	node_offset = 0;
 | |
| }
 | |
| 
 | |
| int XMLParser::get_current_line() const {
 | |
| 
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| XMLParser::XMLParser() {
 | |
| 
 | |
| 	data=NULL;
 | |
| 	close();
 | |
| 	special_characters.push_back("&");
 | |
| 	special_characters.push_back("<lt;");
 | |
| 	special_characters.push_back(">gt;");
 | |
| 	special_characters.push_back("\"quot;");
 | |
| 	special_characters.push_back("'apos;");
 | |
| 
 | |
| 
 | |
| }
 | |
| XMLParser::~XMLParser() {
 | |
| 
 | |
| 
 | |
| 	if (data)
 | |
| 		memdelete_arr(data);
 | |
| }
 | 
