5

我正在努力自动化测试一个接受和返回 XML 的 API,所以我想尽可能地将 API 的记录返回数据转换为模式。基于易用性和学习性,我选择了 RelaxNG 来完成这项任务。

在我输入所有信息之前,这是一个问题:

是否可以描述“无序的元素集,具有相同的名称但不同的属性”?

这是我无法描述的示例对象:

<item>
    <id>d395136e-d060-4a6e-887c-c0337dd7ad09</id>
    <name>The item has a name</name>
    <link rel="self" type="type1" href="url" />
    <link rel="download" type="type2" href="url" />
    <link rel="relatedData" type="type3" href="url" />
</item>

链接对象是我要挂断的部分。这是问题所在:

  • 不能保证 item 内元素的顺序,所以我试图将所有元素放在<interleave>结构中。
  • 里面会有多个<link>元素<item>,具有不同的属性集(即,<item>必须有一个“self”链接、一个“download”链接和一个“relatedData”链接才有效)。
  • 每种链接类型中的一种是必需的,但不能保证再次顺序。

我试图这样描述架构:

<element name="item">
    <interleave>
        <element name="id"><text/></element>
        <element name="name"><text/></element>
        <ref name="selfLink"/>
        <ref name="launchLink"/>
        <ref name="thumbnailLink"/>
    </interleave>
</element>

“链接”引用在其他地方定义,如下所示:

 <define name="selfLink">
 <element name="link">
     <attribute name="href"><text/></attribute>
     <attribute name="rel"><value>self</value></attribute>
     <attribute name="type"><value>type1</value></attribute>
 </element>
 </define>

解析器对此不满意 - 从 jing 我得到error: the element "link" can occur in more than one operand of "interleave"。我可以看到它的意思,但我希望它可以将“具有相同名称但不同属性的元素”的想法作为独特的项目来处理。

将链接引用移出交错可以解析它,但是只要返回数据中的顺序发生变化,我就会等待验证器崩溃。

任何想法,或者这是不可能的?我正在处理的 XML 是否存在固有问题,需要我在测试应用程序中将其中的一些提升到更高的处理逻辑(在运行更通用的 XML 验证后手动检查每种链接类型?)

4

2 回答 2

3

看起来您偶然发现了RELAX NG 中的交错限制。我会尝试在Schematron中执行此操作,或者可能是 RELAX NG 和 Schematron 的组合。

这是一个片段,它使用Jing 支持<link>的 Schematron 版本检查您的元素:

<schema xmlns="http://www.ascc.net/xml/schematron">
  <pattern name="link pattern">
    <rule context="item">
      <assert test='count(link) = 3'>There must be 3 link elements.</assert>
      <assert test="count(link[@rel = 'self' and @type ='type1']) = 1">There must be 1 link element wwhere @rel='self' and @type='type1'.</assert>
      <assert test="count(link[@rel = 'download' and @type ='type2']) = 1">There must be 1 link element where @rel='download' and @type='type2'.</assert>
      <assert test="count(link[@rel = 'relatedData' and @type = 'type3']) = 1">There must be 1 link element where @rel='relatedData' and @type='type3'.</assert>
    </rule>
  </pattern>
</schema>
于 2012-08-08T19:19:14.940 回答
1

看看以下架构是否有帮助

<grammar xmlns="http://relaxng.org/ns/structure/1.0">
<start>
    <element name="item">
        <interleave>
            <element name="id"><text/>
            </element>
            <element name="name"><text/></element>
            <oneOrMore>
                <ref name="link"/>
            </oneOrMore>
        </interleave>
    </element>
</start>

<define name="link">
    <element name="link">
        <attribute name="href"/>
        <choice>
            <group>
                <attribute name="rel"><value>self</value></attribute>
                <attribute name="type"><value>type1</value></attribute>
            </group>
            <group>
                <attribute name="rel"><value>download</value></attribute>
                <attribute name="type"><value>type2</value></attribute>
            </group>
            <group>
                <attribute name="rel"><value>relatedData</value></attribute>
                <attribute name="type"><value>type3</value></attribute>
            </group>
        </choice>
    </element>
</define>
</grammar>
于 2012-08-03T22:03:33.587 回答