forked from RodneyMarsh/pyxb
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathuserref_usebind.txt
303 lines (215 loc) · 11.8 KB
/
userref_usebind.txt
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
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
.. _usebind:
Using Binding Classes
=====================
Python instances corresponding to XML structures can be created in two
primary ways: from XML documents, and directly within Python code.
Generating XML documents from bindings can also be controlled.
.. _from-xml:
Creating Instances from XML Documents
-------------------------------------
XML documents are converted into Python bindings by invoking the
``CreateFromDocument`` function in a binding module. For example:
.. literalinclude:: ../examples/manual/demo3.py
The ``CreateFromDocument`` function in a given binding module is configured
so that documents with no default namespace are assumed to be in the
namespace from which the binding was generated.
.. _invalid-content:
Locating Invalid Content
^^^^^^^^^^^^^^^^^^^^^^^^
If a document does not validate, PyXB will generally through an
:py:obj:`pyxb.UnrecognizedContentError` exception. You can determine where the
problem lies, and what was not recognized, by examining attributes present
on the exception as shown in this example:
.. literalinclude:: ../examples/manual/badcontent.py
which produces:
.. literalinclude:: ../examples/manual/badcontent.out
Coping With Wrong ``xsi:type`` Attributes
^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
Some web services and binding tools mis-use `xsi:type
<http://www.w3.org/TR/xmlschema-1/#xsi_type>`_, providing attribute values
that either are not types, or do not specify a type that is derived from an
abstract type. The
:py:obj:`pyxb.namespace.builtin.XMLSchema_instance.ProcessTypeAttribute
<pyxb.namespace.builtin._XMLSchema_instance.ProcessTypeAttribute>` method
can be used to relax how PyXB processes those attributes.
.. _from-python:
Creating Instances in Python Code
---------------------------------
Creating bindings from XML documents is straightforward, because the
documents contain enough information to identify each element and attribute,
locate the corresponding use in the binding class, and store a value that is
converted to the appropriate type. Creating values in Python is inherently
more complex, because native Python objects like strings and integers do not
contain this information.
As described in :ref:`bindingModel`, binding classes corresponding to simple
types extend the underlying Python type (such as ``str`` or ``int``), and
add XML-specific information like the canonical representation of the value
in `Unicode <http://www.unicode.org/>`_, which is the natural representation
as XML text. These classes also maintain a set of facets that constrain the
values that can be stored as instances when validation is active. Binding
classes for complex types have constructors that parse positional and
keyword parameters to determine the appropriate element or attribute to
which the value belongs. Attributes are assigned using keyword parameters.
Content is assigned using positional parameters. The order of the
positional parameters must be consistent with the order expected by the
content model.
Using the schema in the :ref:`namespace-aware address schema
<nsaddress_xsd>`, we can begin to construct the example document in Python:
.. literalinclude:: ../examples/manual/demo4a.py
.. note::
It is necessary to provide an ``element_name`` parameter to ``to_xml``
because in this case ``USAddress`` is the name of a complex type, not an
top-level schema element. PyXB cannot generate XML for an instance unless
it knows the name to use for the root element. In most situations PyXB can
figure out what element the instance belongs to, as when the instance is
created through an element binding instead of a type binding and when it is
assigned into another instance, both of which are seen in `demo4c`_.
This produces:
.. literalinclude:: ../examples/manual/demo4a.out
Assigning to individual fields like this bypasses the complex type content
model, although each field itself is validated. For example, the address
schema does not include New York as a state, so the following assignment::
addr.state = 'NY'
will cause a :py:obj:`pyxb.exceptions_.BadTypeValueError` exception to be raised:
.. literalinclude:: ../examples/manual/demo4a1.out
However, the order of the field assignments does not matter, as long as all
required fields are present by the time the XML document is generated.
.. literalinclude:: ../examples/manual/demo4a2.py
Alternatively, you can provide the content as positional parameters in the
object creation call:
.. literalinclude:: ../examples/manual/demo4b.py
This has the same effect, and is much more compact, but it does require that
the order match the content model.
.. _demo4c:
Attributes are set using keyword parameters:
.. literalinclude:: ../examples/manual/demo4c.py
This example produces (after reformatting):
.. literalinclude:: ../examples/manual/demo4c.out
.. index::
pair: disabling; validation
Note that, because we're in the middle of the example and have not provided
the ``items`` element that the content model requires, the code
:py:obj:`explicitly disables the requirement for
validation <pyxb.RequireValidWhenGenerating>` when generating XML from a
binding instance. A consequence of this is that the generated XML is not
valid, and validation must be :py:obj:`disabled for parsing
<pyxb.RequireValidWhenParsing>` as well if the resulting document is to be
re-converted into a binding with ``CreateFromDocument``.
.. _pyxb_BIND:
Creating Instances of Anonymous Types
^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
The style of XML schema used for purchase orders uses anonymous types for
the deeper elements of the purchase order:
.. literalinclude:: ../examples/manual/po4.xsd
.. index::
single: BIND
In particular, there is no global ``item`` element that can be used to
create the individual items. For situations like this, we use
:py:obj:`pyxb.BIND`:
.. literalinclude:: ../examples/manual/demo4c1.py
The :py:obj:`pyxb.BIND` reference wraps the content of the inner elements, and
is a cue to PyXB to attempt to build an instance of whatever type of object
would satisfy the content model at that point. The resulting document
(after reformatting) is:
.. literalinclude:: ../examples/manual/demo4c1.out
The complete document is generated by the following program:
.. literalinclude:: ../examples/manual/demo4c2.py
The additional code demonstrates a couple additional features:
- Fixed attribute values (such as ``country``) are present in the bindings,
even though they are only printed if they are set explicitly
- The PyXB types for representing dates and times are extensions of those
used by Python for the same purpose, including the ability to use them in
expressions
.. _to-xml:
Creating XML Documents from Binding Instances
---------------------------------------------
All along we've been seeing how to generate XML from a binding instance.
The ``toxml`` method is short-hand for a sequence that converts the binding
to a DOM instance using ``xml.dom.minidom``, then uses the DOM interface to
generate the XML document.
The :py:obj:`pyxb.utils.domutils.BindingDOMSupport` class provides ways to
control this generation. In particular, you may want to use something more
informative than ``ns#`` to denote namespaces in the generated documents.
This can be done using the following code:
.. literalinclude:: ../examples/manual/demo4c3.py
:lines: 20-
With this, the final document produced is:
.. literalinclude:: ../examples/manual/demo4c3.out
(Surprise: ``addr`` does not appear, because the ``nsaddress.xsd`` schema
uses the default element form ``unqualified``, so none of the address
components in the document have a namespace.)
.. _mixed_content:
Influencing Element and Mixed Content Order
^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
PyXB generally expects that any information reflected in the order of elements
is controlled by the content model in the schema. Where content includes
multiple instances of the same element, they are maintained in order within
the binding attribute corresponding to the name. Historically relative order
with other elements or with mixed content historically was not rigorously
maintained, and generated documents applied only the order enforced by the
content model.
The following example from ``examples/xhtml/generate.py`` hints at the difficulty:
.. literalinclude:: ../examples/xhtml/generate.py
If the relative order of elements and mixed content were not maintained, this
might produce something like:
.. literalinclude:: eltonly.xhtml
Here mixed content is lost, and element content is emitted in the order that
elements appear in the original schema.
As of release 1.2.1 [#content]_, PyXB appends both element and non-element content to a
list in each complex binding instance. The list may be obtained using the
:py:obj:`orderedContent
<pyxb.binding.basis.complexTypeDefinition.orderedContent>` method. The list
comprises instances of :py:obj:`pyxb.binding.basis.ElementContent` and
:py:obj:`pyxb.binding.basis.NonElementContent` added in the order in which they were
added to the binding instance: when creating the instance from a document or
through a constructor, or by invoking the
:py:obj:`append <pyxb.binding.basis.complexTypeDefinition.append>` or
:py:obj:`extend <pyxb.binding.basis.complexTypeDefinition.extend>` methods to add
content consistent with the content model.
The :py:obj:`contentInfluencesGeneration
<pyxb.ValidationConfig.contentInfluencesGeneration>` flag of
:py:obj:`pyxb.ValidationConfig` controls how the ``orderedContent`` list affects
generation of documents (both DOM directly and XML indirectly). With the
default value of :py:obj:`MIXED_ONLY <pyxb.ValidationConfig.MIXED_ONLY>` the
``orderedContent`` list is only consulted when a complex type allows both
element and non-element content.
The bundle for XHTML has been modified to use:
- :py:obj:`ALWAYS <pyxb.ValidationConfig.ALWAYS>` for :py:obj:`contentInfluencesGeneration
<pyxb.ValidationConfig.contentInfluencesGeneration>`
- :py:obj:`RAISE_EXCEPTION <pyxb.ValidationConfig.RAISE_EXCEPTION>` for
:py:obj:`orphanElementInContent <pyxb.ValidationConfig.orphanElementInContent>`
- :py:obj:`RAISE_EXCEPTION <pyxb.ValidationConfig.RAISE_EXCEPTION>` for
:py:obj:`invalidElementInContent <pyxb.ValidationConfig.invalidElementInContent>`
for all binding classes in that module. (See
``pyxb/bundles/common/xhtml1.py`` for the technique used.) This ensures
preservation of element order in cases where no non-element content may appear
(such as the top-level ``body`` element).
With this capability the following document is generated:
.. literalinclude:: ../examples/xhtml/expout.xhtml
Be aware that the automatically-maintained ``orderedContent`` list will be
incorrect in at least two cases:
- When the elements of an instance are mutated through Python code, the list
no longer reflects the correct order;
- When elements are appended directly to sub-elements as with:
.. code-block:: python
p2.b.append('another bit of bold')
the newly added elements do not appear in the ``orderedContent`` list.
The value returned by :py:obj:`orderedContent
<pyxb.binding.basis.complexTypeDefinition.orderedContent>` is a mutable list
so that you can manipulate it to reflect the content you wish to have
generated.
Where the ``orderedContent`` list is not consistent with the content model
(e.g., references elements that are no longer part of the binding instance, or
proposes an order that is not valid) various exceptions may arise. To some
extent this can be controlled through the :py:obj:`orphanElementInContent
<pyxb.ValidationConfig.orphanElementInContent>` and
:py:obj:`invalidElementInContent <pyxb.ValidationConfig.invalidElementInContent>`
flags.
.. [#content] Though previous versions also provided this information through
a ``content`` list, the list did not associate content with the element to
which it belonged making it difficult to reconstruct a valid document.
.. ignored
## Local Variables:
## fill-column:78
## indent-tabs-mode:nil
## End: