我需要在wihch输出一个soap xml请求,其中一些节点<
应替换为<
,>
替换为>
如何使用xslt执行此操作
我的输出应该是下面的
<env:Envelope xmlns:env="http://schemas.xmlsoap.org/soap/envelope/" xmlns:xsd="http://www.w3.org/2001/XMLSchema">
<env:Body><Create xmlns="http://jerseytelecom.com/">
<requestXml>
<ISD_XMLGateway>
<Entity>HLR_ALC</Entity>
<Origin>Comverse One</Origin>
<Log_Level>0</Log_Level>
<Params><Param Name="HLR_System" Value="JT"/>
<Param Name="HLR_ALC_Command" Value="Send_HLR_Command"/>
<Param Name="HLR_Command" Value="CRESBX:MSIN=112210231,MODEL=MODEL001,SNBSV=7797242727-TEL;"/>
</Params>
</ISD_XMLGateway>
</requestXml></Create></env:Body></env:Envelope>
答案 0 :(得分:1)
好吧,XSLT永远不会看到“&lt;”和“&gt;” XML标记中的字符,因为它不适用于词法XML,它适用于XML的树表示。所以你错误地提出了这个问题。您实际想要做的是将树序列化为词汇XML,然后将该词汇XML作为字符串插入到结果树中(之后序列化程序会将该词汇XML中的“&lt;”和“&gt;”字符转换为“&amp; lt;”和“&amp; gt;”)。
在XSLT 2.0中没有内置的serialize()函数,但最近的Saxon版本(仅限商业版)支持XSLT 3.0 serialize()函数。
答案 1 :(得分:0)
我相信这应该适用于在XSLT 1.0中转义XML:
<xsl:template match="*" mode="escape">
<xsl:variable name="currentNode" select="." />
<xsl:value-of select="concat('<', name(.))"/>
<xsl:apply-templates select="@*" mode="escape"/>
<xsl:for-each select="namespace::*[name() != 'xml'][not(. = $currentNode/../namespace::*)]">
<xsl:call-template name="EscapeNamespace">
<xsl:with-param name="namespace" select="." />
</xsl:call-template>
</xsl:for-each>
<xsl:value-of select="'>'"/>
<xsl:apply-templates select="node()" mode="escape" />
<xsl:value-of select="concat('</', local-name(.), '>')"/>
</xsl:template>
<xsl:template match="@*" mode="escape">
<xsl:value-of select="concat(' ', name(.), '="')"/>
<xsl:call-template name="EscapeText">
<xsl:with-param name="text" select="." />
</xsl:call-template>
<xsl:value-of select="'"'"/>
</xsl:template>
<xsl:template match="text()" mode="escape">
<xsl:call-template name="EscapeText">
<xsl:with-param name="text" select="." />
</xsl:call-template>
</xsl:template>
<xsl:template name="EscapeNamespace">
<xsl:param name="namespace" />
<xsl:variable name="prefix">
<xsl:choose>
<xsl:when test="name($namespace) = ''">
<xsl:value-of select="'xmlns'"/>
</xsl:when>
<xsl:otherwise>
<xsl:value-of select="concat('xmlns:',name($namespace))" />
</xsl:otherwise>
</xsl:choose>
</xsl:variable>
<xsl:value-of select="concat(' ', $prefix, '="')"/>
<xsl:call-template name="EscapeText">
<xsl:with-param name="text" select="$namespace" />
</xsl:call-template>
<xsl:value-of select="'"'"/>
</xsl:template>
<xsl:variable name="EntitiesRaw">
<entity value=""" escaped="&quot;" />
<entity value="&" escaped="&amp;" />
<entity value="<" escaped="&lt;" />
<entity value=">" escaped="&gt;" />
<entity value="'" escaped="&apos;" />
</xsl:variable>
<xsl:variable name="Entities" select="msxsl:node-set($EntitiesRaw)" />
<xsl:template name="EscapeText">
<xsl:param name="text" />
<xsl:variable name="foundEntity" select="$Entities/entity[contains($text, @value)]" />
<xsl:choose>
<xsl:when test="$foundEntity">
<xsl:call-template name="EscapeText">
<xsl:with-param name="text" select="substring-before($text, $foundEntity/@value)" />
</xsl:call-template>
<xsl:value-of select="$foundEntity/@escaped" />
<xsl:call-template name="EscapeText">
<xsl:with-param name="text" select="substring-after($text, $foundEntity/@value)" />
</xsl:call-template>
</xsl:when>
<xsl:otherwise>
<xsl:value-of select="$text"/>
</xsl:otherwise>
</xsl:choose>
</xsl:template>
您只需在变量中构建要转义的XML,然后通过模板运行它:
<xsl:template match="something">
<requestXml>
<xsl:variable name="requestXml">
<ISD_XMLGateway>
<Entity><xsl:value-of select="2 + 5" /></Entity>
<Origin>5 < 6 and 7 > 2</Origin>
<xsl:apply-templates select="otherStuff" />
<ElementWithAttributes a="10" b="12" />
</ISD_XMLGateway>
</xsl:variable>
<xsl:apply-templates select="msxsl:node-set($requestXml)" mode="escape" />
</requestXml>
</xsl:template>
以上在两个地方使用msxsl:node-set(),但我相信大多数非MS XSL引擎都具有该功能的等价物。