org.apache.iceberg.flink.source.RowDataFileScanTaskReader Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of iceberg-flink-1.15 Show documentation
Show all versions of iceberg-flink-1.15 Show documentation
A table format for huge analytic datasets
The newest version!
/*
* 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.flink.source;
import java.util.List;
import java.util.Map;
import org.apache.flink.annotation.Internal;
import org.apache.flink.table.data.RowData;
import org.apache.flink.table.types.logical.RowType;
import org.apache.iceberg.FileScanTask;
import org.apache.iceberg.MetadataColumns;
import org.apache.iceberg.Schema;
import org.apache.iceberg.StructLike;
import org.apache.iceberg.avro.Avro;
import org.apache.iceberg.data.DeleteFilter;
import org.apache.iceberg.encryption.InputFilesDecryptor;
import org.apache.iceberg.expressions.Expression;
import org.apache.iceberg.expressions.Expressions;
import org.apache.iceberg.flink.FlinkSchemaUtil;
import org.apache.iceberg.flink.FlinkSourceFilter;
import org.apache.iceberg.flink.RowDataWrapper;
import org.apache.iceberg.flink.data.FlinkAvroReader;
import org.apache.iceberg.flink.data.FlinkOrcReader;
import org.apache.iceberg.flink.data.FlinkParquetReaders;
import org.apache.iceberg.flink.data.RowDataProjection;
import org.apache.iceberg.flink.data.RowDataUtil;
import org.apache.iceberg.io.CloseableIterable;
import org.apache.iceberg.io.CloseableIterator;
import org.apache.iceberg.io.InputFile;
import org.apache.iceberg.mapping.NameMappingParser;
import org.apache.iceberg.orc.ORC;
import org.apache.iceberg.parquet.Parquet;
import org.apache.iceberg.relocated.com.google.common.collect.ImmutableMap;
import org.apache.iceberg.relocated.com.google.common.collect.Sets;
import org.apache.iceberg.types.TypeUtil;
import org.apache.iceberg.util.PartitionUtil;
@Internal
public class RowDataFileScanTaskReader implements FileScanTaskReader {
private final Schema tableSchema;
private final Schema projectedSchema;
private final String nameMapping;
private final boolean caseSensitive;
private final FlinkSourceFilter rowFilter;
public RowDataFileScanTaskReader(
Schema tableSchema,
Schema projectedSchema,
String nameMapping,
boolean caseSensitive,
List filters) {
this.tableSchema = tableSchema;
this.projectedSchema = projectedSchema;
this.nameMapping = nameMapping;
this.caseSensitive = caseSensitive;
if (filters != null && !filters.isEmpty()) {
Expression combinedExpression =
filters.stream().reduce(Expressions.alwaysTrue(), Expressions::and);
this.rowFilter =
new FlinkSourceFilter(this.projectedSchema, combinedExpression, this.caseSensitive);
} else {
this.rowFilter = null;
}
}
@Override
public CloseableIterator open(
FileScanTask task, InputFilesDecryptor inputFilesDecryptor) {
Schema partitionSchema = TypeUtil.select(projectedSchema, task.spec().identitySourceIds());
Map idToConstant =
partitionSchema.columns().isEmpty()
? ImmutableMap.of()
: PartitionUtil.constantsMap(task, RowDataUtil::convertConstant);
FlinkDeleteFilter deletes =
new FlinkDeleteFilter(task, tableSchema, projectedSchema, inputFilesDecryptor);
CloseableIterable iterable =
deletes.filter(
newIterable(task, deletes.requiredSchema(), idToConstant, inputFilesDecryptor));
// Project the RowData to remove the extra meta columns.
if (!projectedSchema.sameSchema(deletes.requiredSchema())) {
RowDataProjection rowDataProjection =
RowDataProjection.create(
deletes.requiredRowType(),
deletes.requiredSchema().asStruct(),
projectedSchema.asStruct());
iterable = CloseableIterable.transform(iterable, rowDataProjection::wrap);
}
return iterable.iterator();
}
private CloseableIterable newIterable(
FileScanTask task,
Schema schema,
Map idToConstant,
InputFilesDecryptor inputFilesDecryptor) {
CloseableIterable iter;
if (task.isDataTask()) {
throw new UnsupportedOperationException("Cannot read data task.");
} else {
switch (task.file().format()) {
case PARQUET:
iter = newParquetIterable(task, schema, idToConstant, inputFilesDecryptor);
break;
case AVRO:
iter = newAvroIterable(task, schema, idToConstant, inputFilesDecryptor);
break;
case ORC:
iter = newOrcIterable(task, schema, idToConstant, inputFilesDecryptor);
break;
default:
throw new UnsupportedOperationException(
"Cannot read unknown format: " + task.file().format());
}
}
if (rowFilter != null) {
return CloseableIterable.filter(iter, rowFilter::filter);
}
return iter;
}
private CloseableIterable newAvroIterable(
FileScanTask task,
Schema schema,
Map idToConstant,
InputFilesDecryptor inputFilesDecryptor) {
Avro.ReadBuilder builder =
Avro.read(inputFilesDecryptor.getInputFile(task))
.reuseContainers()
.project(schema)
.split(task.start(), task.length())
.createReaderFunc(readSchema -> new FlinkAvroReader(schema, readSchema, idToConstant));
if (nameMapping != null) {
builder.withNameMapping(NameMappingParser.fromJson(nameMapping));
}
return builder.build();
}
private CloseableIterable newParquetIterable(
FileScanTask task,
Schema schema,
Map idToConstant,
InputFilesDecryptor inputFilesDecryptor) {
Parquet.ReadBuilder builder =
Parquet.read(inputFilesDecryptor.getInputFile(task))
.split(task.start(), task.length())
.project(schema)
.createReaderFunc(
fileSchema -> FlinkParquetReaders.buildReader(schema, fileSchema, idToConstant))
.filter(task.residual())
.caseSensitive(caseSensitive)
.reuseContainers();
if (nameMapping != null) {
builder.withNameMapping(NameMappingParser.fromJson(nameMapping));
}
return builder.build();
}
private CloseableIterable newOrcIterable(
FileScanTask task,
Schema schema,
Map idToConstant,
InputFilesDecryptor inputFilesDecryptor) {
Schema readSchemaWithoutConstantAndMetadataFields =
TypeUtil.selectNot(
schema, Sets.union(idToConstant.keySet(), MetadataColumns.metadataFieldIds()));
ORC.ReadBuilder builder =
ORC.read(inputFilesDecryptor.getInputFile(task))
.project(readSchemaWithoutConstantAndMetadataFields)
.split(task.start(), task.length())
.createReaderFunc(
readOrcSchema -> new FlinkOrcReader(schema, readOrcSchema, idToConstant))
.filter(task.residual())
.caseSensitive(caseSensitive);
if (nameMapping != null) {
builder.withNameMapping(NameMappingParser.fromJson(nameMapping));
}
return builder.build();
}
private static class FlinkDeleteFilter extends DeleteFilter {
private final RowType requiredRowType;
private final RowDataWrapper asStructLike;
private final InputFilesDecryptor inputFilesDecryptor;
FlinkDeleteFilter(
FileScanTask task,
Schema tableSchema,
Schema requestedSchema,
InputFilesDecryptor inputFilesDecryptor) {
super(task.file().path().toString(), task.deletes(), tableSchema, requestedSchema);
this.requiredRowType = FlinkSchemaUtil.convert(requiredSchema());
this.asStructLike = new RowDataWrapper(requiredRowType, requiredSchema().asStruct());
this.inputFilesDecryptor = inputFilesDecryptor;
}
public RowType requiredRowType() {
return requiredRowType;
}
@Override
protected StructLike asStructLike(RowData row) {
return asStructLike.wrap(row);
}
@Override
protected InputFile getInputFile(String location) {
return inputFilesDecryptor.getInputFile(location);
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy