// Copyright 2015 The Chromium Authors. All rights reserved. // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. #include "base/bind.h" #include "base/callback.h" #include "base/logging.h" #include "base/message_loop/message_loop.h" #include "base/run_loop.h" #include "mojo/public/cpp/bindings/binding_set.h" #include "mojo/public/cpp/bindings/interface_request.h" #include "mojo/public/cpp/bindings/tests/rect_blink.h" #include "mojo/public/cpp/bindings/tests/rect_chromium.h" #include "mojo/public/cpp/bindings/tests/struct_with_traits_impl.h" #include "mojo/public/cpp/bindings/tests/struct_with_traits_impl_traits.h" #include "mojo/public/cpp/bindings/tests/variant_test_util.h" #include "mojo/public/interfaces/bindings/tests/struct_with_traits.mojom.h" #include "mojo/public/interfaces/bindings/tests/test_native_types.mojom-blink.h" #include "mojo/public/interfaces/bindings/tests/test_native_types.mojom.h" #include "testing/gtest/include/gtest/gtest.h" namespace mojo { namespace test { namespace { template <typename T> void DoExpectResult(const T& expected, const base::Closure& callback, const T& actual) { EXPECT_EQ(expected.x(), actual.x()); EXPECT_EQ(expected.y(), actual.y()); EXPECT_EQ(expected.width(), actual.width()); EXPECT_EQ(expected.height(), actual.height()); callback.Run(); } template <typename T> base::Callback<void(const T&)> ExpectResult(const T& r, const base::Closure& callback) { return base::Bind(&DoExpectResult<T>, r, callback); } template <typename T> void DoFail(const std::string& reason, const T&) { EXPECT_TRUE(false) << reason; } template <typename T> base::Callback<void(const T&)> Fail(const std::string& reason) { return base::Bind(&DoFail<T>, reason); } template <typename T> void ExpectError(InterfacePtr<T> *proxy, const base::Closure& callback) { proxy->set_connection_error_handler(callback); } // This implements the generated Chromium variant of RectService. class ChromiumRectServiceImpl : public RectService { public: ChromiumRectServiceImpl() {} // mojo::test::RectService: void AddRect(const RectChromium& r) override { if (r.GetArea() > largest_rect_.GetArea()) largest_rect_ = r; } void GetLargestRect(const GetLargestRectCallback& callback) override { callback.Run(largest_rect_); } private: RectChromium largest_rect_; }; // This implements the generated Blink variant of RectService. class BlinkRectServiceImpl : public blink::RectService { public: BlinkRectServiceImpl() {} // mojo::test::blink::RectService: void AddRect(const RectBlink& r) override { if (r.computeArea() > largest_rect_.computeArea()) { largest_rect_.setX(r.x()); largest_rect_.setY(r.y()); largest_rect_.setWidth(r.width()); largest_rect_.setHeight(r.height()); } } void GetLargestRect(const GetLargestRectCallback& callback) override { callback.Run(largest_rect_); } private: RectBlink largest_rect_; }; // A test which runs both Chromium and Blink implementations of a RectService. class StructTraitsTest : public testing::Test, public TraitsTestService { public: StructTraitsTest() {} protected: void BindToChromiumService(RectServiceRequest request) { chromium_bindings_.AddBinding(&chromium_service_, std::move(request)); } void BindToChromiumService(blink::RectServiceRequest request) { chromium_bindings_.AddBinding( &chromium_service_, ConvertInterfaceRequest<RectService>(std::move(request))); } void BindToBlinkService(blink::RectServiceRequest request) { blink_bindings_.AddBinding(&blink_service_, std::move(request)); } void BindToBlinkService(RectServiceRequest request) { blink_bindings_.AddBinding( &blink_service_, ConvertInterfaceRequest<blink::RectService>(std::move(request))); } TraitsTestServicePtr GetTraitsTestProxy() { return traits_test_bindings_.CreateInterfacePtrAndBind(this); } private: // TraitsTestService: void EchoStructWithTraits( const StructWithTraitsImpl& s, const EchoStructWithTraitsCallback& callback) override { callback.Run(s); } void EchoPassByValueStructWithTraits( PassByValueStructWithTraitsImpl s, const EchoPassByValueStructWithTraitsCallback& callback) override { callback.Run(std::move(s)); } void EchoEnumWithTraits(EnumWithTraitsImpl e, const EchoEnumWithTraitsCallback& callback) override { callback.Run(e); } void EchoStructWithTraitsForUniquePtrTest( std::unique_ptr<int> e, const EchoStructWithTraitsForUniquePtrTestCallback& callback) override { callback.Run(std::move(e)); } base::MessageLoop loop_; ChromiumRectServiceImpl chromium_service_; BindingSet<RectService> chromium_bindings_; BlinkRectServiceImpl blink_service_; BindingSet<blink::RectService> blink_bindings_; BindingSet<TraitsTestService> traits_test_bindings_; }; } // namespace TEST_F(StructTraitsTest, ChromiumProxyToChromiumService) { RectServicePtr chromium_proxy; BindToChromiumService(GetProxy(&chromium_proxy)); { base::RunLoop loop; chromium_proxy->AddRect(RectChromium(1, 1, 4, 5)); chromium_proxy->AddRect(RectChromium(-1, -1, 2, 2)); chromium_proxy->GetLargestRect( ExpectResult(RectChromium(1, 1, 4, 5), loop.QuitClosure())); loop.Run(); } } TEST_F(StructTraitsTest, ChromiumToBlinkService) { RectServicePtr chromium_proxy; BindToBlinkService(GetProxy(&chromium_proxy)); { base::RunLoop loop; chromium_proxy->AddRect(RectChromium(1, 1, 4, 5)); chromium_proxy->AddRect(RectChromium(2, 2, 5, 5)); chromium_proxy->GetLargestRect( ExpectResult(RectChromium(2, 2, 5, 5), loop.QuitClosure())); loop.Run(); } // The Blink service should drop our connection because RectBlink's // deserializer rejects negative origins. { base::RunLoop loop; ExpectError(&chromium_proxy, loop.QuitClosure()); chromium_proxy->AddRect(RectChromium(-1, -1, 2, 2)); chromium_proxy->GetLargestRect( Fail<RectChromium>("The pipe should have been closed.")); loop.Run(); } } TEST_F(StructTraitsTest, BlinkProxyToBlinkService) { blink::RectServicePtr blink_proxy; BindToBlinkService(GetProxy(&blink_proxy)); { base::RunLoop loop; blink_proxy->AddRect(RectBlink(1, 1, 4, 5)); blink_proxy->AddRect(RectBlink(10, 10, 20, 20)); blink_proxy->GetLargestRect( ExpectResult(RectBlink(10, 10, 20, 20), loop.QuitClosure())); loop.Run(); } } TEST_F(StructTraitsTest, BlinkProxyToChromiumService) { blink::RectServicePtr blink_proxy; BindToChromiumService(GetProxy(&blink_proxy)); { base::RunLoop loop; blink_proxy->AddRect(RectBlink(1, 1, 4, 5)); blink_proxy->AddRect(RectBlink(10, 10, 2, 2)); blink_proxy->GetLargestRect( ExpectResult(RectBlink(1, 1, 4, 5), loop.QuitClosure())); loop.Run(); } } void ExpectStructWithTraits(const StructWithTraitsImpl& expected, const base::Closure& closure, const StructWithTraitsImpl& passed) { EXPECT_EQ(expected.get_enum(), passed.get_enum()); EXPECT_EQ(expected.get_bool(), passed.get_bool()); EXPECT_EQ(expected.get_uint32(), passed.get_uint32()); EXPECT_EQ(expected.get_uint64(), passed.get_uint64()); EXPECT_EQ(expected.get_string(), passed.get_string()); EXPECT_EQ(expected.get_string_array(), passed.get_string_array()); EXPECT_EQ(expected.get_struct(), passed.get_struct()); EXPECT_EQ(expected.get_struct_array(), passed.get_struct_array()); EXPECT_EQ(expected.get_struct_map(), passed.get_struct_map()); closure.Run(); } TEST_F(StructTraitsTest, EchoStructWithTraits) { StructWithTraitsImpl input; input.set_enum(EnumWithTraitsImpl::CUSTOM_VALUE_1); input.set_bool(true); input.set_uint32(7); input.set_uint64(42); input.set_string("hello world!"); input.get_mutable_string_array().assign({"hello", "world!"}); input.get_mutable_struct().value = 42; input.get_mutable_struct_array().resize(2); input.get_mutable_struct_array()[0].value = 1; input.get_mutable_struct_array()[1].value = 2; input.get_mutable_struct_map()["hello"] = NestedStructWithTraitsImpl(1024); input.get_mutable_struct_map()["world"] = NestedStructWithTraitsImpl(2048); base::RunLoop loop; TraitsTestServicePtr proxy = GetTraitsTestProxy(); proxy->EchoStructWithTraits( input, base::Bind(&ExpectStructWithTraits, input, loop.QuitClosure())); loop.Run(); } TEST_F(StructTraitsTest, CloneStructWithTraitsContainer) { StructWithTraitsContainerPtr container = StructWithTraitsContainer::New(); container->f_struct.set_uint32(7); container->f_struct.set_uint64(42); StructWithTraitsContainerPtr cloned_container = container.Clone(); EXPECT_EQ(7u, cloned_container->f_struct.get_uint32()); EXPECT_EQ(42u, cloned_container->f_struct.get_uint64()); } void CaptureMessagePipe(ScopedMessagePipeHandle* storage, const base::Closure& closure, PassByValueStructWithTraitsImpl passed) { storage->reset(MessagePipeHandle( passed.get_mutable_handle().release().value())); closure.Run(); } TEST_F(StructTraitsTest, EchoPassByValueStructWithTraits) { MessagePipe mp; PassByValueStructWithTraitsImpl input; input.get_mutable_handle().reset(mp.handle0.release()); base::RunLoop loop; TraitsTestServicePtr proxy = GetTraitsTestProxy(); ScopedMessagePipeHandle received; proxy->EchoPassByValueStructWithTraits( std::move(input), base::Bind(&CaptureMessagePipe, &received, loop.QuitClosure())); loop.Run(); ASSERT_TRUE(received.is_valid()); // Verify that the message pipe handle is correctly passed. const char kHello[] = "hello"; const uint32_t kHelloSize = static_cast<uint32_t>(sizeof(kHello)); EXPECT_EQ(MOJO_RESULT_OK, WriteMessageRaw(mp.handle1.get(), kHello, kHelloSize, nullptr, 0, MOJO_WRITE_MESSAGE_FLAG_NONE)); EXPECT_EQ(MOJO_RESULT_OK, Wait(received.get(), MOJO_HANDLE_SIGNAL_READABLE, MOJO_DEADLINE_INDEFINITE, nullptr)); char buffer[10] = {0}; uint32_t buffer_size = static_cast<uint32_t>(sizeof(buffer)); EXPECT_EQ(MOJO_RESULT_OK, ReadMessageRaw(received.get(), buffer, &buffer_size, nullptr, nullptr, MOJO_READ_MESSAGE_FLAG_NONE)); EXPECT_EQ(kHelloSize, buffer_size); EXPECT_STREQ(kHello, buffer); } void ExpectEnumWithTraits(EnumWithTraitsImpl expected_value, const base::Closure& closure, EnumWithTraitsImpl value) { EXPECT_EQ(expected_value, value); closure.Run(); } TEST_F(StructTraitsTest, EchoEnumWithTraits) { base::RunLoop loop; TraitsTestServicePtr proxy = GetTraitsTestProxy(); proxy->EchoEnumWithTraits( EnumWithTraitsImpl::CUSTOM_VALUE_1, base::Bind(&ExpectEnumWithTraits, EnumWithTraitsImpl::CUSTOM_VALUE_1, loop.QuitClosure())); loop.Run(); } TEST_F(StructTraitsTest, SerializeStructWithTraits) { StructWithTraitsImpl input; input.set_enum(EnumWithTraitsImpl::CUSTOM_VALUE_1); input.set_bool(true); input.set_uint32(7); input.set_uint64(42); input.set_string("hello world!"); input.get_mutable_string_array().assign({"hello", "world!"}); input.get_mutable_struct().value = 42; input.get_mutable_struct_array().resize(2); input.get_mutable_struct_array()[0].value = 1; input.get_mutable_struct_array()[1].value = 2; input.get_mutable_struct_map()["hello"] = NestedStructWithTraitsImpl(1024); input.get_mutable_struct_map()["world"] = NestedStructWithTraitsImpl(2048); mojo::Array<uint8_t> data = StructWithTraits::Serialize(&input); StructWithTraitsImpl output; ASSERT_TRUE(StructWithTraits::Deserialize(std::move(data), &output)); EXPECT_EQ(input.get_enum(), output.get_enum()); EXPECT_EQ(input.get_bool(), output.get_bool()); EXPECT_EQ(input.get_uint32(), output.get_uint32()); EXPECT_EQ(input.get_uint64(), output.get_uint64()); EXPECT_EQ(input.get_string(), output.get_string()); EXPECT_EQ(input.get_string_array(), output.get_string_array()); EXPECT_EQ(input.get_struct(), output.get_struct()); EXPECT_EQ(input.get_struct_array(), output.get_struct_array()); EXPECT_EQ(input.get_struct_map(), output.get_struct_map()); } void ExpectUniquePtr(int expected, const base::Closure& closure, std::unique_ptr<int> value) { EXPECT_EQ(expected, *value); closure.Run(); } TEST_F(StructTraitsTest, TypemapUniquePtr) { base::RunLoop loop; TraitsTestServicePtr proxy = GetTraitsTestProxy(); proxy->EchoStructWithTraitsForUniquePtrTest( base::MakeUnique<int>(12345), base::Bind(&ExpectUniquePtr, 12345, loop.QuitClosure())); loop.Run(); } } // namespace test } // namespace mojo