|  | // Protocol Buffers - Google's data interchange format | 
|  | // Copyright 2008 Google Inc.  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 "google/protobuf/any.pb.h" | 
|  | #include <gtest/gtest.h> | 
|  | #include "absl/strings/str_cat.h" | 
|  | #include "google/protobuf/any_test.pb.h" | 
|  | #include "google/protobuf/unittest.pb.h" | 
|  |  | 
|  |  | 
|  | // Must be included last. | 
|  | #include "google/protobuf/port_def.inc" | 
|  |  | 
|  | namespace google { | 
|  | namespace protobuf { | 
|  | namespace { | 
|  |  | 
|  | TEST(AnyMetadataTest, ConstInit) { | 
|  | PROTOBUF_CONSTINIT static internal::AnyMetadata metadata(nullptr, nullptr); | 
|  | (void)metadata; | 
|  | } | 
|  |  | 
|  | TEST(AnyTest, TestPackAndUnpack) { | 
|  | protobuf_unittest::TestAny submessage; | 
|  | submessage.set_int32_value(12345); | 
|  | protobuf_unittest::TestAny message; | 
|  | ASSERT_TRUE(message.mutable_any_value()->PackFrom(submessage)); | 
|  |  | 
|  | std::string data = message.SerializeAsString(); | 
|  |  | 
|  | ASSERT_TRUE(message.ParseFromString(data)); | 
|  | EXPECT_TRUE(message.has_any_value()); | 
|  | submessage.Clear(); | 
|  | ASSERT_TRUE(message.any_value().UnpackTo(&submessage)); | 
|  | EXPECT_EQ(12345, submessage.int32_value()); | 
|  | } | 
|  |  | 
|  | TEST(AnyTest, TestPackFromSerializationExceedsSizeLimit) { | 
|  | #if defined(_MSC_VER) && defined(_M_IX86) | 
|  | GTEST_SKIP() << "This toolchain can't allocate that much memory."; | 
|  | #endif | 
|  | protobuf_unittest::TestAny submessage; | 
|  | submessage.mutable_text()->resize(INT_MAX, 'a'); | 
|  | protobuf_unittest::TestAny message; | 
|  | EXPECT_FALSE(message.mutable_any_value()->PackFrom(submessage)); | 
|  | } | 
|  |  | 
|  | TEST(AnyTest, TestUnpackWithTypeMismatch) { | 
|  | protobuf_unittest::TestAny payload; | 
|  | payload.set_int32_value(13); | 
|  | google::protobuf::Any any; | 
|  | any.PackFrom(payload); | 
|  |  | 
|  | // Attempt to unpack into the wrong type. | 
|  | protobuf_unittest::TestAllTypes dest; | 
|  | EXPECT_FALSE(any.UnpackTo(&dest)); | 
|  | } | 
|  |  | 
|  | TEST(AnyTest, TestPackAndUnpackAny) { | 
|  | // We can pack a Any message inside another Any message. | 
|  | protobuf_unittest::TestAny submessage; | 
|  | submessage.set_int32_value(12345); | 
|  | google::protobuf::Any any; | 
|  | any.PackFrom(submessage); | 
|  | protobuf_unittest::TestAny message; | 
|  | message.mutable_any_value()->PackFrom(any); | 
|  |  | 
|  | std::string data = message.SerializeAsString(); | 
|  |  | 
|  | ASSERT_TRUE(message.ParseFromString(data)); | 
|  | EXPECT_TRUE(message.has_any_value()); | 
|  | any.Clear(); | 
|  | submessage.Clear(); | 
|  | ASSERT_TRUE(message.any_value().UnpackTo(&any)); | 
|  | ASSERT_TRUE(any.UnpackTo(&submessage)); | 
|  | EXPECT_EQ(12345, submessage.int32_value()); | 
|  | } | 
|  |  | 
|  | TEST(AnyTest, TestPackWithCustomTypeUrl) { | 
|  | protobuf_unittest::TestAny submessage; | 
|  | submessage.set_int32_value(12345); | 
|  | google::protobuf::Any any; | 
|  | // Pack with a custom type URL prefix. | 
|  | any.PackFrom(submessage, "type.myservice.com"); | 
|  | EXPECT_EQ("type.myservice.com/protobuf_unittest.TestAny", any.type_url()); | 
|  | // Pack with a custom type URL prefix ending with '/'. | 
|  | any.PackFrom(submessage, "type.myservice.com/"); | 
|  | EXPECT_EQ("type.myservice.com/protobuf_unittest.TestAny", any.type_url()); | 
|  | // Pack with an empty type URL prefix. | 
|  | any.PackFrom(submessage, ""); | 
|  | EXPECT_EQ("/protobuf_unittest.TestAny", any.type_url()); | 
|  |  | 
|  | // Test unpacking the type. | 
|  | submessage.Clear(); | 
|  | EXPECT_TRUE(any.UnpackTo(&submessage)); | 
|  | EXPECT_EQ(12345, submessage.int32_value()); | 
|  | } | 
|  |  | 
|  | TEST(AnyTest, TestIs) { | 
|  | protobuf_unittest::TestAny submessage; | 
|  | submessage.set_int32_value(12345); | 
|  | google::protobuf::Any any; | 
|  | any.PackFrom(submessage); | 
|  | ASSERT_TRUE(any.ParseFromString(any.SerializeAsString())); | 
|  | EXPECT_TRUE(any.Is<protobuf_unittest::TestAny>()); | 
|  | EXPECT_FALSE(any.Is<google::protobuf::Any>()); | 
|  |  | 
|  | protobuf_unittest::TestAny message; | 
|  | message.mutable_any_value()->PackFrom(any); | 
|  | ASSERT_TRUE(message.ParseFromString(message.SerializeAsString())); | 
|  | EXPECT_FALSE(message.any_value().Is<protobuf_unittest::TestAny>()); | 
|  | EXPECT_TRUE(message.any_value().Is<google::protobuf::Any>()); | 
|  |  | 
|  | any.set_type_url("/protobuf_unittest.TestAny"); | 
|  | EXPECT_TRUE(any.Is<protobuf_unittest::TestAny>()); | 
|  | // The type URL must contain at least one "/". | 
|  | any.set_type_url("protobuf_unittest.TestAny"); | 
|  | EXPECT_FALSE(any.Is<protobuf_unittest::TestAny>()); | 
|  | // The type name after the slash must be fully qualified. | 
|  | any.set_type_url("/TestAny"); | 
|  | EXPECT_FALSE(any.Is<protobuf_unittest::TestAny>()); | 
|  | } | 
|  |  | 
|  | TEST(AnyTest, MoveConstructor) { | 
|  | protobuf_unittest::TestAny payload; | 
|  | payload.set_int32_value(12345); | 
|  |  | 
|  | google::protobuf::Any src; | 
|  | src.PackFrom(payload); | 
|  |  | 
|  | const char* type_url = src.type_url().data(); | 
|  |  | 
|  | google::protobuf::Any dst(std::move(src)); | 
|  | EXPECT_EQ(type_url, dst.type_url().data()); | 
|  | payload.Clear(); | 
|  | ASSERT_TRUE(dst.UnpackTo(&payload)); | 
|  | EXPECT_EQ(12345, payload.int32_value()); | 
|  | } | 
|  |  | 
|  | TEST(AnyTest, MoveAssignment) { | 
|  | protobuf_unittest::TestAny payload; | 
|  | payload.set_int32_value(12345); | 
|  |  | 
|  | google::protobuf::Any src; | 
|  | src.PackFrom(payload); | 
|  |  | 
|  | const char* type_url = src.type_url().data(); | 
|  |  | 
|  | google::protobuf::Any dst; | 
|  | dst = std::move(src); | 
|  | EXPECT_EQ(type_url, dst.type_url().data()); | 
|  | payload.Clear(); | 
|  | ASSERT_TRUE(dst.UnpackTo(&payload)); | 
|  | EXPECT_EQ(12345, payload.int32_value()); | 
|  | } | 
|  |  | 
|  | #if GTEST_HAS_DEATH_TEST | 
|  | #ifndef NDEBUG | 
|  | TEST(AnyTest, PackSelfDeath) { | 
|  | google::protobuf::Any any; | 
|  | EXPECT_DEATH(any.PackFrom(any), "&message"); | 
|  | EXPECT_DEATH(any.PackFrom(any, ""), "&message"); | 
|  | } | 
|  | #endif  // !NDEBUG | 
|  | #endif  // GTEST_HAS_DEATH_TEST | 
|  |  | 
|  |  | 
|  | }  // namespace | 
|  | }  // namespace protobuf | 
|  | }  // namespace google | 
|  |  | 
|  | #include "google/protobuf/port_undef.inc" |