Batch.cpp 3.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121
  1. #define LOG_LOCAL_LEVEL ESP_LOG_VERBOSE
  2. #include "Batch.h"
  3. #include "esp_event.h"
  4. #include "esp_http_client.h"
  5. #include "esp_log.h"
  6. #include "esp_netif.h"
  7. #include "esp_ota_ops.h"
  8. #include "esp_tls.h"
  9. #if CONFIG_MBEDTLS_CERTIFICATE_BUNDLE
  10. #include "esp_crt_bundle.h"
  11. #endif
  12. #include "esp_system.h"
  13. #include "http_handlers.h"
  14. #include "tools.h"
  15. #include <algorithm>
  16. #include <iomanip>
  17. #include <sstream>
  18. #include <string>
  19. #include <sys/param.h>
  20. #if CONFIG_WITH_METRICS
  21. static const char* const TAG = "MetricsBatch";
  22. static const char* const feature_evt_name = "$feature_flag_called";
  23. static const char* const feature_flag_name = "$feature_flag";
  24. static const char* const feature_flag_response_name = "$feature_flag_response";
  25. namespace Metrics {
  26. Event& Batch::add_feature_event() { return add_event(feature_evt_name); }
  27. void Batch::add_remove_feature_event(const char* name, bool active) {
  28. if (!active) {
  29. remove_feature_event(name);
  30. } else {
  31. add_event(feature_evt_name).add_property(feature_flag_name, name);
  32. }
  33. }
  34. Event& Batch::add_feature_variant_event(const char* const name, const char* const value) {
  35. return add_event(feature_evt_name)
  36. .add_property(feature_flag_name, name)
  37. .add_property(feature_flag_response_name, value);
  38. }
  39. void Batch::remove_feature_event(const char* name) {
  40. for (Metrics::Event& e : _events) {
  41. if (strcmp(e.get_name(), feature_evt_name) == 0) {
  42. e.remove_property(feature_flag_name, name);
  43. return;
  44. }
  45. }
  46. }
  47. cJSON* Batch::to_json() {
  48. cJSON* batch_json = cJSON_CreateArray();
  49. for (Metrics::Event& e : _events) {
  50. cJSON_AddItemToArray(batch_json, e.to_json(_metrics_uid.c_str()));
  51. }
  52. cJSON* message = cJSON_CreateObject();
  53. cJSON_AddItemToObject(message, "batch", batch_json);
  54. cJSON_AddStringToObject(message, "api_key", _api_key);
  55. return batch_json;
  56. }
  57. char* Batch::to_json_str() {
  58. cJSON* json = to_json();
  59. char* json_str = cJSON_PrintUnformatted(json);
  60. cJSON_Delete(json);
  61. return json_str;
  62. }
  63. void Batch::push() {
  64. int status_code = 0;
  65. if (_metrics_uid.empty() && !_warned) {
  66. ESP_LOGW(TAG, "Metrics disabled; no CID found");
  67. _warned = true;
  68. return;
  69. }
  70. char* json_str = to_json_str();
  71. ESP_LOGV(TAG, "Metrics payload: %s", json_str);
  72. uint32_t start_time = gettime_ms();
  73. status_code = metrics_http_post_request(json_str, _url);
  74. if (status_code == 200 || status_code == 204) {
  75. _events.clear();
  76. }
  77. FREE_AND_NULL(json_str)
  78. ESP_LOGD(TAG, "Total duration for metrics call: %u. ", gettime_ms() - start_time);
  79. }
  80. void Batch::build_guid() {
  81. uint8_t raw[16];
  82. std::ostringstream oss;
  83. esp_fill_random(raw, 16);
  84. std::for_each(std::begin(raw), std::end(raw), [&oss](const uint8_t& byte) {
  85. oss << std::hex << std::setw(2) << std::setfill('0') << static_cast<int>(byte);
  86. });
  87. _metrics_uid = oss.str();
  88. }
  89. void Batch::assign_id() {
  90. size_t size = 0;
  91. esp_err_t esp_err = ESP_OK;
  92. _metrics_uid = std::string((char*)get_nvs_value_alloc_for_partition(
  93. NVS_DEFAULT_PART_NAME, TAG, NVS_TYPE_BLOB, "cid", &size));
  94. if (_metrics_uid[0] == 'G') {
  95. ESP_LOGW(TAG, "Invalid ID. %s", _metrics_uid.c_str());
  96. _metrics_uid.clear();
  97. }
  98. if (_metrics_uid.empty()) {
  99. build_guid();
  100. if (_metrics_uid.empty()) {
  101. ESP_LOGE(TAG, "ID Failed");
  102. return;
  103. }
  104. ESP_LOGW(TAG, "Metrics ID: %s", _metrics_uid.c_str());
  105. esp_err = store_nvs_value_len_for_partition(NVS_DEFAULT_PART_NAME, TAG, NVS_TYPE_BLOB,
  106. "cid", _metrics_uid.c_str(), _metrics_uid.length() + 1);
  107. if (esp_err != ESP_OK) {
  108. ESP_LOGE(TAG, "Store ID failed: %s", esp_err_to_name(esp_err));
  109. }
  110. }
  111. }
  112. } // namespace Metrics
  113. #endif