HBASE-20885 Removed entry for RPC quota from hbase:quota when RPC quota is removed
Signed-off-by: Josh Elser <elserj@apache.org> Signed-off-by: Mike Drob <mdrob@apache.org>
This commit is contained in:
parent
dff5ba27c3
commit
7e9f8c60e2
|
@ -24,6 +24,7 @@ import org.apache.hadoop.hbase.DoNotRetryIOException;
|
||||||
import org.apache.hadoop.hbase.TableName;
|
import org.apache.hadoop.hbase.TableName;
|
||||||
import org.apache.yetus.audience.InterfaceAudience;
|
import org.apache.yetus.audience.InterfaceAudience;
|
||||||
import org.apache.yetus.audience.InterfaceStability;
|
import org.apache.yetus.audience.InterfaceStability;
|
||||||
|
import org.apache.hbase.thirdparty.com.google.common.annotations.VisibleForTesting;
|
||||||
import org.apache.hadoop.hbase.shaded.protobuf.generated.MasterProtos.SetQuotaRequest;
|
import org.apache.hadoop.hbase.shaded.protobuf.generated.MasterProtos.SetQuotaRequest;
|
||||||
import org.apache.hadoop.hbase.shaded.protobuf.generated.QuotaProtos.TimedQuota;
|
import org.apache.hadoop.hbase.shaded.protobuf.generated.QuotaProtos.TimedQuota;
|
||||||
import org.apache.hadoop.hbase.shaded.protobuf.ProtobufUtil;
|
import org.apache.hadoop.hbase.shaded.protobuf.ProtobufUtil;
|
||||||
|
@ -48,6 +49,14 @@ class ThrottleSettings extends QuotaSettings {
|
||||||
return proto.hasTimedQuota() ? proto.getTimedQuota().getSoftLimit() : -1;
|
return proto.hasTimedQuota() ? proto.getTimedQuota().getSoftLimit() : -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns a copy of the internal state of <code>this</code>
|
||||||
|
*/
|
||||||
|
@VisibleForTesting
|
||||||
|
QuotaProtos.ThrottleRequest getProto() {
|
||||||
|
return proto.toBuilder().build();
|
||||||
|
}
|
||||||
|
|
||||||
public TimeUnit getTimeUnit() {
|
public TimeUnit getTimeUnit() {
|
||||||
return proto.hasTimedQuota() ?
|
return proto.hasTimedQuota() ?
|
||||||
ProtobufUtil.toTimeUnit(proto.getTimedQuota().getTimeUnit()) : null;
|
ProtobufUtil.toTimeUnit(proto.getTimedQuota().getTimeUnit()) : null;
|
||||||
|
|
|
@ -115,70 +115,41 @@ public class GlobalQuotaSettingsImpl extends GlobalQuotaSettings {
|
||||||
validateQuotaTarget(other);
|
validateQuotaTarget(other);
|
||||||
|
|
||||||
// Propagate the Throttle
|
// Propagate the Throttle
|
||||||
QuotaProtos.Throttle.Builder throttleBuilder = (throttleProto == null
|
QuotaProtos.Throttle.Builder throttleBuilder =
|
||||||
? null : throttleProto.toBuilder());
|
throttleProto == null ? null : throttleProto.toBuilder();
|
||||||
|
|
||||||
if (other instanceof ThrottleSettings) {
|
if (other instanceof ThrottleSettings) {
|
||||||
|
ThrottleSettings otherThrottle = (ThrottleSettings) other;
|
||||||
|
if (!otherThrottle.proto.hasType() || !otherThrottle.proto.hasTimedQuota()) {
|
||||||
|
// To prevent the "empty" row in QuotaTableUtil.QUOTA_TABLE_NAME
|
||||||
|
throttleBuilder = null;
|
||||||
|
} else {
|
||||||
|
QuotaProtos.ThrottleRequest otherProto = otherThrottle.proto;
|
||||||
|
validateTimedQuota(otherProto.getTimedQuota());
|
||||||
if (throttleBuilder == null) {
|
if (throttleBuilder == null) {
|
||||||
throttleBuilder = QuotaProtos.Throttle.newBuilder();
|
throttleBuilder = QuotaProtos.Throttle.newBuilder();
|
||||||
}
|
}
|
||||||
ThrottleSettings otherThrottle = (ThrottleSettings) other;
|
|
||||||
|
|
||||||
if (otherThrottle.proto.hasType()) {
|
|
||||||
QuotaProtos.ThrottleRequest otherProto = otherThrottle.proto;
|
|
||||||
if (otherProto.hasTimedQuota()) {
|
|
||||||
if (otherProto.hasTimedQuota()) {
|
|
||||||
validateTimedQuota(otherProto.getTimedQuota());
|
|
||||||
}
|
|
||||||
|
|
||||||
switch (otherProto.getType()) {
|
switch (otherProto.getType()) {
|
||||||
case REQUEST_NUMBER:
|
case REQUEST_NUMBER:
|
||||||
if (otherProto.hasTimedQuota()) {
|
|
||||||
throttleBuilder.setReqNum(otherProto.getTimedQuota());
|
throttleBuilder.setReqNum(otherProto.getTimedQuota());
|
||||||
} else {
|
|
||||||
throttleBuilder.clearReqNum();
|
|
||||||
}
|
|
||||||
break;
|
break;
|
||||||
case REQUEST_SIZE:
|
case REQUEST_SIZE:
|
||||||
if (otherProto.hasTimedQuota()) {
|
|
||||||
throttleBuilder.setReqSize(otherProto.getTimedQuota());
|
throttleBuilder.setReqSize(otherProto.getTimedQuota());
|
||||||
} else {
|
|
||||||
throttleBuilder.clearReqSize();
|
|
||||||
}
|
|
||||||
break;
|
break;
|
||||||
case WRITE_NUMBER:
|
case WRITE_NUMBER:
|
||||||
if (otherProto.hasTimedQuota()) {
|
|
||||||
throttleBuilder.setWriteNum(otherProto.getTimedQuota());
|
throttleBuilder.setWriteNum(otherProto.getTimedQuota());
|
||||||
} else {
|
|
||||||
throttleBuilder.clearWriteNum();
|
|
||||||
}
|
|
||||||
break;
|
break;
|
||||||
case WRITE_SIZE:
|
case WRITE_SIZE:
|
||||||
if (otherProto.hasTimedQuota()) {
|
|
||||||
throttleBuilder.setWriteSize(otherProto.getTimedQuota());
|
throttleBuilder.setWriteSize(otherProto.getTimedQuota());
|
||||||
} else {
|
|
||||||
throttleBuilder.clearWriteSize();
|
|
||||||
}
|
|
||||||
break;
|
break;
|
||||||
case READ_NUMBER:
|
case READ_NUMBER:
|
||||||
if (otherProto.hasTimedQuota()) {
|
|
||||||
throttleBuilder.setReadNum(otherProto.getTimedQuota());
|
throttleBuilder.setReadNum(otherProto.getTimedQuota());
|
||||||
} else {
|
|
||||||
throttleBuilder.clearReqNum();
|
|
||||||
}
|
|
||||||
break;
|
break;
|
||||||
case READ_SIZE:
|
case READ_SIZE:
|
||||||
if (otherProto.hasTimedQuota()) {
|
|
||||||
throttleBuilder.setReadSize(otherProto.getTimedQuota());
|
throttleBuilder.setReadSize(otherProto.getTimedQuota());
|
||||||
} else {
|
|
||||||
throttleBuilder.clearReadSize();
|
|
||||||
}
|
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
} else {
|
|
||||||
clearThrottleBuilder(throttleBuilder);
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
clearThrottleBuilder(throttleBuilder);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -368,7 +368,7 @@ public class TestQuotaAdmin {
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
public void testSetModifyRemoveQuota() throws Exception {
|
public void testSetModifyRemoveSpaceQuota() throws Exception {
|
||||||
Admin admin = TEST_UTIL.getAdmin();
|
Admin admin = TEST_UTIL.getAdmin();
|
||||||
final TableName tn = TableName.valueOf("sq_table2");
|
final TableName tn = TableName.valueOf("sq_table2");
|
||||||
final long originalSizeLimit = 1024L * 1024L * 1024L * 1024L * 5L; // 5TB
|
final long originalSizeLimit = 1024L * 1024L * 1024L * 1024L * 5L; // 5TB
|
||||||
|
@ -453,6 +453,154 @@ public class TestQuotaAdmin {
|
||||||
assertEquals(expected, countResults(filter));
|
assertEquals(expected, countResults(filter));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testSetGetRemoveRPCQuota() throws Exception {
|
||||||
|
Admin admin = TEST_UTIL.getAdmin();
|
||||||
|
final TableName tn = TableName.valueOf("sq_table1");
|
||||||
|
QuotaSettings settings =
|
||||||
|
QuotaSettingsFactory.throttleTable(tn, ThrottleType.REQUEST_SIZE, 2L, TimeUnit.HOURS);
|
||||||
|
admin.setQuota(settings);
|
||||||
|
|
||||||
|
// Verify the Quota in the table
|
||||||
|
verifyRecordPresentInQuotaTable(ThrottleType.REQUEST_SIZE, 2L, TimeUnit.HOURS);
|
||||||
|
|
||||||
|
// Verify we can retrieve it via the QuotaRetriever API
|
||||||
|
verifyFetchableViaAPI(admin, ThrottleType.REQUEST_SIZE, 2L, TimeUnit.HOURS);
|
||||||
|
|
||||||
|
// Now, remove the quota
|
||||||
|
QuotaSettings removeQuota = QuotaSettingsFactory.unthrottleTable(tn);
|
||||||
|
admin.setQuota(removeQuota);
|
||||||
|
|
||||||
|
// Verify that the record doesn't exist in the table
|
||||||
|
verifyRecordNotPresentInQuotaTable();
|
||||||
|
|
||||||
|
// Verify that we can also not fetch it via the API
|
||||||
|
verifyNotFetchableViaAPI(admin);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testSetModifyRemoveRPCQuota() throws Exception {
|
||||||
|
Admin admin = TEST_UTIL.getAdmin();
|
||||||
|
final TableName tn = TableName.valueOf("sq_table1");
|
||||||
|
QuotaSettings settings =
|
||||||
|
QuotaSettingsFactory.throttleTable(tn, ThrottleType.REQUEST_SIZE, 2L, TimeUnit.HOURS);
|
||||||
|
admin.setQuota(settings);
|
||||||
|
|
||||||
|
// Verify the Quota in the table
|
||||||
|
verifyRecordPresentInQuotaTable(ThrottleType.REQUEST_SIZE, 2L, TimeUnit.HOURS);
|
||||||
|
|
||||||
|
// Verify we can retrieve it via the QuotaRetriever API
|
||||||
|
verifyFetchableViaAPI(admin, ThrottleType.REQUEST_SIZE, 2L, TimeUnit.HOURS);
|
||||||
|
|
||||||
|
// Setting a limit and time unit should be reflected
|
||||||
|
QuotaSettings newSettings =
|
||||||
|
QuotaSettingsFactory.throttleTable(tn, ThrottleType.REQUEST_SIZE, 3L, TimeUnit.DAYS);
|
||||||
|
admin.setQuota(newSettings);
|
||||||
|
|
||||||
|
// Verify the new Quota in the table
|
||||||
|
verifyRecordPresentInQuotaTable(ThrottleType.REQUEST_SIZE, 3L, TimeUnit.DAYS);
|
||||||
|
|
||||||
|
// Verify we can retrieve the new quota via the QuotaRetriever API
|
||||||
|
verifyFetchableViaAPI(admin, ThrottleType.REQUEST_SIZE, 3L, TimeUnit.DAYS);
|
||||||
|
|
||||||
|
// Now, remove the quota
|
||||||
|
QuotaSettings removeQuota = QuotaSettingsFactory.unthrottleTable(tn);
|
||||||
|
admin.setQuota(removeQuota);
|
||||||
|
|
||||||
|
// Verify that the record doesn't exist in the table
|
||||||
|
verifyRecordNotPresentInQuotaTable();
|
||||||
|
|
||||||
|
// Verify that we can also not fetch it via the API
|
||||||
|
verifyNotFetchableViaAPI(admin);
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
private void verifyRecordPresentInQuotaTable(ThrottleType type, long limit, TimeUnit tu)
|
||||||
|
throws Exception {
|
||||||
|
// Verify the RPC Quotas in the table
|
||||||
|
try (Table quotaTable = TEST_UTIL.getConnection().getTable(QuotaTableUtil.QUOTA_TABLE_NAME);
|
||||||
|
ResultScanner scanner = quotaTable.getScanner(new Scan())) {
|
||||||
|
Result r = Iterables.getOnlyElement(scanner);
|
||||||
|
CellScanner cells = r.cellScanner();
|
||||||
|
assertTrue("Expected to find a cell", cells.advance());
|
||||||
|
assertRPCQuota(type, limit, tu, cells.current());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private void verifyRecordNotPresentInQuotaTable() throws Exception {
|
||||||
|
// Verify that the record doesn't exist in the QuotaTableUtil.QUOTA_TABLE_NAME
|
||||||
|
try (Table quotaTable = TEST_UTIL.getConnection().getTable(QuotaTableUtil.QUOTA_TABLE_NAME);
|
||||||
|
ResultScanner scanner = quotaTable.getScanner(new Scan())) {
|
||||||
|
assertNull("Did not expect to find a quota entry", scanner.next());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private void verifyFetchableViaAPI(Admin admin, ThrottleType type, long limit, TimeUnit tu)
|
||||||
|
throws Exception {
|
||||||
|
// Verify we can retrieve the new quota via the QuotaRetriever API
|
||||||
|
try (QuotaRetriever quotaScanner = QuotaRetriever.open(admin.getConfiguration())) {
|
||||||
|
assertRPCQuota(type, limit, tu, Iterables.getOnlyElement(quotaScanner));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private void verifyNotFetchableViaAPI(Admin admin) throws Exception {
|
||||||
|
// Verify that we can also not fetch it via the API
|
||||||
|
try (QuotaRetriever quotaScanner = QuotaRetriever.open(admin.getConfiguration())) {
|
||||||
|
assertNull("Did not expect to find a quota entry", quotaScanner.next());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private void assertRPCQuota(ThrottleType type, long limit, TimeUnit tu, Cell cell)
|
||||||
|
throws Exception {
|
||||||
|
Quotas q = QuotaTableUtil
|
||||||
|
.quotasFromData(cell.getValueArray(), cell.getValueOffset(), cell.getValueLength());
|
||||||
|
assertTrue("Quota should have rpc quota defined", q.hasThrottle());
|
||||||
|
|
||||||
|
QuotaProtos.Throttle rpcQuota = q.getThrottle();
|
||||||
|
QuotaProtos.TimedQuota t = null;
|
||||||
|
|
||||||
|
switch (type) {
|
||||||
|
case REQUEST_SIZE:
|
||||||
|
assertTrue(rpcQuota.hasReqSize());
|
||||||
|
t = rpcQuota.getReqSize();
|
||||||
|
break;
|
||||||
|
case READ_NUMBER:
|
||||||
|
assertTrue(rpcQuota.hasReadNum());
|
||||||
|
t = rpcQuota.getReadNum();
|
||||||
|
break;
|
||||||
|
case READ_SIZE:
|
||||||
|
assertTrue(rpcQuota.hasReadSize());
|
||||||
|
t = rpcQuota.getReadSize();
|
||||||
|
break;
|
||||||
|
case REQUEST_NUMBER:
|
||||||
|
assertTrue(rpcQuota.hasReqNum());
|
||||||
|
t = rpcQuota.getReqNum();
|
||||||
|
break;
|
||||||
|
case WRITE_NUMBER:
|
||||||
|
assertTrue(rpcQuota.hasWriteNum());
|
||||||
|
t = rpcQuota.getWriteNum();
|
||||||
|
break;
|
||||||
|
case WRITE_SIZE:
|
||||||
|
assertTrue(rpcQuota.hasWriteSize());
|
||||||
|
t = rpcQuota.getWriteSize();
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
assertEquals(t.getSoftLimit(), limit);
|
||||||
|
assertEquals(t.getTimeUnit(), ProtobufUtil.toProtoTimeUnit(tu));
|
||||||
|
}
|
||||||
|
|
||||||
|
private void assertRPCQuota(ThrottleType type, long limit, TimeUnit tu,
|
||||||
|
QuotaSettings actualSettings) throws Exception {
|
||||||
|
assertTrue(
|
||||||
|
"The actual QuotaSettings was not an instance of " + ThrottleSettings.class + " but of "
|
||||||
|
+ actualSettings.getClass(), actualSettings instanceof ThrottleSettings);
|
||||||
|
QuotaProtos.ThrottleRequest throttleRequest = ((ThrottleSettings) actualSettings).getProto();
|
||||||
|
assertEquals(limit, throttleRequest.getTimedQuota().getSoftLimit());
|
||||||
|
assertEquals(ProtobufUtil.toProtoTimeUnit(tu), throttleRequest.getTimedQuota().getTimeUnit());
|
||||||
|
assertEquals(ProtobufUtil.toProtoThrottleType(type), throttleRequest.getType());
|
||||||
|
}
|
||||||
|
|
||||||
private void assertSpaceQuota(
|
private void assertSpaceQuota(
|
||||||
long sizeLimit, SpaceViolationPolicy violationPolicy, Cell cell) throws Exception {
|
long sizeLimit, SpaceViolationPolicy violationPolicy, Cell cell) throws Exception {
|
||||||
Quotas q = QuotaTableUtil.quotasFromData(
|
Quotas q = QuotaTableUtil.quotasFromData(
|
||||||
|
|
Loading…
Reference in New Issue