File: CallTemplateElement.py

package info (click to toggle)
python-xml 0.8.4-10.1%2Blenny1
  • links: PTS
  • area: main
  • in suites: lenny
  • size: 4,972 kB
  • ctags: 10,628
  • sloc: python: 46,730; ansic: 14,354; xml: 968; makefile: 201; sh: 20
file content (107 lines) | stat: -rw-r--r-- 3,832 bytes parent folder | download | duplicates (3)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
########################################################################
#
# File Name:            CallTemplateElement.py
#
#
"""
Implementation of the XSLT Spec call-template stylesheet element.
WWW: http://4suite.com/4XSLT        e-mail: support@4suite.com

Copyright (c) 1999-2000 Fourthought Inc, USA.   All Rights Reserved.
See  http://4suite.com/COPYRIGHT  for license and copyright information
"""

from xml.dom import EMPTY_NAMESPACE
from xml.dom import Node
import xml.dom.ext
import xml.xslt
from xml.xslt import XsltElement, XsltException, Error, XSL_NAMESPACE
from xml.xpath import Util

class CallTemplateElement(XsltElement):
    legalAttrs = ('name', )

    def __init__(self, doc, uri=xml.xslt.XSL_NAMESPACE,
                 localName='call-template', prefix='xsl', baseUri=''):
        XsltElement.__init__(self, doc, uri, localName, prefix, baseUri)

    def setup(self):
        self.__dict__['_nss'] = xml.dom.ext.GetAllNs(self)
        split_name = Util.ExpandQName(
            self.getAttributeNS(EMPTY_NAMESPACE, 'name'),
            namespaces=self._nss
            )
        self.__dict__['_name'] = split_name
        self.__dict__['_tailRecursive'] = "unknown"
        self.__dict__['_params'] = filter(lambda node:
                                          node.nodeType == Node.ELEMENT_NODE,
                                          self.childNodes)
        for child in self.__dict__['_params']:
            if child.nodeType == Node.ELEMENT_NODE:
                if child.namespaceURI == XSL_NAMESPACE:
                    if child.localName != 'with-param':
                        raise XsltException(Error.ILLEGAL_CALLTEMPLATE_CHILD)
                else:
                    raise XsltException(Error.ILLEGAL_CALLTEMPLATE_CHILD)
        return
        
    def instantiate(self, context, processor, new_level=1):

        if self._tailRecursive == 'unknown':
            self.__dict__['_tailRecursive'] = CheckTailRecursion(self, self._name)

        origState = context.copy()
        context.setNamespaces(self._nss)
        
        params = {}
        for child in self._params:
            param = child.instantiate(context, processor)[1]
            params[param[0]] = param[1]

        if self._tailRecursive:
            if not new_level:
                context.set(origState)
                return (context, params)
            while params is not None:
                params = processor.callTemplate(self._name, context, params, 0)
                if params:
                    context.varBindings.update(params)
        else:
            processor.callTemplate(self._name, context, params, 1)

        context.set(origState)
        return (context, None)

    def __getinitargs__(self):
        return (None, self.namespaceURI, self.localName, self.prefix,
                self.baseUri)

    def __getstate__(self):
         base_state = XsltElement.__getstate__(self)
         new_state = (base_state,
                      self._nss,
                      self._name,
                      self._tailRecursive,
                      self._params)
         return new_state

    def __setstate__(self, state):
        XsltElement.__setstate__(self, state[0])
        self._nss = state[1]
        self._name = state[2]
        self._tailRecursive = state[3]
        self._params = state[4]
        return


def CheckTailRecursion(node, name):
    if node.nextSibling and node.nodeType == Node.ELEMENT_NODE:
        return 0
    p = node.nodeType == Node.ATTRIBUTE_NODE and node.ownerElement or node.parentNode
    if p and p.nodeType == Node.ELEMENT_NODE and p.namespaceURI == XSL_NAMESPACE:
        if p.localName in ['if', 'choose']:
            return CheckTailRecursion(p, name)
        elif p.localName == 'template' and name == p._name:
            return 1
    return 0