ESPHome 2026.1.5
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
6namespace esphome {
7namespace pvvx_mithermometer {
8
9static const char *const TAG = "pvvx_mithermometer";
10
12 ESP_LOGCONFIG(TAG, "PVVX MiThermometer");
13 LOG_SENSOR(" ", "Temperature", this->temperature_);
14 LOG_SENSOR(" ", "Humidity", this->humidity_);
15 LOG_SENSOR(" ", "Battery Level", this->battery_level_);
16 LOG_SENSOR(" ", "Battery Voltage", this->battery_voltage_);
17}
18
20 if (device.address_uint64() != this->address_) {
21 ESP_LOGVV(TAG, "parse_device(): unknown MAC address.");
22 return false;
23 }
24 char addr_buf[MAC_ADDRESS_PRETTY_BUFFER_SIZE];
25 const char *addr_str = device.address_str_to(addr_buf);
26 ESP_LOGVV(TAG, "parse_device(): MAC address %s found.", addr_str);
27
28 bool success = false;
29 for (auto &service_data : device.get_service_datas()) {
30 auto res = parse_header_(service_data);
31 if (!res.has_value()) {
32 continue;
33 }
34 if (!(parse_message_(service_data.data, *res))) {
35 continue;
36 }
37 if (!(report_results_(res, addr_str))) {
38 continue;
39 }
40 if (res->temperature.has_value() && this->temperature_ != nullptr)
41 this->temperature_->publish_state(*res->temperature);
42 if (res->humidity.has_value() && this->humidity_ != nullptr)
43 this->humidity_->publish_state(*res->humidity);
44 if (res->battery_level.has_value() && this->battery_level_ != nullptr)
45 this->battery_level_->publish_state(*res->battery_level);
46 if (res->battery_voltage.has_value() && this->battery_voltage_ != nullptr)
47 this->battery_voltage_->publish_state(*res->battery_voltage);
48 if (this->signal_strength_ != nullptr)
50 success = true;
51 }
52
53 return success;
54}
55
57 ParseResult result;
58 if (!service_data.uuid.contains(0x1A, 0x18)) {
59 ESP_LOGVV(TAG, "parse_header(): no service data UUID magic bytes.");
60 return {};
61 }
62
63 auto raw = service_data.data;
64
65 static uint8_t last_frame_count = 0;
66 if (last_frame_count == raw[13]) {
67 ESP_LOGVV(TAG, "parse_header(): duplicate data packet received (%hhu).", last_frame_count);
68 return {};
69 }
70 last_frame_count = raw[13];
71
72 return result;
73}
74
75bool PVVXMiThermometer::parse_message_(const std::vector<uint8_t> &message, ParseResult &result) {
76 /*
77 All data little endian
78 uint8_t size; // = 19
79 uint8_t uid; // = 0x16, 16-bit UUID
80 uint16_t UUID; // = 0x181A, GATT Service 0x181A Environmental Sensing
81 uint8_t MAC[6]; // [0] - lo, .. [5] - hi digits
82 int16_t temperature; // x 0.01 degree [6,7]
83 uint16_t humidity; // x 0.01 % [8,9]
84 uint16_t battery_mv; // mV [10,11]
85 uint8_t battery_level; // 0..100 % [12]
86 uint8_t counter; // measurement count [13]
87 uint8_t flags; [14]
88 */
89
90 const uint8_t *data = message.data();
91 const int data_length = 15;
92
93 if (message.size() != data_length) {
94 ESP_LOGVV(TAG, "parse_message(): payload has wrong size (%d)!", message.size());
95 return false;
96 }
97
98 // int16_t temperature; // x 0.01 degree [6,7]
99 const int16_t temperature = int16_t(data[6]) | (int16_t(data[7]) << 8);
100 result.temperature = temperature / 1.0e2f;
101
102 // uint16_t humidity; // x 0.01 % [8,9]
103 const int16_t humidity = uint16_t(data[8]) | (uint16_t(data[9]) << 8);
104 result.humidity = humidity / 1.0e2f;
105
106 // uint16_t battery_mv; // mV [10,11]
107 const int16_t battery_voltage = uint16_t(data[10]) | (uint16_t(data[11]) << 8);
108 result.battery_voltage = battery_voltage / 1.0e3f;
109
110 // uint8_t battery_level; // 0..100 % [12]
111 result.battery_level = uint8_t(data[12]);
112
113 return true;
114}
115
117 if (!result.has_value()) {
118 ESP_LOGVV(TAG, "report_results(): no results available.");
119 return false;
120 }
121
122 ESP_LOGD(TAG, "Got PVVX MiThermometer (%s):", address);
123
124 if (result->temperature.has_value()) {
125 ESP_LOGD(TAG, " Temperature: %.2f °C", *result->temperature);
126 }
127 if (result->humidity.has_value()) {
128 ESP_LOGD(TAG, " Humidity: %.2f %%", *result->humidity);
129 }
130 if (result->battery_level.has_value()) {
131 ESP_LOGD(TAG, " Battery Level: %.0f %%", *result->battery_level);
132 }
133 if (result->battery_voltage.has_value()) {
134 ESP_LOGD(TAG, " Battery Voltage: %.3f V", *result->battery_voltage);
135 }
136
137 return true;
138}
139
140} // namespace pvvx_mithermometer
141} // namespace esphome
142
143#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 has_value() const
Definition optional.h:92
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:76
const char * message
Definition component.cpp:38
Providing packet encoding functions for exchanging data with a remote host.
Definition a01nyub.cpp:7
uint16_t temperature
Definition sun_gtil2.cpp:12