ESPHome 2026.6.0-dev
Loading...
Searching...
No Matches
pvvx_mithermometer.cpp
Go to the documentation of this file.
2#include "esphome/core/log.h"
3
4#ifdef USE_ESP32
5
7
8static const char *const TAG = "pvvx_mithermometer";
9
11 ESP_LOGCONFIG(TAG, "PVVX 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 if (this->signal_strength_ != nullptr)
49 success = true;
50 }
51
52 return success;
53}
54
55optional<ParseResult> PVVXMiThermometer::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() < 14) {
64 ESP_LOGVV(TAG, "parse_header_(): service data too short (%zu).", raw.size());
65 return {};
66 }
67
68 if (this->last_frame_count_ == raw[13]) {
69 ESP_LOGVV(TAG, "parse_header(): duplicate data packet received (%hhu).", this->last_frame_count_);
70 return {};
71 }
72 this->last_frame_count_ = raw[13];
73
74 return result;
75}
76
77bool PVVXMiThermometer::parse_message_(const std::vector<uint8_t> &message, ParseResult &result) {
78 /*
79 All data little endian
80 uint8_t size; // = 19
81 uint8_t uid; // = 0x16, 16-bit UUID
82 uint16_t UUID; // = 0x181A, GATT Service 0x181A Environmental Sensing
83 uint8_t MAC[6]; // [0] - lo, .. [5] - hi digits
84 int16_t temperature; // x 0.01 degree [6,7]
85 uint16_t humidity; // x 0.01 % [8,9]
86 uint16_t battery_mv; // mV [10,11]
87 uint8_t battery_level; // 0..100 % [12]
88 uint8_t counter; // measurement count [13]
89 uint8_t flags; [14]
90 */
91
92 const uint8_t *data = message.data();
93 const int data_length = 15;
94
95 if (message.size() != data_length) {
96 ESP_LOGVV(TAG, "parse_message(): payload has wrong size (%d)!", message.size());
97 return false;
98 }
99
100 // int16_t temperature; // x 0.01 degree [6,7]
101 const int16_t temperature = int16_t(data[6]) | (int16_t(data[7]) << 8);
102 result.temperature = temperature / 1.0e2f;
103
104 // uint16_t humidity; // x 0.01 % [8,9]
105 const int16_t humidity = uint16_t(data[8]) | (uint16_t(data[9]) << 8);
106 result.humidity = humidity / 1.0e2f;
107
108 // uint16_t battery_mv; // mV [10,11]
109 const int16_t battery_voltage = uint16_t(data[10]) | (uint16_t(data[11]) << 8);
110 result.battery_voltage = battery_voltage / 1.0e3f;
111
112 // uint8_t battery_level; // 0..100 % [12]
113 result.battery_level = uint8_t(data[12]);
114
115 return true;
116}
117
118bool PVVXMiThermometer::report_results_(const optional<ParseResult> &result, const char *address) {
119 if (!result.has_value()) {
120 ESP_LOGVV(TAG, "report_results(): no results available.");
121 return false;
122 }
123
124 ESP_LOGD(TAG, "Got PVVX MiThermometer (%s):", address);
125
126 if (result->temperature.has_value()) {
127 ESP_LOGD(TAG, " Temperature: %.2f °C", *result->temperature);
128 }
129 if (result->humidity.has_value()) {
130 ESP_LOGD(TAG, " Humidity: %.2f %%", *result->humidity);
131 }
132 if (result->battery_level.has_value()) {
133 ESP_LOGD(TAG, " Battery Level: %.0f %%", *result->battery_level);
134 }
135 if (result->battery_voltage.has_value()) {
136 ESP_LOGD(TAG, " Battery Voltage: %.3f V", *result->battery_voltage);
137 }
138
139 return true;
140}
141
142} // namespace esphome::pvvx_mithermometer
143
144#endif
uint8_t address
Definition bl0906.h:4
uint8_t raw[35]
Definition bl0939.h:0
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
bool parse_message_(const std::vector< uint8_t > &message, ParseResult &result)
bool report_results_(const optional< ParseResult > &result, const char *address)
optional< ParseResult > parse_header_(const esp32_ble_tracker::ServiceData &service_data)
bool parse_device(const esp32_ble_tracker::ESPBTDevice &device) override
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