ESPHome 2026.6.0-dev
Loading...
Searching...
No Matches
atc_mithermometer.cpp
Go to the documentation of this file.
1#include "atc_mithermometer.h"
2#include "esphome/core/log.h"
3
4#ifdef USE_ESP32
5
7
8static const char *const TAG = "atc_mithermometer";
9
11 ESP_LOGCONFIG(TAG, "ATC MiThermometer");
12 LOG_SENSOR(" ", "Temperature", this->temperature_);
13 LOG_SENSOR(" ", "Humidity", this->humidity_);
14 LOG_SENSOR(" ", "Battery Level", this->battery_level_);
15 LOG_SENSOR(" ", "Battery Voltage", this->battery_voltage_);
16}
17
19 if (device.address_uint64() != this->address_) {
20 ESP_LOGVV(TAG, "parse_device(): unknown MAC address.");
21 return false;
22 }
23 char addr_buf[MAC_ADDRESS_PRETTY_BUFFER_SIZE];
24 const char *addr_str = device.address_str_to(addr_buf);
25 ESP_LOGVV(TAG, "parse_device(): MAC address %s found.", addr_str);
26
27 bool success = false;
28 for (auto &service_data : device.get_service_datas()) {
29 auto res = parse_header_(service_data);
30 if (!res.has_value()) {
31 continue;
32 }
33 if (!(parse_message_(service_data.data, *res))) {
34 continue;
35 }
36 if (!(report_results_(res, addr_str))) {
37 continue;
38 }
39 if (res->temperature.has_value() && this->temperature_ != nullptr)
40 this->temperature_->publish_state(*res->temperature);
41 if (res->humidity.has_value() && this->humidity_ != nullptr)
42 this->humidity_->publish_state(*res->humidity);
43 if (res->battery_level.has_value() && this->battery_level_ != nullptr)
44 this->battery_level_->publish_state(*res->battery_level);
45 if (res->battery_voltage.has_value() && this->battery_voltage_ != nullptr)
46 this->battery_voltage_->publish_state(*res->battery_voltage);
47 success = true;
48 }
49 if (this->signal_strength_ != nullptr)
51
52 return success;
53}
54
55optional<ParseResult> ATCMiThermometer::parse_header_(const esp32_ble_tracker::ServiceData &service_data) {
56 ParseResult result;
57 if (!service_data.uuid.contains(0x1A, 0x18)) {
58 ESP_LOGVV(TAG, "parse_header(): no service data UUID magic bytes.");
59 return {};
60 }
61
62 auto raw = service_data.data;
63 if (raw.size() < 13) {
64 ESP_LOGVV(TAG, "parse_header_(): service data too short (%zu).", raw.size());
65 return {};
66 }
67
68 static uint8_t last_frame_count = 0;
69 if (last_frame_count == raw[12]) {
70 ESP_LOGVV(TAG, "parse_header(): duplicate data packet received (%hhu).", last_frame_count);
71 return {};
72 }
73 last_frame_count = raw[12];
74
75 return result;
76}
77
78bool ATCMiThermometer::parse_message_(const std::vector<uint8_t> &message, ParseResult &result) {
79 // Byte 0-5 mac in correct order
80 // Byte 6-7 Temperature in uint16
81 // Byte 8 Humidity in percent
82 // Byte 9 Battery in percent
83 // Byte 10-11 Battery in mV uint16_t
84 // Byte 12 frame packet counter
85
86 const uint8_t *data = message.data();
87 const int data_length = 13;
88
89 if (message.size() != data_length) {
90 ESP_LOGVV(TAG, "parse_message(): payload has wrong size (%d)!", message.size());
91 return false;
92 }
93
94 // temperature, 2 bytes, 16-bit signed integer (LE), 0.1 °C
95 const int16_t temperature = uint16_t(data[7]) | (uint16_t(data[6]) << 8);
96 result.temperature = temperature / 10.0f;
97
98 // humidity, 1 byte, 8-bit unsigned integer, 1.0 %
99 result.humidity = data[8];
100
101 // battery, 1 byte, 8-bit unsigned integer, 1.0 %
102 result.battery_level = data[9];
103
104 // battery, 2 bytes, 16-bit unsigned integer, 0.001 V
105 const int16_t battery_voltage = uint16_t(data[11]) | (uint16_t(data[10]) << 8);
106 result.battery_voltage = battery_voltage / 1.0e3f;
107
108 return true;
109}
110
111bool ATCMiThermometer::report_results_(const optional<ParseResult> &result, const char *address) {
112 if (!result.has_value()) {
113 ESP_LOGVV(TAG, "report_results(): no results available.");
114 return false;
115 }
116
117 ESP_LOGD(TAG, "Got ATC MiThermometer (%s):", address);
118
119 if (result->temperature.has_value()) {
120 ESP_LOGD(TAG, " Temperature: %.1f °C", *result->temperature);
121 }
122 if (result->humidity.has_value()) {
123 ESP_LOGD(TAG, " Humidity: %.0f %%", *result->humidity);
124 }
125 if (result->battery_level.has_value()) {
126 ESP_LOGD(TAG, " Battery Level: %.0f %%", *result->battery_level);
127 }
128 if (result->battery_voltage.has_value()) {
129 ESP_LOGD(TAG, " Battery Voltage: %.3f V", *result->battery_voltage);
130 }
131
132 return true;
133}
134
135} // namespace esphome::atc_mithermometer
136
137#endif
uint8_t address
Definition bl0906.h:4
uint8_t raw[35]
Definition bl0939.h:0
bool report_results_(const optional< ParseResult > &result, const char *address)
bool parse_device(const esp32_ble_tracker::ESPBTDevice &device) override
bool parse_message_(const std::vector< uint8_t > &message, ParseResult &result)
optional< ParseResult > parse_header_(const esp32_ble_tracker::ServiceData &service_data)
bool contains(uint8_t data1, uint8_t data2) const
Definition ble_uuid.cpp:112
const char * address_str_to(std::span< char, MAC_ADDRESS_PRETTY_BUFFER_SIZE > buf) const
Format MAC address into provided buffer, returns pointer to buffer for convenience.
const std::vector< ServiceData > & get_service_datas() const
void publish_state(float state)
Publish a new state to the front-end.
Definition sensor.cpp:68
const LogString * message
Definition component.cpp:35
uint16_t temperature
Definition sun_gtil2.cpp:12