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

org.h2.expression.TimeZoneOperation Maven / Gradle / Ivy

There is a newer version: 1.2.2.1-jre17
Show newest version
/*
 * Copyright 2004-2019 H2 Group. Multiple-Licensed under the MPL 2.0,
 * and the EPL 1.0 (https://h2database.com/html/license.html).
 * Initial Developer: H2 Group
 */
package org.h2.expression;

import org.h2.engine.Session;
import org.h2.message.DbException;
import org.h2.table.ColumnResolver;
import org.h2.table.TableFilter;
import org.h2.util.DateTimeUtils;
import org.h2.util.TimeZoneProvider;
import org.h2.value.DataType;
import org.h2.value.TypeInfo;
import org.h2.value.Value;
import org.h2.value.ValueInterval;
import org.h2.value.ValueNull;
import org.h2.value.ValueTimeTimeZone;
import org.h2.value.ValueTimestamp;
import org.h2.value.ValueTimestampTimeZone;

/**
 * A time zone specification (AT { TIME ZONE | LOCAL }).
 */
public class TimeZoneOperation extends Expression {

    private Expression arg;
    private Expression timeZone;
    private TypeInfo type;

    public TimeZoneOperation(Expression arg) {
        this.arg = arg;
    }

    public TimeZoneOperation(Expression arg, Expression timeZone) {
        this.arg = arg;
        this.timeZone = timeZone;
    }

    @Override
    public StringBuilder getSQL(StringBuilder builder, boolean alwaysQuote) {
        arg.getSQL(builder.append('('), alwaysQuote).append(" AT ");
        if (timeZone != null) {
            timeZone.getSQL(builder.append("TIME ZONE "), alwaysQuote);
        } else {
            builder.append("LOCAL");
        }
        return builder.append(')');
    }

    @Override
    public Value getValue(Session session) {
        Value a = arg.getValue(session).convertTo(type, session, false, null);
        int valueType = a.getValueType();
        if ((valueType == Value.TIMESTAMP_TZ || valueType == Value.TIME_TZ) && timeZone != null) {
            Value b = timeZone.getValue(session);
            if (b != ValueNull.INSTANCE) {
                if (valueType == Value.TIMESTAMP_TZ) {
                    ValueTimestampTimeZone v = (ValueTimestampTimeZone) a;
                    long dateValue = v.getDateValue();
                    long timeNanos = v.getTimeNanos();
                    int offsetSeconds = v.getTimeZoneOffsetSeconds();
                    int newOffset = parseTimeZone(b, dateValue, timeNanos, offsetSeconds, true);
                    if (offsetSeconds != newOffset) {
                        timeNanos += (newOffset - offsetSeconds) * DateTimeUtils.NANOS_PER_SECOND;
                        // Value can be 18+18 hours before or after the limit
                        if (timeNanos < 0) {
                            timeNanos += DateTimeUtils.NANOS_PER_DAY;
                            dateValue = DateTimeUtils.decrementDateValue(dateValue);
                            if (timeNanos < 0) {
                                timeNanos += DateTimeUtils.NANOS_PER_DAY;
                                dateValue = DateTimeUtils.decrementDateValue(dateValue);
                            }
                        } else if (timeNanos >= DateTimeUtils.NANOS_PER_DAY) {
                            timeNanos -= DateTimeUtils.NANOS_PER_DAY;
                            dateValue = DateTimeUtils.incrementDateValue(dateValue);
                            if (timeNanos >= DateTimeUtils.NANOS_PER_DAY) {
                                timeNanos -= DateTimeUtils.NANOS_PER_DAY;
                                dateValue = DateTimeUtils.incrementDateValue(dateValue);
                            }
                        }
                        a = ValueTimestampTimeZone.fromDateValueAndNanos(dateValue, timeNanos, newOffset);
                    }
                } else {
                    ValueTimeTimeZone v = (ValueTimeTimeZone) a;
                    long timeNanos = v.getNanos();
                    int offsetSeconds = v.getTimeZoneOffsetSeconds();
                    int newOffset = parseTimeZone(b, DateTimeUtils.EPOCH_DATE_VALUE, timeNanos, offsetSeconds, false);
                    if (offsetSeconds != newOffset) {
                        timeNanos += (newOffset - offsetSeconds) * DateTimeUtils.NANOS_PER_SECOND;
                        a = ValueTimeTimeZone.fromNanos(DateTimeUtils.normalizeNanosOfDay(timeNanos), newOffset);
                    }
                }
            } else {
                a = ValueNull.INSTANCE;
            }
        }
        return a;
    }

    private static int parseTimeZone(Value b, long dateValue, long timeNanos, int offsetSeconds,
            boolean allowTimeZoneName) {
        int timeZoneType = b.getValueType();
        if (DataType.isStringType(timeZoneType)) {
            String s = b.getString();
            if (s.equals("Z") || s.equals("UTC") || s.equals("GMT")) {
                return 0;
            } else if (!s.isEmpty()) {
                char c = s.charAt(0);
                if (c != '+' && c != '-' && (c < '0' || c > '9')) {
                    TimeZoneProvider timeZone;
                    try {
                        timeZone = TimeZoneProvider.ofId(s);
                    } catch (IllegalArgumentException ex) {
                        throw DbException.getInvalidValueException("time zone", b.getSQL());
                    }
                    if (!allowTimeZoneName && !timeZone.hasFixedOffset()) {
                        throw DbException.getInvalidValueException("time zone", b.getSQL());
                    }
                    return timeZone
                            .getTimeZoneOffsetUTC(DateTimeUtils.getEpochSeconds(dateValue, timeNanos, offsetSeconds));
                }
            }
        }
        return parseInterval(b);
    }

    private static int parseInterval(Value b) {
        ValueInterval i = (ValueInterval) b.convertTo(Value.INTERVAL_HOUR_TO_SECOND);
        long h = i.getLeading(), seconds = i.getRemaining();
        if (h > 18 || h == 18 && seconds != 0 || seconds % DateTimeUtils.NANOS_PER_SECOND != 0) {
            throw DbException.getInvalidValueException("time zone", i.getSQL());
        }
        int newOffset = (int) (h * 3_600 + seconds / DateTimeUtils.NANOS_PER_SECOND);
        if (i.isNegative()) {
            newOffset = -newOffset;
        }
        return newOffset;
    }

    @Override
    public void mapColumns(ColumnResolver resolver, int level, int state) {
        arg.mapColumns(resolver, level, state);
        if (timeZone != null) {
            timeZone.mapColumns(resolver, level, state);
        }
    }

    @Override
    public Expression optimize(Session session) {
        arg = arg.optimize(session);
        if (timeZone != null) {
            timeZone = timeZone.optimize(session);
        }
        TypeInfo type = arg.getType();
        int valueType = Value.TIMESTAMP_TZ, scale = ValueTimestamp.MAXIMUM_SCALE;
        switch (type.getValueType()) {
        case Value.TIMESTAMP:
        case Value.TIMESTAMP_TZ:
            scale = type.getScale();
            break;
        case Value.TIME:
        case Value.TIME_TZ:
            valueType = Value.TIME_TZ;
            scale = type.getScale();
            break;
        default:
            StringBuilder builder = arg.getSQL(new StringBuilder(), false);
            int offset = builder.length();
            builder.append(" AT ");
            if (timeZone != null) {
                timeZone.getSQL(builder.append("TIME ZONE "), false);
            } else {
                builder.append("LOCAL");
            }
            throw DbException.getSyntaxError(builder.toString(), offset, "time, timestamp");
        }
        this.type = TypeInfo.getTypeInfo(valueType, -1, scale, null);
        if (arg.isConstant() && (timeZone == null || timeZone.isConstant())) {
            return ValueExpression.get(getValue(session));
        }
        return this;
    }

    @Override
    public void setEvaluatable(TableFilter tableFilter, boolean b) {
        arg.setEvaluatable(tableFilter, b);
        if (timeZone != null) {
            timeZone.setEvaluatable(tableFilter, b);
        }
    }

    @Override
    public TypeInfo getType() {
        return type;
    }

    @Override
    public void updateAggregate(Session session, int stage) {
        arg.updateAggregate(session, stage);
        if (timeZone != null) {
            timeZone.updateAggregate(session, stage);
        }
    }

    @Override
    public boolean isEverything(ExpressionVisitor visitor) {
        return arg.isEverything(visitor) && (timeZone == null || timeZone.isEverything(visitor));
    }

    @Override
    public int getCost() {
        int cost = arg.getCost() + 1;
        if (timeZone != null) {
            cost += timeZone.getCost();
        }
        return cost;
    }

    @Override
    public int getSubexpressionCount() {
        return timeZone != null ? 2 : 1;
    }

    @Override
    public Expression getSubexpression(int index) {
        if (index == 0) {
            return arg;
        }
        if (index == 1 && timeZone != null) {
            return timeZone;
        }
        throw new IndexOutOfBoundsException();
    }

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy