Protocol Buffers - Google's data interchange format (grpc依赖)
https://developers.google.com/protocol-buffers/
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
197 lines
7.6 KiB
197 lines
7.6 KiB
// Protocol Buffers - Google's data interchange format |
|
// Copyright 2023 Google LLC. All rights reserved. |
|
// https://developers.google.com/protocol-buffers/ |
|
// |
|
// Redistribution and use in source and binary forms, with or without |
|
// modification, are permitted provided that the following conditions are |
|
// met: |
|
// |
|
// * Redistributions of source code must retain the above copyright |
|
// notice, this list of conditions and the following disclaimer. |
|
// * Redistributions in binary form must reproduce the above |
|
// copyright notice, this list of conditions and the following disclaimer |
|
// in the documentation and/or other materials provided with the |
|
// distribution. |
|
// * Neither the name of Google LLC nor the names of its |
|
// contributors may be used to endorse or promote products derived from |
|
// this software without specific prior written permission. |
|
// |
|
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS |
|
// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT |
|
// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR |
|
// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT |
|
// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, |
|
// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT |
|
// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, |
|
// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY |
|
// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT |
|
// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE |
|
// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. |
|
|
|
#include "upb/util/compare.h" |
|
|
|
#include <stdint.h> |
|
|
|
#include <initializer_list> |
|
#include <string> |
|
#include <variant> |
|
#include <vector> |
|
|
|
#include "gtest/gtest.h" |
|
#include "upb/wire/internal/swap.h" |
|
#include "upb/wire/types.h" |
|
|
|
struct UnknownField; |
|
|
|
using UnknownFields = std::vector<UnknownField>; |
|
|
|
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<UnknownField> _val); |
|
UnknownFields val; |
|
}; |
|
|
|
struct UnknownField { |
|
uint32_t field_number; |
|
std::variant<Varint, LongVarint, Delimited, Fixed64, Fixed32, Group> value; |
|
}; |
|
|
|
Group::Group(std::initializer_list<UnknownField> _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<Varint>(&field.value)) { |
|
EncodeVarint(field.field_number << 3 | kUpb_WireType_Varint, &ret); |
|
EncodeVarint(val->val, &ret); |
|
} else if (const auto* val = std::get_if<LongVarint>(&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<Delimited>(&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<Fixed64>(&field.value)) { |
|
EncodeVarint(field.field_number << 3 | kUpb_WireType_64Bit, &ret); |
|
uint64_t swapped = _upb_BigEndian_Swap64(val->val); |
|
ret.append(reinterpret_cast<const char*>(&swapped), sizeof(swapped)); |
|
} else if (const auto* val = std::get_if<Fixed32>(&field.value)) { |
|
EncodeVarint(field.field_number << 3 | kUpb_WireType_32Bit, &ret); |
|
uint32_t swapped = _upb_BigEndian_Swap32(val->val); |
|
ret.append(reinterpret_cast<const char*>(&swapped), sizeof(swapped)); |
|
} else if (const auto* val = std::get_if<Group>(&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)); |
|
}
|
|
|