// Protocol Buffers - Google's data interchange format // Copyright 2023 Google LLC. All rights reserved. // // Use of this source code is governed by a BSD-style // license that can be found in the LICENSE file or at // https://developers.google.com/open-source/licenses/bsd #include "upb/util/compare.h" #include #include #include #include #include #include #include "upb/base/internal/endian.h" #include "upb/wire/types.h" // Must be last. #include "upb/port/def.inc" struct UnknownField; using UnknownFields = std::vector; struct Varint { explicit Varint(uint64_t _val) : val(_val) {} uint64_t val; }; struct LongVarint { explicit LongVarint(uint64_t _val) : val(_val) {} uint64_t val; // Over-encoded. }; struct Delimited { explicit Delimited(std::string _val) : val(_val) {} std::string val; }; struct Fixed64 { explicit Fixed64(uint64_t _val) : val(_val) {} uint64_t val; }; struct Fixed32 { explicit Fixed32(uint32_t _val) : val(_val) {} uint32_t val; }; struct Group { Group(std::initializer_list _val); UnknownFields val; }; struct UnknownField { uint32_t field_number; std::variant value; }; Group::Group(std::initializer_list _val) : val(_val) {} void EncodeVarint(uint64_t val, std::string* str) { do { char byte = val & 0x7fU; val >>= 7; if (val) byte |= 0x80U; str->push_back(byte); } while (val); } std::string ToBinaryPayload(const UnknownFields& fields) { std::string ret; for (const auto& field : fields) { if (const auto* val = std::get_if(&field.value)) { EncodeVarint(field.field_number << 3 | kUpb_WireType_Varint, &ret); EncodeVarint(val->val, &ret); } else if (const auto* val = std::get_if(&field.value)) { EncodeVarint(field.field_number << 3 | kUpb_WireType_Varint, &ret); EncodeVarint(val->val, &ret); ret.back() |= 0x80; ret.push_back(0); } else if (const auto* val = std::get_if(&field.value)) { EncodeVarint(field.field_number << 3 | kUpb_WireType_Delimited, &ret); EncodeVarint(val->val.size(), &ret); ret.append(val->val); } else if (const auto* val = std::get_if(&field.value)) { EncodeVarint(field.field_number << 3 | kUpb_WireType_64Bit, &ret); uint64_t swapped = upb_BigEndian64(val->val); ret.append(reinterpret_cast(&swapped), sizeof(swapped)); } else if (const auto* val = std::get_if(&field.value)) { EncodeVarint(field.field_number << 3 | kUpb_WireType_32Bit, &ret); uint32_t swapped = upb_BigEndian32(val->val); ret.append(reinterpret_cast(&swapped), sizeof(swapped)); } else if (const auto* val = std::get_if(&field.value)) { EncodeVarint(field.field_number << 3 | kUpb_WireType_StartGroup, &ret); ret.append(ToBinaryPayload(val->val)); EncodeVarint(field.field_number << 3 | kUpb_WireType_EndGroup, &ret); } } return ret; } upb_UnknownCompareResult CompareUnknownWithMaxDepth(UnknownFields uf1, UnknownFields uf2, int max_depth) { std::string buf1 = ToBinaryPayload(uf1); std::string buf2 = ToBinaryPayload(uf2); return upb_Message_UnknownFieldsAreEqual(buf1.data(), buf1.size(), buf2.data(), buf2.size(), max_depth); } upb_UnknownCompareResult CompareUnknown(UnknownFields uf1, UnknownFields uf2) { return CompareUnknownWithMaxDepth(uf1, uf2, 64); } TEST(CompareTest, UnknownFieldsReflexive) { EXPECT_EQ(kUpb_UnknownCompareResult_Equal, CompareUnknown({}, {})); EXPECT_EQ(kUpb_UnknownCompareResult_Equal, CompareUnknown({{1, Varint(123)}, {2, Fixed32(456)}}, {{1, Varint(123)}, {2, Fixed32(456)}})); EXPECT_EQ( kUpb_UnknownCompareResult_Equal, CompareUnknown( {{1, Group({{2, Group({{3, Fixed32(456)}, {4, Fixed64(123)}})}})}}, {{1, Group({{2, Group({{3, Fixed32(456)}, {4, Fixed64(123)}})}})}})); } TEST(CompareTest, UnknownFieldsOrdering) { EXPECT_EQ(kUpb_UnknownCompareResult_Equal, CompareUnknown({{1, Varint(111)}, {2, Delimited("ABC")}, {3, Fixed32(456)}, {4, Fixed64(123)}, {5, Group({})}}, {{5, Group({})}, {4, Fixed64(123)}, {3, Fixed32(456)}, {2, Delimited("ABC")}, {1, Varint(111)}})); EXPECT_EQ(kUpb_UnknownCompareResult_NotEqual, CompareUnknown({{1, Varint(111)}, {2, Delimited("ABC")}, {3, Fixed32(456)}, {4, Fixed64(123)}, {5, Group({})}}, {{5, Group({})}, {4, Fixed64(123)}, {3, Fixed32(455)}, // Small difference. {2, Delimited("ABC")}, {1, Varint(111)}})); EXPECT_EQ(kUpb_UnknownCompareResult_Equal, CompareUnknown({{3, Fixed32(456)}, {4, Fixed64(123)}}, {{4, Fixed64(123)}, {3, Fixed32(456)}})); EXPECT_EQ( kUpb_UnknownCompareResult_Equal, CompareUnknown( {{1, Group({{2, Group({{3, Fixed32(456)}, {4, Fixed64(123)}})}})}}, {{1, Group({{2, Group({{4, Fixed64(123)}, {3, Fixed32(456)}})}})}})); } TEST(CompareTest, LongVarint) { EXPECT_EQ(kUpb_UnknownCompareResult_Equal, CompareUnknown({{1, LongVarint(123)}, {2, LongVarint(456)}}, {{1, Varint(123)}, {2, Varint(456)}})); EXPECT_EQ(kUpb_UnknownCompareResult_Equal, CompareUnknown({{2, LongVarint(456)}, {1, LongVarint(123)}}, {{1, Varint(123)}, {2, Varint(456)}})); } TEST(CompareTest, MaxDepth) { EXPECT_EQ( kUpb_UnknownCompareResult_MaxDepthExceeded, CompareUnknownWithMaxDepth( {{1, Group({{2, Group({{3, Fixed32(456)}, {4, Fixed64(123)}})}})}}, {{1, Group({{2, Group({{4, Fixed64(123)}, {3, Fixed32(456)}})}})}}, 2)); }