| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634 |
- #include "tinyxml2.h"
- #include <stdio.h>
- #include <stdlib.h>
- #include <string.h>
- #if defined( _MSC_VER )
- #include <crtdbg.h>
- _CrtMemState startMemState;
- _CrtMemState endMemState;
- #endif
- using namespace tinyxml2;
- int gPass = 0;
- int gFail = 0;
- bool XMLTest (const char* testString, const char* expected, const char* found, bool echo=true )
- {
- bool pass = !strcmp( expected, found );
- if ( pass )
- printf ("[pass]");
- else
- printf ("[fail]");
- if ( !echo )
- printf (" %s\n", testString);
- else
- printf (" %s [%s][%s]\n", testString, expected, found);
- if ( pass )
- ++gPass;
- else
- ++gFail;
- return pass;
- }
- bool XMLTest( const char* testString, int expected, int found, bool echo=true )
- {
- bool pass = ( expected == found );
- if ( pass )
- printf ("[pass]");
- else
- printf ("[fail]");
- if ( !echo )
- printf (" %s\n", testString);
- else
- printf (" %s [%d][%d]\n", testString, expected, found);
- if ( pass )
- ++gPass;
- else
- ++gFail;
- return pass;
- }
- void NullLineEndings( char* p )
- {
- while( p && *p ) {
- if ( *p == '\n' || *p == '\r' ) {
- *p = 0;
- return;
- }
- ++p;
- }
- }
- int main( int /*argc*/, const char* /*argv*/ )
- {
- #if defined( _MSC_VER )
- _CrtMemCheckpoint( &startMemState );
- #endif
- {
- static const char* test[] = { "<element />",
- "<element></element>",
- "<element><subelement/></element>",
- "<element><subelement></subelement></element>",
- "<element><subelement><subsub/></subelement></element>",
- "<!--comment beside elements--><element><subelement></subelement></element>",
- "<!--comment beside elements, this time with spaces--> \n <element> <subelement> \n </subelement> </element>",
- "<element attrib1='foo' attrib2=\"bar\" ></element>",
- "<element attrib1='foo' attrib2=\"bar\" ><subelement attrib3='yeehaa' /></element>",
- "<element>Text inside element.</element>",
- "<element><b></b></element>",
- "<element>Text inside and <b>bolded</b> in the element.</element>",
- "<outer><element>Text inside and <b>bolded</b> in the element.</element></outer>",
- "<element>This & That.</element>",
- "<element attrib='This<That' />",
- 0
- };
- for( int i=0; test[i]; ++i ) {
- XMLDocument doc;
- doc.Parse( test[i] );
- doc.Print();
- printf( "----------------------------------------------\n" );
- }
- }
- #if 1
- {
- static const char* test = "<!--hello world\n"
- " line 2\r"
- " line 3\r\n"
- " line 4\n\r"
- " line 5\r-->";
- XMLDocument doc;
- doc.Parse( test );
- doc.Print();
- }
- {
- static const char* test = "<element>Text before.</element>";
- XMLDocument doc;
- doc.Parse( test );
- XMLElement* root = doc.FirstChildElement();
- XMLElement* newElement = doc.NewElement( "Subelement" );
- root->InsertEndChild( newElement );
- doc.Print();
- }
- {
- XMLDocument* doc = new XMLDocument();
- static const char* test = "<element><sub/></element>";
- doc->Parse( test );
- delete doc;
- }
- {
- // Test: Programmatic DOM
- // Build:
- // <element>
- // <!--comment-->
- // <sub attrib="1" />
- // <sub attrib="2" />
- // <sub attrib="3" >& Text!</sub>
- // <element>
- XMLDocument* doc = new XMLDocument();
- XMLNode* element = doc->InsertEndChild( doc->NewElement( "element" ) );
- XMLElement* sub[3] = { doc->NewElement( "sub" ), doc->NewElement( "sub" ), doc->NewElement( "sub" ) };
- for( int i=0; i<3; ++i ) {
- sub[i]->SetAttribute( "attrib", i );
- }
- element->InsertEndChild( sub[2] );
- XMLNode* comment = element->InsertFirstChild( doc->NewComment( "comment" ) );
- element->InsertAfterChild( comment, sub[0] );
- element->InsertAfterChild( sub[0], sub[1] );
- sub[2]->InsertFirstChild( doc->NewText( "& Text!" ));
- doc->Print();
- XMLTest( "Programmatic DOM", "comment", doc->FirstChildElement( "element" )->FirstChild()->Value() );
- XMLTest( "Programmatic DOM", "0", doc->FirstChildElement( "element" )->FirstChildElement()->Attribute( "attrib" ) );
- XMLTest( "Programmatic DOM", 2, doc->FirstChildElement()->LastChildElement( "sub" )->IntAttribute( "attrib" ) );
- XMLTest( "Programmatic DOM", "& Text!",
- doc->FirstChildElement()->LastChildElement( "sub" )->FirstChild()->ToText()->Value() );
- // And now deletion:
- element->DeleteChild( sub[2] );
- doc->DeleteNode( comment );
- element->FirstChildElement()->SetAttribute( "attrib", true );
- element->LastChildElement()->DeleteAttribute( "attrib" );
- XMLTest( "Programmatic DOM", true, doc->FirstChildElement()->FirstChildElement()->BoolAttribute( "attrib" ) );
- int value = 10;
- int result = doc->FirstChildElement()->LastChildElement()->QueryIntAttribute( "attrib", &value );
- XMLTest( "Programmatic DOM", result, NO_ATTRIBUTE );
- XMLTest( "Programmatic DOM", value, 10 );
- doc->Print();
- XMLPrinter streamer;
- doc->Print( &streamer );
- printf( "%s", streamer.CStr() );
- delete doc;
- }
- {
- // Test: Dream
- // XML1 : 1,187,569 bytes in 31,209 allocations
- // XML2 : 469,073 bytes in 323 allocations
- //int newStart = gNew;
- XMLDocument doc;
- doc.LoadFile( "dream.xml" );
- doc.SaveFile( "dreamout.xml" );
- doc.PrintError();
- XMLTest( "Dream", "xml version=\"1.0\"",
- doc.FirstChild()->ToDeclaration()->Value() );
- XMLTest( "Dream", true, doc.FirstChild()->NextSibling()->ToUnknown() ? true : false );
- XMLTest( "Dream", "DOCTYPE PLAY SYSTEM \"play.dtd\"",
- doc.FirstChild()->NextSibling()->ToUnknown()->Value() );
- XMLTest( "Dream", "And Robin shall restore amends.",
- doc.LastChild()->LastChild()->LastChild()->LastChild()->LastChildElement()->GetText() );
- XMLTest( "Dream", "And Robin shall restore amends.",
- doc.LastChild()->LastChild()->LastChild()->LastChild()->LastChildElement()->GetText() );
- XMLDocument doc2;
- doc2.LoadFile( "dreamout.xml" );
- XMLTest( "Dream-out", "xml version=\"1.0\"",
- doc2.FirstChild()->ToDeclaration()->Value() );
- XMLTest( "Dream-out", true, doc2.FirstChild()->NextSibling()->ToUnknown() ? true : false );
- XMLTest( "Dream-out", "DOCTYPE PLAY SYSTEM \"play.dtd\"",
- doc2.FirstChild()->NextSibling()->ToUnknown()->Value() );
- XMLTest( "Dream-out", "And Robin shall restore amends.",
- doc2.LastChild()->LastChild()->LastChild()->LastChild()->LastChildElement()->GetText() );
- //gNewTotal = gNew - newStart;
- }
- {
- const char* error = "<?xml version=\"1.0\" standalone=\"no\" ?>\n"
- "<passages count=\"006\" formatversion=\"20020620\">\n"
- " <wrong error>\n"
- "</passages>";
- XMLDocument doc;
- doc.Parse( error );
- XMLTest( "Bad XML", doc.ErrorID(), ERROR_PARSING_ATTRIBUTE );
- }
- {
- const char* str = "<doc attr0='1' attr1='2.0' attr2='foo' />";
- XMLDocument doc;
- doc.Parse( str );
- XMLElement* ele = doc.FirstChildElement();
- int iVal, result;
- double dVal;
- result = ele->QueryDoubleAttribute( "attr0", &dVal );
- XMLTest( "Query attribute: int as double", result, XML_NO_ERROR );
- XMLTest( "Query attribute: int as double", (int)dVal, 1 );
- result = ele->QueryDoubleAttribute( "attr1", &dVal );
- XMLTest( "Query attribute: double as double", (int)dVal, 2 );
- result = ele->QueryIntAttribute( "attr1", &iVal );
- XMLTest( "Query attribute: double as int", result, XML_NO_ERROR );
- XMLTest( "Query attribute: double as int", iVal, 2 );
- result = ele->QueryIntAttribute( "attr2", &iVal );
- XMLTest( "Query attribute: not a number", result, WRONG_ATTRIBUTE_TYPE );
- result = ele->QueryIntAttribute( "bar", &iVal );
- XMLTest( "Query attribute: does not exist", result, NO_ATTRIBUTE );
- }
- {
- const char* str = "<doc/>";
- XMLDocument doc;
- doc.Parse( str );
- XMLElement* ele = doc.FirstChildElement();
- int iVal;
- double dVal;
- ele->SetAttribute( "str", "strValue" );
- ele->SetAttribute( "int", 1 );
- ele->SetAttribute( "double", -1.0 );
- const char* cStr = ele->Attribute( "str" );
- ele->QueryIntAttribute( "int", &iVal );
- ele->QueryDoubleAttribute( "double", &dVal );
- XMLTest( "Attribute round trip. c-string.", "strValue", cStr );
- XMLTest( "Attribute round trip. int.", 1, iVal );
- XMLTest( "Attribute round trip. double.", -1, (int)dVal );
- }
- {
- XMLDocument doc;
- doc.LoadFile( "utf8test.xml" );
- // Get the attribute "value" from the "Russian" element and check it.
- XMLElement* element = doc.FirstChildElement( "document" )->FirstChildElement( "Russian" );
- const unsigned char correctValue[] = { 0xd1U, 0x86U, 0xd0U, 0xb5U, 0xd0U, 0xbdU, 0xd0U, 0xbdU,
- 0xd0U, 0xbeU, 0xd1U, 0x81U, 0xd1U, 0x82U, 0xd1U, 0x8cU, 0 };
- XMLTest( "UTF-8: Russian value.", (const char*)correctValue, element->Attribute( "value" ) );
- const unsigned char russianElementName[] = { 0xd0U, 0xa0U, 0xd1U, 0x83U,
- 0xd1U, 0x81U, 0xd1U, 0x81U,
- 0xd0U, 0xbaU, 0xd0U, 0xb8U,
- 0xd0U, 0xb9U, 0 };
- const char russianText[] = "<\xD0\xB8\xD0\xBC\xD0\xB5\xD0\xB5\xD1\x82>";
- XMLText* text = doc.FirstChildElement( "document" )->FirstChildElement( (const char*) russianElementName )->FirstChild()->ToText();
- XMLTest( "UTF-8: Browsing russian element name.",
- russianText,
- text->Value() );
- // Now try for a round trip.
- doc.SaveFile( "utf8testout.xml" );
- // Check the round trip.
- char savedBuf[256];
- char verifyBuf[256];
- int okay = 0;
- #pragma warning ( push )
- #pragma warning ( disable : 4996 ) // Fail to see a compelling reason why this should be deprecated.
- FILE* saved = fopen( "utf8testout.xml", "r" );
- FILE* verify = fopen( "utf8testverify.xml", "r" );
- #pragma warning ( pop )
- if ( saved && verify )
- {
- okay = 1;
- while ( fgets( verifyBuf, 256, verify ) )
- {
- fgets( savedBuf, 256, saved );
- NullLineEndings( verifyBuf );
- NullLineEndings( savedBuf );
- if ( strcmp( verifyBuf, savedBuf ) )
- {
- printf( "verify:%s<\n", verifyBuf );
- printf( "saved :%s<\n", savedBuf );
- okay = 0;
- break;
- }
- }
- }
- if ( saved )
- fclose( saved );
- if ( verify )
- fclose( verify );
- XMLTest( "UTF-8: Verified multi-language round trip.", 1, okay );
- }
- // --------GetText()-----------
- {
- const char* str = "<foo>This is text</foo>";
- XMLDocument doc;
- doc.Parse( str );
- const XMLElement* element = doc.RootElement();
- XMLTest( "GetText() normal use.", "This is text", element->GetText() );
- str = "<foo><b>This is text</b></foo>";
- doc.Parse( str );
- element = doc.RootElement();
- XMLTest( "GetText() contained element.", element->GetText() == 0, true );
- }
- // ---------- CDATA ---------------
- {
- const char* str = "<xmlElement>"
- "<![CDATA["
- "I am > the rules!\n"
- "...since I make symbolic puns"
- "]]>"
- "</xmlElement>";
- XMLDocument doc;
- doc.Parse( str );
- doc.Print();
- XMLTest( "CDATA parse.", doc.FirstChildElement()->FirstChild()->Value(),
- "I am > the rules!\n...since I make symbolic puns",
- false );
- }
- // ----------- CDATA -------------
- {
- const char* str = "<xmlElement>"
- "<![CDATA["
- "<b>I am > the rules!</b>\n"
- "...since I make symbolic puns"
- "]]>"
- "</xmlElement>";
- XMLDocument doc;
- doc.Parse( str );
- doc.Print();
- XMLTest( "CDATA parse. [ tixml1:1480107 ]", doc.FirstChildElement()->FirstChild()->Value(),
- "<b>I am > the rules!</b>\n...since I make symbolic puns",
- false );
- }
- // InsertAfterChild causes crash.
- {
- // InsertBeforeChild and InsertAfterChild causes crash.
- XMLDocument doc;
- XMLElement* parent = doc.NewElement( "Parent" );
- doc.InsertFirstChild( parent );
- XMLElement* childText0 = doc.NewElement( "childText0" );
- XMLElement* childText1 = doc.NewElement( "childText1" );
- XMLNode* childNode0 = parent->InsertEndChild( childText0 );
- XMLNode* childNode1 = parent->InsertAfterChild( childNode0, childText1 );
- XMLTest( "Test InsertAfterChild on empty node. ", ( childNode1 == parent->LastChild() ), true );
- }
- {
- // Entities not being written correctly.
- // From Lynn Allen
- const char* passages =
- "<?xml version=\"1.0\" standalone=\"no\" ?>"
- "<passages count=\"006\" formatversion=\"20020620\">"
- "<psg context=\"Line 5 has "quotation marks" and 'apostrophe marks'."
- " It also has <, >, and &, as well as a fake copyright ©.\"> </psg>"
- "</passages>";
- XMLDocument doc;
- doc.Parse( passages );
- XMLElement* psg = doc.RootElement()->FirstChildElement();
- const char* context = psg->Attribute( "context" );
- const char* expected = "Line 5 has \"quotation marks\" and 'apostrophe marks'. It also has <, >, and &, as well as a fake copyright \xC2\xA9.";
- XMLTest( "Entity transformation: read. ", expected, context, true );
- #pragma warning ( push )
- #pragma warning ( disable : 4996 ) // Fail to see a compelling reason why this should be deprecated.
- FILE* textfile = fopen( "textfile.txt", "w" );
- #pragma warning ( pop )
- if ( textfile )
- {
- XMLPrinter streamer( textfile );
- psg->Accept( &streamer );
- fclose( textfile );
- }
- #pragma warning ( push )
- #pragma warning ( disable : 4996 ) // Fail to see a compelling reason why this should be deprecated.
- textfile = fopen( "textfile.txt", "r" );
- #pragma warning ( pop )
- TIXMLASSERT( textfile );
- if ( textfile )
- {
- char buf[ 1024 ];
- fgets( buf, 1024, textfile );
- XMLTest( "Entity transformation: write. ",
- "<psg context=\"Line 5 has "quotation marks" and 'apostrophe marks'."
- " It also has <, >, and &, as well as a fake copyright \xC2\xA9.\"/>\n",
- buf, false );
- }
- fclose( textfile );
- }
- {
- const char* test = "<?xml version='1.0'?><a.elem xmi.version='2.0'/>";
- XMLDocument doc;
- doc.Parse( test );
- XMLTest( "dot in names", doc.Error(), 0);
- XMLTest( "dot in names", doc.FirstChildElement()->Name(), "a.elem" );
- XMLTest( "dot in names", doc.FirstChildElement()->Attribute( "xmi.version" ), "2.0" );
- }
- {
- const char* test = "<element><Name>1.1 Start easy ignore fin thickness
</Name></element>";
- XMLDocument doc;
- doc.Parse( test );
- XMLText* text = doc.FirstChildElement()->FirstChildElement()->FirstChild()->ToText();
- XMLTest( "Entity with one digit.",
- text->Value(), "1.1 Start easy ignore fin thickness\n",
- false );
- }
- {
- // DOCTYPE not preserved (950171)
- //
- const char* doctype =
- "<?xml version=\"1.0\" ?>"
- "<!DOCTYPE PLAY SYSTEM 'play.dtd'>"
- "<!ELEMENT title (#PCDATA)>"
- "<!ELEMENT books (title,authors)>"
- "<element />";
- XMLDocument doc;
- doc.Parse( doctype );
- doc.SaveFile( "test7.xml" );
- doc.DeleteChild( doc.RootElement() );
- doc.LoadFile( "test7.xml" );
- doc.Print();
-
- const XMLUnknown* decl = doc.FirstChild()->NextSibling()->ToUnknown();
- XMLTest( "Correct value of unknown.", "DOCTYPE PLAY SYSTEM 'play.dtd'", decl->Value() );
- }
- {
- // Comments do not stream out correctly.
- const char* doctype =
- "<!-- Somewhat<evil> -->";
- XMLDocument doc;
- doc.Parse( doctype );
- XMLComment* comment = doc.FirstChild()->ToComment();
- XMLTest( "Comment formatting.", " Somewhat<evil> ", comment->Value() );
- }
- {
- // Double attributes
- const char* doctype = "<element attr='red' attr='blue' />";
- XMLDocument doc;
- doc.Parse( doctype );
-
- XMLTest( "Parsing repeated attributes.", ERROR_PARSING_ATTRIBUTE, doc.ErrorID() ); // is an error to tinyxml (didn't use to be, but caused issues)
- }
- {
- // Embedded null in stream.
- const char* doctype = "<element att\0r='red' attr='blue' />";
- XMLDocument doc;
- doc.Parse( doctype );
- XMLTest( "Embedded null throws error.", true, doc.Error() );
- }
- {
- // Empty documents should return TIXML_ERROR_PARSING_EMPTY, bug 1070717
- const char* str = " ";
- XMLDocument doc;
- doc.Parse( str );
- XMLTest( "Empty document error", ERROR_EMPTY_DOCUMENT, doc.ErrorID() );
- }
- {
- // Low entities
- XMLDocument doc;
- doc.Parse( "<test></test>" );
- const char result[] = { 0x0e, 0 };
- XMLTest( "Low entities.", doc.FirstChildElement()->GetText(), result );
- doc.Print();
- }
- {
- // Attribute values with trailing quotes not handled correctly
- XMLDocument doc;
- doc.Parse( "<foo attribute=bar\" />" );
- XMLTest( "Throw error with bad end quotes.", doc.Error(), true );
- }
- {
- // [ 1663758 ] Failure to report error on bad XML
- XMLDocument xml;
- xml.Parse("<x>");
- XMLTest("Missing end tag at end of input", xml.Error(), true);
- xml.Parse("<x> ");
- XMLTest("Missing end tag with trailing whitespace", xml.Error(), true);
- xml.Parse("<x></y>");
- XMLTest("Mismatched tags", xml.ErrorID(), ERROR_MISMATCHED_ELEMENT);
- }
- {
- // [ 1475201 ] TinyXML parses entities in comments
- XMLDocument xml;
- xml.Parse("<!-- declarations for <head> & <body> -->"
- "<!-- far & away -->" );
- XMLNode* e0 = xml.FirstChild();
- XMLNode* e1 = e0->NextSibling();
- XMLComment* c0 = e0->ToComment();
- XMLComment* c1 = e1->ToComment();
- XMLTest( "Comments ignore entities.", " declarations for <head> & <body> ", c0->Value(), true );
- XMLTest( "Comments ignore entities.", " far & away ", c1->Value(), true );
- }
- {
- XMLDocument xml;
- xml.Parse( "<Parent>"
- "<child1 att=''/>"
- "<!-- With this comment, child2 will not be parsed! -->"
- "<child2 att=''/>"
- "</Parent>" );
- xml.Print();
- int count = 0;
- for( XMLNode* ele = xml.FirstChildElement( "Parent" )->FirstChild();
- ele;
- ele = ele->NextSibling() )
- {
- ++count;
- }
- XMLTest( "Comments iterate correctly.", 3, count );
- }
- {
- // trying to repro ]1874301]. If it doesn't go into an infinite loop, all is well.
- unsigned char buf[] = "<?xml version=\"1.0\" encoding=\"utf-8\"?><feed><![CDATA[Test XMLblablablalblbl";
- buf[60] = 239;
- buf[61] = 0;
- XMLDocument doc;
- doc.Parse( (const char*)buf);
- }
- {
- // bug 1827248 Error while parsing a little bit malformed file
- // Actually not malformed - should work.
- XMLDocument xml;
- xml.Parse( "<attributelist> </attributelist >" );
- XMLTest( "Handle end tag whitespace", false, xml.Error() );
- }
- {
- // This one must not result in an infinite loop
- XMLDocument xml;
- xml.Parse( "<infinite>loop" );
- XMLTest( "Infinite loop test.", true, true );
- }
- #endif
- #if defined( _MSC_VER )
- _CrtMemCheckpoint( &endMemState );
- //_CrtMemDumpStatistics( &endMemState );
- _CrtMemState diffMemState;
- _CrtMemDifference( &diffMemState, &startMemState, &endMemState );
- _CrtMemDumpStatistics( &diffMemState );
- //printf( "new total=%d\n", gNewTotal );
- #endif
- printf ("\nPass %d, Fail %d\n", gPass, gFail);
- return 0;
- }
|