add dv::value support to json::parse

This commit is contained in:
MihailRis 2024-09-16 22:23:22 +03:00
parent ceaa676a3a
commit 271db9a6f1
3 changed files with 139 additions and 13 deletions

View File

@ -23,6 +23,16 @@ public:
std::unique_ptr<dynamic::Map> parse();
};
class ParserDV : BasicParser {
dv::value parseList();
dv::value parseObject();
dv::value parseValue();
public:
ParserDV(std::string_view filename, std::string_view source);
dv::value parse();
};
inline void newline(
std::stringstream& ss, bool nice, uint indent, const std::string& indentstr
) {
@ -362,7 +372,11 @@ std::unique_ptr<List> Parser::parseList() {
Value Parser::parseValue() {
char next = peek();
if (next == '-' || next == '+' || is_digit(next)) {
return parseNumber();
auto numeric = parseNumber();
if (std::holds_alternative<integer_t>(numeric)) {
return std::get<integer_t>(numeric);
}
return std::get<number_t>(numeric);
}
if (is_identifier_start(next)) {
std::string literal = parseName();
@ -400,3 +414,112 @@ dynamic::Map_sptr json::parse(
dynamic::Map_sptr json::parse(std::string_view source) {
return parse("<string>", source);
}
ParserDV::ParserDV(std::string_view filename, std::string_view source)
: BasicParser(filename, source) {
}
dv::value ParserDV::parse() {
char next = peek();
if (next != '{') {
throw error("'{' expected");
}
return parseObject();
}
dv::value ParserDV::parseObject() {
expect('{');
auto object = dv::object();
while (peek() != '}') {
if (peek() == '#') {
skipLine();
continue;
}
std::string key = parseName();
char next = peek();
if (next != ':') {
throw error("':' expected");
}
pos++;
object[key] = parseValue();
next = peek();
if (next == ',') {
pos++;
} else if (next == '}') {
break;
} else {
throw error("',' expected");
}
}
pos++;
return object;
}
dv::value ParserDV::parseList() {
expect('[');
auto list = dv::list();
while (peek() != ']') {
if (peek() == '#') {
skipLine();
continue;
}
list.add(parseValue());
char next = peek();
if (next == ',') {
pos++;
} else if (next == ']') {
break;
} else {
throw error("',' expected");
}
}
pos++;
return list;
}
dv::value ParserDV::parseValue() {
char next = peek();
if (next == '-' || next == '+' || is_digit(next)) {
auto numeric = parseNumber();
if (std::holds_alternative<integer_t>(numeric)) {
return std::get<integer_t>(numeric);
}
return std::get<number_t>(numeric);
}
if (is_identifier_start(next)) {
std::string literal = parseName();
if (literal == "true") {
return true;
} else if (literal == "false") {
return false;
} else if (literal == "inf") {
return INFINITY;
} else if (literal == "nan") {
return NAN;
}
throw error("invalid literal ");
}
if (next == '{') {
return parseObject();
}
if (next == '[') {
return parseList();
}
if (next == '"' || next == '\'') {
pos++;
return parseString(next);
}
throw error("unexpected character '" + std::string({next}) + "'");
}
dv::value json::parseDV(
std::string_view filename, std::string_view source
) {
ParserDV parser(filename, source);
return parser.parse();
}
dv::value json::parseDV(std::string_view source) {
return parseDV("<string>", source);
}

View File

@ -11,6 +11,9 @@ namespace json {
dynamic::Map_sptr parse(std::string_view filename, std::string_view source);
dynamic::Map_sptr parse(std::string_view source);
dv::value parseDV(std::string_view filename, std::string_view source);
dv::value parseDV(std::string_view source);
std::string stringify(
const dynamic::Map* obj, bool nice, const std::string& indent
);

View File

@ -48,22 +48,22 @@ TEST(JSON, EncodeDecodeDV) {
srcBytes[i] = rand();
}
std::string text;
std::string text;
{
auto map = dv::object();
map["name"] = name;
map["year"] = year;
map["score"] = score;
map["data"] = srcBytes;
auto object = dv::object();
object["name"] = name;
object["year"] = year;
object["score"] = score;
object["data"] = srcBytes;
text = json::stringifyDV(map, false, "");
text = json::stringifyDV(object, false, "");
}
{
auto map = json::parse(text);
EXPECT_EQ(map->get<std::string>("name"), name);
EXPECT_EQ(map->get<integer_t>("year"), year);
EXPECT_FLOAT_EQ(map->get<number_t>("score"), score);
auto b64string = map->get<std::string>("data");
auto object = json::parseDV(text);
EXPECT_EQ(object["name"].asString(), name);
EXPECT_EQ(object["year"].asInteger(), year);
EXPECT_FLOAT_EQ(object["score"].asNumber(), score);
auto b64string = object["data"].asString();
auto bytes = util::base64_decode(b64string);
EXPECT_EQ(bytes.size(), bytesSize);