Xalan задача с простой трансформацией XSL - PullRequest
0 голосов
/ 08 июня 2011

Я злюсь из-за простого преобразования XSL.После многих испытаний я увидел, что это зависит от Xalan (потому что с Saxon это работает).Я заставил Java использовать процессор Xalan, поэтому я уверен.

Единственное правило, которое соответствует, - это «правило идентификации», в то время как другие «соответствие» не выполнены.Если я использую saxon, все работает отлично!

Можете ли вы увидеть что-то не так в этом XSL-преобразовании?

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

    <xsl:output method="xml" omit-xml-declaration="no"
        media-type="text/xml" indent="yes" encoding="UTF-8"
        doctype-public="-//W3C//DTD XHTML 1.1//EN" doctype-system="http://www.w3.org/TR/xhtml11/DTD/xhtml11.dtd" />

    <xsl:template match="/">
        <xsl:message>root</xsl:message>
        <xsl:apply-templates select="xhtml:html" />
    </xsl:template>

    <xsl:template match="xhtml:html">
        <xsl:element name="html">
            <xsl:attribute name="xml:lang">it</xsl:attribute>
            <xsl:apply-templates select="xhtml:head|xhtml:body" />
        </xsl:element>
        <xsl:message>xhtml:html</xsl:message>
    </xsl:template>

    <xsl:template match="xhtml:head">
        <xsl:element name="head">
            <xsl:apply-templates select="xhtml:meta" />
            <title>
                <xsl:value-of select="xhtml:title" />
            </title>
            <link href="%stile.css%" rel="stylesheet" type="text/css" />
            <xsl:apply-templates select="xhtml:script" />
        </xsl:element>
        <xsl:message>xhtml:head</xsl:message>
    </xsl:template>

    <xsl:template match="xhtml:body">
        <xsl:element name="body">
            <xsl:apply-templates select="descendant::xhtml:div[@class='sxSmall']" />
        </xsl:element>
        <xsl:message>xhtml:body</xsl:message>
    </xsl:template>

    <xsl:template match="xhtml:script">
        <xsl:message>xhtml:script</xsl:message>
    </xsl:template>

    <xsl:template match="xhtml:meta">
        <xsl:message>xhtml:meta</xsl:message>
    </xsl:template>

    <!-- identity -->
    <xsl:template match="@*|node()">
        <xsl:copy>
            <xsl:apply-templates select="@*|node()" />
        </xsl:copy>
        <xsl:message>identita</xsl:message>
    </xsl:template>

    <xsl:template match="xhtml:div[@class='sxSmall']">
        <xsl:element name="div">
            <xsl:attribute name="class"><xsl:value-of select="@class" /></xsl:attribute>
            <xsl:apply-templates />
        </xsl:element>
        <xsl:message>xhtml:div</xsl:message>
    </xsl:template>

</xsl:stylesheet> 

-------------- ОБНОВЛЕНИЕ ----------------

Проблема связана с Xalan и способом, которым я поместил в него источник XML.Код Java выглядит следующим образом:

StringReader srXslContent = new StringReader(xslContent);
TransformerFactory tFactory = TransformerFactory.newInstance("net.sf.saxon.TransformerFactoryImpl",null);
Transformer transformer = null;         
try
{
    transformer = tFactory.newTransformer(new StreamSource(srXslContent));
    Context.getInstance().getLogger().debug("Transformer created!");

} catch (TransformerConfigurationException e)
{
Context.getInstance().getLogger().error(ExceptionUtils.getStackTrace(e));
}

StringWriter sw = new StringWriter();
Document doc = XmlUtils.parseXmlFile(xmlContent);           
DOMSource domSource = new DOMSource(doc.getDocumentElement());
try
{   // Could be this?
    transformer.transform(domSource, new StreamResult(sw));

    // with streamsource Xalan works fine!!!
    //transformer.transform(new StreamSource(new StringReader(xmlContent)), new StreamResult(sw));
    Context.getInstance().getLogger().debug("Transformation made!");
} catch (TransformerException e)
{
Context.getInstance().getLogger().error(ExceptionUtils.getStackTrace(e));
}   

Здесь используется метод parseXml:

DocumentBuilderFactory dbf = DocumentBuilderFactory.newInstance();

dbf.setNamespaceAware(false);
dbf.setValidating(false);

// Sax
dbf.setFeature("http://xml.org/sax/features/validation", false);

// Xerces: to disable Internet searching...
dbf.setFeature("http://apache.org/xml/features/nonvalidating/load-dtd-grammar", false);
dbf.setFeature("http://apache.org/xml/features/nonvalidating/load-external-dtd", false);

DocumentBuilder db = dbf.newDocumentBuilder();          
InputSource is = new InputSource(new StringReader(contentToParse));
Document document =db.parse(is);

return document;

Важно отметить, что если я передам источник XML в виде строки (через StringReader и StreamSource) с Xalan тоже работает нормально.Я начинаю подозревать, что проблема в DOMSource ...

Ответы [ 2 ]

4 голосов
/ 08 июня 2011

Попробуйте включить пространства имен для DocumentBuilder, так как я вижу, что вы используете пространства имен в своих совпадениях xslt. Также создайте новый DOMSource с документом, а не с методом document.getDocumentElement ().

DocumentBuilderFactory dbf = DocumentBuilderFactory.newInstance();
dbf.setNamespaceAware(true);
dbf.setValidating(false);
DocumentBuilder db = dbf.newDocumentBuilder();
Document d = db.parse(...);
DOMSource ds = new DOMSource(d);
2 голосов
/ 08 июня 2011

Не вижу ничего плохого в XSLT, но трудно сказать без используемого вами источника XML.

Но если бы мне пришлось угадывать, я бы подумал, что проблема в пространстве имен. Исходный документ также определяет и использует это пространство имен xhtml ...? И анализируется ли он с помощью анализатора XML с поддержкой пространства имен?

...