mirror of
https://github.com/esphome/esphome.git
synced 2024-12-02 11:44:13 +01:00
6153bcc6ad
Co-authored-by: Paulus Schoutsen <balloob@gmail.com>
111 lines
3.4 KiB
C++
111 lines
3.4 KiB
C++
#pragma once
|
|
|
|
#include "esphome/components/socket/socket.h"
|
|
#include "esphome/core/component.h"
|
|
#include "esphome/core/preferences.h"
|
|
#include "esphome/core/helpers.h"
|
|
#include "esphome/core/defines.h"
|
|
|
|
namespace esphome {
|
|
namespace ota {
|
|
|
|
enum OTAResponseTypes {
|
|
OTA_RESPONSE_OK = 0,
|
|
OTA_RESPONSE_REQUEST_AUTH = 1,
|
|
|
|
OTA_RESPONSE_HEADER_OK = 64,
|
|
OTA_RESPONSE_AUTH_OK = 65,
|
|
OTA_RESPONSE_UPDATE_PREPARE_OK = 66,
|
|
OTA_RESPONSE_BIN_MD5_OK = 67,
|
|
OTA_RESPONSE_RECEIVE_OK = 68,
|
|
OTA_RESPONSE_UPDATE_END_OK = 69,
|
|
OTA_RESPONSE_SUPPORTS_COMPRESSION = 70,
|
|
|
|
OTA_RESPONSE_ERROR_MAGIC = 128,
|
|
OTA_RESPONSE_ERROR_UPDATE_PREPARE = 129,
|
|
OTA_RESPONSE_ERROR_AUTH_INVALID = 130,
|
|
OTA_RESPONSE_ERROR_WRITING_FLASH = 131,
|
|
OTA_RESPONSE_ERROR_UPDATE_END = 132,
|
|
OTA_RESPONSE_ERROR_INVALID_BOOTSTRAPPING = 133,
|
|
OTA_RESPONSE_ERROR_WRONG_CURRENT_FLASH_CONFIG = 134,
|
|
OTA_RESPONSE_ERROR_WRONG_NEW_FLASH_CONFIG = 135,
|
|
OTA_RESPONSE_ERROR_ESP8266_NOT_ENOUGH_SPACE = 136,
|
|
OTA_RESPONSE_ERROR_ESP32_NOT_ENOUGH_SPACE = 137,
|
|
OTA_RESPONSE_ERROR_NO_UPDATE_PARTITION = 138,
|
|
OTA_RESPONSE_ERROR_MD5_MISMATCH = 139,
|
|
OTA_RESPONSE_ERROR_RP2040_NOT_ENOUGH_SPACE = 140,
|
|
OTA_RESPONSE_ERROR_UNKNOWN = 255,
|
|
};
|
|
|
|
enum OTAState { OTA_COMPLETED = 0, OTA_STARTED, OTA_IN_PROGRESS, OTA_ERROR };
|
|
|
|
/// OTAComponent provides a simple way to integrate Over-the-Air updates into your app using ArduinoOTA.
|
|
class OTAComponent : public Component {
|
|
public:
|
|
OTAComponent();
|
|
#ifdef USE_OTA_PASSWORD
|
|
void set_auth_password(const std::string &password) { password_ = password; }
|
|
#endif // USE_OTA_PASSWORD
|
|
|
|
/// Manually set the port OTA should listen on.
|
|
void set_port(uint16_t port);
|
|
|
|
bool should_enter_safe_mode(uint8_t num_attempts, uint32_t enable_time);
|
|
|
|
/// Set to true if the next startup will enter safe mode
|
|
void set_safe_mode_pending(const bool &pending);
|
|
bool get_safe_mode_pending();
|
|
|
|
#ifdef USE_OTA_STATE_CALLBACK
|
|
void add_on_state_callback(std::function<void(OTAState, float, uint8_t)> &&callback);
|
|
#endif
|
|
|
|
// ========== INTERNAL METHODS ==========
|
|
// (In most use cases you won't need these)
|
|
void setup() override;
|
|
void dump_config() override;
|
|
float get_setup_priority() const override;
|
|
void loop() override;
|
|
|
|
uint16_t get_port() const;
|
|
|
|
void clean_rtc();
|
|
|
|
void on_safe_shutdown() override;
|
|
|
|
protected:
|
|
void write_rtc_(uint32_t val);
|
|
uint32_t read_rtc_();
|
|
|
|
void handle_();
|
|
bool readall_(uint8_t *buf, size_t len);
|
|
bool writeall_(const uint8_t *buf, size_t len);
|
|
|
|
#ifdef USE_OTA_PASSWORD
|
|
std::string password_;
|
|
#endif // USE_OTA_PASSWORD
|
|
|
|
uint16_t port_;
|
|
|
|
std::unique_ptr<socket::Socket> server_;
|
|
std::unique_ptr<socket::Socket> client_;
|
|
|
|
bool has_safe_mode_{false}; ///< stores whether safe mode can be enabled.
|
|
uint32_t safe_mode_start_time_; ///< stores when safe mode was enabled.
|
|
uint32_t safe_mode_enable_time_{60000}; ///< The time safe mode should be on for.
|
|
uint32_t safe_mode_rtc_value_;
|
|
uint8_t safe_mode_num_attempts_;
|
|
ESPPreferenceObject rtc_;
|
|
|
|
static const uint32_t ENTER_SAFE_MODE_MAGIC =
|
|
0x5afe5afe; ///< a magic number to indicate that safe mode should be entered on next boot
|
|
|
|
#ifdef USE_OTA_STATE_CALLBACK
|
|
CallbackManager<void(OTAState, float, uint8_t)> state_callback_{};
|
|
#endif
|
|
};
|
|
|
|
extern OTAComponent *global_ota_component; // NOLINT(cppcoreguidelines-avoid-non-const-global-variables)
|
|
|
|
} // namespace ota
|
|
} // namespace esphome
|