3

读完这篇文章后,我一直在尝试实现一个自定义数据类型,供 RelaxNG XML 验证器 (Jing) 使用。我已经成功地通过命令行运行了由 Jing (他们称之为datatype-sample)提供的示例实现,但我一直无法从 java 代码中做到这一点。

从命令行(Windows):

> set CLASSPATH=path\to\jing-20091111\bin\jing.jar;path\to\jing-20091111\sample\datatype\datatype-sample.jar
> cd path\to\jing-20091111\sample\datatype
> java com.thaiopensource.relaxng.util.Driver datatype-sample.rng valid.xml

验证执行没有任何问题。但现在我正在尝试使用以下 java 代码中的相同数据类型库:

package rngdatatype;

import java.io.File;
import java.io.FileInputStream;
import java.io.FileNotFoundException;
import java.io.IOException;
import java.io.InputStreamReader;
import java.io.UnsupportedEncodingException;
import javax.xml.XMLConstants;
import javax.xml.transform.stream.StreamSource;
import javax.xml.validation.Schema;
import javax.xml.validation.SchemaFactory;
import javax.xml.validation.Validator;
import org.xml.sax.SAXException;

public class Main {

    public static void main(String[] args) throws UnsupportedEncodingException, FileNotFoundException, SAXException, IOException {
        // make sure our jars are on classpath
        System.out.println("Classpath: " + System.getProperty("java.class.path"));

        // args
        String rng = args[0];
        String xml = args[1];
        File rngFile = new File(rng);
        File xmlFile = new File(xml);

        // setup rng validator through JAXP
        System.setProperty(SchemaFactory.class.getName() + ":" + XMLConstants.RELAXNG_NS_URI, "com.thaiopensource.relaxng.jaxp.XMLSyntaxSchemaFactory");
        SchemaFactory rngSchemaFactory = SchemaFactory.newInstance(XMLConstants.RELAXNG_NS_URI);

        // obtain a schema object
        InputStreamReader rngReader = new InputStreamReader(new FileInputStream(rngFile), "UTF-8");
        Schema schema = rngSchemaFactory.newSchema(new StreamSource(rngReader));

        // validate using schema based validator
        Validator validator = schema.newValidator();
        InputStreamReader xmlReader = new InputStreamReader(new FileInputStream(xmlFile), "UTF-8");
        validator.validate(new StreamSource(xmlReader));
    }
}

第一个参数是具有以下内容的文件的路径:

<element name="balancedString"
   xmlns="http://relaxng.org/ns/structure/1.0"
   datatypeLibrary="http://www.thaiopensource.com/relaxng/datatypes/sample">
  <data type="balancedString"/>
</element>

第二个参数是具有以下内容的文件的路径:

<balancedString>foo(bar(baz))</balancedString>

这给了我以下输出:

Classpath: path\to\RNGDataType\lib\datatype-sample.jar;path\to\RNGDataType\lib\jing.jar;path\to\RNGDataType\build\classes;path\to\RNGDataType\src
Exception in thread "main" org.xml.sax.SAXParseException: datatype library "http://www.thaiopensource.com/relaxng/datatypes/sample" not recognized
...

这清楚地表明无法解析数据类型。据我所知,这个工作的唯一要求(在jing.jar类路径上都有)已经得到满足。datatype-sample.jar那么我做错了什么?

PS:要使上述代码正常工作,您必须将jing.jaranddatatype-sample.jar放在类路径上并为其提供参数,其中第一个是路径,datatype-sample.rng第二个是路径valid.xmlinvalid.xml。所有这些都与一起分发。

Edit1:java -jar当作为具有适当MANIFEST.MF文件的 JAR ( ) 运行时,上述程序在我的 IDE 之外也不起作用。手动设置类路径 ( ) 时也不起作用java -classpath。所以我怀疑实际代码有问题。

4

2 回答 2

3

似乎通过 JAXP API 通过 Jing 使用自定义数据类型库在某种程度上被破坏了。即使它应该起作用,它也不起作用。也许需要在某处设置一些额外的属性,而我只是不知道这一点。

所以我想我通过模仿 Jing 找到了一种解决方法com.thaiopensource.relaxng.util.Driver,因此使用 Jing 自己的 API 来执行验证。请注意,这样做会限制您的代码,因此它仅适用于 Jing。

package rngdatatype;

import com.thaiopensource.validate.SchemaReader;
import com.thaiopensource.validate.ValidationDriver;
import com.thaiopensource.validate.auto.AutoSchemaReader;
import java.io.File;
import java.io.IOException;
import org.xml.sax.InputSource;
import org.xml.sax.SAXException;

public class JingApi {

    public static void main(String[] args) throws SAXException, IOException {
        String rng = args[0];
        String xml = args[1];
        File rngFile = new File(rng);
        File xmlFile = new File(xml);

        SchemaReader sr = new AutoSchemaReader();
        ValidationDriver driver = new ValidationDriver(sr);
        InputSource inRng = ValidationDriver.fileInputSource(rngFile);
        inRng.setEncoding("UTF-8");
        driver.loadSchema(inRng);
        InputSource inXml = ValidationDriver.fileInputSource(xmlFile);
        inXml.setEncoding("UTF-8");
        driver.validate(inXml);
    }
}

这使您能够根据使用自定义数据类型库的 RNG 模式从 java 代码验证 XML 文件。请注意,Diver我前面提到的类不能直接使用。

上面的程序使用与我自己的问题中的示例相同的类路径和参数。

编辑1 ---------------------------------------------

在多摆弄之后,我发现了需要设置的属性,以便在使用自定义数据类型库时让我的 JAXP 示例与 Jing 一起使用。在获得 的实例后只需添加以下行SchemaFactory

rngSchemaFactory.setProperty("http://relaxng.org/properties/datatype-library-factory", new org.relaxng.datatype.helpers.DatatypeLibraryLoader());

这是一个使用 Jing 原生 API 的更优雅的解决方案。

/Edit1 --------------------------------------------

于 2012-06-08T09:22:56.247 回答
0

您的 JAR 文件必须以文件 META-INF/services/org.relaxng.datatype.DatatypeLibraryFactory 的形式包含其他元数据,其中必须包含实现接口 org.relaxng.datatype.DatatypeLibraryFactory 的类的名称

于 2012-06-01T08:17:57.660 回答