Open Source Repository

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



org/hibernate/id/MultipleHiLoPerTableGenerator.java
//$Id: MultipleHiLoPerTableGenerator.java 11321 2007-03-20 11:51:09Z [email protected] $
package org.hibernate.id;

import java.io.Serializable;
import java.sql.Connection;
import java.sql.PreparedStatement;
import java.sql.ResultSet;
import java.sql.SQLException;
import java.sql.Types;
import java.util.Properties;

import org.apache.commons.logging.Log;
import org.apache.commons.logging.LogFactory;
import org.hibernate.HibernateException;
import org.hibernate.LockMode;
import org.hibernate.MappingException;
import org.hibernate.dialect.Dialect;
import org.hibernate.engine.SessionImplementor;
import org.hibernate.engine.TransactionHelper;
import org.hibernate.mapping.Table;
import org.hibernate.type.Type;
import org.hibernate.util.PropertiesHelper;

/**
 *
 * A hilo <tt>IdentifierGenerator</tt> that returns a <tt>Long</tt>, constructed using
 * a hi/lo algorithm. The hi value MUST be fetched in a seperate transaction
 * to the <tt>Session</tt> transaction so the generator must be able to obtain
 * a new connection and commit it. Hence this implementation may not
 * be used  when the user is supplying connections. In this
 * case a <tt>SequenceHiLoGenerator</tt> would be a better choice (where
 * supported).<br>
 <br>
 *
 * A hilo <tt>IdentifierGenerator</tt> that uses a database
 * table to store the last generated values. A table can contains
 * several hi values. They are distinct from each other through a key
 <p/>
 <p>This implementation is not compliant with a user connection</p>
 <p/>
 
 <p>Allowed parameters (all of them are optional):</p>
 <ul>
 <li>table: table name (default <tt>hibernate_sequences</tt>)</li>
 <li>primary_key_column: key column name (default <tt>sequence_name</tt>)</li>
 <li>value_column: hi value column name(default <tt>sequence_next_hi_value</tt>)</li>
 <li>primary_key_value: key value for the current entity (default to the entity's primary table name)</li>
 <li>primary_key_length: length of the key column in DB represented as a varchar (default to 255)</li>
 <li>max_lo: max low value before increasing hi (default to Short.MAX_VALUE)</li>
 </ul>
 *
 @author Emmanuel Bernard
 @author <a href="mailto:[email protected]">Klaus Richarz</a>.
 */
public class MultipleHiLoPerTableGenerator 
  extends TransactionHelper
  implements PersistentIdentifierGenerator, Configurable {
  
  private static final Log log = LogFactory.getLog(MultipleHiLoPerTableGenerator.class);
  
  public static final String ID_TABLE = "table";
  public static final String PK_COLUMN_NAME = "primary_key_column";
  public static final String PK_VALUE_NAME = "primary_key_value";
  public static final String VALUE_COLUMN_NAME = "value_column";
  public static final String PK_LENGTH_NAME = "primary_key_length";

  private static final int DEFAULT_PK_LENGTH = 255;
  public static final String DEFAULT_TABLE = "hibernate_sequences";
  private static final String DEFAULT_PK_COLUMN = "sequence_name";
  private static final String DEFAULT_VALUE_COLUMN = "sequence_next_hi_value";
  
  private String tableName;
  private String pkColumnName;
  private String valueColumnName;
  private String query;
  private String insert;
  private String update;

  //hilo params
  public static final String MAX_LO = "max_lo";

  private long hi;
  private int lo;
  private int maxLo;
  private Class returnClass;
  private int keySize;


  public String[] sqlCreateStrings(Dialect dialectthrows HibernateException {
    return new String[] {
      new StringBufferdialect.getCreateTableString() )
          .append' ' )
          .appendtableName )
          .append" ( " )
          .appendpkColumnName )
          .append' ' )
          .appenddialect.getTypeNameTypes.VARCHAR, keySize, 0) )
          .append",  " )
          .appendvalueColumnName )
          .append' ' )
          .appenddialect.getTypeNameTypes.INTEGER ) )
          .append" ) " )
          .toString()
    };
  }

  public String[] sqlDropStrings(Dialect dialectthrows HibernateException {
    StringBuffer sqlDropString = new StringBuffer"drop table " );
    if dialect.supportsIfExistsBeforeTableName() ) {
      sqlDropString.append"if exists " );
    }
    sqlDropString.appendtableName ).appenddialect.getCascadeConstraintsString() );
    if dialect.supportsIfExistsAfterTableName() ) {
      sqlDropString.append" if exists" );
    }
    return new String[] { sqlDropString.toString() };
  }

  public Object generatorKey() {
    return tableName;
  }

  public Serializable doWorkInCurrentTransaction(Connection conn, String sqlthrows SQLException {
    int result;
    int rows;
    do {
      // The loop ensures atomicity of the
      // select + update even for no transaction
      // or read committed isolation level

      //sql = query;
      SQL.debug(query);
      PreparedStatement qps = conn.prepareStatement(query);
      PreparedStatement ips = null;
      try {
        //qps.setString(1, key);
        ResultSet rs = qps.executeQuery();
        boolean isInitialized = rs.next();
        if !isInitialized ) {
          result = 0;
          ips = conn.prepareStatement(insert);
          //ips.setString(1, key);
          ips.setInt(1, result);
          ips.execute();
        }
        else {
          result = rs.getInt(1);
        }
        rs.close();
      }
      catch (SQLException sqle) {
        log.error("could not read or init a hi value", sqle);
        throw sqle;
      }
      finally {
        if (ips != null) {
          ips.close();
        }
        qps.close();
      }

      //sql = update;
      PreparedStatement ups = conn.prepareStatement(update);
      try {
        ups.setInt1, result + );
        ups.setInt2, result );
        //ups.setString( 3, key );
        rows = ups.executeUpdate();
      }
      catch (SQLException sqle) {
        log.error("could not update hi value in: " + tableName, sqle);
        throw sqle;
      }
      finally {
        ups.close();
      }
    }
    while (rows==0);
    return new Integer(result);
  }

  public synchronized Serializable generate(SessionImplementor session, Object obj)
    throws HibernateException {
    if (maxLo < 1) {
      //keep the behavior consistent even for boundary usages
      int val = ( (IntegerdoWorkInNewTransaction(session) ).intValue();
      if (val == 0val = ( (IntegerdoWorkInNewTransaction(session) ).intValue();
      return IdentifierGeneratorFactory.createNumberval, returnClass );
    }
    if (lo>maxLo) {
      int hival = ( (IntegerdoWorkInNewTransaction(session) ).intValue();
      lo = (hival == 00;
      hi = hival * (maxLo+1);
      log.debug("new hi value: " + hival);
    }
    return IdentifierGeneratorFactory.createNumberhi + lo++, returnClass );
  }

  public void configure(Type type, Properties params, Dialect dialectthrows MappingException {
    tableName = PropertiesHelper.getString(ID_TABLE, params, DEFAULT_TABLE);
    pkColumnName = PropertiesHelper.getString(PK_COLUMN_NAME, params, DEFAULT_PK_COLUMN);
    valueColumnName = PropertiesHelper.getString(VALUE_COLUMN_NAME, params, DEFAULT_VALUE_COLUMN);
    String schemaName = params.getProperty(SCHEMA);
    String catalogName = params.getProperty(CATALOG);
    keySize = PropertiesHelper.getInt(PK_LENGTH_NAME, params, DEFAULT_PK_LENGTH);
    String keyValue = PropertiesHelper.getString(PK_VALUE_NAME, params, params.getProperty(TABLE) );

    if tableName.indexOf'.' )<) {
      tableName = Table.qualifycatalogName, schemaName, tableName );
    }

    query = "select " +
      valueColumnName +
      " from " +
      dialect.appendLockHint(LockMode.UPGRADE, tableName+
      " where " + pkColumnName + " = '" + keyValue + "'" +
      dialect.getForUpdateString();

    update = "update " +
      tableName +
      " set " +
      valueColumnName +
      " = ? where " +
      valueColumnName +
      " = ? and " +
      pkColumnName +
      " = '" 
      keyValue 
      "'";
    
    insert = "insert into " + tableName +
      "(" + pkColumnName + ", " +  valueColumnName + ") " +
      "values('"+ keyValue +"', ?)";


    //hilo config
    maxLo = PropertiesHelper.getInt(MAX_LO, params, Short.MAX_VALUE);
    lo = maxLo + 1// so we "clock over" on the first invocation
    returnClass = type.getReturnedClass();
  }
}