/*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package com.facebook.presto.type;
import com.facebook.presto.spi.ConnectorSession;
import com.facebook.presto.spi.PrestoException;
import com.facebook.presto.spi.function.ScalarOperator;
import com.facebook.presto.spi.function.SqlType;
import com.facebook.presto.spi.type.StandardTypes;
import org.joda.time.DateTimeField;
import org.joda.time.chrono.ISOChronology;
import java.util.concurrent.TimeUnit;
import static com.facebook.presto.spi.StandardErrorCode.INVALID_FUNCTION_ARGUMENT;
import static com.facebook.presto.spi.function.OperatorType.ADD;
import static com.facebook.presto.spi.function.OperatorType.SUBTRACT;
import static com.facebook.presto.spi.type.DateTimeEncoding.unpackMillisUtc;
import static com.facebook.presto.spi.type.DateTimeEncoding.updateMillisUtc;
import static com.facebook.presto.util.DateTimeZoneIndex.getChronology;
import static com.facebook.presto.util.DateTimeZoneIndex.unpackChronology;
public final class DateTimeOperators
{
private static final DateTimeField MILLIS_OF_DAY = ISOChronology.getInstanceUTC().millisOfDay();
private static final DateTimeField MONTH_OF_YEAR_UTC = ISOChronology.getInstanceUTC().monthOfYear();
private DateTimeOperators()
{
}
@ScalarOperator(ADD)
@SqlType(StandardTypes.DATE)
public static long datePlusIntervalDayToSecond(@SqlType(StandardTypes.DATE) long left, @SqlType(StandardTypes.INTERVAL_DAY_TO_SECOND) long right)
{
if (MILLIS_OF_DAY.get(right) != 0) {
throw new PrestoException(INVALID_FUNCTION_ARGUMENT, "Cannot add hour, minutes or seconds to a date");
}
return left + TimeUnit.MILLISECONDS.toDays(right);
}
@ScalarOperator(ADD)
@SqlType(StandardTypes.DATE)
public static long intervalDayToSecondPlusDate(@SqlType(StandardTypes.INTERVAL_DAY_TO_SECOND) long left, @SqlType(StandardTypes.DATE) long right)
{
if (MILLIS_OF_DAY.get(left) != 0) {
throw new PrestoException(INVALID_FUNCTION_ARGUMENT, "Cannot add hour, minutes or seconds to a date");
}
return TimeUnit.MILLISECONDS.toDays(left) + right;
}
@ScalarOperator(ADD)
@SqlType(StandardTypes.TIME)
public static long timePlusIntervalDayToSecond(ConnectorSession session, @SqlType(StandardTypes.TIME) long left, @SqlType(StandardTypes.INTERVAL_DAY_TO_SECOND) long right)
{
return modulo24Hour(getChronology(session.getTimeZoneKey()), left + right);
}
@ScalarOperator(ADD)
@SqlType(StandardTypes.TIME)
public static long intervalDayToSecondPlusTime(ConnectorSession session, @SqlType(StandardTypes.INTERVAL_DAY_TO_SECOND) long left, @SqlType(StandardTypes.TIME) long right)
{
return modulo24Hour(getChronology(session.getTimeZoneKey()), left + right);
}
@ScalarOperator(ADD)
@SqlType(StandardTypes.TIME_WITH_TIME_ZONE)
public static long timeWithTimeZonePlusIntervalDayToSecond(@SqlType(StandardTypes.TIME_WITH_TIME_ZONE) long left, @SqlType(StandardTypes.INTERVAL_DAY_TO_SECOND) long right)
{
return updateMillisUtc((long) modulo24Hour(unpackChronology(left), unpackMillisUtc(left) + right), left);
}
@ScalarOperator(ADD)
@SqlType(StandardTypes.TIME_WITH_TIME_ZONE)
public static long intervalDayToSecondPlusTimeWithTimeZone(@SqlType(StandardTypes.INTERVAL_DAY_TO_SECOND) long left, @SqlType(StandardTypes.TIME_WITH_TIME_ZONE) long right)
{
return updateMillisUtc((long) modulo24Hour(unpackChronology(right), left + unpackMillisUtc(right)), right);
}
@ScalarOperator(ADD)
@SqlType(StandardTypes.TIMESTAMP)
public static long timestampPlusIntervalDayToSecond(@SqlType(StandardTypes.TIMESTAMP) long left, @SqlType(StandardTypes.INTERVAL_DAY_TO_SECOND) long right)
{
return left + right;
}
@ScalarOperator(ADD)
@SqlType(StandardTypes.TIMESTAMP)
public static long intervalDayToSecondPlusTimestamp(@SqlType(StandardTypes.INTERVAL_DAY_TO_SECOND) long left, @SqlType(StandardTypes.TIMESTAMP) long right)
{
return left + right;
}
@ScalarOperator(ADD)
@SqlType(StandardTypes.TIMESTAMP_WITH_TIME_ZONE)
public static long timestampWithTimeZonePlusIntervalDayToSecond(@SqlType(StandardTypes.TIMESTAMP_WITH_TIME_ZONE) long left, @SqlType(StandardTypes.INTERVAL_DAY_TO_SECOND) long right)
{
return updateMillisUtc(unpackMillisUtc(left) + right, left);
}
@ScalarOperator(ADD)
@SqlType(StandardTypes.TIMESTAMP_WITH_TIME_ZONE)
public static long intervalDayToSecondPlusTimestampWithTimeZone(@SqlType(StandardTypes.INTERVAL_DAY_TO_SECOND) long left, @SqlType(StandardTypes.TIMESTAMP_WITH_TIME_ZONE) long right)
{
return updateMillisUtc(left + unpackMillisUtc(right), right);
}
@ScalarOperator(ADD)
@SqlType(StandardTypes.DATE)
public static long datePlusIntervalYearToMonth(@SqlType(StandardTypes.DATE) long left, @SqlType(StandardTypes.INTERVAL_YEAR_TO_MONTH) long right)
{
long millis = MONTH_OF_YEAR_UTC.add(TimeUnit.DAYS.toMillis(left), right);
return TimeUnit.MILLISECONDS.toDays(millis);
}
@ScalarOperator(ADD)
@SqlType(StandardTypes.DATE)
public static long intervalYearToMonthPlusDate(@SqlType(StandardTypes.INTERVAL_YEAR_TO_MONTH) long left, @SqlType(StandardTypes.DATE) long right)
{
long millis = MONTH_OF_YEAR_UTC.add(TimeUnit.DAYS.toMillis(right), left);
return TimeUnit.MILLISECONDS.toDays(millis);
}
@ScalarOperator(ADD)
@SqlType(StandardTypes.TIME)
public static long timePlusIntervalYearToMonth(@SqlType(StandardTypes.TIME) long left, @SqlType(StandardTypes.INTERVAL_YEAR_TO_MONTH) long right)
{
return left;
}
@ScalarOperator(ADD)
@SqlType(StandardTypes.TIME)
public static long intervalYearToMonthPlusTime(@SqlType(StandardTypes.INTERVAL_YEAR_TO_MONTH) long left, @SqlType(StandardTypes.TIME) long right)
{
return right;
}
@ScalarOperator(ADD)
@SqlType(StandardTypes.TIME_WITH_TIME_ZONE)
public static long timeWithTimeZonePlusIntervalYearToMonth(@SqlType(StandardTypes.TIME_WITH_TIME_ZONE) long left, @SqlType(StandardTypes.INTERVAL_YEAR_TO_MONTH) long right)
{
return left;
}
@ScalarOperator(ADD)
@SqlType(StandardTypes.TIME_WITH_TIME_ZONE)
public static long intervalYearToMonthPlusTimeWithTimeZone(@SqlType(StandardTypes.INTERVAL_YEAR_TO_MONTH) long left, @SqlType(StandardTypes.TIME_WITH_TIME_ZONE) long right)
{
return right;
}
@ScalarOperator(ADD)
@SqlType(StandardTypes.TIMESTAMP)
public static long timestampPlusIntervalYearToMonth(ConnectorSession session, @SqlType(StandardTypes.TIMESTAMP) long left, @SqlType(StandardTypes.INTERVAL_YEAR_TO_MONTH) long right)
{
return getChronology(session.getTimeZoneKey()).monthOfYear().add(left, right);
}
@ScalarOperator(ADD)
@SqlType(StandardTypes.TIMESTAMP)
public static long intervalYearToMonthPlusTimestamp(ConnectorSession session, @SqlType(StandardTypes.INTERVAL_YEAR_TO_MONTH) long left, @SqlType(StandardTypes.TIMESTAMP) long right)
{
return getChronology(session.getTimeZoneKey()).monthOfYear().add(right, left);
}
@ScalarOperator(ADD)
@SqlType(StandardTypes.TIMESTAMP_WITH_TIME_ZONE)
public static long timestampWithTimeZonePlusIntervalYearToMonth(@SqlType(StandardTypes.TIMESTAMP_WITH_TIME_ZONE) long left, @SqlType(StandardTypes.INTERVAL_YEAR_TO_MONTH) long right)
{
return updateMillisUtc(unpackChronology(left).monthOfYear().add(unpackMillisUtc(left), right), left);
}
@ScalarOperator(ADD)
@SqlType(StandardTypes.TIMESTAMP_WITH_TIME_ZONE)
public static long intervalYearToMonthPlusTimestampWithTimeZone(@SqlType(StandardTypes.INTERVAL_YEAR_TO_MONTH) long left, @SqlType(StandardTypes.TIMESTAMP_WITH_TIME_ZONE) long right)
{
return updateMillisUtc(unpackChronology(right).monthOfYear().add(unpackMillisUtc(right), left), right);
}
@ScalarOperator(SUBTRACT)
@SqlType(StandardTypes.DATE)
public static long dateMinusIntervalDayToSecond(@SqlType(StandardTypes.DATE) long left, @SqlType(StandardTypes.INTERVAL_DAY_TO_SECOND) long right)
{
if (MILLIS_OF_DAY.get(right) != 0) {
throw new PrestoException(INVALID_FUNCTION_ARGUMENT, "Cannot subtract hour, minutes or seconds from a date");
}
return left - TimeUnit.MILLISECONDS.toDays(right);
}
@ScalarOperator(SUBTRACT)
@SqlType(StandardTypes.TIME)
public static long timeMinusIntervalDayToSecond(ConnectorSession session, @SqlType(StandardTypes.TIME) long left, @SqlType(StandardTypes.INTERVAL_DAY_TO_SECOND) long right)
{
return modulo24Hour(getChronology(session.getTimeZoneKey()), left - right);
}
@ScalarOperator(SUBTRACT)
@SqlType(StandardTypes.TIME_WITH_TIME_ZONE)
public static long timeWithTimeZoneMinusIntervalDayToSecond(@SqlType(StandardTypes.TIME_WITH_TIME_ZONE) long left, @SqlType(StandardTypes.INTERVAL_DAY_TO_SECOND) long right)
{
return updateMillisUtc((long) modulo24Hour(unpackChronology(left), unpackMillisUtc(left) - right), left);
}
@ScalarOperator(SUBTRACT)
@SqlType(StandardTypes.TIMESTAMP)
public static long timestampMinusIntervalDayToSecond(@SqlType(StandardTypes.TIMESTAMP) long left, @SqlType(StandardTypes.INTERVAL_DAY_TO_SECOND) long right)
{
return left - right;
}
@ScalarOperator(SUBTRACT)
@SqlType(StandardTypes.TIMESTAMP_WITH_TIME_ZONE)
public static long timestampWithTimeZoneMinusIntervalDayToSecond(@SqlType(StandardTypes.TIMESTAMP_WITH_TIME_ZONE) long left, @SqlType(StandardTypes.INTERVAL_DAY_TO_SECOND) long right)
{
return updateMillisUtc(unpackMillisUtc(left) - right, left);
}
@ScalarOperator(SUBTRACT)
@SqlType(StandardTypes.DATE)
public static long dateMinusIntervalYearToMonth(ConnectorSession session, @SqlType(StandardTypes.DATE) long left, @SqlType(StandardTypes.INTERVAL_YEAR_TO_MONTH) long right)
{
long millis = MONTH_OF_YEAR_UTC.add(TimeUnit.DAYS.toMillis(left), -right);
return TimeUnit.MILLISECONDS.toDays(millis);
}
@ScalarOperator(SUBTRACT)
@SqlType(StandardTypes.TIME)
public static long timeMinusIntervalYearToMonth(@SqlType(StandardTypes.TIME) long left, @SqlType(StandardTypes.INTERVAL_YEAR_TO_MONTH) long right)
{
return left;
}
@ScalarOperator(SUBTRACT)
@SqlType(StandardTypes.TIME_WITH_TIME_ZONE)
public static long timeWithTimeZoneMinusIntervalYearToMonth(@SqlType(StandardTypes.TIME_WITH_TIME_ZONE) long left, @SqlType(StandardTypes.INTERVAL_YEAR_TO_MONTH) long right)
{
return left;
}
@ScalarOperator(SUBTRACT)
@SqlType(StandardTypes.TIMESTAMP)
public static long timestampMinusIntervalYearToMonth(ConnectorSession session, @SqlType(StandardTypes.TIMESTAMP) long left, @SqlType(StandardTypes.INTERVAL_YEAR_TO_MONTH) long right)
{
return getChronology(session.getTimeZoneKey()).monthOfYear().add(left, -right);
}
@ScalarOperator(SUBTRACT)
@SqlType(StandardTypes.TIMESTAMP_WITH_TIME_ZONE)
public static long timestampWithTimeZoneMinusIntervalYearToMonth(@SqlType(StandardTypes.TIMESTAMP_WITH_TIME_ZONE) long left, @SqlType(StandardTypes.INTERVAL_YEAR_TO_MONTH) long right)
{
long dateTimeWithTimeZone = unpackChronology(left).monthOfYear().add(unpackMillisUtc(left), -right);
return updateMillisUtc(dateTimeWithTimeZone, left);
}
public static int modulo24Hour(ISOChronology chronology, long millis)
{
return chronology.millisOfDay().get(millis) - chronology.getZone().getOffset(millis);
}
}