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

org.eclipse.persistence.internal.jpa.metamodel.SingularAttributeImpl Maven / Gradle / Ivy

There is a newer version: 5.0.0-B03
Show newest version
/*******************************************************************************
 * Copyright (c) 2011, 2013 Oracle and/or its affiliates. All rights reserved.
 * This program and the accompanying materials are made available under the 
 * terms of the Eclipse Public License v1.0 and Eclipse Distribution License v. 1.0 
 * which accompanies this distribution. 
 * The Eclipse Public License is available at http://www.eclipse.org/legal/epl-v10.html
 * and the Eclipse Distribution License is available at 
 * http://www.eclipse.org/org/documents/edl-v10.php.
 *
 * Contributors: 
 *     05/26/2009-2.0  mobrien - API update
 *       - 266912: JPA 2.0 Metamodel API (part of the JSR-317 EJB 3.1 Criteria API)
 *     06/30/2009-2.0  mobrien - finish JPA Metadata API modifications in support
 *       of the Metamodel implementation for EclipseLink 2.0 release involving
 *       Map, ElementCollection and Embeddable types on MappedSuperclass descriptors
 *     08/06/2010-2.2 mobrien 322018 - reduce protected instance variables to private to enforce encapsulation       
 *     11/10/2011-2.4 Guy Pelletier 
 *       - 357474: Address primaryKey option from tenant discriminator column
 ******************************************************************************/
package org.eclipse.persistence.internal.jpa.metamodel;

import java.lang.reflect.Field;

import javax.persistence.metamodel.Bindable;
import javax.persistence.metamodel.SingularAttribute;
import javax.persistence.metamodel.Type;

import org.eclipse.persistence.internal.descriptors.OptimisticLockingPolicy;
import org.eclipse.persistence.logging.AbstractSessionLog;
import org.eclipse.persistence.logging.SessionLog;
import org.eclipse.persistence.mappings.AggregateMapping;
import org.eclipse.persistence.mappings.DatabaseMapping;
import org.eclipse.persistence.mappings.ForeignReferenceMapping;
import org.eclipse.persistence.mappings.VariableOneToOneMapping;
import org.eclipse.persistence.mappings.foundation.AbstractDirectMapping;
import org.eclipse.persistence.mappings.structures.ReferenceMapping;

/**
 * 

* Purpose: Provides the implementation for the SingularAttribute interface * of the JPA 2.0 Metamodel API (part of the JSR-317 EJB 3.1 Criteria API) *

* Description: * Instances of the type SingularAttribute represents persistent * single-valued properties or fields. * * @author Michael O'Brien * @see javax.persistence.metamodel.SingularAttribute * @since EclipseLink 1.2 - JPA 2.0 * * @param The type containing the represented attribute * @param The type of the represented attribute * */ public class SingularAttributeImpl extends AttributeImpl implements SingularAttribute { /** Item 54: DI 89: explicit UID will avoid performance hit runtime generation of one */ private static final long serialVersionUID = 3928292425281232234L; /** The Type representing this Entity or Basic type **/ private Type elementType; /** * Create an instance of the Attribute * @param managedType * @param mapping */ protected SingularAttributeImpl(ManagedTypeImpl managedType, DatabaseMapping mapping) { this(managedType, mapping, false); } /** * INTERNAL: * Create an Attribute instance with a passed in validation flag (usually set to true only during Metamodel initialization) * @param managedType * @param mapping * @param validationEnabled */ protected SingularAttributeImpl(ManagedTypeImpl managedType, DatabaseMapping mapping, boolean validationEnabled) { super(managedType, mapping); // Case: Handle primitive or java lang type (non-Entity) targets Class attributeClass = mapping.getAttributeClassification(); /** * Case: Handle Entity targets * Process supported mappings by assigning their elementType. * For unsupported mappings we default to MetamodelImpl.DEFAULT_ELEMENT_TYPE. * If attribute is a primitive type (non-null) - we will wrap it in a BasicType automatically in getType below * The attribute classification is null for non-collection mappings such as embeddable keys. */ if (null == attributeClass) { // We support @OneToOne but not EIS, Reference or VariableOneToOne // Note: OneToMany, ManyToMany are handled by PluralAttributeImpl if(mapping instanceof ForeignReferenceMapping) {// handles @ManyToOne attributeClass = ((ForeignReferenceMapping)mapping).getReferenceClass(); } else if (mapping.isAbstractDirectMapping()) { // Also handles the keys of an EmbeddedId attributeClass = mapping.getField().getType(); if(null == attributeClass) { // lookup the attribute on the containing class attributeClass = managedType.getTypeClassFromAttributeOrMethodLevelAccessor(mapping); } } else if (mapping.isAggregateObjectMapping()) { // IE: EmbeddedId attributeClass = ((AggregateMapping)mapping).getReferenceClass(); } else if (mapping.isVariableOneToOneMapping()) { // interfaces are unsupported in the JPA 2.0 spec for the Metamodel API if(validationEnabled) { AbstractSessionLog.getLog().log(SessionLog.FINEST, SessionLog.METAMODEL, "metamodel_mapping_type_is_unsupported", mapping, this); } // see JUnitCriteriaUnitTestSuite.testSelectPhoneNumberAreaCode() line: 246 // VariableOneToOne mappings are unsupported - default to referenceClass (Interface) anyway // see interface org.eclipse.persistence.testing.models.jpa.relationships.Distributor attributeClass = ((VariableOneToOneMapping)mapping).getReferenceClass(); } else if (mapping.isEISMapping() || mapping.isTransformationMapping()) { // unsupported in the JPA 2.0 spec for the Metamodel API if(validationEnabled) { AbstractSessionLog.getLog().log(SessionLog.FINEST, SessionLog.METAMODEL, "metamodel_mapping_type_is_unsupported", mapping, this); } } else if ( mapping.isReferenceMapping()) { // unsupported in the JPA 2.0 spec for the Metamodel API if(validationEnabled) { AbstractSessionLog.getLog().log(SessionLog.FINEST, SessionLog.METAMODEL, "metamodel_mapping_type_is_unsupported", mapping, this); } // Reference mappings are unsupported - default to referenceClass anyway attributeClass = ((ReferenceMapping)mapping).getReferenceClass(); } } // All unsupported mappings such as TransformationMapping if(null == attributeClass && validationEnabled) { // TODO: refactor attributeClass = MetamodelImpl.DEFAULT_ELEMENT_TYPE_FOR_UNSUPPORTED_MAPPINGS; AbstractSessionLog.getLog().log(SessionLog.FINEST, SessionLog.METAMODEL, "metamodel_attribute_class_type_is_null", this); } elementType = getMetamodel().getType(attributeClass); } /** * Return the Java type of the represented object. * If the bindable type of the object is PLURAL_ATTRIBUTE, * the Java element type is returned. If the bindable type is * SINGULAR_ATTRIBUTE or ENTITY_TYPE, * the Java type of the * represented entity or attribute is returned. * @return Java type */ public Class getBindableJavaType() { // In SingularAttribute our BindableType is SINGLE_ATTRIBUTE - return the java type of the represented entity return this.elementType.getJavaType(); } /** * Is the attribute an id attribute. * @return boolean indicating whether or not attribute is an id */ public boolean isId() { if(this.getManagedTypeImpl().isMappedSuperclass()) { // The field on the mapping is the same field in the pkFields list on the descriptor // 288792: We can use the new isJPAId field here return (this.getDescriptor().getPrimaryKeyFields().contains(this.getMapping().getField())); } else { // 288792: Some id mappings will return false for isPrimaryKeyMapping but true for isJPAId return getMapping().isPrimaryKeyMapping() || getMapping().isJPAId(); } } /** * Can the attribute be null. * @return boolean indicating whether or not the attribute can * be null */ public boolean isOptional() { return getMapping().isOptional(); } /** * INTERNAL: * Return whether the attribute is plural or singular * @return */ @Override public boolean isPlural() { return false; } /** * Is the attribute a version attribute. * @return boolean indicating whether or not attribute is * a version attribute */ public boolean isVersion() { if (getDescriptor().usesOptimisticLocking() && getMapping().isDirectToFieldMapping()) { OptimisticLockingPolicy policy = getDescriptor().getOptimisticLockingPolicy(); return policy.getWriteLockField().equals(((AbstractDirectMapping) getMapping()).getField()); } return false; } public Bindable.BindableType getBindableType() { return Bindable.BindableType.SINGULAR_ATTRIBUTE; } /** * Return the Java type of the represented attribute. * @return Java type */ @Override public Class getJavaType() { if(null == elementType) { Class aJavaType = getMapping().getAttributeClassification(); if(null == aJavaType) { aJavaType = getMapping().getField().getType(); if(null == aJavaType) { // lookup the attribute on the containing class Class containingClass = getMapping().getDescriptor().getJavaClass(); Field aField = null; try { aField = containingClass.getDeclaredField(getMapping().getAttributeName()); aJavaType = aField.getType(); return aJavaType; } catch (NoSuchFieldException nsfe) { // This exception will be warned about below if(null == aJavaType) { AbstractSessionLog.getLog().log(SessionLog.FINEST, SessionLog.METAMODEL, "metamodel_attribute_class_type_is_null", this); return MetamodelImpl.DEFAULT_ELEMENT_TYPE_FOR_UNSUPPORTED_MAPPINGS; } } } } return aJavaType; } else { return this.elementType.getJavaType(); } } /** * Return the type that represents the type of the attribute. * @return type of attribute */ public Type getType() { return elementType; } /** * Return the String representation of the receiver. */ @Override public String toString() { StringBuffer aBuffer = new StringBuffer("SingularAttributeImpl["); aBuffer.append(getType()); aBuffer.append(","); aBuffer.append(getMapping()); aBuffer.append("]"); return aBuffer.toString(); } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy