org.firebirdsql.jdbc.AbstractFieldMetaData Maven / Gradle / Ivy
/*
* Firebird Open Source JavaEE 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 source control history command.
*
* All rights reserved.
*/
package org.firebirdsql.jdbc;
import org.firebirdsql.encodings.EncodingDefinition;
import org.firebirdsql.gds.ISCConstants;
import org.firebirdsql.gds.ng.fields.FieldDescriptor;
import org.firebirdsql.gds.ng.fields.RowDescriptor;
import org.firebirdsql.jdbc.field.JdbcTypeConverter;
import java.sql.SQLException;
import java.sql.Types;
import java.sql.Wrapper;
import java.util.Map;
import static org.firebirdsql.jdbc.JavaTypeNameConstants.*;
import static org.firebirdsql.util.FirebirdSupportInfo.supportInfoFor;
/**
* Base class for {@link org.firebirdsql.jdbc.FBResultSetMetaData} and
* {@link org.firebirdsql.jdbc.FBParameterMetaData} for methods common to both implementations.
*
* @author David Jencks
* @author Nickolay Samofatov
* @author Mark Rotteveel
* @since 3.0
*/
public abstract class AbstractFieldMetaData implements Wrapper {
private static final int SUBTYPE_NUMERIC = 1;
private static final int SUBTYPE_DECIMAL = 2;
private final RowDescriptor rowDescriptor;
private final FBConnection connection;
private Map extendedInfo;
protected AbstractFieldMetaData(RowDescriptor rowDescriptor, FBConnection connection) {
assert rowDescriptor != null : "rowDescriptor is required";
this.rowDescriptor = rowDescriptor;
this.connection = connection;
}
@Override
public boolean isWrapperFor(Class> iface) throws SQLException {
return iface != null && iface.isAssignableFrom(getClass());
}
@Override
public T unwrap(Class iface) throws SQLException {
if (!isWrapperFor(iface))
throw new SQLException("Unable to unwrap to class " + iface.getName());
return iface.cast(this);
}
/**
* @return The row descriptor.
*/
protected final RowDescriptor getRowDescriptor() {
return rowDescriptor;
}
/**
* Retrieves the number of fields in the object for which this AbstractFieldMetaData
object contains
* information.
*
* @return the number of fields
*/
protected final int getFieldCount() {
return rowDescriptor.getCount();
}
/**
* The {@link org.firebirdsql.gds.ng.fields.FieldDescriptor} of the field with index fieldIndex
.
*
* @param fieldIndex
* 1-based index of a field in this metadata object
* @return field descriptor
*/
protected final FieldDescriptor getFieldDescriptor(int fieldIndex) {
return rowDescriptor.getFieldDescriptor(fieldIndex - 1);
}
/**
* Retrieves whether values for the designated field can be signed numbers.
*
* @param field
* the first field is 1, the second is 2, ...
* @return true
if so; false
otherwise
*/
protected final boolean isSignedInternal(int field) {
switch (getFieldDescriptor(field).getType() & ~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:
case ISCConstants.SQL_DEC16:
case ISCConstants.SQL_DEC34:
case ISCConstants.SQL_INT128:
return true;
default:
return false;
}
}
/**
* Retrieves the designated field's number of digits to right of the decimal point.
* 0 is returned for data types where the scale is not applicable.
*
* @param field
* the first field is 1, the second is 2, ...
* @return scale
*/
protected final int getScaleInternal(int field) {
return getFieldDescriptor(field).getScale() * (-1);
}
protected final String getFieldClassName(int field) throws SQLException {
switch (getFieldType(field)) {
case Types.CHAR:
case Types.VARCHAR:
case Types.LONGVARCHAR:
return STRING_CLASS_NAME;
case Types.SMALLINT:
case Types.INTEGER:
return INTEGER_CLASS_NAME;
case Types.FLOAT:
case Types.DOUBLE:
return DOUBLE_CLASS_NAME;
case Types.TIMESTAMP:
return TIMESTAMP_CLASS_NAME;
case Types.BLOB:
return BLOB_CLASS_NAME;
case Types.CLOB:
return CLOB_CLASS_NAME;
case Types.BINARY:
case Types.VARBINARY:
case Types.LONGVARBINARY:
return BYTE_ARRAY_CLASS_NAME;
case Types.ARRAY:
return ARRAY_CLASS_NAME;
case Types.BIGINT:
return LONG_CLASS_NAME;
case Types.TIME:
return TIME_CLASS_NAME;
case Types.DATE:
return SQL_DATE_CLASS_NAME;
case JaybirdTypeCodes.TIME_WITH_TIMEZONE:
return OFFSET_TIME_CLASS_NAME;
case JaybirdTypeCodes.TIMESTAMP_WITH_TIMEZONE:
return OFFSET_DATE_TIME_CLASS_NAME;
case Types.NUMERIC:
case Types.DECIMAL:
case JaybirdTypeCodes.DECFLOAT:
return BIG_DECIMAL_CLASS_NAME;
case Types.BOOLEAN:
return BOOLEAN_CLASS_NAME;
case Types.NULL:
case Types.OTHER:
return OBJECT_CLASS_NAME;
case Types.ROWID:
return ROW_ID_CLASS_NAME;
default:
throw new FBSQLException("Unknown SQL type.", SQLStateConstants.SQL_STATE_INVALID_PARAM_TYPE);
}
}
protected final String getFieldTypeName(int field) {
// Must return the same value as DatabaseMetaData getColumns Type_Name
// TODO Reduce duplication with FBDatabaseMetaData
int sqlType = getFieldDescriptor(field).getType() & ~1;
int sqlScale = getFieldDescriptor(field).getScale();
int sqlSubtype = getFieldDescriptor(field).getSubType();
switch (sqlType) {
case ISCConstants.SQL_SHORT:
if (sqlSubtype == SUBTYPE_NUMERIC || (sqlSubtype == 0 && sqlScale < 0)) {
return "NUMERIC";
} else if (sqlSubtype == SUBTYPE_DECIMAL) {
return "DECIMAL";
} else {
return "SMALLINT";
}
case ISCConstants.SQL_LONG:
if (sqlSubtype == SUBTYPE_NUMERIC || (sqlSubtype == 0 && sqlScale < 0)) {
return "NUMERIC";
} else if (sqlSubtype == SUBTYPE_DECIMAL) {
return "DECIMAL";
} else {
return "INTEGER";
}
case ISCConstants.SQL_INT64:
if (sqlSubtype == SUBTYPE_NUMERIC || (sqlSubtype == 0 && sqlScale < 0)) {
return "NUMERIC";
} else if (sqlSubtype == SUBTYPE_DECIMAL) {
return "DECIMAL";
} else {
return "BIGINT";
}
case ISCConstants.SQL_DOUBLE:
case ISCConstants.SQL_D_FLOAT:
if (sqlSubtype == SUBTYPE_NUMERIC || (sqlSubtype == 0 && sqlScale < 0)) {
return "NUMERIC";
} else if (sqlSubtype == SUBTYPE_DECIMAL) {
return "DECIMAL";
} else {
return "DOUBLE PRECISION";
}
case ISCConstants.SQL_DEC16:
case ISCConstants.SQL_DEC34:
return "DECFLOAT";
case ISCConstants.SQL_INT128:
if (sqlSubtype == SUBTYPE_NUMERIC) {
return "NUMERIC";
} else {
return "DECIMAL";
}
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_TIMESTAMP_TZ:
case ISCConstants.SQL_TIMESTAMP_TZ_EX:
return "TIMESTAMP WITH TIME ZONE";
case ISCConstants.SQL_TIME_TZ:
case ISCConstants.SQL_TIME_TZ_EX:
return "TIME WITH TIME ZONE";
case ISCConstants.SQL_BLOB:
if (sqlSubtype < 0) {
return "BLOB SUB_TYPE <0"; // TODO report actual subtype
} else if (sqlSubtype == ISCConstants.BLOB_SUB_TYPE_BINARY) {
return "BLOB SUB_TYPE 0";
} else if (sqlSubtype == ISCConstants.BLOB_SUB_TYPE_TEXT) {
return "BLOB SUB_TYPE 1";
} else {
return "BLOB SUB_TYPE " + sqlSubtype;
}
case ISCConstants.SQL_QUAD:
return "ARRAY"; // TODO Inconsistent with getFieldType
case ISCConstants.SQL_BOOLEAN:
return "BOOLEAN";
default:
return "NULL";
}
}
protected int getFieldType(int field) {
return JdbcTypeConverter.toJdbcType(getFieldDescriptor(field));
}
/**
* Retrieves the designated parameter's specified column size.
*
* The returned value represents the maximum column size for the given parameter.
* For numeric data, this is the maximum precision. For character data, this is the length in characters.
* For datetime datatypes, this is the length in characters of the String representation (assuming the
* maximum allowed precision of the fractional seconds component). For binary data, this is the length in bytes.
* For the ROWID datatype, this is the length in bytes. 0 is returned for data types where the column size is not
* applicable.
*
* @param field
* the first field is 1, the second is 2, ...
* @return precision
* @throws SQLException
* if a database access error occurs
*/
protected final int getPrecisionInternal(int field) throws SQLException {
final int colType = getFieldType(field);
switch (colType) {
case Types.DECIMAL:
case Types.NUMERIC: {
final ExtendedFieldInfo fieldInfo = getExtFieldInfo(field);
return fieldInfo == null || fieldInfo.fieldPrecision == 0
? estimateFixedPrecision(field)
: fieldInfo.fieldPrecision;
}
case JaybirdTypeCodes.DECFLOAT: {
final FieldDescriptor var = getFieldDescriptor(field);
switch (var.getType() & ~1) {
case ISCConstants.SQL_DEC16:
return 16;
case ISCConstants.SQL_DEC34:
return 34;
default:
return 0;
}
}
case Types.CHAR:
case Types.VARCHAR: {
final FieldDescriptor var = getFieldDescriptor(field);
final EncodingDefinition encodingDefinition =
var.getEncodingFactory().getEncodingDefinitionByCharacterSetId(var.getSubType());
final int charSetSize = encodingDefinition != null ? encodingDefinition.getMaxBytesPerChar() : 1;
return var.getLength() / charSetSize;
}
case Types.BINARY:
case Types.VARBINARY: {
final FieldDescriptor var = getFieldDescriptor(field);
return var.getLength();
}
case Types.FLOAT: {
if (supportInfoFor(connection).supportsFloatBinaryPrecision()) {
return 24;
} else {
return 7;
}
}
case Types.DOUBLE: {
if (supportInfoFor(connection).supportsFloatBinaryPrecision()) {
return 53;
} else {
return 15;
}
}
case Types.INTEGER:
return 10;
case Types.BIGINT:
return 19;
case Types.SMALLINT:
return 5;
case Types.DATE:
return 10;
case Types.TIME:
return 8;
case Types.TIMESTAMP:
return 19;
case JaybirdTypeCodes.TIMESTAMP_WITH_TIMEZONE:
return 30;
case JaybirdTypeCodes.TIME_WITH_TIMEZONE:
return 19;
case Types.BOOLEAN:
return 1;
default:
return 0;
}
}
protected final int estimateFixedPrecision(int fieldIndex) {
final int sqltype = getFieldDescriptor(fieldIndex).getType() & ~1;
switch (sqltype) {
case ISCConstants.SQL_SHORT:
return 4;
case ISCConstants.SQL_LONG:
return 9;
case ISCConstants.SQL_INT64:
return 18;
case ISCConstants.SQL_DOUBLE:
return 18;
case ISCConstants.SQL_DEC16:
return 16;
case ISCConstants.SQL_DEC34:
return 34;
case ISCConstants.SQL_INT128:
return 38;
default:
return 0;
}
}
protected final ExtendedFieldInfo getExtFieldInfo(int columnIndex) throws SQLException {
if (extendedInfo == null) {
extendedInfo = getExtendedFieldInfo(connection);
}
FieldKey key = new FieldKey(
getFieldDescriptor(columnIndex).getOriginalTableName(),
getFieldDescriptor(columnIndex).getOriginalName());
return extendedInfo.get(key);
}
/**
* This method retrieves extended information from the system tables in
* a database. Since this method is expensive, use it with care.
*
* @return mapping between {@link FieldKey} instances and {@link ExtendedFieldInfo} instances,
* or an empty Map if the metadata implementation does not support extended info.
* @throws SQLException
* if a database error occurs while obtaining extended field information.
*/
protected abstract Map getExtendedFieldInfo(FBConnection connection) throws SQLException;
/**
* This class is an old-fashion data structure that stores additional
* information about fields in a database.
*/
protected static class ExtendedFieldInfo {
final FieldKey fieldKey;
final int fieldPrecision;
public ExtendedFieldInfo(String relationName, String fieldName, int precision) {
fieldKey = new FieldKey(relationName, fieldName);
fieldPrecision = precision;
}
}
/**
* This class should be used as a composite key in an internal field
* mapping structures.
*/
protected static final class FieldKey {
private final String relationName;
private final 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 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;
}
}
}