blob: 3e53095b12e4fefd7add9d87d6b31caec4b8848a [file] [log] [blame]
keybuk@chromium.org09715012013-03-26 03:20:08 +09001// Copyright (c) 2013 The Chromium Authors. All rights reserved.
2// Use of this source code is governed by a BSD-style license that can be
3// found in the LICENSE file.
4
5#include "dbus/object_manager.h"
6
7#include <string>
8#include <vector>
9
10#include "base/basictypes.h"
11#include "base/bind.h"
avi@chromium.orga29af562013-07-18 08:00:30 +090012#include "base/message_loop/message_loop.h"
keybuk@chromium.org09715012013-03-26 03:20:08 +090013#include "base/threading/thread.h"
14#include "base/threading/thread_restrictions.h"
15#include "dbus/bus.h"
16#include "dbus/object_path.h"
17#include "dbus/object_proxy.h"
18#include "dbus/property.h"
19#include "dbus/test_service.h"
20#include "testing/gtest/include/gtest/gtest.h"
21
thestig@chromium.orgf0b7eac2013-06-13 15:37:19 +090022namespace dbus {
23
keybuk@chromium.org09715012013-03-26 03:20:08 +090024// The object manager test exercises the asynchronous APIs in ObjectManager,
25// and by extension PropertySet and Property<>.
26class ObjectManagerTest
27 : public testing::Test,
thestig@chromium.orgf0b7eac2013-06-13 15:37:19 +090028 public ObjectManager::Interface {
keybuk@chromium.org09715012013-03-26 03:20:08 +090029 public:
30 ObjectManagerTest() {
31 }
32
thestig@chromium.orgf0b7eac2013-06-13 15:37:19 +090033 struct Properties : public PropertySet {
34 Property<std::string> name;
35 Property<int16> version;
36 Property<std::vector<std::string> > methods;
37 Property<std::vector<ObjectPath> > objects;
keybuk@chromium.org09715012013-03-26 03:20:08 +090038
thestig@chromium.orgf0b7eac2013-06-13 15:37:19 +090039 Properties(ObjectProxy* object_proxy,
keybuk@chromium.org09715012013-03-26 03:20:08 +090040 const std::string& interface_name,
41 PropertyChangedCallback property_changed_callback)
thestig@chromium.orgf0b7eac2013-06-13 15:37:19 +090042 : PropertySet(object_proxy, interface_name, property_changed_callback) {
keybuk@chromium.org09715012013-03-26 03:20:08 +090043 RegisterProperty("Name", &name);
44 RegisterProperty("Version", &version);
45 RegisterProperty("Methods", &methods);
46 RegisterProperty("Objects", &objects);
47 }
48 };
49
thestig@chromium.orgf0b7eac2013-06-13 15:37:19 +090050 virtual PropertySet* CreateProperties(
51 ObjectProxy* object_proxy,
52 const ObjectPath& object_path,
keybuk@chromium.org09715012013-03-26 03:20:08 +090053 const std::string& interface_name) OVERRIDE {
54 Properties* properties = new Properties(
55 object_proxy, interface_name,
56 base::Bind(&ObjectManagerTest::OnPropertyChanged,
57 base::Unretained(this), object_path));
thestig@chromium.orgf0b7eac2013-06-13 15:37:19 +090058 return static_cast<PropertySet*>(properties);
keybuk@chromium.org09715012013-03-26 03:20:08 +090059 }
60
61 virtual void SetUp() {
62 // Make the main thread not to allow IO.
63 base::ThreadRestrictions::SetIOAllowed(false);
64
65 // Start the D-Bus thread.
66 dbus_thread_.reset(new base::Thread("D-Bus Thread"));
67 base::Thread::Options thread_options;
xhwang@chromium.orgdff6b132013-05-02 01:10:30 +090068 thread_options.message_loop_type = base::MessageLoop::TYPE_IO;
keybuk@chromium.org09715012013-03-26 03:20:08 +090069 ASSERT_TRUE(dbus_thread_->StartWithOptions(thread_options));
70
71 // Start the test service, using the D-Bus thread.
thestig@chromium.orgf0b7eac2013-06-13 15:37:19 +090072 TestService::Options options;
keybuk@chromium.org09715012013-03-26 03:20:08 +090073 options.dbus_task_runner = dbus_thread_->message_loop_proxy();
thestig@chromium.orgf0b7eac2013-06-13 15:37:19 +090074 test_service_.reset(new TestService(options));
keybuk@chromium.org09715012013-03-26 03:20:08 +090075 ASSERT_TRUE(test_service_->StartService());
76 ASSERT_TRUE(test_service_->WaitUntilServiceIsStarted());
77 ASSERT_TRUE(test_service_->HasDBusThread());
78
79 // Create the client, using the D-Bus thread.
thestig@chromium.orgf0b7eac2013-06-13 15:37:19 +090080 Bus::Options bus_options;
81 bus_options.bus_type = Bus::SESSION;
82 bus_options.connection_type = Bus::PRIVATE;
keybuk@chromium.org09715012013-03-26 03:20:08 +090083 bus_options.dbus_task_runner = dbus_thread_->message_loop_proxy();
thestig@chromium.orgf0b7eac2013-06-13 15:37:19 +090084 bus_ = new Bus(bus_options);
keybuk@chromium.org09715012013-03-26 03:20:08 +090085 ASSERT_TRUE(bus_->HasDBusThread());
86
87 object_manager_ = bus_->GetObjectManager(
88 "org.chromium.TestService",
thestig@chromium.orgf0b7eac2013-06-13 15:37:19 +090089 ObjectPath("/org/chromium/TestService"));
keybuk@chromium.org09715012013-03-26 03:20:08 +090090 object_manager_->RegisterInterface("org.chromium.TestInterface", this);
91
92 object_manager_->GetManagedObjects();
93 WaitForObject();
94 }
95
96 virtual void TearDown() {
97 bus_->ShutdownOnDBusThreadAndBlock();
98
99 // Shut down the service.
100 test_service_->ShutdownAndBlock();
101
102 // Reset to the default.
103 base::ThreadRestrictions::SetIOAllowed(true);
104
105 // Stopping a thread is considered an IO operation, so do this after
106 // allowing IO.
107 test_service_->Stop();
108 }
109
thestig@chromium.orgf0b7eac2013-06-13 15:37:19 +0900110 void MethodCallback(Response* response) {
keybuk@chromium.org09715012013-03-26 03:20:08 +0900111 method_callback_called_ = true;
112 message_loop_.Quit();
113 }
114
thestig@chromium.orgf0b7eac2013-06-13 15:37:19 +0900115 protected:
keybuk@chromium.org09715012013-03-26 03:20:08 +0900116 // Called when an object is added.
thestig@chromium.orgf0b7eac2013-06-13 15:37:19 +0900117 virtual void ObjectAdded(const ObjectPath& object_path,
118 const std::string& interface_name) OVERRIDE {
keybuk@chromium.org09715012013-03-26 03:20:08 +0900119 added_objects_.push_back(std::make_pair(object_path, interface_name));
120 message_loop_.Quit();
121 }
122
123 // Called when an object is removed.
thestig@chromium.orgf0b7eac2013-06-13 15:37:19 +0900124 virtual void ObjectRemoved(const ObjectPath& object_path,
rsleevi@chromium.org0a2b25e2013-05-02 11:24:12 +0900125 const std::string& interface_name) OVERRIDE {
keybuk@chromium.org09715012013-03-26 03:20:08 +0900126 removed_objects_.push_back(std::make_pair(object_path, interface_name));
127 message_loop_.Quit();
128 }
129
130 // Called when a property value is updated.
thestig@chromium.orgf0b7eac2013-06-13 15:37:19 +0900131 void OnPropertyChanged(const ObjectPath& object_path,
keybuk@chromium.org09715012013-03-26 03:20:08 +0900132 const std::string& name) {
133 updated_properties_.push_back(name);
134 message_loop_.Quit();
135 }
136
137 static const size_t kExpectedObjects = 1;
138 static const size_t kExpectedProperties = 4;
139
140 void WaitForObject() {
141 while (added_objects_.size() < kExpectedObjects ||
142 updated_properties_.size() < kExpectedProperties)
143 message_loop_.Run();
144 for (size_t i = 0; i < kExpectedObjects; ++i)
145 added_objects_.erase(added_objects_.begin());
146 for (size_t i = 0; i < kExpectedProperties; ++i)
147 updated_properties_.erase(updated_properties_.begin());
148 }
149
150 void WaitForRemoveObject() {
151 while (removed_objects_.size() < kExpectedObjects)
152 message_loop_.Run();
153 for (size_t i = 0; i < kExpectedObjects; ++i)
154 removed_objects_.erase(removed_objects_.begin());
155 }
156
157 void WaitForMethodCallback() {
158 message_loop_.Run();
159 method_callback_called_ = false;
160 }
161
thestig@chromium.orgf0b7eac2013-06-13 15:37:19 +0900162 void PerformAction(const std::string& action, const ObjectPath& object_path) {
163 ObjectProxy* object_proxy = bus_->GetObjectProxy(
keybuk@chromium.org09715012013-03-26 03:20:08 +0900164 "org.chromium.TestService",
thestig@chromium.orgf0b7eac2013-06-13 15:37:19 +0900165 ObjectPath("/org/chromium/TestObject"));
keybuk@chromium.org09715012013-03-26 03:20:08 +0900166
thestig@chromium.orgf0b7eac2013-06-13 15:37:19 +0900167 MethodCall method_call("org.chromium.TestInterface", "PerformAction");
168 MessageWriter writer(&method_call);
keybuk@chromium.org09715012013-03-26 03:20:08 +0900169 writer.AppendString(action);
170 writer.AppendObjectPath(object_path);
171
172 object_proxy->CallMethod(&method_call,
thestig@chromium.orgf0b7eac2013-06-13 15:37:19 +0900173 ObjectProxy::TIMEOUT_USE_DEFAULT,
keybuk@chromium.org09715012013-03-26 03:20:08 +0900174 base::Bind(&ObjectManagerTest::MethodCallback,
175 base::Unretained(this)));
176 WaitForMethodCallback();
177 }
178
xhwang@chromium.orgdff6b132013-05-02 01:10:30 +0900179 base::MessageLoop message_loop_;
keybuk@chromium.org09715012013-03-26 03:20:08 +0900180 scoped_ptr<base::Thread> dbus_thread_;
thestig@chromium.orgf0b7eac2013-06-13 15:37:19 +0900181 scoped_refptr<Bus> bus_;
182 ObjectManager* object_manager_;
183 scoped_ptr<TestService> test_service_;
keybuk@chromium.org09715012013-03-26 03:20:08 +0900184
thestig@chromium.orgf0b7eac2013-06-13 15:37:19 +0900185 std::vector<std::pair<ObjectPath, std::string> > added_objects_;
186 std::vector<std::pair<ObjectPath, std::string> > removed_objects_;
keybuk@chromium.org09715012013-03-26 03:20:08 +0900187 std::vector<std::string> updated_properties_;
188
189 bool method_callback_called_;
190};
191
192
193TEST_F(ObjectManagerTest, InitialObject) {
thestig@chromium.orgf0b7eac2013-06-13 15:37:19 +0900194 ObjectProxy* object_proxy = object_manager_->GetObjectProxy(
195 ObjectPath("/org/chromium/TestObject"));
keybuk@chromium.org09715012013-03-26 03:20:08 +0900196 EXPECT_TRUE(object_proxy != NULL);
197
198 Properties* properties = static_cast<Properties*>(
thestig@chromium.orgf0b7eac2013-06-13 15:37:19 +0900199 object_manager_->GetProperties(ObjectPath("/org/chromium/TestObject"),
200 "org.chromium.TestInterface"));
keybuk@chromium.org09715012013-03-26 03:20:08 +0900201 EXPECT_TRUE(properties != NULL);
202
203 EXPECT_EQ("TestService", properties->name.value());
204 EXPECT_EQ(10, properties->version.value());
205
206 std::vector<std::string> methods = properties->methods.value();
207 ASSERT_EQ(4U, methods.size());
208 EXPECT_EQ("Echo", methods[0]);
209 EXPECT_EQ("SlowEcho", methods[1]);
210 EXPECT_EQ("AsyncEcho", methods[2]);
211 EXPECT_EQ("BrokenMethod", methods[3]);
212
thestig@chromium.orgf0b7eac2013-06-13 15:37:19 +0900213 std::vector<ObjectPath> objects = properties->objects.value();
keybuk@chromium.org09715012013-03-26 03:20:08 +0900214 ASSERT_EQ(1U, objects.size());
thestig@chromium.orgf0b7eac2013-06-13 15:37:19 +0900215 EXPECT_EQ(ObjectPath("/TestObjectPath"), objects[0]);
keybuk@chromium.org09715012013-03-26 03:20:08 +0900216}
217
218TEST_F(ObjectManagerTest, UnknownObjectProxy) {
thestig@chromium.orgf0b7eac2013-06-13 15:37:19 +0900219 ObjectProxy* object_proxy = object_manager_->GetObjectProxy(
220 ObjectPath("/org/chromium/UnknownObject"));
keybuk@chromium.org09715012013-03-26 03:20:08 +0900221 EXPECT_TRUE(object_proxy == NULL);
222}
223
224TEST_F(ObjectManagerTest, UnknownObjectProperties) {
225 Properties* properties = static_cast<Properties*>(
thestig@chromium.orgf0b7eac2013-06-13 15:37:19 +0900226 object_manager_->GetProperties(ObjectPath("/org/chromium/UnknownObject"),
227 "org.chromium.TestInterface"));
keybuk@chromium.org09715012013-03-26 03:20:08 +0900228 EXPECT_TRUE(properties == NULL);
229}
230
231TEST_F(ObjectManagerTest, UnknownInterfaceProperties) {
232 Properties* properties = static_cast<Properties*>(
thestig@chromium.orgf0b7eac2013-06-13 15:37:19 +0900233 object_manager_->GetProperties(ObjectPath("/org/chromium/TestObject"),
234 "org.chromium.UnknownService"));
keybuk@chromium.org09715012013-03-26 03:20:08 +0900235 EXPECT_TRUE(properties == NULL);
236}
237
238TEST_F(ObjectManagerTest, GetObjects) {
thestig@chromium.orgf0b7eac2013-06-13 15:37:19 +0900239 std::vector<ObjectPath> object_paths = object_manager_->GetObjects();
keybuk@chromium.org09715012013-03-26 03:20:08 +0900240 ASSERT_EQ(1U, object_paths.size());
thestig@chromium.orgf0b7eac2013-06-13 15:37:19 +0900241 EXPECT_EQ(ObjectPath("/org/chromium/TestObject"), object_paths[0]);
keybuk@chromium.org09715012013-03-26 03:20:08 +0900242}
243
244TEST_F(ObjectManagerTest, GetObjectsWithInterface) {
thestig@chromium.orgf0b7eac2013-06-13 15:37:19 +0900245 std::vector<ObjectPath> object_paths =
keybuk@chromium.org09715012013-03-26 03:20:08 +0900246 object_manager_->GetObjectsWithInterface("org.chromium.TestInterface");
247 ASSERT_EQ(1U, object_paths.size());
thestig@chromium.orgf0b7eac2013-06-13 15:37:19 +0900248 EXPECT_EQ(ObjectPath("/org/chromium/TestObject"), object_paths[0]);
keybuk@chromium.org09715012013-03-26 03:20:08 +0900249}
250
251TEST_F(ObjectManagerTest, GetObjectsWithUnknownInterface) {
thestig@chromium.orgf0b7eac2013-06-13 15:37:19 +0900252 std::vector<ObjectPath> object_paths =
keybuk@chromium.org09715012013-03-26 03:20:08 +0900253 object_manager_->GetObjectsWithInterface("org.chromium.UnknownService");
254 EXPECT_EQ(0U, object_paths.size());
255}
256
257TEST_F(ObjectManagerTest, SameObject) {
thestig@chromium.orgf0b7eac2013-06-13 15:37:19 +0900258 ObjectManager* object_manager = bus_->GetObjectManager(
keybuk@chromium.org09715012013-03-26 03:20:08 +0900259 "org.chromium.TestService",
thestig@chromium.orgf0b7eac2013-06-13 15:37:19 +0900260 ObjectPath("/org/chromium/TestService"));
keybuk@chromium.org09715012013-03-26 03:20:08 +0900261 EXPECT_EQ(object_manager_, object_manager);
262}
263
264TEST_F(ObjectManagerTest, DifferentObjectForService) {
thestig@chromium.orgf0b7eac2013-06-13 15:37:19 +0900265 ObjectManager* object_manager = bus_->GetObjectManager(
keybuk@chromium.org09715012013-03-26 03:20:08 +0900266 "org.chromium.DifferentService",
thestig@chromium.orgf0b7eac2013-06-13 15:37:19 +0900267 ObjectPath("/org/chromium/TestService"));
keybuk@chromium.org09715012013-03-26 03:20:08 +0900268 EXPECT_NE(object_manager_, object_manager);
269}
270
271TEST_F(ObjectManagerTest, DifferentObjectForPath) {
thestig@chromium.orgf0b7eac2013-06-13 15:37:19 +0900272 ObjectManager* object_manager = bus_->GetObjectManager(
keybuk@chromium.org09715012013-03-26 03:20:08 +0900273 "org.chromium.TestService",
thestig@chromium.orgf0b7eac2013-06-13 15:37:19 +0900274 ObjectPath("/org/chromium/DifferentService"));
keybuk@chromium.org09715012013-03-26 03:20:08 +0900275 EXPECT_NE(object_manager_, object_manager);
276}
277
278TEST_F(ObjectManagerTest, SecondObject) {
thestig@chromium.orgf0b7eac2013-06-13 15:37:19 +0900279 PerformAction("AddObject", ObjectPath("/org/chromium/SecondObject"));
keybuk@chromium.org09715012013-03-26 03:20:08 +0900280 WaitForObject();
281
thestig@chromium.orgf0b7eac2013-06-13 15:37:19 +0900282 ObjectProxy* object_proxy = object_manager_->GetObjectProxy(
283 ObjectPath("/org/chromium/SecondObject"));
keybuk@chromium.org09715012013-03-26 03:20:08 +0900284 EXPECT_TRUE(object_proxy != NULL);
285
286 Properties* properties = static_cast<Properties*>(
thestig@chromium.orgf0b7eac2013-06-13 15:37:19 +0900287 object_manager_->GetProperties(ObjectPath("/org/chromium/SecondObject"),
288 "org.chromium.TestInterface"));
keybuk@chromium.org09715012013-03-26 03:20:08 +0900289 EXPECT_TRUE(properties != NULL);
290
thestig@chromium.orgf0b7eac2013-06-13 15:37:19 +0900291 std::vector<ObjectPath> object_paths = object_manager_->GetObjects();
keybuk@chromium.org09715012013-03-26 03:20:08 +0900292 ASSERT_EQ(2U, object_paths.size());
293
294 std::sort(object_paths.begin(), object_paths.end());
thestig@chromium.orgf0b7eac2013-06-13 15:37:19 +0900295 EXPECT_EQ(ObjectPath("/org/chromium/SecondObject"), object_paths[0]);
296 EXPECT_EQ(ObjectPath("/org/chromium/TestObject"), object_paths[1]);
keybuk@chromium.org09715012013-03-26 03:20:08 +0900297
298 object_paths =
299 object_manager_->GetObjectsWithInterface("org.chromium.TestInterface");
300 ASSERT_EQ(2U, object_paths.size());
301
302 std::sort(object_paths.begin(), object_paths.end());
thestig@chromium.orgf0b7eac2013-06-13 15:37:19 +0900303 EXPECT_EQ(ObjectPath("/org/chromium/SecondObject"), object_paths[0]);
304 EXPECT_EQ(ObjectPath("/org/chromium/TestObject"), object_paths[1]);
keybuk@chromium.org09715012013-03-26 03:20:08 +0900305}
306
307TEST_F(ObjectManagerTest, RemoveSecondObject) {
thestig@chromium.orgf0b7eac2013-06-13 15:37:19 +0900308 PerformAction("AddObject", ObjectPath("/org/chromium/SecondObject"));
keybuk@chromium.org09715012013-03-26 03:20:08 +0900309 WaitForObject();
310
thestig@chromium.orgf0b7eac2013-06-13 15:37:19 +0900311 std::vector<ObjectPath> object_paths = object_manager_->GetObjects();
keybuk@chromium.org09715012013-03-26 03:20:08 +0900312 ASSERT_EQ(2U, object_paths.size());
313
thestig@chromium.orgf0b7eac2013-06-13 15:37:19 +0900314 PerformAction("RemoveObject", ObjectPath("/org/chromium/SecondObject"));
keybuk@chromium.org09715012013-03-26 03:20:08 +0900315 WaitForRemoveObject();
316
thestig@chromium.orgf0b7eac2013-06-13 15:37:19 +0900317 ObjectProxy* object_proxy = object_manager_->GetObjectProxy(
318 ObjectPath("/org/chromium/SecondObject"));
keybuk@chromium.org09715012013-03-26 03:20:08 +0900319 EXPECT_TRUE(object_proxy == NULL);
320
321 Properties* properties = static_cast<Properties*>(
thestig@chromium.orgf0b7eac2013-06-13 15:37:19 +0900322 object_manager_->GetProperties(ObjectPath("/org/chromium/SecondObject"),
323 "org.chromium.TestInterface"));
keybuk@chromium.org09715012013-03-26 03:20:08 +0900324 EXPECT_TRUE(properties == NULL);
325
326 object_paths = object_manager_->GetObjects();
327 ASSERT_EQ(1U, object_paths.size());
thestig@chromium.orgf0b7eac2013-06-13 15:37:19 +0900328 EXPECT_EQ(ObjectPath("/org/chromium/TestObject"), object_paths[0]);
keybuk@chromium.org09715012013-03-26 03:20:08 +0900329
330 object_paths =
331 object_manager_->GetObjectsWithInterface("org.chromium.TestInterface");
332 ASSERT_EQ(1U, object_paths.size());
thestig@chromium.orgf0b7eac2013-06-13 15:37:19 +0900333 EXPECT_EQ(ObjectPath("/org/chromium/TestObject"), object_paths[0]);
keybuk@chromium.org09715012013-03-26 03:20:08 +0900334}
thestig@chromium.orgf0b7eac2013-06-13 15:37:19 +0900335
keybuk@chromium.org40b05ba2014-03-07 11:24:33 +0900336TEST_F(ObjectManagerTest, OwnershipLost) {
337 PerformAction("ReleaseOwnership", ObjectPath("/org/chromium/TestService"));
338 WaitForRemoveObject();
339
340 std::vector<ObjectPath> object_paths = object_manager_->GetObjects();
341 ASSERT_EQ(0U, object_paths.size());
342}
343
344TEST_F(ObjectManagerTest, OwnershipLostAndRegained) {
345 PerformAction("Ownership", ObjectPath("/org/chromium/TestService"));
346 WaitForRemoveObject();
347 WaitForObject();
348
349 std::vector<ObjectPath> object_paths = object_manager_->GetObjects();
350 ASSERT_EQ(1U, object_paths.size());
351}
352
thestig@chromium.orgf0b7eac2013-06-13 15:37:19 +0900353} // namespace dbus