1 /* Copyright (c) 2012 The Chromium OS 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 
6 #ifndef CRAS_DBUS_TEST_H_
7 #define CRAS_DBUS_TEST_H_
8 
9 #include <string>
10 #include <vector>
11 
12 #include <stdint.h>
13 #include <dbus/dbus.h>
14 #include <gtest/gtest.h>
15 #include <pthread.h>
16 
17 /* DBusTest, and the related DBusMatch class, are used to provide a
18  * GMock-like experience for testing D-Bus code within cras.
19  *
20  * It works by providing a connection to a private D-Bus Server for use
21  * by code you intend to test. Before making calls, you set expectations
22  * of method calls that the server should receive and reply to, or
23  * instructions for the server to send signals that your connection
24  * should receive and handle.
25  *
26  * The code style is similar to GMock for purposes of familiarity.
27  *
28  * Examples
29  * --------
30  *
31  * To create a test suite class implementing a SetUp and TearDown method,
32  * be sure to call the base-class methods at the appropriate time.
33  *
34  *   class ExampleTestSuite : public DBusTest {
35  *     virtual void SetUp() {
36  *       DBusTest::SetUp();
37  *       // your setup code here
38  *     }
39  *
40  *     virtual void TearDown() {
41  *       // your teardown code here
42  *       DBusTest::TearDown();
43  *     }
44  *   };
45  *
46  * To expect a method call to be made against the server; matching the
47  * object path, interface and method name and then generating an empty
48  * reply. The test code ensures that the reply is received during the
49  * TearDown method.
50  *
51  *   TEST_F(ExampleTestSuite, ExampleTest) {
52  *     ExpectMethodCall("/object/path", "object.Interface", "MethodName")
53  *         .SendReply();
54  *
55  *     // code to generate the method call here
56  *   }
57  *
58  * Due to the asynchronous nature of D-Bus, if you need to check some
59  * state, it's not enough to immediately follow the code that generates
60  * the method call. You must instead ensure that all expectations up to
61  * that point have been met:
62  *
63  *   TEST_F(ExampleTestSuite, ExampleTest) {
64  *     ExpectMethodCall("/object/path", "object.Interface", "MethodName")
65  *         .SendReply();
66  *
67  *     // code to generate the method call here
68  *
69  *     WaitForMatches();
70  *
71  *     // code to examine state here
72  *   }
73  *
74  * To verify the arguments to method calls, place .With*() calls before
75  * sending the reply:
76  *
77  *   ExpectMethodCall("/object/path", "object.Interface", "MethodName")
78  *       .WithObjectPath("/arg0/object/path")
79  *       .WithString("arg1")
80  *       .WithString("arg2")
81  *       .SendReply();
82  *
83  * Normally additional arguments are permitted, since most D-Bus services
84  * don't go out of their way to check they aren't provided; to verify
85  * there are no more arguments use .WithNoMoreArgs():
86  *
87  *   ExpectMethodCall("/object/path", "object.Interface", "MethodName")
88  *       .WithString("arg0")
89  *       .WithNoMoreArgs()
90  *       .SendReply();
91  *
92  * To append arguments to the reply, place .With*() calls after the
93  * instruction to send the reply:
94  *
95  *   ExpectMethodCall("/object/path", "object.Interface", "MethodName")
96  *       .SendReply()
97  *       .WithString("arg0")
98  *       .WithObjectPath("/arg1/object/path");
99  *
100  * Property dictionaries are sufficiently difficult to deal with that
101  * there is special handling for them; to append one to the reply use
102  * .AsPropertyDictionary() and follow with alternate .WithString() and
103  * other .With*() calls for each property:
104  *
105  *  ExpectMethodCall("/object/path", "object.Interface", "GetProperties")
106  *       .SendReply()
107  *       .AsPropertyDictionary()
108  *       .WithString("Keyword")
109  *       .WithObjectPath("/value/of/keyword");
110  *
111  * To reply with an error use .SendError() instead of .SendReply(),
112  * passing the error name and message
113  *
114  *   ExpectMethodCall("/object/path", "object.Interface", "MethodName")
115  *       .SendError("some.error.Name", "Message for error");
116  *
117  * In some cases (notably "AddMatch" method calls) the method call will
118  * be handled by libdbus itself and the mechanism DBusTest uses to verify
119  * that the reply is recieved does not work. In which case you need to use
120  * .SendReplyNoWait() instead.
121  *
122  *    ExpectMethodCall("", DBUS_INTERFACE_DBUS, "AddMatch")
123  *        .SendReplyNoWait();
124  *
125  * Sending signals from the server side is very similar:
126  *
127  *    CreateSignal("/object/path", "object.Interface", "SignalName")
128  *        .WithString("arg0")
129  *        .WithObjectPat("/arg1/object/path")
130  *        .Send();
131  *
132  * Create messages from server side:
133  *    CreateMessageCall("/object/path". "object.Interface", "MethodName")
134  *        .WithString("arg0")
135  *        .WithUnixFd(arg1)
136  *        .Send();
137  *
138  * The TearDown() method will verify that it is received by the client,
139  * use WaitForMatches() to force verification earlier in order to check
140  * state.
141  */
142 
143 class DBusTest;
144 
145 class DBusMatch {
146  public:
147   DBusMatch();
148 
149   struct Arg {
150     int type;
151     bool array;
152     std::string string_value;
153     int int_value;
154     std::vector<std::string> string_values;
155   };
156 
157   // Append arguments to a match.
158   DBusMatch& WithString(std::string value);
159   DBusMatch& WithUnixFd(int value);
160   DBusMatch& WithObjectPath(std::string value);
161   DBusMatch& WithArrayOfStrings(std::vector<std::string> values);
162   DBusMatch& WithArrayOfObjectPaths(std::vector<std::string> values);
163   DBusMatch& WithNoMoreArgs();
164 
165   // Indicates that all arguments in either the method call or reply
166   // should be wrapped into a property dictionary with a string for keys
167   // and a variant for the data.
168   DBusMatch& AsPropertyDictionary();
169 
170   // Send a reply to a method call and wait for it to be received by the
171   // client; may be followed by methods to append arguments.
172   DBusMatch& SendReply();
173 
174   // Send an error in reply to a method call and wait for it to be received
175   // by the client; may also be followed by methods to append arguments.
176   DBusMatch& SendError(std::string error_name, std::string error_message);
177 
178   // Send a reply to a method call but do not wait for it to be received;
179   // mostly needed for internal D-Bus messages.
180   DBusMatch& SendReplyNoWait();
181 
182   // Send a created signal.
183   DBusMatch& Send();
184 
185  private:
186   friend class DBusTest;
187 
188   // Methods used by DBusTest after constructing the DBusMatch instance
189   // to set the type of match.
190   void ExpectMethodCall(std::string path, std::string interface,
191                         std::string method);
192 
193   void CreateSignal(DBusConnection *conn,
194                     std::string path, std::string interface,
195                     std::string signal_name);
196 
197   void CreateMessageCall(DBusConnection *conn,
198                          std::string path, std::string interface,
199                          std::string signal_name);
200 
201   // Determine whether a message matches a set of arguments.
202   bool MatchMessageArgs(DBusMessage *message, std::vector<Arg> *args);
203 
204   // Append a set of arguments to a message.
205   void AppendArgsToMessage(DBusMessage *message, std::vector<Arg> *args);
206 
207   // Send a message on a connection.
208   void SendMessage(DBusConnection *conn, DBusMessage *message);
209 
210   // Handle a message received by the server connection.
211   bool HandleServerMessage(DBusConnection *conn, DBusMessage *message);
212 
213   // Handle a message received by the client connection.
214   bool HandleClientMessage(DBusConnection *conn, DBusMessage *message);
215 
216   // Verify whether the match is complete.
217   bool Complete();
218 
219   int message_type_;
220   std::string path_;
221   std::string interface_;
222   std::string member_;
223 
224   bool as_property_dictionary_;
225   std::vector<Arg> args_;
226 
227   DBusConnection *conn_;
228 
229   bool send_reply_;
230   std::vector<Arg> reply_args_;
231 
232   bool send_error_;
233   std::string error_name_;
234   std::string error_message_;
235 
236   bool expect_serial_;
237   std::vector<dbus_uint32_t> expected_serials_;
238 
239   bool matched_;
240 };
241 
242 class DBusTest : public ::testing::Test {
243  public:
244   DBusTest();
245   virtual ~DBusTest();
246 
247  protected:
248   // Connection to the D-Bus server, this may be used during tests as the
249   // "bus" connection, all messages go to and from the internal D-Bus server.
250   DBusConnection *conn_;
251 
252   // Expect a method call to be received by the server.
253   DBusMatch& ExpectMethodCall(std::string path, std::string interface,
254                               std::string method);
255 
256   // Send a signal from the client to the server.
257   DBusMatch& CreateSignal(std::string path, std::string interface,
258                           std::string signal_name);
259 
260   // Send a message from the client to the server.
261   DBusMatch& CreateMessageCall(std::string path, std::string interface,
262                                std::string signal_name);
263 
264   // Wait for all matches created by Expect*() or Create*() methods to
265   // be complete.
266   void WaitForMatches();
267 
268   // When overriding be sure to call these parent methods to allow the
269   // D-Bus server thread to be cleanly initialized and shut down.
270   virtual void SetUp();
271   virtual void TearDown();
272 
273  private:
274   DBusServer *server_;
275   DBusConnection *server_conn_;
276 
277   std::vector<DBusWatch *> watches_;
278   std::vector<DBusTimeout *> timeouts_;
279 
280   pthread_t thread_id_;
281   pthread_mutex_t mutex_;
282   bool dispatch_;
283 
284   std::vector<DBusMatch> matches_;
285 
286   static void NewConnectionThunk(DBusServer *server, DBusConnection *conn,
287                                  void *data);
288   void NewConnection(DBusServer *server, DBusConnection *conn);
289 
290   static dbus_bool_t AddWatchThunk(DBusWatch *watch, void *data);
291   dbus_bool_t AddWatch(DBusWatch *watch);
292 
293   static void RemoveWatchThunk(DBusWatch *watch, void *data);
294   void RemoveWatch(DBusWatch *watch);
295 
296   static void WatchToggledThunk(DBusWatch *watch, void *data);
297   void WatchToggled(DBusWatch *watch);
298 
299   static dbus_bool_t AddTimeoutThunk(DBusTimeout *timeout, void *data);
300   dbus_bool_t AddTimeout(DBusTimeout *timeout);
301 
302   static void RemoveTimeoutThunk(DBusTimeout *timeout, void *data);
303   void RemoveTimeout(DBusTimeout *timeout);
304 
305   static void TimeoutToggledThunk(DBusTimeout *timeout, void *data);
306   void TimeoutToggled(DBusTimeout *timeout);
307 
308   static DBusHandlerResult HandleMessageThunk(DBusConnection *conn,
309                                               DBusMessage *message, void *data);
310   DBusHandlerResult HandleMessage(DBusConnection *conn, DBusMessage *message);
311 
312   static void *DispatchLoopThunk(void *ptr);
313   void *DispatchLoop();
314   void DispatchOnce();
315 };
316 
317 
318 #endif /* CRAS_DBUS_TEST_H_ */
319