- add sources.
[platform/framework/web/crosswalk.git] / src / dbus / property_unittest.cc
1 // 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"
13 #include "base/message_loop/message_loop.h"
14 #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"
19 #include "dbus/test_service.h"
20 #include "testing/gtest/include/gtest/gtest.h"
21
22 namespace dbus {
23
24 // The property test exerises the asynchronous APIs in PropertySet and
25 // Property<>.
26 class PropertyTest : public testing::Test {
27  public:
28   PropertyTest() {
29   }
30
31   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;
36
37     Properties(ObjectProxy* object_proxy,
38                PropertyChangedCallback property_changed_callback)
39         : PropertySet(object_proxy,
40                       "org.chromium.TestInterface",
41                       property_changed_callback) {
42       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;
56     thread_options.message_loop_type = base::MessageLoop::TYPE_IO;
57     ASSERT_TRUE(dbus_thread_->StartWithOptions(thread_options));
58
59     // Start the test service, using the D-Bus thread.
60     TestService::Options options;
61     options.dbus_task_runner = dbus_thread_->message_loop_proxy();
62     test_service_.reset(new TestService(options));
63     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.
68     Bus::Options bus_options;
69     bus_options.bus_type = Bus::SESSION;
70     bus_options.connection_type = Bus::PRIVATE;
71     bus_options.dbus_task_runner = dbus_thread_->message_loop_proxy();
72     bus_ = new Bus(bus_options);
73     object_proxy_ = bus_->GetObjectProxy(
74         "org.chromium.TestService",
75         ObjectPath("/org/chromium/TestObject"));
76     ASSERT_TRUE(bus_->HasDBusThread());
77
78     // Create the properties structure
79     properties_.reset(new Properties(
80         object_proxy_,
81         base::Bind(&PropertyTest::OnPropertyChanged,
82                    base::Unretained(this))));
83     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
116   // Waits for the given number of updates.
117   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
127   // Waits for initial values to be set.
128   void WaitForGetAll() {
129     WaitForUpdates(kExpectedSignalUpdates);
130   }
131
132   // Waits for the callback. |id| is the string bound to the callback when
133   // 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
141   base::MessageLoop message_loop_;
142   scoped_ptr<base::Thread> dbus_thread_;
143   scoped_refptr<Bus> bus_;
144   ObjectProxy* object_proxy_;
145   scoped_ptr<Properties> properties_;
146   scoped_ptr<TestService> test_service_;
147   // Properties updated.
148   std::vector<std::string> updated_properties_;
149   // Last callback received.
150   std::string last_callback_;
151 };
152
153 TEST_F(PropertyTest, InitialValues) {
154   WaitForGetAll();
155
156   EXPECT_EQ("TestService", properties_->name.value());
157   EXPECT_EQ(10, properties_->version.value());
158
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
166   std::vector<ObjectPath> objects = properties_->objects.value();
167   ASSERT_EQ(1U, objects.size());
168   EXPECT_EQ(ObjectPath("/TestObjectPath"), objects[0]);
169 }
170
171 TEST_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
181   EXPECT_EQ("TestService", properties_->name.value());
182
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
190   EXPECT_EQ(20, properties_->version.value());
191
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
215   std::vector<ObjectPath> objects = properties_->objects.value();
216   ASSERT_EQ(1U, objects.size());
217   EXPECT_EQ(ObjectPath("/TestObjectPath"), objects[0]);
218 }
219
220 TEST_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
232   EXPECT_EQ(20, properties_->version.value());
233 }
234
235 TEST_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
248   EXPECT_EQ("NewService", properties_->name.value());
249 }
250
251 }  // namespace dbus