• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 #include <gio/gio.h>
2 #include <stdlib.h>
3 #include <string.h>
4 
5 /* ---------------------------------------------------------------------------------------------------- */
6 
7 static GDBusNodeInfo *introspection_data = NULL;
8 static GDBusInterfaceInfo *manager_interface_info = NULL;
9 static GDBusInterfaceInfo *block_interface_info = NULL;
10 static GDBusInterfaceInfo *partition_interface_info = NULL;
11 
12 /* Introspection data for the service we are exporting */
13 static const gchar introspection_xml[] =
14   "<node>"
15   "  <interface name='org.gtk.GDBus.Example.Manager'>"
16   "    <method name='Hello'>"
17   "      <arg type='s' name='greeting' direction='in'/>"
18   "      <arg type='s' name='response' direction='out'/>"
19   "    </method>"
20   "  </interface>"
21   "  <interface name='org.gtk.GDBus.Example.Block'>"
22   "    <method name='Hello'>"
23   "      <arg type='s' name='greeting' direction='in'/>"
24   "      <arg type='s' name='response' direction='out'/>"
25   "    </method>"
26   "    <property type='i' name='Major' access='read'/>"
27   "    <property type='i' name='Minor' access='read'/>"
28   "    <property type='s' name='Notes' access='readwrite'/>"
29   "  </interface>"
30   "  <interface name='org.gtk.GDBus.Example.Partition'>"
31   "    <method name='Hello'>"
32   "      <arg type='s' name='greeting' direction='in'/>"
33   "      <arg type='s' name='response' direction='out'/>"
34   "    </method>"
35   "    <property type='i' name='PartitionNumber' access='read'/>"
36   "    <property type='s' name='Notes' access='readwrite'/>"
37   "  </interface>"
38   "</node>";
39 
40 /* ---------------------------------------------------------------------------------------------------- */
41 
42 static void
manager_method_call(GDBusConnection * connection,const gchar * sender,const gchar * object_path,const gchar * interface_name,const gchar * method_name,GVariant * parameters,GDBusMethodInvocation * invocation,gpointer user_data)43 manager_method_call (GDBusConnection       *connection,
44                      const gchar           *sender,
45                      const gchar           *object_path,
46                      const gchar           *interface_name,
47                      const gchar           *method_name,
48                      GVariant              *parameters,
49                      GDBusMethodInvocation *invocation,
50                      gpointer               user_data)
51 {
52   const gchar *greeting;
53   gchar *response;
54 
55   g_assert_cmpstr (interface_name, ==, "org.gtk.GDBus.Example.Manager");
56   g_assert_cmpstr (method_name, ==, "Hello");
57 
58   g_variant_get (parameters, "(&s)", &greeting);
59 
60   response = g_strdup_printf ("Method %s.%s with user_data '%s' on object path %s called with arg '%s'",
61                               interface_name,
62                               method_name,
63                               (const gchar *) user_data,
64                               object_path,
65                               greeting);
66   g_dbus_method_invocation_return_value (invocation,
67                                          g_variant_new ("(s)", response));
68   g_free (response);
69 }
70 
71 const GDBusInterfaceVTable manager_vtable =
72 {
73   manager_method_call,
74   NULL,                 /* get_property */
75   NULL                  /* set_property */
76 };
77 
78 /* ---------------------------------------------------------------------------------------------------- */
79 
80 static void
block_method_call(GDBusConnection * connection,const gchar * sender,const gchar * object_path,const gchar * interface_name,const gchar * method_name,GVariant * parameters,GDBusMethodInvocation * invocation,gpointer user_data)81 block_method_call (GDBusConnection       *connection,
82                    const gchar           *sender,
83                    const gchar           *object_path,
84                    const gchar           *interface_name,
85                    const gchar           *method_name,
86                    GVariant              *parameters,
87                    GDBusMethodInvocation *invocation,
88                    gpointer               user_data)
89 {
90   g_assert_cmpstr (interface_name, ==, "org.gtk.GDBus.Example.Block");
91 
92   if (g_strcmp0 (method_name, "Hello") == 0)
93     {
94       const gchar *greeting;
95       gchar *response;
96 
97       g_variant_get (parameters, "(&s)", &greeting);
98 
99       response = g_strdup_printf ("Method %s.%s with user_data '%s' on object path %s called with arg '%s'",
100                                   interface_name,
101                                   method_name,
102                                   (const gchar *) user_data,
103                                   object_path,
104                                   greeting);
105       g_dbus_method_invocation_return_value (invocation,
106                                              g_variant_new ("(s)", response));
107       g_free (response);
108     }
109   else if (g_strcmp0 (method_name, "DoStuff") == 0)
110     {
111       g_dbus_method_invocation_return_dbus_error (invocation,
112                                                   "org.gtk.GDBus.TestSubtree.Error.Failed",
113                                                   "This method intentionally always fails");
114     }
115   else
116     {
117       g_assert_not_reached ();
118     }
119 }
120 
121 static GVariant *
block_get_property(GDBusConnection * connection,const gchar * sender,const gchar * object_path,const gchar * interface_name,const gchar * property_name,GError ** error,gpointer user_data)122 block_get_property (GDBusConnection  *connection,
123                     const gchar      *sender,
124                     const gchar      *object_path,
125                     const gchar      *interface_name,
126                     const gchar      *property_name,
127                     GError          **error,
128                     gpointer          user_data)
129 {
130   GVariant *ret;
131   const gchar *node;
132   gint major;
133   gint minor;
134 
135   node = strrchr (object_path, '/') + 1;
136   if (g_str_has_prefix (node, "sda"))
137     major = 8;
138   else
139     major = 9;
140   if (strlen (node) == 4)
141     minor = node[3] - '0';
142   else
143     minor = 0;
144 
145   ret = NULL;
146   if (g_strcmp0 (property_name, "Major") == 0)
147     {
148       ret = g_variant_new_int32 (major);
149     }
150   else if (g_strcmp0 (property_name, "Minor") == 0)
151     {
152       ret = g_variant_new_int32 (minor);
153     }
154   else if (g_strcmp0 (property_name, "Notes") == 0)
155     {
156       g_set_error (error,
157                    G_IO_ERROR,
158                    G_IO_ERROR_FAILED,
159                    "Hello %s. I thought I said reading this property "
160                    "always results in an error. kthxbye",
161                    sender);
162     }
163   else
164     {
165       g_assert_not_reached ();
166     }
167 
168   return ret;
169 }
170 
171 static gboolean
block_set_property(GDBusConnection * connection,const gchar * sender,const gchar * object_path,const gchar * interface_name,const gchar * property_name,GVariant * value,GError ** error,gpointer user_data)172 block_set_property (GDBusConnection  *connection,
173                     const gchar      *sender,
174                     const gchar      *object_path,
175                     const gchar      *interface_name,
176                     const gchar      *property_name,
177                     GVariant         *value,
178                     GError          **error,
179                     gpointer          user_data)
180 {
181   /* TODO */
182   g_assert_not_reached ();
183 }
184 
185 const GDBusInterfaceVTable block_vtable =
186 {
187   block_method_call,
188   block_get_property,
189   block_set_property,
190 };
191 
192 /* ---------------------------------------------------------------------------------------------------- */
193 
194 static void
partition_method_call(GDBusConnection * connection,const gchar * sender,const gchar * object_path,const gchar * interface_name,const gchar * method_name,GVariant * parameters,GDBusMethodInvocation * invocation,gpointer user_data)195 partition_method_call (GDBusConnection       *connection,
196                        const gchar           *sender,
197                        const gchar           *object_path,
198                        const gchar           *interface_name,
199                        const gchar           *method_name,
200                        GVariant              *parameters,
201                        GDBusMethodInvocation *invocation,
202                        gpointer               user_data)
203 {
204   const gchar *greeting;
205   gchar *response;
206 
207   g_assert_cmpstr (interface_name, ==, "org.gtk.GDBus.Example.Partition");
208   g_assert_cmpstr (method_name, ==, "Hello");
209 
210   g_variant_get (parameters, "(&s)", &greeting);
211 
212   response = g_strdup_printf ("Method %s.%s with user_data '%s' on object path %s called with arg '%s'",
213                               interface_name,
214                               method_name,
215                               (const gchar *) user_data,
216                               object_path,
217                               greeting);
218   g_dbus_method_invocation_return_value (invocation,
219                                          g_variant_new ("(s)", response));
220   g_free (response);
221 }
222 
223 const GDBusInterfaceVTable partition_vtable =
224 {
225   partition_method_call,
226   //partition_get_property,
227   //partition_set_property
228 };
229 
230 /* ---------------------------------------------------------------------------------------------------- */
231 
232 static gchar **
subtree_enumerate(GDBusConnection * connection,const gchar * sender,const gchar * object_path,gpointer user_data)233 subtree_enumerate (GDBusConnection       *connection,
234                    const gchar           *sender,
235                    const gchar           *object_path,
236                    gpointer               user_data)
237 {
238   gchar **nodes;
239   GPtrArray *p;
240 
241   p = g_ptr_array_new ();
242   g_ptr_array_add (p, g_strdup ("sda"));
243   g_ptr_array_add (p, g_strdup ("sda1"));
244   g_ptr_array_add (p, g_strdup ("sda2"));
245   g_ptr_array_add (p, g_strdup ("sda3"));
246   g_ptr_array_add (p, g_strdup ("sdb"));
247   g_ptr_array_add (p, g_strdup ("sdb1"));
248   g_ptr_array_add (p, g_strdup ("sdc"));
249   g_ptr_array_add (p, g_strdup ("sdc1"));
250   g_ptr_array_add (p, NULL);
251   nodes = (gchar **) g_ptr_array_free (p, FALSE);
252 
253   return nodes;
254 }
255 
256 static GDBusInterfaceInfo **
subtree_introspect(GDBusConnection * connection,const gchar * sender,const gchar * object_path,const gchar * node,gpointer user_data)257 subtree_introspect (GDBusConnection       *connection,
258                     const gchar           *sender,
259                     const gchar           *object_path,
260                     const gchar           *node,
261                     gpointer               user_data)
262 {
263   GPtrArray *p;
264 
265   p = g_ptr_array_new ();
266   if (node == NULL)
267     {
268       g_ptr_array_add (p, g_dbus_interface_info_ref (manager_interface_info));
269     }
270   else
271     {
272       g_ptr_array_add (p, g_dbus_interface_info_ref (block_interface_info));
273       if (strlen (node) == 4)
274         g_ptr_array_add (p,
275                          g_dbus_interface_info_ref (partition_interface_info));
276     }
277 
278   g_ptr_array_add (p, NULL);
279 
280   return (GDBusInterfaceInfo **) g_ptr_array_free (p, FALSE);
281 }
282 
283 static const GDBusInterfaceVTable *
subtree_dispatch(GDBusConnection * connection,const gchar * sender,const gchar * object_path,const gchar * interface_name,const gchar * node,gpointer * out_user_data,gpointer user_data)284 subtree_dispatch (GDBusConnection             *connection,
285                   const gchar                 *sender,
286                   const gchar                 *object_path,
287                   const gchar                 *interface_name,
288                   const gchar                 *node,
289                   gpointer                    *out_user_data,
290                   gpointer                     user_data)
291 {
292   const GDBusInterfaceVTable *vtable_to_return;
293   gpointer user_data_to_return;
294 
295   if (g_strcmp0 (interface_name, "org.gtk.GDBus.Example.Manager") == 0)
296     {
297       user_data_to_return = "The Root";
298       vtable_to_return = &manager_vtable;
299     }
300   else
301     {
302       if (strlen (node) == 4)
303         user_data_to_return = "A partition";
304       else
305         user_data_to_return = "A block device";
306 
307       if (g_strcmp0 (interface_name, "org.gtk.GDBus.Example.Block") == 0)
308         vtable_to_return = &block_vtable;
309       else if (g_strcmp0 (interface_name, "org.gtk.GDBus.Example.Partition") == 0)
310         vtable_to_return = &partition_vtable;
311       else
312         g_assert_not_reached ();
313     }
314 
315   *out_user_data = user_data_to_return;
316 
317   return vtable_to_return;
318 }
319 
320 const GDBusSubtreeVTable subtree_vtable =
321 {
322   subtree_enumerate,
323   subtree_introspect,
324   subtree_dispatch
325 };
326 
327 /* ---------------------------------------------------------------------------------------------------- */
328 
329 static void
on_bus_acquired(GDBusConnection * connection,const gchar * name,gpointer user_data)330 on_bus_acquired (GDBusConnection *connection,
331                  const gchar     *name,
332                  gpointer         user_data)
333 {
334   guint registration_id;
335 
336   registration_id = g_dbus_connection_register_subtree (connection,
337                                                         "/org/gtk/GDBus/TestSubtree/Devices",
338                                                         &subtree_vtable,
339                                                         G_DBUS_SUBTREE_FLAGS_NONE,
340                                                         NULL,  /* user_data */
341                                                         NULL,  /* user_data_free_func */
342                                                         NULL); /* GError** */
343   g_assert (registration_id > 0);
344 }
345 
346 static void
on_name_acquired(GDBusConnection * connection,const gchar * name,gpointer user_data)347 on_name_acquired (GDBusConnection *connection,
348                   const gchar     *name,
349                   gpointer         user_data)
350 {
351 }
352 
353 static void
on_name_lost(GDBusConnection * connection,const gchar * name,gpointer user_data)354 on_name_lost (GDBusConnection *connection,
355               const gchar     *name,
356               gpointer         user_data)
357 {
358   exit (1);
359 }
360 
361 int
main(int argc,char * argv[])362 main (int argc, char *argv[])
363 {
364   guint owner_id;
365   GMainLoop *loop;
366 
367   /* We are lazy here - we don't want to manually provide
368    * the introspection data structures - so we just build
369    * them from XML.
370    */
371   introspection_data = g_dbus_node_info_new_for_xml (introspection_xml, NULL);
372   g_assert (introspection_data != NULL);
373 
374   manager_interface_info = g_dbus_node_info_lookup_interface (introspection_data, "org.gtk.GDBus.Example.Manager");
375   block_interface_info = g_dbus_node_info_lookup_interface (introspection_data, "org.gtk.GDBus.Example.Block");
376   partition_interface_info = g_dbus_node_info_lookup_interface (introspection_data, "org.gtk.GDBus.Example.Partition");
377   g_assert (manager_interface_info != NULL);
378   g_assert (block_interface_info != NULL);
379   g_assert (partition_interface_info != NULL);
380 
381   owner_id = g_bus_own_name (G_BUS_TYPE_SESSION,
382                              "org.gtk.GDBus.TestSubtree",
383                              G_BUS_NAME_OWNER_FLAGS_NONE,
384                              on_bus_acquired,
385                              on_name_acquired,
386                              on_name_lost,
387                              NULL,
388                              NULL);
389 
390   loop = g_main_loop_new (NULL, FALSE);
391   g_main_loop_run (loop);
392 
393   g_bus_unown_name (owner_id);
394 
395   g_dbus_node_info_unref (introspection_data);
396 
397   return 0;
398 }
399