From 1b6a4f4301e455862c25b5dc9051c9cad0ff826c Mon Sep 17 00:00:00 2001 From: AndreyTokmakov Date: Fri, 19 Apr 2024 19:55:20 +0400 Subject: [PATCH] Get rid of unnecessary string copies in the the keys() method --- include/crow/query_string.h | 27 +++++++++++++--------- tests/CMakeLists.txt | 1 + tests/query_string_tests.cpp | 45 ++++++++++++++++++++++++++++++++++++ 3 files changed, 62 insertions(+), 11 deletions(-) create mode 100644 tests/query_string_tests.cpp diff --git a/include/crow/query_string.h b/include/crow/query_string.h index 66d380f29..3ad004ad3 100644 --- a/include/crow/query_string.h +++ b/include/crow/query_string.h @@ -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) @@ -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_) @@ -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(); @@ -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() @@ -472,13 +471,19 @@ namespace crow std::vector keys() const { - std::vector ret; - for (auto element : key_value_pairs_) + std::vector 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: diff --git a/tests/CMakeLists.txt b/tests/CMakeLists.txt index 02976f77c..c892a60ae 100644 --- a/tests/CMakeLists.txt +++ b/tests/CMakeLists.txt @@ -7,6 +7,7 @@ enable_testing() set(TEST_SRCS unittest.cpp + query_string_tests.cpp ) add_executable(unittest ${TEST_SRCS}) diff --git a/tests/query_string_tests.cpp b/tests/query_string_tests.cpp new file mode 100644 index 000000000..73cd4c7f1 --- /dev/null +++ b/tests/query_string_tests.cpp @@ -0,0 +1,45 @@ + +#include +#include + +#include "catch.hpp" +#include "crow/query_string.h" + +namespace +{ + std::string buildQueryStr(const std::vector>& 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 keys = query_params.keys(); + + REQUIRE(keys.empty() == true); +} + +TEST_CASE( "query string keys" ) +{ + const std::vector> 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 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); + } +} \ No newline at end of file