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);
}
}
}
}