flip_social_friends.c 5.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194
  1. #include "flip_social_friends.h"
  2. FlipSocialModel *flip_social_friends_alloc()
  3. {
  4. // Allocate memory for each username only if not already allocated
  5. FlipSocialModel *friends = malloc(sizeof(FlipSocialModel));
  6. for (size_t i = 0; i < MAX_FRIENDS; i++)
  7. {
  8. if (friends->usernames[i] == NULL)
  9. {
  10. friends->usernames[i] = malloc(MAX_USER_LENGTH);
  11. if (friends->usernames[i] == NULL)
  12. {
  13. FURI_LOG_E(TAG, "Failed to allocate memory for username %zu", i);
  14. return NULL; // Return false on memory allocation failure
  15. }
  16. }
  17. }
  18. return friends;
  19. }
  20. void flip_social_free_friends()
  21. {
  22. if (!flip_social_friends)
  23. {
  24. return;
  25. }
  26. for (int i = 0; i < flip_social_friends->count; i++)
  27. {
  28. if (flip_social_friends->usernames[i])
  29. {
  30. free(flip_social_friends->usernames[i]);
  31. }
  32. }
  33. free(flip_social_friends);
  34. flip_social_friends = NULL;
  35. }
  36. // for now we're just listing the current users
  37. // as the feed is upgraded, then we can port more to the friends view
  38. bool flip_social_get_friends()
  39. {
  40. if (!app_instance)
  41. {
  42. FURI_LOG_E(TAG, "App instance is NULL");
  43. return false;
  44. }
  45. if (fhttp.state == INACTIVE)
  46. {
  47. FURI_LOG_E(TAG, "HTTP state is INACTIVE");
  48. return false;
  49. }
  50. // will return true unless the devboard is not connected
  51. char url[100];
  52. snprintf(
  53. fhttp.file_path,
  54. sizeof(fhttp.file_path),
  55. STORAGE_EXT_PATH_PREFIX "/apps_data/flip_social/friends.json");
  56. fhttp.save_received_data = true;
  57. auth_headers_alloc();
  58. snprintf(url, sizeof(url), "https://www.flipsocial.net/api/user/friends/%s/", app_instance->login_username_logged_in);
  59. if (!flipper_http_get_request_with_headers(url, auth_headers))
  60. {
  61. FURI_LOG_E(TAG, "Failed to send HTTP request for friends");
  62. fhttp.state = ISSUE;
  63. return false;
  64. }
  65. fhttp.state = RECEIVING;
  66. return true;
  67. }
  68. bool flip_social_update_friends()
  69. {
  70. if (!app_instance->submenu_friends)
  71. {
  72. FURI_LOG_E(TAG, "Friends submenu is NULL");
  73. return false;
  74. }
  75. if (!flip_social_friends)
  76. {
  77. FURI_LOG_E(TAG, "Friends model is NULL");
  78. return false;
  79. }
  80. // Add submenu items for the users
  81. submenu_reset(app_instance->submenu_friends);
  82. submenu_set_header(app_instance->submenu_friends, "Friends");
  83. for (int i = 0; i < flip_social_friends->count; i++)
  84. {
  85. submenu_add_item(app_instance->submenu_friends, flip_social_friends->usernames[i], FlipSocialSubmenuLoggedInIndexFriendsStart + i, flip_social_callback_submenu_choices, app_instance);
  86. }
  87. return true;
  88. }
  89. bool flip_social_parse_json_friends()
  90. {
  91. // load the received data from the saved file
  92. FuriString *friend_data = flipper_http_load_from_file(fhttp.file_path);
  93. if (friend_data == NULL)
  94. {
  95. FURI_LOG_E(TAG, "Failed to load received data from file.");
  96. return false;
  97. }
  98. char *data_cstr = (char *)furi_string_get_cstr(friend_data);
  99. if (data_cstr == NULL)
  100. {
  101. FURI_LOG_E(TAG, "Failed to get C-string from FuriString.");
  102. furi_string_free(friend_data);
  103. return false;
  104. }
  105. // Allocate memory for each username only if not already allocated
  106. flip_social_friends = flip_social_friends_alloc();
  107. if (flip_social_friends == NULL)
  108. {
  109. FURI_LOG_E(TAG, "Failed to allocate memory for friends usernames.");
  110. furi_string_free(friend_data);
  111. free(data_cstr);
  112. return false;
  113. }
  114. // Remove newlines
  115. char *pos = data_cstr;
  116. while ((pos = strchr(pos, '\n')) != NULL)
  117. {
  118. *pos = ' ';
  119. }
  120. // Initialize friends count
  121. flip_social_friends->count = 0;
  122. // Extract the users array from the JSON
  123. char *json_users = get_json_value("friends", data_cstr, MAX_TOKENS);
  124. if (json_users == NULL)
  125. {
  126. FURI_LOG_E(TAG, "Failed to parse friends array.");
  127. furi_string_free(friend_data);
  128. free(data_cstr);
  129. flip_social_free_friends();
  130. return false;
  131. }
  132. // Manual tokenization for comma-separated values
  133. char *start = json_users + 1; // Skip the opening bracket
  134. char *end;
  135. while ((end = strchr(start, ',')) != NULL && flip_social_friends->count < MAX_FRIENDS)
  136. {
  137. *end = '\0'; // Null-terminate the current token
  138. // Remove quotes
  139. if (*start == '"')
  140. start++;
  141. if (*(end - 1) == '"')
  142. *(end - 1) = '\0';
  143. // Copy username to pre-allocated memory
  144. snprintf(flip_social_friends->usernames[flip_social_friends->count], MAX_USER_LENGTH, "%s", start);
  145. flip_social_friends->count++;
  146. start = end + 1;
  147. }
  148. // Handle the last token
  149. if (*start != '\0' && flip_social_friends->count < MAX_FRIENDS)
  150. {
  151. if (*start == '"')
  152. start++;
  153. if (*(start + strlen(start) - 1) == ']')
  154. *(start + strlen(start) - 1) = '\0';
  155. if (*(start + strlen(start) - 1) == '"')
  156. *(start + strlen(start) - 1) = '\0';
  157. snprintf(flip_social_friends->usernames[flip_social_friends->count], MAX_USER_LENGTH, "%s", start);
  158. flip_social_friends->count++;
  159. }
  160. // Add submenu items for the friends
  161. if (!flip_social_update_friends())
  162. {
  163. FURI_LOG_E(TAG, "Failed to update friends submenu");
  164. furi_string_free(friend_data);
  165. free(data_cstr);
  166. free(json_users);
  167. free(start);
  168. free(end);
  169. return false;
  170. }
  171. furi_string_free(friend_data);
  172. free(data_cstr);
  173. free(json_users);
  174. free(start);
  175. free(end);
  176. return true;
  177. }