blob: 84001a97940e3c8f509608e90129983cd757b299 [file] [log] [blame]
/*
Derby - Class org.apache.derby.impl.sql.execute.WriteCursorConstantAction
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.io.IOException;
import java.io.ObjectInput;
import java.io.ObjectOutput;
import java.util.Properties;
import org.apache.derby.catalog.UUID;
import org.apache.derby.iapi.error.StandardException;
import org.apache.derby.iapi.services.io.ArrayUtil;
import org.apache.derby.iapi.services.io.Formatable;
import org.apache.derby.iapi.services.io.FormatableBitSet;
import org.apache.derby.shared.common.sanity.SanityManager;
import org.apache.derby.iapi.sql.Activation;
import org.apache.derby.iapi.sql.dictionary.IndexRowGenerator;
import org.apache.derby.iapi.sql.execute.ConstantAction;
import org.apache.derby.iapi.store.access.StaticCompiledOpenConglomInfo;
/**
* This abstract class describes compiled constants that are passed into
* Delete, Insert, and Update ResultSets.
*
* This class and its sub-classes are not really implementations
* of ConstantAction, since they are not executed.
*
* A better name for these classes would be 'Constants'.
* E.g. WriteCursorConstants, DeleteConstants.
*
* Ideally one day the split will occur.
*
*/
abstract class WriteCursorConstantAction implements ConstantAction, Formatable
{
/********************************************************
**
** This class implements Formatable. But it is NOT used
** across either major or minor releases. It is only
** written persistently in stored prepared statements,
** not in the replication stage. SO, IT IS OK TO CHANGE
** ITS read/writeExternal.
**
********************************************************/
long conglomId;
StaticCompiledOpenConglomInfo heapSCOCI;
IndexRowGenerator[] irgs;
long[] indexCIDS;
StaticCompiledOpenConglomInfo[] indexSCOCIs;
String[] indexNames;
boolean deferred;
private Properties targetProperties;
UUID targetUUID;
int lockMode;
private FKInfo[] fkInfo;
private TriggerInfo triggerInfo;
private FormatableBitSet baseRowReadList;
private int[] baseRowReadMap;
private int[] streamStorableHeapColIds;
boolean singleRowSource;
/** True if this is an action of a MERGE statement */
private boolean underMerge;
// CONSTRUCTORS
/**
* Public niladic constructor. Needed for Formatable interface to work.
*
*/
public WriteCursorConstantAction() {}
/**
* Make the ConstantAction for a DELETE, INSERT, or UPDATE statement.
*
* @param conglomId Conglomerate ID of heap.
* @param heapSCOCI StaticCompiledOpenConglomInfo for heap.
* @param irgs Index descriptors
* @param indexCIDS Conglomerate IDs of indices
* @param indexSCOCIs StaticCompiledOpenConglomInfos for indexes.
* @param indexNames Names of indices on this table for error reporting.
* @param deferred True means process as a deferred update
* @param targetProperties Properties on the target table
* @param targetUUID UUID of target table
* @param lockMode The lock mode to use on the target table
* @param fkInfo Structure containing foreign key info, if any (may be null)
* @param triggerInfo Structure containing trigger info, if any (may be null)
* @param baseRowReadMap BaseRowReadMap[heapColId]->ReadRowColumnId. (0 based)
* @param streamStorableHeapColIds Null for non rep. (0 based)
* @param singleRowSource Whether or not source is a single row source
* @param underMerge True if this action is under a MERGE statement
*/
public WriteCursorConstantAction(
long conglomId,
StaticCompiledOpenConglomInfo heapSCOCI,
IndexRowGenerator[] irgs,
long[] indexCIDS,
StaticCompiledOpenConglomInfo[] indexSCOCIs,
String[] indexNames,
boolean deferred,
Properties targetProperties,
UUID targetUUID,
int lockMode,
FKInfo[] fkInfo,
TriggerInfo triggerInfo,
FormatableBitSet baseRowReadList,
int[] baseRowReadMap,
int[] streamStorableHeapColIds,
boolean singleRowSource,
boolean underMerge
)
{
this.conglomId = conglomId;
this.heapSCOCI = heapSCOCI;
this.irgs = irgs;
this.indexSCOCIs = indexSCOCIs;
this.indexCIDS = indexCIDS;
this.indexSCOCIs = indexSCOCIs;
this.deferred = deferred;
this.targetProperties = targetProperties;
this.targetUUID = targetUUID;
this.lockMode = lockMode;
this.fkInfo = fkInfo;
this.triggerInfo = triggerInfo;
this.baseRowReadList = baseRowReadList;
this.baseRowReadMap = baseRowReadMap;
this.streamStorableHeapColIds = streamStorableHeapColIds;
this.singleRowSource = singleRowSource;
this.indexNames = indexNames;
this.underMerge = underMerge;
if (SanityManager.DEBUG)
{
if (fkInfo != null)
{
SanityManager.ASSERT(fkInfo.length != 0, "fkinfo array has no elements, if there are no foreign keys, then pass in null");
}
}
}
///////////////////////////////////////////////////////////////////
//
// ACCESSORS
//
///////////////////////////////////////////////////////////////////
/**
* Gets the foreign key information for this constant action.
* A full list of foreign keys was compiled into this constant
* action.
*
*
* @return the list of foreign keys to enforce for this action
*
*/
final FKInfo[] getFKInfo()
{
return fkInfo;
}
/**
* Basically, the same as getFKInfo but for triggers.
*
* @return the triggers that should be fired
*
*/
TriggerInfo getTriggerInfo()
{
return triggerInfo;
}
///////////////////////////////////////////////////////////////////
//
// INTERFACE METHODS
//
///////////////////////////////////////////////////////////////////
/**
* NOP routine. The work is done in InsertResultSet.
*
* @see ConstantAction#executeConstantAction
*
* @exception StandardException Thrown on failure
*/
public final void executeConstantAction( Activation activation )
throws StandardException { }
// Formatable methods
/**
* Read this object from a stream of stored objects.
*
* @param in read this.
*
* @exception IOException thrown on error
* @exception ClassNotFoundException thrown on error
*/
public void readExternal( ObjectInput in )
throws IOException, ClassNotFoundException
{
conglomId = in.readLong();
heapSCOCI = (StaticCompiledOpenConglomInfo) in.readObject();
irgs = new IndexRowGenerator[ArrayUtil.readArrayLength(in)];
ArrayUtil.readArrayItems(in, irgs);
indexCIDS = ArrayUtil.readLongArray(in);
indexSCOCIs = new StaticCompiledOpenConglomInfo[ArrayUtil.readArrayLength(in)];
ArrayUtil.readArrayItems(in, indexSCOCIs);
deferred = in.readBoolean();
targetProperties = (Properties) in.readObject();
targetUUID = (UUID) in.readObject();
lockMode = in.readInt();
fkInfo = new FKInfo[ArrayUtil.readArrayLength(in)];
ArrayUtil.readArrayItems(in, fkInfo);
triggerInfo = (TriggerInfo)in.readObject();
baseRowReadList = (FormatableBitSet)in.readObject();
baseRowReadMap = ArrayUtil.readIntArray(in);
streamStorableHeapColIds = ArrayUtil.readIntArray(in);
singleRowSource = in.readBoolean();
indexNames = ArrayUtil.readStringArray(in);
underMerge = in.readBoolean();
}
/**
* Write this object to a stream of stored objects.
*
* @param out write bytes here.
*
* @exception IOException thrown on error
*/
public void writeExternal( ObjectOutput out )
throws IOException
{
out.writeLong(conglomId);
out.writeObject(heapSCOCI);
ArrayUtil.writeArray(out, irgs);
ArrayUtil.writeLongArray(out, indexCIDS);
ArrayUtil.writeArray(out, indexSCOCIs);
out.writeBoolean(deferred);
out.writeObject(targetProperties);
out.writeObject(targetUUID);
out.writeInt(lockMode);
ArrayUtil.writeArray(out, fkInfo);
//
//Added for Xena.
out.writeObject(triggerInfo);
//
//Moved from super class for Xena.
out.writeObject(baseRowReadList);
//
//Added for Xena
ArrayUtil.writeIntArray(out,baseRowReadMap);
ArrayUtil.writeIntArray(out,streamStorableHeapColIds);
//Added for Buffy
out.writeBoolean(singleRowSource);
// Added for Mulan (Track Bug# 3322)
ArrayUtil.writeArray(out, indexNames);
out.writeBoolean( underMerge );
}
// ACCESSORS
/** Return true if this is an action of a MERGE statement */
public boolean underMerge() { return underMerge; }
/**
* Get the conglomerate id for the changed heap.
* @return the conglomerate id.
*/
public long getConglomerateId() { return conglomId; }
/**
* Get the targetProperties from the constant action.
*
* @return The targetProperties
*/
public Properties getTargetProperties()
{
return targetProperties;
}
/**
* The the value of the specified key, if it exists, from
* the targetProperties.
*
* @param key The key to search for
*
* @return The value for the specified key if it exists, otherwise null.
* (Return null if targetProperties is null.)
*/
public String getProperty(String key)
{
return (targetProperties == null) ? null : targetProperties.getProperty(key);
}
public FormatableBitSet getBaseRowReadList() { return baseRowReadList; }
public int[] getBaseRowReadMap() { return baseRowReadMap; }
public int[] getStreamStorableHeapColIds() { return streamStorableHeapColIds; }
/**
* get the index name given the conglomerate id of the index.
*
* @param indexCID conglomerate ID of the index.
*
* @return index name of given index.
*/
public String getIndexNameFromCID(long indexCID)
{
int size = indexCIDS.length;
if (indexNames == null)
{
return null;
}
for (int i = 0; i < size; i++)
{
if (indexCIDS[i] == indexCID)
return indexNames[i];
}
return null;
}
public String[] getIndexNames()
{
return indexNames;
}
}