REST扩展不会转换JSON。

3

我有一个MarkLogic Rest扩展,需要将JSON转换为OBI对象。首先,我创建了一个XML输入并为其创建了转换,这些都有效。

现在真正的数据略有不同,是JSON格式的,因此我需要将JSON转换为XML。我有一个示例xsl转换,但似乎无法工作...

我想通过rest扩展加载的JSON文档:

{
    "wifi_raw": [
        {
            "id": "4354279",
            "hostname": "rb-0046",
            "mac": "00:0C:43:00:08:F4",
            "firstseen": "2015-08-12 13:54:50",
            "lastseen": "2015-08-12 13:54:50",
            "rssi": "-1",
            "packets": "1",
            "bssid": "24:A4:3C:53:19:62",
            "probes": "",
            "processed": "0"
        },
        {
            "id": "4354257",
            "hostname": "rb-0046",
            "mac": "00:0E:58:BC:E9:03",
            "firstseen": "2015-08-12 13:48:45",
            "lastseen": "2015-08-12 13:52:10",
            "rssi": "-58",
            "packets": "3",
            "bssid": "B8:E9:37:17:DA:EF",
            "probes": "sonos_hbbzdjrspta2htbcqeb0gcjouc",
            "processed": "0"
        },
        {
            "id": "4354273",
            "hostname": "rb-0046",
            "mac": "00:0E:58:BC:E9:03",
            "firstseen": "2015-08-12 13:48:45",
            "lastseen": "2015-08-12 13:54:32",
            "rssi": "-61",
            "packets": "4",
            "bssid": "B8:E9:37:17:DA:EF",
            "probes": "sonos_hbbzdjrspta2htbcqeb0gcjouc",
            "processed": "0"
        }
    ]
}

在接收时,我希望将其转换为已定义的OBI对象,并且这些对象适用于等效的XML输入文档...

如果我执行

json:transform-from-json($source)

在接收扩展中,JSON会转换为以下XML:

<?xml version="1.0"?>
<json xmlns="http://marklogic.com/xdmp/json/basic" type="object">
  <wifi__raw type="array">
    <json type="object">
      <id type="string">4354279</id>
      <hostname type="string">rb-0046</hostname>
      <mac type="string">00:0C:43:00:08:F4</mac>
      <firstseen type="string">2015-08-12 13:54:50</firstseen>
      <lastseen type="string">2015-08-12 13:54:50</lastseen>
      <rssi type="string">-1</rssi>
      <packets type="string">1</packets>
      <bssid type="string">24:A4:3C:53:19:62</bssid>
      <probes type="string"/>
      <processed type="string">0</processed>
    </json>
    <json type="object">
      <id type="string">4354257</id>
      <hostname type="string">rb-0046</hostname>
      <mac type="string">00:0E:58:BC:E9:03</mac>
      <firstseen type="string">2015-08-12 13:48:45</firstseen>
      <lastseen type="string">2015-08-12 13:52:10</lastseen>
      <rssi type="string">-58</rssi>
      <packets type="string">3</packets>
      <bssid type="string">B8:E9:37:17:DA:EF</bssid>
      <probes type="string">sonos_hbbzdjrspta2htbcqeb0gcjouc</probes>
      <processed type="string">0</processed>
    </json>
    <json type="object">
      <id type="string">4354273</id>
      <hostname type="string">rb-0046</hostname>
      <mac type="string">00:0E:58:BC:E9:03</mac>
      <firstseen type="string">2015-08-12 13:48:45</firstseen>
      <lastseen type="string">2015-08-12 13:54:32</lastseen>
      <rssi type="string">-61</rssi>
      <packets type="string">4</packets>
      <bssid type="string">B8:E9:37:17:DA:EF</bssid>
      <probes type="string">sonos_hbbzdjrspta2htbcqeb0gcjouc</probes>
      <processed type="string">0</processed>
    </json>
  </wifi__raw>
</json> 

现在我的最基本的XSL转换如下所示

<?xml version="1.0" encoding="UTF-8"?>
<xsl:stylesheet version="2.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform"

  xmlns:xdmp="http://marklogic.com/xdmp"
  xmlns:json="http://marklogic.com/xdmp/json"
  xmlns:basic="http://marklogic.com/xdmp/json/basic"
  xmlns:map="http://marklogic.com/xdmp/map"

  xmlns:obj="http://marklogic.com/solutions/obi/object"

  extension-element-prefixes="xdmp">

  <xdmp:import-module namespace="http://marklogic.com/xdmp/json" href="/MarkLogic/json/json.xqy"/>

  <xsl:param name="params" as="map:map"/>
  <xsl:variable name="ingest-time" select="(map:get($params, 'ingest-time'), 'false')[1] = 'true'"/>

  <xsl:template match="/text()[empty(../*)]">
    <xsl:apply-templates select="json:transform-from-json(.)"/>
  </xsl:template>

  <xsl:template match="basic:json">
    <results>
      <objects>
        <!-- test -->
        <xsl:text>example</xsl:text>
      </objects>

      <links>
        <!-- links go here-->
        <xsl:text>example</xsl:text>
      </links>
    </results>
  </xsl:template>
</xsl:stylesheet>

我从中没有得到任何结果

let $result := eput:apply-document-transform(fn:concat($dataset, '-transform'), $params, $context, $source)/element()

在摄取扩展中。
据我所知,“match="/text()[empty(../*)]">”匹配JSON文档节点,因此相同的transform-from-json将生成原始JSON文档的XML版本,如上所述,因此我至少可以使用“match="basic:json"”找到根节点。
我在这里错过了什么?
编辑工作解决方案
基本上这就是有效的方法。首先,我们确保匹配新的JSON对象节点,然后您必须非常小心地匹配XSL模板。我有两个匹配“basic:json”的模板,结果只执行最后一个...
现在使用以下内容匹配根节点:
<xsl:template match="/basic:json">

数组中的元素与wifi__raw的任何子元素匹配

<xsl:template match="basic:wifi__raw/*">

完整的XSL现在看起来像这样:
<?xml version="1.0" encoding="UTF-8"?>
<xsl:stylesheet version="2.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform"

  xmlns:xdmp="http://marklogic.com/xdmp"
  xmlns:json="http://marklogic.com/xdmp/json"
  xmlns:basic="http://marklogic.com/xdmp/json/basic"
  xmlns:map="http://marklogic.com/xdmp/map"

  xmlns:obj="http://marklogic.com/solutions/obi/object"

  extension-element-prefixes="xdmp">

  <xdmp:import-module namespace="http://marklogic.com/xdmp/json" href="/MarkLogic/json/json.xqy"/>

  <xsl:param name="params" as="map:map"/>
  <xsl:variable name="ingest-time" select="(map:get($params, 'ingest-time'), 'false')[1] = 'true'"/>

  <!-- voor MarkLogic 7 -->
  <xsl:template match="/text()[empty(../*)]">
    <xsl:apply-templates select="json:transform-from-json(.)"/>
  </xsl:template>

  <!-- voor MarkLogic 8 -->
  <xsl:template match="/node()[not(self::*)]">
    <xsl:apply-templates select="json:transform-from-json(.)"/>
  </xsl:template>

  <xsl:template match="/basic:json">
    <xsl:apply-templates select="basic:wifi__raw"/>
  </xsl:template>

  <xsl:template match="basic:wifi__raw">
    <xsl:variable name="objects" as="element()*">
      <xsl:apply-templates select="*" />
    </xsl:variable>

    <results>
      <objects>
        <xsl:sequence select="$objects" />
      </objects>

    </results>
  </xsl:template>

  <xsl:template match="basic:wifi__raw/*">
    <foundId>
      <xsl:value-of select="basic:id" />
    </foundId>
  </xsl:template>

</xsl:stylesheet>
1个回答

1
这是由于MarkLogic 8与MarkLogic 7及以前的处理JSON数据类型方式不同所致。在MarkLogic 8中,JSON数据被处理为object-node()数据,而不是text()。不幸的是,在XSLT中无法明确测试object-node(),但有几种解决方法。将text()的匹配方式更改为如下:
<xsl:template match="/node()[not(self::*)]">
  <xsl:apply-templates select="json:transform-from-json(.)"/>
</xsl:template>

这应该适用于MarkLogic 7和8。

希望对您有所帮助!


添加此模板会导致“errorResponse”:{“statusCode”:500,“status”:“内部服务器错误”,“messageCode”:“内部错误”,“message”:“XSLT-BADPATTERN:(err:XTSE0340)无效模式:fn:doc(\”/marklogic.rest.transform/csjson-transform/assets/transform.xsl\”)/ *:stylesheet / *:template [1] / @match(XDMP-UNEXPECTED:(err:XPST0003)意外的令牌语法错误,意外的Lpar_,期望$ end) - Hugo Koopmans
抱歉,我忘记了在MarkLogic中XSLT语法比XQuery语法更严格。我改变了我的答案,现在可以正常工作了。 - grtjn
那是一个不同的问题..;-) - grtjn

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接