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

org.apache.hudi.common.table.read.HoodieFileGroupReader 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.common.table.read;

import org.apache.hudi.common.config.HoodieMemoryConfig;
import org.apache.hudi.common.config.HoodieReaderConfig;
import org.apache.hudi.common.config.RecordMergeMode;
import org.apache.hudi.common.config.TypedProperties;
import org.apache.hudi.common.engine.HoodieReaderContext;
import org.apache.hudi.common.model.BaseFile;
import org.apache.hudi.common.model.FileSlice;
import org.apache.hudi.common.model.HoodieBaseFile;
import org.apache.hudi.common.model.HoodieLogFile;
import org.apache.hudi.common.table.HoodieTableConfig;
import org.apache.hudi.common.table.HoodieTableMetaClient;
import org.apache.hudi.common.table.log.HoodieMergedLogRecordReader;
import org.apache.hudi.common.util.ConfigUtils;
import org.apache.hudi.common.util.Option;
import org.apache.hudi.common.util.collection.CachingIterator;
import org.apache.hudi.common.util.collection.ClosableIterator;
import org.apache.hudi.common.util.collection.EmptyIterator;
import org.apache.hudi.common.util.collection.Pair;
import org.apache.hudi.exception.HoodieIOException;
import org.apache.hudi.internal.schema.InternalSchema;
import org.apache.hudi.storage.HoodieStorage;
import org.apache.hudi.storage.StoragePath;
import org.apache.hudi.storage.StoragePathInfo;

import org.apache.avro.Schema;

import java.io.Closeable;
import java.io.IOException;
import java.util.List;
import java.util.function.UnaryOperator;
import java.util.stream.Collectors;

import static org.apache.hudi.common.config.HoodieReaderConfig.RECORD_MERGE_IMPL_CLASSES_DEPRECATED_WRITE_CONFIG_KEY;
import static org.apache.hudi.common.config.HoodieReaderConfig.RECORD_MERGE_IMPL_CLASSES_WRITE_CONFIG_KEY;
import static org.apache.hudi.common.fs.FSUtils.getRelativePartitionPath;
import static org.apache.hudi.common.util.ConfigUtils.getIntWithAltKeys;

/**
 * A file group reader that iterates through the records in a single file group.
 * 

* This should be used by the every engine integration, by plugging in a * {@link HoodieReaderContext} implementation. * * @param The type of engine-specific record representation, e.g.,{@code InternalRow} * in Spark and {@code RowData} in Flink. */ public final class HoodieFileGroupReader implements Closeable { private final HoodieReaderContext readerContext; private final Option hoodieBaseFileOption; private final List logFiles; private final HoodieStorage storage; private final TypedProperties props; // Byte offset to start reading from the base file private final long start; // Length of bytes to read from the base file private final long length; // Core structure to store and process records. private final HoodieFileGroupRecordBuffer recordBuffer; private ClosableIterator baseFileIterator; private final Option> outputConverter; private final HoodieReadStats readStats; public HoodieFileGroupReader(HoodieReaderContext readerContext, HoodieStorage storage, String tablePath, String latestCommitTime, FileSlice fileSlice, Schema dataSchema, Schema requestedSchema, Option internalSchemaOpt, HoodieTableMetaClient hoodieTableMetaClient, TypedProperties props, long start, long length, boolean shouldUseRecordPosition) { this.readerContext = readerContext; this.storage = storage; this.hoodieBaseFileOption = fileSlice.getBaseFile(); this.logFiles = fileSlice.getLogFiles().sorted(HoodieLogFile.getLogFileComparator()).collect(Collectors.toList()); this.props = props; this.start = start; this.length = length; HoodieTableConfig tableConfig = hoodieTableMetaClient.getTableConfig(); readerContext.setRecordMerger(readerContext.getRecordMerger( tableConfig.getRecordMergeMode(), tableConfig.getRecordMergeStrategyId(), props.getString(RECORD_MERGE_IMPL_CLASSES_WRITE_CONFIG_KEY, props.getString(RECORD_MERGE_IMPL_CLASSES_DEPRECATED_WRITE_CONFIG_KEY, "")))); readerContext.setTablePath(tablePath); readerContext.setLatestCommitTime(latestCommitTime); boolean isSkipMerge = ConfigUtils.getStringWithAltKeys(props, HoodieReaderConfig.MERGE_TYPE, true).equalsIgnoreCase(HoodieReaderConfig.REALTIME_SKIP_MERGE); readerContext.setShouldMergeUseRecordPosition(shouldUseRecordPosition && !isSkipMerge); readerContext.setHasLogFiles(!this.logFiles.isEmpty()); if (readerContext.getHasLogFiles() && start != 0) { throw new IllegalArgumentException("Filegroup reader is doing log file merge but not reading from the start of the base file"); } readerContext.setHasBootstrapBaseFile(hoodieBaseFileOption.isPresent() && hoodieBaseFileOption.get().getBootstrapBaseFile().isPresent()); readerContext.setSchemaHandler(readerContext.supportsParquetRowIndex() ? new HoodiePositionBasedSchemaHandler<>(readerContext, dataSchema, requestedSchema, internalSchemaOpt, tableConfig, props) : new HoodieFileGroupReaderSchemaHandler<>(readerContext, dataSchema, requestedSchema, internalSchemaOpt, tableConfig, props)); this.outputConverter = readerContext.getSchemaHandler().getOutputConverter(); this.readStats = new HoodieReadStats(); this.recordBuffer = getRecordBuffer(readerContext, hoodieTableMetaClient, tableConfig.getRecordMergeMode(), props, this.logFiles.isEmpty(), isSkipMerge, shouldUseRecordPosition, readStats); } /** * Initialize correct record buffer */ private static HoodieFileGroupRecordBuffer getRecordBuffer(HoodieReaderContext readerContext, HoodieTableMetaClient hoodieTableMetaClient, RecordMergeMode recordMergeMode, TypedProperties props, boolean hasNoLogFiles, boolean isSkipMerge, boolean shouldUseRecordPosition, HoodieReadStats readStats) { if (hasNoLogFiles) { return null; } else if (isSkipMerge) { return new HoodieUnmergedFileGroupRecordBuffer<>( readerContext, hoodieTableMetaClient, recordMergeMode, Option.empty(), Option.empty(), props, readStats); } else if (shouldUseRecordPosition) { return new HoodiePositionBasedFileGroupRecordBuffer<>( readerContext, hoodieTableMetaClient, recordMergeMode, Option.empty(), Option.empty(), props, readStats); } else { return new HoodieKeyBasedFileGroupRecordBuffer<>( readerContext, hoodieTableMetaClient, recordMergeMode, Option.empty(), Option.empty(), props, readStats); } } /** * Initialize internal iterators on the base and log files. */ public void initRecordIterators() throws IOException { ClosableIterator iter = makeBaseFileIterator(); if (logFiles.isEmpty()) { this.baseFileIterator = CachingIterator.wrap(iter, readerContext); } else { this.baseFileIterator = iter; scanLogFiles(); recordBuffer.setBaseFileIterator(baseFileIterator); } } private ClosableIterator makeBaseFileIterator() throws IOException { if (!hoodieBaseFileOption.isPresent()) { return new EmptyIterator<>(); } HoodieBaseFile baseFile = hoodieBaseFileOption.get(); if (baseFile.getBootstrapBaseFile().isPresent()) { return makeBootstrapBaseFileIterator(baseFile); } StoragePathInfo baseFileStoragePathInfo = baseFile.getPathInfo(); if (baseFileStoragePathInfo != null) { return readerContext.getFileRecordIterator( baseFileStoragePathInfo, start, length, readerContext.getSchemaHandler().getDataSchema(), readerContext.getSchemaHandler().getRequiredSchema(), storage); } else { return readerContext.getFileRecordIterator( baseFile.getStoragePath(), start, length, readerContext.getSchemaHandler().getDataSchema(), readerContext.getSchemaHandler().getRequiredSchema(), storage); } } private ClosableIterator makeBootstrapBaseFileIterator(HoodieBaseFile baseFile) throws IOException { BaseFile dataFile = baseFile.getBootstrapBaseFile().get(); Pair, List> requiredFields = readerContext.getSchemaHandler().getBootstrapRequiredFields(); Pair, List> allFields = readerContext.getSchemaHandler().getBootstrapDataFields(); Option, Schema>> dataFileIterator = makeBootstrapBaseFileIteratorHelper(requiredFields.getRight(), allFields.getRight(), dataFile); Option, Schema>> skeletonFileIterator = makeBootstrapBaseFileIteratorHelper(requiredFields.getLeft(), allFields.getLeft(), baseFile); if (!dataFileIterator.isPresent() && !skeletonFileIterator.isPresent()) { throw new IllegalStateException("should not be here if only partition cols are required"); } else if (!dataFileIterator.isPresent()) { return skeletonFileIterator.get().getLeft(); } else if (!skeletonFileIterator.isPresent()) { return dataFileIterator.get().getLeft(); } else { if (start != 0) { throw new IllegalArgumentException("Filegroup reader is doing bootstrap merge but we are not reading from the start of the base file"); } return readerContext.mergeBootstrapReaders(skeletonFileIterator.get().getLeft(), skeletonFileIterator.get().getRight(), dataFileIterator.get().getLeft(), dataFileIterator.get().getRight()); } } /** * Creates file record iterator to read bootstrap skeleton or data file * * @param requiredFields list of fields that are expected to be read from the file * @param allFields list of all fields in the data file to be read * @param file file to be read * @return pair of the record iterator of the file, and the schema of the data being read */ private Option, Schema>> makeBootstrapBaseFileIteratorHelper(List requiredFields, List allFields, BaseFile file) throws IOException { if (requiredFields.isEmpty()) { return Option.empty(); } Schema requiredSchema = readerContext.getSchemaHandler().createSchemaFromFields(requiredFields); StoragePathInfo fileStoragePathInfo = file.getPathInfo(); if (fileStoragePathInfo != null) { return Option.of(Pair.of(readerContext.getFileRecordIterator(fileStoragePathInfo, 0, file.getFileLen(), readerContext.getSchemaHandler().createSchemaFromFields(allFields), requiredSchema, storage), requiredSchema)); } else { return Option.of(Pair.of(readerContext.getFileRecordIterator(file.getStoragePath(), 0, file.getFileLen(), readerContext.getSchemaHandler().createSchemaFromFields(allFields), requiredSchema, storage), requiredSchema)); } } /** * @return {@code true} if the next record exists; {@code false} otherwise. * @throws IOException on reader error. */ public boolean hasNext() throws IOException { if (recordBuffer == null) { return baseFileIterator.hasNext(); } else { return recordBuffer.hasNext(); } } /** * @return statistics of reading a file group. */ public HoodieReadStats getStats() { return readStats; } /** * @return The next record after calling {@link #hasNext}. */ public T next() { T nextVal = recordBuffer == null ? baseFileIterator.next() : recordBuffer.next(); if (outputConverter.isPresent()) { return outputConverter.get().apply(nextVal); } return nextVal; } private void scanLogFiles() { String path = readerContext.getTablePath(); try (HoodieMergedLogRecordReader logRecordReader = HoodieMergedLogRecordReader.newBuilder() .withHoodieReaderContext(readerContext) .withStorage(storage) .withLogFiles(logFiles) .withReverseReader(false) .withBufferSize(getIntWithAltKeys(props, HoodieMemoryConfig.MAX_DFS_STREAM_BUFFER_SIZE)) .withPartition(getRelativePartitionPath( new StoragePath(path), logFiles.get(0).getPath().getParent())) .withRecordBuffer(recordBuffer) .build()) { readStats.setTotalLogReadTimeMs(logRecordReader.getTotalTimeTakenToReadAndMergeBlocks()); readStats.setTotalUpdatedRecordsCompacted(logRecordReader.getNumMergedRecordsInLog()); readStats.setTotalLogFilesCompacted(logRecordReader.getTotalLogFiles()); readStats.setTotalLogRecords(logRecordReader.getTotalLogRecords()); readStats.setTotalLogBlocks(logRecordReader.getTotalLogBlocks()); readStats.setTotalCorruptLogBlock(logRecordReader.getTotalCorruptBlocks()); readStats.setTotalRollbackBlocks(logRecordReader.getTotalRollbacks()); } } @Override public void close() throws IOException { if (baseFileIterator != null) { baseFileIterator.close(); } if (recordBuffer != null) { recordBuffer.close(); } } public HoodieFileGroupReaderIterator getClosableIterator() { return new HoodieFileGroupReaderIterator<>(this); } public static class HoodieFileGroupReaderIterator implements ClosableIterator { private HoodieFileGroupReader reader; public HoodieFileGroupReaderIterator(HoodieFileGroupReader reader) { this.reader = reader; } @Override public boolean hasNext() { try { return reader.hasNext(); } catch (IOException e) { throw new HoodieIOException("Failed to read record", e); } } @Override public T next() { return reader.next(); } @Override public void close() { try { reader.close(); } catch (IOException e) { throw new HoodieIOException("Failed to close the reader", e); } finally { this.reader = null; } } } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy