2 * Copyright (c) 2016 Samsung Electronics Co., Ltd. All rights reserved.
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
8 * http://www.apache.org/licenses/LICENSE-2.0
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
20 #include <sys/types.h>
26 #include "package-manager.h"
27 #include "pkgmgr_client_debug.h"
28 #include "pkgmgr_client_internal.h"
29 #include "../../installer/pkgmgr_installer.h"
30 #include "../../installer/pkgmgr_installer_config.h"
32 #define CONNECTION_RETRY_MAX 5
33 #define CONNECTION_WAIT_USEC (1000000 / 2) /* 0.5 sec */
34 #define CONNECTION_TIMEOUT_MSEC 5000 /* 5 sec */
35 #define REGULAR_USER 5000
37 static int _is_system_user(void)
41 if (uid < REGULAR_USER)
47 static GBusType __get_bus_type(pkgmgr_client_type type)
49 if (type == PC_REQUEST || _is_system_user())
50 return G_BUS_TYPE_SYSTEM;
52 return G_BUS_TYPE_SESSION;
55 int pkgmgr_client_connection_connect(struct pkgmgr_client_t *pc)
60 #if !GLIB_CHECK_VERSION(2, 35, 0)
63 bus_type = __get_bus_type(pc->pc_type);
64 pc->conn = g_bus_get_sync(bus_type, NULL, &error);
66 ERR("gdbus connection error (%s)", error->message);
68 return PKGMGR_R_ECOMM;
74 void pkgmgr_client_connection_disconnect(struct pkgmgr_client_t *pc)
76 /* flush remaining buffer: blocking mode */
77 g_dbus_connection_flush_sync(pc->conn, NULL, NULL);
78 g_object_unref(pc->conn);
83 const char *signal_str;
87 struct signal_map map[] = {
88 {PKGMGR_INSTALLER_INSTALL_EVENT_STR, PKGMGR_CLIENT_STATUS_INSTALL},
89 {PKGMGR_INSTALLER_UNINSTALL_EVENT_STR, PKGMGR_CLIENT_STATUS_UNINSTALL},
90 {PKGMGR_INSTALLER_UPGRADE_EVENT_STR, PKGMGR_CLIENT_STATUS_UPGRADE},
91 {PKGMGR_INSTALLER_CLEAR_EVENT_STR, PKGMGR_CLIENT_STATUS_CLEAR_DATA},
92 {PKGMGR_INSTALLER_MOVE_EVENT_STR, PKGMGR_CLIENT_STATUS_MOVE},
93 {PKGMGR_INSTALLER_INSTALL_PERCENT_KEY_STR,
94 PKGMGR_CLIENT_STATUS_INSTALL_PROGRESS},
95 {PKGMGR_INSTALLER_GET_SIZE_KEY_STR, PKGMGR_CLIENT_STATUS_GET_SIZE},
96 {PKGMGR_INSTALLER_CLEAR_CACHE_KEY_STR,
97 PKGMGR_CLIENT_STATUS_CLEAR_CACHE},
98 {PKGMGR_INSTALLER_APP_ENABLE_EVENT_STR,
99 PKGMGR_CLIENT_STATUS_ENABLE_APP},
100 {PKGMGR_INSTALLER_APP_DISABLE_EVENT_STR,
101 PKGMGR_CLIENT_STATUS_DISABLE_APP},
102 {PKGMGR_INSTALLER_APP_ENABLE_SPLASH_SCREEN_EVENT_STR,
103 PKGMGR_CLIENT_STATUS_ENABLE_APP_SPLASH_SCREEN},
104 {PKGMGR_INSTALLER_APP_DISABLE_SPLASH_SCREEN_EVENT_STR,
105 PKGMGR_CLIENT_STATUS_DISABLE_APP_SPLASH_SCREEN},
106 {PKGMGR_INSTALLER_RES_COPY_EVENT_STR, PKGMGR_CLIENT_STATUS_RES_COPY},
107 {PKGMGR_INSTALLER_RES_CREATE_DIR_EVENT_STR,
108 PKGMGR_CLIENT_STATUS_RES_CREATE_DIR},
109 {PKGMGR_INSTALLER_RES_REMOVE_EVENT_STR,
110 PKGMGR_CLIENT_STATUS_RES_REMOVE},
111 {PKGMGR_INSTALLER_RES_UNINSTALL_EVENT_STR,
112 PKGMGR_CLIENT_STATUS_RES_UNINSTALL},
116 static int __get_signal_type(const char *name)
123 for (i = 0; map[i].signal_str != NULL; i++) {
124 if (strcmp(map[i].signal_str, name) == 0)
125 return map[i].signal_type;
131 static void __handle_size_info_callback(struct cb_info *cb_info,
132 const char *pkgid, const char *val)
134 pkg_size_info_t size_info;
138 pkgmgr_total_pkg_size_info_receive_cb callback;
140 snprintf(buf, sizeof(buf), "%s", val);
142 DBG("%s, %s", pkgid, val);
144 token = strtok_r(buf, ":", &saveptr);
146 ERR("failed to parse size info");
149 size_info.data_size = atoll(token);
150 token = strtok_r(NULL, ":", &saveptr);
152 ERR("failed to parse size info");
155 size_info.cache_size = atoll(token);
156 token = strtok_r(NULL, ":", &saveptr);
158 ERR("failed to parse size info");
161 size_info.app_size = atoll(token);
162 token = strtok_r(NULL, ":", &saveptr);
164 ERR("failed to parse size info");
167 size_info.ext_data_size = atoll(token);
168 token = strtok_r(NULL, ":", &saveptr);
170 ERR("failed to parse size info");
173 size_info.ext_cache_size = atoll(token);
174 token = strtok_r(NULL, ":", &saveptr);
176 ERR("failed to parse size info");
179 size_info.ext_app_size = atoll(token);
181 DBG("data: %lld, cache: %lld, app: %lld, ext_data: %lld, "
182 "ext_cache: %lld, ext_app: %lld",
183 size_info.data_size, size_info.cache_size,
184 size_info.app_size, size_info.ext_data_size,
185 size_info.ext_cache_size, size_info.ext_app_size);
187 if (strcmp(pkgid, PKG_SIZE_INFO_TOTAL) == 0) {
188 callback = (pkgmgr_total_pkg_size_info_receive_cb)
189 cb_info->size_info_cb;
190 callback(cb_info->client, &size_info, cb_info->data);
192 cb_info->size_info_cb(cb_info->client, pkgid, &size_info,
197 static void __handle_pkg_signal(const gchar *signal_name,
198 GVariant *parameters, gpointer user_data)
202 char *pkg_type = NULL;
207 struct cb_info *cb_info = (struct cb_info *)user_data;
208 GVariantIter *iter = NULL;
211 g_variant_get(parameters, "(u&sa(sss)&s&s)", &target_uid, &req_id, &iter,
213 while (g_variant_iter_loop(iter, "(&s&s&s)", &pkgid, &appid, &pkg_type)) {
214 if (cb_info->req_key) {
215 if (strcmp(cb_info->req_key, req_id) != 0)
218 signal_type = __get_signal_type(signal_name);
219 if (signal_type < 0 || !(cb_info->status_type & signal_type)) {
220 g_variant_iter_free(iter);
225 /* each cb_data can only has one callback */
226 if (cb_info->event_cb) {
227 cb_info->event_cb(target_uid, cb_info->req_id,
228 pkg_type, pkgid, key, val, NULL, cb_info->data);
229 } else if (cb_info->app_event_cb && strcmp(appid, "") != 0) {
230 cb_info->app_event_cb(target_uid, cb_info->req_id,
231 pkg_type, pkgid, appid, key, val, NULL,
233 } else if (cb_info->size_info_cb) {
234 __handle_size_info_callback(cb_info, pkgid, val);
237 /* TODO: unsubscribe request callback */
239 g_variant_iter_free(iter);
242 static void __handle_res_event_signal(const gchar *signal_name,
243 GVariant *parameters, gpointer user_data)
250 pkgmgr_res_event_path_state state;
251 struct cb_info *cb_info = (struct cb_info *)user_data;
253 GVariant *extra_param = NULL;
254 pkgmgr_res_event_info_t *event_info;
257 if (!cb_info->res_event_cb)
260 g_variant_get(parameters, "(u&s&s&sv)", &target_uid, &req_id, &pkgid, &status, &extra_param);
261 if (!g_variant_type_equal(G_VARIANT_TYPE("(ia(si))"),
262 g_variant_get_type(extra_param))) {
263 ERR("invalid extra parameter");
264 g_variant_unref(extra_param);
267 if (cb_info->req_key) {
268 if (strcmp(cb_info->req_key, req_id) != 0) {
269 g_variant_unref(extra_param);
273 signal_type = __get_signal_type(signal_name);
274 if (signal_type < 0 || !(cb_info->status_type & signal_type)) {
275 g_variant_unref(extra_param);
280 event_info = pkgmgr_res_event_info_new();
281 if (event_info == NULL) {
282 ERR("out of memory");
283 g_variant_unref(extra_param);
287 g_variant_get(extra_param, "(ia(si))", &event_info->error_code, &iter);
289 while (g_variant_iter_loop(iter, "(&si)", &path, &state)) {
290 if (pkgmgr_res_event_info_add_path_state(event_info,
291 path, state) != PKGMGR_R_OK) {
292 ERR("Fail to add path state");
293 g_variant_unref(extra_param);
294 g_variant_iter_free(iter);
295 pkgmgr_res_event_info_free(event_info);
299 cb_info->res_event_cb(target_uid, cb_info->req_id, pkgid, signal_name,
300 status, event_info, cb_info->data);
301 g_variant_unref(extra_param);
302 g_variant_iter_free(iter);
303 pkgmgr_res_event_info_free(event_info);
306 static void __signal_handler(GDBusConnection *conn, const gchar *sender_name,
307 const gchar *object_path, const gchar *interface_name,
308 const gchar *signal_name, GVariant *parameters,
311 if (g_variant_type_equal(G_VARIANT_TYPE("(usa(sss)ss)"),
312 g_variant_get_type(parameters))) {
313 __handle_pkg_signal(signal_name, parameters, user_data);
314 } else if (!strcmp(signal_name, PKGMGR_INSTALLER_RES_COPY_EVENT_STR) ||
315 !strcmp(signal_name, PKGMGR_INSTALLER_RES_REMOVE_EVENT_STR) ||
316 !strcmp(signal_name, PKGMGR_INSTALLER_RES_UNINSTALL_EVENT_STR) ||
317 !strcmp(signal_name, PKGMGR_INSTALLER_RES_CREATE_DIR_EVENT_STR)) {
318 __handle_res_event_signal(signal_name, parameters, user_data);
322 static void __set_signal_list(int event, GList **signal_list)
325 if (event == PKGMGR_CLIENT_STATUS_ALL)
328 for (i = 0; map[i].signal_str != NULL; i++) {
329 if (event & map[i].signal_type)
330 *signal_list = g_list_append(*signal_list,
331 (char *)map[i].signal_str);
335 static int __subscribe_signal(struct pkgmgr_client_t *pc,
336 struct cb_info *cb_info, const char *signal)
338 guint sid = g_dbus_connection_signal_subscribe(pc->conn, NULL,
339 PKGMGR_INSTALLER_DBUS_INTERFACE, signal,
340 PKGMGR_INSTALLER_DBUS_OBJECT_PATH, NULL,
341 G_DBUS_SIGNAL_FLAGS_NONE, __signal_handler,
342 (gpointer)cb_info, NULL);
344 ERR("failed to subscribe singal");
345 return PKGMGR_R_ERROR;
347 cb_info->sid_list = g_list_append(cb_info->sid_list,
348 GUINT_TO_POINTER(sid));
353 int pkgmgr_client_connection_set_callback(struct pkgmgr_client_t *pc,
354 struct cb_info *cb_info)
356 GList *signal_list = NULL;
357 GList *tmp_list = NULL;
360 __set_signal_list(pc->status_type, &signal_list);
362 if (g_list_length(signal_list) == 0)
363 return __subscribe_signal(pc, cb_info, NULL);
365 for (tmp_list = signal_list; tmp_list != NULL;
366 tmp_list = g_list_next(tmp_list)) {
367 signal_type = (char *)tmp_list->data;
368 ret = __subscribe_signal(pc, cb_info, signal_type);
370 g_list_free(signal_list);
371 return PKGMGR_R_ERROR;
375 g_list_free(signal_list);
379 void pkgmgr_client_connection_unset_callback(struct pkgmgr_client_t *pc,
380 struct cb_info *cb_info)
384 for (tmp = cb_info->sid_list; tmp != NULL; tmp = g_list_next(tmp)) {
385 tmp_sid = GPOINTER_TO_UINT(tmp->data);
386 g_dbus_connection_signal_unsubscribe(pc->conn, tmp_sid);
390 int pkgmgr_client_connection_send_request(struct pkgmgr_client_t *pc,
391 const char *method, GVariant *params, GVariant **result)
393 GError *error = NULL;
397 int ret = PKGMGR_R_ECOMM;
399 /* convert floating ref into normal ref */
400 g_variant_ref_sink(params);
403 proxy = g_dbus_proxy_new_sync(pc->conn,
404 G_DBUS_PROXY_FLAGS_DO_NOT_LOAD_PROPERTIES,
405 NULL, PKGMGR_DBUS_SERVICE,
406 PKGMGR_DBUS_OBJECT_PATH,
407 PKGMGR_DBUS_INTERFACE, NULL, &error);
409 ERR("failed to get proxy object, sleep and retry[%s]",
413 usleep(CONNECTION_WAIT_USEC);
418 r = g_dbus_proxy_call_sync(proxy, method, params,
419 G_DBUS_CALL_FLAGS_NONE,
420 CONNECTION_TIMEOUT_MSEC, NULL, &error);
421 g_object_unref(proxy);
422 if (error && error->code == G_DBUS_ERROR_ACCESS_DENIED) {
423 ERR("failed to send request, privilege denied[%s]",
426 ret = PKGMGR_R_EPRIV;
435 ERR("failed to send request, sleep and retry[%s]",
439 usleep(CONNECTION_WAIT_USEC);
441 } while (retry_cnt <= CONNECTION_RETRY_MAX);
443 /* decrease ref count to 0 to free resource */
444 g_variant_unref(params);