尝试做主题。
我正在尝试使用来自文件(schemasource = 1)和clob(schemasource = 0)的xsd。 我有两个xsd模式common_types.xsd和migom.xsd。第二个包括第一。 问题是当我从文件中使用common_types模式时出现错误
ORA-29532:Java调用因未捕获的Java异常而终止:oracle.xml.parser.v2.XMLParseException:发生内部错误情况。
当我验证xml时,只有第一个模式已从clob中读取,我获得成功,但是当我添加第二个xsd时,我得到相同的错误,它什么也没说。
create or replace and compile java source named XmlTools AS
import javax.xml.parsers.ParserConfigurationException;
import javax.xml.parsers.SAXParser;
import javax.xml.parsers.SAXParserFactory;
import org.xml.sax.XMLReader;
import org.xml.sax.InputSource;
import oracle.sql.CLOB;
import java.io.IOException;
import org.xml.sax.SAXException;
import java.sql.SQLException;
import java.lang.IllegalArgumentException;
import oracle.xml.parser.v2.XMLParseException;
import javax.xml.parsers.ParserConfigurationException;
import java.io.*;
public class XmlValidator
{
static final String JAXP_SCHEMA_LANGUAGE = "http://java.sun.com/xml/jaxp/properties/schemaLanguage";
static final String W3C_XML_SCHEMA = "http://www.w3.org/2001/XMLSchema";
static final String JAXP_SCHEMA_SOURCE = "http://java.sun.com/xml/jaxp/properties/schemaSource";
public static void ValidateDocument(int schemasource, oracle.sql.CLOB schemadoc, oracle.sql.CLOB schemadoc1, oracle.sql.CLOB xmldoc) throws SAXException, IOException, SQLException, ParserConfigurationException, XMLParseException, IllegalArgumentException {
try
{
File myfile = new File(".//XML//common_types.xsd");
if (myfile.exists())
{
Serv.log("ValidateDocument", "file size" + Long.toString(myfile.length()));
}
/*else
{
Serv.log("ValidateDocument", "file doesn't exists" );
}*/
Serv.log("ValidateDocument", "1" );
SAXParserFactory factory = SAXParserFactory.newInstance();
factory.setValidating(true);
factory.setNamespaceAware(true);
Serv.log("ValidateDocument", "2" );
SAXParser saxParser = factory.newSAXParser();
saxParser.setProperty(JAXP_SCHEMA_LANGUAGE, W3C_XML_SCHEMA);
if (schemasource == 0)
{
InputSource schemaIs = new InputSource(schemadoc.getCharacterStream());
InputSource schemaIs1 = new InputSource(schemadoc1.getCharacterStream());
InputSource[] schemas = {schemaIs, schemaIs1};
//saxParser.setProperty(JAXP_SCHEMA_SOURCE, schemaIs);
saxParser.setProperty(JAXP_SCHEMA_SOURCE, schemas);
}
else
{
saxParser.setProperty(JAXP_SCHEMA_SOURCE, ".//XML//common_types.xsd");
}
XMLReader reader = saxParser.getXMLReader();
//Получаем входной XML документ
InputSource documentIs = new InputSource(xmldoc.getCharacterStream());
Serv.log("ValidateDocument", "3" );
//Запуск разбора
reader.parse(documentIs);
Serv.log("ValidateDocument", "4" );
documentIs = null;
}
/*catch (SAXException e)
{
Serv.log("ValidateDocument", "SAXException" );
Serv.log("ValidateDocument", "document is not valid because ");
Serv.log("ValidateDocument", e.getMessage());
throw(e);
}*/
catch (ParserConfigurationException e)
{
Serv.log("ValidateDocument", "ParserConfigurationException" );
throw(e);
}
catch (IOException e)
{
Serv.log("ValidateDocument", "IOException" );
throw(e);
}
catch (XMLParseException e)
{
Serv.log("ValidateDocument", "XMLParseException" );
Serv.log("ValidateDocument", e.getMessage());
StackTraceElement[] stack = e.getStackTrace();
for (int i = 0; i < stack.length; i++)
{
Serv.log("stacktrace element no " + Integer.toString(i), "toString: " + stack[i].toString());
Serv.log("stacktrace element no " + Integer.toString(i), "file name: " + stack[i].getFileName() + ", class name: " + stack[i].getClassName() + ", method name: " + stack[i].getMethodName() + ", line : " + stack[i].getLineNumber());
}
throw(e);
}
catch (IllegalArgumentException e)
{
Serv.log("ValidateDocument", "IllegalArgumentException" );
Serv.log("ValidateDocument", e.getMessage());
throw(e);
}
}
}
从java stacktrace获取的其他信息:
文件名:XMLError.java,类名:oracle.xml.parser.v2.XMLError,方法名称:flushErrors1,行:320 文件名:NonValidatingParser.java,类名:oracle.xml.parser.v2.NonValidatingParser,方法名:parseDocument,行:300 文件名:XMLParser.java,类名:oracle.xml.parser.v2.XMLParser,方法名:解析,行:200 文件名:XMLTOOLS,类名:XmlValidator,方法名称:ValidateDocument,行:86
我的oracle版本是Oracle数据库10g企业版版本10.2.0.1.0 - Prod 但我的目标是让它适用于从9开始的所有版本
答案 0 :(得分:0)
更新:
所以,您将这些文件作为CLOB加载到数据库中。你把它们插入数据库时是否尊重它们的xml编码?
答案 1 :(得分:0)
根据你的stacktrace,我看到NonValidatingParser正在被使用。即使您没有提到这是一个问题,但这是出乎意料的。我知道xmlparserv2有一个验证解析器,所以我查看了xmlparserv2.jar中反编译的XMLParser(自从我使用OC4J以来,我已将它与我一起使用)。
反编译的来源如下。如您所见,构造函数默认采用非验证解析器。使用setProperty应该将它切换到ValidatingParser,但因为没有发生。
XMLParser() { parser = new NonValidatingParser(); }
我在反编译代码中找不到setProperty方法。这是不寻常的,但我没有调查过。要启用xml验证,我相信您必须使用不同的API方法。我相信setAttribute方法会做你想要的。
public void setAttribute(String s,Object obj)抛出IllegalArgumentException {
............ if(s == "http://java.sun.com/xml/jaxp/properties/schemaSource") schemaSource = obj; else if(s == "http://java.sun.com/xml/jaxp/properties/schemaLanguage") { if(((String)obj).equals("http://www.w3.org/2001/XMLSchema")) setValidationMode(3); getSchemaValidator().setJAXP(true); } ....................... attributes.put(s, obj); }
我在部署在OC4J上的应用程序中使用过它,我知道它使用相同的解析器。 代码示例如下所示
DocumentBuilderFactory factory = DocumentBuilderFactory.newInstance();
factory.setNamespaceAware(真);
factory.setValidating(真);
factory.setAttribute( “http://java.sun.com/xml/jaxp/properties/schemaLanguage”, “http://www.w3.org/2001/XMLSchema”);
factory.setAttribute( “http://java.sun.com/xml/jaxp/properties/schemaSource”, ClassUtil.getResourceAsStream(schemaSourceLocation));
希望这有帮助。