All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.ibm.as400.access.AS400JDBCPreparedStatementImpl Maven / Gradle / Ivy

The newest version!
///////////////////////////////////////////////////////////////////////////////
//
// JTOpen (IBM Toolbox for Java - OSS version)
//
// Filename: AS400JDBCPreparedStatement.java
//
// The source code contained herein is licensed under the IBM Public License
// Version 1.0, which has been approved by the Open Source Initiative.
// Copyright (C) 1997-2006 International Business Machines Corporation and
// others. All rights reserved.
//
///////////////////////////////////////////////////////////////////////////////

package com.ibm.as400.access;

import java.io.IOException;
import java.io.OutputStream;
import java.io.UnsupportedEncodingException;
import java.io.InputStream;
import java.io.Reader;
import java.math.BigDecimal;
import java.net.URL;
import java.sql.Array;
import java.sql.BatchUpdateException;
import java.sql.Blob;
import java.sql.Clob;
import java.sql.DataTruncation;
import java.sql.Date;
import java.sql.SQLWarning;
/* ifdef JDBC40 */
 import java.sql.NClob;
/* endif */ 
import java.sql.ParameterMetaData;
import java.sql.PreparedStatement;
import java.sql.Ref;
import java.sql.ResultSet;
import java.sql.ResultSetMetaData;
/* ifdef JDBC40 */
 import java.sql.RowId;
/* endif */ 
import java.sql.SQLException;
/* ifdef JDBC42
import java.sql.SQLType;
import java.sql.JDBCType;

endif */
/* ifdef JDBC40 */
 import java.sql.SQLXML;
 import java.sql.SQLFeatureNotSupportedException;
/* endif */ 
import java.sql.Statement;
import java.sql.Time;
import java.sql.Timestamp;
import java.sql.Types;
import java.util.Calendar;
import java.util.Enumeration;
import java.util.Vector;

/**
 * 

* The AS400JDBCPreparedStatement class precompiles and stores an SQL statement. * This provides the ability to efficiently run the statement multiple times. In * addition, the statement may contain parameters. Use * Connection.prepareStatement() to create new PreparedStatement objects. * *

* When setting input parameter values, the caller must specify types that are * compatible with the defined SQL type of the input parameter. For example, if * the input parameter has SQL type INTEGER, then the caller must call setInt() * to set the IN parameter value. If arbitrary type conversions are required, * then use setObject() with a target SQL type. * *

* For method that sets parameters, the application should not modify the * parameter value until after the execute completes. Modifying a value between * the setXXXX method and the execute method may result in unpredictable * behavior. **/ // // Implementation notes: // // 1. See implementation note in AS400JDBCStatement.java about // "private protected" methods. // // @F2A // 2. We need to support ?=CALL statements. This is where the stored // procedure returns an INTEGER value, and we treat it as the // first parameter marker. The database and host server support // the return value, but not as a parameter marker. Consequently, // we have to fake it as the first parameter marker. If this appears, // in the SQL statement, we strip it off and maintain this separately. // Of course in that case we are always mapping the caller's parameter // indices to the database's indices by decrementing by 1 as needed. // // @G8c // 3. If there is a return value (ie ?=call xxxx) and the parameter // index is 1 then return data for the return value (always an Integer). // If not, decrement the parm index by one because internally the return // value doesn't count. If there is no return value the count is correct // so don't do anything in that case. Also, the database supports returning // only integers so the metadata will always be an SQLInteger. // public class AS400JDBCPreparedStatementImpl extends AS400JDBCPreparedStatement { static final String copyright2 = "Copyright (C) 1997-2006 International Business Machines Corporation and others."; private int descriptorHandle_; boolean executed_; // private protected private boolean outputParametersExpected_; int parameterCount_; // private protected int parameterInputCount_; // private protected //@array4 boolean batchExecute_; // private protected @G9A private boolean executingBatchedStatement_ = false; // Flag to prevent // clearParameters from // causing execute // exception @DAA private int[] parameterLengths_; private int[] parameterOffsets_; private boolean[] parameterNulls_; private boolean[] parameterDefaults_; // @EIA private boolean[] parameterUnassigned_; // @EIA private String[] parameterNames_; // @pda jdbc40 // @re-prep move to statement JDServerRow JDServerRow parameterRow_; // // private protected Vector batchParameterRows_; // private protected @G9A private int parameterTotalSize_; private int indicatorTotalSize_; // @array Used with array containing data // only. Is total size of all indicators // (including array element indicators) private int headerTotalSize_; // @array Used to calculate size of stream // header boolean[] parameterSet_; // private protected private boolean prepared_; private JDServerRow resultRow_; SQLInteger returnValueParameter_; // private protected @F2A JDSQLStatement sqlStatement_; // @G4c (used by callable statement) boolean useReturnValueParameter_; // private protected @F2A private int maxToLog_ = 10000; // Log value of parameter markers up to this // length // @H1A private boolean containsArrayParameter_ = false; /* @G7A */ private boolean preserveParameters_ = false; private boolean saveParameterValues_ = false; // save the parameters so that can be // reset in a Client Affinities environment private int containsLocator_ = LOCATOR_UNKNOWN; private static final int LOCATOR_UNKNOWN = -1; private static final int LOCATOR_NOT_FOUND = 0; private static final int LOCATOR_FOUND = 1; private static final short INDICATOR_NULL = -1; // @EIA private static final short INDICATOR_DEFAULT = -5; // @EIA private static final short INDICATOR_UNASSIGNED = -7; // @EIA private boolean isjvm16Synchronizer;// @dmy private static boolean isjvm16SynchronizerStatic;// @dmy static { // Changed 2/21/2011 to not use unless the JDBC.jvm16Synchronize property is // true. @C6A /* * Here is some information from service about this error. * * Yes, this trace code was added for a very ugly issue that showed up when * customers started moving to Java 6. While trying to debug it, we found * that the trace points ended up changing the behavior, so they were * altered to trace to a dummy stream so that it would workaround Sun's bug. * The CPS discussion item was 7LXN87. * * Here's the contents of our KB doc on the issue: * * Abstract A problem with the Sun HotSpot Server in the 1.6 JDK causes a * variety of errors. * * Problem Summary: A problem was introduced into the version 1.6 JDK (Java * Development Kit) and JRE (Java Runtime Environment) from Sun. The problem * was introduced somewhere between update number 7 and update 12, which can * cause a number of problems. Java version 1.6.0_7 works; however, version * 1.6.0_12 produces the errors. The problem is specific to the HotSpot * Server which is something like an optimizing compiler that is designed to * provide the best operating speed for long-running applications similar to * a Web server. The problem seems to always manifest itself by 'removing' * parameters that had been bound to a statement. However, it is not * possible to know that this has occurred without tracing the application. * The outward symptoms are exceptions which will vary depending on what * data is missing. The common errors that have been reported are as * follows: * * SQLException: Descriptor index not valid CPD4374 - Field HVR000n and * value N not compatible. <-- where N might a variety of different numbers * SQL0302 - Conversion error on host variable or parameter *N. <-- where n * might a variety of different numbers SQL0406 - Conversion error on * assignment to column N. <-- where N might a variety of different numbers * * Resolution: The problem has been reported to Sun; however, at this time, * no fix is available from them. We have found three ways to circumvent the * problem: * * 1. Do not use JDK 1.6. 2. Use JVM property -client (this turns off * performance code in Sun Hotspot). 3. Use JVM property * -XX:CompileCommand=exclude * ,com/ibm/as400/access/AS400JDBCPreparedStatement,commonExecuteBefore * (more selectively, turn off part of Hotspot). 4. Use the latest version * of jt400.jar (currently 6.6). Additional trace points that were added * while searching for the source of the problem appear to have changed the * Hotspot behavior. * * * * Update 2/24/2011. This was probably a problem with the buffer * synchonization. Before JTOpen 7.1, a flag was set to indicate that a * buffer was available. This flag did not utilize any synchronization. In * JTOpen 7.1, the buffer management code was restructure to used * synchronzation. * * A recreate for the original problem was found. It failed using the JTOpen * 6.4 jar. We then used a jar with the change the set the default * isjvm16SynchronizerStatic to false and set the default value of the * property to false. The problem did not occur with the jar file. */ // Temporary fix for jvm 1.6 memroy stomp issue. (remove //@dmy code when // jvm issue is resolved) // This fix will just trace a few extra traces to a dummy stream // if system property or jdbc property is set to false then extra trace is // not executed // null value for system property means not specified...so true by default String jvm16Synchronize = SystemProperties .getProperty(SystemProperties.JDBC_JVM16_SYNCHRONIZE); // @dmy isjvm16SynchronizerStatic = false; // @dmy //false by default @C6C if ((jvm16Synchronize != null) && (Boolean.valueOf(jvm16Synchronize.trim()).booleanValue() == true)) { try { // @dmy Class.forName("java.sql.SQLXML"); // @dmy isjvm16SynchronizerStatic = true; // @dmy } catch (Exception e) { // @dmy isjvm16SynchronizerStatic = false; // @dmy } } else { // @dmy isjvm16SynchronizerStatic = false; // @dmy } } // @C6C -- Changed to remove the dummy PrimWriter. The dummy PrintWriter uses // a // 16k buffer of storage. This causes storage problems when a lot of // statements are // cached. Instead we'll use the write(byte[]) method instead of the buffered // print writer // // @dmy private dummy outputstream OutputStream dummyOutputStream = new OutputStream() { int b1 = 0; public synchronized void write(int b) throws IOException { b1 = b; } }; /** * Constructs an AS400JDBCPreparedStatement object. * * @param connection * The connection to the system. * @param id * The id. * @param transactionManager * The transaction manager for the connection. * @param packageManager * The package manager for the connection. * @param blockCriteria * The block criteria. * @param blockSize * The block size (in KB). * @param prefetch * Indicates if prefetching data. * @param sqlStatement * The SQL statement. * @param outputParametersExpected * Indicates if output parameters are expected. * @param packageCriteria * The package criteria. * @param resultSetType * The result set type. * @param resultSetConcurrency * The result set concurrency. * @param resultSetHoldability * The result set holdability. * @param autoGeneratedKeys * The auto-generated keys requested * @exception SQLException * If the SQL statement contains a syntax error or an error * occurs. **/ AS400JDBCPreparedStatementImpl(AS400JDBCConnection connection, int id, JDTransactionManager transactionManager, JDPackageManager packageManager, String blockCriteria, int blockSize, boolean prefetch, JDSQLStatement sqlStatement, boolean outputParametersExpected, String packageCriteria, int resultSetType, int resultSetConcurrency, int resultSetHoldability, // @G4A int autoGeneratedKeys) // @G4A throws SQLException { super(connection, id, transactionManager, packageManager, blockCriteria, blockSize, prefetch, packageCriteria, resultSetType, resultSetConcurrency, resultSetHoldability, autoGeneratedKeys); // @dmy // Temporary fix for jvm 1.6 memroy stomp issue. (remove //@dmy code when // jvm issue is resolved) // This fix will just trace a few extra traces to a dummy stream // if system property or jdbc property is set to false then extra trace is // not executed // null value for system property means not specified...so true by default isjvm16Synchronizer = isjvm16SynchronizerStatic; if (connection_.getProperties().getBoolean(JDProperties.JVM16_SYNCHRONIZE)) // @dmy isjvm16Synchronizer = true; // @dmy@C6C batchExecute_ = false; // @G9A outputParametersExpected_ = outputParametersExpected; parameterCount_ = sqlStatement.countParameters(); parameterInputCount_ = 0; // @array4 calculate while we prepare parameterLengths_ = new int[parameterCount_]; parameterNulls_ = new boolean[parameterCount_]; parameterDefaults_ = new boolean[parameterCount_]; // @EIA parameterUnassigned_ = new boolean[parameterCount_]; // @EIA parameterOffsets_ = new int[parameterCount_]; parameterSet_ = new boolean[parameterCount_]; sqlStatement_ = sqlStatement; useReturnValueParameter_ = sqlStatement.hasReturnValueParameter(); // @F2A if (useReturnValueParameter_) // @F2A returnValueParameter_ = new SQLInteger(connection_.getVRM(), settings_); // @F2A // //@trunc3 if (JDTrace.isTraceOn()) { // @D1A @F2C JDTrace .logInformation(this, "isjvm16Synchronizer=" + isjvm16Synchronizer); // @C6A JDTrace.logInformation(this, "Preparing [" + sqlStatement_ + "]"); // @D1A if (useReturnValueParameter_) // @F2A JDTrace.logInformation(this, "Suppressing return value parameter (?=CALL)"); // @F2A } // @F2A // Do not allow statements to be immediately // executed. If we did not do this, then some // statements would get executed at prepare time. allowImmediate_ = false; // Prepare. prepared_ = true; // @L1A Added try catch block around commonPrepare() for JTOpen Bug #3605 // Statement not fully closed on error. // If an error occurs in the preparing of the statement, we need to close // any resources used by the PreparedStatement object. try { resultRow_ = commonPrepare(sqlStatement_); } catch (SQLException e) { try { close(); } catch (Exception e2) { // ignore any nested exceptions } throw e; } executed_ = false; clearParameters(); } // JDBC 2.0 /** * Adds the set of parameters to the current batch. * * @exception SQLException * If the statement is not open or an input parameter has not * been set. **/ public void addBatch() throws SQLException { try { internalLock.lock(); checkOpen(); Object[] parameters = new Object[parameterCount_]; for (int i = 0; i < parameterCount_; ++i) { // Statements with output or inout parameters are not allowed in the // batch. if (parameterRow_.isOutput(i + 1)) JDError.throwSQLException(this, JDError.EXC_PARAMETER_TYPE_INVALID); // If an input parameter is not set, we throw an exception. if (!parameterSet_[i]) JDError.throwSQLException(this, JDError.EXC_PARAMETER_COUNT_MISMATCH); // @KBA Need to check for locators even if the parameter is null // If we don't check for locators and the first row had locator fields // that are all null, the LOCATOR_FOUND // flag was not being set. This meant that we thought we could batch // when executeBatch() was called. We cannot // batch when locators are being used. SQLData sqlData = parameterRow_.getSQLData(i + 1); // @KBA // Save the parameter in the array. If it's null, just leave it null. if (!parameterNulls_[i]) { // @KBD SQLData sqlData = parameterRow_.getSQLData(i+1); // For default and unassigned extended indicator values, we use Byte // to contain the indicator flag if (parameterDefaults_[i]) // @EIA parameters[i] = Byte.valueOf("1"); // default //@EIA else if (parameterUnassigned_[i]) // @EIA parameters[i] = Byte.valueOf("2"); // unassigned //@EIA else // @EIA parameters[i] = sqlData.getBatchableObject(); // @KBD if(containsLocator_ == LOCATOR_UNKNOWN) // @KBD { // @KBD int sqlType = sqlData.getSQLType(); // @KBD if (sqlType == SQLData.CLOB_LOCATOR || // @KBD sqlType == SQLData.BLOB_LOCATOR || // @KBD sqlType == SQLData.DBCLOB_LOCATOR) // @KBD { // @KBD containsLocator_ = LOCATOR_FOUND; // @KBD } // @KBD } } // @KBA check to see if the parameter is a locator field if (containsLocator_ == LOCATOR_UNKNOWN) // @KBA { // @KBA int sqlType = sqlData.getSQLType(); // @KBA if (sqlType == SQLData.CLOB_LOCATOR || // @KBA sqlType == SQLData.BLOB_LOCATOR || // @KBA sqlType == SQLData.DBCLOB_LOCATOR || // @KBA //@pdc jdbc40 sqlType == SQLData.NCLOB_LOCATOR || //@pda jdbc40 sqlType == SQLData.XML_LOCATOR) // @xml3 { // @KBA containsLocator_ = LOCATOR_FOUND; // @KBA } // @KBA } // @KBA } if (containsLocator_ == LOCATOR_UNKNOWN) containsLocator_ = LOCATOR_NOT_FOUND; if (batch_ == null) batch_ = new Vector(); // @P0A if (JDTrace.isTraceOn()) JDTrace.logInformation(this, "addBatch()"); batch_.addElement(parameters); } finally { internalLock.unlock(); } } // JDBC 2.0 /** * Adds an SQL statement to the current batch of SQL statements. * *

* Do not use this form of addBatch() on a prepared statement. * * @param sql * The SQL statement to be added to the current batch. This can be * any SQL statement that does not return a result set. * @exception SQLException * This exception is always thrown. **/ public void addBatch(String sql) throws SQLException { JDError.throwSQLException(this, JDError.EXC_FUNCTION_SEQUENCE); // @B1C } // @BAA /** * Creates or changes the descriptor, which describes the parameter marker * format. * * @exception SQLException * If an error occurs. **/ private void changeDescriptor() throws SQLException { // If a parameter that is not an input parameter has // not been set, we still include it in the format. // // Note that we set the native type, length, scale and // precision of each parameter in the descriptor. // That means that if the user sets a new parameter // value that changes one of these, then the // descriptor needs to be changed. // DBSQLDescriptorDS request2 = null; // @P0C try { descriptorHandle_ = id_; // @P0CDBSQLDescriptorDS request2 = new DBSQLDescriptorDS // (DBSQLDescriptorDS.FUNCTIONID_CHANGE_DESCRIPTOR, // @P0C id_, 0, descriptorHandle_); request2 = DBDSPool.getDBSQLDescriptorDS( DBSQLDescriptorDS.FUNCTIONID_CHANGE_DESCRIPTOR, id_, 0, descriptorHandle_); // @P0C DBDataFormat parameterMarkerDataFormat; if (connection_.useExtendedFormats()) { // @540 We are going to continue using DBExtendedDataFormat for this // part. We use an empty string for the parameter name so we don't need // 128 bytes parameterMarkerDataFormat = new DBExtendedDataFormat(parameterCount_); } else parameterMarkerDataFormat = new DBOriginalDataFormat(parameterCount_); request2.setParameterMarkerDataFormat(parameterMarkerDataFormat); parameterMarkerDataFormat.setConsistencyToken(1); parameterMarkerDataFormat.setRecordSize(parameterTotalSize_); if (isjvm16Synchronizer) { try { dummyOutputStream .write(("!!!changeDescriptor: totalParameterLength_ = " + parameterTotalSize_) .getBytes()); // @dmy@C6C } catch (Exception e) { } } for (int i = 0; i < parameterCount_; ++i) { SQLData sqlData = parameterRow_.getSQLData(i + 1); parameterMarkerDataFormat.setFieldDescriptionLength(i); if (sqlData.getNativeType() == SQLData.NATIVE_ARRAY) // @array { // @array int arrayLen = ((SQLArray) sqlData).getArrayCount(); // @array if (arrayLen > 0) // @array parameterMarkerDataFormat.setFieldLength(i, parameterLengths_[i] / arrayLen); // @array else // @array parameterMarkerDataFormat.setFieldLength(i, parameterLengths_[i]); // @array } // @array else // @array { parameterMarkerDataFormat.setFieldLength(i, parameterLengths_[i]); } parameterMarkerDataFormat.setFieldCCSID(i, parameterRow_.getCCSID(i + 1)); parameterMarkerDataFormat.setFieldNameLength(i, 0); parameterMarkerDataFormat.setFieldNameCCSID(i, 0); parameterMarkerDataFormat.setFieldName(i, "", connection_.getConverter()); // @P0C // @array (arrays sent in as the element type and zda will know they are // arrays) if (sqlData.getNativeType() == SQLData.NATIVE_ARRAY) // @array { // @array parameterMarkerDataFormat.setFieldSQLType(i, (short) (((SQLArray) sqlData).getElementNativeType() | 0x0001)); // @array } // @array else { parameterMarkerDataFormat.setFieldSQLType(i, (short) (sqlData.getNativeType() | 0x0001)); } parameterMarkerDataFormat.setFieldScale(i, (short) sqlData.getScale()); parameterMarkerDataFormat.setFieldPrecision(i, (short) sqlData.getPrecision()); if (isjvm16Synchronizer) { try { dummyOutputStream.write(("!!!changeDescriptor: Parameter " + (i + 1) + " length = " + parameterLengths_[i]).getBytes()); // @C6C } catch (Exception e) { } } } connection_.send(request2, descriptorHandle_); if (JDTrace.isTraceOn()) JDTrace.logInformation(this, "Descriptor " + descriptorHandle_ + " created or changed"); } catch (DBDataStreamException e) { JDError.throwSQLException(this, JDError.EXC_INTERNAL, e); } finally { // @P0C if (isjvm16Synchronizer) { if (request2 != null) { try { dummyOutputStream .write(("!!!changeDescriptor.inUser_(false): request2-id=" + request2 .hashCode()).getBytes()); // @C6C } catch (Exception e) { } } } if (request2 != null) { request2.returnToPool(); request2 = null; } // @P0C } } /** * Releases the resources used by the current input parameter values. In * general, input parameter values remain in effect for repeated executions of * the prepared statement. Setting an input parameter value to a new value * automatically clears its previous value. * * @exception SQLException * If the statement is not open. **/ public void clearParameters() throws SQLException { try { internalLock.lock(); checkOpen(); for (int i = 0; i < parameterCount_; ++i) { // @E1D parameterLengths_[i] = 0; parameterNulls_[i] = false; parameterDefaults_[i] = false; // @EIA parameterUnassigned_[i] = false; // @EIA // @E1D parameterOffsets_[i] = 0; parameterSet_[i] = false; } // @E1D parameterTotalSize_ = 0; if (useReturnValueParameter_) // @F2A returnValueParameter_.set(0); // @F2A } finally { internalLock.unlock(); } } /** * Releases the prepared statement's resources immediately instead of waiting * for them to be automatically released. This closes the current result set. * * @exception SQLException * If an error occurs. **/ public void close() throws SQLException { try { internalLock.lock(); // If this is already closed, then just do nothing. // // The spec does not define what happens when a connection // is closed multiple times. The official word from the Sun // JDBC team is that "the driver's behavior in this case // is implementation defined. Applications that do this are // non-portable." if (isClosed()) return; // If a descriptor was created somewhere along // the lines, then delete it now. if (descriptorHandle_ != 0) { // @P0CDBSQLDescriptorDS request = new DBSQLDescriptorDS ( // @P0C DBSQLDescriptorDS.FUNCTIONID_DELETE_DESCRIPTOR, // @P0C id_, 0, descriptorHandle_); DBSQLDescriptorDS request = null; // @P0C try { // @P0C request = DBDSPool.getDBSQLDescriptorDS( DBSQLDescriptorDS.FUNCTIONID_DELETE_DESCRIPTOR, id_, 0, descriptorHandle_); // @P0C connection_.send(request, descriptorHandle_); } finally { // @P0C if (isjvm16Synchronizer) { try { if (request != null) { dummyOutputStream .write(("!!!close.inUser_(false): request-id=" + request .hashCode()).getBytes()); // @C6C } } catch (Exception e) { } } if (request != null) { request.returnToPool(); request = null; } // @P0C } descriptorHandle_ = 0; } super.close(); } finally { internalLock.unlock(); } } /** * Performs common operations needed after an execute. * * @param sqlStatement * The SQL statement. * @param reply * The execute reply. * @exception SQLException * If an error occurs. **/ void commonExecuteAfter(JDSQLStatement sqlStatement, DBReplyRequestedDS reply) // private // protected throws SQLException { super.commonExecuteAfter(sqlStatement, reply); if (prepared_) { DBData resultData = null; if (outputParametersExpected_) resultData = reply.getResultData(); // Store the output parameters, if needed. if ((outputParametersExpected_) && (resultData != null)) { parameterRow_.setServerData(resultData); parameterRow_.setRowIndex(0); } // Handle the return value parameter, if needed. @F2A try { // @F2A if (useReturnValueParameter_) // @F2A returnValueParameter_.set(reply.getSQLCA().getErrd(1)); // @F2A //@G3C } // @F2A catch (DBDataStreamException e) { // @F2A JDError.throwSQLException(this, JDError.EXC_INTERNAL, e); // @F2A } // @F2A } } /** * Performs common operations needed before an execute. * * @param sqlStatement * The SQL statement. * @param request * The execute request. * @exception SQLException * If an error occurs. **/ void commonExecuteBefore(JDSQLStatement sqlStatement, DBSQLRequestDS request) throws SQLException { super.commonExecuteBefore(sqlStatement, request); int requestLengthOffset = 0; /*@K3A*/ int initialRawOffset = 0; /*@K3A*/ int variableDataLength = 0; //@L9A disableRllCompression_ = false; //@L9A if (prepared_) { // Close the result set before executing again. closeResultSet(JDCursor.REUSE_YES); // Validate each parameters. If a parameter is not an // input parameter, then it is okay for it not to have been // set. However, if an input parameter was not set, // we throw an exception. boolean outputExpected_ = false; // @K2A We do not want to increment our // row index in commonExecuteAfter() if // there are no output parameters parameterInputCount_ = 0; // Initialize before incrementing below. @G1 for (int i = 0; i < parameterCount_; ++i) { // We don't need to validate the parameters if executing a batched // statement. // The parameter were validated during addBatch. // If we attempt to validate, this will fail when // clearParameters is called immediately before executeBatch(). // Problem reported via CPS 8KLGCZ August 2011 @DAA if (!executingBatchedStatement_) { if (!parameterSet_[i] && parameterRow_.isInput(i + 1)) { JDError.throwSQLException(this, JDError.EXC_PARAMETER_COUNT_MISMATCH); } } if (parameterRow_.isOutput(i + 1)) // @K2A outputExpected_ = true; // @K2A if (parameterRow_.isInput(i + 1)) // @array4 parameterInputCount_++; // @array4 } if (!outputExpected_) // @K2A outputParametersExpected_ = false; // @K2A // Create the descriptor if needed. This should only // be done once (on the first execute for the prepared // statement). // This needs to be done every time when there are array parameters. // This is because the size of the array might change each time // execute is called. @G7C if (((parameterCount_ > 0) && (descriptorHandle_ == 0)) || ((parameterCount_ > 0) && containsArrayParameter_)) { /* @G7A */ // Reset the row index when reusing the parameter row. @G7A parameterRow_.setRowIndex(-1); // Get the offset and length for each parameter. // We just use the information that came in the parameter // marker format from reply for the prepare. parameterTotalSize_ = 0; indicatorTotalSize_ = 0; // @array headerTotalSize_ = 2; // @array start with 2 since column count is 2 // bytes for (int i = 0; i < parameterCount_; ++i) { if (!parameterRow_.containsArray_ || parameterRow_.isInput(i + 1)) // @array4 { SQLData sqlData = parameterRow_.getSQLData(i + 1); // @array // boolean arrayIndicatorSet = false; //@array if (sqlData.getType() == java.sql.Types.ARRAY) // @array { // Set the flag so that we still enter this code when // the descriptor handle is not zero @G7A containsArrayParameter_ = true; int arrayLen = ((SQLArray) sqlData).getArrayCount(); // @array // @G2A int elementSize = parameterRow_.getArrayDataLength(i + 1); // @G2A if (parameterNulls_[i] || parameterDefaults_[i] || parameterUnassigned_[i]) // @array headerTotalSize_ += 4; // @array space for x9911ffff else headerTotalSize_ += 12; // @array (array column requires 12 // bytes in header x9911) //@array2 parameterLengths_[i] = elementSize * arrayLen; // @array 0, 1, or // more // datatype-length // blocks indicatorTotalSize_ += (arrayLen * 2);// @array@G2A } else { // non array value headerTotalSize_ += 8; // @array (assuming row has array. x9912 is // length 8) // Set non-array length, avoiding any unnecessary multiplication // @G2A parameterLengths_[i] = parameterRow_.getLength(i + 1); indicatorTotalSize_ += 2; } // @array set input (to host) array lengths of data // @array if null array or 0 length array, then data length is 0 parameterOffsets_[i] = parameterTotalSize_; parameterTotalSize_ += parameterLengths_[i]; } if (isjvm16Synchronizer) { try { dummyOutputStream.write(("!!!commonExecuteBefore: Parameter " + (i + 1) + " length = " + parameterLengths_[i]).getBytes()); // @C6C } catch (Exception e) { } } } /* for i loop */ if (isjvm16Synchronizer) { try { dummyOutputStream .write(("!!!commonExecuteBefore: totalParameterLength_ = " + parameterTotalSize_) .getBytes()); // @C6C } catch (Exception e) { } } changeDescriptor(); } // Add the parameter information to the execute request. try { request.setStatementType(sqlStatement.getNativeType()); // Set the descriptor handle. request.setParameterMarkerDescriptorHandle(descriptorHandle_); // If there are any parameters, then send the parameter // values with the execute data stream. Only the // input parameters are included here. if (parameterCount_ > 0) { // Save the values so that can be restored later if (saveParameterValues_) { if (parameterRow_ != null) { parameterRow_.saveValues(); } } // In building the parameter marker data, we may discover that the // // @BAA // descriptor needs to be changed. If so, we will need to change the // // @BAA // descriptor, then rebuild the parameter marker data based on that // // @BAA // change. This is implemented using a do-while, but it should never // // @BAA // have to loop more than once, since the second time through (in a // // @BAA // particular execute) every thing should be great. // @BAA boolean descriptorChangeNeeded = false; do { descriptorChangeNeeded = false; // Reset our flag every time through // the loop. // Allocate the space for the Extended Parameter Marker Data @G9A // This is the amount of space for all of the rows' data and // indicators @G9A DBData parameterMarkerData; int rowCount = batchExecute_ ? batchParameterRows_.size() : 1; // // Determine if compression on insert is to be used. // @K3A boolean useVariableFieldInsertCompression = false; if (batchExecute_) { if (rowCount > 0 && useVariableFieldInsertCompression()) { useVariableFieldInsertCompression = true; } } // @array create new x382f here if parms contain array if (parameterRow_.containsArray_) // @array { // @array parameterMarkerData = new DBVariableData(parameterInputCount_, 2, headerTotalSize_, indicatorTotalSize_, parameterTotalSize_); // @array // x382f // codepoint // //@array4 // Dont use variableFieldInsertCompression with arrays @K3A if (useVariableFieldInsertCompression) { useVariableFieldInsertCompression = false; } } // @array else if (connection_.useExtendedFormats()) { parameterMarkerData = new DBExtendedData(rowCount, parameterCount_, 2, parameterTotalSize_); } else { parameterMarkerData = new DBOriginalData(rowCount, parameterCount_, 2, parameterTotalSize_); // If extended formats are not used, then // variableFieldInsertCompression cannot be used @K3A if (useVariableFieldInsertCompression) { useVariableFieldInsertCompression = false; } } int parameterOffset = 0; /*Track the offset to the parameter within the row @K3A*/ int rowDataOffset = 0; /*Track the offset to the start of the row @K3A*/ for (int rowLoop = 0; rowLoop < rowCount; ++rowLoop) // @G9a { Object[] parameters = null; // @G9A if (batchExecute_) // @G9A { // @G9A // @CRS - Don't need to synch around this because we have been // called // by executeBatch() which is already inside the synch block. parameters = (Object[]) batchParameterRows_.get(rowLoop); // @G9A } // @G9A // If this is the first of multiple rows. Set the Parameter Marker // @G9A // Data code point and consistency token only once @G9A if (rowLoop == 0) // @G9A { // The call to setParameterMarkerData sets the calculated length of the parameter marker data // in the request. This value is stored in request.lockedLength_ and // request.currentOffset_ has been updated. if (useVariableFieldInsertCompression) { /*@K3A*/ // For variable field insert compression, we need to record where the // parameter length was stored so it can be updated. // This is done after processing the parameters by calling request.updateLength requestLengthOffset = request.setParameterMarkerDataReserve(parameterMarkerData); ((DBExtendedData) parameterMarkerData) .setParameterMarkerInputCompression(true); } else { request.setParameterMarkerData(parameterMarkerData); } parameterMarkerData.setConsistencyToken(1); // @G9M } // @G9A /* calculate the rowDataOffset. This calculation is different * the first time through the look. The calculation is different * if variableFieldInsertCompression is used. */ if (rowLoop == 0) { if (useVariableFieldInsertCompression) { // Save the initialRawOffset so that the size can // be calculated after all the parameter has been added initialRawOffset = ((DBExtendedData)parameterMarkerData).getRawOffset(); } // Get the initial row offset. rowDataOffset = parameterMarkerData.getRowDataOffset(rowLoop); // @G9C } else { if (useVariableFieldInsertCompression) { // calculate the row offset base on the parameters last added rowDataOffset += parameterOffset; } else { rowDataOffset = parameterMarkerData.getRowDataOffset(rowLoop); // @G9C } } parameterOffset = 0; for (int i = 0; i < parameterCount_; ++i) { // @G1 -- zero out the comm buffer if the parameter marker is // null. // If the buffer is not zero'ed out old data will be sent to // the system possibily messing up a future request. if ((batchExecute_ && (parameters != null) && (parameters[i] == null || parameters[i] instanceof Byte)) || // @G9A //@EIC (!batchExecute_ && (parameterNulls_[i] || parameterDefaults_[i] || parameterUnassigned_[i]))) // @B9C // @G9C // //@EIC { short indicatorValue = INDICATOR_NULL; // @EIA if (batchExecute_ && (parameters != null)) // @EIA { // @EIA if (parameters[i] == null) // @EIA indicatorValue = INDICATOR_NULL; // @EIA else if (((Byte) parameters[i]).byteValue() == 1) // @EIA indicatorValue = INDICATOR_DEFAULT; // @EIA else if (((Byte) parameters[i]).byteValue() == 2) // @EIA indicatorValue = INDICATOR_UNASSIGNED; // @EIA } // @EIA else // @EIA { // @EIA if (parameterNulls_[i]) // @EIA indicatorValue = INDICATOR_NULL; // @EIA else if (parameterDefaults_[i]) // @EIA indicatorValue = INDICATOR_DEFAULT; // @EIA else if (parameterUnassigned_[i]) // @EIA indicatorValue = INDICATOR_UNASSIGNED; // @EIA } // @EIA SQLData sqlData = parameterRow_.getSQLType(i + 1); // @array // @array Don't set indicator here for null array, since // setting header below will set it if (sqlData.getType() != java.sql.Types.ARRAY) parameterMarkerData .setIndicator(rowLoop, i, indicatorValue); // @G1a @G9C // @EIC // @array only zero-out data on non-arrays // If the whole array is null, then we do not even include // blank data in the stream since a null array has space for // values (just 0X9911ffff in header of 0X382f) if (sqlData.getType() != java.sql.Types.ARRAY) // @array { byte[] parameterData = parameterMarkerData.getRawBytes(); // @G1a if (useVariableFieldInsertCompression) { /*@K3A*/ if (sqlData instanceof SQLVariableCompressible) { // If a variable compressible field, initialize the // size to zero for null data parameterData[rowDataOffset + parameterOffset]=0; parameterOffset++; parameterData[rowDataOffset + parameterOffset]=0; parameterOffset++; variableDataLength+=2; //@L9A } else { // initialize the value to zero and adjust the parameter offset int parameterDataOffset = rowDataOffset + parameterOffset; int parameterDataLength = parameterLengths_[i] + parameterDataOffset; for (int z = parameterDataOffset; z < parameterDataLength; parameterData[z++] = 0x00) { } parameterOffset += parameterLengths_[i]; } } else { int parameterDataOffset = rowDataOffset + parameterOffsets_[i]; // @G1a int parameterDataLength = parameterLengths_[i] + parameterDataOffset; for (int z = parameterDataOffset; z < parameterDataLength; parameterData[z++] = 0x00) { } } } // @array If the row contains an array, then we must also set // the columnInfo in stream header if (parameterRow_.containsArray_ && parameterRow_.isInput(i + 1)) // @array //@array4 { // @array int arrayLen = -1; // @array int elementType = -1; // @array int size = -1; // @array if (sqlData.getType() == java.sql.Types.ARRAY) // @array { // @array arrayLen = ((SQLArray) sqlData).getArrayCount(); // @array elementType = ((SQLArray) sqlData).getElementNativeType(); // @array size = parameterRow_.getArrayDataLength(i + 1); // @array@G2C } else { size = parameterRow_.getLength(i + 1); /* @X5A */ } // @array ((DBVariableData) parameterMarkerData).setHeaderColumnInfo( i, (short) sqlData.getNativeType(), (short) indicatorValue, (short) elementType, size, (short) arrayLen); // @array } // @array } else { /* Data is not null */ SQLData sqlData = parameterRow_.getSQLType(i + 1); // @array if (!parameterRow_.containsArray_ || parameterRow_.isInput(i + 1)) // @array4 { // Setting array null value here for elements inside of // array) if (sqlData.getType() == java.sql.Types.ARRAY) // @array { // @array // iterate through elements and set null indicators. Array // as a whole null is not set here (see above) for (int e = 0; e < ((SQLArray) sqlData).getArrayCount(); e++) // @array { // @array if (((SQLArray) sqlData).isElementNull(e)) // @array parameterMarkerData.setIndicator(0, i, -1); // @array else // @array parameterMarkerData.setIndicator(0, i, 0); // @array } // @array } else { parameterMarkerData.setIndicator(rowLoop, i, (short) 0); // @G9C } } ConvTable ccsidConverter = connection_ .getConverter(parameterRow_.getCCSID(i + 1)); // @P0C // Convert the data to bytes into the parameter marker data. // // @BAA // If there is an exception here, it means that there were // // @BAA // not enough bytes in the descriptor for the conversion. // // @BAA // If so, we get the correct length via getPrecision() // @BAA // (assume the SQLData implementation has updated its own // // @BAA // precision as needed). // @BAA int correctLength = -1; // @BAA // Put each row's values back into the SQL data type for their // respective columns. if (batchExecute_ && parameters != null) { // @CRS If the type is a locator, we pass -1 here so the // locator will know // not to reset its length, because the length wasn't saved // with the // parameter when addBatch() was called, but since we reuse // the SQLData // objects, it's still saved off inside the SQLLocator. setValue(i + 1, parameters[i], null, -1); } // SQLData sqlData = parameterRow_.getSQLType(i+1); // @BAC // @P0C @G9C //@array move above try { if (!parameterRow_.containsArray_ || parameterRow_.isInput(i + 1)) // @array4 (if array // then only send input // parm data) { // @CRS - This is the only place convertToRawBytes is ever // called. // @K3A compress the data if needed using variable field compression int thisParameterOffset = 0; /*@X4A*/ if (useVariableFieldInsertCompression) { if (sqlData instanceof SQLVariableCompressible) { // @K3A write the compressed bytes for the field. thisParameterOffset = rowDataOffset + parameterOffset; int written = ((SQLVariableCompressible) sqlData) .convertToCompressedBytes( parameterMarkerData.getRawBytes(), thisParameterOffset, ccsidConverter); variableDataLength += written; //@L9A parameterOffset += written; } else { thisParameterOffset = rowDataOffset + parameterOffset; sqlData.convertToRawBytes( parameterMarkerData.getRawBytes(), thisParameterOffset, ccsidConverter); // Increment by the length of the parameter parameterOffset += parameterLengths_[i]; } } else { thisParameterOffset = rowDataOffset + parameterOffsets_[i]; sqlData.convertToRawBytes( parameterMarkerData.getRawBytes(), thisParameterOffset, ccsidConverter); } // Need to check for truncation from convert // Blob truncation not caught at set time if set // using input stream @09A // if (ConvTable.isMixedCCSID(ccsidConverter.getCcsid()) || // sqlData instanceof SQLBlob ) // @trnc // this // is // not caught at // setX() time // @H2C boolean checkRawBytes = testDataTruncation(i + 1, sqlData); // @trnc if (checkRawBytes) { sqlData.validateRawTruncatedData(parameterMarkerData.getRawBytes(), thisParameterOffset, ccsidConverter); } } // @array If the row contains an array, then we must also // set the columnInfo in stream header if (parameterRow_.containsArray_ && parameterRow_.isInput(i + 1)) // @array //@array4 { // @array // Set the stream header info for each column in addition // to data in rawbytes above. int arrayLen = -1; // @array int elementType = -1; // @array int size; // @array if (sqlData.getType() == java.sql.Types.ARRAY) // @array { // @array size = parameterRow_.getArrayDataLength(i + 1); /* @G2A */ arrayLen = ((SQLArray) sqlData).getArrayCount(); // @array elementType = ((SQLArray) sqlData) .getElementNativeType(); // @array } else { // @array size = parameterRow_.getLength(i + 1); } ((DBVariableData) parameterMarkerData) .setHeaderColumnInfo(i, (short) sqlData.getNativeType(), (short) 0, (short) elementType, size, (short) arrayLen); // @array } // @array } catch (SQLException e) { if (e.getSQLState().trim().equals("HY000")) // AN INTERNAL // DRIVER ERROR { // Check error to see if it was thrown from another error if (parameterRow_.containsArray_) // @array always use // prepare/describe // lengths throw e; // @array if (e.getMessage().indexOf("Change Descriptor") != -1) { correctLength = sqlData.getPrecision(); // @BAA } else throw e; } else throw e; } // @BAA // If the length needed is larger than what was allocated in // // @BAA // the descriptor, then change the descriptor, and start // // @BAA // again. // @BAA if (correctLength >= 0) { descriptorChangeNeeded = true; // @BAA parameterLengths_[i] = correctLength; // @BAA parameterTotalSize_ = parameterOffsets_[i] + correctLength; // @BAA if ((i + 1) < parameterCount_) { for (int j = i + 1; j < parameterCount_; ++j) { parameterOffsets_[j] = parameterTotalSize_; // @BAA parameterTotalSize_ += parameterLengths_[j]; // @BAA } // @BAA } // @BAA } // @BAA } } if (descriptorChangeNeeded) changeDescriptor(); } /* for loop */ // For variable field compression update the length of the parameter block // by calculating the parameters length. @K3A if (useVariableFieldInsertCompression) { rowDataOffset += parameterOffset; int parametersLength = rowDataOffset - initialRawOffset; request.updateLength(requestLengthOffset, parametersLength); // Determine if enough space was saved by VFC to disable rll compression // We will disable if at least 80 % of the data was VFC data // This means that 20 % of the data can still be compressed using RLL // It is still possible then when compression is on, that the client // may decide not to send the compressed data if the compression // is large enough. //@L9A if ( variableDataLength > (80 * parametersLength / 100 )) { disableRllCompression_ = true; } } } while (descriptorChangeNeeded); request.setParameterMarkerBlockIndicator(0); } // If we are expecting output parameters // to be returned, then ask for them as result data. if (outputParametersExpected_) request .addOperationResultBitmap(DBSQLRequestDS.ORS_BITMAP_RESULT_DATA); } catch (DBDataStreamException e) { JDError.throwSQLException(this, JDError.EXC_INTERNAL, e); } } } /** * Should variable field compression be used for these parameters. * We will compress if at least 1/3 of the row is VARIABLE data. * * @throws SQLException If a database error occurs. */ /*@K3A*/ private boolean useVariableFieldInsertCompression() throws SQLException { if (useVariableFieldInsertCompression_) { int totalSize = 0; int compressibleSize = 0; for (int parameterNumber = 0; parameterNumber < parameterCount_; parameterNumber++) { int parameterSize = parameterRow_.getLength(parameterNumber+1); SQLData sqlData = parameterRow_.getSQLType(parameterNumber + 1); if (sqlData instanceof SQLVariableCompressible) { compressibleSize += parameterSize; } totalSize += parameterSize; } if (compressibleSize > (totalSize / 3)) { return true; } else { return false; } } else { return false; } } /** * Performs common operations needed after a prepare. * * @param sqlStatement * The SQL statement. * @param reply * The prepare reply. * @exception SQLException * If an error occurs. **/ void commonPrepareAfter(JDSQLStatement sqlStatement, DBReplyRequestedDS reply) throws SQLException { super.commonPrepareAfter(sqlStatement, reply); if (prepared_) { JDServerRow newParameterRow = new JDServerRow(connection_, id_, reply.getParameterMarkerFormat(), settings_); if (preserveParameters_ && (parameterRow_ != null)) { newParameterRow.restoreSavedParameters(parameterRow_); } parameterRow_ = newParameterRow; } } /** * Performs common operations needed before a prepare. * * @param sqlStatement * The SQL statement. * @param request * The prepare request. * @exception SQLException * If an error occurs. **/ void commonPrepareBefore(JDSQLStatement sqlStatement, DBSQLRequestDS request) throws SQLException { super.commonPrepareBefore(sqlStatement, request); if (prepared_) { request .addOperationResultBitmap(DBSQLRequestDS.ORS_BITMAP_PARAMETER_MARKER_FORMAT); } } /** * Performs common operations in leiu of a prepare. * * @param sqlStatement * The SQL statement. * @param statementIndex * The cached statement index. * @exception SQLException * If an error occurs. **/ void commonPrepareBypass(JDSQLStatement sqlStatement, int statementIndex) throws SQLException { super.commonPrepareBypass(sqlStatement, statementIndex); if (prepared_) { parameterRow_ = new JDServerRow(connection_, id_, packageManager_.getCachedParameterMarkerFormat(statementIndex), settings_); } } /** * Runs an SQL statement that may return multiple result sets. This closes the * current result set and clears warnings before executing the SQL statement * again. * *

* Under some situations, a single SQL statement may return multiple result * sets, an update count, or both. This might occur either when executing a * stored procedure that returns multiple result sets or when dynamically * executing an unknown SQL string. * *

* Use Statement.getMoreResults(), Statement.getResultSet(), and * Statement.getUpdateCount() to navigate through multiple result sets, an * update count, or both. * * @return true if a result set was returned; false if an update count was * returned or nothing was returned. * @exception SQLException * If the statement is not open, the query timeout limit is * exceeded, or an error occurs. **/ public boolean execute() throws SQLException { try { internalLock.lock(); checkOpen(); if (!prepared_) { resultRow_ = commonPrepare(sqlStatement_); prepared_ = true; } commonExecute(sqlStatement_, resultRow_); executed_ = true; return (resultSet_ != null); } finally { internalLock.unlock(); } } /** * Runs an SQL statement that may return multiple result sets. This closes the * current result set and clears warnings before executing a new SQL * statement. * *

* Do not use this form of execute() on a prepared statement. * * @param sql * The SQL statement. * @return true if a result set was returned, false if an update count was * returned or nothing was returned. * @exception SQLException * This exception is always thrown. **/ public boolean execute(String sql) throws SQLException { /* * @B1D // Call the super class execute() method. Note that this // results * in the prepare of a different statement, so the // we must mark ours and * not prepared. prepared_ = false; * * return super.execute (sql); */ JDError.throwSQLException(this, JDError.EXC_FUNCTION_SEQUENCE); // @B1A return false; // @B1A } // @GAA /** * Runs an SQL statement that may return multiple result sets and makes any * auto-generated keys available for retrieval using * Statement.getGeneratedKeys(). This closes the current result set and clears * warnings before executing the new SQL statement. * *

* Do not use this form of execute() on a prepared statement. * * @param sql * The SQL statement. * @param autoGeneratedKeys * Indicates whether auto-generated keys should be made available for * retrieval. Valid values are Statement.RETURN_GENERATED_KEYS and * Statement.NO_GENERATED_KEYS. * @return true if a result set was returned, false if an update count was * returned or nothing was returned. * @exception SQLException * This exception is always thrown. * @since Modification 5 **/ public boolean execute(String sql, int autoGeneratedKeys) throws SQLException { JDError.throwSQLException(this, JDError.EXC_FUNCTION_SEQUENCE); return false; // @B1A } public boolean execute(String sql, int [] columns) throws SQLException { JDError.throwSQLException(this, JDError.EXC_FUNCTION_SEQUENCE); return false; // @B1A } public boolean execute(String sql, String [] columnNames) throws SQLException { JDError.throwSQLException(this, JDError.EXC_FUNCTION_SEQUENCE); return false; // @B1A } /** * Runs the batch of SQL statements. Batch updates can be used to submit a set * of SQL statements together as a single unit. The SQL statements are run in * the order in which they were added to the batch. The batch is cleared after * the SQL statements are run. In addition, this closes the current result set * and clears warnings before executing the new SQL statement. * *

* When batch updates are run, autocommit should usually be turned off. This * allows the caller to decide whether or not to commit the transaction in the * event that an error occurs and some of the SQL statements in a batch fail * to run. * * @return An array of row counts for the SQL statements that are run. The * array contains one element for each statement in the batch of SQL * statements. The array is ordered according to the order in which * the SQL statements were added to the batch. * @exception SQLException * If the statement is not open, an SQL statement contains a * syntax error, the query timeout limit is exceeded, an SQL * statement returns a result set, or an error occurs. **/ public int[] executeBatch() throws SQLException { try { internalLock.lock(); checkOpen(); if (batch_ == null || batch_.size() == 0) return new int[0]; batchParameterRows_ = new Vector(); int batchSize = batch_.size(); int[] updateCounts = new int[batchSize]; int numSuccessful = 0; // Number of successfully executed statements in // the batch. boolean canBatch = true; // boolean notInsert = false; //@blksql try { // Only INSERTs can be batched, UPDATE statements must still be done one // at a time. // if(!(sqlStatement_.isInsert_)) //@blksql // { // canBatch = false; // notInsert = true; // } if (!(sqlStatement_.canBatch())) { canBatch = false; } // For sure we have a locator, so we can't batch it, // because the host server only reserves space for one locator handle. if (containsLocator_ == LOCATOR_FOUND) { canBatch = false; } // Set the batch execute flag so common execute knows to use the list // @G9A // of parameter rows. @G9A batchExecute_ = true; // @G9A rowsInserted_ = 0; // @G9A // Prepare the statement if it is not already done. if (!prepared_) { // @H7 Native type should ONLY be BLOCK_INSERT if the statement is of // type // @H7 "INSERT INTO MYTABLE ? ROWS VALUES (?,?)" with a ROWS VALUES // clause, // @H7 not just if we are going to send the values as a batch to the // system. // @H7 We determine whether the statement is of that form in // @H7 JDSQLStatement.java, not here. // @H7D sqlStatement_.setNativeType(JDSQLStatement.TYPE_BLOCK_INSERT); // // @G9A resultRow_ = commonPrepare(sqlStatement_); prepared_ = true; // See if the prepare returned a ResultSet. If so, error out now to // avoid // opening a cursor. Note some stored procedures won't return a // ResultSet // until the execute, so we check in both places. if (resultSet_ != null) { closeResultSet(JDCursor.REUSE_YES); JDError.throwSQLException(this, JDError.EXC_CURSOR_STATE_INVALID); } } // Execute. if (canBatch) { int maximumBlockedInputRows = connection_ .getMaximumBlockedInputRows(); Enumeration list = batch_.elements(); int count = 0; // @K1A Added support for allowing more than 32000 SQL // Statements to be batched and run int totalUpdateCount = 0; /* @A4A */ while (list.hasMoreElements()) { batchParameterRows_.add(list.nextElement()); count++; // @K1A if (count == maximumBlockedInputRows && list.hasMoreElements())// @K1A // Checks // if // 32000 // statements // have // been // added // to // the // batch, // if // so // execute // the // first // 32000, // then // continue // processing // the // batch { // @K1A if (JDTrace.isTraceOn()) JDTrace.logInformation(this, "Begin batching via server-side with " + batchParameterRows_.size() + " rows."); // @K1A executingBatchedStatement_ = true; /* @DAA */ commonExecute(sqlStatement_, resultRow_); // @K1A executingBatchedStatement_ = false; /* @DAA */ totalUpdateCount += updateCount_; /* @A4A */ batchParameterRows_.clear(); // @K1A if (resultSet_ != null) // @K1A { // @K1A closeResultSet(JDCursor.REUSE_YES); // @K1A JDError.throwSQLException(this, JDError.EXC_CURSOR_STATE_INVALID); // @K1A } // @K1A count = 0; // @K1A set the count for the number of statements in // the batch back to zero } // @K1A } if (JDTrace.isTraceOn()) JDTrace.logInformation(this, "Begin batching via server-side with " + batchParameterRows_.size() + " rows."); // // There is a quirk that if clearParameters is called after addBatch // but before executeBatch then // the commonExecute fails because it doesn't think the parameters are // set. // Set a flag that we are doing server side batching. // executingBatchedStatement_ = true; /* @DAA */ commonExecute(sqlStatement_, resultRow_); executingBatchedStatement_ = false; /* @DAA */ totalUpdateCount += updateCount_; /* @A4A */ batchParameterRows_.clear(); if (resultSet_ != null) { closeResultSet(JDCursor.REUSE_YES); JDError.throwSQLException(this, JDError.EXC_CURSOR_STATE_INVALID); } numSuccessful = batchSize; // The host server does not currently report the update counts for // each statement in // the batch. We use -2 here because that is the constant for // Statement.SUCCESS_NO_INFO // as of JDBC 3.0 and JDK 1.4. When we change to build against JDK 1.4 // instead of 1.3, // we can change this to use the actual constant. // However, if the total number of updated rows is the same as the // batch size then // we can set each of the update counts to 1. @A4A // Only set the count to one if the statement is an insert statement. // The logic in JDSQLStatement only allows in insert to be batched if // it is of the // form insert ... VALUES(?,?,?) ... Any other form will not be // batched int updateCount = -2; if (batchSize == totalUpdateCount && sqlStatement_.isInsert_) { updateCount = 1; } for (int i = 0; i < batchSize; ++i) { updateCounts[i] = updateCount; } } else { // We can't really batch because we are not an INSERT, we contain a // locator, or // there is some other reason. Enumeration list = batch_.elements(); if (JDTrace.isTraceOn()) JDTrace.logInformation(this, "Begin batching via client-side multiple executes."); while (list.hasMoreElements()) { batchParameterRows_.addElement(list.nextElement()); // Indicate we are batching to prevent clearParameter /*@DAA*/ executingBatchedStatement_ = true; commonExecute(sqlStatement_, resultRow_); executingBatchedStatement_ = false; /* @DAA */ batchParameterRows_.removeAllElements(); if (resultSet_ != null) { closeResultSet(JDCursor.REUSE_YES); JDError.throwSQLException(this, JDError.EXC_CURSOR_STATE_INVALID); } updateCounts[numSuccessful++] = rowsInserted_; } } executed_ = true; } catch (SQLException e) { // The specification says that if we get an error, // then the size of the update counts array should // reflect the number of statements that were // executed without error. int[] counts = null; if (canBatch) { // @CRS - We must be an INSERT...?? But we don't know which statement // in the batch // caused the exception, so all we can do is return an array of size 1 // with the // number of rows inserted/updated according to the host server. // After some investigation, the above is not true. See below: //@550 // If autocommit is on and we are running under *NONE, then // rowsInserted_ contains the number //@550 // of inserts that executed successfully before the error. // rowsInserted_ is set from the //@550 // the value in SQLERRD3. If autocommit is running under an isolation // level other than *NONE, //@550 // or autocommit is off, no rows are committed. Thus rowsInserted_ // will be zero. //@550 // Since we don't have any update counts for each statement, use // Statement.SUCCESS_NO_INFO //@550 // @550D counts = new int[] { rowsInserted_}; counts = new int[rowsInserted_]; // @550 batch update support for (int i = 0; i < counts.length; i++) // @550 counts[i] = Statement.SUCCESS_NO_INFO; // @550 } else { // @CRS - Since we haven't really been batching, we've been keeping // track of everything // per execute, so we can return more useful information. counts = new int[numSuccessful]; System.arraycopy(updateCounts, 0, counts, 0, numSuccessful); } BatchUpdateException batchUpdateException = new BatchUpdateException( e.getMessage(), e.getSQLState(), e.getErrorCode(), counts); batchUpdateException.initCause(e); throw batchUpdateException; } finally { batch_.removeAllElements(); batchExecute_ = false; // @K1A if (JDTrace.isTraceOn()) JDTrace.logInformation(this, "Done batching."); } return updateCounts; } finally { internalLock.unlock(); } } /** * Runs the SQL statement that returns a single result set. This closes the * current result set and clears warnings before executing the SQL statement * again. * * @return The result set that contains the data produced by the query. * @exception SQLException * If the statement is not open, no result set is returned by the * database, the query timeout limit is exceeded, an input * parameter has not been set, or an error occurs. **/ public ResultSet executeQuery() throws SQLException { try { internalLock.lock(); checkOpen(); if (!prepared_) { resultRow_ = commonPrepare(sqlStatement_); prepared_ = true; } commonExecute(sqlStatement_, resultRow_); executed_ = true; if (resultSet_ == null && ((behaviorOverride_ & 1) == 0)) { JDError.throwSQLException(this, JDError.EXC_CURSOR_STATE_INVALID); } return resultSet_; } finally { internalLock.unlock(); } } /** * Runs an SQL statement that returns a single result set. This closes the * current result set and clears warnings before executing a new SQL * statement. * *

* Do not use this form of executeQuery() on a prepared statement. * * @param sql * The SQL statement. * @return The result set that contains the data produced by the query. * @exception SQLException * This exception is always thrown. **/ public ResultSet executeQuery(String sql) throws SQLException { /* * @B1D // Call the super class execute() method. Note that this // results * in the prepare of a different statement, so the // we must mark ours and * not prepared. prepared_ = false; * * return super.executeQuery (sql); */ JDError.throwSQLException(this, JDError.EXC_FUNCTION_SEQUENCE); // @B1A return null; // @B1A } /** * Runs an SQL INSERT, UPDATE, or DELETE statement, or any SQL statement that * does not return a result set. This closes the current result set and clears * warnings before executing the SQL statement again. * * @return Either the row count for INSERT, UPDATE, or DELETE, or 0 for SQL * statements that return nothing. * @exception SQLException * If the statement is not open, the query timeout limit is * exceeded, the statement returns a result set, an input * parameter has not been set, or an error occurs. **/ public int executeUpdate() throws SQLException { try { internalLock.lock(); checkOpen(); // Prepare and execute. Check for a result set in both // places. It is best to catch it after the prepare (so // we don't open a cursor), but with some stored procedures, // we can't catch it until the execute. // Prepare the statement if it is not already done. if (!prepared_) { resultRow_ = commonPrepare(sqlStatement_); prepared_ = true; if (resultRow_ != null) JDError.throwSQLException(this, JDError.EXC_CURSOR_STATE_INVALID); } // Execute. commonExecute(sqlStatement_, resultRow_); executed_ = true; if (resultSet_ != null) { closeResultSet(JDCursor.REUSE_YES); JDError.throwSQLException(this, JDError.EXC_CURSOR_STATE_INVALID); } return updateCount_; } finally { internalLock.unlock(); } } /** * Runs an SQL INSERT, UPDATE, or DELETE statement, or any SQL statement that * does not return a result set. This closes the current result set and clears * warnings before executing a new SQL statement. * *

* Do not use this form of executeUpdate() on a prepared statement. * * @param sql * The SQL statement. * @return Either the row count for INSERT, UPDATE, or DELETE, or 0 for SQL * statements that return nothing. * @exception SQLException * This exception is always thrown. **/ public int executeUpdate(String sql) throws SQLException { /* * @B1D // Call the super class execute() method. Note that this // results * in the prepare of a different statement, so the // we must mark ours and * not prepared. prepared_ = false; * * return super.executeUpdate (sql); */ JDError.throwSQLException(this, JDError.EXC_FUNCTION_SEQUENCE); // @B1A return 0; // @B1A } // @GAA /** * Runs an SQL INSERT, UPDATE, or DELETE statement, or any SQL statement that * does not return a result set and makes any auto-generated keys available * for retrieval using Statement.getGeneratedKeys(). This closes the current * result set and clears warnings before executing the new SQL statement. * *

* Do not use this form of executeUpdate() on a prepared statement. * * @param sql * The SQL statement. * @return Either the row count for INSERT, UPDATE, or DELETE, or 0 for SQL * statements that return nothing. * @exception SQLException * This exception is always thrown. * @since Modification 5 **/ public int executeUpdate(String sql, int autoGeneratedKeys) throws SQLException { JDError.throwSQLException(this, JDError.EXC_FUNCTION_SEQUENCE); return 0; } public int executeUpdate(String sql, int[] columns) throws SQLException { JDError.throwSQLException(this, JDError.EXC_FUNCTION_SEQUENCE); return 0; } public int executeUpdate(String sql, String[] columnNames) throws SQLException { JDError.throwSQLException(this, JDError.EXC_FUNCTION_SEQUENCE); return 0; } // JDBC 2.0 /** * Returns the ResultSetMetaData object that describes the result set's * columns. Null is returned if the statement does not return a result set. In * the following example rsmd is null since the statement does not return a * result set. * *

   * PreparedStatement ps = connection
   *     .prepareStatement("INSERT INTO COLLECTION.TABLE VALUES(?)");
   * ResultSetMetaData rsmd = ps.getMetaData();
   * 
* * @return The metadata object, or null if the statement does not return a * result set. * @exception SQLException * If the statement is not open. **/ public ResultSetMetaData getMetaData() throws SQLException { try { internalLock.lock(); checkOpen(); if (resultRow_ == null) // @H6a return null; // @H6a ConvTable convTable = null; // @G6A DBExtendedColumnDescriptors extendedDescriptors = extendedColumnDescriptors_; // @G6A@P6C // If we have extendedDescriptors, send a ConvTable to convert them, else // pass null // @G6A if (extendedDescriptors != null) // @G6A { // @G6A convTable = connection_.getConverter(); // @G6A } // @G6A return new AS400JDBCResultSetMetaData(connection_.getCatalog(), resultSetConcurrency_, cursor_.getName(), resultRow_, extendedDescriptors, convTable, connection_); // @in1 // @G6A } finally { internalLock.unlock(); } } /* Validate the parameter index for this PreparedStatement object */ /* It has this name to avoid calling the validateParameterIndex */ /* method in AS400JDBCCallableStatement which does not account */ /* for stored procedure output parameters */ void validatePSParameterIndex(int param) throws SQLException { int parameterCount = getParameterCount(); if (param > parameterCount) { JDError.throwSQLException(this, JDError.EXC_DESCRIPTOR_INDEX_INVALID, param+">"+parameterCount); } if ( param < 1) { JDError.throwSQLException(this, JDError.EXC_DESCRIPTOR_INDEX_INVALID, param+"<1"); } } // @G4A // Return the class name of a parameter for ParameterMetaData support. String getParameterClassName(int param) throws SQLException { validatePSParameterIndex(param); try { internalLock.lock(); checkOpen(); if (useReturnValueParameter_) // @G8a { // @G8a if (param == 1) // @G8a return returnValueParameter_.getJavaClassName(); // @G8a else // @G8a param--; // @G8a } // @G8a return parameterRow_.getSQLData(param).getJavaClassName(); } finally { internalLock.unlock(); } } // @G4A // Return the parameter count for ParameterMetaData support. int getParameterCount() throws SQLException { try { internalLock.lock(); checkOpen(); if (useReturnValueParameter_) { return parameterCount_ + 1; } return parameterCount_; } finally { internalLock.unlock(); } } // @G4A JDBC 3.0 /** * Returns the number, types, and properties of a PreparedStatement object's * parameters. * * @return The ParameterMetaData object that describes this prepared statement * object. * @exception SQLException * If the statement is not open. * @since Modification 5 **/ public ParameterMetaData getParameterMetaData() throws SQLException { try { internalLock.lock(); checkOpen(); return (ParameterMetaData) (Object) new AS400JDBCParameterMetaData(this); } finally { internalLock.unlock(); } } // @G4A // Return the mode of a parameter for ParameterMetaData support. int getParameterMode(int param) throws SQLException { validatePSParameterIndex(param); try { internalLock.lock(); checkOpen(); if (useReturnValueParameter_) // @G8a { // @G8a if (param == 1) // @G8a return ParameterMetaData.parameterModeOut; // @G8a else // @G8a param--; // @G8a } // @G8a boolean input = parameterRow_.isInput(param); boolean output = parameterRow_.isOutput(param); if (input && output) { return ParameterMetaData.parameterModeInOut; } else if (input) { return ParameterMetaData.parameterModeIn; } else if (output) { return ParameterMetaData.parameterModeOut; } else return ParameterMetaData.parameterModeUnknown; } finally { internalLock.unlock(); } } // @G4A // Return the type of a parameter for ParameterMetaData support. int getParameterType(int param) throws SQLException { validatePSParameterIndex(param); try { internalLock.lock(); checkOpen(); if (useReturnValueParameter_) // @G8a { // @G8a if (param == 1) // @G8a return returnValueParameter_.getType(); // @G8a else // @G8a param--; // @G8a } // @G8a return parameterRow_.getSQLData(param).getType(); } finally { internalLock.unlock(); } } // // Return the ccsid of a parameter for ParameterMetaData support. // @V8A int getParameterCcsid(int param) throws SQLException { validatePSParameterIndex(param); try { internalLock.lock(); checkOpen(); if (useReturnValueParameter_) // @G8a { // @G8a if (param == 1) // @G8a return 0; // return value is a number which does not have a CCSID else // @G8a param--; // @G8a } // @G8a return parameterRow_.getCCSID(param); } finally { internalLock.unlock(); } } // @G4A // Return the type name of a parameter for ParameterMetaData support. String getParameterTypeName(int param) throws SQLException { validatePSParameterIndex(param); try { internalLock.lock(); checkOpen(); if (useReturnValueParameter_) // @G8a { // @G8a if (param == 1) // @G8a return returnValueParameter_.getTypeName(); // @G8a else // @G8a param--; // @G8a } // @G8a return parameterRow_.getSQLData(param).getTypeName(); } finally { internalLock.unlock(); } } // @G4A // Return the precision of a parameter for ParameterMetaData support. int getPrecision(int param) throws SQLException { validatePSParameterIndex(param); try { internalLock.lock(); checkOpen(); if (useReturnValueParameter_) // @G8a { // @G8a if (param == 1) // @G8a return returnValueParameter_.getPrecision(); // @G8a else // @G8a param--; // @G8a } // @G8a return parameterRow_.getSQLData(param).getPrecision(); } finally { internalLock.unlock(); } } // @G4A // Return the scale of a parameter for ParameterMetaData support. int getScale(int param) throws SQLException { validatePSParameterIndex(param); try { internalLock.lock(); checkOpen(); if (useReturnValueParameter_) // @G8a { // @G8a if (param == 1) // @G8a return returnValueParameter_.getScale(); // @G8a else // @G8a param--; // @G8a } // @G8a return parameterRow_.getSQLData(param).getScale(); } finally { internalLock.unlock(); } } // @G4A // Return whether a parameter is nullable for ParameterMetaData support. int isNullable(int param) throws SQLException { validatePSParameterIndex(param); try { internalLock.lock(); checkOpen(); if (useReturnValueParameter_) // @G8a { // @G8a if (param == 1) // @G8a return ResultSetMetaData.columnNoNulls; // @G8a else // @G8a param--; // @G8a } // @G8a return parameterRow_.isNullable(param); } finally { internalLock.unlock(); } } // @G4A // Return whether a parameter is signed for ParameterMetaData support. boolean isSigned(int param) throws SQLException { validatePSParameterIndex(param); try { internalLock.lock(); checkOpen(); if (useReturnValueParameter_) // @G8a { // @G8a if (param == 1) // @G8a return returnValueParameter_.isSigned(); // @G8a else // @G8a param--; // @G8a } // @G8a return parameterRow_.getSQLData(param).isSigned(); } finally { internalLock.unlock(); } } // JDBC 2.0 /** * Sets an input parameter to an Array value. DB2 for IBM i only supports * arrays in stored procedures. * * @param parameterIndex * The parameter index (1-based). * @param parameterValue * The parameter value. * @exception SQLException * Always thrown because DB2 for IBM i does not support arrays. **/ public void setArray(int parameterIndex, Array parameterValue) throws SQLException { // @array new support if (JDTrace.isTraceOn()) { JDTrace.logInformation(this, "setArray()"); if (parameterValue == null) JDTrace.logInformation(this, "parameter index: " + parameterIndex + " value: NULL"); else JDTrace.logInformation(this, "parameter index: " + parameterIndex + " value: Array type " + parameterValue.getBaseTypeName()); } if (!sqlStatement_.isProcedureCall()) // @array JDError.throwSQLException(this, JDError.EXC_PARAMETER_TYPE_INVALID); // @array setValue(parameterIndex, parameterValue, null, -1); } /** * Sets an input parameter to an ASCII stream value. The driver reads the data * from the stream as needed until no more bytes are available. The driver * converts this to an SQL VARCHAR value. * * @param parameterIndex * The parameter index (1-based). * @param parameterValue * The parameter value or null to set the value to SQL NULL. * @param length * The number of bytes in the stream. * @exception SQLException * If the statement is not open, the index is not valid, the * parameter is not an input parameter, the length is not valid, * the input stream does not contain all ASCII characters, or an * error occurs while reading the input stream. **/ public void setAsciiStream(int parameterIndex, InputStream parameterValue, int length) throws SQLException { if (JDTrace.isTraceOn()) { // @H1A JDTrace.logInformation(this, "setAsciiStream()"); // @H1A if (parameterValue == null) // @H1A JDTrace.logInformation(this, "parameter index: " + parameterIndex + " value: NULL"); // @H1A else JDTrace.logInformation(this, "parameter index: " + parameterIndex + " length: " + length); // @H1A } // @H1A // Validate the length parameter if (length < 0) JDError.throwSQLException(this, JDError.EXC_BUFFER_LENGTH_INVALID); // @J0A added the code from setValue in this method because streams and // readers are handled specially try { internalLock.lock(); checkOpen(); // Validate the parameter index. validatePSParameterIndex(parameterIndex); // Check if the parameter index refers to the return value parameter. // This is an OUT parameter, so sets are not allowed. If its not // parameter index 1, then decrement the parameter index, since we // are "faking" the return value parameter. if (useReturnValueParameter_) { if (parameterIndex == 1) JDError.throwSQLException(this, JDError.EXC_PARAMETER_TYPE_INVALID); else --parameterIndex; } // Check that the parameter is an input parameter. if (!parameterRow_.isInput(parameterIndex)) JDError.throwSQLException(this, JDError.EXC_PARAMETER_TYPE_INVALID); // Set the parameter data. If there is a type mismatch, // set() with throw an exception. SQLData sqlData = parameterRow_.getSQLType(parameterIndex); if (parameterValue != null) { try { // If the data is a locator, then set its handle. int sqlType = sqlData.getSQLType(); // @xml3 if (sqlType == SQLData.CLOB_LOCATOR || sqlType == SQLData.BLOB_LOCATOR || sqlType == SQLData.DBCLOB_LOCATOR || // @pdc jdbc40 sqlType == SQLData.NCLOB_LOCATOR || //@pda jdbc40 sqlType == SQLData.XML_LOCATOR) // @xml3 { SQLLocator sqlDataAsLocator = (SQLLocator) sqlData; sqlDataAsLocator.setHandle(parameterRow_ .getFieldLOBLocatorHandle(parameterIndex)); if (JDTrace.isTraceOn()) JDTrace.logInformation( this, "locator handle: " + parameterRow_.getFieldLOBLocatorHandle(parameterIndex)); sqlData.set(new ConvTableReader(parameterValue, 819, 0, LOB_BLOCK_SIZE), null, length); // @J0M hacked this to use the // scale parm for the length } else if (sqlType == SQLData.BOOLEAN) { // The stream to boolean conversion is not supported JDError.throwSQLException(this, JDError.EXC_DATA_TYPE_MISMATCH); } else { sqlData.set(JDUtilities.readerToString(new ConvTableReader( parameterValue, 819, 0, LOB_BLOCK_SIZE), length), null, -1); } } catch (UnsupportedEncodingException uee) { /* do nothing */ } testDataTruncation(parameterIndex, sqlData); } // Parameters can be null; you can call one of the set methods to null out // a // field of the database. parameterNulls_[parameterIndex - 1] = (parameterValue == null); parameterDefaults_[parameterIndex - 1] = false; // @EIA parameterUnassigned_[parameterIndex - 1] = false; // @EIA parameterSet_[parameterIndex - 1] = true; } finally { internalLock.unlock(); } // @J0M setValue (parameterIndex, // @J0M (parameterValue == null) ? null : JDUtilities.streamToString // (parameterValue, length, "ISO8859_1"), // @B2C // @J0M null, -1); //@P0C } /** * Sets an input parameter to a BigDecimal value. The driver converts this to * an SQL NUMERIC value. * * @param parameterIndex * The parameter index (1-based). * @param parameterValue * The parameter value or null to set the value to SQL NULL. * @exception SQLException * If the statement is not open, the index is not valid, or the * parameter is not an input parameter. **/ public void setBigDecimal(int parameterIndex, BigDecimal parameterValue) throws SQLException { // @B2D if (parameterValue == null) // @B2D JDError.throwSQLException (this, // JDError.EXC_PARAMETER_TYPE_INVALID); if (JDTrace.isTraceOn()) { // @H1A JDTrace.logInformation(this, "setBigDecimal()"); // @H1A if (parameterValue == null) // @H1A JDTrace.logInformation(this, "parameter index: " + parameterIndex + " value: NULL"); // @H1A else JDTrace.logInformation(this, "parameter index: " + parameterIndex + " value: " + parameterValue.toString()); // @H1A } // @H1A setValue(parameterIndex, parameterValue, null, -1); } /** * Sets an input parameter to a binary stream value. The driver reads the data * from the stream as needed until no more bytes are available. The driver * converts this to an SQL VARBINARY value. * *
* If a parameter is set using setBinaryStream, then the parameter must be * reset prior to the second execute of the PreparedStatement object. * * @param parameterIndex * The parameter index (1-based). * @param parameterValue * The parameter value or null to set the value to SQL NULL. * @param length * The number of bytes in the stream. * @exception SQLException * If the statement is not open, the index is not valid, the * parameter is not an input parameter, the length is not valid, * or an error occurs while reading the input stream. **/ public void setBinaryStream(int parameterIndex, InputStream parameterValue, int length) throws SQLException { if (JDTrace.isTraceOn()) { // @H1A JDTrace.logInformation(this, "setBinaryStream()"); // @H1A if (parameterValue == null) // @H1A JDTrace.logInformation(this, "parameter index: " + parameterIndex + " value: NULL"); // @H1A else JDTrace.logInformation(this, "parameter index: " + parameterIndex + " length: " + length); // @H1A } // @H1A // Validate the length parameter if (length < 0) JDError.throwSQLException(this, JDError.EXC_BUFFER_LENGTH_INVALID); // @J0A added the code from setValue in this method because streams and // readers are handled specially try { internalLock.lock(); checkOpen(); // Validate the parameter index. validatePSParameterIndex(parameterIndex); // Check if the parameter index refers to the return value parameter. // This is an OUT parameter, so sets are not allowed. If it's not // parameter index 1, then decrement the parameter index, since we // are "faking" the return value parameter. if (useReturnValueParameter_) { if (parameterIndex == 1) JDError.throwSQLException(this, JDError.EXC_PARAMETER_TYPE_INVALID); else --parameterIndex; } // Check that the parameter is an input parameter. if (!parameterRow_.isInput(parameterIndex)) JDError.throwSQLException(this, JDError.EXC_PARAMETER_TYPE_INVALID); // Set the parameter data. If there is a type mismatch, // set() will throw an exception. SQLData sqlData = parameterRow_.getSQLType(parameterIndex); if (parameterValue != null) { // If the data is a locator, then set its handle. if (sqlData instanceof SQLLocator) { SQLLocator sqlDataAsLocator = (SQLLocator) sqlData; sqlDataAsLocator.setHandle(parameterRow_ .getFieldLOBLocatorHandle(parameterIndex)); // Don't convert immediately to Bytes. This causes memory problems // with Large lobs @B3A // sqlData.set (JDUtilities.streamToBytes(parameterValue, length), // null, length);//@set1 allow setX one time and reuse execute() // without having to reset stream sqlData.set(parameterValue, null, length); // @J0M hacked this to use // the scale parm for the // length } else { sqlData.set(JDUtilities.streamToBytes(parameterValue, length), null, length); } testDataTruncation(parameterIndex, sqlData); } // Parameters can be null; you can call one of the set methods to null out // a // field of the database. parameterNulls_[parameterIndex - 1] = (parameterValue == null); parameterDefaults_[parameterIndex - 1] = false; // @EIA parameterUnassigned_[parameterIndex - 1] = false; // @EIA parameterSet_[parameterIndex - 1] = true; } finally { internalLock.unlock(); } // @KKC */ // @KKC setValue(parameterIndex, parameterValue, null, length); // @J0D setValue (parameterIndex, // @J0D (parameterValue == null) ? null : JDUtilities.streamToBytes // (parameterValue, length), // @B2C // @J0D null, -1); } // JDBC 2.0 /** * Sets an input parameter to a Blob value. The driver converts this to an SQL * BLOB value.
* If proxy support is in use, the Blob must be serializable. * * @param parameterIndex * The parameter index (1-based). * @param parameterValue * The parameter value or null to set the value to SQL NULL. * @exception SQLException * If the statement is not open, the index is not valid, the * parameter is not an input parameter, or the parameter is not * serializable (when proxy support is in use). **/ public void setBlob(int parameterIndex, Blob parameterValue) throws SQLException { // @B2D if (parameterValue == null) // @B2D JDError.throwSQLException (this, // JDError.EXC_PARAMETER_TYPE_INVALID); if (JDTrace.isTraceOn()) { // @H1A JDTrace.logInformation(this, "setBlob()"); // @H1A if (parameterValue == null) // @H1A JDTrace.logInformation(this, "parameter index: " + parameterIndex + " value: NULL"); // @H1A else JDTrace.logInformation(this, "parameter index: " + parameterIndex + " length: " + parameterValue.length()); // @H1A } // @H1A setValue(parameterIndex, parameterValue, null, -1); } /** * Sets an input parameter to a Java boolean value. The driver converts this * to an SQL SMALLINT value. * * @param parameterIndex * The parameter index (1-based). * @param parameterValue * The parameter value. * @exception SQLException * If the statement is not open, the index is not valid, or the * parameter is not an input parameter. **/ // // Implementation note: // // The spec defines this in terms of SQL BIT, but DB2 for IBM i // does not support that. // public void setBoolean(int parameterIndex, boolean parameterValue) throws SQLException { if (JDTrace.isTraceOn()) { // @H1A JDTrace.logInformation(this, "setBoolean()"); // @H1A JDTrace.logInformation(this, "parameter index: " + parameterIndex + " value: " + parameterValue); // @H1A } // @H1A setValue(parameterIndex, Short.valueOf((short) (parameterValue ? 1 : 0)), null, -1); } /** * Sets an input parameter to a Java byte value. The driver converts this to * an SQL SMALLINT value. * * @param parameterIndex * The parameter index (1-based). * @param parameterValue * The parameter value. * @exception SQLException * If the statement is not open, the index is not valid, or the * parameter is not an input parameter. **/ // // Implementation note: // // The spec defines this in terms of SQL TINYINT, but DB2 for IBM i // does not support that. // public void setByte(int parameterIndex, byte parameterValue) throws SQLException { if (JDTrace.isTraceOn()) { // @H1A JDTrace.logInformation(this, "setByte()"); // @H1A JDTrace.logInformation(this, "parameter index: " + parameterIndex + " value: " + parameterValue); // @H1A } // @H1A setValue(parameterIndex, Short.valueOf(parameterValue), null, -1); } /** * Sets an input parameter to a Java byte array value. The driver converts * this to an SQL VARBINARY value. * * @param parameterIndex * The parameter index (1-based). * @param parameterValue * The parameter value or null to set the value to SQL NULL. * @exception SQLException * If the statement is not open, the index is not valid, or the * parameter is not an input parameter. **/ public void setBytes(int parameterIndex, byte[] parameterValue) throws SQLException { // @B2D if (parameterValue == null) // @B2D JDError.throwSQLException (this, // JDError.EXC_PARAMETER_TYPE_INVALID); if (JDTrace.isTraceOn()) { // @H1A JDTrace.logInformation(this, "setBytes()"); // @H1A if (parameterValue == null) // @H1A JDTrace.logInformation(this, "parameter index: " + parameterIndex + " value: NULL"); // @H1A else if (parameterValue.length > maxToLog_) JDTrace.logInformation(this, "parameter index: " + parameterIndex + " length: " + parameterValue.length); // @H1A else JDTrace.logInformation(this, "parameter index: " + parameterIndex + " length: " + parameterValue.length + " value: " + Trace.toHexString(parameterValue)); // @H1A //@PDC } // @H1A setValue(parameterIndex, parameterValue, null, -1); } // JDBC 2.0 /** * Sets an input parameter to a character stream value. The driver reads the * data from the character stream as needed until no more characters are * available. The driver converts this to an SQL VARCHAR value. * * @param parameterIndex * The parameter index (1-based). * @param parameterValue * The parameter value or null to set the value to SQL NULL. * @param length * The number of characters to read from the reader. * @exception SQLException * If the statement is not open, the index is not valid, the * parameter is not an input parameter, the length is not valid, * or an error occurs while reading the character stream **/ public void setCharacterStream(int parameterIndex, Reader parameterValue, int length) throws SQLException { if (JDTrace.isTraceOn()) { // @H1A JDTrace.logInformation(this, "setCharacterStream()"); // @H1A if (parameterValue == null) // @H1A JDTrace.logInformation(this, "parameter index: " + parameterIndex + " value: NULL"); // @H1A else JDTrace.logInformation(this, "parameter index: " + parameterIndex + " length: " + length); // @H1A } // @H1A // Validate length parameter if (length < 0) JDError.throwSQLException(this, JDError.EXC_BUFFER_LENGTH_INVALID); // @J0A added the code from setValue in this method because streams and // readers are handled specially try { internalLock.lock(); checkOpen(); // Validate the parameter index. validatePSParameterIndex(parameterIndex); // Check if the parameter index refers to the return value parameter. // This is an OUT parameter, so sets are not allowed. If its not // parameter index 1, then decrement the parameter index, since we // are "faking" the return value parameter. if (useReturnValueParameter_) { if (parameterIndex == 1) JDError.throwSQLException(this, JDError.EXC_PARAMETER_TYPE_INVALID); else --parameterIndex; } // Check that the parameter is an input parameter. if (!parameterRow_.isInput(parameterIndex)) JDError.throwSQLException(this, JDError.EXC_PARAMETER_TYPE_INVALID); // Set the parameter data. If there is a type mismatch, // set() with throw an exception. SQLData sqlData = parameterRow_.getSQLType(parameterIndex); if (parameterValue != null) { // If the data is a locator, then set its handle. int sqlType = sqlData.getSQLType(); // @xml3 if (sqlType == SQLData.CLOB_LOCATOR || sqlType == SQLData.BLOB_LOCATOR || sqlType == SQLData.DBCLOB_LOCATOR || // @pdc jdbc40 sqlType == SQLData.NCLOB_LOCATOR || // @pda jdbc40 sqlType == SQLData.XML_LOCATOR) // @xml3 { SQLLocator sqlDataAsLocator = (SQLLocator) sqlData; sqlDataAsLocator.setHandle(parameterRow_ .getFieldLOBLocatorHandle(parameterIndex)); if (JDTrace.isTraceOn()) JDTrace.logInformation( this, "locator handle: " + parameterRow_.getFieldLOBLocatorHandle(parameterIndex)); // @pddsqlData.set(parameterValue, null, length); // @J0M hacked this // to use the scale parameter for the length sqlData.set(JDUtilities.readerToString(parameterValue, length), null, -1); // @pdc length is incorrect for double-byte chars. Use a // slower, but correct method, until we can create a real // ConvTableReader } else if (sqlType == SQLData.BOOLEAN) { // The stream to boolean conversion is not supported JDError.throwSQLException(this, JDError.EXC_DATA_TYPE_MISMATCH); } else { sqlData.set(JDUtilities.readerToString(parameterValue, length), null, -1); } testDataTruncation(parameterIndex, sqlData); } // Parameters can be null; you can call one of the set methods to null out // a // field of the database. parameterNulls_[parameterIndex - 1] = (parameterValue == null); parameterDefaults_[parameterIndex - 1] = false; // @EIA parameterUnassigned_[parameterIndex - 1] = false; // @EIA parameterSet_[parameterIndex - 1] = true; } finally { internalLock.unlock(); } // @J0D setValue (parameterIndex, // @J0D (parameterValue == null) ? null : JDUtilities.readerToString // (parameterValue, length), // @B2C // @J0D null, -1); //@P0C } // JDBC 2.0 /** * Sets an input parameter to a Clob value. The driver converts this to an SQL * CLOB value.
* If proxy support is in use, the Clob must be serializable. * * @param parameterIndex * The parameter index (1-based). * @param parameterValue * The parameter value or null to set the value to SQL NULL. * @exception SQLException * If the statement is not open, the index is not valid, the * parameter is not an input parameter, or the parameter is not * serializable (when proxy support is in use). **/ public void setClob(int parameterIndex, Clob parameterValue) throws SQLException { // @B2D if (parameterValue == null) // @B2D JDError.throwSQLException (this, // JDError.EXC_PARAMETER_TYPE_INVALID); if (JDTrace.isTraceOn()) { // @H1A JDTrace.logInformation(this, "setClob(int,Clob)"); // @H1A if (parameterValue == null) // @H1A JDTrace.logInformation(this, "parameter index: " + parameterIndex + " value: NULL"); // @H1A else if (parameterValue.length() < maxToLog_) // @H1A JDTrace .logInformation( this, "parameter index: " + parameterIndex + " class: "+parameterValue.getClass().getName() + " value: " + parameterValue.getSubString(1, (int) parameterValue.length())); // @H1A else JDTrace.logInformation(this, "parameter index: " + parameterIndex + " class: "+parameterValue.getClass().getName() + " length: " + parameterValue.length()); // @H1A } // @H1A setValue(parameterIndex, parameterValue, null, -1); } /** * Sets an input parameter to a java.sql.Date value using the default * calendar. The driver converts this to an SQL DATE value. * * @param parameterIndex * The parameter index (1-based). * @param parameterValue * The parameter value or null to set the value to SQL NULL. * @exception SQLException * If the statement is not open, the index is not valid, or the * parameter is not an input parameter. **/ public void setDate(int parameterIndex, Date parameterValue) throws SQLException { // @B2D if (parameterValue == null) // @B2D JDError.throwSQLException (this, // JDError.EXC_PARAMETER_TYPE_INVALID); if (JDTrace.isTraceOn()) { // @H1A JDTrace.logInformation(this, "setDate()"); // @H1A if (parameterValue == null) // @H1A JDTrace.logInformation(this, "parameter index: " + parameterIndex + " value: NULL"); // @H1A else JDTrace.logInformation(this, "parameter index: " + parameterIndex + " value: " + parameterValue.toString()); // @H1A } // @H1A setValue(parameterIndex, parameterValue, null, -1); // @P0C } // JDBC 2.0 /** * Sets an input parameter to a java.sql.Date value using a calendar other * than the default. The driver converts this to an SQL DATE value. * * @param parameterIndex * The parameter index (1-based). * @param parameterValue * The parameter value or null to set the value to SQL NULL. * @param calendar * The calendar. * @exception SQLException * If the statement is not open, the index is not valid, the * parameter is not an input parameter, or the calendar is null. **/ public void setDate(int parameterIndex, Date parameterValue, Calendar calendar) throws SQLException { // @B2D if (parameterValue == null) // @B2D JDError.throwSQLException (this, // JDError.EXC_PARAMETER_TYPE_INVALID); if (JDTrace.isTraceOn()) { // @H1A JDTrace.logInformation(this, "setDate()"); // @H1A if (parameterValue == null) // @H1A JDTrace.logInformation(this, "parameter index: " + parameterIndex + " value: NULL"); // @H1A else JDTrace.logInformation(this, "parameter index: " + parameterIndex + " value: " + parameterValue.toString()); // @H1A } // @H1A if (calendar == null) JDError.throwSQLException(this, JDError.EXC_ATTRIBUTE_VALUE_INVALID); setValue(parameterIndex, parameterValue, calendar, -1); } // @EIA 550 extended indicator defaults /** * Sets an input parameter to the default value * * @param parameterIndex * The parameter index (1-based). * @exception SQLException * If the statement is not open, the index is not valid, the * parameter is not an input parameter. **/ public void setDB2Default(int parameterIndex) throws SQLException { if (JDTrace.isTraceOn()) { JDTrace.logInformation(this, "setDB2Default()"); JDTrace.logInformation(this, "parameter index: " + parameterIndex); } setValueExtendedIndicator(parameterIndex, 1); // 1 is default } // @EIA 550 extended indicator defaults /** * Sets an input parameter to the default value. This is a the same as * setDB2Default. * * @param parameterIndex * The parameter index (1-based). * @exception SQLException * If the statement is not open, the index is not valid, the * parameter is not an input parameter. **/ public void setDBDefault(int parameterIndex) throws SQLException { setDB2Default(parameterIndex); } // @EIA 550 extended indicator defaults /** * Sets an input parameter to unassigned * * @param parameterIndex * The parameter index (1-based). * @exception SQLException * If the statement is not open, the index is not valid, the * parameter is not an input parameter. **/ public void setDB2Unassigned(int parameterIndex) throws SQLException { if (JDTrace.isTraceOn()) { JDTrace.logInformation(this, "setDB2Unassigned()"); JDTrace.logInformation(this, "parameter index: " + parameterIndex); } setValueExtendedIndicator(parameterIndex, 2); // 2 is unassigned } // @EIA 550 extended indicator defaults /** * Sets an input parameter to unassigned. This is a the same as * setDB2Unassigned. * * @param parameterIndex * The parameter index (1-based). * @exception SQLException * If the statement is not open, the index is not valid, the * parameter is not an input parameter. **/ public void setDBUnassigned(int parameterIndex) throws SQLException { setDB2Unassigned(parameterIndex); // 2 is unassigned } /** * Sets an input parameter to a Java double value. The driver converts this to * an SQL DOUBLE value. * * @param parameterIndex * The parameter index (1-based). * @param parameterValue * The parameter value. * @exception SQLException * If the statement is not open, the index is not valid or the * parameter is not an input parameter. **/ public void setDouble(int parameterIndex, double parameterValue) throws SQLException { if (JDTrace.isTraceOn()) { // @H1A JDTrace.logInformation(this, "setDouble()"); // @H1A JDTrace.logInformation(this, "parameter index: " + parameterIndex + " value: " + parameterValue); // @H1A } // @H1A setValue(parameterIndex, Double.valueOf(parameterValue), null, -1); } /** * Sets an input parameter to a Java float value. The driver converts this to * an SQL REAL value. * * @param parameterIndex * The parameter index (1-based). * @param parameterValue * The parameter value. * @exception SQLException * If the statement is not open, the index is not valid, or the * parameter is not an input parameter. **/ // // Note: The JDBC 1.22 specification states that this // method should set an SQL FLOAT value. However, // all tables map float to REAL. Otherwise, // nothing is symmetrical and certain INOUT // parameters do not work. // public void setFloat(int parameterIndex, float parameterValue) throws SQLException { if (JDTrace.isTraceOn()) { // @H1A JDTrace.logInformation(this, "setFloat()"); // @H1A JDTrace.logInformation(this, "parameter index: " + parameterIndex + " value: " + parameterValue); // @H1A } // @H1A setValue(parameterIndex, Float.valueOf(parameterValue), null, -1); } /** * Sets an input parameter to a Java int value. The driver converts this to an * SQL INTEGER value. * * @param parameterIndex * The parameter index (1-based). * @param parameterValue * The parameter value. * @exception SQLException * If the statement is not open, the index is not valid or the * parameter is not an input parameter. **/ public void setInt(int parameterIndex, int parameterValue) throws SQLException { if (JDTrace.isTraceOn()) { // @H1A JDTrace.logInformation(this, "setInt()"); // @H1A JDTrace.logInformation(this, "parameter index: " + parameterIndex + " value: " + parameterValue); // @H1A } // @H1A setValue(parameterIndex, Integer.valueOf(parameterValue), null, -1); } // @D0C /** * Sets an input parameter to a Java long value. If the connected system * supports SQL BIGINT data, the driver converts this to an SQL BIGINT value. * Otherwise, the driver converts this to an SQL INTEGER value. SQL BIGINT * data is supported on V4R5 and later. * * @param parameterIndex * The parameter index (1-based). * @param parameterValue * The parameter value. * @exception SQLException * If the statement is not open, the index is not valid, or the * parameter is not an input parameter. **/ // // Implementation note: // // The spec defines this in terms of SQL BIGINT, but DB2 for IBM i // does not support that until V4R5. // public void setLong(int parameterIndex, long parameterValue) throws SQLException { if (JDTrace.isTraceOn()) { // @H1A JDTrace.logInformation(this, "setLong()"); // @H1A JDTrace.logInformation(this, "parameter index: " + parameterIndex + " value: " + parameterValue); // @H1A } // @H1A setValue(parameterIndex, Long.valueOf(parameterValue), null, -1); // @D0C } /** * Sets an input parameter to SQL NULL. * * @param parameterIndex * The parameter index (1-based). * @param sqlType * The SQL type code defined in java.sql.Types. * @exception SQLException * If the statement is not open, the index is not valid, the * parameter is not an input parameter, or the SQL type is not * valid. **/ public void setNull(int parameterIndex, int sqlType) throws SQLException { // @BBD if (sqlType != parameterRow_.getSQLData (parameterIndex).getType ()) // @BBD JDError.throwSQLException (this, // JDError.EXC_PARAMETER_TYPE_INVALID); // @D8 ignore the type supplied by the user. We are checking it // only to rigidly follow the JDBC spec. Ignoring the type // will make us a friendlier driver. // // @D8d testSQLType(sqlType, parameterIndex); // @BBA if (JDTrace.isTraceOn()) { // @H1A JDTrace.logInformation(this, "setNull()"); // @H1A JDTrace.logInformation(this, "parameter index: " + parameterIndex + " value: NULL"); // @H1A } // @H1A setValue(parameterIndex, null, null, -1); } // @B4 - Added for JDK 2.0RC1 - typeName can be ignored, since it is not // relevant to IBM i. /** * Sets an input parameter to SQL NULL. * * @param parameterIndex * The parameter index (1-based). * @param sqlType * The SQL type code defined in java.sql.Types. * @param typeName * The fully-qualified name of an SQL structured type. This value * will be ignored. * @exception SQLException * If the statement is not open, the index is not valid, the * parameter is not an input parameter, or the SQL type is not * valid. **/ public void setNull(int parameterIndex, int sqlType, String typeName) throws SQLException { if (JDTrace.isTraceOn()) { // @H1A JDTrace.logInformation(this, "setNull()"); // @H1A JDTrace.logInformation(this, "parameter index: " + parameterIndex + " value: NULL"); // @H1A } // @H1A setNull(parameterIndex, sqlType); } /** * Sets an input parameter to an Object value. The driver converts this to a * value of an SQL type, depending on the type of the specified value. The * JDBC specification defines a standard mapping from Java types to SQL types. * In the cases where a SQL type is not supported by DB2 for IBM i, the next closest matching type * is used.
* If proxy support is in use, the Object must be serializable. * * @param parameterIndex * The parameter index (1-based). * @param parameterValue * The parameter value or null to set the value to SQL NULL. * @exception SQLException * If the statement is not open, the index is not valid, the * parameter is not an input parameter, the type of value is not * supported, or the parameter is not serializable (when proxy * support is in use). **/ public void setObject(int parameterIndex, Object parameterValue) throws SQLException { // @B2D if (parameterValue == null) // @B2D JDError.throwSQLException (this, // JDError.EXC_PARAMETER_TYPE_INVALID); if (JDTrace.isTraceOn()) { // @H1A JDTrace.logInformation(this, "setObject()"); // @H1A if (parameterValue == null) // @H1A JDTrace.logInformation(this, "parameter index: " + parameterIndex + " value: NULL"); // @H1A else JDTrace.logInformation(this, "parameter index: " + parameterIndex + " type: " + parameterValue.getClass().getName() + " toString():" + parameterValue.toString()); // @H1A } // @H1A setValue(parameterIndex, parameterValue, null, -1); // @P0C } /** * Sets an input parameter to an Object value. The driver converts this to a * value with the specified SQL type.
* If proxy support is in use, the Object must be serializable. * * @param parameterIndex * The parameter index (1-based). * @param parameterValue * The parameter value or null to set the value to SQL NULL. * @param sqlType * The SQL type code defined in java.sql.Types. * @exception SQLException * If the statement is not open, the index is not valid, the * parameter is not an input parameter, the SQL type is not * valid, or the parameter is not serializable (when proxy * support is in use). **/ public void setObject(int parameterIndex, Object parameterValue, int sqlType) throws SQLException { // @B2D if (parameterValue == null) // @B2D JDError.throwSQLException (this, // JDError.EXC_PARAMETER_TYPE_INVALID); // @BBD if (sqlType != parameterRow_.getSQLData (parameterIndex).getType ()) // @BBD JDError.throwSQLException (this, // JDError.EXC_PARAMETER_TYPE_INVALID); // @D8 ignore the type supplied by the user. We are checking it // only to rigidly follow the JDBC spec. Ignoring the type // will make us a friendlier driver. // // @D8d testSQLType(sqlType, parameterIndex); // @BBA if (JDTrace.isTraceOn()) { // @H1A JDTrace.logInformation(this, "setObject()"); // @H1A if (parameterValue == null) // @H1A JDTrace.logInformation(this, "parameter index: " + parameterIndex + " value: NULL"); // @H1A else JDTrace.logInformation(this, "parameter index: " + parameterIndex + " type: " + parameterValue.getClass().getName() + " toString():" + parameterValue.toString()); // @H1A } // @H1A setValue(parameterIndex, parameterValue, null, -1); // @P0C } /** * Sets an input parameter to an Object value. The driver converts this to a * value with the specified SQL type.
* If proxy support is in use, the Object must be serializable. * * @param parameterIndex * The parameter index (1-based). * @param parameterValue * The parameter value or null to set the value to SQL NULL. * @param sqlType * The SQL type code defined in java.sql.Types. * @param scale * The number of digits after the decimal if sqlType is DECIMAL or * NUMERIC. * @exception SQLException * If the statement is not open, the index is not valid, the * parameter is not an input parameter, the SQL type is not * valid, the scale is not valid, or the parameter is not * serializable (when proxy support is in use). **/ public void setObject(int parameterIndex, Object parameterValue, int sqlType, int scale) throws SQLException { // @B2D if (parameterValue == null) // @B2D JDError.throwSQLException (this, // JDError.EXC_PARAMETER_TYPE_INVALID); // @BBD if (sqlType != parameterRow_.getSQLData (parameterIndex).getType ()) // @BBD JDError.throwSQLException (this, // JDError.EXC_PARAMETER_TYPE_INVALID); // @D8 ignore the type supplied by the user. We are checking it // only to rigidly follow the JDBC spec. Ignoring the type // will make us a friendlier driver. // // @D8d testSQLType(sqlType, parameterIndex); // @BBA if (JDTrace.isTraceOn()) { // @H1A JDTrace.logInformation(this, "setObject()"); // @H1A if (parameterValue == null) // @H1A JDTrace.logInformation(this, "parameter index: " + parameterIndex + " value: NULL"); // @H1A else JDTrace.logInformation(this, "parameter index: " + parameterIndex + " type: " + parameterValue.getClass().getName() + " toString():" + parameterValue.toString()); // @H1A } // @H1A if (scale < 0) JDError.throwSQLException(this, JDError.EXC_SCALE_INVALID); /* ifdef JDBC40 */ if (parameterValue instanceof SQLXML) //@xmlspec setSQLXML(parameterIndex, (SQLXML)parameterValue); //@xmlspec else /* endif */ setValue(parameterIndex, parameterValue, null, scale); // @P0C } // JDBC 2.0 /** Sets an input parameter to a Ref value. DB2 for IBM i does not support structured types. @param parameterIndex The parameter index (1-based). @param parameterValue The parameter value. @exception SQLException Always thrown because DB2 for IBM i does not support structured types. **/ public void setRef(int parameterIndex, Ref parameterValue) throws SQLException { JDError.throwSQLException(this, JDError.EXC_PARAMETER_TYPE_INVALID); } /** * Sets an input parameter to a Java short value. The driver converts this to * an SQL SMALLINT value. * * @param parameterIndex * The parameter index (1-based). * @param parameterValue * The parameter value. * @exception SQLException * If the statement is not open, the index is not valid or the * parameter is not an input parameter. **/ public void setShort(int parameterIndex, short parameterValue) throws SQLException { if (JDTrace.isTraceOn()) { // @H1A JDTrace.logInformation(this, "setShort()"); // @H1A JDTrace.logInformation(this, "parameter index: " + parameterIndex + " value: " + parameterValue); // @H1A } // @H1A setValue(parameterIndex, Short.valueOf(parameterValue), null, -1); } /** * Sets an input parameter to a String value. The driver converts this to an * SQL VARCHAR value. * * @param parameterIndex * The parameter index (1-based). * @param parameterValue * The parameter value or null to set the value to SQL NULL. * @exception SQLException * If the statement is not open, the index is not valid, or the * parameter is not an input parameter. **/ public void setString(int parameterIndex, String parameterValue) throws SQLException { // @B2D if (parameterValue == null) // @B2D JDError.throwSQLException (this, // JDError.EXC_PARAMETER_TYPE_INVALID); if (JDTrace.isTraceOn()) { // @H1A JDTrace.logInformation(this, "setString()"); // @H1A if (parameterValue == null) // @H1A JDTrace.logInformation(this, "parameter index: " + parameterIndex + " value: NULL"); // @H1A else if (parameterValue.length() > maxToLog_) // @H1A JDTrace.logInformation(this, "parameter index: " + parameterIndex + " length: " + parameterValue.length()); // @H1A else JDTrace.logInformation(this, "parameter index: " + parameterIndex + " value: " + parameterValue); // @H1A } // @H1A // if(parameterIndex <= parameterCount_ && parameterIndex > 0) //@pdc // parameterValue = // AS400BidiTransform.convertDataToHostCCSID(parameterValue, connection_, // //Bidi-HCG // parameterRow_.getCCSID (parameterIndex)); //Bidi-HCG setValue(parameterIndex, parameterValue, null, -1); // @B7C @P0C } /** * Sets an input parameter to a java.sql.Time value using the default * calendar. The driver converts this to an SQL TIME value. * * @param parameterIndex * The parameter index (1-based). * @param parameterValue * The parameter value or null to set the value to SQL NULL. * @exception SQLException * If the statement is not open, the index is not valid, or the * parameter is not an input parameter. **/ public void setTime(int parameterIndex, Time parameterValue) throws SQLException { // @B2D if (parameterValue == null) // @B2D JDError.throwSQLException (this, // JDError.EXC_PARAMETER_TYPE_INVALID); if (JDTrace.isTraceOn()) { // @H1A JDTrace.logInformation(this, "setTime()"); // @H1A if (parameterValue == null) // @H1A JDTrace.logInformation(this, "parameter index: " + parameterIndex + " value: NULL"); // @H1A else JDTrace.logInformation(this, "parameter index: " + parameterIndex + " value: " + parameterValue.toString()); // @H1A } // @H1A setValue(parameterIndex, parameterValue, null, -1); // @P0C } // JDBC 2.0 /** * Sets an input parameter to a java.sql.Time value using a calendar other * than the default. The driver converts this to an SQL TIME value. * * @param parameterIndex * The parameter index (1-based). * @param parameterValue * The parameter value or null to set the value to SQL NULL. * @param calendar * The calendar. * @exception SQLException * If the statement is not open, the index is not valid, the * parameter is not an input parameter, or the calendar is null. **/ public void setTime(int parameterIndex, Time parameterValue, Calendar calendar) throws SQLException { // @B2D if (parameterValue == null) // @B2D JDError.throwSQLException (this, // JDError.EXC_PARAMETER_TYPE_INVALID); if (JDTrace.isTraceOn()) { // @H1A JDTrace.logInformation(this, "setTime()"); // @H1 if (parameterValue == null) // @H1A JDTrace.logInformation(this, "parameter index: " + parameterIndex + " value: NULL"); // @H1A else JDTrace.logInformation(this, "parameter index: " + parameterIndex + " value: " + parameterValue.toString()); // @H1A } // @H1A if (calendar == null) JDError.throwSQLException(this, JDError.EXC_ATTRIBUTE_VALUE_INVALID); setValue(parameterIndex, parameterValue, calendar, -1); } /** * Sets an input parameter to a java.sql.Timestamp value using the default * calendar. The driver converts this to an SQL TIMESTAMP value. * * @param parameterIndex * The parameter index (1-based). * @param parameterValue * The parameter value or null to set the value to SQL NULL. * @exception SQLException * If the statement is not open, the index is not valid, or the * parameter is not an input parameter. **/ public void setTimestamp(int parameterIndex, Timestamp parameterValue) throws SQLException { // @B2D if (parameterValue == null) // @B2D JDError.throwSQLException (this, // JDError.EXC_PARAMETER_TYPE_INVALID); if (JDTrace.isTraceOn()) { // @H1A JDTrace.logInformation(this, "setTimeStamp()"); // @H1A if (parameterValue == null) // @H1A JDTrace.logInformation(this, "parameter index: " + parameterIndex + " value: NULL"); // @H1A else JDTrace.logInformation(this, "parameter index: " + parameterIndex + " value: " + parameterValue.toString()); // @H1A } // @H1A setValue(parameterIndex, parameterValue, null, -1); // @P0C } // JDBC 2.0 /** * Sets an input parameter to a java.sql.Timestamp value using a calendar * other than the default. The driver converts this to an SQL TIMESTAMP value. * * @param parameterIndex * The parameter index (1-based). * @param parameterValue * The parameter value or null to set the value to SQL NULL. * @param calendar * The calendar. * @exception SQLException * If the statement is not open, the index is not valid, the * parameter is not an input parameter, or the calendar is null. **/ public void setTimestamp(int parameterIndex, Timestamp parameterValue, Calendar calendar) throws SQLException { // @B2D if (parameterValue == null) // @B2D JDError.throwSQLException (this, // JDError.EXC_PARAMETER_TYPE_INVALID); if (JDTrace.isTraceOn()) { // @H1A JDTrace.logInformation(this, "setTimeStamp()"); // @H1A if (parameterValue == null) // @H1A JDTrace.logInformation(this, "parameter index: " + parameterIndex + " value: NULL"); // @H1A else JDTrace.logInformation(this, "parameter index: " + parameterIndex + " value: " + parameterValue.toString()); // @H1A } // @H1A if (calendar == null) JDError.throwSQLException(this, JDError.EXC_ATTRIBUTE_VALUE_INVALID); setValue(parameterIndex, parameterValue, calendar, -1); } /** * Sets an input parameter to a Unicode stream value. The driver reads the * data from the stream as needed until no more bytes are available. The * driver converts this to an SQL VARCHAR value. *

* Note that the number of bytes in a Unicode stream can be computed as 2 * multiplied by the number of characters plus 2 bytes for the byte-order * mark. If an uneven number of bytes is specified, then Java will convert * this to an empty String. * * @param parameterIndex * The parameter index (1-based). * @param parameterValue * The parameter value or null to set the value to SQL NULL. * @param length * The number of bytes in the stream. * @exception SQLException * If the statement is not open, the index is not valid, the * parameter is not an input parameter, the length is not valid, * the input stream does not contain all Unicode characters, or * an error occurs while reading the input stream * @deprecated Use setCharacterStream(int, Reader, int) instead. * @see #setCharacterStream **/ public void setUnicodeStream(int parameterIndex, InputStream parameterValue, int length) throws SQLException { if (JDTrace.isTraceOn()) { // @H1A JDTrace.logInformation(this, "setUnicodeStream()"); // @H1A if (parameterValue == null) // @H1A JDTrace.logInformation(this, "parameter index: " + parameterIndex + " value: NULL"); // @H1A else JDTrace.logInformation(this, "parameter index: " + parameterIndex + " length: " + length); // @H1A } // @H1A // Validate the length parameter if (length < 0) JDError.throwSQLException(this, JDError.EXC_BUFFER_LENGTH_INVALID); // @J0A added the code from setValue in this method because streams and // readers are handled specially try { internalLock.lock(); checkOpen(); // Validate the parameter index. validatePSParameterIndex(parameterIndex); // Check if the parameter index refers to the return value parameter. // This is an OUT parameter, so sets are not allowed. If its not // parameter index 1, then decrement the parameter index, since we // are "faking" the return value parameter. if (useReturnValueParameter_) { if (parameterIndex == 1) JDError.throwSQLException(this, JDError.EXC_PARAMETER_TYPE_INVALID); else --parameterIndex; } // Check that the parameter is an input parameter. if (!parameterRow_.isInput(parameterIndex)) JDError.throwSQLException(this, JDError.EXC_PARAMETER_TYPE_INVALID); // Set the parameter data. If there is a type mismatch, // set() with throw an exception. SQLData sqlData = parameterRow_.getSQLType(parameterIndex); if (parameterValue != null) { try { // If the data is a locator, then set its handle. int sqlType = sqlData.getSQLType(); // @xml3 if (sqlType == SQLData.CLOB_LOCATOR || sqlType == SQLData.BLOB_LOCATOR || sqlType == SQLData.DBCLOB_LOCATOR || // @pdc jdbc40 sqlType == SQLData.NCLOB_LOCATOR || //@pda jdbc40 sqlType == SQLData.XML_LOCATOR) // @xml3 { SQLLocator sqlDataAsLocator = (SQLLocator) sqlData; sqlDataAsLocator.setHandle(parameterRow_ .getFieldLOBLocatorHandle(parameterIndex)); if (JDTrace.isTraceOn()) JDTrace.logInformation( this, "locator handle: " + parameterRow_.getFieldLOBLocatorHandle(parameterIndex)); sqlData.set(new ConvTableReader(parameterValue, 13488, 0, LOB_BLOCK_SIZE), null, length / 2); // @J0M hacked this to use // the scale parm for the // length } else if (sqlType == SQLData.BOOLEAN) { // The stream to boolean conversion is not supported JDError.throwSQLException(this, JDError.EXC_DATA_TYPE_MISMATCH); } else { sqlData.set(JDUtilities.readerToString(new ConvTableReader( parameterValue, 13488, 0, LOB_BLOCK_SIZE), length / 2), null, -1); } } catch (UnsupportedEncodingException uee) { /* do nothing */ } testDataTruncation(parameterIndex, sqlData); } // Parameters can be null; you can call one of the set methods to null out // a // field of the database. parameterNulls_[parameterIndex - 1] = (parameterValue == null); parameterDefaults_[parameterIndex - 1] = false; // @EIA parameterUnassigned_[parameterIndex - 1] = false; // @EIA parameterSet_[parameterIndex - 1] = true; } finally { internalLock.unlock(); } // @J0D setValue (parameterIndex, // @J0D (parameterValue == null) ? null : JDUtilities.streamToString // (parameterValue, length, "UnicodeBig"), // @B2C @B3C @H2C @H3C // @J0D null, -1); //@P0C } // @G4A JDBC 3.0 /** * Sets an input parameter to a URL value. The driver converts this to an SQL * DATALINK value. * * @param parameterIndex * The parameter index (1-based). * @param parameterValue * The parameter value or null to set the value to SQL NULL. * @exception SQLException * If the statement is not open, the index is not valid, or the * parameter is not an input parameter. * @since Modification 5 **/ public void setURL(int parameterIndex, URL parameterValue) throws SQLException { if (JDTrace.isTraceOn()) { // @H1A JDTrace.logInformation(this, "setURL()"); // @H1A if (parameterValue == null) // @H1A JDTrace.logInformation(this, "parameter index: " + parameterIndex + " value: NULL"); // @H1A else JDTrace.logInformation(this, "parameter index: " + parameterIndex + " value: " + parameterValue.toString()); // @H1A } // @H1A setValue(parameterIndex, parameterValue, null, -1); } /** * Sets an input parameter value for the specified index, and performs all * appropriate validation. * * @param parameterIndex * The parameter index (1-based). * @param parameterValue * The parameter value or null if the value is SQL NULL. * @param calendar * The calendar, or null if not applicable. * @param scale * The scale, or -1 if not applicable. * @exception SQLException * If the statement is not open, the index is not valid or the * parameter is not an input parameter. **/ void setValue(int parameterIndex, Object parameterValue, Calendar calendar, int scale) throws SQLException { try { internalLock.lock(); checkOpen(); validatePSParameterIndex(parameterIndex); // Check if the parameter index refers to the return value parameter. @F2A // This is an OUT parameter, so sets are not allowed. If its not @F2A // parameter index 1, then decrement the parameter index, since we @F2A // are "faking" the return value parameter. @F2A if (useReturnValueParameter_) { // @F2A if (parameterIndex == 1) // @F2A JDError.throwSQLException(this, JDError.EXC_PARAMETER_TYPE_INVALID);// @F2A else // @F2A --parameterIndex; // @F2A } // Check that the parameter is an input parameter. if (!parameterRow_.isInput(parameterIndex)) JDError.throwSQLException(this, JDError.EXC_PARAMETER_TYPE_INVALID); // Set the parameter data. If there is a type mismatch, // set() with throw an exception. SQLData sqlData = parameterRow_.getSQLType(parameterIndex); // @P0C if (parameterValue != null) { // @B6C // If the data is a locator, then set its handle. @B6A int sqlType = sqlData.getSQLType(); if((sqlType == SQLData.CLOB_LOCATOR || sqlType == SQLData.BLOB_LOCATOR || sqlType == SQLData.DBCLOB_LOCATOR || //@pdc jdbc40 sqlType == SQLData.NCLOB_LOCATOR || //@pda jdbc40 sqlType == SQLData.XML_LOCATOR)) // @xml3 { // @B6A SQLLocator sqlDataAsLocator = (SQLLocator) sqlData; // @B6A sqlDataAsLocator.setHandle(parameterRow_ .getFieldLOBLocatorHandle(parameterIndex)); // @B6A if (JDTrace.isTraceOn()) JDTrace.logInformation( this, "locator handle: " + parameterRow_.getFieldLOBLocatorHandle(parameterIndex)); } // @B6A sqlData.set(parameterValue, calendar, scale); testDataTruncation(parameterIndex, sqlData); // @B5C @G5move } // Parameters can be null; you can call one of the set methods to null out // a // field of the database. // @B6A parameterNulls_[parameterIndex - 1] = (parameterValue == null); parameterDefaults_[parameterIndex - 1] = false; // @EIA parameterUnassigned_[parameterIndex - 1] = false; // @EIA parameterSet_[parameterIndex - 1] = true; } finally { internalLock.unlock(); } } // @EIA new method /** * Sets an input parameter value for the specified index, and performs all * appropriate validation when the value is one of the valid Extended * Indicator values: default or unassigned. * * Note: this is the same type of method as setValue() above, but we have no * way to pass in the special values without hacking some sort of flag string * for the value, and that seemed to be a messy and slow way to do this. * * @param parameterIndex * The parameter index (1-based). * @param parameterValue * The parameter 1="default" or 2="unassigned". * @exception SQLException * If the statement is not open, the index is not valid or the * parameter is not an input parameter. **/ void setValueExtendedIndicator(int parameterIndex, int parameterValue) throws SQLException { try { internalLock.lock(); checkOpen(); validatePSParameterIndex(parameterIndex); // Check if the parameter index refers to the return value parameter. // This is an OUT parameter, so sets are not allowed. If its not // parameter index 1, then decrement the parameter index, since we // are "faking" the return value parameter. if (useReturnValueParameter_) { if (parameterIndex == 1) JDError.throwSQLException(this, JDError.EXC_PARAMETER_TYPE_INVALID); else --parameterIndex; } // Check that the parameter is an input parameter. if (!parameterRow_.isInput(parameterIndex)) JDError.throwSQLException(this, JDError.EXC_PARAMETER_TYPE_INVALID); parameterNulls_[parameterIndex - 1] = false; parameterDefaults_[parameterIndex - 1] = parameterValue == 1 ? true : false; parameterUnassigned_[parameterIndex - 1] = parameterValue == 2 ? true : false; parameterSet_[parameterIndex - 1] = true; } finally { internalLock.unlock(); } } /** * Tests if a DataTruncation occurred on the write of a piece of data and * throws a DataTruncation exception if so. The data truncation flag is also * taken into consideration for string data. The rules are: 1) If updating * database with numeric data and data truncated, throw exception 2) If * numeric data is part of a query and data truncated, post warning 3) If * string data and suppress truncation, return 4) If updating database with * string data and check truncation and data truncated, throw exception 5) If * string data is part of a query and check truncation and data truncated, * post warning * * @param index * The index (1-based). * @param data * The data that was written or null for SQL NULL. * @return True, if the caller should validate that the raw bytes are valid. **/ /*@X4C*/ private boolean testDataTruncation(int parameterIndex, SQLData data) throws SQLException // @trunc { // Moved logic to connection to be shared with AS400JdbcResultSet return connection_.testDataTruncation(this, null, parameterIndex, true, data, sqlStatement_); } // @BBA /** * Checks that an input SQL type is compatible with the actual parameter type. * * @param sqlType * The SQL type. * @param parameterIndex * The index (1-based). * @exception SQLException * If the SQL type is not compatible. **/ void testSQLType(int sqlType, int parameterIndex) throws SQLException { int parameterType = parameterRow_.getSQLType(parameterIndex).getType(); // @P0C if (sqlType != parameterType) { // If the only reason the types don't match is because one // is a CHAR and the other is a VARCHAR, then let this // slide. if (((sqlType == Types.CHAR) || (sqlType == Types.VARCHAR)) && ((parameterType == Types.CHAR) || (parameterType == Types.VARCHAR))) { // Do nothing! } else { JDError.throwSQLException(this, JDError.EXC_PARAMETER_TYPE_INVALID); } } } // @GKA // Returns the JDServerRow object associated with this statement. JDServerRow getResultRow() { return resultRow_; } // @PDA jdbc40 /** * Sets the designated parameter to the given java.sql.RowId object. The * driver converts this to a SQL ROWID value when it sends it * to the database * * @param parameterIndex The parameter index (1-based). * @param x the parameter value * @throws SQLException if a database access error occurs * */ /* ifdef JDBC40 */ public void setRowId(int parameterIndex, RowId x) throws SQLException { if(JDTrace.isTraceOn()) { JDTrace.logInformation (this, "setRowId()"); if(x == null) JDTrace.logInformation (this, "parameter index: " + parameterIndex + " value: NULL"); else JDTrace.logInformation (this, "parameter index: " + parameterIndex + " value: " + x.toString()); } setValue (parameterIndex, x, null, -1); } /* endif */ // @PDA jdbc40 /** * Sets the designated paramter to the given String object. The * driver converts this to a SQL NCHAR or NVARCHAR * or LONGNVARCHAR value (depending on the argument's size * relative to the driver's limits on NVARCHAR values) when it * sends it to the database. * * @param parameterIndex The parameter index (1-based). * @param value * the parameter value * @throws SQLException If a database error occurs. * if the driver does not support national character sets; if the * driver can detect that a data conversion error could occur ; or * if a database access error occurs */ public void setNString(int parameterIndex, String value) throws SQLException { if (JDTrace.isTraceOn()) { JDTrace.logInformation(this, "setNString()"); if (value == null) JDTrace.logInformation(this, "parameter index: " + parameterIndex + " value: NULL"); else if (value.length() > maxToLog_) JDTrace.logInformation(this, "parameter index: " + parameterIndex + " length: " + value.length()); else JDTrace.logInformation(this, "parameter index: " + parameterIndex + " value: " + value); } setString(parameterIndex, value); } // @PDA jdbc40 /** * Sets the designated parameter to a Reader object. The * Reader reads the data till end-of-file is reached. The driver * does the necessary conversion from Java character format to the national * character set in the database. * * @param parameterIndex The parameter index (1-based). * @param value * the parameter value * @param length * the number of characters in the parameter data. * @throws SQLException If a database error occurs. * if the driver does not support national character sets; if the * driver can detect that a data conversion error could occur ; or * if a database access error occurs */ public void setNCharacterStream(int parameterIndex, Reader value, long length) throws SQLException { if (JDTrace.isTraceOn()) { JDTrace.logInformation(this, "setNCharacterStream()"); if (value == null) JDTrace.logInformation(this, "parameter index: " + parameterIndex + " value: NULL"); else JDTrace.logInformation(this, "parameter index: " + parameterIndex + " length: " + length); } setCharacterStream(parameterIndex, value, (int) length); } // @PDA jdbc40 /** * Sets the designated parameter to a java.sql.NClob object. The driver converts this to a * SQL NCLOB value when it sends it to the database. * @param parameterIndex The parameter index (1-based). * @param value the parameter value * @throws SQLException if the driver does not support national * character sets; if the driver can detect that a data conversion * error could occur ; or if a database access error occurs */ /* ifdef JDBC40 */ public void setNClob(int parameterIndex, NClob value) throws SQLException { if(JDTrace.isTraceOn()) { JDTrace.logInformation (this, "setNClob()"); if(value == null) JDTrace.logInformation (this, "parameter index: " + parameterIndex + " value: NULL"); else if(value.length() > maxToLog_) JDTrace.logInformation (this, "parameter index: " + parameterIndex + " value: " + value.getSubString(1, (int)value.length())); else JDTrace.logInformation (this, "parameter index: " + parameterIndex + " length: " + value.length()); } setClob(parameterIndex, value); } /* endif */ // @PDA jdbc40 /** * Sets the designated parameter to a Reader object. The reader * must contain the number of characters specified by length otherwise a * SQLException will be generated when the * PreparedStatement is executed. * * @param parameterIndex The parameter index (1-based). * @param reader * An object that contains the data to set the parameter value to. * @param length * the number of characters in the parameter data. * @throws SQLException If a database error occurs. * if parameterIndex does not correspond to a parameter marker in * the SQL statement, or if the length specified is less than zero. * */ public void setClob(int parameterIndex, Reader reader, long length) throws SQLException { if (JDTrace.isTraceOn()) { JDTrace.logInformation(this, "setClob(int,Reader,long)"); if (reader == null) JDTrace.logInformation(this, "parameter index: " + parameterIndex + " value: NULL"); else JDTrace.logInformation(this, "parameter index: " + parameterIndex + " class: "+reader.getClass().getName() + " length: " + length); } setCharacterStream(parameterIndex, reader, (int) length); } // @PDA jdbc40 /** * Sets the designated parameter to an InputStream object. The * inputStream must contain the number of characters specified by length * otherwise a SQLException will be generated when the * PreparedStatement is executed. * * @param parameterIndex The parameter index (1-based). * @param inputStream * An object that contains the data to set the parameter value to. * @param length * the number of bytes in the parameter data. * @throws SQLException If a database error occurs. * if parameterIndex does not correspond to a parameter marker in * the SQL statement, if the length specified is less than zero or * if the number of bytes in the inputstream does not match the * specfied length. * */ public void setBlob(int parameterIndex, InputStream inputStream, long length) throws SQLException { if (JDTrace.isTraceOn()) { JDTrace.logInformation(this, "setBlob()"); if (inputStream == null) JDTrace.logInformation(this, "parameter index: " + parameterIndex + " value: NULL"); else JDTrace.logInformation(this, "parameter index: " + parameterIndex + " length: " + length); } setBinaryStream(parameterIndex, inputStream, (int) length); } // @PDA jdbc40 /** * Sets the designated parameter to a Reader object. The reader * must contain the number of characters specified by length otherwise a * SQLException will be generated when the * PreparedStatement is executed. * * @param parameterIndex The parameter index (1-based). * @param reader * An object that contains the data to set the parameter value to. * @param length * the number of characters in the parameter data. * @throws SQLException If a database error occurs. * if parameterIndex does not correspond to a parameter marker in * the SQL statement; if the length specified is less than zero; if * the driver does not support national character sets; if the * driver can detect that a data conversion error could occur; or if * a database access error occurs * */ public void setNClob(int parameterIndex, Reader reader, long length) throws SQLException { if (JDTrace.isTraceOn()) { JDTrace.logInformation(this, "setNClob()"); if (reader == null) JDTrace.logInformation(this, "parameter index: " + parameterIndex + " value: NULL"); else JDTrace.logInformation(this, "parameter index: " + parameterIndex + " length: " + length); } setCharacterStream(parameterIndex, reader, (int) length); } // @PDA jdbc40 /** * Sets the designated parameter to the given java.sql.SQLXML object. * @param parameterIndex The parameter index (1-based). * @param xmlObject a SQLXML object that maps an SQL XML value * @throws SQLException if a database access error occurs */ /* ifdef JDBC40 */ public void setSQLXML(int parameterIndex, SQLXML xmlObject) throws SQLException { if(JDTrace.isTraceOn()) { int len; if(xmlObject == null) len = 0; else len = xmlObject.getString().length(); //no length() method yet in jdbc. JDTrace.logInformation (this, "setSQLXML()"); if(xmlObject == null) JDTrace.logInformation (this, "parameter index: " + parameterIndex + " value: NULL"); else if(len < maxToLog_) JDTrace.logInformation (this, "parameter index: " + parameterIndex + " value: " + xmlObject.getString()); else JDTrace.logInformation (this, "parameter index: " + parameterIndex + " length: " + len); } //@xmlspec special handling of blob/clob column types if(xmlObject == null) //@xmlspec3 { //@xmlspec3 setValue (parameterIndex, xmlObject, null, -1); //@xmlspec3 return; //@xmlspec3 } //@xmlspec3 SQLData sqlData = parameterRow_.getSQLType(parameterIndex); //@xmlspec int sqlDataType = sqlData.getType(); //@xmlspec switch(sqlDataType) { //@xmlspec case Types.CLOB: //@xmlspec setCharacterStream(parameterIndex, xmlObject.getCharacterStream());//@xmlspec break; //@xmlspec case Types.BLOB: //@xmlspec setBinaryStream(parameterIndex, xmlObject.getBinaryStream()); //@xmlspec break; //@xmlspec default: //@xmlspec setValue (parameterIndex, xmlObject, null, -1); } } /* endif */ // @pda jdbc40 protected String[] getValidWrappedList() { return new String[] { "com.ibm.as400.access.AS400JDBCPreparedStatement", "java.sql.PreparedStatement" }; } // @PDA jdbc40 /** * Sets the designated parameter to the given input stream, which will have * the specified number of bytes. When a very large ASCII value is input to a * LONGVARCHAR parameter, it may be more practical to send it via * a java.io.InputStream. Data will be read from the stream as * needed until end-of-file is reached. The JDBC driver will do any necessary * conversion from ASCII to the database char format. * *

* Note: This stream object can either be a standard Java stream object * or your own subclass that implements the standard interface. * * @param parameterIndex The parameter index (1-based). * @param x * the Java input stream that contains the ASCII parameter value * @param length * the number of bytes in the stream * @exception SQLException * if a database access error occurs or this method is called on * a closed PreparedStatement */ public void setAsciiStream(int parameterIndex, InputStream x, long length) throws SQLException { setAsciiStream(parameterIndex, x, (int) length); } // @PDA jdbc40 /** * Sets the designated parameter to the given input stream, which will have * the specified number of bytes. When a very large binary value is input to a * LONGVARBINARY parameter, it may be more practical to send it * via a java.io.InputStream object. The data will be read from * the stream as needed until end-of-file is reached. * *

* Note: This stream object can either be a standard Java stream object * or your own subclass that implements the standard interface. * * @param parameterIndex The parameter index (1-based). * @param x * the java input stream which contains the binary parameter value * @param length * the number of bytes in the stream * @exception SQLException * if a database access error occurs or this method is called on * a closed PreparedStatement */ public void setBinaryStream(int parameterIndex, InputStream x, long length) throws SQLException { setBinaryStream(parameterIndex, x, (int) length); } // @PDA jdbc40 /** * Sets the designated parameter to the given Reader object, * which is the given number of characters long. When a very large UNICODE * value is input to a LONGVARCHAR parameter, it may be more * practical to send it via a java.io.Reader object. The data * will be read from the stream as needed until end-of-file is reached. The * JDBC driver will do any necessary conversion from UNICODE to the database * char format. * *

* Note: This stream object can either be a standard Java stream object * or your own subclass that implements the standard interface. * * @param parameterIndex The parameter index (1-based). * @param reader * the java.io.Reader object that contains the Unicode * data * @param length * the number of characters in the stream * @exception SQLException * if a database access error occurs or this method is called on * a closed PreparedStatement */ public void setCharacterStream(int parameterIndex, Reader reader, long length) throws SQLException { setCharacterStream(parameterIndex, reader, (int) length); } /* initialize the array of parameter names */ void initializeParameterNames() throws SQLException { boolean isProcedure = (sqlStatement_.getProcedure() != null); int count = 0; if (parameterNames_ == null) { // Else, create a new hash table to hold all the column name/number // mappings. parameterNames_ = new String[parameterCount_]; // // Check to see if the parameter names were returned by the super // extended parameter marker format. // This logic added @G5A 8/20/2012 // boolean allFound = true; for (int parameterNumber = 0; allFound && parameterNumber < parameterCount_; parameterNumber++) { try { String name = parameterRow_.getFieldName(parameterNumber + 1); if (name != null) { parameterNames_[parameterNumber] = name; } else { if (isProcedure) { allFound = false; } } } catch (SQLException sqlex) { // // Log the exception and go down the old path // if (JDTrace.isTraceOn()) { JDTrace.logException(this, "exception while retrieving field names", sqlex); } allFound = false; } /* catch SQLException processing names */ } /* for parameterNumber */ if (!allFound) { // Cache all the parm names and numbers. Statement s = null; // @scan1 ResultSet rs = null; // @scan1 try { s = connection_.createStatement(); String catalogSeparator = ""; // @74A Added a check for the naming // used. Need to use separator // appropriate to naming. if (connection_.getProperties().equals(JDProperties.NAMING, JDProperties.NAMING_SQL)) // @74A catalogSeparator = "."; // @74A else // @74A catalogSeparator = "/"; // @74A String schema = sqlStatement_.getSchema(); if (schema == null || schema.equals("")) // no schema in statement { // Derive the schema. schema = connection_.getDefaultSchema(true); // get raw value if (schema == null) // No default schema was set on the connection // url, or by the libraries connection property. { if (catalogSeparator.equals(".")) // using sql naming { schema = connection_.getUserName(); // set to user profile } else // using system naming { // Retrieve the library list from the IBM i - Use ROI Retrieve // Library List. ResultSet rs1 = JDUtilities.getLibraries(this, connection_, null, true); Vector libListV = new Vector(); while (rs1.next()) { libListV.addElement(rs1.getString(1)); } rs1.close(); // @SS String[] libList = new String[libListV.size()]; libListV.toArray(libList); // Get a result set that we can scroll forward/backward through. Statement s1 = connection_.createStatement( ResultSet.TYPE_SCROLL_INSENSITIVE, ResultSet.CONCUR_READ_ONLY); rs = s1.executeQuery("SELECT ROUTINE_SCHEMA FROM QSYS2" + catalogSeparator + "SYSPROCS WHERE ROUTINE_NAME='" + unquote(sqlStatement_.getProcedure()) + "' AND IN_PARMS + OUT_PARMS + INOUT_PARMS = " + parameterCount_); if (!rs.next()) JDError.throwSQLException(this, JDError.EXC_INTERNAL); // didn't // find // the // procedure // in // any // schema // If we get this far, at least one schema contains a procedure // similar to ours. boolean found = false; for (int i = 0; i < libList.length && !found; i++) { if (libList[i].length() != 0) { rs.beforeFirst(); // re-position to before the first row while (rs.next() && !found) { if (rs.getString(1).equals(libList[i])) { schema = rs.getString(1); found = true; // we found a procedure that matches our // criteria } } } } rs.close(); // @SS s1.close(); // @SS if (!found) // none of the libraries in our library list contain // a stored procedure that we are looking for JDError.throwSQLException(this, JDError.EXC_INTERNAL); } } } rs = s.executeQuery("SELECT SPECIFIC_NAME FROM QSYS2" + catalogSeparator + "SYSPROCS WHERE ROUTINE_SCHEMA = '" + unquote(schema) + // @74C @DELIMc "' AND ROUTINE_NAME = '" + unquote(sqlStatement_.getProcedure()) + // @DELIMc "' AND IN_PARMS + OUT_PARMS + INOUT_PARMS = " + parameterCount_); // If there are no rows, throw an internal driver exception if (!rs.next()) JDError.throwSQLException(this, JDError.EXC_INTERNAL); String specificName = rs.getString(1); rs.close(); // @SS rs = s .executeQuery("SELECT PARAMETER_NAME, ORDINAL_POSITION FROM QSYS2" + catalogSeparator + "SYSPARMS WHERE " + // @74A " SPECIFIC_NAME = '" + unquoteNoUppercase(specificName) + "' AND SPECIFIC_SCHEMA = '" + unquote(schema) + "'"); // @DELIMc while (rs.next()) { count++; String colName = rs.getString(1); int colInd = rs.getInt(2); parameterNames_[colInd - 1] = colName; } } finally { if (rs != null) // @scan1 rs.close(); // @SS if (s != null) // @scan1 s.close(); // @SS } // If the number of parm names didn't equal the number of parameters, // throw // an exception (INTERNAL). if (count != parameterCount_) { JDError.throwSQLException(this, JDError.EXC_INTERNAL); } } } } // @pda jdbc40 needed for rowset.setX methods. Moved from callableStatement. /* * Find the column index that matches this parameter name. * * @param parameterName The parameter name to change into a column index * (1-based). */ int findParameterIndex(String parameterName) throws SQLException { // Throw an exception if null was passed in if (parameterName == null) { throw JDError.throwSQLException(this, JDError.EXC_ATTRIBUTE_VALUE_INVALID); } // Throw an exception if the Statement is closed (FUNCTION SEQUENCE) if (isClosed()) JDError.throwSQLException(this, JDError.EXC_FUNCTION_SEQUENCE); boolean caseSensitive = false; int count = 0; int returnParm = 0; // determine if our search should be case insensitive or not if (parameterName.startsWith("\"") && parameterName.endsWith("\"")) // assume // no // leading // or // trailing // blanks { parameterName = JDUtilities.stripOuterDoubleQuotes(parameterName); caseSensitive = true; } if (parameterNames_ == null) { initializeParameterNames(); } // If we have a cache created, try to find the column name in it. // Look up the mapping in our cache. while (count < parameterNames_.length) { if (parameterNames_[count] != null) { if ((caseSensitive && parameterNames_[count].equals(parameterName)) || (!caseSensitive && parameterNames_[count] .equalsIgnoreCase(parameterName))) { returnParm = count + 1; break; } } ++count; } // Throw an exception if the column name is not found (COLUMN NOT FOUND). if (returnParm == 0) JDError.throwSQLException(this, JDError.EXC_COLUMN_NOT_FOUND); // If the statement is using the return value parameter, increment the value to // compensate for the return value parameter. @K2A. if (useReturnValueParameter_) { returnParm++; } return returnParm; } /** * Return the name of the parameter for a stored procedure call. * @param parm the parameter number to get the name for * @return the parameter name. * @throws SQLException If a database error occurs. */ public String getDB2ParameterName(int parm) throws SQLException { try { internalLock.lock(); checkOpen(); validatePSParameterIndex(parm); parm = parm - 1; if (useReturnValueParameter_) { parm = parm - 1; } if (parameterNames_ == null) { initializeParameterNames(); } if ((parm >= 0) && (parm < parameterNames_.length)) { return parameterNames_[parm]; } else { JDError.throwSQLException(this, JDError.EXC_DESCRIPTOR_INDEX_INVALID,"INTERNAL_ERROR"); return null; } } finally { internalLock.unlock(); } } // @PDA jdbc40 move from callableStatement private static final String unquote(String name) { return JDUtilities.prepareForSingleQuotes(name, true); } // @PDA jdbc40 move from callableStatement private static final String unquoteNoUppercase(String name) { return JDUtilities.prepareForSingleQuotes(name, false); } // @PDA jdbc40 helper method private void setInputStream(int parameterIndex, InputStream x) throws SQLException { // @J0A added the code from setValue in this method because streams and // readers are handled specially try { internalLock.lock(); checkOpen(); // Validate the parameter index. validatePSParameterIndex(parameterIndex); // Check if the parameter index refers to the return value parameter. // This is an OUT parameter, so sets are not allowed. If its not // parameter index 1, then decrement the parameter index, since we // are "faking" the return value parameter. if (useReturnValueParameter_) { if (parameterIndex == 1) JDError.throwSQLException(this, JDError.EXC_PARAMETER_TYPE_INVALID); else --parameterIndex; } // Check that the parameter is an input parameter. if (!parameterRow_.isInput(parameterIndex)) JDError.throwSQLException(this, JDError.EXC_PARAMETER_TYPE_INVALID); // Set the parameter data. If there is a type mismatch, // set() with throw an exception. SQLData sqlData = parameterRow_.getSQLType(parameterIndex); if (x != null) { // If the data is a locator, then set its handle. int sqlType = sqlData.getSQLType(); // @xml3 if (sqlType == SQLData.CLOB_LOCATOR || sqlType == SQLData.BLOB_LOCATOR || sqlType == SQLData.DBCLOB_LOCATOR || // @pdc jdbc40 sqlType == SQLData.NCLOB_LOCATOR || // @pda jdbc40 sqlType == SQLData.XML_LOCATOR) // @xml3 { SQLLocator sqlDataAsLocator = (SQLLocator) sqlData; sqlDataAsLocator.setHandle(parameterRow_ .getFieldLOBLocatorHandle(parameterIndex)); if (JDTrace.isTraceOn()) JDTrace.logInformation( this, "locator handle: " + parameterRow_.getFieldLOBLocatorHandle(parameterIndex)); sqlData.set(x, null, SQLData.ALL_READER_BYTES);// new ConvTableReader(x, 819, 0, // LOB_BLOCK_SIZE), null, -2); //@readerlen // -2 flag to read all of reader bytes } else { sqlData.set(x, null, SQLData.ALL_READER_BYTES);// sqlData.set // (JDUtilities.readerToString(new // ConvTableReader(x, 819, 0, // LOB_BLOCK_SIZE)), null, -1); //@readerlen // -2 flag to read all of reader bytes } testDataTruncation(parameterIndex, sqlData); } // Parameters can be null; you can call one of the set methods to null out // a // field of the database. parameterNulls_[parameterIndex - 1] = (x == null); parameterSet_[parameterIndex - 1] = true; } finally { internalLock.unlock(); } } // @PDA jdbc40 /** * Sets the designated parameter to the given input stream. When a very large * ASCII value is input to a LONGVARCHAR parameter, it may be * more practical to send it via a java.io.InputStream. Data will * be read from the stream as needed until end-of-file is reached. The JDBC * driver will do any necessary conversion from ASCII to the database char * format. * *

* Note: This stream object can either be a standard Java stream object * or your own subclass that implements the standard interface. * * @param parameterIndex The parameter index (1-based). * @param x * the Java input stream that contains the ASCII parameter value * @exception SQLException * if parameterIndex does not correspond to a parameter marker in * the SQL statement; if a database access error occurs or this * method is called on a closed PreparedStatement */ public void setAsciiStream(int parameterIndex, InputStream x) throws SQLException { if (JDTrace.isTraceOn()) { JDTrace.logInformation(this, "setAsciiStream(int, InputStream)"); if (x == null) JDTrace.logInformation(this, "parameter index: " + parameterIndex + " value: NULL"); } setInputStream(parameterIndex, x); } // @PDA jdbc40 /** * Sets the designated parameter to the given input stream. When a very large * binary value is input to a LONGVARBINARY parameter, it may be * more practical to send it via a java.io.InputStream object. * The data will be read from the stream as needed until end-of-file is * reached. * *

* Note: This stream object can either be a standard Java stream object * or your own subclass that implements the standard interface. * * @param parameterIndex The parameter index (1-based). * @param x * the java input stream which contains the binary parameter value * @exception SQLException * if parameterIndex does not correspond to a parameter marker in * the SQL statement; if a database access error occurs or this * method is called on a closed PreparedStatement */ public void setBinaryStream(int parameterIndex, InputStream x) throws SQLException { if (JDTrace.isTraceOn()) { JDTrace.logInformation(this, "setBinaryStream(int, InputStream)"); if (x == null) JDTrace.logInformation(this, "parameter index: " + parameterIndex + " value: NULL"); } setInputStream(parameterIndex, x); } // @PDA jdbc40 /** * Sets the designated parameter to a InputStream object. This * method differs from the setBinaryStream (int, InputStream) * method because it informs the driver that the parameter value should be * sent to the server as a BLOB. When the * setBinaryStream method is used, the driver may have to do * extra work to determine whether the parameter data should be sent to the * server as a LONGVARBINARY or a BLOB * * @param parameterIndex The parameter index (1-based). * @param inputStream * An object that contains the data to set the parameter value to. * @throws SQLException If a database error occurs. * if parameterIndex does not correspond to a parameter marker in * the SQL statement; if a database access error occurs; this method * is called on a closed PreparedStatement or if * parameterIndex does not correspond to a parameter marker in the * SQL statement, * */ public void setBlob(int parameterIndex, InputStream inputStream) throws SQLException { if (JDTrace.isTraceOn()) { JDTrace.logInformation(this, "setBlob(int, InputStream)"); if (inputStream == null) JDTrace.logInformation(this, "parameter index: " + parameterIndex + " value: NULL"); } setInputStream(parameterIndex, inputStream); } // @PDA jdbc40 helper private void setReader(int parameterIndex, Reader reader) throws SQLException { // @J0A added the code from setValue in this method because streams and // readers are handled specially try { internalLock.lock(); checkOpen(); // Validate the parameter index. validatePSParameterIndex(parameterIndex); // Check if the parameter index refers to the return value parameter. // This is an OUT parameter, so sets are not allowed. If its not // parameter index 1, then decrement the parameter index, since we // are "faking" the return value parameter. if (useReturnValueParameter_) { if (parameterIndex == 1) JDError.throwSQLException(this, JDError.EXC_PARAMETER_TYPE_INVALID); else --parameterIndex; } // Check that the parameter is an input parameter. if (!parameterRow_.isInput(parameterIndex)) JDError.throwSQLException(this, JDError.EXC_PARAMETER_TYPE_INVALID); // Set the parameter data. If there is a type mismatch, // set() with throw an exception. SQLData sqlData = parameterRow_.getSQLType(parameterIndex); if (reader != null) { // If the data is a locator, then set its handle. int sqlType = sqlData.getSQLType(); // @xml3 if (sqlType == SQLData.CLOB_LOCATOR || sqlType == SQLData.BLOB_LOCATOR || sqlType == SQLData.DBCLOB_LOCATOR || // @pdc jdbc40 sqlType == SQLData.NCLOB_LOCATOR || // @pda jdbc40 sqlType == SQLData.XML_LOCATOR) // @xml3 { SQLLocator sqlDataAsLocator = (SQLLocator) sqlData; sqlDataAsLocator.setHandle(parameterRow_ .getFieldLOBLocatorHandle(parameterIndex)); if (JDTrace.isTraceOn()) JDTrace.logInformation( this, "locator handle: " + parameterRow_.getFieldLOBLocatorHandle(parameterIndex)); sqlData.set(reader, null, SQLData.ALL_READER_BYTES); // @readerlen -2 flag to read all of // reader chars } else if (sqlType == SQLData.BOOLEAN) { // The stream to boolean conversion is not supported JDError.throwSQLException(this, JDError.EXC_DATA_TYPE_MISMATCH); } else { sqlData.set(JDUtilities.readerToString(reader), null, -1); } testDataTruncation(parameterIndex, sqlData); } // Parameters can be null; you can call one of the set methods to null out // a // field of the database. parameterNulls_[parameterIndex - 1] = (reader == null); parameterSet_[parameterIndex - 1] = true; } finally { internalLock.unlock(); } } // @PDA jdbc40 /** * Sets the designated parameter to the given Reader object. When * a very large UNICODE value is input to a LONGVARCHAR * parameter, it may be more practical to send it via a * java.io.Reader object. The data will be read from the stream * as needed until end-of-file is reached. The JDBC driver will do any * necessary conversion from UNICODE to the database char format. * *

* Note: This stream object can either be a standard Java stream object * or your own subclass that implements the standard interface. *

* Note: Consult your JDBC driver documentation to determine if it * might be more efficient to use a version of setCharacterStream * which takes a length parameter. * * @param parameterIndex The parameter index (1-based). * @param reader * the java.io.Reader object that contains the Unicode * data * @exception SQLException * if parameterIndex does not correspond to a parameter marker in * the SQL statement; if a database access error occurs or this * method is called on a closed PreparedStatement */ public void setCharacterStream(int parameterIndex, Reader reader) throws SQLException { if (JDTrace.isTraceOn()) { JDTrace.logInformation(this, "setCharacterStream(int, Reader)"); if (reader == null) JDTrace.logInformation(this, "parameter index: " + parameterIndex + " value: NULL"); } setReader(parameterIndex, reader); } // @PDA jdbc40 /** * Sets the designated parameter to a Reader object. This method * differs from the setCharacterStream (int, Reader) method * because it informs the driver that the parameter value should be sent to * the server as a CLOB. When the setCharacterStream * method is used, the driver may have to do extra work to determine whether * the parameter data should be sent to the server as a * LONGVARCHAR or a CLOB * *

* Note: Consult your JDBC driver documentation to determine if it * might be more efficient to use a version of setClob which * takes a length parameter. * * @param parameterIndex The parameter index (1-based). * @param reader * An object that contains the data to set the parameter value to. * @throws SQLException If a database error occurs. * if parameterIndex does not correspond to a parameter marker in * the SQL statement; if a database access error occurs; this method * is called on a closed PreparedStatementor if * parameterIndex does not correspond to a parameter marker in the * SQL statement * */ public void setClob(int parameterIndex, Reader reader) throws SQLException { if (JDTrace.isTraceOn()) { JDTrace.logInformation(this, "setClob(int, Reader)"); if (reader == null) JDTrace.logInformation(this, "parameter index: " + parameterIndex + " value: NULL"); } setReader(parameterIndex, reader); } // @PDA jdbc40 /** * Sets the designated parameter to a Reader object. The * Reader reads the data till end-of-file is reached. The driver * does the necessary conversion from Java character format to the national * character set in the database. * *

* Note: This stream object can either be a standard Java stream object * or your own subclass that implements the standard interface. *

* Note: Consult your JDBC driver documentation to determine if it * might be more efficient to use a version of * setNCharacterStream which takes a length parameter. * * @param parameterIndex The parameter index (1-based). * @param value * the parameter value * @throws SQLException If a database error occurs. * if parameterIndex does not correspond to a parameter marker in * the SQL statement; if the driver does not support national * character sets; if the driver can detect that a data conversion * error could occur; if a database access error occurs; or this * method is called on a closed PreparedStatement */ public void setNCharacterStream(int parameterIndex, Reader value) throws SQLException { if (JDTrace.isTraceOn()) { JDTrace.logInformation(this, "setNCharacterStream(int, Reader)"); if (value == null) JDTrace.logInformation(this, "parameter index: " + parameterIndex + " value: NULL"); } setReader(parameterIndex, value); } // @PDA jdbc40 /** * Sets the designated parameter to a Reader object. This method * differs from the setCharacterStream (int, Reader) method * because it informs the driver that the parameter value should be sent to * the server as a NCLOB. When the * setCharacterStream method is used, the driver may have to do * extra work to determine whether the parameter data should be sent to the * server as a LONGNVARCHAR or a NCLOB *

* Note: Consult your JDBC driver documentation to determine if it * might be more efficient to use a version of setNClob which * takes a length parameter. * * @param parameterIndex The parameter index (1-based). * @param reader * An object that contains the data to set the parameter value to. * @throws SQLException If a database error occurs. * if parameterIndex does not correspond to a parameter marker in * the SQL statement; if the driver does not support national * character sets; if the driver can detect that a data conversion * error could occur; if a database access error occurs or this * method is called on a closed PreparedStatement */ public void setNClob(int parameterIndex, Reader reader) throws SQLException { if (JDTrace.isTraceOn()) { JDTrace.logInformation(this, "setNClob(int, Reader)"); if (reader == null) JDTrace.logInformation(this, "parameter index: " + parameterIndex + " value: NULL"); } setReader(parameterIndex, reader); } /** * Sets the value of the designated parameter with the given object. If the second * argument is an InputStream then the stream must contain the number of bytes * specified by scaleOrLength. If the second argument is a Reader then the reader * must contain the number of characters specified by scaleOrLength. If these * conditions are not true the driver will generate a SQLException when the * prepared statement is executed. *

The given Java object will be converted to the given targetSqlType * before being sent to the database. If the object has a custom mapping * (is of a class implementing the interface SQLData), the JDBC driver * should call the method SQLData.writeSQL to write it to the SQL data stream. * If, on the other hand, the object is of a class implementing Ref, Blob, * Clob, NClob, Struct, java.net.URL, or Array, the driver should pass it * to the database as a value of the corresponding SQL type. *

Note that this method may be used to pass database-specific abstract data types. * * @param parameterIndex - the first parameter is 1, the second is 2, ... * @param x - the object containing the input parameter value * @param targetSqlType - the SQL type to be sent to the database. The scale argument may * further qualify this type. * @param scaleOrLength - for java.sql.JDBCType.DECIMAL or java.sql.JDBCType.NUMERIC * types, this is the number of digits after the decimal point. For Java Object types * InputStream and Reader, this is the length of the data in the stream or reader. * For all other types, this value will be ignored. * @throws SQLException - if parameterIndex does not correspond to a parameter * marker in the SQL statement; if a database access error occurs or this method * is called on a closed PreparedStatement or if the Java Object specified by x * is an InputStream or Reader object and the value of the scale parameter is * less than zero */ public void setObject(int parameterIndex, Object x, /* ifdef JDBC42 SQLType endif*/ /* ifndef JDBC42 */ Object /* endif */ targetSqlType, int scaleOrLength) throws SQLException { int intType = mapSQLType(targetSqlType); setObject(parameterIndex, x, intType, scaleOrLength); } /** * converts an SQLType to its corresponding java.sql.Types value */ int mapSQLType( /* ifdef JDBC42 SQLType endif*/ /* ifndef JDBC42 */ Object /* endif */ targetSqlType ) throws SQLException { /* ifdef JDBC42 if (targetSqlType instanceof JDBCType) { return targetSqlType.getVendorTypeNumber(); } throw new SQLFeatureNotSupportedException("targetSqlType="+targetSqlType); endif */ /* ifndef JDBC42 */ return 0; /* endif */ } /** * Sets the value of the designated parameter with the given object. This method * is similar to setObject(int parameterIndex, Object x, SQLType targetSqlType, * int scaleOrLength), except that it assumes a scale of zero. * @param parameterIndex - the first parameter is 1, the second is 2, ... * @param x - the object containing the input parameter value * @param targetSqlType - the SQL type to be sent to the database parameterIndex * @throws SQLException - if parameterIndex does not correspond to a parameter marker * in the SQL statement; if a database access error occurs or this method is called * on a closed PreparedStatement */ public void setObject(int parameterIndex, Object x, /* ifdef JDBC42 SQLType endif*/ /* ifndef JDBC42 */ Object /* endif */ targetSqlType) throws SQLException { int intType = mapSQLType(targetSqlType); setObject(parameterIndex, x, intType); } /** * Executes the SQL statement in this PreparedStatement object, which * must be an SQL Data Manipulation Language (DML) statement, such as * INSERT, UPDATE or DELETE; or an SQL statement that returns nothing, * such as a DDL statement. *

This method should be used when the returned row count may exceed Integer.MAX_VALUE. * @return either (1) the row count for SQL Data Manipulation Language (DML) statements or * (2) 0 for SQL statements that return nothing * @throws SQLException - if a database access error occurs; this method is called on a * closed PreparedStatement or the SQL statement returns a ResultSet object */ public long executeLargeUpdate() throws SQLException { return (long) executeUpdate(); } public long executeLargeUpdate(String sql) throws SQLException { JDError.throwSQLException(this, JDError.EXC_FUNCTION_SEQUENCE); return 0; } public long executeLargeUpdate(String sql, int autoGeneratedKeys) throws SQLException { JDError.throwSQLException(this, JDError.EXC_FUNCTION_SEQUENCE); return 0; } public long executeLargeUpdate(String sql, int[] columnIndexes) throws SQLException { JDError.throwSQLException(this, JDError.EXC_FUNCTION_SEQUENCE); return 0; } public long executeLargeUpdate(String sql, String[] columnNames) throws SQLException { JDError.throwSQLException(this, JDError.EXC_FUNCTION_SEQUENCE); return 0; } /** Checks that the statement is open. Public methods that require an open statement should call this first. @exception SQLException If the statement is not open. **/ void checkOpen () throws SQLException { super.checkOpen(); if (connectionReset_) { // Any exception from the prepare will just get thrown preserveParameters_ = true; // Make sure the existing parameters are saved before we prepared again if (parameterRow_ != null) { parameterRow_.saveValues(); } resultRow_ = commonPrepare(sqlStatement_); connectionReset_ = false; } } void setConnectionReset(boolean reset) { descriptorHandle_ = 0; super.setConnectionReset(reset); } void setSaveParameterValues(boolean saveParameterValues) { saveParameterValues_ = saveParameterValues; } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy