src.com.ibm.as400.access.JDUtilities Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of jt400 Show documentation
Show all versions of jt400 Show documentation
The Open Source version of the IBM Toolbox for Java
///////////////////////////////////////////////////////////////////////////////
//
// JTOpen (IBM Toolbox for Java - OSS version)
//
// Filename: JDUtilities.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-2001 International Business Machines Corporation and
// others. All rights reserved.
//
///////////////////////////////////////////////////////////////////////////////
package com.ibm.as400.access;
import java.io.ByteArrayOutputStream;
import java.io.InputStream;
import java.io.IOException;
import java.io.Reader;
import java.math.BigDecimal;
import java.sql.*; // @J1c
import java.util.Hashtable;
/**
* The JDUtilities class provides utilities for use in the implementation of the
* JDBC driver.
**/
class JDUtilities {
static final String copyright = "Copyright (C) 1997-2001 International Business Machines Corporation and others.";
private static final byte escape = (byte) 0x1B; // @D0A
// @j4 the following two constants are now final
static final int vrm740 = AS400.generateVRM(7, 4, 0);
static final int vrm710 = AS400.generateVRM(7, 1, 0); // @710 //@128sch
static final int vrm610 = AS400.generateVRM(6, 1, 0); // @610
static final int vrm540 = AS400.generateVRM(5, 4, 0); // @540
static final int vrm530 = AS400.generateVRM(5, 3, 0); // @G0A
static final int vrm520 = AS400.generateVRM(5, 2, 0); // @J2a @J3a
static final int vrm510 = AS400.generateVRM(5, 1, 0); // @J31a
static final int vrm450 = AS400.generateVRM(4, 5, 0); // @G0A
static final int vrm440 = AS400.generateVRM(4, 4, 0);
static final int vrm430 = AS400.generateVRM(4, 3, 0);
static int JDBCLevel_ = 30; // @J4a
static int JVMLevel_ = 120; // 1.2.0 //@big
private final static Object bigDecimalLock_ = new Object(); // @big
// @J4a
static {
try {
Class.forName("java.util.concurrent.Semaphore"); // @big
JVMLevel_ = 150; // jre 5.0 //@big
Class.forName("java.sql.SQLXML"); // @big
JDBCLevel_ = 40; // @big
JVMLevel_ = 160; // jre 6.0 //@big
} catch (Throwable e) {
}
}
// @D0A
/**
* Decompresses data from one byte array to another.
*
* @param source
* The source (compressed) bytes.
* @param sourceOffset
* The offset in the source bytes to start decompressing.
* @param sourceLength
* The length (compressed) of the bytes to decompress.
* @param destination
* The destination (uncompressed) bytes. It is assumed that this byte
* array is already created.
* @param destinationOffset
* The offset in the destination bytes.
**/
static final void decompress(byte[] source, int sourceOffset,
int sourceLength, byte[] destination, int destinationOffset) {
int i = sourceOffset; // Index into source.
int j = destinationOffset; // Index into destination.
int sourceEnd = sourceOffset + sourceLength;
while (i < sourceEnd) {
if (source[i] == escape) {
if (source[i + 1] == escape) {
destination[j++] = escape;
i += 2;
} else {
int repetitions = BinaryConverter.byteArrayToInt(source, i + 2);
for (int k = 1; k <= repetitions; ++k)
destination[j++] = source[i + 1];
i += 6;
}
} else
destination[j++] = source[i++];
}
}
/**
* Pads a numeric value on the left with zeros. This is a utility for use in
* implementing various pieces of the JDBC driver.
*
* @param value
* The numeric value.
* @param digits
* The number of digits.
* @return The padded string.
**/
static final String padZeros(int value, int digits) {
String temp = "000000000" + Integer.toString(value); // @A1C
return temp.substring(temp.length() - digits);
}
static final String padZeros(long value, int digits) {
String temp = "000000000000" + Long.toString(value); // @A1C
return temp.substring(temp.length() - digits);
}
// @DELIMa
/**
* Returns the names of the libraries on the system. This will return a
* ResultSet with a list of all the libraries.
*
* @param caller
* The calling object.
* @param connection
* The connection to the database.
* @param setting
* The conversion settings (optional).
* @param libraryListOnly
* If true, returns only the libraries on the library list. Otherwise
* returns all libraries on the system.
* @return A ResultSet containing the list of the libraries.
* @exception SQLException
* If the connection is not open or an error occurs.
**/
static final ResultSet getLibraries(Object caller,
AS400JDBCConnection connection, SQLConversionSettings settings,
boolean libraryListOnly) throws SQLException {
// Schema = library
connection.checkOpen();
JDRowCache rowCache = null; // Creates a set of rows that
// are readable one at a time
if (settings == null) {
settings = SQLConversionSettings.getConversionSettings(connection);
}
int id = connection.getID();
DBReplyRequestedDS reply = null;
try {
// Create a request
// @P0C
DBReturnObjectInformationRequestDS request = null;
try {
request = DBDSPool
.getDBReturnObjectInformationRequestDS(
DBReturnObjectInformationRequestDS.FUNCTIONID_RETRIEVE_LIBRARY_INFO,
id, DBBaseRequestDS.ORS_BITMAP_RETURN_DATA
+ DBBaseRequestDS.ORS_BITMAP_DATA_FORMAT
+ DBBaseRequestDS.ORS_BITMAP_RESULT_DATA, 0);
if (!libraryListOnly) // Return list of all libraries on the system
{
request.setLibraryName("%", connection.getConverter());
request.setLibraryNameSearchPatternIndicator(0xF1);
} else {
request.setLibraryName("*LIBL", connection.getConverter()); // @libl
request.setLibraryNameSearchPatternIndicator(0xF0); // @libl
}
// Set the Library Information to Return Bitmap
// Return only the library name
request.setLibraryReturnInfoBitmap(0x80000000);
// Send the request and cache all results from the system
reply = connection.sendAndReceive(request);
// Check for errors - throw exception if errors were
// returned
int errorClass = reply.getErrorClass();
if (errorClass != 0) {
int returnCode = reply.getReturnCode();
JDError.throwSQLException(caller, connection, id, errorClass,
returnCode);
}
// Get the data format and result data
DBDataFormat dataFormat = reply.getDataFormat();
DBData resultData = reply.getResultData();
// Put the result data into a row cache
JDServerRow row = new JDServerRow(connection, id, dataFormat, settings);
// Put the data format into a row format object
JDRowCache serverRowCache = new JDSimpleRowCache(new JDServerRowCache(
row, connection, id, 1, resultData, true,
ResultSet.TYPE_SCROLL_INSENSITIVE));
boolean isJDBC3 = true;
JDFieldMap[] maps = null; // @F2C
String[] fieldNames = null; // @F2C
SQLData[] sqlData = null; // @F2C
int[] fieldNullables = null; // @F2C
// Set up the result set in the format required by JDBC
if (!isJDBC3) {
fieldNames = new String[] { "TABLE_SCHEM" };
sqlData = new SQLData[] { new SQLVarchar(128, settings) }; // schema
// name
fieldNullables = new int[] { AS400JDBCDatabaseMetaData.columnNoNulls };
maps = new JDFieldMap[1];
} else {
fieldNames = new String[] { "TABLE_SCHEM", "TABLE_CATALOG" }; // @G4A
sqlData = new SQLData[] { new SQLVarchar(128, settings), // schema
// name
new SQLVarchar(128, settings) }; // table catalog //@G4A
fieldNullables = new int[] { AS400JDBCDatabaseMetaData.columnNoNulls,
AS400JDBCDatabaseMetaData.columnNullable }; // @G4A
maps = new JDFieldMap[2]; // @G4C
}
// Create the mapped row format that is returned in the
// result set.
// This does not actual move the data, it just sets up
// the mapping
maps[0] = new JDSimpleFieldMap(1); // table schema // @A3C @E4C
if (isJDBC3) // @F2A
{
maps[1] = new JDHardcodedFieldMap(connection.getCatalog()); // table
// catalog
// //@G4A
}
// Create the mapped row cache that is returned in the
// result set
JDMappedRow mappedRow = new JDMappedRow(fieldNames, sqlData,
fieldNullables, maps);
rowCache = new JDMappedRowCache(mappedRow, serverRowCache);
} finally {
if (request != null) {
request.returnToPool();
request = null;
}
// Cannot close this reply. Pass to AS400JDBCResultSet to close
// if (reply != null) { reply.returnToPool(); reply = null; }
}
} // End of try block
catch (DBDataStreamException e) {
JDError.throwSQLException(caller, JDError.EXC_INTERNAL, e);
}
// Return the results
return new AS400JDBCResultSet(rowCache, connection.getCatalog(), "Schemas",
connection, reply); // @in2
}
/**
* Strips outer double-quotes from name (if present).
**/
static final String stripOuterDoubleQuotes(String name/* , boolean uppercase */) {
if (name.startsWith("\"") && name.endsWith("\"")) {
name = name.substring(1, name.length() - 1);
}
return name;
}
/**
* Prepares the name to be enclosed in double-quotes, for example for use as
* column-name values in an INSERT INTO statement. 1. Strip outer
* double-quotes (if present). 2. Double-up any embedded double-quotes.
**/
static final String prepareForDoubleQuotes(String name) {
// Strip outer double-quotes.
name = stripOuterDoubleQuotes(name);
// Double-up any embedded double-quotes.
if (name.indexOf('\"') == -1) {
return name; // name has no embedded double-quotes, so nothing more to do
} else {
StringBuffer buf = new StringBuffer(name);
for (int i = name.length() - 1; i >= 0; i--) // examine char-by-char, from
// end
{
if (buf.charAt(i) == '\"') {
buf.insert(i, '\"'); // double the embedded double-quote
}
}
return buf.toString();
}
}
/**
* Prepares the name to be enclosed in single-quotes, for example for use in
* the WHERE clause of a SELECT statement. 1. Unless name is delimited by
* outer double-quotes, uppercase the name. 2. Strip outer double-quotes (if
* present). 3. Collapse any doubled embedded double-quotes, to single
* double-quotes. 4. Double-up any embedded single-quotes.
**/
static final String prepareForSingleQuotes(String name, boolean uppercase) {
// 1. Unless name is delimited by outer double-quotes, uppercase the name.
if (name.startsWith("\"") && name.endsWith("\"")) {
// 2. Strip outer double-quotes.
name = name.substring(1, name.length() - 1);
} else {
// Don't uppercase if any embedded quotes.
if (uppercase && name.indexOf('\'') == -1) {
name = name.toUpperCase();
}
}
// 3. Collapse any doubled embedded double-quotes, to single double-quotes.
// 4. Double-up any embedded single-quotes.
if (name.indexOf('\"') == -1 && name.indexOf('\'') == -1) {
return name; // name has no embedded double-quotes, so nothing more to do
} else {
StringBuffer buf = new StringBuffer(name);
for (int i = name.length() - 1; i >= 0; i--) // examine char-by-char, from
// end
{
char thisChar = buf.charAt(i);
if (thisChar == '\"') {
if (i > 0 && buf.charAt(i - 1) == '\"') {
buf.deleteCharAt(i);
i--; // don't re-examine the prior double-quote
}
} else if (thisChar == '\'') {
buf.insert(i, '\''); // double the single-quote
}
}
return buf.toString();
}
}
/**
* Strips out beginning/ending matching double-quotes, and internal double
* embedded double-quotes get collapsed to one double-quote. 1. Strip outer
* double-quotes (if present). 2. Collapse any doubled embedded double-quotes,
* to single double-quotes.
**/
static final String stripOutDoubleEmbededQuotes(String name) {
if (name.startsWith("\"") && name.endsWith("\"")) {
// 1. Strip outer double-quotes.
name = name.substring(1, name.length() - 1);
}
// 2. Collapse any doubled embedded double-quotes, to single double-quotes.
if (name.indexOf('\"') == -1) {
return name; // name has no embedded double-quotes, so nothing more to do
} else {
StringBuffer buf = new StringBuffer(name);
for (int i = name.length() - 1; i >= 0; i--) // examine char-by-char, from
// end
{
char thisChar = buf.charAt(i);
if (thisChar == '\"') {
if (i > 0 && buf.charAt(i - 1) == '\"') {
buf.deleteCharAt(i);
i--; // don't re-examine the prior double-quote
}
}
}
return buf.toString();
}
}
/**
* Reads a reader and returns its data as a String.
*
* @param input
* The reader.
* @param length
* The length.
* @return The string.
* @exception SQLException
* If the length is not valid or the conversion is not possible.
**/
static final String readerToString(Reader input, int length)
throws SQLException {
StringBuffer buffer = new StringBuffer();
try {
char[] rawChars = new char[(length == 0) ? 1 : length];
int actualLength = 0;
while (input.ready()) {
int length2 = input.read(rawChars);
if (length2 < 0)
break;
// buffer.append (new String (rawChars, 0, length2)); //@pdd jdbc40
buffer.append(rawChars, 0, length2); // @pda jdbc40 performance
actualLength += length2;
}
// The spec says to throw an exception when the
// actual length does not match the specified length.
// I think this is strange since this means the length
// parameter is essentially not needed. I.e., we always
// read the exact number of bytes in the stream.
if (actualLength != length)
JDError.throwSQLException(JDError.EXC_BUFFER_LENGTH_INVALID);
} catch (IOException e) {
JDError.throwSQLException(JDError.EXC_PARAMETER_TYPE_INVALID);
}
return buffer.toString();
}
// @pda jdbc40 new method for unknown length
/**
* Reads a reader and returns its data as a String. Reads until reader returns
* -1 for eof.
*
* @param input
* The reader.
* @return The string.
**/
static final String readerToString(Reader input) throws SQLException {
StringBuffer buffer = new StringBuffer();
try {
char[] rawChars = new char[32000];
int actualLength = 0;
while (input.ready()) {
int length2 = input.read(rawChars);
if (length2 < 0)
break;
// buffer.append (new String (rawChars, 0, length2)); //@pdd jdbc40
buffer.append(rawChars, 0, length2); // @pda jdbc40 performance
actualLength += length2;
}
} catch (IOException e) {
JDError.throwSQLException(JDError.EXC_PARAMETER_TYPE_INVALID);
}
return buffer.toString();
}
// @j1 new method
/**
* Runs a CL command via the database host server. It uses the QCMDEXC stored
* procedure to run the command
*
* @param connection
* Connection to the system
* @param command
* The CL command to run
* @exception SQLException
* If the command failed.
**/
static final void runCommand(Connection connection, String command,
boolean SQLNaming) throws SQLException {
Statement statement = null; // @scan1
try {
statement = connection.createStatement();
// We run commands via the QCMDEXC stored procedure. That procedure
// requires the length of the command be included with the command
// specified in precision 15, scale 5 format. That is,
// "CALL QSYS.QCMDEXC('command-to-run', 000000nnnn.00000)"
String commandLength = "0000000000" + command.length();
commandLength = commandLength.substring(commandLength.length() - 10)
+ ".00000";
String commandPreface;
if (SQLNaming)
commandPreface = "CALL QSYS.QCMDEXC('";
else
commandPreface = "CALL QSYS/QCMDEXC('";
String SQLCommand = commandPreface + command + "', " + commandLength
+ ")";
statement.executeUpdate(SQLCommand);
} finally // @scan1
{
if (statement != null)
statement.close();
}
}
/**
* Reads an input stream and returns its data as a byte array.
*
* @param input
* The input stream.
* @param length
* The length.
* @return The string.
* @exception SQLException
* If the length is not valid or the conversion is not possible.
**/
static final byte[] streamToBytes(InputStream input, int length)
throws SQLException {
byte[] buffer = new byte[length];
try {
byte[] rawBytes = new byte[(length == 0) ? 1 : length];
int actualLength = 0;
//
// Restructured logic to not use available ..
// @A5C
int length2 = input.read(rawBytes); /* @A5A */
while (length2 >= 0 && actualLength < length) { /* @A5C */
if (length2 > 0) { /* @A5A */
if (actualLength + length2 <= length) {
System.arraycopy(rawBytes, 0, buffer, actualLength, length2);
} else {
// copy part (if needed).
System.arraycopy(rawBytes, 0, buffer, actualLength, length
- actualLength);
}
actualLength += length2;
} /* @A5A */
length2 = input.read(rawBytes); /* @A5A */
}
// The spec says to throw an exception when the
// actual length does not match the specified length.
// I think this is strange since this means the length
// parameter is essentially not needed. I.e., we always
// read the exact number of bytes in the stream.
// @KKC if (actualLength != length)
// @KKC JDError.throwSQLException (JDError.EXC_BUFFER_LENGTH_INVALID);
// @KKC throw an exception if length is greater than the actual length
if (actualLength < length) // @KKC
JDError.throwSQLException(JDError.EXC_DATA_TYPE_MISMATCH); // @KKC
} catch (IOException e) {
JDError.throwSQLException(JDError.EXC_PARAMETER_TYPE_INVALID);
}
return buffer;
}
// @PDA jdbc40
/**
* Reads an input stream and returns its data as a byte array.
*
* @param input
* The input stream.
* @return The string.
* @exception SQLException
* If the length is not valid or the conversion is not possible.
**/
static final byte[] streamToBytes(InputStream input) throws SQLException {
// @pda copy code from native since ByteBuffer is not available on ibm java
ByteArrayOutputStream outBuffer = new ByteArrayOutputStream();
int blocksize = 4096;
byte[] buffer = new byte[blocksize];
try {
int length2 = input.read(buffer);
while (length2 >= 0) {
outBuffer.write(buffer, 0, length2);
length2 = input.read(buffer);
}
} catch (IOException e) {
JDError.throwSQLException(JDError.EXC_PARAMETER_TYPE_INVALID);
}
return outBuffer.toByteArray();
}
// @pda method from native
/**
* Reads an input stream and returns its data as a String.
*
* @param input
* The input stream.
* @param encoding
* The encoding.
* @return The string.
* @exception SQLException
* If the length is not valid or the conversion is not possible.
**/
static String streamToString(InputStream input, String encoding)
throws SQLException {
byte[] rawBytes = streamToBytes(input);
try {
return new String(rawBytes, 0, rawBytes.length, encoding);
} catch (IOException e) {
JDError.throwSQLException(JDError.EXC_PARAMETER_TYPE_INVALID);
return null;
}
}
/**
* Reads an input stream and returns its data as a String.
*
* @param input
* The input stream.
* @param length
* The length.
* @param encoding
* The encoding.
* @return The string.
* @exception SQLException
* If the length is not valid or the conversion is not possible.
**/
static final String streamToString(InputStream input, int length,
String encoding) throws SQLException {
StringBuffer buffer = new StringBuffer();
try {
byte[] rawBytes = new byte[(length == 0) ? 1 : length];
int actualLength = 0;
while (input.available() > 0) {
int length2 = input.read(rawBytes);
buffer.append(new String(rawBytes, 0, length2, encoding));
actualLength += length2;
}
// The spec says to throw an exception when the
// actual length does not match the specified length.
// I think this is strange since this means the length
// parameter is essentially not needed. I.e., we always
// read the exact number of bytes in the stream.
if (actualLength != length)
JDError.throwSQLException(JDError.EXC_BUFFER_LENGTH_INVALID);
} catch (IOException e) {
JDError.throwSQLException(JDError.EXC_PARAMETER_TYPE_INVALID);
}
return buffer.toString();
}
// @DELIMa
/**
* Uppercases the name if it's not enclosed in double-quotes.
**/
static final String upperCaseIfNotQuoted(String name) {
// 1. Unless name is delimited by outer double-quotes, uppercase the name.
if (name.startsWith("\""))
return name;
else
return name.toUpperCase();
}
// @big For 1.4 backlevel support of Decimal and Numeric types
/**
* bigDecimalToPlainString takes a big decimal and converts it into a plain
* string, without an exponent. This was the default behavior of toString
* before JDK 1.5. bigDecimalToPlainString was taken from Native driver for
* java 1.5 support and changed a bit for toolbox Coded so it will compile on
* java 1.4 also
*/
static java.lang.reflect.Method toPlainStringMethod = null;
static Object[] emptyArgs;
public static String bigDecimalToPlainString(BigDecimal bigDecimal) {
if (JVMLevel_ >= 150) {
// We compile using JDK 1.4, so we have to get to the new method via
// reflection
if (toPlainStringMethod == null) {
synchronized (bigDecimalLock_) {
if (toPlainStringMethod == null) {
try {
Class bigDecimalClass = Class.forName("java.math.BigDecimal");
Class[] parameterTypes = new Class[0];
toPlainStringMethod = bigDecimalClass.getMethod("toPlainString",
parameterTypes);
emptyArgs = new Object[0];
} catch (Exception e) {
if (JDTrace.isTraceOn()) {
JDTrace.logException(null,
"Exception while calling BigDecimal.toPlainString.", e);
}
toPlainStringMethod = null;
return bigDecimal.toString();
}
} /* if */
} /* synchronized */
} /* toPlainStringMethod == null */
String returnString;
try {
returnString = (String) toPlainStringMethod.invoke((Object) bigDecimal,
emptyArgs);
} catch (Exception e) {
if (JDTrace.isTraceOn()) {
JDTrace.logException(null,
"Exception while calling BigDecimal.toPlainString.", e);
}
returnString = bigDecimal.toString();
}
return returnString;
} else { /* not JDK15 */
return bigDecimal.toString();
}
}
// @xml3
// removes declaration (header)
// returns input string if there is no xml declaration
// @xmlNat changed to be same as native driver
static final String stripXMLDeclaration(String s) throws SQLException {
int i = 0;
int len = s.length();
while (i < len
&& (s.charAt(i) == '\ufeff' || Character.isWhitespace(s.charAt(i)))) {
i++;
}
if ((i + 1) < len && s.charAt(i) == '<' && s.charAt(i + 1) == '?') {
i += 2;
while ((i + 1) < len && (s.charAt(i) != '?' || s.charAt(i + 1) != '>')) {
i++;
}
if ((i + 1) < len && s.charAt(i) == '?' && s.charAt(i + 1) == '>') {
return s.substring(i + 2);
}
}
return s;
}
// @xmlutf8
static final boolean hasXMLDeclaration(String xml) {
if (xml.length() < 6) // @BE1
return false; // @BE1
if (xml.substring(0, 7).indexOf("");
int encStart = xml.indexOf("encoding=");
if (end != -1 && encStart != -1) {
int encEnd = xml.indexOf(" ", encStart);
if (encEnd == -1 || encEnd > end)
encEnd = end; // end of declaration with no space after encoding
return xml.substring(0, encStart) + xml.substring(encEnd);
}
}
return xml;
}
/**
* returns the type names based on the type from java.sql.Types
*
* @param typeNumber
* java.sql.Types value
* @return name corresponding to the java.sql.Types value
*/
public static String getTypeName(int typeNumber) {
switch (typeNumber) {
case Types.SMALLINT:
return "SMALLINT";
case Types.INTEGER:
return "INTEGER";
case Types.BIGINT:
return "BIGINT";
case Types.FLOAT:
return "FLOAT";
case Types.REAL:
return "REAL";
case Types.DOUBLE:
return "DOUBLE";
case Types.NUMERIC:
return "NUMERIC";
case Types.DECIMAL:
return "DECIMAL";
case -360:
return "DECFLOAT";
case Types.CHAR:
return "CHAR";
case Types.VARCHAR:
return "VARCHAR";
case Types.DATALINK:
return "DATALINK";
case Types.BINARY:
return "BINARY";
case Types.VARBINARY:
return "VARBINARY";
case Types.TIME:
return "TIME";
case Types.DATE:
return "DATE";
case Types.TIMESTAMP:
return "TIMESTAMP";
case Types.BLOB:
return "BLOB";
case Types.CLOB:
return "CLOB";
//
// New for JDK 1.6
//
case -8: /* Types.ROWID */
return "ROWID";
case -15: /* Types.NCHAR */
return "NCHAR";
case -9: /* Types.NVARCHAR */
return "NVARCHAR";
case 2011: /* Types.NCLOB */
return "NCLOB";
case -16: /* Types.LONGNVARCHAR */
return "NVARCHAR";
case 2009: /* Types.SQLXML */
return "SQLXML";
case Types.BOOLEAN: /* boolean */
return "BOOLEAN";
default:
return "UNKNOWN";
}
}
static String[][] typeNameToTypeCode = {
{ "ARRAY", "2003" }, { "BIGINT", "-5" }, { "BINARY", "-2" }, { "BIT", "-7" },
{ "BLOB", "2004" }, { "BOOLEAN", "16" }, { "CHAR", "1" },
{ "CLOB", "2005" }, { "DATALINK", "70" }, { "DATE", "91" },
{ "DBCLOB", "2005" }, { "DECIMAL", "3" }, { "DISTINCT", "2001" },
{ "DOUBLE", "8" }, { "FLOAT", "6" }, { "INTEGER", "4" },
{ "JAVA_OBJECT", "2000" }, { "LONGNVARCHAR", "-16" },
{ "LONGVARBINARY", "-4" }, { "LONGVARCHAR", "-1" }, { "NULL", "0" },
{ "NUMERIC", "2" }, { "DECFLOAT", "-360" }, { "OTHER", "1111" },
{ "REAL", "7" }, { "REF", "2006" }, { "ROWID", "-8" },
{ "SMALLINT", "5" }, { "STRUCT", "2002" }, { "TIME", "92" },
{ "TIMESTAMP", "93" }, { "TINYINT", "-6" }, { "VARBINARY", "-3" },
{ "VARCHAR", "12" },
/* ifndef JDBC40 */
{ "NCHAR", "1" }, { "GRAPHIC", "1" }, { "NCLOB", "2005" },
{ "NVARCHAR", "12" }, { "SQLXML", "2005" }, { "VARGRAPHIC", "12" },
/* endif */
/* ifdef JDBC40
{"NCHAR","-15"},
{"GRAPHIC", "-15"},
{"NCLOB","2011"},
{"NVARCHAR","-9"},
{"SQLXML","2009"},
{"VARGRAPHIC", "-9"},
endif */
};
static Hashtable typeNameHashtable = null;
public static int getTypeCode(String typeName) throws SQLException {
synchronized (typeNameToTypeCode) {
if (typeNameHashtable == null) {
typeNameHashtable = new Hashtable();
for (int i = 0; i < typeNameToTypeCode.length; i++) {
typeNameHashtable.put(typeNameToTypeCode[i][0], new Integer(
typeNameToTypeCode[i][1]));
}
}
}
Integer integer = (Integer) typeNameHashtable.get(typeName);
if (integer == null) {
if (JDTrace.isTraceOn())
JDTrace.logInformation(null, "Unable to get type from " + typeName);
JDError.throwSQLException(JDError.EXC_DATA_TYPE_MISMATCH);
return 0;
} else {
int typecode = integer.intValue();
if (typecode != 0) {
return typecode;
} else {
if (JDTrace.isTraceOn())
JDTrace.logInformation(null, "Unable to get type from " + typeName);
JDError.throwSQLException(JDError.EXC_DATA_TYPE_INVALID);
return 0;
}
}
}
public static Hashtable instanceHashtable;
public static boolean classIsInstanceOf(Class thisClass, String interfaceName) {
synchronized (typeNameToTypeCode) {
if (instanceHashtable == null) {
instanceHashtable = new Hashtable();
}
}
Hashtable interfaceHash = (Hashtable) instanceHashtable.get(thisClass);
if (interfaceHash == null) {
interfaceHash = new Hashtable();
instanceHashtable.put(thisClass, interfaceHash);
}
Boolean answer = (Boolean) interfaceHash.get(interfaceName);
if (answer == null) {
boolean booleanAnswer = false;
Class lookClass = thisClass;
while (lookClass != null && !booleanAnswer) {
if (JDTrace.isTraceOn()) {
JDTrace.logInformation(null, "JDUtilities.classIsInstance checking "
+ lookClass.getName() + " of " + thisClass.getName() + " for "
+ interfaceName);
}
if (interfaceName.equals(lookClass.getName())) {
booleanAnswer = true;
} else {
Class[] interfaces = lookClass.getInterfaces();
for (int i = 0; !booleanAnswer && i < interfaces.length; i++) {
if (JDTrace.isTraceOn()) {
JDTrace.logInformation(
null,
"DB2Utilities.classIsInstance checking "
+ interfaces[i].getName() + " of " + thisClass.getName()
+ " for " + interfaceName);
}
if (interfaces[i].getName().equals(interfaceName)) {
booleanAnswer = true;
}
}
if (!booleanAnswer) {
lookClass = lookClass.getSuperclass();
}
}
}
answer = new Boolean(booleanAnswer);
interfaceHash.put(interfaceName, answer);
}
return answer.booleanValue();
}
/**
* Remove all trailing space from a String.
*
* @param inString
* @return the updated string
*/
/* @O7A */
public static String trimTrailingSpace(String inString) {
int lastCharacterIndex = inString.length() - 1;
if (lastCharacterIndex >= 0) {
if (inString.charAt(lastCharacterIndex) == ' ') {
while ((lastCharacterIndex >= 0)
&& (inString.charAt(lastCharacterIndex) == ' ')) {
lastCharacterIndex--;
}
if (lastCharacterIndex >= 0) {
return inString.substring(0, lastCharacterIndex + 1);
} else {
return "";
}
}
}
return inString;
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy