// Copyright (c) 2012 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/memory/scoped_ptr.h" #include "base/message_loop.h" #include "base/metrics/histogram.h" #include "base/metrics/histogram_samples.h" #include "base/metrics/statistics_recorder.h" #include "base/test/test_timeouts.h" #include "base/threading/platform_thread.h" #include "base/threading/thread_restrictions.h" #include "dbus/bus.h" #include "dbus/message.h" #include "dbus/object_proxy.h" #include "dbus/test_service.h" #include "testing/gtest/include/gtest/gtest.h" // The test for sender verification in ObjectProxy. class SignalSenderVerificationTest : public testing::Test { public: SignalSenderVerificationTest() : on_name_owner_changed_called_(false), on_ownership_called_(false) { } virtual void SetUp() { base::StatisticsRecorder::Initialize(); // Make the main thread not to allow IO. base::ThreadRestrictions::SetIOAllowed(false); // Start the D-Bus thread. dbus_thread_.reset(new base::Thread("D-Bus Thread")); base::Thread::Options thread_options; thread_options.message_loop_type = MessageLoop::TYPE_IO; ASSERT_TRUE(dbus_thread_->StartWithOptions(thread_options)); // Create the client, using the D-Bus thread. dbus::Bus::Options bus_options; bus_options.bus_type = dbus::Bus::SESSION; bus_options.connection_type = dbus::Bus::PRIVATE; bus_options.dbus_thread_message_loop_proxy = dbus_thread_->message_loop_proxy(); bus_ = new dbus::Bus(bus_options); object_proxy_ = bus_->GetObjectProxy( "org.chromium.TestService", dbus::ObjectPath("/org/chromium/TestObject")); ASSERT_TRUE(bus_->HasDBusThread()); object_proxy_->SetNameOwnerChangedCallback( base::Bind(&SignalSenderVerificationTest::OnNameOwnerChanged, base::Unretained(this))); // Connect to the "Test" signal of "org.chromium.TestInterface" from // the remote object. object_proxy_->ConnectToSignal( "org.chromium.TestInterface", "Test", base::Bind(&SignalSenderVerificationTest::OnTestSignal, base::Unretained(this)), base::Bind(&SignalSenderVerificationTest::OnConnected, base::Unretained(this))); // Wait until the object proxy is connected to the signal. message_loop_.Run(); // Start the test service, using the D-Bus thread. dbus::TestService::Options options; options.dbus_thread_message_loop_proxy = dbus_thread_->message_loop_proxy(); test_service_.reset(new dbus::TestService(options)); ASSERT_TRUE(test_service_->StartService()); ASSERT_TRUE(test_service_->WaitUntilServiceIsStarted()); ASSERT_TRUE(test_service_->HasDBusThread()); ASSERT_TRUE(test_service_->has_ownership()); // Same setup for the second TestService. This service should not have the // ownership of the name at this point. test_service2_.reset(new dbus::TestService(options)); ASSERT_TRUE(test_service2_->StartService()); ASSERT_TRUE(test_service2_->WaitUntilServiceIsStarted()); ASSERT_TRUE(test_service2_->HasDBusThread()); ASSERT_FALSE(test_service2_->has_ownership()); // The name should be owned and known at this point. if (!on_name_owner_changed_called_) message_loop_.Run(); ASSERT_FALSE(latest_name_owner_.empty()); } virtual void TearDown() { bus_->ShutdownOnDBusThreadAndBlock(); // Shut down the service. test_service_->ShutdownAndBlock(); test_service2_->ShutdownAndBlock(); // Reset to the default. base::ThreadRestrictions::SetIOAllowed(true); // Stopping a thread is considered an IO operation, so do this after // allowing IO. test_service_->Stop(); test_service2_->Stop(); } void OnOwnership(bool expected, bool success) { ASSERT_EQ(expected, success); // PostTask to quit the MessageLoop as this is called from D-Bus thread. message_loop_.PostTask( FROM_HERE, base::Bind(&SignalSenderVerificationTest::OnOwnershipInternal, base::Unretained(this))); } void OnOwnershipInternal() { on_ownership_called_ = true; message_loop_.Quit(); } void OnNameOwnerChanged(dbus::Signal* signal) { dbus::MessageReader reader(signal); std::string name, old_owner, new_owner; ASSERT_TRUE(reader.PopString(&name)); ASSERT_TRUE(reader.PopString(&old_owner)); ASSERT_TRUE(reader.PopString(&new_owner)); latest_name_owner_ = new_owner; on_name_owner_changed_called_ = true; message_loop_.Quit(); } protected: // Called when the "Test" signal is received, in the main thread. // Copy the string payload to |test_signal_string_|. void OnTestSignal(dbus::Signal* signal) { dbus::MessageReader reader(signal); ASSERT_TRUE(reader.PopString(&test_signal_string_)); message_loop_.Quit(); } // Called when connected to the signal. void OnConnected(const std::string& interface_name, const std::string& signal_name, bool success) { ASSERT_TRUE(success); message_loop_.Quit(); } // Wait for the hey signal to be received. void WaitForTestSignal() { // OnTestSignal() will quit the message loop. message_loop_.Run(); } MessageLoop message_loop_; scoped_ptr dbus_thread_; scoped_refptr bus_; dbus::ObjectProxy* object_proxy_; scoped_ptr test_service_; scoped_ptr test_service2_; // Text message from "Test" signal. std::string test_signal_string_; // The known latest name owner of TestService. Updated in OnNameOwnerChanged. std::string latest_name_owner_; // Boolean flags to record callback calls. bool on_name_owner_changed_called_; bool on_ownership_called_; }; TEST_F(SignalSenderVerificationTest, TestSignalAccepted) { const char kMessage[] = "hello, world"; // Send the test signal from the exported object. test_service_->SendTestSignal(kMessage); // Receive the signal with the object proxy. The signal is handled in // SignalSenderVerificationTest::OnTestSignal() in the main thread. WaitForTestSignal(); ASSERT_EQ(kMessage, test_signal_string_); } TEST_F(SignalSenderVerificationTest, TestSignalRejected) { // To make sure the histogram instance is created. UMA_HISTOGRAM_COUNTS("DBus.RejectedSignalCount", 0); base::Histogram* reject_signal_histogram = base::StatisticsRecorder::FindHistogram("DBus.RejectedSignalCount"); scoped_ptr samples1( reject_signal_histogram->SnapshotSamples()); const char kNewMessage[] = "hello, new world"; test_service2_->SendTestSignal(kNewMessage); // This test tests that our callback is NOT called by the ObjectProxy. // Sleep to have message delivered to the client via the D-Bus service. base::PlatformThread::Sleep(TestTimeouts::action_timeout()); scoped_ptr samples2( reject_signal_histogram->SnapshotSamples()); ASSERT_EQ("", test_signal_string_); EXPECT_EQ(samples1->TotalCount() + 1, samples2->TotalCount()); } TEST_F(SignalSenderVerificationTest, TestOwnerChanged) { const char kMessage[] = "hello, world"; // Send the test signal from the exported object. test_service_->SendTestSignal(kMessage); // Receive the signal with the object proxy. The signal is handled in // SignalSenderVerificationTest::OnTestSignal() in the main thread. WaitForTestSignal(); ASSERT_EQ(kMessage, test_signal_string_); // Release and acquire the name ownership. // latest_name_owner_ should be non empty as |test_service_| owns the name. ASSERT_FALSE(latest_name_owner_.empty()); test_service_->ShutdownAndBlock(); // OnNameOwnerChanged will PostTask to quit the message loop. message_loop_.Run(); // latest_name_owner_ should be empty as the owner is gone. ASSERT_TRUE(latest_name_owner_.empty()); // Reset the flag as NameOwnerChanged is already received in setup. on_name_owner_changed_called_ = false; test_service2_->RequestOwnership( base::Bind(&SignalSenderVerificationTest::OnOwnership, base::Unretained(this), true)); // Both of OnNameOwnerChanged() and OnOwnership() should quit the MessageLoop, // but there's no expected order of those 2 event. message_loop_.Run(); if (!on_name_owner_changed_called_ || !on_ownership_called_) message_loop_.Run(); ASSERT_TRUE(on_name_owner_changed_called_); ASSERT_TRUE(on_ownership_called_); // latest_name_owner_ becomes non empty as the new owner appears. ASSERT_FALSE(latest_name_owner_.empty()); // Now the second service owns the name. const char kNewMessage[] = "hello, new world"; test_service2_->SendTestSignal(kNewMessage); WaitForTestSignal(); ASSERT_EQ(kNewMessage, test_signal_string_); }