ESPHome 2026.5.0b1
Loading...
Searching...
No Matches
mopeka_std_check.cpp
Go to the documentation of this file.
1#include "mopeka_std_check.h"
2
4#include "esphome/core/log.h"
5
6#ifdef USE_ESP32
7
9
10static const char *const TAG = "mopeka_std_check";
11static const uint16_t SERVICE_UUID = 0xADA0;
12static const uint8_t MANUFACTURER_DATA_LENGTH = 23;
13static const uint16_t MANUFACTURER_ID = 0x000D;
14
15// Maximum bytes to log in very verbose hex output
16static constexpr size_t MOPEKA_MAX_LOG_BYTES = 32;
17
19 ESP_LOGCONFIG(TAG,
20 "Mopeka Std Check\n"
21 " Propane Butane mix: %.0f%%\n"
22 " Tank distance empty: %" PRIi32 "mm\n"
23 " Tank distance full: %" PRIi32 "mm",
24 this->propane_butane_mix_ * 100, this->empty_mm_, this->full_mm_);
25 LOG_SENSOR(" ", "Level", this->level_);
26 LOG_SENSOR(" ", "Temperature", this->temperature_);
27 LOG_SENSOR(" ", "Battery Level", this->battery_level_);
28 LOG_SENSOR(" ", "Reading Distance", this->distance_);
29}
30
37 // Validate address.
38 if (device.address_uint64() != this->address_) {
39 return false;
40 }
41
42 // Stack buffer for MAC address formatting - reused throughout function
43 char addr_buf[MAC_ADDRESS_PRETTY_BUFFER_SIZE];
44 const char *addr_str = device.address_str_to(addr_buf);
45
46 ESP_LOGVV(TAG, "parse_device(): MAC address %s found.", addr_str);
47
48 {
49 // Validate service uuid
50 const auto &service_uuids = device.get_service_uuids();
51 if (service_uuids.size() != 1) {
52 return false;
53 }
54 const auto &service_uuid = service_uuids[0];
55 if (service_uuid != esp32_ble_tracker::ESPBTUUID::from_uint16(SERVICE_UUID)) {
56 return false;
57 }
58 }
59
60 const auto &manu_datas = device.get_manufacturer_datas();
61
62 if (manu_datas.size() != 1) {
63 ESP_LOGE(TAG, "[%s] Unexpected manu_datas size (%d)", addr_str, manu_datas.size());
64 return false;
65 }
66
67 const auto &manu_data = manu_datas[0];
68
69#if ESPHOME_LOG_LEVEL >= ESPHOME_LOG_LEVEL_VERY_VERBOSE
70 char hex_buf[format_hex_pretty_size(MOPEKA_MAX_LOG_BYTES)];
71#endif
72 ESP_LOGVV(TAG, "[%s] Manufacturer data: %s", addr_str,
73 format_hex_pretty_to(hex_buf, manu_data.data.data(), manu_data.data.size()));
74
75 if (manu_data.data.size() != MANUFACTURER_DATA_LENGTH) {
76 ESP_LOGE(TAG, "[%s] Unexpected manu_data size (%d)", addr_str, manu_data.data.size());
77 return false;
78 }
79
80 // Now parse the data
81 const auto *mopeka_data = (const mopeka_std_package *) manu_data.data.data();
82
83 const u_int8_t hardware_id = mopeka_data->data_1 & 0xCF;
84 if (static_cast<SensorType>(hardware_id) != STANDARD && static_cast<SensorType>(hardware_id) != XL &&
85 static_cast<SensorType>(hardware_id) != ETRAILER && static_cast<SensorType>(hardware_id) != STANDARD_ALT) {
86 ESP_LOGE(TAG, "[%s] Unsupported Sensor Type (0x%X)", addr_str, hardware_id);
87 return false;
88 }
89
90 ESP_LOGVV(TAG, "[%s] Sensor slow update rate: %d", addr_str, mopeka_data->slow_update_rate);
91 ESP_LOGVV(TAG, "[%s] Sensor sync pressed: %d", addr_str, mopeka_data->sync_pressed);
92 for (u_int8_t i = 0; i < 3; i++) {
93 ESP_LOGVV(TAG, "[%s] %u. Sensor data %u time %u.", addr_str, (i * 4) + 1, mopeka_data->val[i].value_0,
94 mopeka_data->val[i].time_0);
95 ESP_LOGVV(TAG, "[%s] %u. Sensor data %u time %u.", addr_str, (i * 4) + 2, mopeka_data->val[i].value_1,
96 mopeka_data->val[i].time_1);
97 ESP_LOGVV(TAG, "[%s] %u. Sensor data %u time %u.", addr_str, (i * 4) + 3, mopeka_data->val[i].value_2,
98 mopeka_data->val[i].time_2);
99 ESP_LOGVV(TAG, "[%s] %u. Sensor data %u time %u.", addr_str, (i * 4) + 4, mopeka_data->val[i].value_3,
100 mopeka_data->val[i].time_3);
101 }
102
103 // Get battery level first
104 if (this->battery_level_ != nullptr) {
105 uint8_t level = this->parse_battery_level_(mopeka_data);
106 this->battery_level_->publish_state(level);
107 }
108
109 // Get temperature of sensor
110 int8_t temp_in_c = this->parse_temperature_(mopeka_data);
111 if (this->temperature_ != nullptr) {
112 this->temperature_->publish_state(temp_in_c);
113 }
114
115 // Get distance and level if either are sensors
116 if ((this->distance_ != nullptr) || (this->level_ != nullptr)) {
117 // Message contains 12 sensor dataset each 10 bytes long.
118 // each sensor dataset contains 5 byte time and 5 byte value.
119
120 // time in 10us ticks.
121 // value is amplitude.
122
123 std::array<u_int8_t, 12> measurements_time = {};
124 std::array<u_int8_t, 12> measurements_value = {};
125 // Copy measurements over into my array.
126 {
127 u_int8_t measurements_index = 0;
128 for (const auto &val : mopeka_data->val) {
129 measurements_time[measurements_index] = val.time_0 + 1;
130 measurements_value[measurements_index] = val.value_0;
131 measurements_index++;
132 measurements_time[measurements_index] = val.time_1 + 1;
133 measurements_value[measurements_index] = val.value_1;
134 measurements_index++;
135 measurements_time[measurements_index] = val.time_2 + 1;
136 measurements_value[measurements_index] = val.value_2;
137 measurements_index++;
138 measurements_time[measurements_index] = val.time_3 + 1;
139 measurements_value[measurements_index] = val.value_3;
140 measurements_index++;
141 }
142 }
143
144 // Find best(strongest) value(amplitude) and it's belonging time in sensor dataset.
145 u_int8_t number_of_usable_values = 0;
146 u_int16_t best_value = 0;
147 u_int16_t best_time = 0;
148 {
149 u_int16_t measurement_time = 0;
150 for (u_int8_t i = 0; i < 12; i++) {
151 // Time is summed up until a value is reported. This allows time values larger than the 5 bits in transport.
152 measurement_time += measurements_time[i];
153 if (measurements_value[i] != 0) {
154 // I got a value
155 number_of_usable_values++;
156 if (measurements_value[i] > best_value) {
157 // This value is better than a previous one.
158 best_value = measurements_value[i];
159 best_time = measurement_time;
160 }
161 // Reset measurement_time or next values.
162 measurement_time = 0;
163 }
164 }
165 }
166
167 ESP_LOGV(TAG, "[%s] Found %u values with best data %u time %u.", addr_str, number_of_usable_values, best_value,
168 best_time);
169
170 if (number_of_usable_values < 1 || best_value < 2 || best_time < 2) {
171 // At least two measurement values must be present.
172 ESP_LOGW(TAG, "[%s] Poor read quality. Setting distance to 0.", addr_str);
173 if (this->distance_ != nullptr) {
174 this->distance_->publish_state(0);
175 }
176 if (this->level_ != nullptr) {
177 this->level_->publish_state(0);
178 }
179 } else {
180 float lpg_speed_of_sound = this->get_lpg_speed_of_sound_(temp_in_c);
181 ESP_LOGV(TAG, "[%s] Speed of sound in current fluid %f m/s", addr_str, lpg_speed_of_sound);
182
183 uint32_t distance_value = lpg_speed_of_sound * best_time / 100.0f;
184
185 // update distance sensor
186 if (this->distance_ != nullptr) {
187 this->distance_->publish_state(distance_value);
188 }
189
190 // update level sensor
191 if (this->level_ != nullptr) {
192 uint8_t tank_level = 0;
193 if (distance_value >= this->full_mm_) {
194 tank_level = 100; // cap at 100%
195 } else if (distance_value > this->empty_mm_) {
196 tank_level = ((100.0f / (this->full_mm_ - this->empty_mm_)) * (distance_value - this->empty_mm_));
197 }
198 this->level_->publish_state(tank_level);
199 }
200 }
201 }
202
203 return true;
204}
205
207 return 1040.71f - 4.87f * temperature - 137.5f * this->propane_butane_mix_ - 0.0107f * temperature * temperature -
208 1.63f * temperature * this->propane_butane_mix_;
209}
210
212 const float voltage = (float) ((message->raw_voltage / 256.0f) * 2.0f + 1.5f);
213 ESP_LOGVV(TAG, "Sensor battery voltage: %f V", voltage);
214 // convert voltage and scale for CR2032
215 const float percent = (voltage - 2.2f) / 0.65f * 100.0f;
216 if (percent < 0.0f) {
217 return 0;
218 }
219 if (percent > 100.0f) {
220 return 100;
221 }
222 return (uint8_t) percent;
223}
224
226 uint8_t tmp = message->raw_temp;
227 if (tmp == 0x0) {
228 return -40;
229 } else {
230 return static_cast<int8_t>((tmp - 25.0f) * 1.776964f);
231 }
232}
233
234} // namespace esphome::mopeka_std_check
235
236#endif
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_manufacturer_datas() const
const std::vector< ESPBTUUID > & get_service_uuids() const
bool parse_device(const esp32_ble_tracker::ESPBTDevice &device) override
Main parse function that gets called for all ble advertisements.
int8_t parse_temperature_(const mopeka_std_package *message)
uint8_t parse_battery_level_(const mopeka_std_package *message)
void publish_state(float state)
Publish a new state to the front-end.
Definition sensor.cpp:68
const char * message
Definition component.cpp:35
mopeka_std_values val[3]
char * format_hex_pretty_to(char *buffer, size_t buffer_size, const uint8_t *data, size_t length, char separator)
Format byte array as uppercase hex to buffer (base implementation).
Definition helpers.cpp:341
constexpr size_t format_hex_pretty_size(size_t byte_count)
Calculate buffer size needed for format_hex_pretty_to with separator: "XX:XX:...:XX\0".
Definition helpers.h:1386
static void uint32_t
uint16_t temperature
Definition sun_gtil2.cpp:12