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

com.impossibl.postgres.api.data.Interval Maven / Gradle / Ivy

Go to download

A new JDBC driver for PostgreSQL aimed at supporting the advanced features of JDBC and Postgres.

There is a newer version: 0.7.0-89abc52
Show 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.api.data;

import java.text.DecimalFormat;
import java.text.DecimalFormatSymbols;
import java.text.NumberFormat;
import java.text.ParseException;
import java.util.Calendar;
import java.util.Date;
import java.util.Locale;
import java.util.StringTokenizer;

import static java.lang.Double.parseDouble;
import static java.lang.Integer.parseInt;
import static java.lang.Math.round;
import static java.util.concurrent.TimeUnit.HOURS;
import static java.util.concurrent.TimeUnit.MICROSECONDS;
import static java.util.concurrent.TimeUnit.MINUTES;
import static java.util.concurrent.TimeUnit.SECONDS;

public class Interval {

  private int totalMonths;
  private int totalDays;
  private long totalMicros;

  private static final long SECS_TO_MICROS = SECONDS.toMicros(1);
  private static final long SECS_TO_MILLIS = SECONDS.toMillis(1);

  public Interval(int months, int days, long timeMicros) {
    super();
    setValue(months, days, timeMicros);
  }

  public Interval(int years, int months, int days, int hours, int minutes, double seconds) {
    super();
    setValue(years, months, days, hours, minutes, seconds);
  }

  public Interval(String interval) {
    this(interval, Locale.getDefault());
  }

  public Interval(String interval, Locale locale) {
    super();
    setValue(interval, locale);
  }

  public Interval() {
  }

  public void setValue(String value) {
    setValue(value, Locale.getDefault());
  }

  public void setValue(String value, Locale locale) {

    boolean isISOFormat = !value.startsWith("@");

    // Just a simple '0'
    if (!isISOFormat && value.length() == 3 && value.charAt(2) == '0') {
      totalMonths = totalDays = 0;
      totalMicros = 0;
      return;
    }

    int years = 0;
    int months = 0;
    int days = 0;
    int hours = 0;
    int minutes = 0;
    double seconds = 0;
    boolean ago = false;

    try {

      String valueToken;

      final String changedValue = value.replace('+', ' ').replace('@', ' ');

      StringTokenizer st = new StringTokenizer(changedValue);
      while (st.hasMoreTokens()) {

        String token = st.nextToken();

        if (token.equals("ago")) {
          ago = true;
          break;
        }

        int endHours = token.indexOf(':');
        if (endHours == -1) {
          valueToken = token;
        }
        else {

          // This handles hours, minutes, seconds and microseconds for
          // ISO intervals
          int offset = (token.charAt(0) == '-') ? 1 : 0;

          hours = parseInt(token.substring(offset + 0, endHours));
          minutes = parseInt(token.substring(endHours + 1, endHours + 3));

          // Pre 7.4 servers do not put second information into the results
          // unless it is non-zero.
          int endMinutes = token.indexOf(':', endHours + 1);
          if (endMinutes != -1) {
            NumberFormat numberFormat = NumberFormat.getNumberInstance(locale);
            seconds = numberFormat.parse(token.substring(endMinutes + 1)).doubleValue();
          }

          if (offset == 1) {
            hours = -hours;
            minutes = -minutes;
            seconds = -seconds;
          }

          break;
        }

        if (!st.hasMoreTokens()) {
          throw new IllegalArgumentException("invalid interval");
        }

        token = st.nextToken();

        // This handles years, months, days for both, ISO and
        // Non-ISO intervals. Hours, minutes, seconds and microseconds
        // are handled for Non-ISO intervals here.

        if (token.startsWith("year"))
          years = parseInt(valueToken);
        else if (token.startsWith("mon"))
          months = parseInt(valueToken);
        else if (token.startsWith("day"))
          days = parseInt(valueToken);
        else if (token.startsWith("hour"))
          hours = parseInt(valueToken);
        else if (token.startsWith("min"))
          minutes = parseInt(valueToken);
        else if (token.startsWith("sec"))
          seconds = parseDouble(valueToken);

      }
    }
    catch (ParseException e) {
      throw new IllegalArgumentException("invalid interval", e);
    }

    if (!isISOFormat && ago) {
      // Inverse the leading sign
      setValue(-years, -months, -days, -hours, -minutes, -seconds);
    }
    else {
      setValue(years, months, days, hours, minutes, seconds);
    }

  }

  public void setValue(int months, int days, long timeMicros) {
    this.totalMonths = months;
    this.totalDays = days;
    this.totalMicros = timeMicros;
  }

  public void setValue(int years, int months, int days, int hours, int minutes, double seconds) {
    this.totalMonths += years * 12 + months;
    this.totalDays = days;
    this.totalMicros += HOURS.toMicros(hours);
    this.totalMicros += MINUTES.toMicros(minutes);
    this.totalMicros += (long) (seconds * SECONDS.toMicros(1));
  }

  public long getRawTime() {
    return totalMicros;
  }

  public void setRawTime(long time) {
    this.totalMicros = time;
  }

  public int getRawDays() {
    return totalDays;
  }

  public void setRawDays(int days) {
    this.totalDays = days;
  }

  public int getRawMonths() {
    return totalMonths;
  }

  public void setRawMonths(int months) {
    this.totalMonths = months;
  }

  public long getHours() {
    return MICROSECONDS.toHours(totalMicros);
  }

  public void setHours(long hours) {
    totalMicros -= HOURS.toMicros(getHours());
    totalMicros += HOURS.toMicros(hours);
  }

  public long getMinutes() {
    return MICROSECONDS.toMinutes(totalMicros) - HOURS.toMinutes(getHours());
  }

  public void setMinutes(long minutes) {
    totalMicros -= MINUTES.toMicros(getMinutes());
    totalMicros += MINUTES.toMicros(minutes);
  }

  public double getSeconds() {
    double microsDiff = totalMicros - MINUTES.toMicros(MICROSECONDS.toMinutes(totalMicros));
    return microsDiff / SECS_TO_MICROS;
  }

  public void setSeconds(double seconds) {
    totalMicros -= getSeconds() * SECS_TO_MICROS;
    totalMicros += seconds * SECS_TO_MICROS;
  }

  public int getDays() {
    return totalDays;
  }

  public void setDays(int days) {
    this.totalDays = days;
  }

  public int getMonths() {
    return totalMonths % 12;
  }

  public void setMonths(int months) {
    totalMonths -= getMonths();
    totalMonths += months;
  }

  public int getYears() {
    return totalMonths / 12;
  }

  public void setYears(int years) {
    totalMonths -= getYears() * 12;
    totalMonths += years * 12;
  }

  public void addTo(Calendar cal) {

    long millis = round(getSeconds() * SECS_TO_MILLIS);

    cal.add(Calendar.MILLISECOND, (int) millis);
    cal.add(Calendar.MINUTE, (int) getMinutes());
    cal.add(Calendar.HOUR, (int) getHours());
    cal.add(Calendar.DAY_OF_MONTH, getDays());
    cal.add(Calendar.MONTH, getMonths());
    cal.add(Calendar.YEAR, getYears());
  }

  public void addTo(Date date) {

    Calendar cal = Calendar.getInstance();
    cal.setTime(date);

    addTo(cal);

    date.setTime(cal.getTimeInMillis());
  }

  public void addTo(Interval interval) {
    interval.totalMonths += totalMonths;
    interval.totalDays += totalDays;
    interval.totalMicros += totalMicros;
  }

  @Override
  public int hashCode() {
    final int prime = 31;
    int result = 1;
    result = prime * result + totalDays;
    result = prime * result + totalMonths;
    result = prime * result + (int) (totalMicros ^ (totalMicros >>> 32));
    return result;
  }

  @Override
  public boolean equals(Object obj) {
    if (this == obj)
      return true;
    if (obj == null)
      return false;
    if (getClass() != obj.getClass())
      return false;
    Interval other = (Interval) obj;
    if (totalDays != other.totalDays)
      return false;
    if (totalMonths != other.totalMonths)
      return false;
    if (totalMicros != other.totalMicros)
      return false;
    return true;
  }

  @Override
  public String toString() {
    return toString(Locale.getDefault());
  }

  public String toString(Locale locale) {

    DecimalFormat secondsFormat = new DecimalFormat("0.00####", DecimalFormatSymbols.getInstance(locale));

    StringBuilder buffer = new StringBuilder();

    buffer.
        append("@ ").
        append(getYears()).append(" years ").
        append(getMonths()).append(" months ").
        append(getDays()).append(" days ").
        append(getHours()).append(" hours ").
        append(getMinutes()).append(" minutes ").
        append(secondsFormat.format(getSeconds())).append(" seconds");

    return buffer.toString();
  }

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy