Open Source Repository

Home /hibernate/hibernate-3.2.4.ga | Repository Home



org/hibernate/mapping/ForeignKey.java
//$Id: ForeignKey.java 7360 2005-07-01 16:38:03Z maxcsaucdk $
package org.hibernate.mapping;

import java.util.ArrayList;
import java.util.Iterator;
import java.util.List;

import org.hibernate.MappingException;
import org.hibernate.dialect.Dialect;

/**
 * A foreign key constraint
 @author Gavin King
 */
public class ForeignKey extends Constraint {

  private Table referencedTable;
  private String referencedEntityName;
  private boolean cascadeDeleteEnabled;
  private List referencedColumns = new ArrayList();    
    
  public String sqlConstraintString(Dialect dialect, String constraintName, String defaultCatalog, String defaultSchema) {
    String[] cols = new StringgetColumnSpan() ];
    String[] refcols = new StringgetColumnSpan() ];
    int i=0;
    Iterator refiter = null;
    if(isReferenceToPrimaryKey() ) {
      refiter = referencedTable.getPrimaryKey().getColumnIterator();
    
    else {
      refiter = referencedColumns.iterator();
    }
    
    Iterator iter = getColumnIterator();
    while iter.hasNext() ) {
      cols[i( (Columniter.next() ).getQuotedName(dialect);
      refcols[i( (Columnrefiter.next() ).getQuotedName(dialect);
      i++;
    }
    String result = dialect.getAddForeignKeyConstraintString(
      constraintName, cols, referencedTable.getQualifiedName(dialect, defaultCatalog, defaultSchema), refcols, isReferenceToPrimaryKey()
    );
    return cascadeDeleteEnabled && dialect.supportsCascadeDelete() 
      result + " on delete cascade" 
      result;
  }

  public Table getReferencedTable() {
    return referencedTable;
  }

  private void appendColumns(StringBuffer buf, Iterator columns) {
    whilecolumns.hasNext() ) {
      Column column = (Columncolumns.next();
      buf.appendcolumn.getName() );
      if columns.hasNext() ) buf.append(",");
    }
  }

  public void setReferencedTable(Table referencedTablethrows MappingException {
    //if( isReferenceToPrimaryKey() ) alignColumns(referencedTable); // TODO: possibly remove to allow more piecemal building of a foreignkey.  
    
    this.referencedTable = referencedTable;
  }

  /**
   * Validates that columnspan of the foreignkey and the primarykey is the same.
   
   * Furthermore it aligns the length of the underlying tables columns.
   @param referencedTable
   */
  public void alignColumns() {
    if isReferenceToPrimaryKey() ) alignColumns(referencedTable);
  }
  
  private void alignColumns(Table referencedTable) {
    if referencedTable.getPrimaryKey().getColumnSpan()!=getColumnSpan() ) {
      StringBuffer sb = new StringBuffer();
      sb.append("Foreign key (")
                .appendgetName() ":")
        .appendgetTable().getName() )
        .append(" [");
      appendColumnssb, getColumnIterator() );
      sb.append("])")
        .append(") must have same number of columns as the referenced primary key (")
        .appendreferencedTable.getName() )
        .append(" [");
      appendColumnssb, referencedTable.getPrimaryKey().getColumnIterator() );
      sb.append("])");
      throw new MappingExceptionsb.toString() );
    }
    
    Iterator fkCols = getColumnIterator();
    Iterator pkCols = referencedTable.getPrimaryKey().getColumnIterator();
    while pkCols.hasNext() ) {
      ( (ColumnfkCols.next() ).setLength( ( (ColumnpkCols.next() ).getLength() );
    }

  }

  public String getReferencedEntityName() {
    return referencedEntityName;
  }

  public void setReferencedEntityName(String referencedEntityName) {
    this.referencedEntityName = referencedEntityName;
  }

  public String sqlDropString(Dialect dialect, String defaultCatalog, String defaultSchema) {
    return "alter table " 
      getTable().getQualifiedName(dialect, defaultCatalog, defaultSchema
      dialect.getDropForeignKeyString() 
      getName();
  }

  public boolean isCascadeDeleteEnabled() {
    return cascadeDeleteEnabled;
  }

  public void setCascadeDeleteEnabled(boolean cascadeDeleteEnabled) {
    this.cascadeDeleteEnabled = cascadeDeleteEnabled;
  }
  
  public boolean isPhysicalConstraint() {
    return referencedTable.isPhysicalTable() && 
        getTable().isPhysicalTable() && 
        !referencedTable.hasDenormalizedTables();
  }

  /** Returns the referenced columns if the foreignkey does not refer to the primary key */
  public List getReferencedColumns() {
    return referencedColumns;    
  }

  /** Does this foreignkey reference the primary key of the reference table */ 
  public boolean isReferenceToPrimaryKey() {
    return referencedColumns.isEmpty();
  }

  public void addReferencedColumns(Iterator referencedColumnsIterator) {
    while referencedColumnsIterator.hasNext() ) {
      Selectable col = (SelectablereferencedColumnsIterator.next();
      if !col.isFormula() ) addReferencedColumn( (Columncol );
    }
  }

  private void addReferencedColumn(Column column) {
    if !referencedColumns.contains(column) ) referencedColumns.add(column);    
  }
  
  public String toString() {
    if(!isReferenceToPrimaryKey() ) {
      StringBuffer result = new StringBuffer(getClass().getName() '(' + getTable().getName() + getColumns() );
      result.append" ref-columns:" '(' + getReferencedColumns() );
      result.append") as " + getName() );
      return result.toString();
    
    else {
      return super.toString();
    }
    
  }
}