Merge pull request #2496 from zjeffer/tests/json
Fix JSON parsing with hexadecimal characters
This commit is contained in:
		
						commit
						5534fc48b1
					
				|  | @ -3,6 +3,12 @@ | |||
| #include <fmt/ostream.h> | ||||
| #include <json/json.h> | ||||
| 
 | ||||
| #include <algorithm> | ||||
| #include <codecvt> | ||||
| #include <iostream> | ||||
| #include <locale> | ||||
| #include <regex> | ||||
| 
 | ||||
| #if (FMT_VERSION >= 90000) | ||||
| 
 | ||||
| template <> | ||||
|  | @ -12,25 +18,30 @@ struct fmt::formatter<Json::Value> : ostream_formatter {}; | |||
| 
 | ||||
| namespace waybar::util { | ||||
| 
 | ||||
| struct JsonParser { | ||||
|   JsonParser() {} | ||||
| class JsonParser { | ||||
|  public: | ||||
|   JsonParser() = default; | ||||
| 
 | ||||
|   const Json::Value parse(const std::string& data) const { | ||||
|     Json::Value root(Json::objectValue); | ||||
|     if (data.empty()) { | ||||
|       return root; | ||||
|   Json::Value parse(const std::string& jsonStr) { | ||||
|     Json::Value root; | ||||
| 
 | ||||
|     // replace all occurrences of "\x" with "\u00", because JSON doesn't allow "\x" escape sequences
 | ||||
|     std::string modifiedJsonStr = replaceHexadecimalEscape(jsonStr); | ||||
| 
 | ||||
|     std::istringstream jsonStream(modifiedJsonStr); | ||||
|     std::string errs; | ||||
|     if (!Json::parseFromStream(m_readerBuilder, jsonStream, &root, &errs)) { | ||||
|       throw std::runtime_error("Error parsing JSON: " + errs); | ||||
|     } | ||||
|     std::unique_ptr<Json::CharReader> const reader(builder_.newCharReader()); | ||||
|     std::string err; | ||||
|     bool res = reader->parse(data.c_str(), data.c_str() + data.size(), &root, &err); | ||||
|     if (!res) throw std::runtime_error(err); | ||||
|     return root; | ||||
|   } | ||||
| 
 | ||||
|   ~JsonParser() = default; | ||||
| 
 | ||||
|  private: | ||||
|   Json::CharReaderBuilder builder_; | ||||
| }; | ||||
|   Json::CharReaderBuilder m_readerBuilder; | ||||
| 
 | ||||
|   static std::string replaceHexadecimalEscape(const std::string& str) { | ||||
|     static std::regex re("\\\\x"); | ||||
|     return std::regex_replace(str, re, "\\u00"); | ||||
|   } | ||||
| }; | ||||
| }  // namespace waybar::util
 | ||||
|  |  | |||
|  | @ -0,0 +1,45 @@ | |||
| #include "util/json.hpp" | ||||
| 
 | ||||
| #if __has_include(<catch2/catch_test_macros.hpp>) | ||||
| #include <catch2/catch_test_macros.hpp> | ||||
| #else | ||||
| #include <catch2/catch.hpp> | ||||
| #endif | ||||
| 
 | ||||
| TEST_CASE("Simple json", "[json]") { | ||||
|   SECTION("Parse simple json") { | ||||
|     std::string stringToTest = R"({"number": 5, "string": "test"})"; | ||||
|     waybar::util::JsonParser parser; | ||||
|     Json::Value jsonValue = parser.parse(stringToTest); | ||||
|     REQUIRE(jsonValue["number"].asInt() == 5); | ||||
|     REQUIRE(jsonValue["string"].asString() == "test"); | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| TEST_CASE("Json with unicode", "[json]") { | ||||
|   SECTION("Parse json with unicode") { | ||||
|     std::string stringToTest = R"({"test": "\xab"})"; | ||||
|     waybar::util::JsonParser parser; | ||||
|     Json::Value jsonValue = parser.parse(stringToTest); | ||||
|     // compare with "\u00ab" because "\xab" is replaced with "\u00ab" in the parser
 | ||||
|     REQUIRE(jsonValue["test"].asString() == "\u00ab"); | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| TEST_CASE("Json with emoji", "[json]") { | ||||
|   SECTION("Parse json with emoji") { | ||||
|     std::string stringToTest = R"({"test": "😊"})"; | ||||
|     waybar::util::JsonParser parser; | ||||
|     Json::Value jsonValue = parser.parse(stringToTest); | ||||
|     REQUIRE(jsonValue["test"].asString() == "😊"); | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| TEST_CASE("Json with chinese characters", "[json]") { | ||||
|   SECTION("Parse json with chinese characters") { | ||||
|     std::string stringToTest = R"({"test": "你好"})"; | ||||
|     waybar::util::JsonParser parser; | ||||
|     Json::Value jsonValue = parser.parse(stringToTest); | ||||
|     REQUIRE(jsonValue["test"].asString() == "你好"); | ||||
|   } | ||||
| } | ||||
|  | @ -8,6 +8,7 @@ test_dep = [ | |||
| ] | ||||
| test_src = files( | ||||
|     'main.cpp', | ||||
|     'JsonParser.cpp', | ||||
|     'SafeSignal.cpp', | ||||
|     'config.cpp', | ||||
|     '../src/config.cpp', | ||||
|  |  | |||
		Loading…
	
		Reference in New Issue