Skip to content

Commit b5bbdb0

Browse files
authored
Merge pull request protocolbuffers#2037 from abscondment/fix-2036-ruby-hash
Fix protocolbuffers#2036 (Ruby `hash` broken for Messages with repeated fields.)
2 parents c44ca26 + de02863 commit b5bbdb0

File tree

2 files changed

+9
-8
lines changed

2 files changed

+9
-8
lines changed

ruby/ext/google/protobuf_c/repeated_field.c

+5-7
Original file line numberDiff line numberDiff line change
@@ -447,22 +447,20 @@ VALUE RepeatedField_eq(VALUE _self, VALUE _other) {
447447
*/
448448
VALUE RepeatedField_hash(VALUE _self) {
449449
RepeatedField* self = ruby_to_RepeatedField(_self);
450-
451-
VALUE hash = LL2NUM(0);
452-
450+
st_index_t h = rb_hash_start(0);
451+
VALUE hash_sym = rb_intern("hash");
453452
upb_fieldtype_t field_type = self->field_type;
454453
VALUE field_type_class = self->field_type_class;
455454
size_t elem_size = native_slot_size(field_type);
456455
size_t off = 0;
457456
for (int i = 0; i < self->size; i++, off += elem_size) {
458457
void* mem = ((uint8_t *)self->elements) + off;
459458
VALUE elem = native_slot_get(field_type, field_type_class, mem);
460-
hash = rb_funcall(hash, rb_intern("<<"), 1, INT2NUM(2));
461-
hash = rb_funcall(hash, rb_intern("^"), 1,
462-
rb_funcall(elem, rb_intern("hash"), 0));
459+
h = rb_hash_uint(h, NUM2LONG(rb_funcall(elem, hash_sym, 0)));
463460
}
461+
h = rb_hash_end(h);
464462

465-
return hash;
463+
return INT2FIX(h);
466464
}
467465

468466
/*

ruby/tests/basic.rb

+4-1
Original file line numberDiff line numberDiff line change
@@ -183,12 +183,15 @@ def test_inspect
183183

184184
def test_hash
185185
m1 = TestMessage.new(:optional_int32 => 42)
186-
m2 = TestMessage.new(:optional_int32 => 102)
186+
m2 = TestMessage.new(:optional_int32 => 102, repeated_string: ['please', 'work', 'ok?'])
187+
m3 = TestMessage.new(:optional_int32 => 102, repeated_string: ['please', 'work', 'ok?'])
187188
assert m1.hash != 0
188189
assert m2.hash != 0
190+
assert m3.hash != 0
189191
# relying on the randomness here -- if hash function changes and we are
190192
# unlucky enough to get a collision, then change the values above.
191193
assert m1.hash != m2.hash
194+
assert_equal m2.hash, m3.hash
192195
end
193196

194197
def test_unknown_field_errors

0 commit comments

Comments
 (0)