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

org.apache.hudi.io.hadoop.HoodieHBaseAvroHFileReader 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.io.hadoop;

import org.apache.hudi.common.bloom.BloomFilter;
import org.apache.hudi.common.bloom.BloomFilterFactory;
import org.apache.hudi.common.model.HoodieAvroIndexedRecord;
import org.apache.hudi.common.model.HoodieRecord;
import org.apache.hudi.common.model.HoodieRecordLocation;
import org.apache.hudi.common.util.Option;
import org.apache.hudi.common.util.VisibleForTesting;
import org.apache.hudi.common.util.collection.ClosableIterator;
import org.apache.hudi.common.util.collection.CloseableMappingIterator;
import org.apache.hudi.common.util.collection.Pair;
import org.apache.hudi.exception.HoodieException;
import org.apache.hudi.exception.HoodieIOException;
import org.apache.hudi.io.storage.HoodieAvroHFileReaderImplBase;
import org.apache.hudi.io.storage.HoodieFileReader;
import org.apache.hudi.storage.HoodieStorage;
import org.apache.hudi.storage.StorageConfiguration;
import org.apache.hudi.storage.StoragePath;
import org.apache.hudi.storage.hadoop.HoodieHadoopStorage;
import org.apache.hudi.util.Lazy;

import org.apache.avro.Schema;
import org.apache.avro.generic.GenericRecord;
import org.apache.avro.generic.IndexedRecord;
import org.apache.hadoop.conf.Configuration;
import org.apache.hadoop.hbase.Cell;
import org.apache.hadoop.hbase.KeyValue;
import org.apache.hadoop.hbase.io.hfile.CacheConfig;
import org.apache.hadoop.hbase.io.hfile.HFile;
import org.apache.hadoop.hbase.io.hfile.HFileInfo;
import org.apache.hadoop.hbase.io.hfile.HFileScanner;
import org.apache.hadoop.hbase.nio.ByteBuff;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import java.io.IOException;
import java.util.Arrays;
import java.util.Collections;
import java.util.Iterator;
import java.util.List;
import java.util.Objects;
import java.util.Set;
import java.util.SortedSet;
import java.util.TreeSet;
import java.util.stream.Collectors;

import static org.apache.hudi.common.util.StringUtils.getUTF8Bytes;
import static org.apache.hudi.common.util.TypeUtils.unsafeCast;

/**
 * NOTE: PLEASE READ DOCS & COMMENTS CAREFULLY BEFORE MAKING CHANGES
 * 

* {@link HoodieFileReader} implementation allowing to read from {@link HFile}. */ public class HoodieHBaseAvroHFileReader extends HoodieAvroHFileReaderImplBase { private static final Logger LOG = LoggerFactory.getLogger(HoodieHBaseAvroHFileReader.class); private final StoragePath path; private final HoodieStorage storage; private final StorageConfiguration storageConf; private final CacheConfig config; private final Option content; private final Lazy schema; // NOTE: Reader is ONLY THREAD-SAFE for {@code Scanner} operating in Positional Read ("pread") // mode (ie created w/ "pread = true") // Common reader is not used for the iterators since they can be closed independently. // Use {@link getSharedReader()} instead of accessing directly. private Option sharedReader; // NOTE: Scanner caches read blocks, therefore it's important to re-use scanner // wherever possible private Option sharedScanner; private final Object sharedLock = new Object(); public HoodieHBaseAvroHFileReader(StorageConfiguration storageConf, StoragePath path, Option schemaOpt) throws IOException { this(path, new HoodieHadoopStorage(path, storageConf), storageConf, schemaOpt, Option.empty()); } public HoodieHBaseAvroHFileReader(StorageConfiguration storageConf, StoragePath path, HoodieStorage storage, byte[] content, Option schemaOpt) throws IOException { this(path, storage, storageConf, schemaOpt, Option.of(content)); } public HoodieHBaseAvroHFileReader(StorageConfiguration storageConf, StoragePath path) throws IOException { this(storageConf, path, Option.empty()); } public HoodieHBaseAvroHFileReader(StoragePath path, HoodieStorage storage, StorageConfiguration storageConf, Option schemaOpt, Option content) throws IOException { this.path = path; this.storage = storage; this.storageConf = storageConf; this.config = new CacheConfig(storageConf.unwrapAs(Configuration.class)); this.content = content; // Shared reader is instantiated lazily. this.sharedReader = Option.empty(); this.sharedScanner = Option.empty(); this.schema = schemaOpt.map(Lazy::eagerly) .orElseGet(() -> Lazy.lazily(() -> fetchSchema(getSharedHFileReader()))); } @Override public ClosableIterator> getRecordsByKeysIterator(List sortedKeys, Schema schema) throws IOException { // Iterators do not use the shared reader or scanner // We're caching blocks for this scanner to minimize amount of traffic // to the underlying storage as we fetched (potentially) sparsely distributed // keys HFile.Reader reader = getHFileReader(); HFileScanner scanner = getHFileScanner(reader, true); ClosableIterator iterator = new RecordByKeyIterator(reader, scanner, sortedKeys, getSchema(), schema); return new CloseableMappingIterator<>(iterator, data -> unsafeCast(new HoodieAvroIndexedRecord(data))); } @Override public ClosableIterator> getRecordsByKeyPrefixIterator(List sortedKeyPrefixes, Schema schema) throws IOException { // Iterators do not use the shared reader or scanner ClosableIterator iterator = getIndexedRecordsByKeyPrefixIterator(sortedKeyPrefixes, schema); return new CloseableMappingIterator<>(iterator, data -> unsafeCast(new HoodieAvroIndexedRecord(data))); } @Override public String[] readMinMaxRecordKeys() { // NOTE: This access to reader is thread-safe HFileInfo fileInfo = getSharedHFileReader().getHFileInfo(); return new String[] {new String(fileInfo.get(getUTF8Bytes(KEY_MIN_RECORD))), new String(fileInfo.get(getUTF8Bytes(KEY_MAX_RECORD)))}; } @Override public BloomFilter readBloomFilter() { try { // NOTE: This access to reader is thread-safe HFileInfo fileInfo = getSharedHFileReader().getHFileInfo(); ByteBuff buf = getSharedHFileReader().getMetaBlock(KEY_BLOOM_FILTER_META_BLOCK, false).getBufferWithoutHeader(); // We have to copy bytes here, since we can't reuse buffer's underlying // array as is, since it contains additional metadata (header) byte[] bytes = new byte[buf.remaining()]; buf.get(bytes); return BloomFilterFactory.fromString(new String(bytes), new String(fileInfo.get(getUTF8Bytes(KEY_BLOOM_FILTER_TYPE_CODE)))); } catch (IOException e) { throw new HoodieException("Could not read bloom filter from " + path, e); } } @Override public Schema getSchema() { return schema.get(); } /** * Filter keys by availability. *

* Note: This method is performant when the caller passes in a sorted candidate keys. * * @param candidateRowKeys - Keys to check for the availability * @return Subset of candidate keys that are available */ @Override public Set> filterRowKeys(Set candidateRowKeys) { // candidateRowKeys must be sorted SortedSet sortedCandidateRowKeys = new TreeSet<>(candidateRowKeys); synchronized (sharedLock) { if (!sharedScanner.isPresent()) { // For shared scanner, which is primarily used for point-lookups, we're caching blocks // by default, to minimize amount of traffic to the underlying storage sharedScanner = Option.of(getHFileScanner(getSharedHFileReader(), true)); } return sortedCandidateRowKeys.stream() .filter(k -> { try { return isKeyAvailable(k, sharedScanner.get()); } catch (IOException e) { LOG.error("Failed to check key availability: " + k); return false; } }) // Record position is not supported for HFile .map(key -> Pair.of(key, HoodieRecordLocation.INVALID_POSITION)) .collect(Collectors.toSet()); } } @Override public ClosableIterator getIndexedRecordIterator(Schema readerSchema, Schema requestedSchema) { if (!Objects.equals(readerSchema, requestedSchema)) { throw new UnsupportedOperationException("Schema projections are not supported in HFile reader"); } HFile.Reader reader = getHFileReader(); // TODO eval whether seeking scanner would be faster than pread HFileScanner scanner = getHFileScanner(reader, false, false); return new RecordIterator(reader, scanner, getSchema(), readerSchema); } @VisibleForTesting public ClosableIterator getIndexedRecordsByKeysIterator(List keys, Schema readerSchema) throws IOException { // We're caching blocks for this scanner to minimize amount of traffic // to the underlying storage as we fetched (potentially) sparsely distributed // keys HFile.Reader reader = getHFileReader(); HFileScanner scanner = getHFileScanner(reader, true); return new RecordByKeyIterator(reader, scanner, keys, getSchema(), readerSchema); } @VisibleForTesting public ClosableIterator getIndexedRecordsByKeyPrefixIterator(List sortedKeyPrefixes, Schema readerSchema) throws IOException { // We're caching blocks for this scanner to minimize amount of traffic // to the underlying storage as we fetched (potentially) sparsely distributed // keys HFile.Reader reader = getHFileReader(); HFileScanner scanner = getHFileScanner(reader, true); return new RecordByKeyPrefixIterator(reader, scanner, sortedKeyPrefixes, getSchema(), readerSchema); } @Override public long getTotalRecords() { // NOTE: This access to reader is thread-safe return getSharedHFileReader().getEntries(); } @Override public void close() { try { synchronized (this) { if (sharedScanner.isPresent()) { sharedScanner.get().close(); } if (sharedReader.isPresent()) { sharedReader.get().close(); } } } catch (IOException e) { throw new HoodieIOException("Error closing the hfile reader", e); } } /** * Instantiates the shared HFile reader if not instantiated * @return the shared HFile reader */ private HFile.Reader getSharedHFileReader() { if (!sharedReader.isPresent()) { synchronized (sharedLock) { if (!sharedReader.isPresent()) { sharedReader = Option.of(getHFileReader()); } } } return sharedReader.get(); } /** * Instantiate a new reader for HFile files. * @return an instance of {@link HFile.Reader} */ private HFile.Reader getHFileReader() { if (content.isPresent()) { return HoodieHFileUtils.createHFileReader(storage, path, content.get()); } return HoodieHFileUtils.createHFileReader(storage, path, config, storageConf.unwrapAs(Configuration.class)); } private boolean isKeyAvailable(String key, HFileScanner keyScanner) throws IOException { final KeyValue kv = new KeyValue(getUTF8Bytes(key), null, null, null); return keyScanner.seekTo(kv) == 0; } private static Iterator getRecordByKeyPrefixIteratorInternal(HFileScanner scanner, String keyPrefix, Schema writerSchema, Schema readerSchema) throws IOException { KeyValue kv = new KeyValue(getUTF8Bytes(keyPrefix), null, null, null); // NOTE: HFile persists both keys/values as bytes, therefore lexicographical sorted is // essentially employed // // For the HFile containing list of cells c[0], c[1], ..., c[N], `seekTo(cell)` would return // following: // a) -1, if cell < c[0], no position; but w/ prefix search, if the key to be searched is less than first entry, the cursor will be placed at the beginning. // scanner.getCell() could return the next key which could match the key we are interested in. // b) 0, such that c[i] = cell and scanner is left in position i; // c) and 1, such that c[i] < cell, and scanner is left in position i. If key is somewhere in the middle, and if there is no exact match. So, scanner.next() // is expected to return the next entry which could potentially match the prefix we are looking for. If the key being looked up is > last entry in the file, // this could place the cursor in the end. hence we need to check for scanner.next() to ensure there are entries to read or if we reached EOF. // // Consider entries w/ the following keys in HFile: [key01, key02, key03, key04,..., key20]; // In case looked up key-prefix is // - "key", `reseekTo()` will return -1 and place the cursor just before "key01", // `getCell()` will return "key01" entry // - "key03", `reseekTo()` will return 0 (exact match) and place the cursor just before "key03", // `getCell()` will return "key03" entry // - "key1", `reseekTo()` will return 1 (first not lower than) and leave the cursor at key10 (assuming there is no exact match i.e. key1) // // Do remember that reseek will not do any rewind. So, after searching for key05(cursor is placed at key05), if you search for key01(less than current cursor position), // it may not return any results. int val = scanner.reseekTo(kv); if (val == 1) { // Try moving to next entry, matching the prefix key; if we're at the EOF, // `next()` will return false if (!scanner.next()) { return Collections.emptyIterator(); } } class KeyPrefixIterator implements Iterator { private IndexedRecord next = null; private boolean eof = false; @Override public boolean hasNext() { if (next != null) { return true; } else if (eof) { return false; } Cell c = Objects.requireNonNull(scanner.getCell()); byte[] keyBytes = copyKeyFromCell(c); String key = new String(keyBytes); // Check whether we're still reading records corresponding to the key-prefix if (!key.startsWith(keyPrefix)) { return false; } // Extract the byte value before releasing the lock since we cannot hold on to the returned cell afterwards byte[] valueBytes = copyValueFromCell(c); try { next = deserialize(keyBytes, valueBytes, writerSchema, readerSchema); // In case scanner is not able to advance, it means we reached EOF eof = !scanner.next(); } catch (IOException e) { throw new HoodieIOException("Failed to deserialize payload", e); } return true; } @Override public IndexedRecord next() { IndexedRecord next = this.next; this.next = null; return next; } } return new KeyPrefixIterator(); } private static Iterator getRecordByKeyIteratorInternal(HFileScanner scanner, String key, Schema writerSchema, Schema readerSchema) throws IOException { KeyValue kv = new KeyValue(getUTF8Bytes(key), null, null, null); // NOTE: HFile persists both keys/values as bytes, therefore lexicographical sorted is // essentially employed // // For the HFile containing list of cells c[0], c[1], ..., c[N], `seekTo(cell)` would return // following: // a) -1, if cell < c[0], no position; if the key to be searched is less than first entry, the cursor will be placed at the beginning. // b) 0, such that c[i] = cell and scanner is left in position i; // c) and 1, such that c[i] < cell, and scanner is left in position i. // In summary, with exact match, we are interested in return value of 0. in all other cases, key is not found. // Also, do remember we are using reseekTo(), which means, the cursor will not rewind after searching for a key. // Let's say the file contains key01, key02, .., key20. // After searching for key09, if we search for key05, it may not return the matching entry since just after reseeking to key09, the cursor is at key09 and // further reseek calls may not look back in positions. int val = scanner.reseekTo(kv); if (val != 0) { // key is not found. return Collections.emptyIterator(); } // key is found and the cursor is left where the key is found class KeyIterator implements Iterator { private IndexedRecord next = null; private boolean eof = false; @Override public boolean hasNext() { if (next != null) { return true; } else if (eof) { return false; } Cell c = Objects.requireNonNull(scanner.getCell()); byte[] keyBytes = copyKeyFromCell(c); String currentKey = new String(keyBytes); // Check whether we're still reading records corresponding to the key-prefix if (!currentKey.equals(key)) { return false; } // Extract the byte value before releasing the lock since we cannot hold on to the returned cell afterward byte[] valueBytes = copyValueFromCell(c); try { next = deserialize(keyBytes, valueBytes, writerSchema, readerSchema); // In case scanner is not able to advance, it means we reached EOF eof = !scanner.next(); } catch (IOException e) { throw new HoodieIOException("Failed to deserialize payload", e); } return true; } @Override public IndexedRecord next() { IndexedRecord next = this.next; this.next = null; return next; } } return new KeyIterator(); } private static GenericRecord getRecordFromCell(Cell cell, Schema writerSchema, Schema readerSchema) throws IOException { final byte[] keyBytes = copyKeyFromCell(cell); final byte[] valueBytes = copyValueFromCell(cell); return deserialize( keyBytes, 0, keyBytes.length, valueBytes, 0, valueBytes.length, writerSchema, readerSchema); } private static Schema fetchSchema(HFile.Reader reader) { HFileInfo fileInfo = reader.getHFileInfo(); return new Schema.Parser().parse(new String(fileInfo.get(getUTF8Bytes(SCHEMA_KEY)))); } private static byte[] copyKeyFromCell(Cell cell) { return Arrays.copyOfRange(cell.getRowArray(), cell.getRowOffset(), cell.getRowOffset() + cell.getRowLength()); } private static byte[] copyValueFromCell(Cell c) { return Arrays.copyOfRange(c.getValueArray(), c.getValueOffset(), c.getValueOffset() + c.getValueLength()); } private static HFileScanner getHFileScanner(HFile.Reader reader, boolean cacheBlocks) { return getHFileScanner(reader, cacheBlocks, true); } private static HFileScanner getHFileScanner(HFile.Reader reader, boolean cacheBlocks, boolean doSeek) { // NOTE: Only scanners created in Positional Read ("pread") mode could share the same reader, // since scanners in default mode will be seeking w/in the underlying stream try { HFileScanner scanner = reader.getScanner(cacheBlocks, true); if (doSeek) { scanner.seekTo(); // places the cursor at the beginning of the first data block. } return scanner; } catch (IOException e) { throw new HoodieIOException("Failed to initialize HFile scanner for " + reader.getPath(), e); } } private static class RecordByKeyPrefixIterator implements ClosableIterator { private final Iterator sortedKeyPrefixesIterator; private Iterator recordsIterator; private final HFile.Reader reader; private final HFileScanner scanner; private final Schema writerSchema; private final Schema readerSchema; private IndexedRecord next = null; RecordByKeyPrefixIterator(HFile.Reader reader, HFileScanner scanner, List sortedKeyPrefixes, Schema writerSchema, Schema readerSchema) throws IOException { this.sortedKeyPrefixesIterator = sortedKeyPrefixes.iterator(); this.reader = reader; this.scanner = scanner; this.scanner.seekTo(); // position at the beginning of the file this.writerSchema = writerSchema; this.readerSchema = readerSchema; } @Override public boolean hasNext() { try { while (true) { // NOTE: This is required for idempotency if (next != null) { return true; } else if (recordsIterator != null && recordsIterator.hasNext()) { next = recordsIterator.next(); return true; } else if (sortedKeyPrefixesIterator.hasNext()) { String currentKeyPrefix = sortedKeyPrefixesIterator.next(); recordsIterator = getRecordByKeyPrefixIteratorInternal(scanner, currentKeyPrefix, writerSchema, readerSchema); } else { return false; } } } catch (IOException e) { throw new HoodieIOException("Unable to read next record from HFile", e); } } @Override public IndexedRecord next() { IndexedRecord next = this.next; this.next = null; return next; } @Override public void close() { try { scanner.close(); reader.close(); } catch (IOException e) { throw new HoodieIOException("Error closing the hfile reader and scanner", e); } } } private static class RecordByKeyIterator implements ClosableIterator { private final Iterator sortedKeyIterator; private final HFile.Reader reader; private final HFileScanner scanner; private final Schema readerSchema; private final Schema writerSchema; private IndexedRecord next = null; private Iterator currentRecordIterator = null; RecordByKeyIterator(HFile.Reader reader, HFileScanner scanner, List sortedKeys, Schema writerSchema, Schema readerSchema) throws IOException { this.sortedKeyIterator = sortedKeys.iterator(); this.reader = reader; this.scanner = scanner; this.scanner.seekTo(); // position at the beginning of the file this.writerSchema = writerSchema; this.readerSchema = readerSchema; } @Override public boolean hasNext() { try { // NOTE: This is required for idempotency if (next != null) { return true; } // Continue returning records for the current key, if any left while (currentRecordIterator != null && currentRecordIterator.hasNext()) { Option value = Option.of(currentRecordIterator.next()); if (value.isPresent()) { next = value.get(); return true; } } // Move on to the next key and start iterating over its records while (sortedKeyIterator.hasNext()) { currentRecordIterator = getRecordByKeyIteratorInternal(scanner, sortedKeyIterator.next(), writerSchema, readerSchema); if (currentRecordIterator.hasNext()) { Option value = Option.of(currentRecordIterator.next()); if (value.isPresent()) { next = value.get(); return true; } } } // No more keys or records return false; } catch (IOException e) { throw new HoodieIOException("unable to read next record from hfile ", e); } } @Override public IndexedRecord next() { IndexedRecord next = this.next; this.next = null; return next; } @Override public void close() { try { scanner.close(); reader.close(); } catch (IOException e) { throw new HoodieIOException("Error closing the hfile reader and scanner", e); } } } @Override public ClosableIterator getRecordKeyIterator() { HFile.Reader reader = getHFileReader(); final HFileScanner scanner = reader.getScanner(false, false); return new ClosableIterator() { @Override public boolean hasNext() { try { return scanner.next(); } catch (IOException e) { throw new HoodieException("Error while scanning for keys", e); } } @Override public String next() { Cell cell = scanner.getCell(); final byte[] keyBytes = copyKeyFromCell(cell); return new String(keyBytes); } @Override public void close() { try { scanner.close(); reader.close(); } catch (IOException e) { throw new HoodieIOException("Error closing the hfile reader and scanner", e); } } }; } private static class RecordIterator implements ClosableIterator { private final HFile.Reader reader; private final HFileScanner scanner; private final Schema writerSchema; private final Schema readerSchema; private IndexedRecord next = null; private boolean eof = false; RecordIterator(HFile.Reader reader, HFileScanner scanner, Schema writerSchema, Schema readerSchema) { this.reader = reader; this.scanner = scanner; this.writerSchema = writerSchema; this.readerSchema = readerSchema; } @Override public boolean hasNext() { try { // NOTE: This is required for idempotency if (eof) { return false; } if (next != null) { return true; } boolean hasRecords; if (!scanner.isSeeked()) { hasRecords = scanner.seekTo(); } else { hasRecords = scanner.next(); } if (!hasRecords) { eof = true; return false; } this.next = getRecordFromCell(scanner.getCell(), writerSchema, readerSchema); return true; } catch (IOException io) { throw new HoodieIOException("unable to read next record from hfile ", io); } } @Override public IndexedRecord next() { IndexedRecord next = this.next; this.next = null; return next; } @Override public void close() { try { scanner.close(); reader.close(); } catch (IOException e) { throw new HoodieIOException("Error closing the hfile reader and scanner", e); } } } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy