HDDS-1685. Recon: Add support for "start" query param to containers and containers/{id} endpoints.
This commit is contained in:
parent
f8d62a9c4c
commit
db674a0b14
|
@ -38,4 +38,8 @@ public final class ReconConstants {
|
||||||
public static final String CONTAINER_KEY_TABLE =
|
public static final String CONTAINER_KEY_TABLE =
|
||||||
"containerKeyTable";
|
"containerKeyTable";
|
||||||
|
|
||||||
|
public static final String FETCH_ALL = "-1";
|
||||||
|
public static final String RECON_QUERY_PREVKEY = "prev-key";
|
||||||
|
public static final String RECON_QUERY_LIMIT = "limit";
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -38,6 +38,7 @@ import javax.ws.rs.WebApplicationException;
|
||||||
import javax.ws.rs.core.MediaType;
|
import javax.ws.rs.core.MediaType;
|
||||||
import javax.ws.rs.core.Response;
|
import javax.ws.rs.core.Response;
|
||||||
|
|
||||||
|
import org.apache.commons.lang3.StringUtils;
|
||||||
import org.apache.hadoop.ozone.om.helpers.OmKeyInfo;
|
import org.apache.hadoop.ozone.om.helpers.OmKeyInfo;
|
||||||
import org.apache.hadoop.ozone.om.helpers.OmKeyLocationInfo;
|
import org.apache.hadoop.ozone.om.helpers.OmKeyLocationInfo;
|
||||||
import org.apache.hadoop.ozone.om.helpers.OmKeyLocationInfoGroup;
|
import org.apache.hadoop.ozone.om.helpers.OmKeyLocationInfoGroup;
|
||||||
|
@ -50,6 +51,10 @@ import org.apache.hadoop.ozone.recon.spi.ContainerDBServiceProvider;
|
||||||
import org.slf4j.Logger;
|
import org.slf4j.Logger;
|
||||||
import org.slf4j.LoggerFactory;
|
import org.slf4j.LoggerFactory;
|
||||||
|
|
||||||
|
import static org.apache.hadoop.ozone.recon.ReconConstants.FETCH_ALL;
|
||||||
|
import static org.apache.hadoop.ozone.recon.ReconConstants.RECON_QUERY_LIMIT;
|
||||||
|
import static org.apache.hadoop.ozone.recon.ReconConstants.RECON_QUERY_PREVKEY;
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Endpoint for querying keys that belong to a container.
|
* Endpoint for querying keys that belong to a container.
|
||||||
|
@ -69,16 +74,20 @@ public class ContainerKeyService {
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Return @{@link org.apache.hadoop.ozone.recon.api.types.ContainerMetadata}
|
* Return @{@link org.apache.hadoop.ozone.recon.api.types.ContainerMetadata}
|
||||||
* for all the containers.
|
* for the containers starting from the given "prev-key" query param for the
|
||||||
|
* given "limit". The given "prev-key" is skipped from the results returned.
|
||||||
*
|
*
|
||||||
|
* @param limit max no. of containers to get.
|
||||||
|
* @param prevKey the containerID after which results are returned.
|
||||||
* @return {@link Response}
|
* @return {@link Response}
|
||||||
*/
|
*/
|
||||||
@GET
|
@GET
|
||||||
public Response getContainers(
|
public Response getContainers(
|
||||||
@DefaultValue("-1") @QueryParam("limit") int limit) {
|
@DefaultValue(FETCH_ALL) @QueryParam(RECON_QUERY_LIMIT) int limit,
|
||||||
|
@DefaultValue("0") @QueryParam(RECON_QUERY_PREVKEY) long prevKey) {
|
||||||
Map<Long, ContainerMetadata> containersMap;
|
Map<Long, ContainerMetadata> containersMap;
|
||||||
try {
|
try {
|
||||||
containersMap = containerDBServiceProvider.getContainers(limit);
|
containersMap = containerDBServiceProvider.getContainers(limit, prevKey);
|
||||||
} catch (IOException ioEx) {
|
} catch (IOException ioEx) {
|
||||||
throw new WebApplicationException(ioEx,
|
throw new WebApplicationException(ioEx,
|
||||||
Response.Status.INTERNAL_SERVER_ERROR);
|
Response.Status.INTERNAL_SERVER_ERROR);
|
||||||
|
@ -88,20 +97,27 @@ public class ContainerKeyService {
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Return @{@link org.apache.hadoop.ozone.recon.api.types.KeyMetadata} for
|
* Return @{@link org.apache.hadoop.ozone.recon.api.types.KeyMetadata} for
|
||||||
* all keys that belong to the container identified by the id param.
|
* all keys that belong to the container identified by the id param
|
||||||
|
* starting from the given "prev-key" query param for the given "limit".
|
||||||
|
* The given prevKeyPrefix is skipped from the results returned.
|
||||||
*
|
*
|
||||||
* @param containerId Container Id
|
* @param containerID the given containerID.
|
||||||
|
* @param limit max no. of keys to get.
|
||||||
|
* @param prevKeyPrefix the key prefix after which results are returned.
|
||||||
* @return {@link Response}
|
* @return {@link Response}
|
||||||
*/
|
*/
|
||||||
@GET
|
@GET
|
||||||
@Path("/{id}")
|
@Path("/{id}")
|
||||||
public Response getKeysForContainer(
|
public Response getKeysForContainer(
|
||||||
@PathParam("id") Long containerId,
|
@PathParam("id") Long containerID,
|
||||||
@DefaultValue("-1") @QueryParam("limit") int limit) {
|
@DefaultValue(FETCH_ALL) @QueryParam(RECON_QUERY_LIMIT) int limit,
|
||||||
|
@DefaultValue(StringUtils.EMPTY) @QueryParam(RECON_QUERY_PREVKEY)
|
||||||
|
String prevKeyPrefix) {
|
||||||
Map<String, KeyMetadata> keyMetadataMap = new LinkedHashMap<>();
|
Map<String, KeyMetadata> keyMetadataMap = new LinkedHashMap<>();
|
||||||
try {
|
try {
|
||||||
Map<ContainerKeyPrefix, Integer> containerKeyPrefixMap =
|
Map<ContainerKeyPrefix, Integer> containerKeyPrefixMap =
|
||||||
containerDBServiceProvider.getKeyPrefixesForContainer(containerId);
|
containerDBServiceProvider.getKeyPrefixesForContainer(containerID,
|
||||||
|
prevKeyPrefix);
|
||||||
|
|
||||||
// Get set of Container-Key mappings for given containerId.
|
// Get set of Container-Key mappings for given containerId.
|
||||||
for (ContainerKeyPrefix containerKeyPrefix : containerKeyPrefixMap
|
for (ContainerKeyPrefix containerKeyPrefix : containerKeyPrefixMap
|
||||||
|
@ -128,7 +144,7 @@ public class ContainerKeyService {
|
||||||
List<OmKeyLocationInfo> omKeyLocationInfos = omKeyLocationInfoGroup
|
List<OmKeyLocationInfo> omKeyLocationInfos = omKeyLocationInfoGroup
|
||||||
.getLocationList()
|
.getLocationList()
|
||||||
.stream()
|
.stream()
|
||||||
.filter(c -> c.getContainerID() == containerId)
|
.filter(c -> c.getContainerID() == containerID)
|
||||||
.collect(Collectors.toList());
|
.collect(Collectors.toList());
|
||||||
for (OmKeyLocationInfo omKeyLocationInfo : omKeyLocationInfos) {
|
for (OmKeyLocationInfo omKeyLocationInfo : omKeyLocationInfos) {
|
||||||
blockIds.add(new ContainerBlockMetadata(omKeyLocationInfo
|
blockIds.add(new ContainerBlockMetadata(omKeyLocationInfo
|
||||||
|
|
|
@ -66,26 +66,32 @@ public interface ContainerDBServiceProvider {
|
||||||
* @param containerId the given containerId.
|
* @param containerId the given containerId.
|
||||||
* @return Map of Key prefix -> count.
|
* @return Map of Key prefix -> count.
|
||||||
*/
|
*/
|
||||||
Map<ContainerKeyPrefix, Integer> getKeyPrefixesForContainer(long containerId)
|
Map<ContainerKeyPrefix, Integer> getKeyPrefixesForContainer(
|
||||||
throws IOException;
|
long containerId) throws IOException;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Get a Map of containerID, containerMetadata of all the Containers.
|
* Get the stored key prefixes for the given containerId starting
|
||||||
|
* after the given keyPrefix.
|
||||||
*
|
*
|
||||||
* @return Map of containerID -> containerMetadata.
|
* @param containerId the given containerId.
|
||||||
* @throws IOException
|
* @param prevKeyPrefix the key prefix to seek to and start scanning.
|
||||||
|
* @return Map of Key prefix -> count.
|
||||||
*/
|
*/
|
||||||
Map<Long, ContainerMetadata> getContainers() throws IOException;
|
Map<ContainerKeyPrefix, Integer> getKeyPrefixesForContainer(
|
||||||
|
long containerId, String prevKeyPrefix) throws IOException;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Get a Map of containerID, containerMetadata of Containers only for the
|
* Get a Map of containerID, containerMetadata of Containers only for the
|
||||||
* given limit. If the limit is -1 or any integer <0, then return all
|
* given limit. If the limit is -1 or any integer <0, then return all
|
||||||
* the containers without any limit.
|
* the containers without any limit.
|
||||||
*
|
*
|
||||||
|
* @param limit the no. of containers to fetch.
|
||||||
|
* @param prevContainer containerID after which the results are returned.
|
||||||
* @return Map of containerID -> containerMetadata.
|
* @return Map of containerID -> containerMetadata.
|
||||||
* @throws IOException
|
* @throws IOException
|
||||||
*/
|
*/
|
||||||
Map<Long, ContainerMetadata> getContainers(int limit) throws IOException;
|
Map<Long, ContainerMetadata> getContainers(int limit, long prevContainer)
|
||||||
|
throws IOException;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Delete an entry in the container DB.
|
* Delete an entry in the container DB.
|
||||||
|
@ -98,7 +104,6 @@ public interface ContainerDBServiceProvider {
|
||||||
/**
|
/**
|
||||||
* Get iterator to the entire container DB.
|
* Get iterator to the entire container DB.
|
||||||
* @return TableIterator
|
* @return TableIterator
|
||||||
* @throws IOException exception
|
|
||||||
*/
|
*/
|
||||||
TableIterator getContainerTableIterator() throws IOException;
|
TableIterator getContainerTableIterator();
|
||||||
}
|
}
|
||||||
|
|
|
@ -128,8 +128,7 @@ public class ContainerDBServiceProviderImpl
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Use the DB's prefix seek iterator to start the scan from the given
|
* Get key prefixes for the given container ID.
|
||||||
* container ID prefix.
|
|
||||||
*
|
*
|
||||||
* @param containerId the given containerId.
|
* @param containerId the given containerId.
|
||||||
* @return Map of (Key-Prefix,Count of Keys).
|
* @return Map of (Key-Prefix,Count of Keys).
|
||||||
|
@ -137,14 +136,57 @@ public class ContainerDBServiceProviderImpl
|
||||||
@Override
|
@Override
|
||||||
public Map<ContainerKeyPrefix, Integer> getKeyPrefixesForContainer(
|
public Map<ContainerKeyPrefix, Integer> getKeyPrefixesForContainer(
|
||||||
long containerId) throws IOException {
|
long containerId) throws IOException {
|
||||||
|
// set the default startKeyPrefix to empty string
|
||||||
|
return getKeyPrefixesForContainer(containerId, StringUtils.EMPTY);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Use the DB's prefix seek iterator to start the scan from the given
|
||||||
|
* container ID and prev key prefix. The prev key prefix is skipped from
|
||||||
|
* the result.
|
||||||
|
*
|
||||||
|
* @param containerId the given containerId.
|
||||||
|
* @param prevKeyPrefix the given key prefix to start the scan from.
|
||||||
|
* @return Map of (Key-Prefix,Count of Keys).
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
public Map<ContainerKeyPrefix, Integer> getKeyPrefixesForContainer(
|
||||||
|
long containerId, String prevKeyPrefix) throws IOException {
|
||||||
|
|
||||||
Map<ContainerKeyPrefix, Integer> prefixes = new LinkedHashMap<>();
|
Map<ContainerKeyPrefix, Integer> prefixes = new LinkedHashMap<>();
|
||||||
TableIterator<ContainerKeyPrefix, ? extends KeyValue<ContainerKeyPrefix,
|
TableIterator<ContainerKeyPrefix, ? extends KeyValue<ContainerKeyPrefix,
|
||||||
Integer>> containerIterator = containerKeyTable.iterator();
|
Integer>> containerIterator = containerKeyTable.iterator();
|
||||||
containerIterator.seek(new ContainerKeyPrefix(containerId));
|
ContainerKeyPrefix seekKey;
|
||||||
|
boolean skipPrevKey = false;
|
||||||
|
if (StringUtils.isNotBlank(prevKeyPrefix)) {
|
||||||
|
skipPrevKey = true;
|
||||||
|
seekKey = new ContainerKeyPrefix(containerId, prevKeyPrefix);
|
||||||
|
} else {
|
||||||
|
seekKey = new ContainerKeyPrefix(containerId);
|
||||||
|
}
|
||||||
|
KeyValue<ContainerKeyPrefix, Integer> seekKeyValue =
|
||||||
|
containerIterator.seek(seekKey);
|
||||||
|
|
||||||
|
// check if RocksDB was able to seek correctly to the given key prefix
|
||||||
|
// if not, then return empty result
|
||||||
|
// In case of an empty prevKeyPrefix, all the keys in the container are
|
||||||
|
// returned
|
||||||
|
if (seekKeyValue == null ||
|
||||||
|
(StringUtils.isNotBlank(prevKeyPrefix) &&
|
||||||
|
!seekKeyValue.getKey().getKeyPrefix().equals(prevKeyPrefix))) {
|
||||||
|
return prefixes;
|
||||||
|
}
|
||||||
|
|
||||||
while (containerIterator.hasNext()) {
|
while (containerIterator.hasNext()) {
|
||||||
KeyValue<ContainerKeyPrefix, Integer> keyValue = containerIterator.next();
|
KeyValue<ContainerKeyPrefix, Integer> keyValue = containerIterator.next();
|
||||||
ContainerKeyPrefix containerKeyPrefix = keyValue.getKey();
|
ContainerKeyPrefix containerKeyPrefix = keyValue.getKey();
|
||||||
|
|
||||||
|
// skip the prev key if prev key is present
|
||||||
|
if (skipPrevKey &&
|
||||||
|
containerKeyPrefix.getKeyPrefix().equals(prevKeyPrefix)) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
// The prefix seek only guarantees that the iterator's head will be
|
// The prefix seek only guarantees that the iterator's head will be
|
||||||
// positioned at the first prefix match. We still have to check the key
|
// positioned at the first prefix match. We still have to check the key
|
||||||
// prefix.
|
// prefix.
|
||||||
|
@ -164,36 +206,46 @@ public class ContainerDBServiceProviderImpl
|
||||||
return prefixes;
|
return prefixes;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* Get all the containers.
|
|
||||||
*
|
|
||||||
* @return Map of containerID -> containerMetadata.
|
|
||||||
* @throws IOException
|
|
||||||
*/
|
|
||||||
@Override
|
|
||||||
public Map<Long, ContainerMetadata> getContainers() throws IOException {
|
|
||||||
// Set a negative limit to get all the containers.
|
|
||||||
return getContainers(-1);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Iterate the DB to construct a Map of containerID -> containerMetadata
|
* Iterate the DB to construct a Map of containerID -> containerMetadata
|
||||||
* only for the given limit.
|
* only for the given limit from the given start key. The start containerID
|
||||||
|
* is skipped from the result.
|
||||||
*
|
*
|
||||||
* Return all the containers if limit < 0.
|
* Return all the containers if limit < 0.
|
||||||
*
|
*
|
||||||
|
* @param limit No of containers to get.
|
||||||
|
* @param prevContainer containerID after which the
|
||||||
|
* list of containers are scanned.
|
||||||
* @return Map of containerID -> containerMetadata.
|
* @return Map of containerID -> containerMetadata.
|
||||||
* @throws IOException
|
* @throws IOException
|
||||||
*/
|
*/
|
||||||
@Override
|
@Override
|
||||||
public Map<Long, ContainerMetadata> getContainers(int limit)
|
public Map<Long, ContainerMetadata> getContainers(int limit,
|
||||||
|
long prevContainer)
|
||||||
throws IOException {
|
throws IOException {
|
||||||
Map<Long, ContainerMetadata> containers = new LinkedHashMap<>();
|
Map<Long, ContainerMetadata> containers = new LinkedHashMap<>();
|
||||||
TableIterator<ContainerKeyPrefix, ? extends KeyValue<ContainerKeyPrefix,
|
TableIterator<ContainerKeyPrefix, ? extends KeyValue<ContainerKeyPrefix,
|
||||||
Integer>> containerIterator = containerKeyTable.iterator();
|
Integer>> containerIterator = containerKeyTable.iterator();
|
||||||
|
ContainerKeyPrefix seekKey;
|
||||||
|
if (prevContainer > 0L) {
|
||||||
|
seekKey = new ContainerKeyPrefix(prevContainer);
|
||||||
|
KeyValue<ContainerKeyPrefix,
|
||||||
|
Integer> seekKeyValue = containerIterator.seek(seekKey);
|
||||||
|
// Check if RocksDB was able to correctly seek to the given
|
||||||
|
// prevContainer containerId. If not, then return empty result
|
||||||
|
if (seekKeyValue != null &&
|
||||||
|
seekKeyValue.getKey().getContainerId() != prevContainer) {
|
||||||
|
return containers;
|
||||||
|
} else {
|
||||||
|
// seek to the prevContainer+1 containerID to start scan
|
||||||
|
seekKey = new ContainerKeyPrefix(prevContainer + 1);
|
||||||
|
containerIterator.seek(seekKey);
|
||||||
|
}
|
||||||
|
}
|
||||||
while (containerIterator.hasNext()) {
|
while (containerIterator.hasNext()) {
|
||||||
KeyValue<ContainerKeyPrefix, Integer> keyValue = containerIterator.next();
|
KeyValue<ContainerKeyPrefix, Integer> keyValue = containerIterator.next();
|
||||||
Long containerID = keyValue.getKey().getContainerId();
|
ContainerKeyPrefix containerKeyPrefix = keyValue.getKey();
|
||||||
|
Long containerID = containerKeyPrefix.getContainerId();
|
||||||
Integer numberOfKeys = keyValue.getValue();
|
Integer numberOfKeys = keyValue.getValue();
|
||||||
|
|
||||||
// break the loop if limit has been reached
|
// break the loop if limit has been reached
|
||||||
|
@ -220,7 +272,7 @@ public class ContainerDBServiceProviderImpl
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public TableIterator getContainerTableIterator() throws IOException {
|
public TableIterator getContainerTableIterator() {
|
||||||
return containerKeyTable.iterator();
|
return containerKeyTable.iterator();
|
||||||
}
|
}
|
||||||
}
|
}
|
|
@ -36,6 +36,7 @@ import java.util.Map;
|
||||||
|
|
||||||
import javax.ws.rs.core.Response;
|
import javax.ws.rs.core.Response;
|
||||||
|
|
||||||
|
import org.apache.commons.lang3.StringUtils;
|
||||||
import org.apache.hadoop.hdds.client.BlockID;
|
import org.apache.hadoop.hdds.client.BlockID;
|
||||||
import org.apache.hadoop.hdds.conf.OzoneConfiguration;
|
import org.apache.hadoop.hdds.conf.OzoneConfiguration;
|
||||||
import org.apache.hadoop.hdds.scm.pipeline.Pipeline;
|
import org.apache.hadoop.hdds.scm.pipeline.Pipeline;
|
||||||
|
@ -132,7 +133,7 @@ public class TestContainerKeyService extends AbstractOMMetadataManagerTest {
|
||||||
OmKeyLocationInfoGroup omKeyLocationInfoGroup = new
|
OmKeyLocationInfoGroup omKeyLocationInfoGroup = new
|
||||||
OmKeyLocationInfoGroup(0, omKeyLocationInfoList);
|
OmKeyLocationInfoGroup(0, omKeyLocationInfoList);
|
||||||
|
|
||||||
//key = key_one, Blocks = [ {CID = 1, LID = 1}, {CID = 2, LID = 1} ]
|
//key = key_one, Blocks = [ {CID = 1, LID = 101}, {CID = 2, LID = 102} ]
|
||||||
writeDataToOm(omMetadataManager,
|
writeDataToOm(omMetadataManager,
|
||||||
"key_one", "bucketOne", "sampleVol",
|
"key_one", "bucketOne", "sampleVol",
|
||||||
Collections.singletonList(omKeyLocationInfoGroup));
|
Collections.singletonList(omKeyLocationInfoGroup));
|
||||||
|
@ -156,7 +157,7 @@ public class TestContainerKeyService extends AbstractOMMetadataManagerTest {
|
||||||
infoGroups.add(new OmKeyLocationInfoGroup(1,
|
infoGroups.add(new OmKeyLocationInfoGroup(1,
|
||||||
omKeyLocationInfoListNew));
|
omKeyLocationInfoListNew));
|
||||||
|
|
||||||
//key = key_two, Blocks = [ {CID = 1, LID = 2}, {CID = 1, LID = 3} ]
|
//key = key_two, Blocks = [ {CID = 1, LID = 103}, {CID = 1, LID = 104} ]
|
||||||
writeDataToOm(omMetadataManager,
|
writeDataToOm(omMetadataManager,
|
||||||
"key_two", "bucketOne", "sampleVol", infoGroups);
|
"key_two", "bucketOne", "sampleVol", infoGroups);
|
||||||
|
|
||||||
|
@ -201,46 +202,83 @@ public class TestContainerKeyService extends AbstractOMMetadataManagerTest {
|
||||||
@Test
|
@Test
|
||||||
public void testGetKeysForContainer() {
|
public void testGetKeysForContainer() {
|
||||||
|
|
||||||
Response response = containerKeyService.getKeysForContainer(1L, -1);
|
Response response = containerKeyService.getKeysForContainer(1L, -1, "");
|
||||||
|
|
||||||
Collection<KeyMetadata> keyMetadataList =
|
Collection<KeyMetadata> keyMetadataList =
|
||||||
(Collection<KeyMetadata>) response.getEntity();
|
(Collection<KeyMetadata>) response.getEntity();
|
||||||
assertEquals(keyMetadataList.size(), 2);
|
assertEquals(2, keyMetadataList.size());
|
||||||
|
|
||||||
Iterator<KeyMetadata> iterator = keyMetadataList.iterator();
|
Iterator<KeyMetadata> iterator = keyMetadataList.iterator();
|
||||||
|
|
||||||
KeyMetadata keyMetadata = iterator.next();
|
KeyMetadata keyMetadata = iterator.next();
|
||||||
assertEquals(keyMetadata.getKey(), "key_one");
|
assertEquals("key_one", keyMetadata.getKey());
|
||||||
assertEquals(keyMetadata.getVersions().size(), 1);
|
assertEquals(1, keyMetadata.getVersions().size());
|
||||||
assertEquals(keyMetadata.getBlockIds().size(), 1);
|
assertEquals(1, keyMetadata.getBlockIds().size());
|
||||||
Map<Long, List<KeyMetadata.ContainerBlockMetadata>> blockIds =
|
Map<Long, List<KeyMetadata.ContainerBlockMetadata>> blockIds =
|
||||||
keyMetadata.getBlockIds();
|
keyMetadata.getBlockIds();
|
||||||
assertEquals(blockIds.get(0L).iterator().next().getLocalID(), 101);
|
assertEquals(101, blockIds.get(0L).iterator().next().getLocalID());
|
||||||
|
|
||||||
keyMetadata = iterator.next();
|
keyMetadata = iterator.next();
|
||||||
assertEquals(keyMetadata.getKey(), "key_two");
|
assertEquals("key_two", keyMetadata.getKey());
|
||||||
assertEquals(keyMetadata.getVersions().size(), 2);
|
assertEquals(2, keyMetadata.getVersions().size());
|
||||||
assertTrue(keyMetadata.getVersions().contains(0L) && keyMetadata
|
assertTrue(keyMetadata.getVersions().contains(0L) && keyMetadata
|
||||||
.getVersions().contains(1L));
|
.getVersions().contains(1L));
|
||||||
assertEquals(keyMetadata.getBlockIds().size(), 2);
|
assertEquals(2, keyMetadata.getBlockIds().size());
|
||||||
blockIds = keyMetadata.getBlockIds();
|
blockIds = keyMetadata.getBlockIds();
|
||||||
assertEquals(blockIds.get(0L).iterator().next().getLocalID(), 103);
|
assertEquals(103, blockIds.get(0L).iterator().next().getLocalID());
|
||||||
assertEquals(blockIds.get(1L).iterator().next().getLocalID(), 104);
|
assertEquals(104, blockIds.get(1L).iterator().next().getLocalID());
|
||||||
|
|
||||||
response = containerKeyService.getKeysForContainer(3L, -1);
|
response = containerKeyService.getKeysForContainer(3L, -1, "");
|
||||||
keyMetadataList = (Collection<KeyMetadata>) response.getEntity();
|
keyMetadataList = (Collection<KeyMetadata>) response.getEntity();
|
||||||
assertTrue(keyMetadataList.isEmpty());
|
assertTrue(keyMetadataList.isEmpty());
|
||||||
|
|
||||||
// test if limit works as expected
|
// test if limit works as expected
|
||||||
response = containerKeyService.getKeysForContainer(1L, 1);
|
response = containerKeyService.getKeysForContainer(1L, 1, "");
|
||||||
keyMetadataList = (Collection<KeyMetadata>) response.getEntity();
|
keyMetadataList = (Collection<KeyMetadata>) response.getEntity();
|
||||||
assertEquals(keyMetadataList.size(), 1);
|
assertEquals(1, keyMetadataList.size());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testGetKeysForContainerWithPrevKey() {
|
||||||
|
// test if prev-key param works as expected
|
||||||
|
Response response = containerKeyService.getKeysForContainer(
|
||||||
|
1L, -1, "/sampleVol/bucketOne/key_one");
|
||||||
|
|
||||||
|
Collection<KeyMetadata> keyMetadataList =
|
||||||
|
(Collection<KeyMetadata>) response.getEntity();
|
||||||
|
assertEquals(1, keyMetadataList.size());
|
||||||
|
|
||||||
|
Iterator<KeyMetadata> iterator = keyMetadataList.iterator();
|
||||||
|
KeyMetadata keyMetadata = iterator.next();
|
||||||
|
|
||||||
|
assertEquals("key_two", keyMetadata.getKey());
|
||||||
|
assertEquals(2, keyMetadata.getVersions().size());
|
||||||
|
assertEquals(2, keyMetadata.getBlockIds().size());
|
||||||
|
|
||||||
|
response = containerKeyService.getKeysForContainer(
|
||||||
|
1L, -1, StringUtils.EMPTY);
|
||||||
|
keyMetadataList = (Collection<KeyMetadata>) response.getEntity();
|
||||||
|
assertEquals(2, keyMetadataList.size());
|
||||||
|
iterator = keyMetadataList.iterator();
|
||||||
|
keyMetadata = iterator.next();
|
||||||
|
assertEquals("key_one", keyMetadata.getKey());
|
||||||
|
|
||||||
|
// test for negative cases
|
||||||
|
response = containerKeyService.getKeysForContainer(
|
||||||
|
1L, -1, "/sampleVol/bucketOne/invalid_key");
|
||||||
|
keyMetadataList = (Collection<KeyMetadata>) response.getEntity();
|
||||||
|
assertEquals(0, keyMetadataList.size());
|
||||||
|
|
||||||
|
response = containerKeyService.getKeysForContainer(
|
||||||
|
5L, -1, "");
|
||||||
|
keyMetadataList = (Collection<KeyMetadata>) response.getEntity();
|
||||||
|
assertEquals(0, keyMetadataList.size());
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
public void testGetContainers() {
|
public void testGetContainers() {
|
||||||
|
|
||||||
Response response = containerKeyService.getContainers(-1);
|
Response response = containerKeyService.getContainers(-1, 0L);
|
||||||
|
|
||||||
List<ContainerMetadata> containers = new ArrayList<>(
|
List<ContainerMetadata> containers = new ArrayList<>(
|
||||||
(Collection<ContainerMetadata>) response.getEntity());
|
(Collection<ContainerMetadata>) response.getEntity());
|
||||||
|
@ -248,20 +286,55 @@ public class TestContainerKeyService extends AbstractOMMetadataManagerTest {
|
||||||
Iterator<ContainerMetadata> iterator = containers.iterator();
|
Iterator<ContainerMetadata> iterator = containers.iterator();
|
||||||
|
|
||||||
ContainerMetadata containerMetadata = iterator.next();
|
ContainerMetadata containerMetadata = iterator.next();
|
||||||
assertEquals(containerMetadata.getContainerID(), 1L);
|
assertEquals(1L, containerMetadata.getContainerID());
|
||||||
// Number of keys for CID:1 should be 3 because of two different versions
|
// Number of keys for CID:1 should be 3 because of two different versions
|
||||||
// of key_two stored in CID:1
|
// of key_two stored in CID:1
|
||||||
assertEquals(containerMetadata.getNumberOfKeys(), 3L);
|
assertEquals(3L, containerMetadata.getNumberOfKeys());
|
||||||
|
|
||||||
containerMetadata = iterator.next();
|
containerMetadata = iterator.next();
|
||||||
assertEquals(containerMetadata.getContainerID(), 2L);
|
assertEquals(2L, containerMetadata.getContainerID());
|
||||||
assertEquals(containerMetadata.getNumberOfKeys(), 2L);
|
assertEquals(2L, containerMetadata.getNumberOfKeys());
|
||||||
|
|
||||||
// test if limit works as expected
|
// test if limit works as expected
|
||||||
response = containerKeyService.getContainers(1);
|
response = containerKeyService.getContainers(1, 0L);
|
||||||
containers = new ArrayList<>(
|
containers = new ArrayList<>(
|
||||||
(Collection<ContainerMetadata>) response.getEntity());
|
(Collection<ContainerMetadata>) response.getEntity());
|
||||||
assertEquals(containers.size(), 1);
|
assertEquals(1, containers.size());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testGetContainersWithPrevKey() {
|
||||||
|
|
||||||
|
Response response = containerKeyService.getContainers(1, 1L);
|
||||||
|
|
||||||
|
List<ContainerMetadata> containers = new ArrayList<>(
|
||||||
|
(Collection<ContainerMetadata>) response.getEntity());
|
||||||
|
|
||||||
|
Iterator<ContainerMetadata> iterator = containers.iterator();
|
||||||
|
|
||||||
|
ContainerMetadata containerMetadata = iterator.next();
|
||||||
|
|
||||||
|
assertEquals(1, containers.size());
|
||||||
|
assertEquals(2L, containerMetadata.getContainerID());
|
||||||
|
|
||||||
|
response = containerKeyService.getContainers(-1, 0L);
|
||||||
|
containers = new ArrayList<>(
|
||||||
|
(Collection<ContainerMetadata>) response.getEntity());
|
||||||
|
assertEquals(2, containers.size());
|
||||||
|
iterator = containers.iterator();
|
||||||
|
containerMetadata = iterator.next();
|
||||||
|
assertEquals(1L, containerMetadata.getContainerID());
|
||||||
|
|
||||||
|
// test for negative cases
|
||||||
|
response = containerKeyService.getContainers(-1, 5L);
|
||||||
|
containers = new ArrayList<>(
|
||||||
|
(Collection<ContainerMetadata>) response.getEntity());
|
||||||
|
assertEquals(0, containers.size());
|
||||||
|
|
||||||
|
response = containerKeyService.getContainers(-1, -1L);
|
||||||
|
containers = new ArrayList<>(
|
||||||
|
(Collection<ContainerMetadata>) response.getEntity());
|
||||||
|
assertEquals(2, containers.size());
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|
|
@ -20,6 +20,7 @@ package org.apache.hadoop.ozone.recon.spi.impl;
|
||||||
|
|
||||||
import static org.apache.hadoop.ozone.recon.ReconServerConfigKeys.OZONE_RECON_DB_DIR;
|
import static org.apache.hadoop.ozone.recon.ReconServerConfigKeys.OZONE_RECON_DB_DIR;
|
||||||
import static org.junit.Assert.assertEquals;
|
import static org.junit.Assert.assertEquals;
|
||||||
|
import static org.junit.Assert.assertNull;
|
||||||
import static org.junit.Assert.assertTrue;
|
import static org.junit.Assert.assertTrue;
|
||||||
|
|
||||||
import java.io.File;
|
import java.io.File;
|
||||||
|
@ -29,6 +30,7 @@ import java.util.Map;
|
||||||
|
|
||||||
import org.apache.hadoop.hdds.conf.OzoneConfiguration;
|
import org.apache.hadoop.hdds.conf.OzoneConfiguration;
|
||||||
import org.apache.hadoop.ozone.recon.api.types.ContainerKeyPrefix;
|
import org.apache.hadoop.ozone.recon.api.types.ContainerKeyPrefix;
|
||||||
|
import org.apache.hadoop.ozone.recon.api.types.ContainerMetadata;
|
||||||
import org.apache.hadoop.ozone.recon.spi.ContainerDBServiceProvider;
|
import org.apache.hadoop.ozone.recon.spi.ContainerDBServiceProvider;
|
||||||
import org.apache.hadoop.utils.db.DBStore;
|
import org.apache.hadoop.utils.db.DBStore;
|
||||||
import org.junit.After;
|
import org.junit.After;
|
||||||
|
@ -204,6 +206,112 @@ public class TestContainerDBServiceProviderImpl {
|
||||||
assertTrue(keyPrefixMap.get(containerKeyPrefix3) == 3);
|
assertTrue(keyPrefixMap.get(containerKeyPrefix3) == 3);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testGetKeyPrefixesForContainerWithKeyPrefix() throws Exception {
|
||||||
|
long containerId = System.currentTimeMillis();
|
||||||
|
|
||||||
|
String keyPrefix1 = "V3/B1/K1";
|
||||||
|
String keyPrefix2 = "V3/B1/K2";
|
||||||
|
String keyPrefix3 = "V3/B2/K1";
|
||||||
|
|
||||||
|
ContainerKeyPrefix containerKeyPrefix1 = new
|
||||||
|
ContainerKeyPrefix(containerId, keyPrefix1, 0);
|
||||||
|
containerDbServiceProvider.storeContainerKeyMapping(containerKeyPrefix1,
|
||||||
|
1);
|
||||||
|
|
||||||
|
ContainerKeyPrefix containerKeyPrefix2 = new ContainerKeyPrefix(
|
||||||
|
containerId, keyPrefix2, 0);
|
||||||
|
containerDbServiceProvider.storeContainerKeyMapping(containerKeyPrefix2,
|
||||||
|
2);
|
||||||
|
|
||||||
|
long nextContainerId = containerId + 1000L;
|
||||||
|
ContainerKeyPrefix containerKeyPrefix3 = new ContainerKeyPrefix(
|
||||||
|
nextContainerId, keyPrefix3, 0);
|
||||||
|
containerDbServiceProvider.storeContainerKeyMapping(containerKeyPrefix3,
|
||||||
|
3);
|
||||||
|
|
||||||
|
Map<ContainerKeyPrefix, Integer> keyPrefixMap =
|
||||||
|
containerDbServiceProvider.getKeyPrefixesForContainer(containerId,
|
||||||
|
keyPrefix1);
|
||||||
|
assertEquals(1, keyPrefixMap.size());
|
||||||
|
assertEquals(2, keyPrefixMap.get(containerKeyPrefix2).longValue());
|
||||||
|
|
||||||
|
keyPrefixMap = containerDbServiceProvider.getKeyPrefixesForContainer(
|
||||||
|
nextContainerId, keyPrefix3);
|
||||||
|
assertEquals(0, keyPrefixMap.size());
|
||||||
|
|
||||||
|
// test for negative cases
|
||||||
|
keyPrefixMap = containerDbServiceProvider.getKeyPrefixesForContainer(
|
||||||
|
containerId, "V3/B1/invalid");
|
||||||
|
assertEquals(0, keyPrefixMap.size());
|
||||||
|
|
||||||
|
keyPrefixMap = containerDbServiceProvider.getKeyPrefixesForContainer(
|
||||||
|
containerId, keyPrefix3);
|
||||||
|
assertEquals(0, keyPrefixMap.size());
|
||||||
|
|
||||||
|
keyPrefixMap = containerDbServiceProvider.getKeyPrefixesForContainer(
|
||||||
|
1L, "");
|
||||||
|
assertEquals(0, keyPrefixMap.size());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testGetContainersWithPrevKey() throws Exception {
|
||||||
|
long containerId = System.currentTimeMillis();
|
||||||
|
|
||||||
|
String keyPrefix1 = "V3/B1/K1";
|
||||||
|
String keyPrefix2 = "V3/B1/K2";
|
||||||
|
String keyPrefix3 = "V3/B2/K1";
|
||||||
|
|
||||||
|
ContainerKeyPrefix containerKeyPrefix1 = new
|
||||||
|
ContainerKeyPrefix(containerId, keyPrefix1, 0);
|
||||||
|
containerDbServiceProvider.storeContainerKeyMapping(containerKeyPrefix1,
|
||||||
|
1);
|
||||||
|
|
||||||
|
ContainerKeyPrefix containerKeyPrefix2 = new ContainerKeyPrefix(
|
||||||
|
containerId, keyPrefix2, 0);
|
||||||
|
containerDbServiceProvider.storeContainerKeyMapping(containerKeyPrefix2,
|
||||||
|
2);
|
||||||
|
|
||||||
|
long nextContainerId = containerId + 1000L;
|
||||||
|
ContainerKeyPrefix containerKeyPrefix3 = new ContainerKeyPrefix(
|
||||||
|
nextContainerId, keyPrefix3, 0);
|
||||||
|
containerDbServiceProvider.storeContainerKeyMapping(containerKeyPrefix3,
|
||||||
|
3);
|
||||||
|
|
||||||
|
Map<Long, ContainerMetadata> containerMap =
|
||||||
|
containerDbServiceProvider.getContainers(-1, 0L);
|
||||||
|
assertEquals(2, containerMap.size());
|
||||||
|
|
||||||
|
assertEquals(3, containerMap.get(containerId).getNumberOfKeys());
|
||||||
|
assertEquals(3, containerMap.get(nextContainerId).getNumberOfKeys());
|
||||||
|
|
||||||
|
// test if limit works
|
||||||
|
containerMap = containerDbServiceProvider.getContainers(
|
||||||
|
1, 0L);
|
||||||
|
assertEquals(1, containerMap.size());
|
||||||
|
assertNull(containerMap.get(nextContainerId));
|
||||||
|
|
||||||
|
// test for prev key
|
||||||
|
containerMap = containerDbServiceProvider.getContainers(
|
||||||
|
-1, containerId);
|
||||||
|
assertEquals(1, containerMap.size());
|
||||||
|
// containerId must be skipped from containerMap result
|
||||||
|
assertNull(containerMap.get(containerId));
|
||||||
|
|
||||||
|
containerMap = containerDbServiceProvider.getContainers(
|
||||||
|
-1, nextContainerId);
|
||||||
|
assertEquals(0, containerMap.size());
|
||||||
|
|
||||||
|
// test for negative cases
|
||||||
|
containerMap = containerDbServiceProvider.getContainers(
|
||||||
|
-1, 1L);
|
||||||
|
assertEquals(0, containerMap.size());
|
||||||
|
|
||||||
|
containerMap = containerDbServiceProvider.getContainers(
|
||||||
|
0, containerId);
|
||||||
|
assertEquals(0, containerMap.size());
|
||||||
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
public void testDeleteContainerMapping() throws IOException {
|
public void testDeleteContainerMapping() throws IOException {
|
||||||
long containerId = System.currentTimeMillis();
|
long containerId = System.currentTimeMillis();
|
||||||
|
|
Loading…
Reference in New Issue