如何在OSGi Bundle中使用Java扩展执行XSLT转换

时间:2011-09-11 19:42:17

标签: osgi classloader xslt

我们正在将现有代码转换为OSGi环境。在我们的一个(尚未)OSGi包中,我们有代码执行XSLT转换。一段XSLT包含一个java扩展函数,用于创建唯一的数值。 Java类也驻留在bundle中。这是样式表的一个片段:

<xsl:template match="m:property">
   <xsl:variable name="uniqueDataStreamName" select="concat(../@id,'/',@name)" />
   <xsl:variable name="uniqueDataStreamId"
       select="java:com.xyz.TransformationUtils.makeDataStreamIdFromUniqueName($uniqueDataStreamName)" />
   <data id="{number($uniqueDataStreamId)}">
   <tag>
      <xsl:value-of select="$uniqueDataStreamName" />
   </tag>
   <current_value>
     <xsl:value-of select="@value" />
  </current_value>
</data>

作为参考,这是设置和调用转换的方式:

protected Templates createTemplates(Source xsltSource) {
    try {
        TransformerFactory tf = TransformerFactory.newInstance();
        Templates templates = tf.newTemplates(xsltSource);
        return templates;
    } catch (TransformerException e) {
        throw new RuntimeException(e);
    }
}


protected byte[] transform(byte[] input) throws TransformerException {
    ByteArrayOutputStream out = new ByteArrayOutputStream(2048);
    templates.newTransformer().transform(
            new StreamSource(new ByteArrayInputStream(input)),
            new StreamResult(out));
    return out.toByteArray();
}

在非OSGi environemnt中运行时,它可以正常工作。在OSGi框架中运行时,它会失败,因为无法编译样式表,因为找不到类TransformationUtils。我有点明白 - 加载jaxp转换器实现的类加载器在我们的bundle中没有看到扩展类。但是,我很难找到解决方案。我尝试过使用OSGi:使用Xalan和Xerces捆绑无济于事。

我的问题是:如何解决这个问题?可以吗?

1 个答案:

答案 0 :(得分:5)

答案取决于XSLT处理器如何查找扩展类。您可能需要阅读源代码和/或通过调试器运行它以找到它。

例如,如果XSLT处理器使用线程上下文类加载器(TCCL),那么这通常会失败,因为OSGi中未定义TCCL。您可以通过在XSLT处理器调用期间显式设置TCCL来解决此问题,例如:

ClassLoader orig = Thread.currentThread().getContextClassLoader();
Thread.currentThread().setContextClassLoader(MyClass.class.getClassLoader());
try {
    // invoke XSLT processor
} finally {
    Thread.currentThread.setContextClassLoader(orig);
}

其中MyClass是您的捆绑中的一个类,并且可以看到扩展类。

在最坏的情况下,处理器可能会使用自己的类加载器查找扩展名,即只需调用Class.forName()即可。在这里做的最好的事情是打败图书馆的开发人员是如此愚蠢。在你完成之后,你可以使用一个片段将扩展类附加到处理器包......这是一个讨厌的黑客,但比其他一些可能讨厌的黑客更好。