/* * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS HEADER. * * Copyright (c) 1997-2012 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.LinkedHashSet; import java.util.Set; import javax.xml.bind.JAXBElement; import javax.xml.bind.annotation.XmlAnyElement; import javax.xml.bind.annotation.XmlElementRef; import javax.xml.bind.annotation.XmlElementRefs; import javax.xml.bind.annotation.XmlMixed; import javax.xml.bind.annotation.XmlSchema; import javax.xml.bind.annotation.XmlNsForm; import javax.xml.namespace.QName; import com.sun.xml.bind.v2.model.annotation.AnnotationReader; import com.sun.xml.bind.v2.model.core.ClassInfo; import com.sun.xml.bind.v2.model.core.Element; import com.sun.xml.bind.v2.model.core.ElementInfo; import com.sun.xml.bind.v2.model.core.NonElement; import com.sun.xml.bind.v2.model.core.PropertyKind; import com.sun.xml.bind.v2.model.core.ReferencePropertyInfo; import com.sun.xml.bind.v2.model.core.WildcardMode; import com.sun.xml.bind.v2.model.nav.Navigator; import com.sun.xml.bind.v2.runtime.IllegalAnnotationException; import java.util.Iterator; /** * Implementation of {@link ReferencePropertyInfo}. * * @author Kohsuke Kawaguchi */ class ReferencePropertyInfoImpl<T,C,F,M> extends ERPropertyInfoImpl<T,C,F,M> implements ReferencePropertyInfo<T,C>, DummyPropertyInfo<T, C, F, M> { /** * Lazily computed. * @see #getElements() */ private Set<Element<T,C>> types; private Set<ReferencePropertyInfoImpl<T,C,F,M>> subTypes = new LinkedHashSet<ReferencePropertyInfoImpl<T,C,F,M>>(); private final boolean isMixed; private final WildcardMode wildcard; private final C domHandler; /** * Lazily computed. * @see #isRequired() */ private Boolean isRequired; public ReferencePropertyInfoImpl( ClassInfoImpl<T,C,F,M> classInfo, PropertySeed<T,C,F,M> seed) { super(classInfo, seed); isMixed = seed.readAnnotation(XmlMixed.class) != null; XmlAnyElement xae = seed.readAnnotation(XmlAnyElement.class); if(xae==null) { wildcard = null; domHandler = null; } else { wildcard = xae.lax()?WildcardMode.LAX:WildcardMode.SKIP; domHandler = nav().asDecl(reader().getClassValue(xae,"value")); } } public Set<? extends Element<T,C>> ref() { return getElements(); } public PropertyKind kind() { return PropertyKind.REFERENCE; } public Set<? extends Element<T,C>> getElements() { if(types==null) calcTypes(false); assert types!=null; return types; } /** * Compute {@link #types}. * * @param last * if true, every {@link XmlElementRef} must yield at least one type. */ private void calcTypes(boolean last) { XmlElementRef[] ann; types = new LinkedHashSet<Element<T,C>>(); XmlElementRefs refs = seed.readAnnotation(XmlElementRefs.class); XmlElementRef ref = seed.readAnnotation(XmlElementRef.class); if(refs!=null && ref!=null) { parent.builder.reportError(new IllegalAnnotationException( Messages.MUTUALLY_EXCLUSIVE_ANNOTATIONS.format( nav().getClassName(parent.getClazz())+'#'+seed.getName(), ref.annotationType().getName(), refs.annotationType().getName()), ref, refs )); } if(refs!=null) ann = refs.value(); else { if(ref!=null) ann = new XmlElementRef[]{ref}; else ann = null; } isRequired = !isCollection(); // this is by default, to remain compatible with 2.1 if(ann!=null) { Navigator<T,C,F,M> nav = nav(); AnnotationReader<T,C,F,M> reader = reader(); final T defaultType = nav.ref(XmlElementRef.DEFAULT.class); final C je = nav.asDecl(JAXBElement.class); for( XmlElementRef r : ann ) { boolean yield; T type = reader.getClassValue(r,"type"); if(nav().isSameType(type, defaultType)) type = nav.erasure(getIndividualType()); if(nav.getBaseClass(type,je)!=null) yield = addGenericElement(r); else yield = addAllSubtypes(type); // essentially "isRequired &= isRequired(r)" except that we'd like to skip evaluating isRequird(r) // if the value is already false. if(isRequired && !isRequired(r)) isRequired = false; if(last && !yield) { // a reference didn't produce any type. // diagnose the problem if(nav().isSameType(type, nav.ref(JAXBElement.class))) { // no XmlElementDecl parent.builder.reportError(new IllegalAnnotationException( Messages.NO_XML_ELEMENT_DECL.format( getEffectiveNamespaceFor(r), r.name()), this )); } else { parent.builder.reportError(new IllegalAnnotationException( Messages.INVALID_XML_ELEMENT_REF.format(type),this)); } // reporting one error would do. // often the element ref field is using @XmlElementRefs // to point to multiple JAXBElements. // reporting one error for each @XmlElemetnRef is thus often redundant. return; } } } for (ReferencePropertyInfoImpl<T, C, F, M> info : subTypes) { PropertySeed sd = info.seed; refs = sd.readAnnotation(XmlElementRefs.class); ref = sd.readAnnotation(XmlElementRef.class); if (refs != null && ref != null) { parent.builder.reportError(new IllegalAnnotationException( Messages.MUTUALLY_EXCLUSIVE_ANNOTATIONS.format( nav().getClassName(parent.getClazz())+'#'+seed.getName(), ref.annotationType().getName(), refs.annotationType().getName()), ref, refs )); } if (refs != null) { ann = refs.value(); } else { if (ref != null) { ann = new XmlElementRef[]{ref}; } else { ann = null; } } if (ann != null) { Navigator<T,C,F,M> nav = nav(); AnnotationReader<T,C,F,M> reader = reader(); final T defaultType = nav.ref(XmlElementRef.DEFAULT.class); final C je = nav.asDecl(JAXBElement.class); for( XmlElementRef r : ann ) { boolean yield; T type = reader.getClassValue(r,"type"); if (nav().isSameType(type, defaultType)) { type = nav.erasure(getIndividualType()); } if (nav.getBaseClass(type,je) != null) { yield = addGenericElement(r, info); } else { yield = addAllSubtypes(type); } if(last && !yield) { // a reference didn't produce any type. // diagnose the problem if(nav().isSameType(type, nav.ref(JAXBElement.class))) { // no XmlElementDecl parent.builder.reportError(new IllegalAnnotationException( Messages.NO_XML_ELEMENT_DECL.format( getEffectiveNamespaceFor(r), r.name()), this )); } else { parent.builder.reportError(new IllegalAnnotationException( Messages.INVALID_XML_ELEMENT_REF.format(),this)); } // reporting one error would do. // often the element ref field is using @XmlElementRefs // to point to multiple JAXBElements. // reporting one error for each @XmlElemetnRef is thus often redundant. return; } } } } types = Collections.unmodifiableSet(types); } public boolean isRequired() { if(isRequired==null) calcTypes(false); return isRequired; } /** * If we find out that we are working with 2.1 API, remember the fact so that * we don't waste time generating exceptions every time we call {@link #isRequired(XmlElementRef)}. */ private static boolean is2_2 = true; /** * Reads the value of {@code XmlElementRef.required()}. * * If we are working as 2.1 RI, this defaults to true. */ private boolean isRequired(XmlElementRef ref) { if(!is2_2) return true; try { return ref.required(); } catch(LinkageError e) { is2_2 = false; return true; // the value defaults to true } } /** * @return * true if the reference yields at least one type */ private boolean addGenericElement(XmlElementRef r) { String nsUri = getEffectiveNamespaceFor(r); // TODO: check spec. defaulting of localName. return addGenericElement(parent.owner.getElementInfo(parent.getClazz(),new QName(nsUri,r.name()))); } private boolean addGenericElement(XmlElementRef r, ReferencePropertyInfoImpl<T,C,F,M> info) { String nsUri = info.getEffectiveNamespaceFor(r); ElementInfo ei = parent.owner.getElementInfo(info.parent.getClazz(), new QName(nsUri, r.name())); types.add(ei); return true; } private String getEffectiveNamespaceFor(XmlElementRef r) { String nsUri = r.namespace(); XmlSchema xs = reader().getPackageAnnotation( XmlSchema.class, parent.getClazz(), this ); if(xs!=null && xs.attributeFormDefault()== XmlNsForm.QUALIFIED) { // JAX-RPC doesn't want the default namespace URI swapping to take effect to // local "unqualified" elements. UGLY. if(nsUri.length()==0) nsUri = parent.builder.defaultNsUri; } return nsUri; } private boolean addGenericElement(ElementInfo<T,C> ei) { if(ei==null) return false; types.add(ei); for( ElementInfo<T,C> subst : ei.getSubstitutionMembers() ) addGenericElement(subst); return true; } private boolean addAllSubtypes(T type) { Navigator<T,C,F,M> nav = nav(); // this allows the explicitly referenced type to be sucked in to the model NonElement<T,C> t = parent.builder.getClassInfo(nav.asDecl(type),this); if(!(t instanceof ClassInfo)) // this is leaf. return false; boolean result = false; ClassInfo<T,C> c = (ClassInfo<T,C>) t; if(c.isElement()) { types.add(c.asElement()); result = true; } // look for other possible types for( ClassInfo<T,C> ci : parent.owner.beans().values() ) { if(ci.isElement() && nav.isSubClassOf(ci.getType(),type)) { types.add(ci.asElement()); result = true; } } // don't allow local elements to substitute. for( ElementInfo<T,C> ei : parent.owner.getElementMappings(null).values()) { if(nav.isSubClassOf(ei.getType(),type)) { types.add(ei); result = true; } } return result; } @Override protected void link() { super.link(); // until we get the whole thing into TypeInfoSet, // we never really know what are all the possible types that can be assigned on this field. // so recompute this value when we have all the information. calcTypes(true); } public final void addType(PropertyInfoImpl<T,C,F,M> info) { //noinspection unchecked subTypes.add((ReferencePropertyInfoImpl)info); } public final boolean isMixed() { return isMixed; } public final WildcardMode getWildcard() { return wildcard; } public final C getDOMHandler() { return domHandler; } }