#ifndef XSLT_XSLT_TEST_HPP #define XSLT_XSLT_TEST_HPP #include #include #include #include #include //#include //#include //#include //#include #ifdef ARABICA_WINDOWS const std::string PATH_PREFIX="../tests/XSLT/testsuite/TESTS/"; const std::string SEPERATOR = "/"; #else #include "test_path.hpp" const std::string PATH_PREFIX=test_path; const std::string SEPERATOR = "/"; #endif Arabica::DOM::Document buildDOM(const std::string& filename) { Arabica::SAX::InputSource is(filename); Arabica::SAX2DOM::Parser parser; parser.parse(is); Arabica::DOM::Document d = parser.getDocument(); if(d != 0) d.normalize(); return d; } // buildDOM std::string readFile(const std::string& filename) { std::ifstream in(filename.c_str()); return std::string(std::istreambuf_iterator(in), std::istreambuf_iterator()); } // readFile Arabica::XPath::NodeSet selectNodes(const std::string& path, const Arabica::DOM::Node& node) { Arabica::XPath::XPath xpath; return xpath.evaluate(path, node).asNodeSet(); } // selectNodes Arabica::DOM::Node selectNode(const std::string& path, const Arabica::DOM::Node& node) { return selectNodes(path, node)[0]; } // selectNode std::string selectString(const std::string& path, const Arabica::DOM::Node& node) { Arabica::XPath::XPath xpath; return xpath.evaluate_expr(path, node).asString(); } // selectString std::string make_path(const std::string& path, const std::string& filename) { return PATH_PREFIX + path + SEPERATOR + filename; } // make_path class SkipTest : public TestCase { public: SkipTest(const std::string& name, const std::string& reason) : TestCase(name), reason_(reason) { } // SkipTest protected: virtual void runTest() { throw SkipException(reason_); } // runTest private: std::string reason_; }; // class SkipTest class CompileFailsTest : public TestCase { public: CompileFailsTest(const std::string& name, const std::string& input_xslt, const std::string& reason) : TestCase(name), input_xslt_(input_xslt), reason_(reason) { } // CompileFailsTest protected: virtual void runTest() { Arabica::XSLT::StylesheetCompiler compiler; Arabica::SAX::InputSource source(input_xslt_); std::auto_ptr stylesheet = compiler.compile(source); if(stylesheet.get() != 0) assertImplementation(false, "Expected " + input_xslt_ + " not to compile. But it did :o"); throw SkipException(reason_ + " : " + compiler.error()); } // runTest private: std::string input_xslt_; std::string reason_; }; // CompileFailsTest class RunFailsTest : public TestCase { public: RunFailsTest(const std::string& name, const std::string& input_xml, const std::string& input_xslt, const std::string& reason) : TestCase(name), input_xml_(input_xml), input_xslt_(input_xslt), reason_(reason) { } // RunFailsTest protected: virtual void runTest() { Arabica::XSLT::StylesheetCompiler compiler; Arabica::SAX::InputSource source(input_xslt_); std::auto_ptr stylesheet = compiler.compile(source); if(stylesheet.get() == 0) assertImplementation(false, "Failed to compile " + input_xslt_ + " : " + compiler.error()); Arabica::XSLT::DOMSink output; stylesheet->set_output(output); std::ostringstream errors; stylesheet->set_error_output(errors); Arabica::DOM::Document document = buildDOM(input_xml_); try { stylesheet->execute(document); } catch(const std::exception& e) { throw SkipException(reason_ + " : " + e.what()); } assertImplementation(false, "Expected " + input_xslt_ + " to fail to execute. But it did :o"); } // runTest private: std::string input_xml_; std::string input_xslt_; std::string reason_; }; // RunFailsTest class CompareAsTextXSLTTest : public TestCase { public: CompareAsTextXSLTTest(const std::string& name, const std::string& input_xml, const std::string& input_xslt, const std::string& output_xml) : TestCase(name), input_xml_(input_xml), input_xslt_(input_xslt), output_xml_(output_xml) { } // CompareAsTextXSLTTest protected: virtual void runTest() { Arabica::XSLT::StylesheetCompiler compiler; Arabica::SAX::InputSource source(input_xslt_); std::auto_ptr stylesheet = compiler.compile(source); if(stylesheet.get() == 0) assertImplementation(false, "Failed to compile " + input_xslt_ + " : " + compiler.error()); std::ostringstream xml_output; Arabica::XSLT::StreamSink output(xml_output); stylesheet->set_output(output); std::ostringstream errors; stylesheet->set_error_output(errors); Arabica::DOM::Document document = buildDOM(input_xml_); try { stylesheet->execute(document); } catch(const std::exception& e) { assertImplementation(false, "Failed to run " + input_xslt_ + " : " + e.what()); } // catch std::string ref = readFile(output_xml_); std::string out = xml_output.str(); if(ref == out) return; std::string refs = trimXMLDecl(ref); std::string outs = trimXMLDecl(out); if(refs == outs) return; refs = stripWhitespace(refs); outs = stripWhitespace(outs); if(refs == outs) return; assertImplementation(false, "Expected text:\n" + ref + "\nbut got:\n" + out + "\n" + refs + "\nbut got:" + outs + "\n====="); } // runTest private: std::string stripWhitespace(const std::string& str) { std::string s = Arabica::text::normalize_whitespace >(str); std::string::size_type i = s.find("> "); while(i != std::string::npos) { s.erase(i+1, 1); i = s.find("> "); } // while .. i = s.find(" <"); while(i != std::string::npos) { s.erase(i, 1); i = s.find(" <"); } // while .. return s; } // stripWhitespace std::string trimXMLDecl(const std::string& str) { if(str.find("") + 2); } // trimXMLDecl std::string input_xml_; std::string input_xslt_; std::string output_xml_; }; // class CompareAsTextXSLTTest class StandardXSLTTest : public TestCase { public: StandardXSLTTest(const std::string& name, const std::string& input_xml, const std::string& input_xslt, const std::string& output_xml) : TestCase(name), input_xml_(input_xml), input_xslt_(input_xslt), output_xml_(output_xml) { } // StandardXSLTTest protected: virtual void runTest() { Arabica::XSLT::StylesheetCompiler compiler; Arabica::SAX::InputSource source(input_xslt_); std::auto_ptr stylesheet = compiler.compile(source); if(stylesheet.get() == 0) assertImplementation(false, "Failed to compile " + input_xslt_ + " : " + compiler.error()); Arabica::XSLT::DOMSink output; stylesheet->set_output(output); std::ostringstream errors; stylesheet->set_error_output(errors); Arabica::DOM::Document document = buildDOM(input_xml_); try { stylesheet->execute(document); } catch(const std::exception& e) { assertImplementation(false, "Failed to run " + input_xslt_ + " : " + e.what()); } Arabica::DOM::Document ref = buildDOM(output_xml_); if(ref == 0) assertImplementation(false, "Couldn't read " + output_xml_ + ". Perhaps it isn't well-formed XML?"); Arabica::DOM::Node out = output.node(); out.normalize(); std::string refs = docToString(ref.getFirstChild()); std::string outs = docToString(out.getFirstChild()); if(refs == outs) return; stripWhitespace(ref); stripWhitespace(out); std::string refs2 = docToString(ref.getFirstChild()); std::string outs2 = docToString(out.getFirstChild()); if(refs2 == outs2) return; assertImplementation(false, "Expected:\n" + refs + "\nbut got:\n" + outs); } // runTest std::string docToString(Arabica::DOM::Node node) { std::ostringstream ss; ss << node; return Arabica::text::normalize_whitespace >(ss.str()); } // docToString void stripWhitespace(Arabica::DOM::Node doc) { Arabica::XPath::NodeSet textNodes = selectNodes("//text()", doc); for(int i = 0; i != textNodes.size(); ++i) { Arabica::DOM::Node t = textNodes[i]; std::string text = t.getNodeValue(); text = Arabica::text::normalize_whitespace >(text); size_t index = text.find_first_of(" "); while(index != std::string::npos) { text.replace(index, 1, ""); index = text.find_first_of(" "); } t.setNodeValue(text); } } // stripWhitespace std::string input_xml_; std::string input_xslt_; std::string output_xml_; }; // class StandardXSLTTest class Expected { public: Expected() { Arabica::DOM::Document fail_doc = buildDOM(PATH_PREFIX + "arabica-expected-fails.xml"); Arabica::XPath::NodeSet failcases = selectNodes("/test-suite/test-case", fail_doc); for(int i = 0; i != failcases.size(); ++i) { std::string name = selectString("@id", failcases[i]); std::string compiles = selectString("@compiles", failcases[i]); std::string runs = selectString("@runs", failcases[i]); std::string skip = selectString("@skip", failcases[i]); std::string reason = selectString("@reason", failcases[i]); std::string compare = selectString("@compare", failcases[i]); if(compiles == "no") fails[name] = "compile"; else if(runs == "no") fails[name] = "run"; else if(skip == "yes") fails[name] = "skip"; else if(compare == "text") fails[name] = "text"; reasons[name] = reason; } // for ... } std::map& Fails() { return fails; } std::map& Reasons() { return reasons; } private: std::map fails; std::map reasons; }; static Expected expected; TestSuite* XSLTTest_suite(const std::string& path) { //#define new ELEPHANTNEW // using namespace elephant; // LeakDetector leakDetector; // MemoryMonitorHolder().Instance().AddObserver(&leakDetector); // // poo(); // // MemoryMonitorHolder().Instance().RemoveObserver(&leakDetector); // LeakDisplayFunc leakDisplay(std::cout); // std::for_each(leakDetector.begin(), leakDetector.end(), leakDisplay); //#undef new static Arabica::DOM::Document catalog = buildDOM(PATH_PREFIX + "catalog.xml"); TestSuite *suiteOfTests = new TestSuite; Arabica::XPath::NodeSet tests = // selectNodes("/test-suite/test-catalog/test-case[scenario/@operation='standard']", catalog); // selectNodes("/test-suite/test-catalog/test-case[@id='boolean_boolean86']", catalog); selectNodes("/test-suite/test-catalog/test-case[scenario/@operation='standard' and file-path='" + path + "']", catalog); for(int i = 0; i != tests.size(); ++i) { std::string name = selectString("@id", tests[i]); std::string path = selectString("concat(../major-path, '" + SEPERATOR + "', file-path)", tests[i]); std::string out_path = selectString("concat(../major-path, '" + SEPERATOR + "REF_OUT" + SEPERATOR + "', file-path)", tests[i]); std::string input_xml = selectString(".//input-file[@role='principal-data']", tests[i]); std::string input_xslt = selectString(".//input-file[@role='principal-stylesheet']", tests[i]); std::string output_xml = selectString(".//output-file[@role='principal']", tests[i]); if(expected.Fails().find(name) == expected.Fails().end()) suiteOfTests->addTest(new StandardXSLTTest(name, make_path(path, input_xml), make_path(path, input_xslt), make_path(out_path, output_xml))); else if(expected.Fails()[name] == "compile") suiteOfTests->addTest(new CompileFailsTest(name, make_path(path, input_xslt), expected.Reasons()[name])); else if(expected.Fails()[name] == "run") suiteOfTests->addTest(new RunFailsTest(name, make_path(path, input_xml), make_path(path, input_xslt), expected.Reasons()[name])); else if(expected.Fails()[name] == "skip") suiteOfTests->addTest(new SkipTest(name, expected.Reasons()[name])); else if(expected.Fails()[name] == "text") suiteOfTests->addTest(new CompareAsTextXSLTTest(name, make_path(path, input_xml), make_path(path, input_xslt), make_path(out_path, output_xml))); } // for ... return suiteOfTests; } // XSLTTest_suitoo #endif