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

com.impossibl.postgres.jdbc.PGResultSetMetaData Maven / Gradle / Ivy

The newest version!
/**
 * Copyright (c) 2013, impossibl.com
 * All rights reserved.
 *
 * Redistribution and use in source and binary forms, with or without
 * modification, are permitted provided that the following conditions are met:
 *
 *  * Redistributions of source code must retain the above copyright notice,
 *    this list of conditions and the following disclaimer.
 *  * Redistributions in binary form must reproduce the above copyright
 *    notice, this list of conditions and the following disclaimer in the
 *    documentation and/or other materials provided with the distribution.
 *  * Neither the name of impossibl.com nor the names of its contributors may
 *    be used to endorse or promote products derived from this software
 *    without specific prior written permission.
 *
 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
 * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
 * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE
 * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
 * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
 * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
 * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
 * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
 * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
 * POSSIBILITY OF SUCH DAMAGE.
 */
package com.impossibl.postgres.jdbc;

import com.impossibl.postgres.protocol.ResultField;
import com.impossibl.postgres.types.CompositeType;
import com.impossibl.postgres.types.Registry;
import com.impossibl.postgres.types.Type;

import static com.impossibl.postgres.jdbc.ErrorUtils.makeSQLException;
import static com.impossibl.postgres.jdbc.Exceptions.COLUMN_INDEX_OUT_OF_BOUNDS;
import static com.impossibl.postgres.jdbc.Exceptions.UNWRAP_ERROR;
import static com.impossibl.postgres.system.CustomTypes.lookupCustomType;
import static com.impossibl.postgres.system.SystemSettings.DATABASE_NAME;

import java.io.IOException;
import java.sql.ResultSetMetaData;
import java.sql.SQLException;
import java.util.Arrays;
import java.util.HashMap;
import java.util.List;
import java.util.Map;

import static java.util.Collections.emptyList;
import static java.util.Collections.singletonList;

class PGResultSetMetaData extends PGMetaData implements ResultSetMetaData {

  private ResultField[] resultFields;
  private int resultFieldsHash;
  private Map> relationsColumnsData;
  private Map> typeMap;
  private int typeMapHash;

  PGResultSetMetaData(PGDirectConnection connection, ResultField[] resultFields, Map> typeMap) {
    super(connection);
    this.resultFields = resultFields;
    this.resultFieldsHash = Arrays.hashCode(resultFields);
    this.relationsColumnsData = new HashMap<>();
    this.typeMap = typeMap;
    this.typeMapHash = typeMap.hashCode();
  }

  boolean matches(ResultField[] resultFields, Map> typeMap) {
    return this.resultFieldsHash == Arrays.hashCode(resultFields) && this.typeMapHash == typeMap.hashCode();
  }

  private ColumnData getRelationColumnData(ResultField field) throws SQLException {
    List relationColumnsData = getRelationColumnsData(field.getRelationId());
    for (ColumnData relationColumnData : relationColumnsData) {
      if (relationColumnData.relationAttributeNumber == field.getRelationAttributeNumber()) {
        return relationColumnData;
      }
    }
    return null;
  }

  private List getRelationColumnsData(int relationId) throws SQLException {
    if (relationId == 0) return emptyList();

    List data;
    if ((data = relationsColumnsData.get(relationId)) == null) {
      data = loadRelationColumsData(relationId);
      relationsColumnsData.put(relationId, data);
    }
    return data;
  }

  private List loadRelationColumsData(int relationId) throws SQLException {
    String sql = getColumnSQL(" AND c.oid = ?").toString();

    return getColumnData(sql, singletonList(relationId));
  }

  /**
   * Returns the ResultField associated with the requested column
   *
   * @param columnIndex Requested column index
   * @return ResultField of column
   * @throws SQLException If columnIndex is out of bounds
   */
  ResultField get(int columnIndex) throws SQLException {

    if (columnIndex < 1 || columnIndex > resultFields.length) {
      throw COLUMN_INDEX_OUT_OF_BOUNDS;
    }

    return resultFields[columnIndex - 1];
  }

  /**
   * Returns the CompositeType representing the requested column's table.
   *
   * @param columnIndex Requested column index
   * @return CompositeType of columns table
   * @throws SQLException If columnIndex is out of bounds
   */
  private CompositeType getRelType(int columnIndex) throws SQLException {

    ResultField field = get(columnIndex);
    if (field.getRelationId() == 0)
      return null;

    try {
      return connection.getRegistry().loadRelationType(field.getRelationId());
    }
    catch (IOException e) {
      throw makeSQLException(e);
    }
  }

  @Override
  public  T unwrap(Class iface) throws SQLException {
    if (!iface.isAssignableFrom(getClass())) {
      throw UNWRAP_ERROR;
    }

    return iface.cast(this);
  }

  @Override
  public boolean isWrapperFor(Class iface) {
    return iface.isAssignableFrom(getClass());
  }

  @Override
  public int getColumnCount() {
    return resultFields.length;
  }

  @Override
  public boolean isAutoIncrement(int column) throws SQLException {

    ResultField field = get(column);
    ColumnData columnData = getRelationColumnData(field);
    if (columnData == null) {
      try {
        return connection.getRegistry().resolve(field.getTypeRef()).isAutoIncrement();
      }
      catch (IOException e) {
        throw makeSQLException(e);
      }
    }

    return Type.isAutoIncrement(columnData.defaultValue) || columnData.type.isAutoIncrement();
  }

  @Override
  public boolean isCaseSensitive(int column) throws SQLException {

    Registry registry = connection.getRegistry();

    try {
      return JDBCTypeMetaData.isCaseSensitive(registry.resolve(get(column).getTypeRef()));
    }
    catch (IOException e) {
      throw makeSQLException(e);
    }
  }

  @Override
  public boolean isSearchable(int column) {
    return true;
  }

  @Override
  public boolean isCurrency(int column) throws SQLException {

    try {
      return JDBCTypeMetaData.isCurrency(connection.getRegistry().resolve(get(column).getTypeRef()));
    }
    catch (IOException e) {
      throw makeSQLException(e);
    }
  }

  @Override
  public int isNullable(int column) throws SQLException {

    ResultField field = get(column);
    ColumnData columnData = getRelationColumnData(field);

    Boolean nullable;
    if (columnData == null) {
      try {
        nullable = connection.getRegistry().resolve(field.getTypeRef()).isNullable();
      }
      catch (IOException e) {
        throw makeSQLException(e);
      }
    }
    else {
      nullable = columnData.nullable != null ? columnData.nullable : columnData.type.isNullable();
    }

    return nullable != null ? (nullable ? columnNullable : columnNoNulls) : columnNullableUnknown;
  }

  @Override
  public boolean isSigned(int column) throws SQLException {

    try {
      return JDBCTypeMetaData.isSigned(connection.getRegistry().resolve(get(column).getTypeRef()));
    }
    catch (IOException e) {
      throw makeSQLException(e);
    }
  }

  @Override
  public String getColumnLabel(int column) throws SQLException {

    String val = get(column).getName();
    if (val == null)
      val = getColumnName(column);

    return val;
  }

  @Override
  public String getCatalogName(int column) {

    return connection.getSetting(DATABASE_NAME);
  }

  @Override
  public String getSchemaName(int column) throws SQLException {

    Type relType = getRelType(column);
    if (relType == null)
      return "";

    return relType.getNamespace();
  }

  @Override
  public String getTableName(int column) throws SQLException {

    //Note: there seems to be some debate about whether this should return
    //query aliases or table names. We are returning table names, if
    //available, as this is at least more useful than always returning
    //null since we never have the query table aliases

    CompositeType relType = getRelType(column);
    if (relType == null)
      return "";

    return relType.getName();
  }

  @Override
  public String getColumnName(int column) throws SQLException {
    if (!connection.isStrictMode()) {
      String val = get(column).getName();
      if (val != null)
        return val;
    }

    ResultField field = get(column);
    ColumnData columnData = getRelationColumnData(field);
    if (columnData == null) {
      return field.getName();
    }

    return columnData.columnName;
  }

  @Override
  public int getColumnType(int column) throws SQLException {
    ResultField field = get(column);
    try {
      return JDBCTypeMapping.getJDBCTypeCode(connection.getRegistry().resolve(field.getTypeRef()));
    }
    catch (IOException e) {
      throw makeSQLException(e);
    }
  }

  @Override
  public String getColumnTypeName(int column) throws SQLException {

    ResultField field = get(column);
    ColumnData columnData = getRelationColumnData(field);
    if (columnData == null) {
      try {
        return connection.getRegistry().resolve(field.getTypeRef()).getName();
      }
      catch (IOException e) {
        throw makeSQLException(e);
      }
    }

    return JDBCTypeMetaData.getTypeName(columnData.type, columnData.defaultValue);
  }

  @Override
  public String getColumnClassName(int column) throws SQLException {

    Type type;
    try {
      type = connection.getRegistry().resolve(get(column).getTypeRef());
    }
    catch (IOException e) {
      throw makeSQLException(e);
    }
    return lookupCustomType(type, typeMap, type.getCodec(type.getResultFormat()).getDecoder().getDefaultClass()).getName();
  }

  @Override
  public int getPrecision(int column) throws SQLException {

    ResultField field = get(column);
    try {
      return JDBCTypeMetaData.getPrecision(connection.getRegistry().resolve(field.getTypeRef()), field.getTypeLength(), field.getTypeModifier());
    }
    catch (IOException e) {
      throw makeSQLException(e);
    }
  }

  @Override
  public int getScale(int column) throws SQLException {

    ResultField field = get(column);
    try {
      return JDBCTypeMetaData.getScale(connection.getRegistry().resolve(field.getTypeRef()), field.getTypeModifier());
    }
    catch (IOException e) {
      throw makeSQLException(e);
    }
  }

  @Override
  public int getColumnDisplaySize(int column) throws SQLException {

    ResultField field = get(column);
    try {
      return JDBCTypeMetaData.getDisplaySize(connection.getRegistry().resolve(field.getTypeRef()), field.getTypeLength(), field.getTypeModifier());
    }
    catch (IOException e) {
      throw makeSQLException(e);
    }
  }

  @Override
  public boolean isReadOnly(int column) throws SQLException {

    //If it's a computed column we assume it's read only
    return get(column).getRelationAttributeNumber() == 0;
  }

  @Override
  public boolean isWritable(int column) throws SQLException {
    return !isReadOnly(column);
  }

  @Override
  public boolean isDefinitelyWritable(int column) throws SQLException {
    //TODO determine what this is really asking
    return isWritable(column);
  }

}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy