Mule XSLTTransformer正在转换后添加命名空间



问题:我正试图将uuid作为新元素添加到传入的xml消息中。我可以在记录结果时看到添加了它,但mule将其命名空间和java-util-uuid命名空间添加到结果中,这导致我路由此消息的另一个服务无法识别它,因为它有一个它不知道的命名空间。

有没有一种方法可以配置xslt转换器来实现我在这里要做的事情?还有其他可以替代的建议吗?

目标是让xslt生成一个uuid并标记传入消息,然后将其传递给服务端点。感谢在这方面提供的所有帮助。

骡子配置:

<mule xmlns:mulexml="http://www.mulesoft.org/schema/mule/xml"
xmlns:tracking="http://www.mulesoft.org/schema/mule/ee/tracking"
xmlns:http="http://www.mulesoft.org/schema/mule/http" xmlns="http://www.mulesoft.org/schema/mule/core"
xmlns:doc="http://www.mulesoft.org/schema/mule/documentation"
xmlns:spring="http://www.springframework.org/schema/beans" version="EE-3.4.1"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:xsl="http://www.w3.org/1999/XSL/Transform"
xmlns:billing="http://mycompany.com/billing"
xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-current.xsd
http://www.mulesoft.org/schema/mule/core http://www.mulesoft.org/schema/mule/core/current/mule.xsd
http://www.mulesoft.org/schema/mule/http http://www.mulesoft.org/schema/mule/http/current/mule-http.xsd
http://www.mulesoft.org/schema/mule/ee/tracking http://www.mulesoft.org/schema/mule/ee/tracking/current/mule-tracking-ee.xsd
http://www.mulesoft.org/schema/mule/xml http://www.mulesoft.org/schema/mule/xml/current/mule-xml.xsd">
<mulexml:xslt-transformer name="xslt"
doc:name="XSLT">
<mulexml:xslt-text>
<xsl:stylesheet version="2.0" xmlns:uuid="java:java.util.UUID">
<xsl:variable name="uid" select="uuid:randomUUID()" />
<xsl:template match="/">
<xsl:apply-templates />
</xsl:template>
<xsl:template match="node()|@*">
<xsl:copy>
<xsl:apply-templates select="node()|@*" />
</xsl:copy>
</xsl:template>
<xsl:template match="Request">
<Request>
<xsl:apply-templates select="node()|@*" />
<RequestId>
<xsl:value-of select="$uid" />
</RequestId>
</Request>
</xsl:template>
</xsl:stylesheet>
</mulexml:xslt-text>
</mulexml:xslt-transformer>
<flow name="rsi_invoiceFlow1" doc:name="rsi_invoiceFlow1">
<http:inbound-endpoint exchange-pattern="request-response"
address="${listener.hostname}:${listener.port}/${listener.path.invoice.rsi}"
doc:name="HTTP" transformer-refs="xslt" />

<logger message="#[message.payloadAs(java.lang.String)]" level="ERROR"
doc:name="Logger" />
<http:outbound-endpoint exchange-pattern="request-response"
method="POST" address="${destination.dev2.url}/" doc:name="HTTP"
doc:description="The '/' at the end of the URL is required on the RSI outbound call" />
</flow>
</mule>

传入XML:

<?xml version="1.0" encoding="UTF-8"?>
<billing:RQ xmlns:billing="http://mycompany.com/billing">
<Request>
<CallingHostOrWeblogicInstance>SOAPUI</CallingHostOrWeblogicInstance>
</Request>

输出:

<?xml version="1.0" encoding="UTF-8"?>
<billing:RQ xmlns:billing="http://mycompany.com/billing">
<Request xmlns="http://www.mulesoft.org/schema/mule/core" xmlns:uuid="java:java.util.UUID"><CallingHostOrWeblogicInstance xmlns="">SOAPUI</CallingHostOrWeblogicInstance>
<RequestId>bff3e1d6-ecdd-41ae-8807-ec04085a2b54</RequestId>
</Request>

XSLT创建的新元素在有效的默认命名空间中没有指定的命名空间。

在您的情况下,默认名称空间声明为xmlns="http://www.mulesoft.org/schema/mule/core",因此您输出的<Request>元素将位于该名称空间中。

一种方法是完全避免使用默认名称空间。请使用前缀xmlns:mule="http://www.mulesoft.org/schema/mule/core",然后使用<mule:mule><mule:flow><mule:logger>等。如果发现mule:过长,请使用m:

这可能是最干净的方法。无论如何,您已经在使用大量其他名称空间,明确说明这一点不会有什么坏处。

另一种变体是在XSLT声明中将默认名称空间重置为空:

<xsl:stylesheet 
version="2.0" 
xmlns:xsl="http://www.w3.org/1999/XSL/Transform"
xmlns:uuid="java:java.util.UUID" 
xmlns=""
>
<!-- ... -->
</xsl:stylesheet>

您也可以使用<xsl:copy>而不是显式地创建新的<Request>,但使用<RequestId>仍然会遇到同样的问题。<RequestId xmlns="">可以解决这个问题,但它相当丑陋,尤其是如果它不是您创建的唯一元素。

如果您不想将XSLT外部化(正如David所建议的那样),我建议您采用"无默认名称空间"的方法。

不要将XSL嵌入Mule配置本身,而是存储在一个独立的文件中。您将避免名称空间污染,这是一种最佳实践。

在xsl:stylesheet元素上使用xmlns="重置默认名称空间。

在xsl:stylesheet元素中使用exclude结果前缀,以避免其他mule名称空间泄漏到样式表中。

(您也可以使用XML 1.1"名称空间声明",这是为这种目的而发明的,但我不会这么做:XML 1.1没有得到足够广泛的支持,无法使该技术变得强大。)

我尝试了所有3个建议,我认为将转换与流配置分离最适合我的情况,因为它提供了最干净和解耦的方法来解决这个问题。

感谢大家的回复。

最新更新