~bzr-pqm/bzr/bzr.dev

1 by mbp at sourcefrog
import from baz patch-364
1
# -*- coding: UTF-8 -*-
1887.1.1 by Adeodato Simó
Do not separate paragraphs in the copyright statement with blank lines,
2
#
1 by mbp at sourcefrog
import from baz patch-364
3
# This program is free software; you can redistribute it and/or modify
4
# it under the terms of the GNU General Public License as published by
5
# the Free Software Foundation; either version 2 of the License, or
6
# (at your option) any later version.
1887.1.1 by Adeodato Simó
Do not separate paragraphs in the copyright statement with blank lines,
7
#
1 by mbp at sourcefrog
import from baz patch-364
8
# This program is distributed in the hope that it will be useful,
9
# but WITHOUT ANY WARRANTY; without even the implied warranty of
10
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
11
# GNU General Public License for more details.
1887.1.1 by Adeodato Simó
Do not separate paragraphs in the copyright statement with blank lines,
12
#
1 by mbp at sourcefrog
import from baz patch-364
13
# You should have received a copy of the GNU General Public License
14
# along with this program; if not, write to the Free Software
15
# Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
16
17
"""XML externalization support."""
18
48 by Martin Pool
witty comment
19
# "XML is like violence: if it doesn't solve your problem, you aren't
20
# using enough of it." -- various
21
1180 by Martin Pool
- start splitting code for xml (de)serialization away from objects
22
# importing this module is fairly slow because it has to load several
23
# ElementTree bits
24
1248 by Martin Pool
- new weave based cleanup [broken]
25
from bzrlib.trace import mutter, warning
26
802 by Martin Pool
- Remove XMLMixin class in favour of simple pack_xml, unpack_xml functions
27
try:
1283 by Martin Pool
- cElementTree is typically not installed in util
28
    from cElementTree import (ElementTree, SubElement, Element,
29
                              XMLTreeBuilder, fromstring, tostring)
1772.1.1 by mbp at sourcefrog
Fix up loading of fallback ElementTree
30
    import elementtree
802 by Martin Pool
- Remove XMLMixin class in favour of simple pack_xml, unpack_xml functions
31
except ImportError:
1185.33.68 by Martin Pool
Emit warning to trace file only if using cElementTree.
32
    mutter('WARNING: using slower ElementTree; consider installing cElementTree'
33
           " and make sure it's on your PYTHONPATH")
1227 by Martin Pool
- methods to deserialize objects from strings
34
    from util.elementtree.ElementTree import (ElementTree, SubElement,
1248 by Martin Pool
- new weave based cleanup [broken]
35
                                              Element, XMLTreeBuilder,
36
                                              fromstring, tostring)
1772.1.1 by mbp at sourcefrog
Fix up loading of fallback ElementTree
37
    import util.elementtree as elementtree
802 by Martin Pool
- Remove XMLMixin class in favour of simple pack_xml, unpack_xml functions
38
1910.2.31 by Aaron Bentley
Fix bugs in basis inventory handling, change filename
39
from bzrlib import errors
1180 by Martin Pool
- start splitting code for xml (de)serialization away from objects
40
41
42
class Serializer(object):
43
    """Abstract object serialize/deserialize"""
44
    def write_inventory(self, inv, f):
45
        """Write inventory to a file"""
46
        elt = self._pack_inventory(inv)
47
        self._write_element(elt, f)
48
1248 by Martin Pool
- new weave based cleanup [broken]
49
    def write_inventory_to_string(self, inv):
1185.16.123 by Martin Pool
Fix syntax of serializer_v5.pack_revision_to_string
50
        return tostring(self._pack_inventory(inv)) + '\n'
1248 by Martin Pool
- new weave based cleanup [broken]
51
1227 by Martin Pool
- methods to deserialize objects from strings
52
    def read_inventory_from_string(self, xml_string):
1910.2.31 by Aaron Bentley
Fix bugs in basis inventory handling, change filename
53
        try:
54
            return self._unpack_inventory(fromstring(xml_string))
55
        except SyntaxError, e:
56
            raise errors.UnexpectedInventoryFormat(e)
1227 by Martin Pool
- methods to deserialize objects from strings
57
1180 by Martin Pool
- start splitting code for xml (de)serialization away from objects
58
    def read_inventory(self, f):
1910.2.31 by Aaron Bentley
Fix bugs in basis inventory handling, change filename
59
        try:
60
            return self._unpack_inventory(self._read_element(f))
61
        except SyntaxError, e:
62
            raise errors.UnexpectedInventoryFormat(e)
1180 by Martin Pool
- start splitting code for xml (de)serialization away from objects
63
1182 by Martin Pool
- more disentangling of xml storage format from objects
64
    def write_revision(self, rev, f):
65
        self._write_element(self._pack_revision(rev), f)
66
1248 by Martin Pool
- new weave based cleanup [broken]
67
    def write_revision_to_string(self, rev):
1185.16.123 by Martin Pool
Fix syntax of serializer_v5.pack_revision_to_string
68
        return tostring(self._pack_revision(rev)) + '\n'
1248 by Martin Pool
- new weave based cleanup [broken]
69
1182 by Martin Pool
- more disentangling of xml storage format from objects
70
    def read_revision(self, f):
71
        return self._unpack_revision(self._read_element(f))
72
1227 by Martin Pool
- methods to deserialize objects from strings
73
    def read_revision_from_string(self, xml_string):
1248 by Martin Pool
- new weave based cleanup [broken]
74
        return self._unpack_revision(fromstring(xml_string))
1227 by Martin Pool
- methods to deserialize objects from strings
75
1180 by Martin Pool
- start splitting code for xml (de)serialization away from objects
76
    def _write_element(self, elt, f):
77
        ElementTree(elt).write(f, 'utf-8')
78
        f.write('\n')
79
80
    def _read_element(self, f):
81
        return ElementTree().parse(f)
1713.1.12 by Robert Collins
Improve serialisation of xml performance by overriding elementree's escape routines.
82
83
1772.1.1 by mbp at sourcefrog
Fix up loading of fallback ElementTree
84
# performance tuning for elementree's serialiser. This should be
1713.1.14 by Robert Collins
Review feedback.
85
# sent upstream - RBC 20060523.
1759.2.1 by Jelmer Vernooij
Fix some types (found using aspell).
86
# the functions here are patched into elementtree at runtime.
1713.1.12 by Robert Collins
Improve serialisation of xml performance by overriding elementree's escape routines.
87
import re
1713.1.14 by Robert Collins
Review feedback.
88
escape_re = re.compile("[&'\"<>]")
1713.1.12 by Robert Collins
Improve serialisation of xml performance by overriding elementree's escape routines.
89
escape_map = {
90
    "&":'&amp;',
91
    "'":"&apos;", # FIXME: overkill
92
    "\"":"&quot;",
93
    "<":"&lt;",
94
    ">":"&gt;",
95
    }
96
def _escape_replace(match, map=escape_map):
97
    return map[match.group()]
98
 
99
def _escape_attrib(text, encoding=None, replace=None):
100
    # escape attribute value
101
    try:
102
        if encoding:
103
            try:
104
                text = elementtree.ElementTree._encode(text, encoding)
105
            except UnicodeError:
106
                return elementtree.ElementTree._encode_entity(text)
107
        if replace is None:
108
            return escape_re.sub(_escape_replace, text)
109
        else:
110
            text = replace(text, "&", "&amp;")
111
            text = replace(text, "'", "&apos;") # FIXME: overkill
112
            text = replace(text, "\"", "&quot;")
113
            text = replace(text, "<", "&lt;")
114
            text = replace(text, ">", "&gt;")
115
            return text
116
    except (TypeError, AttributeError):
117
        elementtree.ElementTree._raise_serialization_error(text)
118
119
elementtree.ElementTree._escape_attrib = _escape_attrib
120
1713.1.14 by Robert Collins
Review feedback.
121
escape_cdata_re = re.compile("[&<>]")
1713.1.12 by Robert Collins
Improve serialisation of xml performance by overriding elementree's escape routines.
122
escape_cdata_map = {
123
    "&":'&amp;',
124
    "<":"&lt;",
125
    ">":"&gt;",
126
    }
127
def _escape_cdata_replace(match, map=escape_cdata_map):
128
    return map[match.group()]
129
 
130
def _escape_cdata(text, encoding=None, replace=None):
131
    # escape character data
132
    try:
133
        if encoding:
134
            try:
135
                text = elementtree.ElementTree._encode(text, encoding)
136
            except UnicodeError:
137
                return elementtree.ElementTree._encode_entity(text)
138
        if replace is None:
139
            return escape_cdata_re.sub(_escape_cdata_replace, text)
140
        else:
141
            text = replace(text, "&", "&amp;")
142
            text = replace(text, "<", "&lt;")
143
            text = replace(text, ">", "&gt;")
144
            return text
145
    except (TypeError, AttributeError):
146
        elementtree.ElementTree._raise_serialization_error(text)
147
148
elementtree.ElementTree._escape_cdata = _escape_cdata