org.apache.hudi.table.format.CastMap Maven / Gradle / Ivy
/*
* 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.hudi.table.format;
import org.apache.hudi.common.util.Option;
import org.apache.hudi.table.format.TypeConverters.TypeConverter;
import org.apache.hudi.util.RowDataCastProjection;
import org.apache.hudi.util.RowDataProjection;
import org.apache.flink.annotation.VisibleForTesting;
import org.apache.flink.table.types.DataType;
import org.apache.flink.table.types.logical.LogicalType;
import java.io.Serializable;
import java.util.HashMap;
import java.util.Map;
import java.util.stream.Collectors;
/**
* CastMap is responsible for conversion of flink types when full schema evolution enabled.
*
* Supported cast conversions:
*
* - Integer => Long, Float, Double, Decimal, String
* - Long => Float, Double, Decimal, String
* - Float => Double, Decimal, String
* - Double => Decimal, String
* - Decimal => Decimal, String
* - String => Decimal, Date
* - Date => String
*
*/
public final class CastMap implements Serializable {
private static final long serialVersionUID = 1L;
// Maps position to corresponding cast
private final Map castMap = new HashMap<>();
private DataType[] fileFieldTypes;
public Option toRowDataProjection(int[] selectedFields) {
if (castMap.isEmpty()) {
return Option.empty();
}
LogicalType[] requiredType = new LogicalType[selectedFields.length];
for (int i = 0; i < selectedFields.length; i++) {
requiredType[i] = fileFieldTypes[selectedFields[i]].getLogicalType();
}
return Option.of(new RowDataCastProjection(requiredType, this));
}
public Object castIfNeeded(int pos, Object val) {
Cast cast = castMap.get(pos);
if (cast == null) {
return val;
}
return cast.convert(val);
}
public DataType[] getFileFieldTypes() {
return fileFieldTypes;
}
public void setFileFieldTypes(DataType[] fileFieldTypes) {
this.fileFieldTypes = fileFieldTypes;
}
@VisibleForTesting
void add(int pos, LogicalType fromType, LogicalType toType) {
TypeConverter converter = TypeConverters.getInstance(fromType, toType);
if (converter == null) {
throw new IllegalArgumentException(String.format("Cannot create cast %s => %s at pos %s", fromType, toType, pos));
}
add(pos, new Cast(fromType, toType, converter));
}
private void add(int pos, Cast cast) {
castMap.put(pos, cast);
}
/**
* Fields {@link Cast#from} and {@link Cast#to} are redundant due to {@link Cast#convert(Object)} determines conversion.
* However, it is convenient to debug {@link CastMap} when {@link Cast#toString()} prints types.
*/
private static final class Cast implements Serializable {
private static final long serialVersionUID = 1L;
private final LogicalType from;
private final LogicalType to;
private final TypeConverter castMapConverter;
Cast(LogicalType from, LogicalType to, TypeConverter conversion) {
this.from = from;
this.to = to;
this.castMapConverter = conversion;
}
Object convert(Object val) {
return castMapConverter.convert(val);
}
@Override
public String toString() {
return from + " => " + to;
}
}
@Override
public String toString() {
return castMap.entrySet().stream()
.map(e -> e.getKey() + ": " + e.getValue())
.collect(Collectors.joining(", ", "{", "}"));
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy