Fix a bug regarding category data
[platform/core/appfw/pkgmgr-info.git] / src / pkgmgrinfo_private.c
index 2889ce5..4fa7551 100644 (file)
@@ -73,7 +73,8 @@ static struct _pkginfo_bool_map_t pkginfo_bool_prop_map[] = {
        {E_PMINFO_PKGINFO_PROP_PACKAGE_APPSETTING,      PMINFO_PKGINFO_PROP_PACKAGE_APPSETTING},
        {E_PMINFO_PKGINFO_PROP_PACKAGE_NODISPLAY_SETTING,       PMINFO_PKGINFO_PROP_PACKAGE_NODISPLAY_SETTING},
        {E_PMINFO_PKGINFO_PROP_PACKAGE_SUPPORT_DISABLE, PMINFO_PKGINFO_PROP_PACKAGE_SUPPORT_DISABLE},
-       {E_PMINFO_PKGINFO_PROP_PACKAGE_DISABLE, PMINFO_PKGINFO_PROP_PACKAGE_DISABLE}
+       {E_PMINFO_PKGINFO_PROP_PACKAGE_DISABLE, PMINFO_PKGINFO_PROP_PACKAGE_DISABLE},
+       {E_PMINFO_PKGINFO_PROP_PACKAGE_CHECK_STORAGE,   PMINFO_PKGINFO_PROP_PACKAGE_CHECK_STORAGE}
 };
 
 struct _appinfo_str_map_t {
@@ -93,8 +94,6 @@ static struct _appinfo_str_map_t appinfo_str_prop_map[] = {
        {E_PMINFO_APPINFO_PROP_APP_CATEGORY,    PMINFO_APPINFO_PROP_APP_CATEGORY},
        {E_PMINFO_APPINFO_PROP_APP_HWACCELERATION,      PMINFO_APPINFO_PROP_APP_HWACCELERATION},
        {E_PMINFO_APPINFO_PROP_APP_SCREENREADER,        PMINFO_APPINFO_PROP_APP_SCREENREADER},
-       {E_PMINFO_APPINFO_PROP_APP_METADATA_KEY,        PMINFO_APPINFO_PROP_APP_METADATA_KEY},
-       {E_PMINFO_APPINFO_PROP_APP_METADATA_VALUE,      PMINFO_APPINFO_PROP_APP_METADATA_VALUE},
        {E_PMINFO_APPINFO_PROP_APP_PACKAGE,     PMINFO_APPINFO_PROP_APP_PACKAGE},
        {E_PMINFO_APPINFO_PROP_APP_INSTALLED_STORAGE,   PMINFO_APPINFO_PROP_APP_INSTALLED_STORAGE}
 };
@@ -105,7 +104,7 @@ struct _appinfo_int_map_t {
 };
 
 static struct _appinfo_int_map_t appinfo_int_prop_map[] = {
-       {E_PMINFO_APPINFO_PROP_APP_DISABLE_FOR_USER,    PMINFO_APPINFO_PROP_APP_DISABLE_FOR_USER}
+       {E_PMINFO_APPINFO_PROP_APP_SUPPORT_MODE,        PMINFO_APPINFO_PROP_APP_SUPPORT_MODE}
 };
 
 struct _appinfo_bool_map_t {
@@ -122,7 +121,8 @@ static struct _appinfo_bool_map_t appinfo_bool_prop_map[] = {
        {E_PMINFO_APPINFO_PROP_APP_LAUNCHCONDITION,             PMINFO_APPINFO_PROP_APP_LAUNCHCONDITION},
        {E_PMINFO_APPINFO_PROP_APP_UI_GADGET,           PMINFO_APPINFO_PROP_APP_UI_GADGET},
        {E_PMINFO_APPINFO_PROP_APP_SUPPORT_DISABLE,             PMINFO_APPINFO_PROP_APP_SUPPORT_DISABLE},
-       {E_PMINFO_APPINFO_PROP_APP_DISABLE,             PMINFO_APPINFO_PROP_APP_DISABLE}
+       {E_PMINFO_APPINFO_PROP_APP_DISABLE,             PMINFO_APPINFO_PROP_APP_DISABLE},
+       {E_PMINFO_APPINFO_PROP_APP_CHECK_STORAGE,               PMINFO_APPINFO_PROP_APP_CHECK_STORAGE}
 };
 
 inline pkgmgrinfo_pkginfo_filter_prop_str _pminfo_pkginfo_convert_to_prop_str(const char *property)
@@ -233,13 +233,15 @@ inline pkgmgrinfo_appinfo_filter_prop_bool _pminfo_appinfo_convert_to_prop_bool(
        return prop;
 }
 
-int __get_filter_condition(gpointer data, char **condition, GList **params)
+int __get_filter_condition(gpointer data, uid_t uid, char **condition, GList **params)
 {
        pkgmgrinfo_node_x *node = (pkgmgrinfo_node_x *)data;
        char buf[MAX_QUERY_LEN] = {'\0'};
        int flag = 0;
+       char *value;
        char *ptr = NULL;
        char *saveptr = NULL;
+       size_t len = 0;
 
        switch (node->prop) {
        case E_PMINFO_PKGINFO_PROP_PACKAGE_ID:
@@ -328,18 +330,29 @@ int __get_filter_condition(gpointer data, char **condition, GList **params)
                break;
        case E_PMINFO_APPINFO_PROP_APP_CATEGORY:
                snprintf(buf, sizeof(buf), "package_app_app_category.category IN (");
-               ptr = strtok_r(node->value, ",", &saveptr);
-               if (ptr == NULL)
+               value = strdup(node->value);
+               if (value == NULL) {
+                       _LOGE("out of memeory");
                        return 0;
-               strncat(buf, "?", MAX_QUERY_LEN - 2);
+               }
+               ptr = strtok_r(value, ",", &saveptr);
+               if (ptr == NULL) {
+                       free(value);
+                       return 0;
+               }
+               strncat(buf, "?", MAX_QUERY_LEN - len - 1);
+               len += strlen("?");
                *params = g_list_append(*params, strdup(ptr));
                while ((ptr = strtok_r(NULL, ",", &saveptr))) {
-                       strncat(buf, ", ?", MAX_QUERY_LEN - strlen(", ?") - 1);
+                       strncat(buf, ", ?", MAX_QUERY_LEN - len - 1);
+                       len += strlen(", ?");
                        *params = g_list_append(*params, strdup(ptr));
                }
-               strncat(buf, ")", MAX_QUERY_LEN - 2);
+               strncat(buf, ")", MAX_QUERY_LEN - len - 1);
+               len += strlen("?");
                *condition = strdup(buf);
                flag = E_PMINFO_APPINFO_JOIN_CATEGORY;
+               free(value);
 
                return flag;
        case E_PMINFO_APPINFO_PROP_APP_NODISPLAY:
@@ -370,30 +383,32 @@ int __get_filter_condition(gpointer data, char **condition, GList **params)
                snprintf(buf, sizeof(buf), "ai.package=?");
                break;
        case E_PMINFO_APPINFO_PROP_APP_INSTALLED_STORAGE:
-               snprintf(buf, sizeof(buf), "ai.installed_storage=?");
+               snprintf(buf, sizeof(buf), "ai.app_installed_storage=?");
                break;
        case E_PMINFO_APPINFO_PROP_APP_UI_GADGET:
                snprintf(buf, sizeof(buf), "ai.app_ui_gadget=? COLLATE NOCASE");
                break;
-       case E_PMINFO_APPINFO_PROP_APP_METADATA_KEY:
-               snprintf(buf, sizeof(buf), "package_app_app_metadata.md_key=?");
-               flag = E_PMINFO_APPINFO_JOIN_METADATA;
-               break;
-       case E_PMINFO_APPINFO_PROP_APP_METADATA_VALUE:
-               snprintf(buf, sizeof(buf), "package_app_app_metadata.md_value=?");
-               flag = E_PMINFO_APPINFO_JOIN_METADATA;
-               break;
        case E_PMINFO_APPINFO_PROP_APP_DISABLE:
-               snprintf(buf, MAX_QUERY_LEN, "ai.app_disable=? COLLATE NOCASE");
+               if (strcasecmp(node->value, "true") == 0)
+                       snprintf(buf, MAX_QUERY_LEN, "(ai.app_disable=? COLLATE NOCASE OR "
+                                       "ai.app_id IN (SELECT app_id FROM package_app_info_for_uid "
+                                       "WHERE uid=%d AND is_disabled='true'))", uid);
+               else
+                       snprintf(buf, MAX_QUERY_LEN, "(ai.app_disable=? COLLATE NOCASE AND "
+                                       "ai.app_id NOT IN (SELECT app_id FROM package_app_info_for_uid "
+                                       "WHERE uid=%d AND is_disabled='true'))", uid);
+               flag = E_PMINFO_APPINFO_JOIN_APPINFO_FOR_UID;
                break;
        case E_PMINFO_APPINFO_PROP_APP_SUPPORT_DISABLE:
                snprintf(buf, MAX_QUERY_LEN, "ai.app_support_disable=? COLLATE NOCASE");
                break;
-       case E_PMINFO_APPINFO_PROP_APP_DISABLE_FOR_USER:
-               snprintf(buf, MAX_QUERY_LEN, "ai.app_id NOT IN "
-                               "(SELECT app_id from package_app_info_for_uid WHERE uid=? " \
-                               "AND is_disabled='true' COLLATE NOCASE)");
+       case E_PMINFO_APPINFO_PROP_APP_SUPPORT_MODE:
+               snprintf(buf, sizeof(buf), "ai.app_support_mode=?");
                break;
+       case E_PMINFO_PKGINFO_PROP_PACKAGE_CHECK_STORAGE:
+       case E_PMINFO_APPINFO_PROP_APP_CHECK_STORAGE:
+               *condition = NULL;
+               return 0;
        default:
                _LOGE("Invalid Property Type\n");
                *condition = NULL;
@@ -404,6 +419,31 @@ int __get_filter_condition(gpointer data, char **condition, GList **params)
        return flag;
 }
 
+int __get_metadata_filter_condition(gpointer data, char **condition,
+               GList **params)
+{
+       pkgmgrinfo_metadata_node_x *node = (pkgmgrinfo_metadata_node_x *)data;
+       char buf[MAX_QUERY_LEN];
+       size_t len = 0;
+
+       snprintf(buf, sizeof(buf), "(package_app_app_metadata.md_key=?");
+       len += strlen("(package_app_app_metadata.md_key=?");
+       if (node->value) {
+               strncat(buf, " AND package_app_app_metadata.md_value=?",
+                               sizeof(buf) - len - 1);
+               len += strlen(" AND package_app_app_metadata.md_value=?");
+       }
+       strncat(buf, ")", sizeof(buf) - len - 1);
+       len += strlen(")");
+
+       *condition = strdup(buf);
+       *params = g_list_append(*params, strdup(node->key));
+       if (node->value)
+               *params = g_list_append(*params, strdup(node->value));
+
+       return E_PMINFO_APPINFO_JOIN_METADATA;
+}
+
 int _add_icon_info_into_list(const char *locale, char *value, GList **icon)
 {
        icon_x *info;
@@ -480,7 +520,7 @@ int __pkginfo_check_installed_storage(package_x *pkginfo)
 
        if (strcmp(pkginfo->installed_storage, "installed_external") == 0) {
                snprintf(buf, MAX_QUERY_LEN - 1, "%s", pkginfo->external_path);
-               ret = access(buf, F_OK);
+               ret = access(buf, R_OK);
                if (ret != 0) {
                        _LOGE("can not access [%s] - %d", buf, ret);
                        return PMINFO_R_ERROR;
@@ -507,3 +547,34 @@ int __appinfo_check_installed_storage(application_x *appinfo)
        return PMINFO_R_OK;
 }
 
+#define BUSY_WAITING_USEC (1000000 / 10 / 2) /* 0.05 sec */
+#define BUSY_WAITING_MAX 20 /* wait for max 1 sec */
+static int __db_busy_handler(void *data, int count)
+{
+       if (count < BUSY_WAITING_MAX) {
+               usleep(BUSY_WAITING_USEC);
+               return 1;
+       } else {
+               /* sqlite3_prepare_v2 will return SQLITE_BUSY */
+               return 0;
+       }
+}
+
+int __open_db(const char *path, sqlite3 **db, int flags)
+{
+       int ret;
+
+       ret = sqlite3_open_v2(path, db, flags, NULL);
+       if (ret != SQLITE_OK)
+               return ret;
+
+       ret = sqlite3_busy_handler(*db, __db_busy_handler, NULL);
+       if (ret != SQLITE_OK) {
+               _LOGE("failed to register busy handler: %s",
+                               sqlite3_errmsg(*db));
+               sqlite3_close_v2(*db);
+               return ret;
+       }
+
+       return ret;
+}