/*
 * Hibernate, Relational Persistence for Idiomatic Java
 *
 * License: GNU Lesser General Public License (LGPL), version 2.1 or later.
 * See the lgpl.txt file in the root directory or <http://www.gnu.org/licenses/lgpl-2.1.html>.
 */
package org.hibernate.cfg;

import java.util.Iterator;
import java.util.Map;
import javax.persistence.ConstraintMode;

import org.hibernate.AnnotationException;
import org.hibernate.MappingException;
import org.hibernate.annotations.ForeignKey;
import org.hibernate.annotations.LazyGroup;
import org.hibernate.annotations.common.reflection.XClass;
import org.hibernate.boot.spi.MetadataBuildingContext;
import org.hibernate.cfg.annotations.PropertyBinder;
import org.hibernate.internal.util.StringHelper;
import org.hibernate.mapping.Column;
import org.hibernate.mapping.Component;
import org.hibernate.mapping.DependantValue;
import org.hibernate.mapping.Join;
import org.hibernate.mapping.ManyToOne;
import org.hibernate.mapping.OneToOne;
import org.hibernate.mapping.PersistentClass;
import org.hibernate.mapping.Property;
import org.hibernate.mapping.SimpleValue;
import org.hibernate.type.ForeignKeyDirection;

We have to handle OneToOne in a second pass because: -
/** * We have to handle OneToOne in a second pass because: * - */
public class OneToOneSecondPass implements SecondPass { private MetadataBuildingContext buildingContext; private String mappedBy; private String ownerEntity; private String ownerProperty; private PropertyHolder propertyHolder; private boolean ignoreNotFound; private PropertyData inferredData; private XClass targetEntity; private boolean cascadeOnDelete; private boolean optional; private String cascadeStrategy; private Ejb3JoinColumn[] joinColumns; //that suck, we should read that from the property mainly public OneToOneSecondPass( String mappedBy, String ownerEntity, String ownerProperty, PropertyHolder propertyHolder, PropertyData inferredData, XClass targetEntity, boolean ignoreNotFound, boolean cascadeOnDelete, boolean optional, String cascadeStrategy, Ejb3JoinColumn[] columns, MetadataBuildingContext buildingContext) { this.ownerEntity = ownerEntity; this.ownerProperty = ownerProperty; this.mappedBy = mappedBy; this.propertyHolder = propertyHolder; this.buildingContext = buildingContext; this.ignoreNotFound = ignoreNotFound; this.inferredData = inferredData; this.targetEntity = targetEntity; this.cascadeOnDelete = cascadeOnDelete; this.optional = optional; this.cascadeStrategy = cascadeStrategy; this.joinColumns = columns; } //TODO refactor this code, there is a lot of duplication in this method public void doSecondPass(Map persistentClasses) throws MappingException { org.hibernate.mapping.OneToOne value = new org.hibernate.mapping.OneToOne( buildingContext, propertyHolder.getTable(), propertyHolder.getPersistentClass() ); final String propertyName = inferredData.getPropertyName(); value.setPropertyName( propertyName ); String referencedEntityName = ToOneBinder.getReferenceEntityName( inferredData, targetEntity, buildingContext ); value.setReferencedEntityName( referencedEntityName ); AnnotationBinder.defineFetchingStrategy( value, inferredData.getProperty() ); //value.setFetchMode( fetchMode ); value.setCascadeDeleteEnabled( cascadeOnDelete ); //value.setLazy( fetchMode != FetchMode.JOIN ); if ( !optional ) { value.setConstrained( true ); } if ( value.isReferenceToPrimaryKey() ) { value.setForeignKeyType( ForeignKeyDirection.TO_PARENT ); } else { value.setForeignKeyType( value.isConstrained() ? ForeignKeyDirection.FROM_PARENT : ForeignKeyDirection.TO_PARENT ); } PropertyBinder binder = new PropertyBinder(); binder.setName( propertyName ); binder.setValue( value ); binder.setCascade( cascadeStrategy ); binder.setAccessType( inferredData.getDefaultAccess() ); final LazyGroup lazyGroupAnnotation = inferredData.getProperty().getAnnotation( LazyGroup.class ); if ( lazyGroupAnnotation != null ) { binder.setLazyGroup( lazyGroupAnnotation.value() ); } Property prop = binder.makeProperty(); prop.setOptional( optional ); if ( BinderHelper.isEmptyAnnotationValue( mappedBy ) ) { /* * we need to check if the columns are in the right order * if not, then we need to create a many to one and formula * but actually, since entities linked by a one to one need * to share the same composite id class, this cannot happen in hibernate */ boolean rightOrder = true; if ( rightOrder ) { String path = StringHelper.qualify( propertyHolder.getPath(), propertyName ); final ToOneFkSecondPass secondPass = new ToOneFkSecondPass( value, joinColumns, !optional, //cannot have nullabe and unique on certain DBs propertyHolder.getEntityOwnerClassName(), path, buildingContext ); secondPass.doSecondPass( persistentClasses ); //no column associated since its a one to one propertyHolder.addProperty( prop, inferredData.getDeclaringClass() ); } else { //this is a many to one with Formula } } else { PersistentClass otherSide = (PersistentClass) persistentClasses.get( value.getReferencedEntityName() ); Property otherSideProperty; try { if ( otherSide == null ) { throw new MappingException( "Unable to find entity: " + value.getReferencedEntityName() ); } otherSideProperty = BinderHelper.findPropertyByName( otherSide, mappedBy ); } catch (MappingException e) { throw new AnnotationException( "Unknown mappedBy in: " + StringHelper.qualify( ownerEntity, ownerProperty ) + ", referenced property unknown: " + StringHelper.qualify( value.getReferencedEntityName(), mappedBy ) ); } if ( otherSideProperty == null ) { throw new AnnotationException( "Unknown mappedBy in: " + StringHelper.qualify( ownerEntity, ownerProperty ) + ", referenced property unknown: " + StringHelper.qualify( value.getReferencedEntityName(), mappedBy ) ); } if ( otherSideProperty.getValue() instanceof OneToOne ) { propertyHolder.addProperty( prop, inferredData.getDeclaringClass() ); } else if ( otherSideProperty.getValue() instanceof ManyToOne ) { Iterator it = otherSide.getJoinIterator(); Join otherSideJoin = null; while ( it.hasNext() ) { Join otherSideJoinValue = (Join) it.next(); if ( otherSideJoinValue.containsProperty( otherSideProperty ) ) { otherSideJoin = otherSideJoinValue; break; } } if ( otherSideJoin != null ) { //@OneToOne @JoinTable Join mappedByJoin = buildJoinFromMappedBySide( (PersistentClass) persistentClasses.get( ownerEntity ), otherSideProperty, otherSideJoin ); ManyToOne manyToOne = new ManyToOne( buildingContext, mappedByJoin.getTable() ); //FIXME use ignore not found here manyToOne.setIgnoreNotFound( ignoreNotFound ); manyToOne.setCascadeDeleteEnabled( value.isCascadeDeleteEnabled() ); manyToOne.setFetchMode( value.getFetchMode() ); manyToOne.setLazy( value.isLazy() ); manyToOne.setReferencedEntityName( value.getReferencedEntityName() ); manyToOne.setUnwrapProxy( value.isUnwrapProxy() ); prop.setValue( manyToOne ); Iterator otherSideJoinKeyColumns = otherSideJoin.getKey().getColumnIterator(); while ( otherSideJoinKeyColumns.hasNext() ) { Column column = (Column) otherSideJoinKeyColumns.next(); Column copy = new Column(); copy.setLength( column.getLength() ); copy.setScale( column.getScale() ); copy.setValue( manyToOne ); copy.setName( column.getQuotedName() ); copy.setNullable( column.isNullable() ); copy.setPrecision( column.getPrecision() ); copy.setUnique( column.isUnique() ); copy.setSqlType( column.getSqlType() ); copy.setCheckConstraint( column.getCheckConstraint() ); copy.setComment( column.getComment() ); copy.setDefaultValue( column.getDefaultValue() ); manyToOne.addColumn( copy ); } mappedByJoin.addProperty( prop ); } else { propertyHolder.addProperty( prop, inferredData.getDeclaringClass() ); } value.setReferencedPropertyName( mappedBy ); // HHH-6813 // Foo: @Id long id, @OneToOne(mappedBy="foo") Bar bar // Bar: @Id @OneToOne Foo foo boolean referencesDerivedId = false; try { referencesDerivedId = otherSide.getIdentifier() instanceof Component && ( (Component) otherSide.getIdentifier() ).getProperty( mappedBy ) != null; } catch ( MappingException e ) { // ignore } boolean referenceToPrimaryKey = referencesDerivedId || mappedBy == null; value.setReferenceToPrimaryKey( referenceToPrimaryKey ); // If the other side is a derived ID, prevent an infinite // loop of attempts to resolve identifiers. if ( referencesDerivedId ) { ( (ManyToOne) otherSideProperty.getValue() ).setReferenceToPrimaryKey( false ); } String propertyRef = value.getReferencedPropertyName(); if ( propertyRef != null ) { buildingContext.getMetadataCollector().addUniquePropertyReference( value.getReferencedEntityName(), propertyRef ); } } else { throw new AnnotationException( "Referenced property not a (One|Many)ToOne: " + StringHelper.qualify( otherSide.getEntityName(), mappedBy ) + " in mappedBy of " + StringHelper.qualify( ownerEntity, ownerProperty ) ); } } final ForeignKey fk = inferredData.getProperty().getAnnotation( ForeignKey.class ); if ( fk != null && !BinderHelper.isEmptyAnnotationValue( fk.name() ) ) { value.setForeignKeyName( fk.name() ); } else { final javax.persistence.ForeignKey jpaFk = inferredData.getProperty().getAnnotation( javax.persistence.ForeignKey.class ); if ( jpaFk != null ) { if ( jpaFk.value() == ConstraintMode.NO_CONSTRAINT ) { value.setForeignKeyName( "none" ); } else { value.setForeignKeyName( StringHelper.nullIfEmpty( jpaFk.name() ) ); value.setForeignKeyDefinition( StringHelper.nullIfEmpty( jpaFk.foreignKeyDefinition() ) ); } } } }
Builds the Join instance for the mapped by side of a OneToOne association using a join tables.

Note:

  • From the mappedBy side we should not create the PK nor the FK, this is handled from the other side.
  • This method is a dirty dupe of EntityBinder.bindSecondaryTable.

/** * Builds the <code>Join</code> instance for the mapped by side of a <i>OneToOne</i> association using * a join tables. * <p> * Note:<br/> * <ul> * <li>From the mappedBy side we should not create the PK nor the FK, this is handled from the other side.</li> * <li>This method is a dirty dupe of EntityBinder.bindSecondaryTable</i>. * </p> */
private Join buildJoinFromMappedBySide(PersistentClass persistentClass, Property otherSideProperty, Join originalJoin) { Join join = new Join(); join.setPersistentClass( persistentClass ); //no check constraints available on joins join.setTable( originalJoin.getTable() ); join.setInverse( true ); SimpleValue key = new DependantValue( buildingContext, join.getTable(), persistentClass.getIdentifier() ); //TODO support @ForeignKey join.setKey( key ); join.setSequentialSelect( false ); //TODO support for inverse and optional join.setOptional( true ); //perhaps not quite per-spec, but a Good Thing anyway key.setCascadeDeleteEnabled( false ); Iterator mappedByColumns = otherSideProperty.getValue().getColumnIterator(); while ( mappedByColumns.hasNext() ) { Column column = (Column) mappedByColumns.next(); Column copy = new Column(); copy.setLength( column.getLength() ); copy.setScale( column.getScale() ); copy.setValue( key ); copy.setName( column.getQuotedName() ); copy.setNullable( column.isNullable() ); copy.setPrecision( column.getPrecision() ); copy.setUnique( column.isUnique() ); copy.setSqlType( column.getSqlType() ); copy.setCheckConstraint( column.getCheckConstraint() ); copy.setComment( column.getComment() ); copy.setDefaultValue( column.getDefaultValue() ); key.addColumn( copy ); } persistentClass.addJoin( join ); return join; } }