com.sun.xml.bind.v2.runtime.unmarshaller.XmlVisitor Maven / Gradle / Ivy
/*
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS HEADER.
*
* Copyright (c) 1997-2011 Oracle and/or its affiliates. All rights reserved.
*
* The contents of this file are subject to the terms of either the GNU
* General Public License Version 2 only ("GPL") or the Common Development
* and Distribution License("CDDL") (collectively, the "License"). You
* may not use this file except in compliance with the License. You can
* obtain a copy of the License at
* https://glassfish.dev.java.net/public/CDDL+GPL_1_1.html
* or packager/legal/LICENSE.txt. See the License for the specific
* language governing permissions and limitations under the License.
*
* When distributing the software, include this License Header Notice in each
* file and include the License file at packager/legal/LICENSE.txt.
*
* GPL Classpath Exception:
* Oracle designates this particular file as subject to the "Classpath"
* exception as provided by Oracle in the GPL Version 2 section of the License
* file that accompanied this code.
*
* Modifications:
* If applicable, add the following below the License Header, with the fields
* enclosed by brackets [] replaced by your own identifying information:
* "Portions Copyright [year] [name of copyright owner]"
*
* Contributor(s):
* If you wish your version of this file to be governed by only the CDDL or
* only the GPL Version 2, indicate your decision by adding "[Contributor]
* elects to include this software in this distribution under the [CDDL or GPL
* Version 2] license." If you don't indicate a single choice of license, a
* recipient has the option to distribute your version of this file under
* either the CDDL, the GPL Version 2 or to extend the choice of license to
* its licensees as provided above. However, if you add GPL Version 2 code
* and therefore, elected the GPL Version 2 license, then the option applies
* only if the new code is made subject to such option by the copyright
* holder.
*/
package com.sun.xml.bind.v2.runtime.unmarshaller;
import javax.xml.namespace.NamespaceContext;
import org.xml.sax.SAXException;
/**
* Walks the XML document structure.
*
* Implemented by the unmarshaller and called by the API-specific connectors.
*
* Event Call Sequence
*
* The {@link XmlVisitor} expects the event callbacks in the following order:
*
* CALL SEQUENCE := startDocument ELEMENT endDocument
* ELEMENT := startPrefixMapping ELEMENT endPrefixMapping
* | startElement BODY endElement
* BODY := text? (ELEMENT text?)*
*
* Note in particular that text events may not be called in a row;
* consecutive characters (even those separated by PIs and comments)
* must be reported as one event, unlike SAX.
*
*
* All namespace URIs, local names, and prefixes of element and attribute
* names must be interned. qnames need not be interned.
*
*
*
Typed PCDATA
* For efficiency, JAXB RI defines a few {@link CharSequence} implementations
* that can be used as a parameter to the {@link #text(CharSequence)} method.
* For example, see {@link Base64Data}.
*
* Error Handling
* The visitor may throw {@link SAXException} to abort the unmarshalling process
* in the middle.
*
* @author Kohsuke Kawaguchi
*/
public interface XmlVisitor {
/**
* Notifies a start of the document.
*
* @param locator
* This live object returns the location information as the parsing progresses.
* must not be null.
* @param nsContext
* Some broken XML APIs can't iterate all the in-scope namespace bindings,
* which makes it impossible to emulate {@link #startPrefixMapping(String, String)} correctly
* when unmarshalling a subtree. Connectors that use such an API can
* pass in additional {@link NamespaceContext} object that knows about the
* in-scope namespace bindings. Otherwise (and normally) it is null.
*
*
* Ideally this object should be immutable and only represent the namespace URI bindings
* in the context (those done above the element that JAXB started unmarshalling),
* but it can also work even if it changes as the parsing progress (to include
* namespaces declared on the current element being parsed.)
*/
void startDocument(LocatorEx locator, NamespaceContext nsContext) throws SAXException;
void endDocument() throws SAXException;
/**
* Notifies a start tag of a new element.
*
* namespace URIs and local names must be interned.
*/
void startElement(TagName tagName) throws SAXException;
void endElement(TagName tagName) throws SAXException;
/**
* Called before {@link #startElement} event to notify a new namespace binding.
*/
void startPrefixMapping( String prefix, String nsUri ) throws SAXException;
/**
* Called after {@link #endElement} event to notify the end of a binding.
*/
void endPrefixMapping( String prefix ) throws SAXException;
/**
* Text events.
*
*
* The caller should consult {@link TextPredictor} to see
* if the unmarshaller is expecting any PCDATA. If the above is returning
* false, the caller is OK to skip any text in XML. The net effect is
* that we can ignore whitespaces quickly.
*
* @param pcdata
* represents character data. This object can be mutable
* (such as {@link StringBuilder}); it only needs to be fixed
* while this method is executing.
*/
void text( CharSequence pcdata ) throws SAXException;
/**
* Returns the {@link UnmarshallingContext} at the end of the chain.
*
* @return
* always return the same object, so caching the result is recommended.
*/
UnmarshallingContext getContext();
/**
* Gets the predictor that can be used for the caller to avoid
* calling {@link #text(CharSequence)} unnecessarily.
*/
TextPredictor getPredictor();
interface TextPredictor {
/**
* Returns true if the visitor is expecting a text event as the next event.
*
*
* This is primarily intended to be used for optimization to avoid buffering
* characters unnecessarily. If this method returns false and the connector
* sees whitespace it can safely skip it.
*
*
* If this method returns true, all the whitespaces are considered significant
* and thus need to be reported as a {@link XmlVisitor#text} event. Furthermore,
* if the element has no children (like <foo/>), then it has to be reported
* an empty {@link XmlVisitor#text} event.
*/
boolean expectText();
}
}