All Downloads are FREE. Search and download functionalities are using the official Maven repository.

spinjar.com.sun.xml.bind.v2.model.impl.TypeInfoSetImpl Maven / Gradle / Ivy

There is a newer version: 1.23.0
Show newest version
/*
 * 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.model.impl;

import java.util.Collections;
import java.util.HashMap;
import java.util.Iterator;
import java.util.LinkedHashMap;
import java.util.Map;

import javax.xml.bind.JAXBContext;
import javax.xml.bind.JAXBException;
import javax.xml.bind.Marshaller;
import javax.xml.bind.annotation.XmlNs;
import javax.xml.bind.annotation.XmlNsForm;
import javax.xml.bind.annotation.XmlRegistry;
import javax.xml.bind.annotation.XmlSchema;
import javax.xml.bind.annotation.XmlTransient;
import javax.xml.bind.annotation.adapters.XmlJavaTypeAdapter;
import javax.xml.namespace.QName;
import javax.xml.transform.Result;

import com.sun.xml.bind.v2.model.annotation.AnnotationReader;
import com.sun.xml.bind.v2.model.core.BuiltinLeafInfo;
import com.sun.xml.bind.v2.model.core.ClassInfo;
import com.sun.xml.bind.v2.model.core.LeafInfo;
import com.sun.xml.bind.v2.model.core.NonElement;
import com.sun.xml.bind.v2.model.core.Ref;
import com.sun.xml.bind.v2.model.core.TypeInfo;
import com.sun.xml.bind.v2.model.core.TypeInfoSet;
import com.sun.xml.bind.v2.model.nav.Navigator;
import com.sun.xml.bind.v2.runtime.IllegalAnnotationException;
import com.sun.xml.bind.v2.runtime.RuntimeUtil;
import com.sun.xml.bind.v2.util.FlattenIterator;

/**
 * Set of {@link TypeInfo}s.
 *
 * 

* This contains a fixed set of {@link LeafInfo}s and arbitrary set of {@link ClassInfo}s. * *

* Members are annotated with JAXB annotations so that we can dump it easily. * * @author Kohsuke Kawaguchi */ class TypeInfoSetImpl implements TypeInfoSet { @XmlTransient public final Navigator nav; @XmlTransient public final AnnotationReader reader; /** * All the leaves. */ private final Map> builtins = new LinkedHashMap>(); /** All {@link EnumLeafInfoImpl}s. */ private final Map> enums = new LinkedHashMap>(); /** All {@link ArrayInfoImpl}s. */ private final Map> arrays = new LinkedHashMap>(); /** * All the user-defined classes. * * Using {@link LinkedHashMap} allows us to process classes * in the order they are given to us. When the user incorrectly * puts an unexpected class into a reference graph, this causes * an error to be reported on a class closer to the user's code. */ @XmlJavaTypeAdapter(RuntimeUtil.ToStringAdapter.class) private final Map> beans = new LinkedHashMap>(); @XmlTransient private final Map> beansView = Collections.unmodifiableMap(beans); /** * The element mapping. */ private final Map>> elementMappings = new LinkedHashMap>>(); private final Iterable> allElements = new Iterable>() { public Iterator> iterator() { return new FlattenIterator>(elementMappings.values()); } }; /** * {@link TypeInfo} for xs:anyType. * * anyType is the only {@link TypeInfo} that works with an interface, * and accordingly it requires a lot of special casing. */ private final NonElement anyType; /** * Lazily parsed set of {@link XmlNs}s. * * @see #getXmlNs(String) */ private Map> xmlNsCache; public TypeInfoSetImpl(Navigator nav, AnnotationReader reader, Map> leaves) { this.nav = nav; this.reader = reader; this.builtins.putAll(leaves); this.anyType = createAnyType(); // register primitive types. for (Map.Entry e : RuntimeUtil.primitiveToBox.entrySet()) { this.builtins.put( nav.getPrimitive(e.getKey()), leaves.get(nav.ref(e.getValue())) ); } // make sure at lease we got a map for global ones. elementMappings.put(null,new LinkedHashMap>()); } protected NonElement createAnyType() { return new AnyTypeImpl(nav); } public Navigator getNavigator() { return nav; } /** * Adds a new {@link ClassInfo} to the set. */ public void add( ClassInfoImpl ci ) { beans.put( ci.getClazz(), ci ); } /** * Adds a new {@link LeafInfo} to the set. */ public void add( EnumLeafInfoImpl li ) { enums.put( li.clazz, li ); } public void add(ArrayInfoImpl ai) { arrays.put( ai.getType(), ai ); } /** * Returns a {@link TypeInfo} for the given type. * * @return * null if the specified type cannot be bound by JAXB, or * not known to this set. */ public NonElement getTypeInfo( T type ) { type = nav.erasure(type); // replace type variables by their bounds LeafInfo l = builtins.get(type); if(l!=null) return l; if( nav.isArray(type) ) { return arrays.get(type); } C d = nav.asDecl(type); if(d==null) return null; return getClassInfo(d); } public NonElement getAnyTypeInfo() { return anyType; } /** * This method is used to add a root reference to a model. */ public NonElement getTypeInfo(Ref ref) { // TODO: handle XmlValueList assert !ref.valueList; C c = nav.asDecl(ref.type); if(c!=null && reader.getClassAnnotation(XmlRegistry.class,c,null/*TODO: is this right?*/)!=null) { return null; // TODO: is this correct? } else return getTypeInfo(ref.type); } /** * Returns all the {@link ClassInfo}s known to this set. */ public Map> beans() { return beansView; } public Map> builtins() { return builtins; } public Map> enums() { return enums; } public Map> arrays() { return arrays; } /** * Returns a {@link ClassInfo} for the given bean. * *

* This method is almost like refinement of {@link #getTypeInfo(Object)} except * our C cannot derive from T. * * @return * null if the specified type is not bound by JAXB or otherwise * unknown to this set. */ public NonElement getClassInfo( C type ) { LeafInfo l = builtins.get(nav.use(type)); if(l!=null) return l; l = enums.get(type); if(l!=null) return l; if(nav.asDecl(Object.class).equals(type)) return anyType; return beans.get(type); } public ElementInfoImpl getElementInfo( C scope, QName name ) { while(scope!=null) { Map> m = elementMappings.get(scope); if(m!=null) { ElementInfoImpl r = m.get(name); if(r!=null) return r; } scope = nav.getSuperClass(scope); } return elementMappings.get(null).get(name); } /** * @param builder * used for reporting errors. */ public final void add( ElementInfoImpl ei, ModelBuilder builder ) { C scope = null; if(ei.getScope()!=null) scope = ei.getScope().getClazz(); Map> m = elementMappings.get(scope); if(m==null) elementMappings.put(scope,m=new LinkedHashMap>()); ElementInfoImpl existing = m.put(ei.getElementName(),ei); if(existing!=null) { QName en = ei.getElementName(); builder.reportError( new IllegalAnnotationException( Messages.CONFLICTING_XML_ELEMENT_MAPPING.format(en.getNamespaceURI(),en.getLocalPart()), ei, existing )); } } public Map> getElementMappings( C scope ) { return elementMappings.get(scope); } public Iterable> getAllElements() { return allElements; } public Map getXmlNs(String namespaceUri) { if(xmlNsCache==null) { xmlNsCache = new HashMap>(); for (ClassInfoImpl ci : beans().values()) { XmlSchema xs = reader.getPackageAnnotation( XmlSchema.class, ci.getClazz(), null ); if(xs==null) continue; String uri = xs.namespace(); Map m = xmlNsCache.get(uri); if(m==null) xmlNsCache.put(uri,m=new HashMap()); for( XmlNs xns : xs.xmlns() ) { m.put(xns.prefix(),xns.namespaceURI()); } } } Map r = xmlNsCache.get(namespaceUri); if(r!=null) return r; else return Collections.emptyMap(); } public Map getSchemaLocations() { Map r = new HashMap(); for (ClassInfoImpl ci : beans().values()) { XmlSchema xs = reader.getPackageAnnotation( XmlSchema.class, ci.getClazz(), null ); if(xs==null) continue; String loc = xs.location(); if(loc.equals(XmlSchema.NO_LOCATION)) continue; // unspecified r.put(xs.namespace(),loc); } return r; } public final XmlNsForm getElementFormDefault(String nsUri) { for (ClassInfoImpl ci : beans().values()) { XmlSchema xs = reader.getPackageAnnotation( XmlSchema.class, ci.getClazz(), null ); if(xs==null) continue; if(!xs.namespace().equals(nsUri)) continue; XmlNsForm xnf = xs.elementFormDefault(); if(xnf!=XmlNsForm.UNSET) return xnf; } return XmlNsForm.UNSET; } public final XmlNsForm getAttributeFormDefault(String nsUri) { for (ClassInfoImpl ci : beans().values()) { XmlSchema xs = reader.getPackageAnnotation( XmlSchema.class, ci.getClazz(), null ); if(xs==null) continue; if(!xs.namespace().equals(nsUri)) continue; XmlNsForm xnf = xs.attributeFormDefault(); if(xnf!=XmlNsForm.UNSET) return xnf; } return XmlNsForm.UNSET; } /** * Dumps this model into XML. * * For debug only. * * TODO: not sure if this actually works. We don't really know what are T,C. */ public void dump( Result out ) throws JAXBException { JAXBContext context = JAXBContext.newInstance(this.getClass()); Marshaller m = context.createMarshaller(); m.marshal(this,out); } }





© 2015 - 2024 Weber Informatics LLC | Privacy Policy