blob: 390e1ed77e6f840fbc3a8834df28c08523e47167 [file] [log] [blame]
// Copyright 2021 The Pigweed Authors
//
// Licensed under the Apache License, Version 2.0 (the "License"); you may not
// use this file except in compliance with the License. You may obtain a copy of
// the License at
//
// https://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
// WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
// License for the specific language governing permissions and limitations under
// the License.
#include "pw_protobuf/message.h"
#include "gtest/gtest.h"
#include "pw_stream/memory_stream.h"
#define ASSERT_OK(status) ASSERT_EQ(OkStatus(), status)
namespace pw::protobuf {
TEST(ProtoHelper, IterateMessage) {
// clang-format off
constexpr uint8_t encoded_proto[] = {
// type=uint32, k=1, v=1
0x08, 0x01,
// type=uint32, k=2, v=2
0x10, 0x02,
// type=uint32, k=3, v=3
0x18, 0x03,
};
// clang-format on
stream::MemoryReader reader(std::as_bytes(std::span(encoded_proto)));
Message parser = Message(reader, sizeof(encoded_proto));
uint32_t count = 0;
for (Message::Field field : parser) {
++count;
EXPECT_EQ(field.field_number(), count);
Uint32 value = field.As<Uint32>();
ASSERT_OK(value.status());
EXPECT_EQ(value.value(), count);
}
EXPECT_EQ(count, static_cast<uint32_t>(3));
}
TEST(ProtoHelper, MessageIterator) {
// clang-format off
std::uint8_t encoded_proto[] = {
// key = 1, str = "foo 1"
0x0a, 0x05, 'f', 'o', 'o', ' ', '1',
// type=uint32, k=2, v=2
0x10, 0x02,
};
// clang-format on
stream::MemoryReader reader(std::as_bytes(std::span(encoded_proto)));
Message parser = Message(reader, sizeof(encoded_proto));
Message::iterator iter = parser.begin();
Message::iterator first = iter++;
ASSERT_EQ(first, first);
ASSERT_EQ(first->field_number(), static_cast<uint32_t>(1));
String str = first->As<String>();
ASSERT_OK(str.status());
Result<bool> cmp = str.Equal("foo 1");
ASSERT_OK(cmp.status());
ASSERT_TRUE(cmp.value());
Message::iterator second = iter++;
ASSERT_EQ(second, second);
ASSERT_EQ(second->field_number(), static_cast<uint32_t>(2));
Uint32 uint32_val = second->As<Uint32>();
ASSERT_OK(uint32_val.status());
ASSERT_EQ(uint32_val.value(), static_cast<uint32_t>(2));
ASSERT_NE(first, second);
ASSERT_NE(first, iter);
ASSERT_NE(second, iter);
ASSERT_EQ(iter, parser.end());
}
TEST(ProtoHelper, AsProtoInteger) {
// clang-format off
std::uint8_t encoded_proto[] = {
// type: int32, k = 1, val = -123
0x08, 0x85, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0x01,
// type: uint32, k = 2, val = 123
0x10, 0x7b,
// type: sint32, k = 3, val = -456
0x18, 0x8f, 0x07,
// type: fixed32, k = 4, val = 268435457
0x25, 0x01, 0x00, 0x00, 0x10,
// type: sfixed32, k = 5, val = -268435457
0x2d, 0xff, 0xff, 0xff, 0xef,
// type: int64, k = 6, val = -1099511627776
0x30, 0x80, 0x80, 0x80, 0x80, 0x80, 0xe0, 0xff, 0xff, 0xff, 0x01,
// type: uint64, k = 7, val = 1099511627776
0x38, 0x80, 0x80, 0x80, 0x80, 0x80, 0x20,
// type: sint64, k = 8, val = -2199023255552
0x40, 0xff, 0xff, 0xff, 0xff, 0xff, 0x7f,
// type: fixed64, k = 9, val = 72057594037927937
0x49, 0x01, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x01,
// type: sfixed64, k = 10, val = -72057594037927937
0x51, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xfe,
};
// clang-format on
stream::MemoryReader reader(std::as_bytes(std::span(encoded_proto)));
Message parser = Message(reader, sizeof(encoded_proto));
{
Int32 value = parser.AsInt32(1);
ASSERT_OK(value.status());
ASSERT_EQ(value.value(), static_cast<int32_t>(-123));
}
{
Uint32 value = parser.AsUint32(2);
ASSERT_OK(value.status());
ASSERT_EQ(value.value(), static_cast<uint32_t>(123));
}
{
Sint32 value = parser.AsSint32(3);
ASSERT_OK(value.status());
ASSERT_EQ(value.value(), static_cast<int32_t>(-456));
}
{
Fixed32 value = parser.AsFixed32(4);
ASSERT_OK(value.status());
ASSERT_EQ(value.value(), static_cast<uint32_t>(268435457));
}
{
Sfixed32 value = parser.AsSfixed32(5);
ASSERT_OK(value.status());
ASSERT_EQ(value.value(), static_cast<int32_t>(-268435457));
}
{
Int64 value = parser.AsInt64(6);
ASSERT_OK(value.status());
ASSERT_EQ(value.value(), static_cast<int64_t>(-1099511627776));
}
{
Uint64 value = parser.AsUint64(7);
ASSERT_OK(value.status());
ASSERT_EQ(value.value(), static_cast<uint64_t>(1099511627776));
}
{
Sint64 value = parser.AsSint64(8);
ASSERT_OK(value.status());
ASSERT_EQ(value.value(), static_cast<int64_t>(-2199023255552));
}
{
Fixed64 value = parser.AsFixed64(9);
ASSERT_OK(value.status());
ASSERT_EQ(value.value(), static_cast<uint64_t>(72057594037927937));
}
{
Sfixed64 value = parser.AsSfixed64(10);
ASSERT_OK(value.status());
ASSERT_EQ(value.value(), static_cast<int64_t>(-72057594037927937));
}
}
TEST(ProtoHelper, AsString) {
// message {
// string str = 1;
// }
// clang-format off
std::uint8_t encoded_proto[] = {
// `str`, k = 1, "string"
0x0a, 0x06, 's', 't', 'r', 'i', 'n', 'g',
};
// clang-format on
stream::MemoryReader reader(std::as_bytes(std::span(encoded_proto)));
Message parser = Message(reader, sizeof(encoded_proto));
constexpr uint32_t kFieldNumber = 1;
String value = parser.AsString(kFieldNumber);
ASSERT_OK(value.status());
Result<bool> cmp = value.Equal("string");
ASSERT_OK(cmp.status());
ASSERT_TRUE(cmp.value());
cmp = value.Equal("other");
ASSERT_OK(cmp.status());
ASSERT_FALSE(cmp.value());
// The string is a prefix of the target string to compare.
cmp = value.Equal("string and more");
ASSERT_OK(cmp.status());
ASSERT_FALSE(cmp.value());
// The target string to compare is a sub prefix of this string
cmp = value.Equal("str");
ASSERT_OK(cmp.status());
ASSERT_FALSE(cmp.value());
}
TEST(ProtoHelper, AsRepeatedStrings) {
// Repeated field of string i.e.
//
// message RepeatedString {
// repeated string msg_a = 1;
// repeated string msg_b = 2;
// }
// clang-format off
std::uint8_t encoded_proto[] = {
// key = 1, str = "foo 1"
0x0a, 0x05, 'f', 'o', 'o', ' ', '1',
// key = 2, str = "foo 2"
0x12, 0x05, 'f', 'o', 'o', ' ', '2',
// key = 1, str = "bar 1"
0x0a, 0x05, 'b', 'a', 'r', ' ', '1',
// key = 2, str = "bar 2"
0x12, 0x05, 'b', 'a', 'r', ' ', '2',
};
// clang-format on
constexpr uint32_t kMsgAFieldNumber = 1;
constexpr uint32_t kMsgBFieldNumber = 2;
constexpr uint32_t kNonExistFieldNumber = 3;
stream::MemoryReader reader(std::as_bytes(std::span(encoded_proto)));
Message parser = Message(reader, sizeof(encoded_proto));
// Field 'msg_a'
{
RepeatedStrings msg = parser.AsRepeatedStrings(kMsgAFieldNumber);
std::string_view expected[] = {
"foo 1",
"bar 1",
};
size_t count = 0;
for (String ele : msg) {
ASSERT_OK(ele.status());
Result<bool> res = ele.Equal(expected[count++]);
ASSERT_OK(res.status());
ASSERT_TRUE(res.value());
}
ASSERT_EQ(count, static_cast<size_t>(2));
}
// Field `msg_b`
{
RepeatedStrings msg = parser.AsRepeatedStrings(kMsgBFieldNumber);
std::string_view expected[] = {
"foo 2",
"bar 2",
};
size_t count = 0;
for (String ele : msg) {
ASSERT_OK(ele.status());
Result<bool> res = ele.Equal(expected[count++]);
ASSERT_OK(res.status());
ASSERT_TRUE(res.value());
}
ASSERT_EQ(count, static_cast<size_t>(2));
}
// non-existing field
{
RepeatedStrings msg = parser.AsRepeatedStrings(kNonExistFieldNumber);
size_t count = 0;
for ([[maybe_unused]] String ele : msg) {
count++;
}
ASSERT_EQ(count, static_cast<size_t>(0));
}
}
TEST(ProtoHelper, RepeatedFieldIterator) {
// Repeated field of string i.e.
//
// message RepeatedString {
// repeated string msg = 1;
// }
// clang-format off
std::uint8_t encoded_proto[] = {
// key = 1, str = "foo 1"
0x0a, 0x05, 'f', 'o', 'o', ' ', '1',
// key = 1, str = "bar 1"
0x0a, 0x05, 'b', 'a', 'r', ' ', '1',
};
// clang-format on
constexpr uint32_t kFieldNumber = 1;
stream::MemoryReader reader(std::as_bytes(std::span(encoded_proto)));
Message parser = Message(reader, sizeof(encoded_proto));
RepeatedStrings repeated_str = parser.AsRepeatedStrings(kFieldNumber);
RepeatedStrings::iterator iter = repeated_str.begin();
RepeatedStrings::iterator first = iter++;
ASSERT_EQ(first, first);
Result<bool> cmp = first->Equal("foo 1");
ASSERT_OK(cmp.status());
ASSERT_TRUE(cmp.value());
RepeatedStrings::iterator second = iter++;
ASSERT_EQ(second, second);
cmp = second->Equal("bar 1");
ASSERT_OK(cmp.status());
ASSERT_TRUE(cmp.value());
ASSERT_NE(first, second);
ASSERT_NE(first, iter);
ASSERT_NE(second, iter);
ASSERT_EQ(iter, repeated_str.end());
}
TEST(ProtoHelper, AsMessage) {
// A nested message:
//
// message Contact {
// string number = 1;
// string email = 2;
// }
//
// message Person {
// Contact info = 2;
// }
// clang-format off
std::uint8_t encoded_proto[] = {
// Person.info.number = "123456", .email = "foo@email.com"
0x12, 0x17,
0x0a, 0x06, '1', '2', '3', '4', '5', '6',
0x12, 0x0d, 'f', 'o', 'o', '@', 'e', 'm', 'a', 'i', 'l', '.', 'c', 'o', 'm',
};
// clang-format on
constexpr uint32_t kInfoFieldNumber = 2;
constexpr uint32_t kNumberFieldNumber = 1;
constexpr uint32_t kEmailFieldNumber = 2;
stream::MemoryReader reader(std::as_bytes(std::span(encoded_proto)));
Message parser = Message(reader, sizeof(encoded_proto));
Message info = parser.AsMessage(kInfoFieldNumber);
ASSERT_OK(info.status());
String number = info.AsString(kNumberFieldNumber);
ASSERT_OK(number.status());
Result<bool> cmp = number.Equal("123456");
ASSERT_OK(cmp.status());
ASSERT_TRUE(cmp.value());
String email = info.AsString(kEmailFieldNumber);
ASSERT_OK(email.status());
cmp = email.Equal("foo@email.com");
ASSERT_OK(cmp.status());
ASSERT_TRUE(cmp.value());
}
TEST(ProtoHelper, AsRepeatedMessages) {
// message Contact {
// string number = 1;
// string email = 2;
// }
//
// message Person {
// repeated Contact info = 1;
// }
// clang-format off
std::uint8_t encoded_proto[] = {
// Person.Contact.number = "12345", .email = "foo@email.com"
0x0a, 0x16,
0x0a, 0x05, '1', '2', '3', '4', '5',
0x12, 0x0d, 'f', 'o', 'o', '@', 'e', 'm', 'a', 'i', 'l', '.', 'c', 'o', 'm',
// Person.Contact.number = "67890", .email = "bar@email.com"
0x0a, 0x16,
0x0a, 0x05, '6', '7', '8', '9', '0',
0x12, 0x0d, 'b', 'a', 'r', '@', 'e', 'm', 'a', 'i', 'l', '.', 'c', 'o', 'm',
};
// clang-format on
constexpr uint32_t kInfoFieldNumber = 1;
constexpr uint32_t kNumberFieldNumber = 1;
constexpr uint32_t kEmailFieldNumber = 2;
stream::MemoryReader reader(std::as_bytes(std::span(encoded_proto)));
Message parser = Message(reader, sizeof(encoded_proto));
RepeatedMessages messages = parser.AsRepeatedMessages(kInfoFieldNumber);
ASSERT_OK(messages.status());
struct {
std::string_view number;
std::string_view email;
} expected[] = {
{"12345", "foo@email.com"},
{"67890", "bar@email.com"},
};
size_t count = 0;
for (Message message : messages) {
String number = message.AsString(kNumberFieldNumber);
ASSERT_OK(number.status());
Result<bool> cmp = number.Equal(expected[count].number);
ASSERT_OK(cmp.status());
ASSERT_TRUE(cmp.value());
String email = message.AsString(kEmailFieldNumber);
ASSERT_OK(email.status());
cmp = email.Equal(expected[count].email);
ASSERT_OK(cmp.status());
ASSERT_TRUE(cmp.value());
count++;
}
ASSERT_EQ(count, static_cast<size_t>(2));
}
TEST(ProtoHelper, AsStringToBytesMap) {
// message Maps {
// map<string, string> map_a = 1;
// map<string, string> map_b = 2;
// }
// clang-format off
std::uint8_t encoded_proto[] = {
// map_a["key_bar"] = "bar_a", key = 1
0x0a, 0x10,
0x0a, 0x07, 'k', 'e', 'y', '_', 'b', 'a', 'r', // map key
0x12, 0x05, 'b', 'a', 'r', '_', 'a', // map value
// map_a["key_foo"] = "foo_a", key = 1
0x0a, 0x10,
0x0a, 0x07, 'k', 'e', 'y', '_', 'f', 'o', 'o',
0x12, 0x05, 'f', 'o', 'o', '_', 'a',
// map_b["key_foo"] = "foo_b", key = 2
0x12, 0x10,
0x0a, 0x07, 'k', 'e', 'y', '_', 'f', 'o', 'o',
0x12, 0x05, 'f', 'o', 'o', 0x5f, 0x62,
// map_b["key_bar"] = "bar_b", key = 2
0x12, 0x10,
0x0a, 0x07, 'k', 'e', 'y', '_', 'b', 'a', 'r',
0x12, 0x05, 'b', 'a', 'r', 0x5f, 0x62,
};
// clang-format on
stream::MemoryReader reader(std::as_bytes(std::span(encoded_proto)));
Message parser = Message(reader, sizeof(encoded_proto));
{
// Parse field 'map_a'
constexpr uint32_t kFieldNumber = 1;
StringMapParser<String> string_map =
parser.AsStringToStringMap(kFieldNumber);
String value = string_map["key_foo"];
ASSERT_OK(value.status());
Result<bool> cmp = value.Equal("foo_a");
ASSERT_OK(cmp.status());
ASSERT_TRUE(cmp.value());
value = string_map["key_bar"];
ASSERT_OK(value.status());
cmp = value.Equal("bar_a");
ASSERT_OK(cmp.status());
ASSERT_TRUE(cmp.value());
// Non-existing key
value = string_map["non-existing"];
ASSERT_EQ(value.status(), Status::NotFound());
}
{
// Parse field 'map_b'
constexpr uint32_t kFieldNumber = 2;
StringMapParser<String> string_map =
parser.AsStringToStringMap(kFieldNumber);
String value = string_map["key_foo"];
ASSERT_OK(value.status());
Result<bool> cmp = value.Equal("foo_b");
ASSERT_OK(cmp.status());
ASSERT_TRUE(cmp.value());
value = string_map["key_bar"];
ASSERT_OK(value.status());
cmp = value.Equal("bar_b");
ASSERT_OK(cmp.status());
ASSERT_TRUE(cmp.value());
// Non-existing key
value = string_map["non-existing"];
ASSERT_EQ(value.status(), Status::NotFound());
}
}
TEST(ProtoHelper, AsStringToMessageMap) {
// message Contact {
// string number = 1;
// string email = 2;
// }
//
// message Contacts {
// map<string, Contact> staffs = 1;
// }
// clang-format off
std::uint8_t encoded_proto[] = {
// staffs['bar'] = {.number = '456, .email = "bar@email.com"}
0x0a, 0x1b,
0x0a, 0x03, 0x62, 0x61, 0x72,
0x12, 0x14, 0x0a, 0x03, 0x34, 0x35, 0x36, 0x12, 0x0d, 0x62, 0x61, 0x72, 0x40, 0x65, 0x6d, 0x61, 0x69, 0x6c, 0x2e, 0x63, 0x6f, 0x6d,
// staffs['foo'] = {.number = '123', .email = "foo@email.com"}
0x0a, 0x1b,
0x0a, 0x03, 0x66, 0x6f, 0x6f,
0x12, 0x14, 0x0a, 0x03, 0x31, 0x32, 0x33, 0x12, 0x0d, 0x66, 0x6f, 0x6f, 0x40, 0x65, 0x6d, 0x61, 0x69, 0x6c, 0x2e, 0x63, 0x6f, 0x6d,
};
// clang-format on
constexpr uint32_t kStaffsFieldId = 1;
constexpr uint32_t kNumberFieldId = 1;
constexpr uint32_t kEmailFieldId = 2;
stream::MemoryReader reader(std::as_bytes(std::span(encoded_proto)));
Message parser = Message(reader, sizeof(encoded_proto));
StringMapParser<Message> staffs = parser.AsStringToMessageMap(kStaffsFieldId);
ASSERT_OK(staffs.status());
Message foo_staff = staffs["foo"];
ASSERT_OK(foo_staff.status());
String foo_number = foo_staff.AsString(kNumberFieldId);
ASSERT_OK(foo_number.status());
Result<bool> foo_number_cmp = foo_number.Equal("123");
ASSERT_OK(foo_number_cmp.status());
ASSERT_TRUE(foo_number_cmp.value());
String foo_email = foo_staff.AsString(kEmailFieldId);
ASSERT_OK(foo_email.status());
Result<bool> foo_email_cmp = foo_email.Equal("foo@email.com");
ASSERT_OK(foo_email_cmp.status());
ASSERT_TRUE(foo_email_cmp.value());
Message bar_staff = staffs["bar"];
ASSERT_OK(bar_staff.status());
String bar_number = bar_staff.AsString(kNumberFieldId);
ASSERT_OK(bar_number.status());
Result<bool> bar_number_cmp = bar_number.Equal("456");
ASSERT_OK(bar_number_cmp.status());
ASSERT_TRUE(bar_number_cmp.value());
String bar_email = bar_staff.AsString(kEmailFieldId);
ASSERT_OK(bar_email.status());
Result<bool> bar_email_cmp = bar_email.Equal("bar@email.com");
ASSERT_OK(bar_email_cmp.status());
ASSERT_TRUE(bar_email_cmp.value());
}
} // namespace pw::protobuf