HBASE-23782 We still reference the hard coded meta descriptor in some places when listing table descriptors (#1115)

Signed-off-by: Viraj Jasani <vjasani@apache.org>
This commit is contained in:
Duo Zhang 2020-02-03 14:15:57 +08:00
parent 4de06915b8
commit f94dbebffa
3 changed files with 58 additions and 56 deletions

View File

@ -17,6 +17,7 @@
*/ */
package org.apache.hadoop.hbase.util; package org.apache.hadoop.hbase.util;
import edu.umd.cs.findbugs.annotations.Nullable;
import java.io.FileNotFoundException; import java.io.FileNotFoundException;
import java.io.IOException; import java.io.IOException;
import java.util.Comparator; import java.util.Comparator;
@ -27,8 +28,6 @@ import java.util.concurrent.ConcurrentHashMap;
import java.util.function.Function; import java.util.function.Function;
import java.util.regex.Matcher; import java.util.regex.Matcher;
import java.util.regex.Pattern; import java.util.regex.Pattern;
import edu.umd.cs.findbugs.annotations.Nullable;
import org.apache.commons.lang3.NotImplementedException; import org.apache.commons.lang3.NotImplementedException;
import org.apache.hadoop.conf.Configuration; import org.apache.hadoop.conf.Configuration;
import org.apache.hadoop.fs.FSDataInputStream; import org.apache.hadoop.fs.FSDataInputStream;
@ -37,23 +36,24 @@ import org.apache.hadoop.fs.FileStatus;
import org.apache.hadoop.fs.FileSystem; import org.apache.hadoop.fs.FileSystem;
import org.apache.hadoop.fs.Path; import org.apache.hadoop.fs.Path;
import org.apache.hadoop.fs.PathFilter; import org.apache.hadoop.fs.PathFilter;
import org.apache.hadoop.hbase.client.CoprocessorDescriptorBuilder;
import org.apache.hadoop.hbase.coprocessor.MultiRowMutationEndpoint;
import org.apache.yetus.audience.InterfaceAudience;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.apache.hadoop.hbase.client.ColumnFamilyDescriptorBuilder;
import org.apache.hadoop.hbase.client.TableDescriptor;
import org.apache.hadoop.hbase.client.TableDescriptorBuilder;
import org.apache.hadoop.hbase.Coprocessor; import org.apache.hadoop.hbase.Coprocessor;
import org.apache.hadoop.hbase.exceptions.DeserializationException;
import org.apache.hadoop.hbase.HConstants; import org.apache.hadoop.hbase.HConstants;
import org.apache.hadoop.hbase.regionserver.BloomType;
import org.apache.hbase.thirdparty.com.google.common.annotations.VisibleForTesting;
import org.apache.hbase.thirdparty.com.google.common.primitives.Ints;
import org.apache.hadoop.hbase.TableDescriptors; import org.apache.hadoop.hbase.TableDescriptors;
import org.apache.hadoop.hbase.TableInfoMissingException; import org.apache.hadoop.hbase.TableInfoMissingException;
import org.apache.hadoop.hbase.TableName; import org.apache.hadoop.hbase.TableName;
import org.apache.hadoop.hbase.client.ColumnFamilyDescriptorBuilder;
import org.apache.hadoop.hbase.client.CoprocessorDescriptorBuilder;
import org.apache.hadoop.hbase.client.TableDescriptor;
import org.apache.hadoop.hbase.client.TableDescriptorBuilder;
import org.apache.hadoop.hbase.coprocessor.MultiRowMutationEndpoint;
import org.apache.hadoop.hbase.exceptions.DeserializationException;
import org.apache.hadoop.hbase.regionserver.BloomType;
import org.apache.yetus.audience.InterfaceAudience;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.apache.hbase.thirdparty.com.google.common.annotations.VisibleForTesting;
import org.apache.hbase.thirdparty.com.google.common.primitives.Ints;
/** /**
* Implementation of {@link TableDescriptors} that reads descriptors from the * Implementation of {@link TableDescriptors} that reads descriptors from the
@ -99,11 +99,6 @@ public class FSTableDescriptors implements TableDescriptors {
// TODO. // TODO.
private final Map<TableName, TableDescriptor> cache = new ConcurrentHashMap<>(); private final Map<TableName, TableDescriptor> cache = new ConcurrentHashMap<>();
/**
* Table descriptor for <code>hbase:meta</code> catalog table
*/
private final TableDescriptor metaTableDescriptor;
/** /**
* Construct a FSTableDescriptors instance using the hbase root dir of the given * Construct a FSTableDescriptors instance using the hbase root dir of the given
* conf and the filesystem where that root dir lives. * conf and the filesystem where that root dir lives.
@ -135,31 +130,29 @@ public class FSTableDescriptors implements TableDescriptors {
* see HMaster#finishActiveMasterInitialization * see HMaster#finishActiveMasterInitialization
* TODO: This is a workaround. Should remove this ugly code... * TODO: This is a workaround. Should remove this ugly code...
*/ */
public FSTableDescriptors(final Configuration conf, final FileSystem fs, public FSTableDescriptors(final Configuration conf, final FileSystem fs, final Path rootdir,
final Path rootdir, final boolean fsreadonly, final boolean usecache, final boolean fsreadonly, final boolean usecache,
Function<TableDescriptorBuilder, TableDescriptorBuilder> metaObserver) throws IOException { Function<TableDescriptorBuilder, TableDescriptorBuilder> metaObserver) throws IOException {
this.fs = fs; this.fs = fs;
this.rootdir = rootdir; this.rootdir = rootdir;
this.fsreadonly = fsreadonly; this.fsreadonly = fsreadonly;
this.usecache = usecache; this.usecache = usecache;
TableDescriptor td = null; if (!fsreadonly) {
try { // see if we already have meta descriptor on fs. Write one if not.
td = getTableDescriptorFromFs(fs, rootdir, TableName.META_TABLE_NAME); try {
} catch (TableInfoMissingException e) { getTableDescriptorFromFs(fs, rootdir, TableName.META_TABLE_NAME);
td = metaObserver == null? createMetaTableDescriptor(conf): } catch (TableInfoMissingException e) {
metaObserver.apply(createMetaTableDescriptorBuilder(conf)).build(); TableDescriptorBuilder builder = createMetaTableDescriptorBuilder(conf);
if (!fsreadonly) { if (metaObserver != null) {
metaObserver.apply(builder);
}
TableDescriptor td = builder.build();
LOG.info("Creating new hbase:meta table default descriptor/schema {}", td); LOG.info("Creating new hbase:meta table default descriptor/schema {}", td);
updateTableDescriptor(td); updateTableDescriptor(td);
} }
} }
this.metaTableDescriptor = td;
} }
/**
*
* Make this private as soon as we've undone test dependency.
*/
@VisibleForTesting @VisibleForTesting
public static TableDescriptorBuilder createMetaTableDescriptorBuilder(final Configuration conf) public static TableDescriptorBuilder createMetaTableDescriptorBuilder(final Configuration conf)
throws IOException { throws IOException {
@ -200,12 +193,6 @@ public class FSTableDescriptors implements TableDescriptors {
.build()); .build());
} }
@VisibleForTesting
public static TableDescriptor createMetaTableDescriptor(final Configuration conf)
throws IOException {
return createMetaTableDescriptorBuilder(conf).build();
}
@Override @Override
public void setCacheOn() throws IOException { public void setCacheOn() throws IOException {
this.cache.clear(); this.cache.clear();
@ -266,16 +253,12 @@ public class FSTableDescriptors implements TableDescriptors {
* Returns a map from table name to table descriptor for all tables. * Returns a map from table name to table descriptor for all tables.
*/ */
@Override @Override
public Map<String, TableDescriptor> getAll() public Map<String, TableDescriptor> getAll() throws IOException {
throws IOException {
Map<String, TableDescriptor> tds = new TreeMap<>(); Map<String, TableDescriptor> tds = new TreeMap<>();
if (fsvisited && usecache) { if (fsvisited && usecache) {
for (Map.Entry<TableName, TableDescriptor> entry: this.cache.entrySet()) { for (Map.Entry<TableName, TableDescriptor> entry: this.cache.entrySet()) {
tds.put(entry.getKey().getNameWithNamespaceInclAsString(), entry.getValue()); tds.put(entry.getKey().getNameWithNamespaceInclAsString(), entry.getValue());
} }
// add hbase:meta to the response
tds.put(this.metaTableDescriptor.getTableName().getNameAsString(), metaTableDescriptor);
} else { } else {
LOG.trace("Fetching table descriptors from the filesystem."); LOG.trace("Fetching table descriptors from the filesystem.");
boolean allvisited = true; boolean allvisited = true;
@ -558,15 +541,17 @@ public class FSTableDescriptors implements TableDescriptors {
* @throws IOException Thrown if failed update. * @throws IOException Thrown if failed update.
* @throws NotImplementedException if in read only mode * @throws NotImplementedException if in read only mode
*/ */
@VisibleForTesting Path updateTableDescriptor(TableDescriptor td) @VisibleForTesting
throws IOException { Path updateTableDescriptor(TableDescriptor td) throws IOException {
if (fsreadonly) { if (fsreadonly) {
throw new NotImplementedException("Cannot update a table descriptor - in read only mode"); throw new NotImplementedException("Cannot update a table descriptor - in read only mode");
} }
TableName tableName = td.getTableName(); TableName tableName = td.getTableName();
Path tableDir = getTableDir(tableName); Path tableDir = getTableDir(tableName);
Path p = writeTableDescriptor(fs, td, tableDir, getTableInfoPath(tableDir)); Path p = writeTableDescriptor(fs, td, tableDir, getTableInfoPath(tableDir));
if (p == null) throw new IOException("Failed update"); if (p == null) {
throw new IOException("Failed update");
}
LOG.info("Updated tableinfo=" + p); LOG.info("Updated tableinfo=" + p);
if (usecache) { if (usecache) {
this.cache.put(td.getTableName(), td); this.cache.put(td.getTableName(), td);

View File

@ -22,7 +22,7 @@ import static org.junit.Assert.assertTrue;
import static org.junit.Assert.fail; import static org.junit.Assert.fail;
import java.io.IOException; import java.io.IOException;
import java.util.Collections;
import org.apache.hadoop.hbase.client.Admin; import org.apache.hadoop.hbase.client.Admin;
import org.apache.hadoop.hbase.client.ColumnFamilyDescriptor; import org.apache.hadoop.hbase.client.ColumnFamilyDescriptor;
import org.apache.hadoop.hbase.client.ColumnFamilyDescriptorBuilder; import org.apache.hadoop.hbase.client.ColumnFamilyDescriptorBuilder;
@ -30,7 +30,7 @@ import org.apache.hadoop.hbase.client.RegionInfoBuilder;
import org.apache.hadoop.hbase.client.TableDescriptor; import org.apache.hadoop.hbase.client.TableDescriptor;
import org.apache.hadoop.hbase.io.encoding.DataBlockEncoding; import org.apache.hadoop.hbase.io.encoding.DataBlockEncoding;
import org.apache.hadoop.hbase.regionserver.Region; import org.apache.hadoop.hbase.regionserver.Region;
import org.apache.hadoop.hbase.testclassification.LargeTests; import org.apache.hadoop.hbase.testclassification.MediumTests;
import org.apache.hadoop.hbase.testclassification.MiscTests; import org.apache.hadoop.hbase.testclassification.MiscTests;
import org.apache.hadoop.hbase.util.Bytes; import org.apache.hadoop.hbase.util.Bytes;
import org.junit.After; import org.junit.After;
@ -44,7 +44,7 @@ import org.junit.rules.TestName;
/** /**
* Test being able to edit hbase:meta. * Test being able to edit hbase:meta.
*/ */
@Category({MiscTests.class, LargeTests.class}) @Category({ MiscTests.class, MediumTests.class })
public class TestHBaseMetaEdit { public class TestHBaseMetaEdit {
@ClassRule @ClassRule
public static final HBaseClassTestRule CLASS_RULE = public static final HBaseClassTestRule CLASS_RULE =
@ -63,6 +63,23 @@ public class TestHBaseMetaEdit {
UTIL.shutdownMiniCluster(); UTIL.shutdownMiniCluster();
} }
// make sure that with every possible way, we get the same meta table descriptor.
private TableDescriptor getMetaDescriptor() throws TableNotFoundException, IOException {
Admin admin = UTIL.getAdmin();
TableDescriptor get = admin.getDescriptor(TableName.META_TABLE_NAME);
TableDescriptor list = admin.listTableDescriptors(null, true).stream()
.filter(td -> td.isMetaTable()).findAny().get();
TableDescriptor listByName =
admin.listTableDescriptors(Collections.singletonList(TableName.META_TABLE_NAME)).get(0);
TableDescriptor listByNs =
admin.listTableDescriptorsByNamespace(NamespaceDescriptor.SYSTEM_NAMESPACE_NAME).stream()
.filter(td -> td.isMetaTable()).findAny().get();
assertEquals(get, list);
assertEquals(get, listByName);
assertEquals(get, listByNs);
return get;
}
/** /**
* Set versions, set HBASE-16213 indexed block encoding, and add a column family. * Set versions, set HBASE-16213 indexed block encoding, and add a column family.
* Delete the column family. Then try to delete a core hbase:meta family (should fail). * Delete the column family. Then try to delete a core hbase:meta family (should fail).
@ -73,7 +90,7 @@ public class TestHBaseMetaEdit {
public void testEditMeta() throws IOException { public void testEditMeta() throws IOException {
Admin admin = UTIL.getAdmin(); Admin admin = UTIL.getAdmin();
admin.tableExists(TableName.META_TABLE_NAME); admin.tableExists(TableName.META_TABLE_NAME);
TableDescriptor originalDescriptor = admin.getDescriptor(TableName.META_TABLE_NAME); TableDescriptor originalDescriptor = getMetaDescriptor();
ColumnFamilyDescriptor cfd = originalDescriptor.getColumnFamily(HConstants.CATALOG_FAMILY); ColumnFamilyDescriptor cfd = originalDescriptor.getColumnFamily(HConstants.CATALOG_FAMILY);
int oldVersions = cfd.getMaxVersions(); int oldVersions = cfd.getMaxVersions();
// Add '1' to current versions count. Set encoding too. // Add '1' to current versions count. Set encoding too.
@ -85,11 +102,11 @@ public class TestHBaseMetaEdit {
ColumnFamilyDescriptor newCfd = ColumnFamilyDescriptor newCfd =
ColumnFamilyDescriptorBuilder.newBuilder(extraColumnFamilyName).build(); ColumnFamilyDescriptorBuilder.newBuilder(extraColumnFamilyName).build();
admin.addColumnFamily(TableName.META_TABLE_NAME, newCfd); admin.addColumnFamily(TableName.META_TABLE_NAME, newCfd);
TableDescriptor descriptor = admin.getDescriptor(TableName.META_TABLE_NAME); TableDescriptor descriptor = getMetaDescriptor();
// Assert new max versions is == old versions plus 1. // Assert new max versions is == old versions plus 1.
assertEquals(oldVersions + 1, assertEquals(oldVersions + 1,
descriptor.getColumnFamily(HConstants.CATALOG_FAMILY).getMaxVersions()); descriptor.getColumnFamily(HConstants.CATALOG_FAMILY).getMaxVersions());
descriptor = admin.getDescriptor(TableName.META_TABLE_NAME); descriptor = getMetaDescriptor();
// Assert new max versions is == old versions plus 1. // Assert new max versions is == old versions plus 1.
assertEquals(oldVersions + 1, assertEquals(oldVersions + 1,
descriptor.getColumnFamily(HConstants.CATALOG_FAMILY).getMaxVersions()); descriptor.getColumnFamily(HConstants.CATALOG_FAMILY).getMaxVersions());
@ -107,7 +124,7 @@ public class TestHBaseMetaEdit {
assertTrue(r.getStore(extraColumnFamilyName) != null); assertTrue(r.getStore(extraColumnFamilyName) != null);
// Assert we can't drop critical hbase:meta column family but we can drop any other. // Assert we can't drop critical hbase:meta column family but we can drop any other.
admin.deleteColumnFamily(TableName.META_TABLE_NAME, newCfd.getName()); admin.deleteColumnFamily(TableName.META_TABLE_NAME, newCfd.getName());
descriptor = admin.getDescriptor(TableName.META_TABLE_NAME); descriptor = getMetaDescriptor();
assertTrue(descriptor.getColumnFamily(newCfd.getName()) == null); assertTrue(descriptor.getColumnFamily(newCfd.getName()) == null);
try { try {
admin.deleteColumnFamily(TableName.META_TABLE_NAME, HConstants.CATALOG_FAMILY); admin.deleteColumnFamily(TableName.META_TABLE_NAME, HConstants.CATALOG_FAMILY);

View File

@ -949,7 +949,7 @@ public class TestDefaultMemStore {
edge.setCurrentTimeMillis(1234); edge.setCurrentTimeMillis(1234);
WALFactory wFactory = new WALFactory(conf, "1234"); WALFactory wFactory = new WALFactory(conf, "1234");
HRegion meta = HRegion.createHRegion(RegionInfoBuilder.FIRST_META_REGIONINFO, testDir, HRegion meta = HRegion.createHRegion(RegionInfoBuilder.FIRST_META_REGIONINFO, testDir,
conf, FSTableDescriptors.createMetaTableDescriptor(conf), conf, FSTableDescriptors.createMetaTableDescriptorBuilder(conf).build(),
wFactory.getWAL(RegionInfoBuilder.FIRST_META_REGIONINFO)); wFactory.getWAL(RegionInfoBuilder.FIRST_META_REGIONINFO));
// parameterized tests add [#] suffix get rid of [ and ]. // parameterized tests add [#] suffix get rid of [ and ].
TableDescriptor desc = TableDescriptorBuilder TableDescriptor desc = TableDescriptorBuilder