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

net.snowflake.client.core.ResultUtil Maven / Gradle / Ivy

The newest version!
/*
 * Copyright (c) 2012-2019 Snowflake Computing Inc. All rights reserved.
 */

package net.snowflake.client.core;

import com.fasterxml.jackson.databind.JsonNode;
import java.math.BigDecimal;
import java.sql.Date;
import java.sql.SQLException;
import java.sql.Timestamp;
import java.sql.Types;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Calendar;
import java.util.Collections;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.TimeZone;
import net.snowflake.client.jdbc.ErrorCode;
import net.snowflake.client.jdbc.SnowflakeUtil;
import net.snowflake.client.log.ArgSupplier;
import net.snowflake.client.log.SFLogger;
import net.snowflake.client.log.SFLoggerFactory;
import net.snowflake.common.core.SFTime;
import net.snowflake.common.core.SFTimestamp;
import net.snowflake.common.core.SnowflakeDateTimeFormat;
import net.snowflake.common.util.TimeUtil;

public class ResultUtil {
  private static final SFLogger logger = SFLoggerFactory.getLogger(ResultUtil.class);

  public static final int MILLIS_IN_ONE_DAY = 86400000;
  public static final int DEFAULT_SCALE_OF_SFTIME_FRACTION_SECONDS =
      3; // default scale for SFTime fraction seconds

  // Map of default parameter values, used by effectiveParamValue().
  private static final Map defaultParameters;

  static {
    Map map = new HashMap<>();

    // IMPORTANT: This must be consistent with CommonParameterEnum
    map.put("TIMEZONE", "America/Los_Angeles");
    map.put("TIMESTAMP_OUTPUT_FORMAT", "DY, DD MON YYYY HH24:MI:SS TZHTZM");
    map.put("TIMESTAMP_NTZ_OUTPUT_FORMAT", "");
    map.put("TIMESTAMP_LTZ_OUTPUT_FORMAT", "");
    map.put("TIMESTAMP_TZ_OUTPUT_FORMAT", "");
    map.put("DATE_OUTPUT_FORMAT", "YYYY-MM-DD");
    map.put("TIME_OUTPUT_FORMAT", "HH24:MI:SS");
    map.put("CLIENT_HONOR_CLIENT_TZ_FOR_TIMESTAMP_NTZ", Boolean.TRUE);
    map.put("CLIENT_DISABLE_INCIDENTS", Boolean.TRUE);
    map.put("BINARY_OUTPUT_FORMAT", "HEX");
    defaultParameters = map;
  }

  /**
   * Returns the effective parameter value, using the value explicitly provided in parameters, or
   * the default if absent
   *
   * @param parameters keyed in parameter name and valued in parameter value
   * @param paramName Parameter to return the value of
   * @return Effective value
   */
  public static Object effectiveParamValue(Map parameters, String paramName) {
    String upper = paramName.toUpperCase();
    Object value = parameters.get(upper);

    if (value != null) {
      return value;
    }

    value = defaultParameters.get(upper);
    if (value != null) {
      return value;
    }

    logger.debug("Unknown Common Parameter: {}", paramName);
    return null;
  }

  /**
   * Helper function building a formatter for a specialized timestamp type. Note that it will be
   * based on either the 'param' value if set, or the default format provided.
   *
   * @param parameters keyed in parameter name and valued in parameter value
   * @param id id
   * @param param timestamp output format param
   * @param defaultFormat default format
   * @return {@link SnowflakeDateTimeFormat}
   */
  public static SnowflakeDateTimeFormat specializedFormatter(
      Map parameters, String id, String param, String defaultFormat) {
    String sqlFormat =
        SnowflakeDateTimeFormat.effectiveSpecializedTimestampFormat(
            (String) effectiveParamValue(parameters, param), defaultFormat);
    SnowflakeDateTimeFormat formatter = SnowflakeDateTimeFormat.fromSqlFormat(sqlFormat);
    logger.debug(
        "sql {} format: {}, java {} format: {}",
        id,
        sqlFormat,
        id,
        (ArgSupplier) formatter::toSimpleDateTimePattern);
    return formatter;
  }

  /**
   * Adjust timestamp for dates before 1582-10-05
   *
   * @param timestamp needs to be adjusted
   * @return adjusted timestamp
   */
  public static Timestamp adjustTimestamp(Timestamp timestamp) {
    long milliToAdjust = ResultUtil.msDiffJulianToGregorian(timestamp);

    if (milliToAdjust != 0) {
      logger.debug(
          "adjust timestamp by {} days", (ArgSupplier) () -> milliToAdjust / MILLIS_IN_ONE_DAY);

      Timestamp newTimestamp = new Timestamp(timestamp.getTime() + milliToAdjust);

      newTimestamp.setNanos(timestamp.getNanos());

      return newTimestamp;
    } else {
      return timestamp;
    }
  }

  /**
   * For dates before 1582-10-05, calculate the number of millis to adjust.
   *
   * @param date date before 1582-10-05
   * @return millis needs to be adjusted
   */
  public static long msDiffJulianToGregorian(java.util.Date date) {
    // if date is before 1582-10-05, apply the difference
    // by (H-(H/4)-2) where H is the hundreds digit of the year according to:
    // http://en.wikipedia.org/wiki/Gregorian_calendar
    if (date.getTime() < -12220156800000L) {
      // get the year of the date
      Calendar cal = Calendar.getInstance();
      cal.setTime(date);
      int year = cal.get(Calendar.YEAR);
      int month = cal.get(Calendar.MONTH);
      int dayOfMonth = cal.get(Calendar.DAY_OF_MONTH);

      // for dates on or before 02/28, use the previous year otherwise use
      // current year.
      // TODO: we need to revisit this since there is a potential issue using
      // the year/month/day from the calendar since that may not be the same
      // year/month/day as the original date (which is the problem we are
      // trying to solve here).

      if (month == 0 || (month == 1 && dayOfMonth <= 28)) {
        year = year - 1;
      }

      int hundreds = year / 100;
      int differenceInDays = hundreds - (hundreds / 4) - 2;

      return differenceInDays * MILLIS_IN_ONE_DAY;
    } else {
      return 0;
    }
  }

  /**
   * Convert a timestamp internal value (scaled number of seconds + fractional seconds) into a
   * SFTimestamp.
   *
   * @param timestampStr timestamp object
   * @param scale timestamp scale
   * @param internalColumnType snowflake timestamp type
   * @param resultVersion For new result version, timestamp with timezone is formatted as the
   *     seconds since epoch with fractional part in the decimal followed by time zone index. E.g.:
   *     "123.456 1440". Here 123.456 is the * number of seconds since epoch and 1440 is the
   *     timezone index.
   * @param sessionTZ session timezone
   * @param session session object
   * @return converted snowflake timestamp object
   * @throws SFException if timestampStr is an invalid timestamp
   */
  public static SFTimestamp getSFTimestamp(
      String timestampStr,
      int scale,
      int internalColumnType,
      long resultVersion,
      TimeZone sessionTZ,
      SFBaseSession session)
      throws SFException {
    logger.trace("Timestamp getTimestamp(int columnIndex)", false);

    try {
      TimeUtil.TimestampType tsType = null;

      switch (internalColumnType) {
        case Types.TIMESTAMP:
          tsType = TimeUtil.TimestampType.TIMESTAMP_NTZ;
          break;
        case SnowflakeUtil.EXTRA_TYPES_TIMESTAMP_TZ:
          tsType = TimeUtil.TimestampType.TIMESTAMP_TZ;
          logger.trace(
              "Handle timestamp with timezone {} encoding: {}",
              (resultVersion > 0 ? "new" : "old"),
              timestampStr);
          break;
        case SnowflakeUtil.EXTRA_TYPES_TIMESTAMP_LTZ:
          tsType = TimeUtil.TimestampType.TIMESTAMP_LTZ;
          break;
      }

      // Construct a timestamp
      return TimeUtil.getSFTimestamp(timestampStr, scale, tsType, resultVersion, sessionTZ);
    } catch (IllegalArgumentException ex) {
      throw new SFException(ErrorCode.IO_ERROR, "Invalid timestamp value: " + timestampStr);
    }
  }

  /**
   * Convert a time internal value (scaled number of seconds + fractional seconds) into an SFTime.
   *
   * 

Example: getSFTime("123.456", 5) returns an SFTime for 00:02:03.45600. * * @param obj time object * @param scale time scale * @param session session object * @return snowflake time object * @throws SFException if time is invalid */ public static SFTime getSFTime(String obj, int scale, SFBaseSession session) throws SFException { try { return TimeUtil.getSFTime(obj, scale); } catch (IllegalArgumentException ex) { throw new SFException(ErrorCode.INTERNAL_ERROR, "Invalid time value: " + obj); } } /** * Convert a time value into a string * * @param sft snowflake time object * @param scale time scale * @param timeFormatter time formatter * @return time in string */ public static String getSFTimeAsString( SFTime sft, int scale, SnowflakeDateTimeFormat timeFormatter) { return timeFormatter.format(sft, scale); } /** * Convert a boolean to a string * * @param bool boolean * @return boolean in string */ public static String getBooleanAsString(boolean bool) { return bool ? "TRUE" : "FALSE"; } /** * Convert a SFTimestamp to a string value. * * @param sfTS snowflake timestamp object * @param columnType internal snowflake t * @param scale timestamp scale * @param timestampNTZFormatter snowflake timestamp ntz format * @param timestampLTZFormatter snowflake timestamp ltz format * @param timestampTZFormatter snowflake timestamp tz format * @param session session object * @return timestamp in string in desired format * @throws SFException timestamp format is missing */ public static String getSFTimestampAsString( SFTimestamp sfTS, int columnType, int scale, SnowflakeDateTimeFormat timestampNTZFormatter, SnowflakeDateTimeFormat timestampLTZFormatter, SnowflakeDateTimeFormat timestampTZFormatter, SFBaseSession session) throws SFException { // Derive the timestamp formatter to use SnowflakeDateTimeFormat formatter; if (columnType == Types.TIMESTAMP || columnType == SnowflakeUtil.EXTRA_TYPES_TIMESTAMP_NTZ) { formatter = timestampNTZFormatter; } else if (columnType == SnowflakeUtil.EXTRA_TYPES_TIMESTAMP_LTZ) { formatter = timestampLTZFormatter; } else // TZ { formatter = timestampTZFormatter; } if (formatter == null) { throw new SFException(ErrorCode.INTERNAL_ERROR, "missing timestamp formatter"); } try { Timestamp adjustedTimestamp = ResultUtil.adjustTimestamp(sfTS.getTimestamp()); return formatter.format(adjustedTimestamp, sfTS.getTimeZone(), scale); } catch (SFTimestamp.TimestampOperationNotAvailableException e) { // this timestamp doesn't fit into a Java timestamp, and therefore we // can't format it (for now). Just print it out as seconds since epoch. BigDecimal nanosSinceEpoch = sfTS.getNanosSinceEpoch(); BigDecimal secondsSinceEpoch = nanosSinceEpoch.scaleByPowerOfTen(-9); return secondsSinceEpoch.setScale(scale).toPlainString(); } } /** * Convert a date value into a string * * @param date date will be converted * @param dateFormatter date format * @return date in string */ public static String getDateAsString(Date date, SnowflakeDateTimeFormat dateFormatter) { return dateFormatter.format(date, TimeZone.getDefault()); } /** * Adjust date for before 1582-10-05 * * @param date date before 1582-10-05 * @return adjusted date */ public static Date adjustDate(Date date) { long milliToAdjust = ResultUtil.msDiffJulianToGregorian(date); if (milliToAdjust != 0) { // add the difference to the new date return new Date(date.getTime() + milliToAdjust); } else { return date; } } /** * Convert a date internal object to a Date object in specified timezone. * * @param str snowflake date object * @param tz timezone we want convert to * @param session snowflake session object * @return java date object * @throws SFException if date is invalid */ @Deprecated public static Date getDate(String str, TimeZone tz, SFBaseSession session) throws SFException { try { long milliSecsSinceEpoch = Long.valueOf(str) * MILLIS_IN_ONE_DAY; SFTimestamp tsInUTC = SFTimestamp.fromDate(new Date(milliSecsSinceEpoch), 0, TimeZone.getTimeZone("UTC")); SFTimestamp tsInClientTZ = tsInUTC.moveToTimeZone(tz); logger.debug( "getDate: tz offset={}", (ArgSupplier) () -> tsInClientTZ.getTimeZone().getOffset(tsInClientTZ.getTime())); // return the date adjusted to the JVM default time zone Date preDate = new Date(tsInClientTZ.getTime()); // if date is on or before 1582-10-04, apply the difference // by (H-H/4-2) where H is the hundreds digit of the year according to: // http://en.wikipedia.org/wiki/Gregorian_calendar Date newDate = adjustDate(preDate); logger.debug( "Adjust date from {} to {}", (ArgSupplier) preDate::toString, (ArgSupplier) newDate::toString); return newDate; } catch (NumberFormatException ex) { throw new SFException(ErrorCode.INTERNAL_ERROR, "Invalid date value: " + str); } } /** * Convert snowflake bool to java boolean * * @param str boolean type in string representation * @return true if the value indicates true otherwise false */ public static boolean getBoolean(String str) { return str.equalsIgnoreCase(Boolean.TRUE.toString()) || str.equals("1"); } /** * Calculate number of rows updated given a result set Interpret result format based on result * set's statement type * * @param resultSet result set to extract update count from * @return the number of rows updated * @throws SFException if failed to calculate update count * @throws SQLException if failed to calculate update count */ public static long calculateUpdateCount(SFBaseResultSet resultSet) throws SFException, SQLException { long updateCount = 0; SFStatementType statementType = resultSet.getStatementType(); if (statementType.isDML()) { while (resultSet.next()) { if (statementType == SFStatementType.COPY) { SFResultSetMetaData resultSetMetaData = resultSet.getMetaData(); int columnIndex = resultSetMetaData.getColumnIndex("rows_loaded"); updateCount += columnIndex == -1 ? 0 : resultSet.getLong(columnIndex + 1); } else if (statementType == SFStatementType.INSERT || statementType == SFStatementType.UPDATE || statementType == SFStatementType.DELETE || statementType == SFStatementType.MERGE || statementType == SFStatementType.MULTI_INSERT) { int columnCount = resultSet.getMetaData().getColumnCount(); for (int i = 0; i < columnCount; i++) { updateCount += resultSet.getLong(i + 1); // add up number of rows updated } } else { updateCount = 0; } } } else { updateCount = statementType.isGenerateResultSet() ? -1 : 0; } return updateCount; } /** * Given a list of String, do a case insensitive search for target string Used by * resultsetMetadata to search for target column name * * @param source source string list * @param target target string to match * @return index in the source string list that matches the target string index starts from zero */ public static int listSearchCaseInsensitive(List source, String target) { for (int i = 0; i < source.size(); i++) { if (target.equalsIgnoreCase(source.get(i))) { return i; } } return -1; } /** * Return the list of result IDs provided in a result, if available; otherwise return an empty * list. * * @param result result json * @return list of result IDs which can be used for result scans */ private static List getResultIds(JsonNode result) { JsonNode resultIds = result.path("data").path("resultIds"); if (resultIds.isNull() || resultIds.isMissingNode() || resultIds.asText().isEmpty()) { return Collections.emptyList(); } return new ArrayList<>(Arrays.asList(resultIds.asText().split(","))); } /** * Return the list of result types provided in a result, if available; otherwise return an empty * list. * * @param result result json * @return list of result IDs which can be used for result scans */ private static List getResultTypes(JsonNode result) { JsonNode resultTypes = result.path("data").path("resultTypes"); if (resultTypes.isNull() || resultTypes.isMissingNode() || resultTypes.asText().isEmpty()) { return Collections.emptyList(); } String[] typeStrs = resultTypes.asText().split(","); List res = new ArrayList<>(); for (String typeStr : typeStrs) { long typeId = Long.valueOf(typeStr); res.add(SFStatementType.lookUpTypeById(typeId)); } return res; } /** * Return the list of child results provided in a result, if available; otherwise return an empty * list * * @param session the current session * @param requestId the current request id * @param result result json * @return list of child results * @throws SFException if the number of child IDs does not match child statement types */ public static List getChildResults( SFBaseSession session, String requestId, JsonNode result) throws SFException { List ids = getResultIds(result); List types = getResultTypes(result); if (ids.size() != types.size()) { throw new SFException( ErrorCode.CHILD_RESULT_IDS_AND_TYPES_DIFFERENT_SIZES, ids.size(), types.size()); } List res = new ArrayList<>(); for (int i = 0; i < ids.size(); i++) { res.add(new SFChildResult(ids.get(i), types.get(i))); } return res; } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy