blob: cbff1dcfc0daf39600bbfc5cfcb45b6b1ae81220 [file] [log] [blame]
keybuk@chromium.org7e0c4932012-02-15 13:21:08 +09001// Copyright (c) 2012 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/property.h"
6
7#include <string>
8#include <vector>
9
10#include "base/basictypes.h"
11#include "base/bind.h"
12#include "base/logging.h"
avi@chromium.orga29af562013-07-18 08:00:30 +090013#include "base/message_loop/message_loop.h"
keybuk@chromium.org7e0c4932012-02-15 13:21:08 +090014#include "base/threading/thread.h"
15#include "base/threading/thread_restrictions.h"
16#include "dbus/bus.h"
17#include "dbus/object_path.h"
18#include "dbus/object_proxy.h"
keybuk@chromium.org7e0c4932012-02-15 13:21:08 +090019#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.org7e0c4932012-02-15 13:21:08 +090024// The property test exerises the asynchronous APIs in PropertySet and
25// Property<>.
26class PropertyTest : public testing::Test {
27 public:
28 PropertyTest() {
29 }
30
thestig@chromium.orgf0b7eac2013-06-13 15:37:19 +090031 struct Properties : public PropertySet {
32 Property<std::string> name;
33 Property<int16> version;
34 Property<std::vector<std::string> > methods;
35 Property<std::vector<ObjectPath> > objects;
keybuk@chromium.org7e0c4932012-02-15 13:21:08 +090036
thestig@chromium.orgf0b7eac2013-06-13 15:37:19 +090037 Properties(ObjectProxy* object_proxy,
keybuk@chromium.org7e0c4932012-02-15 13:21:08 +090038 PropertyChangedCallback property_changed_callback)
thestig@chromium.orgf0b7eac2013-06-13 15:37:19 +090039 : PropertySet(object_proxy,
40 "org.chromium.TestInterface",
41 property_changed_callback) {
keybuk@chromium.org7e0c4932012-02-15 13:21:08 +090042 RegisterProperty("Name", &name);
43 RegisterProperty("Version", &version);
44 RegisterProperty("Methods", &methods);
45 RegisterProperty("Objects", &objects);
46 }
47 };
48
49 virtual void SetUp() {
50 // Make the main thread not to allow IO.
51 base::ThreadRestrictions::SetIOAllowed(false);
52
53 // Start the D-Bus thread.
54 dbus_thread_.reset(new base::Thread("D-Bus Thread"));
55 base::Thread::Options thread_options;
xhwang@chromium.orgdff6b132013-05-02 01:10:30 +090056 thread_options.message_loop_type = base::MessageLoop::TYPE_IO;
keybuk@chromium.org7e0c4932012-02-15 13:21:08 +090057 ASSERT_TRUE(dbus_thread_->StartWithOptions(thread_options));
58
59 // Start the test service, using the D-Bus thread.
thestig@chromium.orgf0b7eac2013-06-13 15:37:19 +090060 TestService::Options options;
thestig@chromium.org074b1db2013-02-20 10:36:53 +090061 options.dbus_task_runner = dbus_thread_->message_loop_proxy();
thestig@chromium.orgf0b7eac2013-06-13 15:37:19 +090062 test_service_.reset(new TestService(options));
keybuk@chromium.org7e0c4932012-02-15 13:21:08 +090063 ASSERT_TRUE(test_service_->StartService());
64 ASSERT_TRUE(test_service_->WaitUntilServiceIsStarted());
65 ASSERT_TRUE(test_service_->HasDBusThread());
66
67 // Create the client, using the D-Bus thread.
thestig@chromium.orgf0b7eac2013-06-13 15:37:19 +090068 Bus::Options bus_options;
69 bus_options.bus_type = Bus::SESSION;
70 bus_options.connection_type = Bus::PRIVATE;
thestig@chromium.org074b1db2013-02-20 10:36:53 +090071 bus_options.dbus_task_runner = dbus_thread_->message_loop_proxy();
thestig@chromium.orgf0b7eac2013-06-13 15:37:19 +090072 bus_ = new Bus(bus_options);
keybuk@chromium.org7e0c4932012-02-15 13:21:08 +090073 object_proxy_ = bus_->GetObjectProxy(
74 "org.chromium.TestService",
thestig@chromium.orgf0b7eac2013-06-13 15:37:19 +090075 ObjectPath("/org/chromium/TestObject"));
keybuk@chromium.org7e0c4932012-02-15 13:21:08 +090076 ASSERT_TRUE(bus_->HasDBusThread());
77
78 // Create the properties structure
satorux@chromium.org5e612da2012-05-31 15:55:53 +090079 properties_.reset(new Properties(
80 object_proxy_,
81 base::Bind(&PropertyTest::OnPropertyChanged,
82 base::Unretained(this))));
keybuk@chromium.org7e0c4932012-02-15 13:21:08 +090083 properties_->ConnectSignals();
84 properties_->GetAll();
85 }
86
87 virtual void TearDown() {
88 bus_->ShutdownOnDBusThreadAndBlock();
89
90 // Shut down the service.
91 test_service_->ShutdownAndBlock();
92
93 // Reset to the default.
94 base::ThreadRestrictions::SetIOAllowed(true);
95
96 // Stopping a thread is considered an IO operation, so do this after
97 // allowing IO.
98 test_service_->Stop();
99 }
100
101 // Generic callback, bind with a string |id| for passing to
102 // WaitForCallback() to ensure the callback for the right method is
103 // waited for.
104 void PropertyCallback(const std::string& id, bool success) {
105 last_callback_ = id;
106 message_loop_.Quit();
107 }
108
109 protected:
110 // Called when a property value is updated.
111 void OnPropertyChanged(const std::string& name) {
112 updated_properties_.push_back(name);
113 message_loop_.Quit();
114 }
115
keybuk@chromium.org683dd8c2012-03-23 05:34:05 +0900116 // Waits for the given number of updates.
keybuk@chromium.org7e0c4932012-02-15 13:21:08 +0900117 void WaitForUpdates(size_t num_updates) {
118 while (updated_properties_.size() < num_updates)
119 message_loop_.Run();
120 for (size_t i = 0; i < num_updates; ++i)
121 updated_properties_.erase(updated_properties_.begin());
122 }
123
124 // Name, Version, Methods, Objects
125 static const int kExpectedSignalUpdates = 4;
126
keybuk@chromium.org683dd8c2012-03-23 05:34:05 +0900127 // Waits for initial values to be set.
keybuk@chromium.org7e0c4932012-02-15 13:21:08 +0900128 void WaitForGetAll() {
129 WaitForUpdates(kExpectedSignalUpdates);
130 }
131
keybuk@chromium.org683dd8c2012-03-23 05:34:05 +0900132 // Waits for the callback. |id| is the string bound to the callback when
keybuk@chromium.org7e0c4932012-02-15 13:21:08 +0900133 // the method call is made that identifies it and distinguishes from any
134 // other; you can set this to whatever you wish.
135 void WaitForCallback(const std::string& id) {
136 while (last_callback_ != id) {
137 message_loop_.Run();
138 }
139 }
140
xhwang@chromium.orgdff6b132013-05-02 01:10:30 +0900141 base::MessageLoop message_loop_;
keybuk@chromium.org7e0c4932012-02-15 13:21:08 +0900142 scoped_ptr<base::Thread> dbus_thread_;
thestig@chromium.orgf0b7eac2013-06-13 15:37:19 +0900143 scoped_refptr<Bus> bus_;
144 ObjectProxy* object_proxy_;
satorux@chromium.org5e612da2012-05-31 15:55:53 +0900145 scoped_ptr<Properties> properties_;
thestig@chromium.orgf0b7eac2013-06-13 15:37:19 +0900146 scoped_ptr<TestService> test_service_;
keybuk@chromium.org7e0c4932012-02-15 13:21:08 +0900147 // Properties updated.
148 std::vector<std::string> updated_properties_;
149 // Last callback received.
150 std::string last_callback_;
151};
152
153TEST_F(PropertyTest, InitialValues) {
154 WaitForGetAll();
155
keybuk@chromium.org683dd8c2012-03-23 05:34:05 +0900156 EXPECT_EQ("TestService", properties_->name.value());
157 EXPECT_EQ(10, properties_->version.value());
keybuk@chromium.org7e0c4932012-02-15 13:21:08 +0900158
159 std::vector<std::string> methods = properties_->methods.value();
160 ASSERT_EQ(4U, methods.size());
161 EXPECT_EQ("Echo", methods[0]);
162 EXPECT_EQ("SlowEcho", methods[1]);
163 EXPECT_EQ("AsyncEcho", methods[2]);
164 EXPECT_EQ("BrokenMethod", methods[3]);
165
thestig@chromium.orgf0b7eac2013-06-13 15:37:19 +0900166 std::vector<ObjectPath> objects = properties_->objects.value();
keybuk@chromium.org7e0c4932012-02-15 13:21:08 +0900167 ASSERT_EQ(1U, objects.size());
thestig@chromium.orgf0b7eac2013-06-13 15:37:19 +0900168 EXPECT_EQ(ObjectPath("/TestObjectPath"), objects[0]);
keybuk@chromium.org7e0c4932012-02-15 13:21:08 +0900169}
170
keybuk@google.comf056b962012-03-22 08:43:45 +0900171TEST_F(PropertyTest, UpdatedValues) {
172 WaitForGetAll();
173
174 // Update the value of the "Name" property, this value should not change.
175 properties_->name.Get(base::Bind(&PropertyTest::PropertyCallback,
176 base::Unretained(this),
177 "Name"));
178 WaitForCallback("Name");
179 WaitForUpdates(1);
180
keybuk@chromium.org683dd8c2012-03-23 05:34:05 +0900181 EXPECT_EQ("TestService", properties_->name.value());
keybuk@google.comf056b962012-03-22 08:43:45 +0900182
183 // Update the value of the "Version" property, this value should be changed.
184 properties_->version.Get(base::Bind(&PropertyTest::PropertyCallback,
185 base::Unretained(this),
186 "Version"));
187 WaitForCallback("Version");
188 WaitForUpdates(1);
189
keybuk@chromium.org683dd8c2012-03-23 05:34:05 +0900190 EXPECT_EQ(20, properties_->version.value());
keybuk@google.comf056b962012-03-22 08:43:45 +0900191
192 // Update the value of the "Methods" property, this value should not change
193 // and should not grow to contain duplicate entries.
194 properties_->methods.Get(base::Bind(&PropertyTest::PropertyCallback,
195 base::Unretained(this),
196 "Methods"));
197 WaitForCallback("Methods");
198 WaitForUpdates(1);
199
200 std::vector<std::string> methods = properties_->methods.value();
201 ASSERT_EQ(4U, methods.size());
202 EXPECT_EQ("Echo", methods[0]);
203 EXPECT_EQ("SlowEcho", methods[1]);
204 EXPECT_EQ("AsyncEcho", methods[2]);
205 EXPECT_EQ("BrokenMethod", methods[3]);
206
207 // Update the value of the "Objects" property, this value should not change
208 // and should not grow to contain duplicate entries.
209 properties_->objects.Get(base::Bind(&PropertyTest::PropertyCallback,
210 base::Unretained(this),
211 "Objects"));
212 WaitForCallback("Objects");
213 WaitForUpdates(1);
214
thestig@chromium.orgf0b7eac2013-06-13 15:37:19 +0900215 std::vector<ObjectPath> objects = properties_->objects.value();
keybuk@google.comf056b962012-03-22 08:43:45 +0900216 ASSERT_EQ(1U, objects.size());
thestig@chromium.orgf0b7eac2013-06-13 15:37:19 +0900217 EXPECT_EQ(ObjectPath("/TestObjectPath"), objects[0]);
keybuk@google.comf056b962012-03-22 08:43:45 +0900218}
219
keybuk@chromium.org7e0c4932012-02-15 13:21:08 +0900220TEST_F(PropertyTest, Get) {
221 WaitForGetAll();
222
223 // Ask for the new Version property.
224 properties_->version.Get(base::Bind(&PropertyTest::PropertyCallback,
225 base::Unretained(this),
226 "Get"));
227 WaitForCallback("Get");
228
229 // Make sure we got a property update too.
230 WaitForUpdates(1);
231
keybuk@chromium.org683dd8c2012-03-23 05:34:05 +0900232 EXPECT_EQ(20, properties_->version.value());
keybuk@chromium.org7e0c4932012-02-15 13:21:08 +0900233}
234
235TEST_F(PropertyTest, Set) {
236 WaitForGetAll();
237
238 // Set a new name.
239 properties_->name.Set("NewService",
240 base::Bind(&PropertyTest::PropertyCallback,
241 base::Unretained(this),
242 "Set"));
243 WaitForCallback("Set");
244
245 // TestService sends a property update.
246 WaitForUpdates(1);
247
keybuk@chromium.org683dd8c2012-03-23 05:34:05 +0900248 EXPECT_EQ("NewService", properties_->name.value());
keybuk@chromium.org7e0c4932012-02-15 13:21:08 +0900249}
thestig@chromium.orgf0b7eac2013-06-13 15:37:19 +0900250
251} // namespace dbus