/*
* 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
*
* 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; } }