esp32_improv.cpp.txt 9.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285
  1. #include "esp32_improv.h"
  2. namespace esp32_improv {
  3. static const char *const TAG = "esp32_improv.component";
  4. static const char *const ESPHOME_MY_LINK = "https://my.home-assistant.io/redirect/config_flow_start?domain=esphome";
  5. namespace esp32_improv {
  6. void ESP32Improv::setup() {
  7. this->service_ = global_ble_server->create_service(improv::SERVICE_UUID, true);
  8. this->setup_characteristics();
  9. }
  10. void ESP32Improv::setup_characteristics() {
  11. this->status_ = this->service_->create_characteristic(
  12. improv::STATUS_UUID,
  13. BLECharacteristic::PROPERTY_READ | BLECharacteristic::PROPERTY_NOTIFY);
  14. BLEDescriptor *status_descriptor = new BLE2902();
  15. this->status_->add_descriptor(status_descriptor);
  16. this->error_ = this->service_->create_characteristic(
  17. improv::ERROR_UUID,
  18. BLECharacteristic::PROPERTY_READ | BLECharacteristic::PROPERTY_NOTIFY);
  19. BLEDescriptor *error_descriptor = new BLE2902();
  20. this->error_->add_descriptor(error_descriptor);
  21. this->rpc_ = this->service_->create_characteristic(
  22. improv::RPC_COMMAND_UUID,
  23. BLECharacteristic::PROPERTY_WRITE);
  24. this->rpc_->on_write([this](const std::vector<uint8_t> &data) {
  25. if (!data.empty()) {
  26. this->incoming_data_.insert(this->incoming_data_.end(), data.begin(), data.end());
  27. }
  28. });
  29. BLEDescriptor *rpc_descriptor = new BLE2902();
  30. this->rpc_->add_descriptor(rpc_descriptor);
  31. this->rpc_response_ = this->service_->create_characteristic(
  32. improv::RPC_RESULT_UUID,
  33. BLECharacteristic::PROPERTY_READ | BLECharacteristic::PROPERTY_NOTIFY);
  34. BLEDescriptor *rpc_response_descriptor = new BLE2902();
  35. this->rpc_response_->add_descriptor(rpc_response_descriptor);
  36. this->capabilities_ =
  37. this->service_->create_characteristic(
  38. improv::CAPABILITIES_UUID,
  39. BLECharacteristic::PROPERTY_READ);
  40. BLEDescriptor *capabilities_descriptor = new BLE2902();
  41. this->capabilities_->add_descriptor(capabilities_descriptor);
  42. uint8_t capabilities = 0x00;
  43. if (this->status_indicator_ != nullptr)
  44. capabilities |= improv::CAPABILITY_IDENTIFY;
  45. this->capabilities_->set_value(capabilities);
  46. this->setup_complete_ = true;
  47. }
  48. void ESP32Improv::loop() {
  49. if (!this->incoming_data_.empty())
  50. this->process_incoming_data_();
  51. uint32_t now = gettime_ms();
  52. switch (this->state_) {
  53. case improv::STATE_STOPPED:
  54. if (this->status_indicator_ != nullptr)
  55. this->status_indicator_->turn_off();
  56. if (this->service_->is_created() && this->should_start_ && this->setup_complete_) {
  57. if (this->service_->is_running()) {
  58. esp32_ble::global_ble->get_advertising()->start();
  59. this->set_state_(improv::STATE_AWAITING_AUTHORIZATION);
  60. this->set_error_(improv::ERROR_NONE);
  61. this->should_start_ = false;
  62. ESP_LOGD(TAG, "Service started!");
  63. } else {
  64. this->service_->start();
  65. }
  66. }
  67. break;
  68. case improv::STATE_AWAITING_AUTHORIZATION: {
  69. if (this->authorizer_ == nullptr || this->authorizer_->state) {
  70. this->set_state_(improv::STATE_AUTHORIZED);
  71. this->authorized_start_ = now;
  72. } else {
  73. if (this->status_indicator_ != nullptr) {
  74. if (!this->check_identify_())
  75. this->status_indicator_->turn_on();
  76. }
  77. }
  78. break;
  79. }
  80. case improv::STATE_AUTHORIZED: {
  81. if (this->authorizer_ != nullptr) {
  82. if (now - this->authorized_start_ > this->authorized_duration_) {
  83. ESP_LOGD(TAG, "Authorization timeout");
  84. this->set_state_(improv::STATE_AWAITING_AUTHORIZATION);
  85. return;
  86. }
  87. }
  88. if (this->status_indicator_ != nullptr) {
  89. if (!this->check_identify_()) {
  90. if ((now % 1000) < 500) {
  91. this->status_indicator_->turn_on();
  92. } else {
  93. this->status_indicator_->turn_off();
  94. }
  95. }
  96. }
  97. break;
  98. }
  99. case improv::STATE_PROVISIONING: {
  100. if (this->status_indicator_ != nullptr) {
  101. if ((now % 200) < 100) {
  102. this->status_indicator_->turn_on();
  103. } else {
  104. this->status_indicator_->turn_off();
  105. }
  106. }
  107. if (wifi::global_wifi_component->is_connected()) {
  108. wifi::global_wifi_component->save_wifi_sta(this->connecting_sta_.get_ssid(),
  109. this->connecting_sta_.get_password());
  110. this->connecting_sta_ = {};
  111. this->cancel_timeout("wifi-connect-timeout");
  112. this->set_state_(improv::STATE_PROVISIONED);
  113. std::vector<std::string> urls = {ESPHOME_MY_LINK};
  114. #ifdef USE_WEBSERVER
  115. auto ip = wifi::global_wifi_component->wifi_sta_ip();
  116. std::string webserver_url = "http://" + ip.str() + ":" + to_string(USE_WEBSERVER_PORT);
  117. urls.push_back(webserver_url);
  118. #endif
  119. std::vector<uint8_t> data = improv::build_rpc_response(improv::WIFI_SETTINGS, urls);
  120. this->send_response_(data);
  121. this->set_timeout("end-service", 1000, [this] {
  122. this->service_->stop();
  123. this->set_state_(improv::STATE_STOPPED);
  124. });
  125. }
  126. break;
  127. }
  128. case improv::STATE_PROVISIONED: {
  129. this->incoming_data_.clear();
  130. if (this->status_indicator_ != nullptr)
  131. this->status_indicator_->turn_off();
  132. break;
  133. }
  134. }
  135. }
  136. bool ESP32Improv::check_identify_() {
  137. uint32_t now = gettime_ms();
  138. bool identify = this->identify_start_ != 0 && now - this->identify_start_ <= this->identify_duration_;
  139. if (identify) {
  140. uint32_t time = now % 1000;
  141. if (time < 600 && time % 200 < 100) {
  142. this->status_indicator_->turn_on();
  143. } else {
  144. this->status_indicator_->turn_off();
  145. }
  146. }
  147. return identify;
  148. }
  149. void ESP32Improv::set_state_(improv::State state) {
  150. ESP_LOGV(TAG, "Setting state: %d", state);
  151. this->state_ = state;
  152. if (this->status_->get_value().empty() || this->status_->get_value()[0] != state) {
  153. uint8_t data[1]{state};
  154. this->status_->set_value(data, 1);
  155. if (state != improv::STATE_STOPPED)
  156. this->status_->notify();
  157. }
  158. }
  159. void ESP32Improv::set_error_(improv::Error error) {
  160. if (error != improv::ERROR_NONE)
  161. ESP_LOGE(TAG, "Error: %d", error);
  162. if (this->error_->get_value().empty() || this->error_->get_value()[0] != error) {
  163. uint8_t data[1]{error};
  164. this->error_->set_value(data, 1);
  165. if (this->state_ != improv::STATE_STOPPED)
  166. this->error_->notify();
  167. }
  168. }
  169. void ESP32Improv::send_response_(std::vector<uint8_t> &response) {
  170. this->rpc_response_->set_value(response);
  171. if (this->state_ != improv::STATE_STOPPED)
  172. this->rpc_response_->notify();
  173. }
  174. void ESP32Improv::start() {
  175. if (this->state_ != improv::STATE_STOPPED)
  176. return;
  177. ESP_LOGD(TAG, "Setting Improv to start");
  178. this->should_start_ = true;
  179. }
  180. void ESP32Improv::stop() {
  181. this->set_timeout("end-service", 1000, [this] {
  182. this->service_->stop();
  183. this->set_state_(improv::STATE_STOPPED);
  184. });
  185. }
  186. float ESP32Improv::get_setup_priority() const { return setup_priority::AFTER_BLUETOOTH; }
  187. void ESP32Improv::dump_config() {
  188. ESP_LOGCONFIG(TAG, "ESP32 Improv:");
  189. LOG_BINARY_SENSOR(" ", "Authorizer", this->authorizer_);
  190. ESP_LOGCONFIG(TAG, " Status Indicator: '%s'", YESNO(this->status_indicator_ != nullptr));
  191. }
  192. void ESP32Improv::process_incoming_data_() {
  193. uint8_t length = this->incoming_data_[1];
  194. ESP_LOGD(TAG, "Processing bytes - %s", format_hex_pretty(this->incoming_data_).c_str());
  195. if (this->incoming_data_.size() - 3 == length) {
  196. this->set_error_(improv::ERROR_NONE);
  197. improv::ImprovCommand command = improv::parse_improv_data(this->incoming_data_);
  198. switch (command.command) {
  199. case improv::BAD_CHECKSUM:
  200. ESP_LOGW(TAG, "Error decoding Improv payload");
  201. this->set_error_(improv::ERROR_INVALID_RPC);
  202. this->incoming_data_.clear();
  203. break;
  204. case improv::WIFI_SETTINGS: {
  205. if (this->state_ != improv::STATE_AUTHORIZED) {
  206. ESP_LOGW(TAG, "Settings received, but not authorized");
  207. this->set_error_(improv::ERROR_NOT_AUTHORIZED);
  208. this->incoming_data_.clear();
  209. return;
  210. }
  211. wifi::WiFiAP sta{};
  212. sta.set_ssid(command.ssid);
  213. sta.set_password(command.password);
  214. this->connecting_sta_ = sta;
  215. wifi::global_wifi_component->set_sta(sta);
  216. wifi::global_wifi_component->start_scanning();
  217. this->set_state_(improv::STATE_PROVISIONING);
  218. ESP_LOGD(TAG, "Received Improv wifi settings ssid=%s, password=" LOG_SECRET("%s"), command.ssid.c_str(),
  219. command.password.c_str());
  220. auto f = std::bind(&ESP32Improv::on_wifi_connect_timeout_, this);
  221. this->set_timeout("wifi-connect-timeout", 30000, f);
  222. this->incoming_data_.clear();
  223. break;
  224. }
  225. case improv::IDENTIFY:
  226. this->incoming_data_.clear();
  227. this->identify_start_ = gettime_ms();
  228. break;
  229. default:
  230. ESP_LOGW(TAG, "Unknown Improv payload");
  231. this->set_error_(improv::ERROR_UNKNOWN_RPC);
  232. this->incoming_data_.clear();
  233. }
  234. } else if (this->incoming_data_.size() - 2 > length) {
  235. ESP_LOGV(TAG, "Too much data came in, or malformed resetting buffer...");
  236. this->incoming_data_.clear();
  237. } else {
  238. ESP_LOGV(TAG, "Waiting for split data packets...");
  239. }
  240. }
  241. void ESP32Improv::on_wifi_connect_timeout_() {
  242. this->set_error_(improv::ERROR_UNABLE_TO_CONNECT);
  243. this->set_state_(improv::STATE_AUTHORIZED);
  244. if (this->authorizer_ != nullptr)
  245. this->authorized_start_ = gettime_ms();
  246. ESP_LOGW(TAG, "Timed out trying to connect to given WiFi network");
  247. wifi::global_wifi_component->clear_sta();
  248. }
  249. void ESP32Improv::on_client_disconnect() { this->set_error_(improv::ERROR_NONE); };
  250. ESP32Improv *global_improv_component = nullptr; // NOLINT(cppcoreguidelines-avoid-non-const-global-variables)
  251. } // namespace esp32_improv