net.snowflake.client.core.SFBaseResultSet Maven / Gradle / Ivy
/*
* Copyright (c) 2012-2017 Snowflake Computing Inc. All rights reserved.
*/
package net.snowflake.client.core;
import net.snowflake.client.jdbc.ErrorCode;
import net.snowflake.client.jdbc.SnowflakeSQLException;
import net.snowflake.client.jdbc.SnowflakeUtil;
import net.snowflake.common.core.SFBinary;
import net.snowflake.common.core.SFBinaryFormat;
import net.snowflake.common.core.SFTime;
import net.snowflake.common.core.SFTimestamp;
import net.snowflake.common.core.SnowflakeDateTimeFormat;
import java.math.BigDecimal;
import java.math.RoundingMode;
import java.sql.Date;
import java.sql.SQLException;
import java.sql.Time;
import java.sql.Timestamp;
import java.sql.Types;
import java.util.HashMap;
import java.util.Map;
import java.util.TimeZone;
import net.snowflake.client.log.SFLogger;
import net.snowflake.client.log.SFLoggerFactory;
/**
* Base class for query result set and metadata result set
*
* @author jhuang
*/
public abstract class SFBaseResultSet
{
static final SFLogger logger = SFLoggerFactory.getLogger(SFBaseResultSet.class);
protected boolean wasNull = false;
protected Object[] nextRow = null;
protected SFResultSetMetaData resultSetMetaData = null;
protected int row = 0;
protected Map parameters = new HashMap<>();
protected TimeZone timeZone;
// Timezone used for TimestampNTZ
private static TimeZone timeZoneUTC = TimeZone.getTimeZone("UTC");
// Formatters for different datatypes
protected SnowflakeDateTimeFormat timestampNTZFormatter;
protected SnowflakeDateTimeFormat timestampLTZFormatter;
protected SnowflakeDateTimeFormat timestampTZFormatter;
protected SnowflakeDateTimeFormat dateFormatter;
protected SnowflakeDateTimeFormat timeFormatter;
protected boolean honorClientTZForTimestampNTZ = true;
protected SFBinaryFormat binaryFormatter;
protected long resultVersion = 0;
protected int numberOfBinds = 0;
// For creating incidents
protected SFSession session;
// indicate whether the result set has been closed or not.
protected boolean isClosed = true;
public void setSession(SFSession session)
{
this.session = session;
}
// default implementation
public boolean next() throws SFException, SnowflakeSQLException
{
logger.debug("public boolean next()");
return false;
}
public void close()
{
if (logger.isDebugEnabled())
logger.debug("Time: {} public void close()", System.currentTimeMillis());
resultSetMetaData = null;
isClosed = true;
}
public boolean wasNull()
{
logger.debug("public boolean wasNull() returning {}", wasNull);
return wasNull;
}
public String getString(int columnIndex) throws SFException
{
logger.debug("public String getString(int columnIndex)");
// Column index starts from 1, not 0.
Object obj = getObjectInternal(columnIndex);
if (obj == null)
{
return null;
}
// print timestamp in string format
int columnType = resultSetMetaData.getInternalColumnType(columnIndex);
switch (columnType)
{
case Types.BOOLEAN:
return ResultUtil.getBooleanAsString(
ResultUtil.getBoolean(obj.toString()));
case Types.TIMESTAMP:
case SnowflakeUtil.EXTRA_TYPES_TIMESTAMP_LTZ:
case SnowflakeUtil.EXTRA_TYPES_TIMESTAMP_TZ:
SFTimestamp sfTS = getSFTimestamp(columnIndex);
int columnScale = resultSetMetaData.getScale(columnIndex);
String timestampStr = ResultUtil.getSFTimestampAsString(
sfTS, columnType, columnScale, timestampNTZFormatter,
timestampLTZFormatter, timestampTZFormatter, session);
if (logger.isDebugEnabled())
logger.debug("Converting timestamp to string from: {} to: {}",
obj.toString(), timestampStr);
return timestampStr;
case Types.DATE:
Date date = getDate(columnIndex, timeZoneUTC);
if (dateFormatter == null)
{
throw IncidentUtil.
generateIncidentWithException(session, null, null,
ErrorCode.INTERNAL_ERROR,
"missing date formatter");
}
String dateStr = ResultUtil.getDateAsString(date, dateFormatter);
if (logger.isDebugEnabled())
logger.debug("Converting date to string from: {} to: {}",
obj.toString(), dateStr);
return dateStr;
case Types.TIME:
SFTime sfTime = getSFTime(columnIndex);
if (timeFormatter == null)
{
throw IncidentUtil
.generateIncidentWithException(session, null, null,
ErrorCode.INTERNAL_ERROR,
"missing time formatter");
}
int scale = resultSetMetaData.getScale(columnIndex);
String timeStr = ResultUtil.getSFTimeAsString(sfTime, scale, timeFormatter);
if (logger.isDebugEnabled())
logger.debug("Converting time to string from: {} to: {}",
obj.toString(), timeStr);
return timeStr;
case Types.BINARY:
if (binaryFormatter == null)
{
throw IncidentUtil
.generateIncidentWithException(session, null, null,
ErrorCode.INTERNAL_ERROR,
"missing binary formatter");
}
if (binaryFormatter == SFBinaryFormat.HEX)
{
// Shortcut: the values are already passed with hex encoding, so just
// return the string unchanged rather than constructing an SFBinary.
return obj.toString();
}
SFBinary sfb = new SFBinary(getBytes(columnIndex));
return binaryFormatter.format(sfb);
default:
break;
}
return obj.toString();
}
public boolean getBoolean(int columnIndex) throws SFException
{
logger.debug(
"public boolean getBoolean(int columnIndex)");
Object obj = getObjectInternal(columnIndex);
if (obj == null)
{
return false;
}
if (obj instanceof Boolean)
{
return (Boolean) obj;
}
else
{
return ResultUtil.getBoolean(obj.toString());
}
}
public short getShort(int columnIndex) throws SFException
{
logger.debug("public short getShort(int columnIndex)");
// Column index starts from 1, not 0.
Object obj = getObjectInternal(columnIndex);
if (obj == null)
return 0;
if (obj instanceof String)
{
return Short.parseShort((String) obj);
}
else
{
return ((Number) obj).shortValue();
}
}
public int getInt(int columnIndex) throws SFException
{
logger.debug("public int getInt(int columnIndex)");
// Column index starts from 1, not 0.
Object obj = getObjectInternal(columnIndex);
if (obj == null)
return 0;
if (obj instanceof String)
{
return Integer.parseInt((String) obj);
}
else
{
return ((Number) obj).intValue();
}
}
public long getLong(int columnIndex) throws SFException
{
logger.debug("public long getLong(int columnIndex)");
// Column index starts from 1, not 0.
Object obj = getObjectInternal(columnIndex);
if (obj == null)
return 0;
try
{
if (obj instanceof String)
{
return Long.parseLong((String) obj);
}
else
{
return ((Number) obj).longValue();
}
}
catch (NumberFormatException nfe)
{
int columnType = resultSetMetaData.getColumnType(columnIndex);
if (Types.INTEGER == columnType
|| Types.SMALLINT == columnType)
{
SFException sfe = new SFException(ErrorCode.INTERNAL_ERROR,
"Invalid long: " + obj.toString());
IncidentUtil.generateIncident(session, "Unable to Convert to Long",
null, null, null, sfe);
throw sfe;
}
else
{
throw new SFException(ErrorCode.INVALID_VALUE_CONVERT,
columnType, "LONG", obj);
}
}
}
public float getFloat(int columnIndex) throws SFException
{
logger.debug("public float getFloat(int columnIndex)");
// Column index starts from 1, not 0.
Object obj = getObjectInternal(columnIndex);
if (obj == null)
return 0;
if (obj instanceof String)
{
return Float.parseFloat((String) obj);
}
else
{
return ((Number) obj).floatValue();
}
}
public double getDouble(int columnIndex) throws SFException
{
logger.debug("public double getDouble(int columnIndex)");
// Column index starts from 1, not 0.
Object obj = getObjectInternal(columnIndex);
// snow-11974: null for getDouble should return 0
if (obj == null)
return 0;
if (obj instanceof String)
{
return Double.parseDouble((String) obj);
}
else
{
return ((Number) obj).doubleValue();
}
}
public byte[] getBytes(int columnIndex) throws SFException
{
logger.debug("public byte[] getBytes(int columnIndex)");
// Column index starts from 1, not 0.
Object obj = getObjectInternal(columnIndex);
if (obj == null)
return null;
try
{
return SFBinary.fromHex(obj.toString()).getBytes();
}
catch (IllegalArgumentException ex)
{
throw new SFException(ErrorCode.INTERNAL_ERROR,
"Invalid binary value: " + obj.toString());
}
}
public Date getDate(int columnIndex, TimeZone tz) throws SFException
{
if (tz == null)
{
tz = TimeZone.getDefault();
}
logger.debug("public Date getDate(int columnIndex)");
// Column index starts from 1, not 0.
Object obj = getObjectInternal(columnIndex);
if (obj == null)
return null;
int columnType = resultSetMetaData.getColumnType(columnIndex);
if (Types.TIMESTAMP == columnType)
{
return new Date(getTimestamp(columnIndex, tz).getTime());
}
else if (Types.DATE == columnType)
{
return ResultUtil.getDate(obj.toString(), tz, session);
}
// for Types.TIME and all other type, throw user error
else
{
throw new SFException(ErrorCode.INVALID_VALUE_CONVERT, columnType, "DATE", obj);
}
}
public Date getDate(int columnIndex) throws SFException
{
return getDate(columnIndex, TimeZone.getDefault());
}
public Time getTime(int columnIndex) throws SFException
{
logger.debug("public Time getTime(int columnIndex)");
int columnType = resultSetMetaData.getColumnType(columnIndex);
if (Types.TIME == columnType)
{
SFTime sfTime = getSFTime(columnIndex);
if (sfTime == null)
{
return null;
}
return new Time(sfTime.getFractionalSeconds(3));
}
else if (Types.TIMESTAMP == columnType)
{
return new Time(getTimestamp(columnIndex).getTime());
}
else
{
throw new SFException(ErrorCode.INVALID_VALUE_CONVERT, columnType, "Time",
getObjectInternal(columnIndex));
}
}
private SFTime getSFTime(int columnIndex) throws SFException
{
Object obj = getObjectInternal(columnIndex);
if (obj == null)
return null;
int scale = resultSetMetaData.getScale(columnIndex);
return ResultUtil.getSFTime(obj.toString(), scale, session);
}
private Timestamp getTimestamp(int columnIndex) throws SFException
{
return getTimestamp(columnIndex, TimeZone.getDefault());
}
public Timestamp getTimestamp(int columnIndex, TimeZone tz)
throws SFException
{
int columnType = resultSetMetaData.getColumnType(columnIndex);
if (Types.TIMESTAMP == columnType)
{
SFTimestamp sfTS = getSFTimestamp(columnIndex);
if (sfTS == null)
{
return null;
}
Timestamp res = sfTS.getTimestamp();
if (res == null)
{
return null;
}
// SNOW-14777: for timestamp_ntz, we should treat the time as in client time
// zone so adjust the timestamp by subtracting the offset of the client
// timezone
if (honorClientTZForTimestampNTZ &&
resultSetMetaData.getInternalColumnType(columnIndex) == Types.TIMESTAMP)
{
return sfTS.moveToTimeZone(tz).getTimestamp();
}
return res;
}
else if (Types.DATE == columnType)
{
return new Timestamp(getDate(columnIndex, tz).getTime());
}
else if (Types.TIME == columnType)
{
return new Timestamp(getTime(columnIndex).getTime());
}
else
{
throw new SFException(ErrorCode.INVALID_VALUE_CONVERT, columnType, "Timestamp",
getObjectInternal(columnIndex));
}
}
private SFTimestamp getSFTimestamp(int columnIndex) throws SFException
{
logger.debug(
"public Timestamp getTimestamp(int columnIndex)");
Object obj = getObjectInternal(columnIndex);
if (obj == null)
return null;
return ResultUtil.getSFTimestamp(
obj.toString(),
resultSetMetaData.getScale(columnIndex),
resultSetMetaData.getInternalColumnType(columnIndex),
resultVersion, timeZone, session);
}
public SFResultSetMetaData getMetaData() throws SFException
{
logger.debug("public ResultSetMetaData getMetaData()");
return resultSetMetaData;
}
protected abstract Object getObjectInternal(int columnIndex) throws SFException;
public Object getObject(int columnIndex) throws SFException
{
logger.debug(
"public Object getObject(int columnIndex)");
int type = resultSetMetaData.getColumnType(columnIndex);
Object obj = getObjectInternal(columnIndex);
if (obj == null)
return null;
switch(type)
{
case Types.VARCHAR:
case Types.CHAR:
return getString(columnIndex);
case Types.BINARY:
return getBytes(columnIndex);
case Types.INTEGER:
return getInt(columnIndex);
case Types.DECIMAL:
return getBigDecimal(columnIndex);
case Types.DOUBLE:
return getDouble(columnIndex);
case Types.TIMESTAMP:
return getTimestamp(columnIndex);
case Types.DATE:
return getDate(columnIndex);
case Types.TIME:
return getTime(columnIndex);
case Types.BOOLEAN:
return getBoolean(columnIndex);
default:
throw IncidentUtil.
generateIncidentWithException(session, null, null,
ErrorCode.FEATURE_UNSUPPORTED,
"data type: " + type);
}
}
public BigDecimal getBigDecimal(int columnIndex) throws SFException
{
logger.debug(
"public BigDecimal getBigDecimal(int columnIndex)");
// Column index starts from 1, not 0.
Object obj = getObjectInternal(columnIndex);
if (obj == null)
return null;
return new BigDecimal(obj.toString());
}
public BigDecimal getBigDecimal(int columnIndex, int scale) throws SFException
{
logger.debug(
"public BigDecimal getBigDecimal(int columnIndex)");
Object obj = getObjectInternal(columnIndex);
if (obj == null)
return null;
BigDecimal value = new BigDecimal(obj.toString());
value = value.setScale(scale, RoundingMode.HALF_UP);
return value;
}
public int getRow() throws SQLException
{
logger.debug("public int getRow()");
return row;
}
public boolean absolute(int row) throws SFException
{
logger.debug("public boolean absolute(int row)");
throw new SFException(
ErrorCode.FEATURE_UNSUPPORTED, "seek to a specific row");
}
public boolean relative(int rows) throws SFException
{
logger.debug("public boolean relative(int rows)");
throw new SFException(
ErrorCode.FEATURE_UNSUPPORTED, "seek to a row relative to current row");
}
public boolean previous() throws SFException
{
logger.debug("public boolean previous()");
throw new SFException(
ErrorCode.FEATURE_UNSUPPORTED, "seek to a previous row");
}
protected int getNumberOfBinds()
{
return numberOfBinds;
}
public boolean isFirst()
{
logger.debug("public boolean isFirst()");
return row == 1;
}
public boolean isClosed()
{
return isClosed;
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy