Commit 0b5111db authored by bmoyles0117's avatar bmoyles0117 Committed by Paul Yang

Raise error for JSON overflow encoding in Ruby (#5752) (#5861)

* add check for overflow

* de-nestify

* break long lines
parent 582743bf
...@@ -118,11 +118,11 @@ file "tests/test_ruby_package_proto2.rb" => "tests/test_ruby_package_proto2.prot ...@@ -118,11 +118,11 @@ file "tests/test_ruby_package_proto2.rb" => "tests/test_ruby_package_proto2.prot
end end
file "tests/basic_test.rb" => "tests/basic_test.proto" do |file_task| file "tests/basic_test.rb" => "tests/basic_test.proto" do |file_task|
sh "../src/protoc --ruby_out=. tests/basic_test.proto" sh "../src/protoc -I../src -I. --ruby_out=. tests/basic_test.proto"
end end
file "tests/basic_test_proto2.rb" => "tests/basic_test_proto2.proto" do |file_task| file "tests/basic_test_proto2.rb" => "tests/basic_test_proto2.proto" do |file_task|
sh "../src/protoc --ruby_out=. tests/basic_test_proto2.proto" sh "../src/protoc -I../src -I. --ruby_out=. tests/basic_test_proto2.proto"
end end
task :genproto => genproto_output task :genproto => genproto_output
......
...@@ -1061,6 +1061,11 @@ static void put_ruby_value(VALUE value, ...@@ -1061,6 +1061,11 @@ static void put_ruby_value(VALUE value,
upb_sink *sink, upb_sink *sink,
bool emit_defaults, bool emit_defaults,
bool is_json) { bool is_json) {
if (depth > ENCODE_MAX_NESTING) {
rb_raise(rb_eRuntimeError,
"Maximum recursion depth exceeded during encoding.");
}
upb_selector_t sel = 0; upb_selector_t sel = 0;
if (upb_fielddef_isprimitive(f)) { if (upb_fielddef_isprimitive(f)) {
sel = getsel(f, upb_handlers_getprimitivehandlertype(f)); sel = getsel(f, upb_handlers_getprimitivehandlertype(f));
......
...@@ -2,6 +2,8 @@ syntax = "proto3"; ...@@ -2,6 +2,8 @@ syntax = "proto3";
package basic_test; package basic_test;
import "google/protobuf/struct.proto";
message Foo { message Foo {
Bar bar = 1; Bar bar = 1;
repeated Baz baz = 2; repeated Baz baz = 2;
...@@ -107,3 +109,13 @@ message Outer { ...@@ -107,3 +109,13 @@ message Outer {
message Inner { message Inner {
} }
message MyRepeatedStruct {
repeated MyStruct structs = 1;
}
message MyStruct {
string string = 1;
google.protobuf.Struct struct = 2;
}
...@@ -2,6 +2,8 @@ syntax = "proto2"; ...@@ -2,6 +2,8 @@ syntax = "proto2";
package basic_test_proto2; package basic_test_proto2;
import "google/protobuf/struct.proto";
message Foo { message Foo {
optional Bar bar = 1; optional Bar bar = 1;
repeated Baz baz = 2; repeated Baz baz = 2;
...@@ -115,3 +117,12 @@ message OneofMessage { ...@@ -115,3 +117,12 @@ message OneofMessage {
TestEnum d = 4; TestEnum d = 4;
} }
} }
message MyRepeatedStruct {
repeated MyStruct structs = 1;
}
message MyStruct {
optional string string = 1;
optional google.protobuf.Struct struct = 2;
}
...@@ -1114,6 +1114,67 @@ module CommonTests ...@@ -1114,6 +1114,67 @@ module CommonTests
assert JSON.parse(actual, :symbolize_names => true) == expected assert JSON.parse(actual, :symbolize_names => true) == expected
end end
def value_from_ruby(value)
ret = Google::Protobuf::Value.new
case value
when String
ret.string_value = value
when Google::Protobuf::Struct
ret.struct_value = value
when Hash
ret.struct_value = struct_from_ruby(value)
when Google::Protobuf::ListValue
ret.list_value = value
when Array
ret.list_value = list_from_ruby(value)
else
@log.error "Unknown type: #{value.class}"
raise Google::Protobuf::Error, "Unknown type: #{value.class}"
end
ret
end
def list_from_ruby(arr)
ret = Google::Protobuf::ListValue.new
arr.each do |v|
ret.values << value_from_ruby(v)
end
ret
end
def struct_from_ruby(hash)
ret = Google::Protobuf::Struct.new
hash.each do |k, v|
ret.fields[k] ||= value_from_ruby(v)
end
ret
end
def test_deep_json
# will not overflow
json = '{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":{"a":'\
'{"a":{"a":{"a":{"a":{}}}}}}}}}}}}}}}}'
struct = struct_from_ruby(JSON.parse(json))
assert_equal json, struct.to_json
encoded = proto_module::MyRepeatedStruct.encode(
proto_module::MyRepeatedStruct.new(structs: [proto_module::MyStruct.new(struct: struct)]))
assert_equal json, proto_module::MyRepeatedStruct.decode(encoded).structs[0].struct.to_json
# will overflow
json = '{"a":{"a":{"a":[{"a":{"a":[{"a":[{"a":{"a":[{"a":[{"a":'\
'{"a":[{"a":[{"a":{"a":{"a":[{"a":"a"}]}}}]}]}}]}]}}]}]}}]}}}'
struct = struct_from_ruby(JSON.parse(json))
assert_equal json, struct.to_json
assert_raise(RuntimeError, "Maximum recursion depth exceeded during encoding") do
proto_module::MyRepeatedStruct.encode(
proto_module::MyRepeatedStruct.new(structs: [proto_module::MyStruct.new(struct: struct)]))
end
end
def test_comparison_with_arbitrary_object def test_comparison_with_arbitrary_object
assert proto_module::TestMessage.new != nil assert proto_module::TestMessage.new != nil
end end
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment