3

我正在生成一些 XML 模式,并希望确保我们的生成器正在创建有效的 XML 模式文档(不是 XML)。我试图提出验证 XML Schema 文档的代码,但失败得很惨。我没想到会这么复杂。

  private void validateXsd( String xsdAsString ) {
    try {
      SAXParserFactory factory = SAXParserFactory.newInstance();
      factory.setValidating(true);
      factory.setNamespaceAware(true);
      factory.setFeature( "http://apache.org/xml/features/validation/schema", true );

      SchemaFactory schemaFactory = SchemaFactory.newInstance( XMLConstants.W3C_XML_SCHEMA_NS_URI );

      URL xmlSchemaXSD = this.getClass().getClassLoader().getResource( "com/metamodel/xsd/XMLSchema.xsd" );
      URL xmlSchemaDTD = this.getClass().getClassLoader().getResource( "com/metamodel/xsd/XMLSchema.dtd" );
      URL xmlSchemaDataTypes = this.getClass().getClassLoader().getResource( "com/metamodel/xsd/datatypes.dtd" );

      // requires that XMLSchema.dtd and datatypes.dtd are present in the same directory with the XMLSchema.xsd.
      factory.setSchema( schemaFactory.newSchema( xmlSchemaXSD ) );

      SAXParser parser = factory.newSAXParser();
      // parser.setProperty( "http://java.sun.com/xml/jaxp/properties/schemaLanguage", "http://www.w3.org/2001/XMLSchema" );

      XMLReader reader = parser.getXMLReader();
      reader.setErrorHandler( new SimpleErrorHandler() );
      reader.parse( new InputSource( IOUtils.toInputStream( xsdAsString ) ) );
    } catch( SAXParseException e ) {
      e.printStackTrace( System.err );
    } catch ( ParserConfigurationException e ) {
      e.printStackTrace( System.err );
    } catch ( SAXException e ) {
      e.printStackTrace( System.err );
    } catch ( IOException e ) {
      e.printStackTrace( System.err );
    }
  }

这段代码非常接近我需要做的,但我收到以下错误。

org.xml.sax.SAXParseException: schema_reference.4: Failed to read schema document 'jar:file:/r:/as4ot/lib/metamodel.jar!/com/metamodel/xsd/XMLSchema.xsd', because 1) could not find the document; 2) the document could not be read; 3) the root element of the document is not <xsd:schema>.
    at com.sun.org.apache.xerces.internal.util.ErrorHandlerWrapper.createSAXParseException(ErrorHandlerWrapper.java:236)
    at com.sun.org.apache.xerces.internal.util.ErrorHandlerWrapper.error(ErrorHandlerWrapper.java:172)
    at com.sun.org.apache.xerces.internal.impl.XMLErrorReporter.reportError(XMLErrorReporter.java:382)
    at com.sun.org.apache.xerces.internal.impl.XMLErrorReporter.reportError(XMLErrorReporter.java:316)
    at com.sun.org.apache.xerces.internal.impl.xs.traversers.XSDHandler.reportSchemaError(XSDHandler.java:2245)
    at com.sun.org.apache.xerces.internal.impl.xs.traversers.XSDHandler.getSchema(XSDHandler.java:1590)
    at com.sun.org.apache.xerces.internal.impl.xs.traversers.XSDHandler.parseSchema(XSDHandler.java:438)
    at com.sun.org.apache.xerces.internal.impl.xs.XMLSchemaLoader.loadSchema(XMLSchemaLoader.java:556)
    at com.sun.org.apache.xerces.internal.impl.xs.XMLSchemaLoader.loadGrammar(XMLSchemaLoader.java:523)
    at com.sun.org.apache.xerces.internal.jaxp.validation.xs.SchemaFactoryImpl.newSchema(SchemaFactoryImpl.java:206)
    at javax.xml.validation.SchemaFactory.newSchema(SchemaFactory.java:489)
    at javax.xml.validation.SchemaFactory.newSchema(SchemaFactory.java:521)

我在调用 setSchema(uri) 时使用了 URL,以确保 XMLSchema.dtd 和 datatypes.dtd 是相对的,我希望它们可用,但我不确定如何检查。

我猜它不喜欢 XMLSchema.xsd、XMLSchema.dtd 和 datatypes.dtd 捆绑在我的 jar 中并因此由类加载器加载的事实。无论如何要解决这个问题,以便我可以进入下一个障碍,即org.xml.sax.SAXParseException:src-resolve:无法将名称“xml:lang”解析为(n)“属性声明”组件。假设它的行为与我从 jar 外部的目录路径加载 XMLSchema 时的行为相同。

我的目标是:

  • 从 jar 中加载 XMLSchema.xsd,以便它与代码一起整齐地打包。
  • 克服关于 xml:lang 的下一个错误。

希望我没有要求太多!

谢谢,斯图尔特

4

2 回答 2

3

出于几个原因,我不会使用您描述的方法来验证 XSD,最重要的原因是 XSD 作为一种语言很弱,因此它本身并没有捕获完整的规范。因此,您很可能会验证实际上无效的内容。

对于 XSD 验证,您有专门的处理器;Java 有XSOM;遵循用户指南

于 2012-07-27T18:05:48.763 回答
2

我同意 Petru 的观点,即针对模式文档的模式验证模式并不是一件非常有用的事情,因为它不会检测到模式中的所有错误。确保您的模式有效的唯一真正方法是将其传递给模式处理器。

例如,对于撒克逊

Processor p = new Processor(true);
StreamSource ss = new StreamSource(new File('mySchema.xsd');
p.getSchemaManager().load(ss);

会告诉你你的模式是否有效。

于 2012-07-27T22:03:38.357 回答