HBASE-25988 Store the store file list by a file (#3578)
Signed-off-by: Wellington Chevreuil <wchevreuil@apache.org>
This commit is contained in:
parent
04e1980dea
commit
826e54ce78
|
@ -0,0 +1,36 @@
|
||||||
|
/**
|
||||||
|
* 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.
|
||||||
|
*/
|
||||||
|
syntax = "proto2";
|
||||||
|
// This file contains protocol buffers that are used for store file tracker.
|
||||||
|
package hbase.pb;
|
||||||
|
|
||||||
|
option java_package = "org.apache.hadoop.hbase.shaded.protobuf.generated";
|
||||||
|
option java_outer_classname = "StoreFileTrackerProtos";
|
||||||
|
option java_generic_services = true;
|
||||||
|
option java_generate_equals_and_hash = true;
|
||||||
|
option optimize_for = SPEED;
|
||||||
|
|
||||||
|
message StoreFileEntry {
|
||||||
|
required string name = 1;
|
||||||
|
required uint64 size = 2;
|
||||||
|
}
|
||||||
|
|
||||||
|
message StoreFileList {
|
||||||
|
required uint64 timestamp = 1;
|
||||||
|
repeated StoreFileEntry store_file = 2;
|
||||||
|
}
|
|
@ -22,6 +22,7 @@ import java.util.Collection;
|
||||||
import java.util.function.Supplier;
|
import java.util.function.Supplier;
|
||||||
import org.apache.hadoop.fs.Path;
|
import org.apache.hadoop.fs.Path;
|
||||||
import org.apache.hadoop.hbase.CellComparator;
|
import org.apache.hadoop.hbase.CellComparator;
|
||||||
|
import org.apache.hadoop.hbase.TableName;
|
||||||
import org.apache.hadoop.hbase.client.ColumnFamilyDescriptor;
|
import org.apache.hadoop.hbase.client.ColumnFamilyDescriptor;
|
||||||
import org.apache.hadoop.hbase.client.RegionInfo;
|
import org.apache.hadoop.hbase.client.RegionInfo;
|
||||||
import org.apache.hadoop.hbase.io.HeapSize;
|
import org.apache.hadoop.hbase.io.HeapSize;
|
||||||
|
@ -109,6 +110,10 @@ public final class StoreContext implements HeapSize {
|
||||||
return coprocessorHost;
|
return coprocessorHost;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public TableName getTableName() {
|
||||||
|
return getRegionInfo().getTable();
|
||||||
|
}
|
||||||
|
|
||||||
public RegionInfo getRegionInfo() {
|
public RegionInfo getRegionInfo() {
|
||||||
return regionFileSystem.getRegionInfo();
|
return regionFileSystem.getRegionInfo();
|
||||||
}
|
}
|
||||||
|
|
|
@ -173,9 +173,9 @@ public abstract class StoreEngine<SF extends StoreFlusher, CP extends Compaction
|
||||||
return this.storeFlusher;
|
return this.storeFlusher;
|
||||||
}
|
}
|
||||||
|
|
||||||
private StoreFileTracker createStoreFileTracker(HStore store) {
|
private StoreFileTracker createStoreFileTracker(Configuration conf, HStore store) {
|
||||||
return StoreFileTrackerFactory.create(store.conf, store.getRegionInfo().getTable(),
|
return StoreFileTrackerFactory.create(conf, store.isPrimaryReplicaStore(),
|
||||||
store.isPrimaryReplicaStore(), store.getStoreContext());
|
store.getStoreContext());
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -206,7 +206,7 @@ public abstract class StoreEngine<SF extends StoreFlusher, CP extends Compaction
|
||||||
this.ctx = store.getStoreContext();
|
this.ctx = store.getStoreContext();
|
||||||
this.coprocessorHost = store.getHRegion().getCoprocessorHost();
|
this.coprocessorHost = store.getHRegion().getCoprocessorHost();
|
||||||
this.openStoreFileThreadPoolCreator = store.getHRegion()::getStoreFileOpenAndCloseThreadPool;
|
this.openStoreFileThreadPoolCreator = store.getHRegion()::getStoreFileOpenAndCloseThreadPool;
|
||||||
this.storeFileTracker = createStoreFileTracker(store);
|
this.storeFileTracker = createStoreFileTracker(conf, store);
|
||||||
assert compactor != null && compactionPolicy != null && storeFileManager != null &&
|
assert compactor != null && compactionPolicy != null && storeFileManager != null &&
|
||||||
storeFlusher != null && storeFileTracker != null;
|
storeFlusher != null && storeFileTracker != null;
|
||||||
}
|
}
|
||||||
|
|
|
@ -21,7 +21,6 @@ import java.io.IOException;
|
||||||
import java.util.Collection;
|
import java.util.Collection;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
import org.apache.hadoop.conf.Configuration;
|
import org.apache.hadoop.conf.Configuration;
|
||||||
import org.apache.hadoop.hbase.TableName;
|
|
||||||
import org.apache.hadoop.hbase.regionserver.StoreContext;
|
import org.apache.hadoop.hbase.regionserver.StoreContext;
|
||||||
import org.apache.hadoop.hbase.regionserver.StoreFileInfo;
|
import org.apache.hadoop.hbase.regionserver.StoreFileInfo;
|
||||||
import org.apache.yetus.audience.InterfaceAudience;
|
import org.apache.yetus.audience.InterfaceAudience;
|
||||||
|
@ -33,9 +32,9 @@ import org.apache.yetus.audience.InterfaceAudience;
|
||||||
@InterfaceAudience.Private
|
@InterfaceAudience.Private
|
||||||
class DefaultStoreFileTracker extends StoreFileTrackerBase {
|
class DefaultStoreFileTracker extends StoreFileTrackerBase {
|
||||||
|
|
||||||
public DefaultStoreFileTracker(Configuration conf, TableName tableName, boolean isPrimaryReplica,
|
public DefaultStoreFileTracker(Configuration conf, boolean isPrimaryReplica,
|
||||||
StoreContext ctx) {
|
StoreContext ctx) {
|
||||||
super(conf, tableName, isPrimaryReplica, ctx);
|
super(conf, isPrimaryReplica, ctx);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
|
|
@ -0,0 +1,142 @@
|
||||||
|
/**
|
||||||
|
* 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.hadoop.hbase.regionserver.storefiletracker;
|
||||||
|
|
||||||
|
import java.io.IOException;
|
||||||
|
import java.util.ArrayList;
|
||||||
|
import java.util.Collection;
|
||||||
|
import java.util.Collections;
|
||||||
|
import java.util.HashMap;
|
||||||
|
import java.util.List;
|
||||||
|
import java.util.Map;
|
||||||
|
import java.util.Set;
|
||||||
|
import java.util.stream.Collectors;
|
||||||
|
import org.apache.hadoop.conf.Configuration;
|
||||||
|
import org.apache.hadoop.fs.FileSystem;
|
||||||
|
import org.apache.hadoop.fs.Path;
|
||||||
|
import org.apache.hadoop.hbase.regionserver.StoreContext;
|
||||||
|
import org.apache.hadoop.hbase.regionserver.StoreFileInfo;
|
||||||
|
import org.apache.hadoop.hbase.util.ServerRegionReplicaUtil;
|
||||||
|
import org.apache.yetus.audience.InterfaceAudience;
|
||||||
|
|
||||||
|
import org.apache.hadoop.hbase.shaded.protobuf.generated.StoreFileTrackerProtos.StoreFileEntry;
|
||||||
|
import org.apache.hadoop.hbase.shaded.protobuf.generated.StoreFileTrackerProtos.StoreFileList;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* A file based store file tracker.
|
||||||
|
* <p/>
|
||||||
|
* For this tracking way, the store file list will be persistent into a file, so we can write the
|
||||||
|
* new store files directly to the final data directory, as we will not load the broken files. This
|
||||||
|
* will greatly reduce the time for flush and compaction on some object storages as a rename is
|
||||||
|
* actual a copy on them. And it also avoid listing when loading store file list, which could also
|
||||||
|
* speed up the loading of store files as listing is also not a fast operation on most object
|
||||||
|
* storages.
|
||||||
|
*/
|
||||||
|
@InterfaceAudience.Private
|
||||||
|
public class FileBasedStoreFileTracker extends StoreFileTrackerBase {
|
||||||
|
|
||||||
|
private final StoreFileListFile backedFile;
|
||||||
|
|
||||||
|
private final Map<String, StoreFileInfo> storefiles = new HashMap<>();
|
||||||
|
|
||||||
|
public FileBasedStoreFileTracker(Configuration conf, boolean isPrimaryReplica, StoreContext ctx) {
|
||||||
|
super(conf, isPrimaryReplica, ctx);
|
||||||
|
backedFile = new StoreFileListFile(ctx);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public List<StoreFileInfo> load() throws IOException {
|
||||||
|
StoreFileList list = backedFile.load();
|
||||||
|
if (list == null) {
|
||||||
|
return Collections.emptyList();
|
||||||
|
}
|
||||||
|
FileSystem fs = ctx.getRegionFileSystem().getFileSystem();
|
||||||
|
List<StoreFileInfo> infos = new ArrayList<>();
|
||||||
|
for (StoreFileEntry entry : list.getStoreFileList()) {
|
||||||
|
infos.add(ServerRegionReplicaUtil.getStoreFileInfo(conf, fs, ctx.getRegionInfo(),
|
||||||
|
ctx.getRegionFileSystem().getRegionInfoForFS(), ctx.getFamily().getNameAsString(),
|
||||||
|
new Path(ctx.getFamilyStoreDirectoryPath(), entry.getName())));
|
||||||
|
}
|
||||||
|
// In general, for primary replica, the load method should only be called once when
|
||||||
|
// initialization, so we do not need synchronized here. And for secondary replicas, though the
|
||||||
|
// load method could be called multiple times, we will never call other methods so no
|
||||||
|
// synchronized is also fine.
|
||||||
|
// But we have a refreshStoreFiles method in the Region interface, which can be called by CPs,
|
||||||
|
// and we have a RefreshHFilesEndpoint example to expose the refreshStoreFiles method as RPC, so
|
||||||
|
// for safety, let's still keep the synchronized here.
|
||||||
|
synchronized (storefiles) {
|
||||||
|
for (StoreFileInfo info : infos) {
|
||||||
|
storefiles.put(info.getPath().getName(), info);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return infos;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected boolean requireWritingToTmpDirFirst() {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
private StoreFileEntry toStoreFileEntry(StoreFileInfo info) {
|
||||||
|
return StoreFileEntry.newBuilder().setName(info.getPath().getName()).setSize(info.getSize())
|
||||||
|
.build();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected void doAddNewStoreFiles(Collection<StoreFileInfo> newFiles) throws IOException {
|
||||||
|
synchronized (storefiles) {
|
||||||
|
StoreFileList.Builder builder = StoreFileList.newBuilder();
|
||||||
|
for (StoreFileInfo info : storefiles.values()) {
|
||||||
|
builder.addStoreFile(toStoreFileEntry(info));
|
||||||
|
}
|
||||||
|
for (StoreFileInfo info : newFiles) {
|
||||||
|
builder.addStoreFile(toStoreFileEntry(info));
|
||||||
|
}
|
||||||
|
backedFile.update(builder);
|
||||||
|
for (StoreFileInfo info : newFiles) {
|
||||||
|
storefiles.put(info.getPath().getName(), info);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected void doAddCompactionResults(Collection<StoreFileInfo> compactedFiles,
|
||||||
|
Collection<StoreFileInfo> newFiles) throws IOException {
|
||||||
|
Set<String> compactedFileNames =
|
||||||
|
compactedFiles.stream().map(info -> info.getPath().getName()).collect(Collectors.toSet());
|
||||||
|
synchronized (storefiles) {
|
||||||
|
StoreFileList.Builder builder = StoreFileList.newBuilder();
|
||||||
|
storefiles.forEach((name, info) -> {
|
||||||
|
if (compactedFileNames.contains(name)) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
builder.addStoreFile(toStoreFileEntry(info));
|
||||||
|
});
|
||||||
|
for (StoreFileInfo info : newFiles) {
|
||||||
|
builder.addStoreFile(toStoreFileEntry(info));
|
||||||
|
}
|
||||||
|
backedFile.update(builder);
|
||||||
|
for (String name : compactedFileNames) {
|
||||||
|
storefiles.remove(name);
|
||||||
|
}
|
||||||
|
for (StoreFileInfo info : newFiles) {
|
||||||
|
storefiles.put(info.getPath().getName(), info);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,142 @@
|
||||||
|
/**
|
||||||
|
* 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.hadoop.hbase.regionserver.storefiletracker;
|
||||||
|
|
||||||
|
import java.io.FileNotFoundException;
|
||||||
|
import java.io.IOException;
|
||||||
|
import org.apache.hadoop.fs.FSDataInputStream;
|
||||||
|
import org.apache.hadoop.fs.FSDataOutputStream;
|
||||||
|
import org.apache.hadoop.fs.FileSystem;
|
||||||
|
import org.apache.hadoop.fs.Path;
|
||||||
|
import org.apache.hadoop.hbase.regionserver.StoreContext;
|
||||||
|
import org.apache.hadoop.hbase.util.EnvironmentEdgeManager;
|
||||||
|
import org.apache.yetus.audience.InterfaceAudience;
|
||||||
|
import org.slf4j.Logger;
|
||||||
|
import org.slf4j.LoggerFactory;
|
||||||
|
|
||||||
|
import org.apache.hbase.thirdparty.com.google.common.base.Preconditions;
|
||||||
|
import org.apache.hbase.thirdparty.com.google.common.io.ByteStreams;
|
||||||
|
import org.apache.hbase.thirdparty.com.google.protobuf.InvalidProtocolBufferException;
|
||||||
|
|
||||||
|
import org.apache.hadoop.hbase.shaded.protobuf.generated.StoreFileTrackerProtos.StoreFileList;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* To fully avoid listing, here we use two files for tracking. When loading, we will try to read
|
||||||
|
* both the two files, if only one exists, we will trust this one, if both exist, we will compare
|
||||||
|
* the timestamp to see which one is newer and trust that one. And we will record in memory that
|
||||||
|
* which one is trusted by us, and when we need to update the store file list, we will write to the
|
||||||
|
* other file.
|
||||||
|
* <p/>
|
||||||
|
* So in this way, we could avoid listing when we want to load the store file list file.
|
||||||
|
*/
|
||||||
|
@InterfaceAudience.Private
|
||||||
|
class StoreFileListFile {
|
||||||
|
|
||||||
|
private static final Logger LOG = LoggerFactory.getLogger(StoreFileListFile.class);
|
||||||
|
|
||||||
|
private static final String TRACK_FILE_DIR = ".filelist";
|
||||||
|
|
||||||
|
private static final String TRACK_FILE = "f1";
|
||||||
|
|
||||||
|
private static final String TRACK_FILE_ROTATE = "f2";
|
||||||
|
|
||||||
|
private final StoreContext ctx;
|
||||||
|
|
||||||
|
private final Path trackFileDir;
|
||||||
|
|
||||||
|
private final Path[] trackFiles = new Path[2];
|
||||||
|
|
||||||
|
// this is used to make sure that we do not go backwards
|
||||||
|
private long prevTimestamp = -1;
|
||||||
|
|
||||||
|
private int nextTrackFile = -1;
|
||||||
|
|
||||||
|
StoreFileListFile(StoreContext ctx) {
|
||||||
|
this.ctx = ctx;
|
||||||
|
trackFileDir = new Path(ctx.getFamilyStoreDirectoryPath(), TRACK_FILE_DIR);
|
||||||
|
trackFiles[0] = new Path(trackFileDir, TRACK_FILE);
|
||||||
|
trackFiles[1] = new Path(trackFileDir, TRACK_FILE_ROTATE);
|
||||||
|
}
|
||||||
|
|
||||||
|
private StoreFileList load(Path path) throws IOException {
|
||||||
|
FileSystem fs = ctx.getRegionFileSystem().getFileSystem();
|
||||||
|
byte[] bytes;
|
||||||
|
try (FSDataInputStream in = fs.open(path)) {
|
||||||
|
bytes = ByteStreams.toByteArray(in);
|
||||||
|
}
|
||||||
|
// Read all the bytes and then parse it, so we will only throw InvalidProtocolBufferException
|
||||||
|
// here. This is very important for upper layer to determine whether this is the normal case,
|
||||||
|
// where the file does not exist or is incomplete. If there is another type of exception, the
|
||||||
|
// upper layer should throw it out instead of just ignoring it, otherwise it will lead to data
|
||||||
|
// loss.
|
||||||
|
return StoreFileList.parseFrom(bytes);
|
||||||
|
}
|
||||||
|
|
||||||
|
private int select(StoreFileList[] lists) {
|
||||||
|
if (lists[0] == null) {
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
if (lists[1] == null) {
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
return lists[0].getTimestamp() >= lists[1].getTimestamp() ? 0 : 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
StoreFileList load() throws IOException {
|
||||||
|
StoreFileList[] lists = new StoreFileList[2];
|
||||||
|
for (int i = 0; i < 2; i++) {
|
||||||
|
try {
|
||||||
|
lists[i] = load(trackFiles[i]);
|
||||||
|
} catch (FileNotFoundException | InvalidProtocolBufferException e) {
|
||||||
|
// this is normal case, so use info and do not log stacktrace
|
||||||
|
LOG.info("Failed to load track file {}: {}", trackFiles[i], e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
int winnerIndex = select(lists);
|
||||||
|
if (lists[winnerIndex] != null) {
|
||||||
|
nextTrackFile = 1 - winnerIndex;
|
||||||
|
prevTimestamp = lists[winnerIndex].getTimestamp();
|
||||||
|
} else {
|
||||||
|
nextTrackFile = 0;
|
||||||
|
}
|
||||||
|
return lists[winnerIndex];
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* We will set the timestamp in this method so just pass the builder in
|
||||||
|
*/
|
||||||
|
void update(StoreFileList.Builder builder) throws IOException {
|
||||||
|
Preconditions.checkState(nextTrackFile >= 0, "should call load first before calling update");
|
||||||
|
FileSystem fs = ctx.getRegionFileSystem().getFileSystem();
|
||||||
|
long timestamp = Math.max(prevTimestamp + 1, EnvironmentEdgeManager.currentTime());
|
||||||
|
try (FSDataOutputStream out = fs.create(trackFiles[nextTrackFile], true)) {
|
||||||
|
builder.setTimestamp(timestamp).build().writeTo(out);
|
||||||
|
}
|
||||||
|
// record timestamp
|
||||||
|
prevTimestamp = timestamp;
|
||||||
|
// rotate the file
|
||||||
|
nextTrackFile = 1 - nextTrackFile;
|
||||||
|
try {
|
||||||
|
fs.delete(trackFiles[nextTrackFile], false);
|
||||||
|
} catch (IOException e) {
|
||||||
|
// we will create new file with overwrite = true, so not a big deal here, only for speed up
|
||||||
|
// loading as we do not need to read this file when loading(we will hit FileNotFoundException)
|
||||||
|
LOG.debug("failed to delete old track file {}, not a big deal, just ignore", e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -21,7 +21,6 @@ import java.io.IOException;
|
||||||
import java.util.Collection;
|
import java.util.Collection;
|
||||||
import org.apache.hadoop.conf.Configuration;
|
import org.apache.hadoop.conf.Configuration;
|
||||||
import org.apache.hadoop.fs.Path;
|
import org.apache.hadoop.fs.Path;
|
||||||
import org.apache.hadoop.hbase.TableName;
|
|
||||||
import org.apache.hadoop.hbase.client.ColumnFamilyDescriptor;
|
import org.apache.hadoop.hbase.client.ColumnFamilyDescriptor;
|
||||||
import org.apache.hadoop.hbase.io.compress.Compression;
|
import org.apache.hadoop.hbase.io.compress.Compression;
|
||||||
import org.apache.hadoop.hbase.io.crypto.Encryption;
|
import org.apache.hadoop.hbase.io.crypto.Encryption;
|
||||||
|
@ -51,18 +50,14 @@ abstract class StoreFileTrackerBase implements StoreFileTracker {
|
||||||
|
|
||||||
protected final Configuration conf;
|
protected final Configuration conf;
|
||||||
|
|
||||||
protected final TableName tableName;
|
|
||||||
|
|
||||||
protected final boolean isPrimaryReplica;
|
protected final boolean isPrimaryReplica;
|
||||||
|
|
||||||
protected final StoreContext ctx;
|
protected final StoreContext ctx;
|
||||||
|
|
||||||
private volatile boolean cacheOnWriteLogged;
|
private volatile boolean cacheOnWriteLogged;
|
||||||
|
|
||||||
protected StoreFileTrackerBase(Configuration conf, TableName tableName, boolean isPrimaryReplica,
|
protected StoreFileTrackerBase(Configuration conf, boolean isPrimaryReplica, StoreContext ctx) {
|
||||||
StoreContext ctx) {
|
|
||||||
this.conf = conf;
|
this.conf = conf;
|
||||||
this.tableName = tableName;
|
|
||||||
this.isPrimaryReplica = isPrimaryReplica;
|
this.isPrimaryReplica = isPrimaryReplica;
|
||||||
this.ctx = ctx;
|
this.ctx = ctx;
|
||||||
}
|
}
|
||||||
|
@ -95,7 +90,7 @@ abstract class StoreFileTrackerBase implements StoreFileTracker {
|
||||||
.withBlockSize(family.getBlocksize()).withHBaseCheckSum(true)
|
.withBlockSize(family.getBlocksize()).withHBaseCheckSum(true)
|
||||||
.withDataBlockEncoding(family.getDataBlockEncoding()).withEncryptionContext(encryptionContext)
|
.withDataBlockEncoding(family.getDataBlockEncoding()).withEncryptionContext(encryptionContext)
|
||||||
.withCreateTime(EnvironmentEdgeManager.currentTime()).withColumnFamily(family.getName())
|
.withCreateTime(EnvironmentEdgeManager.currentTime()).withColumnFamily(family.getName())
|
||||||
.withTableName(tableName.getName()).withCellComparator(ctx.getComparator()).build();
|
.withTableName(ctx.getTableName().getName()).withCellComparator(ctx.getComparator()).build();
|
||||||
return hFileContext;
|
return hFileContext;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -153,7 +148,7 @@ abstract class StoreFileTrackerBase implements StoreFileTracker {
|
||||||
outputDir =
|
outputDir =
|
||||||
new Path(ctx.getRegionFileSystem().getTempDir(), ctx.getFamily().getNameAsString());
|
new Path(ctx.getRegionFileSystem().getTempDir(), ctx.getFamily().getNameAsString());
|
||||||
} else {
|
} else {
|
||||||
throw new UnsupportedOperationException("not supported yet");
|
outputDir = ctx.getFamilyStoreDirectoryPath();
|
||||||
}
|
}
|
||||||
StoreFileWriter.Builder builder =
|
StoreFileWriter.Builder builder =
|
||||||
new StoreFileWriter.Builder(conf, writerCacheConf, ctx.getRegionFileSystem().getFileSystem())
|
new StoreFileWriter.Builder(conf, writerCacheConf, ctx.getRegionFileSystem().getFileSystem())
|
||||||
|
|
|
@ -18,8 +18,8 @@
|
||||||
package org.apache.hadoop.hbase.regionserver.storefiletracker;
|
package org.apache.hadoop.hbase.regionserver.storefiletracker;
|
||||||
|
|
||||||
import org.apache.hadoop.conf.Configuration;
|
import org.apache.hadoop.conf.Configuration;
|
||||||
import org.apache.hadoop.hbase.TableName;
|
|
||||||
import org.apache.hadoop.hbase.regionserver.StoreContext;
|
import org.apache.hadoop.hbase.regionserver.StoreContext;
|
||||||
|
import org.apache.hadoop.hbase.util.ReflectionUtils;
|
||||||
import org.apache.yetus.audience.InterfaceAudience;
|
import org.apache.yetus.audience.InterfaceAudience;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -28,8 +28,12 @@ import org.apache.yetus.audience.InterfaceAudience;
|
||||||
@InterfaceAudience.Private
|
@InterfaceAudience.Private
|
||||||
public final class StoreFileTrackerFactory {
|
public final class StoreFileTrackerFactory {
|
||||||
|
|
||||||
public static StoreFileTracker create(Configuration conf, TableName tableName,
|
public static final String TRACK_IMPL = "hbase.store.file-tracker.impl";
|
||||||
boolean isPrimaryReplica, StoreContext ctx) {
|
|
||||||
return new DefaultStoreFileTracker(conf, tableName, isPrimaryReplica, ctx);
|
public static StoreFileTracker create(Configuration conf, boolean isPrimaryReplica,
|
||||||
|
StoreContext ctx) {
|
||||||
|
Class<? extends StoreFileTracker> tracker =
|
||||||
|
conf.getClass(TRACK_IMPL, DefaultStoreFileTracker.class, StoreFileTracker.class);
|
||||||
|
return ReflectionUtils.newInstance(tracker, conf, isPrimaryReplica, ctx);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -0,0 +1,109 @@
|
||||||
|
/**
|
||||||
|
* 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.hadoop.hbase.regionserver.storefiletracker;
|
||||||
|
|
||||||
|
import static org.junit.Assert.assertEquals;
|
||||||
|
|
||||||
|
import java.io.IOException;
|
||||||
|
import org.apache.hadoop.conf.Configuration;
|
||||||
|
import org.apache.hadoop.hbase.HBaseClassTestRule;
|
||||||
|
import org.apache.hadoop.hbase.HBaseTestingUtility;
|
||||||
|
import org.apache.hadoop.hbase.TableName;
|
||||||
|
import org.apache.hadoop.hbase.client.ColumnFamilyDescriptorBuilder;
|
||||||
|
import org.apache.hadoop.hbase.client.Get;
|
||||||
|
import org.apache.hadoop.hbase.client.Put;
|
||||||
|
import org.apache.hadoop.hbase.client.RegionInfo;
|
||||||
|
import org.apache.hadoop.hbase.client.RegionInfoBuilder;
|
||||||
|
import org.apache.hadoop.hbase.client.Result;
|
||||||
|
import org.apache.hadoop.hbase.client.TableDescriptor;
|
||||||
|
import org.apache.hadoop.hbase.client.TableDescriptorBuilder;
|
||||||
|
import org.apache.hadoop.hbase.regionserver.HRegion;
|
||||||
|
import org.apache.hadoop.hbase.testclassification.MediumTests;
|
||||||
|
import org.apache.hadoop.hbase.testclassification.RegionServerTests;
|
||||||
|
import org.apache.hadoop.hbase.util.Bytes;
|
||||||
|
import org.junit.After;
|
||||||
|
import org.junit.Before;
|
||||||
|
import org.junit.ClassRule;
|
||||||
|
import org.junit.Rule;
|
||||||
|
import org.junit.Test;
|
||||||
|
import org.junit.experimental.categories.Category;
|
||||||
|
import org.junit.rules.TestName;
|
||||||
|
|
||||||
|
@Category({ RegionServerTests.class, MediumTests.class })
|
||||||
|
public class TestRegionWithFileBasedStoreFileTracker {
|
||||||
|
|
||||||
|
@ClassRule
|
||||||
|
public static final HBaseClassTestRule CLASS_RULE =
|
||||||
|
HBaseClassTestRule.forClass(TestRegionWithFileBasedStoreFileTracker.class);
|
||||||
|
|
||||||
|
private static final HBaseTestingUtility UTIL = new HBaseTestingUtility();
|
||||||
|
|
||||||
|
private static final byte[] CF = Bytes.toBytes("cf");
|
||||||
|
|
||||||
|
private static final byte[] CQ = Bytes.toBytes("cq");
|
||||||
|
|
||||||
|
private static final TableDescriptor TD =
|
||||||
|
TableDescriptorBuilder.newBuilder(TableName.valueOf("file_based_tracker"))
|
||||||
|
.setColumnFamily(ColumnFamilyDescriptorBuilder.of(CF)).build();
|
||||||
|
|
||||||
|
private static final RegionInfo RI = RegionInfoBuilder.newBuilder(TD.getTableName()).build();
|
||||||
|
|
||||||
|
@Rule
|
||||||
|
public TestName name = new TestName();
|
||||||
|
|
||||||
|
private HRegion region;
|
||||||
|
|
||||||
|
@Before
|
||||||
|
public void setUp() throws IOException {
|
||||||
|
Configuration conf = new Configuration(UTIL.getConfiguration());
|
||||||
|
conf.setClass(StoreFileTrackerFactory.TRACK_IMPL, FileBasedStoreFileTracker.class,
|
||||||
|
StoreFileTracker.class);
|
||||||
|
region = HBaseTestingUtility.createRegionAndWAL(RI, UTIL.getDataTestDir(name.getMethodName()),
|
||||||
|
conf, TD);
|
||||||
|
}
|
||||||
|
|
||||||
|
@After
|
||||||
|
public void tearDown() throws IOException {
|
||||||
|
if (region != null) {
|
||||||
|
HBaseTestingUtility.closeRegionAndWAL(region);
|
||||||
|
}
|
||||||
|
UTIL.cleanupTestDir();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testFlushAndCompaction() throws IOException {
|
||||||
|
for (int i = 0; i < 10; i++) {
|
||||||
|
for (int j = 0; j < 10; j++) {
|
||||||
|
int v = i * 10 + j;
|
||||||
|
region.put(new Put(Bytes.toBytes(v)).addColumn(CF, CQ, Bytes.toBytes(v)));
|
||||||
|
}
|
||||||
|
region.flush(true);
|
||||||
|
if (i % 3 == 2) {
|
||||||
|
region.compact(true);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// reopen the region, make sure the store file tracker works, i.e, we can get all the records
|
||||||
|
// back
|
||||||
|
region.close();
|
||||||
|
region = HRegion.openHRegion(region, null);
|
||||||
|
for (int i = 0; i < 100; i++) {
|
||||||
|
Result result = region.get(new Get(Bytes.toBytes(i)));
|
||||||
|
assertEquals(i, Bytes.toInt(result.getValue(CF, CQ)));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
Loading…
Reference in New Issue