Amend HBASE-18786 FileNotFoundException should not be silently handled for primary region replicas
Remove now invalid unit test TestCorruptedRegionStoreFile
This commit is contained in:
parent
5a3d9cb225
commit
d74bdc3100
|
@ -1,263 +0,0 @@
|
|||
/**
|
||||
* 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;
|
||||
|
||||
import java.io.FileNotFoundException;
|
||||
import java.io.IOException;
|
||||
import java.util.ArrayList;
|
||||
|
||||
import org.apache.commons.logging.Log;
|
||||
import org.apache.commons.logging.LogFactory;
|
||||
import org.apache.hadoop.conf.Configuration;
|
||||
import org.apache.hadoop.fs.FSDataInputStream;
|
||||
import org.apache.hadoop.fs.FileSystem;
|
||||
import org.apache.hadoop.fs.Path;
|
||||
import org.apache.hadoop.hbase.HBaseTestingUtility;
|
||||
import org.apache.hadoop.hbase.TableName;
|
||||
import org.apache.hadoop.hbase.client.HTable;
|
||||
import org.apache.hadoop.hbase.client.Scan;
|
||||
import org.apache.hadoop.hbase.client.Result;
|
||||
import org.apache.hadoop.hbase.client.ResultScanner;
|
||||
import org.apache.hadoop.hbase.client.Durability;
|
||||
import org.apache.hadoop.hbase.client.Put;
|
||||
import org.apache.hadoop.hbase.client.Table;
|
||||
import org.apache.hadoop.hbase.io.HFileLink;
|
||||
import org.apache.hadoop.hbase.testclassification.LargeTests;
|
||||
import org.apache.hadoop.hbase.util.JVMClusterUtil.RegionServerThread;
|
||||
import org.apache.hadoop.hbase.util.Bytes;
|
||||
import org.apache.hadoop.hbase.util.FSUtils;
|
||||
import org.apache.hadoop.hbase.util.FSVisitor;
|
||||
import org.apache.hadoop.hbase.util.TestTableName;
|
||||
|
||||
import org.junit.After;
|
||||
import org.junit.Before;
|
||||
import org.junit.Rule;
|
||||
import org.junit.Test;
|
||||
import org.junit.experimental.categories.Category;
|
||||
|
||||
import static org.junit.Assert.assertEquals;
|
||||
import static org.junit.Assert.assertTrue;
|
||||
|
||||
@Category(LargeTests.class)
|
||||
public class TestCorruptedRegionStoreFile {
|
||||
private static final Log LOG = LogFactory.getLog(TestCorruptedRegionStoreFile.class);
|
||||
|
||||
private static final HBaseTestingUtility UTIL = new HBaseTestingUtility();
|
||||
|
||||
private static final String FAMILY_NAME_STR = "f";
|
||||
private static final byte[] FAMILY_NAME = Bytes.toBytes(FAMILY_NAME_STR);
|
||||
|
||||
private static final int NUM_FILES = 10;
|
||||
private static final int ROW_PER_FILE = 2000;
|
||||
private static final int NUM_ROWS = NUM_FILES * ROW_PER_FILE;
|
||||
|
||||
@Rule public TestTableName TEST_TABLE = new TestTableName();
|
||||
|
||||
private final ArrayList<Path> storeFiles = new ArrayList<Path>();
|
||||
private Path tableDir;
|
||||
private int rowCount;
|
||||
|
||||
private static void setupConf(Configuration conf) {
|
||||
// Disable compaction so the store file count stays constant
|
||||
conf.setLong("hbase.hstore.compactionThreshold", NUM_FILES + 1);
|
||||
conf.setLong("hbase.hstore.blockingStoreFiles", NUM_FILES * 2);
|
||||
}
|
||||
|
||||
private void setupTable(final TableName tableName) throws IOException {
|
||||
// load the table
|
||||
Table table = UTIL.createTable(tableName, FAMILY_NAME);
|
||||
try {
|
||||
rowCount = 0;
|
||||
byte[] value = new byte[1024];
|
||||
byte[] q = Bytes.toBytes("q");
|
||||
while (rowCount < NUM_ROWS) {
|
||||
Put put = new Put(Bytes.toBytes(String.format("%010d", rowCount)));
|
||||
put.setDurability(Durability.SKIP_WAL);
|
||||
put.add(FAMILY_NAME, q, value);
|
||||
table.put(put);
|
||||
|
||||
if ((rowCount++ % ROW_PER_FILE) == 0) {
|
||||
// flush it
|
||||
((HTable)table).flushCommits();
|
||||
UTIL.getHBaseAdmin().flush(tableName);
|
||||
}
|
||||
}
|
||||
} finally {
|
||||
UTIL.getHBaseAdmin().flush(tableName);
|
||||
table.close();
|
||||
}
|
||||
|
||||
assertEquals(NUM_ROWS, rowCount);
|
||||
|
||||
// get the store file paths
|
||||
storeFiles.clear();
|
||||
tableDir = FSUtils.getTableDir(getRootDir(), tableName);
|
||||
FSVisitor.visitTableStoreFiles(getFileSystem(), tableDir, new FSVisitor.StoreFileVisitor() {
|
||||
@Override
|
||||
public void storeFile(final String region, final String family, final String hfile)
|
||||
throws IOException {
|
||||
HFileLink link = HFileLink.build(UTIL.getConfiguration(), tableName, region, family, hfile);
|
||||
storeFiles.add(link.getOriginPath());
|
||||
}
|
||||
});
|
||||
assertTrue("Expected at least " + NUM_FILES + " store files", storeFiles.size() >= NUM_FILES);
|
||||
LOG.info("Store files: " + storeFiles);
|
||||
}
|
||||
|
||||
@Before
|
||||
public void setup() throws Exception {
|
||||
setupConf(UTIL.getConfiguration());
|
||||
UTIL.startMiniCluster(2, 3);
|
||||
|
||||
setupTable(TEST_TABLE.getTableName());
|
||||
}
|
||||
|
||||
@After
|
||||
public void tearDown() throws Exception {
|
||||
try {
|
||||
UTIL.shutdownMiniCluster();
|
||||
} catch (Exception e) {
|
||||
LOG.warn("failure shutting down cluster", e);
|
||||
}
|
||||
}
|
||||
|
||||
private void removeStoreFile(FileSystem fs, Path tmpStoreFilePath) throws Exception {
|
||||
try (FSDataInputStream input = fs.open(storeFiles.get(0))) {
|
||||
fs.copyToLocalFile(true, storeFiles.get(0), tmpStoreFilePath);
|
||||
LOG.info("Move file to local");
|
||||
evictHFileCache(storeFiles.get(0));
|
||||
// make sure that all the replicas have been deleted on DNs.
|
||||
for (;;) {
|
||||
try {
|
||||
input.read(0, new byte[1], 0, 1);
|
||||
} catch (FileNotFoundException e) {
|
||||
break;
|
||||
}
|
||||
Thread.sleep(1000);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
@Test(timeout=180000)
|
||||
public void testLosingFileDuringScan() throws Exception {
|
||||
assertEquals(rowCount, fullScanAndCount(TEST_TABLE.getTableName()));
|
||||
|
||||
final FileSystem fs = getFileSystem();
|
||||
final Path tmpStoreFilePath = new Path(UTIL.getDataTestDir(), "corruptedHFile");
|
||||
|
||||
// try to query with the missing file
|
||||
int count = fullScanAndCount(TEST_TABLE.getTableName(), new ScanInjector() {
|
||||
private boolean hasFile = true;
|
||||
|
||||
@Override
|
||||
public void beforeScanNext(Table table) throws Exception {
|
||||
// move the path away (now the region is corrupted)
|
||||
if (hasFile) {
|
||||
removeStoreFile(fs, tmpStoreFilePath);
|
||||
hasFile = false;
|
||||
}
|
||||
}
|
||||
});
|
||||
assertTrue("expected one file lost: rowCount=" + count + " lostRows=" + (NUM_ROWS - count),
|
||||
count >= (NUM_ROWS - ROW_PER_FILE));
|
||||
}
|
||||
|
||||
@Test(timeout=180000)
|
||||
public void testLosingFileAfterScannerInit() throws Exception {
|
||||
assertEquals(rowCount, fullScanAndCount(TEST_TABLE.getTableName()));
|
||||
|
||||
final FileSystem fs = getFileSystem();
|
||||
final Path tmpStoreFilePath = new Path(UTIL.getDataTestDir(), "corruptedHFile");
|
||||
|
||||
// try to query with the missing file
|
||||
int count = fullScanAndCount(TEST_TABLE.getTableName(), new ScanInjector() {
|
||||
private boolean hasFile = true;
|
||||
|
||||
@Override
|
||||
public void beforeScan(Table table, Scan scan) throws Exception {
|
||||
// move the path away (now the region is corrupted)
|
||||
if (hasFile) {
|
||||
removeStoreFile(fs, tmpStoreFilePath);
|
||||
hasFile = false;
|
||||
}
|
||||
}
|
||||
});
|
||||
assertTrue("expected one file lost: rowCount=" + count + " lostRows=" + (NUM_ROWS - count),
|
||||
count >= (NUM_ROWS - ROW_PER_FILE));
|
||||
}
|
||||
|
||||
// ==========================================================================
|
||||
// Helpers
|
||||
// ==========================================================================
|
||||
private FileSystem getFileSystem() {
|
||||
return UTIL.getHBaseCluster().getMaster().getMasterFileSystem().getFileSystem();
|
||||
}
|
||||
|
||||
private Path getRootDir() {
|
||||
return UTIL.getHBaseCluster().getMaster().getMasterFileSystem().getRootDir();
|
||||
}
|
||||
|
||||
private void evictHFileCache(final Path hfile) throws Exception {
|
||||
for (RegionServerThread rst: UTIL.getMiniHBaseCluster().getRegionServerThreads()) {
|
||||
HRegionServer rs = rst.getRegionServer();
|
||||
rs.getCacheConfig().getBlockCache().evictBlocksByHfileName(hfile.getName());
|
||||
}
|
||||
}
|
||||
|
||||
private int fullScanAndCount(final TableName tableName) throws Exception {
|
||||
return fullScanAndCount(tableName, new ScanInjector());
|
||||
}
|
||||
|
||||
private int fullScanAndCount(final TableName tableName, final ScanInjector injector)
|
||||
throws Exception {
|
||||
Table table = UTIL.getConnection().getTable(tableName);
|
||||
int count = 0;
|
||||
try {
|
||||
Scan scan = new Scan();
|
||||
scan.setCaching(1);
|
||||
scan.setCacheBlocks(false);
|
||||
injector.beforeScan(table, scan);
|
||||
ResultScanner scanner = table.getScanner(scan);
|
||||
try {
|
||||
while (true) {
|
||||
injector.beforeScanNext(table);
|
||||
Result result = scanner.next();
|
||||
injector.afterScanNext(table, result);
|
||||
if (result == null) break;
|
||||
if ((count++ % (ROW_PER_FILE / 2)) == 0) {
|
||||
LOG.debug("scan next " + count);
|
||||
}
|
||||
}
|
||||
} finally {
|
||||
scanner.close();
|
||||
injector.afterScan(table);
|
||||
}
|
||||
} finally {
|
||||
table.close();
|
||||
}
|
||||
return count;
|
||||
}
|
||||
|
||||
private class ScanInjector {
|
||||
protected void beforeScan(Table table, Scan scan) throws Exception {}
|
||||
protected void beforeScanNext(Table table) throws Exception {}
|
||||
protected void afterScanNext(Table table, Result result) throws Exception {}
|
||||
protected void afterScan(Table table) throws Exception {}
|
||||
}
|
||||
}
|
Loading…
Reference in New Issue