/* * Copyright 2005-2006 Sun Microsystems, Inc. All Rights Reserved. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * * This code is free software; you can redistribute it and/or modify it * under the terms of the GNU General Public License version 2 only, as * published by the Free Software Foundation. Sun designates this * particular file as subject to the "Classpath" exception as provided * by Sun in the LICENSE file that accompanied this code. * * This code is distributed in the hope that it will be useful, but WITHOUT * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License * version 2 for more details (a copy is included in the LICENSE file that * accompanied this code). * * You should have received a copy of the GNU General Public License version * 2 along with this work; if not, write to the Free Software Foundation, * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. * * Please contact Sun Microsystems, Inc., 4150 Network Circle, Santa Clara, * CA 95054 USA or visit www.sun.com if you need additional information or * have any questions. */ package com.sun.xml.internal.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.internal.bind.v2.model.annotation.AnnotationReader; import com.sun.xml.internal.bind.v2.model.core.ClassInfo; import com.sun.xml.internal.bind.v2.model.core.Element; import com.sun.xml.internal.bind.v2.model.core.ElementInfo; import com.sun.xml.internal.bind.v2.model.core.NonElement; import com.sun.xml.internal.bind.v2.model.core.PropertyKind; import com.sun.xml.internal.bind.v2.model.core.ReferencePropertyInfo; import com.sun.xml.internal.bind.v2.model.core.WildcardMode; import com.sun.xml.internal.bind.v2.model.nav.Navigator; import com.sun.xml.internal.bind.v2.runtime.IllegalAnnotationException; /** * Implementation of {@link ReferencePropertyInfo}. * * @author Kohsuke Kawaguchi */ class ReferencePropertyInfoImpl<T,C,F,M> extends ERPropertyInfoImpl<T,C,F,M> implements ReferencePropertyInfo<T,C> { /** * Lazily computed. * @see #getElements() */ private Set<Element<T,C>> types; private final boolean isMixed; private final WildcardMode wildcard; private final C domHandler; 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; } 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( type.equals(defaultType) ) type = nav.erasure(getIndividualType()); if(nav.getBaseClass(type,je)!=null) yield = addGenericElement(r); else yield = addAllSubtypes(type); if(last && !yield) { // a reference didn't produce any type. // diagnose the problem if(type.equals(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); } /** * @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 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; } 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 boolean isMixed() { return isMixed; } public final WildcardMode getWildcard() { return wildcard; } public final C getDOMHandler() { return domHandler; } }