Skip to content

Commit

Permalink
HDDS-11205. Implement a search feature for users to locate keys pendi…
Browse files Browse the repository at this point in the history
…ng Deletion within the OM Deleted Keys Insights section (#6969)
  • Loading branch information
ArafatKhan2198 authored Oct 18, 2024
1 parent f7b428d commit 86b7aae
Show file tree
Hide file tree
Showing 8 changed files with 928 additions and 180 deletions.
Original file line number Diff line number Diff line change
Expand Up @@ -43,20 +43,20 @@ private ReconConstants() {
public static final int DISK_USAGE_TOP_RECORDS_LIMIT = 30;
public static final String DEFAULT_OPEN_KEY_INCLUDE_NON_FSO = "false";
public static final String DEFAULT_OPEN_KEY_INCLUDE_FSO = "false";
public static final String DEFAULT_START_PREFIX = "/";
public static final String DEFAULT_FETCH_COUNT = "1000";
public static final String DEFAULT_KEY_SIZE = "0";
public static final String DEFAULT_BATCH_NUMBER = "1";
public static final String RECON_QUERY_BATCH_PARAM = "batchNum";
public static final String RECON_QUERY_PREVKEY = "prevKey";
public static final String RECON_QUERY_START_PREFIX = "startPrefix";
public static final String RECON_OPEN_KEY_INCLUDE_NON_FSO = "includeNonFso";
public static final String RECON_OPEN_KEY_INCLUDE_FSO = "includeFso";
public static final String RECON_OPEN_KEY_DEFAULT_SEARCH_LIMIT = "1000";
public static final String RECON_OPEN_KEY_SEARCH_DEFAULT_PREV_KEY = "";
public static final String RECON_OM_INSIGHTS_DEFAULT_START_PREFIX = "/";
public static final String RECON_OM_INSIGHTS_DEFAULT_SEARCH_LIMIT = "1000";
public static final String RECON_OM_INSIGHTS_DEFAULT_SEARCH_PREV_KEY = "";
public static final String RECON_QUERY_FILTER = "missingIn";
public static final String PREV_CONTAINER_ID_DEFAULT_VALUE = "0";
public static final String PREV_DELETED_BLOCKS_TRANSACTION_ID_DEFAULT_VALUE =
"0";
public static final String PREV_DELETED_BLOCKS_TRANSACTION_ID_DEFAULT_VALUE = "0";
// Only include containers that are missing in OM by default
public static final String DEFAULT_FILTER_FOR_MISSING_CONTAINERS = "SCM";
public static final String RECON_QUERY_LIMIT = "limit";
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -44,7 +44,7 @@ public static Response noMatchedKeysResponse(String startPrefix) {
String jsonResponse = String.format(
"{\"message\": \"No keys matched the search prefix: '%s'.\"}",
startPrefix);
return Response.status(Response.Status.NOT_FOUND)
return Response.status(Response.Status.NO_CONTENT)
.entity(jsonResponse)
.type(MediaType.APPLICATION_JSON)
.build();
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -32,11 +32,14 @@
import java.text.ParseException;
import java.text.SimpleDateFormat;
import java.time.Instant;
import java.util.List;
import java.util.TimeZone;
import java.util.ArrayList;
import java.util.Collections;
import java.util.Date;
import java.util.LinkedHashMap;
import java.util.List;
import java.util.Map;
import java.util.Set;
import java.util.ArrayList;
import java.util.TimeZone;
import java.util.concurrent.BlockingQueue;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
Expand All @@ -54,6 +57,8 @@
import org.apache.hadoop.hdds.scm.ha.SCMNodeDetails;
import org.apache.hadoop.hdds.scm.server.SCMDatanodeHeartbeatDispatcher;
import org.apache.hadoop.hdds.utils.HddsServerUtil;
import org.apache.hadoop.hdds.utils.db.Table;
import org.apache.hadoop.hdds.utils.db.TableIterator;
import org.apache.hadoop.hdfs.web.URLConnectionFactory;
import org.apache.hadoop.io.IOUtils;

Expand Down Expand Up @@ -596,6 +601,109 @@ public static long convertToEpochMillis(String dateString, String dateFormat, Ti
}
}

public static boolean validateStartPrefix(String startPrefix) {

// Ensure startPrefix starts with '/' for non-empty values
startPrefix = startPrefix.startsWith("/") ? startPrefix : "/" + startPrefix;

// Split the path to ensure it's at least at the bucket level (volume/bucket).
String[] pathComponents = startPrefix.split("/");
if (pathComponents.length < 3 || pathComponents[2].isEmpty()) {
return false; // Invalid if not at bucket level or deeper
}

return true;
}

/**
* Retrieves keys from the specified table based on pagination and prefix filtering.
* This method handles different scenarios based on the presence of {@code startPrefix}
* and {@code prevKey}, enabling efficient key retrieval from the table.
*
* The method handles the following cases:
*
* 1. {@code prevKey} provided, {@code startPrefix} empty:
* - Seeks to {@code prevKey}, skips it, and returns subsequent records up to the limit.
*
* 2. {@code prevKey} empty, {@code startPrefix} empty:
* - Iterates from the beginning of the table, retrieving all records up to the limit.
*
* 3. {@code startPrefix} provided, {@code prevKey} empty:
* - Seeks to the first key matching {@code startPrefix} and returns all matching keys up to the limit.
*
* 4. {@code startPrefix} provided, {@code prevKey} provided:
* - Seeks to {@code prevKey}, skips it, and returns subsequent keys that match {@code startPrefix},
* up to the limit.
*
* This method also handles the following {@code limit} scenarios:
* - If {@code limit == 0} or {@code limit < -1}, no records are returned.
* - If {@code limit == -1}, all records are returned.
* - For positive {@code limit}, it retrieves records up to the specified {@code limit}.
*
* @param table The table to retrieve keys from.
* @param startPrefix The search prefix to match keys against.
* @param limit The maximum number of keys to retrieve.
* @param prevKey The key to start after for the next set of records.
* @return A map of keys and their corresponding {@code OmKeyInfo} or {@code RepeatedOmKeyInfo} objects.
* @throws IOException If there are problems accessing the table.
*/
public static <T> Map<String, T> extractKeysFromTable(
Table<String, T> table, String startPrefix, int limit, String prevKey)
throws IOException {

Map<String, T> matchedKeys = new LinkedHashMap<>();

// Null check for the table to prevent NPE during omMetaManager initialization
if (table == null) {
log.error("Table object is null. omMetaManager might still be initializing.");
return Collections.emptyMap();
}

// If limit = 0, return an empty result set
if (limit == 0 || limit < -1) {
return matchedKeys;
}

// If limit = -1, set it to Integer.MAX_VALUE to return all records
int actualLimit = (limit == -1) ? Integer.MAX_VALUE : limit;

try (TableIterator<String, ? extends Table.KeyValue<String, T>> keyIter = table.iterator()) {

// Scenario 1 & 4: prevKey is provided (whether startPrefix is empty or not)
if (!prevKey.isEmpty()) {
keyIter.seek(prevKey);
if (keyIter.hasNext()) {
keyIter.next(); // Skip the previous key record
}
} else if (!startPrefix.isEmpty()) {
// Scenario 3: startPrefix is provided but prevKey is empty, so seek to startPrefix
keyIter.seek(startPrefix);
}

// Scenario 2: Both startPrefix and prevKey are empty (iterate from the start of the table)
// No seeking needed; just start iterating from the first record in the table
// This is implicit in the following loop, as the iterator will start from the beginning

// Iterate through the keys while adhering to the limit (if the limit is not zero)
while (keyIter.hasNext() && matchedKeys.size() < actualLimit) {
Table.KeyValue<String, T> entry = keyIter.next();
String dbKey = entry.getKey();

// Scenario 3 & 4: If startPrefix is provided, ensure the key matches startPrefix
if (!startPrefix.isEmpty() && !dbKey.startsWith(startPrefix)) {
break; // If the key no longer matches the prefix, exit the loop
}

// Add the valid key-value pair to the results
matchedKeys.put(dbKey, entry.getValue());
}
} catch (IOException exception) {
log.error("Error retrieving keys from table for path: {}", startPrefix, exception);
throw exception;
}
return matchedKeys;
}

/**
* Finds all subdirectories under a parent directory in an FSO bucket. It builds
* a list of paths for these subdirectories. These sub-directories are then used
Expand Down
Loading

0 comments on commit 86b7aae

Please sign in to comment.