#ifndef XSLT_XSLT_TEST_HPP #define XSLT_XSLT_TEST_HPP #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 DOM::Document buildDOM(const std::string& filename) { Arabica::SAX::InputSource is(filename); SAX2DOM::Parser parser; parser.parse(is); DOM::Document d = parser.getDocument(); if(d != 0) d.normalize(); return d; } // buildDOM Arabica::XPath::NodeSet selectNodes(const std::string& path, const DOM::Node& node) { Arabica::XPath::XPath xpath; return xpath.evaluate(path, node)->asNodeSet(); } // selectNodes DOM::Node selectNode(const std::string& path, const DOM::Node& node) { return selectNodes(path, node)[0]; } // selectNode std::string selectString(const std::string& path, const 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) : TestCase(name), input_xslt_(input_xslt) { } // 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(compiler.error()); } // runTest private: std::string input_xslt_; }; // CompileFailsTest class RunFailsTest : public TestCase { public: RunFailsTest(const std::string& name, const std::string& input_xml, const std::string& input_xslt) : TestCase(name), input_xml_(input_xml), input_xslt_(input_xslt) { } // 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); DOM::Document document = buildDOM(input_xml_); try { stylesheet->execute(document); } catch(const std::exception& e) { throw SkipException(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_; }; // RunFailsTest 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); DOM::Document document = buildDOM(input_xml_); try { stylesheet->execute(document); } catch(const std::exception& e) { assertImplementation(false, "Failed to run " + input_xslt_ + " : " + e.what()); } DOM::Document ref = buildDOM(output_xml_); if(ref == 0) assertImplementation(false, "Couldn't read " + output_xml_ + ". Perhaps it isn't well-formed XML?"); DOM::Node out = output.node(); std::string refs = docToString(ref.getFirstChild()); std::string outs = docToString(out.getFirstChild()); if(refs != outs) { stripWhitespace(ref); stripWhitespace(out); std::string refs2 = docToString(ref.getFirstChild()); std::string outs2 = docToString(out.getFirstChild()); if(refs2 != outs2) assertImplementation(false, "Expected:\n" + refs + "\nbut got:\n" + outs); } } // runTest std::string docToString(DOM::Node node) { std::ostringstream ss; ss << node; return Arabica::string::normalize_whitespace >(ss.str()); } // docToString void stripWhitespace(DOM::Node doc) { Arabica::XPath::NodeSet textNodes = selectNodes("//text()", doc); for(int i = 0; i != textNodes.size(); ++i) { DOM::Node t = textNodes[i]; std::string text = t.getNodeValue(); size_t index = text.find_first_of(" \n"); while(index != std::string::npos) { text.replace(index, 1, ""); index = text.find_first_of(" \n"); } t.setNodeValue(text); } } // stripWhitespace std::string input_xml_; std::string input_xslt_; std::string output_xml_; }; // class StandardXSLTTest class Expected { public: Expected() { 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]); if(compiles == "no") fails[name] = "compile"; else if(runs == "no") fails[name] = "run"; else if(skip == "yes") { fails[name] = "skip"; skips[name] = reason; } } // for ... } std::map& Fails() { return fails; } std::map& Skips() { return skips; } private: std::map fails; std::map skips; }; static Expected expected; TestSuite* XSLTTest_suite(const std::string& path) { static 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))); else if(expected.Fails()[name] == "run") suiteOfTests->addTest(new RunFailsTest(name, make_path(path, input_xml), make_path(path, input_xslt))); else if(expected.Fails()[name] == "skip") suiteOfTests->addTest(new SkipTest(name, expected.Skips()[name])); } // for ... return suiteOfTests; } // XSLTTest_suitoo #endif