在工作中,我们刚刚将一个旧的 web 应用程序从 struts 1.1 迁移到 1.2.9(希望是迁移到 1.3 的第一步),但是我们现在遇到了 commons digester 的问题。Struts 1.2.9 使用 commons-digester 1.6。
当我们尝试解析我们的 XML 文件之一时,我们得到了异常:
org.xml.sax.SAXParseException: Attribute "" bound to namespace "null" was already specified for element "metric".
at org.apache.xerces.util.ErrorHandlerWrapper.createSAXParseException(ErrorHandlerWrapper.java:232)
at org.apache.xerces.util.ErrorHandlerWrapper.fatalError(ErrorHandlerWrapper.java:213)
at org.apache.xerces.impl.XMLErrorReporter.reportError(XMLErrorReporter.java:385)
at org.apache.xerces.impl.XMLErrorReporter.reportError(XMLErrorReporter.java:315)
at org.apache.xerces.impl.dtd.XMLNSDTDValidator.startNamespaceScope(XMLNSDTDValidator.java:242)
at org.apache.xerces.impl.dtd.XMLDTDValidator.handleStartElement(XMLDTDValidator.java:1980)
at org.apache.xerces.impl.dtd.XMLDTDValidator.startElement(XMLDTDValidator.java:802)
at org.apache.xerces.impl.XMLNSDocumentScannerImpl.scanStartElement(XMLNSDocumentScannerImpl.java:313)
at org.apache.xerces.impl.XMLNSDocumentScannerImpl$NSContentDispatcher.scanRootElementHook(XMLNSDocumentScannerImpl.java:610)
at org.apache.xerces.impl.XMLDocumentFragmentScannerImpl$FragmentContentDispatcher.dispatch(XMLDocumentFragmentScannerImpl.java:1608)
at org.apache.xerces.impl.XMLDocumentFragmentScannerImpl.scanDocument(XMLDocumentFragmentScannerImpl.java:346)
at org.apache.xerces.parsers.DTDConfiguration.parse(DTDConfiguration.java:529)
at org.apache.xerces.parsers.DTDConfiguration.parse(DTDConfiguration.java:585)
at org.apache.xerces.parsers.XMLParser.parse(XMLParser.java:152)
at org.apache.xerces.parsers.AbstractSAXParser.parse(AbstractSAXParser.java:1142)
at org.apache.commons.digester.Digester.parse(Digester.java:1572)
at com.foo.ctms.framework.metrics.parser.MetricsXMLParser$InternalDigester.parse(MetricsXMLParser.java:54)
at com.foo.ctms.framework.metrics.parser.MetricsXMLParser.parse(MetricsXMLParser.java:40)
在调查这个问题时,我试图找到一个最简单的情况,这就是我目前所拥有的:
package com.foo.ctms.framework.metrics.parser;
import org.apache.commons.digester.Digester;
import org.xml.sax.InputSource;
import org.xml.sax.SAXException;
import java.io.IOException;
import java.net.URL;
/**
* Class that provides methods for parsing metrics definitions defined via XML and populating an object graph of JavaBeans
* representing the definition.
* @version $Revision: 41470 $
*/
public final class MetricsXMLParser {
/**
* The set of public identifiers, and corresponding resource names, for the versions of the configuration file DTD that we know
* about. The key is the name of the resource as in the XMl file, and the value is the location of the resource with respect to
* the <code>ClassLoader</code> that this code in running in.
*/
private static final String registrations[] = {"-//Foo Inc.//DTD Portal Metrics 1.0//EN", "metrics.dtd"};
private MetricsXMLParser() {
}
/**
* Parses a metric definition specified as an <code>InputStream</code>.
* @param url The metrics definition to parse. Must not be <code>null</code>.
* @throws IOException if an I/O error occured while attempting to parse
* @throws SAXException if an XML parsing error occured
*/
public static MetricDefinition parse(URL url)
throws IOException, SAXException {
InternalDigester digester = new InternalDigester();
return digester.parse(url);
}
private static final class InternalDigester {
private final Digester digester;
/**
* Parses a metric definition specified as an <code>InputStream</code>.
* @param input The metrics definition to parse. Must not be <code>null</code>.
* @throws IOException if an I/O error occured while attempting to parse
* @throws SAXException if an XML parsing error occured
*/
public MetricDefinition parse(URL input)
throws IOException, SAXException {
return (MetricDefinition)digester.parse(new InputSource(input.toString()));
}
private InternalDigester() {
digester = new Digester();
digester.setValidating(true);
for (int i = 0; i < MetricsXMLParser.registrations.length; i += 2) {
URL url = getClass().getResource(MetricsXMLParser.registrations[i + 1]);
if (url != null) {
digester.register(MetricsXMLParser.registrations[i], url.toString());
}
}
digester.addObjectCreate("metric", MetricDefinition.class);
digester.addSetProperties("metric");
}
}
}
这是给定的XML:
<?xml version='1.0' encoding='windows-1252'?>
<!DOCTYPE metric PUBLIC "-//Foo Inc.//DTD Portal Metrics 1.0//EN" "metrics.dtd">
<metric name="metricsConfig" defaultView="trials">
</metric>
DTD 目前归结为:
<!-- A metric element is the document root -->
<!ELEMENT metric ANY>
<!-- A metric has a name and a default view. The default view must
exactly match the name of one of the nested views -->
<!ATTLIST metric
name CDATA #REQUIRED
defaultView CDATA #IMPLIED
>
有谁知道我做错了什么?
如果我删除 defaultView 属性,我不会收到错误消息。
根据 sfussenegger 的建议,我现在尝试了以下(非消化器)代码:
try {
SAXParserFactory factory = SAXParserFactory.newInstance();
factory.setNamespaceAware(false);
factory.setValidating(true);
SAXParser parser = factory.newSAXParser();
XMLReader reader = parser.getXMLReader();
reader.parse(new InputSource(url.toString()));
} catch (Exception e) {
e.printStackTrace();
}
并且无法重现该问题,但在使用我的工厂之前添加以下内容(commons-digester 在 XercesParser 中也这样做)给出了相同的异常:
factory.setFeature("http://apache.org/xml/features/validation/dynamic", true);
factory.setFeature("http://apache.org/xml/features/validation/schema", true);
最后,我们决定尝试更现代的 xerces (2.7.1) 版本,这似乎可行。