#ifndef ARABICA_XSLT_VARIABLE_STACK_HPP #define ARABICA_XSLT_VARIABLE_STACK_HPP #include #include #include #include "xslt_precedence.hpp" namespace Arabica { namespace XSLT { class Variable_instance; typedef boost::shared_ptr Variable_instance_ptr; typedef std::map Scope; class Variable_instance { public: Variable_instance() { } virtual ~Variable_instance() { } virtual const std::string& namespace_uri() const = 0; virtual const std::string& name() const = 0; virtual const Precedence& precedence() const = 0; virtual Arabica::XPath::XPathValue value() const = 0; virtual void injectGlobalScope(const Scope& scope) const = 0; private: Variable_instance(const Variable_instance&); Variable_instance& operator=(const Variable_instance&); bool operator==(const Variable_instance&) const; }; // Variable_instance class VariableStack : public Arabica::XPath::VariableResolver { public: VariableStack() { stack_.push_back(Scope()); params_.push_back(Scope()); params_.push_back(Scope()); } // VariableStack VariableStack(const VariableStack& rhs) : stack_(rhs.stack_), params_(rhs.params_) { } // VariableStack void pushScope() { stack_.push_back(Scope()); params_.push_back(Scope()); } // pushScope void chainScope() { stack_.push_back(Scope(stack_.back())); params_.push_back(Scope(params_.back())); } // chainsScope void popScope() { params_.pop_back(); stack_.pop_back(); } // popScope void topLevelParam(Variable_instance_ptr param) { params_.front()[clarkName(param)] = param; } // topLevelParam std::string passParam(Variable_instance_ptr param) { std::string clark_name = clarkName(param); Scope& params = params_.back(); if(params.find(clark_name) != params.end()) throw std::runtime_error("Duplicate parameter name in xsl:with-param - " + clark_name); params[clark_name] = param; return clark_name; } // passParam void unpassParam(const std::string& name) { params_.back().erase(name); } // unpassParam void declareParam(Variable_instance_ptr param) { ScopeStack::reverse_iterator p = params_.rbegin()+1; Scope::iterator i = p->find(clarkName(param)); if(i == p->end()) declareVariable(param); else declareVariable(i->second); } // declareParam void declareVariable(Variable_instance_ptr var) { std::string name = clarkName(var); Scope& stack = stack_.back(); if(stack.find(name) != stack.end()) { const Precedence& current_p = stack[name]->precedence(); if(var->precedence() == current_p) throw std::runtime_error("Duplicate variable name : " + clarkName(var)); if(current_p.is_descendant(var->precedence())) return; if(current_p > var->precedence()) return; } // if ... if(var->precedence() == Precedence::FrozenPrecedence()) // we're running, so resolve immediately var->value(); stack[name] = var; } // declareVariable void freezeTopLevel() { const Scope& top = stack_.front(); for(Scope::const_iterator v = top.begin(), ve = top.end(); v != ve; ++v) v->second->injectGlobalScope(top); for(Scope::const_iterator v = top.begin(), ve = top.end(); v != ve; ++v) lookup(top, v->first); } // freezeTopLevel void injectGlobalScope(const Scope& scope) { stack_.front() = scope; } // injectGlobalScope virtual Arabica::XPath::XPathValue resolveVariable(const std::string& namespace_uri, const std::string& name) const { std::string clarkName = "{" + namespace_uri + "}" + name; if(std::find(resolutionStack_.begin(), resolutionStack_.end(), clarkName) != resolutionStack_.end()) throw std::runtime_error("Circular dependency: " + clarkName + " refers to itself directly or indirectly."); resolutionStack_.push_back(clarkName); Arabica::XPath::XPathValue val = lookup(stack_.back(), clarkName); resolutionStack_.pop_back(); if(val != 0) return val; val = lookup(stack_.front(), clarkName); // try our "global" scope if(val == 0) throw Arabica::XPath::UnboundVariableException(clarkName); return val; } // resolveVariable private: typedef std::vector ScopeStack; std::string clarkName(Variable_instance_ptr var) { return "{" + var->namespace_uri() + "}" + var->name(); } // clarkName Arabica::XPath::XPathValue lookup(const Scope& scope, const std::string& name) const { Scope::const_iterator i = scope.find(name); if(i == scope.end()) return Arabica::XPath::XPathValue(0); return i->second->value(); } // lookup ScopeStack stack_; ScopeStack params_; mutable std::vector resolutionStack_; }; // class VariableStack } // namespace XSLT } // namespace Arabica #endif // ARABICA_XSLT_VARIABLE_STACK_HPP