#include "dbus/bus.h"
#include <memory>
#include <utility>
#include "base/files/file_descriptor_watcher_posix.h"
#include "base/functional/bind.h"
#include "base/functional/callback_helpers.h"
#include "base/memory/ref_counted.h"
#include "base/message_loop/message_pump_type.h"
#include "base/run_loop.h"
#include "base/test/task_environment.h"
#include "base/threading/thread.h"
#include "dbus/error.h"
#include "dbus/exported_object.h"
#include "dbus/object_path.h"
#include "dbus/object_proxy.h"
#include "dbus/test_service.h"
#include "testing/gtest/include/gtest/gtest.h"
namespace dbus {
namespace {
class RunLoopWithExpectedCount {
public:
RunLoopWithExpectedCount() : expected_quit_calls_(0), actual_quit_calls_(0) {}
RunLoopWithExpectedCount(const RunLoopWithExpectedCount&) = delete;
RunLoopWithExpectedCount& operator=(const RunLoopWithExpectedCount&) = delete;
~RunLoopWithExpectedCount() = default;
void Run(int expected_quit_calls) {
DCHECK_EQ(0, expected_quit_calls_);
DCHECK_EQ(0, actual_quit_calls_);
expected_quit_calls_ = expected_quit_calls;
run_loop_ = std::make_unique<base::RunLoop>();
run_loop_->Run();
}
void QuitIfConditionIsSatisified() {
if (++actual_quit_calls_ != expected_quit_calls_)
return;
run_loop_->Quit();
expected_quit_calls_ = 0;
actual_quit_calls_ = 0;
}
private:
std::unique_ptr<base::RunLoop> run_loop_;
int expected_quit_calls_;
int actual_quit_calls_;
};
void OnServiceOwnerChanged(RunLoopWithExpectedCount* run_loop_state,
std::string* service_owner,
int* num_of_owner_changes,
const std::string& new_service_owner) {
*service_owner = new_service_owner;
++(*num_of_owner_changes);
run_loop_state->QuitIfConditionIsSatisified();
}
}
TEST(BusTest, GetObjectProxy) {
scoped_refptr<Bus> bus = new Bus(Bus::Options());
ObjectProxy* object_proxy1 =
bus->GetObjectProxy("org.chromium.TestService",
ObjectPath("/org/chromium/TestObject"));
ASSERT_TRUE(object_proxy1);
ObjectProxy* object_proxy2 =
bus->GetObjectProxy("org.chromium.TestService",
ObjectPath("/org/chromium/TestObject"));
ASSERT_TRUE(object_proxy2);
EXPECT_EQ(object_proxy1, object_proxy2);
ObjectProxy* object_proxy3 =
bus->GetObjectProxy(
"org.chromium.TestService",
ObjectPath("/org/chromium/DifferentTestObject"));
ASSERT_TRUE(object_proxy3);
EXPECT_NE(object_proxy1, object_proxy3);
bus->ShutdownAndBlock();
}
TEST(BusTest, GetObjectProxyIgnoreUnknownService) {
scoped_refptr<Bus> bus = new Bus(Bus::Options());
ObjectProxy* object_proxy1 =
bus->GetObjectProxyWithOptions(
"org.chromium.TestService",
ObjectPath("/org/chromium/TestObject"),
ObjectProxy::IGNORE_SERVICE_UNKNOWN_ERRORS);
ASSERT_TRUE(object_proxy1);
ObjectProxy* object_proxy2 =
bus->GetObjectProxyWithOptions(
"org.chromium.TestService",
ObjectPath("/org/chromium/TestObject"),
ObjectProxy::IGNORE_SERVICE_UNKNOWN_ERRORS);
ASSERT_TRUE(object_proxy2);
EXPECT_EQ(object_proxy1, object_proxy2);
ObjectProxy* object_proxy3 =
bus->GetObjectProxyWithOptions(
"org.chromium.TestService",
ObjectPath("/org/chromium/DifferentTestObject"),
ObjectProxy::IGNORE_SERVICE_UNKNOWN_ERRORS);
ASSERT_TRUE(object_proxy3);
EXPECT_NE(object_proxy1, object_proxy3);
bus->ShutdownAndBlock();
}
TEST(BusTest, RemoveObjectProxy) {
base::test::SingleThreadTaskEnvironment task_environment;
base::Thread::Options thread_options;
thread_options.message_pump_type = base::MessagePumpType::IO;
base::Thread dbus_thread("D-Bus thread");
dbus_thread.StartWithOptions(std::move(thread_options));
Bus::Options options;
options.dbus_task_runner = dbus_thread.task_runner();
scoped_refptr<Bus> bus = new Bus(std::move(options));
ASSERT_FALSE(bus->shutdown_completed());
ASSERT_FALSE(bus->RemoveObjectProxy("org.chromium.TestService",
ObjectPath("/org/chromium/TestObject"),
base::DoNothing()));
ObjectProxy* object_proxy1 =
bus->GetObjectProxy("org.chromium.TestService",
ObjectPath("/org/chromium/TestObject"));
ASSERT_TRUE(object_proxy1);
object_proxy1->AddRef();
ASSERT_TRUE(bus->RemoveObjectProxy("org.chromium.TestService",
ObjectPath("/org/chromium/TestObject"),
base::DoNothing()));
ObjectProxy* object_proxy2 =
bus->GetObjectProxy("org.chromium.TestService",
ObjectPath("/org/chromium/TestObject"));
ASSERT_TRUE(object_proxy2);
EXPECT_NE(object_proxy1, object_proxy2);
object_proxy1->Release();
bus->ShutdownOnDBusThreadAndBlock();
EXPECT_TRUE(bus->shutdown_completed());
dbus_thread.Stop();
}
TEST(BusTest, GetExportedObject) {
scoped_refptr<Bus> bus = new Bus(Bus::Options());
ExportedObject* object_proxy1 =
bus->GetExportedObject(ObjectPath("/org/chromium/TestObject"));
ASSERT_TRUE(object_proxy1);
ExportedObject* object_proxy2 =
bus->GetExportedObject(ObjectPath("/org/chromium/TestObject"));
ASSERT_TRUE(object_proxy2);
EXPECT_EQ(object_proxy1, object_proxy2);
ExportedObject* object_proxy3 =
bus->GetExportedObject(
ObjectPath("/org/chromium/DifferentTestObject"));
ASSERT_TRUE(object_proxy3);
EXPECT_NE(object_proxy1, object_proxy3);
bus->ShutdownAndBlock();
}
TEST(BusTest, UnregisterExportedObject) {
base::Thread::Options thread_options;
thread_options.message_pump_type = base::MessagePumpType::IO;
base::Thread dbus_thread("D-Bus thread");
dbus_thread.StartWithOptions(std::move(thread_options));
Bus::Options options;
options.dbus_task_runner = dbus_thread.task_runner();
scoped_refptr<Bus> bus = new Bus(std::move(options));
ASSERT_FALSE(bus->shutdown_completed());
ExportedObject* object_proxy1 =
bus->GetExportedObject(ObjectPath("/org/chromium/TestObject"));
ASSERT_TRUE(object_proxy1);
object_proxy1->AddRef();
bus->UnregisterExportedObject(ObjectPath("/org/chromium/TestObject"));
ExportedObject* object_proxy2 =
bus->GetExportedObject(ObjectPath("/org/chromium/TestObject"));
ASSERT_TRUE(object_proxy2);
EXPECT_NE(object_proxy1, object_proxy2);
object_proxy1->Release();
bus->ShutdownOnDBusThreadAndBlock();
EXPECT_TRUE(bus->shutdown_completed());
dbus_thread.Stop();
}
TEST(BusTest, ShutdownAndBlock) {
scoped_refptr<Bus> bus = new Bus(Bus::Options());
ASSERT_FALSE(bus->shutdown_completed());
bus->ShutdownAndBlock();
EXPECT_TRUE(bus->shutdown_completed());
}
TEST(BusTest, ShutdownAndBlockWithDBusThread) {
base::Thread::Options thread_options;
thread_options.message_pump_type = base::MessagePumpType::IO;
base::Thread dbus_thread("D-Bus thread");
dbus_thread.StartWithOptions(std::move(thread_options));
Bus::Options options;
options.dbus_task_runner = dbus_thread.task_runner();
scoped_refptr<Bus> bus = new Bus(std::move(options));
ASSERT_FALSE(bus->shutdown_completed());
bus->ShutdownOnDBusThreadAndBlock();
EXPECT_TRUE(bus->shutdown_completed());
dbus_thread.Stop();
}
TEST(BusTest, DoubleAddAndRemoveMatch) {
scoped_refptr<Bus> bus = new Bus(Bus::Options());
dbus::Error error;
bus->Connect();
bus->AddMatch("type='signal',interface='org.chromium.TestService',path='/'",
&error);
ASSERT_FALSE(error.IsValid());
bus->AddMatch("type='signal',interface='org.chromium.TestService',path='/'",
&error);
ASSERT_FALSE(error.IsValid());
ASSERT_TRUE(bus->RemoveMatch(
"type='signal',interface='org.chromium.TestService',path='/'", &error));
ASSERT_FALSE(error.IsValid());
ASSERT_TRUE(bus->RemoveMatch(
"type='signal',interface='org.chromium.TestService',path='/'", &error));
ASSERT_FALSE(error.IsValid());
ASSERT_FALSE(bus->RemoveMatch(
"type='signal',interface='org.chromium.TestService',path='/'", &error));
bus->ShutdownAndBlock();
}
TEST(BusTest, ListenForServiceOwnerChange) {
base::test::SingleThreadTaskEnvironment task_environment(
base::test::SingleThreadTaskEnvironment::MainThreadType::IO);
RunLoopWithExpectedCount run_loop_state;
scoped_refptr<Bus> bus = new Bus(Bus::Options());
std::string service_owner1;
int num_of_owner_changes1 = 0;
Bus::ServiceOwnerChangeCallback callback1 =
base::BindRepeating(&OnServiceOwnerChanged, &run_loop_state,
&service_owner1, &num_of_owner_changes1);
bus->ListenForServiceOwnerChange("org.chromium.TestService", callback1);
bus->ListenForServiceOwnerChange("org.chromium.TestService", callback1);
base::RunLoop().RunUntilIdle();
EXPECT_TRUE(service_owner1.empty());
EXPECT_EQ(0, num_of_owner_changes1);
ASSERT_TRUE(bus->RequestOwnershipAndBlock("org.chromium.TestService",
Bus::REQUIRE_PRIMARY));
run_loop_state.Run(1);
{
std::string current_service_owner =
bus->GetServiceOwnerAndBlock("org.chromium.TestService",
Bus::REPORT_ERRORS);
ASSERT_FALSE(current_service_owner.empty());
EXPECT_EQ(current_service_owner, service_owner1);
EXPECT_EQ(1, num_of_owner_changes1);
}
std::string service_owner2;
int num_of_owner_changes2 = 0;
Bus::ServiceOwnerChangeCallback callback2 =
base::BindRepeating(&OnServiceOwnerChanged, &run_loop_state,
&service_owner2, &num_of_owner_changes2);
bus->ListenForServiceOwnerChange("org.chromium.TestService", callback2);
base::RunLoop().RunUntilIdle();
ASSERT_TRUE(bus->ReleaseOwnership("org.chromium.TestService"));
run_loop_state.Run(2);
EXPECT_TRUE(service_owner1.empty());
EXPECT_TRUE(service_owner2.empty());
EXPECT_EQ(2, num_of_owner_changes1);
EXPECT_EQ(1, num_of_owner_changes2);
bus->UnlistenForServiceOwnerChange("org.chromium.TestService", callback1);
bus->UnlistenForServiceOwnerChange("org.chromium.TestService", callback2);
base::RunLoop().RunUntilIdle();
bus->ShutdownAndBlock();
EXPECT_TRUE(bus->shutdown_completed());
}
TEST(BusTest, GetConnectionName) {
scoped_refptr<Bus> bus = new Bus(Bus::Options());
EXPECT_FALSE(bus->IsConnected());
EXPECT_TRUE(bus->GetConnectionName().empty());
bus->Connect();
EXPECT_TRUE(bus->IsConnected());
EXPECT_FALSE(bus->GetConnectionName().empty());
bus->ShutdownAndBlock();
EXPECT_TRUE(bus->shutdown_completed());
}
}