arabica/include/XSLT/impl/handler/xslt_include_handler.hpp

209 lines
5.9 KiB
C++
Raw Normal View History

2007-07-19 19:01:42 +02:00
#ifndef ARABICA_XSLT_INCLUDE_HANDLER_HPP
#define ARABICA_XSLT_INCLUDE_HANDLER_HPP
#include <algorithm>
#include "xslt_constants.hpp"
#include "xslt_value_validation.hpp"
namespace Arabica
{
namespace XSLT
{
class IncludeHandler : public SAX::DefaultHandler<std::string>
2007-07-19 19:01:42 +02:00
{
public:
IncludeHandler() :
context_(0),
compiler_(0),
no_content_(false)
2007-07-19 19:01:42 +02:00
{
} // IncludeHandler
void context(CompilationContext& context, SAX::DefaultHandler<std::string>* compiler)
2007-07-19 19:01:42 +02:00
{
context_ = &context;
compiler_ = compiler;
} // context
void start_include(const std::string& namespaceURI,
const std::string& localName,
const std::string& qName,
const SAX::Attributes<std::string>& atts)
2007-07-19 19:01:42 +02:00
{
context_->parser().setContentHandler(*this);
startElement(namespaceURI, localName, qName, atts);
} // start_include
virtual void startDocument()
{
context_->parentHandler().startDocument();
} // startDocument
2007-07-19 19:01:42 +02:00
virtual void startElement(const std::string& namespaceURI,
const std::string& localName,
const std::string& qName,
const SAX::Attributes<std::string>& atts)
2007-07-19 19:01:42 +02:00
{
if(no_content_)
throw SAX::SAXException("xsl:include must be empty");
bool start_pass_through = false;
if(namespaceURI == StylesheetConstant::NamespaceURI())
{
if(localName == "import")
{
std::string href = validate_href(qName, atts);
2008-12-29 22:14:44 +01:00
import_stack_.push_back(href, context_->next_precedence());
2007-07-19 19:01:42 +02:00
return;
} // if(localName == "import")
if(localName == "include")
{
href_.push_back(validate_href(qName, atts));
return;
} // if(localName == "include")
} // if ...
context_->parentHandler().startElement(namespaceURI,
localName,
qName,
atts);
2007-07-19 19:01:42 +02:00
} // startElement
virtual void startPrefixMapping(const std::string& prefix, const std::string& uri)
{
context_->parentHandler().startPrefixMapping(prefix, uri);
} // startPrefixMapping
2007-07-19 19:01:42 +02:00
virtual void endElement(const std::string& namespaceURI,
const std::string& localName,
const std::string& qName)
{
if(no_content_ &&
(namespaceURI == StylesheetConstant::NamespaceURI()))
{
no_content_ = false;
if(localName == "include")
{
include_stylesheet(href_.back(), context_->precedence());
2007-07-19 19:01:42 +02:00
href_.pop_back();
} // if ...
if(href_.empty())
context_->parser().setContentHandler(*compiler_);
return;
} // if ...
context_->parentHandler().endElement(namespaceURI,
localName,
qName);
2007-07-19 19:01:42 +02:00
} // endElement
virtual void characters(const std::string& ch)
{
if(no_content_)
throw SAX::SAXException("xsl:include/xsl:import must be empty");
context_->parentHandler().characters(ch);
} // characters
void unwind_imports()
{
while(!import_stack_.empty())
{
2008-12-29 22:14:44 +01:00
include_stylesheet(import_stack_.current().first, import_stack_.current().second);
import_stack_.pop();
2007-07-19 19:01:42 +02:00
} // while ...
} // unwind_imports
private:
std::string validate_href(const std::string& qName, const SAX::Attributes<std::string>& atts)
2007-07-19 19:01:42 +02:00
{
static const ValueRule rules[] = { { "href", true, 0 },
{ 0, false, 0 } };
std::string href = gatherAttributes(qName, atts, rules)["href"];
no_content_ = true;
return context_->makeAbsolute(href);
} // validate_href
2008-12-29 22:14:44 +01:00
void check_for_loops(const std::string& href)
2007-07-19 19:01:42 +02:00
{
2008-12-29 22:14:44 +01:00
if(std::find(current_includes_.begin(), current_includes_.end(), href) != current_includes_.end())
2007-07-19 19:01:42 +02:00
{
std::string error = "Stylesheet '" + href + "' includes/imports itself ";
2008-12-29 22:14:44 +01:00
for(std::vector<std::string>::const_iterator i = current_includes_.begin(), ie = current_includes_.end(); i != ie; ++i)
2007-07-19 19:01:42 +02:00
error += "\n " + *i;
throw std::runtime_error(error);
} // if ...
} // check_for_loops
2008-12-29 22:14:44 +01:00
void include_stylesheet(const std::string& href, const Precedence& precedence)
2007-07-19 19:01:42 +02:00
{
2008-12-29 22:14:44 +01:00
check_for_loops(href);
2007-07-19 19:01:42 +02:00
current_includes_.push_back(href);
std::string prev = context_->setBase(href);
context_->set_precedence(precedence);
SAX::InputSource<std::string> source(href);
2007-07-19 19:01:42 +02:00
SAX::XMLReader<std::string> include_parser;
SAX::CatchErrorHandler<std::string> errorHandler;
2007-07-19 19:01:42 +02:00
include_parser.setContentHandler(*this);
include_parser.setErrorHandler(errorHandler);
2007-07-19 19:01:42 +02:00
include_parser.parse(source);
context_->setBase(prev);
2007-07-19 19:01:42 +02:00
if(errorHandler.errorsReported())
throw std::runtime_error("Could not import/include stylesheet '" + href + "' - " + errorHandler.errors());
current_includes_.pop_back();
} // include_stylesheet
SAX::DefaultHandler<std::string>* compiler_;
2007-07-19 19:01:42 +02:00
CompilationContext* context_;
bool no_content_;
2008-12-29 22:14:44 +01:00
class ImportStack
{
public:
typedef std::pair<std::string, Precedence> ImportHref;
ImportStack() { }
void push_back(const std::string href, const Precedence& precedence)
{
stack_.push_back(std::make_pair<std::string, Precedence>(href, precedence));
} // push_back
bool empty() const { return stack_.empty(); }
const ImportHref& current() const
{
most_recent_ = stack_.size() - 1;
return stack_[most_recent_];
} // current
void pop()
{
stack_.erase(stack_.begin() + most_recent_);
} // pop
private:
std::vector<ImportHref> stack_;
mutable size_t most_recent_;
}; // class ImportStack
ImportStack import_stack_;
2007-07-19 19:01:42 +02:00
std::vector<std::string> current_includes_;
std::vector<std::string> href_;
}; // class IncludeHandler
} // namespace XSLT
} // namespace Arabica
#endif