configurationengine/source/cone/public/_etree_wrapper.py
author m2lahtel
Thu, 21 Oct 2010 16:36:53 +0300
changeset 5 d2c80f5cab53
parent 3 e7e0ae78773e
permissions -rw-r--r--
Updated to version 1.2.14
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
0
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
     1
#
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
     2
# Copyright (c) 2009 Nokia Corporation and/or its subsidiary(-ies).
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
     3
# All rights reserved.
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
     4
# This component and the accompanying materials are made available
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
     5
# under the terms of "Eclipse Public License v1.0"
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
     6
# which accompanies this distribution, and is available
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
     7
# at the URL "http://www.eclipse.org/legal/epl-v10.html".
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
     8
#
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
     9
# Initial Contributors:
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
    10
# Nokia Corporation - initial contribution.
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
    11
#
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
    12
# Contributors:
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
    13
#
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
    14
# Description: 
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
    15
# 
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
    16
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
    17
from xml.parsers import expat
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
    18
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
    19
# Import ElementTree (should always be available)
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
    20
try:
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
    21
    from elementtree import ElementTree
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
    22
except ImportError:
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
    23
    from xml.etree import ElementTree
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
    24
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
    25
import exceptions
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
    26
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
    27
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
    28
class ElementTreeBackendWrapperBase(object):
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
    29
    def get_module(self):
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
    30
        raise NotImplementedError()
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
    31
    
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
    32
    def get_lineno(self, element):
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
    33
        raise NotImplementedError()
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
    34
3
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents: 0
diff changeset
    35
    def get_elem_from_lineno(self, root, lineno):
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents: 0
diff changeset
    36
        raise NotImplementedError()
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents: 0
diff changeset
    37
0
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
    38
class ElementTreeBackendWrapper(ElementTreeBackendWrapperBase):
3
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents: 0
diff changeset
    39
0
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
    40
    class CustomTreeBuilder(ElementTree.TreeBuilder):
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
    41
        """
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
    42
        Custom TreeBuilder for ElementTree that records line numbers
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
    43
        of the elements.
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
    44
        """
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
    45
        def start(self, tag, attrs):
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
    46
            elem = ElementTree.TreeBuilder.start(self, tag, attrs)
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
    47
            lineno = self._xmltreebuilder._parser.CurrentLineNumber
3
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents: 0
diff changeset
    48
            # print "Tag: %s, line: %r" % (tag, lineno)
0
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
    49
            elem.sourceline = lineno
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
    50
            return elem
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
    51
    
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
    52
    def get_module(self):
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
    53
        return ElementTree
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
    54
    
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
    55
    def fromstring(self, text):
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
    56
        try:
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
    57
            treebuilder = self.CustomTreeBuilder()
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
    58
            parser = ElementTree.XMLTreeBuilder(target=treebuilder)
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
    59
            treebuilder._xmltreebuilder = parser
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
    60
            parser.feed(text)
3
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents: 0
diff changeset
    61
            self.root = parser.close()
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents: 0
diff changeset
    62
            return self.root
0
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
    63
        except expat.ExpatError, e:
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
    64
            raise exceptions.XmlParseError(
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
    65
                "XML parse error on line %d: %s" % (e.lineno, e),
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
    66
                e.lineno, str(e))
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
    67
    
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
    68
    def tostring(self, etree, encoding=None):
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
    69
        return ElementTree.tostring(etree, encoding)
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
    70
    
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
    71
    def get_lineno(self, element):
3
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents: 0
diff changeset
    72
        try:
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents: 0
diff changeset
    73
            return element.sourceline
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents: 0
diff changeset
    74
        except AttributeError:
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents: 0
diff changeset
    75
            return None
0
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
    76
3
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents: 0
diff changeset
    77
    def get_elem_from_lineno(self, root, lineno):
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents: 0
diff changeset
    78
        for elem in root.getiterator():
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents: 0
diff changeset
    79
            if elem.sourceline == lineno:
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents: 0
diff changeset
    80
                return elem
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents: 0
diff changeset
    81
        return None
0
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
    82
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
    83
class CElementTreeBackendWrapper(ElementTreeBackendWrapperBase):
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
    84
    def __init__(self):
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
    85
        try:
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
    86
            from cElementTree import cElementTree
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
    87
        except ImportError:
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
    88
            from xml.etree import cElementTree
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
    89
        
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
    90
        self.cElementTree = cElementTree
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
    91
    
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
    92
    def get_module(self):
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
    93
        return self.cElementTree
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
    94
    
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
    95
    def fromstring(self, text):
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
    96
        try:
3
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents: 0
diff changeset
    97
            self.root = self.cElementTree.fromstring(text)
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents: 0
diff changeset
    98
            return self.root
0
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
    99
        except SyntaxError, e:
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   100
            # cElementTree raises a SyntaxError, but does not set
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   101
            # its lineno attribute, so look for the line number
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   102
            # in the exception text
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   103
            import re
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   104
            match = re.search(r'line (\d+)\, column \d+$', str(e))
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   105
            if match:   lineno = int(match.group(1))
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   106
            else:       lineno = None
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   107
            
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   108
            raise exceptions.XmlParseError(
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   109
                "XML parse error on line %s: %s" % (lineno, e),
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   110
                lineno, str(e))
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   111
    
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   112
    def tostring(self, etree, encoding=None):
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   113
        return self.cElementTree.tostring(etree, encoding)
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   114
    
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   115
    def get_lineno(self, element):
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   116
        # cElementTree does not support line numbers
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   117
        return None
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   118
3
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents: 0
diff changeset
   119
    def get_elem_from_lineno(self, root, lineno):
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents: 0
diff changeset
   120
        # cElementTree does not support line numbers
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents: 0
diff changeset
   121
        return None
0
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   122
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   123
class LxmlBackendWrapper(ElementTreeBackendWrapperBase):
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   124
    
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   125
    def __init__(self):
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   126
        import lxml.etree
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   127
        self.lxml = lxml
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   128
    
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   129
    def get_module(self):
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   130
        return self.lxml.etree
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   131
    
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   132
    def fromstring(self, text):
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   133
        try:
3
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents: 0
diff changeset
   134
            self.root = self.lxml.etree.fromstring(text)
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents: 0
diff changeset
   135
            self.remove_comments(self.root)
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents: 0
diff changeset
   136
            return self.root
0
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   137
        except self.lxml.etree.XMLSyntaxError, e:
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   138
            raise exceptions.XmlParseError(
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   139
                "XML parse error on line %d: %s" % (e.position[0], e),
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   140
                e.position[0], str(e))
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   141
    
3
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents: 0
diff changeset
   142
    def remove_comments(self, elem):
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents: 0
diff changeset
   143
        """
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents: 0
diff changeset
   144
        lxml parses also comments, but ConE does not expect those,
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents: 0
diff changeset
   145
        so remove them to prevent any errors on that account
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents: 0
diff changeset
   146
        """
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents: 0
diff changeset
   147
        # Find the comments under this element
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents: 0
diff changeset
   148
        comments = []
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents: 0
diff changeset
   149
        for x in elem:
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents: 0
diff changeset
   150
            if isinstance(x, self.lxml.etree._Comment):
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents: 0
diff changeset
   151
                comments.append(x)
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents: 0
diff changeset
   152
        
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents: 0
diff changeset
   153
        # Remove them
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents: 0
diff changeset
   154
        for c in comments:
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents: 0
diff changeset
   155
            elem.remove(c)
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents: 0
diff changeset
   156
        
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents: 0
diff changeset
   157
        # Recurse to sub-elements
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents: 0
diff changeset
   158
        for subelem in elem:
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents: 0
diff changeset
   159
            self.remove_comments(subelem)
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents: 0
diff changeset
   160
            
0
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   161
    def tostring(self, etree, encoding=None):
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   162
        return self.lxml.etree.tostring(etree, encoding=encoding)
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   163
    
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   164
    def get_lineno(self, element):
3
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents: 0
diff changeset
   165
        try:
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents: 0
diff changeset
   166
            return element.sourceline
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents: 0
diff changeset
   167
        except AttributeError:
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents: 0
diff changeset
   168
            return None
0
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   169
3
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents: 0
diff changeset
   170
    def get_elem_from_lineno(self, root, lineno):
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents: 0
diff changeset
   171
        for elem in root.getiterator():
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents: 0
diff changeset
   172
            if elem.sourceline == lineno:
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents: 0
diff changeset
   173
                return elem
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents: 0
diff changeset
   174
        return None
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents: 0
diff changeset
   175
    
0
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   176
# ============================================================================
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   177
#
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   178
# ============================================================================
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   179
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   180
class ElementTreeWrapper(object):
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   181
    """
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   182
    ElementTree wrapper class for providing a unified interface to different
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   183
    ElementTree implementations.
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   184
    
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   185
    Currently supported are the pure Python ElementTree implementation,
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   186
    cElementTree and lxml.etree
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   187
    """
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   188
    BACKEND_ELEMENT_TREE     = 'ElementTree'
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   189
    BACKEND_C_ELEMENT_TREE   = 'cElementTree'
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   190
    BACKEND_LXML             = 'lxml'
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   191
    
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   192
    # Import order for the default back-end. The list is traversed
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   193
    # top-down and the first back-end whose importing is successful is
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   194
    # used as the default back-end
3
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents: 0
diff changeset
   195
    DEFAULT_BACKEND_IMPORT_ORDER = [BACKEND_LXML,
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents: 0
diff changeset
   196
                                    BACKEND_C_ELEMENT_TREE,
0
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   197
                                    BACKEND_ELEMENT_TREE]
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   198
    
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   199
    _backend_mapping = {BACKEND_ELEMENT_TREE:     ElementTreeBackendWrapper,
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   200
                        BACKEND_C_ELEMENT_TREE:   CElementTreeBackendWrapper,
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   201
                        BACKEND_LXML:             LxmlBackendWrapper}
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   202
    
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   203
    _backend_id = None
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   204
    _backend_wrapper = None
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   205
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   206
    def get_backend_id(self):
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   207
        """
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   208
        Return the ID of the currently used ElementTree back-end.
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   209
        """
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   210
        # Make sure that the default back-end is set, so _backend_id
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   211
        # will not be None
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   212
        self._get_backend()
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   213
        assert self._backend_id is not None
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   214
        return self._backend_id
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   215
    
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   216
    def set_backend_id(self, backend_id):
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   217
        """
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   218
        Set the used ElementTree back-end by back-end ID.
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   219
        """
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   220
        if backend_id not in self._backend_mapping:
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   221
            raise ValueError("Invalid ElementTree back-end ID: %r" % backend_id)
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   222
        
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   223
        if backend_id == self._backend_id:
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   224
            return
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   225
        
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   226
        backend_wrapper_class = self._backend_mapping[backend_id]
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   227
        self._backend_wrapper = backend_wrapper_class()
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   228
        self._backend_id = backend_id
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   229
    
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   230
    def _get_backend(self):
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   231
        """
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   232
        Return the currently set ElementTree back-end wrapper object.
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   233
        """
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   234
        if self._backend_wrapper is None:
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   235
            # Back-end not set, so set the default back-end.
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   236
            # The default is the C version of ElementTree, but if that
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   237
            # is not available, the pure Python version is used
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   238
            for backend_id in self.DEFAULT_BACKEND_IMPORT_ORDER:
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   239
                try:
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   240
                    self.set_backend_id(backend_id)
5
d2c80f5cab53 Updated to version 1.2.14
m2lahtel
parents: 3
diff changeset
   241
                    #break
0
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   242
                except ImportError:
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   243
                    pass
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   244
            
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   245
            if self._backend_wrapper is None:
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   246
                raise RuntimeError("Failed to set any ElementTree backend! Tried these: %r" % self.DEFAULT_BACKEND_IMPORT_ORDER)
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   247
        
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   248
        return self._backend_wrapper
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   249
    
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   250
    def get_lineno(self, element):
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   251
        """
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   252
        Return the source line number of the given XML element.
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   253
        
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   254
        Note that for the cElementTree parser this will always return
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   255
        None, since that parser does not support line numbers.
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   256
        """
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   257
        return self._get_backend().get_lineno(element)
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   258
    
3
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents: 0
diff changeset
   259
    def get_elem_from_lineno(self, root, lineno):
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents: 0
diff changeset
   260
        """
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents: 0
diff changeset
   261
        Return the element from the given line number of the given XML element.
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents: 0
diff changeset
   262
        
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents: 0
diff changeset
   263
        @param root: the root element to search from
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents: 0
diff changeset
   264
        @param lineno: the line number to search for  
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents: 0
diff changeset
   265
        
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents: 0
diff changeset
   266
        Note that for the cElementTree parser this will always return
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents: 0
diff changeset
   267
        None, since that parser does not support line numbers.
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents: 0
diff changeset
   268
        """
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents: 0
diff changeset
   269
        return self._get_backend().get_elem_from_lineno(root, lineno)
e7e0ae78773e ConE 1.2.11 release
m2lahtel
parents: 0
diff changeset
   270
0
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   271
    def __getattribute__(self, attrname):
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   272
        try:
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   273
            # Try to get the attribute from this object (the top-level wrapper)
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   274
            return object.__getattribute__(self, attrname)
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   275
        except AttributeError:
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   276
            # If not overridden here, try to get it from the back-end wrapper
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   277
            backend = self._get_backend()
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   278
            try:
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   279
                return getattr(backend, attrname)
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   280
            except AttributeError:
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   281
                # Last resort: try to get it from the module
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   282
                # the back-end wrapper wraps
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   283
                backend_module = backend.get_module()
2e8eeb919028 Adding EPL version of configurationengine.
terytkon
parents:
diff changeset
   284
                return getattr(backend_module, attrname)