Merge pull request #2496 from zjeffer/tests/json
Fix JSON parsing with hexadecimal characterspull/2874/head
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()) {
|
||||
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);
|
||||
}
|
||||
return root;
|
||||
}
|
||||
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