mirror of
https://github.com/qpdf/qpdf.git
synced 2025-01-03 15:17:29 +00:00
In JSONParser::handleToken avoid creating JSON objects for dictionary keys
This commit is contained in:
parent
f2e46c20b6
commit
a39043f65e
@ -1138,8 +1138,10 @@ JSONParser::handleToken()
|
|||||||
std::string s_value;
|
std::string s_value;
|
||||||
std::shared_ptr<JSON> item;
|
std::shared_ptr<JSON> item;
|
||||||
auto tos = stack.empty() ? nullptr : stack.back();
|
auto tos = stack.empty() ? nullptr : stack.back();
|
||||||
|
auto ls = lex_state;
|
||||||
|
lex_state = ls_top;
|
||||||
|
|
||||||
switch (lex_state) {
|
switch (ls) {
|
||||||
case ls_begin_dict:
|
case ls_begin_dict:
|
||||||
item = std::make_shared<JSON>(JSON::makeDictionary());
|
item = std::make_shared<JSON>(JSON::makeDictionary());
|
||||||
break;
|
break;
|
||||||
@ -1156,7 +1158,6 @@ JSONParser::handleToken()
|
|||||||
": unexpected colon");
|
": unexpected colon");
|
||||||
}
|
}
|
||||||
parser_state = ps_dict_after_colon;
|
parser_state = ps_dict_after_colon;
|
||||||
lex_state = ls_top;
|
|
||||||
return;
|
return;
|
||||||
|
|
||||||
case ls_comma:
|
case ls_comma:
|
||||||
@ -1175,7 +1176,6 @@ JSONParser::handleToken()
|
|||||||
throw std::logic_error("JSONParser::handleToken: unexpected parser"
|
throw std::logic_error("JSONParser::handleToken: unexpected parser"
|
||||||
" state for comma");
|
" state for comma");
|
||||||
}
|
}
|
||||||
lex_state = ls_top;
|
|
||||||
return;
|
return;
|
||||||
|
|
||||||
case ls_end_array:
|
case ls_end_array:
|
||||||
@ -1195,7 +1195,6 @@ JSONParser::handleToken()
|
|||||||
if (parser_state != ps_done) {
|
if (parser_state != ps_done) {
|
||||||
stack.pop_back();
|
stack.pop_back();
|
||||||
}
|
}
|
||||||
lex_state = ls_top;
|
|
||||||
return;
|
return;
|
||||||
|
|
||||||
case ls_end_dict:
|
case ls_end_dict:
|
||||||
@ -1215,7 +1214,6 @@ JSONParser::handleToken()
|
|||||||
if (parser_state != ps_done) {
|
if (parser_state != ps_done) {
|
||||||
stack.pop_back();
|
stack.pop_back();
|
||||||
}
|
}
|
||||||
lex_state = ls_top;
|
|
||||||
return;
|
return;
|
||||||
|
|
||||||
case ls_number:
|
case ls_number:
|
||||||
@ -1243,7 +1241,15 @@ JSONParser::handleToken()
|
|||||||
throw std::logic_error("JSON string length < 2");
|
throw std::logic_error("JSON string length < 2");
|
||||||
}
|
}
|
||||||
s_value = decode_string(token, token_start);
|
s_value = decode_string(token, token_start);
|
||||||
item = std::make_shared<JSON>(JSON::makeString(s_value));
|
if (parser_state == ps_dict_begin ||
|
||||||
|
parser_state == ps_dict_after_comma) {
|
||||||
|
dict_key = s_value;
|
||||||
|
dict_key_offset = token_start;
|
||||||
|
parser_state = ps_dict_after_key;
|
||||||
|
return;
|
||||||
|
} else {
|
||||||
|
item = std::make_shared<JSON>(JSON::makeString(s_value));
|
||||||
|
}
|
||||||
break;
|
break;
|
||||||
|
|
||||||
default:
|
default:
|
||||||
@ -1260,16 +1266,10 @@ JSONParser::handleToken()
|
|||||||
switch (parser_state) {
|
switch (parser_state) {
|
||||||
case ps_dict_begin:
|
case ps_dict_begin:
|
||||||
case ps_dict_after_comma:
|
case ps_dict_after_comma:
|
||||||
if (lex_state != ls_string) {
|
QTC::TC("libtests", "JSON parse string as dict key");
|
||||||
QTC::TC("libtests", "JSON parse string as dict key");
|
throw std::runtime_error(
|
||||||
throw std::runtime_error(
|
"JSON: offset " + std::to_string(offset) +
|
||||||
"JSON: offset " + std::to_string(offset) +
|
": expect string as dictionary key");
|
||||||
": expect string as dictionary key");
|
|
||||||
}
|
|
||||||
this->dict_key = s_value;
|
|
||||||
this->dict_key_offset = item->getStart();
|
|
||||||
item = nullptr;
|
|
||||||
next_state = ps_dict_after_key;
|
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case ps_dict_after_colon:
|
case ps_dict_after_colon:
|
||||||
@ -1320,7 +1320,7 @@ JSONParser::handleToken()
|
|||||||
"JSONParser::handleToken: unexpected parser state");
|
"JSONParser::handleToken: unexpected parser state");
|
||||||
}
|
}
|
||||||
|
|
||||||
if (reactor && item.get()) {
|
if (reactor) {
|
||||||
// Calling container start method is postponed until after
|
// Calling container start method is postponed until after
|
||||||
// adding the containers to their parent containers, if any.
|
// adding the containers to their parent containers, if any.
|
||||||
// This makes it much easier to keep track of the current
|
// This makes it much easier to keep track of the current
|
||||||
@ -1333,26 +1333,25 @@ JSONParser::handleToken()
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Prepare for next token
|
// Prepare for next token
|
||||||
if (item.get()) {
|
|
||||||
if (item->isDictionary()) {
|
if (item->isDictionary()) {
|
||||||
stack.push_back(item);
|
stack.push_back(item);
|
||||||
ps_stack.push_back(next_state);
|
ps_stack.push_back(next_state);
|
||||||
next_state = ps_dict_begin;
|
next_state = ps_dict_begin;
|
||||||
} else if (item->isArray()) {
|
} else if (item->isArray()) {
|
||||||
stack.push_back(item);
|
stack.push_back(item);
|
||||||
ps_stack.push_back(next_state);
|
ps_stack.push_back(next_state);
|
||||||
next_state = ps_array_begin;
|
next_state = ps_array_begin;
|
||||||
} else if (parser_state == ps_top) {
|
} else if (parser_state == ps_top) {
|
||||||
stack.push_back(item);
|
stack.push_back(item);
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (ps_stack.size() > 500) {
|
if (ps_stack.size() > 500) {
|
||||||
throw std::runtime_error(
|
throw std::runtime_error(
|
||||||
"JSON: offset " + std::to_string(offset) +
|
"JSON: offset " + std::to_string(offset) +
|
||||||
": maximum object depth exceeded");
|
": maximum object depth exceeded");
|
||||||
}
|
}
|
||||||
parser_state = next_state;
|
parser_state = next_state;
|
||||||
lex_state = ls_top;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
std::shared_ptr<JSON>
|
std::shared_ptr<JSON>
|
||||||
|
Loading…
Reference in New Issue
Block a user