org.glassfish.hk2.xml.internal.ModelImpl Maven / Gradle / Ivy
/*
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS HEADER.
*
* Copyright (c) 2015-2017 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 org.glassfish.hk2.xml.internal;
import java.lang.reflect.Method;
import java.util.Collection;
import java.util.Collections;
import java.util.HashSet;
import java.util.LinkedHashMap;
import java.util.LinkedHashSet;
import java.util.Map;
import java.util.Set;
import javax.xml.bind.annotation.XmlAttribute;
import javax.xml.bind.annotation.XmlElement;
import javax.xml.namespace.QName;
import org.glassfish.hk2.utilities.general.GeneralUtilities;
import org.glassfish.hk2.utilities.reflection.ClassReflectionHelper;
import org.glassfish.hk2.utilities.reflection.MethodWrapper;
import org.glassfish.hk2.xml.internal.alt.AltMethod;
import org.glassfish.hk2.xml.internal.alt.clazz.MethodAltMethodImpl;
import org.glassfish.hk2.xml.spi.Model;
/**
* This model is a description of the children and non-children nodes
* of a Bean. It contains only Strings or other easy to constructs
* structures so that it can be added to the proxy at build time
* and hence save some reflection at runtime
*
* @author jwells
*
*/
public class ModelImpl implements Model {
private static final long serialVersionUID = 752816761552710497L;
/** For thread safety on the computed fields */
private final static Object lock = new Object();
/** The interface from which the JAXB proxy was created, fully qualified */
private String originalInterface;
/** Calculated at runtime lazily this is the original interface as a class */
private volatile Class> originalInterfaceAsClass;
/** The JAXB proxy of the originalInterface, fully qualified */
private String translatedClass;
/** Calculated at runtime lazily this is the proxy as a class */
private volatile Class> translatedClassAsClass;
/** If this node can be a root, the xml tag of the root of the document */
private QName rootName;
/** A map from the xml tag to the parented child node */
private final Map childrenByName = new LinkedHashMap();
/** A map from xml tag to information about the non-child property */
private final Map nonChildProperty = new LinkedHashMap();
/** A map from xml tag to child data, ordered */
private final Map allChildren = new LinkedHashMap();
/** If this node has a key, this is the property name of the key */
private QName keyProperty;
/**
* These are calculated values and only filled in when asked for
*/
private Set unKeyedChildren = null;
private Set keyedChildren = null;
private transient JAUtilities jaUtilities = null;
private ClassLoader myLoader;
private Map keyToJavaNameMap = null;
private Set allXmlWrappers;
private String valuePropertyNamespace;
private String valueProperty;
private ChildDataModel valueData;
public ModelImpl() {
}
public ModelImpl(String originalInterface,
String translatedClass) {
this.originalInterface = originalInterface;
this.translatedClass = translatedClass;
}
public void setRootName(QName root) {
this.rootName = root;
}
public void setRootName(String rootNamespace, String rootName) {
this.rootName = QNameUtilities.createQName(rootNamespace, rootName);
}
public void setKeyProperty(QName qName) {
String namespace = QNameUtilities.getNamespace(qName);
String name = qName.getLocalPart();
setKeyProperty(namespace, name);
}
public void setKeyProperty(String keyNamespace, String keyProperty) {
this.keyProperty = QNameUtilities.createQName(keyNamespace, keyProperty);
}
public void addChild(
String childInterface,
String namespace,
String xmlTag,
String xmlAlias,
ChildType childType,
String givenDefault,
AliasType aliased,
String childWrapperTag,
String adapter,
boolean required) {
ParentedModel pm = new ParentedModel(childInterface,
namespace,
xmlTag,
xmlAlias,
childType,
givenDefault,
aliased,
childWrapperTag,
adapter,
required);
childrenByName.put(QNameUtilities.createQName(namespace, xmlTag), pm);
allChildren.put(QNameUtilities.createQName(namespace, xmlTag), new ChildDescriptor(pm));
}
public void addNonChild(
QName qName,
String defaultValue,
String childType,
String childListType,
boolean isReference,
Format format,
AliasType aliasType,
String aliasOf,
boolean required) {
String namespace = QNameUtilities.getNamespace(qName);
String xmlTag = qName.getLocalPart();
addNonChild(namespace,
xmlTag,
defaultValue,
childType,
childListType,
isReference,
format,
aliasType,
aliasOf,
required);
}
public void addNonChild(
String namespace,
String xmlTag,
String defaultValue,
String childType,
String childListType,
boolean isReference,
Format format,
AliasType aliasType,
String aliasOf,
boolean required) {
ChildDataModel cdm = new ChildDataModel(childType, childListType, defaultValue, isReference, format, aliasType, aliasOf, required);
nonChildProperty.put(QNameUtilities.createQName(namespace, xmlTag), cdm);
allChildren.put(QNameUtilities.createQName(namespace, xmlTag), new ChildDescriptor(cdm));
if (Format.VALUE.equals(format)) {
valuePropertyNamespace = namespace;
valueProperty = xmlTag;
valueData = cdm;
}
}
/**
* @return the originalInterface
*/
@Override
public String getOriginalInterface() {
return originalInterface;
}
/**
* @return the translatedClass
*/
@Override
public String getTranslatedClass() {
return translatedClass;
}
/**
* @return the rootName
*/
@Override
public QName getRootName() {
return rootName;
}
/**
* @return the keyProperty
*/
@Override
public QName getKeyProperty() {
return keyProperty;
}
public Map getChildrenByName() {
return childrenByName;
}
public Map getNonChildProperties() {
return nonChildProperty;
}
public Map getAllChildrenDescriptors() {
return allChildren;
}
public String getValuePropertyNamespace() {
return valuePropertyNamespace;
}
public String getValueProperty() {
return valueProperty;
}
public ChildDataModel getValueData() {
return valueData;
}
public Set getAllXmlWrappers() {
synchronized (lock) {
if (allXmlWrappers != null) return allXmlWrappers;
allXmlWrappers = new LinkedHashSet();
for (ParentedModel pm : childrenByName.values()) {
if (pm.getXmlWrapperTag() != null) {
allXmlWrappers.add(pm.getXmlWrapperTag());
}
}
return allXmlWrappers;
}
}
public ChildDescriptor getChildDescriptor(QName xmlTag) {
return allChildren.get(xmlTag);
}
public Set getUnKeyedChildren() {
synchronized (lock) {
if (unKeyedChildren != null) return unKeyedChildren;
unKeyedChildren = new HashSet();
for (Map.Entry entry : childrenByName.entrySet()) {
if (entry.getValue().getChildModel().getKeyProperty() != null) continue;
unKeyedChildren.add(entry.getKey());
}
return unKeyedChildren;
}
}
public Set getKeyedChildren() {
synchronized (lock) {
if (keyedChildren != null) return keyedChildren;
keyedChildren = new HashSet();
for (Map.Entry entry : childrenByName.entrySet()) {
if (entry.getValue().getChildModel().getKeyProperty() == null) continue;
keyedChildren.add(entry.getKey());
}
return keyedChildren;
}
}
public void setJAUtilities(JAUtilities jaUtilities, ClassLoader myLoader) {
synchronized (lock) {
if (this.jaUtilities != null) return;
this.jaUtilities = jaUtilities;
this.myLoader = myLoader;
for (ParentedModel pm : childrenByName.values()) {
pm.setRuntimeInformation(jaUtilities, myLoader);
}
for (ChildDataModel cdm : nonChildProperty.values()) {
cdm.setLoader(myLoader);
}
}
}
public String getDefaultChildValue(String propNamespace, String propName) {
QName propQName = QNameUtilities.createQName(propNamespace, propName);
synchronized (lock) {
ChildDataModel cd = nonChildProperty.get(propQName);
if (cd == null) return null;
return cd.getDefaultAsString();
}
}
public ModelPropertyType getModelPropertyType(String propNamespace, String propName) {
QName propQName = QNameUtilities.createQName(propNamespace, propName);
synchronized (lock) {
if (nonChildProperty.containsKey(propQName)) return ModelPropertyType.FLAT_PROPERTY;
if (childrenByName.containsKey(propQName)) return ModelPropertyType.TREE_ROOT;
return ModelPropertyType.UNKNOWN;
}
}
public Class> getNonChildType(String propNamespace, String propName) {
QName propQName = QNameUtilities.createQName(propNamespace, propName);
synchronized (lock) {
ChildDataModel cd = nonChildProperty.get(propQName);
if (cd == null) return null;
return cd.getChildTypeAsClass();
}
}
public ParentedModel getChild(String propNamespace, String propName) {
QName propQName = QNameUtilities.createQName(propNamespace, propName);
synchronized (lock) {
return childrenByName.get(propQName);
}
}
@Override
public Class> getOriginalInterfaceAsClass() {
if (originalInterfaceAsClass != null) return originalInterfaceAsClass;
synchronized (lock) {
if (originalInterfaceAsClass != null) return originalInterfaceAsClass;
originalInterfaceAsClass = GeneralUtilities.loadClass(myLoader, originalInterface);
return originalInterfaceAsClass;
}
}
@Override
public Class> getProxyAsClass() {
if (translatedClassAsClass != null) return translatedClassAsClass;
synchronized (lock) {
if (translatedClassAsClass != null) return translatedClassAsClass;
translatedClassAsClass = GeneralUtilities.loadClass(myLoader, translatedClass);
return translatedClassAsClass;
}
}
public Collection getAllChildren() {
synchronized (lock) {
return Collections.unmodifiableCollection(childrenByName.values());
}
}
public Map getChildrenProperties() {
synchronized (lock) {
return Collections.unmodifiableMap(childrenByName);
}
}
public Map getAllAttributeChildren() {
Map retVal = new LinkedHashMap();
for (Map.Entry candidate : nonChildProperty.entrySet()) {
QName xmlKey = candidate.getKey();
ChildDataModel childDataModel = candidate.getValue();
if (!Format.ATTRIBUTE.equals(childDataModel.getFormat())) continue;
retVal.put(xmlKey, childDataModel);
}
return retVal;
}
public Map getAllElementChildren() {
Map retVal = new LinkedHashMap();
for (Map.Entry candidate : allChildren.entrySet()) {
QName xmlKey = candidate.getKey();
ChildDescriptor childDescriptor = candidate.getValue();
if (childDescriptor.getParentedModel() != null) {
// Is an element since it is a child!
retVal.put(xmlKey, childDescriptor);
continue;
}
ChildDataModel childDataModel = childDescriptor.getChildDataModel();
if (!Format.ELEMENT.equals(childDataModel.getFormat())) continue;
retVal.put(xmlKey, childDescriptor);
}
return retVal;
}
public synchronized String getJavaNameFromKey(String key, ClassReflectionHelper reflectionHelper) {
if (keyToJavaNameMap == null) {
keyToJavaNameMap = new LinkedHashMap();
}
String result = keyToJavaNameMap.get(key);
if (result != null) return result;
if (reflectionHelper == null) return null;
Class> originalInterface = getOriginalInterfaceAsClass();
for (MethodWrapper wrapper : reflectionHelper.getAllMethods(originalInterface)) {
Method m = wrapper.getMethod();
String xmlName;
XmlElement element = m.getAnnotation(XmlElement.class);
if (element == null) {
XmlAttribute attribute = m.getAnnotation(XmlAttribute.class);
if (attribute == null) continue;
xmlName = attribute.name();
}
else {
xmlName = element.name();
}
String keyName;
String javaName = getJavaNameFromGetterOrSetter(m, reflectionHelper);
if ("##default".equals(xmlName)) {
keyName = javaName;
}
else {
keyName = xmlName;
}
if (!key.equals(keyName)) continue;
// Found it!
keyToJavaNameMap.put(key, javaName);
return javaName;
}
return null;
}
private static String getJavaNameFromGetterOrSetter(Method m, ClassReflectionHelper reflectionHelper) {
AltMethod alt = new MethodAltMethodImpl(m, reflectionHelper);
String retVal = Utilities.isGetter(alt);
if (retVal != null) return retVal;
return Utilities.isSetter(alt);
}
@Override
public int hashCode() {
return translatedClass.hashCode();
}
@Override
public boolean equals(Object o) {
if (o == null) return false;
if (!(o instanceof ModelImpl)) return false;
return translatedClass.equals(((ModelImpl) o).getTranslatedClass());
}
@Override
public String toString() {
return "ModelImpl(interface=" + originalInterface +
",class=" + translatedClass +
",root=" + rootName +
",keyProperty=" + keyProperty +
",valuePropertyNamespace=" + valuePropertyNamespace +
",valueProperty=" + valueProperty +
"," + System.identityHashCode(this) + ")";
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy