#ifndef ARABICA_XSLT_STYLESHEETCONSTANT_HPP #define ARABICA_XSLT_STYLESHEETCONSTANT_HPP #include #include #include #include "xslt_stylesheet_parser.hpp" #include "xslt_compiled_stylesheet.hpp" #include "xslt_compilation_context.hpp" #include "handler/xslt_template_handler.hpp" #include "handler/xslt_include_handler.hpp" #include "handler/xslt_output_handler.hpp" #include "handler/xslt_namespace_alias_handler.hpp" #include "handler/xslt_key_handler.hpp" #include "handler/xslt_foreign_element_handler.hpp" namespace Arabica { namespace XSLT { class StylesheetHandler : public SAX::DefaultHandler { public: StylesheetHandler(CompilationContext& context) : context_(context), top_(false), foreign_(0) { context_.root(*this); includer_.context(context_, this); } // StylesheetHandler virtual void startDocument() { top_ = true; } // startDocument virtual void startElement(const std::string& namespaceURI, const std::string& localName, const std::string& qName, const SAX::Attributes& atts) { if(top_) { startStylesheet(namespaceURI, localName, qName, atts); return; } // if(top_) if(namespaceURI == StylesheetConstant::NamespaceURI()) startXSLTElement(namespaceURI, localName, qName, atts); else if(!namespaceURI.empty()) startForeignElement(namespaceURI, localName, qName, atts); else oops(qName); } // startElement virtual void endElement(const std::string& namespaceURI, const std::string& localName, const std::string& qName) { if(foreign_) --foreign_; } // endElement virtual void characters(const std::string& ch) { if(foreign_) return; for(std::string::const_iterator s = ch.begin(), e = ch.end(); s != e; ++s) if(!Arabica::XML::is_space(*s)) throw SAX::SAXException("stylesheet element can not contain character data :'" + ch +"'"); } // characters virtual void endDocument() { includer_.unwind_imports(); context_.stylesheet().prepare(); } // endDocument private: void startStylesheet(const std::string& namespaceURI, const std::string& localName, const std::string& qName, const SAX::Attributes& atts) { if(namespaceURI != StylesheetConstant::NamespaceURI()) throw SAX::SAXException("The source file does not look like a stylesheet."); if(localName != "stylesheet" && localName != "transform") throw SAX::SAXException("Top-level element must be 'stylesheet' or 'transform'."); static const ValueRule rules[] = { { "version", true, 0 }, { "extension-element-prefixes", false, 0 }, { "exclude-result-prefixes", false, 0 }, { "id", false, 0 }, { 0, false, 0 } }; std::map attributes = gatherAttributes(qName, atts, rules); if(attributes["version"] != StylesheetConstant::Version()) throw SAX::SAXException("I'm only a poor version 1.0 XSLT Transformer."); if(!attributes["extension-element-prefixes"].empty()) throw SAX::SAXException("Haven't implemented extension-element-prefixes yet"); top_ = false; } // startStylesheet void startXSLTElement(const std::string& namespaceURI, const std::string& localName, const std::string& qName, const SAX::Attributes& atts) { if((localName == "import") || (localName == "include")) { include_stylesheet(namespaceURI, localName, qName, atts); return; } // if ... for(const ChildElement* c = allowedChildren; c->name != 0; ++c) if(c->name == localName) { context_.push(0, c->createHandler(context_), namespaceURI, localName, qName, atts); return; } // if ... oops(qName); } // startXSLTElement void startForeignElement(const std::string& namespaceURI, const std::string& localName, const std::string& qName, const SAX::Attributes& atts) { context_.push(0, new ForeignElementHandler(context_), namespaceURI, localName, qName, atts); } // startForeignElement void include_stylesheet(const std::string& namespaceURI, const std::string& localName, const std::string& qName, const SAX::Attributes& atts) { includer_.start_include(namespaceURI, localName, qName, atts); } // include_stylesheet void oops(const std::string& qName) const { throw SAX::SAXException("xsl:stylesheet does not allow " + qName + " here."); } // oops CompilationContext& context_; SAX::DefaultHandler* child_; IncludeHandler includer_; bool top_; unsigned int foreign_; static const ChildElement allowedChildren[]; }; // class StylesheetHandler const ChildElement StylesheetHandler::allowedChildren[] = { { "attribute-set", CreateHandler}, { "decimal-format", CreateHandler}, //"import" //"include" { "key", CreateHandler}, { "namespace-alias", CreateHandler}, { "output", CreateHandler}, { "param", CreateHandler >}, { "preserve-space", CreateHandler}, { "strip-space", CreateHandler}, { "template", CreateHandler }, { "variable", CreateHandler > }, { 0, 0 } }; // StylesheetHandler::allowedChildren class StylesheetCompiler { public: StylesheetCompiler() { } // StylesheetCompiler ~StylesheetCompiler() { } // ~StylesheetCompiler std::auto_ptr compile(SAX::InputSource& source) { error_ = ""; std::auto_ptr stylesheet(new CompiledStylesheet()); StylesheetParser parser; CompilationContext context(parser, *stylesheet.get()); StylesheetHandler stylesheetHandler(context); parser.setContentHandler(stylesheetHandler); //parser.setErrorHandler(*this); //if(entityResolver_) // parser.setEntityResolver(*entityResolver_); try { parser.parse(source); } // try catch(std::exception& ex) { error_ = ex.what(); //std::cerr << "Compilation Failed : " << ex.what() << std::endl; stylesheet.reset(); } // catch return std::auto_ptr(stylesheet.release()); } // compile const std::string& error() const { return error_; } // error private: std::string error_; }; // class StylesheetCompiler } // namespace XSLT } // namespace Arabica #endif // ARABICA_XSLT_STYLESHEETCOMPILER_HPP