Skip to content

Commit

Permalink
Get rid of unnecessary string copies in the the keys() method
Browse files Browse the repository at this point in the history
  • Loading branch information
AndreyTokmakov authored and gittiver committed May 6, 2024
1 parent 4fbd9b4 commit 1b6a4f4
Show file tree
Hide file tree
Showing 3 changed files with 62 additions and 11 deletions.
27 changes: 16 additions & 11 deletions include/crow/query_string.h
Original file line number Diff line number Diff line change
Expand Up @@ -138,7 +138,7 @@ inline size_t qs_parse(char* qs, char* qs_kv[], size_t qs_kv_size, bool parse_ur
#endif

return i;
}
}


inline int qs_decode(char * qs)
Expand Down Expand Up @@ -295,9 +295,7 @@ namespace crow
public:
static const int MAX_KEY_VALUE_PAIRS_COUNT = 256;

query_string()
{
}
query_string() = default;

query_string(const query_string& qs):
url_(qs.url_)
Expand All @@ -319,7 +317,7 @@ namespace crow
return *this;
}

query_string& operator=(query_string&& qs)
query_string& operator=(query_string&& qs) noexcept
{
key_value_pairs_ = std::move(qs.key_value_pairs_);
char* old_data = (char*)qs.url_.c_str();
Expand All @@ -339,9 +337,10 @@ namespace crow
return;

key_value_pairs_.resize(MAX_KEY_VALUE_PAIRS_COUNT);

size_t count = qs_parse(&url_[0], &key_value_pairs_[0], MAX_KEY_VALUE_PAIRS_COUNT, url);

key_value_pairs_.resize(count);
key_value_pairs_.shrink_to_fit();
}

void clear()
Expand Down Expand Up @@ -472,13 +471,19 @@ namespace crow

std::vector<std::string> keys() const
{
std::vector<std::string> ret;
for (auto element : key_value_pairs_)
std::vector<std::string> keys;
keys.reserve(key_value_pairs_.size());

for (const char* const element : key_value_pairs_)
{
std::string str_element(element);
ret.emplace_back(str_element.substr(0, str_element.find('=')));
const char* delimiter = strchr(element, '=');
if (delimiter)
keys.emplace_back(element, delimiter);
else
keys.emplace_back(element);
}
return ret;

return keys;
}

private:
Expand Down
1 change: 1 addition & 0 deletions tests/CMakeLists.txt
Original file line number Diff line number Diff line change
Expand Up @@ -7,6 +7,7 @@ enable_testing()

set(TEST_SRCS
unittest.cpp
query_string_tests.cpp
)

add_executable(unittest ${TEST_SRCS})
Expand Down
45 changes: 45 additions & 0 deletions tests/query_string_tests.cpp
Original file line number Diff line number Diff line change
@@ -0,0 +1,45 @@

#include <iostream>
#include <vector>

#include "catch.hpp"
#include "crow/query_string.h"

namespace
{
std::string buildQueryStr(const std::vector<std::pair<std::string, std::string>>& paramList)
{
std::string paramsStr{};
for (const auto& param : paramList)
paramsStr.append(param.first).append(1, '=').append(param.second).append(1, '&');
if (!paramsStr.empty())
paramsStr.resize(paramsStr.size() - 1);
return paramsStr;
}
}

TEST_CASE( "empty query params" )
{
const crow::query_string query_params("");
const std::vector<std::string> keys = query_params.keys();

REQUIRE(keys.empty() == true);
}

TEST_CASE( "query string keys" )
{
const std::vector<std::pair<std::string, std::string>> params {
{"foo", "bar"}, {"mode", "night"}, {"page", "2"},
{"tag", "js"}, {"name", "John Smith"}, {"age", "25"},
};

const crow::query_string query_params("params?" + buildQueryStr(params));
const std::vector<std::string> keys = query_params.keys();

for (const auto& entry: params)
{
const bool exist = std::any_of(keys.cbegin(), keys.cend(), [&](const std::string& key) {
return key == entry.first;});
REQUIRE(exist == true);
}
}

0 comments on commit 1b6a4f4

Please sign in to comment.