org.firebirdsql.jdbc.FBResultSetMetaData Maven / Gradle / Ivy
Show all versions of jaybird-jdk17 Show documentation
/*
* $Id: FBResultSetMetaData.java,v 1.35 2011/12/18 13:47:17 mrotteveel Exp $
*
* Firebird Open Source J2ee connector - jdbc driver
*
* Distributable under LGPL license.
* You may obtain a copy of the License at http://www.gnu.org/copyleft/lgpl.html
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* LGPL License for more details.
*
* This file was created by members of the firebird development team.
* All individual contributions remain the Copyright (C) of those
* individuals. Contributors to this file are either listed here or
* can be obtained from a CVS history command.
*
* All rights reserved.
*/
package org.firebirdsql.jdbc;
import java.math.BigDecimal;
import java.sql.*;
import java.sql.Date;
import java.util.*;
import org.firebirdsql.encodings.EncodingFactory;
import org.firebirdsql.gds.ISCConstants;
import org.firebirdsql.gds.XSQLVAR;
import org.firebirdsql.gds.impl.GDSHelper;
/**
* Implementation of {@link java.sql.ResultSetMetaData} interface.
*
* @author David Jencks
* @version 1.0
*/
public class FBResultSetMetaData implements FirebirdResultSetMetaData {
private final XSQLVAR[] xsqlvars;
private Map extendedInfo;
private final GDSHelper connection;
/**
* Creates a new FBResultSetMetaData
instance.
*
* @param xsqlvars a XSQLVAR[]
value
* @param connection a AbstractConnection
value
* @exception SQLException if an error occurs
*
* TODO Need another constructor for metadata from constructed
* result set, where we supply the ext field info.
*/
protected FBResultSetMetaData(XSQLVAR[] xsqlvars, GDSHelper connection) throws SQLException {
this.xsqlvars = xsqlvars;
this.connection = connection;
}
private String getIscEncoding() {
if (connection != null)
return connection.getIscEncoding();
else
return "NONE";
}
/**
* Returns the number of columns in this ResultSet
object.
*
* @return the number of columns
*/
public int getColumnCount() {
return xsqlvars.length;
}
/**
* Indicates whether the designated column is automatically numbered, thus read-only.
*
* @param column the first column is 1, the second is 2, ...
* @return true
if so; false
otherwise
*/
public boolean isAutoIncrement(int column) {
return false;
}
/**
* Indicates whether a column's case matters.
*
* @param column the first column is 1, the second is 2, ...
* @return true
if so; false
otherwise
* @exception SQLException if a database access error occurs
*/
public boolean isCaseSensitive(int column) throws SQLException {
return true;
}
/**
* Indicates whether the designated column can be used in a where clause.
*
* @param column the first column is 1, the second is 2, ...
* @return true
if so; false
otherwise
* @exception SQLException if a database access error occurs
*/
public boolean isSearchable(int column) throws SQLException {
if (((getXsqlvar(column).sqltype & ~1) == ISCConstants.SQL_ARRAY)
|| ((getXsqlvar(column).sqltype & ~1) == ISCConstants.SQL_BLOB)) {
return false;
}
else {
return true;
}
}
/**
* Indicates whether the designated column is a cash value.
*
* @param column the first column is 1, the second is 2, ...
* @return true
if so; false
otherwise
* @exception SQLException if a database access error occurs
*/
public boolean isCurrency(int column) throws SQLException {
return false;
}
/**
* Indicates the nullability of values in the designated column.
*
* @param column the first column is 1, the second is 2, ...
* @return the nullability status of the given column; one of columnNoNulls
,
* columnNullable
or columnNullableUnknown
* @exception SQLException if a database access error occurs
*/
public int isNullable(int column) throws SQLException {
if ((getXsqlvar(column).sqltype & 1) == 1) {
return columnNullable;
}
else {
return columnNoNulls;
}
}
/**
* The constant indicating that a
* column does not allow NULL
values.
*/
int columnNoNulls = 0;
/**
* The constant indicating that a
* column allows NULL
values.
*/
int columnNullable = 1;
/**
* The constant indicating that the
* nullability of a column's values is unknown.
*/
int columnNullableUnknown = 2;
/**
* Indicates whether values in the designated column are signed numbers.
*
* @param column the first column is 1, the second is 2, ...
* @return true
if so; false
otherwise
* @exception SQLException if a database access error occurs
*/
public boolean isSigned(int column) throws SQLException {
switch (getXsqlvar(column).sqltype & ~1) {
case ISCConstants.SQL_SHORT:
case ISCConstants.SQL_LONG:
case ISCConstants.SQL_FLOAT:
case ISCConstants.SQL_DOUBLE:
case ISCConstants.SQL_D_FLOAT:
case ISCConstants.SQL_INT64:
return true;
default:
return false;
}
}
/**
* Indicates the designated column's normal maximum width in characters.
*
* @param column the first column is 1, the second is 2, ...
* @return the normal maximum number of characters allowed as the width
* of the designated column
* @exception SQLException if a database access error occurs
*/
public int getColumnDisplaySize(int column) throws SQLException {
int colType = getColumnType(column);
switch (colType){
case Types.DECIMAL:
case Types.NUMERIC: {
ExtendedFieldInfo fieldInfo = getExtFieldInfo(column);
if (fieldInfo == null)
return estimatePrecision(column);
else
return fieldInfo.fieldPrecision;
}
case Types.CHAR:
case Types.VARCHAR: {
XSQLVAR var = getXsqlvar(column);
int charset = var.sqlsubtype & 0xFF;
int charSetSize = charset == 127 /* CS_dynamic */ ?
EncodingFactory.getIscEncodingSize(getIscEncoding()) :
EncodingFactory.getCharacterSetSize(charset);
return var.sqllen / charSetSize;
}
case Types.FLOAT:
return 9;
case Types.DOUBLE:
return 17;
case Types.INTEGER:
return 11;
case Types.BIGINT:
return 21;
case Types.SMALLINT:
return 6;
case Types.DATE:
return 10;
case Types.TIME:
return 8;
case Types.TIMESTAMP:
return 19;
default:
return 0;
}
}
/**
* Gets the designated column's suggested title for use in printouts and
* displays.
*
* @param column the first column is 1, the second is 2, ...
* @return the suggested column title
* @exception SQLException if a database access error occurs
*/
public String getColumnLabel(int column) throws SQLException {
return (getXsqlvar(column).aliasname == null) ?
(getXsqlvar(column).sqlname != null ? getXsqlvar(column).sqlname : "") :
getXsqlvar(column).aliasname;
}
/**
* Get the designated column's name.
*
* @param column the first column is 1, the second is 2, ...
* @return column name
* @exception SQLException if a database access error occurs
*/
public String getColumnName(int column) throws SQLException {
if (getXsqlvar(column).sqlname == null)
return getColumnLabel(column);
else
return getXsqlvar(column).sqlname;
}
public String getSourceColumnName(int column) throws SQLException {
String result = getXsqlvar(column).sqlname;
if (result == null)
result = "";
return result;
}
/**
* Get the designated column's table's schema.
*
* @param column the first column is 1, the second is 2, ...
* @return schema name or "" if not applicable
* @exception SQLException if a database access error occurs
*/
public String getSchemaName(int column) throws SQLException {
//not really implemented
return "";
}
/**
* Get the designated column's number of decimal digits.
*
* @param column the first column is 1, the second is 2, ...
* @return precision
* @exception SQLException if a database access error occurs
*/
public int getPrecision(int column) throws SQLException {
int colType = getColumnType(column);
switch (colType){
case Types.DECIMAL:
case Types.NUMERIC: {
ExtendedFieldInfo fieldInfo = getExtFieldInfo(column);
if (fieldInfo == null)
return estimatePrecision(column);
else
return fieldInfo.fieldPrecision;
}
case Types.CHAR:
case Types.VARCHAR: {
XSQLVAR var = getXsqlvar(column);
int charset = var.sqlsubtype & 0xFF;
int charSetSize = charset == 127 /* CS_dynamic */ ?
EncodingFactory.getIscEncodingSize(getIscEncoding()) :
EncodingFactory.getCharacterSetSize(charset);
return var.sqllen / charSetSize;
}
case Types.FLOAT:
return 7;
case Types.DOUBLE:
return 15;
case Types.INTEGER:
return 10;
case Types.BIGINT:
return 20;
case Types.SMALLINT:
return 5;
case Types.DATE:
return 10;
case Types.TIME:
return 8;
case Types.TIMESTAMP:
return 19;
default:
return 0;
}
}
/**
* Gets the designated column's number of digits to right of the decimal point.
*
* @param column the first column is 1, the second is 2, ...
* @return scale
* @exception SQLException if a database access error occurs
*/
public int getScale(int column) throws SQLException {
return getXsqlvar(column).sqlscale * (-1);
}
/**
* Gets the designated column's table name.
*
* @param column the first column is 1, the second is 2, ...
* @return table name or "" if not applicable
* @exception SQLException if a database access error occurs
*/
public String getTableName(int column) throws SQLException {
String result = getXsqlvar(column).relname;
if (result == null) result = "";
return result;
}
/**
* Gets the designated column's table alias.
*
* @param column the first column is 1, the second is 2, ...
* @return table alias or "" if not applicable
* @exception SQLException if a database access error occurs
*/
public String getTableAlias(int column) throws SQLException {
String result = getXsqlvar(column).relaliasname;
if (result == null) result = getTableName(column);
return result;
}
/**
* Gets the designated column's table's catalog name.
*
* @param column the first column is 1, the second is 2, ...
* @return column name or "" if not applicable
* @exception SQLException if a database access error occurs
*/
public String getCatalogName(int column) throws SQLException {
return "";
}
/**
* Retrieves the designated column's SQL type.
*
* @param column the first column is 1, the second is 2, ...
* @return SQL type from java.sql.Types
* @exception SQLException if a database access error occurs
* @see Types
*/
public int getColumnType(int column) throws SQLException {
int sqltype = getXsqlvar(column).sqltype & ~1;
int sqlscale = getXsqlvar(column).sqlscale;
int sqlsubtype = getXsqlvar(column).sqlsubtype;
if (sqlscale < 0) {
switch (sqltype) {
case ISCConstants.SQL_SHORT:
case ISCConstants.SQL_LONG:
case ISCConstants.SQL_INT64:
case ISCConstants.SQL_DOUBLE:
// NOTE: can't be BIGINT because of scale
if (sqlsubtype == 2)
return Types.DECIMAL;
else
return Types.NUMERIC;
default:
break;
}
}
switch (sqltype) {
case ISCConstants.SQL_SHORT:
return Types.SMALLINT;
case ISCConstants.SQL_LONG:
return Types.INTEGER;
case ISCConstants.SQL_DOUBLE:
case ISCConstants.SQL_D_FLOAT:
return Types.DOUBLE;
case ISCConstants.SQL_FLOAT:
return Types.FLOAT;
case ISCConstants.SQL_TEXT:
return Types.CHAR;
case ISCConstants.SQL_VARYING:
return Types.VARCHAR;
case ISCConstants.SQL_TIMESTAMP:
return Types.TIMESTAMP;
case ISCConstants.SQL_TYPE_TIME:
return Types.TIME;
case ISCConstants.SQL_TYPE_DATE:
return Types.DATE;
case ISCConstants.SQL_INT64:
if (sqlsubtype == 1)
return Types.NUMERIC;
else if (sqlsubtype == 2)
return Types.DECIMAL;
else
return Types.BIGINT;
case ISCConstants.SQL_BLOB:
if (sqlsubtype < 0)
return Types.BLOB;
else if (sqlsubtype == 0 || sqlsubtype > 1)
return Types.LONGVARBINARY;
else if (sqlsubtype == 1)
return Types.LONGVARCHAR;
else
return Types.OTHER;
case ISCConstants.SQL_QUAD:
return Types.OTHER;
default:
return Types.NULL;
}
}
/**
* Retrieves the designated column's database-specific type name.
*
* @param column the first column is 1, the second is 2, ...
* @return type name used by the database. If the column type is
* a user-defined type, then a fully-qualified type name is returned.
* @exception SQLException if a database access error occurs
*/
public String getColumnTypeName(int column) throws SQLException {
// Must return the same value as DatabaseMetaData getColumns Type_Name
int sqltype = getXsqlvar(column).sqltype & ~1;
int sqlscale = getXsqlvar(column).sqlscale;
int sqlsubtype = getXsqlvar(column).sqlsubtype;
if (sqlscale < 0) {
switch (sqltype) {
case ISCConstants.SQL_SHORT:
case ISCConstants.SQL_LONG:
case ISCConstants.SQL_INT64:
case ISCConstants.SQL_DOUBLE:
// NOTE: can't be BIGINT because of scale
if (sqlsubtype == 2)
return "DECIMAL";
else
return "NUMERIC";
default:
break;
}
}
switch (sqltype) {
case ISCConstants.SQL_SHORT:
return "SMALLINT";
case ISCConstants.SQL_LONG:
return "INTEGER";
case ISCConstants.SQL_DOUBLE:
case ISCConstants.SQL_D_FLOAT:
return "DOUBLE PRECISION";
case ISCConstants.SQL_FLOAT:
return "FLOAT";
case ISCConstants.SQL_TEXT:
return "CHAR";
case ISCConstants.SQL_VARYING:
return "VARCHAR";
case ISCConstants.SQL_TIMESTAMP:
return "TIMESTAMP";
case ISCConstants.SQL_TYPE_TIME:
return "TIME";
case ISCConstants.SQL_TYPE_DATE:
return "DATE";
case ISCConstants.SQL_INT64:
//this might need some help for long mapping
if (sqlsubtype == 1)
return "NUMERIC";
else if (sqlsubtype == 2)
return "DECIMAL";
else
return "BIGINT";
case ISCConstants.SQL_BLOB:
if (sqlsubtype < 0)
return "BLOB SUB_TYPE <0";
else if (sqlsubtype == 0)
return "BLOB SUB_TYPE 0";
else if (sqlsubtype == 1)
return "BLOB SUB_TYPE 1";
else
return "BLOB SUB_TYPE " + sqlsubtype;
case ISCConstants.SQL_QUAD:
return "ARRAY";
default:
return "NULL";
}
}
/**
* Indicates whether the designated column is definitely not writable.
*
* @param column the first column is 1, the second is 2, ...
* @return true
if so; false
otherwise
* @exception SQLException if a database access error occurs
*/
public boolean isReadOnly(int column) throws SQLException {
//Need to consider priveleges!!
return false;
}
/**
* Indicates whether it is possible for a write on the designated column to succeed.
*
* @param column the first column is 1, the second is 2, ...
* @return true
if so; false
otherwise
* @exception SQLException if a database access error occurs
*/
public boolean isWritable(int column) throws SQLException {
//Needs priveleges???
return true;
}
/**
* Indicates whether a write on the designated column will definitely succeed.
*
* @param column the first column is 1, the second is 2, ...
* @return true
if so; false
otherwise
* @exception SQLException if a database access error occurs
*/
public boolean isDefinitelyWritable(int column) throws SQLException {
//Need to consider privileges!!!
return true;
}
//--------------------------JDBC 2.0-----------------------------------
/**
* Returns the fully-qualified name of the Java class whose instances
* are manufactured if the method ResultSet.getObject
* is called to retrieve a value
* from the column. ResultSet.getObject
may return a subclass of the
* class returned by this method.
*
* @return the fully-qualified name of the class in the Java programming
* language that would be used by the method
* ResultSet.getObject
to retrieve the value in the specified
* column. This is the class name used for custom mapping.
* @exception SQLException if a database access error occurs
* @since 1.2
* @see What Is in the JDBC
* 2.0 API
*/
public String getColumnClassName(int column) throws SQLException {
switch (getXsqlvar(column).sqltype & ~1) {
case ISCConstants.SQL_TEXT:
case ISCConstants.SQL_VARYING:
return String.class.getName();
case ISCConstants.SQL_SHORT:
case ISCConstants.SQL_LONG:
return Integer.class.getName();
case ISCConstants.SQL_FLOAT:
case ISCConstants.SQL_DOUBLE:
case ISCConstants.SQL_D_FLOAT:
return Double.class.getName();
case ISCConstants.SQL_TIMESTAMP:
return Timestamp.class.getName();
case ISCConstants.SQL_BLOB:
XSQLVAR field = getXsqlvar(column);
if (field.sqlsubtype < 0)
return Blob.class.getName();
if (field.sqlsubtype == 1)
return String.class.getName();
else
return byte[].class.getName();
case ISCConstants.SQL_ARRAY:
return Array.class.getName();
case ISCConstants.SQL_QUAD:
return Long.class.getName();
case ISCConstants.SQL_TYPE_TIME:
return Time.class.getName();
case ISCConstants.SQL_TYPE_DATE:
return Date.class.getName();
case ISCConstants.SQL_INT64:
if (getXsqlvar(column).sqlscale == 0) {
return Long.class.getName();
}
else {
return BigDecimal.class.getName();
}
default:
throw new FBSQLException("Unkown SQL type.",
FBSQLException.SQL_STATE_INVALID_PARAM_TYPE);
}
}
//private methods
private XSQLVAR getXsqlvar(int columnIndex) {
//return stmt.getOutSqlda().sqlvar[columnIndex - 1];
return xsqlvars[columnIndex - 1];
}
private ExtendedFieldInfo getExtFieldInfo(int columnIndex)
throws SQLException
{
if (extendedInfo == null)
{
this.extendedInfo = getExtendedFieldInfo(connection);
} // end of if ()
FieldKey key = new FieldKey(
getXsqlvar(columnIndex).relname,
getXsqlvar(columnIndex).sqlname);
return (ExtendedFieldInfo)extendedInfo.get(key);
}
private int estimatePrecision(int columnIndex) {
int sqltype = getXsqlvar(columnIndex).sqltype & ~1;
// TODO: Why unused?
int sqlscale = getXsqlvar(columnIndex).sqlscale;
switch(sqltype) {
case ISCConstants.SQL_SHORT : return 5;
case ISCConstants.SQL_LONG : return 10;
case ISCConstants.SQL_INT64 : return 19;
case ISCConstants.SQL_DOUBLE : return 19;
default : return 0;
}
}
private static final String GET_FIELD_INFO = "SELECT "
+ " RF.RDB$RELATION_NAME as RELATION_NAME"
+ ", RF.RDB$FIELD_NAME as FIELD_NAME"
+ ", F.RDB$FIELD_LENGTH as FIELD_LENGTH"
+ ", F.RDB$FIELD_PRECISION as FIELD_PRECISION"
+ ", F.RDB$FIELD_SCALE as FIELD_SCALE"
+ ", F.RDB$FIELD_SUB_TYPE as FIELD_SUB_TYPE"
+ ", F.RDB$CHARACTER_LENGTH as CHAR_LEN"
+ ", F.RDB$CHARACTER_SET_ID as CHARACTER_SET_ID"
+ " FROM"
+ " RDB$RELATION_FIELDS RF "
+ ", RDB$FIELDS F "
+ " WHERE "
+ " RF.RDB$FIELD_SOURCE = F.RDB$FIELD_NAME"
+ " AND"
+ " RF.RDB$FIELD_NAME = ?"
+ " AND"
+ " RF.RDB$RELATION_NAME = ?"
;
/**
* This class is an old-fashion data structure that stores additional
* information about fields in a database.
*/
private static class ExtendedFieldInfo {
// TODO Find out why there are unused fields here
String relationName;
String fieldName;
int fieldLength;
int fieldPrecision;
int fieldScale;
int fieldSubtype;
int characterLength;
int characterSetId;
}
/**
* This class should be used as a composite key in an internal field
* mapping structures.
*/
private static final class FieldKey {
private String relationName;
private String fieldName;
/**
* Create instance of this class for the specified relation and field
* names.
*
* @param relationName relation name.
* @param fieldName field name.
*/
FieldKey(String relationName, String fieldName) {
this.relationName = relationName;
this.fieldName = fieldName;
}
/**
* Check if obj
is equal to this object.
*
* @param obj object to check.
*
* @return true
if obj
is instance of this
* class and has equal relation and field names.
*/
public boolean equals(Object obj) {
if (obj == this) return true;
if (obj == null || !(obj instanceof FieldKey)) return false;
FieldKey that = (FieldKey)obj;
return (relationName != null ? relationName.equals(that.relationName) : that.relationName == null)
&&
(fieldName != null ? fieldName.equals(that.fieldName) : that.fieldName == null);
}
/**
* Get hash code of this instance.
*
* @return combination of hash codes of relationName
field
* and fieldName
field.
*/
public int hashCode() {
int result = 971;
result = 23 * result + (relationName != null ? relationName.hashCode() : 0);
result = 23 * result + (fieldName != null ? fieldName.hashCode() : 0);
return result;
}
}
/**
* This method retrieves extended information from the system tables in
* a database. Since this method is expensinve, use it with care.
*
* @return mapping between {@link FieldKey} instances and
* {@link ExtendedFieldInfo} instances.
*
* @throws SQLException if extended field information cannot be obtained.
*/
private Map getExtendedFieldInfo(GDSHelper gdsHelper) throws SQLException {
if (gdsHelper == null) return Collections.EMPTY_MAP;
//
// Apparently there is a limit in the UNION
// It is necesary to split in several querys
// Although the problem reported with 93 UNION use only 70
//
int pending = xsqlvars.length;
HashMap result = new HashMap();
while (pending > 0){
StringBuffer sb = new StringBuffer();
ArrayList params = new ArrayList();
int maxLength = (pending>70) ? 70 : pending;
for (int i = 0; i < maxLength; i++) {
String relationName = xsqlvars[i].relname;
String fieldName = xsqlvars[i].sqlname;
if (relationName == null || fieldName == null) continue;
sb.append(GET_FIELD_INFO);
params.add(fieldName);
params.add(relationName);
if (i < maxLength - 1)
sb.append("\n").append("UNION").append("\n");
}
FBDatabaseMetaData metaData = new FBDatabaseMetaData(gdsHelper);
ResultSet rs = metaData.doQuery(sb.toString(), params);
try {
while(rs.next()) {
ExtendedFieldInfo fieldInfo = new ExtendedFieldInfo();
fieldInfo.relationName = rs.getString("RELATION_NAME");
fieldInfo.fieldName = rs.getString("FIELD_NAME");
fieldInfo.fieldLength = rs.getInt("FIELD_LENGTH");
fieldInfo.fieldPrecision = rs.getInt("FIELD_PRECISION");
fieldInfo.fieldScale = rs.getInt("FIELD_SCALE");
fieldInfo.fieldSubtype = rs.getInt("FIELD_SUB_TYPE");
fieldInfo.characterSetId = rs.getInt("CHARACTER_SET_ID");
fieldInfo.characterLength = rs.getInt("CHAR_LEN");
if (rs.wasNull())
fieldInfo.characterLength =
fieldInfo.fieldLength /
EncodingFactory.getCharacterSetSize(fieldInfo.characterSetId);
result.put(
new FieldKey(fieldInfo.relationName, fieldInfo.fieldName),
fieldInfo);
}
} finally {
rs.close();
}
pending -= maxLength;
}
return result;
}
public boolean isWrapperFor(Class> iface) throws SQLException {
return iface != null && iface.isAssignableFrom(getClass());
}
public T unwrap(Class iface) throws SQLException {
if (!isWrapperFor(iface))
throw new FBDriverNotCapableException();
return iface.cast(this);
}
}