From dc13914cd698b49cbae491c1e1aebbcd44fbe932 Mon Sep 17 00:00:00 2001 From: Brad King Date: Thu, 5 Feb 2009 16:31:37 -0500 Subject: ENH: Create cmXMLSafe to help escapes in XML This class provides easy syntax to efficiently insert blocks of data into XML documents with proper escapes. It replaces the old cmCTest::MakeXMLSafe and cmSystemTools::MakeXMLSafe methods which allocated extra memory instead of directly streaming the data. --- Source/CMakeLists.txt | 2 + Source/CPack/cmCPackGenerator.cxx | 3 +- Source/CTest/cmCTestBuildHandler.cxx | 9 ++-- Source/CTest/cmCTestConfigureHandler.cxx | 3 +- Source/CTest/cmCTestCoverageHandler.cxx | 23 ++++---- Source/CTest/cmCTestMemCheckHandler.cxx | 15 +++--- Source/CTest/cmCTestTestHandler.cxx | 23 ++++---- Source/CTest/cmCTestUpdateHandler.cxx | 39 +++++++------- Source/cmCTest.cxx | 59 ++------------------ Source/cmCTest.h | 3 -- Source/cmSystemTools.cxx | 50 ----------------- Source/cmSystemTools.h | 3 -- Source/cmXMLSafe.cxx | 93 ++++++++++++++++++++++++++++++++ Source/cmXMLSafe.h | 49 +++++++++++++++++ 14 files changed, 209 insertions(+), 165 deletions(-) create mode 100644 Source/cmXMLSafe.cxx create mode 100644 Source/cmXMLSafe.h diff --git a/Source/CMakeLists.txt b/Source/CMakeLists.txt index 210c733..1103c37 100644 --- a/Source/CMakeLists.txt +++ b/Source/CMakeLists.txt @@ -213,6 +213,8 @@ SET(SRCS cmVersion.h cmXMLParser.cxx cmXMLParser.h + cmXMLSafe.cxx + cmXMLSafe.h cmake.cxx cmake.h cmakewizard.cxx diff --git a/Source/CPack/cmCPackGenerator.cxx b/Source/CPack/cmCPackGenerator.cxx index a94736a..a8d500c 100644 --- a/Source/CPack/cmCPackGenerator.cxx +++ b/Source/CPack/cmCPackGenerator.cxx @@ -24,6 +24,7 @@ #include "cmLocalGenerator.h" #include "cmGeneratedFileStream.h" #include "cmCPackComponentGroup.h" +#include "cmXMLSafe.h" #include #include @@ -143,7 +144,7 @@ int cmCPackGenerator::PrepareNames() "Read description file: " << descFileName << std::endl); while ( ifs && cmSystemTools::GetLineFromStream(ifs, line) ) { - ostr << cmSystemTools::MakeXMLSafe(line.c_str()) << std::endl; + ostr << cmXMLSafe(line) << std::endl; } this->SetOptionIfNotSet("CPACK_PACKAGE_DESCRIPTION", ostr.str().c_str()); } diff --git a/Source/CTest/cmCTestBuildHandler.cxx b/Source/CTest/cmCTestBuildHandler.cxx index 9e67f73..76d9425 100644 --- a/Source/CTest/cmCTestBuildHandler.cxx +++ b/Source/CTest/cmCTestBuildHandler.cxx @@ -23,6 +23,7 @@ #include "cmLocalGenerator.h" #include "cmGlobalGenerator.h" #include "cmGeneratedFileStream.h" +#include "cmXMLSafe.h" //#include #include @@ -490,7 +491,7 @@ void cmCTestBuildHandler::GenerateXMLHeader(std::ostream& os) static_cast(this->StartBuildTime) << "\n" << "" - << this->CTest->MakeXMLSafe( + << cmXMLSafe( this->CTest->GetCTestConfiguration("MakeCommand")) << "" << std::endl; } @@ -526,7 +527,7 @@ void cmCTestBuildHandler::GenerateXMLLogScraped(std::ostream& os) } os << "\t<" << (cm->Error ? "Error" : "Warning") << ">\n" << "\t\t" << cm->LogLine << "\n" - << "\t\t" << this->CTest->MakeXMLSafe(cm->Text) + << "\t\t" << cmXMLSafe(cm->Text).Quotes(false) << "\n" << std::endl; std::vector::iterator rit; for ( rit = this->ErrorWarningFileLineRegex.begin(); @@ -579,9 +580,9 @@ void cmCTestBuildHandler::GenerateXMLLogScraped(std::ostream& os) << "" << std::endl; } } - os << "\t\t" << this->CTest->MakeXMLSafe(cm->PreContext) + os << "\t\t" << cmXMLSafe(cm->PreContext).Quotes(false) << "\n" - << "\t\t" << this->CTest->MakeXMLSafe(cm->PostContext); + << "\t\t" << cmXMLSafe(cm->PostContext).Quotes(false); // is this the last warning or error, if so notify if (cm->Error && !numErrorsAllowed || !cm->Error && !numWarningsAllowed) diff --git a/Source/CTest/cmCTestConfigureHandler.cxx b/Source/CTest/cmCTestConfigureHandler.cxx index 6004508..12fe75b 100644 --- a/Source/CTest/cmCTestConfigureHandler.cxx +++ b/Source/CTest/cmCTestConfigureHandler.cxx @@ -20,6 +20,7 @@ #include "cmCTest.h" #include "cmGeneratedFileStream.h" #include "cmake.h" +#include "cmXMLSafe.h" #include @@ -106,7 +107,7 @@ int cmCTestConfigureHandler::ProcessHandler() os << "" << cCommand.c_str() << "" << std::endl; cmCTestLog(this->CTest, DEBUG, "End" << std::endl); - os << "" << cmCTest::MakeXMLSafe(output) << "" << std::endl; + os << "" << cmXMLSafe(output) << "" << std::endl; std::string end_time = this->CTest->CurrentTime(); os << "\t" << retVal << "\n" << "\t" << end_time << "\n" diff --git a/Source/CTest/cmCTestCoverageHandler.cxx b/Source/CTest/cmCTestCoverageHandler.cxx index 8f73220..40dac00 100644 --- a/Source/CTest/cmCTestCoverageHandler.cxx +++ b/Source/CTest/cmCTestCoverageHandler.cxx @@ -19,6 +19,7 @@ #include "cmake.h" #include "cmSystemTools.h" #include "cmGeneratedFileStream.h" +#include "cmXMLSafe.h" #include #include @@ -470,8 +471,8 @@ int cmCTestCoverageHandler::ProcessHandler() const cmCTestCoverageHandlerContainer::SingleFileCoverageVector& fcov = fileIterator->second; covLogFile << "\tCTest->MakeXMLSafe(fileName.c_str()) - << "\" FullPath=\"" << this->CTest->MakeXMLSafe( + << cmXMLSafe(fileName) + << "\" FullPath=\"" << cmXMLSafe( this->CTest->GetShortPathToFile( fileIterator->first.c_str())) << "\">" << std::endl << "\t\t" << std::endl; @@ -507,7 +508,7 @@ int cmCTestCoverageHandler::ProcessHandler() } covLogFile << "\t\t" - << this->CTest->MakeXMLSafe(line.c_str()) << "" << std::endl; + << cmXMLSafe(line) << "" << std::endl; if ( fcov[cc] == 0 ) { untested ++; @@ -536,8 +537,8 @@ int cmCTestCoverageHandler::ProcessHandler() total_untested += untested; covLogFile << "\t\t" << std::endl << "\t" << std::endl; - covSumFile << "\tCTest->MakeXMLSafe(fileName) - << "\" FullPath=\"" << this->CTest->MakeXMLSafe( + covSumFile << "\tCTest->GetShortPathToFile(fullFileName.c_str())) << "\" Covered=\"" << (tested > 0 ? "true":"false") << "\">\n" << "\t\t" << tested << "\n" @@ -1329,8 +1330,8 @@ int cmCTestCoverageHandler::RunBullseyeCoverageBranch( << std::endl); // start the file output covLogFile << "\tCTest->MakeXMLSafe(i->first.c_str()) - << "\" FullPath=\"" << this->CTest->MakeXMLSafe( + << cmXMLSafe(i->first) + << "\" FullPath=\"" << cmXMLSafe( this->CTest->GetShortPathToFile( i->second.c_str())) << "\">" << std::endl << "\t\t" << std::endl; @@ -1339,7 +1340,7 @@ int cmCTestCoverageHandler::RunBullseyeCoverageBranch( for(int k =0; bullseyeHelp[k] != 0; ++k) { covLogFile << "\t\t" - << this->CTest->MakeXMLSafe(bullseyeHelp[k]) + << cmXMLSafe(bullseyeHelp[k]) << "" << std::endl; line++; } @@ -1355,7 +1356,7 @@ int cmCTestCoverageHandler::RunBullseyeCoverageBranch( else if(valid) { covLogFile << "\t\t" - << this->CTest->MakeXMLSafe(lineIn.c_str()) + << cmXMLSafe(lineIn) << "" << std::endl; line++; } @@ -1561,8 +1562,8 @@ int cmCTestCoverageHandler::RunBullseyeSourceSummary( tmpLog << "percentBranch: " << percentBranch << "\n"; tmpLog << "percentCoverage: " << percent_coverage << "\n"; tmpLog << "coverage metric: " << cmet << "\n"; - covSumFile << "\tCTest->MakeXMLSafe(sourceFile) - << "\" FullPath=\"" << this->CTest->MakeXMLSafe( + covSumFile << "\tCTest->GetShortPathToFile(file.c_str())) << "\" Covered=\"" << (cmet>0?"true":"false") << "\">\n" << "\t\t" diff --git a/Source/CTest/cmCTestMemCheckHandler.cxx b/Source/CTest/cmCTestMemCheckHandler.cxx index 97a6ee0..12d178d 100644 --- a/Source/CTest/cmCTestMemCheckHandler.cxx +++ b/Source/CTest/cmCTestMemCheckHandler.cxx @@ -24,6 +24,7 @@ #include #include #include "cmMakefile.h" +#include "cmXMLSafe.h" #include #include @@ -77,8 +78,8 @@ public: int i = 0; for(; atts[i] != 0; i+=2) { - ostr << " " << this->CTest->MakeXMLSafe(atts[i]).c_str() - << " - " << this->CTest->MakeXMLSafe(atts[i+1]).c_str() << "\n"; + ostr << " " << cmXMLSafe(atts[i]) + << " - " << cmXMLSafe(atts[i+1]) << "\n"; } ostr << "\n"; this->Log += ostr.str(); @@ -313,7 +314,7 @@ void cmCTestMemCheckHandler::GenerateDartOutput(std::ostream& os) { cmCTestTestResult *result = &this->TestResults[cc]; std::string testPath = result->Path + "/" + result->Name; - os << "\t\t" << cmCTest::MakeXMLSafe( + os << "\t\t" << cmXMLSafe( this->CTest->GetShortPathToFile(testPath.c_str())) << "" << std::endl; } @@ -610,7 +611,7 @@ bool cmCTestMemCheckHandler::ProcessMemCheckPurifyOutput( results[failure] ++; defects ++; } - ostr << cmCTest::MakeXMLSafe(*i) << std::endl; + ostr << cmXMLSafe(*i) << std::endl; } log = ostr.str(); @@ -750,7 +751,7 @@ bool cmCTestMemCheckHandler::ProcessMemCheckValgrindOutput( defects ++; } totalOutputSize += lines[cc].size(); - ostr << cmCTest::MakeXMLSafe(lines[cc]) << std::endl; + ostr << cmXMLSafe(lines[cc]) << std::endl; } else { @@ -767,9 +768,9 @@ bool cmCTestMemCheckHandler::ProcessMemCheckValgrindOutput( cmCTestLog(this->CTest, DEBUG, "before xml safe " << lines[*i] << std::endl); cmCTestLog(this->CTest, DEBUG, "after xml safe " - << cmCTest::MakeXMLSafe(lines[*i]) << std::endl); + << cmXMLSafe(lines[*i]) << std::endl); - ostr << cmCTest::MakeXMLSafe(lines[*i]) << std::endl; + ostr << cmXMLSafe(lines[*i]) << std::endl; if(!unlimitedOutput && totalOutputSize > static_cast(this->CustomMaximumFailedTestOutputSize)) { diff --git a/Source/CTest/cmCTestTestHandler.cxx b/Source/CTest/cmCTestTestHandler.cxx index 5e46c69..cb2c1aa 100644 --- a/Source/CTest/cmCTestTestHandler.cxx +++ b/Source/CTest/cmCTestTestHandler.cxx @@ -28,6 +28,7 @@ #include "cmLocalGenerator.h" #include "cmCommand.h" #include "cmSystemTools.h" +#include "cmXMLSafe.h" #include #include @@ -1464,7 +1465,7 @@ void cmCTestTestHandler::GenerateDartOutput(std::ostream& os) { cmCTestTestResult *result = &this->TestResults[cc]; std::string testPath = result->Path + "/" + result->Name; - os << "\t\t" << cmCTest::MakeXMLSafe( + os << "\t\t" << cmXMLSafe( this->CTest->GetShortPathToFile(testPath.c_str())) << "" << std::endl; } @@ -1481,7 +1482,7 @@ void cmCTestTestHandler::GenerateDartOutput(std::ostream& os) { os << "\t\t\t" - << cmCTest::MakeXMLSafe(this->GetTestStatus(result->Status)) + << cmXMLSafe(this->GetTestStatus(result->Status)) << "" "\n" << "\t\t\t" - << cmCTest::MakeXMLSafe(result->CompletionStatus) + << cmXMLSafe(result->CompletionStatus) << "\n"; } os << "\t\t\t" - << cmCTest::MakeXMLSafe(result->FullCommandLine) + << cmXMLSafe(result->FullCommandLine) << "\n"; std::map::iterator measureIt; for ( measureIt = result->Properties->Measurements.begin(); @@ -1514,13 +1515,13 @@ void cmCTestTestHandler::GenerateDartOutput(std::ostream& os) os << "\t\t\tfirst.c_str() << "\">" - << cmCTest::MakeXMLSafe(measureIt->second.c_str()) + << cmXMLSafe(measureIt->second) << "\n"; } os << "\t\t\t\n" << "\t\t\t\t"; - os << cmCTest::MakeXMLSafe(result->Output); + os << cmXMLSafe(result->Output); os << "\n" << "\t\t\t\n" @@ -1556,13 +1557,13 @@ void cmCTestTestHandler::WriteTestResultHeader(std::ostream& os, } std::string testPath = result->Path + "/" + result->Name; os << "\">\n" - << "\t\t" << cmCTest::MakeXMLSafe(result->Name) << "\n" - << "\t\t" << cmCTest::MakeXMLSafe( + << "\t\t" << cmXMLSafe(result->Name) << "\n" + << "\t\t" << cmXMLSafe( this->CTest->GetShortPathToFile(result->Path.c_str())) << "\n" - << "\t\t" << cmCTest::MakeXMLSafe( + << "\t\t" << cmXMLSafe( this->CTest->GetShortPathToFile(testPath.c_str())) << "\n" << "\t\t" - << cmCTest::MakeXMLSafe(result->FullCommandLine) + << cmXMLSafe(result->FullCommandLine) << "\n"; } @@ -1577,7 +1578,7 @@ void cmCTestTestHandler::WriteTestResultFooter(std::ostream& os, for(std::vector::const_iterator li = labels.begin(); li != labels.end(); ++li) { - os << "\t\t\t\n"; + os << "\t\t\t\n"; } os << "\t\t\n"; } diff --git a/Source/CTest/cmCTestUpdateHandler.cxx b/Source/CTest/cmCTestUpdateHandler.cxx index e1eeeb1..bd1aead 100644 --- a/Source/CTest/cmCTestUpdateHandler.cxx +++ b/Source/CTest/cmCTestUpdateHandler.cxx @@ -25,6 +25,7 @@ #include "cmVersion.h" #include "cmGeneratedFileStream.h" #include "cmXMLParser.h" +#include "cmXMLSafe.h" //#include #include @@ -653,9 +654,9 @@ int cmCTestUpdateHandler::ProcessHandler() << this->CTest->GetTestModelString() << "" << std::endl; os << "\t" << start_time << "\n" << "\t" << start_time_time << "\n" - << "\t" << this->CTest->MakeXMLSafe(command) + << "\t" << cmXMLSafe(command) << "\n" - << "\t" << this->CTest->MakeXMLSafe( + << "\t" << cmXMLSafe( cmCTestUpdateHandlerUpdateToString(updateType)) << "\n"; @@ -1009,17 +1010,17 @@ int cmCTestUpdateHandler::ProcessHandler() << path.c_str() << " / " << fname.c_str() << " was updated by " << sauthor1.c_str() << " to revision: " << srevision1.c_str() << " from revision: " << srevision2.c_str() << std::endl); - os << "\t\t" - << cmCTest::MakeXMLSafe(fname) + os << "\t\t" + << cmXMLSafe(fname) << "\n" - << "\t\t" << cmCTest::MakeXMLSafe(path) + << "\t\t" << cmXMLSafe(path) << "\n" - << "\t\t" << cmCTest::MakeXMLSafe(file) << "\n" - << "\t\t" << cmCTest::MakeXMLSafe(sdate1) + << "\t\t" << cmXMLSafe(file) << "\n" + << "\t\t" << cmXMLSafe(sdate1) << "\n" - << "\t\t" << cmCTest::MakeXMLSafe(sauthor1) << "\n" - << "\t\t" << cmCTest::MakeXMLSafe(semail1) << "\n" - << "\t\t" << cmCTest::MakeXMLSafe(comment1) << "\n" + << "\t\t" << cmXMLSafe(sauthor1) << "\n" + << "\t\t" << cmXMLSafe(semail1) << "\n" + << "\t\t" << cmXMLSafe(comment1) << "\n" << "\t\t" << srevision1 << "\n" << "\t\t" << srevision2 << "" << std::endl; @@ -1030,26 +1031,26 @@ int cmCTestUpdateHandler::ProcessHandler() << "\t\t\t" << srevision1 << "\n" << "\t\t\t" << srevision2 << "\n" - << "\t\t\t" << cmCTest::MakeXMLSafe(sauthor1) + << "\t\t\t" << cmXMLSafe(sauthor1) << "\n" - << "\t\t\t" << cmCTest::MakeXMLSafe(sdate1) + << "\t\t\t" << cmXMLSafe(sdate1) << "\n" - << "\t\t\t" << cmCTest::MakeXMLSafe(comment1) + << "\t\t\t" << cmXMLSafe(comment1) << "\n" - << "\t\t\t" << cmCTest::MakeXMLSafe(semail1) + << "\t\t\t" << cmXMLSafe(semail1) << "\n" << "\t\t\n" << "\t\t\n" << "\t\t\t" << srevision2 << "\n" << "\t\t\t" << srevision2 << "\n" - << "\t\t\t" << cmCTest::MakeXMLSafe(sauthor2) + << "\t\t\t" << cmXMLSafe(sauthor2) << "\n" - << "\t\t\t" << cmCTest::MakeXMLSafe(sdate2) + << "\t\t\t" << cmXMLSafe(sdate2) << "\n" - << "\t\t\t" << cmCTest::MakeXMLSafe(comment2) + << "\t\t\t" << cmXMLSafe(comment2) << "\n" - << "\t\t\t" << cmCTest::MakeXMLSafe(semail2) + << "\t\t\t" << cmXMLSafe(semail2) << "\n" << "\t\t" << std::endl; } @@ -1147,7 +1148,7 @@ int cmCTestUpdateHandler::ProcessHandler() if ( updateProducedError ) { os << "Update error: "; - os << this->CTest->MakeXMLSafe(checkoutErrorMessages); + os << cmXMLSafe(checkoutErrorMessages); cmCTestLog(this->CTest, ERROR_MESSAGE, " Update with command: " << command << " failed" << std::endl); } diff --git a/Source/cmCTest.cxx b/Source/cmCTest.cxx index e86fc7a..1b44226 100644 --- a/Source/cmCTest.cxx +++ b/Source/cmCTest.cxx @@ -25,6 +25,7 @@ #include #include "cmDynamicLoader.h" #include "cmGeneratedFileStream.h" +#include "cmXMLSafe.h" #include "cmCTestCommand.h" #include "cmCTestBuildHandler.h" @@ -149,59 +150,7 @@ std::string cmCTest::CurrentTime() strftime(current_time, 1000, "%a %b %d %H:%M:%S %Z %Y", t); } cmCTestLog(this, DEBUG, " Current_Time: " << current_time << std::endl); - return cmCTest::MakeXMLSafe(cmCTest::CleanString(current_time)); -} - - -//---------------------------------------------------------------------- -std::string cmCTest::MakeXMLSafe(const std::string& str) -{ - std::vector result; - result.reserve(500); - const char* pos = str.c_str(); - for ( ;*pos; ++pos) - { - char ch = *pos; - if ( (ch > 126 || ch < 32) && ch != 9 && - ch != 10 && ch != 13 && ch != '\r' ) - { - char buffer[33]; - sprintf(buffer, "<%d>", (int)ch); - //sprintf(buffer, "&#x%0x;", (unsigned int)ch); - result.insert(result.end(), buffer, buffer+strlen(buffer)); - } - else - { - const char* const encodedChars[] = { - "&", - "<", - ">" - }; - switch ( ch ) - { - case '&': - result.insert(result.end(), encodedChars[0], encodedChars[0]+5); - break; - case '<': - result.insert(result.end(), encodedChars[1], encodedChars[1]+4); - break; - case '>': - result.insert(result.end(), encodedChars[2], encodedChars[2]+4); - break; - case '\n': - result.push_back('\n'); - break; - case '\r': break; // Ignore \r - default: - result.push_back(ch); - } - } - } - if ( result.size() == 0 ) - { - return ""; - } - return std::string(&*result.begin(), result.size()); + return cmXMLSafe(cmCTest::CleanString(current_time)).str(); } //---------------------------------------------------------------------- @@ -1386,7 +1335,7 @@ int cmCTest::GenerateCTestNotesOutput(std::ostream& os, { cmCTestLog(this, OUTPUT, "\tAdd file: " << it->c_str() << std::endl); std::string note_time = this->CurrentTime(); - os << "MakeXMLSafe(it->c_str()) << "\">\n" + os << "\n" << "\n" << "" << note_time << "\n" << "" << std::endl; @@ -1396,7 +1345,7 @@ int cmCTest::GenerateCTestNotesOutput(std::ostream& os, std::string line; while ( cmSystemTools::GetLineFromStream(ifs, line) ) { - os << this->MakeXMLSafe(line) << std::endl; + os << cmXMLSafe(line) << std::endl; } ifs.close(); } diff --git a/Source/cmCTest.h b/Source/cmCTest.h index 7c3d41e..4f58b19 100644 --- a/Source/cmCTest.h +++ b/Source/cmCTest.h @@ -200,9 +200,6 @@ public: cmGeneratedFileStream& stream, bool compress = false); - ///! Convert string to something that is XML safe - static std::string MakeXMLSafe(const std::string&); - ///! Should we only show what we would do? bool GetShowOnly(); diff --git a/Source/cmSystemTools.cxx b/Source/cmSystemTools.cxx index 0ef7508..84ed729 100644 --- a/Source/cmSystemTools.cxx +++ b/Source/cmSystemTools.cxx @@ -1644,56 +1644,6 @@ void cmSystemTools::EnableVSConsoleOutput() #endif } -std::string cmSystemTools::MakeXMLSafe(const char* str) -{ - std::vector result; - result.reserve(500); - const char* pos = str; - for ( ;*pos; ++pos) - { - char ch = *pos; - if ( (ch > 126 || ch < 32) && ch != 9 && ch != 10 && ch != 13 - && ch != '\r' ) - { - char buffer[33]; - sprintf(buffer, "<%d>", static_cast(ch)); - //sprintf(buffer, "&#x%0x;", (unsigned int)ch); - result.insert(result.end(), buffer, buffer+strlen(buffer)); - } - else - { - const char* const encodedChars[] = { - "&", - "<", - ">" - }; - switch ( ch ) - { - case '&': - result.insert(result.end(), encodedChars[0], encodedChars[0]+5); - break; - case '<': - result.insert(result.end(), encodedChars[1], encodedChars[1]+4); - break; - case '>': - result.insert(result.end(), encodedChars[2], encodedChars[2]+4); - break; - case '\n': - result.push_back('\n'); - break; - case '\r': break; // Ignore \r - default: - result.push_back(ch); - } - } - } - if ( result.size() == 0 ) - { - return ""; - } - return std::string(&*result.begin(), result.size()); -} - bool cmSystemTools::IsPathToFramework(const char* path) { if(cmSystemTools::FileIsFullPath(path)) diff --git a/Source/cmSystemTools.h b/Source/cmSystemTools.h index 3726c44..8e639ac 100644 --- a/Source/cmSystemTools.h +++ b/Source/cmSystemTools.h @@ -351,9 +351,6 @@ public: /** Setup the environment to enable VS 8 IDE output. */ static void EnableVSConsoleOutput(); - /** Make string XML safe */ - static std::string MakeXMLSafe(const char* str); - /** Create tar */ static bool ListTar(const char* outFileName, std::vector& files, diff --git a/Source/cmXMLSafe.cxx b/Source/cmXMLSafe.cxx new file mode 100644 index 0000000..861d75c --- /dev/null +++ b/Source/cmXMLSafe.cxx @@ -0,0 +1,93 @@ +/*========================================================================= + + Program: CMake - Cross-Platform Makefile Generator + Module: $RCSfile$ + Language: C++ + Date: $Date$ + Version: $Revision$ + + Copyright (c) 2002 Kitware, Inc., Insight Consortium. All rights reserved. + See Copyright.txt or http://www.cmake.org/HTML/Copyright.html for details. + + This software is distributed WITHOUT ANY WARRANTY; without even + the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR + PURPOSE. See the above copyright notices for more information. + +=========================================================================*/ +#include "cmXMLSafe.h" + +#include +#include + +#include +#include + +//---------------------------------------------------------------------------- +cmXMLSafe::cmXMLSafe(const char* s): + Data(s), + Size(static_cast(strlen(s))), + DoQuotes(true) +{ +} + +//---------------------------------------------------------------------------- +cmXMLSafe::cmXMLSafe(cmsys_stl::string const& str): + Data(str.c_str()), + Size(static_cast(str.length())), + DoQuotes(true) +{ +} + +//---------------------------------------------------------------------------- +cmXMLSafe& cmXMLSafe::Quotes(bool b) +{ + this->DoQuotes = b; + return *this; +} + +//---------------------------------------------------------------------------- +cmsys_stl::string cmXMLSafe::str() +{ + cmsys_ios::ostringstream ss; + ss << *this; + return ss.str(); +} + +//---------------------------------------------------------------------------- +cmsys_ios::ostream& operator<<(cmsys_ios::ostream& os, cmXMLSafe const& self) +{ + char const* first = self.Data; + char const* last = self.Data + self.Size; + for(char const* ci = first; ci != last; ++ci) + { + char c = *ci; + switch(c) + { + case '&': os << "&"; break; + case '<': os << "<"; break; + case '>': os << ">"; break; + case '"': os << (self.DoQuotes? """ : "\""); break; + case '\'': os << (self.DoQuotes? "'" : "'"); break; + case '\t': os << "\t"; break; + case '\n': os << "\n"; break; + case '\r': break; // Ignore CR + default: + if(c >= 0x20 && c <= 0x7f) + { + os.put(c); + } + else + { + // TODO: More complete treatment of program output character + // encoding. Instead of escaping these bytes, we should + // handle the current locale and its encoding. + unsigned char uc = static_cast(c); + char buf[16]; + sprintf(buf, "&#x%hx;", static_cast(uc)); + os << buf; + } + break; + } + } + return os; +} diff --git a/Source/cmXMLSafe.h b/Source/cmXMLSafe.h new file mode 100644 index 0000000..6352ad6 --- /dev/null +++ b/Source/cmXMLSafe.h @@ -0,0 +1,49 @@ +/*========================================================================= + + Program: CMake - Cross-Platform Makefile Generator + Module: $RCSfile$ + Language: C++ + Date: $Date$ + Version: $Revision$ + + Copyright (c) 2002 Kitware, Inc., Insight Consortium. All rights reserved. + See Copyright.txt or http://www.cmake.org/HTML/Copyright.html for details. + + This software is distributed WITHOUT ANY WARRANTY; without even + the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR + PURPOSE. See the above copyright notices for more information. + +=========================================================================*/ +#ifndef cmXMLSafe_h +#define cmXMLSafe_h + +#include +#include + +/** \class cmXMLSafe + * \brief Write strings to XML with proper escapes + */ +class cmXMLSafe +{ +public: + /** Construct with the data to be written. This assumes the data + will exist for the duration of this object's life. */ + cmXMLSafe(const char* s); + cmXMLSafe(cmsys_stl::string const& str); + + /** Specify whether to escape quotes too. This is needed when + writing the content of an attribute value. By default quotes + are escaped. */ + cmXMLSafe& Quotes(bool b = true); + + /** Get the escaped data as a string. */ + cmsys_stl::string str(); +private: + char const* Data; + unsigned long Size; + bool DoQuotes; + friend cmsys_ios::ostream& operator<<(cmsys_ios::ostream&, + cmXMLSafe const&); +}; + +#endif -- cgit v0.12