Package org.apache.derby.impl.sql.execute

Source Code of org.apache.derby.impl.sql.execute.CreateIndexConstantAction

/*

   Derby - Class org.apache.derby.impl.sql.execute.CreateIndexConstantAction

   Licensed to the Apache Software Foundation (ASF) under one or more
   contributor license agreements.  See the NOTICE file distributed with
   this work for additional information regarding copyright ownership.
   The ASF licenses this file to you under the Apache License, Version 2.0
   (the "License"); you may not use this file except in compliance with
   the License.  You may obtain a copy of the License at

      http://www.apache.org/licenses/LICENSE-2.0

   Unless required by applicable law or agreed to in writing, software
   distributed under the License is distributed on an "AS IS" BASIS,
   WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
   See the License for the specific language governing permissions and
   limitations under the License.

*/

package org.apache.derby.impl.sql.execute;

import java.util.Properties;

import org.apache.derby.catalog.UUID;
import org.apache.derby.catalog.types.StatisticsImpl;
import org.apache.derby.iapi.error.StandardException;
import org.apache.derby.iapi.reference.SQLState;
import org.apache.derby.iapi.services.io.FormatableBitSet;
import org.apache.derby.iapi.services.loader.ClassFactory;
import org.apache.derby.iapi.services.sanity.SanityManager;
import org.apache.derby.iapi.sql.Activation;
import org.apache.derby.iapi.sql.conn.LanguageConnectionContext;
import org.apache.derby.iapi.sql.depend.DependencyManager;
import org.apache.derby.iapi.sql.dictionary.ColumnDescriptor;
import org.apache.derby.iapi.sql.dictionary.ColumnDescriptorList;
import org.apache.derby.iapi.sql.dictionary.ConglomerateDescriptor;
import org.apache.derby.iapi.sql.dictionary.ConglomerateDescriptorList;
import org.apache.derby.iapi.sql.dictionary.ConstraintDescriptor;
import org.apache.derby.iapi.sql.dictionary.DataDescriptorGenerator;
import org.apache.derby.iapi.sql.dictionary.DataDictionary;
import org.apache.derby.iapi.sql.dictionary.IndexRowGenerator;
import org.apache.derby.iapi.sql.dictionary.SchemaDescriptor;
import org.apache.derby.iapi.sql.dictionary.StatisticsDescriptor;
import org.apache.derby.iapi.sql.dictionary.TableDescriptor;
import org.apache.derby.iapi.sql.execute.ConstantAction;
import org.apache.derby.iapi.sql.execute.ExecIndexRow;
import org.apache.derby.iapi.sql.execute.ExecRow;
import org.apache.derby.iapi.store.access.AccessFactoryGlobals;
import org.apache.derby.iapi.store.access.ColumnOrdering;
import org.apache.derby.iapi.store.access.ConglomerateController;
import org.apache.derby.iapi.store.access.GroupFetchScanController;
import org.apache.derby.iapi.store.access.RowLocationRetRowSource;
import org.apache.derby.iapi.store.access.ScanController;
import org.apache.derby.iapi.store.access.SortController;
import org.apache.derby.iapi.store.access.SortObserver;
import org.apache.derby.iapi.store.access.TransactionController;
import org.apache.derby.iapi.types.DataTypeDescriptor;
import org.apache.derby.iapi.types.DataValueDescriptor;
import org.apache.derby.iapi.types.RowLocation;
import org.apache.derby.iapi.types.TypeId;

// Used only to access a debug flag, will be removed or replaced.
import org.apache.derby.impl.services.daemon.IndexStatisticsDaemonImpl;

/**
* ConstantAction to create an index either through
* a CREATE INDEX statement or as a backing index to
* a constraint.
*/

class CreateIndexConstantAction extends IndexConstantAction
{
    /**
     * Is this for a CREATE TABLE, i.e. it is
     * for a constraint declared in a CREATE TABLE
     * statement that requires a backing index.
     */
    private final boolean forCreateTable;

  private boolean      unique;
  private boolean      uniqueWithDuplicateNulls;
  private String      indexType;
  private String[]    columnNames;
  private boolean[]    isAscending;
  private boolean      isConstraint;
  private UUID      conglomerateUUID;
  private Properties    properties;

  private ExecRow indexTemplateRow;

  /** Conglomerate number for the conglomerate created by this
   * constant action; -1L if this constant action has not been
   * executed.  If this constant action doesn't actually create
   * a new conglomerate--which can happen if it finds an existing
   * conglomerate that satisfies all of the criteria--then this
   * field will hold the conglomerate number of whatever existing
   * conglomerate was found.
   */
  private long conglomId;

  /** Conglomerate number of the physical conglomerate that we
   * will "replace" using this constant action.  That is, if
   * the purpose of this constant action is to create a new physical
   * conglomerate to replace a dropped physical conglomerate, then
   * this field holds the conglomerate number of the dropped physical
   * conglomerate. If -1L then we are not replacing a conglomerate,
   * we're simply creating a new index (and backing physical
   * conglomerate) as normal.
   */
  private long droppedConglomNum;

  // CONSTRUCTORS
  /**
     *   Make the ConstantAction to create an index.
     *
     * @param forCreateTable                Being executed within a CREATE TABLE
     *                                      statement
     * @param unique                    True means it will be a unique index
     * @param uniqueWithDuplicateNulls      True means index check and disallow
     *                                      any duplicate key if key has no
     *                                      column with a null value.  If any
     *                                      column in the key has a null value,
     *                                      no checking is done and insert will
     *                                      always succeed.
     * @param indexType                      type of index (BTREE, for example)
     * @param schemaName                  schema that table (and index)
     *                                      lives in.
     * @param indexName                      Name of the index
     * @param tableName                      Name of table the index will be on
     * @param tableId                    UUID of table
     * @param columnNames                  Names of the columns in the index,
     *                                      in order
     * @param isAscending                  Array of booleans telling asc/desc
     *                                      on each column
     * @param isConstraint                  TRUE if index is backing up a
     *                                      constraint, else FALSE
     * @param conglomerateUUID              ID of conglomerate
     * @param properties                  The optional properties list
     *                                      associated with the index.
     */
  CreateIndexConstantAction(
            boolean         forCreateTable,
            boolean      unique,
            boolean      uniqueWithDuplicateNulls,
            String      indexType,
            String      schemaName,
            String      indexName,
            String      tableName,
            UUID      tableId,
            String[]    columnNames,
            boolean[]    isAscending,
            boolean      isConstraint,
            UUID      conglomerateUUID,
            Properties    properties)
  {
    super(tableId, indexName, tableName, schemaName);

        this.forCreateTable             = forCreateTable;
    this.unique                     = unique;
    this.uniqueWithDuplicateNulls   = uniqueWithDuplicateNulls;
    this.indexType                  = indexType;
    this.columnNames                = columnNames;
    this.isAscending                = isAscending;
    this.isConstraint               = isConstraint;
    this.conglomerateUUID           = conglomerateUUID;
    this.properties                 = properties;
    this.conglomId                  = -1L;
    this.droppedConglomNum          = -1L;
  }

  /**
   * Make a ConstantAction that creates a new physical conglomerate
   * based on index information stored in the received descriptors.
   * Assumption is that the received ConglomerateDescriptor is still
   * valid (meaning it has corresponding entries in the system tables
   * and it describes some constraint/index that has _not_ been
   * dropped--though the physical conglomerate underneath has).
   *
   * This constructor is used in cases where the physical conglomerate
   * for an index has been dropped but the index still exists. That
   * can happen if multiple indexes share a physical conglomerate but
   * then the conglomerate is dropped as part of "drop index" processing
   * for one of the indexes. (Note that "indexes" here includes indexes
   * which were created to back constraints.) In that case we have to
   * create a new conglomerate to satisfy the remaining sharing indexes,
   * so that's what we're here for.  See ConglomerateDescriptor.drop()
   * for details on when that is necessary.
   */
  CreateIndexConstantAction(ConglomerateDescriptor srcCD,
    TableDescriptor td, Properties properties)
  {
    super(td.getUUID(),
      srcCD.getConglomerateName(), td.getName(), td.getSchemaName());

    this.forCreateTable = false;

    /* We get here when a conglomerate has been dropped and we
     * need to create (or find) another one to fill its place.
     * At this point the received conglomerate descriptor still
     * references the old (dropped) conglomerate, so we can
     * pull the conglomerate number from there.
     */
    this.droppedConglomNum = srcCD.getConglomerateNumber();

    /* Plug in the rest of the information from the received
     * descriptors.
     */
    IndexRowGenerator irg = srcCD.getIndexDescriptor();
    this.unique = irg.isUnique();
    this.uniqueWithDuplicateNulls = irg.isUniqueWithDuplicateNulls();
    this.indexType = irg.indexType();
    this.columnNames = srcCD.getColumnNames();
    this.isAscending = irg.isAscending();
    this.isConstraint = srcCD.isConstraint();
    this.conglomerateUUID = srcCD.getUUID();
    this.properties = properties;
    this.conglomId = -1L;

    /* The ConglomerateDescriptor may not know the names of
     * the columns it includes.  If that's true (which seems
     * to be the more common case) then we have to build the
     * list of ColumnNames ourselves.
     */
    if (columnNames == null)
    {
      int [] baseCols = irg.baseColumnPositions();
      columnNames = new String[baseCols.length];
      ColumnDescriptorList colDL = td.getColumnDescriptorList();
      for (int i = 0; i < baseCols.length; i++)
      {
        columnNames[i] =
          colDL.elementAt(baseCols[i]-1).getColumnName();
      }
    }
  }
       
  ///////////////////////////////////////////////
  //
  // OBJECT SHADOWS
  //
  ///////////////////////////////////////////////

  public  String  toString()
  {
    // Do not put this under SanityManager.DEBUG - it is needed for
    // error reporting.
    return "CREATE INDEX " + indexName;
  }

  // INTERFACE METHODS


  /**
   *  This is the guts of the Execution-time logic for
     *  creating an index.
     *
     *  <P>
     *  A index is represented as:
     *  <UL>
     <LI> ConglomerateDescriptor.
     *  </UL>
     *  No dependencies are created.
      *
     *  @see ConglomerateDescriptor
     *  @see SchemaDescriptor
   *  @see ConstantAction#executeConstantAction
   *
   * @exception StandardException    Thrown on failure
   */
  public void  executeConstantAction( Activation activation )
            throws StandardException
  {
    TableDescriptor       td;
    UUID             toid;
    ColumnDescriptor      columnDescriptor;
    int[]            baseColumnPositions;
    IndexRowGenerator      indexRowGenerator = null;
    ExecRow[]          baseRows;
    ExecIndexRow[]        indexRows;
    ExecRow[]          compactBaseRows;
    GroupFetchScanController    scan;
    RowLocationRetRowSource      rowSource;
    long            sortId;
    int              maxBaseColumnPosition = -1;

    LanguageConnectionContext lcc = activation.getLanguageConnectionContext();
    DataDictionary dd = lcc.getDataDictionary();
    DependencyManager dm = dd.getDependencyManager();
    TransactionController tc = lcc.getTransactionExecute();

    /*
    ** Inform the data dictionary that we are about to write to it.
    ** There are several calls to data dictionary "get" methods here
    ** that might be done in "read" mode in the data dictionary, but
    ** it seemed safer to do this whole operation in "write" mode.
    **
    ** We tell the data dictionary we're done writing at the end of
    ** the transaction.
    */
    dd.startWriting(lcc);

    /*
    ** If the schema descriptor is null, then
    ** we must have just read ourselves in. 
    ** So we will get the corresponding schema
    ** descriptor from the data dictionary.
    */
    SchemaDescriptor sd = dd.getSchemaDescriptor(schemaName, tc, true) ;


    /* Get the table descriptor. */
    /* See if we can get the TableDescriptor
     * from the Activation.  (Will be there
     * for backing indexes.)
     */
    td = activation.getDDLTableDescriptor();

    if (td == null)
    {
      /* tableId will be non-null if adding an index to
       * an existing table (as opposed to creating a
       * table with a constraint with a backing index).
       */
      if (tableId != null)
      {
        td = dd.getTableDescriptor(tableId);
      }
      else
      {
        td = dd.getTableDescriptor(tableName, sd, tc);
      }
    }

    if (td == null)
    {
      throw StandardException.newException(SQLState.LANG_CREATE_INDEX_NO_TABLE,
            indexName, tableName);
    }

    if (td.getTableType() == TableDescriptor.SYSTEM_TABLE_TYPE)
    {
      throw StandardException.newException(SQLState.LANG_CREATE_SYSTEM_INDEX_ATTEMPTED,
            indexName, tableName);
    }

    /* Get a shared table lock on the table. We need to lock table before
     * invalidate dependents, otherwise, we may interfere with the
     * compilation/re-compilation of DML/DDL.  See beetle 4325 and $WS/
     * docs/language/SolutionsToConcurrencyIssues.txt (point f).
     */
    lockTableForDDL(tc, td.getHeapConglomerateId(), false);

    // invalidate any prepared statements that
    // depended on this table (including this one)
    if (! forCreateTable)
    {
      dm.invalidateFor(td, DependencyManager.CREATE_INDEX, lcc);
    }

    // Translate the base column names to column positions
    baseColumnPositions = new int[columnNames.length];
    for (int i = 0; i < columnNames.length; i++)
    {
      // Look up the column in the data dictionary
      columnDescriptor = td.getColumnDescriptor(columnNames[i]);
      if (columnDescriptor == null)
      {
        throw StandardException.newException(SQLState.LANG_COLUMN_NOT_FOUND_IN_TABLE,
                              columnNames[i],
                              tableName);
      }

      TypeId typeId = columnDescriptor.getType().getTypeId();

      // Don't allow a column to be created on a non-orderable type
      ClassFactory cf = lcc.getLanguageConnectionFactory().getClassFactory();
      boolean isIndexable = typeId.orderable(cf);

      if (isIndexable && typeId.userType()) {
        String userClass = typeId.getCorrespondingJavaTypeName();

        // Don't allow indexes to be created on classes that
        // are loaded from the database. This is because recovery
        // won't be able to see the class and it will need it to
        // run the compare method.
        try {
          if (cf.isApplicationClass(cf.loadApplicationClass(userClass)))
            isIndexable = false;
        } catch (ClassNotFoundException cnfe) {
          // shouldn't happen as we just check the class is orderable
          isIndexable = false;
        }
      }

      if (!isIndexable) {
        throw StandardException.newException(SQLState.LANG_COLUMN_NOT_ORDERABLE_DURING_EXECUTION,
          typeId.getSQLTypeName());
      }

      // Remember the position in the base table of each column
      baseColumnPositions[i] = columnDescriptor.getPosition();

      if (maxBaseColumnPosition < baseColumnPositions[i])
        maxBaseColumnPosition = baseColumnPositions[i];
    }

    /* The code below tries to determine if the index that we're about
     * to create can "share" a conglomerate with an existing index.
     * If so, we will use a single physical conglomerate--namely, the
     * one that already exists--to support both indexes. I.e. we will
     * *not* create a new conglomerate as part of this constant action.
     */

    // check if we have similar indices already for this table
    ConglomerateDescriptor[] congDescs = td.getConglomerateDescriptors();
    boolean shareExisting = false;
    for (int i = 0; i < congDescs.length; i++)
    {
      ConglomerateDescriptor cd = congDescs[i];
      if ( ! cd.isIndex())
        continue;

      if (droppedConglomNum == cd.getConglomerateNumber())
      {
        /* We can't share with any conglomerate descriptor
         * whose conglomerate number matches the dropped
         * conglomerate number, because that descriptor's
         * backing conglomerate was dropped, as well.  If
         * we're going to share, we have to share with a
         * descriptor whose backing physical conglomerate
         * is still around.
         */
        continue;
      }

      IndexRowGenerator irg = cd.getIndexDescriptor();
      int[] bcps = irg.baseColumnPositions();
      boolean[] ia = irg.isAscending();
      int j = 0;

      /* The conditions which allow an index to share an existing
       * conglomerate are as follows:
       *
       * 1. the set of columns (both key and include columns) and their
       *  order in the index is the same as that of an existing index AND
       *
       * 2. the ordering attributes are the same AND
       *
       * 3. one of the following is true:
       *    a) the existing index is unique, OR
       *    b) the existing index is non-unique with uniqueWhenNotNulls
       *       set to TRUE and the index being created is non-unique, OR
       *    c) both the existing index and the one being created are
       *       non-unique and have uniqueWithDuplicateNulls set to FALSE.
       */
      boolean possibleShare = (irg.isUnique() || !unique) &&
          (bcps.length == baseColumnPositions.length);

      //check if existing index is non unique and uniqueWithDuplicateNulls
      //is set to true (backing index for unique constraint)
      if (possibleShare && !irg.isUnique ())
      {
        /* If the existing index has uniqueWithDuplicateNulls set to
         * TRUE it can be shared by other non-unique indexes; otherwise
         * the existing non-unique index has uniqueWithDuplicateNulls
         * set to FALSE, which means the new non-unique conglomerate
         * can only share if it has uniqueWithDuplicateNulls set to
         * FALSE, as well.
         */
        possibleShare = (irg.isUniqueWithDuplicateNulls() ||
                ! uniqueWithDuplicateNulls);
      }

      if (possibleShare && indexType.equals(irg.indexType()))
      {
        for (; j < bcps.length; j++)
        {
          if ((bcps[j] != baseColumnPositions[j]) || (ia[j] != isAscending[j]))
            break;
        }
      }

      if (j == baseColumnPositions.length// share
      {
        /*
         * Don't allow users to create a duplicate index. Allow if being done internally
         * for a constraint
         */
        if (!isConstraint)
        {
          activation.addWarning(
              StandardException.newWarning(
                SQLState.LANG_INDEX_DUPLICATE,
                cd.getConglomerateName()));

          return;
        }

        /* Sharing indexes share the physical conglomerate
         * underneath, so pull the conglomerate number from
         * the existing conglomerate descriptor.
         */
        conglomId = cd.getConglomerateNumber();

        /* We create a new IndexRowGenerator because certain
         * attributes--esp. uniqueness--may be different between
         * the index we're creating and the conglomerate that
         * already exists.  I.e. even though we're sharing a
         * conglomerate, the new index is not necessarily
         * identical to the existing conglomerate. We have to
         * keep track of that info so that if we later drop
         * the shared physical conglomerate, we can figure out
         * what this index (the one we're creating now) is
         * really supposed to look like.
         */
        indexRowGenerator =
          new IndexRowGenerator(
            indexType, unique, uniqueWithDuplicateNulls,
            baseColumnPositions,
            isAscending,
            baseColumnPositions.length);

        //DERBY-655 and DERBY-1343 
        // Sharing indexes will have unique logical conglomerate UUIDs.
        conglomerateUUID = dd.getUUIDFactory().createUUID();
        shareExisting = true;
        break;
      }
    }

    /* If we have a droppedConglomNum then the index we're about to
     * "create" already exists--i.e. it has an index descriptor and
     * the corresponding information is already in the system catalogs.
     * The only thing we're missing, then, is the physical conglomerate
     * to back the index (because the old conglomerate was dropped).
     */
    boolean alreadyHaveConglomDescriptor = (droppedConglomNum > -1L);

    /* If this index already has an essentially same one, we share the
     * conglomerate with the old one, and just simply add a descriptor
     * entry into SYSCONGLOMERATES--unless we already have a descriptor,
     * in which case we don't even need to do that.
     */
    DataDescriptorGenerator ddg = dd.getDataDescriptorGenerator();
    if (shareExisting && !alreadyHaveConglomDescriptor)
    {
      ConglomerateDescriptor cgd =
        ddg.newConglomerateDescriptor(conglomId, indexName, true,
                      indexRowGenerator, isConstraint,
                      conglomerateUUID, td.getUUID(), sd.getUUID() );
      dd.addDescriptor(cgd, sd, DataDictionary.SYSCONGLOMERATES_CATALOG_NUM, false, tc);
      // add newly added conglomerate to the list of conglomerate
      // descriptors in the td.
      ConglomerateDescriptorList cdl =
        td.getConglomerateDescriptorList();
      cdl.add(cgd);

      // can't just return yet, need to get member "indexTemplateRow"
      // because create constraint may use it
    }

    // Describe the properties of the index to the store using Properties
    // RESOLVE: The following properties assume a BTREE index.
    Properties  indexProperties;
   
    if (properties != null)
    {
      indexProperties = properties;
    }
    else
    {
      indexProperties = new Properties();
    }

    // Tell it the conglomerate id of the base table
    indexProperties.put("baseConglomerateId",
              Long.toString(td.getHeapConglomerateId()));
       
    if (uniqueWithDuplicateNulls)
        {
            if (dd.checkVersion(DataDictionary.DD_VERSION_DERBY_10_4, null))
            {
        indexProperties.put(
                    "uniqueWithDuplicateNulls", Boolean.toString(true));
      }
      else
            {
        // for lower version of DD there is no unique with nulls
                // index creating a unique index instead.
        if (uniqueWithDuplicateNulls)
                {
          unique = true;
        }
      }
    }

    // All indexes are unique because they contain the RowLocation.
    // The number of uniqueness columns must include the RowLocation
    // if the user did not specify a unique index.
    indexProperties.put("nUniqueColumns",
          Integer.toString(unique ? baseColumnPositions.length :
                        baseColumnPositions.length + 1)
              );
    // By convention, the row location column is the last column
    indexProperties.put("rowLocationColumn",
              Integer.toString(baseColumnPositions.length));

    // For now, all columns are key fields, including the RowLocation
    indexProperties.put("nKeyFields",
              Integer.toString(baseColumnPositions.length + 1));

    // For now, assume that all index columns are ordered columns
    if (! shareExisting)
    {
            if (dd.checkVersion(DataDictionary.DD_VERSION_DERBY_10_4, null))
            {
                indexRowGenerator = new IndexRowGenerator(
                                            indexType,
                                            unique,
                                            uniqueWithDuplicateNulls,
                                            baseColumnPositions,
                                            isAscending,
                                            baseColumnPositions.length);
      }
      else
            {
        indexRowGenerator = new IndexRowGenerator(
                                            indexType,
                                            unique,
                                            baseColumnPositions,
                                            isAscending,
                                            baseColumnPositions.length);
      }
    }

    /* Now add the rows from the base table to the conglomerate.
     * We do this by scanning the base table and inserting the
     * rows into a sorter before inserting from the sorter
     * into the index.  This gives us better performance
     * and a more compact index.
     */

    rowSource = null;
    sortId = 0;
    boolean needToDropSort = false// set to true once the sorter is created

    /* bulkFetchSIze will be 16 (for now) unless
     * we are creating the table in which case it
     * will be 1.  Too hard to remove scan when
     * creating index on new table, so minimize
     * work where we can.
     */
    int bulkFetchSize = (forCreateTable) ? 1 : 16
    int numColumns = td.getNumberOfColumns();
    int approximateRowSize = 0;

    // Create the FormatableBitSet for mapping the partial to full base row
    FormatableBitSet bitSet = new FormatableBitSet(numColumns+1);
    for (int index = 0; index < baseColumnPositions.length; index++)
    {
      bitSet.set(baseColumnPositions[index]);
    }
    FormatableBitSet zeroBasedBitSet = RowUtil.shift(bitSet, 1);

    // Start by opening a full scan on the base table.
    scan = tc.openGroupFetchScan(
                            td.getHeapConglomerateId(),
              false,  // hold
              0// open base table read only
                            TransactionController.MODE_TABLE,
                            TransactionController.ISOLATION_SERIALIZABLE,
              zeroBasedBitSet,    // all fields as objects
              (DataValueDescriptor[]) null,  // startKeyValue
              0,    // not used when giving null start posn.
              null,  // qualifier
              (DataValueDescriptor[]) null,  // stopKeyValue
              0);    // not used when giving null stop posn.

    // Create an array to put base row template
    baseRows = new ExecRow[bulkFetchSize];
    indexRows = new ExecIndexRow[bulkFetchSize];
    compactBaseRows = new ExecRow[bulkFetchSize];

    try
    {
      // Create the array of base row template
      for (int i = 0; i < bulkFetchSize; i++)
      {
        // create a base row template
        baseRows[i] = activation.getExecutionFactory().getValueRow(maxBaseColumnPosition);

        // create an index row template
        indexRows[i] = indexRowGenerator.getIndexRowTemplate();

        // create a compact base row template
        compactBaseRows[i] = activation.getExecutionFactory().getValueRow(
                          baseColumnPositions.length);
      }

      indexTemplateRow = indexRows[0];

      // Fill the partial row with nulls of the correct type
      ColumnDescriptorList cdl = td.getColumnDescriptorList();
      int           cdlSize = cdl.size();
      for (int index = 0, numSet = 0; index < cdlSize; index++)
      {
        if (! zeroBasedBitSet.get(index))
        {
          continue;
        }
        numSet++;
        ColumnDescriptor cd = (ColumnDescriptor) cdl.elementAt(index);
        DataTypeDescriptor dts = cd.getType();


        for (int i = 0; i < bulkFetchSize; i++)
        {
          // Put the column in both the compact and sparse base rows
          baseRows[i].setColumn(index + 1,
                  dts.getNull());
          compactBaseRows[i].setColumn(numSet,
                  baseRows[i].getColumn(index + 1));
        }

        // Calculate the approximate row size for the index row
        approximateRowSize += dts.getTypeId().getApproximateLengthInBytes(dts);
      }

      // Get an array of RowLocation template
      RowLocation rl[] = new RowLocation[bulkFetchSize];
      for (int i = 0; i < bulkFetchSize; i++)
      {
        rl[i] = scan.newRowLocationTemplate();

        // Get an index row based on the base row
        indexRowGenerator.getIndexRow(compactBaseRows[i], rl[i], indexRows[i], bitSet);
      }

      /* now that we got indexTemplateRow, done for sharing index
       */
      if (shareExisting)
        return;

      /* For non-unique indexes, we order by all columns + the RID.
       * For unique indexes, we just order by the columns.
       * We create a unique index observer for unique indexes
       * so that we can catch duplicate key.
       * We create a basic sort observer for non-unique indexes
       * so that we can reuse the wrappers during an external
       * sort.
       */
      int             numColumnOrderings;
      SortObserver    sortObserver   = null;
            Properties      sortProperties = null;
      if (unique || uniqueWithDuplicateNulls)
      {
        // if the index is a constraint, use constraintname in
                // possible error message
        String indexOrConstraintName = indexName;
        if  (conglomerateUUID != null)
        {
          ConglomerateDescriptor cd =
                        dd.getConglomerateDescriptor(conglomerateUUID);
          if ((isConstraint) &&
                        (cd != null && cd.getUUID() != null && td != null))
          {
            ConstraintDescriptor conDesc =
                            dd.getConstraintDescriptor(td, cd.getUUID());
            indexOrConstraintName = conDesc.getConstraintName();
          }
        }

        if (unique)
        {
                    numColumnOrderings = baseColumnPositions.length;

          sortObserver =
                        new UniqueIndexSortObserver(
                                true,
                                isConstraint,
                                indexOrConstraintName,
                                indexTemplateRow,
                                true,
                                td.getName());
        }
        else
                {
                    // unique with duplicate nulls allowed.

          numColumnOrderings = baseColumnPositions.length + 1;

                    // tell transaction controller to use the unique with
                    // duplicate nulls sorter, when making createSort() call.
          sortProperties = new Properties();
          sortProperties.put(
                        AccessFactoryGlobals.IMPL_TYPE,
                        AccessFactoryGlobals.SORT_UNIQUEWITHDUPLICATENULLS_EXTERNAL);
          //use sort operator which treats nulls unequal
          sortObserver =
                        new UniqueWithDuplicateNullsIndexSortObserver(
                                true,
                                isConstraint,
                                indexOrConstraintName,
                                indexTemplateRow,
                                true,
                                td.getName());
        }
      }
      else
      {
        numColumnOrderings = baseColumnPositions.length + 1;
        sortObserver = new BasicSortObserver(true, false,
                           indexTemplateRow,
                           true);
      }

      ColumnOrdering[]  order = new ColumnOrdering[numColumnOrderings];
      for (int i=0; i < numColumnOrderings; i++)
      {
        order[i] =
                    new IndexColumnOrder(
                        i,
                        unique || i < numColumnOrderings - 1 ?
                            isAscending[i] : true);
      }

      // create the sorter
      sortId = tc.createSort((Properties)sortProperties,
          indexTemplateRow.getRowArrayClone(),
          order,
          sortObserver,
          false,      // not in order
          scan.getEstimatedRowCount(),
          approximateRowSize  // est row size, -1 means no idea 
          );

      needToDropSort = true;

      // Populate sorter and get the output of the sorter into a row
      // source.  The sorter has the indexed columns only and the columns
      // are in the correct order.
      rowSource = loadSorter(baseRows, indexRows, tc,
                   scan, sortId, rl);

      conglomId =
                tc.createAndLoadConglomerate(
          indexType,
          indexTemplateRow.getRowArray()// index row template
          order, //colums sort order
                    indexRowGenerator.getColumnCollationIds(
                        td.getColumnDescriptorList()),
          indexProperties,
          TransactionController.IS_DEFAULT, // not temporary
          rowSource,
          (long[]) null);
     
    }
    finally
    {

      /* close the table scan */
      if (scan != null)
        scan.close();

      /* close the sorter row source before throwing exception */
      if (rowSource != null)
        rowSource.closeRowSource();

      /*
      ** drop the sort so that intermediate external sort run can be
      ** removed from disk
      */
      if (needToDropSort)
         tc.dropSort(sortId);
    }

    ConglomerateController indexController =
      tc.openConglomerate(
                conglomId, false, 0, TransactionController.MODE_TABLE,
                TransactionController.ISOLATION_SERIALIZABLE);

    // Check to make sure that the conglomerate can be used as an index
    if ( ! indexController.isKeyed())
    {
      indexController.close();
      throw StandardException.newException(SQLState.LANG_NON_KEYED_INDEX, indexName,
                               indexType);
    }
    indexController.close();

    //
    // Create a conglomerate descriptor with the conglomId filled
    // in and add it--if we don't have one already.
    //
    if (!alreadyHaveConglomDescriptor)
    {
      ConglomerateDescriptor cgd =
        ddg.newConglomerateDescriptor(
          conglomId, indexName, true,
          indexRowGenerator, isConstraint,
          conglomerateUUID, td.getUUID(), sd.getUUID() );

      dd.addDescriptor(cgd, sd,
        DataDictionary.SYSCONGLOMERATES_CATALOG_NUM, false, tc);

      // add newly added conglomerate to the list of conglomerate
      // descriptors in the td.
      ConglomerateDescriptorList cdl = td.getConglomerateDescriptorList();
      cdl.add(cgd);

      /* Since we created a new conglomerate descriptor, load
       * its UUID into the corresponding field, to ensure that
       * it is properly set in the StatisticsDescriptor created
       * below.
       */
      conglomerateUUID = cgd.getUUID();
    }

    CardinalityCounter cCount = (CardinalityCounter)rowSource;

        long numRows = cCount.getRowCount();
        if (addStatistics(dd, indexRowGenerator, numRows))
    {
      long[] c = cCount.getCardinality();
      for (int i = 0; i < c.length; i++)
      {
        StatisticsDescriptor statDesc =
          new StatisticsDescriptor(dd,
            dd.getUUIDFactory().createUUID(),
            conglomerateUUID, td.getUUID(), "I",
            new StatisticsImpl(numRows, c[i]), i + 1);

        dd.addDescriptor(statDesc, null,
                 DataDictionary.SYSSTATISTICS_CATALOG_NUM,
                 true, tc);
      }
    }
  }

    /**
     * Determines if a statistics entry is to be added for the index.
     * <p>
     * As an optimization, it may be better to not write a statistics entry to
     * SYS.SYSSTATISTICS. If it isn't needed by Derby as part of query
     * optimization there is no reason to spend resources keeping the
     * statistics up to date.
     *
     * @param dd the data dictionary
     * @param irg the index row generator
     * @param numRows the number of rows in the index
     * @return {@code true} if statistics should be written to
     *      SYS.SYSSTATISTICS, {@code false} otherwise.
     * @throws StandardException if accessing the data dictionary fails
     */
    private boolean addStatistics(DataDictionary dd,
                                  IndexRowGenerator irg,
                                  long numRows)
            throws StandardException {
        boolean add = (numRows > 0);
        if (dd.checkVersion(DataDictionary.DD_VERSION_DERBY_10_9, null) &&
                // This horrible piece of code will hopefully go away soon!
               ((IndexStatisticsDaemonImpl)dd.getIndexStatsRefresher(false)).
                    skipDisposableStats) {
            if (add && irg.isUnique() && irg.numberOfOrderedColumns() == 1) {
                // Do not add statistics for single-column unique indexes.
                add = false;
            }
        }
        return add;
    }

  // CLASS METHODS
 
  ///////////////////////////////////////////////////////////////////////
  //
  //  GETTERs called by CreateConstraint
  //
  ///////////////////////////////////////////////////////////////////////
  ExecRow getIndexTemplateRow()
  {
    return indexTemplateRow;
  }

  /**
   * Get the conglomerate number for the conglomerate that was
   * created by this constant action.  Will return -1L if the
   * constant action has not yet been executed.  This is used
   * for updating conglomerate descriptors which share a
   * conglomerate that has been dropped, in which case those
   * "sharing" descriptors need to point to the newly-created
   * conglomerate (the newly-created conglomerate replaces
   * the dropped one).
   */
  long getCreatedConglomNumber()
  {
    if (SanityManager.DEBUG)
    {
      if (conglomId == -1L)
      {
        SanityManager.THROWASSERT(
          "Called getCreatedConglomNumber() on a CreateIndex" +
          "ConstantAction before the action was executed.");
      }
    }

    return conglomId;
  }

  /**
   * If the purpose of this constant action was to "replace" a
   * dropped physical conglomerate, then this method returns the
   * conglomerate number of the dropped conglomerate.  Otherwise
   * this method will end up returning -1.
   */
  long getReplacedConglomNumber()
  {
    return droppedConglomNum;
  }

  /**
   * Get the UUID for the conglomerate descriptor that was created
   * (or re-used) by this constant action.
   */
  UUID getCreatedUUID()
  {
    return conglomerateUUID;
  }

  /**
   * Scan the base conglomerate and insert the keys into a sorter,
   * returning a rowSource on the sorter.
   *
   * @return RowSource on the sorted index keys.
   *
   * @exception StandardException          thrown on error
   */
  private RowLocationRetRowSource loadSorter(ExecRow[] baseRows,
                               ExecIndexRow[] indexRows,
                               TransactionController tc,
                               GroupFetchScanController scan,
                               long sortId,
                               RowLocation rl[])
    throws StandardException
  {
    SortController    sorter;
    long        rowCount = 0;

    sorter = tc.openSort(sortId);

    try
    {
      // Step through all the rows in the base table
      // prepare an array or rows for bulk fetch
      int bulkFetchSize = baseRows.length;

      if (SanityManager.DEBUG)
      {
        SanityManager.ASSERT(bulkFetchSize == indexRows.length,
          "number of base rows and index rows does not match");
        SanityManager.ASSERT(bulkFetchSize == rl.length,
          "number of base rows and row locations does not match");
      }

      DataValueDescriptor[][] baseRowArray = new DataValueDescriptor[bulkFetchSize][];

      for (int i = 0; i < bulkFetchSize; i++)
        baseRowArray[i] = baseRows[i].getRowArray();

      // rl[i] and baseRowArray[i] and indexRows[i] are all tied up
      // beneath the surface.  Fetching the base row and row location
      // from the table scan will automagically set up the indexRow
      // fetchNextGroup will return how many rows are actually fetched.
      int bulkFetched = 0;

      while ((bulkFetched = scan.fetchNextGroup(baseRowArray, rl)) > 0)
      {
        for (int i = 0; i < bulkFetched; i++)
        {
          sorter.insert(indexRows[i].getRowArray());
          rowCount++;
        }
      }

      /*
      ** We've just done a full scan on the heap, so set the number
      ** of rows so the optimizer will have an accurate count.
      */
      scan.setEstimatedRowCount(rowCount);
    }
    finally
    {
      sorter.completedInserts();
    }

    return new CardinalityCounter(tc.openSortRowSource(sortId));
  }
}
TOP

Related Classes of org.apache.derby.impl.sql.execute.CreateIndexConstantAction

TOP
Copyright © 2018 www.massapi.com. All rights reserved.
All source code are property of their respective owners. Java is a trademark of Sun Microsystems, Inc and owned by ORACLE Inc. Contact coftware#gmail.com.