1 /* -*- Mode: C ; c-basic-offset: 2 -*- */
3 * LADI Session Handler (ladish)
5 * Copyright (C) 2009,2010,2011 Nedko Arnaudov <nedko@arnaudov.name>
7 **************************************************************************
8 * This file contains implementation of code that interfaces
9 * app supervisor object through D-Bus
10 **************************************************************************
12 * LADI Session Handler is free software; you can redistribute it and/or modify
13 * it under the terms of the GNU General Public License as published by
14 * the Free Software Foundation; either version 2 of the License, or
15 * (at your option) any later version.
17 * LADI Session Handler is distributed in the hope that it will be useful,
18 * but WITHOUT ANY WARRANTY; without even the implied warranty of
19 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
20 * GNU General Public License for more details.
22 * You should have received a copy of the GNU General Public License
23 * along with LADI Session Handler. If not, see <http://www.gnu.org/licenses/>
24 * or write to the Free Software Foundation, Inc.,
25 * 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
28 #include "app_supervisor_proxy.h"
30 struct ladish_app_supervisor_proxy
37 void (* app_added
)(void * context
, uint64_t id
, const char * name
, bool running
, bool terminal
, const char * level
);
38 void (* app_state_changed
)(void * context
, uint64_t id
, const char * name
, bool running
, bool terminal
, const char * level
);
39 void (* app_removed
)(void * context
, uint64_t id
);
42 #define proxy_ptr ((struct ladish_app_supervisor_proxy *)context)
44 static void on_app_added(void * context
, DBusMessage
* message_ptr
)
46 uint64_t new_list_version
;
53 if (!dbus_message_get_args(
56 DBUS_TYPE_UINT64
, &new_list_version
,
57 DBUS_TYPE_UINT64
, &id
,
58 DBUS_TYPE_STRING
, &name
,
59 DBUS_TYPE_BOOLEAN
, &running
,
60 DBUS_TYPE_BOOLEAN
, &terminal
,
61 DBUS_TYPE_STRING
, &level
,
64 log_error("dbus_message_get_args() failed to extract AppAdded2 signal arguments (%s)", cdbus_g_dbus_error
.message
);
65 dbus_error_free(&cdbus_g_dbus_error
);
69 //log_info("AppAdded2 signal received. id=%"PRIu64", name='%s', %srunning, %s, level '%s'", id, name, running ? "" : "not ", terminal ? "terminal" : "shell", level);
71 if (new_list_version
<= proxy_ptr
->version
)
73 log_info("Ignoring signal for older version of the app list");
77 //log_info("got new list version %llu", (unsigned long long)version);
78 proxy_ptr
->version
= new_list_version
;
80 level
= ladish_map_app_level_constant(level
);
83 proxy_ptr
->app_added(proxy_ptr
->context
, id
, name
, running
, terminal
, level
);
87 log_error("Ignoring app added signal for '%s' because of invalid app level value", name
);
92 static void on_app_removed(void * context
, DBusMessage
* message_ptr
)
94 uint64_t new_list_version
;
97 if (!dbus_message_get_args(
100 DBUS_TYPE_UINT64
, &new_list_version
,
101 DBUS_TYPE_UINT64
, &id
,
104 log_error("dbus_message_get_args() failed to extract AppRemoved signal arguments (%s)", cdbus_g_dbus_error
.message
);
105 dbus_error_free(&cdbus_g_dbus_error
);
109 //log_info("AppRemoved signal received, id=%"PRIu64, id);
110 if (new_list_version
<= proxy_ptr
->version
)
112 log_info("Ignoring signal for older version of the app list");
116 //log_info("got new list version %llu", (unsigned long long)version);
117 proxy_ptr
->version
= new_list_version
;
118 proxy_ptr
->app_removed(proxy_ptr
->context
, id
);
122 static void on_app_state_changed(void * context
, DBusMessage
* message_ptr
)
124 uint64_t new_list_version
;
128 dbus_bool_t terminal
;
131 if (!dbus_message_get_args(
134 DBUS_TYPE_UINT64
, &new_list_version
,
135 DBUS_TYPE_UINT64
, &id
,
136 DBUS_TYPE_STRING
, &name
,
137 DBUS_TYPE_BOOLEAN
, &running
,
138 DBUS_TYPE_BOOLEAN
, &terminal
,
139 DBUS_TYPE_STRING
, &level
,
142 log_error("dbus_message_get_args() failed to extract AppStateChanged2 signal arguments (%s)", cdbus_g_dbus_error
.message
);
143 dbus_error_free(&cdbus_g_dbus_error
);
147 //log_info("AppStateChanged2 signal received");
148 //log_info("AppRemoved signal received, id=%"PRIu64, id);
149 if (new_list_version
<= proxy_ptr
->version
)
151 log_info("Ignoring signal for older version of the app list");
155 //log_info("got new list version %llu", (unsigned long long)version);
156 proxy_ptr
->version
= new_list_version
;
158 level
= ladish_map_app_level_constant(level
);
161 proxy_ptr
->app_state_changed(proxy_ptr
->context
, id
, name
, running
, terminal
, level
);
165 log_error("Ignoring app state changed signal for '%s' because of invalid app level value", name
);
172 /* this must be static because it is referenced by the
173 * dbus helper layer when hooks are active */
174 static struct cdbus_signal_hook g_signal_hooks
[] =
176 {"AppAdded2", on_app_added
},
177 {"AppRemoved", on_app_removed
},
178 {"AppStateChanged2", on_app_state_changed
},
182 static void refresh_internal(struct ladish_app_supervisor_proxy
* proxy_ptr
, bool force
)
184 DBusMessage
* reply_ptr
;
185 DBusMessageIter iter
;
186 dbus_uint64_t version
;
187 const char * reply_signature
;
188 DBusMessageIter array_iter
;
189 DBusMessageIter struct_iter
;
193 dbus_bool_t terminal
;
196 log_info("refresh_internal() called");
198 version
= proxy_ptr
->version
;
200 if (!cdbus_call(0, proxy_ptr
->service
, proxy_ptr
->object
, IFACE_APP_SUPERVISOR
, "GetAll2", "t", &version
, NULL
, &reply_ptr
))
202 log_error("GetAll2() failed.");
206 reply_signature
= dbus_message_get_signature(reply_ptr
);
208 if (strcmp(reply_signature
, "ta(tsbbs)") != 0)
210 log_error("GetAll2() reply signature mismatch. '%s'", reply_signature
);
214 dbus_message_iter_init(reply_ptr
, &iter
);
216 //log_info_msg("version " + (char)dbus_message_iter_get_arg_type(&iter));
217 dbus_message_iter_get_basic(&iter
, &version
);
218 dbus_message_iter_next(&iter
);
220 if (!force
&& version
<= proxy_ptr
->version
)
225 //log_info("got new list version %llu", (unsigned long long)version);
226 proxy_ptr
->version
= version
;
228 for (dbus_message_iter_recurse(&iter
, &array_iter
);
229 dbus_message_iter_get_arg_type(&array_iter
) != DBUS_TYPE_INVALID
;
230 dbus_message_iter_next(&array_iter
))
232 dbus_message_iter_recurse(&array_iter
, &struct_iter
);
234 dbus_message_iter_get_basic(&struct_iter
, &id
);
235 dbus_message_iter_next(&struct_iter
);
237 dbus_message_iter_get_basic(&struct_iter
, &name
);
238 dbus_message_iter_next(&struct_iter
);
240 dbus_message_iter_get_basic(&struct_iter
, &running
);
241 dbus_message_iter_next(&struct_iter
);
243 dbus_message_iter_get_basic(&struct_iter
, &terminal
);
244 dbus_message_iter_next(&struct_iter
);
246 dbus_message_iter_get_basic(&struct_iter
, &level
);
247 dbus_message_iter_next(&struct_iter
);
249 level
= ladish_map_app_level_constant(level
);
252 //log_info("App id=%"PRIu64", name='%s', %srunning, %s, level '%s'", id, name, running ? "" : "not ", terminal ? "terminal" : "shell", level);
253 proxy_ptr
->app_added(proxy_ptr
->context
, id
, name
, running
, terminal
, level
);
257 log_error("Ignoring app '%s' because of invalid app level value", name
);
260 dbus_message_iter_next(&struct_iter
);
264 dbus_message_unref(reply_ptr
);
268 ladish_app_supervisor_proxy_create(
269 const char * service
,
272 void (* app_added
)(void * context
, uint64_t id
, const char * name
, bool running
, bool terminal
, const char * level
),
273 void (* app_state_changed
)(void * context
, uint64_t id
, const char * name
, bool running
, bool terminal
, const char * level
),
274 void (* app_removed
)(void * context
, uint64_t id
),
275 ladish_app_supervisor_proxy_handle
* handle_ptr
)
277 struct ladish_app_supervisor_proxy
* proxy_ptr
;
279 proxy_ptr
= malloc(sizeof(struct ladish_app_supervisor_proxy
));
280 if (proxy_ptr
== NULL
)
282 log_error("malloc() failed to allocate struct proxy");
286 proxy_ptr
->service
= strdup(service
);
287 if (proxy_ptr
->service
== NULL
)
289 log_error("strdup() failed too duplicate service name '%s'", service
);
293 proxy_ptr
->object
= strdup(object
);
294 if (proxy_ptr
->object
== NULL
)
296 log_error("strdup() failed too duplicate object name '%s'", object
);
300 proxy_ptr
->version
= 0;
302 proxy_ptr
->context
= context
;
303 proxy_ptr
->app_added
= app_added
;
304 proxy_ptr
->app_state_changed
= app_state_changed
;
305 proxy_ptr
->app_removed
= app_removed
;
307 if (!cdbus_register_object_signal_hooks(
308 cdbus_g_dbus_connection
,
311 IFACE_APP_SUPERVISOR
,
315 log_error("dbus_register_object_signal_hooks() failed for app supervisor");
319 refresh_internal(proxy_ptr
, true);
321 *handle_ptr
= (ladish_app_supervisor_proxy_handle
)proxy_ptr
;
326 free(proxy_ptr
->object
);
328 free(proxy_ptr
->service
);
335 #define proxy_ptr ((struct ladish_app_supervisor_proxy *)proxy)
337 void ladish_app_supervisor_proxy_destroy(ladish_app_supervisor_proxy_handle proxy
)
339 cdbus_unregister_object_signal_hooks(cdbus_g_dbus_connection
, proxy_ptr
->service
, proxy_ptr
->object
, IFACE_APP_SUPERVISOR
);
341 free(proxy_ptr
->object
);
342 free(proxy_ptr
->service
);
347 ladish_app_supervisor_proxy_run_custom(
348 ladish_app_supervisor_proxy_handle proxy
,
349 const char * command
,
351 bool run_in_terminal
,
354 dbus_bool_t terminal
;
356 terminal
= run_in_terminal
;
358 if (!cdbus_call(0, proxy_ptr
->service
, proxy_ptr
->object
, IFACE_APP_SUPERVISOR
, "RunCustom2", "bsss", &terminal
, &command
, &name
, &level
, ""))
360 log_error("RunCustom2() failed.");
367 bool ladish_app_supervisor_proxy_start_app(ladish_app_supervisor_proxy_handle proxy
, uint64_t id
)
369 if (!cdbus_call(0, proxy_ptr
->service
, proxy_ptr
->object
, IFACE_APP_SUPERVISOR
, "StartApp", "t", &id
, ""))
371 log_error("StartApp() failed.");
378 bool ladish_app_supervisor_proxy_stop_app(ladish_app_supervisor_proxy_handle proxy
, uint64_t id
)
380 if (!cdbus_call(0, proxy_ptr
->service
, proxy_ptr
->object
, IFACE_APP_SUPERVISOR
, "StopApp", "t", &id
, ""))
382 log_error("StopApp() failed.");
389 bool ladish_app_supervisor_proxy_kill_app(ladish_app_supervisor_proxy_handle proxy
, uint64_t id
)
391 if (!cdbus_call(0, proxy_ptr
->service
, proxy_ptr
->object
, IFACE_APP_SUPERVISOR
, "KillApp", "t", &id
, ""))
393 log_error("KillApp() failed.");
400 bool ladish_app_supervisor_proxy_remove_app(ladish_app_supervisor_proxy_handle proxy
, uint64_t id
)
402 if (!cdbus_call(0, proxy_ptr
->service
, proxy_ptr
->object
, IFACE_APP_SUPERVISOR
, "RemoveApp", "t", &id
, ""))
404 log_error("RemoveApp() failed.");
412 ladish_app_supervisor_get_app_properties(
413 ladish_app_supervisor_proxy_handle proxy
,
415 char ** name_ptr_ptr
,
416 char ** command_ptr_ptr
,
419 const char ** level_ptr
)
421 DBusMessage
* reply_ptr
;
423 const char * commandline
;
425 dbus_bool_t terminal
;
428 char * commandline_buffer
;
430 if (!cdbus_call(0, proxy_ptr
->service
, proxy_ptr
->object
, IFACE_APP_SUPERVISOR
, "GetAppProperties2", "t", &id
, NULL
, &reply_ptr
))
432 log_error("GetAppProperties2() failed.");
436 if (!dbus_message_get_args(
439 DBUS_TYPE_STRING
, &name
,
440 DBUS_TYPE_STRING
, &commandline
,
441 DBUS_TYPE_BOOLEAN
, &running
,
442 DBUS_TYPE_BOOLEAN
, &terminal
,
443 DBUS_TYPE_STRING
, &level
,
446 dbus_message_unref(reply_ptr
);
447 dbus_error_free(&cdbus_g_dbus_error
);
448 log_error("decoding reply of GetAppProperties failed.");
452 level
= ladish_map_app_level_constant(level
);
455 dbus_message_unref(reply_ptr
);
456 log_error("decoding reply of GetAppProperties failed.");
460 name_buffer
= strdup(name
);
461 if (name_buffer
== NULL
)
463 log_error("strdup() for app name failed.");
464 dbus_message_unref(reply_ptr
);
468 commandline_buffer
= strdup(commandline
);
469 if (commandline_buffer
== NULL
)
471 log_error("strdup() for app commandline failed.");
473 dbus_message_unref(reply_ptr
);
477 *name_ptr_ptr
= name_buffer
;
478 *command_ptr_ptr
= commandline_buffer
;
479 *running_ptr
= running
;
480 *terminal_ptr
= terminal
;
483 dbus_message_unref(reply_ptr
);
489 ladish_app_supervisor_set_app_properties(
490 ladish_app_supervisor_proxy_handle proxy
,
493 const char * command
,
494 bool run_in_terminal
,
497 dbus_bool_t terminal
;
499 terminal
= run_in_terminal
;
505 IFACE_APP_SUPERVISOR
,
515 log_error("SetAppProperties2() failed.");