使用 zeep / python 创建 XML 序列

maj*_*son 2 python xsd soap wsdl zeep

我正在使用 zeep (Python 3.6) 与 SOAP API 交互,并使用包含此部分的 WSDL 模式:

<xs:element name="passengers">
    <xs:complexType>
        <xs:sequence>
            <xs:element maxOccurs="unbounded" name="passenger" type="com:PassengerType"/>
        </xs:sequence>
    </xs:complexType>
</xs:element>
Run Code Online (Sandbox Code Playgroud)

所以我希望我的 zeep 生成的 XML 看起来像这样:

<book:passengers>
    <book:passenger>
        ...redacted...
    </book:passenger>
</book:passengers>
Run Code Online (Sandbox Code Playgroud)

我用 Zeep 实现这一目标的第一次尝试如下所示:

passengers = [factories.PassengerType()]
Run Code Online (Sandbox Code Playgroud)

但是,将其发送到我的 SOAP API 时,会产生以下错误:

File "/usr/local/lib/python3.6/site-packages/zeep/xsd/elements/element.py", line 220, in validate
  "Missing element %s" % (self.name), path=render_path)
zeep.exceptions.ValidationError: Missing element passenger (createBookingRecordRequest.passengers)
Run Code Online (Sandbox Code Playgroud)

我相信这是因为我的 'passengers' 属性应该包含一个带有标签名称“passenger”的 Zeep 对象,该对象将包含我的元素列表。我已经尝试修补zeep.xsd.AnyType以实现这一目标,但尚未成功。

任何建议,将不胜感激。

maj*_*son 5

回答我自己的问题,因为我现在已经解决了,没有收到任何其他答案。

这个问题的根源是我试图创建没有被我的 SOAP API 的 WSDL 明确定义为类型的 XML 元素。但这没关系,因为 Zeep 仍会为其生成类型对象,它只是不会将这些类型分配给特定名称,因此我们必须跳过一些额外的环节来获取这些类型。这是我花了一点时间才弄明白的。

您可以通过任何父类型访问这些对象来获取它们。它们存储在一个名为elements2 元组列表的属性中。在这种情况下,我的PassengerType对象应该包含在属性名称为“passengers”的序列容器中。例如,如果我的父类型名为ParentType,我可以像这样使用这个“乘客”序列:

passengers = dict(ParentType.elements)['passengers'](
    PassengerType(),
    ...
    PassengerType()
)
Run Code Online (Sandbox Code Playgroud)

在这里,我们将元素对象转换为字典(利用它是一个 2 元组列表的事实,其中第一项是属性名称的字符串),然后按名称提取元素。

结果对象可以直接传递给ParentType像:

ParentType(passengers=passengers)
Run Code Online (Sandbox Code Playgroud)

简单的。

我发现的另一种替代方法是使用 zeep 的xsd对象显式构建类型。示例如下所示。

from lxml import etree
from zeep import xsd

PassengersType = xsd.ComplexType(
    xsd.Sequence([
        xsd.Element('passengers', PassengerType, min_occurs=1, max_occurs='unbounded')
    ]), qname=etree.QName("{http://example.com/schema}passengers")
)
Run Code Online (Sandbox Code Playgroud)

我认为这不是很好,但可能对登陆这里的人有用。