/*
* Copyright (C) 2017 The Android Open Source Project
*
* 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
*
* http://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 <limits>
#include <memory>
#include <vector>
#include "gtest/gtest.h"
#include "perfetto/protozero/message_handle.h"
#include "src/protozero/test/fake_scattered_buffer.h"
// Autogenerated headers in out/*/gen/
#include "src/protozero/test/example_proto/library.pbzero.h"
#include "src/protozero/test/example_proto/test_messages.pb.h"
#include "src/protozero/test/example_proto/test_messages.pbzero.h"
namespace pbtest = foo::bar::pbzero; // Generated by the protozero plugin.
namespace pbgold = foo::bar; // Generated by the official protobuf compiler.
namespace protozero {
namespace {
constexpr size_t kChunkSize = 42;
class ProtoZeroConformanceTest : public ::testing::Test {
public:
void SetUp() override {
buffer_.reset(new FakeScatteredBuffer(kChunkSize));
stream_writer_.reset(new ScatteredStreamWriter(buffer_.get()));
}
void TearDown() override {
root_messages_.clear();
stream_writer_.reset();
buffer_.reset();
}
protected:
template <class T>
T* CreateMessage() {
T* message = new T();
root_messages_.push_back(std::unique_ptr<T>(message));
message->Reset(stream_writer_.get());
return message;
}
size_t GetNumSerializedBytes() {
return buffer_->chunks().size() * kChunkSize -
stream_writer_->bytes_available();
}
void GetSerializedBytes(size_t start, size_t length, uint8_t* buffer) {
return buffer_->GetBytes(start, length, buffer);
}
private:
std::unique_ptr<FakeScatteredBuffer> buffer_;
std::unique_ptr<ScatteredStreamWriter> stream_writer_;
std::vector<std::unique_ptr<Message>> root_messages_;
};
TEST_F(ProtoZeroConformanceTest, SimpleFieldsNoNesting) {
auto* msg = CreateMessage<pbtest::EveryField>();
msg->set_field_int32(-1);
msg->set_field_int64(-333123456789ll);
msg->set_field_uint32(600);
msg->set_field_uint64(333123456789ll);
msg->set_field_sint32(-5);
msg->set_field_sint64(-9000);
msg->set_field_fixed32(12345);
msg->set_field_fixed64(444123450000ll);
msg->set_field_sfixed32(-69999);
msg->set_field_sfixed64(-200);
msg->set_field_float(3.14f);
msg->set_field_double(0.5555);
msg->set_field_bool(true);
msg->set_small_enum(pbtest::SmallEnum::TO_BE);
msg->set_signed_enum(pbtest::SignedEnum::NEGATIVE);
msg->set_big_enum(pbtest::BigEnum::BEGIN);
msg->set_field_string("FizzBuzz");
msg->set_field_bytes(reinterpret_cast<const uint8_t*>("\x11\x00\xBE\xEF"), 4);
msg->add_repeated_int32(1);
msg->add_repeated_int32(-1);
msg->add_repeated_int32(100);
msg->add_repeated_int32(2000000);
size_t msg_size = GetNumSerializedBytes();
EXPECT_EQ(126u, msg_size);
std::unique_ptr<uint8_t[]> msg_binary(new uint8_t[msg_size]);
GetSerializedBytes(0, msg_size, msg_binary.get());
pbgold::EveryField gold_msg;
gold_msg.ParseFromArray(msg_binary.get(), static_cast<int>(msg_size));
EXPECT_EQ(-1, gold_msg.field_int32());
EXPECT_EQ(-333123456789ll, gold_msg.field_int64());
EXPECT_EQ(600u, gold_msg.field_uint32());
EXPECT_EQ(333123456789ull, gold_msg.field_uint64());
EXPECT_EQ(-5, gold_msg.field_sint32());
EXPECT_EQ(-9000, gold_msg.field_sint64());
EXPECT_EQ(12345u, gold_msg.field_fixed32());
EXPECT_EQ(444123450000ull, gold_msg.field_fixed64());
EXPECT_EQ(-69999, gold_msg.field_sfixed32());
EXPECT_EQ(-200, gold_msg.field_sfixed64());
EXPECT_FLOAT_EQ(3.14f, gold_msg.field_float());
EXPECT_DOUBLE_EQ(0.5555, gold_msg.field_double());
EXPECT_EQ(true, gold_msg.field_bool());
EXPECT_EQ(pbgold::SmallEnum::TO_BE, gold_msg.small_enum());
EXPECT_EQ(pbgold::SignedEnum::NEGATIVE, gold_msg.signed_enum());
EXPECT_EQ(pbgold::BigEnum::BEGIN, gold_msg.big_enum());
EXPECT_EQ("FizzBuzz", gold_msg.field_string());
EXPECT_EQ(std::string("\x11\x00\xBE\xEF", 4), gold_msg.field_bytes());
EXPECT_EQ(4, gold_msg.repeated_int32_size());
EXPECT_EQ(1, gold_msg.repeated_int32(0));
EXPECT_EQ(-1, gold_msg.repeated_int32(1));
EXPECT_EQ(100, gold_msg.repeated_int32(2));
EXPECT_EQ(2000000, gold_msg.repeated_int32(3));
}
TEST_F(ProtoZeroConformanceTest, NestedMessages) {
auto* msg_a = CreateMessage<pbtest::NestedA>();
pbtest::NestedA::NestedB* msg_b = msg_a->add_repeated_a();
pbtest::NestedA::NestedB::NestedC* msg_c = msg_b->set_value_b();
msg_c->set_value_c(321);
msg_b = msg_a->add_repeated_a();
msg_c = msg_a->set_super_nested();
msg_c->set_value_c(1000);
msg_a->Finalize();
size_t msg_size = GetNumSerializedBytes();
EXPECT_EQ(26u, msg_size);
std::unique_ptr<uint8_t[]> msg_binary(new uint8_t[msg_size]);
GetSerializedBytes(0, msg_size, msg_binary.get());
pbgold::NestedA gold_msg_a;
gold_msg_a.ParseFromArray(msg_binary.get(), static_cast<int>(msg_size));
EXPECT_EQ(2, gold_msg_a.repeated_a_size());
EXPECT_EQ(321, gold_msg_a.repeated_a(0).value_b().value_c());
EXPECT_FALSE(gold_msg_a.repeated_a(1).has_value_b());
EXPECT_EQ(1000, gold_msg_a.super_nested().value_c());
}
TEST(ProtoZeroTest, Simple) {
// Test the includes for indirect public import: library.pbzero.h ->
// library_internals/galaxies.pbzero.h -> upper_import.pbzero.h .
EXPECT_LE(0u, sizeof(pbtest::TrickyPublicImport));
}
TEST(ProtoZeroTest, Reflection) {
// Tests camel case conversion as well.
EXPECT_EQ(1, pbtest::CamelCaseFields::kFooBarBazFieldNumber);
EXPECT_EQ(2, pbtest::CamelCaseFields::kBarBazFieldNumber);
EXPECT_EQ(3, pbtest::CamelCaseFields::kMooMooFieldNumber);
EXPECT_EQ(4, pbtest::CamelCaseFields::kURLEncoderFieldNumber);
EXPECT_EQ(5, pbtest::CamelCaseFields::kXMapFieldNumber);
EXPECT_EQ(6, pbtest::CamelCaseFields::kUrLENcoDerFieldNumber);
EXPECT_EQ(7, pbtest::CamelCaseFields::kBigBangFieldNumber);
EXPECT_EQ(8, pbtest::CamelCaseFields::kU2FieldNumber);
EXPECT_EQ(9, pbtest::CamelCaseFields::kBangBigFieldNumber);
const ProtoFieldDescriptor* reflection =
pbtest::EveryField::GetFieldDescriptor(
pbtest::EveryField::kFieldInt32FieldNumber);
EXPECT_STREQ("field_int32", reflection->name());
EXPECT_EQ(ProtoFieldDescriptor::Type::TYPE_INT32, reflection->type());
EXPECT_EQ(1u, reflection->number());
EXPECT_FALSE(reflection->is_repeated());
EXPECT_TRUE(reflection->is_valid());
EXPECT_FALSE(pbtest::TransgalacticParcel::GetFieldDescriptor(42)->is_valid());
}
} // namespace
} // namespace protozero