Open Source Repository

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



org/hibernate/hql/ast/tree/AssignmentSpecification.java
// $Id: AssignmentSpecification.java 8273 2005-09-30 17:54:42Z steveebersole $
package org.hibernate.hql.ast.tree;

import java.util.Iterator;
import java.util.List;
import java.util.Set;
import java.util.HashSet;
import java.util.Collections;

import org.hibernate.QueryException;
import org.hibernate.engine.SessionFactoryImplementor;
import org.hibernate.hql.antlr.HqlSqlTokenTypes;
import org.hibernate.hql.ast.SqlGenerator;
import org.hibernate.hql.ast.util.ASTUtil;
import org.hibernate.param.ParameterSpecification;
import org.hibernate.persister.entity.Queryable;
import org.hibernate.persister.entity.UnionSubclassEntityPersister;

import antlr.collections.AST;

/**
 * Encapsulates the information relating to an individual assignment within the
 * set clause of an HQL update statement.  This information is used during execution
 * of the update statements when the updates occur against "multi-table" stuff.
 *
 @author Steve Ebersole
 */
public class AssignmentSpecification {

  private final Set tableNames;
  private final ParameterSpecification[] hqlParameters;
  private final AST eq;
  private final SessionFactoryImplementor factory;

  private String sqlAssignmentString;

  public AssignmentSpecification(AST eq, Queryable persister) {
    if eq.getType() != HqlSqlTokenTypes.EQ ) {
      throw new QueryException"assignment in set-clause not associated with equals" );
    }

    this.eq = eq;
    this.factory = persister.getFactory();

    // Needed to bump this up to DotNode, because that is the only thing which currently
    // knows about the property-ref path in the correct format; it is either this, or
    // recurse over the DotNodes constructing the property path just like DotNode does
    // internally
    DotNode lhs = DotNode eq.getFirstChild();
    SqlNode rhs = SqlNode lhs.getNextSibling();

    validateLhslhs );

    final String propertyPath = lhs.getPropertyPath();
    Set temp = new HashSet();
    // yuck!
    if persister instanceof UnionSubclassEntityPersister ) {
      UnionSubclassEntityPersister usep = UnionSubclassEntityPersister persister;
      String[] tables = persister.getConstraintOrderedTableNameClosure();
      int size = tables.length;
      for int i = 0; i < size; i ++ ) {
        temp.addtables[i] );
      }
    }
    else {
      temp.add(
          persister.getSubclassTableNamepersister.getSubclassPropertyTableNumberpropertyPath ) )
      );
    }
    this.tableNames = Collections.unmodifiableSettemp );

    if (rhs==null) {
      hqlParameters = new ParameterSpecification[0];
    }
    else if isParamrhs ) ) {
      hqlParameters = new ParameterSpecification[] { ( ( ParameterNode rhs ).getHqlParameterSpecification() };
    }
    else {
      List parameterList = ASTUtil.collectChildren(
              rhs,
              new ASTUtil.IncludePredicate() {
                public boolean include(AST node) {
                  return isParamnode );
                  }
              }
      );
      hqlParameters = new ParameterSpecificationparameterList.size() ];
      Iterator itr = parameterList.iterator();
      int i = 0;
      whileitr.hasNext() ) {
        hqlParameters[i++( ( ParameterNode itr.next() ).getHqlParameterSpecification();
      }
    }
  }

  public boolean affectsTable(String tableName) {
    return this.tableNames.containstableName );
  }

  public ParameterSpecification[] getParameters() {
    return hqlParameters;
  }

  public String getSqlAssignmentFragment() {
    if sqlAssignmentString == null ) {
      try {
        SqlGenerator sqlGenerator = new SqlGeneratorfactory );
        sqlGenerator.comparisonExpreq, false );  // false indicates to not generate parens around the assignment
        sqlAssignmentString = sqlGenerator.getSQL();
      }
      catchThrowable t ) {
        throw new QueryException"cannot interpret set-clause assignment" );
      }
    }
    return sqlAssignmentString;
  }

  private static boolean isParam(AST node) {
    return node.getType() == HqlSqlTokenTypes.PARAM || node.getType() == HqlSqlTokenTypes.NAMED_PARAM;
  }

  private void validateLhs(FromReferenceNode lhs) {
    // make sure the lhs is "assignable"...
    if !lhs.isResolved() ) {
      throw new UnsupportedOperationException"cannot validate assignablity of unresolved node" );
    }

    if lhs.getDataType().isCollectionType() ) {
      throw new QueryException"collections not assignable in update statements" );
    }
    else if lhs.getDataType().isComponentType() ) {
      throw new QueryException"Components currently not assignable in update statements" );
    }
    else if lhs.getDataType().isEntityType() ) {
      // currently allowed...
    }

    // TODO : why aren't these the same?
    if lhs.getImpliedJoin() != null || lhs.getFromElement().isImplied() ) {
      throw new QueryException"Implied join paths are not assignable in update statements" );
    }
  }
}