Remove storage type
[platform/core/api/media-content.git] / src / media_content.c
1 /*
2 * Copyright (c) 2011 Samsung Electronics Co., Ltd All Rights Reserved
3 *
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
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
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.
15 */
16
17 #include <media_info_private.h>
18 #include <media_util_private.h>
19 #include <mntent.h>
20 #include <tzplatform_config.h>
21
22 static sqlite3 *db_handle = NULL;
23 static int ref_count = 0;
24 static GMutex db_mutex;
25 static uid_t content_g_uid = 0;
26
27 sqlite3 * _content_get_db_handle(void)
28 {
29         return db_handle;
30 }
31
32 uid_t _content_get_uid(void)
33 {
34         if (content_g_uid == 0)
35                 return tzplatform_getuid(TZ_USER_NAME);
36         else
37                 return content_g_uid;
38 }
39
40 int _content_query_prepare(const char *select_query, const char *condition_query, const char *option_query, sqlite3_stmt **stmt)
41 {
42         int ret = MEDIA_CONTENT_ERROR_NONE;
43         char *query = NULL;
44
45         content_retip_if_fail(select_query);
46
47         /* If 'option_query' is NULL, 'condition_query' is also NULL. */
48         if (option_query) {
49                 if (STRING_VALID(condition_query))
50                         query = sqlite3_mprintf("%s AND %s %s", select_query, condition_query, option_query);
51                 else
52                         query = sqlite3_mprintf("%s %s", select_query, option_query);
53
54         } else {
55                 query = sqlite3_mprintf("%s", select_query);
56         }
57
58         ret = _content_get_result(query, stmt);
59         SQLITE3_SAFE_FREE(query);
60
61         return ret;
62 }
63
64 int _content_get_result(char *query, sqlite3_stmt **stmt)
65 {
66         int err = MEDIA_CONTENT_ERROR_NONE;
67
68         content_retip_if_fail(_content_get_db_handle());
69         content_retip_if_fail(query);
70
71         content_sec_debug("Query[%s]", query);
72
73         err = sqlite3_prepare_v2(_content_get_db_handle(), query, strlen(query), stmt, NULL);
74         if (err != SQLITE_OK) {
75                 content_error("Failed to sqlite3_prepare_v2[%s]", sqlite3_errmsg(_content_get_db_handle()));
76                 if (err == SQLITE_BUSY)
77                         return MEDIA_CONTENT_ERROR_DB_BUSY;
78                 else if (err == SQLITE_PERM)
79                         return MEDIA_CONTENT_ERROR_PERMISSION_DENIED;
80                 else
81                         return MEDIA_CONTENT_ERROR_DB_FAILED;
82         }
83
84         return MEDIA_CONTENT_ERROR_NONE;
85 }
86
87 int _content_error_capi(int internal_error)
88 {
89         if (internal_error == MS_MEDIA_ERR_NONE)
90                 return MEDIA_CONTENT_ERROR_NONE;
91
92         content_error("MS Error[%d]", internal_error);
93
94         switch (internal_error) {
95         case MS_MEDIA_ERR_INVALID_PARAMETER:
96                 return MEDIA_CONTENT_ERROR_INVALID_PARAMETER;
97         case MS_MEDIA_ERR_OUT_OF_MEMORY:
98                 return MEDIA_CONTENT_ERROR_OUT_OF_MEMORY;
99         case MS_MEDIA_ERR_DB_BUSY_FAIL:
100                 return MEDIA_CONTENT_ERROR_DB_BUSY;
101         case MS_MEDIA_ERR_DB_CONSTRAINT_FAIL:
102         case MS_MEDIA_ERR_DB_NO_RECORD:
103         case MS_MEDIA_ERR_DB_CORRUPT:
104         case MS_MEDIA_ERR_DB_FULL_FAIL:
105         case MS_MEDIA_ERR_DB_RESET:
106         case MS_MEDIA_ERR_DB_INTERNAL:
107                 return MEDIA_CONTENT_ERROR_DB_FAILED;
108         case MS_MEDIA_ERR_IPC:
109                 return MEDIA_CONTENT_ERROR_NETWORK;
110         case MS_MEDIA_ERR_PERMISSION_DENIED:
111                 return MEDIA_CONTENT_ERROR_PERMISSION_DENIED;
112         case MS_MEDIA_ERR_THUMB_UNSUPPORTED:
113                 return MEDIA_CONTENT_ERROR_UNSUPPORTED_CONTENT;
114         default:
115                 return MEDIA_CONTENT_ERROR_INVALID_OPERATION;
116         }
117 }
118
119 int _content_query_sql(char *query_str)
120 {
121         int ret = MEDIA_CONTENT_ERROR_NONE;
122
123         /*DB will be updated by Media Server.*/
124         ret = media_db_request_update_db(query_str, _content_get_uid());
125
126         return _content_error_capi(ret);
127 }
128
129 int media_content_connect(void)
130 {
131         int ret = MEDIA_CONTENT_ERROR_NONE;
132
133         g_mutex_lock(&db_mutex);
134         content_info("ref count : %d", ref_count);
135
136         if (ref_count == 0) {
137                 if (db_handle == NULL) {
138                         ret = media_db_connect(&db_handle, _content_get_uid(), false);
139                         ret = _content_error_capi(ret);
140                         if (ret == MEDIA_CONTENT_ERROR_NONE)
141                                 ref_count++;
142                 } else {
143                         content_error("Wrong DB Connection status");
144                         ret = MEDIA_CONTENT_ERROR_DB_FAILED;
145                 }
146         } else {
147                 if (db_handle != NULL) {
148                         ref_count++;
149                 } else {
150                         content_error("Wrong DB Handle status");
151                         ret = MEDIA_CONTENT_ERROR_DB_FAILED;
152                 }
153         }
154
155         content_info("ref count changed to: %d", ref_count);
156         g_mutex_unlock(&db_mutex);
157
158         return ret;
159 }
160
161 int media_content_connect_with_uid(uid_t uid)
162 {
163         content_sec_debug("media_content_connect_with_uid [%d]", uid);
164         content_g_uid = uid;
165
166         return media_content_connect();
167 }
168
169 int media_content_disconnect(void)
170 {
171         g_mutex_lock(&db_mutex);
172         content_debug("ref count : %d", ref_count);
173
174         if (db_handle && ref_count > 0) {
175                 if (--ref_count == 0) {
176                         media_db_disconnect(db_handle);
177                         db_handle = NULL;
178                 }
179         } else {
180                 content_error("Database is not connected");
181                 g_mutex_unlock(&db_mutex);
182                 return MEDIA_CONTENT_ERROR_DB_FAILED;
183         }
184
185         g_mutex_unlock(&db_mutex);
186
187         content_info("ref count changed to: %d", ref_count);
188
189         return MEDIA_CONTENT_ERROR_NONE;
190 }
191
192 int media_content_scan_file(const char *path)
193 {
194         int ret = MEDIA_CONTENT_ERROR_NONE;
195         char *folder_path = NULL;
196         int check_file = MEDIA_CONTENT_ERROR_NONE;
197         char storage_id[MEDIA_CONTENT_UUID_SIZE+1] = {0,};
198
199         content_retip_if_fail(STRING_VALID(path));
200
201         content_sec_debug("Path : %s", path);
202
203         content_retvm_if(_media_util_is_ignorable_file(path), MEDIA_CONTENT_ERROR_INVALID_PARAMETER, "Invalid path");
204
205         ret = media_svc_get_storage_id(_content_get_db_handle(), path, storage_id, _content_get_uid());
206         if (ret != MS_MEDIA_ERR_NONE) {
207                 content_error("media_svc_get_storage_id failed : %d", ret);
208                 return _content_error_capi(ret);
209         }
210
211         check_file = _media_util_check_file_exist(path);
212         if (check_file == MEDIA_CONTENT_ERROR_NONE) {
213                 /* This means this path has to be inserted or refreshed */
214                 folder_path = g_path_get_dirname(path);
215
216                 if (_media_util_is_ignorable_dir(folder_path)) {
217                         g_free(folder_path);
218                         return MEDIA_CONTENT_ERROR_INVALID_PARAMETER;
219                 }
220
221                 g_free(folder_path);
222
223                 /* check feature */
224                 content_retvm_if(!_media_util_check_support_media_type(path), MEDIA_CONTENT_ERROR_NOT_SUPPORTED, "Unsupported media type");
225
226                 ret = media_svc_check_item_exist_by_path(_content_get_db_handle(), storage_id, path);
227                 if (ret == MS_MEDIA_ERR_NONE) {
228                         /* Refresh */
229                         ret = media_svc_refresh_item(_content_get_db_handle(), false, storage_id, path, _content_get_uid());
230                         if (ret != MS_MEDIA_ERR_NONE) {
231                                 content_error("media_svc_refresh_item failed : %d", ret);
232                                 return _content_error_capi(ret);
233                         }
234
235                 } else if (ret == MS_MEDIA_ERR_DB_NO_RECORD) {
236                         /* Insert */
237                         ret = media_svc_insert_item_immediately(_content_get_db_handle(), storage_id, path, _content_get_uid());
238                         if (ret != MS_MEDIA_ERR_NONE) {
239                                 if (ret == MS_MEDIA_ERR_DB_CONSTRAINT_FAIL) {
240                                         content_sec_error("This item is already inserted. This may be normal operation because other process already did this (%s)", path);
241                                         ret = MEDIA_CONTENT_ERROR_NONE;
242                                 } else {
243                                         content_sec_error("media_svc_insert_item_immediately failed : %d (%s)", ret, path);
244                                 }
245
246                                 return _content_error_capi(ret);
247                         }
248                 } else {
249                         content_error("media_svc_check_item_exist_by_path failed : %d", ret);
250                         return _content_error_capi(ret);
251                 }
252         } else if (check_file == MEDIA_CONTENT_ERROR_PERMISSION_DENIED) {
253                 content_error("You have no permission for this file %d", ret);
254                 return MEDIA_CONTENT_ERROR_PERMISSION_DENIED;
255         } else {
256                 /* This means this path has to be deleted */
257                 content_debug("This path doesn't exists in file system... So now start to delete it from DB");
258                 ret = media_svc_delete_item_by_path(_content_get_db_handle(), storage_id, path, _content_get_uid());
259                 if (ret != MS_MEDIA_ERR_NONE) {
260                         if (ret == MS_MEDIA_ERR_DB_NO_RECORD) {
261                                 content_error("Does not exist in media DB also... So, this is an invalid parameter");
262                                 return MEDIA_CONTENT_ERROR_INVALID_PARAMETER;
263                         }
264
265                         content_error("media_svc_delete_item_by_path failed : %d", ret);
266                         return _content_error_capi(ret);
267                 }
268         }
269
270         return _content_error_capi(ret);
271 }
272
273 void _media_content_scan_cb(media_request_result_s* result, void *user_data)
274 {
275         int err = -1;
276         media_content_scan_cb_data *cb_data = user_data;
277
278         err = _content_error_capi(result->result);
279 #ifdef _USE_TVPD_MODE
280         if (result->request_type != MEDIA_REQUEST_SCAN_COMPLETE &&
281                 result->request_type != MEDIA_REQUEST_SCAN_PARTIAL &&
282                 result->request_type != MEDIA_RECURSIVE_START) {
283                 if (cb_data && cb_data->callback) {
284                         content_debug("begin:User callback is being called now, result=%d", err);
285                         cb_data->callback(err, cb_data->user_data);
286                         content_debug("end:User callback is being called now, result=%d", err);
287                 }
288
289                 g_free(cb_data);
290         }
291 #else
292         if (cb_data && cb_data->callback) {
293                 content_debug("User callback is being called now");
294                 cb_data->callback(err, cb_data->user_data);
295         }
296
297         g_free(cb_data);
298 #endif
299
300         return;
301 }
302
303 #ifdef _USE_TVPD_MODE
304 void _media_content_scan_cb_v2(media_request_result_s* result, void *user_data)
305 {
306         int err = -1;
307         media_content_scan_cb_data_v2 *cb_data = user_data;
308         media_content_complete_phase_e complete_phase = -1;
309         if (!cb_data)
310                 content_debug("cb_data is NULL");
311         err = _content_error_capi(result->result);
312         content_debug("result is %d", err);
313
314         if (result->request_type == MEDIA_REQUEST_SCAN_PARTIAL)
315                 complete_phase = MEDIA_CONTENT_SCAN_PARTIAL_COMPLETE;
316         else if (result->request_type == MEDIA_REQUEST_SCAN_COMPLETE)
317                 complete_phase = MEDIA_CONTENT_SCAN_COMPLETE;
318         else if (result->request_type == MEDIA_REQUEST_EXTRACT_COMPLETE)
319                 complete_phase = MEDIA_CONTENT_EXTRACT_COMPLETE;
320         else if (result->request_type == MEDIA_RECURSIVE_START)
321                 complete_phase = MEDIA_CONTENT_RECURSIVE_START;
322
323         if (cb_data && cb_data->callback)
324                 cb_data->callback(err, complete_phase, cb_data->user_data);
325         else
326                 content_debug("run error");
327
328         if ((result->request_type != MEDIA_REQUEST_SCAN_COMPLETE) &&
329         (result->request_type != MEDIA_REQUEST_SCAN_PARTIAL) &&
330         (result->request_type != MEDIA_RECURSIVE_START))
331                 g_free(cb_data);
332
333         return;
334 }
335 #endif
336
337 int media_content_scan_folder(const char *path, bool is_recursive, media_scan_completed_cb callback, void *user_data)
338 {
339         int ret = MEDIA_CONTENT_ERROR_NONE;
340         char storage_id[MEDIA_CONTENT_UUID_SIZE+1] = {0, };
341
342         content_retip_if_fail(STRING_VALID(path));
343         content_retip_if_fail(callback);
344
345         ret = _media_content_check_dir(path);
346         content_retvm_if(ret == MEDIA_CONTENT_ERROR_PERMISSION_DENIED, ret, "Permission Denied");
347
348         if (ret == MEDIA_CONTENT_ERROR_NONE) {
349                 /* If directory exist check that's ignore directory or not*/
350                 content_retvm_if(_media_util_is_ignorable_dir(path), MEDIA_CONTENT_ERROR_INVALID_PARAMETER, "Invalid folder path");
351         } else {
352                 /* This means this folder has to be deleted */
353                 /* Or, it is real invalid path.. check storage type */
354                 if (!ms_user_is_valid_path(_content_get_uid(), path)) {
355                         content_sec_error("ms_user_is_valid_path failed : %d (%s)", ret, path);
356                         return _content_error_capi(ret);
357                 }
358
359                 content_debug("This path doesn't exists in file system... So will be deleted it from DB");
360         }
361
362         ret = media_svc_get_storage_id(_content_get_db_handle(), path, storage_id, _content_get_uid());
363         if (ret != MS_MEDIA_ERR_NONE) {
364                 content_error("media_svc_get_storage_id failed : %d", ret);
365                 return _content_error_capi(ret);
366         }
367
368         media_content_scan_cb_data *cb_data = NULL;
369         cb_data = g_new0(media_content_scan_cb_data, 1);
370
371         cb_data->callback = callback;
372         cb_data->user_data = user_data;
373
374         ret = media_directory_scanning_async(path, storage_id, is_recursive, _media_content_scan_cb, cb_data, _content_get_uid());
375         if (ret != MS_MEDIA_ERR_NONE) {
376                 content_error("media_directory_scanning_async failed : %d", ret);
377                 g_free(cb_data);
378         }
379
380         return _content_error_capi(ret);
381 }
382
383 #ifdef _USE_TVPD_MODE
384 bool _is_mounted(const char *dir_path)
385 {
386         bool ret = false;
387         char mount_path[MAX_PATH_LEN] = {0,};
388         char root_path[MAX_PATH_LEN] = {0,};
389         struct mntent *mnt;
390         const char *table = "/etc/mtab";
391         FILE *fp;
392
393         snprintf(root_path, MAX_PATH_LEN, "%s%s", tzplatform_getenv(TZ_SYS_STORAGE), "/USBDrive");
394         if (!g_str_has_prefix(dir_path, root_path))
395                 return true;
396
397         char *p = strstr(dir_path + strlen(root_path), "/");
398         if(p && p - dir_path < MAX_PATH_LEN)
399                 strncpy(mount_path, dir_path, p - dir_path);
400         else
401                 snprintf(mount_path, MAX_PATH_LEN, "%s", dir_path);
402
403         fp = setmntent(table, "r");
404         if (!fp) {
405                 content_error("open /etc/mtab failed");
406                 return false;
407         }
408         while ((mnt = getmntent(fp))) {
409                 if (!strncmp(mnt->mnt_dir, mount_path, strlen(mount_path))) {
410                         ret = true;
411                         break;
412                 }
413         }
414         endmntent(fp);
415
416         content_error("[NO ERROR] dir_path[%s] mount_path[%s] ret[%d]", dir_path, mount_path, ret);
417         return ret;
418 }
419
420 int media_content_scan_folder_v2(const char *path, bool is_recursive, media_scan_completed_cb_v2 callback, void *user_data)
421 {
422         int ret = MEDIA_CONTENT_ERROR_NONE;
423         char storage_id[MEDIA_CONTENT_UUID_SIZE+1] = {0, };
424
425         content_retip_if_fail(STRING_VALID(path));
426         content_retip_if_fail(callback);
427
428         content_retvm_if(!_is_mounted(path), MEDIA_CONTENT_ERROR_PERMISSION_DENIED, "path is not mounted");
429         content_retvm_if(_media_util_is_ignorable_dir(path), MEDIA_CONTENT_ERROR_INVALID_PARAMETER, "Invalid folder path");
430
431         ret = _media_content_check_dir(path);
432         content_retvm_if(ret == MEDIA_CONTENT_ERROR_PERMISSION_DENIED, ret, "Permission Denied");
433         content_retvm_if(ret == MEDIA_CONTENT_ERROR_INVALID_PARAMETER, ret, "invalid path[%s]", path);
434
435         media_content_scan_cb_data_v2* cb_data = NULL;
436         cb_data = g_new0(media_content_scan_cb_data_v2, 1);
437
438         cb_data->callback = callback;
439         cb_data->user_data = user_data;
440
441         ret = media_svc_get_storage_id(_content_get_db_handle(), path, storage_id, _content_get_uid());
442         /*FIX ME. need to check ret value?*/
443
444         ret = media_directory_scanning_async(path, storage_id, is_recursive, _media_content_scan_cb_v2, cb_data, _content_get_uid());
445         if (ret != MS_MEDIA_ERR_NONE)
446                 content_error("media_directory_scanning_async failed : %d", ret);
447
448         return _content_error_capi(ret);
449 }
450 #endif
451
452 int media_content_cancel_scan_folder(const char *path)
453 {
454         int ret = MEDIA_CONTENT_ERROR_NONE;
455
456         content_retip_if_fail(STRING_VALID(path));
457
458         ret = media_directory_scanning_cancel(path, _content_get_uid());
459         if (ret != MS_MEDIA_ERR_NONE)
460                 content_error("media_directory_scanning_async failed : %d", ret);
461
462         return _content_error_capi(ret);
463 }
464
465 void _media_content_db_update_noti_cb(
466                                                         int pid,
467                                                         media_item_type_e item,
468                                                         media_item_update_type_e update_type,
469                                                         char* path,
470                                                         char* uuid,
471                                                         media_type_e content_type,
472                                                         char *mime_type,
473                                                         void *user_data)
474 {
475         media_noti_cb_s *_noti_info = (media_noti_cb_s *)user_data;
476
477         if (_noti_info != NULL && _noti_info->update_noti_cb)
478                 _noti_info->update_noti_cb(
479                                         MEDIA_CONTENT_ERROR_NONE,
480                                         pid,
481                                         item,
482                                         update_type,
483                                         content_type,
484                                         uuid,
485                                         path,
486                                         mime_type,
487                                         _noti_info->user_data);
488 }
489
490 int media_content_add_db_updated_cb(media_content_db_update_cb callback, void *user_data, media_content_noti_h *noti_handle)
491 {
492         int ret = MEDIA_CONTENT_ERROR_NONE;
493         media_noti_cb_s *noti_info = NULL;
494
495         content_retip_if_fail(callback);
496         content_retip_if_fail(noti_handle);
497
498         noti_info = g_new0(media_noti_cb_s, 1);
499
500         noti_info->update_noti_cb = callback;
501         noti_info->user_data = user_data;
502
503         ret = media_db_update_subscribe((MediaNotiHandle*)noti_handle, _media_content_db_update_noti_cb, (void *)noti_info);
504         if (ret != MS_MEDIA_ERR_NONE)
505                 g_free(noti_info);
506
507         return _content_error_capi(ret);
508 }
509
510 int media_content_remove_db_updated_cb(media_content_noti_h noti_handle)
511 {
512         int ret = MEDIA_CONTENT_ERROR_NONE;
513
514         ret = media_db_update_unsubscribe((MediaNotiHandle)noti_handle);
515
516         return _content_error_capi(ret);
517 }
518 #ifdef _USE_TVPD_MODE
519 GMutex* _content_get_db_mutex(void)
520 {
521         return &db_mutex;
522 }
523 #endif