Bundles / src /

data.c

  1. /*
  2. * Copyright (c) 2016 Samsung Electronics Co., Ltd
  3. *
  4. * Licensed under the Flora License, Version 1.1 (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://floralicense.org/license/
  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 <Eina.h>
  18. #include <bundle.h>
  19. #include <message_port.h>
  20. #include "$(appName).h"
  21. #include "data.h"
  22. #include <efl_extension.h>
  23.  
  24. #define MESSAGE_PORT_RCV_NAME PACKAGE"_msg_port_local_rcv"
  25.  
  26. typedef struct data_callbacks {
  27. message_data_new_cb new_data_cb;
  28. } data_callbacks_t;
  29.  
  30. static struct data_info {
  31. data_callbacks_t callbacks;
  32. Eina_List *bundles_received;
  33. bundle *bundle_obj;
  34. int msg_port_rcv_id;
  35. } s_info = {
  36. .callbacks = {0,},
  37. .bundles_received = NULL,
  38. .bundle_obj = NULL,
  39. .msg_port_rcv_id = 0,
  40. };
  41.  
  42. static void _header_iterator_cb(const char *key, const int type, const bundle_keyval_t *kv, void *user_data);
  43. static void _bundle_clear_iterator_cb(const char *key, const int type, const bundle_keyval_t *kv, void *user_data);
  44. static void _keys_iterator_cb(const char *key, const int type, const bundle_keyval_t *kv, void *user_data);
  45. static void _message_received_cb(int local_port_id, const char *remote_app_id, const char *remote_port, bool trusted_remote_port, bundle *message, void *user_data);
  46. static inline bundle *_get_bundle(int index);
  47. static inline void _free_string_array(int count, char** keys);
  48.  
  49. /**
  50. * @brief Initialize functions for data module.
  51. * @param[in] new_data_cb Callback invoked when a new message is received. It is used to inform the view module about the message.
  52. */
  53. void data_initialize(message_data_new_cb new_data_cb)
  54. {
  55. s_info.callbacks.new_data_cb = new_data_cb;
  56.  
  57. int ret = message_port_register_local_port(MESSAGE_PORT_RCV_NAME, _message_received_cb, NULL);
  58. if (ret < 0) {
  59. dlog_print(DLOG_ERROR, LOG_TAG, "[%s:%d] message_port_register_local_port() failed. Error: %s", __FILE__, __LINE__, get_error_message(ret));
  60. return;
  61. } else {
  62. s_info.msg_port_rcv_id = ret;
  63. }
  64.  
  65. data_create_bundle();
  66. }
  67.  
  68. /**
  69. * @brief Finalize function for data module.
  70. */
  71. void data_finalize(void)
  72. {
  73. data_destroy_bundle();
  74.  
  75. /*
  76. * all the s_info.items_list items shall be released
  77. */
  78. if (s_info.bundles_received)
  79. s_info.bundles_received = eina_list_free(s_info.bundles_received);
  80.  
  81. if (s_info.msg_port_rcv_id > 0)
  82. message_port_unregister_local_port(s_info.msg_port_rcv_id);
  83. }
  84.  
  85. /**
  86. * @brief Function creates a bundle handle, adds a header and fills it with relevant data.
  87. * If the bundle handle already exists, then it is destroyed and a new one is created.
  88. * The newly created bundle handle is stored internally.
  89. * @return The function returns 'true' if the bundle handle is created successfully,
  90. * otherwise 'false' is returned.
  91. */
  92. bool data_create_bundle(void)
  93. {
  94. s_info.bundle_obj = bundle_create();
  95. if (!s_info.bundle_obj) {
  96. dlog_print(DLOG_ERROR, LOG_TAG, "bundle_create() failed.");
  97. return false;
  98. }
  99.  
  100. return true;
  101. }
  102.  
  103. /**
  104. * @brief Function destroys a bundle handle.
  105. * If the bundle handle does not exist, then it nothing happens.
  106. */
  107. void data_destroy_bundle(void)
  108. {
  109. int ret;
  110.  
  111. if (!s_info.bundle_obj)
  112. return;
  113.  
  114. ret = bundle_free(s_info.bundle_obj);
  115. if (ret != BUNDLE_ERROR_NONE) {
  116. dlog_print(DLOG_ERROR, LOG_TAG, "bundle_free() failed. Err = %d.", ret);
  117. return;
  118. }
  119.  
  120. s_info.bundle_obj = NULL;
  121. }
  122.  
  123. /**
  124. * @brief Function which adds a numeric value to the bundle object.
  125. * If a bundle handle is not prior created, the function fails.
  126. * @param[in] key The key name for value identification.
  127. * @param[in] value The value assigned to the given key.
  128. * @return The function returns 'true' if the value is added successfully,
  129. * otherwise 'false' is returned.
  130. */
  131. bool data_add_byte(const char *key, int value)
  132. {
  133. int ret;
  134.  
  135. data_delete_key(-1, key);
  136.  
  137. ret = bundle_add_byte(s_info.bundle_obj, key, &value, sizeof(value));
  138. if (ret != BUNDLE_ERROR_NONE) {
  139. dlog_print(DLOG_ERROR, LOG_TAG, "_add_byte() failed. Err = %s.", get_error_message(ret));
  140. return false;
  141. }
  142.  
  143. return true;
  144. }
  145.  
  146. /**
  147. * @brief Function which adds a string value to the bundle object.
  148. * If a bundle handle is not prior created, the function fails.
  149. * @param[in] key The key name for value identification.
  150. * @param[in] value The value assigned to the given key.
  151. * @return The function returns 'true' if the value is added successfully,
  152. * otherwise 'false' is returned.
  153. */
  154. bool data_add_string(const char *key, const char *value)
  155. {
  156. int ret;
  157.  
  158. data_delete_key(-1, key);
  159.  
  160. ret = bundle_add_str(s_info.bundle_obj, key, value);
  161. if (ret != BUNDLE_ERROR_NONE) {
  162. dlog_print(DLOG_ERROR, LOG_TAG, "_add_string() failed. Err = %s.", get_error_message(ret));
  163. return false;
  164. }
  165.  
  166. return true;
  167. }
  168.  
  169. /**
  170. * @brief Internal function that creates a key-value pair of the string array type. The array is then filled with key names and the data types.
  171. * @param[in] count Number of keys stored in a bundle
  172. * @return true on success or false on fail.
  173. */
  174. static bool _add_header_data(int count)
  175. {
  176. int ret;
  177. char** keys = calloc(count, sizeof(char*));
  178.  
  179. data_delete_key(-1, BUNDLE_HEADER_DATA_KEY);
  180. bundle_foreach(s_info.bundle_obj, _header_iterator_cb, (void*)keys);
  181.  
  182. ret = bundle_add_str_array(s_info.bundle_obj, BUNDLE_HEADER_DATA_KEY, (const char**)keys, count);
  183. if (ret != BUNDLE_ERROR_NONE) {
  184. dlog_print(DLOG_ERROR, LOG_TAG, "_add_string() failed. Err = %s.", get_error_message(ret));
  185.  
  186. _free_string_array(count, keys);
  187. return false;
  188. }
  189.  
  190. _free_string_array(count, keys);
  191. return true;
  192. }
  193.  
  194. /**
  195. * @brief Creates a header pair containing the count of the remaining pairs
  196. * @return true on success or false on fail.
  197. */
  198. bool data_add_header(void)
  199. {
  200. int count = 0;
  201. int ret = 0;
  202.  
  203. bundle_del(s_info.bundle_obj, BUNDLE_HEADER_KEY);
  204. bundle_del(s_info.bundle_obj, BUNDLE_HEADER_DATA_KEY);
  205.  
  206. count = bundle_get_count(s_info.bundle_obj);
  207.  
  208. if (!_add_header_data(count)) {
  209. dlog_print(DLOG_ERROR, LOG_TAG, "[%s:%d] Function _add_header_data() failed", __FILE__, __LINE__);
  210. return false;
  211. }
  212.  
  213. ret = bundle_add_byte(s_info.bundle_obj, BUNDLE_HEADER_KEY, (void *)&count, sizeof(int));
  214. if (ret != BUNDLE_ERROR_NONE) {
  215. dlog_print(DLOG_ERROR, LOG_TAG, "[%s:%d] bundle_add_byte() error: %s", __FILE__, __LINE__, get_error_message(ret));
  216. return false;
  217. }
  218.  
  219. return true;
  220. }
  221.  
  222. /**
  223. * @brief Check if the given key exist in the bundle with the given index.
  224. * @param[in] index Bundle's index (>=0 bundles received in a message, <0 bundle prepared using the source view).
  225. * @param[in] key The key to search for.
  226. * @return true - key exists, false the key doesn't exist.
  227. */
  228. bool data_key_exists(int index, const char *key)
  229. {
  230. bundle* bun = _get_bundle(index);
  231. bundle_get_type(bun, key);
  232. return !(get_last_result() == BUNDLE_ERROR_KEY_NOT_AVAILABLE);
  233. }
  234.  
  235. /**
  236. * @brief Check whether the given key contains a byte or a string value.
  237. * @param[in] index The bundle index.
  238. * @param[in] key The key name.
  239. * @return false - value type is byte, false - otherwise.
  240. */
  241. int data_is_byte(int index, const char *key)
  242. {
  243. bundle* bun = _get_bundle(index);
  244. set_last_result(BUNDLE_ERROR_NONE);
  245. int type = bundle_get_type(bun, key);
  246. int ret = get_last_result();
  247. if (ret != BUNDLE_ERROR_NONE) {
  248. dlog_print(DLOG_ERROR, LOG_TAG, "[%s:%d] bundle_get_type error: %s", __FILE__, __LINE__, get_error_message(ret));
  249. return 0;
  250. }
  251.  
  252. return (type == BUNDLE_TYPE_BYTE);
  253. }
  254.  
  255. /**
  256. * @brief Function returns a text with the key's type.
  257. * @param[in] index The bundle index.
  258. * @param[in] key The key name.
  259. * @return String containing the value's type name (e.g. "byte", "string").
  260. */
  261. char *data_get_type_text(int index, const char *key)
  262. {
  263. int type;
  264. int ret;
  265. bundle* bun = _get_bundle(index);
  266.  
  267. set_last_result(BUNDLE_ERROR_NONE);
  268. type = bundle_get_type(bun, key);
  269. ret = get_last_result();
  270. if (ret != BUNDLE_ERROR_NONE) {
  271. dlog_print(DLOG_ERROR, LOG_TAG, "[%s:%d] bundle_get_type error: %s", __FILE__, __LINE__, get_error_message(ret));
  272. return NULL;
  273. }
  274.  
  275. if (type == BUNDLE_TYPE_BYTE)
  276. return "byte";
  277. else
  278. return "string";
  279. }
  280.  
  281. /**
  282. * @brief Function returns the number of key-value pairs in the given bundle.
  283. * @param[in] index The bundle's index.
  284. * @return Number of key-value pairs stored in a bundle.
  285. */
  286. int data_get_count(int index)
  287. {
  288. bundle* bun = _get_bundle(index);
  289. return bundle_get_count(bun);
  290. }
  291.  
  292. /**
  293. * @brief Function returns a byte value associated with given key stored in a bundle under given index.
  294. * @param[in] index The bundle index.
  295. * @param[in] key The key name.
  296. * @return The value stored in a bundle under provided index and associated with given key.
  297. */
  298. int data_get_byte(int index, const char *key)
  299. {
  300. int *val = NULL;
  301. size_t size = 0;
  302. bundle *bun = _get_bundle(index);
  303.  
  304. int ret = bundle_get_byte(bun, key, (void**)&val, &size);
  305. if (ret != BUNDLE_ERROR_NONE) {
  306. dlog_print(DLOG_ERROR, LOG_TAG, "[%s:%d] bundle_get_byte() error: %s", __FILE__, __LINE__, get_error_message(ret));
  307. return 0;
  308. }
  309.  
  310. return *val;
  311. }
  312.  
  313. /**
  314. * @brief Function returns a string value associated with given key stored in a bundle under given index.
  315. * @param[in] index The bundle index.
  316. * @param[in] key The key name.
  317. * @return The value stored in a bundle under provided index and associated with given key.
  318. */
  319. char *data_get_string(int index, const char *key)
  320. {
  321. char *str = NULL;
  322. bundle *bun = _get_bundle(index);
  323.  
  324. int ret = bundle_get_str(bun, key, &str);
  325. if (ret != BUNDLE_ERROR_NONE) {
  326. dlog_print(DLOG_ERROR, LOG_TAG, "[%s:%d] data_get_string() error: %s", __FILE__, __LINE__, get_error_message(ret));
  327. return NULL;
  328. }
  329.  
  330. return str;
  331. }
  332.  
  333. /**
  334. * @brief Function returns a byte value associated with given key stored in a bundle under given index.
  335. * @param[in] index The bundle index.
  336. * @param[in] key The key name.
  337. * @param[out] len Length of the string array
  338. * @return The value stored in a bundle under provided index and associated with given key.
  339. */
  340. const char **data_get_string_array(int index, const char *key, int *len)
  341. {
  342. int ret;
  343. const char **values = NULL;
  344. bundle *bun = _get_bundle(index);
  345.  
  346. values = bundle_get_str_array(bun, key, len);
  347. ret = get_last_result();
  348. if (ret != BUNDLE_ERROR_NONE) {
  349. dlog_print(DLOG_ERROR, LOG_TAG, "[%s:%d] bundle_get_str_array() error: %s", __FILE__, __LINE__, get_error_message(ret));
  350. return NULL;
  351. }
  352.  
  353. return values;
  354. }
  355.  
  356. /**
  357. * @brief Functions deletes the given key-value pair from the bundle.
  358. * @param[in] index bundle's index.
  359. * @param[in] key Key to remove.
  360. */
  361. void data_delete_key(int index, const char *key)
  362. {
  363. int ret;
  364. bundle *bun = _get_bundle(index);
  365.  
  366. ret = bundle_del(bun, key);
  367. if (ret != BUNDLE_ERROR_NONE && ret != BUNDLE_ERROR_KEY_NOT_AVAILABLE) {
  368. dlog_print(DLOG_ERROR, LOG_TAG, "[%s:%d] bundle_del() error: %s", __FILE__, __LINE__, get_error_message(ret));
  369. return;
  370. }
  371. }
  372.  
  373. /**
  374. * @brief Function removes all key-value pairs from a bundle.
  375. */
  376. void data_clear_bundle(void)
  377. {
  378. bundle_foreach(s_info.bundle_obj, _bundle_clear_iterator_cb, NULL);
  379. }
  380.  
  381. Eina_List *data_get_keys(int index)
  382. {
  383. bundle *bun = _get_bundle(index);
  384. Eina_List *keys = NULL;
  385. bundle_foreach(bun, _keys_iterator_cb, (void*)&keys);
  386.  
  387. return keys;
  388. }
  389.  
  390. /**
  391. * @brief Function sends a message containing the bundle data.
  392. * @return true on success or false on fail.
  393. */
  394. bool data_send_message(void)
  395. {
  396. int ret = message_port_send_message(PACKAGE, MESSAGE_PORT_RCV_NAME, s_info.bundle_obj);
  397. if (ret != MESSAGE_PORT_ERROR_NONE) {
  398. dlog_print(DLOG_ERROR, LOG_TAG, "message_port_send_message() failed. Err = %s.", get_error_message(ret));
  399. return false;
  400. }
  401.  
  402. return true;
  403. }
  404.  
  405. /**
  406. * @brief Internal function used to retrieve a bundle with the given index.
  407. * @param[in] index Index of a bundle object to be returned.
  408. * @return A bundle object. If a non-negative index value is provided, then a relevant bundle object, received via Message Port, is returned. For negative index value, the bundle object prepared for sending is returned.
  409. */
  410. static inline bundle *_get_bundle(int index)
  411. {
  412. bundle *bun = NULL;
  413.  
  414. if (index < 0)
  415. bun = s_info.bundle_obj;
  416. else
  417. bun = eina_list_nth(s_info.bundles_received, index);
  418.  
  419. return bun;
  420. }
  421.  
  422. /**
  423. * @brief Internal callback function invoked by a bundle iterator. It is used to delete all the key-value pairs from the given bundle.
  424. * @param[in] key Current key.
  425. * @param[in] type Value type.
  426. * @param[in] kv Bundle value container.
  427. * @param[in] user_data User data.
  428. */
  429. static void _bundle_clear_iterator_cb(const char *key, const int type, const bundle_keyval_t *kv, void *user_data)
  430. {
  431. bundle_del(s_info.bundle_obj, key);
  432. }
  433.  
  434. /**
  435. * @brief Internal callback function invoked by a bundle iterator. It is used to fill the header data array.
  436. * @param[in] key Current key.
  437. * @param[in] type Value type.
  438. * @param[in] kv Bundle value container.
  439. * @param[out] user_data The array to store the keys.
  440. */
  441. static void _header_iterator_cb(const char *key, const int type, const bundle_keyval_t *kv, void *user_data)
  442. {
  443. char *type_name = NULL;
  444. char **keys = (char**)user_data;
  445. char buf[NAME_MAX];
  446. static int iter = 0;
  447. int count = bundle_get_count(s_info.bundle_obj);
  448.  
  449. if (type == BUNDLE_TYPE_BYTE)
  450. type_name = "byte";
  451. else
  452. type_name = "string";
  453.  
  454. snprintf(buf, NAME_MAX, "%s\2%s", key, type_name);
  455. keys[iter] = strdup(buf);
  456.  
  457. if (iter < count -1)
  458. iter++;
  459. else
  460. iter = 0;
  461. }
  462.  
  463. /**
  464. * @brief Internal callback function invoked by a bundle iterator used to create an eina_list for all of the available keys from the given bundle storage.
  465. * @param[in] key Current key.
  466. * @param[in] type Value type.
  467. * @param[in] kv Bundle value container.
  468. * @param[out] user_data List to store the keys.
  469. */
  470. static void _keys_iterator_cb(const char *key, const int type, const bundle_keyval_t *kv, void *user_data)
  471. {
  472. Eina_List **keys = (Eina_List **)user_data;
  473. *keys = eina_list_append(*keys, strdup(key));
  474. }
  475.  
  476. /**
  477. * @brief Internal callback function invoked when a message is received. The bundle received is appended to an eina_list. This callback function is invoked to inform the view module that the message had just been received.
  478. * @param[in] local_port_id Local port id
  479. * @param[in] remote_app_id Remote app id (In this case "org.example.bundles")
  480. * @param[in] remote_port Remote port name
  481. * @param[in] trusted_remote_port true - the port is trusted.
  482. * @param[in] message The message received.
  483. * @param[in] user_data User data.
  484. */
  485. static void _message_received_cb(int local_port_id, const char *remote_app_id, const char *remote_port, bool trusted_remote_port, bundle *message, void *user_data)
  486. {
  487. bundle *message_cpy = bundle_dup(message);
  488. if (!message_cpy) {
  489. dlog_print(DLOG_ERROR, LOG_TAG, "[%s:%d] message_cpy == NULL", __FILE__, __LINE__);
  490. return;
  491. }
  492.  
  493. s_info.bundles_received = eina_list_append(s_info.bundles_received, (void*)message_cpy);
  494. s_info.callbacks.new_data_cb(eina_list_count(s_info.bundles_received) - 1);
  495. }
  496.  
  497. /**
  498. * @brief Internal function that frees a string array and all of its content.
  499. * @param[in] count The array length.
  500. * @param[in] keys The array to free.
  501. */
  502. static inline void _free_string_array(int count, char** keys)
  503. {
  504. int i;
  505. for (i = 0; i < count; ++i)
  506. free(keys[i]);
  507.  
  508. free(keys);
  509. }