flip_weather_parse.c 5.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150
  1. #include "parse/flip_weather_parse.h"
  2. bool sent_get_request = false;
  3. bool get_request_success = false;
  4. bool got_ip_address = false;
  5. bool geo_information_processed = false;
  6. bool weather_information_processed = false;
  7. bool flip_weather_parse_ip_address() {
  8. // load the received data from the saved file
  9. FuriString* returned_data = flipper_http_load_from_file(fhttp.file_path);
  10. if(returned_data == NULL) {
  11. FURI_LOG_E(TAG, "Failed to load received data from file.");
  12. return false;
  13. }
  14. const char* data_cstr = furi_string_get_cstr(returned_data);
  15. if(data_cstr == NULL) {
  16. FURI_LOG_E(TAG, "Failed to get C-string from FuriString.");
  17. furi_string_free(returned_data);
  18. return false;
  19. }
  20. char* ip = get_json_value("origin", (char*)data_cstr, MAX_TOKENS);
  21. if(ip == NULL) {
  22. FURI_LOG_E(TAG, "Failed to get IP address");
  23. sent_get_request = true;
  24. get_request_success = false;
  25. fhttp.state = ISSUE;
  26. free(ip);
  27. furi_string_free(returned_data);
  28. return false;
  29. }
  30. snprintf(ip_address, 16, "%s", ip);
  31. free(ip);
  32. furi_string_free(returned_data);
  33. return true;
  34. }
  35. // handle the async-to-sync process to get and set the IP address
  36. bool flip_weather_handle_ip_address() {
  37. if(fhttp.state == INACTIVE) {
  38. FURI_LOG_E(TAG, "Board is INACTIVE");
  39. flipper_http_ping(); // ping the device
  40. return false;
  41. }
  42. if(!got_ip_address) {
  43. got_ip_address = true;
  44. snprintf(
  45. fhttp.file_path,
  46. sizeof(fhttp.file_path),
  47. STORAGE_EXT_PATH_PREFIX "/apps_data/flip_weather/ip.txt");
  48. fhttp.save_received_data = true;
  49. if(!flipper_http_get_request("https://httpbin.org/get")) {
  50. FURI_LOG_E(TAG, "Failed to get IP address");
  51. fhttp.state = ISSUE;
  52. return false;
  53. } else {
  54. fhttp.state = RECEIVING;
  55. furi_timer_start(fhttp.get_timeout_timer, TIMEOUT_DURATION_TICKS);
  56. }
  57. while(fhttp.state == RECEIVING && furi_timer_is_running(fhttp.get_timeout_timer) > 0) {
  58. // Wait for the feed to be received
  59. furi_delay_ms(10);
  60. }
  61. furi_timer_stop(fhttp.get_timeout_timer);
  62. if(!flip_weather_parse_ip_address()) {
  63. FURI_LOG_E(TAG, "Failed to get IP address");
  64. sent_get_request = true;
  65. get_request_success = false;
  66. fhttp.state = ISSUE;
  67. return false;
  68. }
  69. }
  70. return true;
  71. }
  72. bool send_geo_location_request() {
  73. if(fhttp.state == INACTIVE) {
  74. FURI_LOG_E(TAG, "Board is INACTIVE");
  75. flipper_http_ping(); // ping the device
  76. return false;
  77. }
  78. if(!sent_get_request && fhttp.state == IDLE) {
  79. sent_get_request = true;
  80. char* headers = jsmn("Content-Type", "application/json");
  81. get_request_success =
  82. flipper_http_get_request_with_headers("https://ipwhois.app/json/", headers);
  83. free(headers);
  84. if(!get_request_success) {
  85. FURI_LOG_E(TAG, "Failed to send GET request");
  86. fhttp.state = ISSUE;
  87. return false;
  88. }
  89. fhttp.state = RECEIVING;
  90. }
  91. return true;
  92. }
  93. void process_geo_location() {
  94. if(!geo_information_processed && fhttp.last_response != NULL) {
  95. geo_information_processed = true;
  96. char* city = get_json_value("city", fhttp.last_response, MAX_TOKENS);
  97. char* region = get_json_value("region", fhttp.last_response, MAX_TOKENS);
  98. char* country = get_json_value("country", fhttp.last_response, MAX_TOKENS);
  99. char* latitude = get_json_value("latitude", fhttp.last_response, MAX_TOKENS);
  100. char* longitude = get_json_value("longitude", fhttp.last_response, MAX_TOKENS);
  101. snprintf(city_data, 64, "City: %s", city);
  102. snprintf(region_data, 64, "Region: %s", region);
  103. snprintf(country_data, 64, "Country: %s", country);
  104. snprintf(lat_data, 64, "Latitude: %s", latitude);
  105. snprintf(lon_data, 64, "Longitude: %s", longitude);
  106. snprintf(ip_data, 64, "IP Address: %s", ip_address);
  107. fhttp.state = IDLE;
  108. }
  109. }
  110. void process_weather() {
  111. if(!weather_information_processed && fhttp.last_response != NULL) {
  112. weather_information_processed = true;
  113. char* current_data = get_json_value("current", fhttp.last_response, MAX_TOKENS);
  114. char* temperature = get_json_value("temperature_2m", current_data, MAX_TOKENS);
  115. char* precipitation = get_json_value("precipitation", current_data, MAX_TOKENS);
  116. char* rain = get_json_value("rain", current_data, MAX_TOKENS);
  117. char* showers = get_json_value("showers", current_data, MAX_TOKENS);
  118. char* snowfall = get_json_value("snowfall", current_data, MAX_TOKENS);
  119. char* time = get_json_value("time", current_data, MAX_TOKENS);
  120. // replace the "T" in time with a space
  121. char* ptr = strstr(time, "T");
  122. if(ptr != NULL) {
  123. *ptr = ' ';
  124. }
  125. snprintf(temperature_data, 64, "Temperature (C): %s", temperature);
  126. snprintf(precipitation_data, 64, "Precipitation: %s", precipitation);
  127. snprintf(rain_data, 64, "Rain: %s", rain);
  128. snprintf(showers_data, 64, "Showers: %s", showers);
  129. snprintf(snowfall_data, 64, "Snowfall: %s", snowfall);
  130. snprintf(time_data, 64, "Time: %s", time);
  131. fhttp.state = IDLE;
  132. } else if(!weather_information_processed && fhttp.last_response == NULL) {
  133. FURI_LOG_E(TAG, "Failed to process weather data");
  134. // store error message
  135. snprintf(temperature_data, 64, "Failed. Update WiFi settings.");
  136. fhttp.state = ISSUE;
  137. }
  138. }