blob: 660a182f0e78230af933c2a944205ad7a89487db [file] [log] [blame]
/*
* Licensed to the Apache Software Foundation (ASF) under one
* or more contributor license agreements. See the NOTICE file
* distributed with this work for additional information
* regarding copyright ownership. The ASF licenses this file
* to you 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 org.apache.iceberg.spark.functions;
import org.apache.iceberg.util.DateTimeUtil;
import org.apache.spark.sql.catalyst.InternalRow;
import org.apache.spark.sql.connector.catalog.functions.BoundFunction;
import org.apache.spark.sql.types.DataType;
import org.apache.spark.sql.types.DataTypes;
import org.apache.spark.sql.types.TimestampNTZType;
import org.apache.spark.sql.types.TimestampType;
/**
* A Spark function implementation for the Iceberg hour transform.
*
* <p>Example usage: {@code SELECT system.hours('source_col')}.
*/
public class HoursFunction extends UnaryUnboundFunction {
@Override
protected BoundFunction doBind(DataType valueType) {
if (valueType instanceof TimestampType) {
return new TimestampToHoursFunction();
} else if (valueType instanceof TimestampNTZType) {
return new TimestampNtzToHoursFunction();
} else {
throw new UnsupportedOperationException(
"Expected value to be timestamp: " + valueType.catalogString());
}
}
@Override
public String description() {
return name()
+ "(col) - Call Iceberg's hour transform\n"
+ " col :: source column (must be timestamp)";
}
@Override
public String name() {
return "hours";
}
public static class TimestampToHoursFunction extends BaseScalarFunction<Integer> {
// magic method used in codegen
public static int invoke(long micros) {
return DateTimeUtil.microsToHours(micros);
}
@Override
public String name() {
return "hours";
}
@Override
public DataType[] inputTypes() {
return new DataType[] {DataTypes.TimestampType};
}
@Override
public DataType resultType() {
return DataTypes.IntegerType;
}
@Override
public String canonicalName() {
return "iceberg.hours(timestamp)";
}
@Override
public Integer produceResult(InternalRow input) {
// return null for null input to match what Spark does in codegen
return input.isNullAt(0) ? null : invoke(input.getLong(0));
}
}
public static class TimestampNtzToHoursFunction extends BaseScalarFunction<Integer> {
// magic method used in codegen
public static int invoke(long micros) {
return DateTimeUtil.microsToHours(micros);
}
@Override
public String name() {
return "hours";
}
@Override
public DataType[] inputTypes() {
return new DataType[] {DataTypes.TimestampNTZType};
}
@Override
public DataType resultType() {
return DataTypes.IntegerType;
}
@Override
public String canonicalName() {
return "iceberg.hours(timestamp_ntz)";
}
@Override
public Integer produceResult(InternalRow input) {
// return null for null input to match what Spark does in codegen
return input.isNullAt(0) ? null : invoke(input.getLong(0));
}
}
}