69df521b71fefa10e06cfab508a939cfff4ad4f5
[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
18 #include <media_info_private.h>
19 #include <media_util_private.h>
20
21 static sqlite3 *db_handle = NULL;
22 static int ref_count = 0;
23 static GMutex db_mutex;
24 static uid_t content_g_uid = 0;
25
26 sqlite3 * _content_get_db_handle(void)
27 {
28         return db_handle;
29 }
30
31 uid_t _content_get_uid(void)
32 {
33         if (content_g_uid == 0)
34                 return tzplatform_getuid(TZ_USER_NAME);
35         else
36                 return content_g_uid;
37 }
38
39 int _content_query_prepare(const char *select_query, const char *condition_query, const char *option_query, sqlite3_stmt **stmt)
40 {
41         int ret = MEDIA_CONTENT_ERROR_NONE;
42         char *query = NULL;
43
44         content_retip_if_fail(select_query);
45
46         /* If 'option_query' is NULL, 'condition_query' is also NULL. */
47         if (option_query) {
48                 if (STRING_VALID(condition_query))
49                         query = sqlite3_mprintf("%s AND %s %s", select_query, condition_query, option_query);
50                 else
51                         query = sqlite3_mprintf("%s %s", select_query, option_query);
52
53         } else {
54                 query = sqlite3_mprintf("%s", select_query);
55         }
56
57         ret = _content_get_result(query, stmt);
58         SQLITE3_SAFE_FREE(query);
59
60         return ret;
61 }
62
63 int _content_get_result(char *query, sqlite3_stmt **stmt)
64 {
65         int err = MEDIA_CONTENT_ERROR_NONE;
66
67         content_retip_if_fail(_content_get_db_handle());
68         content_retip_if_fail(query);
69
70         content_sec_debug("Query[%s]", query);
71
72         err = sqlite3_prepare_v2(_content_get_db_handle(), query, strlen(query), stmt, NULL);
73         if (err != SQLITE_OK) {
74                 content_error("Failed to sqlite3_prepare_v2[%s]", sqlite3_errmsg(_content_get_db_handle()));
75                 if (err == SQLITE_BUSY)
76                         return MEDIA_CONTENT_ERROR_DB_BUSY;
77                 else if (err == SQLITE_PERM)
78                         return MEDIA_CONTENT_ERROR_PERMISSION_DENIED;
79                 else
80                         return MEDIA_CONTENT_ERROR_DB_FAILED;
81         }
82
83         return MEDIA_CONTENT_ERROR_NONE;
84 }
85
86 int _content_error_capi(int internal_error)
87 {
88         if (internal_error == MS_MEDIA_ERR_NONE)
89                 return MEDIA_CONTENT_ERROR_NONE;
90
91         content_error("MS Error[%d]", internal_error);
92
93         switch (internal_error) {
94         case MS_MEDIA_ERR_INVALID_PARAMETER:
95                 return MEDIA_CONTENT_ERROR_INVALID_PARAMETER;
96         case MS_MEDIA_ERR_OUT_OF_MEMORY:
97                 return MEDIA_CONTENT_ERROR_OUT_OF_MEMORY;
98         case MS_MEDIA_ERR_DB_BUSY_FAIL:
99                 return MEDIA_CONTENT_ERROR_DB_BUSY;
100         case MS_MEDIA_ERR_DB_CONSTRAINT_FAIL:
101         case MS_MEDIA_ERR_DB_NO_RECORD:
102         case MS_MEDIA_ERR_DB_CORRUPT:
103         case MS_MEDIA_ERR_DB_FULL_FAIL:
104         case MS_MEDIA_ERR_DB_RESET:
105         case MS_MEDIA_ERR_DB_INTERNAL:
106                 return MEDIA_CONTENT_ERROR_DB_FAILED;
107         case MS_MEDIA_ERR_IPC:
108                 return MEDIA_CONTENT_ERROR_NETWORK;
109         case MS_MEDIA_ERR_PERMISSION_DENIED:
110                 return MEDIA_CONTENT_ERROR_PERMISSION_DENIED;
111         case MS_MEDIA_ERR_THUMB_TOO_BIG:
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         char repl_path[MAX_PATH_LEN] = {0,};
199
200         content_retip_if_fail(STRING_VALID(path));
201
202         content_sec_debug("Path : %s", path);
203
204         ret = _media_content_replace_path(path, repl_path);
205         content_retvm_if(!STRING_VALID(repl_path), MEDIA_CONTENT_ERROR_INVALID_OPERATION, "path replacement failed");
206
207         content_retvm_if(_media_util_is_ignorable_file(repl_path), MEDIA_CONTENT_ERROR_INVALID_PARAMETER, "Invalid path");
208
209         ret = media_svc_get_storage_id(_content_get_db_handle(), repl_path, storage_id, _content_get_uid());
210         if (ret != MS_MEDIA_ERR_NONE) {
211                 content_error("media_svc_get_storage_id failed : %d", ret);
212                 return _content_error_capi(ret);
213         }
214
215         check_file = _media_util_check_file_exist(repl_path);
216         if (check_file == MEDIA_CONTENT_ERROR_NONE) {
217                 /* This means this path has to be inserted or refreshed */
218                 folder_path = g_path_get_dirname(repl_path);
219
220                 if (_media_util_is_ignorable_dir(folder_path)) {
221                         g_free(folder_path);
222                         return MEDIA_CONTENT_ERROR_INVALID_PARAMETER;
223                 }
224
225                 g_free(folder_path);
226
227                 /* check feature */
228                 content_retvm_if(!_media_util_check_support_media_type(repl_path), MEDIA_CONTENT_ERROR_NOT_SUPPORTED, "Unsupported media type");
229
230                 ms_user_storage_type_e storage_type;
231
232                 ret = ms_user_get_storage_type(_content_get_uid(), repl_path, &storage_type);
233                 if (ret != MS_MEDIA_ERR_NONE) {
234                         content_sec_error("ms_user_get_storage_type failed : %d (%s)", ret, repl_path);
235                         return _content_error_capi(ret);
236                 }
237                 ret = media_svc_check_item_exist_by_path(_content_get_db_handle(), storage_id, repl_path);
238                 if (ret == MS_MEDIA_ERR_NONE) {
239                         /* Refresh */
240                         ret = media_svc_refresh_item(_content_get_db_handle(), false, storage_id, storage_type, repl_path, _content_get_uid());
241                         if (ret != MS_MEDIA_ERR_NONE) {
242                                 content_error("media_svc_refresh_item failed : %d", ret);
243                                 return _content_error_capi(ret);
244                         }
245
246                 } else if (ret == MS_MEDIA_ERR_DB_NO_RECORD) {
247                         /* Insert */
248                         ret = media_svc_insert_item_immediately(_content_get_db_handle(), storage_id, storage_type, repl_path, _content_get_uid());
249                         if (ret != MS_MEDIA_ERR_NONE) {
250                                 if (ret == MS_MEDIA_ERR_DB_CONSTRAINT_FAIL) {
251                                         content_sec_error("This item is already inserted. This may be normal operation because other process already did this (%s)", repl_path);
252                                         ret = MEDIA_CONTENT_ERROR_NONE;
253                                 } else {
254                                         content_sec_error("media_svc_insert_item_immediately failed : %d (%s)", ret, repl_path);
255                                 }
256
257                                 return _content_error_capi(ret);
258                         }
259                 } else {
260                         content_error("media_svc_check_item_exist_by_path failed : %d", ret);
261                         return _content_error_capi(ret);
262                 }
263         } else if (check_file == MEDIA_CONTENT_ERROR_PERMISSION_DENIED) {
264                 content_error("You have no permission for this file %d", ret);
265                 return MEDIA_CONTENT_ERROR_PERMISSION_DENIED;
266         } else {
267                 /* This means this path has to be deleted */
268                 content_debug("This path doesn't exists in file system... So now start to delete it from DB");
269                 ret = media_svc_delete_item_by_path(_content_get_db_handle(), storage_id, repl_path, _content_get_uid());
270                 if (ret != MS_MEDIA_ERR_NONE) {
271                         if (ret == MS_MEDIA_ERR_DB_NO_RECORD) {
272                                 content_error("Does not exist in media DB also... So, this is an invalid parameter");
273                                 return MEDIA_CONTENT_ERROR_INVALID_PARAMETER;
274                         }
275
276                         content_error("media_svc_delete_item_by_path failed : %d", ret);
277                         return _content_error_capi(ret);
278                 }
279         }
280
281         return _content_error_capi(ret);
282 }
283
284 void _media_content_scan_cb(media_request_result_s* result, void *user_data)
285 {
286         int err = -1;
287         media_content_scan_cb_data *cb_data = user_data;
288
289         err = _content_error_capi(result->result);
290 #ifdef _USE_TVPD_MODE
291         if (result->request_type != MEDIA_REQUEST_SCAN_COMPLETE &&
292                 result->request_type != MEDIA_REQUEST_SCAN_PARTIAL &&
293                 result->request_type != MEDIA_RECURSIVE_START) {
294                 if (cb_data && cb_data->callback) {
295                         content_debug("begin:User callback is being called now, result=%d", err);
296                         cb_data->callback(err, cb_data->user_data);
297                         content_debug("end:User callback is being called now, result=%d", err);
298                 }
299
300                 g_free(cb_data);
301         }
302 #else
303         if (cb_data && cb_data->callback) {
304                 content_debug("User callback is being called now");
305                 cb_data->callback(err, cb_data->user_data);
306         }
307
308         g_free(cb_data);
309 #endif
310
311         return;
312 }
313
314 #ifdef _USE_TVPD_MODE
315 void _media_content_scan_cb_v2(media_request_result_s* result, void *user_data)
316 {
317         int err = -1;
318         media_content_scan_cb_data_v2 *cb_data = user_data;
319         media_content_complete_phase_e complete_phase = -1;
320         if (!cb_data)
321                 content_debug("cb_data is NULL");
322         err = _content_error_capi(result->result);
323         content_debug("result is %d", err);
324
325         if (result->request_type == MEDIA_REQUEST_SCAN_PARTIAL)
326                 complete_phase = MEDIA_CONTENT_SCAN_PARTIAL_COMPLETE;
327         else if (result->request_type == MEDIA_REQUEST_SCAN_COMPLETE)
328                 complete_phase = MEDIA_CONTENT_SCAN_COMPLETE;
329         else if (result->request_type == MEDIA_REQUEST_EXTRACT_COMPLETE)
330                 complete_phase = MEDIA_CONTENT_EXTRACT_COMPLETE;
331         else if (result->request_type == MEDIA_RECURSIVE_START)
332                 complete_phase = MEDIA_CONTENT_RECURSIVE_START;
333
334         if (cb_data && cb_data->callback)
335                 cb_data->callback(err, complete_phase, cb_data->user_data);
336         else
337                 content_debug("run error");
338
339         if ((result->request_type != MEDIA_REQUEST_SCAN_COMPLETE) &&
340         (result->request_type != MEDIA_REQUEST_SCAN_PARTIAL) &&
341         (result->request_type != MEDIA_RECURSIVE_START))
342                 g_free(cb_data);
343
344         return;
345 }
346 #endif
347
348 int media_content_scan_folder(const char *path, bool is_recursive, media_scan_completed_cb callback, void *user_data)
349 {
350         int ret = MEDIA_CONTENT_ERROR_NONE;
351         char storage_id[MEDIA_CONTENT_UUID_SIZE+1] = {0, };
352         char repl_path[MAX_PATH_LEN] = {0, };
353         ms_user_storage_type_e storage_type = MS_USER_STORAGE_INTERNAL;
354
355         content_retip_if_fail(STRING_VALID(path));
356         content_retip_if_fail(callback);
357
358         ret = _media_content_replace_path(path, repl_path);
359         content_retvm_if(!STRING_VALID(repl_path), MEDIA_CONTENT_ERROR_INVALID_OPERATION, "path replacement failed");
360
361         ret = _media_content_check_dir(repl_path);
362         content_retvm_if(ret == MEDIA_CONTENT_ERROR_PERMISSION_DENIED, ret, "Permission Denied");
363
364         if (ret == MEDIA_CONTENT_ERROR_NONE) {
365                 /* If directory exist check that's ignore directory or not*/
366                 content_retvm_if(_media_util_is_ignorable_dir(repl_path), MEDIA_CONTENT_ERROR_INVALID_PARAMETER, "Invalid folder path");
367         } else {
368                 /* This means this folder has to be deleted */
369                 /* Or, it is real invalid path.. check storage type */
370                 ret = ms_user_get_storage_type(_content_get_uid(), repl_path, &storage_type);
371                 if (ret != MS_MEDIA_ERR_NONE) {
372                         content_sec_error("ms_user_get_storage_type failed : %d (%s)", ret, repl_path);
373                         return _content_error_capi(ret);
374                 }
375
376                 content_debug("This path doesn't exists in file system... So will be deleted it from DB");
377         }
378
379         ret = media_svc_get_storage_id(_content_get_db_handle(), repl_path, storage_id, _content_get_uid());
380         if (ret != MS_MEDIA_ERR_NONE) {
381                 content_error("media_svc_get_storage_id failed : %d", ret);
382                 return _content_error_capi(ret);
383         }
384
385         media_content_scan_cb_data *cb_data = NULL;
386         cb_data = g_new0(media_content_scan_cb_data, 1);
387
388         cb_data->callback = callback;
389         cb_data->user_data = user_data;
390
391         ret = media_directory_scanning_async(repl_path, storage_id, is_recursive, _media_content_scan_cb, cb_data, _content_get_uid());
392         if (ret != MS_MEDIA_ERR_NONE) {
393                 content_error("media_directory_scanning_async failed : %d", ret);
394                 g_free(cb_data);
395         }
396
397         return _content_error_capi(ret);
398 }
399
400 #ifdef _USE_TVPD_MODE
401 int media_content_scan_folder_v2(const char *path, bool is_recursive, media_scan_completed_cb_v2 callback, void *user_data)
402 {
403         int ret = MEDIA_CONTENT_ERROR_NONE;
404         char storage_id[MEDIA_CONTENT_UUID_SIZE+1] = {0, };
405
406         content_retip_if_fail(STRING_VALID(path));
407         content_retip_if_fail(callback);
408
409         content_retvm_if(_media_util_is_ignorable_dir(path), MEDIA_CONTENT_ERROR_INVALID_PARAMETER, "Invalid folder path");
410
411         ret = _media_content_check_dir(path);
412         content_retvm_if(ret == MEDIA_CONTENT_ERROR_PERMISSION_DENIED, ret, "Permission Denied");
413         content_retvm_if(ret == MEDIA_CONTENT_ERROR_INVALID_PARAMETER, ret, "invalid path[%s]", path);
414
415         media_content_scan_cb_data_v2* cb_data = NULL;
416         cb_data = g_new0(media_content_scan_cb_data_v2, 1);
417
418         cb_data->callback = callback;
419         cb_data->user_data = user_data;
420
421         ret = media_svc_get_storage_id(_content_get_db_handle(), path, storage_id, _content_get_uid());
422         /*FIX ME. need to check ret value?*/
423
424         ret = media_directory_scanning_async(path, storage_id, is_recursive, _media_content_scan_cb_v2, cb_data, _content_get_uid());
425         if (ret != MS_MEDIA_ERR_NONE)
426                 content_error("media_directory_scanning_async failed : %d", ret);
427
428         return _content_error_capi(ret);
429 }
430 #endif
431
432 int media_content_cancel_scan_folder(const char *path)
433 {
434         int ret = MEDIA_CONTENT_ERROR_NONE;
435         char repl_path[MAX_PATH_LEN] = {0, };
436
437         content_retip_if_fail(STRING_VALID(path));
438
439         ret = _media_content_replace_path(path, repl_path);
440         content_retvm_if(!STRING_VALID(repl_path), MEDIA_CONTENT_ERROR_INVALID_OPERATION, "path replacement failed");
441
442         ret = media_directory_scanning_cancel(repl_path, _content_get_uid());
443         if (ret != MS_MEDIA_ERR_NONE)
444                 content_error("media_directory_scanning_async failed : %d", ret);
445
446         return _content_error_capi(ret);
447 }
448
449 void _media_content_db_update_noti_cb(
450                                                         int pid,
451                                                         media_item_type_e item,
452                                                         media_item_update_type_e update_type,
453                                                         char* path,
454                                                         char* uuid,
455                                                         media_type_e content_type,
456                                                         char *mime_type,
457                                                         void *user_data)
458 {
459         media_noti_cb_s *_noti_info = (media_noti_cb_s *)user_data;
460
461         if (_noti_info != NULL && _noti_info->update_noti_cb)
462                 _noti_info->update_noti_cb(
463                                         MEDIA_CONTENT_ERROR_NONE,
464                                         pid,
465                                         item,
466                                         update_type,
467                                         content_type,
468                                         uuid,
469                                         path,
470                                         mime_type,
471                                         _noti_info->user_data);
472 }
473
474 int media_content_add_db_updated_cb(media_content_db_update_cb callback, void *user_data, media_content_noti_h *noti_handle)
475 {
476         int ret = MEDIA_CONTENT_ERROR_NONE;
477         media_noti_cb_s *noti_info = NULL;
478
479         content_retip_if_fail(callback);
480         content_retip_if_fail(noti_handle);
481
482         noti_info = g_new0(media_noti_cb_s, 1);
483
484         noti_info->update_noti_cb = callback;
485         noti_info->user_data = user_data;
486
487         ret = media_db_update_subscribe((MediaNotiHandle*)noti_handle, _media_content_db_update_noti_cb, (void *)noti_info);
488         if (ret != MS_MEDIA_ERR_NONE)
489                 g_free(noti_info);
490
491         return _content_error_capi(ret);
492 }
493
494 #ifdef TIZEN_FEATURE_COMPATIBILITY
495 int media_content_set_db_updated_cb(media_content_db_update_cb callback, void *user_data)
496 {
497         content_warn("DEPRECATION WARNING: media_content_set_db_updated_cb() is removed from 5.0.");
498
499         return MEDIA_CONTENT_ERROR_NONE;
500 }
501 #endif
502
503 int media_content_remove_db_updated_cb(media_content_noti_h noti_handle)
504 {
505         int ret = MEDIA_CONTENT_ERROR_NONE;
506
507         ret = media_db_update_unsubscribe((MediaNotiHandle)noti_handle);
508
509         return _content_error_capi(ret);
510 }
511 #ifdef _USE_TVPD_MODE
512 GMutex* _content_get_db_mutex(void)
513 {
514         return &db_mutex;
515 }
516 #endif