/* * The Apache Software License, Version 1.1 * * * Copyright (c) 2000-2002 The Apache Software Foundation. All rights * reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions * are met: * * 1. Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * * 2. Redistributions in binary form must reproduce the above copyright * notice, this list of conditions and the following disclaimer in * the documentation and/or other materials provided with the * distribution. * * 3. The end-user documentation included with the redistribution, * if any, must include the following acknowledgment: * "This product includes software developed by the * Apache Software Foundation (http://www.apache.org/)." * Alternately, this acknowledgment may appear in the software itself, * if and wherever such third-party acknowledgments normally appear. * * 4. The names "Xerces" and "Apache Software Foundation" must * not be used to endorse or promote products derived from this * software without prior written permission. For written * permission, please contact apache@apache.org. * * 5. Products derived from this software may not be called "Apache", * nor may "Apache" appear in their name, without prior written * permission of the Apache Software Foundation. * * THIS SOFTWARE IS PROVIDED ``AS IS'' AND ANY EXPRESSED OR IMPLIED * WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE * DISCLAIMED. IN NO EVENT SHALL THE APACHE SOFTWARE FOUNDATION OR * ITS CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF * USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT * OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF * SUCH DAMAGE. * ==================================================================== * * This software consists of voluntary contributions made by many * individuals on behalf of the Apache Software Foundation and was * originally based on software copyright (c) 1999, International * Business Machines, Inc., http://www.apache.org. For more * information on the Apache Software Foundation, please see * . */ package com.sun.org.apache.xerces.internal.jaxp.validation; import javax.xml.validation.TypeInfoProvider; import javax.xml.validation.ValidatorHandler; import com.sun.org.apache.xerces.internal.impl.Constants; import com.sun.org.apache.xerces.internal.impl.XMLEntityManager; import com.sun.org.apache.xerces.internal.impl.XMLErrorReporter; import com.sun.org.apache.xerces.internal.impl.dv.XSSimpleType; import com.sun.org.apache.xerces.internal.impl.validation.ValidationManager; import com.sun.org.apache.xerces.internal.impl.xs.XSMessageFormatter; import com.sun.org.apache.xerces.internal.util.DraconianErrorHandler; import com.sun.org.apache.xerces.internal.util.LocatorWrapper; import com.sun.org.apache.xerces.internal.util.NamespaceSupport; import com.sun.org.apache.xerces.internal.util.SymbolTable; import com.sun.org.apache.xerces.internal.util.XMLAttributesImpl; import com.sun.org.apache.xerces.internal.xni.Augmentations; import com.sun.org.apache.xerces.internal.xni.QName; import com.sun.org.apache.xerces.internal.xni.XMLAttributes; import com.sun.org.apache.xerces.internal.xni.XMLLocator; import com.sun.org.apache.xerces.internal.xni.XMLString; import com.sun.org.apache.xerces.internal.xni.parser.XMLComponentManager; import com.sun.org.apache.xerces.internal.xni.parser.XMLConfigurationException; import com.sun.org.apache.xerces.internal.xni.parser.XMLDocumentFilter; import com.sun.org.apache.xerces.internal.xs.AttributePSVI; import com.sun.org.apache.xerces.internal.xs.ElementPSVI; import com.sun.org.apache.xerces.internal.xs.ItemPSVI; import com.sun.org.apache.xerces.internal.xs.XSTypeDefinition; import org.w3c.dom.TypeInfo; import org.w3c.dom.ls.LSResourceResolver; import org.xml.sax.Attributes; import org.xml.sax.ContentHandler; import org.xml.sax.ErrorHandler; import org.xml.sax.Locator; import org.xml.sax.SAXException; import org.xml.sax.SAXNotRecognizedException; import org.xml.sax.SAXNotSupportedException; /** * {@link ValidatorHandler} implementation that wraps * {@link InsulatedValidatorComponent}. * *

* This class implements all the SAX {@link org.xml.sax.ContentHandler} * methods and turn SAX events into XNI events. * *

* This class also implements {@link XMLComponentManager} * to host a validator. * * @author * Kohsuke Kawaguchi (kohsuke.kawaguchi@sun.com) */ final class ValidatorHandlerImpl extends ValidatorHandler { // TODO: reuse SAX2XNI /** * The actual validator. */ private final InsulatedValidatorComponent validator; /** * validator.getValidator(). */ private final XMLDocumentFilter validatorFilter; /** * Used to adopt the output from a validtor to the input of * the user specified {@link ContentHandler}. */ private final XNI2SAXEx xni2sax = new XNI2SAXEx(); // XMLSchemaValidator needs various helper components to work. private final SymbolTable symbolTable = new SymbolTable(); private final NamespaceSupport nsContext = new NamespaceSupport(); private final ValidationManager validationManager = new ValidationManager(); private final XMLEntityManager entityManager = new XMLEntityManager(); /** error reporter is used to format error messages. */ private final XMLErrorReporter errorReporter = new XMLErrorReporter(); /** User-specified error handler. Maybe null. */ private ErrorHandler errorHandler; /** This flag is set to true while we are processing the startElement event. */ private boolean inStartElement; /** The value of the http://xml.org/sax/features/namespace-prefixes feature. */ private boolean namespacePrefixesFeature = false; //private Hashtable fProperties; /** * Used by {@link XMLDTDValidator} to report errors. */ private final ErrorHandlerAdaptor xercesErrorHandler = new ErrorHandlerAdaptor() { protected ErrorHandler getErrorHandler() { if(errorHandler==null ) return DraconianErrorHandler.theInstance; else return errorHandler; } }; /** User-specified entity resolver. Maybe null. */ private LSResourceResolver resourceResolver; ValidatorHandlerImpl( InsulatedValidatorComponent validator ) { this.validator = validator; this.validatorFilter = validator.getValidator(); // format error message with Schema aware formatter errorReporter.putMessageFormatter( XSMessageFormatter.SCHEMA_DOMAIN, new XSMessageFormatter()); } /** * Obtains the current augmentation. *

* used for {@link javax.xml.validation.TypeInfoProvider}. * * @return * may return null. */ private final Augmentations getCurrentAugmentation() { return xni2sax.getCurrentAugmentation(); } /** * Obtains the current attributes. * * @throws IllegalStateException */ private final XMLAttributes getCurrentAttributes() { return xni2sax.getCurrentAttributes(); } public boolean getFeature(String name) throws SAXNotRecognizedException, SAXNotSupportedException { if( name.equals("http://xml.org/sax/features/namespace-prefixes") ) return namespacePrefixesFeature; return super.getFeature(name); } public void setFeature(String name, boolean value) throws SAXNotRecognizedException, SAXNotSupportedException { if( name.equals("http://xml.org/sax/features/namespace-prefixes") ) { namespacePrefixesFeature = value; return; } super.setFeature(name, value); } // // // ValidaorHandler implementation // // public boolean isValidSoFar() { return !xercesErrorHandler.hadError(); } public void setErrorHandler(ErrorHandler errorHandler) { this.errorHandler = errorHandler; } public ErrorHandler getErrorHandler() { return errorHandler; } public void setResourceResolver(LSResourceResolver entityResolver) { this.resourceResolver = entityResolver; } public LSResourceResolver getResourceResolver() { return resourceResolver; } public final void setContentHandler(ContentHandler result) { xni2sax.setContentHandler(result); if(result==null) validatorFilter.setDocumentHandler(null); else validatorFilter.setDocumentHandler(xni2sax); } public final ContentHandler getContentHandler() { return xni2sax.getContentHandler(); } // // // XMLComponentManager implementation // // private final XMLComponentManager manager = new XMLComponentManager() { public Object getProperty( String propName ) { if( propName.equals(XercesConstants.SYMBOL_TABLE) ) return symbolTable; if( propName.equals(XercesConstants.VALIDATION_MANAGER) ) return validationManager; if( propName.equals(XercesConstants.ERROR_REPORTER) ) return errorReporter; if( propName.equals(XercesConstants.ERROR_HANDLER) ) return xercesErrorHandler; if( propName.equals(XercesConstants.ENTITY_MANAGER) ) return entityManager; if( propName.equals(XercesConstants.ENTITY_RESOLVER) ) return entityManager; throw new XMLConfigurationException( XMLConfigurationException.NOT_RECOGNIZED, propName ); } public boolean getFeature( String propName ) { // if( propName.equals(XercesConstants.SCHEMA_VALIDATION) ) // // this flag will turn on the validation // return true; if( propName.equals(XercesConstants.VALIDATION) ) return true;//TODO:: Configure throw new XMLConfigurationException( XMLConfigurationException.NOT_RECOGNIZED, propName ); } }; // // // ContentHandler implementation // // public void startDocument() throws SAXException { try { resetComponents(); XMLLocator xmlLocator = (locator==null)?null:new LocatorWrapper(locator); // set the locator to the error reporter errorReporter.setDocumentLocator(xmlLocator); validatorFilter.startDocument( xmlLocator, null, nsContext, null); } catch( WrappedSAXException e ) { throw e.exception; } } /** * Resets the components we use internally. */ private void resetComponents() { // reset the error flag when we start a new validation. xercesErrorHandler.reset(); nsContext.reset(); errorReporter.reset(manager); validator.reset(manager); } public void endDocument() throws SAXException { try { validatorFilter.endDocument(null); } catch( WrappedSAXException e ) { throw e.exception; } } public void startElement( String uri, String local, String qname, Attributes att ) throws SAXException { try { inStartElement = true; validatorFilter.startElement(createQName(uri,local,qname),createAttributes(att),null); } catch( WrappedSAXException e ) { throw e.exception; } finally { inStartElement = false; } } public void endElement( String uri, String local, String qname ) throws SAXException { try { validatorFilter.endElement(createQName(uri,local,qname),null); } catch( WrappedSAXException e ) { throw e.exception; } } public void characters( char[] buf, int offset, int len ) throws SAXException { try { validatorFilter.characters(new XMLString(buf,offset,len),null); } catch( WrappedSAXException e ) { throw e.exception; } } public void ignorableWhitespace( char[] buf, int offset, int len ) throws SAXException { try { validatorFilter.ignorableWhitespace(new XMLString(buf,offset,len),null); } catch( WrappedSAXException e ) { throw e.exception; } } public void startPrefixMapping( String prefix, String uri ) { nsContext.pushContext(); nsContext.declarePrefix(prefix,uri); } public void endPrefixMapping( String prefix ) { nsContext.popContext(); } public void processingInstruction( String target, String data ) throws SAXException { try { validatorFilter.processingInstruction( symbolize(target),createXMLString(data),null); } catch( WrappedSAXException e ) { throw e.exception; } } public void skippedEntity( String name ) { // there seems to be no corresponding method on XMLDocumentFilter. // just pass it down to the output, if any. ContentHandler handler = getContentHandler(); if( handler!=null ) skippedEntity(name); } private Locator locator; public void setDocumentLocator( Locator _loc ) { this.locator = _loc; } public TypeInfoProvider getTypeInfoProvider() { return typeInfoProvider; } /** * {@link TypeInfoProvider} implementation. * * REVISIT: I'm not sure if this code should belong here. */ private final TypeInfoProvider typeInfoProvider = new TypeInfoProvider() { /** * Throws a {@link IllegalStateException} if we are not in * the startElement callback. the JAXP API requires this * for most of the public methods. */ private void checkState() { if( !inStartElement ) throw new IllegalStateException(); } public TypeInfo getAttributeTypeInfo(int index) { checkState(); return getAttributeType(index); } private XSTypeDefinition getAttributeType( int index ) { checkState(); XMLAttributes atts = getCurrentAttributes(); if( index<0 || atts.getLength()<=index ) throw new IndexOutOfBoundsException(Integer.toString(index)); Augmentations augs = atts.getAugmentations(index); if(augs==null) return null; AttributePSVI psvi = (AttributePSVI)augs.getItem(Constants.ATTRIBUTE_PSVI); return getTypeInfoFromPSVI(psvi); } public TypeInfo getAttributeTypeInfo(String attributeUri, String attributeLocalName) { checkState(); return getAttributeTypeInfo(getCurrentAttributes().getIndex(attributeUri,attributeLocalName)); } public TypeInfo getAttributeTypeInfo(String attributeQName) { checkState(); return getAttributeTypeInfo(getCurrentAttributes().getIndex(attributeQName)); } public TypeInfo getElementTypeInfo() { checkState(); Augmentations augs = getCurrentAugmentation(); if(augs==null) return null; ElementPSVI psvi = (ElementPSVI)augs.getItem(Constants.ELEMENT_PSVI); return getTypeInfoFromPSVI(psvi); } private XSTypeDefinition getTypeInfoFromPSVI( ItemPSVI psvi ) { if(psvi==null) return null; // TODO: make sure if this is correct. // TODO: since the number of types in a schema is quite limited, // TypeInfoImpl should be pooled. Even better, it should be a part // of the element decl. if( psvi.getValidity()== ElementPSVI.VALIDITY_VALID ) { XSTypeDefinition t = psvi.getMemberTypeDefinition(); if(t!=null) return t; } XSTypeDefinition t = psvi.getTypeDefinition(); if(t!=null) return t; // TODO: can t be null? return null; } public boolean isIdAttribute(int index) { checkState(); XSSimpleType type = (XSSimpleType)getAttributeType(index); if(type==null) return false; return type.isIDType(); } public boolean isSpecified(int index) { checkState(); return getCurrentAttributes().isSpecified(index); } }; // // // helper methods // // /** Symbolizes the specified string. */ private String symbolize(String s) { if (s == null) return null; else return symbolTable.addSymbol(s); } /** Creates a QName object. */ private QName createQName(String uri, String local, String raw) { if( local.length()==0 ) { // if naemspace processing is turned off, local could be "". // in that case, treat everything to be in the no namespace. uri = ""; local = raw; } int idx = raw.indexOf(':'); String prefix; if (idx < 0) prefix = null; else prefix = raw.substring(0, idx); if (uri != null && uri.length() == 0) uri = null; // XNI uses null whereas SAX uses the empty string return new QName(symbolize(prefix), symbolize(local), symbolize(raw), symbolize(uri)); } /** only one instance of XMLAttributes is used. */ private final XMLAttributes xa = new XMLAttributesImpl(); /** Creates an XMLAttributes object. */ private XMLAttributes createAttributes(Attributes att) { xa.removeAllAttributes(); int len = att.getLength(); for (int i = 0; i < len; i++) { int idx = xa.addAttribute( createQName(att.getURI(i), att.getLocalName(i), att.getQName(i)), att.getType(i), att.getValue(i)); // attributes present in the original SAX event streams // are considered as "specified". xa.setSpecified(idx,true); } return xa; } private XMLString createXMLString(String str) { // with my patch // return new XMLString(str); // for now return new XMLString(str.toCharArray(), 0, str.length()); } /** * Resets this handler. *

* Meaning resets all the user-specified configurations to the * initial state, then also resets all the components. */ public void reset() { resetComponents(); errorHandler = null; resourceResolver = null; } }