src/share/jaxws_classes/com/sun/xml/internal/bind/v2/runtime/unmarshaller/UnmarshallerImpl.java

Tue, 06 Mar 2012 16:09:35 -0800

author
ohair
date
Tue, 06 Mar 2012 16:09:35 -0800
changeset 286
f50545b5e2f1
child 368
0989ad8c0860
permissions
-rw-r--r--

7150322: Stop using drop source bundles in jaxws
Reviewed-by: darcy, ohrstrom

     1 /*
     2  * Copyright (c) 1997, 2011, Oracle and/or its affiliates. All rights reserved.
     3  * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
     4  *
     5  * This code is free software; you can redistribute it and/or modify it
     6  * under the terms of the GNU General Public License version 2 only, as
     7  * published by the Free Software Foundation.  Oracle designates this
     8  * particular file as subject to the "Classpath" exception as provided
     9  * by Oracle in the LICENSE file that accompanied this code.
    10  *
    11  * This code is distributed in the hope that it will be useful, but WITHOUT
    12  * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
    13  * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
    14  * version 2 for more details (a copy is included in the LICENSE file that
    15  * accompanied this code).
    16  *
    17  * You should have received a copy of the GNU General Public License version
    18  * 2 along with this work; if not, write to the Free Software Foundation,
    19  * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
    20  *
    21  * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
    22  * or visit www.oracle.com if you need additional information or have any
    23  * questions.
    24  */
    26 package com.sun.xml.internal.bind.v2.runtime.unmarshaller;
    28 import java.io.IOException;
    29 import java.io.InputStream;
    31 import javax.xml.bind.JAXBContext;
    32 import javax.xml.bind.JAXBElement;
    33 import javax.xml.bind.JAXBException;
    34 import javax.xml.bind.PropertyException;
    35 import javax.xml.bind.UnmarshalException;
    36 import javax.xml.bind.Unmarshaller;
    37 import javax.xml.bind.UnmarshallerHandler;
    38 import javax.xml.bind.ValidationEvent;
    39 import javax.xml.bind.ValidationEventHandler;
    40 import javax.xml.bind.annotation.adapters.XmlAdapter;
    41 import javax.xml.bind.attachment.AttachmentUnmarshaller;
    42 import javax.xml.bind.helpers.AbstractUnmarshallerImpl;
    43 import javax.xml.stream.XMLEventReader;
    44 import javax.xml.stream.XMLStreamConstants;
    45 import javax.xml.stream.XMLStreamException;
    46 import javax.xml.stream.XMLStreamReader;
    47 import javax.xml.stream.events.XMLEvent;
    48 import javax.xml.transform.Source;
    49 import javax.xml.transform.dom.DOMSource;
    50 import javax.xml.transform.sax.SAXSource;
    51 import javax.xml.transform.stream.StreamSource;
    52 import javax.xml.validation.Schema;
    54 import com.sun.xml.internal.bind.IDResolver;
    55 import com.sun.xml.internal.bind.api.ClassResolver;
    56 import com.sun.xml.internal.bind.unmarshaller.DOMScanner;
    57 import com.sun.xml.internal.bind.unmarshaller.InfosetScanner;
    58 import com.sun.xml.internal.bind.unmarshaller.Messages;
    59 import com.sun.xml.internal.bind.v2.ClassFactory;
    60 import com.sun.xml.internal.bind.v2.runtime.AssociationMap;
    61 import com.sun.xml.internal.bind.v2.runtime.JAXBContextImpl;
    62 import com.sun.xml.internal.bind.v2.runtime.JaxBeanInfo;
    64 import java.io.Closeable;
    65 import org.w3c.dom.Document;
    66 import org.w3c.dom.Element;
    67 import org.w3c.dom.Node;
    68 import org.xml.sax.InputSource;
    69 import org.xml.sax.SAXException;
    70 import org.xml.sax.XMLReader;
    71 import org.xml.sax.helpers.DefaultHandler;
    73 /**
    74  * Default Unmarshaller implementation.
    75  *
    76  * <p>
    77  * This class can be extended by the generated code to provide
    78  * type-safe unmarshall methods.
    79  *
    80  * @author
    81  *  <a href="mailto:kohsuke.kawaguchi@sun.com">Kohsuke KAWAGUCHI</a>
    82  */
    83 public final class UnmarshallerImpl extends AbstractUnmarshallerImpl implements ValidationEventHandler, Closeable
    84 {
    85     /** Owning {@link JAXBContext} */
    86     protected final JAXBContextImpl context;
    88     /**
    89      * schema which will be used to validate during calls to unmarshal
    90      */
    91     private Schema schema;
    93     public final UnmarshallingContext coordinator;
    95     /** Unmarshaller.Listener */
    96     private Listener externalListener;
    98     /**
    99      * The attachment unmarshaller used to support MTOM and swaRef.
   100      */
   101     private AttachmentUnmarshaller attachmentUnmarshaller;
   102     private IDResolver idResolver = new DefaultIDResolver();
   104     public UnmarshallerImpl( JAXBContextImpl context, AssociationMap assoc ) {
   105         this.context = context;
   106         this.coordinator = new UnmarshallingContext( this, assoc );
   108         try {
   109             setEventHandler(this);
   110         } catch (JAXBException e) {
   111             throw new AssertionError(e);    // impossible
   112         }
   113     }
   115     public UnmarshallerHandler getUnmarshallerHandler() {
   116         return getUnmarshallerHandler(true,null);
   117     }
   119     private SAXConnector getUnmarshallerHandler( boolean intern, JaxBeanInfo expectedType ) {
   120         XmlVisitor h = createUnmarshallerHandler(null,false,expectedType);
   121         if(intern)
   122             h = new InterningXmlVisitor(h);
   123         return new SAXConnector(h,null);
   124     }
   126     /**
   127      * Creates and configures a new unmarshalling pipe line.
   128      * Depending on the setting, we put a validator as a filter.
   129      *
   130      * @return
   131      *      A component that implements both {@link UnmarshallerHandler}
   132      *      and {@link ValidationEventHandler}. All the parsing errors
   133      *      should be reported to this error handler for the unmarshalling
   134      *      process to work correctly.
   135      *
   136      *      Also, returned handler expects all the XML names to be interned.
   137      *
   138      */
   139     public final XmlVisitor createUnmarshallerHandler(InfosetScanner scanner, boolean inplace, JaxBeanInfo expectedType ) {
   141         coordinator.reset(scanner,inplace,expectedType,idResolver);
   142         XmlVisitor unmarshaller = coordinator;
   144         // delegate to JAXP 1.3 for validation if the client provided a schema
   145         if (schema != null)
   146             unmarshaller = new ValidatingUnmarshaller(schema,unmarshaller);
   148         if(attachmentUnmarshaller!=null && attachmentUnmarshaller.isXOPPackage())
   149             unmarshaller = new MTOMDecorator(this,unmarshaller,attachmentUnmarshaller);
   151         return unmarshaller;
   152     }
   154     private static final DefaultHandler dummyHandler = new DefaultHandler();
   156     public static boolean needsInterning( XMLReader reader ) {
   157         // attempt to set it to true, which could fail
   158         try {
   159             reader.setFeature("http://xml.org/sax/features/string-interning",true);
   160         } catch (SAXException e) {
   161             // if it fails that's fine. we'll work around on our side
   162         }
   164         try {
   165             if( reader.getFeature("http://xml.org/sax/features/string-interning") )
   166                 return false;   // no need for intern
   167         } catch (SAXException e) {
   168             // unrecognized/unsupported
   169         }
   170         // otherwise we need intern
   171         return true;
   172     }
   174     protected Object unmarshal( XMLReader reader, InputSource source ) throws JAXBException {
   175         return unmarshal0(reader,source,null);
   176     }
   178     protected <T> JAXBElement<T> unmarshal( XMLReader reader, InputSource source, Class<T> expectedType ) throws JAXBException {
   179         if(expectedType==null)
   180             throw new IllegalArgumentException();
   181         return (JAXBElement)unmarshal0(reader,source,getBeanInfo(expectedType));
   182     }
   184     private Object unmarshal0( XMLReader reader, InputSource source, JaxBeanInfo expectedType ) throws JAXBException {
   186         SAXConnector connector = getUnmarshallerHandler(needsInterning(reader),expectedType);
   188         reader.setContentHandler(connector);
   189         // saxErrorHandler will be set by the getUnmarshallerHandler method.
   190         // configure XMLReader so that the error will be sent to it.
   191         // This is essential for the UnmarshallerHandler to be able to abort
   192         // unmarshalling when an error is found.
   193         //
   194         // Note that when this XMLReader is provided by the client code,
   195         // it might be already configured to call a client error handler.
   196         // This will clobber such handler, if any.
   197         //
   198         // Ryan noted that we might want to report errors to such a client
   199         // error handler as well.
   200         reader.setErrorHandler(coordinator);
   202         try {
   203             reader.parse(source);
   204         } catch( IOException e ) {
   205             coordinator.clearStates();
   206             throw new UnmarshalException(e);
   207         } catch( SAXException e ) {
   208             coordinator.clearStates();
   209             throw createUnmarshalException(e);
   210         }
   212         Object result = connector.getResult();
   214         // avoid keeping unnecessary references too long to let the GC
   215         // reclaim more memory.
   216         // setting null upsets some parsers, so use a dummy instance instead.
   217         reader.setContentHandler(dummyHandler);
   218         reader.setErrorHandler(dummyHandler);
   220         return result;
   221     }
   223     @Override
   224     public <T> JAXBElement<T> unmarshal( Source source, Class<T> expectedType ) throws JAXBException {
   225         if(source instanceof SAXSource) {
   226             SAXSource ss = (SAXSource)source;
   228             XMLReader reader = ss.getXMLReader();
   229             if( reader == null )
   230                 reader = getXMLReader();
   232             return unmarshal( reader, ss.getInputSource(), expectedType );
   233         }
   234         if(source instanceof StreamSource) {
   235             return unmarshal( getXMLReader(), streamSourceToInputSource((StreamSource)source), expectedType );
   236         }
   237         if(source instanceof DOMSource)
   238             return unmarshal( ((DOMSource)source).getNode(), expectedType );
   240         // we don't handle other types of Source
   241         throw new IllegalArgumentException();
   242     }
   244     public Object unmarshal0( Source source, JaxBeanInfo expectedType ) throws JAXBException {
   245         if(source instanceof SAXSource) {
   246             SAXSource ss = (SAXSource)source;
   248             XMLReader reader = ss.getXMLReader();
   249             if( reader == null )
   250                 reader = getXMLReader();
   252             return unmarshal0( reader, ss.getInputSource(), expectedType );
   253         }
   254         if(source instanceof StreamSource) {
   255             return unmarshal0( getXMLReader(), streamSourceToInputSource((StreamSource)source), expectedType );
   256         }
   257         if(source instanceof DOMSource)
   258             return unmarshal0( ((DOMSource)source).getNode(), expectedType );
   260         // we don't handle other types of Source
   261         throw new IllegalArgumentException();
   262     }
   265     @Override
   266     public final ValidationEventHandler getEventHandler() {
   267         try {
   268             return super.getEventHandler();
   269         } catch (JAXBException e) {
   270             // impossible
   271             throw new AssertionError();
   272         }
   273     }
   275     /**
   276      * Returns true if an event handler is installed.
   277      * <p>
   278      * The default handler ignores any errors, and for that this method returns false.
   279      */
   280     public final boolean hasEventHandler() {
   281         return getEventHandler()!=this;
   282     }
   284     @Override
   285     public <T> JAXBElement<T> unmarshal(Node node, Class<T> expectedType) throws JAXBException {
   286         if(expectedType==null)
   287             throw new IllegalArgumentException();
   288         return (JAXBElement)unmarshal0(node,getBeanInfo(expectedType));
   289     }
   291     public final Object unmarshal( Node node ) throws JAXBException {
   292         return unmarshal0(node,null);
   293     }
   295     // just to make the the test harness happy by making this method accessible
   296     @Deprecated
   297     public final Object unmarshal( SAXSource source ) throws JAXBException {
   298         return super.unmarshal(source);
   299     }
   301     public final Object unmarshal0( Node node, JaxBeanInfo expectedType ) throws JAXBException {
   302         try {
   303             final DOMScanner scanner = new DOMScanner();
   305             InterningXmlVisitor handler = new InterningXmlVisitor(createUnmarshallerHandler(null,false,expectedType));
   306             scanner.setContentHandler(new SAXConnector(handler,scanner));
   308             if(node.getNodeType() == Node.ELEMENT_NODE)
   309                 scanner.scan((Element)node);
   310             else
   311             if(node.getNodeType() == Node.DOCUMENT_NODE)
   312                 scanner.scan((Document)node);
   313             else
   314                 // no other type of input is supported
   315                 throw new IllegalArgumentException("Unexpected node type: "+node);
   317             Object retVal = handler.getContext().getResult();
   318             handler.getContext().clearResult();
   319             return retVal;
   320         } catch( SAXException e ) {
   321             throw createUnmarshalException(e);
   322         }
   323     }
   325     @Override
   326     public Object unmarshal(XMLStreamReader reader) throws JAXBException {
   327         return unmarshal0(reader,null);
   328     }
   330     @Override
   331     public <T> JAXBElement<T> unmarshal(XMLStreamReader reader, Class<T> expectedType) throws JAXBException {
   332         if(expectedType==null)
   333             throw new IllegalArgumentException();
   334         return (JAXBElement)unmarshal0(reader,getBeanInfo(expectedType));
   335     }
   337     public Object unmarshal0(XMLStreamReader reader, JaxBeanInfo expectedType) throws JAXBException {
   338         if (reader == null) {
   339             throw new IllegalArgumentException(
   340                 Messages.format(Messages.NULL_READER));
   341         }
   343         int eventType = reader.getEventType();
   344         if (eventType != XMLStreamConstants.START_ELEMENT
   345             && eventType != XMLStreamConstants.START_DOCUMENT) {
   346             // TODO: convert eventType into event name
   347             throw new IllegalStateException(
   348                 Messages.format(Messages.ILLEGAL_READER_STATE,eventType));
   349         }
   351         XmlVisitor h = createUnmarshallerHandler(null,false,expectedType);
   352         StAXConnector connector=StAXStreamConnector.create(reader,h);
   354         try {
   355             connector.bridge();
   356         } catch (XMLStreamException e) {
   357             throw handleStreamException(e);
   358         }
   360         Object retVal = h.getContext().getResult();
   361         h.getContext().clearResult();
   362         return retVal;
   363     }
   365     @Override
   366     public <T> JAXBElement<T> unmarshal(XMLEventReader reader, Class<T> expectedType) throws JAXBException {
   367         if(expectedType==null)
   368             throw new IllegalArgumentException();
   369         return (JAXBElement)unmarshal0(reader,getBeanInfo(expectedType));
   370     }
   372     @Override
   373     public Object unmarshal(XMLEventReader reader) throws JAXBException {
   374         return unmarshal0(reader,null);
   375     }
   377     private Object unmarshal0(XMLEventReader reader,JaxBeanInfo expectedType) throws JAXBException {
   378         if (reader == null) {
   379             throw new IllegalArgumentException(
   380                     Messages.format(Messages.NULL_READER));
   381         }
   383         try {
   384             XMLEvent event = reader.peek();
   386             if (!event.isStartElement() && !event.isStartDocument()) {
   387                 // TODO: convert event into event name
   388                 throw new IllegalStateException(
   389                     Messages.format(
   390                         Messages.ILLEGAL_READER_STATE,event.getEventType()));
   391             }
   393             // Quick hack until SJSXP fixes 6270116
   394             boolean isZephyr = reader.getClass().getName().equals("com.sun.xml.internal.stream.XMLReaderImpl");
   395             XmlVisitor h = createUnmarshallerHandler(null,false,expectedType);
   396             if(!isZephyr)
   397                 h = new InterningXmlVisitor(h);
   398             new StAXEventConnector(reader,h).bridge();
   399             return h.getContext().getResult();
   400         } catch (XMLStreamException e) {
   401             throw handleStreamException(e);
   402         }
   403     }
   405     public Object unmarshal0( InputStream input, JaxBeanInfo expectedType ) throws JAXBException {
   406         return unmarshal0(getXMLReader(),new InputSource(input),expectedType);
   407     }
   409     private static JAXBException handleStreamException(XMLStreamException e) {
   410         // StAXStreamConnector wraps SAXException to XMLStreamException.
   411         // XMLStreamException doesn't print its nested stack trace when it prints
   412         // its stack trace, so if we wrap XMLStreamException in JAXBException,
   413         // it becomes harder to find out the real problem.
   414         // So we unwrap them here. But we don't want to unwrap too eagerly, because
   415         // that could throw away some meaningful exception information.
   416         Throwable ne = e.getNestedException();
   417         if(ne instanceof JAXBException)
   418             return (JAXBException)ne;
   419         if(ne instanceof SAXException)
   420             return new UnmarshalException(ne);
   421         return new UnmarshalException(e);
   422     }
   424     @Override
   425     public Object getProperty(String name) throws PropertyException {
   426         if(name.equals(IDResolver.class.getName())) {
   427             return idResolver;
   428         }
   429         return super.getProperty(name);
   430     }
   432     @Override
   433     public void setProperty(String name, Object value) throws PropertyException {
   434         if(name.equals(FACTORY)) {
   435             coordinator.setFactories(value);
   436             return;
   437         }
   438         if(name.equals(IDResolver.class.getName())) {
   439             idResolver = (IDResolver)value;
   440             return;
   441         }
   442         if(name.equals(ClassResolver.class.getName())) {
   443             coordinator.classResolver = (ClassResolver)value;
   444             return;
   445         }
   446         if(name.equals(ClassLoader.class.getName())) {
   447             coordinator.classLoader = (ClassLoader)value;
   448             return;
   449         }
   450         super.setProperty(name, value);
   451     }
   453     public static final String FACTORY = "com.sun.xml.internal.bind.ObjectFactory";
   455     @Override
   456     public void setSchema(Schema schema) {
   457         this.schema = schema;
   458     }
   460     @Override
   461     public Schema getSchema() {
   462         return schema;
   463     }
   465     @Override
   466     public AttachmentUnmarshaller getAttachmentUnmarshaller() {
   467         return attachmentUnmarshaller;
   468     }
   470     @Override
   471     public void setAttachmentUnmarshaller(AttachmentUnmarshaller au) {
   472         this.attachmentUnmarshaller = au;
   473     }
   475     /**
   476      * @deprecated since 2.0
   477      */
   478     @Override
   479     public boolean isValidating() {
   480         throw new UnsupportedOperationException();
   481     }
   483     /**
   484      * @deprecated since 2.0
   485      */
   486     @Override
   487     public void setValidating(boolean validating) {
   488         throw new UnsupportedOperationException();
   489     }
   491     @Override
   492     public <A extends XmlAdapter> void setAdapter(Class<A> type, A adapter) {
   493         if(type==null)
   494             throw new IllegalArgumentException();
   495         coordinator.putAdapter(type,adapter);
   496     }
   498     @Override
   499     public <A extends XmlAdapter> A getAdapter(Class<A> type) {
   500         if(type==null)
   501             throw new IllegalArgumentException();
   502         if(coordinator.containsAdapter(type))
   503             // so as not to create a new instance when this method is called
   504             return coordinator.getAdapter(type);
   505         else
   506             return null;
   507     }
   509     // opening up for public use
   510     @Override
   511     public UnmarshalException createUnmarshalException( SAXException e ) {
   512         return super.createUnmarshalException(e);
   513     }
   516     /**
   517      * Default error handling behavior for {@link Unmarshaller}.
   518      */
   519     public boolean handleEvent(ValidationEvent event) {
   520         return event.getSeverity()!=ValidationEvent.FATAL_ERROR;
   521     }
   523     private static InputSource streamSourceToInputSource( StreamSource ss ) {
   524         InputSource is = new InputSource();
   525         is.setSystemId( ss.getSystemId() );
   526         is.setByteStream( ss.getInputStream() );
   527         is.setCharacterStream( ss.getReader() );
   529         return is;
   530     }
   532     public <T> JaxBeanInfo<T> getBeanInfo(Class<T> clazz) throws JAXBException {
   533         return context.getBeanInfo(clazz,true);
   534     }
   536     @Override
   537     public Listener getListener() {
   538         return externalListener;
   539     }
   541     @Override
   542     public void setListener(Listener listener) {
   543         externalListener = listener;
   544     }
   546     public UnmarshallingContext getContext() {
   547         return coordinator;
   548     }
   550     @Override
   551     @SuppressWarnings("FinalizeDeclaration")
   552     protected void finalize() throws Throwable {
   553         try {
   554             ClassFactory.cleanCache();
   555         } finally {
   556             super.finalize();
   557         }
   558     }
   560     /**
   561      *  Must be called from same thread which created the UnmarshallerImpl instance.
   562      * @throws IOException
   563      */
   564     public void close() throws IOException {
   565         ClassFactory.cleanCache();
   566     }
   568 }

mercurial