Implement $top and $skip
$top and $skip are parameters for controlling the responses of
collections, to limit their size, per the Redfish specification section
7.4.
$skip=integer "Applies to resource collections. Returns a subset of the
members in a resource collection, or an empty set of members if the
$skip value is greater than or equal to the member count. This paging
query parameter defines the number of members in the resource collection
to skip."
$top=<integer> "Applies to resource collections. Defines the number of
members to show in the response. Minimum value is 0 , though a value of
0 returns an empty set of members."
This commit implements them within the resource query.
Tested:
curl --insecure --user root:0penBmc https://localhost:18080/redfish/v1/Registries\?\$top\=1
Returns 1 value. Walking through values of 1-5 (there are 4 registries
currently) returns the appropriate sizes of collection (with 5 returning
4 entries).
curl --insecure --user root:0penBmc https://localhost:18080/redfish/v1/Registries\?\$skip\=0
Returns the collection. $skip values of 0-5 return descending number of
results.
Signed-off-by: Ed Tanous <edtanous@google.com>
Change-Id: Ied8a8f8338f119173509fb4b7ba2bd4a6c49cae8
diff --git a/redfish-core/include/utils/query_param.hpp b/redfish-core/include/utils/query_param.hpp
index b4a18a3..5c43255 100644
--- a/redfish-core/include/utils/query_param.hpp
+++ b/redfish-core/include/utils/query_param.hpp
@@ -494,9 +494,49 @@
std::shared_ptr<bmcweb::AsyncResp> finalRes;
};
+inline void processTopAndSkip(const Query& query, crow::Response& res)
+{
+ nlohmann::json::object_t* obj =
+ res.jsonValue.get_ptr<nlohmann::json::object_t*>();
+ if (obj == nullptr)
+ {
+ // Shouldn't be possible. All responses should be objects.
+ messages::internalError(res);
+ return;
+ }
+
+ BMCWEB_LOG_DEBUG << "Handling top/skip";
+ nlohmann::json::object_t::iterator members = obj->find("Members");
+ if (members == obj->end())
+ {
+ // From the Redfish specification 7.3.1
+ // ... the HTTP 400 Bad Request status code with the
+ // QueryNotSupportedOnResource message from the Base Message Registry
+ // for any supported query parameters that apply only to resource
+ // collections but are used on singular resources.
+ messages::queryNotSupportedOnResource(res);
+ return;
+ }
+
+ nlohmann::json::array_t* arr =
+ members->second.get_ptr<nlohmann::json::array_t*>();
+ if (arr == nullptr)
+ {
+ messages::internalError(res);
+ return;
+ }
+
+ // Per section 7.3.1 of the Redfish specification, $skip is run before $top
+ // Can only skip as many values as we have
+ size_t skip = std::min(arr->size(), query.skip);
+ arr->erase(arr->begin(), arr->begin() + static_cast<ssize_t>(skip));
+
+ size_t top = std::min(arr->size(), query.top);
+ arr->erase(arr->begin() + static_cast<ssize_t>(top), arr->end());
+}
+
inline void
processAllParams(crow::App& app, const Query query,
-
std::function<void(crow::Response&)>& completionHandler,
crow::Response& intermediateResponse)
{
@@ -520,6 +560,12 @@
processOnly(app, intermediateResponse, completionHandler);
return;
}
+
+ if (query.top != std::numeric_limits<size_t>::max() || query.skip != 0)
+ {
+ processTopAndSkip(query, intermediateResponse);
+ }
+
if (query.expandType != ExpandType::None)
{
BMCWEB_LOG_DEBUG << "Executing expand query";
diff --git a/redfish-core/include/utils/query_param_test.cpp b/redfish-core/include/utils/query_param_test.cpp
index e5d8de7..93013c2 100644
--- a/redfish-core/include/utils/query_param_test.cpp
+++ b/redfish-core/include/utils/query_param_test.cpp
@@ -150,6 +150,73 @@
}
}
+TEST(QueryParams, ParseParametersTop)
+{
+ auto ret = boost::urls::parse_relative_ref("/redfish/v1?$top=1");
+ ASSERT_TRUE(ret);
+
+ crow::Response res;
+
+ using redfish::query_param::parseParameters;
+ using redfish::query_param::Query;
+ std::optional<Query> query = parseParameters(ret->params(), res);
+ ASSERT_TRUE(query != std::nullopt);
+ EXPECT_EQ(query->top, 1);
+}
+
+TEST(QueryParams, ParseParametersTopOutOfRangeNegative)
+{
+ auto ret = boost::urls::parse_relative_ref("/redfish/v1?$top=-1");
+ ASSERT_TRUE(ret);
+
+ crow::Response res;
+
+ using redfish::query_param::parseParameters;
+ using redfish::query_param::Query;
+ std::optional<Query> query = parseParameters(ret->params(), res);
+ ASSERT_TRUE(query == std::nullopt);
+}
+
+TEST(QueryParams, ParseParametersTopOutOfRangePositive)
+{
+ auto ret = boost::urls::parse_relative_ref("/redfish/v1?$top=1001");
+ ASSERT_TRUE(ret);
+
+ crow::Response res;
+
+ using redfish::query_param::parseParameters;
+ using redfish::query_param::Query;
+ std::optional<Query> query = parseParameters(ret->params(), res);
+ ASSERT_TRUE(query == std::nullopt);
+}
+
+TEST(QueryParams, ParseParametersSkip)
+{
+ auto ret = boost::urls::parse_relative_ref("/redfish/v1?$skip=1");
+ ASSERT_TRUE(ret);
+
+ crow::Response res;
+
+ using redfish::query_param::parseParameters;
+ using redfish::query_param::Query;
+ std::optional<Query> query = parseParameters(ret->params(), res);
+ ASSERT_TRUE(query != std::nullopt);
+ EXPECT_EQ(query->skip, 1);
+}
+TEST(QueryParams, ParseParametersSkipOutOfRange)
+{
+ auto ret = boost::urls::parse_relative_ref(
+ "/redfish/v1?$skip=99999999999999999999");
+ ASSERT_TRUE(ret);
+
+ crow::Response res;
+
+ using redfish::query_param::parseParameters;
+ using redfish::query_param::Query;
+ std::optional<Query> query = parseParameters(ret->params(), res);
+ ASSERT_EQ(query, std::nullopt);
+}
+
TEST(QueryParams, ParseParametersUnexpectedGetsIgnored)
{
auto ret = boost::urls::parse_relative_ref("/redfish/v1?unexpected_param");